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

javascript - How I can render a component from a string with vue 3? - Stack Overflow

programmeradmin2浏览0评论

I'm trying render a ponent from string but I didn't succeed. My codes are bellow:

<template>
<div v-html="beautifyNotification(notification)"></div>
</template>

<script>
import { Link } from '@inertiajs/inertia-vue3'
import {pile,h} from "vue"

export default {
    ponents: {
    },
    props: {
        notifications: Object
    },
    methods: {
        beautifyNotification (ntfction) {
            return h(pile(`<Link :href="`+ntfction.from.username+`"
                    class="h6 notification-friend">`+ntfction.from.name+`
            </Link>, mented on your new
            <Link href="#" class="notification-link">profile status</Link>.`))
        },
    }
}
</script>

I tried render ponent with h and pile but it returned object object

I'm trying render a ponent from string but I didn't succeed. My codes are bellow:

<template>
<div v-html="beautifyNotification(notification)"></div>
</template>

<script>
import { Link } from '@inertiajs/inertia-vue3'
import {pile,h} from "vue"

export default {
    ponents: {
    },
    props: {
        notifications: Object
    },
    methods: {
        beautifyNotification (ntfction) {
            return h(pile(`<Link :href="`+ntfction.from.username+`"
                    class="h6 notification-friend">`+ntfction.from.name+`
            </Link>, mented on your new
            <Link href="#" class="notification-link">profile status</Link>.`))
        },
    }
}
</script>

I tried render ponent with h and pile but it returned object object

Share Improve this question edited Feb 5, 2022 at 10:54 Nikola Pavicevic 23.5k9 gold badges29 silver badges51 bronze badges asked Feb 2, 2022 at 14:52 neographyneography 313 silver badges6 bronze badges 4
  • I don't see why you need to render your ponents like that in your example. Just use the <Link /> inside the template. That's exactly how the ponents are supposed to be used 99% of the time. – captainskippah Commented Feb 2, 2022 at 15:04
  • Check out how <todo-item /> is used in the docs: v3.vuejs/guide/introduction.html#posing-with-ponents – captainskippah Commented Feb 2, 2022 at 15:06
  • Actually i want to learn how to render a string but if I don't find a solution I'll do like what you said – neography Commented Feb 2, 2022 at 15:08
  • if you want to render an HTML string (literally a "<div>This is a string HTML</div>") then you use v-html. Docs: v3.vuejs/guide/template-syntax.html#raw-html – captainskippah Commented Feb 2, 2022 at 15:11
Add a ment  | 

3 Answers 3

Reset to default 4

To anyone else hopefully finding this, so you don't have to go down the hellish path of wrong answers that I did (very surprisingly no one's SO answers work properly, even though they specify vue 3). Even within the 'advanced' area of vue discord i was getting weird/wrong answers, which was surprsing.

The answer above by maembe didn't work for me either (it doesnt have access to any other ponents/scope)

This is what worked perfect for me.

Heres mine CompiledContent ponent that accepts html + vue ponents mixed together in a single string from backend.

Notice it doesn't use the pile function at all

<script>
import { h } from 'vue';
import AppAlert from '@/ponents/AppAlert.vue';

export default {
  props: {
    content: {
      type: String,
      default: '',
    },
  },
  render() {
    const r = {
      ponents: {
        AppAlert,
      },
      template: `<div class="content">${this.content || ''}</div>`,
      methods: {
        hello() {
          // method "hello" is also available here
        },
      },
    };
    return h(r);
  },
};
</script>

If you have tons of ponents in your content you can also make them all async ponents:

