Trigger events from a parent Vue 3 component to one or more child components.
There is also a Vue 2 version of this library.
This small library provides an intentionally limited event bus. It allows parents to emit events and children to listen for these events in a local, understandable manner.
npm install vue-parent-emit
- Parent Component
- Create an event source using
myEventSource = newEventSource()
- Pass as prop to child (e.g.
:my-event="myEventSource"
)
- Create an event source using
- Child Component
- in
setup()
:useExternalEvent(props.myEvent, fetchSomeData)
This registers an event listener when mounted. - or e.g. in
mounted()
:useExternalEvent(props.myEvent, fetchSomeData, {immediate: true})
This registers an event listener immediately. - The event listener will be deregistered automatically on unmount.
- in
- Somewhere in parent
- emit events using e.g.
myEventSource.emit('hello child!')
- call
emit()
without a parameter, or use a single parameter to pass arbitrary data to the event listener(s)
- emit events using e.g.
See Usage Notes for further discussion.
Also see a live sandbox example
// parent-component.vue
<template>
<div>
<ChildComponent :my-event="myEventSource" other-prop="hello" />
<button @click="sendEvent">Notify child</button>
</div>
</template>
<script>
import { defineComponent } from 'vue';
import { newEventSource } from 'vue-parent-emit';
export default defineComponent({
// ...
data() {
return {
// ...
myEventSource: newEventSource(), // TS: newEventSource<MyEventPayload>()
};
},
methods: {
sendEvent() {
// use this anywhere in the parent component
this.myEventSource.emit();
// or this.myEventSource.emit(someEventPayload)
},
},
});
</script>
// child-component.vue
<template>
<!-- child template -->
</template>
<script>
import { defineComponent } from 'vue';
import { useExternalEvent } from 'vue-parent-emit';
export default defineComponent({
// ...
props: {
// ...
myEvent: Object, // TS: as PropType<EventSource<MyEventPayload>>
},
setup(props) {
const myData = ref("initial value");
useExternalEvent(props.myEvent, (eventPayload) => fetchSomeData(myData, eventPayload));
return { myData };
},
});
function fetchSomeData(myData, eventPayload /*TS: :MyEventData*/) {
// handle the event received from the parent, e.g.:
console.log('child: fetching new data', eventPayload);
myData.value = 'fetching ...';
}
</script>
// child-component.vue
<template>
<!-- child template -->
</template>
<script>
import { defineComponent } from 'vue';
import { useExternalEvent } from 'vue-parent-emit';
export default defineComponent({
// ...
props: {
// ...
myEvent: Object, // TS: as PropType<EventSource<MyEventPayload>>
},
mounted() {
// register child event listener
useExternalEvent(this.myEvent, this.fetchSomeData, {immediate: true});
},
methods: {
fetchSomeData(eventPayload /*TS: :MyEventData*/) {
// handle the event received from the parent, e.g.:
console.log('child: fetching new data', eventPayload);
this.myData = 'fetching ...';
},
});
</script>
Triggering actions, like re-fetching child data is a very good use-case for this mechanism. It is not (primarily) intended to pass data to the child. Props can do that just fine.
Most of the time you should not need an event payload at all.
You can use a single event source for multiple kinds of events by passing in different payloads (this.myEventSource.emit('foo-event')
and this.myEventSource.emit('bar-event')
) and dispatch them according to the payload in the child component.
We however recommend to create multiple specific event sources instead. You will need to pass multiple props, but both emitting and handling the events is easier to understand this way. As an example, you could create two separate event sources and just invoke this.fooEvent.emit()
or this.barEvent.emit()
respectively.