Bạn có thực sự hiểu về Slots?

Bạn có thực sự hiểu về Slots?

·

7 min read

Trong quá trình làm việc với Vuejs thì hẳn ai cũng đã từng sử dụng slot rồi, nhưng liệu ngoài chức năng là thêm nội dung html từ component cha vào component con thì nó còn có tác dụng nào nữa

Hãy đi qua các ví dụ và phương pháp dưới đây để biết xem rằng bạn có thật sự hiểu về slot không?

Đóng gói component

Vấn đề

Ví dụ với 1 component dialog đơn giản như này với thư viện là ElementUI

<template>
  <div>
    <el-button @click="handleOpen">Open </el-button>
    <el-dialog :visible="isVisible">
      <el-button @click="handleClose">Cancel</el-button>
    </el-dialog>
  </div>
</template>

<script>
export default {
  data() {
    return {
      isVisible: false,
    }
  },
  methods: {
    handleOpen() {
      this.isVisible = true;
    },
    handleClose() {
      this.isVisible = false;
    }
  }
}
</script>

Component này có 1 biến isVisible và các hàm có chức năng đóng mở dialog.

Nhưng khi component mở rộng và có nhiều dialog hơn, các biến và hàm nhiều lên.

<script>
export default {
 data() {
   return {
     isVisible1: false,
     isVisible2: false,
     isVisible3: false,
     isVisible4: false,
     isVisible5: false,
   }
 },
 methods: {
   handleOpen1() {
     this.isVisible1 = true;
   },
   handleClose1() {
     this.isVisible1 = false;
   },
   handleOpen2() {
     this.isVisible2 = true;
   },
   handleClose2() {
     this.isVisible2 = false;
   },
   handleOpen3() {
     this.isVisible3 = true;
   },
   handleClose3() {
     this.isVisible3 = false;
   }
   ...
 }
}
</script>

Các hàm đóng mở lặp đi lặp lại, vậy làm thế nào để tối ưu, chúng ta sẽ đi đến phần sau.

Giải quyết

Tạo một component là SlotDialog, component này sẽ đặt 1 slot default, và có biến isVisiblevới 2 function dùng để đóng mở dialog.

// SlotDialog

<template>
 <div>
   <!-- Truyền ngược lại các biến và hàm cần dùng vào slot theo kiểu truyền prop -->
   <slot :isVisible="isVisible" :open="handleOpen" :close="handleClose" />
 </div>
</template>

<script>
export default {
 data() {
   return {
     isVisible: false,
   }
 },
 methods: {
   handleOpen() {
     this.isVisible = true;
   },
   handleClose() {
     this.isVisible = false;
   }
 }
}
</script>

Và khi dùng thì chúng ta có thể lấy được các props được truyền vào bằng 2 cách là v-slot hoặc là slot-scope

<!-- Cách 1 -->
<SlotDialog>
  <!-- Dùng destructing để lấy các biến -->
  <template v-slot="{ isVisible, open, close }">
    <el-button @click="open">click </el-button>
    <el-dialog :visible="isVisible">
      <el-button @click="close">Cancel</el-button>
    </el-dialog>
  </template>
</SlotDialog>

<!-- Cách 2 -->
<SlotDialog>
  <template slot-scope="scope">
    <el-button @click="scope.open">click </el-button>
    <el-dialog :visible="scope.isVisible">
      <!-- Nếu muốn xử lý thêm logic thì chỉ cần truyền vào 1 function callback -->
      <el-button @click="handleClose(scope.close)">Cancel</el-button>
    </el-dialog>
  </template>
</SlotDialog>

<script>
export default {
  methods: {
    handleClose(close) {
      // Hàm này sẽ xử lý logic và gọi lại hàm đóng dialog
      close()
    }
  }
}
</script>

Đối với slot default thì chúng ta có thể viết như trên, còn với ví dụ slot có tên là body thì cú pháp sẽ là:

<!-- Cách 1 -->
<template v-slot:body="scope"></template>

<!-- Cách 2 -->
<template slot="body" slot-scope="scope"></template>

Vậy là quá ví dụ trên chúng ta đã thấy là có thể sử dựng nhiều dialog mà không cần phải tạo các biến và hàm giống nhau

Slot lồng nhau

Vấn đề

Component Counter có 1 biến counter sẽ tăng dần theo mỗi giây được đặt trong slot body

Các thẻ HeaderFooter cũng được dặt trong slot tương ứng.

// Counter.vue

<template>
  <div>
    <slot name="header">
      <h1>Header</h1>
    </slot>
    <slot name="body">
      <p>Counter: {{ counter }}</p>
    </slot>
    <slot name="footer">
      <h2>Footer</h2>
    </slot>
  </div>
</template>

<script>
export default {
  data() {
    return {
      counter: 0,
    }
  },
  beforeMount() {
    this.$options.counter = setInterval(() => {
      this.counter += 1;
    }, 1000); 
    this.$on('hook:beforeDestroy', () => {
      clearInterval(this.$options.counter)
    })
  },
}
</script>

Component này sẽ được dùng ở rất nhiều nơi, mỗi nơi lại có 1 yêu cầu khác nhau, là thay đổi nội dung của body hoặc header, footer

Cách làm chúng ta nghĩ đến đầu tiên sẽ là truyền các props điều kiện vào và if else để thay đổi.

Và khi đó, mỗi 1 yêu cầu sẽ là 1 prop mới và thêm các if else, cuối cùng nó sẽ trở thành 1 đống rác.

