ProjectManagerFme.pas 60 KB

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