프로젝트

일반

사용자정보

통계
| 브랜치(Branch): | 개정판:

markus / ConvertService / ServiceBase / Markus.Service.StationController / ViewModel / DataBaseItemsModel.cs @ e6281033

이력 | 보기 | 이력해설 | 다운로드 (25 KB)

1
using Markus.Message;
2
using Markus.Service.DataBase;
3
using Markus.Service.Extensions;
4
using Markus.Service.StationController.Data;
5
using Markus.Service.StationController.Behaviors;
6
using Markus.Service.StationController.Extensions;
7
using Microsoft.Win32;
8
using System;
9
using System.Collections.Generic;
10
using System.ComponentModel;
11
using System.Linq;
12
using System.Threading.Tasks;
13
using System.Web;
14
using System.Windows;
15
using Telerik.Windows.Controls;
16
using Telerik.Windows.Data;
17
using ConvertItem = Markus.Service.Interface.ConvertItem;
18
using System.Net;
19
using System.Windows.Threading;
20

    
21
namespace Markus.Service.StationController.ViewModel
22
{
23
    //세미
24
    public class DataBaseItemsModel : Mvvm.ToolKit.ViewModelBase
25
    {
26
        #region Constructor
27

    
28
        /// <summary>
29
        /// 실행하면 처음 들어가는 부분
30
        /// </summary>
31
        public DataBaseItemsModel()
32
        {
33
            DataSaveFileGemBoxCommand = new DelegateCommand(DataExportData);
34
            ConvertCommand = new DelegateCommand(DataConvert);
35
            DeleteCommand = new DelegateCommand(DataDelete);
36
            ValidateCommand = new DelegateCommand(DataValidate);
37
        }
38

    
39
        #endregion
40

    
41
        #region Properties
42

    
43
        private System.Collections.ObjectModel.ObservableCollection<ConvertItem> _AliveItems;
44
        public System.Collections.ObjectModel.ObservableCollection<ConvertItem> AliveItems
45
        {
46
            get => _AliveItems;
47
            set
48
            {
49
                _AliveItems = value;
50
                OnPropertyChanged(() => AliveItems);
51
            }
52
        }
53

    
54
        private System.Collections.ObjectModel.ObservableCollection<ConvertItem> _FilterConvertSource;
55
        public System.Collections.ObjectModel.ObservableCollection<ConvertItem> FilterConvertSource
56
        {
57
            get => _FilterConvertSource;
58
            set
59
            {
60
                _FilterConvertSource = value;
61
                OnPropertyChanged(() => FilterConvertSource);
62
            }
63
        }
64

    
65

    
66
        private System.Collections.ObjectModel.ObservableCollection<ConvertItem> _RealConvertSource;
67
        public System.Collections.ObjectModel.ObservableCollection<ConvertItem> RealConvertSource
68
        {
69
            get => _RealConvertSource;
70
            set
71
            {
72
                _RealConvertSource = value;
73
                OnPropertyChanged(() => RealConvertSource);
74
            }
75
        }
76

    
77

    
78
        private System.Windows.Documents.FlowDocument connectionLog;
79
        public System.Windows.Documents.FlowDocument ConnectionLog
80
        {
81
            get => connectionLog;
82
            set
83
            {
84
                if (connectionLog != value)
85
                {
86
                    connectionLog = value;
87
                    OnPropertyChanged(() => ConnectionLog);
88
                }
89
            }
90
        }
91

    
92

    
93
        private Telerik.Windows.Data.EnumMemberViewModel _SelectedStatus;
94
        public Telerik.Windows.Data.EnumMemberViewModel SelectedStatus
95
        {
96
            get => _SelectedStatus;
97
            set
98
            {
99
                _SelectedStatus = value;
100
                OnPropertyChanged(() => SelectedStatus);
101
            }
102
        }
103

    
104

    
105
        private SelectedCountItem _SelectedCount;
106
        public SelectedCountItem SelectedCount
107
        {
108
            get => _SelectedCount;
109
            set
110
            {
111
                _SelectedCount = value;
112
                OnPropertyChanged(() => SelectedCount);
113
            }
114
        }
115

    
116
        List<SelectedCountItem> _SelectedCountList;
117
        public List<SelectedCountItem> SelectedCountList
118
        {
119
            get
120
            {
121
                if (_SelectedCountList == null)
122
                {
123
                    _SelectedCountList = new List<SelectedCountItem>
124
                    {
125
                        new SelectedCountItem{DisplayMember = "50",ValueMember = 50},
126
                        new SelectedCountItem{DisplayMember = "100",ValueMember = 100},
127
                        new SelectedCountItem{DisplayMember = "150",ValueMember = 150},
128
                        new SelectedCountItem{DisplayMember = "200",ValueMember = 200}
129
                    };
130
                }
131

    
132
                return _SelectedCountList;
133
            }
134
        }
135

    
136
        private ConvertItem _SelectFilterConvert;
137
        public ConvertItem SelectFilterConvert
138
        {
139
            get => _SelectFilterConvert;
140
            set
141
            {
142
                _SelectFilterConvert = value;
143
                OnPropertyChanged(() => SelectFilterConvert);
144
            }
145
        }
146

    
147
        private ConvertItem _SelectRealConvert;
148
        public ConvertItem SelectRealConvert
149
        {
150
            get => _SelectRealConvert;
151
            set
152
            {
153
                _SelectRealConvert = value;
154
                OnPropertyChanged(() => SelectRealConvert);
155
            }
156
        }
157

    
158

    
159
        private StatusTypeList _StatusType;
160
        public StatusTypeList StatusType
161
        {
162
            get => _StatusType;
163
            set
164
            {
165
                _StatusType = value;
166
                OnPropertyChanged(() => StatusType);
167
            }
168
        }
169

    
170
        private bool _IsLoading;
171
        public bool IsLoading
172
        {
173
            get => _IsLoading;
174
            set
175
            {
176
                if (_IsLoading != value)
177
                {
178
                    _IsLoading = value;
179
                    OnPropertyChanged(() => IsLoading);
180
                }
181
            }
182
        }
183

    
184
        IEnumerable<Telerik.Windows.Data.EnumMemberViewModel> _StatusCodeList;
185
        public IEnumerable<Telerik.Windows.Data.EnumMemberViewModel> StatusCodeList
186
        {
187
            get
188
            {
189
                if (_StatusCodeList == null)
190
                {
191
                    _StatusCodeList = Telerik.Windows.Data.EnumDataSource.FromType<StatusCodeType>();
192
                }
193

    
194
                return _StatusCodeList;
195
            }
196
        }
197
        #endregion
198

    
199
        #region Command
200

    
201
        public DelegateCommand ConvertCommand { get; private set; }
202
        public DelegateCommand DeleteCommand { get; private set; }
203
        public DelegateCommand ValidateCommand { get; private set; }
204
        public DelegateCommand DataSaveFileGemBoxCommand { get; private set; }
205

    
206
        #endregion
207

    
208
        #region Main Logic
209

    
210
        /// <summary>
211
        /// 각각의 Grid row 객체들 업데이트
212
        /// </summary>
213

    
214
        private DispatcherTimer dispatcherTimer;
215
        public override void Loaded()
216
        {
217
            base.Loaded();
218

    
219
            if (!App.IsDesignMode)
220
            {
221
                dispatcherTimer = new DispatcherTimer();
222
                dispatcherTimer.Tick += new EventHandler(Timer_Tick);
223
                dispatcherTimer.Interval = new TimeSpan(0, 0, 1);
224
                dispatcherTimer.Start();
225
            }
226
        }
227

    
228
        private void Timer_Tick(object sender, EventArgs e)
229
        {
230
            dispatcherTimer.Stop();
231

    
232
            if (IsAcitve)
233
            {
234
                App.Current.Dispatcher.InvokeAsync(() =>
235
                {
236
                    DataSelect();
237

    
238
                    AliveDataSelect();
239
                });
240
            }
241

    
242
            System.Threading.Thread.Sleep(new TimeSpan(0,0,0,0,100));
243

    
244
            dispatcherTimer.Start();
245
        }
246

    
247
        public override void Closed()
248
        {
249
            if (dispatcherTimer != null)
250
            {
251
                dispatcherTimer.Stop();
252
            }
253

    
254
            base.Closed();
255
        }
256

    
257

    
258
        #endregion
259

    
260
        #region Function
261

    
262
        #region Data Select
263

    
264
        /// <summary>
265
        /// 상단 그리드 중앙 그리드 출력 데이터
266
        /// </summary>
267
        private void DataSelect()
268
        {
269

    
270
            if (FilterConvertSource == null)
271
            {
272
                FilterConvertSource = new System.Collections.ObjectModel.ObservableCollection<ConvertItem>();
273
            }
274

    
275
            if (RealConvertSource == null)
276
            {
277
                RealConvertSource = new System.Collections.ObjectModel.ObservableCollection<ConvertItem>();
278
            }
279

    
280
            /// combobox 에서 선택된 items
281
            if (SelectedStatus != null)
282
            {
283
                DataSelect(new[] { (StatusCodeType)(SelectedStatus.Value) }, FilterConvertSource);
284
            }
285

    
286
            /// 컨버터중인 items
287
            RealDataSelect(new[] { StatusCodeType.None, StatusCodeType.Wait, StatusCodeType.PageLoading, StatusCodeType.Saving }, RealConvertSource);
288

    
289
        }
290

    
291
        private void RealDataSelect(IEnumerable<StatusCodeType> statusCodeTypeList, System.Collections.ObjectModel.ObservableCollection<ConvertItem> collection)
292
        {
293
            try
294
            {
295
                using (Markus.Service.DataBase.ConvertDatabase database = new Markus.Service.DataBase.ConvertDatabase(App.MarkusDataBaseConnecitonString))
296
                {
297
                    var items = database.GetConvertProjects(collection)//x:database객체 y:statusCodeTypeList의값  Count()안에 predicator 조건 만족하면 count개수안에 넣음
298
                                          .Take(SelectedCount.ValueMember).ToList();//
299

    
300
                    items.ForEach(newitem =>
301
                    {
302
                        collection.UpdateWhere(changeitem =>
303
                        ConvertItemEx.ChangeValues(changeitem, newitem), x => x.ConvertID == newitem.ConvertID && x.ProjectNumber == newitem.ProjectNumber);
304
                    });
305
                }
306
            }
307
            catch (Exception ex)
308
            {
309
                MessageBox.Show(ex.ToString());
310
            }
311
        }
312

    
313
        private void DataSelect(IEnumerable<StatusCodeType> statusCodeTypeList, System.Collections.ObjectModel.ObservableCollection<ConvertItem> collection)
314
        {
315
            try
316
            {
317
                using (Markus.Service.DataBase.ConvertDatabase database = new Markus.Service.DataBase.ConvertDatabase(App.MarkusDataBaseConnecitonString))
318
                {
319

    
320
                    int _status = 0;
321
                    if (SelectedStatus != null)
322
                    {
323
                        _status = (int)SelectedStatus.Value;
324
                    }
325

    
326
                    var items = database.GetConvertProjects(_status)//x:database객체 y:statusCodeTypeList의값  Count()안에 predicator 조건 만족하면 count개수안에 넣음
327
                                          .Take(SelectedCount.ValueMember).ToList();
328

    
329
                    if (collection.Count() == 0)
330
                    {
331
                        if (statusCodeTypeList.Count() == 1)
332
                        {
333
                            items.ForEach(x => collection.Add(x));
334
                        }
335
                    }
336
                    else
337
                    {
338

    
339
                        ////세미 업데이트
340
                        items.ForEach(newitem =>
341
                        {
342
                            collection.UpdateWhere(changeitem =>
343
                            ConvertItemEx.ChangeValues(changeitem, newitem), x => x.ProjectNumber == newitem.ProjectNumber && x.ConvertID == newitem.ConvertID);
344
                        });
345

    
346

    
347
                        if (statusCodeTypeList.Count() == 1)
348
                        {
349
                            //삭제
350
                            for (int i = collection.Count() - 1; i >= 0; --i)
351
                            {
352
                                var item = collection[i];
353

    
354
                                if (items.Count(x => x.ConvertID == item.ConvertID && x.ProjectNumber == item.ProjectNumber) == 0)//디비에서 가져온 값과 마지막값부터 차례대로 비교
355
                                {//참=> 0제외한 모든 수
356
                                    collection.RemoveAt(i);
357
                                }
358
                            }
359
                        }
360

    
361
                        if (statusCodeTypeList.Count() == 1)
362
                        {
363
                            //추가 convert 후 추가됨
364
                            foreach (var item in items)
365
                            {
366
                                if (collection.Count(x => x.ConvertID == item.ConvertID && x.ProjectNumber == item.ProjectNumber) == 0)//그리드와 디비 변동 없으면 안들어감
367
                                {
368
                                    /*for (int i = 0; i < collection.Count()+1; i++)//위 그리드에서 카운드 개수 변함 없고 컨버터 끝난 후 추가할때 createtime 비교 하며 order by 순으로 추가*/
369
                                    for (int i = 0; i < 200; i++)
370
                                    {
371
                                        if (i < collection.Count() - 1)
372
                                        {
373
                                            if (DateTime.Compare(collection[i].CreateTime, item.CreateTime) < 0)
374
                                            {
375
                                                collection.Insert(i, item);
376
                                                break;
377
                                            }
378
                                        }
379
                                        else
380
                                        {
381
                                            collection.Add(item);
382
                                            break;
383
                                        }
384
                                    }
385

    
386
                                }
387

    
388
                            }
389
                        }
390

    
391
                    }
392
                }
393

    
394
            }
395
            catch (Exception ex)
396
            {
397
                MessageBox.Show(ex.ToString());
398
            }
399
        }
400

    
401
        /// <summary>
402
        /// 서비스의 실시간 컨버터 Item
403
        /// </summary>
404
        private async void AliveDataSelect()
405
        {
406
            try
407
            {
408
                List<ConvertItem> newitems = new List<ConvertItem>();
409

    
410
                foreach (var client in App.StationClientList)
411
                {
412
                    if (await SimplePingAsync(client.Endpoint.Address.ToString()))
413
                    {
414
                        try
415
                        {
416
                            List<ConvertItem> itemsToList = new List<ConvertItem>();
417
                            var items = await client.AliveConvertListAsync();
418
                            foreach (var item in items)
419
                            {
420
                                ConvertItem itemsToEach = new ConvertItem();
421
                                itemsToEach.ServiceID = item.ServiceID;
422
                                itemsToEach.ConvertID = item.ConvertID;
423
                                itemsToEach.ProjectNumber = item.ProjectNumber;
424
                                try
425
                                {
426
                                    itemsToEach.ConvertState = (StatusCodeType)Enum.Parse(typeof(StatusCodeType), item.ConvertState);
427
                                }
428
                                catch (Exception ex)
429
                                {
430

    
431
                                    throw ex;
432
                                }
433

    
434
                                itemsToEach.CurrentPageNo = item.CurrentPageNo;
435
                                itemsToEach.TotalPage = item.TotalPage;
436
                                itemsToEach.OriginfilePath = item.OriginfilePath;
437
                                itemsToEach.ConvertPath = item.ConvertPath;
438

    
439
                                itemsToList.Add(itemsToEach);
440
                            }
441
                            newitems.AddRange(itemsToList);
442
                            System.Diagnostics.Trace.WriteLine($"{client.Endpoint.Address} ping");
443

    
444
                            if (items.Count() == 0)
445
                            {
446
                                System.Diagnostics.Trace.WriteLine($"{client.Endpoint.Address} Alive Items is zero.");
447
                            }
448
                        }
449
                        catch (Exception ex)
450
                        {
451
                            System.Diagnostics.Trace.Fail($"{client.Endpoint.Address} {ex.Message}");
452
                        }
453
                    }
454
                    else
455
                    {
456
                        System.Diagnostics.Trace.Fail($"{client.Endpoint.Address} ping Error");
457
                    }
458
                }
459

    
460
                ItemsUpdate(newitems);
461
                await System.Windows.Application.Current.Dispatcher.InvokeAsync(() => ItemsUpdate(newitems));
462
            }
463
            catch (Exception ex)
464
            {
465
                System.Diagnostics.Debug.WriteLine(ex.ToString());
466
            }
467
        }
468

    
469
        /// <summary>
470
        /// AliveDataSelect의 Data Update
471
        /// </summary>
472
        /// <param name="newitems"></param>
473
        private void ItemsUpdate(List<ConvertItem> newitems)
474
        {
475

    
476
            foreach (var item in newitems)
477
            {
478
                item.OriginfilePath = HttpUtility.UrlDecode(item.OriginfilePath);
479
            }
480

    
481
            if (AliveItems == null)
482
            {
483
                AliveItems = new System.Collections.ObjectModel.ObservableCollection<ConvertItem>();
484

    
485
                foreach (var item in newitems)
486
                {
487
                    AliveItems.Add(item);
488
                }
489
            }
490
            else
491
            {
492
                /// 데이터 업데이트
493
                newitems.ForEach(newitem =>
494
                {
495
                    AliveItems.UpdateWhere(changeitem => ConvertItemEx.ChangeValues(changeitem, newitem), x => x.ProjectNumber == newitem.ProjectNumber && x.ConvertID == newitem.ConvertID);
496
                });
497

    
498
                // 추가
499
                foreach (var item in newitems)
500
                {
501
                    if (AliveItems.Count(x => x.ConvertID == item.ConvertID && x.ProjectNumber == item.ProjectNumber) == 0)
502
                    {
503
                        AliveItems.Add(item);
504
                    }
505
                }
506

    
507
                /// 삭제
508

    
509
                for (int i = AliveItems.Count() - 1; i > -1; --i)
510
                {
511
                    var item = AliveItems[i];
512

    
513
                    if (newitems.Count(x => x.ConvertID == item.ConvertID && x.ProjectNumber == item.ProjectNumber) == 0)
514
                    {
515
                        try
516
                        {
517
                            AliveItems.RemoveAt(i);
518
                        }
519
                        catch (Exception ex)
520
                        {
521
                            System.Diagnostics.Debug.WriteLine(ex.ToString());
522
                        }
523
                    }
524
                }
525
            }
526
        }
527

    
528

    
529
        public static async Task<bool> SimplePingAsync(string uri)
530
        {
531
            bool result = false;
532

    
533
            try
534
            {
535
                using (System.Net.Http.HttpClient Client = new System.Net.Http.HttpClient())
536
                {
537
                    Client.Timeout = new TimeSpan(0, 0, 60);
538

    
539
                    var message = await Client.GetAsync(uri);
540

    
541
                    System.Net.HttpStatusCode StatusCode = message.StatusCode;
542

    
543
                    switch (StatusCode)
544
                    {
545

    
546
                        case System.Net.HttpStatusCode.Accepted:
547
                        case System.Net.HttpStatusCode.OK:
548
                            result = true;
549
                            break;
550
                    }
551
                }
552
            }
553
            catch (Exception ex)
554
            {
555
                result = false;
556
                System.Diagnostics.Debug.WriteLine(ex.ToString());
557
            }
558

    
559
            return result;
560
        }
561

    
562
        #endregion
563

    
564
        #region Data Convert
565

    
566
        private void DataConvert(object obj)
567
        {
568
            if (SelectFilterConvert == null && SelectRealConvert == null)
569
            {
570
                MessageBox.Show("왼쪽 버튼 클릭 후 Converter 해주세요!");
571
            }
572
            else
573
            {
574
                ConvertDatabase _DataConvert = new ConvertDatabase(App.MarkusDataBaseConnecitonString);
575
                var resultRealConvert = 0;
576
                var resultFiltertConvert = 0;
577

    
578
                if (SelectRealConvert != null)
579
                {
580
                    resultRealConvert = _DataConvert.SetCleanUpItem(SelectRealConvert);
581
                }
582
                else if (SelectFilterConvert != null)
583
                {
584
                    resultFiltertConvert = _DataConvert.SetCleanUpItem(SelectFilterConvert);
585
                }
586
                System.Diagnostics.Debug.WriteLine(resultRealConvert + "  " + resultFiltertConvert);
587

    
588
                using (Markus.Service.DataBase.ConvertDatabase database = new Markus.Service.DataBase.ConvertDatabase(App.MarkusDataBaseConnecitonString))
589
                {
590
                    var items = database.GetConvertProjects(SelectFilterConvert);
591

    
592
                    foreach (var item in items)
593
                    {
594
                        RealConvertSource.Add(item);
595
                    }
596

    
597
                }
598
            }
599
        }
600

    
601
        #endregion
602

    
603
        #region Validation
604

    
605
        private void DataValidate(object obj)
606
        {
607

    
608
            bool result = false;
609

    
610
            WebRequest webRequest = WebRequest.Create(SelectFilterConvert.OriginfilePath);
611
            webRequest.Timeout = 1200; // miliseconds
612
            webRequest.Method = "HEAD";
613

    
614
            HttpWebResponse response = null;
615

    
616
            try
617
            {
618
                response = (HttpWebResponse)webRequest.GetResponse();
619
                result = true;
620
            }
621
            catch (WebException webException)
622
            {
623
                MessageBox.Show(SelectFilterConvert.FileName + " doesn't exist: " + webException.Message);
624
                result = true;
625
            }
626
            finally
627
            {
628
                if (response != null)
629
                {
630
                    response.Close();
631
                }
632
            }
633
            if (result == true)
634
            {
635
                MessageBox.Show("File exists");
636
            }
637
        }
638

    
639
        #endregion
640

    
641
        #region Data Delete
642

    
643
        private void DataDelete(object obj)
644
        {
645
            RadWindow.Alert("정말로 삭제 하시겠습니까?", this.OnClosed);
646
        }
647

    
648
        private void OnClosed(object sender, WindowClosedEventArgs e)
649
        {
650
            var result = e.DialogResult;
651
            if (result == true)
652
            {
653
                ConvertDatabase _DataConvert = new ConvertDatabase(App.MarkusDataBaseConnecitonString);
654
                var resultRealConvert = _DataConvert.RemoveItem(SelectRealConvert.ConvertID);
655
                var resultFiltertConvert = _DataConvert.RemoveItem(SelectFilterConvert.ConvertID);
656
                System.Diagnostics.Debug.WriteLine(resultRealConvert + "  " + resultFiltertConvert);
657
            }
658
        }
659

    
660

    
661
        #endregion
662

    
663
        #region Data Export
664

    
665

    
666
        /// <summary>
667
        /// 필터된 상단 그리드 엑셀로 출력
668
        /// </summary>
669

    
670
        public void DataExportData(object obj)
671
        {
672
            SaveFileDialog saveFileDialog = new SaveFileDialog();
673

    
674
            saveFileDialog.FileName = "Document"; // Default file name
675
            saveFileDialog.DefaultExt = ".txt"; // Default file extension
676
            saveFileDialog.Filter = "Csv documents (.Csv)|*.csv|Excel(2017~2019)Worksheets|*.xlsx"; // Filter files by extension
677

    
678

    
679
            if (saveFileDialog.ShowDialog() == true)
680
            {
681

    
682

    
683
                if (!string.IsNullOrWhiteSpace(saveFileDialog.FileName))
684
                {
685

    
686
                    var extension = new System.IO.FileInfo(saveFileDialog.FileName).Extension;
687

    
688
                    if (extension == ".xlsx" || extension == ".csv")
689
                    {
690

    
691
                        var headers = new List<HeaderMember>
692
                        {
693
                            new HeaderMember{HeaderName = "ServiceID", Property = "ServiceID" },
694
                            new HeaderMember{HeaderName = "ConvertID", Property = "ConvertID" },
695
                            new HeaderMember{HeaderName = "ProjectNumber", Property = "ProjectNumber" },
696
                            new HeaderMember{HeaderName = "ConvertState", Property = "ConvertState" },
697
                            new HeaderMember{HeaderName = "CurrentPageNo", Property = "CurrentPageNo" },
698
                            new HeaderMember{HeaderName = "TotalPage", Property = "TotalPage" },
699
                            new HeaderMember{HeaderName = "OriginfilePath", Property = "OriginfilePath" },
700
                            new HeaderMember{HeaderName = "ConvertPath", Property = "ConvertPath" },
701
                            new HeaderMember{HeaderName = "CreateTime", Property = "CreateTime" },
702
                            new HeaderMember{HeaderName = "Exception", Property = "Exception" },
703
                            new HeaderMember{HeaderName = "ProcessorAffinity", Property = "ProcessorAffinity" },
704
                            new HeaderMember{HeaderName = "ReConverter", Property = "ReConverter" },
705
                            new HeaderMember{HeaderName = "UniqueKey", Property = "UniqueKey" }
706
                        };
707

    
708

    
709
                        DataExport dataExport = new DataExport();
710
                        dataExport.DataExportExcel(saveFileDialog.FileName, "Hello world", FilterConvertSource, headers);
711
                        //_dataExport.DataExportExcel(saveFileDialog.FileName, saveFileDialog.FileName.ToString(),  Projectsource, headers);
712
                        //GemBoxFileSave(obj);
713
                    }
714
                }
715
            }
716
        }
717

    
718
        #endregion
719

    
720
        #endregion
721
    }
722
}
723

    
클립보드 이미지 추가 (최대 크기: 500 MB)