ProjectManagerFme.pas 50 KB

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