ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/server/ext/reseller.ext
Revision: 1.3
Committed: Thu Dec 21 22:41:34 2006 UTC (17 years, 5 months ago) by root
Branch: MAIN
Changes since 1.2: +2 -2 lines
Log Message:
- updated cf.pm to use a more generic and extendable syntax,
  now that it is clear that we will have multiple "attachable" objects.
  maybe bite the bullet in C++ and make attachable virtual?
- completely rework the syntax for attaching and attachments
- update all extensions

File Contents

# User Rev Content
1 root 1.1 #!perl
2    
3     my %unit = (
4     silver => 1,
5     gold => 10,
6     platina => 50,
7     royalty => 5000,
8     );
9    
10     my %aliases = (
11     platinum => 'platina',
12     royalties => 'royalty',
13     );
14    
15 elmex 1.2 sub ob2info {
16     my ($item, $rval) = @_;
17     sprintf "[%s from %s (%d:%d%s) nrof: %d uuid: %s]",
18     $item->name, $item->get_ob_key_value ('ext_reseller_seller'),
19     $item->get_ob_key_value ('ext_reseller_orig_value'), $item->value,
20     (defined $rval ? ":$rval" : ""), $item->nrof, $item->uuid;
21     }
22    
23     sub audit_log {
24     my ($who, $action, $info) = @_;
25     warn
26     sprintf "RESELLER_AUDIT(%s) %s %s: %s\n",
27     $who->map->path, $who->name, $action, $info;
28     }
29    
30 root 1.1 sub find_rec {
31     my ($ob, $cb) = @_;
32    
33     my @unpaid;
34     for my $i ($ob->inv) {
35     push @unpaid, $i if $cb->($i);
36    
37     push @unpaid, find_unpaid ($i, $cb)
38     if $i->inv;
39     }
40     return @unpaid;
41     }
42    
43     sub find_unpaid {
44     my ($ob) = @_;
45     my @r = find_rec ($ob, sub { $_[0]->flag (cf::FLAG_UNPAID) });
46     return @r;
47     }
48    
49     sub find_traded {
50     my ($ob) = @_;
51     my @r = find_rec ($ob, sub { $_[0]->get_ob_key_value ('ext_reseller_seller') ne '' });
52     return @r;
53     }
54    
55     cf::register_script_function "reseller::list_sells" => sub {
56     my ($who, $msg, $npc) = @_;
57     my $sells = cf::from_json $npc->get_ob_key_value ('ext_reseller_sales');
58     my $hissells = $sells->{$who->name};
59    
60     unless (keys %{$hissells || {}}) {
61     $who->reply ($npc, "I'm sorry, but you sold nothing.\n");
62     return 0;
63     }
64    
65     $who->reply ($npc, "You sold:\n", cf::NDI_BROWN);
66     for (keys %$hissells) {
67     my $n = $_;
68     $n =~ s/\s*\(unpaid\)//g;
69     $who->reply ($npc, "$n for " . cf::cost_string_from_value ($hissells->{$_}), cf::NDI_BROWN);
70     }
71    
72     0
73     };
74    
75     cf::register_script_function "reseller::pay_player" => sub {
76     my ($who, $msg, $npc) = @_;
77     my $sells = cf::from_json $npc->get_ob_key_value ('ext_reseller_sales');
78     my $hissells = $sells->{$who->name};
79    
80     unless (keys %{$hissells || {}}) {
81     $who->reply ($npc, "I'm sorry, but you sold nothing.\n");
82     return 0;
83     }
84    
85     my $sum = 0;
86     $sum += $_ for values %$hissells;
87    
88     $who->pay_player ($sum);
89     $who->reply ($npc, "Here are the " . cf::cost_string_from_value ($sum) . " for your sales");
90    
91 elmex 1.2 audit_log ($who, 'collects', "$sum silver");
92    
93 root 1.1 $sells->{$who->name} = {};
94    
95     $npc->set_ob_key_value (ext_reseller_sales => cf::to_json $sells);
96    
97     0
98     };
99    
100 root 1.3 cf::object::attachment "reseller_shopmat",
101 root 1.1 on_move_trigger => sub {
102     my ($self, $who_caused, $who) = @_;
103    
104     my @obs = grep { $_->name eq $self->{reseller_shopmat}{npc_name} }
105     $who->map->at ($self->{reseller_shopmat}{npc_x}, $self->{reseller_shopmat}{npc_y});
106    
107     unless (@obs) {
108 elmex 1.2 warn "Couldn't find shop keeper in " . $who->map->path . "\n";
109 root 1.1 return cf::override;
110     }
111    
112     my $sells = cf::from_json $obs[0]->get_ob_key_value ('ext_reseller_sales');
113    
114     my $unpaid_items = {};
115    
116     for my $item (find_unpaid ($who)) {
117     if ($item->get_ob_key_value ('ext_reseller_seller') eq $who->name) {
118 elmex 1.2 audit_log ($who, 'removes', ob2info ($item));
119 root 1.1 $item->flag (cf::FLAG_UNPAID, 0);
120     $item->remove;
121     $item->insert_ob_in_ob ($who);
122     next;
123     }
124    
125     my $value = $item->query_cost ($who, cf::F_BUY | cf::F_SHOP);
126    
127     warn "Object " . $item->name . " bought by " . $who->name . " on map "
128     . $who->map->path . " for $value silver has no seller set\n"
129     if $item->get_ob_key_value ('ext_reseller_seller') eq '';
130    
131     $unpaid_items->{$item} = [$value, $item];
132     }
133    
134 elmex 1.2 audit_log ($who, 'wants', (join ",", map { ob2info ($_->[1], $_->[0]) } values %$unpaid_items))
135     if %$unpaid_items;
136    
137 root 1.1 $self->apply_shop_mat ($who);
138    
139 elmex 1.2 my @seller_noted;
140    
141 root 1.1 for my $item (find_traded ($who)) {
142     next if $item->flag (cf::FLAG_UNPAID);
143     if (my $value = $unpaid_items->{$item}[0]) {
144 elmex 1.2 push @seller_noted, ob2info ($item, $value)."P";
145 root 1.1 $sells->{$item->get_ob_key_value ('ext_reseller_seller')}->{$item->name} += $value;
146 elmex 1.2 } else {
147     push @seller_noted, ob2info ($item)."T";
148 root 1.1 }
149    
150     $item->value ($item->get_ob_key_value ('ext_reseller_orig_value'));
151     $item->set_ob_key_value (ext_reseller_seller => '');
152     }
153    
154 elmex 1.2 audit_log ($who, 'removed', (join ",", @seller_noted))
155     if @seller_noted;
156    
157 root 1.1 $obs[0]->set_ob_key_value (ext_reseller_sales => cf::to_json $sells);
158    
159     cf::override;
160     },
161     ;
162    
163 root 1.3 cf::object::attachment "reseller_floor",
164 root 1.1 on_drop_on => sub {
165     my ($on, $what, $who) = @_;
166     my $name = $what->custom_name;
167    
168     return if $what->flag (cf::FLAG_UNPAID);
169    
170     if ($what->type == cf::MONEY) {
171     $who->message ("The shopkeeper says: Sorry, you can't sell money here.", cf::NDI_BROWN);
172     $what->insert_ob_in_ob ($who);
173     return cf::override;
174     }
175    
176     if (!$what->flag (cf::FLAG_IDENTIFIED) && $what->need_identify) {
177     $who->message ("The shopkeeper says: Sorry, you can't sell unidentified stuff here.", cf::NDI_BROWN);
178     $what->insert_ob_in_ob ($who);
179     return cf::override;
180     }
181    
182     my $orig_value = $what->value;
183     my $value = 0;
184    
185     if ($name =~ m/\S/) {
186     unless ($name =~ m/\d+\s*\S+/) {
187     $who->message ("The shopkeeper says: Sorry, I don't recognize '$name' as currency. Please name your item like '10 royalty' or '10 platinum 2 silver'", cf::NDI_BROWN);
188     $what->insert_ob_in_ob ($who);
189     return cf::override;
190     }
191    
192     while ($name =~ s/^\s*(\d+)\s*(\S+)//) {
193     if ($aliases{lc $2} or $unit{lc $2}) {
194     $value += $1 * ($unit{lc $2} ? $unit{lc $2} : $unit{$aliases{lc $2}});
195     } else {
196     $what->insert_ob_in_ob ($who);
197     $who->message ("The shopkeeper says: I don't know the currency '$2'", cf::NDI_BROWN);
198     return cf::override;
199     }
200     }
201     } else {
202     $value = $what->query_cost ($who, cf::F_SELL | cf::F_SHOP) / ($what->nrof || 1);
203     }
204    
205     if ($value < 0) {
206     $what->insert_ob_in_ob ($who);
207     $who->message ("The shopkeeper says: You can't sell something for a negative value: $value", cf::NDI_BROWN);
208     return cf::override;
209     }
210    
211     my $fee = $value / 100; # 1% selling fee
212    
213     unless ($who->pay_amount ($fee)) {
214     $who->message (
215     "The shopkeeper says: You need " . cf::cost_string_from_value ($fee) . " to pay the 1% fee for this item",
216     cf::NDI_BROWN
217     );
218     $what->insert_ob_in_ob ($who);
219     return cf::override;
220     } else {
221     $who->message (
222     "The shopkeeper says: Ok, got the fee of " . cf::cost_string_from_value ($fee) . " for the item",
223     cf::NDI_BROWN
224     );
225     }
226    
227     $what->value ($value);
228     my $cost = $what->query_cost ($who, cf::F_BUY | cf::F_SHOP) / ($what->nrof || 1);
229    
230     my $fact = 0;
231     if ($cost) {
232     $fact = $value / $cost;
233     $what->value (cf::ceil ($value * $fact));
234     }
235    
236     # warn "END VALUE: $value * $fact => " . $what->value . "\n";
237    
238     # my $cost = $what->query_cost ($who, cf::F_BUY | cf::F_SHOP) / $what->nrof;
239     # warn "COSTS NOW: $cost\n";
240    
241     $who->message (
242     "The shopkeeper says: Ok, I marked "
243     . ($what->nrof || 1) . " " . $what->name . " to be sold for at least "
244     . cf::cost_string_from_value ($value)
245     . ($what->nrof > 1 ? " each" : ""), cf::NDI_BROWN
246     );
247    
248     $what->set_ob_key_value (ext_reseller_seller => $who->name);
249     $what->set_ob_key_value (ext_reseller_orig_value => $orig_value);
250     # warn "SET SELLER ON " . $what->name . " + " . $what->{seller}->[0] . "\n";
251     $what->custom_name ($what->name . " (by " . $who->name . ")");
252     $what->flag (cf::FLAG_UNPAID, 1);
253     $what->insert_ob_in_map_at ($who->map, $who, cf::INS_BELOW_ORIGINATOR, $who->x, $who->y);
254    
255 elmex 1.2 audit_log ($who, 'sells', ob2info ($what));
256    
257 root 1.1 cf::override;
258     },
259     ;