onCast(FightCastScope)   A
last analyzed

Complexity

Conditions 2

Size

Total Lines 4
Code Lines 4

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 2

Importance

Changes 0
Metric Value
eloc 4
c 0
b 0
f 0
dl 0
loc 4
ccs 2
cts 2
cp 1
rs 10
cc 2
crap 2
1
/*
2
 * This file is part of Araknemu.
3
 *
4
 * Araknemu is free software: you can redistribute it and/or modify
5
 * it under the terms of the GNU Lesser General Public License as published by
6
 * the Free Software Foundation, either version 3 of the License, or
7
 * (at your option) any later version.
8
 *
9
 * Araknemu is distributed in the hope that it will be useful,
10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
 * GNU Lesser General Public License for more details.
13
 *
14
 * You should have received a copy of the GNU Lesser General Public License
15
 * along with Araknemu.  If not, see <https://www.gnu.org/licenses/>.
16
 *
17
 * Copyright (c) 2017-2019 Vincent Quatrevieux
18
 */
19
20
package fr.quatrevieux.araknemu.game.fight.castable.effect.buff;
21
22
import fr.quatrevieux.araknemu.game.fight.castable.FightCastScope;
23
import fr.quatrevieux.araknemu.game.fight.castable.effect.EffectValue;
24
import fr.quatrevieux.araknemu.game.fight.castable.effect.handler.damage.Damage;
25
import fr.quatrevieux.araknemu.game.fight.castable.effect.handler.damage.ReflectedDamage;
26
import fr.quatrevieux.araknemu.game.fight.fighter.Fighter;
27
import fr.quatrevieux.araknemu.game.fight.turn.Turn;
28
import fr.quatrevieux.araknemu.network.game.out.fight.AddBuff;
29
import fr.quatrevieux.araknemu.util.SafeLinkedList;
30
import org.checkerframework.checker.index.qual.Positive;
31
32
import java.util.Iterator;
33
import java.util.function.Predicate;
34
import java.util.stream.Stream;
35
import java.util.stream.StreamSupport;
36
37
/**
38
 * Handle buff list for a fighter
39
 */