Tư duy khi thiết kế 1 component dùng chung cho nhiều nơi đó là làm thế nào vẫn có thể chỉnh sửa và mở rộng được mà vẫn không làm ảnh hưởng đến những phần khác.

Component Counter được lồng bên trong component ChildParent

// ChildComponent

<template>
  <Counter>
    <template slot="header">
      <slot name="header" />
    </template>
    <template slot="body">
      <slot name="body" />
    </template>
    <template slot="footer">
      <slot name="footer" />
    </template>
  </Counter>
</template>

<script>
import Counter from './Counter.vue'

export default {
  components: { Counter },
}
</script>
// ParentComponent

<template>
  <ChildComponent />
</template>

<script>
import ChildComponent from "./ChildComponent.vue"

export default {
  components: { ChildComponent },
}
</script>

Component Child sẽ gọi component Counter và sẽ để sẵn các thẻ slot để có thể truyền slot từ component Parent

Lưu ý khi truyền slot 2 tầng thế này thì chúng ta sẽ sử dụng thẻ template, vì nếu sử dụng thẻ div thì component Counter sẽ hiểu rằng ta đang truyền slot với nội dung rỗng từ ngoài vào và nó sẽ đè mất phần slot mặc định bên trong slot đấy.

Nếu dùng thẻ div, phải thêm điều kiện để chỉ khi nào có slot truyền vào từ Parent thì mới render ra.

<div v-if="$slots.body" slot="body">
  <slot name="body" />
</div>

Giải quyết

Và bây giờ, chúng ta sẽ sử dụng kỹ thuật scoped slots để lấy giá trị counter từ component Counter ra ngoài Parent

// Counter.vue

<template>
  <div>
    <slot name="header">
      <h1>Header</h1>
    </slot>
    <!-- Truyền biến counter vào slot body -->
    <slot name="body" :counter="counter" >
      <p>Counter: {{ counter }}</p>
    </slot>
    <slot name="footer">
      <h2>Footer</h2>
    </slot>
  </div>
</template>
// ChildComponent

<template>
  <Counter>
    <template slot="header">
      <slot name="header" />
    </template>
    <!-- Lấy biến counter từ scope rồi truyền ngược ra 1 lần nữa vào slot body -->
    <template slot="body" slot-scope="scope">
      <slot name="body" :counter="scope.counter" />
    </template>
    <template slot="footer">
      <slot name="footer" />
    </template>
  </Counter>
</template>
// ParentComponent

<template>
  <ChildComponent>
    <template slot="header">
      <h1>Custom Header</h1>
    </template>
    <template slot="body" slot-scope="scope">
      <h1>
        {{ scope.counter }}
      </h1>
    </template>
  </ChildComponent>
</template>

Cuỗi cùng chúng ta đã có thể thay đổi được nội dung từ Parent mà vẫn lấy được các giá trị từ bên trong component Counter

Tối ưu

Cách viết của Child component bên trên đã dùng được, nhưng khi có nhiều slot và prop thì chúng ta lại phải viết đi viết lại rất nhiều

// ChildComponent

<template>
  <Counter>
    <template slot="header">
      <slot name="header" />
    </template>
    <template slot="body" slot-scope="scope">
      <slot
        name="body"
        :prop1="scope.prop1"
        :prop2="scope.prop2"
        :prop3="scope.prop3"
        ...
      />
    </template>
    <template slot="footer" slot-scope="scope">
      <slot name="footer"
        :prop1="scope.prop1"
        :prop2="scope.prop2"
        :prop3="scope.prop3"
        ...
      />
      ...
    </template>
  </Counter>
</template>

Nên chúng ta sẽ dùng vòng for để render ra các thẻ slot

// ChildComponent

<template>
  <Counter>
    <!-- Lấy các slot name từ $slots và truyền vào thẻ <slot> -->
    <template v-for="slotName in $slots">
      <slot :name="slotName"></slot>
    </template>

    <!-- Đôi với scoped slots thì sẽ lấy các key $scopedSlots  -->
    <!-- Truyền slotName theo cú pháp dynamic slot để lấy scope -->
    <template
      v-for="slotName in Object.keys($scopedSlots)"
      v-slot:[slotName]="scope"
    >
      <!-- Thay vì phải viết từng prop ra, chúng ta sẽ v-bind scope luôn -->
      <slot :name="slotName" v-bind="scope"></slot>
    </template>

    <!-- Scoped slots cũng có thể loop theo cách này -->
    <!-- Biến đầu tiền sẽ là 1 function, chúng ta không dùng mà sẽ chỉ dùng slot name thôi -->
    <template v-for="(_, slotName) in $scopedSlots" #[slotName]="scope">
      <!-- Có thể truyền thêm các prop sau v-bind -->
      <slot :name="slotName" v-bind="scope" :newProp="1"></slot>
    </template>
  </Counter>
</template>

<script>
import Counter from './Counter.vue'

export default {
  components: { Counter },
}
</script>

Kết luận

Scoped slot là một phương pháp giúp chúng ta có thể gói gọn các biến và logic trong 1 component, nhằm tránh dư thừa code.

Nhưng đúng theo tên gọi của nó, phạm vi của các prop được lấy ra từ slot chỉ có thể hoạt động được trong scope đấy.

Vậy nên tùy vào mỗi trường hợp mà chúng ta sẽ cân nhắc có nên sử dụng hay không, nếu được sử dụng đúng cách thì hiệu quả của phương pháp này đem lại là rất tuyệt vời.