--- deliantra/Deliantra-Client/DC.pm 2006/05/26 18:28:23 1.70 +++ deliantra/Deliantra-Client/DC.pm 2006/05/29 21:20:15 1.76 @@ -70,22 +70,27 @@ close CFG; } -mkdir "$Crossfire::VARDIR/pclient", 0777; +mkdir "$Crossfire::VARDIR/cfplus", 0777; -our $DB_ENV = new BerkeleyDB::Env - -Home => "$Crossfire::VARDIR/pclient", - -Cachesize => 1_000_000, - -ErrFile => "$Crossfire::VARDIR/pclient/errorlog.txt", +{ + use strict; + + our $DB_ENV = new BerkeleyDB::Env + -Home => "$Crossfire::VARDIR/cfplus", + -Cachesize => 1_000_000, + -ErrFile => "$Crossfire::VARDIR/cfplus/errorlog.txt", # -ErrPrefix => "DATABASE", - -Verbose => 1, - -Flags => DB_CREATE | DB_RECOVER | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_TXN, - or die "unable to create/open database home $Crossfire::VARDIR/pclient: $BerkeleyDB::Error"; + -Verbose => 1, + -Flags => DB_CREATE | DB_RECOVER | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_TXN, + -SetFlags => DB_AUTO_COMMIT | DB_LOG_AUTOREMOVE | DB_TXN_WRITE_NOSYNC, + or die "unable to create/open database home $Crossfire::VARDIR/cfplus: $BerkeleyDB::Error"; +} sub db_table($) { my ($table) = @_; $table =~ s/([^a-zA-Z0-9_\-])/sprintf "=%x=", ord $1/ge; - + new CFClient::Database -Env => $DB_ENV, -Filename => $table, @@ -93,7 +98,7 @@ # -Subname => $table, -Property => DB_CHKSUM, -Flags => DB_CREATE | DB_UPGRADE, - or die "unable to create/open database table $_[0]: $BerkeleyDB::Error"; + or die "unable to create/open database table $_[0]: $BerkeleyDB::Error" } sub pod_to_pango($) { @@ -192,6 +197,9 @@ package CFClient::Item; +use strict; +use Crossfire::Protocol::Constants; + sub desc_string { my ($self) = @_; @@ -200,19 +208,19 @@ ? $self->{name} : "$self->{nrof} × $self->{name_pl}"; - $self->{flags} & Crossfire::Protocol::F_OPEN + $self->{flags} & F_OPEN and $desc .= " (open)"; - $self->{flags} & Crossfire::Protocol::F_APPLIED + $self->{flags} & F_APPLIED and $desc .= " (applied)"; - $self->{flags} & Crossfire::Protocol::F_UNPAID + $self->{flags} & F_UNPAID and $desc .= " (unpaid)"; - $self->{flags} & Crossfire::Protocol::F_MAGIC + $self->{flags} & F_MAGIC and $desc .= " (magic)"; - $self->{flags} & Crossfire::Protocol::F_CURSED + $self->{flags} & F_CURSED and $desc .= " (cursed)"; - $self->{flags} & Crossfire::Protocol::F_DAMNED + $self->{flags} & F_DAMNED and $desc .= " (damned)"; - $self->{flags} & Crossfire::Protocol::F_LOCKED + $self->{flags} & F_LOCKED and $desc .= " *"; $desc @@ -250,7 +258,7 @@ ["mark", sub { $::CONN->send ("mark ". pack "N", $self->{tag}) }], ["apply", sub { $::CONN->send ("apply $self->{tag}") }], ( - $self->{flags} & Crossfire::Protocol::F_LOCKED + $self->{flags} & F_LOCKED ? ( ["unlock", sub { $::CONN->send ("lock " . pack "CN", 0, $self->{tag}) }], ) @@ -279,7 +287,7 @@ can_hover => 1, anim => $self->{anim}, animspeed => $self->{animspeed}, # TODO# must be set at creation time - connect_button_down => $button_cb, + on_button_down => $button_cb, ; $self->{face_widget}{face} = $self->{face}; $self->{face_widget}{anim} = $self->{anim}; @@ -296,7 +304,7 @@ can_hover => 1, ellipsise => 2, align => -1, - connect_button_down => $button_cb, + on_button_down => $button_cb, ; my $desc = CFClient::Item::desc_string $self; $self->{desc_widget}->set_text ($desc); @@ -307,7 +315,7 @@ can_hover => 1, ellipsise => 0, align => 0, - connect_button_down => $button_cb, + on_button_down => $button_cb, ; $self->{weight_widget}->set_text (CFClient::Item::weight_string $self); @@ -319,6 +327,217 @@ ); } +package CFClient::Recorder; + +our $RECORD_WINDOW; + +my $CMDBOX; +my $CURRENT_CMDS; +my $REC_BTN; + +my @ALLOWED_MODIFIER_KEYS = ( + (CFClient::SDLK_LSHIFT) => "LSHIFT", + (CFClient::SDLK_LCTRL ) => "LCTRL", + (CFClient::SDLK_LALT ) => "LALT", + (CFClient::SDLK_LMETA ) => "LMETA", + + (CFClient::SDLK_RSHIFT) => "RSHIFT", + (CFClient::SDLK_RCTRL ) => "RCTRL", + (CFClient::SDLK_RALT ) => "RALT", + (CFClient::SDLK_RMETA ) => "RMETA", +); + +my %ALLOWED_MODIFIERS = ( + (CFClient::KMOD_LSHIFT) => "LSHIFT", + (CFClient::KMOD_LCTRL ) => "LCTRL", + (CFClient::KMOD_LALT ) => "LALT", + (CFClient::KMOD_LMETA ) => "LMETA", + + (CFClient::KMOD_RSHIFT) => "RSHIFT", + (CFClient::KMOD_RCTRL ) => "RCTRL", + (CFClient::KMOD_RALT ) => "RALT", + (CFClient::KMOD_RMETA ) => "RMETA", +); + +my %DIRECT_BIND_CHARS = map { $_ => 1 } qw/0 1 2 3 4 5 6 7 8 9/; +my @DIRECT_BIND_KEYS = ( + CFClient::SDLK_F1, + CFClient::SDLK_F2, + CFClient::SDLK_F3, + CFClient::SDLK_F4, + CFClient::SDLK_F5, + CFClient::SDLK_F6, + CFClient::SDLK_F7, + CFClient::SDLK_F8, + CFClient::SDLK_F9, + CFClient::SDLK_F10, + CFClient::SDLK_F11, + CFClient::SDLK_F12, + CFClient::SDLK_F13, + CFClient::SDLK_F14, + CFClient::SDLK_F15, +); + +# this binding dialog asks for a key-combo to be pressed +# and if successful it binds the modifier+symbol to the +# supplied actions in $cmd. +# (Bindings are stored in $::CFG->{bindings}->{$mod}->{$sym}) +sub open_binding_dialog { + my ($cmd) = @_; + + my $w = new CFClient::UI::FancyFrame + title => "Bind Action"; + + $w->add (my $vb = new CFClient::UI::VBox); + $vb->add (new CFClient::UI::Label + text => "Press a modifier (CTRL, ALT and/or SHIFT) and a key." + ."You can only bind 0-9 and F1-F15 without modifiers." + ); + $vb->add (my $entry = new CFClient::UI::Entry + text => "", + on_key_down => sub { + my ($entry, $ev) = @_; + + my $mod = $ev->{mod}; + my $sym = $ev->{sym}; + + # XXX: This seems a little bit hackisch to me, but i have to ignore them + if (grep { $_ == $sym } @ALLOWED_MODIFIER_KEYS) { + return; + } + + if ($mod == CFClient::KMOD_NONE + and not $DIRECT_BIND_CHARS{chr ($ev->{unicode})} + and not grep { $sym == $_ } @DIRECT_BIND_KEYS) + { + $::STATUSBOX->add ( + "Can't bind key ".CFClient::SDL_GetKeyName ($sym) + ." directly without modifier! It would damage the completer handling." + ); + return; + } + + $entry->focus_out; + + $::CFG->{bindings}->{$mod}->{$sym} = $cmd; + $::STATUSBOX->add ("Bound actions to '".keycombo_to_name ($mod, $sym)."'. Don't forget 'Save Config'!"); + + $w->destroy + }); + + $entry->focus_in; + $w->center; + $w->show; +} + +sub keycombo_to_name { + my ($mod, $sym) = @_; + + my $mods = join '+', + map { $ALLOWED_MODIFIERS{$_} } + grep { $_ & $mod } + keys %ALLOWED_MODIFIERS; + $mods .= "+" if $mods ne ''; + + return $mods . CFClient::SDL_GetKeyName ($sym); +} + +sub clear_command_list { + $CMDBOX->clear () if $CMDBOX; +} + +sub set_command_list { + my ($list) = @_; + + return unless $CMDBOX; + + $CMDBOX->clear (); + $CURRENT_CMDS = $list; + + my $idx = 0; + + for (@$list) { + $CMDBOX->add (my $hb = new CFClient::UI::HBox); + + my $i = $idx; + $hb->add (new CFClient::UI::Button + text => "delete", + tooltip => "Deletes the action from the record", + on_activate => sub { + $CMDBOX->remove ($hb); + $list->[$i] = undef; + }); + + $hb->add (new CFClient::UI::Label text => $_); + + $idx++ + } +} + +# if $show is 1 the recorder will be shown +sub start { + my ($show) = @_; + + $RECORD_WINDOW->show if $show; + + $REC_BTN->set_text ("stop recording"); + $REC_BTN->{recording} = 1; + clear_command_list; + $::CONN->start_record; +} + +# if $autobind is 1 the recorder will be automatically +# jump into the binding query and hide the recorder window +sub stop { + my ($autobind) = @_; + + $REC_BTN->set_text ("start recording"); + $REC_BTN->{recording} = 0; + + my $rec = $::CONN->stop_record; + return unless ref $rec eq 'ARRAY'; + set_command_list ($rec); + + if ($autobind) { + open_binding_dialog ([ grep { defined $_ } @$CURRENT_CMDS ]); + $RECORD_WINDOW->hide; + } +} + +sub make_window { + $RECORD_WINDOW = new CFClient::UI::FancyFrame + req_y => 1, + req_x => -1, + title => "Action Recorder"; + + $RECORD_WINDOW->add (my $vb = new CFClient::UI::VBox); + $vb->add ($REC_BTN = new CFClient::UI::Button + text => "start recording", + tooltip => "Start/Stops recording of actions." + ."(CTRL+Insert Starts the recorder, Insert Stops recorder and binds automatically)" + ."All subsequent actions after the recording started will be captured." + ."The actions are displayed after the record was stopped." + ."To bind the action you have to click on the 'Bind' button", + on_activate => sub { + my ($btn) = @_; + + unless ($btn->{recording}) { + start; + } else { + stop; + } + }); + $vb->add ($CMDBOX = new CFClient::UI::VBox); + $vb->add (new CFClient::UI::Button + text => "bind", + tooltip => "This opens a query where you have to press the key combination to bind the recorded actions", + on_activate => sub { + open_binding_dialog ([ grep { defined $_ } @$CURRENT_CMDS ]); + }); + + $RECORD_WINDOW +} + 1; =back