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