1 | #! perl |
1 | #! perl |
2 | |
2 | |
3 | use List::Util; |
3 | use List::Util; |
4 | |
4 | |
5 | my %apartment = ( |
5 | our %apartment = ( |
6 | "/scorn/apartment/apartments" => [ 1, "scorn", "skorn"], |
6 | "/scorn/apartment/apartments" => [ 1, "scorn", "skorn"], |
7 | "/santo_dominion/sdomino_appartment" => [ 10, "santo dominion", "domino"], |
7 | "/santo_dominion/sdomino_appartment" => [ 10, "santo dominion", "domino"], |
8 | "/darcap/darcap/apartment" => [ 30, "darcap", "thecap"], |
8 | "/darcap/darcap/apartment" => [ 30, "darcap", "thecap"], |
|
|
9 | "/gotischerbereich/steinwandstadt/sapartment1" => [ 100, "steinwandstadt", "konkret"], |
9 | "/navar_city/apartments/apartment" => [ 250, "navar", "navar"], |
10 | "/navar_city/apartments/apartment" => [ 250, "navar", "navar"], |
|
|
11 | "/celvear_port/tower2/tower2" => [ 300, "celvear port", "kevlar"], |
10 | "/azumauindo/ranbounagisatoshi/apartments/sapartment" => [ 100, "乱暴渚都市", "benjo"], |
12 | "/azumauindo/ranbounagisatoshi/apartments/sapartment" => [ 100, "乱暴渚都市", "benjo"], |
11 | "/azumauindo/suno-yamatoshi/apartments/lapartment1" => [ 1000, "スノー大和島根", "sama"], |
13 | "/azumauindo/suno-yamatoshi/apartments/lapartment1" => [ 1000, "スノー大和島根", "sama"], |
|
|
14 | "/elmex/jeweler/jeweler_inn_upper" => [ 500, "jeweler town", "jewelor"], |
12 | "/pup_land/nurnberg/apartment/main" => [ 300, "nürnberg", "sauerkraut"], |
15 | "/pup_land/nurnberg/apartment/main" => [ 300, "nürnberg", "sauerkraut"], |
|
|
16 | "/lostwages/petapartment" => [ 5000, "lostwages", "losvegas"], |
|
|
17 | "/gotischerbereich/towerapartment/tower1" => [10000, "gotisch", "retreat"], |
|
|
18 | "/lake_country/Butakis/apartment" => [25000, "castle butakis", "expandor"], |
|
|
19 | "/brest/apartments/brest_town_house" => [30000, "brest", "brecht"], |
13 | "/pup_land/lone_town/apartment/groundfloor" => [50000, "lone town", "looney"], |
20 | "/pup_land/lone_town/apartment/groundfloor" => [50000, "lone town", "looney"], |
14 | "/brest/apartments/brest_town_house" => [30000, "brest", "brecht"], |
|
|
15 | ); |
21 | ); |
16 | |
|
|
17 | sub teleport { |
|
|
18 | my ($pl, $map, $x, $y) = @_; |
|
|
19 | |
|
|
20 | my $portal = cf::object::new "exit"; |
|
|
21 | |
|
|
22 | $portal->slaying ($map); |
|
|
23 | $portal->stats->hp ($x); |
|
|
24 | $portal->stats->sp ($y); |
|
|
25 | $portal->apply ($pl->ob); |
|
|
26 | $portal->destroy; |
|
|
27 | } |
|
|
28 | |
22 | |
29 | # we have to special case some special cases :) |
23 | # we have to special case some special cases :) |
30 | sub reject_entry { |
24 | sub reject_entry { |
31 | my ($pl) = @_; |
25 | my ($pl) = @_; |
32 | |
26 | |
|
|
27 | my $prev_pos = $pl->ob->{_prev_pos}; |
|
|
28 | $pl->ob->goto ($prev_pos ? @$prev_pos : ("/world/world_105_115", 2, 34)); |
|
|
29 | |
33 | cf::override; |
30 | cf::override; |
34 | |
|
|
35 | teleport $pl, "/world/world_105_115", 2, 34 |
|
|
36 | unless |
|
|
37 | $pl->ob->map |
|
|
38 | && $pl->ob->map->path !~ /nimbus/ |
|
|
39 | && $pl->ob->map->path !~ m%/var/crossfire/players/%; |
|
|
40 | } |
31 | } |
41 | |
32 | |
42 | sub update_balance { |
33 | sub update_balance { |
43 | my ($pl) = @_; |
34 | my ($pl) = @_; |
44 | |
35 | |
… | |
… | |
60 | } |
51 | } |
61 | |
52 | |
62 | sub pay_balance { |
53 | sub pay_balance { |
63 | my ($pl) = @_; |
54 | my ($pl) = @_; |
64 | |
55 | |
|
|
56 | cf::cede_to_tick; |
|
|
57 | |
65 | update_balance $pl; |
58 | update_balance $pl; |
66 | |
59 | |
67 | return unless $pl->{rent}{balance} > 0; |
60 | return unless $pl->{rent}{balance} > 0; |
68 | |
61 | |
69 | my $deduct = cf::ceil $pl->{rent}{balance}; |
62 | my $deduct = cf::ceil $pl->{rent}{balance}; |
… | |
… | |
72 | |
65 | |
73 | if ($deduct <= $pl->ob->{bank_balance}) { |
66 | if ($deduct <= $pl->ob->{bank_balance}) { |
74 | cf::db_put rent => balance => $deduct + cf::db_get rent => "balance"; |
67 | cf::db_put rent => balance => $deduct + cf::db_get rent => "balance"; |
75 | $pl->ob->{bank_balance} -= $deduct; |
68 | $pl->ob->{bank_balance} -= $deduct; |
76 | $pl->{rent}{balance} -= $deduct; |
69 | $pl->{rent}{balance} -= $deduct; |
77 | $pl->ob->reply (undef, "Something whispers into your ear:\n" |
70 | $pl->ob->reply (undef, "Something whispers into your ear: " |
78 | . "Sir, we deducted your apartment rent ($deduct_string) from your bank account."); |
71 | . "Your highness, we deducted your apartment rent ($deduct_string) from your bank account."); |
79 | } else { |
72 | } else { |
80 | $pl->ob->reply (undef, "Something whispers into your ear:\n" |
73 | $pl->ob->reply (undef, "Something whispers into your ear: " |
81 | . "Sir, we want to deduct the apartment rent ($deduct_string), but the bank informed us that they cannot perform the transaction. " |
74 | . "Your highness, we want to deduct the apartment rent ($deduct_string), but the bank informed us that they cannot perform the transaction. " |
82 | . "Please even out your balance so we can deduct the fees, otherwise we will be forced to shut down your access to the apartment."); |
75 | . "Please even out your balance so we can deduct the fees, otherwise we will be forced to shut down your access to the apartment."); |
83 | } |
76 | } |
84 | } |
77 | } |
85 | |
78 | |
86 | sub check_balance { |
79 | sub check_balance { |
… | |
… | |
104 | my ($pl, $types) = @_; |
97 | my ($pl, $types) = @_; |
105 | |
98 | |
106 | while (my ($k, $v) = each %apartment) { |
99 | while (my ($k, $v) = each %apartment) { |
107 | my $type = exists $pl->{rent}{apartment}{$k} ? 1 : 2; |
100 | my $type = exists $pl->{rent}{apartment}{$k} ? 1 : 2; |
108 | |
101 | |
109 | $pl->ob->reply (undef, "model \"$v->[2]\", situated in $v->[1] (" |
102 | $pl->ob->reply (undef, "model \"$v->[2]\", located in $v->[1] (" |
110 | . (cf::cost_string_from_value $v->[0]) . "/hr)") |
103 | . (cf::cost_string_from_value $v->[0]) . "/hr)\n") |
111 | if $type & $types; |
104 | if $type & $types; |
112 | } |
105 | } |
113 | }; |
106 | }; |
114 | |
107 | |
115 | cf::register_script_function "rent::status" => sub { |
108 | cf::register_script_function "rent::status" => sub { |
… | |
… | |
130 | |
123 | |
131 | update_balance $pl; |
124 | update_balance $pl; |
132 | |
125 | |
133 | $pl->{rent}{apartment}{$apartment} = undef; |
126 | $pl->{rent}{apartment}{$apartment} = undef; |
134 | |
127 | |
135 | $pl->ob->reply (undef, "Wonderful decision, sir! " |
128 | $pl->ob->reply (undef, "Wonderful decision, your highness! " |
136 | . "We told the proprietor in $apartment{$apartment}[1] to expect you and let you in. " |
129 | . "We told the proprietor in $apartment{$apartment}[1] to expect you and let you in. " |
137 | . "We are sure you will be satisfied!"); |
130 | . "We are sure you will be satisfied!"); |
138 | }; |
131 | }; |
139 | |
132 | |
140 | cf::register_script_function "rent::stop" => sub { |
133 | cf::register_script_function "rent::stop" => sub { |
… | |
… | |
158 | $pl->{rent}{last_offline_check} ||= time; |
151 | $pl->{rent}{last_offline_check} ||= time; |
159 | |
152 | |
160 | if ($pl->{rent}{last_online_check}) { |
153 | if ($pl->{rent}{last_online_check}) { |
161 | $pl->{rent}{last_online_check} = time |
154 | $pl->{rent}{last_online_check} = time |
162 | - List::Util::min 3600, |
155 | - List::Util::min 3600, |
163 | $pl->ob->get_ob_key_value ("schmorplog_last_save") - $pl->{rent}{last_online_check}; |
156 | $pl->ob->kv_get ("schmorplog_last_save") - $pl->{rent}{last_online_check}; |
164 | } else { |
157 | } else { |
165 | $pl->{rent}{last_online_check} = time; |
158 | $pl->{rent}{last_online_check} = time; |
166 | } |
159 | } |
167 | |
160 | |
168 | update_balance $pl; |
161 | update_balance $pl; |
… | |
… | |
171 | |
164 | |
172 | cf::map::attachment rent => |
165 | cf::map::attachment rent => |
173 | on_enter => sub { |
166 | on_enter => sub { |
174 | my ($map, $pl, $x, $y) = @_; |
167 | my ($map, $pl, $x, $y) = @_; |
175 | |
168 | |
176 | # can freely enter homes of other people |
169 | return if $pl->ob->flag (cf::FLAG_WIZ); |
177 | { |
|
|
178 | my $path = sprintf "%s/%s/%s/", |
|
|
179 | cf::localdir, cf::playerdir, $pl->ob->name; |
|
|
180 | |
170 | |
|
|
171 | my $pfx = sprintf "~%s/", $pl->ob->name; |
|
|
172 | |
|
|
173 | # only do something if entering ones own apartment |
181 | return if $path ne substr $map->path, 0, length $path; |
174 | if ($pfx eq substr $map->path, 0, length $pfx) { |
|
|
175 | for my $path (keys %{ $pl->{rent}{apartment} }) { |
|
|
176 | $path = sprintf "~%s%s", $pl->ob->name, $path; |
|
|
177 | |
|
|
178 | if ($map->path eq $path) { |
|
|
179 | if (check_balance $pl) { |
|
|
180 | $pl->ob->reply (undef, "Welcome to your apartment, your highness!"); |
|
|
181 | } else { |
|
|
182 | $pl->ob->reply (undef, "We are sorry, your highness, you have to pay your rent first."); |
|
|
183 | reject_entry $pl; |
|
|
184 | } |
|
|
185 | |
|
|
186 | return; |
|
|
187 | } |
|
|
188 | } |
|
|
189 | |
|
|
190 | $pl->ob->reply (undef, "Your highness, you have to rent this apartment in The Apartment Shop in Scorn or other apartment shops first!"); |
|
|
191 | reject_entry $pl; |
182 | } |
192 | } |
183 | |
|
|
184 | for my $path (keys %{ $pl->{rent}{apartment} }) { |
|
|
185 | $path =~ y/\//_/; |
|
|
186 | $path = sprintf "%s/%s/%s/%s", |
|
|
187 | cf::localdir, cf::playerdir, $pl->ob->name, $path; |
|
|
188 | |
|
|
189 | if ($map->path eq $path) { |
|
|
190 | if (check_balance $pl) { |
|
|
191 | $pl->ob->reply (undef, "Welcome to your apartment, sir!"); |
|
|
192 | } else { |
|
|
193 | $pl->ob->reply (undef, "We are sorry, sir, you have to pay your rent first."); |
|
|
194 | reject_entry $pl; |
|
|
195 | } |
|
|
196 | |
|
|
197 | return; |
|
|
198 | } |
|
|
199 | } |
|
|
200 | |
|
|
201 | $pl->ob->reply (undef, "Sir, you have to rent this apartment in The Apartment Shop in Scorn first!"); |
|
|
202 | reject_entry $pl; |
|
|
203 | }, |
193 | }, |
204 | ; |
194 | ; |
205 | |
195 | |
206 | Event->timer (after => 60, interval => 3600, data => cf::WF_AUTOCANCEL, cb => sub { |
196 | our $RENT_TIMER = cf::periodic 3600, Coro::unblock_sub { |
207 | pay_balance $_ for cf::player::list; |
197 | pay_balance $_ for cf::player::list; |
208 | }); |
198 | }; |
209 | |
199 | |