1 | #! perl # mandatory depends=login |
1 | #! perl # mandatory depends=login |
2 | |
2 | |
3 | # sends the following ext message types |
3 | # sends the following ext message types |
|
|
4 | # ws_a id name... # associate well-known widget with given id |
4 | # ws_n id # widgetset new |
5 | # ws_n ws # widgetset new |
5 | # ws_d id # widgetset destroy |
6 | # ws_d ws # widgetset destroy |
6 | # ws_c ws id class args # widgetset create |
7 | # ws_c ws id class @args # widgetset create |
7 | # w_c id [rid] name args # widget method call |
8 | # w_c id rid name @args # widget method call |
8 | # w_s id name value # widget member set |
9 | # w_s id @attr # widget member set |
9 | # w_g id rid name # widget member get |
10 | # w_g id rid @attr # widget member get |
10 | # |
11 | # |
11 | # and expects the following exti message types |
12 | # and expects the following exti message types |
12 | # w_r rid res # widget call return |
|
|
13 | # w_e id name args # widget_event |
13 | # w_e id @args # widget_call |
|
|
14 | |
|
|
15 | our $DEBUG = 1; |
14 | |
16 | |
15 | cf::client->attach ( |
17 | cf::client->attach ( |
16 | on_connect => sub { |
18 | on_connect => sub { |
17 | my ($ns) = @_; |
19 | my ($ns) = @_; |
18 | |
20 | |
19 | Scalar::Util::weaken (my $weakns = $ns); |
21 | Scalar::Util::weaken (my $weakns = $ns); |
20 | |
22 | |
21 | $ns->{id} = "a"; |
23 | $ns->{id} = "a"; |
22 | $ns->{json_coder}->filter_json_single_key_object (__widget_ref__ => sub { |
24 | $ns->{json_coder}->filter_json_single_key_object ("\fw" => sub { |
23 | # cannot deserialise ATM |
25 | $weakns->{widget}{$_[0]} |
24 | undef |
|
|
25 | }); |
26 | }); |
26 | }, |
27 | }, |
27 | ); |
28 | ); |
28 | |
29 | |
29 | sub csc_update_stats { |
30 | sub csc_update_stats { |
30 | my ($ns) = @_; |
31 | my ($ns) = @_; |
31 | |
32 | |
32 | while (my ($k, $v) = each %{ $ns->{csc}{stat} }) { |
33 | while (my ($k, $v) = each %{ $ns->{csc}{stat} }) { |
33 | $v->set_text ($ns->pl->ob->stats->$k); |
34 | $v->set_text ($ns->pl->ob->stats->$k); |
34 | } |
35 | } |
|
|
36 | } |
|
|
37 | |
|
|
38 | sub demo_start { |
|
|
39 | my ($ns) = @_; |
|
|
40 | |
|
|
41 | my $ws = $ns->{csc} = $ns->new_widgetset; |
|
|
42 | |
|
|
43 | $ws->{tab} = $ws->new (Label => text => "dumb tst", c_tab => ["hull"]); |
|
|
44 | |
|
|
45 | $ws->find ("setup_notebook")->add ($ws->{tab}); |
|
|
46 | $ws->find ("setup_dialog")->toggle_visibility; |
35 | } |
47 | } |
36 | |
48 | |
37 | sub csc_start { |
49 | sub csc_start { |
38 | my ($ns) = @_; |
50 | my ($ns) = @_; |
39 | |
51 | |
… | |
… | |
51 | }, |
63 | }, |
52 | ); |
64 | ); |
53 | |
65 | |
54 | $w->add (my $ntb = $ws->new (Notebook => expand => 1)); |
66 | $w->add (my $ntb = $ws->new (Notebook => expand => 1)); |
55 | |
67 | |
56 | $ntb->add (Statistics => (my $stats = $ws->new (Table => expand => 1)), "Basic statistics of your new character"); |
68 | $ntb->add_tab (Statistics => (my $stats = $ws->new (Table => expand => 1)), "Basic statistics of your new character"); |
57 | |
69 | |
58 | $stats->add (0, 0, (my $statstable = $ws->new ("Table"))); |
70 | $stats->add_at (0, 0, (my $statstable = $ws->new ("Table"))); |
59 | |
71 | |
60 | for ( |
72 | for ( |
61 | [0, "Str"], |
73 | [0, "Str"], |
62 | [1, "Dex"], |
74 | [1, "Dex"], |
63 | [2, "Con"], |
75 | [2, "Con"], |
… | |
… | |
66 | [5, "Pow"], |
78 | [5, "Pow"], |
67 | [6, "Cha"], |
79 | [6, "Cha"], |
68 | ) { |
80 | ) { |
69 | my ($x, $label) = @$_; |
81 | my ($x, $label) = @$_; |
70 | |
82 | |
71 | $statstable->add ($x, 0, $ws->new (Label => |
83 | $statstable->add_at ($x, 0, $ws->new (Label => |
72 | can_hover => 1, can_events => 1, |
84 | can_hover => 1, can_events => 1, |
73 | align => +1, text => $label, tooltip => "#stat_$label", |
85 | align => +1, text => $label, tooltip => "#stat_$label", |
74 | )); |
86 | )); |
75 | $statstable->add ($x, 1, $ws->{stat}{$label} = $ws->new (Label => |
87 | $statstable->add_at ($x, 1, $ws->{stat}{$label} = $ws->new (Label => |
76 | can_hover => 1, can_events => 1, |
88 | can_hover => 1, can_events => 1, |
77 | align => +1, template => "88", tooltip => "#stat_$label", |
89 | align => +1, template => "88", tooltip => "#stat_$label", |
78 | )); |
90 | )); |
79 | } |
91 | } |
80 | |
92 | |
81 | csc_update_stats $ns; |
93 | csc_update_stats $ns; |
82 | |
94 | |
|
|
95 | $ws->{tl} = $w; |
83 | $w->show; |
96 | $w->show; |
84 | } |
97 | } |
85 | |
98 | |
86 | cf::player->attach ( |
99 | cf::player->attach ( |
87 | on_login => sub { |
100 | on_login => sub { |
… | |
… | |
90 | return unless $cf::CFG{devel}; |
103 | return unless $cf::CFG{devel}; |
91 | |
104 | |
92 | my $ns = $pl->ns; |
105 | my $ns = $pl->ns; |
93 | |
106 | |
94 | return unless $ns->{can_widget}; |
107 | return unless $ns->{can_widget}; |
95 | |
|
|
96 | csc_start $ns; |
108 | #csc_start $ns; |
|
|
109 | #demo_start $ns; |
97 | }, |
110 | }, |
98 | ); |
111 | ); |
99 | |
112 | |
100 | cf::register_exticmd w_e => sub { |
113 | cf::register_exticmd w_e => sub { |
101 | my ($ns, $pkt) = @_; |
114 | my ($ns, $id, @args) = @_; |
102 | |
115 | |
103 | if (my $w = $ns->{widget}{$pkt->{id}}) { |
116 | if (my $cb = $ns->{widget_cb}{$id}) { |
104 | if (my $cb = $w->{ev}{$pkt->{name}}) { |
117 | $cb->(@args); |
105 | $_->($w, @{ $pkt->{args} || [] }) |
|
|
106 | for @$cb; |
|
|
107 | } |
|
|
108 | } |
|
|
109 | |
|
|
110 | () |
|
|
111 | }; |
|
|
112 | |
|
|
113 | cf::register_exticmd w_r => sub { |
|
|
114 | my ($ns, $pkt) = @_; |
|
|
115 | |
|
|
116 | if (my $cb = delete $ns->{widget_return}{$pkt->{rid}}) { |
|
|
117 | $cb->(@{$pkt->{res} || [] }); |
|
|
118 | } |
118 | } |
119 | |
119 | |
120 | () |
120 | () |
121 | }; |
121 | }; |
122 | |
122 | |
… | |
… | |
126 | my $id = ++$self->{id}; |
126 | my $id = ++$self->{id}; |
127 | |
127 | |
128 | my $ws = bless { |
128 | my $ws = bless { |
129 | id => $id, |
129 | id => $id, |
130 | ns => $self, |
130 | ns => $self, |
131 | w => {}, |
131 | _w => {}, |
132 | }, "ext::widget::set"; |
132 | }, "ext::widget::set"; |
133 | |
133 | |
134 | $ws->msg (ws_n => id => $id); |
134 | $ws->msg (ws_n => $id); |
135 | |
135 | |
136 | $ws |
136 | $ws |
|
|
137 | } |
|
|
138 | |
|
|
139 | sub cf::client::alloc_wid { |
|
|
140 | pop @{ $_[0]{ids} } |
|
|
141 | or ++$_[0]{id} |
|
|
142 | } |
|
|
143 | |
|
|
144 | sub cf::client::free_wid { |
|
|
145 | push @{ $_[0]{ids} }, $_[1]; |
137 | } |
146 | } |
138 | |
147 | |
139 | ############################################################################# |
148 | ############################################################################# |
140 | |
149 | |
141 | package ext::widget::set; |
150 | package ext::widget::set; |
… | |
… | |
145 | } |
154 | } |
146 | |
155 | |
147 | sub destroy { |
156 | sub destroy { |
148 | my ($self) = @_; |
157 | my ($self) = @_; |
149 | |
158 | |
150 | $self->msg (ws_d => id => $self->{id}); |
159 | $self->msg (ws_d => $self->{id}); |
151 | delete $self->{ns}; |
160 | delete $self->{ns}; |
152 | $_->destroy |
161 | $_->destroy |
153 | for values %{ $self->{w} }; |
162 | for values %{ $self->{w} }; |
154 | } |
163 | } |
155 | |
164 | |
156 | sub msg { |
165 | sub msg { |
157 | my ($self, $type, %msg) = @_; |
166 | my ($self, @msg) = @_; |
158 | |
167 | |
159 | if (my $ns = shift->{ns}) { |
168 | if (my $ns = shift->{ns}) { |
160 | $msg{msgtype} = $type; |
169 | warn "msg " . $ns->{json_coder}->encode (\@msg) if $DEBUG;#d# |
161 | $ns->send_packet ("ext " . $ns->{json_coder}->encode (\%msg)); |
170 | $ns->send_packet ("ext " . $ns->{json_coder}->encode (\@msg)); |
162 | } |
171 | } |
|
|
172 | } |
|
|
173 | |
|
|
174 | sub alloc { |
|
|
175 | my ($self) = @_; |
|
|
176 | |
|
|
177 | my $id = $self->{ns}->alloc_wid; |
|
|
178 | |
|
|
179 | my $proxy = bless { |
|
|
180 | id => $id, |
|
|
181 | }, "ext::widget::proxy"; |
|
|
182 | |
|
|
183 | Scalar::Util::weaken ($proxy->{ns} = $self->{ns}); |
|
|
184 | Scalar::Util::weaken ($self->{ns}{widget}{$id} = $proxy); |
|
|
185 | |
|
|
186 | $proxy |
163 | } |
187 | } |
164 | |
188 | |
165 | sub new { |
189 | sub new { |
166 | my ($self, $class, %args) = @_; |
190 | my ($self, $class, %args) = @_; |
167 | |
191 | |
168 | my $id = ++$self->{ns}{id}; |
192 | my $proxy = $self->alloc; |
169 | |
193 | |
170 | my $proxy = $self->{w}{$id} = bless { |
194 | Scalar::Util::weaken ($self->{_w}{$proxy->{id}} = $proxy); |
171 | id => $id, |
|
|
172 | }, "ext::widget::proxy"; |
|
|
173 | |
|
|
174 | Scalar::Util::weaken ($proxy->{ws} = $self); |
195 | Scalar::Util::weaken ($proxy->{ws} = $self); |
175 | Scalar::Util::weaken ($proxy->{ns} = $self->{ns}); |
|
|
176 | Scalar::Util::weaken ($self->{ns}{widget}{$id} = $proxy); |
|
|
177 | |
196 | |
178 | for my $ev (grep /^on_/, keys %args) { |
197 | for my $ev (grep /^on_/, keys %args) { |
179 | push @{$proxy->{ev}{$ev}}, $args{$ev}; |
198 | $args{$ev} = $proxy->{"_$ev"} = $proxy->cb ($args{$ev}); |
180 | $args{$ev} = 0; |
|
|
181 | } |
199 | } |
182 | |
200 | |
183 | $self->msg (ws_c => |
201 | $self->msg (ws_c => |
184 | ws => $self->{w}{id}, |
202 | $self->{id}, |
185 | id => $id, |
203 | $proxy->{id}, |
186 | class => $class, |
204 | $class, |
187 | args => \%args, |
205 | \%args, |
188 | ); |
206 | ); |
189 | |
207 | |
190 | $proxy |
208 | $proxy |
191 | } |
209 | } |
192 | |
210 | |
|
|
211 | sub find { |
|
|
212 | my ($self, @names) = @_; |
|
|
213 | |
|
|
214 | my @res; |
|
|
215 | my @alloc; |
|
|
216 | |
|
|
217 | for my $name (@names) { |
|
|
218 | push @res, $self->{ns}{widget_wkw}{$name} ||= do { |
|
|
219 | my $proxy = $self->alloc; |
|
|
220 | |
|
|
221 | push @alloc, $proxy->{id} => $name; |
|
|
222 | |
|
|
223 | $proxy |
|
|
224 | }; |
|
|
225 | } |
|
|
226 | |
|
|
227 | $self->msg (ws_a => @alloc) |
|
|
228 | if @alloc; |
|
|
229 | |
|
|
230 | wantarray ? @res : $res[0] |
|
|
231 | } |
|
|
232 | |
193 | ############################################################################# |
233 | ############################################################################# |
194 | |
234 | |
195 | package ext::widget::proxy; |
235 | package ext::widget::proxy; |
196 | |
236 | |
197 | sub DESTROY { |
237 | sub DESTROY { |
198 | my ($self) = @_; |
238 | my ($self) = @_; |
199 | |
239 | |
200 | delete $self->{ns}{widget}{$self->{id}}; |
240 | delete $self->{ns}{widget}{$self->{id}}; |
201 | |
241 | |
202 | if (my $ws = $self->{ws}) { |
242 | if (my $ws = $self->{ws}) { |
|
|
243 | $self->msg (w_c => 0, "destroy"); |
203 | delete $ws->{w}{$self->{id}}; |
244 | delete $ws->{_w}{$self->{id}}; |
204 | $self->msg (w_c => name => "destroy"); |
|
|
205 | } |
245 | } |
|
|
246 | } |
|
|
247 | |
|
|
248 | sub cb { |
|
|
249 | my ($self, $cb) = @_; |
|
|
250 | |
|
|
251 | my $proxy = bless { |
|
|
252 | ns => $self->{ns}, |
|
|
253 | id => $self->{ns}->alloc_wid, |
|
|
254 | }, "ext::widget::callback"; |
|
|
255 | |
|
|
256 | Scalar::Util::weaken $proxy->{ns}; |
|
|
257 | |
|
|
258 | $self->{ns}{widget_cb}{$proxy->{id}} = $cb; |
|
|
259 | |
|
|
260 | $proxy |
|
|
261 | } |
|
|
262 | |
|
|
263 | sub oneshot_cb { |
|
|
264 | my ($self, $cb) = @_; |
|
|
265 | |
|
|
266 | if ("CODE" eq ref $cb) { |
|
|
267 | my $ocb = $cb; |
|
|
268 | $cb = $self->cb (sub { |
|
|
269 | undef $cb; |
|
|
270 | &$ocb |
|
|
271 | }); |
|
|
272 | } |
|
|
273 | |
|
|
274 | $cb |
206 | } |
275 | } |
207 | |
276 | |
208 | sub msg { |
277 | sub msg { |
209 | my ($self, $type, %msg) = @_; |
278 | my ($self, $type, @arg) = @_; |
|
|
279 | |
|
|
280 | if (my $ns = $self->{ns}) { |
|
|
281 | my @msg = ($type, $self->{id}, @arg); |
|
|
282 | warn "MSG " . $ns->{json_coder}->encode (\@msg) if $DEBUG;#d# |
|
|
283 | $ns->send_packet ("ext " . $ns->{json_coder}->encode (\@msg)); |
|
|
284 | } |
|
|
285 | } |
|
|
286 | |
|
|
287 | sub msg_cb { |
|
|
288 | my ($self, $cb, $type, @arg) = @_; |
210 | |
289 | |
211 | if (my $ws = $self->{ws}) { |
290 | if (my $ws = $self->{ws}) { |
212 | $ws->msg ($type, |
|
|
213 | %msg, |
|
|
214 | id => $self->{id}, |
|
|
215 | ); |
|
|
216 | } |
|
|
217 | } |
|
|
218 | |
|
|
219 | sub msg_cb { |
|
|
220 | my ($self, $cb, $type, %msg) = @_; |
|
|
221 | |
|
|
222 | if (my $ws = $self->{ws}) { |
|
|
223 | |
|
|
224 | my $rid = ++$ws->{ns}{id}; |
291 | my $rid = $ws->{ns}->alloc_wid; |
225 | |
|
|
226 | $self->msg ($type, %msg, rid => $rid); |
|
|
227 | |
292 | |
228 | if ($cb) { |
293 | if ($cb) { |
229 | $ws->{ns}{widget_return}{$rid} = $cb; |
294 | $ws->{ns}{widget_cb}{$rid} = sub { |
|
|
295 | delete $ws->{ns}{widget_cb}{$rid}; |
|
|
296 | $ws->{ns}->free_wid ($rid); |
|
|
297 | &$cb |
|
|
298 | }; |
|
|
299 | |
|
|
300 | $self->msg ($type, $rid, @arg); |
230 | } else { |
301 | } else { |
231 | # synchronous case |
302 | # synchronous case |
232 | my $wait = new Coro::Signal; |
303 | my $wait = new Coro::Signal; |
233 | my @res; |
304 | my @res; |
234 | |
305 | |
235 | $ws->{ns}{widget_return}{$rid} = sub { |
306 | $ws->{ns}{widget_cb}{$rid} = sub { |
|
|
307 | delete $ws->{ns}{widget_cb}{$rid}; |
|
|
308 | $ws->{ns}->free_wid ($rid); |
|
|
309 | |
236 | @res = @_; |
310 | @res = @_; |
237 | $wait->send; |
311 | $wait->send; |
238 | }; |
312 | }; |
|
|
313 | $self->msg ($type, $rid, @arg); |
239 | $wait->wait; |
314 | $wait->wait; |
240 | |
315 | |
241 | return @res; |
316 | return @res; |
242 | } |
317 | } |
243 | } |
318 | } |
244 | |
319 | |
245 | () |
320 | () |
246 | } |
321 | } |
247 | |
322 | |
248 | sub set { |
323 | sub set { |
249 | my ($self, $member, $value) = @_; |
324 | my ($self, @kv) = @_; |
250 | |
325 | |
251 | $self->msg (w_s => name => $member, value => $value); |
326 | $self->msg (w_s => \@kv); |
252 | } |
327 | } |
253 | |
328 | |
254 | sub get { |
329 | sub get { |
255 | my ($self, $member, $cb) = @_; |
330 | my ($self, $member, $cb) = @_; |
256 | |
331 | |
257 | $self->msg_cb ($cb, w_g => name => $member); |
332 | $self->msg_cb ($cb, w_g => ref $member ? @$member : $member); |
258 | } |
333 | } |
259 | |
334 | |
260 | sub TO_JSON { |
335 | sub TO_JSON { |
261 | { __widget_ref__ => $_[0]{id} } |
336 | { "\fw" => $_[0]{id} } |
262 | } |
337 | } |
263 | |
338 | |
264 | our $AUTOLOAD; |
339 | our $AUTOLOAD; |
265 | |
340 | |
266 | sub AUTOLOAD { |
341 | sub AUTOLOAD { |
267 | $AUTOLOAD =~ s/^.*:// |
342 | $AUTOLOAD =~ s/^.*:// |
268 | or return; |
343 | or return; |
269 | |
344 | |
270 | my $self = shift; |
345 | my $self = shift; |
271 | |
346 | |
|
|
347 | #TODO: handle non-void context |
272 | $self->msg (w_c => name => $AUTOLOAD, args => \@_); |
348 | $self->msg (w_c => 0, $AUTOLOAD, @_); |
273 | |
349 | |
274 | () |
350 | () |
275 | } |
351 | } |
276 | |
352 | |
|
|
353 | package ext::widget::callback; |
|
|
354 | |
|
|
355 | sub DESTROY { |
|
|
356 | my ($self) = @_; |
|
|
357 | |
|
|
358 | if (my $ns = $self->{ns}) { |
|
|
359 | delete $ns->{widget_cb}{$self->{id}}; |
|
|
360 | $ns->free_wid ($self->{id}); |
|
|
361 | } |
|
|
362 | } |
|
|
363 | |
|
|
364 | sub TO_JSON { |
|
|
365 | { "\fc" => $_[0]{id} } |
|
|
366 | } |
|
|
367 | |