1 |
root |
1.1 |
#! perl |
2 |
|
|
|
3 |
root |
1.15 |
use strict; |
4 |
|
|
|
5 |
root |
1.1 |
use Jeweler; |
6 |
|
|
use List::Util qw/max min sum/; |
7 |
|
|
|
8 |
|
|
sub ingred_alias { |
9 |
|
|
my ($ing) = @_; |
10 |
|
|
|
11 |
|
|
my %aliases = ( |
12 |
|
|
pow => 'power', |
13 |
|
|
cha => 'charisma', |
14 |
|
|
wis => 'wisdom', |
15 |
|
|
int => 'intelligence', |
16 |
|
|
dex => 'dexterity', |
17 |
|
|
con => 'constitution', |
18 |
|
|
str => 'strength', |
19 |
elmex |
1.7 |
gem => 'diamond', |
20 |
root |
1.1 |
); |
21 |
|
|
|
22 |
|
|
if ($ing =~ m/resist_(\S+)/) { |
23 |
|
|
my $a = $aliases{lc $1} || $1; |
24 |
|
|
"something for '". lc ($a). "' resistance"; |
25 |
|
|
|
26 |
|
|
} elsif ($ing =~ m/stat_(\S+)/) { |
27 |
|
|
my $a = $aliases{lc $1} || $1; |
28 |
|
|
"something for the ". lc ($a). " stat"; |
29 |
|
|
|
30 |
|
|
} elsif ($ing =~ m/spec_(\S+)/) { |
31 |
|
|
my $a = $aliases{lc $1} || $1; |
32 |
|
|
"something for the ". lc ($a). "' special"; |
33 |
|
|
|
34 |
elmex |
1.7 |
} elsif ($aliases{$ing}) { |
35 |
|
|
$aliases{$ing} |
36 |
|
|
|
37 |
root |
1.1 |
} else { |
38 |
|
|
$ing |
39 |
|
|
} |
40 |
|
|
} |
41 |
|
|
|
42 |
|
|
my $DEBUG = 1; |
43 |
|
|
|
44 |
|
|
sub merge { |
45 |
|
|
my ($chdl, $sk, $pl, $do_analyze) = @_; |
46 |
|
|
|
47 |
elmex |
1.5 |
my $ingred = get_ingred ($pl, $chdl) || return; |
48 |
elmex |
1.4 |
|
49 |
root |
1.1 |
my @ring = $ingred->get_ring; |
50 |
|
|
my @rings = map { Jeweler::Object->new (object => $_) } @ring; |
51 |
|
|
|
52 |
|
|
@rings >= 2 |
53 |
|
|
or return $pl->reply (undef, "You slap yourself, you forgot to put at least 2 jewels in!"); |
54 |
|
|
|
55 |
elmex |
1.12 |
my $input_level = 0; |
56 |
elmex |
1.7 |
my $value; |
57 |
|
|
for (@rings) { |
58 |
|
|
$input_level = max ($_->power_to_level, $input_level); |
59 |
|
|
$value += $_->{hash}->{value}; |
60 |
|
|
} |
61 |
elmex |
1.5 |
|
62 |
root |
1.1 |
my $ring = shift @rings; |
63 |
|
|
$ring->improve_by_ring (@rings); |
64 |
|
|
|
65 |
|
|
if ($do_analyze) { |
66 |
elmex |
1.12 |
$pl->reply (undef, "You want to make a " . $ring->to_string . ": " . $ring->analyze ($sk, $pl, $input_level)); |
67 |
root |
1.1 |
$ring->wiz_analyze ($pl) |
68 |
|
|
if $pl->flag (cf::FLAG_WIZ); |
69 |
|
|
return; |
70 |
|
|
} |
71 |
|
|
|
72 |
elmex |
1.7 |
make_ring ($chdl, $ingred, $ring, $value, $sk, $pl, $input_level); |
73 |
root |
1.1 |
} |
74 |
|
|
|
75 |
|
|
sub make_ring { |
76 |
elmex |
1.7 |
my ($chdl, $ingred, $ring, $value, $sk, $pl, $input_level) = @_; |
77 |
root |
1.1 |
|
78 |
|
|
if (!$pl->flag (cf::FLAG_WIZ)) { |
79 |
|
|
$ingred->remove ('rings'); |
80 |
|
|
$ingred->remove ('ammys'); |
81 |
|
|
} |
82 |
|
|
|
83 |
|
|
my $ch = $ring->get_chance_perc ($sk); |
84 |
|
|
my $succ = 0; |
85 |
|
|
my $r = cf::random_roll (0, 100, $pl, cf::PREFER_HIGH); |
86 |
elmex |
1.5 |
|
87 |
elmex |
1.7 |
my $make_status; |
88 |
|
|
my $exp; |
89 |
|
|
|
90 |
root |
1.1 |
if ($r <= $ch or $pl->flag (cf::FLAG_WIZ)) { |
91 |
elmex |
1.12 |
$exp = $ring->projected_exp ($input_level); |
92 |
elmex |
1.5 |
|
93 |
root |
1.1 |
$pl->change_exp ($exp, "jeweler", cf::SK_EXP_SKILL_ONLY); |
94 |
elmex |
1.7 |
$pl->message ("You succeed and get $exp experience points."); |
95 |
|
|
$make_status = "succeeded"; |
96 |
|
|
|
97 |
|
|
$ring->set_value ($value * 0.8); # 20% of the input values will vanish |
98 |
|
|
|
99 |
root |
1.1 |
} else { |
100 |
|
|
$pl->message ("You fail!"); |
101 |
|
|
$ring->negate; |
102 |
elmex |
1.7 |
$make_status = "fail"; |
103 |
|
|
$exp = 0; |
104 |
|
|
} |
105 |
|
|
|
106 |
|
|
my $ring_ob = $ring->to_object; |
107 |
|
|
|
108 |
|
|
{ # some audit info calculation |
109 |
|
|
my $sklvl = cf::exp_to_level ($sk->stats->exp); |
110 |
|
|
|
111 |
|
|
my $make_info = sprintf |
112 |
|
|
"JEWELER AUDIT: '%s' made '%s' (%s) (sk lvl %d, ring lvl %d, got %d exp): %s", |
113 |
|
|
$pl->name, $ring->to_string, $ring_ob->uuid, $sklvl, |
114 |
|
|
$ring->power_to_level, $exp, $make_status; |
115 |
|
|
|
116 |
|
|
warn "$make_info\n" if $make_status eq 'succeeded'; |
117 |
root |
1.1 |
} |
118 |
elmex |
1.7 |
|
119 |
|
|
$chdl->put ($ring_ob); |
120 |
root |
1.1 |
} |
121 |
|
|
|
122 |
elmex |
1.5 |
sub get_ingred { |
123 |
|
|
my ($pl, $chdl) = @_; |
124 |
|
|
my $ingred = eval { $chdl->extract_jeweler_ingredients }; |
125 |
|
|
if ($@ =~ /cursed/) { |
126 |
|
|
$pl->message ("There are cursed items in the workbench, take them out before you do anything."). |
127 |
|
|
return |
128 |
|
|
} elsif ($@ =~ /unidentified/) { |
129 |
|
|
$pl->message ("There are unidentified items in the workbench, identify them before you do anything."). |
130 |
|
|
return |
131 |
|
|
} elsif ($@) { |
132 |
|
|
warn "error in jeweler ingredient extraction: $@"; |
133 |
|
|
return; |
134 |
|
|
} |
135 |
|
|
$ingred; |
136 |
|
|
} |
137 |
|
|
|
138 |
elmex |
1.6 |
cf::object::attachment check_ring_drop_on => |
139 |
|
|
on_drop_on => sub { |
140 |
|
|
my ($self, $obj, $who) = @_; |
141 |
|
|
my $cfg = $self->{check_ring_drop_on}; |
142 |
|
|
if ($obj->type == cf::RING |
143 |
|
|
&& !$obj->flag (cf::FLAG_CURSED) |
144 |
|
|
&& !$obj->flag (cf::FLAG_DAMNED) |
145 |
|
|
) { |
146 |
|
|
my $ringo = Jeweler::Object->new (object => $obj); |
147 |
|
|
for (grep { /^resist_/ } keys %$cfg) { |
148 |
|
|
if (/^resist_(\S+)$/) { |
149 |
|
|
if ($ringo->has_resist ($1)) { |
150 |
|
|
$self->map->trigger ( |
151 |
|
|
$cfg->{connection}, |
152 |
|
|
$cfg->{state} |
153 |
|
|
); |
154 |
|
|
cf::override; |
155 |
|
|
} |
156 |
|
|
} |
157 |
|
|
} |
158 |
|
|
} |
159 |
|
|
}; |
160 |
|
|
|
161 |
root |
1.2 |
cf::object->attach ( |
162 |
|
|
type => cf::SKILL, |
163 |
|
|
subtype => cf::SK_JEWELER, |
164 |
root |
1.1 |
on_use_skill => sub { |
165 |
|
|
my ($sk, $ob, $part, $dir, $msg) = @_; |
166 |
|
|
my $pl = $ob; |
167 |
|
|
|
168 |
|
|
my $skobj = $sk; |
169 |
|
|
|
170 |
|
|
my $chdl = new Jeweler::CauldronHandler; |
171 |
|
|
|
172 |
|
|
my $rv = 1; |
173 |
|
|
eval { |
174 |
|
|
$DEBUG = 1; |
175 |
|
|
|
176 |
|
|
my $player = $ob->contr; |
177 |
|
|
|
178 |
|
|
unless ($chdl->find_cauldron ('jeweler_bench', $ob->map->at ($ob->x, $ob->y))) { |
179 |
|
|
return; |
180 |
|
|
} |
181 |
|
|
|
182 |
|
|
cf::override; |
183 |
|
|
|
184 |
|
|
if ($msg =~ m/^\s*analy[sz]e\s*$/i) { |
185 |
|
|
Jeweler::analyze ($sk, $chdl, $pl); |
186 |
|
|
|
187 |
|
|
} elsif ($msg =~ m/^\s*make\s*$/i) { |
188 |
|
|
$pl->message ("You can make: " . (join ', ', keys %{Jeweler::getcfg ('conversions') || {}})); |
189 |
|
|
|
190 |
|
|
} elsif ($msg =~ m/^\s*make\s+(\S+)\s*$/i) { |
191 |
elmex |
1.5 |
my $ingred = get_ingred ($pl, $chdl) || return; |
192 |
root |
1.1 |
|
193 |
|
|
unless ($Jeweler::CFG->{conversions}->{lc $1}) { |
194 |
|
|
$pl->message ("You don't know how to make '$1', is does such a thing even exist?"); |
195 |
|
|
return |
196 |
|
|
} |
197 |
|
|
|
198 |
|
|
Jeweler::simple_converter ($player, $ingred, $chdl, $1); |
199 |
|
|
|
200 |
|
|
} elsif ($msg =~ m/^\s*merge\s*analy[sz]e\s*$/i) { |
201 |
|
|
merge ($chdl, $sk, $pl, 1); |
202 |
|
|
|
203 |
|
|
} elsif ($msg =~ m/^\s*merge\s*$/i) { |
204 |
|
|
merge ($chdl, $sk, $pl, 0); |
205 |
|
|
|
206 |
|
|
} else { |
207 |
elmex |
1.5 |
my $ingred = get_ingred ($pl, $chdl) || return; |
208 |
elmex |
1.4 |
|
209 |
root |
1.1 |
my $plan = $ingred->get_plan; |
210 |
|
|
|
211 |
|
|
if ($plan) { |
212 |
|
|
my @ring = $ingred->get_ring; |
213 |
|
|
|
214 |
elmex |
1.14 |
if (!@ring) { |
215 |
root |
1.1 |
# actually the algorithm cant |
216 |
|
|
$pl->message ("You slap yourself, you forgot the jewelery!"); |
217 |
|
|
return; |
218 |
|
|
|
219 |
elmex |
1.14 |
} elsif ((@ring > 1) || (grep { $_->nrof > 1 } @ring)) { |
220 |
|
|
# actually the algorithm cant handle more than one improvement at a time |
221 |
|
|
$pl->message ("You can't manage to improve more than one thing at a time!"); |
222 |
|
|
return; |
223 |
root |
1.1 |
} else { |
224 |
|
|
my $ringo = Jeweler::Object->new (object => $ring[0]); |
225 |
|
|
my $iring = $ingred->improve_ring_by_plan ($plan, $ringo); |
226 |
elmex |
1.7 |
my $c1 = $ringo->calc_costs; |
227 |
|
|
my $c2 = $iring->calc_costs; |
228 |
|
|
my $value = $iring->calc_value_from_cost ($c2); |
229 |
root |
1.1 |
|
230 |
elmex |
1.12 |
if ((not defined $c1) || (not defined $c2)) { |
231 |
|
|
$pl->message ("The jewel has or will become a resistancy above 99%,\n" |
232 |
|
|
."that is completly impossible to make!"); |
233 |
|
|
return; |
234 |
|
|
} |
235 |
|
|
|
236 |
root |
1.1 |
my %keys; |
237 |
|
|
my %cdiff; |
238 |
|
|
for (keys %$c1, keys %$c2) { $keys{$_} = 1 } |
239 |
elmex |
1.7 |
for (keys %keys) { $cdiff{$_} = $c2->{$_} - $c1->{$_}; } |
240 |
root |
1.1 |
|
241 |
elmex |
1.3 |
unless ($iring->is_better_than ($ringo)) { |
242 |
root |
1.1 |
$pl->message ("This plan doesn't improve anything, you find yourself puzzled about what you missed..."); |
243 |
|
|
return; |
244 |
|
|
} |
245 |
|
|
|
246 |
|
|
my $remcosts = $ingred->check_costs (\%cdiff); |
247 |
|
|
|
248 |
|
|
if (grep { $_ > 0 } values %$remcosts) { |
249 |
|
|
$pl->message ("You want to make a " . $iring->to_string . ": " . $iring->analyze ($sk, $pl)); |
250 |
|
|
$pl->message ("You recognize that you are short of: " |
251 |
|
|
. (join ", ", |
252 |
|
|
map { my $cost = $remcosts->{$_}; $cost . " " . ($cost > 1 ? "times" : "time") . " " . ingred_alias ($_) } |
253 |
|
|
grep { $remcosts->{$_} > 0 } keys %$remcosts)); |
254 |
|
|
|
255 |
|
|
if ($pl->flag (cf::FLAG_WIZ)) { |
256 |
|
|
$iring->wiz_analyze ($pl); |
257 |
|
|
} |
258 |
|
|
} else { |
259 |
|
|
if (!$pl->flag (cf::FLAG_WIZ)) { |
260 |
|
|
$ingred->check_costs (\%cdiff, 1); |
261 |
|
|
} |
262 |
elmex |
1.7 |
make_ring ($chdl, $ingred, $iring, $value, $sk, $pl); |
263 |
root |
1.1 |
} |
264 |
|
|
} |
265 |
|
|
} else { |
266 |
|
|
$pl->message ("You've got no idea what you are planning to do!"); |
267 |
|
|
} |
268 |
|
|
} |
269 |
|
|
}; |
270 |
|
|
$@ and warn "ERROR: $@\n"; |
271 |
|
|
} |
272 |
root |
1.2 |
); |
273 |
root |
1.1 |
|
274 |
root |
1.15 |
Jeweler::read_config "res/jeweler.yaml"; |
275 |
root |
1.10 |
|