markus / KCOM / PageManager / PageStorage.cs @ 8de0cf00
이력 | 보기 | 이력해설 | 다운로드 (11.5 KB)
1 |
using System; |
---|---|
2 |
using System.Collections.Concurrent; |
3 |
using System.Collections.Generic; |
4 |
using System.ComponentModel; |
5 |
using System.Linq; |
6 |
using System.Text; |
7 |
using System.Threading.Tasks; |
8 |
using System.Windows.Media.Imaging; |
9 |
|
10 |
namespace KCOM.PageManager |
11 |
{ |
12 |
public class PageStorage |
13 |
{ |
14 |
private const int DEFUALT_TALK_PAGE_COUNT = 10; |
15 |
|
16 |
public event EventHandler<PageLoadCompletedEventArgs> PageLoadCompleted; |
17 |
|
18 |
List<int> WorkItems = new List<int>(); |
19 |
ConcurrentBag<PageItem> fileItems = new ConcurrentBag<PageItem>(); |
20 |
BackgroundWorker backgroundWorker; |
21 |
|
22 |
//List<PageItem> fileItems = new List<PageItem>(); |
23 |
public string LocalStorage; |
24 |
string _fileExt; |
25 |
string _BaseUri; |
26 |
int _TotalPages; |
27 |
int _TakeCount; |
28 |
|
29 |
bool IsDownload = false; |
30 |
|
31 |
BitmapFrame PageImage; |
32 |
|
33 |
public PageStorage(string BaseUri, string localStoragePath, string fileExt, int totalPages, int takeCount = 5) |
34 |
{ |
35 |
try |
36 |
{ |
37 |
backgroundWorker = new BackgroundWorker { WorkerSupportsCancellation = true, WorkerReportsProgress = true }; |
38 |
backgroundWorker.DoWork += BackgroundWorker_DoWork; |
39 |
backgroundWorker.ProgressChanged += BackgroundWorker_ProgressChanged; |
40 |
_fileExt = fileExt; |
41 |
_BaseUri = BaseUri; |
42 |
LocalStorage = localStoragePath; |
43 |
_TotalPages = totalPages; |
44 |
_TakeCount = takeCount; |
45 |
|
46 |
System.IO.Directory.CreateDirectory(LocalStorage); |
47 |
|
48 |
//backgroundWorker.RunWorkerAsync(new int[] { 1, 10 }); |
49 |
} |
50 |
catch (Exception ex) |
51 |
{ |
52 |
throw new Exception("PageStorage", ex); |
53 |
} |
54 |
} |
55 |
|
56 |
private void BackgroundWorker_ProgressChanged(object sender, ProgressChangedEventArgs e) |
57 |
{ |
58 |
System.Diagnostics.Debug.WriteLine(_fileExt + ":" + e.ProgressPercentage.ToString() + "%"); |
59 |
} |
60 |
|
61 |
private async void BackgroundWorker_DoWork(object sender, DoWorkEventArgs e) |
62 |
{ |
63 |
|
64 |
while (WorkItems.Count > 0) |
65 |
{ |
66 |
if (!IsDownload) |
67 |
{ |
68 |
int item = -1; |
69 |
try |
70 |
{ |
71 |
|
72 |
if (WorkItems.TryFrist(true, out item)) |
73 |
{ |
74 |
var result = await DownloadPageAsync(item, null); |
75 |
|
76 |
if (!result.IsDownLoad) |
77 |
{ |
78 |
System.Threading.Thread.Sleep(100); |
79 |
} |
80 |
await Task.Delay(100); |
81 |
//System.Threading.Thread.Sleep(10); |
82 |
//backgroundWorker.ReportProgress(fileItems.Count / _TotalPages * 100); |
83 |
} |
84 |
} |
85 |
catch (Exception ex) |
86 |
{ |
87 |
System.Diagnostics.Debug.WriteLine(ex.ToString()); |
88 |
} |
89 |
} |
90 |
} |
91 |
} |
92 |
|
93 |
public void ResetImage() |
94 |
{ |
95 |
//if (PageImage != null) |
96 |
//{ |
97 |
// PageImage = null; |
98 |
//} |
99 |
|
100 |
//PageImage.CreateOptions = BitmapCreateOptions.IgnoreImageCache; |
101 |
|
102 |
} |
103 |
|
104 |
public async Task<BitmapFrame> GetPageImageAsync(System.Threading.CancellationToken cts, int PageNo, bool IsRestart = false) |
105 |
{ |
106 |
try |
107 |
{ |
108 |
|
109 |
var localUri = await GetPageUriAsync(null, PageNo); |
110 |
|
111 |
if (localUri != null || !cts.IsCancellationRequested) |
112 |
{ |
113 |
PageImage = BitmapFrame.Create(localUri, BitmapCreateOptions.None, BitmapCacheOption.OnLoad); |
114 |
} |
115 |
} |
116 |
catch (Exception ex) |
117 |
{ |
118 |
PageImage = BitmapFrame.Create(new Uri(_BaseUri.Replace("{PageNo}", PageNo.ToString())), BitmapCreateOptions.None, BitmapCacheOption.OnLoad); |
119 |
} |
120 |
finally |
121 |
{ |
122 |
} |
123 |
|
124 |
return PageImage; |
125 |
} |
126 |
|
127 |
private void DownloadWorkAsync(int startPage, int TakeCount) |
128 |
{ |
129 |
|
130 |
lock (WorkItems) |
131 |
{ |
132 |
WorkItems.AddRange(Enumerable.Range(startPage, TakeCount)); |
133 |
} |
134 |
|
135 |
if (TakeCount == _TotalPages) |
136 |
{ |
137 |
_TakeCount = 0; |
138 |
} |
139 |
|
140 |
// var files = fileItems.Select(x => x.PageNo); |
141 |
|
142 |
|
143 |
//for (int i = startPage; i < TakeCount + 1; i++) |
144 |
//{ |
145 |
// if (i < _TotalPages && !WorkItems.Contains(i) && !files.Contains(i)) |
146 |
// { |
147 |
// WorkItems.Add(i); |
148 |
// // System.Threading.Thread.Sleep(1); |
149 |
// } |
150 |
// else if(i == _TotalPages) |
151 |
// { |
152 |
// _TakeCount = 0; |
153 |
// } |
154 |
//} |
155 |
|
156 |
if (!backgroundWorker.IsBusy && WorkItems.Count() > 0) |
157 |
{ |
158 |
backgroundWorker.RunWorkerAsync(); |
159 |
} |
160 |
|
161 |
//await Task.Delay(10);// System.Threading.Thread.Sleep(10); |
162 |
//while (WorkItems.Count > 0) |
163 |
//{ |
164 |
// int item = -1; |
165 |
|
166 |
// if (WorkItems.TryDequeue(out item)) |
167 |
// { |
168 |
// var result = await DownloadPageAsync(item, null); |
169 |
// } |
170 |
|
171 |
//} |
172 |
|
173 |
} |
174 |
|
175 |
/// <summary> |
176 |
/// |
177 |
/// </summary> |
178 |
/// <param name="cts"></param> |
179 |
/// <param name="PageNo"></param> |
180 |
/// <param name="IsClone"></param> |
181 |
/// <param name="IsRestart"></param> |
182 |
/// <returns></returns> |
183 |
public async Task<Uri> GetPageUriAsync(System.Threading.CancellationToken? cts, int PageNo, bool IsRestart = false) |
184 |
{ |
185 |
Uri result = null; |
186 |
|
187 |
try |
188 |
{ |
189 |
System.Diagnostics.Debug.WriteLine("GetPageAsync"); |
190 |
|
191 |
var pageItem = await DownloadPageAsync(PageNo, cts); |
192 |
|
193 |
if (pageItem != null && 0 < _TakeCount && PageNo + 1 < _TotalPages) |
194 |
{ |
195 |
int takecount = _TakeCount; |
196 |
|
197 |
if (_TotalPages < PageNo + 1 + takecount) |
198 |
{ |
199 |
takecount = takecount - (PageNo + 1 + takecount - _TotalPages) - 1; |
200 |
} |
201 |
|
202 |
DownloadWorkAsync(PageNo + 1, takecount); |
203 |
//var takePageNoList = Enumerable.Range(PageNo + 1, _TakeCount); |
204 |
} |
205 |
|
206 |
result = pageItem.LocalUri; |
207 |
} |
208 |
catch (Exception ex) |
209 |
{ |
210 |
if (cts != null) |
211 |
{ |
212 |
if (cts.Value.IsCancellationRequested) |
213 |
{ |
214 |
return result; |
215 |
} |
216 |
} |
217 |
|
218 |
if (!IsRestart) |
219 |
{ |
220 |
await Task.Delay(100); |
221 |
|
222 |
result = await GetPageUriAsync(cts, PageNo, true); |
223 |
} |
224 |
//throw new Exception("GetPageAsync(string BasePageUri,int PageNo)", ex); |
225 |
} |
226 |
finally |
227 |
{ |
228 |
} |
229 |
|
230 |
return result; |
231 |
} |
232 |
|
233 |
public async Task<PageItem> DownloadPageAsync(int PageNo, System.Threading.CancellationToken? cts) |
234 |
{ |
235 |
PageItem result = new PageItem { PageNo = PageNo }; |
236 |
|
237 |
try |
238 |
{ |
239 |
var page = fileItems.Where(x => x.PageNo == PageNo).ToList(); |
240 |
|
241 |
if (page.Count > 0) |
242 |
{ |
243 |
System.Diagnostics.Debug.WriteLine("DownloadPageAsync fileItems"); |
244 |
|
245 |
result = page.First(); |
246 |
|
247 |
/// 파일 체크 후 없으면 다시 다운로드 |
248 |
if (!System.IO.File.Exists(result.LocalFilePath)) |
249 |
{ |
250 |
//fileItems.(result); |
251 |
|
252 |
result = await DownloadPageAsync(PageNo, cts); |
253 |
} |
254 |
} |
255 |
else |
256 |
{ |
257 |
|
258 |
System.Diagnostics.Debug.WriteLine("DownloadPageAsync down"); |
259 |
|
260 |
string downloadFilePath = System.IO.Path.Combine(LocalStorage, System.IO.Path.GetRandomFileName()); /// PageNo.ToString() + "." + _fileExt); |
261 |
|
262 |
Uri originalUri = new Uri(_BaseUri.Replace("{PageNo}", PageNo.ToString())); |
263 |
|
264 |
result = new PageItem |
265 |
{ |
266 |
PageNo = PageNo, |
267 |
OriginalUri = originalUri, |
268 |
LocalUri = new Uri(downloadFilePath, UriKind.Absolute), |
269 |
LocalFilePath = downloadFilePath |
270 |
}; |
271 |
|
272 |
if (fileItems.Where(x => x.PageNo == PageNo).Count() == 0) |
273 |
{ |
274 |
fileItems.Add(result); |
275 |
} |
276 |
|
277 |
using (System.Net.WebClient client = new System.Net.WebClient()) |
278 |
{ |
279 |
client.UseDefaultCredentials = true; |
280 |
System.Net.IWebProxy webProxy = client.Proxy; |
281 |
|
282 |
if (webProxy != null) |
283 |
{ |
284 |
// Use the default credentials of the logged on user. |
285 |
webProxy.Credentials = System.Net.CredentialCache.DefaultCredentials; |
286 |
} |
287 |
|
288 |
//client.DownloadFileCompleted += (snd, evt) => |
289 |
//{ |
290 |
|
291 |
|
292 |
// //(snd as System.Net.WebClient).Dispose(); |
293 |
//}; |
294 |
|
295 |
if (cts != null) |
296 |
{ |
297 |
client.DownloadProgressChanged += (snd, ect) => |
298 |
{ |
299 |
IsDownload = client.IsBusy; |
300 |
|
301 |
if (cts.Value.IsCancellationRequested) |
302 |
{ |
303 |
client.CancelAsync(); |
304 |
} |
305 |
}; |
306 |
} |
307 |
|
308 |
client.DownloadFileCompleted += (snd, evt) => |
309 |
{ |
310 |
result.IsDownLoad = true; |
311 |
System.Diagnostics.Debug.WriteLine("Download : " + downloadFilePath); |
312 |
|
313 |
PageLoadCompleted?.Invoke(this, new PageLoadCompletedEventArgs(result)); |
314 |
}; |
315 |
|
316 |
await client.DownloadFileTaskAsync(originalUri, downloadFilePath); |
317 |
|
318 |
} |
319 |
|
320 |
//} |
321 |
//System.Diagnostics.Debug.WriteLine("Download : " + result.LocalFilePath); |
322 |
} |
323 |
} |
324 |
catch (Exception ex) |
325 |
{ |
326 |
throw new Exception("DownloadPageAsync : ", ex); |
327 |
} |
328 |
finally |
329 |
{ |
330 |
IsDownload = false; |
331 |
} |
332 |
|
333 |
return result; |
334 |
} |
335 |
|
336 |
public void Clear() |
337 |
{ |
338 |
try |
339 |
{ |
340 |
ResetImage(); |
341 |
} |
342 |
catch (Exception ex) |
343 |
{ |
344 |
System.Diagnostics.Debug.WriteLine(ex.ToString()); |
345 |
} |
346 |
|
347 |
try |
348 |
{ |
349 |
backgroundWorker.CancelAsync(); |
350 |
backgroundWorker.Dispose(); |
351 |
|
352 |
/// downloadmanager에서 삭제함 |
353 |
//foreach (var item in fileItems) |
354 |
//{ |
355 |
// System.IO.File.Delete(item.LocalFilePath); |
356 |
//} |
357 |
|
358 |
//System.IO.Directory.Delete(LocalStorage, true); |
359 |
} |
360 |
catch (Exception ex) |
361 |
{ |
362 |
System.Diagnostics.Debug.WriteLine(ex.ToString()); |
363 |
} |
364 |
} |
365 |
} |
366 |
} |