ProjectManagerFme.pas 56 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843
  1. unit ProjectManagerFme;
  2. interface
  3. uses
  4. ProjectManagerDm, ZhAPI,
  5. NewProjectFrm,
  6. Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  7. Dialogs, ZjGridDBA, ZJGrid, ComCtrls, ToolWin, ActnList,
  8. dxBar, sdGridDBA, sdGridTreeDBA, sdIDTree, CslJson, ExtCtrls,
  9. StdCtrls, sdDB, CslButton, OrderCheckerFme, Contnrs;
  10. type
  11. TStrArr = array of string;
  12. TProjectManagerFrame = class(TFrame)
  13. ToolBar: TToolBar;
  14. tobtnOpen: TToolButton;
  15. zgProjects: TZJGrid;
  16. tobtnDelete: TToolButton;
  17. ActionList1: TActionList;
  18. actnOpen: TAction;
  19. actnDelete: TAction;
  20. dxpmProjectManager: TdxBarPopupMenu;
  21. actnReceiveProject: TAction;
  22. stdProjects: TsdGridTreeDBA;
  23. actnNewProject: TAction;
  24. actnNewSubProject: TAction;
  25. actnNewTender: TAction;
  26. pnlTenderProperty: TPanel;
  27. sdTenderProperty: TsdGridDBA;
  28. sprProperty: TSplitter;
  29. tobtnRenane: TToolButton;
  30. actnRename: TAction;
  31. tobtnImport: TToolButton;
  32. tobtnExport: TToolButton;
  33. actnImport: TAction;
  34. actnExport: TAction;
  35. tobtn1: TToolButton;
  36. pnlWeb: TPanel;
  37. pnlProject: TPanel;
  38. shp2: TShape;
  39. shp1: TShape;
  40. shp3: TShape;
  41. shp4: TShape;
  42. pnlTenderTitle: TPanel;
  43. lblBidName: TLabel;
  44. pnlShadow: TPanel;
  45. pnlProgress: TPanel;
  46. lblPeriodTotal: TLabel;
  47. lblPeriodState: TLabel;
  48. lblProgress: TLabel;
  49. lblPeriod: TLabel;
  50. pnlBelongProject: TPanel;
  51. lblBelongProject: TLabel;
  52. lblProjName: TLabel;
  53. lblLeftHalfBracket: TLabel;
  54. lblOnwerCompany: TLabel;
  55. lblOnwerName: TLabel;
  56. pnlProjectType: TPanel;
  57. lblProjectType: TLabel;
  58. lblWebProjCtgyName: TLabel;
  59. zgTenderProperty: TZJGrid;
  60. actnOpenBackupFolder: TAction;
  61. sbChecker: TScrollBox;
  62. procedure actnOpenExecute(Sender: TObject);
  63. procedure actnDeleteExecute(Sender: TObject);
  64. procedure zgProjectsMouseDown(Sender: TObject; Button: TMouseButton;
  65. Shift: TShiftState; X, Y: Integer);
  66. procedure actnReceiveProjectExecute(Sender: TObject);
  67. procedure actnNewProjectExecute(Sender: TObject);
  68. procedure actnNewSubProjectExecute(Sender: TObject);
  69. procedure actnNewTenderExecute(Sender: TObject);
  70. procedure zgProjectsDrawCellText(ACanvas: TCanvas; const ARect: TRect;
  71. const ACoord: TPoint; AGrid: TZJGrid; const Text: String;
  72. var ADefaultDraw: Boolean);
  73. procedure actnNewSubProjectUpdate(Sender: TObject);
  74. procedure actnNewTenderUpdate(Sender: TObject);
  75. procedure zgProjectsCurrentChanged(Sender: TObject; Col, Row: Integer);
  76. procedure actnSubmitProjectUpdate(Sender: TObject);
  77. procedure actnReplyProjectUpdate(Sender: TObject);
  78. procedure actnRenameExecute(Sender: TObject);
  79. procedure actnOpenUpdate(Sender: TObject);
  80. procedure actnImportExecute(Sender: TObject);
  81. procedure actnExportExecute(Sender: TObject);
  82. procedure actnDeleteUpdate(Sender: TObject);
  83. procedure actnOpenBackupFolderExecute(Sender: TObject);
  84. procedure actnRenameUpdate(Sender: TObject);
  85. procedure zgProjectsShowHint(var HintStr: String; var CanShow: Boolean;
  86. var HintInfo: THintInfo; const ACoord: TPoint);
  87. procedure actnExportUpdate(Sender: TObject);
  88. procedure actnOpenBackupFolderUpdate(Sender: TObject);
  89. procedure pnlProgressClick(Sender: TObject);
  90. private
  91. FProjectManagerData: TProjectManagerData;
  92. // Chenshilong,2016.03.24
  93. // 这部分线上和本地一致,无需区分
  94. FID: Integer; // 本地标段文件ID
  95. FWebID: Integer; // 关联服务器用的ID(服务器有自己的ID体系)
  96. FWebAuthorID: Integer; // 编制人
  97. FWebOwnerID: Integer; // 业主
  98. // 本地存储值、线上存储值。线上修改后,需要同步到本地
  99. FWebMD5_Local: string; // 本地存储的MD5
  100. FWebMD5_OnLine: string; // 线上存储的最新的MD5(下同)
  101. FWebBidName_Local: string; // 标段名 (当无法同线上取得联系时,本地需要用到该名称来提示)
  102. FWebBidName_OnLine: string;
  103. FWebFolder_OnLine: string; // 这个命名不妥,但很直观。线上的项目名称、项目类型概念跟本地颠倒,很混乱。
  104. FWebSubFolder_OnLine: string;
  105. FWebOwnerCompany: string;
  106. FWebOwnerRole: string;
  107. FWebOwnerName: string;
  108. FWebCheckStatusMy: TCheckStatus; // 登陆用户在当前项目中的工作状态。
  109. FWebCheckStatusProject: TCheckStatus; // 项目的审核状态。
  110. FOnLineCheckerBegin: Integer; // 线上审批的起始人。 010110111 起7止9。 0101101110 起0止0。
  111. FOnLineCheckerEnd: Integer; // 线上审批的截止人。
  112. FOnLineCheckerEndIsOwner: Boolean; // 终审是线上审批
  113. FPhaseTotal: Integer;
  114. FPhaseNo: Integer;
  115. FCurPos: Integer; // 用来控制审核人的添加位置
  116. FCheckerList: TObjectList;
  117. function ReceiveFile(const AFileName: string; AIsReback: Boolean = False): Boolean;
  118. function ImportFile(const AFileName: string; AFileMD5: string = ''): Boolean;
  119. procedure ConnectButtonWithAction;
  120. function GetImportProjectName(const AFileName: string; AParent: TsdIDTreeNode): string;
  121. function IsProject(ANode: TsdIDTreeNode): Boolean;
  122. function IsLeafProject(ANode: TsdIDTreeNode): Boolean;
  123. function IsUnEmptyLeafProject(ANode: TsdIDTreeNode): Boolean;
  124. function CheckOpened(ANode: TsdIDTreeNode): Boolean;
  125. procedure SetPropertyVisible(AVisible: Boolean);
  126. procedure ShowProjectInfoTopAndCheckers;
  127. // 网络上的目录结构,本地有则定位,没有则创建。
  128. procedure CheckWebFolders(AFolderID, ASubFolderID: Integer;
  129. AFolderName, ASubFolderName: string);
  130. // ANewBidName: 项目的最新标段名(取自服务器,有人改名了,本地的就变成旧的)
  131. procedure CheckBidName(AID: Integer; ANewBidName: string); overload;
  132. procedure CheckBidName(AUserID, AWebID: Integer; ANewBidName: string); overload;
  133. procedure ClearLocalValues;
  134. procedure GetLocalValues(ARec: TsdDataRecord); overload;
  135. // 用户ID、网络标段ID、Type=1可以定位一个标段。
  136. procedure GetLocalValues(AUserID, AWebID: Integer); overload;
  137. // 1 等待我审核的标段文件; 2 我参与的全部标段文件
  138. procedure DoBatchReceiveOnline(ARequestType: Integer);
  139. function LocalMD5(AUserID, AWebID: Integer): string;
  140. procedure BubbleSortProjects;
  141. // AReceiveKind: 1 接收; 2 导入
  142. function FileDownAndReceive(ADownURL: string; AReceiveKind: Integer): Boolean;
  143. // 线上审批的起止人
  144. procedure OnLineChecker(AAr: TOVArr; var ABegin, AEnd: Integer; var AOnLineEndIsOwner: Boolean);
  145. public
  146. constructor Create(AOwner: TComponent); override;
  147. destructor Destroy; override;
  148. procedure DoBatchReceiveAllOnline;
  149. // AType: -2 繁忙; -1 正常读取; 0 第0期; 1 第1期。
  150. procedure ShowProjectInfoTop(AType: Integer = -1);
  151. function Rec(AProjectID: Integer): TsdDataRecord;
  152. function CurRec: TsdDataRecord;
  153. function CurRecAttachmentPath: string;
  154. function AttachmentFileCountsWithoutManageFile(ANode: TsdIDTreeNode): Integer;
  155. property ProjectCheckStatus: TCheckStatus read FWebCheckStatusProject;
  156. end;
  157. implementation
  158. uses
  159. MainFrm, UtilMethods, ProjectCommands, Globals, ConfigDoc, ConstUnit,
  160. WebNewTenderFrm, PHPWebDm, Math, mProgressFrm, ProgressHintFrm,
  161. ShellAPI, ProjectFme;
  162. {$R *.dfm}
  163. procedure TProjectManagerFrame.ConnectButtonWithAction;
  164. begin
  165. SetDxBtnAction(actnNewProject, MainForm.dxbtnNewProject);
  166. SetDxBtnAction(actnNewSubProject, MainForm.dxbtnNewSubProject);
  167. SetDxBtnAction(actnNewTender, MainForm.dxbtnNewTender);
  168. SetDxBtnAction(actnOpen, MainForm.dxbtnOpenProject);
  169. SetDxBtnAction(actnDelete, MainForm.dxbtnDeleteProject);
  170. SetDxBtnAction(actnReceiveProject, MainForm.dxbtnReceiveProject);
  171. SetDxBtnAction(actnOpenBackupFolder, MainForm.dxbtnOpenBackupFolder);
  172. SetDxBtnAction(actnRename, MainForm.dxbtnRename);
  173. end;
  174. constructor TProjectManagerFrame.Create(AOwner: TComponent);
  175. begin
  176. inherited;
  177. FCheckerList := TObjectList.Create;
  178. FProjectManagerData := ProjectManager;
  179. FProjectManagerData.Open;
  180. stdProjects.IDTree := FProjectManagerData.ProjectsTree;
  181. sdTenderProperty.DataView := FProjectManagerData.sdvTenderProperty;
  182. ConnectButtonWithAction;
  183. SetPropertyVisible(False);
  184. sbChecker.Height := 0;
  185. if G_IsCloud then
  186. begin
  187. Application.HintPause := 200;
  188. Application.HintHidePause := 60000;
  189. tobtnImport.Visible := False;
  190. stdProjects.TreeOptions := stdProjects.TreeOptions - [aoAllowUpLevel, aoAllowDownLevel];
  191. stdProjects.Options := stdProjects.Options - [aoAllowUpMove, aoAllowDownMove];
  192. CreateProgress('正在从云端下载新项目');
  193. try
  194. actnReceiveProject.Execute;
  195. finally
  196. CloseProgress;
  197. end;
  198. end;
  199. end;
  200. // 双击打开项目 TagB
  201. procedure TProjectManagerFrame.actnOpenExecute(Sender: TObject);
  202. var
  203. vSel: TsdIDTreeNode;
  204. vRec: TsdDataRecord;
  205. sHint: string;
  206. function SearchFileOnline(AURL: string; var ADownURL, AFolder, ASubFolder, ABidName, AMD5Web, AError: string; var AFolderID, ASubFolderID: Integer): Integer;
  207. var vArr: TOVArr;
  208. begin
  209. Result := PHPWeb.Search(AURL, [''], [''], vArr);
  210. AError := '';
  211. if Result = 1 then
  212. begin
  213. if High(vArr) >= 0 then
  214. begin
  215. ADownURL := vArr[0, 0];
  216. AFolder := vArr[0, 2];
  217. ASubFolder := vArr[0, 3];
  218. AMD5Web := vArr[0, 1];
  219. AFolderID := StrToInt(vArr[0, 4]);
  220. ASubFolderID := StrToInt(vArr[0, 5]);
  221. ABidName := vArr[0, 6];
  222. end
  223. else
  224. Result := 10; // 返回10,表示无记录。用这个数字代表是否觉得怪异?没办法,0被占用了。
  225. end
  226. else if Result = 0 then
  227. AError := PHPWeb.PageError('标段更新数据失败')
  228. else if Result = -1 then
  229. AError := PHPWeb.NetError('标段更新数据失败');
  230. end;
  231. function CanOpen: Boolean;
  232. var
  233. sSearchURL, sDownURL,
  234. sMD5_UnLock, sError, sLocalFile: string;
  235. iSearch, iFolderID, iSubFolderID: Integer;
  236. bLock, bCanImp: Boolean;
  237. vFileCheck: TTenderFileChecker;
  238. begin
  239. Result := False;
  240. // 先按正常接口找到最新的MD5码看是否需要更新
  241. sSearchURL := Format('%stender/get/%d/update', [PHPWeb.MeasureURL, FWebID]);
  242. iSearch := SearchFileOnline(sSearchURL, sDownURL, FWebFolder_OnLine, FWebSubFolder_OnLine, FWebBidName_OnLine, FWebMD5_OnLine, sError, iFolderID, iSubFolderID);
  243. if iSearch = 1 then
  244. begin
  245. try
  246. CheckWebFolders(iFolderID, iSubFolderID, FWebFolder_OnLine, FWebSubFolder_OnLine);
  247. CheckBidName(FID, FWebBidName_OnLine);
  248. finally
  249. if vSel <> nil then
  250. vSel.LocateInControl;
  251. end;
  252. // 打开前一定要先下载最新的标段文件(无论审核有没有通过)
  253. if FWebMD5_OnLine <> FWebMD5_Local then
  254. if not FileDownAndReceive(sDownURL, 1) then Exit;
  255. // 编制人且项目末通过
  256. if (FWebAuthorID = PHPWeb.UserID) and (FWebCheckStatusProject = csNotPass) then
  257. begin
  258. sHint := '本期计量审批不通过,你现在可以:' + #10#13 + '点击【是(Y)】重新开始本期计量,软件将打开本期上报时的数据,开始重新计量;' +
  259. #10#13 +'点击【否(N)】查看不通过计量,软件将打开本期最后审批的数据,重新打开标段' + '可再次打开本确认窗口。';
  260. if Application.MessageBox(PChar(sHint), '询问', MB_YESNO + MB_ICONQUESTION) = ID_Yes then
  261. begin
  262. // 无锁文件机制取消。直接从原位置再下载一遍,这次不插表。
  263. if FileDownAndReceive(sDownURL, 2) then
  264. begin
  265. // 这里主要是为了线上创建新一期审批人列表、更改标段状态。不再使用同步更新等。
  266. sSearchURL := Format('%suser/create/%d/%d/new/audit', [PHPWeb.MeasureURL, FWebID, FPhaseNo]);
  267. case SearchFileOnline(sSearchURL, sDownURL, FWebFolder_OnLine, FWebSubFolder_OnLine, FWebBidName_OnLine, sMD5_UnLock, sError, iFolderID, iSubFolderID) of
  268. 1:
  269. begin
  270. sHint := '【十分重要】:本期计量已重新开始,原报在本次上报完成前,请勿删除该标段或更换电脑。';
  271. Application.MessageBox(PChar(sHint), '提示', MB_OK + MB_ICONINFORMATION);
  272. end;
  273. 0, -1:
  274. begin
  275. sHint := sError + '(因网络出错,无法连接到云端以获取本期原报上传的无锁文件,无法重新开始本期,本次操作已取消。请重试。';
  276. Application.MessageBox(PChar(sHint), '警告', MB_OK + MB_ICONWARNING);
  277. Exit;
  278. end;
  279. end;
  280. end
  281. else
  282. Exit;
  283. end;
  284. end;
  285. end
  286. else if (iSearch = 10) then
  287. begin
  288. if not PHPWeb.ExistInServer(FWebID) then
  289. begin
  290. sHint :='该项目[' + FWebBidName_Local + ']在云端已被删除,点击"确定"后,可手动删除该项目。';
  291. Application.MessageBox(PChar(sHint), '提示', MB_OK + MB_ICONINFORMATION);
  292. Exit;
  293. end;
  294. end
  295. else if (iSearch = 0) or (iSearch = -1) then
  296. begin
  297. sHint := sError + '(因网络出错,无法检测[' + FWebBidName_Local + ']在云端是否有更新,本次操作已取消,请重试)。';
  298. Application.MessageBox(PChar(sHint), '警告', MB_OK + MB_ICONWARNING);
  299. Exit;
  300. end;
  301. Result := True;
  302. end;
  303. var
  304. CurProjectFme: TProjectFrame;
  305. begin
  306. // 打开前先下载更新
  307. Screen.Cursor := crHourGlass;
  308. try
  309. vSel := stdProjects.IDTree.Selected;
  310. vRec := vSel.Rec;
  311. if G_IsCloud then
  312. begin
  313. GetLocalValues(CurRec);
  314. // 以下这段已经包含在MainForm.OpenProject(vRec)中了,但在调用这句之前,网络版要提前用一下。
  315. if MainForm.HasOpened(FID) then
  316. begin
  317. MainForm.LocateProject(FID);
  318. Exit;
  319. end;
  320. if not CanOpen then
  321. Exit;
  322. end;
  323. CurProjectFme := MainForm.OpenProject(vRec);
  324. if G_IsCloud then
  325. begin
  326. // 云版,如果用户在打开项目的过程中,切换界面回项目管理,MainFrom.CurProjectFrame值为nil,后面的调用,包括MainForm.actnCloseProject.Execute都会出错
  327. // 其中由于前面调用的MainForm.LocateProject触发的一系列事件中,调用到了SetPropertyVisible,其中执行了Screen.Cursor := crDefault
  328. // 导致鼠标状态还原,用户可以点击,单机版下测试,属性状态为crHourGlass时,亦可以切换会项目管理
  329. // 按逻辑,CheckFileAndCloudCheckerList这种方法不应在界面,应在控制器,亦不可直接调用MainForm.actnCloseProject.Execute
  330. // 怕按逻辑改动引起更多Bug,故继续错下去,检查MainForm.CurProjectFrame是否正确,并再次禁用鼠标
  331. if CurProjectFme <> MainForm.CurProjectFrame then
  332. begin
  333. Screen.Cursor := crHourGlass;
  334. MainForm.LocateProject(CurProjectFme.ProjectData.ProjectID);
  335. end;
  336. if not MainForm.CurProjectFrame.CheckFileAndCloudChekerList then
  337. begin
  338. sHint := '项目校验:“' + FWebBidName_OnLine + '”文件中的审批人和云端的审批人不一致,' +
  339. '禁止继续操作,项目即将关闭!请删除本地项目重新从云端获取,' +
  340. '重新获取后如果仍然存在同样的问题,请联系纵横服务人员以寻求帮助。';
  341. Application.MessageBox(PChar(sHint), '文件错误', MB_OK +MB_ICONWARNING);
  342. MainForm.actnCloseProject.Execute;
  343. Exit;
  344. end;
  345. end;
  346. finally
  347. Screen.Cursor := crDefault;
  348. end;
  349. end;
  350. procedure TProjectManagerFrame.actnDeleteExecute(Sender: TObject);
  351. begin
  352. if stdProjects.IDTree.Count = 0 then Exit;
  353. with stdProjects.IDTree.Selected.Rec do
  354. if QuestMessage(Format('确定要删除[%s]吗?', [ValueByName('Name').AsString])) then
  355. begin
  356. Screen.Cursor := crHourGlass;
  357. try
  358. FProjectManagerData.Delete;
  359. finally
  360. Screen.Cursor := crDefault;
  361. end;
  362. end;
  363. end;
  364. procedure TProjectManagerFrame.zgProjectsMouseDown(Sender: TObject;
  365. Button: TMouseButton; Shift: TShiftState; X, Y: Integer);
  366. begin
  367. if Button = mbRight then
  368. dxpmProjectManager.PopupFromCursorPos
  369. else if (zgProjects.CurCol = 1) and (Button = mbLeft) and (ssDouble in Shift)
  370. and Assigned(stdProjects.IDTree.Selected) then
  371. begin
  372. if IsProject(stdProjects.IDTree.Selected) then
  373. stdProjects.IDTree.Selected.Expand
  374. else
  375. actnOpen.Execute;
  376. end;
  377. end;
  378. function TProjectManagerFrame.ReceiveFile(const AFileName: string; AIsReback: Boolean): Boolean;
  379. var
  380. vRP: TReceiveProject;
  381. vNode: TsdIDTreeNode;
  382. begin
  383. Result := False;
  384. vRP := TReceiveProject.Create(stdProjects.IDTree.Selected);
  385. try
  386. ProjectManager.RefreshSeedID;
  387. if G_IsCloud then
  388. begin
  389. vRP.IsReback := AIsReback;
  390. vRP.Lock := (FWebAuthorID = PHPWeb.UserID) or
  391. ((FWebAuthorID <> PHPWeb.UserID) and (FWebCheckStatusMy <> csChecking));
  392. vNode := vRP.ReceiveForLost(AFileName, FOnLineCheckerBegin, FOnLineCheckerEnd, FOnLineCheckerEndIsOwner);
  393. if vNode <> nil then
  394. begin
  395. vNode.Rec.BeginUpdate;
  396. vNode.Rec.ValueByName('WebMD5').AsString := FWebMD5_OnLine;
  397. vNode.Rec.ValueByName('WebUserID').AsInteger := PHPWeb.UserID;
  398. vNode.Rec.EndUpdate;
  399. end;
  400. end
  401. else
  402. vNode := vRP.Receive(AFileName);
  403. finally
  404. case vRP.MessageID of
  405. 0: begin
  406. Result := True;
  407. vNode.LocateInControl;
  408. end;
  409. 1: ErrorMessage('当前标段处于打开状态,未能成功接收,请先关闭标段再次接收。');
  410. 2: ErrorMessage('下载数据与审批状态不一致,未能成功接收,请再次接收。');
  411. 3: ErrorMessage('升级数据失败,未能成功接收,请再次接收。');
  412. end;
  413. vRP.Free;
  414. FProjectManagerData.Save;
  415. end;
  416. end;
  417. // 登录后自动扫描等待我审核的项目 TagC
  418. procedure TProjectManagerFrame.actnReceiveProjectExecute(Sender: TObject);
  419. procedure DoReceiveLocal;
  420. var
  421. sFileName: string;
  422. begin
  423. if SelectFile(sFileName, '.rmf;*.arf') then
  424. begin
  425. ShowProgressHint('正在接收项目并升级数据');
  426. try
  427. ReceiveFile(sFileName);
  428. finally
  429. CloseProgressHint;
  430. end;
  431. end;
  432. end;
  433. var OnCC: TZjCellNotifyEvent;
  434. begin
  435. Screen.Cursor := crHourGlass;
  436. try
  437. if G_IsCloud then
  438. begin
  439. OnCC := zgProjects.OnCurrentChanged;
  440. try
  441. zgProjects.OnCurrentChanged := nil;
  442. DoBatchReceiveOnline(1);
  443. if stdProjects.IDTree.FirstNode <> nil then
  444. stdProjects.IDTree.FirstNode.LocateInControl;
  445. finally
  446. zgProjects.OnCurrentChanged := OnCC;
  447. end;
  448. end
  449. else
  450. DoReceiveLocal;
  451. finally
  452. Screen.Cursor := crDefault;
  453. end;
  454. end;
  455. function TProjectManagerFrame.GetImportProjectName(
  456. const AFileName: string; AParent: TsdIDTreeNode): string;
  457. begin
  458. Result := ExtractSimpleFileName(AFileName);
  459. while FProjectManagerData.ExistProject(Result, AParent) do
  460. if not InputNewProjectName(Result, '导入', AParent) then Abort;
  461. end;
  462. procedure TProjectManagerFrame.actnNewProjectExecute(Sender: TObject);
  463. var
  464. sName: string;
  465. begin
  466. if G_IsCloud then Exit; // 云版线上与本地要保持同步,不允许本地新建
  467. if InputNewProjectName(sName, '新建', stdProjects.IDTree.Selected) then
  468. FProjectManagerData.InsertProject(sName, stdProjects.IDTree.Selected);
  469. end;
  470. procedure TProjectManagerFrame.actnNewSubProjectExecute(Sender: TObject);
  471. var
  472. sName: string;
  473. begin
  474. if G_IsCloud then Exit;
  475. if InputNewProjectName(sName, '新建', stdProjects.IDTree.Selected) then
  476. FProjectManagerData.InsertSubProject(sName, stdProjects.IDTree.Selected);
  477. end;
  478. procedure TProjectManagerFrame.actnNewTenderExecute(Sender: TObject);
  479. function AddAndOpenTender(const ATenderName: string): TsdIDTreeNode;
  480. begin
  481. Result := FProjectManagerData.InsertTender(ATenderName, stdProjects.IDTree.Selected);
  482. MainForm.OpenProject(Result.Rec);
  483. end;
  484. // 网络版新建标段 TagD
  485. procedure NewProjectWithOnline;
  486. var
  487. WebNewTenderForm: TWebNewTenderForm;
  488. sName, sKey, sURL: string;
  489. stnNew: TsdIDTreeNode;
  490. iID, iFolderID, iSubFolderID: Integer;
  491. vRec: TsdDataRecord;
  492. vArr: array of string;
  493. begin
  494. WebNewTenderForm := TWebNewTenderForm.Create(nil);
  495. try
  496. WebNewTenderForm.ShowModal;
  497. if WebNewTenderForm.ModalResult = mrOK then
  498. begin
  499. sKey := WebNewTenderForm.edtKey.Text;
  500. sName := WebNewTenderForm.edtTenderName.Text;
  501. // 同服务器取得联系
  502. iID := -1;
  503. vArr := VarArrayOf(['catid', 'name', 'company', 'phone', 'mobile', 'qq',
  504. 'pname', 'ptype', 'jobs', 'avatar', 'ownuid', 'pnameid', 'ptypeid']);
  505. sURL := Format('%s%d/%s/%s/creatmeasure', [PHPWeb.MeasureURL, PHPWeb.UserID, sName, sKey]); // AnsiToUtf8(sName)
  506. case PHPWeb.Search(sURL, [], [], vArr) of
  507. 1:
  508. begin
  509. iID := StrToInt(vArr[0]);
  510. iFolderID := StrToInt(vArr[11]);
  511. iSubFolderID := StrToInt(vArr[12]);
  512. CheckWebFolders(iFolderID, iSubFolderID, vArr[6], vArr[7]);
  513. end;
  514. 0:
  515. begin
  516. Application.MessageBox(PChar(PHPWeb.PageError('创建标段失败' + '[' + vArr[0] + ']')),
  517. '警告', MB_OK + MB_ICONWARNING);
  518. Exit;
  519. end;
  520. -1:
  521. begin
  522. Application.MessageBox(PChar(PHPWeb.NetError('创建标段失败')),
  523. '警告', MB_OK + MB_ICONWARNING);
  524. Exit;
  525. end;
  526. end;
  527. // 本地创建
  528. stnNew := FProjectManagerData.InsertTender(sName, stdProjects.IDTree.Selected);
  529. // 这里把Web获取的信息存储到项目管理里面。
  530. vRec := stnNew.Rec;
  531. vRec.BeginUpdate;
  532. vRec.ValueByName('WebID').AsInteger := iID;
  533. vRec.ValueByName('WebOwnerID').AsInteger := StrToInt(vArr[10]); // 业主
  534. vRec.ValueByName('WebAuthorID').AsInteger := PHPWeb.UserID; // 编制人
  535. vRec.ValueByName('WebUserID').AsInteger := PHPWeb.UserID; // 当前用户,用于只显示自己的项目
  536. vRec.ValueByName('WebKey').AsString := sKey;
  537. vRec.EndUpdate;
  538. GetLocalValues(vRec);
  539. FWebOwnerName := vArr[1];
  540. FWebOwnerCompany := vArr[2];
  541. FWebOwnerRole := vArr[8];
  542. FWebFolder_OnLine := vArr[6];
  543. FWebSubFolder_OnLine := vArr[7];
  544. // WebOwnerImage := vArr[9];
  545. // WebOwnerPhone := vArr[3];
  546. // WebOwnerMobile := vArr[4];
  547. // WebOwnerQQ := vArr[5];
  548. ShowProjectInfoTop(0);
  549. FProjectManagerData.Save;
  550. MainForm.OpenProject(vRec);
  551. end;
  552. finally
  553. WebNewTenderForm.Free;
  554. end;
  555. end;
  556. procedure NewProject;
  557. var
  558. sName: string;
  559. begin
  560. if InputNewProjectName(sName, '新建', stdProjects.IDTree.Selected) then
  561. AddAndOpenTender(sName);
  562. end;
  563. begin
  564. if G_IsCloud then
  565. NewProjectWithOnline
  566. else
  567. NewProject;
  568. end;
  569. procedure TProjectManagerFrame.zgProjectsDrawCellText(ACanvas: TCanvas;
  570. const ARect: TRect; const ACoord: TPoint; AGrid: TZJGrid;
  571. const Text: String; var ADefaultDraw: Boolean);
  572. procedure GetBitmap(AImage: TBitmap; ANode: TsdIDTreeNode);
  573. begin
  574. if Assigned(ANode) and Assigned(ANode.Rec) then
  575. begin
  576. if ANode.Rec.ValueByName('Type').AsInteger = 0 then
  577. if ANode.Expanded and ANode.HasChildren then
  578. MainForm.Images.GetBitmap(34, AImage)
  579. else
  580. MainForm.Images.GetBitmap(34, AImage)
  581. else
  582. MainForm.Images.GetBitmap(11, AImage);
  583. end
  584. else
  585. AImage := nil;
  586. end;
  587. const
  588. rIconWidth = 16;
  589. rIconHeight = 16;
  590. var
  591. Img: TBitmap;
  592. Cell: TZjCell;
  593. rImg: TRect;
  594. vNode: TsdIDTreeNode;
  595. begin
  596. if (ACoord.X = 1) and (ACoord.Y > zgProjects.FixedRowCount - 1) then
  597. begin
  598. Cell := zgProjects.Cells[ACoord.X, ACoord.Y];
  599. Img := TBitmap.Create;
  600. try
  601. vNode := stdProjects.IDTree.Items[ACoord.Y-zgProjects.FixedRowCount];
  602. GetBitmap(Img, vNode);
  603. case Cell.Align of
  604. gaTopLeft, gaTopCenter, gaTopRight:
  605. rImg := Rect(ARect.Left + 2, ARect.Top, ARect.Left + rIconWidth, ARect.Top + rIconHeight);
  606. gaCenterLeft, gaCenterCenter, gaCenterRight:
  607. rImg := Rect(ARect.Left + 2, ARect.Top + (ARect.Bottom - ARect.Top - rIconHeight) div 2, ARect.Left + rIconWidth, ARect.Bottom - (ARect.Bottom - ARect.Top - rIconHeight) div 2);
  608. gaBottomLeft, gaBottomCenter, gaBottomRight:
  609. rImg := Rect(ARect.Left + 2, ARect.Bottom - rIconHeight, ARect.Left + rIconWidth, ARect.Bottom);
  610. end;
  611. ACanvas.StretchDraw(rImg, Img);
  612. WriteText(ACanvas, Rect(ARect.Left + rIconWidth, ARect.Top, ARect.Right, ARect.Bottom)
  613. , 2, 2, Text, Cell.Align, False);
  614. ADefaultDraw := False;
  615. finally
  616. Img.Free;
  617. end;
  618. end;
  619. end;
  620. procedure TProjectManagerFrame.actnNewSubProjectUpdate(Sender: TObject);
  621. begin
  622. TAction(Sender).Enabled := Assigned(stdProjects.IDTree.Selected)
  623. and IsProject(stdProjects.IDTree.Selected)
  624. and (not IsUnEmptyLeafProject(stdProjects.IDTree.Selected));
  625. end;
  626. procedure TProjectManagerFrame.actnNewTenderUpdate(Sender: TObject);
  627. var bCloud: Boolean;
  628. vNode: TsdIDTreeNode;
  629. begin
  630. bCloud := G_IsCloud;
  631. // 只有编制人才能创建新标段?逻辑先后有问题:编制人是在创建标段之后产生的。
  632. // 创建前,当前用户只是一个帐户,它是不是编制人末知,因为它还可以是其它角色。
  633. // 同一帐户在不同的标段可以作为不同的角色。
  634. if bCloud then
  635. begin
  636. TAction(Sender).Enabled := True;
  637. end
  638. else
  639. begin
  640. vNode := stdProjects.IDTree.Selected;
  641. TAction(Sender).Enabled := Assigned(vNode) and IsProject(vNode) and IsLeafProject(vNode);
  642. end;
  643. end;
  644. function TProjectManagerFrame.IsLeafProject(ANode: TsdIDTreeNode): Boolean;
  645. begin
  646. if ANode.HasChildren then
  647. Result := ANode.FirstChild.Rec.ValueByName('Type').AsInteger = 1
  648. else
  649. Result := ANode.Rec.ValueByName('Type').AsInteger = 0;
  650. end;
  651. function TProjectManagerFrame.IsProject(ANode: TsdIDTreeNode): Boolean;
  652. begin
  653. Result := ANode.Rec.ValueByName('Type').AsInteger = 0;
  654. end;
  655. function TProjectManagerFrame.IsUnEmptyLeafProject(
  656. ANode: TsdIDTreeNode): Boolean;
  657. begin
  658. Result := ANode.HasChildren and (ANode.Rec.ValueByName('Type').AsInteger = 1);
  659. end;
  660. procedure TProjectManagerFrame.SetPropertyVisible(AVisible: Boolean);
  661. begin
  662. // 单击刷新项目信息 TagA
  663. if G_IsCloud then
  664. begin
  665. if CurRec = nil then Exit;
  666. pnlTenderProperty.Visible := False;
  667. pnlWeb.Visible := AVisible;
  668. if AVisible then
  669. begin
  670. CreateProgress('云端读取项目信息');
  671. try
  672. ShowProjectInfoTopAndCheckers;
  673. finally
  674. CloseProgress;
  675. end;
  676. end;
  677. end
  678. else
  679. begin
  680. pnlWeb.Visible := False;
  681. pnlTenderProperty.Visible := AVisible;
  682. sprProperty.Visible := AVisible;
  683. end;
  684. end;
  685. procedure TProjectManagerFrame.zgProjectsCurrentChanged(Sender: TObject;
  686. Col, Row: Integer);
  687. begin
  688. if G_IsCloud then
  689. begin
  690. if CurRec <> nil then
  691. begin
  692. // 加这句后产生Bug:上报项目后,记录不曾移动,FID不变,不会刷新
  693. // if FID <> CurRec.ValueByName('ID').AsInteger then
  694. SetPropertyVisible(CurRec.ValueByName('Type').AsInteger = 1);
  695. // OnCurrentChanged取得的 CurRec.ValueByName() 值并不总是可靠,这里加保险。
  696. // 如调用locateInControl后,执行到这里取得的CurRec.ValueByName('ID')值还是上一条的。
  697. if (CurRec.ValueByName('Type').AsInteger = 1) and (not pnlWeb.Visible) then
  698. pnlWeb.Visible := True;
  699. end;
  700. end;
  701. end;
  702. procedure TProjectManagerFrame.actnSubmitProjectUpdate(Sender: TObject);
  703. begin
  704. TAction(Sender).Enabled := Assigned(stdProjects.IDTree.Selected)
  705. and (stdProjects.IDTree.Selected.Rec.ValueByName('Type').AsInteger = 1)
  706. and (stdProjects.IDTree.Selected.Rec.ValueByName('AuditStatus').AsInteger < iMaxStageCount-1);
  707. end;
  708. procedure TProjectManagerFrame.actnReplyProjectUpdate(Sender: TObject);
  709. begin
  710. TAction(Sender).Enabled := Assigned(stdProjects.IDTree.Selected)
  711. and (stdProjects.IDTree.Selected.Rec.ValueByName('Type').AsInteger = 1);
  712. end;
  713. procedure TProjectManagerFrame.actnRenameExecute(Sender: TObject);
  714. function CanRename(ARec: TsdDataRecord; const ANewName: string): Boolean;
  715. var
  716. sURL: string;
  717. iRename: Integer;
  718. vArr: array [0..0] of string;
  719. begin
  720. Result := True;
  721. if not G_IsCloud then Exit;
  722. // 云版 重命名须确保同步服务器
  723. sURL := Format('%stender/%d/%s/update', [PHPWeb.MeasureURL, ARec.ValueByName('WebID').AsInteger, ANewName]);
  724. iRename := PHPWeb.Search(sURL, [], [], vArr);
  725. Result := iRename = 1;
  726. case iRename of
  727. 1: ShowMessage('新的标段名称已同步到服务器!');
  728. 0: Application.MessageBox(PChar(PHPWeb.PageError('重命名同步到云端失败' + '[' + vArr[0] + ']')),
  729. '警告', MB_OK + MB_ICONWARNING);
  730. -1: Application.MessageBox(PChar(PHPWeb.NetError('重命名同步到云端失败')),
  731. '警告', MB_OK + MB_ICONWARNING);
  732. end;
  733. end;
  734. var
  735. stnNode: TsdIDTreeNode;
  736. sName: string;
  737. begin
  738. stnNode := stdProjects.IDTree.Selected;
  739. sName := stnNode.Rec.ValueByName('Name').AsString;
  740. if not Assigned(OpenProjectManager.FindProjectData(stnNode.ID)) then
  741. begin
  742. if InputNewProjectName(sName, '重命名', stnNode.Parent, stnNode.ID) then
  743. begin
  744. if (sName <> stnNode.Rec.ValueByName('Name').AsString) and CanRename(stnNode.Rec, sName) then
  745. begin
  746. stnNode.Rec.ValueByName('Name').AsString := sName;
  747. ProjectManager.Save;
  748. end;
  749. end;
  750. end
  751. else
  752. ErrorMessage(Format('项目[%s]已经打开,无法重命名!', [sName]));
  753. end;
  754. procedure TProjectManagerFrame.actnOpenUpdate(Sender: TObject);
  755. begin
  756. TAction(Sender).Enabled := Assigned(stdProjects.IDTree.Selected)
  757. and (stdProjects.IDTree.Selected.Rec.ValueByName('Type').AsInteger = 1);
  758. end;
  759. procedure TProjectManagerFrame.actnImportExecute(Sender: TObject);
  760. procedure ImportTender(const AFileName, AProjectName: string);
  761. var
  762. Importor: TTenderImport;
  763. begin
  764. Importor := TTenderImport.Create(stdProjects.IDTree.Selected,
  765. AProjectName, AFileName);
  766. try
  767. Importor.Execute;
  768. finally
  769. Importor.Free;
  770. end;
  771. end;
  772. procedure ImportProject(const AFileName, AProjectName: string);
  773. var
  774. Importor: TProjectImport;
  775. begin
  776. Importor := TProjectImport.Create(stdProjects.IDTree.Selected,
  777. AProjectName, AFileName);
  778. try
  779. Importor.Execute;
  780. finally
  781. Importor.Free;
  782. end;
  783. end;
  784. var
  785. sFileName, sProjectName: string;
  786. vCur: TsdIDTreeNode;
  787. begin
  788. if SelectFile(sFileName, '.mtf;*.mpf') then
  789. begin
  790. vCur := stdProjects.IDTree.Selected;
  791. if Assigned(vCur) then
  792. begin
  793. if SameText(ExtractFileExt(sFileName), '.mtf')
  794. and (vCur.Rec.ValueByName('Type').AsInteger = 0) then
  795. sProjectName := GetImportProjectName(sFileName, stdProjects.IDTree.Selected)
  796. else
  797. sProjectName := GetImportProjectName(sFileName, stdProjects.IDTree.Selected.Parent);
  798. end
  799. else
  800. sProjectName := GetImportProjectName(sFileName, vCur);
  801. Screen.Cursor := crHourGlass;
  802. try
  803. if SameText(ExtractFileExt(sFileName), '.mtf') then
  804. ImportTender(sFileName, sProjectName)
  805. else
  806. ImportProject(sFileName, sProjectName);
  807. finally
  808. Screen.Cursor := crDefault;
  809. end;
  810. end;
  811. FProjectManagerData.Save;
  812. end;
  813. procedure TProjectManagerFrame.actnExportExecute(Sender: TObject);
  814. procedure ExportTender(ANode: TsdIDTreeNode);
  815. var
  816. Exportor: TTenderExport;
  817. sFileName, sHint: string;
  818. bExpAtch: Boolean;
  819. iCount: Integer;
  820. begin
  821. bExpAtch := False;
  822. sFileName := SupportManager.ConfigInfo.OutputPath + ANode.Rec.ValueByName('Name').AsString + '.mtf';
  823. if SaveFile(sFileName, '.mtf') then
  824. begin
  825. if FileExists(sFileName) and not QuestMessage(Format('存在同名文件“%s”,是否替换?', [ExtractFileName(sFileName)])) then
  826. Exit;
  827. Screen.Cursor := crHourGlass;
  828. try
  829. Exportor := TTenderExport.Create(ANode.Rec, sFileName);
  830. try
  831. { if not G_IsCloud then
  832. begin
  833. iCount := FileCount(CurRecAttachmentPath);
  834. if iCount > 1 then // 排除管理文件库
  835. begin
  836. sHint := Format('本标段包含 %d 个附件,是否将附件一起导出?', [iCount - 1]);
  837. bExpAtch := Application.MessageBox(PChar(sHint), '询问', MB_YESNO + MB_ICONQUESTION + MB_DEFBUTTON2) = ID_Yes;
  838. end;
  839. end; } // FFFFF
  840. Exportor.Execute(bExpAtch);
  841. finally
  842. Exportor.Free;
  843. end;
  844. finally
  845. Screen.Cursor := crDefault;
  846. end;
  847. end;
  848. end;
  849. procedure ExportProject(ANode: TsdIDTreeNode);
  850. var
  851. Exportor: TProjectExport;
  852. sFileName, sHint: string;
  853. bExpAtch: Boolean;
  854. iCount: Integer;
  855. begin
  856. sFileName := SupportManager.ConfigInfo.OutputPath + ANode.Rec.ValueByName('Name').AsString + '.mpf';
  857. if SaveFile(sFileName, '.mpf') then
  858. begin
  859. if FileExists(sFileName) and not QuestMessage(Format('存在同名文件“%s”,是否替换?', [ExtractFileName(sFileName)])) then
  860. Exit;
  861. Screen.Cursor := crHourGlass;
  862. try
  863. bExpAtch := False;
  864. { if not G_IsCloud then
  865. begin
  866. iCount := AttachmentFileCountsWithoutManageFile(ANode);
  867. if iCount > 0 then
  868. begin
  869. sHint := Format('本建设项目共包含 %d 个附件,是否将附件一起导出?', [iCount]);
  870. bExpAtch := Application.MessageBox(PChar(sHint), '询问', MB_YESNO + MB_ICONQUESTION + MB_DEFBUTTON2) = ID_Yes;
  871. end;
  872. end; } // FFFFF
  873. Exportor := TProjectExport.Create(ANode, sFileName, bExpAtch);
  874. try
  875. Exportor.Execute;
  876. finally
  877. Exportor.Free;
  878. end;
  879. finally
  880. Screen.Cursor := crDefault;
  881. end;
  882. end;
  883. end;
  884. var
  885. stnNode: TsdIDTreeNode;
  886. begin
  887. stnNode := stdProjects.IDTree.Selected;
  888. if stnNode.Rec.ValueByName('Type').AsInteger = 1 then
  889. begin
  890. ExportTender(stnNode);
  891. end
  892. else
  893. ExportProject(stnNode);
  894. end;
  895. procedure TProjectManagerFrame.actnDeleteUpdate(Sender: TObject);
  896. begin
  897. if Assigned(stdProjects.IDTree.Selected) then
  898. TAction(Sender).Enabled := not CheckOpened(stdProjects.IDTree.Selected)
  899. else
  900. TAction(Sender).Enabled := False;
  901. end;
  902. function TProjectManagerFrame.CheckOpened(ANode: TsdIDTreeNode): Boolean;
  903. var
  904. iChild: Integer;
  905. begin
  906. Result := False;
  907. if ANode.Rec.ValueByName('Type').AsInteger = 1 then
  908. Result := OpenProjectManager.ProjectIndex(ANode.ID) > -1
  909. else
  910. begin
  911. if not ANode.HasChildren then
  912. Result := False
  913. else
  914. begin
  915. for iChild := 0 to ANode.ChildCount - 1 do
  916. Result := Result or CheckOpened(ANode.ChildNodes[iChild]);
  917. end
  918. end;
  919. end;
  920. procedure TProjectManagerFrame.ShowProjectInfoTopAndCheckers;
  921. var
  922. vPSArr: TStrArr;
  923. vCArr: TOVArr; // Checkers
  924. vChecker: TOrderCheckerFrame;
  925. sPicPath, sURL: string;
  926. procedure AddChecker(AType: TCheckerFrameType; AArr: array of string);
  927. begin
  928. vChecker := TOrderCheckerFrame.Create(self);
  929. FCheckerList.Add(vChecker);
  930. vChecker.Owner := Self;
  931. sbChecker.VertScrollBar.Range := sbChecker.VertScrollBar.Range + vChecker.Height;
  932. sbChecker.Height := Min(sbChecker.Height + vChecker.Height, pnlWeb.Height - pnlProject.Height);
  933. vChecker.Parent := sbChecker;
  934. vChecker.Top := FCurPos;
  935. FCurPos := FCurPos + vChecker.Height;
  936. vChecker.Align := alTop;
  937. sPicPath := PHPWeb.UserPath + '1_' + AArr[0] + '.jpg';
  938. PHPWeb.DownFile(AArr[4], sPicPath);
  939. vChecker.Init(AType, StrToInt(AArr[0]), AArr[1], AArr[3],
  940. AArr[2], sPicPath, AArr[6], TCheckStatus(StrToInt(AArr[5])-1), AArr[8], StrToInt(AArr[7]));
  941. vChecker.Name := 'ProjectOrderFrame' + AArr[0];
  942. end;
  943. procedure RepairOrder;
  944. var k: Integer;
  945. begin
  946. for k := 0 to sbChecker.ControlCount - 1 do
  947. TOrderCheckerFrame(sbChecker.Controls[k]).Order := k + 1;
  948. end;
  949. procedure ShowProjectCheckers;
  950. var i, j, n: Integer;
  951. vOwner: array of string; // 业主信息
  952. begin
  953. FCheckerList.Clear;
  954. sbChecker.Height := 0;
  955. FCurPos := 0;
  956. n := Length(vCArr[Low(vCArr)]);
  957. SetLength(vOwner, n);
  958. sbChecker.VertScrollBar.Range := 0;
  959. for i := Low(vCArr) to High(vCArr) do
  960. begin
  961. if StrToInt(vCArr[i, 0]) = PHPWeb.UserID then
  962. FWebCheckStatusMy := TCheckStatus(StrToInt(vCArr[i, 5])-1);
  963. if StrToInt(vCArr[i, 0]) = FWebOwnerID then
  964. begin
  965. for j := 0 to n - 1 do
  966. vOwner[j] := vCArr[i, j];
  967. Continue;
  968. end;
  969. AddChecker(cftChecker, vCArr[i]);
  970. end;
  971. if vOwner[0] <> '' then
  972. AddChecker(cftOwner, vOwner);
  973. RepairOrder;
  974. OnLineChecker(vCArr, FOnLineCheckerBegin, FOnLineCheckerEnd, FOnLineCheckerEndIsOwner);
  975. end;
  976. begin
  977. GetLocalValues(CurRec);
  978. if FWebID = 0 then Exit;
  979. SetLength(vPSArr, 8);
  980. sURL := Format('%smeasure/status/%d/get', [PHPWeb.MeasureURL, FWebID]);
  981. if PHPWeb.Search(sURL, [''], [''], 3, vPSArr, vCArr) = 1 then
  982. begin
  983. LockWindowUpdate(pnlWeb.Handle);
  984. try
  985. FPhaseNo := StrToInt(vPSArr[0]);
  986. FWebCheckStatusProject := TCheckStatus(StrToInt(vPSArr[1])-1);
  987. FPhaseTotal := StrToInt(vPSArr[2]);
  988. FWebFolder_OnLine := vPSArr[3];
  989. FWebSubFolder_OnLine := vPSArr[4];
  990. FWebOwnerName := vPSArr[5];
  991. FWebOwnerCompany := vPSArr[6];
  992. FWebOwnerRole := vPSArr[7];
  993. ShowProjectInfoTop;
  994. ShowProjectCheckers;
  995. finally
  996. LockWindowUpdate(0);
  997. end;
  998. end
  999. else
  1000. begin
  1001. FPhaseNo := 0;
  1002. FWebCheckStatusProject := csNotBegin;
  1003. FPhaseTotal := 0;
  1004. FWebFolder_OnLine := '';
  1005. FWebSubFolder_OnLine := '';
  1006. FWebOwnerName := '';
  1007. FWebOwnerCompany := '';
  1008. FWebOwnerRole := '';
  1009. ShowProjectInfoTop;
  1010. sbChecker.Height := 0;
  1011. end;
  1012. end;
  1013. procedure TProjectManagerFrame.ShowProjectInfoTop(AType: Integer);
  1014. procedure ShowOwner;
  1015. begin
  1016. lblBidName.Caption := FWebBidName_Local;
  1017. lblBidName.Update;
  1018. lblProjName.Caption := FWebFolder_OnLine;
  1019. lblProjName.Update;
  1020. lblWebProjCtgyName.Caption := FWebSubFolder_OnLine;
  1021. lblWebProjCtgyName.Update;
  1022. lblOnwerName.Caption := FWebOwnerName;
  1023. lblOnwerName.Update;
  1024. lblOnwerCompany.Caption := Format('-%s)', [FWebOwnerCompany]);
  1025. lblOnwerCompany.Update;
  1026. lblOnwerCompany.Left := lblOnwerName.Left + lblOnwerName.Width;
  1027. end;
  1028. procedure ShowStatus(ANo: Integer; AState: TCheckStatus);
  1029. begin
  1030. lblPeriod.Caption := Format('第%d期', [ANo]);
  1031. lblPeriod.Update;
  1032. lblPeriodState.Caption := CheckStatusNames[AState];
  1033. lblPeriodState.Font.Color := CheckStatusColors[AState];
  1034. lblPeriodState.Update;
  1035. lblPeriodState.Left := lblPeriod.Left + lblPeriod.Width + 5;
  1036. lblPeriodTotal.Caption := Format('(共%d期)', [ANo]);
  1037. lblPeriodTotal.Update;
  1038. lblPeriodTotal.Left := lblPeriodState.Left + lblPeriodState.Width + 3;
  1039. end;
  1040. begin
  1041. GetLocalValues(CurRec);
  1042. case AType of
  1043. -2:
  1044. begin
  1045. lblPeriod.Caption := '正在从云端读取状态信息...';
  1046. lblPeriod.Update;
  1047. end;
  1048. -1:
  1049. begin
  1050. ShowOwner;
  1051. ShowStatus(FPhaseNo, FWebCheckStatusProject);
  1052. end;
  1053. 0:
  1054. begin
  1055. ShowOwner;
  1056. ShowStatus(0, csNotBegin);
  1057. end;
  1058. end;
  1059. end;
  1060. // 检查后,应该定位到最后一层目录,不应该回到原先的选择节点。否则从网络拉下来的项目无法组织正确的树结构。
  1061. procedure TProjectManagerFrame.CheckWebFolders(AFolderID, ASubFolderID: Integer;
  1062. AFolderName, ASubFolderName: string);
  1063. var
  1064. vTree: TsdIDTree;
  1065. vNode, vSubNode: TsdIDTreeNode;
  1066. i: Integer;
  1067. sName: string;
  1068. iUserID, iWebID, iWebFolderLevel: Integer;
  1069. bExist, bSubExist, bModified: Boolean;
  1070. begin
  1071. bExist := False;
  1072. bSubExist := False;
  1073. bModified := False;
  1074. vTree := stdProjects.IDTree;
  1075. for i := 0 to vTree.Count - 1 do
  1076. begin
  1077. vNode := vTree.Items[i];
  1078. sName := vNode.Rec.ValueByName('Name').AsString;
  1079. iUserID := vNode.Rec.ValueByName('WebUserID').AsInteger;
  1080. iWebID := vNode.Rec.ValueByName('WebID').AsInteger;
  1081. iWebFolderLevel := vNode.Rec.ValueByName('WebFolderLevel').AsInteger;
  1082. if (iWebID = AFolderID) and (iWebFolderLevel = G_WFL_ProjName) and (iUserID = PHPWeb.UserID) then
  1083. begin
  1084. bExist := True;
  1085. vNode.LocateInControl;
  1086. if not SameText(sName, AFolderName) then
  1087. begin
  1088. vNode.Rec.ValueByName('Name').AsString := AFolderName;
  1089. bModified := True;
  1090. end;
  1091. Break;
  1092. end;
  1093. end;
  1094. if not bExist then
  1095. begin
  1096. vNode := vTree.Items[0];
  1097. if Assigned(vNode) then
  1098. vNode.LocateInControl;
  1099. vNode := FProjectManagerData.InsertProject(AFolderName, stdProjects.IDTree.Selected, AFolderID, G_WFL_ProjName);
  1100. vNode.LocateInControl;
  1101. end;
  1102. for i := 0 to vNode.ChildCount - 1 do
  1103. begin
  1104. vSubNode := vNode.ChildNodes[i];
  1105. sName := vSubNode.Rec.ValueByName('Name').AsString;
  1106. iUserID := vSubNode.Rec.ValueByName('WebUserID').AsInteger;
  1107. iWebID := vSubNode.Rec.ValueByName('WebID').AsInteger;
  1108. iWebFolderLevel := vSubNode.Rec.ValueByName('WebFolderLevel').AsInteger;
  1109. if (iWebID = ASubFolderID) and (iWebFolderLevel = G_WFL_BidType) and (iUserID = PHPWeb.UserID) then
  1110. begin
  1111. bSubExist := True;
  1112. vSubNode.LocateInControl;
  1113. if not SameText(sName, ASubFolderName) then
  1114. begin
  1115. vSubNode.Rec.ValueByName('Name').AsString := ASubFolderName;
  1116. bModified := True;
  1117. end;
  1118. Break;
  1119. end;
  1120. end;
  1121. if not bSubExist then
  1122. begin
  1123. vNode.LocateInControl;
  1124. vNode := FProjectManagerData.InsertSubProject(ASubFolderName, stdProjects.IDTree.Selected, ASubFolderID, G_WFL_BidType);
  1125. vNode.LocateInControl;
  1126. end;
  1127. if bModified then
  1128. ProjectManager.Save;
  1129. end;
  1130. function TProjectManagerFrame.Rec(AProjectID: Integer): TsdDataRecord;
  1131. var i: Integer;
  1132. vTree: TsdIDTree;
  1133. begin
  1134. vTree := stdProjects.IDTree;
  1135. if vTree.Selected.Rec.ValueByName('ID').AsInteger = AProjectID then
  1136. begin
  1137. Result := stdProjects.IDTree.Selected.Rec;
  1138. Exit;
  1139. end;
  1140. for i := 0 to vTree.Count - 1 do
  1141. begin
  1142. if vTree.Items[i].Rec.ValueByName('ID').AsInteger = AProjectID then
  1143. begin
  1144. Result := vTree.Items[i].Rec;
  1145. vTree.Items[i].LocateInControl;
  1146. Break;
  1147. end;
  1148. end;
  1149. end;
  1150. destructor TProjectManagerFrame.Destroy;
  1151. begin
  1152. FCheckerList.Free;
  1153. inherited;
  1154. end;
  1155. procedure TProjectManagerFrame.DoBatchReceiveOnline(ARequestType: Integer);
  1156. var
  1157. sURL, sCheckersURL, sHint: string;
  1158. vArr: TOVArr;
  1159. i, iFolderID, iSubFolderID: Integer;
  1160. vPSArr: TStrArr;
  1161. vCArr: TOVArr; // Checkers
  1162. begin
  1163. // 查询等待我审核的标段文件,杰哥说分三种:①业主未审核 ②业主审核中 ③审核人审核中 (为什么加①?问杰哥)
  1164. case PHPWeb.Search(PHPWeb.MeasureURL + 'user/get/audit/project', ['audituid', 'RequestType'],
  1165. [IntToStr(PHPWeb.UserID), IntToStr(ARequestType)], vArr) of
  1166. 1:
  1167. begin
  1168. CreateProgress('正在从云端下载新项目');
  1169. try
  1170. for i := Low(vArr) to High(vArr) do
  1171. begin
  1172. sURL := vArr[i, 0];
  1173. FWebFolder_OnLine := vArr[i, 1];
  1174. FWebSubFolder_OnLine := vArr[i, 2];
  1175. FWebMD5_OnLine := vArr[i, 3];
  1176. FWebID := StrToInt(vArr[i, 5]);
  1177. iFolderID := StrToInt(vArr[i, 6]);
  1178. iSubFolderID := StrToInt(vArr[i, 7]);
  1179. FWebCheckStatusMy := TCheckStatus(StrToInt(vArr[i, 8])-1); // vArr[i, 4]项目审核状态;vArr[i, 8]当前登陆用户的审核状态
  1180. FWebAuthorID := StrToInt(vArr[i, 9]);
  1181. FWebBidName_OnLine := vArr[i, 10];
  1182. FWebMD5_Local := LocalMD5(PHPWeb.UserID, FWebID);
  1183. CheckWebFolders(iFolderID, iSubFolderID, FWebFolder_OnLine, FWebSubFolder_OnLine);
  1184. CheckBidName(PHPWeb.UserID, FWebID, FWebBidName_OnLine);
  1185. if FWebMD5_OnLine <> FWebMD5_Local then
  1186. begin
  1187. sCheckersURL := Format('%smeasure/status/%d/get', [PHPWeb.MeasureURL, FWebID]);
  1188. if PHPWeb.Search(sCheckersURL, [''], [''], 3, vPSArr, vCArr) = 1 then
  1189. OnLineChecker(vCArr, FOnLineCheckerBegin, FOnLineCheckerEnd, FOnLineCheckerEndIsOwner)
  1190. else
  1191. begin
  1192. FOnLineCheckerBegin := 0;
  1193. FOnLineCheckerEnd := 0;
  1194. end;
  1195. if not FileDownAndReceive(sURL, 1) then Exit;
  1196. end;
  1197. end;
  1198. BubbleSortProjects;
  1199. finally
  1200. CloseProgress;
  1201. end;
  1202. end;
  1203. 0, -1:
  1204. begin
  1205. sHint := '网络出错,无法查询云端项目的更新情况,请重试!';
  1206. Application.MessageBox(PChar(sHint), '警告', MB_OK + MB_ICONWARNING);
  1207. Exit;
  1208. end;
  1209. end;
  1210. end;
  1211. function TProjectManagerFrame.ImportFile(const AFileName: string; AFileMD5: string): Boolean;
  1212. var
  1213. vImport: TTenderImport;
  1214. vNode: TsdIDTreeNode;
  1215. begin
  1216. Result := False;
  1217. vNode := stdProjects.IDTree.Selected;
  1218. vImport := TTenderImport.Create(vNode, '', AFileName);
  1219. try
  1220. try
  1221. vImport.ImportToSelect;
  1222. vNode.LocateInControl;
  1223. Result := True;
  1224. except
  1225. Result := False;
  1226. end;
  1227. finally
  1228. vImport.Free;
  1229. vNode.Rec.BeginUpdate;
  1230. vNode.Rec.ValueByName('WebMD5').AsString := AFileMD5;
  1231. vNode.Rec.ValueByName('WebUserID').AsInteger := PHPWeb.UserID;
  1232. vNode.Rec.EndUpdate;
  1233. FProjectManagerData.Save;
  1234. end;
  1235. end;
  1236. procedure TProjectManagerFrame.actnOpenBackupFolderExecute(
  1237. Sender: TObject);
  1238. var
  1239. stnNode: TsdIDTreeNode;
  1240. begin
  1241. stnNode := stdProjects.IDTree.Selected;
  1242. if stnNode.Rec.ValueByName('BackupFolder').AsString = '' then
  1243. TipMessage('该项目暂无备份数据!')
  1244. else
  1245. ShellExecute(Handle, 'open', 'Explorer.exe',
  1246. PChar(FProjectManagerData.BackupPath(stnNode.ID)), nil, 1);
  1247. end;
  1248. procedure TProjectManagerFrame.actnRenameUpdate(Sender: TObject);
  1249. var
  1250. Rec: TsdDataRecord;
  1251. bNet, bEnabled: Boolean;
  1252. begin
  1253. if Assigned(stdProjects.IDTree.Selected) and Assigned(stdProjects.IDTree.Selected.Rec) then
  1254. begin
  1255. bEnabled := True;
  1256. Rec := stdProjects.IDTree.Selected.Rec;
  1257. bNet := G_IsCloud;
  1258. if bNet then
  1259. begin
  1260. bEnabled := (Rec.ValueByName('Type').AsInteger = 1) and
  1261. (Rec.ValueByName('WebAuthorID').AsInteger = PHPWeb.UserID);
  1262. end;
  1263. end
  1264. else
  1265. bEnabled := False;
  1266. if bEnabled <> TAction(Sender).Enabled then
  1267. TAction(Sender).Enabled := bEnabled;
  1268. end;
  1269. procedure TProjectManagerFrame.CheckBidName(AID: Integer; ANewBidName: string);
  1270. var vNode: TsdIDTreeNode;
  1271. begin
  1272. vNode := stdProjects.IDTree.FindNode(AID);
  1273. if vNode = nil then Exit;
  1274. if vNode.Rec.ValueByName('Name').AsString <> ANewBidName then
  1275. begin
  1276. vNode.Rec.ValueByName('Name').AsString := ANewBidName;
  1277. ProjectManager.Save;
  1278. end;
  1279. end;
  1280. procedure TProjectManagerFrame.DoBatchReceiveAllOnline;
  1281. var
  1282. OnCC: TZjCellNotifyEvent;
  1283. begin
  1284. OnCC := zgProjects.OnCurrentChanged;
  1285. try
  1286. zgProjects.OnCurrentChanged := nil;
  1287. DoBatchReceiveOnline(2);
  1288. if stdProjects.IDTree.FirstNode <> nil then
  1289. stdProjects.IDTree.FirstNode.LocateInControl;
  1290. finally
  1291. zgProjects.OnCurrentChanged := OnCC;
  1292. end;
  1293. end;
  1294. procedure TProjectManagerFrame.GetLocalValues(ARec: TsdDataRecord);
  1295. begin
  1296. if not Assigned(ARec) then
  1297. begin
  1298. ClearLocalValues;
  1299. Exit;
  1300. end;
  1301. // 加这句后产生Bug:上报项目后,记录不曾移动,FID不变,不会刷新
  1302. // if ARec.ValueByName('ID').AsInteger <> FID then
  1303. begin
  1304. FID := ARec.ValueByName('ID').AsInteger;
  1305. FWebID := ARec.ValueByName('WebID').AsInteger;
  1306. FWebAuthorID := ARec.ValueByName('WebAuthorID').AsInteger;
  1307. FWebOwnerID := ARec.ValueByName('WebOwnerID').AsInteger;
  1308. FWebMD5_Local := ARec.ValueByName('WebMD5').AsString;
  1309. FWebBidName_Local := ARec.ValueByName('Name').AsString;
  1310. end;
  1311. end;
  1312. procedure TProjectManagerFrame.GetLocalValues(AUserID, AWebID: Integer);
  1313. var i: Integer;
  1314. vTree: TsdIDTree;
  1315. vRec: TsdDataRecord;
  1316. begin
  1317. ClearLocalValues; // 先清空,以防没找到。
  1318. if (CurRec <> nil) and
  1319. (CurRec.ValueByName('WebUserID').AsInteger = AUserID) and
  1320. (CurRec.ValueByName('WebID').AsInteger = AWebID) and
  1321. (CurRec.ValueByName('Type').AsInteger = 1) then
  1322. begin
  1323. GetLocalValues(vRec);
  1324. Exit;
  1325. end;
  1326. vTree := stdProjects.IDTree;
  1327. for i := 0 to vTree.Count - 1 do
  1328. begin
  1329. vRec := vTree.Items[i].Rec;
  1330. if (vRec.ValueByName('WebUserID').AsInteger = AUserID) and
  1331. (vRec.ValueByName('WebID').AsInteger = AWebID) and
  1332. (vRec.ValueByName('Type').AsInteger = 1) then
  1333. begin
  1334. GetLocalValues(vRec);
  1335. Break;
  1336. end;
  1337. end;
  1338. end;
  1339. procedure TProjectManagerFrame.ClearLocalValues;
  1340. begin
  1341. FID := -1;
  1342. FWebID := -1;
  1343. FWebAuthorID := -1;
  1344. FWebOwnerID := -1;
  1345. FWebMD5_Local := '';
  1346. FWebBidName_Local := '';
  1347. end;
  1348. function TProjectManagerFrame.LocalMD5(AUserID, AWebID: Integer): string;
  1349. var i: Integer;
  1350. vTree: TsdIDTree;
  1351. vRec: TsdDataRecord;
  1352. begin
  1353. Result := '本地无MD5码';
  1354. if (CurRec <> nil) and
  1355. (CurRec.ValueByName('WebUserID').AsInteger = AUserID) and
  1356. (CurRec.ValueByName('WebID').AsInteger = AWebID) and
  1357. (CurRec.ValueByName('Type').AsInteger = 1) then
  1358. begin
  1359. Result := vRec.ValueByName('WebMD5').AsString;
  1360. Exit;
  1361. end;
  1362. vTree := stdProjects.IDTree;
  1363. for i := 0 to vTree.Count - 1 do
  1364. begin
  1365. vRec := vTree.Items[i].Rec;
  1366. if (vRec.ValueByName('WebUserID').AsInteger = AUserID) and
  1367. (vRec.ValueByName('WebID').AsInteger = AWebID) and
  1368. (vRec.ValueByName('Type').AsInteger = 1) then
  1369. begin
  1370. Result := vRec.ValueByName('WebMD5').AsString;
  1371. Break;
  1372. end;
  1373. end;
  1374. end;
  1375. function TProjectManagerFrame.CurRec: TsdDataRecord;
  1376. begin
  1377. if stdProjects.IDTree.Selected = nil then
  1378. Result := nil
  1379. else
  1380. Result := stdProjects.IDTree.Selected.Rec;
  1381. end;
  1382. procedure TProjectManagerFrame.CheckBidName(AUserID, AWebID: Integer;
  1383. ANewBidName: string);
  1384. var i: Integer;
  1385. vTree: TsdIDTree;
  1386. vRec: TsdDataRecord;
  1387. begin
  1388. if (CurRec <> nil) and
  1389. (CurRec.ValueByName('WebUserID').AsInteger = AUserID) and
  1390. (CurRec.ValueByName('WebID').AsInteger = AWebID) and
  1391. (CurRec.ValueByName('Type').AsInteger = 1) then
  1392. begin
  1393. if (CurRec.ValueByName('Name').AsString <> ANewBidName) then
  1394. begin
  1395. CurRec.ValueByName('Name').AsString := ANewBidName;
  1396. ProjectManager.Save;
  1397. end;
  1398. Exit;
  1399. end;
  1400. vTree := stdProjects.IDTree;
  1401. for i := 0 to vTree.Count - 1 do
  1402. begin
  1403. vRec := vTree.Items[i].Rec;
  1404. if (vRec.ValueByName('WebUserID').AsInteger = AUserID) and
  1405. (vRec.ValueByName('WebID').AsInteger = AWebID) and
  1406. (vRec.ValueByName('Type').AsInteger = 1) then
  1407. begin
  1408. if vRec.ValueByName('Name').AsString <> ANewBidName then
  1409. begin
  1410. vRec.ValueByName('Name').AsString := ANewBidName;
  1411. ProjectManager.Save;
  1412. end;
  1413. Break;
  1414. end;
  1415. end;
  1416. end;
  1417. function TProjectManagerFrame.CurRecAttachmentPath: string;
  1418. begin
  1419. if G_IsCloud then
  1420. Result := PHPWeb.WebPath + 'Projects\' + CurRec.ValueByName('WebID').AsString + '\Attachment\'
  1421. else
  1422. Result := GetMyProjectsFilePath + 'Attachment\' + CurRec.ValueByName('FileName').AsString + '\';
  1423. end;
  1424. function TProjectManagerFrame.AttachmentFileCountsWithoutManageFile(ANode: TsdIDTreeNode): Integer;
  1425. function GetCount(ANode: TsdIDTreeNode): Integer;
  1426. var sPath: string;
  1427. begin
  1428. if not Assigned(ANode) then Exit;
  1429. Result := 0;
  1430. if ANode.Rec.ValueByName('Type').AsInteger = 0 then
  1431. Result := Result + 0
  1432. else
  1433. begin
  1434. sPath := GetMyProjectsFilePath + 'Attachment\' + ANode.Rec.ValueByName('FileName').AsString + '\';
  1435. Result := Result + FileCount(sPath) - 1;
  1436. end;
  1437. if Assigned(ANode.FirstChild) then
  1438. Result := Result + GetCount(ANode.FirstChild);
  1439. if Assigned(ANode.NextSibling) then
  1440. Result := Result + GetCount(ANode.NextSibling);
  1441. end;
  1442. begin
  1443. if not Assigned(ANode) then Exit;
  1444. if Assigned(ANode.FirstChild) then
  1445. Result := GetCount(ANode.FirstChild)
  1446. else
  1447. Result := 0;
  1448. end;
  1449. procedure TProjectManagerFrame.BubbleSortProjects;
  1450. // 不能排最顶层
  1451. procedure BubbleSort(ANode: TsdIDTreeNode);
  1452. var n, t, c, temp: Integer;
  1453. bSwap: Boolean;
  1454. vNode1, vNode2, vTempNode: TsdIDTreeNode;
  1455. begin
  1456. if ANode = nil then Exit;
  1457. // if ANode.Rec.ValueByName('WebFolderLevel').AsInteger = G_WFL_ProjName then Exit;
  1458. n := ANode.ChildCount;
  1459. for t := 1 to n - 1 do
  1460. begin
  1461. bSwap := False;
  1462. for c := 1 to (n - t) do
  1463. begin
  1464. vNode1 := ANode.ChildNodes[c - 1];
  1465. vNode2 := ANode.ChildNodes[c];
  1466. if AnsiCompareStr(vNode1.Rec.ValueByName('Name').AsString,
  1467. vNode2.Rec.ValueByName('Name').AsString) = 1 then
  1468. begin
  1469. vNode1.DownMove;
  1470. bSwap := True;
  1471. end;
  1472. end;
  1473. if bSwap = False then Break;
  1474. end;
  1475. if Assigned(ANode.FirstChild) then
  1476. BubbleSort(ANode.FirstChild);
  1477. if Assigned(ANode.NextSibling) then
  1478. BubbleSort(ANode.NextSibling);
  1479. end;
  1480. begin
  1481. BubbleSort(stdProjects.IDTree.FirstNode);
  1482. end;
  1483. procedure TProjectManagerFrame.zgProjectsShowHint(var HintStr: String;
  1484. var CanShow: Boolean; var HintInfo: THintInfo; const ACoord: TPoint);
  1485. var
  1486. vCell: TZjCell;
  1487. vNode: TsdIDTreeNode;
  1488. iLevelWidth: Integer;
  1489. rText: TRect;
  1490. procedure CalcTextRect(var R: TRect);
  1491. var
  1492. DC: HDC;
  1493. iTextHeight: Integer;
  1494. begin
  1495. DC := CreateCompatibleDC(0);
  1496. try
  1497. SelectObject(DC, vCell.Font.Handle);
  1498. iTextHeight := DrawText(DC, PChar(vCell.Text), Length(vCell.Text), R, DT_SINGLELINE or DT_VCenter
  1499. or DT_NOCLIP or DT_CALCRECT);
  1500. finally
  1501. DeleteDC(DC);
  1502. end;
  1503. end;
  1504. begin
  1505. if (ACoord.Y < 1) and (ACoord.X <> 1) then Exit;
  1506. vCell := zgProjects.Cells[ACoord.X, ACoord.Y];
  1507. with HintInfo do
  1508. begin
  1509. vNode := stdProjects.IDTree.Items[ACoord.Y - 1];
  1510. if not Assigned(vNode) then Exit;
  1511. iLevelWidth := (vNode.Level + 1) * 20 + 16;
  1512. rText := CursorRect;
  1513. CalcTextRect(rText);
  1514. if (rText.Right - rText.Left + iLevelWidth > CursorRect.Right - CursorRect.Left) or
  1515. (rText.Right > ClientWidth) then
  1516. begin
  1517. CanShow := True;
  1518. HintStr := vCell.Text;
  1519. GetCursorPos(HintPos);
  1520. end;
  1521. end;
  1522. end;
  1523. function TProjectManagerFrame.FileDownAndReceive(ADownURL: string; AReceiveKind: Integer): Boolean;
  1524. var sLocalFile, sHint: string;
  1525. bCanImp: Boolean;
  1526. vFileCheck: TTenderFileChecker;
  1527. begin
  1528. Result := False;
  1529. // 下载
  1530. sLocalFile := PHPWeb.UserPath + ExtractFileName(ADownURL);
  1531. if not PHPWeb.DownFile(ADownURL, sLocalFile) then
  1532. begin
  1533. sHint := Format('云端已找到 [%s] 的新文件,但由于网络原因下载失败!请重试!', [FWebBidName_Local]);
  1534. Application.MessageBox(PChar(sHint), '系统提醒', MB_OK + MB_ICONWARNING);
  1535. Exit;
  1536. end;
  1537. // 接收前先检验原报文件是否正确(审核不通过打回)
  1538. if AReceiveKind = 2 then
  1539. begin
  1540. // 有时原报文件出错:包含了1审2审的数据。
  1541. vFileCheck := TTenderFileChecker.Create;
  1542. try
  1543. // 有一期以上数据,且最新期数据审核状态为原报
  1544. bCanImp := vFileCheck.CheckFileValid(sLocalFile) and (vFileCheck.PhaseCount > 0) and (vFileCheck.AuditStatus = 0);
  1545. if not bCanImp then
  1546. begin
  1547. Application.MessageBox(PChar('已从云端下载原报文件到本地,但文件有错误(包含1审数据)禁止接收!请致电纵横服务人员以获取帮助。'),
  1548. '警告', MB_OK + MB_ICONWARNING);
  1549. Exit;
  1550. end;
  1551. finally
  1552. vFileCheck.Free;
  1553. end;
  1554. {注意: 审核末通过的导入更新,这里的MD5码应取最终审核不通过项目的,否则会带来下载循环问题。
  1555. 问题描述:
  1556. ①编制人运行软件,双击项目,发现审核不通过,自动下载无锁文件,开始新一期,保存关闭。
  1557. ②编制人再次运行软件,双击项目,MD5码不同,自动下载旧的审核不通过文件,覆盖本地。
  1558. ③编制人再次运行软件,双击项目,发现审核不通过,重复①,循环....
  1559. 问:①那里下载无锁文件后不能改成新的MD5码吗?
  1560. 答:不能,因为审核人的项目会变成无锁文件,看不到不通过项目。MD5码只能在上传后更新。
  1561. 且编制人有个交互界面2种选择:a.下载更新不通过项目查看;b.下载更新无锁文件开始新一期。}
  1562. end;
  1563. // 接收更新
  1564. if not ReceiveFile(sLocalFile, (AReceiveKind = 2)) then
  1565. begin
  1566. sHint := Format('已从云端下载新的 [%s] 到本地 [%s],但接收失败,请删除该项目然后重新从云端获取!', [FWebBidName_OnLine, sLocalFile]);
  1567. Application.MessageBox(PChar(sHint), '系统提醒', MB_OK + MB_ICONWARNING);
  1568. Exit;
  1569. end;
  1570. if FileExists(sLocalFile) then
  1571. DeleteFile(sLocalFile);
  1572. Result := True;
  1573. end;
  1574. // 为了跟PHP的数组兼容,这里限制数组的第一个元素是A[0](不能是A[1])
  1575. procedure TProjectManagerFrame.OnLineChecker(AAr: TOVArr; var ABegin,
  1576. AEnd: Integer; var AOnLineEndIsOwner: Boolean);
  1577. var i, j: Integer;
  1578. vCS: TCheckStatus;
  1579. begin
  1580. i := 0;
  1581. j := 0;
  1582. ABegin := 0;
  1583. AEnd := 0;
  1584. AOnLineEndIsOwner := AAr[High(AAr), 7] = '1'; // 接口返回的第7列是线上审批标记
  1585. // 数组必须在位置n处截断,后面的部分作废。n的取值情况:
  1586. // ①轮到工作中的那个人的前一个人 ②审核不通过的那个人 ③ 审核通过的人是业主
  1587. if AOnLineEndIsOwner then
  1588. AEnd := High(AAr) + 1
  1589. else
  1590. begin
  1591. for i := High(AAr) downto 0 do
  1592. begin
  1593. vCS := TCheckStatus(StrToInt(AAr[i, 5]) - 1);
  1594. if vCS = csChecking then
  1595. begin
  1596. if (i > 0) and (AAr[i - 1, 7] = '1') then // 前一个人且不是第一个人
  1597. AEnd := i;
  1598. Break;
  1599. end
  1600. else if vCS = csNotPass then
  1601. begin
  1602. if AAr[i, 7] = '1' then
  1603. AEnd := i + 1;
  1604. Break;
  1605. end;
  1606. end;
  1607. end;
  1608. if AEnd = 0 then Exit;
  1609. if AEnd = 1 then
  1610. begin
  1611. ABegin := 1;
  1612. Exit;
  1613. end;
  1614. for j := AEnd -2 downto 0 do
  1615. begin
  1616. if AAr[j, 7] = '0' then
  1617. begin
  1618. ABegin := j + 2;
  1619. Break;
  1620. end
  1621. else
  1622. begin
  1623. if j = 0 then
  1624. ABegin := 1;
  1625. end;
  1626. end;
  1627. end;
  1628. procedure TProjectManagerFrame.actnExportUpdate(Sender: TObject);
  1629. begin
  1630. TAction(Sender).Enabled := Assigned(stdProjects.IDTree.Selected);
  1631. end;
  1632. procedure TProjectManagerFrame.actnOpenBackupFolderUpdate(Sender: TObject);
  1633. begin
  1634. TAction(Sender).Enabled := Assigned(stdProjects.IDTree.Selected) and
  1635. (stdProjects.IDTree.Selected.Rec.ValueByName('Type').AsInteger = 1);
  1636. end;
  1637. procedure TProjectManagerFrame.pnlProgressClick(Sender: TObject);
  1638. var s: string;
  1639. begin
  1640. if (GetKeyState(VK_LSHIFT) < 0) and (GetKeyState(VK_LCONTROL) < 0) then
  1641. begin
  1642. s := Format('%d, %d, %s', [FOnLineCheckerBegin, FOnLineCheckerEnd, BoolToStr(FOnLineCheckerEndIsOwner)]);
  1643. Application.MessageBox(Pchar(s), 'Hint');
  1644. end;
  1645. end;
  1646. end.