OSDN Git Service

ShutdownProxyを非同期に実行する
[kancollesniffer/KancolleSniffer.git] / KancolleSniffer / MainForm.cs
1 // Copyright (C) 2013, 2014, 2015 Kazuhiro Fujieda <fujieda@users.osdn.me>\r
2 // \r
3 // This program is part of KancolleSniffer.\r
4 //\r
5 // KancolleSniffer is free software: you can redistribute it and/or modify\r
6 // it under the terms of the GNU General Public License as published by\r
7 // the Free Software Foundation, either version 3 of the License, or\r
8 // (at your option) any later version.\r
9 //\r
10 // This program is distributed in the hope that it will be useful,\r
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of\r
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
13 // GNU General Public License for more details.\r
14 //\r
15 // You should have received a copy of the GNU General Public License\r
16 // along with this program; if not, see <http://www.gnu.org/licenses/>.\r
17 \r
18 using System;\r
19 using System.Collections.Generic;\r
20 using System.Diagnostics;\r
21 using System.Drawing;\r
22 using System.IO;\r
23 using System.Linq;\r
24 using System.Runtime.InteropServices;\r
25 using System.Text;\r
26 using System.Windows.Forms;\r
27 using System.Threading.Tasks;\r
28 using Codeplex.Data;\r
29 using Nekoxy;\r
30 \r
31 namespace KancolleSniffer\r
32 {\r
33     public partial class MainForm : Form\r
34     {\r
35         private readonly Sniffer _sniffer = new Sniffer();\r
36         private readonly Config _config = new Config();\r
37         private readonly ConfigDialog _configDialog;\r
38         private int _currentFleet;\r
39         private readonly Label[] _labelCheckFleets;\r
40         private readonly ShipLabels _shipLabels;\r
41         private readonly ShipListForm _shipListForm;\r
42         private readonly NoticeQueue _noticeQueue;\r
43         private bool _started;\r
44         private string _debugLogFile;\r
45         private IEnumerator<string> _playLog;\r
46         private LogServer _logServer;\r
47         private readonly ProxyConfig _prevProxy = new ProxyConfig();\r
48 \r
49         public MainForm()\r
50         {\r
51             InitializeComponent();\r
52             HttpProxy.AfterSessionComplete += HttpProxy_AfterSessionComplete;\r
53             _configDialog = new ConfigDialog(_config, this);\r
54             _labelCheckFleets = new[] {labelCheckFleet1, labelCheckFleet2, labelCheckFleet3, labelCheckFleet4};\r
55 \r
56             // この時点でAutoScaleDimensions == CurrentAutoScaleDimensionsなので、\r
57             // MainForm.Designer.csのAutoScaleDimensionsの6f,12fを使う。\r
58             ShipLabel.ScaleFactor = new SizeF(CurrentAutoScaleDimensions.Width / 6f,\r
59                 CurrentAutoScaleDimensions.Height / 12f);\r
60 \r
61             SetupFleetClick();\r
62             _shipLabels = new ShipLabels();\r
63             _shipLabels.CreateAkashiTimers(panelShipInfo);\r
64             _shipLabels.CreateLabels(panelShipInfo, ShowShipOnShipList);\r
65             _shipLabels.CreateDamagedShipList(panelDamagedShipList);\r
66             _shipLabels.CreateNDockLabels(panelDock);\r
67             _shipListForm = new ShipListForm(_sniffer, _config) {Owner = this};\r
68             _noticeQueue = new NoticeQueue(Ring);\r
69         }\r
70 \r
71         private void HttpProxy_AfterSessionComplete(Session session)\r
72         {\r
73             Invoke(new Action<Session>(ProcessRequest), session);\r
74         }\r
75 \r
76         private void ProcessRequest(Session session)\r
77         {\r
78             var response = session.Response.BodyAsString;\r
79             if (response == null || !response.StartsWith("svdata="))\r
80                 return;\r
81             response = response.Remove(0, "svdata=".Length);\r
82             var json = DynamicJson.Parse(response);\r
83             var request = session.Request.BodyAsString;\r
84             if (_debugLogFile != null)\r
85             {\r
86                 File.AppendAllText(_debugLogFile,\r
87                     string.Format("url: {0}\nrequest: {1}\nresponse: {2}\n", session.Request.PathAndQuery, request, json.ToString()));\r
88             }\r
89             UpdateInfo(_sniffer.Sniff(session.Request.PathAndQuery, request, json));\r
90         }\r
91 \r
92         private void UpdateInfo(Sniffer.Update update)\r
93         {\r
94             if (update == Sniffer.Update.Start)\r
95             {\r
96                 labelLogin.Visible = false;\r
97                 _started = true;\r
98                 return;\r
99             }\r
100             if (!_started)\r
101                 return;\r
102             if ((update & Sniffer.Update.Item) != 0)\r
103                 UpdateItemInfo();\r
104             if ((update & Sniffer.Update.Timer) != 0)\r
105                 UpdateTimers();\r
106             if ((update & Sniffer.Update.NDock) != 0)\r
107                 UpdateNDocLabels();\r
108             if ((update & Sniffer.Update.Mission) != 0)\r
109                 UpdateMissionLabels();\r
110             if ((update & Sniffer.Update.QuestList) != 0)\r
111                 UpdateQuestList();\r
112             if ((update & Sniffer.Update.Ship) != 0)\r
113                 UpdateShipInfo();\r
114             if ((update & Sniffer.Update.Battle) != 0)\r
115                 UpdateBattleInfo();\r
116         }\r
117 \r
118         private void MainForm_Load(object sender, EventArgs e)\r
119         {\r
120             _config.Load();\r
121             RestoreLocation();\r
122             if (_config.HideOnMinimized && WindowState == FormWindowState.Minimized)\r
123                 ShowInTaskbar = false;\r
124             ApplyConfig();\r
125             ApplyDebugLogSetting();\r
126             ApplyLogSetting();\r
127             _sniffer.LoadState();\r
128             StartProxy();\r
129         }\r
130 \r
131         private void StartProxy()\r
132         {\r
133             if (_config.Proxy.UseUpstream)\r
134             {\r
135                 HttpProxy.UpstreamProxyHost = "127.0.0.1";\r
136                 HttpProxy.UpstreamProxyPort = _config.Proxy.UpstreamPort;\r
137             }\r
138             HttpProxy.Startup(_config.Proxy.Listen, false, false);\r
139             _prevProxy.Listen = _config.Proxy.Listen;\r
140             _prevProxy.UseUpstream = _config.Proxy.UseUpstream;\r
141             _prevProxy.UpstreamPort = _config.Proxy.UpstreamPort;\r
142         }\r
143 \r
144         private void MainForm_FormClosing(object sender, FormClosingEventArgs e)\r
145         {\r
146             e.Cancel = false;\r
147             _config.Location = (WindowState == FormWindowState.Normal ? Bounds : RestoreBounds).Location;\r
148             _config.Save();\r
149             Task.Run(() => ShutdownProxy());\r
150             if (_logServer != null)\r
151                 _logServer.Stop();\r
152         }\r
153 \r
154         private void ShutdownProxy()\r
155         {\r
156             lock (typeof (HttpProxy))\r
157             {\r
158                 HttpProxy.Shutdown();\r
159             }\r
160         }\r
161 \r
162         private void MainForm_Resize(object sender, EventArgs e)\r
163         {\r
164             ShowInTaskbar = !(_config.HideOnMinimized && WindowState == FormWindowState.Minimized);\r
165         }\r
166 \r
167         private void notifyIconMain_MouseDoubleClick(object sender, MouseEventArgs e)\r
168         {\r
169             NotifyIconOpenToolStripMenuItem_Click(sender, e);\r
170         }\r
171 \r
172         private void NotifyIconOpenToolStripMenuItem_Click(object sender, EventArgs e)\r
173         {\r
174             ShowInTaskbar = true;\r
175             WindowState = FormWindowState.Normal;\r
176             TopMost = _config.TopMost; // 最前面に表示されなくなることがあるのを回避する\r
177             Activate();\r
178         }\r
179 \r
180         private void ExitToolStripMenuItem_Click(object sender, EventArgs e)\r
181         {\r
182             Close();\r
183         }\r
184 \r
185         private void ConfigToolStripMenuItem_Click(object sender, EventArgs e)\r
186         {\r
187             if (_configDialog.ShowDialog(this) == DialogResult.OK)\r
188                 ApplyConfig();\r
189         }\r
190 \r
191         private void RestoreLocation()\r
192         {\r
193             if (_config.Location.X == int.MinValue)\r
194                 return;\r
195             var newBounds = Bounds;\r
196             newBounds.Location = _config.Location;\r
197             if (IsVisibleOnAnyScreen(newBounds))\r
198                 Location = _config.Location;\r
199         }\r
200 \r
201         private void ApplyConfig()\r
202         {\r
203             _shipListForm.TopMost = TopMost = _config.TopMost;\r
204             _sniffer.Item.MarginShips = _config.MarginShips;\r
205             _sniffer.Item.MarginEquips = _config.MarginEquips;\r
206             _sniffer.Achievement.ResetHours = _config.ResetHours;\r
207         }\r
208 \r
209         public void ApplyDebugLogSetting()\r
210         {\r
211             _debugLogFile = _config.DebugLogging ? _config.DebugLogFile : null;\r
212         }\r
213 \r
214         public void ApplyProxySetting()\r
215         {\r
216             if (_config.Proxy.Listen == _prevProxy.Listen &&\r
217                 _config.Proxy.UseUpstream == _prevProxy.UseUpstream &&\r
218                 _config.Proxy.UpstreamPort == _prevProxy.UpstreamPort)\r
219                 return;\r
220             if (_config.Proxy.UseUpstream)\r
221             {\r
222                 HttpProxy.UpstreamProxyHost = "127.0.0.1";\r
223                 HttpProxy.UpstreamProxyPort = _config.Proxy.UpstreamPort;\r
224             }\r
225             else\r
226             {\r
227                 HttpProxy.UpstreamProxyHost = null;\r
228             }\r
229             Task.Run(() =>\r
230             {\r
231                 ShutdownProxy();\r
232                 StartProxy();\r
233             });\r
234         }\r
235 \r
236         public void ApplyLogSetting()\r
237         {\r
238             if (_logServer != null && (!_config.Log.ServerOn || _config.Log.Listen != _logServer.Port))\r
239             {\r
240                 _logServer.Stop();\r
241                 _logServer = null;\r
242             }\r
243             if (_logServer == null && _config.Log.ServerOn)\r
244             {\r
245                 _logServer = new LogServer(_config.Log.Listen);\r
246                 _logServer.Start();\r
247             }\r
248             if (_logServer != null)\r
249                 _logServer.OutputDir = _config.Log.OutputDir;\r
250             _sniffer.EnableLog(_config.Log.On ? LogType.All : LogType.None);\r
251             _sniffer.MaterialLogInterval = _config.Log.MaterialLogInterval;\r
252             _sniffer.LogOutputDir = _config.Log.OutputDir;\r
253         }\r
254 \r
255         public static bool IsVisibleOnAnyScreen(Rectangle rect)\r
256         {\r
257             return Screen.AllScreens.Any(screen => screen.WorkingArea.IntersectsWith(rect));\r
258         }\r
259 \r
260         private void timerMain_Tick(object sender, EventArgs e)\r
261         {\r
262             if (_started)\r
263                 UpdateTimers();\r
264             if (_playLog == null || _configDialog.Visible)\r
265             {\r
266                 labelPlayLog.Visible = false;\r
267                 return;\r
268             }\r
269             PlayLog();\r
270         }\r
271 \r
272         public void SetPlayLog(string file)\r
273         {\r
274             _playLog = File.ReadLines(file).GetEnumerator();\r
275         }\r
276 \r
277         private void PlayLog()\r
278         {\r
279             var lines = new List<string>();\r
280             foreach (var s in new[] {"url: ", "request: ", "response: "})\r
281             {\r
282                 if (!_playLog.MoveNext() || !_playLog.Current.StartsWith(s))\r
283                 {\r
284                     labelPlayLog.Visible = false;\r
285                     return;\r
286                 }\r
287                 lines.Add(_playLog.Current.Substring(s.Count()));\r
288             }\r
289             labelPlayLog.Visible = !labelPlayLog.Visible;\r
290             var json = DynamicJson.Parse(lines[2]);\r
291             UpdateInfo(_sniffer.Sniff(lines[0], lines[1], json));\r
292         }\r
293 \r
294         private void ShowShipOnShipList(object sender, EventArgs ev)\r
295         {\r
296             if (!_shipListForm.Visible)\r
297                 return;\r
298             var idx = (int)((Control)sender).Tag;\r
299             var statuses = _sniffer.GetShipStatuses(_currentFleet);\r
300             if (statuses.Length <= idx)\r
301                 return;\r
302             _shipListForm.ShowShip(statuses[idx].Id);\r
303         }\r
304 \r
305         private void UpdateItemInfo()\r
306         {\r
307             UpdateNumOfShips();\r
308             UpdateNumOfEquips();\r
309             labelNumOfBuckets.Text = _sniffer.Item.MaterialHistory[(int)Material.Bucket].Now.ToString("D");\r
310             UpdateBucketHistory();\r
311             var ac = _sniffer.Achievement.Value;\r
312             if (ac >= 10000)\r
313                 ac = 9999;\r
314             labelAchievement.Text = ac >= 1000 ? ((int)ac).ToString("D") : ac.ToString("F1");\r
315             toolTipAchievement.SetToolTip(labelAchievement,\r
316                 "今月 " + _sniffer.Achievement.ValueOfMonth.ToString("F1") + "\n" +\r
317                 "EO " + _sniffer.ExMap.Achievement);\r
318             UpdateMaterialHistry();\r
319             if (_shipListForm.Visible)\r
320                 _shipListForm.UpdateList();\r
321         }\r
322 \r
323         private void UpdateNumOfShips()\r
324         {\r
325             var item = _sniffer.Item;\r
326             labelNumOfShips.Text = string.Format("{0:D}/{1:D}", item.NowShips, item.MaxShips);\r
327             labelNumOfShips.ForeColor = item.TooManyShips ? Color.Red : Color.Black;\r
328             if (item.RingShips)\r
329             {\r
330                 var message = string.Format("残り{0:D}隻", _sniffer.Item.MaxShips - _sniffer.Item.NowShips);\r
331                 _noticeQueue.Enqueue("艦娘が多すぎます", message, _config.MaxShipsSoundFile);\r
332                 item.RingShips = false;\r
333             }\r
334         }\r
335 \r
336         private void UpdateNumOfEquips()\r
337         {\r
338             var item = _sniffer.Item;\r
339             labelNumOfEquips.Text = string.Format("{0:D}/{1:D}", item.NowEquips, item.MaxEquips);\r
340             labelNumOfEquips.ForeColor = item.TooManyEquips ? Color.Red : Color.Black;\r
341             if (item.RingEquips)\r
342             {\r
343                 var message = string.Format("残り{0:D}個", _sniffer.Item.MaxEquips - _sniffer.Item.NowEquips);\r
344                 _noticeQueue.Enqueue("装備が多すぎます", message, _config.MaxEquipsSoundFile);\r
345                 item.RingEquips = false;\r
346             }\r
347         }\r
348 \r
349         private void UpdateBucketHistory()\r
350         {\r
351             var count = _sniffer.Item.MaterialHistory[(int)Material.Bucket];\r
352             var day = count.Now - count.BegOfDay;\r
353             var week = count.Now - count.BegOfWeek;\r
354             if (day >= 1000)\r
355                 day = 999;\r
356             if (week >= 1000)\r
357                 week = 999;\r
358             labelBucketHistory.Text = string.Format("{0:+#;-#;±0} 今日\n{1:+#;-#;±0} 今週", day, week);\r
359         }\r
360 \r
361         private void UpdateMaterialHistry()\r
362         {\r
363             var labels = new[] {labelFuelHistory, labelBulletHistory, labelSteelHistory, labelBouxiteHistory};\r
364             var text = new[] {"燃料", "弾薬", "鋼材", "ボーキ"};\r
365             for (var i = 0; i < labels.Length; i++)\r
366             {\r
367                 var count = _sniffer.Item.MaterialHistory[i];\r
368                 var day = count.Now - count.BegOfDay;\r
369                 if (day >= 100000)\r
370                     day = 99999;\r
371                 var week = count.Now - count.BegOfWeek;\r
372                 if (week >= 100000)\r
373                     week = 99999;\r
374                 labels[i].Text = string.Format("{0}\n{1:+#;-#;±0}\n{2:+#;-#;±0}", text[i], day, week);\r
375             }\r
376         }\r
377 \r
378         private void UpdateShipInfo()\r
379         {\r
380             UpdatePanelShipInfo();\r
381             NotifyDamagedShip();\r
382             UpdateChargeInfo();\r
383             UpdateDamagedShipList();\r
384             if (_shipListForm.Visible)\r
385                 _shipListForm.UpdateList();\r
386         }\r
387 \r
388         private void UpdatePanelShipInfo()\r
389         {\r
390             var statuses = _sniffer.GetShipStatuses(_currentFleet);\r
391             _shipLabels.SetShipInfo(statuses);\r
392             labelFighterPower.Text = _sniffer.GetFighterPower(_currentFleet).ToString("D");\r
393             UpdateAkashiTimer();\r
394             UpdateLoS();\r
395             UpdateCondTimers();\r
396         }\r
397 \r
398         private void NotifyDamagedShip()\r
399         {\r
400             if (_sniffer.Battle.HasDamagedShip)\r
401                 _noticeQueue.Enqueue("大破した艦娘がいます", string.Join(" ", _sniffer.Battle.DamagedShipNames),\r
402                     _config.DamagedShipSoundFile);\r
403         }\r
404 \r
405         private void NotifyAkashiTimer()\r
406         {\r
407             var msgs = _sniffer.GetAkashiTimerNotice();\r
408             if (msgs.Length == 0)\r
409                 return;\r
410             if (msgs[0] == "20分経過しました。")\r
411             {\r
412                 _noticeQueue.Enqueue("泊地修理", msgs[0], _config.Akashi20MinSoundFile);\r
413                 return;\r
414             }\r
415             var fn = new[] {"第一艦隊", "第二艦隊", "第三艦隊", "第四艦隊"};\r
416             for (var i = 0; i < fn.Length; i++)\r
417             {\r
418                 if (msgs[i] == "")\r
419                     continue;\r
420                 _noticeQueue.Enqueue("泊地修理 " + fn[i], msgs[i], _config.AkashiProgressSoundFile);\r
421             }\r
422         }\r
423 \r
424         private void UpdateLoS()\r
425         {\r
426             labelLoS.Text = _sniffer.GetFleetLineOfSights(_currentFleet).ToString("F1");\r
427         }\r
428 \r
429         private void UpdateBattleInfo()\r
430         {\r
431             labelFormation.Text = "";\r
432             labelEnemyFighterPower.Text = "";\r
433             labelFighterPower.ForeColor = DefaultForeColor;\r
434             labelResultRank.Text = "判定";\r
435             panelBattleInfo.Visible = _sniffer.Battle.InBattle;\r
436             if (!_sniffer.Battle.InBattle)\r
437                 return;\r
438             panelBattleInfo.BringToFront();\r
439             var battle = _sniffer.Battle;\r
440             var color = new[] {DefaultForeColor, DefaultForeColor, Color.Blue, Color.Green, Color.Orange, Color.Red};\r
441             labelFormation.Text = battle.Formation;\r
442             labelEnemyFighterPower.Text = battle.EnemyFighterPower.ToString("D");\r
443             labelFighterPower.ForeColor = color[battle.AirControlLevel + 1];\r
444             if (_config.AlwaysShowResultRank)\r
445                 ShowResultRank();\r
446         }\r
447 \r
448         private void ShowResultRank()\r
449         {\r
450             var result = new[] {"完全S", "勝利S", "勝利A", "勝利B", "敗北C", "敗北D", "敗北E"};\r
451             labelResultRank.Text = result[(int)_sniffer.Battle.ResultRank];\r
452         }\r
453 \r
454         private void labelResultRank_Click(object sender, EventArgs e)\r
455         {\r
456             ShowResultRank();\r
457         }\r
458 \r
459         private void UpdateChargeInfo()\r
460         {\r
461             var fuelSq = new[] {labelFuelSq1, labelFuelSq2, labelFuelSq3, labelFuelSq4};\r
462             var bullSq = new[] {labelBullSq1, labelBullSq2, labelBullSq3, labelBullSq4};\r
463 \r
464             for (var i = 0; i < fuelSq.Length; i++)\r
465             {\r
466                 var stat = _sniffer.ChargeStatuses[i];\r
467                 fuelSq[i].ImageIndex = stat.Fuel;\r
468                 bullSq[i].ImageIndex = stat.Bull;\r
469             }\r
470         }\r
471 \r
472         private void UpdateNDocLabels()\r
473         {\r
474             _shipLabels.SetNDockLabels(_sniffer.NDock);\r
475         }\r
476 \r
477         private void UpdateMissionLabels()\r
478         {\r
479             foreach (var entry in\r
480                 new[] {labelMissionName1, labelMissionName2, labelMissionName3}.Zip(_sniffer.Missions,\r
481                     (label, mission) => new {label, mission.Name}))\r
482                 entry.label.Text = entry.Name;\r
483         }\r
484 \r
485         private void UpdateTimers()\r
486         {\r
487             foreach (var entry in\r
488                 new[] {labelMission1, labelMission2, labelMission3}.Zip(_sniffer.Missions,\r
489                     (label, mission) => new {label, mission.Name, mission.Timer}))\r
490             {\r
491                 entry.Timer.Update();\r
492                 SetTimerColor(entry.label, entry.Timer);\r
493                 var rest = entry.Timer.Rest;\r
494                 entry.label.Text = rest.Days == 0 ? rest.ToString(@"hh\:mm\:ss") : rest.ToString(@"d\.hh\:mm");\r
495                 if (!entry.Timer.NeedRing)\r
496                     continue;\r
497                 _noticeQueue.Enqueue("遠征が終わりました", entry.Name, _config.MissionSoundFile);\r
498                 entry.Timer.NeedRing = false;\r
499             }\r
500             for (var i = 0; i < _sniffer.NDock.Length; i++)\r
501             {\r
502                 var entry = _sniffer.NDock[i];\r
503                 entry.Timer.Update();\r
504                 _shipLabels.SetNDockTimer(i, entry.Timer);\r
505                 if (!entry.Timer.NeedRing)\r
506                     continue;\r
507                 _noticeQueue.Enqueue("入渠が終わりました", entry.Name, _config.NDockSoundFile);\r
508                 entry.Timer.NeedRing = false;\r
509             }\r
510             var kdock = new[] {labelConstruct1, labelConstruct2, labelConstruct3, labelConstruct4};\r
511             for (var i = 0; i < kdock.Length; i++)\r
512             {\r
513                 var timer = _sniffer.KDock[i];\r
514                 timer.Update();\r
515                 SetTimerColor(kdock[i], timer);\r
516                 kdock[i].Text = timer.Rest.ToString(@"hh\:mm\:ss");\r
517                 if (!timer.NeedRing)\r
518                     continue;\r
519                 _noticeQueue.Enqueue("建造が終わりました", string.Format("第{0:D}ドック", i + 1), _config.KDockSoundFile);\r
520                 timer.NeedRing = false;\r
521             }\r
522             UpdateCondTimers();\r
523             UpdateAkashiTimer();\r
524         }\r
525 \r
526         private void SetTimerColor(Label label, RingTimer timer)\r
527         {\r
528             label.ForeColor = timer.IsFinished ? Color.Red : Color.Black;\r
529         }\r
530 \r
531         private void UpdateCondTimers()\r
532         {\r
533             var timer = _sniffer.GetConditionTimer(_currentFleet);\r
534             var now = DateTime.Now;\r
535             if (timer == DateTime.MinValue)\r
536             {\r
537                 labelCondTimerTitle.Text = "";\r
538                 labelCondTimer.Text = "";\r
539                 return;\r
540             }\r
541             var span = TimeSpan.FromSeconds(Math.Ceiling((timer - now).TotalSeconds));\r
542             if (span >= TimeSpan.FromMinutes(9))\r
543             {\r
544                 labelCondTimerTitle.Text = "cond40まで";\r
545                 labelCondTimer.Text = (span - TimeSpan.FromMinutes(9)).ToString(@"mm\:ss");\r
546             }\r
547             else\r
548             {\r
549                 labelCondTimerTitle.Text = "cond49まで";\r
550                 labelCondTimer.Text = (span >= TimeSpan.Zero ? span : TimeSpan.Zero).ToString(@"mm\:ss");\r
551             }\r
552             var notice = _sniffer.GetConditionNotice();\r
553             if (notice == null)\r
554                 return;\r
555             var fn = new[] {"第一艦隊", "第二艦隊", "第三艦隊", "第四艦隊"};\r
556             for (var i = 0; i < fn.Length; i++)\r
557             {\r
558                 if (!_config.NotifyConditions.Contains(notice[i]))\r
559                     return;\r
560                 _noticeQueue.Enqueue("疲労が回復しました", fn[i] + " cond" + notice[i].ToString("D"), _config.ConditionSoundFile);\r
561             }\r
562         }\r
563 \r
564         private void UpdateAkashiTimer()\r
565         {\r
566             _shipLabels.SetAkashiTimer(_sniffer.GetShipStatuses(_currentFleet),\r
567                 _sniffer.GetAkashiTimers(_currentFleet));\r
568             NotifyAkashiTimer();\r
569         }\r
570 \r
571         private void UpdateDamagedShipList()\r
572         {\r
573             _shipLabels.SetDamagedShipList(_sniffer.DamagedShipList);\r
574         }\r
575 \r
576         private void UpdateQuestList()\r
577         {\r
578             var category = new[]\r
579             {\r
580                 labelQuestColor1, labelQuestColor2, labelQuestColor3, labelQuestColor4, labelQuestColor5,\r
581                 labelQuestColor6\r
582             };\r
583             var name = new[] {labelQuest1, labelQuest2, labelQuest3, labelQuest4, labelQuest5, labelQuest6};\r
584             var progress = new[]\r
585             {labelProgress1, labelProgress2, labelProgress3, labelProgress4, labelProgress5, labelProgress6};\r
586             var color = new[]\r
587             {\r
588                 Color.FromArgb(60, 141, 76), Color.FromArgb(232, 57, 41), Color.FromArgb(136, 204, 120),\r
589                 Color.FromArgb(52, 147, 185), Color.FromArgb(220, 198, 126), Color.FromArgb(168, 111, 76),\r
590                 Color.FromArgb(200, 148, 231)\r
591             };\r
592 \r
593             var quests = _sniffer.Quests;\r
594             for (var i = 0; i < name.Length; i++)\r
595             {\r
596                 if (i < quests.Length)\r
597                 {\r
598                     category[i].BackColor = color[quests[i].Category - 1];\r
599                     name[i].Text = quests[i].Name;\r
600                     progress[i].Text = string.Format("{0:D}%", quests[i].Progress);\r
601                 }\r
602                 else\r
603                 {\r
604                     category[i].BackColor = DefaultBackColor;\r
605                     name[i].Text = progress[i].Text = "";\r
606                 }\r
607             }\r
608         }\r
609 \r
610         private class NoticeQueue\r
611         {\r
612             private readonly Action<string, string, string> _ring;\r
613             private readonly Queue<Tuple<string, string, string>> _queue = new Queue<Tuple<string, string, string>>();\r
614             private readonly Timer _timer = new Timer {Interval = 2000};\r
615 \r
616             public NoticeQueue(Action<string, string, string> ring)\r
617             {\r
618                 _ring = ring;\r
619                 _timer.Tick += TimerOnTick;\r
620             }\r
621 \r
622             private void TimerOnTick(object obj, EventArgs e)\r
623             {\r
624                 if (_queue.Count == 0)\r
625                 {\r
626                     _timer.Stop();\r
627                     return;\r
628                 }\r
629                 var notice = _queue.Dequeue();\r
630                 _ring(notice.Item1, notice.Item2, notice.Item3);\r
631             }\r
632 \r
633             public void Enqueue(string title, string message, string soundFile)\r
634             {\r
635                 if (_timer.Enabled)\r
636                 {\r
637                     _queue.Enqueue(new Tuple<string, string, string>(title, message, soundFile));\r
638                 }\r
639                 else\r
640                 {\r
641                     _ring(title, message, soundFile);\r
642                     _timer.Start();\r
643                 }\r
644             }\r
645         }\r
646 \r
647         private void Ring(string baloonTitle, string baloonMessage, string soundFile)\r
648         {\r
649             if (_config.FlashWindow)\r
650                 Win32API.FlashWindow(Handle);\r
651             if (_config.ShowBaloonTip)\r
652                 notifyIconMain.ShowBalloonTip(20000, baloonTitle, baloonMessage, ToolTipIcon.Info);\r
653             if (_config.PlaySound)\r
654                 PlaySound(soundFile, _config.SoundVolume);\r
655         }\r
656 \r
657         [DllImport("winmm.dll")]\r
658         private static extern int mciSendString(String command,\r
659             StringBuilder buffer, int bufferSize, IntPtr hwndCallback);\r
660 \r
661 // ReSharper disable InconsistentNaming\r
662         private const int MM_MCINOTIFY = 0x3B9;\r
663         private const int MCI_NOTIFY_SUCCESSFUL = 1;\r
664 // ReSharper restore InconsistentNaming\r
665 \r
666         public void PlaySound(string file, int volume)\r
667         {\r
668             if (!File.Exists(file))\r
669                 return;\r
670             mciSendString("close sound", null, 0, IntPtr.Zero);\r
671             if (mciSendString("open \"" + file + "\" type mpegvideo alias sound", null, 0, IntPtr.Zero) != 0)\r
672                 return;\r
673             mciSendString("setaudio sound volume to " + volume * 10, null, 0, IntPtr.Zero);\r
674             mciSendString("play sound notify", null, 0, Handle);\r
675         }\r
676 \r
677         protected override void WndProc(ref Message m)\r
678         {\r
679             if (m.Msg == MM_MCINOTIFY && (int)m.WParam == MCI_NOTIFY_SUCCESSFUL)\r
680                 mciSendString("close sound", null, 0, IntPtr.Zero);\r
681             base.WndProc(ref m);\r
682         }\r
683 \r
684         private void SetupFleetClick()\r
685         {\r
686             var labels = new[]\r
687             {\r
688                 new[] {labelFleet1, labelFleet2, labelFleet3, labelFleet4},\r
689                 new[] {labelFuelSq1, labelFuelSq2, labelFuelSq3, labelFuelSq4},\r
690                 new[] {labelBullSq1, labelBullSq2, labelBullSq3, labelBullSq4}\r
691             };\r
692             foreach (var a in labels)\r
693             {\r
694                 for (var fleet = 0; fleet < labels[0].Length; fleet++)\r
695                 {\r
696                     a[fleet].Tag = fleet;\r
697                     a[fleet].Click += labelFleet_Click;\r
698                 }\r
699             }\r
700         }\r
701 \r
702         private void labelFleet_Click(object sender, EventArgs e)\r
703         {\r
704             var fleet = (int)((Label)sender).Tag;\r
705             if (_currentFleet == fleet)\r
706                 return;\r
707             _currentFleet = fleet;\r
708             foreach (var label in _labelCheckFleets)\r
709                 label.Visible = false;\r
710             _labelCheckFleets[fleet].Visible = true;\r
711             if (!_started)\r
712                 return;\r
713             UpdatePanelShipInfo();\r
714         }\r
715 \r
716         private void labelBucketHistoryButton_Click(object sender, EventArgs e)\r
717         {\r
718             if (labelBucketHistory.Visible)\r
719             {\r
720                 labelBucketHistory.Visible = false;\r
721                 labelBucketHistoryButton.BackColor = DefaultBackColor;\r
722             }\r
723             else\r
724             {\r
725                 labelBucketHistory.Visible = true;\r
726                 labelBucketHistory.BringToFront();\r
727                 labelBucketHistoryButton.BackColor = SystemColors.ActiveCaption;\r
728             }\r
729         }\r
730 \r
731         private void labelMaterialHistoryButton_Click(object sender, EventArgs e)\r
732         {\r
733             if (panelMaterialHistory.Visible)\r
734             {\r
735                 panelMaterialHistory.Visible = false;\r
736                 labelMaterialHistoryButton.BackColor = DefaultBackColor;\r
737             }\r
738             else\r
739             {\r
740                 panelMaterialHistory.Visible = true;\r
741                 panelMaterialHistory.BringToFront();\r
742                 labelMaterialHistoryButton.BackColor = SystemColors.ActiveCaption;\r
743             }\r
744         }\r
745 \r
746         public void ResetAchievemnt()\r
747         {\r
748             _sniffer.Achievement.Reset();\r
749             UpdateItemInfo();\r
750         }\r
751 \r
752         private void labelDamgedShipListButton_Click(object sender, EventArgs e)\r
753         {\r
754             if (panelDamagedShipList.Visible)\r
755             {\r
756                 panelDamagedShipList.Visible = false;\r
757                 labelDamgedShipListButton.BackColor = DefaultBackColor;\r
758             }\r
759             else\r
760             {\r
761                 panelDamagedShipList.Visible = true;\r
762                 panelDamagedShipList.BringToFront();\r
763                 labelDamgedShipListButton.BackColor = SystemColors.ActiveCaption;\r
764             }\r
765         }\r
766 \r
767         private void ShipListToolStripMenuItem_Click(object sender, EventArgs e)\r
768         {\r
769             _shipListForm.UpdateList();\r
770             _shipListForm.Show();\r
771             if (_shipListForm.WindowState == FormWindowState.Minimized)\r
772                 _shipListForm.WindowState = FormWindowState.Normal;\r
773             _shipListForm.Activate();\r
774         }\r
775 \r
776         private void LogToolStripMenuItem_Click(object sender, EventArgs e)\r
777         {\r
778             Process.Start("http://localhost:" + _config.Log.Listen + "/");\r
779         }\r
780     }\r
781 }