最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Vue pass slot template to extended component - Stack Overflow

programmeradmin1浏览0评论

Is any way to pass a template to extended ponent? Example, that illustrates the problem:

There is a ponent named Slide with a template like so:

<template>
    <div class="swiper-slide swiper-slide-one">
        <div class="swiper-filter"></div>
        <slot />
    </div>
</template>

In other place i need to create and mount that ponent in imperative style:

import SlideComponent from '@/ponents/Slide'
const Slide = Vue.extend(SlideComponent);
new Slide().$mount('#container');

The problem is that i have no idea how to pass a template that will be piled in the slot of the extended ponent.

jsfiddle

Is any way to pass a template to extended ponent? Example, that illustrates the problem:

There is a ponent named Slide with a template like so:

<template>
    <div class="swiper-slide swiper-slide-one">
        <div class="swiper-filter"></div>
        <slot />
    </div>
</template>

In other place i need to create and mount that ponent in imperative style:

import SlideComponent from '@/ponents/Slide'
const Slide = Vue.extend(SlideComponent);
new Slide().$mount('#container');

The problem is that i have no idea how to pass a template that will be piled in the slot of the extended ponent.

jsfiddle

Share Improve this question edited Mar 27, 2017 at 9:21 Kirill asked Mar 27, 2017 at 8:14 KirillKirill 5341 gold badge5 silver badges16 bronze badges
Add a ment  | 

5 Answers 5

Reset to default 5

You can use Vue.pile bined with $createElement.

const slotTemplate = `<div><h4>This is the slot content</h4>{{message}}</div>`;
const renderer = Vue.pile(slotTemplate)
const slotContent = {
  data(){
    return {
      message: "some data in the slot"
    }
  },
  render: renderer.render,
  staticRenderFns: renderer.staticRenderFns
}

const instance = new Slide();
instance.$slots.default = [instance.$createElement(slotContent)];
instance.$mount("#container");

Above is an example of a slot that requires some data. If your slot content is just HTML you could get away with just this:

const slotTemplate = `<div><h4>This is the slot content</h4></div>`
const instance = new Slide();
instance.$slots.default = [instance.$createElement(Vue.pile(slotTemplate))];
instance.$mount("#container");

Here is an example.

Note: $slots[key] should be an array of vNodes. It will render correctly if you use a vNode, but there will be bugs (e.g. when calling vm._update)

If you are open to change your approach a little bit, a <ponent> instead of <slot>:

https://jsfiddle/jonataswalker/z5Lgbjv8/

const Parent = {
  template: `<div>
        <span>parent content</span>
      <ponent :is="view"></ponent>
    </div>`,
  data() {
    return { view: '' }
  }
}

const Other = {
  template: `<div>
        <h1>from Other</h1>
    </div>`
}

const Slide = Vue.extend(Parent)

// how to pass template to the Parent`s slot?
new Slide({
  data: { view: Other }
}).$mount('#container')

(This question is similar to this SO question. I am just gonna share a similar answer I post for that question here.)

The trick is simple. Basically, when extending, create a new ponent that registers Parent as a local ponent.

Demo: https://jsfiddle/jacobgoh101/omjgmb3f/

const Slide = Vue.extend({
  ponents: {
    Parent
  },
  template: `
  <Parent>
    <div style="color:red">I am a slot</div>
  </Parent>
`
});
new Slide().$mount('#container')

If You can change Your source code structure, You should also be able to use jsx. In Vue-Cli3 it requires additional babel plugin, in Nuxt it works out of the box). You simply create some hook function as method or pute, for example:

slotHook () {
  return(
  <template v-slot.../>
)}

and place it inside the render function. Then, in the child (ponent extending the default one), You overwrite the slotHook method:

slotHook () {
  return (
    <template v-slot... some-differen-params:.../>
)}

This approach does not cause any errors/issues during render and works flawless. Other approach might be to use PUG templates and their block & extend functionality. I know I'm posting this response quite late, but maybe it could help somebody.

You have to pass it like this:

<Slide>
   //HTML code here goes to slot
</Slide>
发布评论

评论列表(0)

  1. 暂无评论