1 |
package GCE::AttrEdit; |
2 |
|
3 |
=head1 NAME |
4 |
|
5 |
GCE::AttrEdit - an edit wiget for attributes |
6 |
|
7 |
=cut |
8 |
|
9 |
use Gtk2; |
10 |
use Gtk2::Gdk::Keysyms; |
11 |
use Gtk2::SimpleList; |
12 |
|
13 |
use GCE::Util; |
14 |
use GCE::InventoryEditor; |
15 |
use GCE::AttachEditor; |
16 |
|
17 |
use Glib::Object::Subclass |
18 |
Gtk2::HPaned; |
19 |
|
20 |
use AnyEvent::EditText; |
21 |
|
22 |
use Deliantra; |
23 |
use strict; |
24 |
|
25 |
sub save_layout { |
26 |
my ($self) = @_; |
27 |
|
28 |
$::CFG->{attr_view_hpane_pos} = $self->get_position; |
29 |
$::CFG->{attr_view_show_use} = $self->{use_btn}->get_active * 1; |
30 |
|
31 |
$self->{attach_editor}->save_layout |
32 |
if $self->{attach_editor}; |
33 |
} |
34 |
|
35 |
sub load_layout { |
36 |
my ($self) = @_; |
37 |
|
38 |
$self->set_position ($::CFG->{attr_view_hpane_pos} || 350); |
39 |
$self->{use_btn}->set_active ($::CFG->{attr_view_show_use} * 1); |
40 |
|
41 |
$self->{use_al}->remove ($_) for $self->{use_al}->get_children; |
42 |
if ($self->{use_btn}->get_active) { |
43 |
$self->{use_al}->add ($self->{use_lbl}); |
44 |
$self->{use_lbl}->show; |
45 |
} |
46 |
} |
47 |
|
48 |
sub INIT_INSTANCE { |
49 |
my ($self) = @_; |
50 |
|
51 |
my $pb = $self->{arch_pb} = new_arch_pb; |
52 |
|
53 |
$self->add (my $topvb = Gtk2::VBox->new); |
54 |
$topvb->pack_start (my $menubar = Gtk2::MenuBar->new, 0, 0, 0); |
55 |
$menubar->append (my $actions_menu = Gtk2::MenuItem->new ('Actions')); |
56 |
my $actmenu = Gtk2::Menu->new; |
57 |
$actmenu->append (my $defitm = Gtk2::MenuItem->new ('reset to defaults')); |
58 |
$defitm->signal_connect (activate => sub { |
59 |
my $ar = $self->{archref}; |
60 |
$ar->reset_to_defaults; |
61 |
#XXXAR $self->set_arch ($arch, $self->{change_cb}); |
62 |
}); |
63 |
|
64 |
$actmenu->append (my $attitm = Gtk2::MenuItem->new ('attach')); |
65 |
$attitm->signal_connect (activate => sub { |
66 |
my $ar = $self->{archref}; |
67 |
return unless $ar; |
68 |
unless ($self->{attach_editor}) { |
69 |
my $w = GCE::AttachEditor->new; |
70 |
$w->set_attachment ( |
71 |
$ar->get ('attach'), |
72 |
sub { |
73 |
if (@{$_[0]}) { |
74 |
$ar->set_silent (attach => $_[0]) |
75 |
} else { |
76 |
$ar->set_silent (attach => undef) |
77 |
} |
78 |
} |
79 |
); |
80 |
$self->{attach_editor} = $w; |
81 |
$w->signal_connect (destroy => sub { delete $self->{attach_editor} }); |
82 |
$w->show_all; |
83 |
} |
84 |
}); |
85 |
|
86 |
$actions_menu->set_submenu ($actmenu); |
87 |
$actmenu->show_all; |
88 |
|
89 |
$topvb->pack_start (my $hb2 = Gtk2::HBox->new, 0, 1, 0); |
90 |
$hb2->pack_start (my $img = $self->{arch_img} = (new_from_pixbuf Gtk2::Image $pb), 0, 0, 0); |
91 |
$img->set_alignment (0, 0.5); |
92 |
|
93 |
$hb2->pack_start (my $lbl = $self->{arch_name_lbl} = Gtk2::Label->new, 0, 0, 0); |
94 |
$lbl->set_alignment (0, 0.5); |
95 |
|
96 |
$hb2->pack_start (my $statbtn = $self->{arch_stat_btn} = Gtk2::Button->new, 0, 0, 0); |
97 |
|
98 |
$topvb->pack_start (my $docal = Gtk2::Alignment->new (0, 0.5, 0, 1), 0, 1, 0); |
99 |
$topvb->pack_start (my $usebtn = $self->{use_btn} = Gtk2::ToggleButton->new ('show use'), 0, 1, 0); |
100 |
$topvb->pack_start (my $useal = $self->{use_al} = Gtk2::Alignment->new (0, 0.5, 0, 1), 0, 1, 0); |
101 |
$topvb->pack_start (my $ntbook = $self->{ntbook} = Gtk2::Notebook->new, 1, 1, 0); |
102 |
$ntbook->set_scrollable (1); |
103 |
$docal->add ($self->{doc_lbl} = Gtk2::Label->new); |
104 |
|
105 |
$usebtn->set_active (0); |
106 |
$self->{use_lbl} = Gtk2::Label->new; |
107 |
$usebtn->signal_connect (toggled => sub { |
108 |
my ($usebtn) = @_; |
109 |
|
110 |
$useal->remove ($_) for $useal->get_children; |
111 |
if ($usebtn->get_active) { |
112 |
$useal->add ($self->{use_lbl}); |
113 |
$self->{use_lbl}->show; |
114 |
} |
115 |
}); |
116 |
$self->{doc_lbl}->set_line_wrap (1); |
117 |
$self->{use_lbl}->set_line_wrap (1); |
118 |
|
119 |
$self->add2 (my $sw = Gtk2::ScrolledWindow->new); |
120 |
$sw->set_policy ('automatic', 'automatic'); |
121 |
$sw->add_with_viewport (my $inv = $self->{inv_edit} = GCE::InventoryEditor->new); |
122 |
} |
123 |
|
124 |
#sub spawn_editor { |
125 |
# my ($arch, $cb) = @_; |
126 |
# |
127 |
# my $w = Gtk2::Window->new; |
128 |
# $w->set_title ("gce - edit attrs"); |
129 |
# $w->add (my $ae = GCE::AttrEdit->new); |
130 |
# |
131 |
# main::set_pos_and_size ($w, $main::CFG->{attr_view}, 200, 200); |
132 |
# |
133 |
# $ae->set_arch ($arch, $cb); |
134 |
# $w->set_title ("gce - edit $arch->{_name}"); |
135 |
# |
136 |
# $w->show_all; |
137 |
#} |
138 |
|
139 |
sub update_arch { |
140 |
my ($self, $ar, $key, $value) = @_; |
141 |
$ar->set ($key, $value); |
142 |
} |
143 |
|
144 |
sub set_attr { |
145 |
my ($self, $key, $value) = @_; |
146 |
|
147 |
my $attr = $self->{archref}->get ($key); |
148 |
|
149 |
unless (ref $attr) { |
150 |
|
151 |
$self->{archref}->set ($key, $value); |
152 |
} |
153 |
} |
154 |
|
155 |
sub get_arch { |
156 |
my ($self) = @_; |
157 |
|
158 |
$self->{archref} |
159 |
} |
160 |
|
161 |
#sub update { |
162 |
# my ($self, $narch) = @_; |
163 |
# |
164 |
# if (ref ($narch) ne 'GCE::ArchRef') { require Carp; Carp::confess ("NO ARCH REF!") } |
165 |
# |
166 |
# if ($narch) { |
167 |
# $self->set_arch ($narch); |
168 |
# } else { |
169 |
# $self->set_arch ($self->{archref}); |
170 |
# } |
171 |
#} |
172 |
|
173 |
sub set_arch { |
174 |
my ($self, $ar, $clear_inv) = @_; |
175 |
|
176 |
if ((defined $self->{archref}) && ($self->{archref} != $ar)) { |
177 |
$self->{archref}->remove_on_change ('attredit') |
178 |
if defined $self->{archref}; |
179 |
$ar->add_on_change (attredit => sub { $self->set_arch ($_[0]) }); |
180 |
$self->{attach_editor}->destroy if $self->{attach_editor}; |
181 |
|
182 |
} elsif (not defined $self->{archref}) { |
183 |
$ar->add_on_change (attredit => sub { $self->set_arch ($_[0]) }); |
184 |
} |
185 |
|
186 |
$self->{inv_edit}->clear_inv_hist if $clear_inv; |
187 |
$self->{inv_edit}->set_arch ($ar); |
188 |
|
189 |
$self->{archref} = $ar; |
190 |
|
191 |
$self->{arch_name_lbl}->set_text ($ar->longname); |
192 |
|
193 |
fill_pb_from_arch ($self->{arch_pb}, $ar->getarch); |
194 |
$self->{arch_img}->set_from_pixbuf ($self->{arch_pb}); |
195 |
$self->label_set_color ($self->{arch_name_lbl}, 0); |
196 |
|
197 |
$self->{arch_stat_btn}->set_label ($ar->{source}); |
198 |
|
199 |
# get current page (to remember it for later) |
200 |
my $pgnum = $self->{ntbook}->get_current_page; |
201 |
my $curwid = $self->{ntbook}->get_nth_page ($pgnum); |
202 |
my $curpage_text = |
203 |
defined $curwid |
204 |
? $self->{ntbook}->get_tab_label_text ($curwid) |
205 |
: undef; |
206 |
|
207 |
my $al_arch = $ar->archetype; |
208 |
|
209 |
$self->hide; |
210 |
$self->{ntbook}->remove ($_) |
211 |
for $self->{ntbook}->get_children; |
212 |
|
213 |
$self->{ttip} = Gtk2::Tooltips->new; |
214 |
|
215 |
my $type = $ar->type; |
216 |
|
217 |
for my $sec (@{$type->{section}}) { |
218 |
my $secname = shift @$sec; |
219 |
$self->add_section_edit_widgets ($self->{ntbook}, $secname, $ar, $sec); |
220 |
} |
221 |
|
222 |
for my $key (qw/lore msg/) { |
223 |
|
224 |
$self->{ntbook}->append_page (my $v = Gtk2::VBox->new, $key); |
225 |
$v->pack_start (my $ed = Gtk2::Button->new ("Open Editor"), 0, 1, 0); |
226 |
$v->pack_start (my $sw = Gtk2::ScrolledWindow->new, 1, 1, 0); |
227 |
$sw->set_policy ('automatic', 'automatic'); |
228 |
$sw->add (my $tb = $self->{"${key}_txt"} = Gtk2::TextView->new); |
229 |
|
230 |
my $change_field = sub { |
231 |
my ($txt) = @_; |
232 |
|
233 |
my $buf = $tb->get_buffer; |
234 |
|
235 |
my $old_txt = $buf->get_text ($buf->get_start_iter, $buf->get_end_iter, 0); |
236 |
$txt = $old_txt unless defined $txt; |
237 |
|
238 |
$ar->set_silent ($key, $txt) |
239 |
if $txt ne $ar->get ($key); |
240 |
$buf->set_text ($txt) |
241 |
if $txt ne $old_txt; |
242 |
}; |
243 |
|
244 |
my $buf = $tb->get_buffer; |
245 |
$change_field->($ar->get ($key)); |
246 |
$buf->signal_connect (changed => sub { |
247 |
$change_field->(); |
248 |
1 |
249 |
}); |
250 |
|
251 |
$ed->signal_connect (clicked => sub { |
252 |
AnyEvent::EditText::edit ($ar->get ($key), sub { |
253 |
my ($new, $change, $err) = @_; |
254 |
|
255 |
if (not defined $new) { |
256 |
quick_msg ("Error while starting editor: $err"); |
257 |
return; |
258 |
} |
259 |
|
260 |
$change_field->($new) |
261 |
if $change; |
262 |
}); |
263 |
}); |
264 |
} |
265 |
|
266 |
my $desc = pseudohtml2txt $type->{desc}; |
267 |
my $use = pseudohtml2txt $type->{use}; |
268 |
$self->{doc_lbl}->set_text ($desc); |
269 |
$self->{use_lbl}->set_text ($use); |
270 |
|
271 |
$self->{ttip}->enable; |
272 |
|
273 |
$self->show_all; |
274 |
|
275 |
# reset the current page if found |
276 |
# XXX: it's braindamaged: it has to be done AFTER show all for some reason |
277 |
if (defined $curpage_text) { |
278 |
|
279 |
for (my $i = 0; $i <= $self->{ntbook}->get_n_pages; $i++) { |
280 |
my $w = $self->{ntbook}->get_nth_page ($i); |
281 |
|
282 |
if ($w && $self->{ntbook}->get_tab_label_text ($w) eq $curpage_text) { |
283 |
$self->{ntbook}->set_current_page ($i); |
284 |
last; |
285 |
} |
286 |
} |
287 |
} |
288 |
} |
289 |
|
290 |
sub add_section_edit_widgets { |
291 |
my ($self, $ntbook, $name, $ar, $section) = @_; |
292 |
|
293 |
$self->{ntbook}->append_page (my $sw = Gtk2::ScrolledWindow->new, $name); |
294 |
$sw->set_policy ('automatic', 'automatic'); |
295 |
$sw->add_with_viewport (my $vb = Gtk2::VBox->new); |
296 |
$vb->pack_start (my $table = new Gtk2::Table (2, (scalar @$section) + 1), 0, 1, 0); |
297 |
|
298 |
my $i = 0; |
299 |
for my $sec (@$section) { |
300 |
my ($key, $sec) = ($sec->[0], $sec->[1]); |
301 |
|
302 |
next if grep { $key eq $_ } qw/msg lore/; |
303 |
|
304 |
my $bwid = Gtk2::EventBox->new; |
305 |
$bwid->add (my $al = Gtk2::Alignment->new (0.0, 0.5, 0, 1)); |
306 |
$al->add (Gtk2::Label->new (def ($sec->{name}, $key))); |
307 |
|
308 |
if ($sec->{desc} !~ m/^\s*$/s) { |
309 |
$self->{ttip}->set_tip ($bwid, $sec->{desc}); |
310 |
} |
311 |
|
312 |
$table->attach ($bwid, 0, 1, $i, $i + 1, ['shrink','fill'], 'fill', 5, 0); |
313 |
|
314 |
$al = Gtk2::Alignment->new (0.0, 0.5, 1, 0); |
315 |
$table->attach ($al, 1, 2, $i, $i + 1, ['expand', 'fill'], 'expand', 0, 0); |
316 |
$al->add ($self->get_edit_widget ($key, $sec, $ar, $bwid)); |
317 |
|
318 |
$i++; |
319 |
} |
320 |
} |
321 |
|
322 |
sub label_set_color { |
323 |
my ($self, $lbl, $dark) = @_; |
324 |
if ($dark) { |
325 |
for (qw/normal active prelight selected insensitive/) { |
326 |
$lbl->modify_bg ($_, $lbl->get_default_style->bg ('active')); |
327 |
$lbl->modify_fg ($_, $lbl->get_default_style->fg ('active')); |
328 |
} |
329 |
|
330 |
} else { |
331 |
for (qw/normal active prelight selected insensitive/) { |
332 |
$lbl->modify_bg ($_, $lbl->get_default_style->bg ($_)); |
333 |
} |
334 |
} |
335 |
|
336 |
} |
337 |
|
338 |
sub label_set_color_default { |
339 |
my ($self, $lbl, $ar, $key, $val) = @_; |
340 |
require Carp; $ar or Carp::confess ("UNDEF"); |
341 |
my $al_arch = $ar->archetype; |
342 |
$self->label_set_color ($lbl, $ar->field_value_is_default ($key, $val)); |
343 |
|
344 |
} |
345 |
|
346 |
# XXX: Warning: Ugly code ahead: |
347 |
sub get_edit_widget { |
348 |
my ($self, $key, $edspec, $ar, $lbl) = @_; |
349 |
|
350 |
my $type = $edspec->{type}; |
351 |
my $al_arch = $ar->archetype; |
352 |
|
353 |
if ($type eq 'bool') { |
354 |
my $boolval = def ($edspec->{value}, [0, 1]); |
355 |
|
356 |
|
357 |
$self->label_set_color_default ($lbl, $ar, $key, $ar->get_or_default ($key)); |
358 |
|
359 |
my $chk = new Gtk2::CheckButton (def ($edspec->{name}, $key)); |
360 |
$chk->set_active ($ar->get_or_default ($key) == $boolval->[1]); |
361 |
$chk->signal_connect (clicked => sub { |
362 |
my ($chk) = @_; |
363 |
|
364 |
$ar->set_silent ($key, $boolval->[$chk->get_active * 1], $type); |
365 |
|
366 |
$self->label_set_color_default ($lbl, $ar, $key, $boolval->[$chk->get_active * 1]); |
367 |
}); |
368 |
|
369 |
$self->{ttip}->set_tip ($chk, $al_arch->{$key} * 1); |
370 |
|
371 |
return $chk |
372 |
|
373 |
} elsif (grep { $type eq $_ } qw/string int treasurelist float/) { |
374 |
$self->label_set_color_default ($lbl, $ar, $key, $ar->get_or_default ($key)); |
375 |
|
376 |
my $entry = new Gtk2::Entry; |
377 |
$entry->set_text ($ar->get_or_default ($key)); |
378 |
$entry->signal_connect (changed => sub { |
379 |
my ($entry) = @_; |
380 |
$self->label_set_color_default ($lbl, $ar, $key, $entry->get_text); |
381 |
$ar->set_silent ($key, $entry->get_text); |
382 |
1 |
383 |
}); |
384 |
|
385 |
$self->{ttip}->set_tip ($entry, $ar->archetype->{$key}); |
386 |
|
387 |
return $entry |
388 |
|
389 |
} elsif ($type eq 'spell' or $type eq 'nz_spell') { # XXX: nz_spell bug in datafiles? |
390 |
my $comb = Gtk2::ComboBox->new_text; |
391 |
my $spells_idx = {}; |
392 |
my $spells_cmd_idx = {}; |
393 |
my $sp = \%Deliantra::Data::SPELL; |
394 |
|
395 |
$comb->append_text ("<none>"); |
396 |
|
397 |
my $idx = 1; # XXX: replace this idx with a more save/correct method? |
398 |
for (sort { $sp->{$a} cmp $sp->{$b} } keys %$sp) { |
399 |
$spells_cmd_idx->{$idx} = $_; |
400 |
$spells_idx->{$_} = $idx++; |
401 |
|
402 |
$comb->append_text ($sp->{$_}); |
403 |
} |
404 |
#XXX: FIXME: $self->{ttip}->set_tip ($comb, $sp->{$al_arch->{$key}}); |
405 |
|
406 |
$comb->set_active ($spells_idx->{$ar->get_or_default ($key)}); |
407 |
$self->label_set_color_default ($lbl, $ar, $key, $ar->get_or_default ($key)); |
408 |
|
409 |
$comb->signal_connect (changed => sub { |
410 |
my ($comb) = @_; |
411 |
$self->label_set_color_default ($lbl, $ar, $key, $spells_cmd_idx->{$comb->get_active}); |
412 |
$ar->set_silent ($key, $spells_cmd_idx->{$comb->get_active}); |
413 |
}); |
414 |
return $comb |
415 |
|
416 |
} elsif ($type eq 'bitmask') { |
417 |
my $chval = $ar->get_or_default ($key); |
418 |
my $btn = Gtk2::Button->new; |
419 |
$btn->add (my $lblb = Gtk2::Label->new ("bitmask: " . ($chval * 1))); |
420 |
$self->{ttip}->set_tip ($btn, $al_arch->{$key}); |
421 |
|
422 |
my $menu = $self->create_bitmask_menu ($edspec->{value}, $lbl, $lblb, $ar, $key, \$chval); |
423 |
|
424 |
$self->label_set_color_default ($lbl, $ar, $key, $chval); |
425 |
|
426 |
$btn->signal_connect (button_press_event => sub { |
427 |
my ($btn, $ev) = @_; |
428 |
$menu->popup (undef, undef, undef, undef, $ev->button, 0); |
429 |
}); |
430 |
|
431 |
return $btn; |
432 |
|
433 |
} elsif ($type eq 'list') { |
434 |
my $lblb = Gtk2::Label->new ($edspec->{value}->{$ar->get_or_default ($key) * 1}); |
435 |
my $btn = Gtk2::Button->new; |
436 |
$self->{ttip}->set_tip ($btn, $edspec->{value}->{$al_arch->{$key}}); |
437 |
$btn->add ($lblb); |
438 |
my $menu = $self->create_list_menu ($edspec->{value}, $lbl, $lblb, $ar, $key); |
439 |
|
440 |
$self->label_set_color_default ($lbl, $ar, $key, $ar->get_or_default ($key)); |
441 |
|
442 |
$btn->signal_connect (button_press_event => sub { |
443 |
my ($btn, $ev) = @_; |
444 |
$menu->popup (undef, undef, undef, undef, $ev->button, 0); |
445 |
}); |
446 |
return $btn; |
447 |
|
448 |
} elsif ($type eq 'fixed') { |
449 |
return Gtk2::Label->new ("$edspec->{name} = $edspec->{value}"); |
450 |
|
451 |
} elsif ($type eq 'text') { |
452 |
return Gtk2::Label->new ("<see $key tab>"); |
453 |
|
454 |
} elsif ($type eq 'movement_type') { # ok... this is quite a big one... awww |
455 |
my $a = Gtk2::Alignment->new (0, 0, 0, 0); |
456 |
$a->add (my $v = Gtk2::VBox->new (0, 0)); |
457 |
$a->set_padding (4, 4, 0, 0); |
458 |
my $btns1 = Gtk2::HButtonBox->new; |
459 |
my $btns2 = Gtk2::HButtonBox->new; |
460 |
$btns1->set_layout ('start'); |
461 |
$btns2->set_layout ('start'); |
462 |
|
463 |
$v->pack_start ($btns1, 0, 1, 0); |
464 |
$v->pack_start ($btns2, 0, 1, 0); |
465 |
|
466 |
my @lblbtns; |
467 |
my $calc_lbl = sub { # update callback for the buttons |
468 |
for (@lblbtns) { |
469 |
my $btn = $_->[0]; |
470 |
my $lbl = $_->[1]; |
471 |
|
472 |
if (defined $ar->get_or_default ($key)) { |
473 |
my $res = |
474 |
Deliantra::MoveType->new ($ar->get_or_default ($key)) |
475 |
>= $lbl; |
476 |
|
477 |
if (defined $res) { |
478 |
$self->label_set_color ($btn, 0); |
479 |
$lbl = ($res ? "+" : "-") . $lbl; |
480 |
|
481 |
} else { |
482 |
$self->label_set_color ($btn, 1); |
483 |
$lbl = "$lbl?"; |
484 |
} |
485 |
} else { |
486 |
$self->label_set_color ($btn, 1); |
487 |
$lbl = "$lbl?"; |
488 |
} |
489 |
|
490 |
$btn->set_label ($lbl); |
491 |
} |
492 |
}; |
493 |
|
494 |
$self->label_set_color_default ($lbl, $ar, $key, $ar->get_or_default ($key)); |
495 |
|
496 |
my $mid = (scalar @Deliantra::MOVE_TYPE) / 2; |
497 |
my $cnt = 0; |
498 |
my $box = $btns1; |
499 |
|
500 |
for my $mty (@Deliantra::MOVE_TYPE) { |
501 |
$box->pack_start (my $btn = Gtk2::Button->new ($calc_lbl->()), 0, 1, 0); |
502 |
push @lblbtns, [$btn, $mty]; |
503 |
|
504 |
$self->{ttip}->set_tip ($btn, $ar->archetype->{$key}); |
505 |
$btn->signal_connect (clicked => sub { |
506 |
my $v = $ar->get ($key) || Deliantra::MoveType->new; |
507 |
|
508 |
$v x= $mty; |
509 |
$v = $v eq '' ? undef : $v; |
510 |
$ar->set_silent ($key, $v); |
511 |
|
512 |
$self->label_set_color_default ( |
513 |
$lbl, $ar, $key, $ar->get_or_default ($key)); |
514 |
$calc_lbl->(); |
515 |
}); |
516 |
|
517 |
if (++$cnt >= $mid) { $box = $btns2 } # for wrapping into the second button box |
518 |
} |
519 |
$calc_lbl->(); |
520 |
|
521 |
return $a; |
522 |
|
523 |
} else { |
524 |
return Gtk2::Label->new ("$key => $edspec->{name} ($type)"); |
525 |
|
526 |
} |
527 |
} |
528 |
|
529 |
sub bitmask_to_list { |
530 |
my ($self, $bitlist, $bits) = @_; |
531 |
|
532 |
my @l; |
533 |
for (%$bitlist) { |
534 |
if ($bits & (1 << $_)) { |
535 |
push @l, $bitlist->{$_}; |
536 |
} |
537 |
} |
538 |
return @l; |
539 |
} |
540 |
|
541 |
sub create_list_menu { |
542 |
my ($self, $list, $clbl, $lbl, $ar, $key) = @_; |
543 |
|
544 |
my $menu = Gtk2::Menu->new; |
545 |
|
546 |
for my $item (sort keys %$list) { |
547 |
my $lbltxt = $list->{$item}; |
548 |
my $menuitem = Gtk2::MenuItem->new_with_label ($lbltxt); |
549 |
$menuitem->signal_connect (activate => sub { |
550 |
my ($menuitem) = @_; |
551 |
$lbl->set_text ($list->{$item}); |
552 |
$self->label_set_color_default ($clbl, $ar, $key, $item); |
553 |
$ar->set_silent ($key, $item); |
554 |
}); |
555 |
$menu->append ($menuitem); |
556 |
$menuitem->show; |
557 |
} |
558 |
|
559 |
return $menu; |
560 |
} |
561 |
|
562 |
sub create_bitmask_menu { |
563 |
my ($self, $bits, $clbl, $lbl, $ar, $key, $rval) = @_; |
564 |
|
565 |
my $menu = Gtk2::Menu->new; |
566 |
|
567 |
for my $bit (sort { $a <=> $b } keys %$bits) { |
568 |
my $lbltxt = $bits->{$bit}; |
569 |
my $menuitem = Gtk2::CheckMenuItem->new_with_label ($lbltxt); |
570 |
if ($$rval & (1 << $bit)) { |
571 |
$menuitem->set_active (1);#$arch->{$key} & (1 << $bit)); |
572 |
} |
573 |
$menuitem->signal_connect (toggled => sub { |
574 |
my ($menuitem) = @_; |
575 |
my $newval = $ar->get ($key); |
576 |
$$rval &= ~(1 << $bit); |
577 |
$$rval |= (1 << $bit) if $menuitem->get_active; |
578 |
$lbl->set_text ("bitmask: " . ($$rval * 1)); |
579 |
$self->label_set_color_default ($clbl, $ar, $key, $$rval); |
580 |
$ar->set_silent ($key, $$rval); |
581 |
}); |
582 |
$menu->append ($menuitem); |
583 |
$menuitem->show; |
584 |
} |
585 |
|
586 |
return $menu; |
587 |
} |
588 |
|
589 |
|
590 |
=head1 AUTHOR |
591 |
|
592 |
Marc Lehmann <schmorp@schmorp.de> |
593 |
http://home.schmorp.de/ |
594 |
|
595 |
Robin Redeker <elmex@ta-sa.org> |
596 |
http://www.ta-sa.org/ |
597 |
|
598 |
=cut |
599 |
1; |