using AR; using FarPoint.CalcEngine; using FarPoint.Excel; using FarPoint.Excel.EntityClassLibrary.DrawingEx; using NetOffice.OfficeApi.Enums; using NetOffice.OutlookApi; using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Data.Entity.Core.Mapping; using System.Drawing; using System.Linq; using System.Security.Cryptography.X509Certificates; using System.Text; using System.Threading.Tasks; using System.Windows.Forms; namespace FPJ0000.JobReport_ { public partial class fJobProcessManager : Form { string GroupCode = "15"; public fJobProcessManager() { InitializeComponent(); this.dsJobReport.Common.TableNewRow += Common_TableNewRow; this.FormClosing += FJobProcessManager_FormClosing; // 이벤트 핸들러 연결 treeView1.ItemDrag += TreeView1_ItemDrag; treeView1.DragEnter += TreeView1_DragEnter; treeView1.DragDrop += TreeView1_DragDrop; } private void TreeView1_ItemDrag(object sender, ItemDragEventArgs e) { // Ctrl 키가 눌린 상태면 복사, 아니면 이동 DragDropEffects effect = (ModifierKeys == Keys.Control) ? DragDropEffects.Copy : DragDropEffects.Move; DoDragDrop(e.Item, effect); } private void TreeView1_DragEnter(object sender, DragEventArgs e) { // 드래그한 항목이 TreeNode인 경우에만 허용 if (e.Data.GetDataPresent(typeof(TreeNode))) { // Ctrl 키가 눌린 상태면 복사 효과 표시 e.Effect = (ModifierKeys == Keys.Control) ? DragDropEffects.Copy : DragDropEffects.Move; } else { e.Effect = DragDropEffects.None; } } private void TreeView1_DragDrop(object sender, DragEventArgs e) { // 드롭된 위치의 노드 찾기 Point pt = treeView1.PointToClient(new Point(e.X, e.Y)); TreeNode destinationNode = treeView1.GetNodeAt(pt); TreeNode draggedNode = (TreeNode)e.Data.GetData(typeof(TreeNode)); // 자기 자신이나 자식 노드로는 이동할 수 없음 if (destinationNode == null || draggedNode == destinationNode || IsChildNode(draggedNode, destinationNode)) { return; } var dr = draggedNode.Tag as dsJobReport.CommonRow; if (e.Effect == DragDropEffects.Move) //이동한다면 기존 데이터는 삭제한다 { draggedNode.Remove(); } var inseritem = e.Effect == DragDropEffects.Copy; if (draggedNode.Level == 1) //분류를 움직이고 있다 { if (destinationNode.Level == 2) { var idx = destinationNode.Parent.Parent.Nodes.Add((TreeNode)draggedNode.Clone()); var cnd = destinationNode.Parent.Parent.Nodes[idx]; UpdateParentInfo(cnd, inseritem); } else if (destinationNode.Level == 1) { var idx = destinationNode.Parent.Nodes.Add((TreeNode)draggedNode.Clone()); var cnd = destinationNode.Parent.Nodes[idx]; UpdateParentInfo(cnd, inseritem); } else { var idx = destinationNode.Nodes.Add((TreeNode)draggedNode.Clone()); var cnd = destinationNode.Nodes[idx]; UpdateParentInfo(cnd, inseritem); } } else { // 노드 이동 if (destinationNode.Level > 1) { var idx = destinationNode.Parent.Nodes.Add((TreeNode)draggedNode.Clone()); var cnd = destinationNode.Parent.Nodes[idx]; UpdateParentInfo(cnd, inseritem); } else { var idx = destinationNode.Nodes.Add((TreeNode)draggedNode.Clone()); var cnd = destinationNode.Nodes[idx]; UpdateParentInfo(cnd, inseritem); } } } void UpdateParentInfo(TreeNode cnd, bool newitem) { if (cnd.Level == 2) //최하위 { var grp = cnd.Parent.Text; var prc = cnd.Parent.Parent.Text; //신규로추가를 해준다 if (newitem) { dsJobReport.CommonRow ndr = this.dsJobReport.Common.NewCommonRow(); ndr.svalue2 = prc; ndr.svalue = grp; ndr.memo = cnd.Text; ndr.wuid = FCOMMON.info.Login.no; ndr.wdate = DateTime.Now; ndr.grp = this.GroupCode; ndr.code = makenewcode(); this.dsJobReport.Common.AddCommonRow(ndr); cnd.Tag = ndr; } else { //기존정보를 업데이트한다 var dr = cnd.Tag as dsJobReport.CommonRow; dr.svalue = grp; dr.svalue2 = prc; dr.EndEdit(); } } else if (cnd.Level == 1) //분류(하위항목업데이트필요) { var prc = cnd.Parent.Text; foreach (TreeNode nd in cnd.Nodes) { //신규로추가를 해준다 if (newitem) { dsJobReport.CommonRow ndr = this.dsJobReport.Common.NewCommonRow(); ndr.svalue2 = prc; ndr.svalue = cnd.Text; ndr.memo = nd.Text; ndr.wuid = FCOMMON.info.Login.no; ndr.wdate = DateTime.Now; ndr.grp = this.GroupCode; ndr.code = makenewcode(); this.dsJobReport.Common.AddCommonRow(ndr); nd.Tag = cnd; } else { var dr = nd.Tag as dsJobReport.CommonRow; dr.svalue2 = prc; dr.EndEdit(); } } } else if (cnd.Level == 0) //프로세스(하위항모업데이트필요) { } } private bool IsChildNode(TreeNode parent, TreeNode node) { // 대상 노드가 현재 노드의 자식인지 확인 if (node.Parent == null) return false; if (node.Parent == parent) return true; return IsChildNode(parent, node.Parent); } private void FJobProcessManager_FormClosing(object sender, FormClosingEventArgs e) { if (haschanged() == true) { var dlg = UTIL.MsgQ("변경된 내용이 있습니다 그래도 화면을 닫을까요?"); if (dlg != DialogResult.Yes) { e.Cancel = true; return; } } } bool haschanged() { var dt = this.dsJobReport.Common.GetChanges(); if (dt == null) return false; if (dt.Rows.Count < 1) return false; return true; } private void Common_TableNewRow(object sender, DataTableNewRowEventArgs e) { e.Row["wdate"] = DateTime.Now; e.Row["wuid"] = FCOMMON.info.Login.no; e.Row["gcode"] = FCOMMON.info.Login.gcode; e.Row["grp"] = this.GroupCode; } private void commonBindingNavigatorSaveItem_Click(object sender, EventArgs e) { this.Validate(); this.bs.EndEdit(); if (this.haschanged() == false) { UTIL.MsgI("변경된 내용이 없습니다"); return; } try { var cnt = this.tam.UpdateAll(this.dsJobReport); AR.UTIL.MsgI($"{cnt}건의 자료가 업데이트 되었습니다"); RefreshTreeview(); } catch (System.Exception ex) { UTIL.MsgE(ex.Message); } } private void RefreshData() { try { this.ta.Fill(this.dsJobReport.Common, FCOMMON.info.Login.gcode, GroupCode); this.commonDataGridView.AutoResizeColumns(); } catch (System.Exception ex) { AR.UTIL.MsgE(ex.Message); } RefreshTreeview(); } private void fJobProcessManager_Load(object sender, EventArgs e) { RefreshData(); } void RefreshTreeview() { this.treeView1.Nodes.Clear(); //dsJobReport.AcceptChanges(); var items = this.dsJobReport.Common.Where(t => t.RowState != DataRowState.Detached && t.RowState != DataRowState.Deleted && string.IsNullOrEmpty(t.svalue2)); foreach (var item in items) { item.svalue2 = "N/A"; item.EndEdit(); } var grp_process = dsJobReport.Common.OrderBy(t => t.svalue2).GroupBy(t => t.svalue2); foreach (var prc in grp_process) { var prcname = prc.Key; if (prcname.isEmpty()) prcname = "N/A"; var nd = treeView1.Nodes.Add(prcname, prcname); //add process node var grp_jobtype = prc.OrderBy(t => t.svalue).GroupBy(t => t.svalue); foreach (var grp in grp_jobtype) //job type { var nd2 = nd.Nodes.Add(grp.Key, grp.Key); foreach (var item in grp.OrderBy(t => t.memo)) { var nd3 = nd2.Nodes.Add(item.memo, item.memo); //job item if (prcname.Equals("N/A")) nd3.ForeColor = Color.DimGray; else nd3.ForeColor = Color.Black; nd3.Tag = item; } } } treeView1.ExpandAll(); //show all groupBox3.Enabled = false; } private void tbFind_KeyDown(object sender, KeyEventArgs e) { var search = tbFind.Text.Trim(); if (e.KeyCode == Keys.Enter) { try { var cols = new string[] { "svalue2", "svalue", "memo" }; var filterstring = ""; foreach (var col in cols) { if (filterstring.Length > 0) filterstring += $" or "; filterstring += $"{col} like '%{search}%'"; } if (search.isEmpty()) { this.bs.Filter = ""; tbFind.BackColor = Color.White; } else { this.bs.Filter = filterstring; tbFind.BackColor = Color.Lime; } } catch (System.Exception ex) { AR.UTIL.MsgE(ex.Message); tbFind.BackColor = Color.HotPink; } tbFind.Focus(); tbFind.SelectAll(); } } private void groupBox2_Enter(object sender, EventArgs e) { } private void button1_Click(object sender, EventArgs e) { RefreshTreeview(); } private void button2_Click(object sender, EventArgs e) { //edit var prc = tbProcessE.Text.Trim(); var grp = tbTypeE.Text.Trim(); var item = tbItemE.Text.Trim(); if (item.isEmpty()) { UTIL.MsgE("형태는 반드시 입력되어야 합니다"); tbItemE.Focus(); return; } var nd = this.treeView1.SelectedNode; if (nd == null) { UTIL.MsgE("선택된 노드가 없습니다"); return; } if (nd.Tag == null) { UTIL.MsgE("선택노드에 정보가 없습니다\n개발자 문의"); return; } //동일데이터가있는지확인 var exist = this.dsJobReport.Common.Where(t => t.RowState != DataRowState.Deleted && t.svalue2.Equals(prc) && t.svalue.Equals(grp) && t.memo.Equals(item)).Any(); if (exist) { UTIL.MsgE("동일한 항목이 존재합니다 / 확인 후 다시 진행 하세요"); return; } var dr = nd.Tag as dsJobReport.CommonRow; if (dr == null) return; var changecate = dr.svalue2.Equals(prc) == false || dr.svalue.Equals(grp) == false; dr.svalue2 = prc; dr.svalue = grp; dr.memo = item; dr.EndEdit(); //카테고리가 변경되었다면 위치를 이동시켜야한다 nd.Remove(); //선택노드를 삭제한다. AddNode(prc, grp, item, dr); } void AddNode(string prc, string grp, string item, dsJobReport.CommonRow dr) { //프로세스 노드 선택 TreeNode tn_prc = null; if (prc.isEmpty()) prc = "N/A"; var prc_nodes = this.treeView1.Nodes.Find(prc, false); if (prc_nodes.Any() == false) { tn_prc = treeView1.Nodes.Add(prc, prc); //신규노드추가 } else { tn_prc = prc_nodes.First(); //첫번째 노드로 변경 } //분류노드확인 TreeNode tn_type = null; var type_nodes = tn_prc.Nodes.Find(grp, false); if (type_nodes.Any() == false) { tn_type = tn_prc.Nodes.Add(grp, grp); //신규노드추가 } else { tn_type = type_nodes.First(); //첫번째 노드로 변경 } //아이템노드확인 TreeNode tn_item = null; var item_nodes = tn_type.Nodes.Find(item, false); if (item_nodes.Any() == false) { tn_item = tn_type.Nodes.Add(item, item); //신규노드추가 } else { tn_item = item_nodes.First(); //동일한 노드이다 } tn_item.Tag = dr; treeView1.SelectedNode = tn_item; } private void button3_Click(object sender, EventArgs e) { //add var prc = tbProcessA.Text.Trim(); var grp = tbTypeA.Text.Trim(); var item = tbItemA.Text.Trim(); if (item.isEmpty()) { UTIL.MsgE("형태는 반드시 입력되어야 합니다"); tbItemA.Focus(); return; } //동일데이터가있는지확인 var exist = this.dsJobReport.Common.Where(t => t.RowState != DataRowState.Deleted && t.svalue2.Equals(prc) && t.svalue.Equals(grp) && t.memo.Equals(item)).Any(); if (exist) { UTIL.MsgE("동일한 항목이 존재합니다 / 확인 후 다시 진행 하세요"); tbItemA.Focus(); tbItemA.SelectAll(); return; } var dr = this.dsJobReport.Common.NewCommonRow(); dr.svalue2 = prc; dr.svalue = grp; dr.memo = item; dr.wuid = FCOMMON.info.Login.no; dr.wdate = DateTime.Now; dr.grp = this.GroupCode; dr.code = makenewcode(); this.dsJobReport.Common.AddCommonRow(dr); AddNode(prc, grp, item, dr); tbItemA.Focus(); tbItemA.SelectAll(); } string makenewcode() { string lastcodename = ""; var lastcodedr = dsJobReport.Common.OrderByDescending(t => t.code).FirstOrDefault(); if (lastcodedr != null) lastcodename = lastcodedr.code; int newnum = 0; int codenameleng = lastcodename.Length; if (codenameleng == 0) codenameleng = 4; if (int.TryParse(lastcodename, out int lastcodenum) == false) { //숫자로되어있지 않다 newnum = 1;// newnum.PadLeft(codenameleng, '0'); } else { //숫자로되어있으니 +1해서 처리한다. newnum = (lastcodenum + 1); } var newnumstr = newnum.ToString().PadLeft(codenameleng, '0'); return newnumstr; } private void tv_AfterSelect(object sender, TreeViewEventArgs e) { label8.Text = "Path://"; var tn = this.treeView1.SelectedNode; if (tn == null) return; if (tn.Tag == null) return; this.label8.Text = tn.FullPath; var dr = tn.Tag as dsJobReport.CommonRow; if (dr == null) return; tbProcessE.Text = dr.svalue2; tbTypeE.Text = dr.svalue; tbItemE.Text = dr.memo; tbProcessA.Text = dr.svalue2; tbTypeA.Text = dr.svalue; groupBox3.Enabled = true; groupBox3.Text = $"선택자료 편집({dr.idx})"; } private void tbItemA_KeyDown(object sender, KeyEventArgs e) { if (e.KeyCode == Keys.Enter) btAdd.PerformClick(); } private void tbItemE_KeyDown(object sender, KeyEventArgs e) { if (e.KeyCode == Keys.Enter) btEdit.PerformClick(); } private void 클립보드에서ToolStripMenuItem_Click(object sender, EventArgs e) { UTIL.MsgI("목록화면고 동일한 배치의 데이터를 클립보드에서 가져옵니다\n(항목/분류/프로세스)"); var data = Clipboard.GetText(); var lines = data.Split(new string[] { Environment.NewLine }, StringSplitOptions.RemoveEmptyEntries); var dlg = UTIL.MsgQ($"{lines.Length} 개의 데이터가 있습니다.\n\n입력 할까요?"); if (dlg != DialogResult.Yes) return; var icnt = 0; var scnt = 0; var ecnt = 0; foreach (var line in lines) { var buf = line.Split(new char[] { '\t' }, StringSplitOptions.RemoveEmptyEntries); var item = buf[0].Trim(); if (item.isEmpty()) { ecnt += 1; continue; } var grp = buf.Length > 1 ? buf[1].Trim() : string.Empty; var prc = buf.Length > 2 ? buf[2].Trim() : string.Empty; //동일항목체크 if (this.dsJobReport.Common.Where(t => t.RowState != DataRowState.Detached && t.RowState != DataRowState.Deleted && t.memo.Equals(item) && t.svalue.Equals(grp) && t.svalue2.Equals(prc)).Any()) { scnt += 1; continue; } icnt += 1; var dr = this.dsJobReport.Common.NewCommonRow(); dr.svalue2 = prc; dr.svalue = grp; dr.memo = item; dr.wuid = FCOMMON.info.Login.no; dr.wdate = DateTime.Now; dr.grp = this.GroupCode; dr.code = makenewcode(); this.dsJobReport.Common.AddCommonRow(dr); } var msg = $"가져오기 결과 입니다\n\n추가:{icnt}\n중복:{scnt}\n오류:{ecnt}"; AR.UTIL.MsgI(msg); if (icnt > 0) RefreshTreeview(); } private void button2_Click_1(object sender, EventArgs e) { //edit var nd = this.treeView1.SelectedNode; if (nd == null) { UTIL.MsgE("선택된 노드가 없습니다"); return; } if (nd.Tag == null) { UTIL.MsgE("선택노드에 정보가 없습니다\n최하위 아이템을 선택하세요\n오류가 반복되면 개발사에 문의 하세요"); return; } //동일데이터가있는지확인 var dr = nd.Tag as dsJobReport.CommonRow; var prc = dr.svalue2; var grp = dr.svalue; var item = dr.memo; var exist = this.dsJobReport.Common.Where(t => t.RowState != DataRowState.Deleted && t.svalue2.Equals(prc) && t.svalue.Equals(grp) && t.memo.Equals(item)).FirstOrDefault(); if (exist == null) { UTIL.MsgE("해당 데이터를 찾지 못했습니다\n잠시 후 다시 시도하세요"); return; } exist.Delete(); // dsJobReport.Common.RemoveCommonRow(exist); RefreshTreeview(); } private void btDelete_Click(object sender, EventArgs e) { RefreshTreeview(); } private void btInsert_Click(object sender, EventArgs e) { } private void treeView1_KeyDown(object sender, KeyEventArgs e) { if(e.KeyCode == Keys.Delete) { var nd = this.treeView1.SelectedNode; if (nd == null) return; if(nd.Level == 2) { var dr = nd.Tag as dsJobReport.CommonRow; nd.Remove(); dr.Delete(); } else { UTIL.MsgE("최하위 항목만 삭제 가능합니다"); } } } } }