1 | #! perl |
1 | #! perl # depends=irc mandatory |
2 | #CONVERSION: PARTIAL |
|
|
3 | |
2 | |
4 | # implement a replacement for the built-in say/chat/shout/tell/reply commands |
3 | # implement a replacement for the built-in say/chat/shout/tell commands |
5 | # adds ignore/unignore functionality |
4 | # adds ignore/unignore functionality |
6 | |
5 | |
7 | use NPC_Dialogue; |
6 | use NPC_Dialogue; |
8 | use POSIX (); # for strftime only |
7 | use POSIX (); # for strftime only |
|
|
8 | |
|
|
9 | sub tell_channel($) { |
|
|
10 | my ($target) = @_; |
|
|
11 | |
|
|
12 | { |
|
|
13 | id => "tell-$target", |
|
|
14 | title => "$target", |
|
|
15 | reply => "tell $target ", |
|
|
16 | tooltip => "Private messages from/to $target", |
|
|
17 | } |
|
|
18 | } |
|
|
19 | |
|
|
20 | sub send_msg($$$$$) { |
|
|
21 | my ($pl, $channel, $msg, $flags, $sound) = @_; |
|
|
22 | $pl->play_sound (cf::sound::find $sound) if defined $sound; |
|
|
23 | $pl->send_msg ($channel, $msg, $flags); |
|
|
24 | () |
|
|
25 | } |
9 | |
26 | |
10 | sub clean_timeouts($) { |
27 | sub clean_timeouts($) { |
11 | my ($player) = @_; |
28 | my ($player) = @_; |
12 | my $NOW = time; |
29 | my $NOW = time; |
13 | |
30 | |
14 | for my $hash (@$player{qw(ext_ignore_shout ext_ignore_tell)}) { |
31 | for my $hash (@$player{qw(ext_ignore_shout ext_ignore_tell)}) { |
15 | while (my ($k, $v) = each %$hash) { |
32 | while (my ($k, $v) = each %$hash) { |
16 | if ($v < $NOW) { |
33 | if ($v < $NOW) { |
17 | $player->message ("Your ignore on $k has expired.", cf::NDI_GREEN | cf::NDI_UNIQUE); |
34 | $player->message ("Your ignore on $k has expired.", cf::NDI_GREEN); |
18 | delete $hash->{$k}; |
35 | delete $hash->{$k}; |
19 | } elsif (!cf::player::exists $k) { |
36 | } elsif (!cf::player::exists $k) { |
20 | $player->message ("Your ignore on $k is no longer valid (no such user).", cf::NDI_GREEN | cf::NDI_UNIQUE); |
37 | $player->message ("Your ignore on $k is no longer valid (no such user).", cf::NDI_GREEN); |
21 | delete $hash->{$k}; |
38 | delete $hash->{$k}; |
22 | } |
39 | } |
23 | } |
40 | } |
24 | } |
41 | } |
25 | } |
42 | } |
26 | |
43 | |
|
|
44 | # send_irc ($format, @args, $msg) |
|
|
45 | # make sure the last argument is the message! |
|
|
46 | sub send_irc { |
|
|
47 | my ($format, @args) = @_; |
|
|
48 | my $msg = pop @args; |
|
|
49 | for (split /\n/, $msg) { |
|
|
50 | ext::irc::do_notice (sprintf $format, @args, $_) |
|
|
51 | } |
|
|
52 | } |
|
|
53 | |
27 | cf::player->attach ( |
54 | cf::player->attach ( |
28 | prio => -1000, |
55 | prio => -1000, |
29 | on_login => sub { |
56 | on_login => sub { |
30 | my ($pl) = @_; |
57 | my ($pl) = @_; |
31 | |
58 | |
32 | clean_timeouts $pl->ob; |
59 | clean_timeouts $pl->ob; |
|
|
60 | |
|
|
61 | $pl->send_msg ($cf::SAY_CHANNEL); |
|
|
62 | $pl->send_msg ($cf::CHAT_CHANNEL); |
33 | }, |
63 | }, |
34 | ); |
64 | ); |
35 | |
65 | |
|
|
66 | # TODO: remove once safe |
36 | cf::register_command listen => sub { |
67 | cf::register_command listen => sub { |
37 | my ($pl, $msg) = @_; |
|
|
38 | my $player = cf::player::find_active $pl->name; |
|
|
39 | |
|
|
40 | if ($msg ne "") { |
|
|
41 | my $prev_listen = $player->listening; |
|
|
42 | $player->listening ($msg); |
|
|
43 | if ($prev_listen == $player->listening) { |
|
|
44 | $pl->message ("Your verbose level stayed $prev_listen.", cf::NDI_UNIQUE); |
|
|
45 | } else { |
|
|
46 | $pl->message ("Your verbose level is now " . $player->listening . ". (previously: $prev_listen)", cf::NDI_UNIQUE); |
|
|
47 | } |
|
|
48 | } else { |
|
|
49 | $pl->message ("Your verbose level is " . $player->listening . ".", cf::NDI_UNIQUE); |
|
|
50 | } |
|
|
51 | }; |
68 | }; |
52 | |
69 | |
53 | cf::register_command cointoss => sub { |
70 | cf::register_command cointoss => sub { |
54 | my ($pl, $msg) = @_; |
71 | my ($ob, $msg) = @_; |
55 | |
72 | |
|
|
73 | my $pl = $ob->contr; |
56 | my $name = $pl->name; |
74 | my $name = $ob->name; |
57 | |
75 | |
58 | if (int rand 2) { |
76 | my $coin = int rand 2 ? "Heads" : "Tails"; |
|
|
77 | |
|
|
78 | send_msg $_, $cf::SAY_CHANNEL => "$name flips a coin.... $coin!", cf::NDI_GREY, "msg_say" |
59 | for my $other ( grep { $pl->on_same_map_as ($_->ob) } cf::player::list ) { |
79 | for grep { $ob->on_same_map_as ($_->ob) && $_ != $pl} cf::player::list; |
60 | next |
|
|
61 | if $other->ob == $pl; |
|
|
62 | $other->ob->message ("$name flips a coin.... Heads!", cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
63 | } |
|
|
64 | |
|
|
65 | $pl->message ("You flip a coin.... Heads!", cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
66 | } else { |
|
|
67 | for my $other ( grep { $pl->on_same_map_as ($_->ob) } cf::player::list ) { |
|
|
68 | next |
|
|
69 | if $other->ob == $pl; |
|
|
70 | $other->ob->message ("$name flips a coin.... Tails!", cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
71 | } |
|
|
72 | |
|
|
73 | $pl->message ("You flip a coin.... Tails!", cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
74 | } |
80 | |
|
|
81 | $pl->send_msg ($cf::SAY_CHANNEL => "You flip a coin.... $coin!", cf::NDI_GREY | cf::NDI_REPLY); |
|
|
82 | }; |
|
|
83 | |
|
|
84 | cf::register_command orcknuckle => sub { |
|
|
85 | my ($ob, $msg) = @_; |
|
|
86 | |
|
|
87 | my $pl = $ob->contr; |
|
|
88 | my $name = $ob->name; |
|
|
89 | |
|
|
90 | my @orcknuckle = ("beholder", "ghost", "knight", "princess", "dragon", "orc"); |
|
|
91 | my ($i, $j, $k, $l) = (rand 5, rand 5, rand 5, rand 6); |
|
|
92 | my $result = "$orcknuckle[$i], $orcknuckle[$j], $orcknuckle[$k], $orcknuckle[$l]"; |
|
|
93 | |
|
|
94 | send_msg $_, $cf::SAY_CHANNEL => "$name throws his orc-knuckles and rolls $result!", cf::NDI_GREY, "msg_say" |
|
|
95 | for grep { $ob->on_same_map_as ($_->ob) && $_ != $pl} cf::player::list; |
|
|
96 | |
|
|
97 | $pl->send_msg ($cf::SAY_CHANNEL => "You roll $result!", cf::NDI_GREY | cf::NDI_REPLY); |
75 | }; |
98 | }; |
76 | |
99 | |
77 | my $emotes = { |
100 | my $emotes = { |
78 | growl => { |
101 | growl => { |
79 | noparams => { |
102 | noparams => { |
… | |
… | |
203 | self => "You kiss <other>.", |
226 | self => "You kiss <other>.", |
204 | }, |
227 | }, |
205 | self => { |
228 | self => { |
206 | }, |
229 | }, |
207 | }, |
230 | }, |
|
|
231 | smother => { |
|
|
232 | noparams => { |
|
|
233 | other => "<self> makes weird facial contortions", |
|
|
234 | self => "All the lonely people..", |
|
|
235 | }, |
|
|
236 | params => { |
|
|
237 | target => "<self> smothers you with kisses.", |
|
|
238 | other => "<self> smothers <other> with kisses.", |
|
|
239 | self => "You smother <other> with kisses.", |
|
|
240 | }, |
|
|
241 | self => { |
|
|
242 | }, |
|
|
243 | }, |
208 | wink => { |
244 | wink => { |
209 | noparams => { |
245 | noparams => { |
210 | other => "<self> winks suggestively.", |
246 | other => "<self> winks suggestively.", |
211 | self => "Have you got something in your eye?", |
247 | self => "Have you got something in your eye?", |
212 | }, |
248 | }, |
213 | params => { |
249 | params => { |
|
|
250 | target => "<self> winks at you.", |
214 | target => "<self> winks at <other>.", |
251 | other => "<self> winks at <other>.", |
215 | self => "You wink suggestively at <other>.", |
252 | self => "You wink suggestively at <other>.", |
216 | }, |
253 | }, |
217 | self => { |
254 | self => { |
218 | other => "<self> winks at himself - something strange is going on...", |
255 | other => "<self> winks at himself - something strange is going on...", |
219 | self => "You wink at yourself?? What are you up to?", |
256 | self => "You wink at yourself?? What are you up to?", |
… | |
… | |
309 | noparams => { |
346 | noparams => { |
310 | other => "<self> is bleeding all over the carpet - got a spare tourniquet?", |
347 | other => "<self> is bleeding all over the carpet - got a spare tourniquet?", |
311 | self => "You bleed all over your nice new armour.", |
348 | self => "You bleed all over your nice new armour.", |
312 | }, |
349 | }, |
313 | params => { |
350 | params => { |
|
|
351 | target => "<self> slashes his wrist and bleeds all over you.", |
314 | target => "<self> slashes his wrist and bleeds all over <other>.", |
352 | other => "<self> slashes his wrist and bleeds all over <other>.", |
315 | self => "You slash your wrist and bleed all over <other>", |
353 | self => "You slash your wrist and bleed all over <other>", |
316 | }, |
354 | }, |
317 | self => { |
355 | self => { |
318 | other => "<self> performs some satanic ritual while wiping his blood on himself.", |
356 | other => "<self> performs some satanic ritual while wiping his blood on himself.", |
319 | self => "Very impressive! You wipe your blood all over yourself.", |
357 | self => "Very impressive! You wipe your blood all over yourself.", |
… | |
… | |
654 | }, |
692 | }, |
655 | }; |
693 | }; |
656 | |
694 | |
657 | for my $emotion (keys %$emotes) { |
695 | for my $emotion (keys %$emotes) { |
658 | cf::register_command $emotion => sub { |
696 | cf::register_command $emotion => sub { |
659 | my ($pl, $tname) = @_; |
697 | my ($ob, $tname) = @_; |
|
|
698 | |
|
|
699 | my $pl = $ob->contr; |
660 | |
700 | |
661 | cf::async { |
701 | cf::async { |
662 | my $name = $pl->name; |
702 | my $name = $ob->name; |
|
|
703 | $Coro::current->{desc} = "emote handler for $name"; |
663 | |
704 | |
664 | if ($tname eq $name) { |
705 | if ($tname eq $name) { |
665 | my $emote = $emotes->{$emotion}->{self}; |
706 | my %emote = %{ $emotes->{$emotion}->{self} || {} }; |
666 | |
707 | |
667 | $emote->{other} = "You look away from <self>." |
708 | $emote{other} ||= "You look away from <self>."; |
668 | if !$emote->{other}; |
|
|
669 | $emote->{self} = "My god! Is that LEGAL?" |
709 | $emote{self} ||= "My god! Is that LEGAL?"; |
670 | if !$emote->{self}; |
|
|
671 | |
710 | |
672 | $emote->{other} =~ s/<self>/$name/; |
711 | $emote{other} =~ s/<self>/$name/; |
673 | |
712 | |
|
|
713 | send_msg $_, $cf::CHAT_CHANNEL, $emote{other}, cf::NDI_GREY, "msg_chat" |
674 | for my $other ( grep { $pl->on_same_map_as ($_->ob) } cf::player::list ) { |
714 | for grep { $ob->on_same_map_as ($_->ob) && $_ != $ob} cf::player::list; |
675 | next |
|
|
676 | if $other->ob == $pl; |
|
|
677 | $other->ob->message ($emote->{other}, cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
678 | } |
|
|
679 | |
715 | |
680 | $pl->message ($emote->{self}, cf::NDI_GREY | cf::NDI_UNIQUE); |
716 | $pl->send_msg ($emote{self}, cf::NDI_GREY | cf::NDI_REPLY); |
681 | } elsif ($tname) { |
717 | } elsif ($tname) { |
682 | my $target = cf::player::find $tname |
718 | my $target = cf::player::find $tname |
683 | or return $pl->reply (undef, "$tname is not around."); |
719 | or return send_msg $pl, tell_channel $tname, "$tname is not around.", cf::NDI_DK_ORANGE | cf::NDI_REPLY, "msg_chat"; |
684 | |
720 | |
685 | my $emote = $emotes->{$emotion}->{params}; |
721 | my %emote = %{ $emotes->{$emotion}->{params} || {} }; |
686 | |
722 | |
687 | $emote->{other} = "<self> is eyeing <other> quizzically." |
723 | $emote{other} ||= "<self> is eyeing <other> quizzically."; |
688 | if !$emote->{other}; |
|
|
689 | $emote->{self} = "You are still nuts." |
724 | $emote{self} ||= "You are still nuts."; |
690 | if !$emote->{self}; |
|
|
691 | $emote->{target} = "You get the distinct feeling that <other> is nuts." |
725 | $emote{target} ||= "You get the distinct feeling that <self> is nuts."; |
692 | if !$emote->{target}; |
|
|
693 | |
726 | |
694 | $emote->{self} =~ s/<other>/$tname/; |
727 | $emote{self} =~ s/<other>/$tname/; |
695 | $emote->{target} =~ s/<self>/$name/; |
728 | $emote{target} =~ s/<self>/$name/; |
696 | $emote->{other} =~ s/<other>/$tname/; |
729 | $emote{other} =~ s/<other>/$tname/; |
697 | $emote->{other} =~ s/<self>/$name/; |
730 | $emote{other} =~ s/<self>/$name/; |
698 | |
731 | |
699 | for my $other ( grep { $pl->on_same_map_as ($_->ob) } cf::player::list ) { |
732 | send_msg $_, $cf::CHAT_CHANNEL, $emote{other}, cf::NDI_GREY, "msg_chat" |
700 | next |
733 | for grep { $_ != $pl && $_ != $target && $ob->on_same_map_as ($_->ob) } cf::player::list; |
701 | if $other->ob == $pl or $other == $target; |
|
|
702 | $other->ob->message ($emote->{other}, cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
703 | } |
|
|
704 | |
734 | |
705 | $target->ob->message ($emote->{target}, cf::NDI_GREY | cf::NDI_UNIQUE); |
735 | send_msg $target, tell_channel $name, $emote{target}, cf::NDI_GREY, "msg_shout"; |
706 | $pl->message ($emote->{self}, cf::NDI_GREY | cf::NDI_UNIQUE); |
736 | $pl->send_msg (tell_channel $tname, $emote{self}, cf::NDI_GREY | cf::NDI_REPLY); |
707 | } else { |
737 | } else { |
708 | my $emote = $emotes->{$emotion}->{noparams}; |
738 | my %emote = %{ $emotes->{$emotion}->{noparams} || {} }; |
709 | $emote->{other} =~ s/<self>/$name/; |
|
|
710 | |
739 | |
711 | $emote->{other} = "<self> dances with glee." |
740 | $emote{other} ||= "<self> dances with glee."; |
712 | if !$emote->{other}; |
|
|
713 | $emote->{self} = "You are a nut." |
741 | $emote{self} ||= "You are a nut."; |
714 | if !$emote->{self}; |
|
|
715 | |
742 | |
|
|
743 | $emote{other} =~ s/<self>/$name/; |
|
|
744 | |
|
|
745 | send_msg $_, $cf::CHAT_CHANNEL, $emote{other}, cf::NDI_GREY, "msg_chat" |
716 | for my $other ( grep { $pl->on_same_map_as ($_->ob) } cf::player::list ) { |
746 | for grep { $ob->on_same_map_as ($_->ob) && $_ != $pl } cf::player::list; |
717 | next |
|
|
718 | if $other->ob == $pl; |
|
|
719 | $other->ob->message ($emote->{other}, cf::NDI_GREY | cf::NDI_UNIQUE); |
|
|
720 | } |
|
|
721 | |
747 | |
722 | $pl->message ($emote->{self}, cf::NDI_GREY | cf::NDI_UNIQUE); |
748 | $pl->send_msg ($cf::CHAT_CHANNEL, $emote{self}, cf::NDI_GREY | cf::NDI_REPLY); |
723 | } |
749 | } |
724 | }; |
750 | }; |
725 | }; |
751 | }; |
726 | } |
752 | } |
727 | |
753 | |
728 | cf::register_command me => sub { |
754 | cf::register_command me => sub { |
729 | my ($pl, $msg) = @_; |
755 | my ($pl, $msg) = @_; |
730 | |
756 | |
731 | my $name = $pl->name; |
757 | my $name = $pl->name; |
732 | |
758 | |
733 | $_->ob->message ("* $name $msg", cf::NDI_GREY | cf::NDI_UNIQUE) |
759 | send_msg $_, $cf::SAY_CHANNEL => "* $name $msg", cf::NDI_GREY | cf::NDI_DEF | ($_ == $pl ? cf::NDI_REPLY : 0), "msg_say" |
734 | for grep $pl->on_same_map_as ($_->ob), cf::player::list; |
760 | for grep $pl->on_same_map_as ($_->ob), cf::player::list; |
735 | }; |
761 | }; |
736 | |
762 | |
737 | cf::register_command say => sub { |
763 | cf::register_command say => sub { |
738 | my ($pl, $msg) = @_; |
764 | my ($ob, $msg) = @_; |
739 | |
765 | |
740 | utf8::decode $msg; |
766 | utf8::decode $msg; |
741 | |
767 | |
742 | return if $pl->contr->invoke (cf::EVENT_PLAYER_SAY, $msg); |
768 | return if $ob->contr->invoke (cf::EVENT_PLAYER_SAY, $msg); |
743 | |
769 | |
744 | if ($msg) { |
770 | if ($msg) { |
745 | my $name = $pl->name; |
771 | my $name = $ob->name; |
746 | |
|
|
747 | utf8::encode $msg; # ->message not yet utf8-ified |
|
|
748 | $_->ob->message ("$name says: $msg", cf::NDI_GREY | cf::NDI_UNIQUE) |
|
|
749 | for grep $pl->on_same_map_as ($_->ob), cf::player::list; |
772 | my @plonmap = grep $ob->on_same_map_as ($_->ob), cf::player::list; |
750 | utf8::decode $msg; |
773 | |
|
|
774 | send_msg $_, $cf::SAY_CHANNEL => "$name says: $msg", cf::NDI_GREY, "msg_say" |
|
|
775 | for grep $_ != $ob->contr, @plonmap; |
|
|
776 | $ob->contr->send_msg ($cf::SAY_CHANNEL => "$name says: $msg", cf::NDI_GREY | cf::NDI_REPLY); |
751 | |
777 | |
752 | # npcs, magic_ears etc. |
778 | # npcs, magic_ears etc. |
753 | # first find all objects and theirt-level inventories |
779 | # first find all objects and their first-level inventories |
754 | # within a 5x5 square # that have something resembling |
780 | # within a 5x5 square that have something resembling |
755 | # dialogue or support on_say. |
781 | # dialogue or support on_say. |
756 | my ($map, $x, $y) = ($pl->map, $pl->x - 2, $pl->y - 2); |
782 | my ($map, $x, $y) = ($ob->map, $ob->x - 2, $ob->y - 2); |
757 | |
783 | |
758 | for my $npc ( |
784 | for my $npc ( |
759 | grep +($_->invoke (cf::EVENT_OBJECT_SAY, $pl->contr, $msg) && return) || NPC_Dialogue::has_dialogue $_, |
785 | grep +($_->invoke (cf::EVENT_OBJECT_SAY, $ob->contr, $msg) && return) || $_->has_dialogue, |
760 | map +($_, $_->inv), |
786 | map +($_, $_->inv), |
761 | grep $_, |
787 | grep $_, |
762 | map $map->at ($x + $_ % 5, $y + (int $_ / 5)), |
788 | map $map->at ($x + $_ % 5, $y + (int $_ / 5)), |
763 | 0..24 |
789 | 0..24 |
764 | ) { |
790 | ) { |
765 | # if some listener teleported us somewhere else, stop right here |
791 | # if some listener teleported us somewhere else, stop right here |
766 | last unless $map->path == $pl->map->path; |
792 | last unless $map->path == $ob->map->path; |
767 | |
793 | |
768 | my $dialog = new NPC_Dialogue ob => $pl, npc => $npc; |
794 | my $dialog = new NPC_Dialogue pl => $ob->contr, npc => $npc; |
769 | my ($reply, @kw) = $dialog->tell ($msg); |
795 | my ($reply, @kw) = $dialog->tell ($msg); |
770 | |
796 | |
771 | if (defined $reply) { |
797 | if (defined $reply) { |
772 | if ($npc->type == cf::MAGIC_EAR) { |
798 | if ($npc->type == cf::MAGIC_EAR) { |
773 | if (length $reply) { |
799 | if (length $reply) { |
774 | $_->ob->message ($reply, cf::NDI_BROWN | cf::NDI_UNIQUE) |
800 | send_msg $_, $cf::SAY_CHANNEL => $reply, cf::NDI_BROWN, "msg_say" |
775 | for grep $pl->on_same_map_as ($_->ob), cf::player::list; |
801 | for @plonmap; |
776 | } |
802 | } |
777 | $npc->use_trigger; |
803 | $npc->use_trigger; |
778 | } else { |
804 | } else { |
779 | if (length $reply) { |
805 | if (length $reply) { |
780 | $_->ob->message ($npc->name . " says: $reply", cf::NDI_BROWN | cf::NDI_UNIQUE) |
806 | send_msg $_, $cf::SAY_CHANNEL => $npc->name . " says: $reply", cf::NDI_BROWN, "msg_say" |
781 | for grep $pl->on_same_map_as ($_->ob), cf::player::list; |
807 | for @plonmap; |
782 | } |
808 | } |
783 | } |
809 | } |
784 | } |
810 | } |
785 | |
811 | |
786 | if (@kw) { |
812 | if (@kw) { |
787 | $_->ob->message ("[further topics: " . (join ", ", @kw) . "]", cf::NDI_BROWN | cf::NDI_UNIQUE) |
813 | $_->send_msg ($cf::SAY_CHANNEL => "[further topics: " . (join ", ", @kw) . "]", cf::NDI_BROWN) |
788 | for grep $pl->on_same_map_as ($_->ob), cf::player::list; |
814 | for @plonmap; |
789 | } |
815 | } |
790 | } |
816 | } |
791 | |
817 | |
792 | } else { |
818 | } else { |
793 | $pl->message ("What do you want to say?", cf::NDI_UNIQUE); |
819 | $ob->send_msg ($cf::SAY_CHANNEL => "What do you want to say?", cf::NDI_GREY | cf::NDI_REPLY); |
794 | } |
820 | } |
795 | }; |
821 | }; |
796 | |
822 | |
797 | cf::register_command chat => sub { |
823 | cf::register_command chat => sub { |
798 | my ($pl, $msg) = @_; |
824 | my ($ob, $msg) = @_; |
799 | |
825 | |
800 | utf8::decode $msg; |
826 | utf8::decode $msg; |
801 | |
827 | |
|
|
828 | my $pl = $ob->contr; |
|
|
829 | |
802 | return if $pl->contr->invoke (cf::EVENT_PLAYER_CHAT, $msg); |
830 | return if $pl->invoke (cf::EVENT_PLAYER_CHAT, $msg); |
803 | |
831 | |
804 | if ($msg) { |
832 | if ($msg) { |
805 | my $name = $pl->name; |
833 | my $name = $ob->name; |
806 | my $NOW = time; |
834 | my $NOW = time; |
807 | |
835 | |
808 | utf8::encode $msg; # ->message not yet utf8-ified |
|
|
809 | cf::LOG cf::llevDebug, sprintf "QBERT [%s] %s\n", $name, $msg; |
836 | cf::LOG cf::llevDebug, sprintf "QBERT [%s] %s\n", $name, $msg; |
810 | ext::schmorp_irc::do_notice (sprintf "[%s] %s", $name, $msg); |
837 | send_irc ("[%s] %s", $name, $msg); |
811 | |
838 | |
812 | $_->ob->message ("$name chats: $msg", cf::NDI_BLUE) |
839 | send_msg $_, $cf::CHAT_CHANNEL => "$name chats: $msg", cf::NDI_BLUE | cf::NDI_DEF | ($_ == $pl ? cf::NDI_REPLY : 0), "msg_chat" |
813 | for grep { $_->ob->{ext_ignore_shout}{$name} < $NOW && $_->listening >= 10 } cf::player::list; |
840 | for grep { $_->ob->{ext_ignore_shout}{$name} < $NOW } cf::player::list; |
814 | |
841 | |
815 | } else { |
842 | } else { |
816 | $pl->message ("Chat what?", cf::NDI_UNIQUE); |
843 | $pl->send_msg ($cf::CHAT_CHANNEL => "Chat what?", cf::NDI_BLUE | cf::NDI_DEF | cf::NDI_REPLY); |
817 | } |
844 | } |
818 | }; |
845 | }; |
819 | |
846 | |
820 | cf::register_command shout => sub { |
847 | cf::register_command shout => sub { |
821 | my ($pl, $msg) = @_; |
848 | my ($ob, $msg) = @_; |
822 | |
849 | |
823 | utf8::decode $msg; |
850 | utf8::decode $msg; |
824 | |
851 | |
|
|
852 | my $pl = $ob->contr; |
|
|
853 | |
825 | return if $pl->contr->invoke (cf::EVENT_PLAYER_SHOUT, $msg); |
854 | return if $pl->invoke (cf::EVENT_PLAYER_SHOUT, $msg); |
826 | |
855 | |
827 | if ($msg) { |
856 | if ($msg) { |
828 | my $NOW = time; |
857 | my $NOW = time; |
829 | my $name = $pl->name; |
858 | my $name = $ob->name; |
830 | |
859 | |
831 | cf::LOG cf::llevDebug, sprintf "QBERT {%s} %s\n", $name, $msg; |
860 | cf::LOG cf::llevDebug, sprintf "QBERT {%s} %s\n", $name, $msg; |
832 | ext::schmorp_irc::do_notice (sprintf "\007\0034{%s} %s\n", $name, $msg); |
861 | send_irc ("\007\0034{%s} %s\n", $name, $msg); |
833 | |
862 | |
834 | utf8::encode $msg; # ->message not yet utf8-ified |
863 | send_msg $_, $cf::CHAT_CHANNEL => "$name shouts: $msg", cf::NDI_RED | cf::NDI_DEF | ($_ == $pl ? cf::NDI_REPLY : 0), "msg_shout" |
835 | $_->ob->message ("$name shouts: $msg", cf::NDI_RED) |
|
|
836 | for grep { $_->ob->{ext_ignore_shout}{$name} < $NOW && $_->listening >= 2 } cf::player::list; |
864 | for grep { $_->ob->{ext_ignore_shout}{$name} < $NOW } cf::player::list; |
837 | |
865 | |
838 | } else { |
866 | } else { |
839 | $pl->message ("Shout what?", cf::NDI_UNIQUE); |
867 | $pl->send_msg ($cf::CHAT_CHANNEL => "Shout what?", cf::NDI_RED | cf::NDI_DEF | cf::NDI_REPLY); |
840 | } |
868 | } |
841 | }; |
869 | }; |
842 | |
870 | |
843 | cf::register_command tell => sub { |
871 | cf::register_command tell => sub { |
844 | my ($pl, $args) = @_; |
872 | my ($ob, $args) = @_; |
845 | my ($target, $msg) = split /\s+/, $args, 2; |
873 | my ($target, $msg) = split /\s+/, $args, 2; |
846 | |
874 | |
847 | utf8::decode $msg; |
875 | utf8::decode $msg; |
848 | |
876 | |
|
|
877 | my $pl = $ob->contr; |
|
|
878 | my $ns = $pl->ns |
|
|
879 | or return; |
|
|
880 | my $name = $ob->name; |
|
|
881 | |
849 | return if $pl->contr->invoke (cf::EVENT_PLAYER_TELL, $target, $msg); |
882 | return if $pl->invoke (cf::EVENT_PLAYER_TELL, $target, $msg); |
850 | |
883 | |
851 | my $name = $pl->name; |
884 | my $pl_channel = tell_channel $target; |
852 | |
885 | |
853 | if ($target =~ /irc\//) { |
886 | if ($target =~ /irc\//) { |
854 | my (undef, $nick) = split /\//, $target, 2; |
887 | my (undef, $nick) = split /\//, $target, 2; |
855 | $pl->message ("You tell $target: $args"); |
888 | $ns->send_msg ($pl_channel => "You tell $target: $args", cf::NDI_DK_ORANGE | cf::NDI_DEF | cf::NDI_REPLY); |
856 | ext::schmorp_irc::do_notice (sprintf "(%s) %s: %s\n", $name, $nick, $msg); |
889 | send_irc ("(%s) %s: %s\n", $name, $nick, $msg); |
|
|
890 | |
857 | } elsif (my $other = cf::player::find_active $target) { |
891 | } elsif (my $other = cf::player::find_active $target) { |
|
|
892 | my $other_channel = tell_channel $name; |
858 | |
893 | |
859 | if ($msg) { |
894 | if ($msg) { |
860 | if ($target eq $name) { |
895 | if ($target eq $name) { |
861 | $pl->message ("You are talking to yourself, you freak!", cf::NDI_UNIQUE); |
896 | $ns->send_msg ($pl_channel => "You are talking to yourself, you freak!", cf::NDI_DK_ORANGE | cf::NDI_DEF | cf::NDI_REPLY); |
862 | } elsif ($other->ob->{ext_ignore_tell}{$name} >= time) { |
897 | } elsif ($other->ob->{ext_ignore_tell}{$name} >= time) { |
863 | $pl->message ("$target ignores what you say. Give up on it.", cf::NDI_UNIQUE); |
898 | $ns->send_msg ($pl_channel => "$target ignores what you say. Give up on it.", cf::NDI_DK_ORANGE | cf::NDI_DEF | cf::NDI_REPLY); |
864 | } else { |
899 | } else { |
865 | utf8::encode $msg; # ->message not yet utf8-ified |
900 | return if $other->invoke (cf::EVENT_PLAYER_TOLD, $pl, $msg); |
866 | cf::LOG cf::llevDebug, sprintf "TELL [%s>%s] %s\n", $name, $target, $msg; |
901 | cf::LOG cf::llevDebug, sprintf "TELL [%s>%s] %s\n", $name, $target, $msg; |
867 | |
902 | |
868 | $pl->message ("You tell $target: $msg"); |
903 | $ns->send_msg ($pl_channel => "You tell $target: $msg", cf::NDI_DK_ORANGE | cf::NDI_DEF | cf::NDI_REPLY); |
869 | $other->ob->message ("$name tells you: $msg"); |
904 | send_msg $other, $other_channel => "$name tells you: $msg", cf::NDI_DK_ORANGE | cf::NDI_DEF, "msg_tell"; |
870 | $other->ob->{ext_last_tell} = $name; |
|
|
871 | } |
905 | } |
872 | } else { |
906 | } else { |
873 | $pl->message ("What do you want to tell $target?", cf::NDI_UNIQUE); |
907 | $ns->send_msg ($pl_channel => "What do you want to tell $target?", cf::NDI_DK_ORANGE | cf::NDI_DEF | cf::NDI_REPLY); |
874 | } |
908 | } |
875 | |
909 | |
876 | } else { |
910 | } else { |
877 | $pl->message ("No such player. Your message: $msg", cf::NDI_UNIQUE); |
911 | $ns->send_msg ($pl_channel => "No such player. Your message: $msg", cf::NDI_DK_ORANGE | cf::NDI_DEF | cf::NDI_REPLY); |
878 | } |
|
|
879 | }; |
|
|
880 | |
|
|
881 | cf::register_command reply => sub { |
|
|
882 | my ($pl, $args) = @_; |
|
|
883 | my $name = $pl->name; |
|
|
884 | |
|
|
885 | utf8::decode $args; |
|
|
886 | |
|
|
887 | return if $pl->contr->invoke (cf::EVENT_PLAYER_TELL, $pl->{ext_last_tell}, $args); |
|
|
888 | |
|
|
889 | if ($pl->{ext_last_tell} =~ /irc\//) { |
|
|
890 | my (undef, $nick) = split /\//, $pl->{ext_last_tell}, 2; |
|
|
891 | $pl->message ("You tell " . $pl->{ext_last_tell} . ": $args"); |
|
|
892 | ext::schmorp_irc::do_notice (sprintf "(%s) %s: %s\n", $name, $nick, $args); |
|
|
893 | } elsif (my $other = cf::player::find_active $pl->{ext_last_tell}) { |
|
|
894 | |
|
|
895 | if ($args) { |
|
|
896 | $other->ob->{ext_ignore_tell}{$name} >= time |
|
|
897 | or delete $other->ob->{ext_ignore_tell}{$name}; |
|
|
898 | |
|
|
899 | if ($other->ob->{ext_ignore_tell}{$name} < time) { |
|
|
900 | utf8::encode $args; # ->message not yet utf8-ified |
|
|
901 | cf::LOG cf::llevDebug, sprintf "TELL [%s>%s] %s\n", $name, $other->ob->name, $args; |
|
|
902 | |
|
|
903 | $pl->message ("You tell " . $other->ob->name . ": $args"); |
|
|
904 | $other->ob->message ("$name tells you: $args"); |
|
|
905 | $pl->{ext_last_tell} = $other->ob->name; |
|
|
906 | } else { |
|
|
907 | $pl->message ($other->ob->name . " ignores what you say. Give up on it.", cf::NDI_UNIQUE); |
|
|
908 | } |
|
|
909 | } else { |
|
|
910 | $pl->message ("What do you want to tell ".$other->ob->name."?", cf::NDI_UNIQUE); |
|
|
911 | } |
|
|
912 | |
|
|
913 | } else { |
|
|
914 | $pl->message ("Can't reply, player left. Your message: $args".$pl->{ext_last_tell}, cf::NDI_UNIQUE); |
|
|
915 | } |
912 | } |
916 | }; |
913 | }; |
917 | |
914 | |
918 | cf::register_command ignore => sub { |
915 | cf::register_command ignore => sub { |
919 | my ($pl, $args) = @_; |
916 | my ($pl, $args) = @_; |
… | |
… | |
922 | if ($args eq "list") { |
919 | if ($args eq "list") { |
923 | clean_timeouts $pl; |
920 | clean_timeouts $pl; |
924 | |
921 | |
925 | if ((my @ignored_tell = sort keys %{$pl->{ext_ignore_tell}}) |
922 | if ((my @ignored_tell = sort keys %{$pl->{ext_ignore_tell}}) |
926 | + (my @ignored_shout = sort keys %{$pl->{ext_ignore_shout}})) { |
923 | + (my @ignored_shout = sort keys %{$pl->{ext_ignore_shout}})) { |
927 | $pl->message ("Currently ignoring private messages from: ", cf::NDI_UNIQUE); |
924 | $pl->message ("Currently ignoring private messages from: ", cf::NDI_REPLY); |
928 | $pl->message ((join ", ", @ignored_tell), cf::NDI_UNIQUE); |
925 | $pl->message ((join ", ", @ignored_tell), cf::NDI_REPLY); |
929 | $pl->message ("Currently ignoring shouts from: ", cf::NDI_UNIQUE); |
926 | $pl->message ("Currently ignoring shouts from: ", cf::NDI_REPLY); |
930 | $pl->message ((join ", ", @ignored_shout), cf::NDI_UNIQUE); |
927 | $pl->message ((join ", ", @ignored_shout), cf::NDI_REPLY); |
931 | $pl->message ("To stop ignoring one, use unignore.", cf::NDI_UNIQUE); |
928 | $pl->message ("To stop ignoring one, use unignore.", cf::NDI_REPLY); |
932 | } else { |
929 | } else { |
933 | $pl->message ("Not ignoring anyone", cf::NDI_UNIQUE); |
930 | $pl->message ("Not ignoring anyone", cf::NDI_REPLY); |
934 | } |
931 | } |
935 | |
932 | |
936 | } elsif ($target && $type) { |
933 | } elsif ($target && $type) { |
937 | |
934 | |
938 | $timeout ne "" or $timeout = 24; |
935 | $timeout ne "" or $timeout = 24; |
939 | my $absolute_timeout = time + $timeout * 3600; |
936 | my $absolute_timeout = time + $timeout * 3600; |
940 | |
937 | |
941 | if (cf::player::exists $target) { |
938 | if (cf::player::exists $target) { |
942 | if ($type eq "tell") { |
939 | if ($type eq "tell") { |
943 | $pl->message ("Now ignoring private messages from $target for $timeout hours.", cf::NDI_UNIQUE); |
940 | $pl->message ("Now ignoring private messages from $target for $timeout hours.", cf::NDI_REPLY); |
944 | $pl->{ext_ignore_tell}{$target} = $absolute_timeout; |
941 | $pl->{ext_ignore_tell}{$target} = $absolute_timeout; |
945 | } elsif ($type eq "shout") { |
942 | } elsif ($type eq "shout") { |
946 | $pl->message ("Now ignoring shouts from $target for $timeout hours.", cf::NDI_UNIQUE); |
943 | $pl->message ("Now ignoring shouts from $target for $timeout hours.", cf::NDI_REPLY); |
947 | $pl->{ext_ignore_shout}{$target} = $absolute_timeout; |
944 | $pl->{ext_ignore_shout}{$target} = $absolute_timeout; |
948 | } elsif ($type eq "all") { |
945 | } elsif ($type eq "all") { |
949 | $pl->message ("Now ignoring everything from $target for $timeout hours.", cf::NDI_UNIQUE); |
946 | $pl->message ("Now ignoring everything from $target for $timeout hours.", cf::NDI_REPLY); |
950 | $pl->{ext_ignore_tell}{$target} = $absolute_timeout; |
947 | $pl->{ext_ignore_tell}{$target} = $absolute_timeout; |
951 | $pl->{ext_ignore_shout}{$target} = $absolute_timeout; |
948 | $pl->{ext_ignore_shout}{$target} = $absolute_timeout; |
952 | } else { |
949 | } else { |
953 | $pl->message ("You need to specify tell, shout or all.", cf::NDI_UNIQUE); |
950 | $pl->message ("You need to specify tell, shout or all.", cf::NDI_REPLY); |
954 | } |
951 | } |
955 | } else { |
952 | } else { |
956 | $pl->message ("No such player: $target", cf::NDI_UNIQUE); |
953 | $pl->message ("No such player: $target", cf::NDI_REPLY); |
957 | } |
954 | } |
958 | |
955 | |
959 | } else { |
956 | } else { |
960 | $pl->message ("Usage: ignore <player> <tell|shout|all> <timeout>\n" |
957 | $pl->message ("Usage: ignore <player> <tell|shout|all> <timeout>\n" |
961 | . "will ignore a player for <timeout> hours.\n" |
958 | . "will ignore a player for <timeout> hours.\n" |
962 | . "Usage: ignore list\n" |
959 | . "Usage: ignore list\n" |
963 | . "will show you a list of players currently ignored.", cf::NDI_UNIQUE); |
960 | . "will show you a list of players currently ignored.", cf::NDI_REPLY); |
964 | } |
961 | } |
965 | }; |
962 | }; |
966 | |
963 | |
967 | cf::register_command unignore => sub { |
964 | cf::register_command unignore => sub { |
968 | my ($pl, $args) = @_; |
965 | my ($pl, $args) = @_; |
969 | my ($target, $type) = split /\s+/, $args; |
966 | my ($target, $type) = split /\s+/, $args; |
970 | |
967 | |
971 | if ($args eq "") { |
968 | if ($args eq "") { |
972 | if ($pl->{ext_ignore_tell}) { |
969 | if ($pl->{ext_ignore_tell}) { |
973 | $pl->message ("Currently ignoring private messages from: ", cf::NDI_UNIQUE); |
970 | $pl->message ("Currently ignoring private messages from: ", cf::NDI_REPLY); |
974 | $pl->message ((join ", ", sort keys %{ $pl->{ext_ignore_tell} }), cf::NDI_UNIQUE); |
971 | $pl->message ((join ", ", sort keys %{ $pl->{ext_ignore_tell} }), cf::NDI_REPLY); |
975 | $pl->message ("Currently ignoring shouts from: ", cf::NDI_UNIQUE); |
972 | $pl->message ("Currently ignoring shouts from: ", cf::NDI_REPLY); |
976 | $pl->message ((join ", ", sort keys %{ $pl->{ext_ignore_shout} }), cf::NDI_UNIQUE); |
973 | $pl->message ((join ", ", sort keys %{ $pl->{ext_ignore_shout} }), cf::NDI_REPLY); |
977 | } else { |
974 | } else { |
978 | $pl->message ("Not ignoring anyone", cf::NDI_UNIQUE); |
975 | $pl->message ("Not ignoring anyone", cf::NDI_REPLY); |
979 | } |
976 | } |
980 | } else { |
977 | } else { |
981 | if (cf::player::exists $target) { |
978 | if (cf::player::exists $target) { |
982 | if ($type eq "tell") { |
979 | if ($type eq "tell") { |
983 | $pl->message ("Not ignoring private messages from $target anymore.", cf::NDI_UNIQUE); |
980 | $pl->message ("Not ignoring private messages from $target anymore.", cf::NDI_REPLY); |
984 | delete $pl->{ext_ignore_tell} {$target}; |
981 | delete $pl->{ext_ignore_tell} {$target}; |
985 | } elsif ($type eq "shout") { |
982 | } elsif ($type eq "shout") { |
986 | $pl->message ("Not ignoring shouts from $target anymore.", cf::NDI_UNIQUE); |
983 | $pl->message ("Not ignoring shouts from $target anymore.", cf::NDI_REPLY); |
987 | delete $pl->{ext_ignore_shout}{$target}; |
984 | delete $pl->{ext_ignore_shout}{$target}; |
988 | } elsif ($type eq "all") { |
985 | } elsif ($type eq "all") { |
989 | $pl->message ("Not ignoring anything from $target anymore.", cf::NDI_UNIQUE); |
986 | $pl->message ("Not ignoring anything from $target anymore.", cf::NDI_REPLY); |
990 | delete $pl->{ext_ignore_tell} {$target}; |
987 | delete $pl->{ext_ignore_tell} {$target}; |
991 | delete $pl->{ext_ignore_shout}{$target}; |
988 | delete $pl->{ext_ignore_shout}{$target}; |
992 | } else { |
989 | } else { |
993 | $pl->message ("You need to specify tell, shout or all.", cf::NDI_UNIQUE); |
990 | $pl->message ("You need to specify tell, shout or all.", cf::NDI_REPLY); |
994 | } |
991 | } |
995 | } else { |
992 | } else { |
996 | $pl->message ("No such player or ambiguous name: $target", cf::NDI_UNIQUE); |
993 | $pl->message ("No such player or ambiguous name: $target", cf::NDI_REPLY); |
997 | } |
994 | } |
998 | } |
995 | } |
999 | }; |
996 | }; |
1000 | |
997 | |
1001 | cf::register_command seen => sub { |
|
|
1002 | my ($pl, $args) = @_; |
|
|
1003 | |
|
|
1004 | if (my ($login) = $args =~ /(\S+)/) { |
|
|
1005 | if ($login eq $pl->name) { |
|
|
1006 | $pl->message ("Very funny, $login. Ha. Ha.", cf::NDI_UNIQUE); |
|
|
1007 | } elsif (cf::player::find_active $login) { |
|
|
1008 | $pl->message ("$login is right here on this server!", cf::NDI_UNIQUE); |
|
|
1009 | } elsif (cf::player::exists $login |
|
|
1010 | and stat sprintf "%s/%s/%s/%s.pl", cf::localdir, cf::playerdir, ($login) x 2) { |
|
|
1011 | my $time = (stat _)[9]; |
|
|
1012 | |
|
|
1013 | $pl->message ("$login was last seen here " |
|
|
1014 | . (POSIX::strftime "%Y-%m-%d %H:%M:%S +0000", gmtime $time) |
|
|
1015 | . " which was " . (int +(time - $time) / 3600) . " hours ago.", cf::NDI_UNIQUE); |
|
|
1016 | } else { |
|
|
1017 | $pl->message ("No player named $login is known to me.", cf::NDI_UNIQUE); |
|
|
1018 | } |
|
|
1019 | } else { |
|
|
1020 | $pl->message ("Usage: seen <player>", cf::NDI_UNIQUE); |
|
|
1021 | } |
|
|
1022 | }; |
|
|
1023 | |
|
|