40
public final class BuffList implements Iterable<Buff>, Buffs {
41
    private final Fighter fighter;
42 1
    private final SafeLinkedList<Buff> buffs = new SafeLinkedList<>();
43
44 1
    public BuffList(Fighter fighter) {
45 1
        this.fighter = fighter;
46 1
    }
47
48
    @Override
49
    public Iterator<Buff> iterator() {
50 1
        return buffs.iterator();
51
    }
52
53
    /**
54
     * Get the Buff stream
55
     */
56
    public Stream<Buff> stream() {
57 1
        return StreamSupport.stream(spliterator(), false);
58
    }
59
60
    @Override
61
    public void add(Buff buff) {
62 1
        buffs.add(buff);
63 1
        buff.hook().onBuffStarted(buff);
64
65 1
        if (buff.remainingTurns() == 0) {
66 1
            buff.incrementRemainingTurns();
67
        }
68
69 1
        fighter.fight().send(new AddBuff(buff));
70
71
        // Add one turn when it's the turn of the current fighter
72 1
        if (fighter.isPlaying()) {
73 1
            buff.incrementRemainingTurns();
74
        }
75 1
    }
76
77
    @Override
78
    public boolean onStartTurn() {
79 1
        boolean result = true;
80
81 1
        for (Buff buff : buffs) {
82 1
            result &= buff.hook().onStartTurn(buff);
83 1
        }
84
85 1
        return result;
86
    }
87
88
    @Override
89
    public void onEndTurn(Turn turn) {
90 1
        for (Buff buff : buffs) {
91 1
            buff.hook().onEndTurn(buff, turn);
92 1
        }
93 1
    }
94
95
    @Override
96
    public void onCast(FightCastScope cast) {
97 1
        for (Buff buff : buffs) {
98 1
            buff.hook().onCast(buff, cast);
99 1
        }
100 1
    }
101
102
    @Override
103
    public boolean onCastTarget(FightCastScope cast) {
104 1
        for (Buff buff : buffs) {
105 1
            if (!buff.hook().onCastTarget(buff, cast)) {
106 1
                return false;
107
            }
108 1
        }
109
110 1
        return true;
111
    }
112
113
    @Override
114
    public void onDirectDamage(Fighter caster, Damage value) {
115 1
        for (Buff buff : buffs) {
116 1
            buff.hook().onDirectDamage(buff, caster, value);
117 1
        }
118 1
    }
119
120
    @Override
121
    public void onIndirectDamage(Fighter caster, Damage value) {
122 1
        for (Buff buff : buffs) {
123 1
            buff.hook().onIndirectDamage(buff, caster, value);
124 1
        }
125 1
    }
126
127
    @Override
128
    public void onBuffDamage(Buff poison, Damage value) {
129 1
        for (Buff buff : buffs) {
130 1
            buff.hook().onBuffDamage(buff, poison, value);
131 1
        }
132 1
    }
133
134
    @Override
135
    public void onDirectDamageApplied(Fighter caster, @Positive int value) {
136 1
        for (Buff buff : buffs) {
137 1
            buff.hook().onDirectDamageApplied(buff, caster, value);
138 1
        }
139 1
    }
140
141
    @Override
142
    public void onLifeAltered(int value) {
143 1
        for (Buff buff : buffs) {
144 1
            buff.hook().onLifeAltered(buff, value);
145 1
        }
146 1
    }
147
148
    @Override
149
    public void onReflectedDamage(ReflectedDamage damage) {
150 1
        for (Buff buff : buffs) {
151 1
            buff.hook().onReflectedDamage(buff, damage);
152 1
        }
153 1
    }
154
155
    @Override
156
    public void onCastDamage(Damage damage, Fighter target) {
157 1
        for (Buff buff : buffs) {
158 1
            buff.hook().onCastDamage(buff, damage, target);
159 1
        }
160 1
    }
161
162
    @Override
163
    public void onEffectValueCast(EffectValue value) {
164 1
        for (Buff buff : buffs) {
165 1
            buff.hook().onEffectValueCast(buff, value);
166 1
        }
167 1
    }
168
169
    @Override
170
    public void onEffectValueTarget(EffectValue value) {
171 1
        for (Buff buff : buffs) {
172 1
            buff.hook().onEffectValueTarget(buff, value);
173 1
        }
174 1
    }
175
176
    @Override
177
    public void refresh() {
178
        // invalidate buffs before removing it in case of buffs are iterated by onBuffTerminated() hook
179 1
        buffs.forEach(Buff::decrementRemainingTurns);
180 1
        removeIf(buff -> !buff.valid());
181 1
    }
182
183
    @Override
184
    public boolean removeAll() {
185 1
        return removeIf(Buff::canBeDispelled);
186
    }
187
188
    @Override
189
    public boolean removeByCaster(Fighter caster) {
190 1
        return removeIf(buff -> buff.caster().equals(caster));
191
    }
192
193
    /**
194
     * Remove buff by a predicate
195
     *
196
     * @param predicate Takes the buff as parameter, and return true to delete (and terminate) the buff
197
     *
198
     * @return true if there is a change (i.e. a buff is terminated)
199
     */
200
    private boolean removeIf(Predicate<Buff> predicate) {
201 1
        final Iterator<Buff> iterator = buffs.iterator();
202
203 1
        boolean hasChanged = false;
204
205 1
        while (iterator.hasNext()) {
206 1
            final Buff buff = iterator.next();
207
208 1
            if (predicate.test(buff)) {
209 1
                iterator.remove();
210 1
                buff.hook().onBuffTerminated(buff);
211 1
                hasChanged = true;
212
            }
213 1
        }
214
215 1
        return hasChanged;
216
    }
217
}
218