1 // Copyright (C) 2014, 2015 Kazuhiro Fujieda <fujieda@users.osdn.me>
\r
3 // Licensed under the Apache License, Version 2.0 (the "License");
\r
4 // you may not use this file except in compliance with the License.
\r
5 // You may obtain a copy of the License at
\r
7 // http://www.apache.org/licenses/LICENSE-2.0
\r
9 // Unless required by applicable law or agreed to in writing, software
\r
10 // distributed under the License is distributed on an "AS IS" BASIS,
\r
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
\r
12 // See the License for the specific language governing permissions and
\r
13 // limitations under the License.
\r
16 using System.Collections.Generic;
\r
19 namespace KancolleSniffer
\r
21 public class AkashiTimer
\r
23 private readonly ShipInfo _shipInfo;
\r
24 private readonly DockInfo _dockInfo;
\r
25 private readonly PresetDeck _presetDeck;
\r
26 private readonly RepairStatus[] _repairStatuses = new RepairStatus[ShipInfo.FleetCount];
\r
27 private DateTime _start;
\r
28 private DateTime _prev;
\r
30 public class RepairSpan
\r
32 public int Diff { get; set; }
\r
33 public TimeSpan Span { get; set; }
\r
35 public RepairSpan(int diff, TimeSpan span)
\r
42 private class RepairStatus
\r
44 private ShipStatus[] _target = new ShipStatus[0];
\r
45 private int[] _deck = new int[0];
\r
49 set { _deck = value; }
\r
52 public State State { get; set; }
\r
54 public bool IsRepaired(ShipStatus[] target) => _target.Zip(target, (a, b) => a.NowHp < b.NowHp).Any(x => x);
\r
56 public bool DeckChanged(IEnumerable<int> deck) => !_deck.SequenceEqual(deck);
\r
58 public void UpdateTarget(ShipStatus[] target)
\r
63 public RepairSpan[] GetTimers(DateTime start, DateTime now)
\r
65 var spent = TimeSpan.FromSeconds((int)(now - start).TotalSeconds);
\r
66 return _target.Select(s =>
\r
68 var damage = s.MaxHp - s.NowHp;
\r
70 return new RepairSpan(0, TimeSpan.MinValue);
\r
71 if (spent < TimeSpan.FromMinutes(20))
\r
72 return new RepairSpan(0, TimeSpan.FromMinutes(20) - spent);
\r
74 return new RepairSpan(1, TimeSpan.Zero);
\r
75 for (var d = 2; d <= damage; d++)
\r
77 var sec = s.CalcRepairSec(d) + 60;
\r
80 if (TimeSpan.FromSeconds(sec) > spent)
\r
81 return new RepairSpan(d - 1, TimeSpan.FromSeconds(sec) - spent);
\r
83 return new RepairSpan(damage, TimeSpan.Zero);
\r
87 public Notice GetNotice(DateTime start, DateTime prev, DateTime now)
\r
89 var m20 = TimeSpan.FromMinutes(20);
\r
90 var proc = new List<string>();
\r
91 var comp = new List<string>();
\r
92 foreach (var s in _target)
\r
94 var damage = s.MaxHp - s.NowHp;
\r
99 if (prev - start < m20 && now - start >= m20)
\r
103 // スリープで時間が飛んだときに修理完了だけを表示するために、
\r
104 // 完全回復から減らしながら所要時間と経過時間と比較する。
\r
105 for (var d = damage; d >= 2; d--)
\r
107 var sec = s.CalcRepairSec(d) + 70;
\r
108 if (sec <= 20 * 60)
\r
110 if (d == damage && (prev - start < m20 && now - start >= m20))
\r
114 var span = TimeSpan.FromSeconds(sec);
\r
115 if (span <= prev - start || now - start < span)
\r
126 Proceeded = proc.Count == 0 ? "" : string.Join(" ", proc),
\r
127 Completed = comp.Count == 0 ? "" : string.Join(" ", comp)
\r
132 public class Notice
\r
134 public string Proceeded { get; set; }
\r
135 public string Completed { get; set; }
\r
138 public AkashiTimer(ShipInfo ship, DockInfo dock, PresetDeck preset)
\r
142 _presetDeck = preset;
\r
143 for (var i = 0; i < _repairStatuses.Length; i++)
\r
144 _repairStatuses[i] = new RepairStatus();
\r
156 var now = DateTime.Now;
\r
157 var reset = _repairStatuses.Any(r => r.State == State.Reset);
\r
158 if (_start == DateTime.MinValue || now - _start > TimeSpan.FromMinutes(20) || reset)
\r
162 public void InspectChange(string request)
\r
165 var values = HttpUtility.ParseQueryString(request);
\r
166 if (int.Parse(values["api_ship_idx"]) == -1)
\r
168 if (_repairStatuses.Any(r => r.State == State.Reset))
\r
169 _start = DateTime.Now;
\r
172 public void CheckFleet()
\r
174 for (var fleet = 0; fleet < ShipInfo.FleetCount; fleet++)
\r
178 private void CheckFleet(int fleet)
\r
180 var deck = _shipInfo.GetDeck(fleet).ToArray();
\r
181 var repair = _repairStatuses[fleet];
\r
182 var fs = _shipInfo.GetStatus(deck[0]);
\r
183 repair.State = State.Continue;
\r
184 if (!fs.Spec.IsRepairShip)
\r
186 repair.UpdateTarget(new ShipStatus[0]);
\r
187 repair.Deck = deck;
\r
190 if (repair.DeckChanged(deck))
\r
192 repair.State = State.Reset;
\r
193 repair.Deck = deck;
\r
195 var target = RepairTarget(deck);
\r
196 if (repair.IsRepaired(target))
\r
197 repair.State = State.Reset;
\r
198 repair.UpdateTarget(target);
\r
201 private ShipStatus[] RepairTarget(int[] deck)
\r
203 var fs = _shipInfo.GetStatus(deck[0]);
\r
204 if (!fs.Spec.IsRepairShip || _dockInfo.InNDock(fs.Id) || fs.DamageLevel >= ShipStatus.Damage.Half)
\r
205 return new ShipStatus[0];
\r
206 var cap = fs.Slot.Count(item => item.Spec.IsRepairFacility) + 2;
\r
208 * 泊地修理の条件を満たさない艦はMaxHp==NowHpのダミーを設定する。
\r
209 * - 入渠中の艦娘は終わったときに回復扱いされないようNowHp=MaxHpに
\r
210 * - 中破以上でNowHp=MaxHpにすると回復扱いされるのでNowHp=MaxHp=0に
\r
212 return (from id in deck.Take(cap)
\r
213 let s = (ShipStatus)_shipInfo.GetStatus(id).Clone()
\r
214 let full = new ShipStatus {NowHp = s.MaxHp, MaxHp = s.MaxHp}
\r
215 let zero = new ShipStatus()
\r
216 select _dockInfo.InNDock(id) ? full : s.DamageLevel >= ShipStatus.Damage.Half ? zero : s).ToArray();
\r
219 public RepairSpan[] GetTimers(int fleet)
\r
220 => _start == DateTime.MinValue ? new RepairSpan[0] : _repairStatuses[fleet].GetTimers(_start, DateTime.Now);
\r
222 public TimeSpan PresetDeckTimer
\r
226 if (_start == DateTime.MinValue)
\r
227 return TimeSpan.MinValue;
\r
228 var r = TimeSpan.FromMinutes(20) - TimeSpan.FromSeconds((int)(DateTime.Now - _start).TotalSeconds);
\r
229 return r >= TimeSpan.Zero ? r : TimeSpan.Zero;
\r
233 public bool CheckReparing(int fleet) => GetTimers(fleet).Any(r => r.Span != TimeSpan.MinValue);
\r
235 public bool CheckReparing() => Enumerable.Range(0, ShipInfo.FleetCount).Any(CheckReparing);
\r
237 public bool CheckPresetReparing()
\r
238 => _presetDeck.Decks.Where(deck => deck != null)
\r
239 .Any(deck => RepairTarget(deck).Any(s => s.NowHp < s.MaxHp));
\r
241 public Notice[] GetNotice()
\r
243 var now = DateTime.Now;
\r
246 if (prev == DateTime.MinValue || _start == DateTime.MinValue)
\r
247 return new Notice[0];
\r
248 var r = _repairStatuses.Select(repair => repair.GetNotice(_start, prev, now)).ToArray();
\r
249 var m20 = TimeSpan.FromMinutes(20);
\r
250 if (prev - _start < m20 && now - _start >= m20)
\r
251 r[0].Proceeded = "20分経過しました。";
\r