ProjectManagerFme.pas 52 KB

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