//Card.vue <template> <div class="card"> <img class="card-img" :src="img" /> <p class="card-desc">{{ desc }}</p> </div> </template> <style lang="scss" scoped> .card { // 這些以"--"開頭的變數就是我們開放可自定義的樣式屬性了,可以在元件使用檔案中明確開放 //卡片根元素樣式 --width: 150px; --height: auto; --border-size: 1px; --border-color: #ccc; --border-radius: 5px; --bg: #eee; // 圖片可定樣式 --img-width: 130px; --img-height: 130px; --img-radius: 50%; // 卡片描述樣式 --desc-size: 16px; --desc-line-height: 32px; --desc-color: #333; height: var(--height); width: var(--width); border: var(--border-size) solid var(--border-color); border-radius: var(--border-radius); background: var(--bg); padding: 10px; &-img { width: var(--img-width); height: var(--img-height); border-radius: var(--img-radius); overflow: hidden; } &-desc { font-size: var(--desc-size); color: var(--desc-color); line-height: var(--desc-line-height); text-align: center; } } </style>
//demo.vue <template> <div > <Card desc="我是預設的樣式我是預設的樣式我是預設的樣式" :img="img" /> <Card class="card_1" desc="自定義樣式,子元素圖片變小了" :img="img" /> <Card class="card_2" desc="自定義樣式,圓角沒了,描述字變小了,高度高了" :img="img" /> </div> </template> <script>...</script> <style lang="scss" scoped> .card_1 { --width: 100px; --height: 200px; --border-radius: 20px; --img-width: 80px; --img-height: 50px; --img-radius: 10px; --desc-color: #f00; --desc-size: 12px; --desc-line-height: 21px; } .card_2 { --height: 300px; --border-radius: 0px; --bg: #fff; --img-radius: 50px; --desc-size: 14px; --desc-line-height: 21px; } </style>
這種我們也可以把這些全提出來使用style屬性。讓樣式與js引數徹底隔離
//demo.vue <template> <card desc="我是預設的樣式我是預設的樣式" :img="img" :style="hoverStyle" @mouseout="hoverStyle = {}" @mouseover="handleHover" /> </template> <script setup> let hoverStyle = ref({}); const handleHover = () => { hoverStyle.value = { '--bg': '#f0f', '--width': '180px' }; }; </script>
我們在元件內
//card.vue <template> <div class="card" :style="style"> {{ width }} <img class="card-img" :src="img" /> <p class="card-desc">{{ desc }}</p> </div> </template> <script setup> const $props = defineProps({ img: { type: String, default: '', }, desc: { type: String, default: '', }, style: { type: Object, default: () => ({}), }, }); //假如你在js中需要用到寬度 let width = computed(() => { return parseInt($props.style['--width'] || 150); }); </script> <style lang="scss" scoped> .card{ ... //假如你有個子級元素需要基於寬度計算 .item{ width: calc(var(--width) - 100) } ... } </style>
但是這種實現有名稱空間的問題
//比如這樣 .ch-card{ --ch-card-width:100px; --ch-card-height:100px; }