ponents: {
  AppAlert: defineAsyncComponent(() => import('@/ponents/AppAlert.vue')), 
  ...

I will answer this question even though it seems late Below I use the static pilation method

Advantage:

  • Fast and less CPU intensive: it uses native pilation instead of bundling the entire heavy vue piler into your package
  • Saves memory: This method only uses vue's prototype functions instead of including the piler so it doesn't consume any memory

Defect:

  • Does not support SFC syntax: this is a native piler so it will not accept regular {{ value }} (although you can add one) but it reluctantly supports event syntax @click=... and popnent prop support

import { Fragment, h, Component, defineComponent } from "vue"

function buildElement(element: ChildNode, ponents: Record<string, Component>) {
  if (element.nodeType === 1) {
    const { attributes } = element as HTMLElement

    const name = element.nodeName
    const attrs: Record<string, unknown> = {}
    for (let index = 0; index < attributes.length; index++) {
      const attr = attributes.item(index)
      attrs[attr.name] = attr.value
    }

    const children = pilerChildren(element as HTMLElement, ponents)

    const ponent = ponents[element.nodeName.toLowerCase()]
    console.log(element.nodeName, ponents)

    return ponent ? h(ponent, attrs, {
      default: () => children
    }) : h(name, attrs, children)
    // console.log({ name, attrs, children })
  }

  return element.textContent
}

function pilerChildren(parent: HTMLElement, ponents: Record<string, Component>) {
  // deflag
  const children = []
  parent.childNodes.forEach(item => {
    switch (item.nodeType) {
      case 1:
      case 3:
        // normal element
        children.push(buildElement(item, ponents))
        break
      default:
        console.warn("Can't resolve this element: ", item)
    }
  })

  return children
}


export function piler(html: string, ponents: Record<string, Component>) {
  const ps = Object.fromEntries(
    Object.entries(ponents).map(([name, ponent]) => [name.toLowerCase(), ponent])
  )
  return () => h(Fragment, null, pilerChildren(
    new DOMParser().parseFromString(html, "text/html").body,
    ps
  ))
}

Here is the mon version so people who don't have in-depth knowledge of vue's low-level APIs can understand I would normally use openBlock

Demo:

https://play.vuejs/#eNqVVktv4zYQ/itT9VAZ60gJclNsF002QVrkhU3QSxSgskRLSihSIKnYgeH/3uFQkqW4u+ge/OBwHt88OVvvj7oO3hvmRd5Mp6qsDWhmmhp4IvJ57Bkde4tYhCE8FaUGbRJlmALDqponhgHSGl2KHP5uGJzCSMcCHq8uNAlfFCx9A9kYKIypdRSGaPNVB1LlYVKXoV6lR070iESDwlT81yElFmVVS2VgC6ms6pIjih2slKwg9oKQy7xMw+4m9nr2c8mzPZs9WXctQyxSKbRFVHGYwz+xuGacS2DvTH1IwX6B2VKh77G4/QCRVOTsjPSlkkuF0Vlyq2rxlKRFuUxEAo9FKWZkhQQ7BRS7A+lcMSZQ/BtLDdChFY1QMivfMdofnCHjusxMEcHJ8XG9OYOClXlh+uMySd9yJRuRHZHiyKlCvbMQlTgcQ22fuGIBcF2CaSEyziomDJQCrNAYcV2KNxR5wB9H633tbGEUu7g+FnKNce1y4ts4TzF9pHE3sTIuwQ5iV1IEaEbCISkeXOCRvMC/S5l9wNbyHgZgXZSGncViRzZIwJtiKSOuVZkHr1oKrHcStuFw+O5rUyLu2IucWnuXYD2s/yKaUQ2bdvTUlvN/0F/1xtJi70ExzdQ7Fkd/h52TM+OuLx/v2Ab/95eVzBqO3D+4/Ma05I3F6NjO0WEq9Z6P0P5JZY8d+aQvN4YJ3TllgVrOHfHHHnbBxQ9c38M9DU5JDuOJURw3WmA0RrLvzCuV5LZ6plBMwWrHNrKnjK1KQeaIsG9c14oKM4ufVSNSCwSWTcmzS1eIfluQEc4QpN7JjE2ppkiTjgC7R6oM60Kh0wOriwk6g1oByhV0WgKB8k8fNYP5fA4nPQugSluzW0gMKlo2hmlEOe/bIdFw/XR704LqEO8FaT707GTmDkljJqv7EHEj3oRciwWKb3edwEoq8DmzfZixDV4dn7V/ZwOIAWciN0V79eXLwJ+hTRQfyGBvVD4JTPbMBO3ZfgfWlZdWJHhP7Ihr2Qjd2CUcfDxTTAwanfKEpC7mn2I3zB4B+KSwLxKn0TE+fw5sYOSNXKOxRDN/8jLUITkLsEpHKbci3zWs8H1RYmD5dyj8/jR1scHJtY8W1nPScKxJfwLzRR+EPk4TiFCFDWQv3fH0McdXcYh2SyV0wG4HJQkMYt/i7fwzOC0upMBep7rsGft2OkhMneA3ov9OVv5PTyF6DAJPcnc8KIbnNiXOVEA3tnd1gJV9ie+lb8vQBq8Pq16XJi2ALvo+HVc0JhtOok+E0wGBgAmpqoR38RledgCDutGFPxoz1uqoQgbNgW+MYsnbYf5HuttkrhMl/Ni7SMRvBlOFtHfmXtd+kuHMBmuvN9G2vcv1KMnD0uozaz9sQzP3IMn0zEb42LvU/VxW9y2oMYf3y1fcTAI7qi8FCjHtd4DbKxQm8iBsQZXUvv/sirmnv1CfEHXcuSMep739aQPgOqzw90+LaDjuEQdF3UETbA1f728fEoUvsD8JsAI1u0IfHsnvdg3BBw7bJrSH2JsEdpuY7ocI+t9iITD23es2R7soj1eVJUTdamV3U9xAMJ/e4vz+5muECzGXBheZWbh03ON1Bgntxkz7tsENmzbnGMezfTIflKz1jB7ldrdx2UNMCx9XKKuwXaK83b/bkA9X

You're using the h function wrong here. h() returns virtual node that is meant to be used inside a render() function instead of a template. You don't need a <template> or <v-html> at all in this case:

//no template element
<script>
import { Link } from '@inertiajs/inertia-vue3'
import {pile,h} from "vue"

export default {
    props: {
        notifications: Object
    },
    render() {
        return h(
           // abbreviated template string for brevity
           pile('<Link href="#" class="notification-link">profile status</Link>')
        ) 
    }
}
</script>
发布评论

评论列表(0)

  1. 暂无评论