First pass at allowing multiple event types to be emitted by an entity
This commit is contained in:
parent
c81440424b
commit
26fc36ee0e
6 changed files with 123 additions and 74 deletions
|
@ -201,7 +201,8 @@ pub struct AppContext {
|
||||||
pub(crate) pending_notifications: HashSet<EntityId>,
|
pub(crate) pending_notifications: HashSet<EntityId>,
|
||||||
pub(crate) pending_global_notifications: HashSet<TypeId>,
|
pub(crate) pending_global_notifications: HashSet<TypeId>,
|
||||||
pub(crate) observers: SubscriberSet<EntityId, Handler>,
|
pub(crate) observers: SubscriberSet<EntityId, Handler>,
|
||||||
pub(crate) event_listeners: SubscriberSet<EntityId, Listener>,
|
// (Entity, Event Type)
|
||||||
|
pub(crate) event_listeners: SubscriberSet<EntityId, (TypeId, Listener)>,
|
||||||
pub(crate) release_listeners: SubscriberSet<EntityId, ReleaseListener>,
|
pub(crate) release_listeners: SubscriberSet<EntityId, ReleaseListener>,
|
||||||
pub(crate) global_observers: SubscriberSet<TypeId, Handler>,
|
pub(crate) global_observers: SubscriberSet<TypeId, Handler>,
|
||||||
pub(crate) quit_observers: SubscriberSet<(), QuitHandler>,
|
pub(crate) quit_observers: SubscriberSet<(), QuitHandler>,
|
||||||
|
@ -351,14 +352,15 @@ impl AppContext {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn subscribe<T, E>(
|
pub fn subscribe<T, E, Evt>(
|
||||||
&mut self,
|
&mut self,
|
||||||
entity: &E,
|
entity: &E,
|
||||||
mut on_event: impl FnMut(E, &T::Event, &mut AppContext) + 'static,
|
mut on_event: impl FnMut(E, &Evt, &mut AppContext) + 'static,
|
||||||
) -> Subscription
|
) -> Subscription
|
||||||
where
|
where
|
||||||
T: 'static + EventEmitter,
|
T: 'static + EventEmitter<Evt>,
|
||||||
E: Entity<T>,
|
E: Entity<T>,
|
||||||
|
Evt: 'static,
|
||||||
{
|
{
|
||||||
self.subscribe_internal(entity, move |entity, event, cx| {
|
self.subscribe_internal(entity, move |entity, event, cx| {
|
||||||
on_event(entity, event, cx);
|
on_event(entity, event, cx);
|
||||||
|
@ -366,27 +368,32 @@ impl AppContext {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn subscribe_internal<T, E>(
|
pub(crate) fn subscribe_internal<T, E, Evt>(
|
||||||
&mut self,
|
&mut self,
|
||||||
entity: &E,
|
entity: &E,
|
||||||
mut on_event: impl FnMut(E, &T::Event, &mut AppContext) -> bool + 'static,
|
mut on_event: impl FnMut(E, &Evt, &mut AppContext) -> bool + 'static,
|
||||||
) -> Subscription
|
) -> Subscription
|
||||||
where
|
where
|
||||||
T: 'static + EventEmitter,
|
T: 'static + EventEmitter<Evt>,
|
||||||
E: Entity<T>,
|
E: Entity<T>,
|
||||||
|
Evt: 'static,
|
||||||
{
|
{
|
||||||
let entity_id = entity.entity_id();
|
let entity_id = entity.entity_id();
|
||||||
let entity = entity.downgrade();
|
let entity = entity.downgrade();
|
||||||
|
|
||||||
self.event_listeners.insert(
|
self.event_listeners.insert(
|
||||||
entity_id,
|
entity_id,
|
||||||
Box::new(move |event, cx| {
|
(
|
||||||
let event: &T::Event = event.downcast_ref().expect("invalid event type");
|
TypeId::of::<Evt>(),
|
||||||
if let Some(handle) = E::upgrade_from(&entity) {
|
Box::new(move |event, cx| {
|
||||||
on_event(handle, event, cx)
|
let event: &Evt = event.downcast_ref().expect("invalid event type");
|
||||||
} else {
|
if let Some(handle) = E::upgrade_from(&entity) {
|
||||||
false
|
on_event(handle, event, cx)
|
||||||
}
|
} else {
|
||||||
}),
|
false
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -509,7 +516,11 @@ impl AppContext {
|
||||||
Effect::Notify { emitter } => {
|
Effect::Notify { emitter } => {
|
||||||
self.apply_notify_effect(emitter);
|
self.apply_notify_effect(emitter);
|
||||||
}
|
}
|
||||||
Effect::Emit { emitter, event } => self.apply_emit_effect(emitter, event),
|
Effect::Emit {
|
||||||
|
emitter,
|
||||||
|
event_type,
|
||||||
|
event,
|
||||||
|
} => self.apply_emit_effect(emitter, event_type, event),
|
||||||
Effect::FocusChanged {
|
Effect::FocusChanged {
|
||||||
window_handle,
|
window_handle,
|
||||||
focused,
|
focused,
|
||||||
|
@ -604,10 +615,16 @@ impl AppContext {
|
||||||
.retain(&emitter, |handler| handler(self));
|
.retain(&emitter, |handler| handler(self));
|
||||||
}
|
}
|
||||||
|
|
||||||
fn apply_emit_effect(&mut self, emitter: EntityId, event: Box<dyn Any>) {
|
fn apply_emit_effect(&mut self, emitter: EntityId, event_type: TypeId, event: Box<dyn Any>) {
|
||||||
self.event_listeners
|
self.event_listeners
|
||||||
.clone()
|
.clone()
|
||||||
.retain(&emitter, |handler| handler(event.as_ref(), self));
|
.retain(&emitter, |(stored_type, handler)| {
|
||||||
|
if *stored_type == event_type {
|
||||||
|
handler(event.as_ref(), self)
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn apply_focus_changed_effect(
|
fn apply_focus_changed_effect(
|
||||||
|
@ -978,6 +995,7 @@ pub(crate) enum Effect {
|
||||||
},
|
},
|
||||||
Emit {
|
Emit {
|
||||||
emitter: EntityId,
|
emitter: EntityId,
|
||||||
|
event_type: TypeId,
|
||||||
event: Box<dyn Any>,
|
event: Box<dyn Any>,
|
||||||
},
|
},
|
||||||
FocusChanged {
|
FocusChanged {
|
||||||
|
|
|
@ -59,15 +59,16 @@ impl<'a, T: 'static> ModelContext<'a, T> {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn subscribe<T2, E>(
|
pub fn subscribe<T2, E, Evt>(
|
||||||
&mut self,
|
&mut self,
|
||||||
entity: &E,
|
entity: &E,
|
||||||
mut on_event: impl FnMut(&mut T, E, &T2::Event, &mut ModelContext<'_, T>) + 'static,
|
mut on_event: impl FnMut(&mut T, E, &Evt, &mut ModelContext<'_, T>) + 'static,
|
||||||
) -> Subscription
|
) -> Subscription
|
||||||
where
|
where
|
||||||
T: 'static,
|
T: 'static,
|
||||||
T2: 'static + EventEmitter,
|
T2: 'static + EventEmitter<Evt>,
|
||||||
E: Entity<T2>,
|
E: Entity<T2>,
|
||||||
|
Evt: 'static,
|
||||||
{
|
{
|
||||||
let this = self.weak_model();
|
let this = self.weak_model();
|
||||||
self.app.subscribe_internal(entity, move |e, event, cx| {
|
self.app.subscribe_internal(entity, move |e, event, cx| {
|
||||||
|
@ -189,13 +190,15 @@ impl<'a, T: 'static> ModelContext<'a, T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, T> ModelContext<'a, T>
|
impl<'a, T> ModelContext<'a, T> {
|
||||||
where
|
pub fn emit<Evt>(&mut self, event: Evt)
|
||||||
T: EventEmitter,
|
where
|
||||||
{
|
T: EventEmitter<Evt>,
|
||||||
pub fn emit(&mut self, event: T::Event) {
|
Evt: 'static,
|
||||||
|
{
|
||||||
self.app.pending_effects.push_back(Effect::Emit {
|
self.app.pending_effects.push_back(Effect::Emit {
|
||||||
emitter: self.model_state.entity_id,
|
emitter: self.model_state.entity_id,
|
||||||
|
event_type: TypeId::of::<Evt>(),
|
||||||
event: Box::new(event),
|
event: Box::new(event),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -197,12 +197,12 @@ impl TestAppContext {
|
||||||
rx
|
rx
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn events<T: 'static + EventEmitter>(
|
pub fn events<Evt, T: 'static + EventEmitter<Evt>>(
|
||||||
&mut self,
|
&mut self,
|
||||||
entity: &Model<T>,
|
entity: &Model<T>,
|
||||||
) -> futures::channel::mpsc::UnboundedReceiver<T::Event>
|
) -> futures::channel::mpsc::UnboundedReceiver<Evt>
|
||||||
where
|
where
|
||||||
T::Event: 'static + Clone,
|
Evt: 'static + Clone,
|
||||||
{
|
{
|
||||||
let (tx, rx) = futures::channel::mpsc::unbounded();
|
let (tx, rx) = futures::channel::mpsc::unbounded();
|
||||||
entity
|
entity
|
||||||
|
@ -240,10 +240,11 @@ impl TestAppContext {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Send + EventEmitter> Model<T> {
|
impl<T: Send> Model<T> {
|
||||||
pub fn next_event(&self, cx: &mut TestAppContext) -> T::Event
|
pub fn next_event<Evt>(&self, cx: &mut TestAppContext) -> Evt
|
||||||
where
|
where
|
||||||
T::Event: Send + Clone,
|
Evt: Send + Clone + 'static,
|
||||||
|
T: EventEmitter<Evt>,
|
||||||
{
|
{
|
||||||
let (tx, mut rx) = futures::channel::mpsc::unbounded();
|
let (tx, mut rx) = futures::channel::mpsc::unbounded();
|
||||||
let _subscription = self.update(cx, |_, cx| {
|
let _subscription = self.update(cx, |_, cx| {
|
||||||
|
|
|
@ -138,6 +138,8 @@ pub trait Entity<T>: Sealed {
|
||||||
Self: Sized;
|
Self: Sized;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait EventEmitter<E: Any>: 'static {}
|
||||||
|
|
||||||
pub enum GlobalKey {
|
pub enum GlobalKey {
|
||||||
Numeric(usize),
|
Numeric(usize),
|
||||||
View(EntityId),
|
View(EntityId),
|
||||||
|
@ -171,10 +173,6 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait EventEmitter: 'static {
|
|
||||||
type Event: Any;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait Flatten<T> {
|
pub trait Flatten<T> {
|
||||||
fn flatten(self) -> Result<T>;
|
fn flatten(self) -> Result<T>;
|
||||||
}
|
}
|
||||||
|
|
|
@ -75,6 +75,8 @@ where
|
||||||
.flatten()
|
.flatten()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Call the given callback for each subscriber to the given emitter.
|
||||||
|
/// If the callback returns false, the subscriber is removed.
|
||||||
pub fn retain<F>(&self, emitter: &EmitterKey, mut f: F)
|
pub fn retain<F>(&self, emitter: &EmitterKey, mut f: F)
|
||||||
where
|
where
|
||||||
F: FnMut(&mut Callback) -> bool,
|
F: FnMut(&mut Callback) -> bool,
|
||||||
|
|
|
@ -439,33 +439,37 @@ impl<'a> WindowContext<'a> {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn subscribe<Emitter, E>(
|
pub fn subscribe<Emitter, E, Evt>(
|
||||||
&mut self,
|
&mut self,
|
||||||
entity: &E,
|
entity: &E,
|
||||||
mut on_event: impl FnMut(E, &Emitter::Event, &mut WindowContext<'_>) + 'static,
|
mut on_event: impl FnMut(E, &Evt, &mut WindowContext<'_>) + 'static,
|
||||||
) -> Subscription
|
) -> Subscription
|
||||||
where
|
where
|
||||||
Emitter: EventEmitter,
|
Emitter: EventEmitter<Evt>,
|
||||||
E: Entity<Emitter>,
|
E: Entity<Emitter>,
|
||||||
|
Evt: 'static,
|
||||||
{
|
{
|
||||||
let entity_id = entity.entity_id();
|
let entity_id = entity.entity_id();
|
||||||
let entity = entity.downgrade();
|
let entity = entity.downgrade();
|
||||||
let window_handle = self.window.handle;
|
let window_handle = self.window.handle;
|
||||||
self.app.event_listeners.insert(
|
self.app.event_listeners.insert(
|
||||||
entity_id,
|
entity_id,
|
||||||
Box::new(move |event, cx| {
|
(
|
||||||
window_handle
|
TypeId::of::<Evt>(),
|
||||||
.update(cx, |_, cx| {
|
Box::new(move |event, cx| {
|
||||||
if let Some(handle) = E::upgrade_from(&entity) {
|
window_handle
|
||||||
let event = event.downcast_ref().expect("invalid event type");
|
.update(cx, |_, cx| {
|
||||||
on_event(handle, event, cx);
|
if let Some(handle) = E::upgrade_from(&entity) {
|
||||||
true
|
let event = event.downcast_ref().expect("invalid event type");
|
||||||
} else {
|
on_event(handle, event, cx);
|
||||||
false
|
true
|
||||||
}
|
} else {
|
||||||
})
|
false
|
||||||
.unwrap_or(false)
|
}
|
||||||
}),
|
})
|
||||||
|
.unwrap_or(false)
|
||||||
|
}),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1809,14 +1813,33 @@ impl<'a, V: 'static> ViewContext<'a, V> {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn subscribe<V2, E>(
|
// Options for simplifying this new event API:
|
||||||
|
//
|
||||||
|
// - Make a new stlye of API which does partial application of the arguments to capture
|
||||||
|
// the types involved e.g.
|
||||||
|
// `cx.for_entity(handle).subscribe::<ItemEvents>(..)`
|
||||||
|
//
|
||||||
|
// - Make it so there are less types:
|
||||||
|
// - Bail on this idea all together, go back to associated types.
|
||||||
|
// causes our event enums to be a blob of anything that could happen ever, and
|
||||||
|
// makes applications have some translation boilerplate
|
||||||
|
//
|
||||||
|
// - Move some of the types into the method names,
|
||||||
|
// `cx.subscribe_model::<_, ItemEvents>(handle)`
|
||||||
|
//
|
||||||
|
// - Do something drastic like removing views and models, or removing the multiple
|
||||||
|
// kind of contexts. (Not going to happen, we already tried this before.)
|
||||||
|
//
|
||||||
|
// - Accept it, and use `cx.subscribe::<_, _, ItemEvents>(handle, ...)`
|
||||||
|
pub fn subscribe<V2, E, Evt>(
|
||||||
&mut self,
|
&mut self,
|
||||||
entity: &E,
|
entity: &E,
|
||||||
mut on_event: impl FnMut(&mut V, E, &V2::Event, &mut ViewContext<'_, V>) + 'static,
|
mut on_event: impl FnMut(&mut V, E, &Evt, &mut ViewContext<'_, V>) + 'static,
|
||||||
) -> Subscription
|
) -> Subscription
|
||||||
where
|
where
|
||||||
V2: EventEmitter,
|
V2: EventEmitter<Evt>,
|
||||||
E: Entity<V2>,
|
E: Entity<V2>,
|
||||||
|
Evt: 'static,
|
||||||
{
|
{
|
||||||
let view = self.view().downgrade();
|
let view = self.view().downgrade();
|
||||||
let entity_id = entity.entity_id();
|
let entity_id = entity.entity_id();
|
||||||
|
@ -1824,19 +1847,22 @@ impl<'a, V: 'static> ViewContext<'a, V> {
|
||||||
let window_handle = self.window.handle;
|
let window_handle = self.window.handle;
|
||||||
self.app.event_listeners.insert(
|
self.app.event_listeners.insert(
|
||||||
entity_id,
|
entity_id,
|
||||||
Box::new(move |event, cx| {
|
(
|
||||||
window_handle
|
TypeId::of::<Evt>(),
|
||||||
.update(cx, |_, cx| {
|
Box::new(move |event, cx| {
|
||||||
if let Some(handle) = E::upgrade_from(&handle) {
|
window_handle
|
||||||
let event = event.downcast_ref().expect("invalid event type");
|
.update(cx, |_, cx| {
|
||||||
view.update(cx, |this, cx| on_event(this, handle, event, cx))
|
if let Some(handle) = E::upgrade_from(&handle) {
|
||||||
.is_ok()
|
let event = event.downcast_ref().expect("invalid event type");
|
||||||
} else {
|
view.update(cx, |this, cx| on_event(this, handle, event, cx))
|
||||||
false
|
.is_ok()
|
||||||
}
|
} else {
|
||||||
})
|
false
|
||||||
.unwrap_or(false)
|
}
|
||||||
}),
|
})
|
||||||
|
.unwrap_or(false)
|
||||||
|
}),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2181,15 +2207,16 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<V> ViewContext<'_, V>
|
impl<V> ViewContext<'_, V> {
|
||||||
where
|
pub fn emit<Evt>(&mut self, event: Evt)
|
||||||
V: EventEmitter,
|
where
|
||||||
V::Event: 'static,
|
Evt: 'static,
|
||||||
{
|
V: EventEmitter<Evt>,
|
||||||
pub fn emit(&mut self, event: V::Event) {
|
{
|
||||||
let emitter = self.view.model.entity_id;
|
let emitter = self.view.model.entity_id;
|
||||||
self.app.push_effect(Effect::Emit {
|
self.app.push_effect(Effect::Emit {
|
||||||
emitter,
|
emitter,
|
||||||
|
event_type: TypeId::of::<Evt>(),
|
||||||
event: Box::new(event),
|
event: Box::new(event),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue