프로젝트

일반

사용자정보

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

markus / KCOM / Services / BaseServices.cs @ 53deabaf

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

1
using Microsoft.AspNet.SignalR.Client;
2
using IKCOM;
3
using KCOM.Common;
4
using KCOM.ServiceDeepView;
5
using KCOMDataModel.DataModel;
6
using Newtonsoft.Json;
7
using RestSharp;
8
using System;
9
using System.Collections.Generic;
10
using System.Collections.ObjectModel;
11
using System.ComponentModel;
12
using System.Linq;
13
using System.Runtime.CompilerServices;
14
using System.Text;
15
using System.Windows;
16
using System.Windows.Controls;
17
using Telerik.Windows.Controls;
18
using MarkupToPDF.Controls.Parsing;
19
using Telerik.Windows.Data;
20
using System.Web;
21

    
22
namespace KCOM.Views
23
{
24
    public partial class MainMenu : UserControl, INotifyPropertyChanged
25
    {
26
        //const string DeepZoomUrl = @"http://{baseUrl}/TileSource/{TileSourcePath}/{ItemsPath}/{SharepointItemID}/{PageNo}.jpg";
27
        public ServiceDeepViewClient BaseClient;
28
        public KcomService.ServiceDeepViewClient BaseTaskClient;
29
        public ViewInfo _ViewInfo;
30
        public DOCINFO _DocInfo;
31
        public DOCUMENT_ITEM _DocItem;
32
        public MEMBER userData;
33
        public List<int> MarkupSortList = new List<int>();
34

    
35
        public event PropertyChangedEventHandler PropertyChanged;
36
        private void NotifyPropertyChanged([CallerMemberName] String propertyName = "")
37
        {
38
            if (PropertyChanged != null)
39
            {
40
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
41
            }
42
        }
43

    
44
        private Common.DocumentSet _CurrentDoc { get; set; }
45
        public Common.DocumentSet CurrentDoc
46
        {
47
            get
48
            {
49
                return _CurrentDoc;
50
            }
51
            set
52
            {
53
                if (value != CurrentDoc)
54
                {
55
                    _CurrentDoc = value;
56
                    NotifyPropertyChanged();
57
                }
58
            }
59
        }
60
        public void ServiceOn()
61
        {
62
            BaseTaskClient = new KcomService.ServiceDeepViewClient(App._binding, App._EndPoint);
63
            BaseClient = new ServiceDeepViewClient(App._binding, App._EndPoint);
64

    
65
            BaseClient.GetDocInfoCompleted += BaseClient_GetDocInfoCompleted; //문서 섬네일, 페이지크기, 마크업 정보 [3]
66
            BaseClient.GetConversionStateCompleted += BaseClient_GetConversionStateCompleted;  //컨버팅의 여부 [2] 
67
            BaseClient.GetSystemDataCompleted += BaseClient_GetSystemDataCompleted; //시스템 정보 [1] 
68
            BaseClient.GetDocumentItemInfoCompleted += BaseClient_GetDocumentItemInfoCompleted; // 문서 자체의 정보 [4]
69
            BaseClient.GetMarkupInfoItemsCompleted += BaseClient_GetMarkupInfoItemsCompleted;
70
            BaseClient.GetDeptDataCompleted += BaseClient_GetDeptDataCompleted;
71
            BaseClient.DeleteMarkupCompleted += BaseClient_DeleteMarkupCompleted;
72
            BaseClient.GetVPRevisionHistoryCompleted += BaseClient_GetVPRevisionHistoryCompleted;
73
            BaseClient.SetFinalPDFCompleted += BaseClient_SetFinalPDFCompleted;
74
            BaseClient.GetCompareRectCompleted += BaseClient_GetCompareRectCompleted;
75
            BaseClient.DeleteMarkupCompleted += BaseClient_DeleteMarkupCompleted;
76
            BaseClient.DelFavoriteVPCompleted += BaseClient_DelFavoriteVPCompleted;
77
            BaseClient.EditFavoriteVPCompleted += BaseClient_EditFavoriteVPCompleted;
78
        }
79

    
80

    
81

    
82
        private void BaseClient_EditFavoriteVPCompleted(object sender, EditFavoriteVPCompletedEventArgs e)
83
        {
84
            Logger.sendResLog("EditFavoriteVPCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
85
            
86
            if (e.Error == null && e.Result != false)
87
            {
88
                Logger.sendReqLog("GetFavoriteVPAsync", App.ViewInfo.ProjectNO + "," + App.ViewInfo.UserID + "," + App.ViewInfo.DocumentItemID, 1);
89
                BaseClient.GetFavoriteVPAsync(App.ViewInfo.ProjectNO, App.ViewInfo.UserID, App.ViewInfo.DocumentItemID);
90
            }
91
            else
92
            {
93
                DialogMessage_Alert("Unable to delete the file due to a system failure", "Warning");
94
            }
95
        }
96

    
97
        private void BaseClient_DelFavoriteVPCompleted(object sender, DelFavoriteVPCompletedEventArgs e)
98
        {
99
            Logger.sendResLog("DelFavoriteVPCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
100

    
101
            if (e.Error == null && e.Result != false)
102
            {
103
                Logger.sendReqLog("GetFavoriteVPAsync", App.ViewInfo.ProjectNO + "," + App.ViewInfo.UserID + "," + App.ViewInfo.DocumentItemID, 1);
104
                BaseClient.GetFavoriteVPAsync(App.ViewInfo.ProjectNO, App.ViewInfo.UserID, App.ViewInfo.DocumentItemID);
105
            }
106
            else
107
            {
108
                DialogMessage_Alert("Unable to delete the file due to a system failure", "Warning");
109
            }
110
        }
111

    
112
        /// <summary>
113
        /// Compare 결과를 받아 화면에 표시한다
114
        /// </summary>
115
        /// <param name="sender"></param>
116
        /// <param name="e"></param>
117
        private void BaseClient_GetCompareRectCompleted(object sender, GetCompareRectCompletedEventArgs e)
118
        {
119
            if (e.Error != null || e.Result == null)
120
            {
121
                if (e.Error != null)
122
                {
123
                    DialogMessage_Alert(e.Error.ToString(), "Error");
124
                }
125
                else
126
                {
127
                    DialogMessage_Alert("Compare 기능을 원활히 실행하기 위한 데이터가 부족합니다", "안내");
128
                }
129
            }
130
            else
131
            {
132
                e.Result.ForEach(d =>
133
                {
134
                    d.Width = d.Width * 3;
135
                    d.Height = d.Height * 3;
136
                    var point = MarkupToPDF.Controls.Common.MathSet.getRectMiddlePoint(d);
137
                    System.Windows.Shapes.Rectangle myEllipse = new System.Windows.Shapes.Rectangle();
138
                    myEllipse.Fill = new System.Windows.Media.SolidColorBrush(System.Windows.Media.Color.FromArgb(100,255,8,0));
139
                    myEllipse.Opacity = 0.8;
140
                    myEllipse.Width = d.Width;
141
                    myEllipse.Height = d.Height;
142
                    Canvas.SetLeft(myEllipse, d.X - ((point.X - d.X) / 2.0));
143
                    Canvas.SetTop(myEllipse, d.Y - ((point.Y - d.Y) / 2.0));
144
                    myEllipse.RenderTransformOrigin = point;
145
                    canvas_compareBorder.Children.Add(myEllipse);
146
                });
147

    
148
                da.From = 0.8;
149
                da.To = 0.1;
150
                da.Duration = new Duration(TimeSpan.FromSeconds(1));
151
                da.AutoReverse = true;
152
                canvas_compareBorder.BeginAnimation(OpacityProperty, da);
153
            }
154
        }
155

    
156
        private void BaseClient_SetFinalPDFCompleted(object sender, SetFinalPDFCompletedEventArgs e)
157
        {
158
            Logger.sendResLog("SetFinalPDFCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
159
            string errormsg = CommonLib.Common.GetAlertMessageString("SetFinalPDFError", "MSG", "최종 파일을 만드는데 문제가 발생하였습니다");
160
            string successmsg = CommonLib.Common.GetAlertMessageString("SetFinalPDFSuccess", "MSG", "최종 파일 생성 중입니다. 문서관리시스템을 확인해주세요");
161
            if (e.Error != null || e.Result.Status == FinalStatus.Error)
162
            {
163
                DialogMessage_Alert(errormsg, "안내");
164
            }
165
            else
166
            {
167
                DialogMessage_Alert(successmsg, "안내");
168
            }
169
        }
170

    
171
        private void BaseClient_GetVPRevisionHistoryCompleted(object sender, GetVPRevisionHistoryCompletedEventArgs e)
172
        {
173
            Logger.sendResLog("GetVPRevisionHistoryCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
174

    
175
            if (e.Result != null && e.Error == null)
176
            {
177
                List<VPRevision> list = e.Result;
178
                foreach (var item in list)
179
                {
180

    
181
                    if (App.isExternal)
182
                    {
183
                        var sBaseServiceURL = CommonLib.Common.GetConfigString("ExternalOriginalFileUrl", "URL", "", App.isExternal);
184

    
185
                        if (!string.IsNullOrWhiteSpace(item.TO_VENDOR))
186
                        {
187
                            var toUri = new Uri(HttpUtility.UrlDecode(item.TO_VENDOR));
188
                            item.TO_VENDOR = $"{sBaseServiceURL}{toUri.LocalPath}";
189
                        }
190

    
191
                        if (!string.IsNullOrWhiteSpace(item.FROM_VENDOR))
192
                        {
193
                            var fromUri = new Uri(HttpUtility.UrlDecode(item.FROM_VENDOR));
194
                            item.FROM_VENDOR = $"{sBaseServiceURL}{fromUri.LocalPath}";
195
                        }
196
                    }
197
                    else
198
                    {
199

    
200
                        item.FROM_VENDOR = HttpUtility.UrlDecode(item.FROM_VENDOR);
201
                    }
202
                }
203
                gridViewHistory.ItemsSource = list;
204
            }
205
            //TempFile.TempLoad();
206
        }
207

    
208
        private void BaseClient_DeleteMarkupCompleted(object sender, DeleteMarkupCompletedEventArgs e)
209
        {
210
            Logger.sendResLog("DeleteMarkupCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
211

    
212
            foreach(var info in ViewerDataModel.Instance._markupInfoList)
213
            {
214
                if(info.UserID == App.ViewInfo.UserID)
215
                {
216
                    info.userDelete = true;
217
                    info.DisplayColor = "FFFF0000";
218
                }
219
                else
220
                {
221
                    info.userDelete = false;
222
                }
223
            }
224
        }
225

    
226
        private void BaseClient_GetDeptDataCompleted(object sender, GetDeptDataCompletedEventArgs e)
227
        {
228
            if (e.Error == null)
229
            {
230
                Logger.sendResLog("GetDeptDataCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
231
            }
232
            else
233
            {
234
                Logger.sendResLog("GetDeptDataCompleted",  "\r Error :" + e.Error, 1);
235
            }
236

    
237
            if (e.Result != null)
238
            {
239
                e.Result.Sort();
240
                //cbSymbolPublic.ItemsSource = e.Result.ToList();
241
            }
242
        }
243

    
244
        public MarkupInfoItem PreviewUserMarkupInfoItem { get; set; }
245

    
246
        /// <summary>
247
        /// 문서 정보를 가져온다
248
        /// </summary>
249
        /// <param name="sender"></param>
250
        /// <param name="e"></param>
251
        private async void BaseClient_GetDocInfoCompleted(object sender, GetDocInfoCompletedEventArgs e)
252
        {
253
            //Logger.sendResLog("GetDocInfoCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
254

    
255
            if (e.Error != null)
256
            {
257
                DialogMessage_Alert("The page cannot be displayed.", "Alert");
258
                return;
259
            }
260

    
261
            string sFolder = _ViewInfo.DocumentItemID.All(char.IsDigit) ? (Convert.ToUInt32(_ViewInfo.DocumentItemID) / 100).ToString() : (_ViewInfo.DocumentItemID.Length >= 5 ? _ViewInfo.DocumentItemID.Substring(0, 5) : _ViewInfo.DocumentItemID);
262
            var MainUrl = string.Format(CommonLib.Common.GetConfigString("mainServerImageWebPath", "URL", "", App.isExternal).Replace("png","jpg"), _ViewInfo.ProjectNO, sFolder, _ViewInfo.DocumentItemID, "{PageNo}");
263
            _DocInfo = e.Result;
264
            _DocInfo.ORIGINAL_FILE = HttpUtility.UrlDecode(_DocInfo.ORIGINAL_FILE);
265
            foreach (var item in _DocInfo.DOCPAGE.OrderBy(order => order.PAGE_NUMBER))
266
            {
267
                ViewerDataModel.Instance.Document_Info.Add(item);
268
            }
269
            Logger.sendReqLog("GetMarkupInfoItemsAsync", _ViewInfo.ProjectNO + "," + _DocInfo.ID, 1);
270
           CurrentDoc = new Common.DocumentSet
271
            {
272
                docInfo = _DocInfo,
273
                Document_Id = _DocInfo.DOCUMENT_ID,
274
                File_Original = _DocInfo.ORIGINAL_FILE,
275

    
276
                Document_NO = _DocItem?.DOCUMENT_NO,
277
                File_Result = _DocItem?.RESULT_FILE,
278
                Group_No = _DocItem?.GROUP_NO,
279
                ProjectNo = _DocItem?.PROJECT_NO,
280
                Revision = _DocItem?.REVISION,
281
            };
282

    
283
            var instanceMain = this.ParentOfType<MainWindow>();
284
            string firstpage = "1";
285
            string lastpage = _DocInfo.PAGE_COUNT.ToString();
286
            instanceMain.dzTopMenu.tlcurrentPage.Text = firstpage;
287
            instanceMain.dzTopMenu.tlcurrentPage_readonly.Text = lastpage;
288
            string apptitle = string.Format("MARKUS(ver.{2}) - {0}(Rev.{1})", _DocItem?.DOCUMENT_NO, _DocItem?.REVISION, System.Reflection.Assembly.GetExecutingAssembly().GetName().Version.ToString());
289

    
290
            instanceMain.Title = apptitle;
291

    
292
            #region 페이지 이미지를 로컬에 저장
293
            string tempStoragePath = System.IO.Path.Combine(System.IO.Path.GetTempPath(),"MARKUS", System.IO.Path.GetRandomFileName());
294

    
295
            //.Replace("jpg", "png")을 붙이는 이유는 MainUrl에 확장명이 jpg로 설정한 경우가 있기 때문
296

    
297
            #endregion
298
            App.PageStorage = new PageManager.PageStorage(MainUrl.Replace("jpg","png"), tempStoragePath,"png", _DocInfo.PAGE_COUNT);
299
         
300
            await this.pageNavigator.SetPageNaviAsync(CurrentDoc.docInfo.DOCPAGE.ToList(), MainUrl);
301

    
302
        }
303

    
304
        private void pageNavigator_ThumbInitialized(object sender,EventArgs e)
305
        {
306
            BaseClient.GetMarkupInfoItemsAsync(_ViewInfo.ProjectNO, _DocInfo.ID);
307

    
308
            ViewerDataModel.Instance.MarkupControls_USER.Clear();
309
            //ViewerDataModel.Instance.SystemMain.dzMainMenu.TempLoad();
310

    
311
            #region 단어 검색용 파일 다운로드
312

    
313
            DownloadOriginalFileAsync().RunAndForget();
314

    
315
            #endregion
316

    
317
            Logger.sendReqLog("GetVPRevisionHistoryAsync", CurrentDoc.ProjectNo + "," + CurrentDoc.Document_NO + "," + CurrentDoc.Document_Id, 1);
318
            BaseClient.GetVPRevisionHistoryAsync(CurrentDoc.ProjectNo, CurrentDoc.Document_NO, CurrentDoc.Document_Id);
319

    
320
        }
321

    
322

    
323
        /// <summary>
324
        /// get markupinfo and markupitem from database
325
        ///  - 1. called when document is loaded
326
        ///  - 2. called when user do consolidation
327
        /// </summary>
328
        /// <param name="sender"></param>
329
        /// <param name="e"></param>
330
        private void BaseClient_GetMarkupInfoItemsCompleted(object sender, GetMarkupInfoItemsCompletedEventArgs e)
331
        {
332
            Logger.sendResLog("GetMarkupInfoItemsCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
333

    
334
            ViewerDataModel.Instance.MarkupControls_USER.Clear();
335
            ViewerDataModel.Instance.MarkupControls.Clear();
336

    
337
            if (e.Result != null)
338
            {
339
                ViewerDataModel.Instance._markupInfoList.Clear();
340
                if (e.Result.Count > 0) ViewerDataModel.Instance.MyMarkupList.Clear();
341

    
342
                foreach (var info in e.Result)
343
                {
344
                    if (info.Depatment == null) info.Depatment = "Unknown";
345

    
346
                    if (info.UserID == App.ViewInfo.UserID)
347
                    {
348
                        info.userDelete = true;
349
                        info.DisplayColor = "#FFFF0000";
350
                    }
351
                    else
352
                    {
353
                        info.userDelete = false;
354
                    }
355

    
356
                    ViewerDataModel.Instance._markupInfoList.Add(info);
357
                }
358

    
359
                var userInfoList = ViewerDataModel.Instance._markupInfoList.GroupBy(f => f.UserID).Select(f => new MarkupUserInfo { UserId = f.Key, UserName = f.First().UserName });
360

    
361
                userInfoList.ToList().ForEach(x => ViewerDataModel.Instance._markupUserList.Add(x));
362

    
363
                e.Result.ToList().ForEach(d =>
364
                {
365
                    if (d.MarkupList != null)
366
                    {
367
                        d.MarkupList.ForEach(p =>
368
                        {
369
                            if (d.UserID == App.ViewInfo.UserID)
370
                            {
371
                                d.userDelete = true;
372
                                d.DisplayColor = "#FFFF0000";
373

    
374
                                MarkupItemEx markup = new MarkupItemEx
375
                                {
376
                                    Data = p.Data,
377
                                    Data_Type = p.Data_Type,
378
                                    ID = p.ID,
379
                                    IsUpdate = false,
380
                                    MarkupInfoID = d.MarkupInfoID,
381
                                    PageNumber = p.PageNumber,
382
                                    Symbol_ID = p.Symbol_ID,
383
                                    //Group_ID = p.Group_ID
384
                                };
385
                                ViewerDataModel.Instance.MyMarkupList.Add(markup);
386
                            }
387
                            else
388
                            {
389
                                MarkupItemEx markup = new MarkupItemEx
390
                                {
391
                                    Data = p.Data,
392
                                    Data_Type = p.Data_Type,
393
                                    ID = p.ID,
394
                                    IsUpdate = false,
395
                                    MarkupInfoID = d.MarkupInfoID,
396
                                    PageNumber = p.PageNumber,
397
                                    Symbol_ID = p.Symbol_ID,
398
                                    //Group_ID = p.Group_ID
399
                                };
400
                                ViewerDataModel.Instance.MarkupList_Pre.Add(markup);
401
                            }
402
                        });
403
                    }
404
                });
405

    
406
                var currentUser = e.Result.Where(data => data.UserID == App.ViewInfo.UserID && data.AvoidConsolidate == 0).FirstOrDefault();
407

    
408
                if (currentUser == null)
409
                {
410
                    /// create a new markupinfo to do comments
411
                    /// 
412
                    if (App.ViewInfo.NewCommentPermission)
413
                    {
414
                        var infoId = Commons.shortGuid();
415
                        PreviewUserMarkupInfoItem = new MarkupInfoItem
416
                        {
417
                            CreateTime = DateTime.Now,
418
                            UpdateTime = DateTime.Now,
419
                            Depatment = userData.DEPARTMENT,
420
                            DisplayColor = "#FFFF0000",
421
                            UserID = userData.ID,
422
                            UserName = userData.NAME,
423
                            PageCount = 1,
424
                            Description = "",
425
                            MarkupInfoID = infoId,
426
                            MarkupList = null,
427
                            MarkupVersionID = Commons.shortGuid(),
428
                            Consolidate = 0,
429
                            PartConsolidate = 0,
430
                            userDelete = true,
431
                            AvoidConsolidate = 0,
432
                            IsPreviewUser = false
433
                        };
434

    
435
                        App.Custom_ViewInfoId = infoId;
436

    
437
                        ViewerDataModel.Instance._markupInfoList.Insert(0, PreviewUserMarkupInfoItem);
438
                    }
439
                }
440
                else
441
                {
442
                    App.Custom_ViewInfoId = currentUser.MarkupInfoID;
443
                }
444

    
445
                ViewerDataModel.Instance.PageNumber = 1;
446

    
447
                /// Consolidation 된 데이터가 최상단에 올 수 있도록 변경
448
                //var markupItem = ViewerDataModel.Instance._markupInfoList.OrderByDescending(p => p.Consolidate == Convert.ToInt32(true)).ToList();
449

    
450
                gridViewMarkup.ItemsSource = ViewerDataModel.Instance._markupInfoList;
451

    
452
                SetCommentPages();
453

    
454
                // GRIDVIEW에서 팀별 GROUPING
455
                // 효성은 안함
456
#if (Hyosung)
457
#else
458
                /// create a group descriptor
459
                gridViewMarkup.GroupDescriptors.Clear();
460
                GroupDescriptor descriptor = new GroupDescriptor();
461
                descriptor.Member = "Depatment";
462
                descriptor.DisplayContent = "DEPT";
463
                //descriptor.SortDirection = ListSortDirection.Ascending;
464
                gridViewMarkup.GroupDescriptors.Add(descriptor);
465
#endif
466

    
467
                GridViewMarkup_Sorting(gridViewMarkup, new GridViewSortingEventArgs(gridViewMarkup.Columns[0], SortingState.None, SortingState.None));
468

    
469
                var SelectedItem = ViewerDataModel.Instance._markupInfoList.Where(info => info.UserID == App.ViewInfo.UserID).OrderByDescending(order => order.UpdateTime).FirstOrDefault();
470

    
471
                if (PreviewUserMarkupInfoItem != null)
472
                {
473
                    this.gridViewMarkup.SelectedItem = PreviewUserMarkupInfoItem;
474
                }
475
                else if (SelectedItem != null)
476
                {
477
                    this.gridViewMarkup.SelectedItem = SelectedItem;
478
                }
479
                else if(ViewerDataModel.Instance._markupInfoList.Count() > 0)
480
                {
481
                    this.gridViewMarkup.SelectedItem = ViewerDataModel.Instance._markupInfoList.First();
482
                }
483
            }
484
        }
485

    
486
        private void GridViewMarkup_Sorting(object sender, GridViewSortingEventArgs e)
487
        {
488
            IEnumerable<IKCOM.MarkupInfoItem> markuplist = e.DataControl.ItemsSource as IEnumerable<IKCOM.MarkupInfoItem>;
489

    
490
            if(markuplist == null)
491
            {
492
                e.Cancel = true;
493
                return;
494
            }
495

    
496
            if(e.NewSortingState == SortingState.None)
497
            {
498
                markuplist = markuplist.OrderByDescending(x => SortIndex(x))
499
                                            .ThenBy(x => x.Depatment)
500
                                            .ThenBy(x => x.UserID);
501

    
502
                e.DataControl.ItemsSource = markuplist.ToList();
503
            }
504
        }
505

    
506

    
507
        // 마크업 리스트 정렬
508
        private int SortIndex(IKCOM.MarkupInfoItem markupInfoItem)
509
        {
510
            int result = 0;
511

    
512
            // Consolidate은 최상단
513
            if (markupInfoItem.Consolidate == 1)
514
            {
515
                result += 2;
516
            }
517

    
518
            /// Consolidate 다음으로 우선순위
519
            if (markupInfoItem.UserID == App.ViewInfo.UserID)
520
            {
521
                result += 1;
522
            }
523

    
524
            return result;
525
        }
526

    
527

    
528
        private void GridViewMarkup_Sorted(object sender, GridViewSortedEventArgs e)
529
        {
530

    
531
        }
532

    
533
        private async void gridViewRevMarkup_SelectionChanged(object sender, Telerik.Windows.Controls.SelectionChangeEventArgs e)
534
        {
535
            List<MarkupInfoItem> gridSelectionItem = gridViewRevMarkup.SelectedItems.Cast<MarkupInfoItem>().ToList(); //선택 된 마크업
536
            //if (ViewerDataModel.Instance.PageBalanceNumber == 0)
537
            //{
538
            //    ViewerDataModel.Instance.PageBalanceNumber = 1;
539
            //}
540

    
541
            if (ViewerDataModel.Instance.PageNumber == 0)
542
            {
543
                ViewerDataModel.Instance.PageNumber = 1;
544
            }
545
            if (e.AddedItems.Count() > 0)
546
            {
547
                foreach (var item in gridSelectionItem)
548
                {
549
                    //item.MarkupList.Where(pageItem => pageItem.PageNumber == ViewerDataModel.Instance.PageBalanceNumber).ToList().ForEach(delegate (MarkupItem markupitem)
550
                    //{
551
                    //    layerControl.markupParse(markupitem.Data, Common.ViewerDataModel.Instance.MarkupControls_Sync, item.DisplayColor, "", item.MarkupInfoID);
552
                    //});
553

    
554
                    var pageItems = item.MarkupList.Where(pageItem => pageItem.PageNumber == ViewerDataModel.Instance.PageNumber).ToList();
555

    
556
                    foreach (var markupitem in pageItems)
557
                    {
558
                       await MarkupParser.ParseExAsync(App.ViewInfo.ProjectNO, markupitem.Data, Common.ViewerDataModel.Instance.MarkupControls_Sync, item.DisplayColor, "", item.MarkupInfoID);
559
                    }
560
                }
561
            }
562
            else if (e.RemovedItems.Count > 0)
563
            {
564
                var _infoItem = e.RemovedItems.Cast<MarkupInfoItem>().ToList();
565

    
566
                (from A in _infoItem
567
                 from B in Common.ViewerDataModel.Instance.MarkupControls_Sync
568
                 where A.MarkupInfoID == B.MarkupInfoID
569
                 select B).ToList().ForEach(data =>
570
                 {
571
                     Common.ViewerDataModel.Instance.MarkupControls_Sync.Remove(data);
572
                 });
573
            }
574
        }
575

    
576
        /// <summary>
577
        /// 선택한 항목은 Visibility = Visible, 해제한 항목은 Visibility = Hidden 처리한다.
578
        /// </summary>
579
        /// <param name="sender"></param>
580
        /// <param name="e"></param>
581
        private async void gridViewMarkup_SelectionChanged(object sender, Telerik.Windows.Controls.SelectionChangeEventArgs e)
582
        {
583
            SelectionSet.Instance.UnSelect(Common.ViewerDataModel.Instance.SystemMain.dzMainMenu);
584

    
585
            var dataSet = gridViewMarkup.SelectedItems.Cast<MarkupInfoItem>().Where(data => data.UserID == _ViewInfo.UserID);
586
            var dataSet_User = gridViewMarkup.Items.Cast<MarkupInfoItem>().Where(data => data.UserID == _ViewInfo.UserID);
587
            var resultA = (from item_Select in dataSet
588
                           from item_Non in dataSet_User
589
                           where item_Select == item_Non
590
                           select item_Non).ToList();
591

    
592
            if (resultA.Count() >= 2)
593
            {
594
                // 동일한 사용자 데이터가 두개 이상 선택됩니다. Consolidation을 선택할까요?
595
                RadWindow.Confirm(new DialogParameters
596
                {
597
                    Owner = Application.Current.MainWindow,
598
                    Header = "Info",
599
                    Content = "Selected more than two identical user data. Do you want to select consolidation?",
600
                    Closed = new EventHandler<WindowClosedEventArgs>((sen, ea) =>
601
                    {
602
                        if (e.AddedItems.Count() == 1)
603
                        {
604
                            MarkupInfoItem instanceItem = e.AddedItems.FirstOrDefault() as MarkupInfoItem;
605
                            gridViewMarkup.SelectedItem = null;
606

    
607
                        
608
                            if (ea.DialogResult.Value)
609
                            {
610
                                if (instanceItem.Consolidate == 1 || instanceItem.PartConsolidate == 1)
611
                                {
612
                                    gridViewMarkup.SelectedItem = instanceItem;
613
                                }
614
                                else
615
                                {
616
                                    //1. Consolidations이 있다면 하나 선택
617
                                    var Consolidations = resultA.Where(item => (item.Consolidate == 1 || item.PartConsolidate == 1));
618

    
619
                                    if (Consolidations.Count() > 0)
620
                                    {
621
                                        gridViewMarkup.SelectedItem = Consolidations.First();
622
                                    }
623
                                }
624
                              
625
                            }
626
                            else {
627
                                //1. Consolidations 아닌게 있다면 Change
628
                                var notConsolidations = resultA.Where(item => !(item.Consolidate == 1 || item.PartConsolidate == 1));
629

    
630
                                if(notConsolidations.Count() > 0) {
631
                                    gridViewMarkup.SelectedItem = notConsolidations.First();
632
                                }
633
                                else {//1. Consolidations 아닌게 없다면 그대로
634
                                    gridViewMarkup.SelectedItem = instanceItem;
635
                                }
636
                                
637
                            }
638
                        }
639
                    }),
640
                    Theme = new Windows8Theme(),
641
                    ModalBackground = new System.Windows.Media.SolidColorBrush { Color = System.Windows.Media.Colors.Black, Opacity = 0.6 },
642
                });
643
            }
644
            else //파라미터로 넘어온 사용자가 걸리지 않은 경우
645
            {
646
                List<MarkupInfoItem> gridItem = gridViewMarkup.Items.Cast<MarkupInfoItem>().ToList(); //전체 마크업
647
                List<MarkupInfoItem> gridSelectionItem = gridViewMarkup.SelectedItems.Cast<MarkupInfoItem>().ToList(); //선택 된 마크업
648
                List<MarkupInfoItem> gridNonSelectionItem = gridItem.Except(gridSelectionItem).ToList(); //선택 되지 않은 마크업
649
#region 코멘트 보기
650
                if (e.AddedItems.Count() > 0 && this.pageNavigator.CurrentPage != null)
651
                {
652
                    foreach (var item in gridSelectionItem)
653
                    {
654
                        if (item.UserID == App.ViewInfo.UserID)
655
                        {
656
                            var controls = Common.ViewerDataModel.Instance.MarkupControls_USER.Where(d => d.MarkupInfoID == item.MarkupInfoID).ToList();
657
                            if (controls.Count == 0)
658
                            {
659
                                var instance = ViewerDataModel.Instance.MyMarkupList.Where(d => d.PageNumber == this.pageNavigator.CurrentPage.PageNumber && d.MarkupInfoID == item.MarkupInfoID).ToList();
660

    
661
                                foreach (var markup in instance)
662
                                {
663
                                    await MarkupParser.ParseExAsync(App.ViewInfo.ProjectNO, markup.Data, Common.ViewerDataModel.Instance.MarkupControls_USER, item.DisplayColor, "", 
664
                                        item.MarkupInfoID, markup.ID);
665
                                }
666
                            }
667

    
668
                            controls = Common.ViewerDataModel.Instance.MarkupControls_USER.Where(d => d.MarkupInfoID == item.MarkupInfoID).ToList();
669
                            foreach (var control in controls)
670
                            {
671
                                Logger.sendCheckLog(String.Format("=={0}", control), 1);
672
                                control.Visibility = Visibility.Visible;
673
                            }
674
                        }
675
                        else
676
                        {
677
                            var controls = Common.ViewerDataModel.Instance.MarkupControls.Where(d => d.MarkupInfoID == item.MarkupInfoID).ToList();
678
                            if (controls.Count == 0)
679
                            {
680
                                var instance = ViewerDataModel.Instance.MarkupList_Pre.Where(d => d.PageNumber == this.pageNavigator.CurrentPage.PageNumber && d.MarkupInfoID == item.MarkupInfoID).ToList();
681
                                foreach (var markup in instance)
682
                                {
683
                                    await MarkupParser.ParseExAsync(App.ViewInfo.ProjectNO, markup.Data, Common.ViewerDataModel.Instance.MarkupControls, item.DisplayColor, "",
684
                                        item.MarkupInfoID, markup.ID);
685
                                }
686
                            }
687

    
688
                            controls = Common.ViewerDataModel.Instance.MarkupControls.Where(d => d.MarkupInfoID == item.MarkupInfoID).ToList();
689
                            foreach (var control in controls)
690
                            {
691
                                control.Visibility = Visibility.Visible;
692
                            }
693
                        }
694
                    }
695
                }
696
#endregion
697
#region 코멘트 숨기기
698
                else if (e.RemovedItems.Count > 0)
699
                {
700
                    var _infoItem = e.RemovedItems.Cast<MarkupInfoItem>().ToList();
701

    
702
                    (from A in _infoItem
703
                     from B in Common.ViewerDataModel.Instance.MarkupControls
704
                     where A.MarkupInfoID == B.MarkupInfoID
705
                     select B).ToList().ForEach(data =>
706
                     {
707
                         data.Visibility = Visibility.Hidden;                         
708
                     });
709

    
710

    
711
                    var myComment = _infoItem.Where(d => d.UserID == App.ViewInfo.UserID).FirstOrDefault();
712
                    if (myComment != null)
713
                    {
714
                        Common.ViewerDataModel.Instance.MarkupControls_USER.ToList().ForEach(delegate (MarkupToPDF.Common.CommentUserInfo markupitem)
715
                        {
716
                            markupitem.Visibility = Visibility.Hidden;
717
                        });
718
                    }
719
                }
720
#endregion
721
            }
722
        }
723

    
724
        private void BaseClient_GetDocumentItemInfoCompleted(object sender, GetDocumentItemInfoCompletedEventArgs e)
725
        {
726
            Logger.sendResLog("GetDocumentItemInfoCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
727

    
728
            if (e.Result != null)
729
            {
730
                _DocItem = e.Result;
731

    
732
                if (e.Result.DOCUMENT_NO == "" && e.Result.DOCUMENT_NAME == "" && e.Result.GROUP_NO == "0")
733
                {
734
                    //장원
735
                    _ViewInfo.UserID = _ViewInfo.UserID.Replace(@"DAELIM\", "");
736
                }
737
                Logger.sendReqLog("GetDocInfoAsync", _ViewInfo.ProjectNO + "," + _ViewInfo.DocumentItemID, 1);
738
                BaseClient.GetDocInfoAsync(new KCOM_BasicParam { projectNo = _ViewInfo.ProjectNO, documentID = _ViewInfo.DocumentItemID });
739
            }
740
            else
741
            {
742
                DialogMessage_Alert("Document Information Not Found.", "Error");
743

    
744
                BaseClient.GetCheckSystemAdminCompleted += (sen, ea) =>
745
                {
746
                    Logger.sendResLog("GetCheckSystemAdminCompleted", "UserState : " + ea.UserState + "\r Result :" + ea.Result + "\r Cancelled :" + ea.Cancelled + "\r Error :" + ea.Error, 1);
747

    
748
                    if (ea.Error == null)
749
                    {
750
                        if (ea.Result)
751
                        {
752
                            this._ViewInfo.isAdmin = true;
753
                            Logger.sendReqLog("GetDocInfoAsync", _ViewInfo.ProjectNO + "," + _ViewInfo.DocumentItemID, 1);
754
                            BaseClient.GetDocInfoAsync(new KCOM_BasicParam { projectNo = _ViewInfo.ProjectNO, documentID = _ViewInfo.DocumentItemID });
755
                        }
756
                        else
757
                        {
758
                            //System.Windows.Forms.MessageBox.Show("해당 프로젝트에 가입되지 않은 유저입니다. 관리자에게 등록신청을 해주세요", "안내");
759
                            DialogMessage_Alert("You are not belong to this project. Please ask admin for an authority.", "Info");
760
                            //this.ParentOfType<MainPage>().DialogMessage_Alert("해당 프로젝트에 가입되지 않은 유저입니다. 관리자에게 등록신청을 해주세요", "안내");
761
                        }
762

    
763
                    }
764
                    else
765
                    {
766
                        //System.Windows.Browser.HtmlPage.Window.Invoke("close");
767
                    }
768
                };
769
                Logger.sendReqLog("GetCheckSystemAdminAsync", _ViewInfo.UserID, 1);
770
                BaseClient.GetCheckSystemAdminAsync(this._ViewInfo.UserID);
771
            }
772
        }
773

    
774
        private void BaseClient_GetSystemDataCompleted(object sender, GetSystemDataCompletedEventArgs e)
775
        {
776
            Logger.sendResLog("GetSystemDataCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
777

    
778
            if (e.Error == null && e.Result != null)
779
            {
780
                App.SystemInfo = e.Result;
781

    
782
                //App.urlPort_DB = App.SystemInfo.HostPort;
783
                //App.urlHost_DB = App.SystemInfo.HostName;
784
                Logger.sendReqLog("GetConversionStateAsync", _ViewInfo.ProjectNO + "," + _ViewInfo.DocumentItemID, 1);
785
                BaseClient.GetConversionStateAsync(new KCOM_BasicParam { projectNo = _ViewInfo.ProjectNO, documentID = _ViewInfo.DocumentItemID });
786

    
787
                //if (App.urlHost.Contains("localhost"))
788
                //{
789
                //    App.urlHost = App.urlHost_DB;
790
                //}
791
            }
792
        }
793

    
794
        private void BaseClient_GetConversionStateCompleted(object sender, GetConversionStateCompletedEventArgs e)
795
        {
796
            
797
            Logger.sendResLog("GetConversionStateCompleted", "UserState : " + e.UserState + "\r Result :" + e.Result + "\r Cancelled :" + e.Cancelled + "\r Error :" + e.Error, 1);
798
            string failedmsg = CommonLib.Common.GetAlertMessageString("GetConversionStateFailed", "MSG", "Too Many V/P Files are uploading instantaneously, This V/P file can not be open now. Please feel free to re-open wait a moment again!\n\n현재 일시적으로 많은 양의 V/P가 등록되고 있어 이 문서를 열람할 수 없습니다.\n잠시만 기다려 주신 후 다시 열람하여 주시기 바랍니다");
799
            if (e.Error == null && e.Result == true)
800
            {
801
                Logger.sendReqLog("GetDocumentItemInfoAsync", _ViewInfo.ProjectNO + "," + _ViewInfo.DocumentItemID + "," + _ViewInfo.UserID, 1);
802

    
803
                BaseClient.GetDocumentItemInfoAsync(new KCOM_BasicParam { projectNo = _ViewInfo.ProjectNO, documentID = _ViewInfo.DocumentItemID, userID = _ViewInfo.UserID });
804

    
805
                if (App.ViewInfo != null && App.ViewInfo.ProjectNO != "")
806
                {
807
                    Logger.sendReqLog("GetMember", _ViewInfo.ProjectNO + "," + _ViewInfo.UserID, 1);
808
                    userData = BaseClient.GetMember(App.ViewInfo.ProjectNO, App.ViewInfo.UserID);
809
                    if (userData != null)
810
                    {
811
                        Logger.sendReqLog("GetDeptDataAsync", userData.DEPARTMENT, 1);
812
                        BaseClient.GetDeptDataAsync(userData.DEPARTMENT);
813
                    }
814
                    else
815
                    {
816
                        userData = new MEMBER();
817
                        userData.DEPARTMENT = "ALL";
818
                        Logger.sendReqLog("GetDeptDataAsync", userData.DEPARTMENT, 1);
819
                        BaseClient.GetDeptDataAsync(userData.DEPARTMENT);
820
                    }
821
                }
822
            }
823
            else
824
            {
825
                
826
                DialogMessage_Alert(failedmsg, "Warning");
827
            }
828
        }
829

    
830
        public void SetView(ViewInfo info)
831
        { 
832
            this._ViewInfo = info;
833
            try
834
            {
835
                /// 대림의 경우 EnsembleID를 가지고 DocumentItemID를 구함
836
                /// 다른 Site를 대비하여 에러 처리함
837
                string docid = BaseClient.GetDocItemID(info.EnsembleID);
838
                if (!string.IsNullOrEmpty(docid))
839
                {
840
                    this._ViewInfo.DocumentItemID = docid;
841
                }
842
            }
843
            catch(Exception ex)
844
            {
845
                App.FileLogger.Fatal(ex);
846
            }
847

    
848
            Logger.sendReqLog("GetSystemDataAsync", "", 1);
849
            BaseClient.GetSystemDataAsync();
850
        }
851

    
852
        public void HubSet()
853
        {
854
            // 허브 연결 구성
855
            var hubConnection = new HubConnection(CommonLib.Common.GetConfigString("HubAddress", "URL", "", App.isExternal));
856
            IHubProxy MarkusLicenseProxy = hubConnection.CreateHubProxy("MarkusLicenseHub");
857

    
858

    
859
            // 허브 연결 
860
            hubConnection.Start().ContinueWith(task =>
861
            {
862
                if (task.IsFaulted)
863
                {
864
                    Console.WriteLine("There was an error opening the connection:{0}",
865
                                      task.Exception.GetBaseException());
866
                    //throw task.Exception.GetBaseException();
867
                }
868
                else
869
                {
870
                    Console.WriteLine("Connected :: " + task.Id);
871
                }
872

    
873
            }).Wait();
874

    
875

    
876
            // [ 프로그램 시작 ]
877
            // 클라이언트에서 서버를 호출 ( 접속 시 )  : 라이센스 체크
878
            //( "method name", arg1, arg2 )
879
            //MarkusLicenseProxy.Invoke<string>("ConnectionMarkus", "H2009115", "고동균").ContinueWith(task =>
880
            //{
881
            //    if (task.IsFaulted)
882
            //    {
883
            //        Console.WriteLine("ConnectionMarkus :: There was an error calling send: {0}", task.Exception.GetBaseException());
884
            //    }
885
            //    else
886
            //    {
887
            //        Console.WriteLine(task.Result);
888
            //    }
889
            //});
890
            MarkusLicenseProxy.Invoke<string>("ConnectionMarkus", App.ViewInfo.UserID,App.ViewInfo.ProjectNO).ContinueWith(task =>
891
            {
892
                if (task.IsFaulted)
893
                {
894
                    Console.WriteLine("ConnectionMarkus :: There was an error calling send: {0}", task.Exception.GetBaseException());
895
                    //throw task.Exception.GetBaseException();
896
                }
897
                else
898
                {
899
                    switch (task.Result)
900
                    {
901
                        case "2":
902
                            {
903
                                var result = MessageBox.Show("권한이 없습니다.");
904
                                if (result == MessageBoxResult.OK)
905
                                    System.Diagnostics.Process.GetCurrentProcess().Kill();
906
                            }
907
                            break;
908
                        case "404":
909
                            {
910
                                var result = MessageBox.Show("등록된 사용자가 아닙니다.");
911
                                if(result == MessageBoxResult.OK)
912
                                    System.Diagnostics.Process.GetCurrentProcess().Kill();
913
                            }
914
                            break;
915
                    }
916
                }
917
            });
918

    
919
            // 서버에서 클라이언트 호출 ( 프로그램 구동여부 ) : 라이센스 체크 후 응답 ( 반환값 : true, false ) 
920
            //MarkusLicenseProxy.On<bool>("IsConnection", status =>
921
            //{
922

    
923
            //    if (status)
924
            //    {
925
            //        // 프로그램 시작 
926
            //        Console.WriteLine("connection status : " + status);
927
            //    }
928
            //    else
929
            //    {
930
            //        // 프로그램 종료 ( signalr 접속 끊기 )  
931
            //        hubConnection.Stop();
932
            //    }
933
            //});
934

    
935
            // 공지 메세지 ( 서버에서 클라이언트 호출 ) 
936
            MarkusLicenseProxy.On<string>("MarkusNotification", msg =>
937
                Console.WriteLine("Noficiation Message : " + msg)
938
            );
939

    
940

    
941
            // 서버 메서드 호출 
942
            //MarkusLicenseProxy.Invoke<string>("DoSomething", "I'm doing something!!!").Wait();
943

    
944
            //Console.ReadKey();
945

    
946
            //hubConnection.Stop();
947
        }
948

    
949
        public async void MarkupUserListDisplay()
950
        {
951
            List<MarkupInfoItem> gridItem = gridViewMarkup.Items.Cast<MarkupInfoItem>().ToList(); //전체 마크업
952
            List<MarkupInfoItem> gridSelectionItem = gridViewMarkup.SelectedItems.Cast<MarkupInfoItem>().ToList(); //선택 된 마크업
953
            List<MarkupInfoItem> gridNonSelectionItem = gridItem.Except(gridSelectionItem).ToList(); //선택 되지 않은 마크업
954
                                                                                                     //MarkupToPDF.Controls.Parsing.LayerControl layerControl = new MarkupToPDF.Controls.Parsing.LayerControl();
955
            Common.ViewerDataModel.Instance.MarkupControls_USER.Clear();  //전체 제거
956
            Common.ViewerDataModel.Instance.MarkupControls.Clear();  //전체 제거
957
            //foreach (var item in gridSelectionItem)
958
            //{
959
            //    if (item.UserID == App.ViewInfo.UserID)
960
            //    {
961
            //        ViewerDataModel.Instance.MyMarkupList.Where(data => data.PageNumber == pageNavigator.CurrentPage.PageNumber && data.MarkupInfoID == item.MarkupInfoID).ToList().ForEach(delegate (MarkupItemEx markupitem)
962
            //        {
963
            //            layerControl.markupParseEx(markupitem.Data, Common.ViewerDataModel.Instance.MarkupControls_USER, item.DisplayColor, "", item.MarkupInfoID, markupitem.ID);
964
            //        });
965
            //    }
966
            //    else
967
            //    {
968
            //        ViewerDataModel.Instance.MarkupList_Pre.Where(data => data.PageNumber == pageNavigator.CurrentPage.PageNumber && data.MarkupInfoID == item.MarkupInfoID).ToList().ForEach(delegate (MarkupItemEx markupitem)
969
            //        {
970
            //            layerControl.markupParse(markupitem.Data, Common.ViewerDataModel.Instance.MarkupControls, item.DisplayColor, "", item.MarkupInfoID);
971
            //        });
972
            //    }
973
            //}
974
                (from A in gridSelectionItem
975
                 from B in Common.ViewerDataModel.Instance.MarkupControls_USER
976
                 where A.MarkupInfoID == B.MarkupInfoID
977
                 select B).ToList().ForEach(data =>
978
                 {
979
                     Common.ViewerDataModel.Instance.MarkupControls_USER.Remove(data);
980
                 });
981
            (from A in gridSelectionItem
982
             from B in Common.ViewerDataModel.Instance.MarkupControls
983
             where A.MarkupInfoID == B.MarkupInfoID
984
             select B).ToList().ForEach(data =>
985
             {
986
                 Common.ViewerDataModel.Instance.MarkupControls.Remove(data);
987
             });
988

    
989
            foreach (var item in gridSelectionItem)
990
            {
991

    
992
                if (item.UserID == App.ViewInfo.UserID)
993
                {
994
                    var markupItems = ViewerDataModel.Instance.MyMarkupList.Where(data => data.PageNumber == pageNavigator.CurrentPage.PageNumber && data.MarkupInfoID == item.MarkupInfoID).ToList();
995

    
996
                    foreach (var markupitem in markupItems)
997
                    {
998
                        await MarkupParser.ParseExAsync(App.ViewInfo.ProjectNO, markupitem.Data, Common.ViewerDataModel.Instance.MarkupControls_USER, item.DisplayColor, "", 
999
                                item.MarkupInfoID, markupitem.ID);
1000
                    }
1001
                }
1002
                else
1003
                {
1004
                    var markupItems = ViewerDataModel.Instance.MarkupList_Pre.Where(data => data.PageNumber == pageNavigator.CurrentPage.PageNumber && data.MarkupInfoID == item.MarkupInfoID).ToList();
1005

    
1006
                    foreach (var markupitem in markupItems)
1007
                    {
1008
                        await MarkupParser.ParseExAsync(App.ViewInfo.ProjectNO, markupitem.Data, Common.ViewerDataModel.Instance.MarkupControls, item.DisplayColor, "", item.MarkupInfoID);
1009
                    }
1010
                }
1011
            }
1012
        }
1013
    }
1014
}
클립보드 이미지 추가 (최대 크기: 500 MB)