| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183 | unit BillsCompileDm;interfaceuses  BillsDm, StandardBillsFme,  SysUtils, Classes, sdDB, BillsTree, sdIDTree, DB;type  TBillsCompileData = class(TDataModule)    sdvBillsCompile: TsdDataView;    procedure sdvBillsCompileGetText(var Text: String;      ARecord: TsdDataRecord; AValue: TsdValue; AColumn: TsdViewColumn;      DisplayText: Boolean);    procedure sdvBillsCompileAfterValueChanged(AValue: TsdValue);    procedure sdvBillsCompileBeforeValueChange(AValue: TsdValue;      const NewValue: Variant; var Allow: Boolean);    procedure sdvBillsCompileSetText(var Text: String;      ARecord: TsdDataRecord; AValue: TsdValue; AColumn: TsdViewColumn;      var Allow: Boolean);    procedure sdvBillsCompileAfterOpen(Sender: TObject);    procedure sdvBillsCompileAfterClose(Sender: TObject);    procedure sdvBillsCompileAfterAddRecord(ARecord: TsdDataRecord);    procedure sdvBillsCompileCurrentChanged(ARecord: TsdDataRecord);  private    FProjectData: TObject;    FBillsData: TBillsData;    FBillsCompileTree: TCompileBillsIDTree;    FOnRecChange: TRecChangeEvent;    function GatherChildrenOrg(ANode: TsdIDTreeNode): Double;    procedure UpdateRecordOrg(ABillsID: Integer; ATotalPrice: Double);    function FindChild(AParentNode, ANode: TsdIDTreeNode): TsdIDTreeNode;    function InsertChild(AParentNode, ANode: TsdIDTreeNode): TsdIDTreeNode;    function CompareNodeCode(ANode, ACompareNode: TsdIDTreeNode): Integer;    function GetNextSiblingID(AParent, ANode: TsdIDTreeNode): Integer;    function IsSameNode(ANode, ACompareNode: TsdIDTreeNode): Boolean;    function GetTopParentNode(ANode: TsdIDTreeNode; ALevel: Integer): TsdIDTreeNode;    procedure AddXmjBillsFromLib(AStdBillsNode: TsdIDTreeNode);    function CanAddGclBills: Boolean;    function GetGclBillsParent(AChildNode: TsdIDTreeNode): TsdIDTreeNode;    procedure AddGclBillsFromLib(AStdBillsNode: TsdIDTreeNode);    procedure DoOnAfterDeleteNode(AParent: TsdIDTreeNode);    function GatherChildren(ANode: TsdIDTreeNode; const AFieldName: string): Double;    procedure UpdateParent(ABillsID: Integer; ADifferTotalPrice: Double; const AFieldName: string);    // 经济指标[与其他节点无关]    procedure CalculateDesignPrice(ANode: TBillsIDTreeNode);    // 施工图原设计[增量]    procedure CalculateOrg(ABillsID: Integer);    // 设计错漏增减[增量]    procedure CalculateMis(ABillsID: Integer);    // 其他错漏增减[增量]    procedure CalculateOth(ABillsID: Integer);    procedure CalculateTotal(ABillsID: Integer);    procedure CalculateLeaf(ANode: TBillsIDTreeNode);    procedure GatherNode(ANode: TBillsIDTreeNode);    procedure CalculateBills(ANode: TsdIDTreeNode);    function GetActive: Boolean;    procedure SetOnRecChange(const Value: TRecChangeEvent);  public    constructor Create(AProjectData: TObject);    destructor Destroy; override;    procedure Open;    procedure Close;    procedure ReConnectTree;    procedure AddBillsFromLib(ANode: TsdIDTreeNode; ABillsType: TBillsType);    procedure AddBillsFromDealBills(ARec: TsdDataRecord);    procedure Calculate(ABillsID: Integer);    procedure CalculateAll;    function GetLeafXmjParentID(ABillsID: Integer): Integer;    procedure ExpandNodeTo(ALevel: Integer);    procedure ExpandXmjNode;    procedure ExpandPegXmjNode;    procedure ReorderChildrenCode(ANode: TsdIDTreeNode);    // 所有解锁的节点全部重新锁定    procedure ReLockBaseData;    property ProjectData: TObject read FProjectData;    property BillsData: TBillsData read FBillsData;    property BillsCompileTree: TCompileBillsIDTree read FBillsCompileTree;    property Active: Boolean read GetActive;    property OnRecChange: TRecChangeEvent read FOnRecChange write SetOnRecChange;  end;implementationuses  ProjectData, Math, ZhAPI, UtilMethods, ConstUnit, mDataRecord, Variants;{$R *.dfm}{ TBillsCompileData }constructor TBillsCompileData.Create(AProjectData: TObject);begin  inherited Create(nil);  FProjectData := AProjectData;  FBillsData := TProjectData(FProjectData).BillsData;  FBillsCompileTree := TCompileBillsIDTree.Create;  FBillsCompileTree.KeyFieldName := 'ID';  FBillsCompileTree.ParentFieldName := 'ParentID';  FBillsCompileTree.NextSiblingFieldName := 'NextSiblingID';  FBillsCompileTree.AutoCreateKeyID := True;  FBillsCompileTree.AutoExpand := True;  FBillsCompileTree.DataView := sdvBillsCompile;  FBillsCompileTree.SeedID := Max(FBillsCompileTree.SeedID, 100);  FBillsCompileTree.OnReCalcNode := Calculate;end;destructor TBillsCompileData.Destroy;begin  FBillsCompileTree.Free;  inherited;end;procedure TBillsCompileData.Open;begin  sdvBillsCompile.DataSet := TProjectData(FProjectData).BillsData.sddBills;  sdvBillsCompile.Open;  FBillsCompileTree.SeedID := Max(FBillsCompileTree.SeedID, 100);end;procedure TBillsCompileData.ReConnectTree;begin  FBillsCompileTree.DataView := nil;  FBillsCompileTree.DataView := sdvBillsCompile;end;procedure TBillsCompileData.sdvBillsCompileGetText(var Text: String;  ARecord: TsdDataRecord; AValue: TsdValue; AColumn: TsdViewColumn;  DisplayText: Boolean);  procedure GetEditText;  var    sFormula: string;    sFormulaField: string;  begin    sFormula := '';    if ARecord.ValueByName('CalcType').AsInteger = 0 then    begin      if SameText('OrgQuantity', AColumn.FieldName) then        sFormula := ARecord.ValueByName('OrgFormula').AsString      else if SameText('MisQuantity', AColumn.FieldName) then        sFormula := ARecord.ValueByName('MisFormula').AsString      else if SameText('OthQuantity', AColumn.FieldName) then        sFormula := ARecord.ValueByName('OthFormula').AsString;    end    else if ARecord.ValueByName('CalcType').AsInteger = 1 then    begin      if SameText('OrgTotalPrice', AColumn.FieldName) then        sFormula := ARecord.ValueByName('OrgFormula').AsString      else if SameText('MisTotalPrice', AColumn.FieldName) then        sFormula := ARecord.ValueByName('MisFormula').AsString      else if SameText('OthTotalPrice', AColumn.FieldName) then        sFormula := ARecord.ValueByName('OthFormula').AsString;    end;    if sFormula <> '' then      Text := sFormula;  end;  procedure GetDisplayText;  begin    if AValue.DataType = ftFloat then    begin      if not Assigned(AValue) or (AValue.AsFloat = 0) then        Text := '';    end;  end;begin  if DisplayText then    GetDisplayText  else    GetEditText;end;procedure TBillsCompileData.ExpandNodeTo(ALevel: Integer);begin  BillsCompileTree.ExpandLevel := ALevel;end;procedure TBillsCompileData.ExpandXmjNode;var  iIndex: Integer;  stnNode: TBillsIDTreeNode;begin  for iIndex := 0 to BillsCompileTree.Count - 1 do  begin    stnNode := TBillsIDTreeNode(BillsCompileTree.Items[iIndex]);    if (stnNode.ParentID <> -1) then      stnNode.Parent.Expanded := stnNode.Rec.B_Code.AsString = '';  end;end;procedure TBillsCompileData.sdvBillsCompileAfterValueChanged(  AValue: TsdValue);  procedure ResetChildrenLockedInfo(ANode: TsdIDTreeNode; ALockedInfo: Boolean);  var    iChild: Integer;  begin    if not Assigned(ANode) then Exit;    if ANode.Rec.ValueByName('LockedLevel').AsBoolean then      ANode.Rec.ValueByName('LockedInfo').AsBoolean := ALockedInfo;    if ANode.HasChildren then      for iChild := 0 to ANode.ChildCount - 1 do        ResetChildrenLockedInfo(ANode.ChildNodes[iChild], ALockedInfo);  end;var  vNode: TBillsIDTreeNode;begin  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(AValue.Owner.ValueByName('ID').AsInteger));  if SameText(AValue.FieldName, 'OrgQuantity') or      SameText(AValue.FieldName, 'OrgTotalPrice') then    CalculateOrg(AValue.Owner.ValueByName('ID').AsInteger)  else if SameText(AValue.FieldName, 'MisQuantity') or      SameText(AValue.FieldName, 'MisTotalPrice') then    CalculateMis(AValue.Owner.ValueByName('ID').AsInteger)  else if SameText(AValue.FieldName, 'OthQuantity') or      SameText(AValue.FieldName, 'OthTotalPrice') then    CalculateOth(AValue.Owner.ValueByName('ID').AsInteger)  else if SameText(AValue.FieldName, 'Price') then    CalculateTotal(AValue.Owner.ValueByName('ID').AsInteger)  else if SameText(AValue.FieldName, 'DgnQuantity1') then    CalculateDesignPrice(vNode);  if (AValue.FieldName = 'LockedInfo') then    ResetChildrenLockedInfo(vNode, AValue.AsBoolean);  if (AValue.FieldName = 'B_Code') then  begin    AValue.Owner.ValueByName('DgnQuantity1').Clear;    AValue.Owner.ValueByName('DgnQuantity2').Clear;    AValue.Owner.ValueByName('DgnPrice').Clear;  end;end;function TBillsCompileData.GatherChildrenOrg(ANode: TsdIDTreeNode): Double;var  iChild: Integer;begin  if ANode = nil then Exit;  if ANode.HasChildren and Assigned(ANode.FirstChild) then  begin    Result := 0;    for iChild := 0 to ANode.ChildCount - 1 do      Result := Result + GatherChildrenOrg(ANode.ChildNodes[iChild]);    Result := TotalPriceRoundTo(Result);  end  else    if Assigned(ANode.Rec) and Assigned(ANode.Rec.ValueByName('TotalPrice')) then      Result := ANode.Rec.ValueByName('TotalPrice').AsFloat    else      Result := 0;end;procedure TBillsCompileData.UpdateRecordOrg(ABillsID: Integer;  ATotalPrice: Double);var  stnNode: TsdIDTreeNode;begin  stnNode := BillsCompileTree.FindNode(ABillsID);  if not Assigned(stnNode) then Exit;  with stnNode.Rec do  begin    ValueByName('TotalPrice').AsFloat := TotalPriceRoundTo(      ValueByName('TotalPrice').AsFloat + ATotalPrice);    if ValueByName('DgnQuantity1').AsFloat <> 0 then      ValueByName('DgnPrice').AsFloat := PriceRoundTo(          ValueByName('TotalPrice').AsFloat/ValueByName('DgnQuantity1').AsFloat);  end;  UpdateRecordOrg(stnNode.ParentID, ATotalPrice);end;procedure TBillsCompileData.sdvBillsCompileBeforeValueChange(  AValue: TsdValue; const NewValue: Variant; var Allow: Boolean);begin  if SameText(AValue.FieldName, 'OrgQuantity') or      SameText(AValue.FieldName, 'MisQuantity') or      SameText(AValue.FieldName, 'OthQuantity') or      SameText(AValue.FieldName, 'OrgTotalPrice') or      SameText(AValue.FieldName, 'MisTotalPrice') or      SameText(AValue.FieldName, 'OthTotalPrice') or      SameText(AValue.FieldName, 'Price') then  begin    TBillsRecord(AValue.Owner).CacheOrgTP := AValue.Owner.ValueByName('OrgTotalPrice').AsFloat;    TBillsRecord(AValue.Owner).CacheMisTP := AValue.Owner.ValueByName('MisTotalPrice').AsFloat;    TBillsRecord(AValue.Owner).CacheOthTP := AValue.Owner.ValueByName('OthTotalPrice').AsFloat;  end;end;procedure TBillsCompileData.CalculateAll;  procedure RecursiveCalc(ANode: TsdIDTreeNode);  begin    if not Assigned(ANode) then Exit;    if ANode.HasChildren then    begin      RecursiveCalc(ANode.FirstChild);      GatherNode(TBillsIDTreeNode(ANode));    end    else      CalculateLeaf(TBillsIDTreeNode(ANode));    RecursiveCalc(ANode.NextSibling);  end;  procedure BeginCalc;  begin    sdvBillsCompile.BeforeValueChange := nil;    sdvBillsCompile.AfterValueChanged := nil;  end;  procedure EndCalc;  begin    sdvBillsCompile.BeforeValueChange := sdvBillsCompileBeforeValueChange;    sdvBillsCompile.AfterValueChanged := sdvBillsCompileAfterValueChanged;  end;begin  BeginCalc;  try    RecursiveCalc(BillsCompileTree.FirstNode);  finally    EndCalc;  end;end;procedure TBillsCompileData.AddBillsFromLib(ANode: TsdIDTreeNode;  ABillsType: TBillsType);begin  if not Assigned(ANode) then Exit;  if ABillsType = btXm then    AddXmjBillsFromLib(ANode)  else if ABillsType = btGcl then    AddGclBillsFromLib(ANode);end;procedure TBillsCompileData.AddGclBillsFromLib(  AStdBillsNode: TsdIDTreeNode);var  stnParent, stnStdNode: TsdIDTreeNode;  iLevel: Integer;begin  if not CanAddGclBills then    raise Exception.Create('当前节点下不可添加工程量清单!');  stnParent := GetGclBillsParent(BillsCompileTree.Selected);  if TBillsIDTreeNode(stnParent).HasLedger or     (not stnParent.HasChildren and TBillsIDTreeNode(stnParent).HasMeasure) then    raise Exception.Create('当前节点不可添加工程量清单!');  stnStdNode := GetTopParentNode(AStdBillsNode, AStdBillsNode.Level);  for iLevel := 1 to AStdBillsNode.Level + 1 do  begin    if stnStdNode.Rec.ValueByName('B_Code').AsString <> '' then      if FindChild(stnParent, stnStdNode) <> nil then        stnParent := FindChild(stnParent, stnStdNode)      else        stnParent := InsertChild(stnParent, stnStdNode);    stnStdNode := GetTopParentNode(AStdBillsNode, AStdBillsNode.Level - iLevel);  end;end;procedure TBillsCompileData.AddXmjBillsFromLib(  AStdBillsNode: TsdIDTreeNode);var  stnStdNode, stnCurNode: TsdIDTreeNode;  iLevel: Integer;begin  stnStdNode := GetTopParentNode(AStdBillsNode, AStdBillsNode.Level);  stnCurNode := nil;  for iLevel := 1 to AStdBillsNode.Level + 1 do  begin    if FindChild(stnCurNode, stnStdNode) <> nil then      stnCurNode := FindChild(stnCurNode, stnStdNode)    else    begin      if TBillsIDTreeNode(stnCurNode).HasLedger or         (not stnCurNode.HasChildren and TBillsIDTreeNode(stnCurNode).HasMeasure) then        raise Exception.Create('不可添加该项目节数据!')      else        stnCurNode := InsertChild(stnCurNode, stnStdNode);    end;    stnStdNode := GetTopParentNode(AStdBillsNode, AStdBillsNode.Level - iLevel);  end;end;function TBillsCompileData.CanAddGclBills: Boolean;  function CheckChildrenHasXmj(ANode: TsdIDTreeNode): Boolean;  var    stnCurNode: TsdIDTreeNode;  begin    Result := False;    if not ANode.HasChildren then Exit;    stnCurNode := ANode.FirstChild;    while not Result and Assigned(stnCurNode) do    begin      Result := Result or (stnCurNode.Rec.ValueByName('Code').AsString <> '');      if stnCurNode.HasChildren then        Result := Result or CheckChildrenHasXmj(stnCurNode);      stnCurNode := stnCurNode.NextSibling;    end;  end;  function CheckParentIsXmj(ANode: TsdIDTreeNode): Boolean;  begin    Result := False;    if not Assigned(ANode) then Exit;    Result := ANode.Rec.ValueByName('Code').AsString <> '';    if not Result then      Result := Result or CheckParentIsXmj(ANode.Parent);  end;begin  Result := False;  if not Assigned(BillsCompileTree.Selected) then Exit;  Result := CheckParentIsXmj(BillsCompileTree.Selected)    and not CheckChildrenHasXmj(BillsCompileTree.Selected);end;function TBillsCompileData.CompareNodeCode(ANode,  ACompareNode: TsdIDTreeNode): Integer;begin  if ANode.Rec.ValueByName('Code').AsString <> '' then    Result := CompareCode(ANode.Rec.ValueByName('Code').AsString,      ACompareNode.Rec.ValueByName('Code').AsString)  else if ANode.Rec.ValueByName('B_Code').AsString <> '' then    Result := CompareCode(ANode.Rec.ValueByName('B_Code').AsString,      ACompareNode.Rec.ValueByName('B_Code').AsString);end;function TBillsCompileData.GetGclBillsParent(  AChildNode: TsdIDTreeNode): TsdIDTreeNode;begin  if AChildNode.Rec.ValueByName('B_Code').AsString <> '' then    Result := GetGclBillsParent(AChildNode.Parent)  else    Result := AChildNode;end;function TBillsCompileData.GetNextSiblingID(AParent,  ANode: TsdIDTreeNode): Integer;var  stnCurNode: TsdIDTreeNode;begin  Result := -1;  if Assigned(AParent) then    stnCurNode := AParent.FirstChild  else    stnCurNode := BillsCompileTree.FirstNode;  if not Assigned(stnCurNode) then Exit;  while Assigned(stnCurNode) do  begin    if CompareNodeCode(stnCurNode, ANode) >= 0 then    begin      Result := stnCurNode.ID;      Exit;    end;    stnCurNode := stnCurNode.NextSibling;  end;end;function TBillsCompileData.GetTopParentNode(ANode: TsdIDTreeNode;  ALevel: Integer): TsdIDTreeNode;begin  Result := ANode;  while Assigned(Result.Parent) and (Result.Level + ALevel > ANode.Level) do    Result := Result.Parent;end;function TBillsCompileData.IsSameNode(ANode,  ACompareNode: TsdIDTreeNode): Boolean;begin  Result := (ANode.Rec.ValueByName('Code').AsString = ACompareNode.Rec.ValueByName('Code').AsString)        and (ANode.Rec.ValueByName('B_Code').AsString = ACompareNode.Rec.ValueByName('B_Code').AsString)        and (ANode.Rec.ValueByName('Name').AsString = ACompareNode.Rec.ValueByName('Name').AsString);end;function TBillsCompileData.FindChild(AParentNode,  ANode: TsdIDTreeNode): TsdIDTreeNode;  function FindSibling(AFirstNode, ANode: TsdIDTreeNode): TsdIDTreeNode;  var    stnCurNode: TsdIDTreeNode;  begin    Result := nil;    stnCurNode := AFirstNode;    while Assigned(stnCurNode) and not Assigned(Result) do    begin      if IsSameNode(ANode, stnCurNode) then        Result := stnCurNode;      stnCurNode := stnCurNode.NextSibling;    end;  end;begin  if not Assigned(AParentNode) then    Result := FindSibling(BillsCompileTree.FirstNode, ANode)  else    Result := FindSibling(AParentNode.FirstChild, ANode);end;function TBillsCompileData.InsertChild(AParentNode,  ANode: TsdIDTreeNode): TsdIDTreeNode;var  iNextSiblingID: Integer;begin  iNextSiblingID := GetNextSiblingID(AParentNode, ANode);  if Assigned(AParentNode) then    Result := BillsCompileTree.Add(AParentNode.ID, iNextSiblingID)  else    Result := BillsCompileTree.Add(-1, iNextSiblingID);  Result.Rec.ValueByName('Code').AsString := ANode.Rec.ValueByName('Code').AsString;  Result.Rec.ValueByName('B_Code').AsString := ANode.Rec.ValueByName('B_Code').AsString;  Result.Rec.ValueByName('Name').AsString := ANode.Rec.ValueByName('Name').AsString;  Result.Rec.ValueByName('Units').AsString := ANode.Rec.ValueByName('Unit').AsString;end;procedure TBillsCompileData.sdvBillsCompileSetText(var Text: String;  ARecord: TsdDataRecord; AValue: TsdValue; AColumn: TsdViewColumn;  var Allow: Boolean);  procedure SetTextErrorHint(const AHint: string);  begin    ErrorMessage(AHint);    Allow := False;  end;  procedure SetQuantity(const AFieldName: string);  var    sPre: string;  begin    sPre := StringReplace(AFieldName, 'Quantity', '', [rfIgnoreCase, rfReplaceAll]);    if CheckStringNull(Text) or CheckNumeric(Text) then    begin      ARecord.ValueByName(sPre + 'Formula').AsString := '';      Text := FloatToStr(QuantityRoundTo(StrToFloatDef(Text, 0)));    end    else    begin      ARecord.ValueByName(sPre + 'Formula').AsString := Text;      Text := FloatToStr(QuantityRoundTo(EvaluateExprs(Text)));    end;    ARecord.ValueByName('CalcType').AsInteger := 0;  end;  procedure SetTotalPrice(const AFieldName: string);  var    sPre: string;  begin    sPre := StringReplace(AFieldName, 'TotalPrice', '', [rfIgnoreCase, rfReplaceAll]);    if CheckStringNull(Text) or CheckNumeric(Text) then    begin      ARecord.ValueByName(sPre + 'Formula').AsString := '';      Text := FloatToStr(TotalPriceRoundTo(StrToFloatDef(Text, 0)));    end    else    begin      ARecord.ValueByName(sPre + 'Formula').AsString := Text;      Text := FloatToStr(TotalPriceRoundTo(EvaluateExprs(Text)));    end;    ARecord.ValueByName('CalcType').AsInteger := 1;  end;  procedure SetDgnQuantity;  begin    Text := FloatToStr(QuantityRoundTo(StrToFloatDef(Text, 0)));  end;  procedure SetPrice;  begin    Text := FloatToStr(PriceRoundTo(StrToFloatDef(Text, 0)));    ARecord.ValueByName('CalcType').AsInteger := 0;  end;  procedure DoCurChanged(ANode: TBillsIDTreeNode);  begin    if SameText(AColumn.FieldName, 'OrgQuantity') or        SameText(AColumn.FieldName, 'MisQuantity') or        SameText(AColumn.FieldName, 'OthQuantity')then      SetQuantity(AColumn.FieldName)    else if SameText(AColumn.FieldName, 'OrgTotalPrice') or        SameText(AColumn.FieldName, 'MisTotalPrice') or        SameText(AColumn.FieldName, 'OthTotalPrice') then      SetTotalPrice(AColumn.FieldName)    else if Pos('DgnQuantity', AColumn.FieldName) = 1 then      SetDgnQuantity    else if SameText(AColumn.FieldName, 'Price') then      SetPrice    else if SameText(AColumn.FieldName, 'Code') then      BillsCompileTree.RecodeChildrenCode(ANode, AValue.AsString, Text)    else if SameText(AColumn.FieldName, 'B_Code') then      BillsCompileTree.RecodeChildrenB_Code(ANode, AValue.AsString, Text);  end;  procedure CheckLockedData;  begin    if SameText(AColumn.FieldName, 'Code') or       SameText(AColumn.FieldName, 'B_Code') or       SameText(AColumn.FieldName, 'Name') or       SameText(AColumn.FieldName, 'Units') or       SameText(AColumn.FieldName, 'Price') or       SameText(AColumn.FieldName, 'OrgQuantity') or       SameText(AColumn.FieldName, 'OrgTotalPrice') or       SameText(AColumn.FieldName, 'MisQuantity') or       SameText(AColumn.FieldName, 'MisTotalPrice') or       SameText(AColumn.FieldName, 'OthQuantity') or       SameText(AColumn.FieldName, 'OthTotalPrice') or       SameText(AColumn.FieldName, 'DrawingCode')then       if ARecord.ValueByName('LockedInfo').AsBoolean then         SetTextErrorHint('清单信息已被锁定,不允许修改编号、名称、单位、清单单价、0号台账数量与金额、图号!');  end;  procedure CheckNodeWritable(ANode: TBillsIDTreeNode);  var    iCreatePhase: Integer;  begin    if not Allow then Exit;    iCreatePhase := ANode.Rec.ValueByName('CreatePhaseID').AsInteger;    if ANode.ID = iPriceMarginID then      SetTextErrorHint(sBills_PMHint);    if ANode.HasChildren then    begin      if Text = '' then        Exit      else if ((Pos('Quantity', AColumn.FieldName) > 0) and (Pos('Dgn', AColumn.FieldName) <=0)) or          (Pos('TotalPrice', AColumn.FieldName) > 0) then        SetTextErrorHint('该清单有子计算项,不能直接修改!')      else if (Pos('Price', AColumn.FieldName) > 0) then        SetTextErrorHint('仅最底层清单可输入单价!');      if not Allow then Exit;    end    else    begin      if SameText('OrgTotalPrice', AColumn.FieldName) or          SameText('MisTotalPrice', AColumn.FieldName) or          SameText('OthTotalPrice', AColumn.FieldName) then      begin        if not ANode.TotalPriceEnable then          SetTextErrorHint('该清单不可直接输入金额,如需直接输入金额,请先清空所有数量、单价!');      end;      if not Allow then Exit;      if SameText('Price', AColumn.FieldName) or          SameText('OrgQuantity', AColumn.FieldName) or          SameText('MisQuantity', AColumn.FieldName) or          SameText('OthQuantity', AColumn.FieldName) then      begin        if not ANode.CountPriceEnable then          SetTextErrorHint('该清单不可输入数量单价,如需使用数量×单价计算,请先清空所有直接输入的金额!');      end;      if not Allow then Exit;    end;    // 清单编号和项目节编号不可同时存在    if SameText(AValue.FieldName, 'Code') then    begin      if AValue.Owner.ValueByName('B_Code').AsString <> '' then        SetTextErrorHint('已存在清单编号,不可输入项目节编号!');    end    else if SameText(AValue.FieldName, 'B_Code') then    begin      if AValue.Owner.ValueByName('Code').AsString <> '' then        SetTextErrorHint('已存在项目节编号,不可输入清单编号!');    end    //    else if SameText(AValue.FieldName, 'Price') then    begin      if AValue.Owner.ValueByName('AddGatherTotalPrice').AsFloat <> 0 then        SetTextErrorHint('该清单已经开始计量,不可修改单价!');    end    // 变更清单不可修改0号台账数据    else if SameText(AValue.FieldName, 'OrgQuantity') or        SameText(AValue.FieldName, 'OrgTotalPrice') or        SameText(AValue.FieldName, 'MisQuantity') or        SameText(AValue.FieldName, 'MisTotalPrice') or        SameText(AValue.FieldName, 'OthQuantity') or        SameText(AValue.FieldName, 'OthTotalPrice') then    begin      if AValue.Owner.ValueByName('IsMeasureAdd').AsBoolean then        SetTextErrorHint('变更清单不可填写0号台账数量与金额');    end;    if not Allow then Exit;    if SameText('Code', AColumn.FieldName) or       SameText('B_Code', AColumn.FieldName) or       SameText('Name', AColumn.FieldName) or       SameText('Units', AColumn.FieldName) or       SameText('Price', AColumn.FieldName) then      if TBillsIDTreeNode(ANode).HasMeasure then        SetTextErrorHint('该清单已经计量,不可修改清单编号');  end;  function CheckValidData: Boolean;  begin    Result := (AValue.AsString <> Text);    if SameText(AColumn.FieldName, 'OrgQuantity') or       SameText(AColumn.FieldName, 'OrgTotalPrice') or       SameText(AColumn.FieldName, 'MisQuantity') or       SameText(AColumn.FieldName, 'MisTotalPrice') or       SameText(AColumn.FieldName, 'OthQuantity') or       SameText(AColumn.FieldName, 'OthTotalPrice') or       SameText(AColumn.FieldName, 'Price') then    begin      if (AValue.AsFloat = 0) and (Text = '') then        Result := False;    end;  end;var  vNode: TBillsIDTreeNode;begin  if not Assigned(AValue) then Exit;  // 修改后数据与原数据相同则不提交  if not CheckValidData then    Allow := False;  if not Allow then Exit;  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(ARecord.ValueByName('ID').AsInteger));  CheckLockedData;  if not Allow then Exit;  CheckNodeWritable(vNode);  if not Allow then Exit;  Text := Trim(Text);  if Pos('=', Text) = 1 then    Text := Copy(Text, 2, Length(Text) - 1);  DoCurChanged(vNode);end;function TBillsCompileData.GetActive: Boolean;begin  Result := sdvBillsCompile.Active;end;function TBillsCompileData.GetLeafXmjParentID(ABillsID: Integer): Integer;var  stnNode: TsdIDTreeNode;begin  stnNode := BillsCompileTree.FindNode(ABillsID);  Result := GetGclBillsParent(stnNode).ID;end;procedure TBillsCompileData.sdvBillsCompileAfterOpen(Sender: TObject);begin  BillsCompileTree.Active := True;end;procedure TBillsCompileData.sdvBillsCompileAfterClose(Sender: TObject);begin  BillsCompileTree.Active := False;end;procedure TBillsCompileData.ReorderChildrenCode(ANode: TsdIDTreeNode);var  iChild: Integer;  sParentCode: string;  stnChild: TsdIDTreeNode;begin  if not Assigned(ANode) then Exit;  sParentCode := ANode.Rec.ValueByName('Code').AsString;  for iChild := 0 to ANode.ChildCount - 1 do  begin    stnChild := ANode.ChildNodes[iChild];    if stnChild.Rec.ValueByName('Code').AsString <> '' then      stnChild.Rec.ValueByName('Code').AsString := sParentCode + '-' + IntToStr(iChild + 1);    ReorderChildrenCode(stnChild);  end;end;procedure TBillsCompileData.sdvBillsCompileAfterAddRecord(  ARecord: TsdDataRecord);begin  // 解锁前,新增清单为变更清单,解锁后,新增清单为0号台账清单  if TProjectData(FProjectData).ProjProperties.PhaseCount > 0 then    ARecord.ValueByName('IsMeasureAdd').AsBoolean := not TProjectData(FProjectData).CanUnlockInfo;end;procedure TBillsCompileData.DoOnAfterDeleteNode(AParent: TsdIDTreeNode);begin  if Assigned(AParent) and (AParent.ID > 0) then    Calculate(AParent.ID);end;procedure TBillsCompileData.Close;begin  sdvBillsCompile.Close;end;procedure TBillsCompileData.SetOnRecChange(const Value: TRecChangeEvent);begin  FOnRecChange := Value;end;procedure TBillsCompileData.sdvBillsCompileCurrentChanged(  ARecord: TsdDataRecord);begin  if Assigned(FOnRecChange) then    FOnRecChange(ARecord);end;procedure TBillsCompileData.ReLockBaseData;  procedure LockNodeBaseData(ANode: TsdIDTreeNode);  begin    if not Assigned(ANode) then Exit;    if ANode.Rec.ValueByName('LockedLevel').AsBoolean then      if not ANode.Rec.ValueByName('LockedInfo').AsBoolean then        ANode.Rec.ValueByName('LockedInfo').AsBoolean := True;    LockNodeBaseData(ANode.FirstChild);    LockNodeBaseData(ANode.NextSibling);  end;begin  sdvBillsCompile.AfterValueChanged := nil;  try    LockNodeBaseData(FBillsCompileTree.FirstNode);  finally    sdvBillsCompile.AfterValueChanged := sdvBillsCompileAfterValueChanged;  end;end;procedure TBillsCompileData.AddBillsFromDealBills(ARec: TsdDataRecord);var  stnParent, stnNode: TsdIDTreeNode;begin  if not CanAddGclBills then    raise Exception.Create('当前节点下不可添加工程量清单!');  stnParent := GetGclBillsParent(BillsCompileTree.Selected);  if TBillsIDTreeNode(stnParent).HasLedger or     (not stnParent.HasChildren and TBillsIDTreeNode(stnParent).HasMeasure) then    raise Exception.Create('当前节点不可添加工程量清单!');  stnNode := BillsCompileTree.Add(stnParent.ID, -1);  stnNode.Rec.ValueByName('B_Code').AsString := ARec.ValueByName('B_Code').AsString;  stnNode.Rec.ValueByName('Name').AsString := ARec.ValueByName('Name').AsString;  stnNode.Rec.ValueByName('Units').AsString := ARec.ValueByName('Units').AsString;  stnNode.Rec.ValueByName('Price').AsString := ARec.ValueByName('Price').AsString;end;procedure TBillsCompileData.CalculateMis(ABillsID: Integer);var  vNode: TBillsIDTreeNode;  iChild: Integer;begin  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(ABillsID));  if not Assigned(vNode) then Exit;  if vNode.HasChildren then  begin    for iChild := 0 to vNode.ChildCount - 1 do      CalculateMis(vNode.ChildNodes[iChild].ID);  end  else  begin    with vNode.Rec do    begin      // 数量单价模式则计算金额      if CalcType.AsInteger = 0 then        MisTotalPrice.AsFloat := TotalPriceRoundTo(MisQuantity.AsFloat * Price.AsFloat);      SetFloatValue(Quantity, QuantityRoundTo(          OrgQuantity.AsFloat + MisQuantity.AsFloat + OthQuantity.AsFloat));              // 金额与修改前不一样,则向父项增量      if MisTotalPrice.AsFloat <> CacheMisTP then      begin        UpdateParent(vNode.ParentID, MisTotalPrice.AsFloat - CacheMisTP, 'MisTotalPrice');        TotalPrice.AsFloat := TotalPriceRoundTo(            OrgTotalPrice.AsFloat + MisTotalPrice.AsFloat + OthTotalPrice.AsFloat);      end;    end;  end;  CalculateDesignPrice(vNode);end;procedure TBillsCompileData.CalculateOrg(ABillsID: Integer);var  vNode: TBillsIDTreeNode;  iChild: Integer;begin  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(ABillsID));  if not Assigned(vNode) then Exit;  if vNode.HasChildren then  begin    for iChild := 0 to vNode.ChildCount - 1 do      CalculateOrg(vNode.ChildNodes[iChild].ID);  end  else  begin    with vNode.Rec do    begin      // 数量单价模式则计算金额      if CalcType.AsInteger = 0 then        OrgTotalPrice.AsFloat := TotalPriceRoundTo(OrgQuantity.AsFloat * Price.AsFloat);      SetFloatValue(Quantity, QuantityRoundTo(          OrgQuantity.AsFloat + MisQuantity.AsFloat + OthQuantity.AsFloat));              // 金额与修改前不一样,则向父项增量      if CacheOrgTP <> OrgTotalPrice.AsFloat then      begin        UpdateParent(vNode.ParentID, OrgTotalPrice.AsFloat - CacheOrgTP, 'OrgTotalPrice');        TotalPrice.AsFloat := TotalPriceRoundTo(            OrgTotalPrice.AsFloat + MisTotalPrice.AsFloat + OthTotalPrice.AsFloat);      end;    end;  end;  CalculateDesignPrice(vNode);end;procedure TBillsCompileData.CalculateOth(ABillsID: Integer);var  vNode: TBillsIDTreeNode;  iChild: Integer;begin  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(ABillsID));  if not Assigned(vNode) then Exit;  if vNode.HasChildren then  begin    for iChild := 0 to vNode.ChildCount - 1 do      CalculateOth(vNode.ChildNodes[iChild].ID);  end  else  begin    with vNode.Rec do    begin      // 数量单价模式则计算金额      if CalcType.AsInteger = 0 then        OthTotalPrice.AsFloat := TotalPriceRoundTo(OthQuantity.AsFloat * Price.AsFloat);      SetFloatValue(Quantity, QuantityRoundTo(          OrgQuantity.AsFloat + MisQuantity.AsFloat + OthQuantity.AsFloat));              // 金额与修改前不一样,则向父项增量      if OthTotalPrice.AsFloat <> CacheOthTP then      begin        UpdateParent(vNode.ParentID, OthTotalPrice.AsFloat - CacheOthTP, 'OthTotalPrice');        TotalPrice.AsFloat := TotalPriceRoundTo(            OrgTotalPrice.AsFloat + MisTotalPrice.AsFloat+ OthTotalPrice.AsFloat);      end;    end;  end;  CalculateDesignPrice(vNode);end;function TBillsCompileData.GatherChildren(ANode: TsdIDTreeNode;  const AFieldName: string): Double;var  iChild: Integer;begin  Result := 0;  if not Assigned(ANode) then Exit;  if ANode.HasChildren and Assigned(ANode.FirstChild) then  begin    Result := 0;    for iChild := 0 to ANode.ChildCount - 1 do      Result := Result + ANode.Rec.ValueByName(AFieldName).AsFloat;    Result := TotalPriceRoundTo(Result);  end  else    if Assigned(ANode.Rec) and Assigned(ANode.Rec.ValueByName(AFieldName)) then      Result := ANode.Rec.ValueByName(AFieldName).AsFloat;end;procedure TBillsCompileData.UpdateParent(ABillsID: Integer;  ADifferTotalPrice: Double; const AFieldName: string);var  vNode: TBillsIDTreeNode;begin  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(ABillsID));  if not Assigned(vNode) then Exit;  with vNode.Rec do  begin    ValueByName(AFieldName).AsFloat := TotalPriceRoundTo(        ValueByName(AFieldName).AsFloat + ADifferTotalPrice);    TotalPrice.AsFloat := TotalPriceRoundTo(TotalPrice.AsFloat + ADifferTotalPrice);  end;  CalculateDesignPrice(vNode);  UpdateParent(vNode.ParentID, ADifferTotalPrice, AFieldName);end;procedure TBillsCompileData.CalculateTotal(ABillsID: Integer);begin  CalculateOrg(ABillsID);  CalculateMis(ABillsID);  CalculateOth(ABillsID);end;procedure TBillsCompileData.CalculateBills(ANode: TsdIDTreeNode);var  iChild: Integer;begin  if not Assigned(ANode) then Exit;  if ANode.HasChildren then  begin    for iChild := 0 to ANode.ChildCount - 1 do      CalculateBills(ANode.ChildNodes[iChild]);    GatherNode(TBillsIDTreeNode(ANode));  end  else    CalculateLeaf(TBillsIDTreeNode(ANode));end;procedure TBillsCompileData.CalculateLeaf(ANode: TBillsIDTreeNode);begin  if not Assigned(ANode) or ANode.HasChildren then Exit;  with ANode.Rec do  begin    // 分项    if CalcType.AsFloat = 0 then    begin      OrgTotalPrice.AsFloat := TotalPriceRoundTo(Price.AsFloat * OrgQuantity.AsFloat);      MisTotalPrice.AsFloat := TotalPriceRoundTo(Price.AsFloat * MisQuantity.AsFloat);      OthTotalPrice.AsFloat := TotalPriceRoundTo(Price.AsFloat * OthQuantity.AsFloat);    end;    // 汇总    Quantity.AsFloat := QuantityRoundTo(        OrgQuantity.AsFloat + MisQuantity.AsFloat + OthQuantity.AsFloat);    TotalPrice.AsFloat := TotalPriceRoundTo(        OrgTotalPrice.AsFloat + MisTotalPrice.AsFloat + OthTotalPrice.AsFloat);  end;  CalculateDesignPrice(ANode);end;procedure TBillsCompileData.GatherNode(ANode: TBillsIDTreeNode);var  iChild: Integer;  fOrg, fMis, fOth: Double;  vChild: TBillsIDTreeNode;begin  fOrg := 0;  fMis := 0;  fOth := 0;  for iChild := 0 to ANode.ChildCount - 1 do  begin    vChild := TBillsIDTreeNode(ANode.ChildNodes[iChild]);    fOrg := fOrg + vChild.Rec.OrgTotalPrice.AsFloat;    fMis := fMis + vChild.Rec.MisTotalPrice.AsFloat;    fOth := fOth + vChild.Rec.OthTotalPrice.AsFloat;  end;  ANode.Rec.OrgTotalPrice.AsFloat := TotalPriceRoundTo(fOrg);  ANode.Rec.MisTotalPrice.AsFloat := TotalPriceRoundTo(fMis);  ANode.Rec.OthTotalPrice.AsFloat := TotalPriceRoundTo(fOth);  ANode.Rec.TotalPrice.AsFloat := TotalPriceRoundTo(fOrg + fMis + fOth);  CalculateDesignPrice(ANode);end;procedure TBillsCompileData.Calculate(ABillsID: Integer);  procedure UpdateParent(ANode: TBillsIDTreeNode; ADifferOrg, ADifferMis, ADifferOth: Double);  begin    if not Assigned(ANode) then Exit;    with ANode.Rec do    begin      OrgTotalPrice.AsFloat := TotalPriceRoundTo(OrgTotalPrice.AsFloat + ADifferOrg);      MisTotalPrice.AsFloat := TotalPriceRoundTo(MisTotalPrice.AsFloat + ADifferMis);      OthTotalPrice.AsFloat := TotalPriceRoundTo(OthTotalPrice.AsFloat + ADifferOth);      TotalPrice.AsFloat := TotalPriceRoundTo(          TotalPrice.AsFloat + ADifferOrg + ADifferMis + ADifferOth);      if DgnQuantity1.AsFloat <> 0 then        DgnPrice.AsFloat := PriceRoundTo(TotalPrice.AsFloat/DgnQuantity1.AsFloat);    end;    UpdateParent(TBillsIDTreeNode(ANode.Parent), ADifferOrg, ADifferMis, ADifferOth);  end;var  vNode: TBillsIDTreeNode;  iChild: Integer;  fOrg, fMis, fOth: Double;begin  vNode := TBillsIDTreeNode(BillsCompileTree.FindNode(ABillsID));  if not Assigned(vNode) then Exit;  fOrg := vNode.Rec.OrgTotalPrice.AsFloat;  fMis := vNode.Rec.MisTotalPrice.AsFloat;  fOth := vNode.Rec.OthTotalPrice.AsFloat;  CalculateBills(vNode);  fOrg := vNode.Rec.OrgTotalPrice.AsFloat - fOrg;  fMis := vNode.Rec.MisTotalPrice.AsFloat - fMis;  fOth := vNode.Rec.OthTotalPrice.AsFloat - fOth;  UpdateParent(TBillsIDTreeNode(vNode.Parent), fOrg, fMis, fOth);end;procedure TBillsCompileData.CalculateDesignPrice(ANode: TBillsIDTreeNode);begin  if ANode.Rec.DgnQuantity1.AsFloat <> 0 then    ANode.Rec.DgnPrice.AsFloat := PriceRoundTo(        ANode.Rec.TotalPrice.AsFloat/ANode.Rec.DgnQuantity1.AsFloat)  else    ANode.Rec.DgnPrice.Clear;end;procedure TBillsCompileData.ExpandPegXmjNode;  function HasPegChild(ANode: TBillsIDTreeNode): Boolean;  var    NextNode: TBillsIDTreeNode;  begin    Result := False;    NextNode := TBillsIDTreeNode(ANode.NextNode);    while ((NextNode.MajorIndex - ANode.MajorIndex) <= ANode.PosterityCount) do    begin      if CheckPeg(NextNode.Rec.Name.AsString) then      begin        Result := True;        Break;      end;      NextNode := TBillsIDTreeNode(NextNode.NextNode);    end;  end;  function HasGclChild(ANode: TBillsIDTreeNode): Boolean;  var    vChild: TBillsIDTreeNode;  begin    Result := True;    vChild := TBillsIDTreeNode(ANode.FirstChild);    while Assigned(vChild) and not Result do    begin      if vChild.Rec.B_Code.AsString <> '' then        Result := False;      vChild := TBillsIDTreeNode(vChild.NextSibling);    end;  end;var  iIndex: Integer;  vNode: TBillsIDTreeNode;begin  for iIndex := 0 to BillsCompileTree.Count - 1 do  begin    vNode := TBillsIDTreeNode(BillsCompileTree.Items[iIndex]);    if vNode.HasChildren then      vNode.Expanded := HasPegChild(vNode) or not HasGclChild(vNode);  end;end;end.
 |