ProjectManagerFme.pas 63 KB

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