4

是否可以创建一个组件并以编程方式将事件侦听器附加到它?

我知道这对于使用<svelte:component/>传播的道具很容易实现{ ...props }。我想知道是否可以实现类似的附加事件侦听器。

例如,在以下示例中,我想以编程方式附加on:messageAon:countB

<!-- App.svelte -->
<script>
    import A from './A.svelte';
    import B from './B.svelte';

    let message = 'Hi there ';
    let count = 0;

    const components = [{
        component: A,
        props: { message },
        listeners: { message: (m) => { console.log(`They say "${m}"`); } }
    }, {
        component: B,
        props: { count },
        listeners: { click: () => { count++; } }
    }];
</script>

{#each components as component}
    <div><svelte:component this={component.component} { ...component.props }/></div>
{/each}

<div>
    <p>They say "{message}"!</p>
    <p>They clicked {count} times!</p>
</div>

<!-- A.svelte -->
<script>
    import { createEventDispatcher } from 'svelte';
    const dispatch = createEventDispatcher();
    export let message = '';
    function changeHandler(e) { dispatch('message', message); }
</script>

<input on:change={changeHandler} on:value={message} value={message} />

<!-- B.svelte -->
<script>
    import { createEventDispatcher } from 'svelte';
    const dispatch = createEventDispatcher();
    export let count = 0;
    function clickHandler() { dispatch('count', count); }
</script>

<button on:click={clickHandler}>Click me</button>

这是一个现场演示:https ://svelte.dev/repl/af1bd30ab75b43f19b72a306340b7282?version=3.18.2

即,我希望有一种方法可以将components数组扩展到

<A message={message} on:message={e => { message = e.detail; }}/>
<B count={count} on:count={e => { count = e.detail; }}/>
4

3 回答 3

7

您可以使用$on动态注册组件事件的侦听器。

在您的用例中,您可以使用此组件:

组件事件.svelte

<svelte:component this={component.component} { ...component.props } bind:this={instance}/>

<script>
  export let component;

    let instance;
    
    $: if (instance && component.listeners) {
        for (let [key, listener] of Object.entries(component.listeners)) {
            instance.$on(key, listener);
        }
    }
</script>

并在您的 App.svelte 中将每个循环替换为:

{#each components as component}
    <ComponentEvent {component}/>
{/each}

看到这个REPL

于 2020-07-04T18:51:42.957 回答
1

您不能动态附加侦听器,不 - 不要忘记 Svelte 是一种编译语言,它在编译时完成所有繁重的工作,因此需要提前了解。

我为动态侦听器所做的是触发单个已知事件,然后让事件详细信息包含差异逻辑,例如事件名称等。如下所示:

// SomeComponent.svelte
<script>
  import { createEventDispatcher } from 'svelte'

  const dispatch = createEventDispatcher()

  dispatch('component-event', { name: 'alert', value: 'oh noes' })
  dispatch('component-event', { name: 'log', value: 'some message' })
</script>
// App.svelte
<svelte:component this={someComponent} on:component-event={e => handleEvent(e)} />

<script>
  function handleEvent ({ detail }) {
    const { name, value } = detail
    if (name === 'alert') { alert(value) }
    if (name === 'log') { console.log(value) }
  }
</script>
于 2020-02-19T09:33:03.610 回答
0

不确定您是否可以像您正在寻找的那样以编程方式附加它们,但您可以在对象中传递一个函数或回调,并解释 svelte:component 标记中的每个潜在调度事件,如下所示:

<svelte:component
    this={component.component}
    { ...component.props }
    on:message={ component.function}
    on:count={ component.function}
/>

这是REPL中的一个工作示例,它显示了对象中的箭头函数和回调(如果需要更丰富的函数),回调仍会获取事件详细信息。

如果组件中不存在分派的事件,我认为不会发生任何事情,但是如果您需要单个组件来分派多个事件,则需要其他东西。

于 2020-02-16T05:33:27.280 回答