File Coverage

blib/lib/Mojo/EventEmitter.pm
Criterion Covered Total %
statement 39 39 100.0
branch 10 12 83.3
condition 3 4 75.0
subroutine 11 11 100.0
pod 7 7 100.0
total 70 73 95.8


line stmt bran cond sub pod time code
1             package Mojo::EventEmitter;
2 77     77   77384 use Mojo::Base -base;
  77         147  
  77         586  
3              
4 77     77   510 use Scalar::Util qw(blessed weaken);
  77         202  
  77         8029  
5              
6 77   50 77   417 use constant DEBUG => $ENV{MOJO_EVENTEMITTER_DEBUG} || 0;
  77         138  
  77         62122  
7              
8 111 50   111 1 562 sub catch { $_[0]->on(error => $_[1]) and return $_[0] }
9              
10             sub emit {
11 55634     55634 1 85146 my ($self, $name) = (shift, shift);
12              
13 55634 100       98234 if (my $s = $self->{events}{$name}) {
14 21259         20711 warn "-- Emit $name in @{[blessed $self]} (@{[scalar @$s]})\n" if DEBUG;
15 21259         30026 for my $cb (@$s) { $self->$cb(@_) }
  21757         43372  
16             }
17             else {
18 34375         33745 warn "-- Emit $name in @{[blessed $self]} (0)\n" if DEBUG;
19 34375 100       52984 die "@{[blessed $self]}: $_[0]" if $name eq 'error';
  1         10  
20             }
21              
22 55629         158163 return $self;
23             }
24              
25 4026     4026 1 17032 sub has_subscribers { !!shift->{events}{shift()} }
26              
27 22849 50   22849 1 32895 sub on { push @{$_[0]{events}{$_[1]}}, $_[2] and return $_[2] }
  22849         83751  
28              
29             sub once {
30 9621     9621 1 22927 my ($self, $name, $cb) = @_;
31              
32 9621         11568 weaken $self;
33             my $wrapper = sub {
34 9535     9535   21917 $self->unsubscribe($name => __SUB__);
35 9535         18730 $cb->(@_);
36 9621         31265 };
37 9621         19665 $self->on($name => $wrapper);
38              
39 9621         15609 return $wrapper;
40             }
41              
42 8214   100 8214 1 29949 sub subscribers { shift->{events}{shift()} //= [] }
43              
44             sub unsubscribe {
45 9683     9683 1 14372 my ($self, $name, $cb) = @_;
46              
47             # One
48 9683 100       12705 if ($cb) {
49 9599         10357 $self->{events}{$name} = [grep { $cb ne $_ } @{$self->{events}{$name}}];
  9760         27869  
  9599         16713  
50 9599 100       11215 delete $self->{events}{$name} unless @{$self->{events}{$name}};
  9599         21561  
51             }
52              
53             # All
54 84         330 else { delete $self->{events}{$name} }
55              
56 9683         12952 return $self;
57             }
58              
59             1;
60              
61             =encoding utf8
62              
63             =head1 NAME
64              
65             Mojo::EventEmitter - Event emitter base class
66              
67             =head1 SYNOPSIS
68              
69             package Cat;
70             use Mojo::Base 'Mojo::EventEmitter', -signatures;
71              
72             # Emit events
73             sub poke ($self) { $self->emit(roar => 3) }
74              
75             package main;
76              
77             # Subscribe to events
78             my $tiger = Cat->new;
79             $tiger->on(roar => sub ($tiger, $times) { say 'RAWR!' for 1 .. $times });
80             $tiger->poke;
81              
82             =head1 DESCRIPTION
83              
84             L is a simple base class for event emitting objects.
85              
86             =head1 EVENTS
87              
88             L can emit the following events.
89              
90             =head2 error
91              
92             $e->on(error => sub ($e, $err) {...});
93              
94             This is a special event for errors, it will not be emitted directly by this class, but is fatal if unhandled.
95             Subclasses may choose to emit it, but are not required to do so.
96              
97             $e->on(error => sub ($e, $err) { say "This looks bad: $err" });
98              
99             =head1 METHODS
100              
101             L inherits all methods from L and implements the following new ones.
102              
103             =head2 catch
104              
105             $e = $e->catch(sub {...});
106              
107             Subscribe to L event.
108              
109             # Longer version
110             $e->on(error => sub {...});
111              
112             =head2 emit
113              
114             $e = $e->emit('foo');
115             $e = $e->emit('foo', 123);
116              
117             Emit event.
118              
119             =head2 has_subscribers
120              
121             my $bool = $e->has_subscribers('foo');
122              
123             Check if event has subscribers.
124              
125             =head2 on
126              
127             my $cb = $e->on(foo => sub {...});
128              
129             Subscribe to event.
130              
131             $e->on(foo => sub ($e, @args) {...});
132              
133             =head2 once
134              
135             my $cb = $e->once(foo => sub {...});
136              
137             Subscribe to event and unsubscribe again after it has been emitted once.
138              
139             $e->once(foo => sub ($e, @args) {...});
140              
141             =head2 subscribers
142              
143             my $subscribers = $e->subscribers('foo');
144              
145             All subscribers for event.
146              
147             # Unsubscribe last subscriber
148             $e->unsubscribe(foo => $e->subscribers('foo')->[-1]);
149              
150             # Change order of subscribers
151             @{$e->subscribers('foo')} = reverse @{$e->subscribers('foo')};
152              
153             =head2 unsubscribe
154              
155             $e = $e->unsubscribe('foo');
156             $e = $e->unsubscribe(foo => $cb);
157              
158             Unsubscribe from event.
159              
160             =head1 DEBUGGING
161              
162             You can set the C environment variable to get some advanced diagnostics information printed to
163             C.
164              
165             MOJO_EVENTEMITTER_DEBUG=1
166              
167             =head1 SEE ALSO
168              
169             L, L, L.
170              
171             =cut