I have trouble figuring out how to get the following to work:
My parent template
<comp>
<a href="#" slot="links">link 1</a>
<a href="#" slot="links">link 2</a>
</comp>
and my component comp
template looks like the following:
<ul class="comp">
<li class="comp-item"><slot name="links"></slot></li>
</ul>
currently all my anchors goes to that single li
tag (which is expected) but I would like to be able to produce multiple li
for every named slot I inserted like the following:
<ul class="comp">
<li class="comp-item"><a href="#" slot="links">link 1</a></li>
<li class="comp-item"><a href="#" slot="links">link 2</a></li>
</ul>
Is there any way to achieve what I need without using scoped slot? Because my content is pure HTML so I feel it is unnecessary to put static content inside prop in order to render them.
From what I have seen, most vue UI framework requires you to use another custom component for the list item, which I feel is over killed for the problem. Is there any other way to do this?
This is easily accomplished with a render function.
Vue.component("comp", {
render(h){
let links = this.$slots.links.map(l => h('li', {class: "comp-item"}, [l]))
return h('ul', {class: 'comp'}, links)
}
})
Here is a working example.
console.clear() Vue.component("comp", { render(h){ let links = this.$slots.links.map(l => h('li', {class: "comp-item"}, [l])) return h('ul', {class: 'comp'}, links) } }) new Vue({ el: "#app" })
<script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.13/vue.js"></script> <div id="app"> <comp> <a href="#" slot="links">link 1</a> <a href="#" slot="links">link 2</a> </comp> </div>
Or with the help of a small utility component for rendering vNodes you could do it like this with a template.
Vue.component("vnode", { functional: true, render(h, context){ return context.props.node } }) Vue.component("comp", { template: ` <ul class="comp"> <li class="comp-item" v-for="link in $slots.links"><vnode :node="link" /></li> </ul> ` }) new Vue({ el: "#app" })
<script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.13/vue.js"></script> <div id="app"> <comp> <a href="#" slot="links">link 1</a> <a href="#" slot="links">link 2</a> </comp> </div>
You can make use of scoped slots instead of slots
Your comp
component receives a prop links
which is an array of links(since static initialized as a custom option ). Iterate over the links
and pass link
as data to the slot just like passing props to a component
Vue.component("comp", {
template: `
<ul class="comp">
<li class="comp-item" v-for="link in links">
<slot v-bind="link"></slot>
</li>
</ul>
`,
props: ["links"]
})
new Vue({
el: "#app",
// custom static option , accessed using vm.$options.links
links: [
{text: "link1"},
{text: "link2"},
{text: "lin3"}
]
})
In the parent where the comp
component is used make use of a <template>
element with a special attribute slot-scope
, indicating that it is a template for a scoped slot.
The value of slot-scope
will be used as the name of a temporary variable that holds the props object passed from the child:
<script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.13/vue.js"></script>
<div id="app">
<comp :links="$options.links">
<template slot-scope="link">
<a href="#">{{link.text}}</a>
</template>
</comp>
</div>
Here is the working fiddle
If you don't like to put your data in array, and render list with v-for
You can put all of them in the component, no slot:
<ul class="comp">
<li class="comp-item"><a href="#">link 1</a></li>
<li class="comp-item"><a href="#">link 2</a></li>
</ul>
or with slot:
<comp>
<ul class="comp" slot="links">
<li class="comp-item"><a href="#">link 1</a></li>
<li class="comp-item"><a href="#">link 2</a></li>
</ul>
</comp>
Do you know if it's possible to use the @Bert's solution #48975908 with a scoped slot ? In my case, I would like to reach something like below:
Vue.component("vnode", { functional: true, render(h, context){ return context.props.node } }) Vue.component("comp", { template: ` <ul class="comp"> <li class="comp-item" v-for="link in $slots.links"> <vnode :node="link" :handleClose="close" /> </li> </ul> `, methods: { close() { console.log('close!'); } } }) new Vue({ el: "#app" })
<script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.13/vue.js"></script> <div id="app"> <comp> <a href="#" #links="{closeHandler}" @click.prevent="closeHandler">link 1</a> <a href="#" slot="links">link 2</a> </comp> </div>
but I don't understand what I should do into the vnode
component or somewhere else :/
Thank you for your help and the all your previous answers :D
The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.