Small knowledge, big challenge! This article is participating in the creation activity of “Essential Tips for Programmers”.
preface
Before we introduce Teleport, let’s take a look at what this new feature does: the Action panel pops up below the top button when clicked, common features in H5 such as Modal,Toast, Dialog of PC side component library, etc.
According to our thinking, the Dom position of the elastic layer should be on the top layer, which is convenient to control the display level and avoid the CSS level problems caused by nesting. Logically, we would apply this code as a child component to the parent component it should belong to for easy management. It is important to note that we are talking logically here because the Dom is no longer in the parent component.
First, how do you achieve a similar effect in Vue2.X?
What is the Teleport
The official documentation
Teleport is a new built-in component in Vue3 that can mount a contained node to a specified DOM node. For example, in the case of this article, mounting a Body at the same level as
(common in third-party component libraries) can be done simply by adding a to attribute to specify the target location,
To: Mandatory attribute, HtmlElement or selector, such as p,.class, # ID, etc
<teleport to="body">
<div>content</div>
</teleport>
Copy the code
The code above will be inserted under the body tag at the same level as #app. I’ve prepared a complete example.
Implement an Action panel with Teleport
Use Vite here for simple reasons – fast
yarn create @vitejs/app my-vue-app --template vue
Copy the code
Less and less-loader need to be installed in the project
Create two components, one parent (helloWord.vue) and one child (action.vue)
HelloWord.vue
<template>
<div class="">
<button class="btn" @click="openAction">Open the Action</button>
<Action
:show="showAction"
@close="showAction = false"
:options="options"
></Action>
</div>
</template>
<script>
import Action from "./Action.vue";
import { ref } from "vue";
export default {
name: "Hello".components: {
Action,
},
setup() {
const showAction = ref(false);
const options = ["Option 1"."Option 2"."Option 3"."Option 4"];
const openAction = () = > {
showAction.value = true;
};
return{ showAction, options, openAction }; }};</script>
<style scoped >
.btn {
background-color: pink;
border: none;
padding: 10px 20px;
cursor: pointer;
}
</style>
Copy the code
Action.vue
<template>
<teleport to="body">
<transition name="action-fade">
<div v-if="show" class="action-list" @click="$emit('close')">
<div class="action-wrapper" @click.stop>
<div class="action-container">
<li v-for="(item, index) in options" :key="index" @click="clickThis(item)">
{{ item }}
</li>
</div>
</div>
</div>
</transition>
</teleport>
</template>
<script>
export default {
name: "Action".props: {
show: {
type: Boolean.default: false,},options: {
type: Array.default: () = >[],}},methods: {
clickThis(item) {
this.$emit("close"); alert(item); ,}}};</script>
<style lang="less">
.action-list {
position: fixed;
z-index: 9998;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0.0.0.0.5);
}
.action-wrapper {
position: fixed;
bottom: 0;
left: 0;
right: 0;
}
.action-container {
width: 100%;
max-height: 300px;
padding: 10px 0;
overflow-y: auto;
z-index: 9999;
background-color: #fff;
border-radius: 10px 10px 0 0;
box-shadow: 0 2px 8px rgba(0.0.0.0.33);
transition: all 0.3 s ease;
}
.action-container li {
display: flex;
flex-direction: column;
align-items: center;
justify-content: space-around;
height: 60px;
list-style: none;
}
.action-container li:active {
background: #f2f3f5;
}
.action-body {
margin: 20px 0;
}
.action-fade-enter-active..action-fade-leave-active {
transition: opacity 0.3 s;
.action-wrapper {
transition: all 0.3 s; }}.action-fade-enter-from..action-fade-leave-to {
opacity: 0;
.action-wrapper {
transform: translate3d(0.100%.0); }}</style>
Copy the code
The overall layout uses positioning and the use of transation components to enrich the transition effect. The Action component uses the Teleport component, and the value of the TO attribute is the body tag. Let’s look at the Dom structure after rendering.
Looking at the image above, you can see that the content in the Action component is rendered under the body and is at the same level as the app root component.
So what do I do if I want to render this Action component under any node in Vue2.X? The Action component is dynamically added and removed from the node as a child node. Now let’s look at the implementation of the Element part of the source code, which actually adds a pop-up layer Dom to the Body of the code, and removes that PART of the Dom when the component is destroyed.
Element source
This part of the code is located in the packages/dialog/SRC/component. The vue, only keep related logic. Visible controls the overt and covert of Dialog; appendToBody’s el-Dialog props; it is worth mentioning here that the component destruction removes the Dialog below the body, because the Dialog component is actually rendered below the body node. Bugs occur when removeChild is not executed.
watch: {
visible(val) {
if (val) {
if (this.appendToBody) {
document.body.appendChild(this.$el); }}}},destroyed() {
// if appendToBody is true, remove DOM node after destroy
if (this.appendToBody && this.$el && this.$el.parentNode) {
this.$el.parentNode.removeChild(this.$el); }}Copy the code
conclusion
The code examples in this article are very simple, and if you want to make encapsulation completely easy to use, you can use slots. Teleport can be used to Toast, Modal, Dialog, etc.
We seem to be talking about putting the Teleport component under the Body all the time, and remember that there are many options.
Source sandbox address
Codesandbox. IO/s/vue3 – tele…