📄 datamod.pas
字号:
CustByOrd.Close;
end;
procedure TMastData.OrdersBeforeDelete(DataSet: TDataSet);
begin
if not Confirm('Delete order and line items?') then
Abort
else
DeleteItems;
end;
procedure TMastData.OrdersBeforeInsert(DataSet: TDataSet);
begin
if Orders.State in dsEditModes then
begin
if Confirm('An order is being processed. Save changes and start a new one?') then
Orders.Post
else
Abort;
end;
FItemNo := 1;
end;
procedure TMastData.OrdersBeforeOpen(DataSet: TDataSet);
begin
CustByComp.Open;
CustByOrd.Open;
Cust.Open;
Emps.Open;
Items.Open;
end;
{ Calculate the order's tax totals and amount due }
procedure TMastData.OrdersCalcFields(DataSet: TDataSet);
begin
OrdersTaxTotal.Value := OrdersItemsTotal.Value * (OrdersTaxRate.Value / 100);
OrdersAmountDue.Value := OrdersItemsTotal.Value + OrdersTaxTotal.Value +
OrdersFreight.Value - OrdersAmountPaid.Value;
end;
{ Inititializes the record values as a result of an Orders.Insert. }
procedure TMastData.OrdersNewRecord(DataSet: TDataSet);
begin
{ Get the Next Order Value from the NextOrd Table }
with NextOrd do
begin
Open;
try
Edit;
OrdersOrderNo.Value := NextOrdNewKey.Value;
NextOrdNewKey.Value := NextOrdNewKey.Value + 1;
Post;
finally
Close;
end;
end;
OrdersSaleDate.Value := Date;
OrdersShipVia.Value := 'UPS';
OrdersTerms.Value := 'net 30';
OrdersPaymentMethod.Value := 'Check';
OrdersItemsTotal.Value := 0;
OrdersTaxRate.Value := 0;
OrdersFreight.Value := 0;
OrdersAmountPaid.Value := 0;
end;
procedure TMastData.ItemsAfterDelete(DataSet: TDataSet);
begin
UpdateTotals;
end;
{ Update the order totals and the Parts table }
procedure TMastData.ItemsAfterPost(DataSet: TDataSet);
{ Reduce/increase Parts table's OnOrder field }
procedure UpdateParts(PartNo: Double; Qty: Longint);
begin
if (PartNo > 0) and (Qty <> 0) then
try
if not Parts.Locate('PartNo', PartNo, []) then Abort;
Parts.Edit;
PartsOnOrder.Value := PartsOnOrder.Value + Qty;
Parts.Post;
except
on E: Exception do
ShowMessage(Format('Error updating parts table for PartNo: %d', [PartNo]));
end;
end;
begin
{ Maintain next available item number }
Inc(FItemNo);
UpdateTotals;
if not ((PrevPartNo = ItemsPartNo.Value) and (PrevQty = ItemsQty.Value)) then
begin
{ Reduce previous Part#'s OnOrder field by previous Qty }
UpdateParts(PrevPartNo, -PrevQty);
{ Increase new Part#'s OnOrder field by previous Qty }
UpdateParts(ItemsPartNo.Value, ItemsQty.Value);
end;
end;
{ When a change to the detail table affects a field in the master, always make
sure the master (orders) table is in edit or insert mode before allowing the
detail table to be modified. }
procedure TMastData.EnsureOrdersEdit(DataSet: TDataSet);
begin
Orders.Edit;
end;
{ Remember previous PartNo and Qty for updating Parts.OnOrder after post.
When a change to the detail table affects a field in the master, always make
sure the master table is in edit or insert mode before allowing the
detail table to be modified. }
procedure TMastData.ItemsBeforeEdit(DataSet: TDataSet);
begin
Orders.Edit;
PrevPartNo := ItemsPartNo.Value;
PrevQty := ItemsQty.Value;
end;
{ Make sure the Parts table opens before the Items table, since there are
lookups which depend on it. }
procedure TMastData.ItemsBeforeOpen(DataSet: TDataSet);
begin
Parts.Open;
end;
{ Complete the item's key by initializing its NextItemNo field }
procedure TMastData.ItemsBeforePost(DataSet: TDataSet);
begin
ItemsItemNo.Value := FItemNo;
end;
{ Lookup PartNo info for the item; calculate its extended price }
procedure TMastData.ItemsCalcFields(DataSet: TDataSet);
begin
ItemsExtPrice.Value := ItemsQty.Value *
ItemsSellPrice.Value * (100 - ItemsDiscount.Value) / 100;
end;
{ New item. Zero the "prev" buckets, initialize the key }
procedure TMastData.ItemsNewRecord(DataSet: TDataSet);
begin
PrevPartNo := 0;
PrevQty := 0;
ItemsOrderNo.Value := OrdersOrderNo.Value;
ItemsQty.Value := 1;
ItemsDiscount.Value := 0;
end;
{ Concatenate last name + first name for the order's SoldBy DBLookupCombo }
procedure TMastData.EmpsCalcFields(DataSet: TDataSet);
begin
EmpsFullName.Value := Format('%s, %s', [EmpsLastName.Value, EmpsFirstName.Value]);
end;
procedure TMastData.DeleteItems;
begin
DeletingItems := True; { suppress recalc of totals during delete }
Items.DisableControls; { for faster table traversal }
try
Items.First;
while not Items.EOF do Items.Delete;
finally
DeletingItems := False;
Items.EnableControls; { always re-enable controls after disabling }
end;
end;
{ Steps through Items and gathers sum of ExtPrice. After OrdersItemsTotal
is calculated, OrdersCalcFields is automatically called (which
updates other calculated fields. }
procedure TMastData.UpdateTotals;
var
TempTotal: Extended;
PrevRecord: TBookmark;
begin
if DeletingItems then Exit; { don't calculate if deleting all items }
PrevRecord := Items.GetBookmark; { returns nil if table is empty }
try
Items.DisableControls;
Items.First;
TempTotal := 0; { use temp for efficiency }
while not Items.EOF do
begin
TempTotal := TempTotal + ItemsExtPrice.Value;
Items.Next;
end;
OrdersItemsTotal.Value := TempTotal;
finally
Items.EnableControls;
if PrevRecord <> nil then
begin
Items.GoToBookmark(PrevRecord);
Items.FreeBookmark(PrevRecord);
end;
end;
end;
procedure TMastData.OrdersCustNoChange(Sender: TField);
var
TaxRate: Variant;
begin
OrdersShipToContact.Value := '';
OrdersShipToPhone.Value := '';
OrdersShipToAddr1.Value := '';
OrdersShipToAddr2.Value := '';
OrdersShipToCity.Value := '';
OrdersShipToState.Value := '';
OrdersShipToZip.Value := '';
OrdersShipToCountry.Value := '';
TaxRate := Cust.Lookup('CustNo', OrdersCustNo.Value, 'TaxRate');
if not VarIsNull(TaxRate) then
OrdersTaxRate.Value := TaxRate;
end;
{ Alternatively, could set the Qty field's Min and Max values in code
or in the Object Inspector. }
procedure TMastData.ItemsQtyValidate(Sender: TField);
begin
if ItemsQty.Value < 1 then
raise Exception.Create('Must specify quantity');
end;
{ Alternatively, could set the Freight field's Min and Max values in code
or in the Object Inspector. }
procedure TMastData.OrdersFreightValidate(Sender: TField);
begin
if OrdersFreight.Value < 0 then
raise Exception.Create('Freight cannot be less than zero');
end;
procedure TMastData.ItemsPartNoValidate(Sender: TField);
begin
if not Parts.Locate('PartNo', ItemsPartNo.Value, []) then
raise Exception.Create('You must specify a valid PartNo');
end;
procedure TMastData.OrdersSaleDateValidate(Sender: TField);
begin
if OrdersSaleDate.Value > Now then
raise Exception.Create('Cannot enter a future date');
end;
{ Browse Customers }
procedure TMastData.CustBeforeOpen(DataSet: TDataSet);
begin
OrdByCust.Open;
end;
procedure TMastData.OrdByCustCalcFields(DataSet: TDataSet);
begin
OrdByCustAmountDue.Value := OrdByCustItemsTotal.Value +
OrdByCustItemsTotal.Value * OrdByCustTaxRate.Value / 100 +
OrdByCustFreight.Value - OrdByCustAmountPaid.Value;
end;
{ Get the next available customer number from the NextCust table }
procedure TMastData.CustBeforePost(DataSet: TDataSet);
begin
if Cust.State = dsInsert then
with NextCust do
begin
Open;
try
Edit;
CustCustNo.Value := NextCustNewCust.Value;
NextCustNewCust.Value := NextCustNewCust.Value + 1;
Post;
finally
Close;
end;
end;
end;
function TMastData.DataSetApplyUpdates(DataSet: TDataSet; Apply: Boolean): Boolean;
begin
Result := True;
with TDBDataSet(DataSet) do
begin
if (State in dsEditModes) or UpdatesPending then
begin
if Apply then
begin
Database.ApplyUpdates([DataSet as TDBDataSet]);
{ Always call CancelUpdates to remove any discard changes }
CancelUpdates;
end
else
begin
if (MessageDlg('Unsaved changes, exit anyway?', mtConfirmation,
[mbYes, mbCancel], 0) = mrYes) then
CancelUpdates
else
Result := False;
end;
end;
end;
end;
{ Determine the next available ItemNo for this order }
procedure TMastData.OrdersBeforeEdit(DataSet: TDataSet);
begin
LastItemQuery.Close;
LastItemQuery.Open;
{ SQL servers return Null for some aggregates if no items are present }
with LastItemQuery.Fields[0] do
if IsNull then FItemNo := 1
else FItemNo := AsInteger + 1;
end;
procedure TMastData.EditUpdateError(DataSet: TDataSet; E: EDatabaseError;
UpdateKind: TUpdateKind; var UpdateAction: TUpdateAction);
var
Key: Variant;
const
UpdErrMsg = '%s.'#13#10'Discard the edits to %S %S and continue updating?';
begin
if UpdateKind = ukDelete then
Key := Dataset.Fields[0].OldValue else
Key := Dataset.Fields[0].NewValue;
if MessageDlg(Format(UpdErrMsg, [E.Message, DataSet.Fields[0].DisplayLabel, Key]),
mtConfirmation, [mbYes, mbCancel], 0) = mrYes then
UpdateAction := uaSkip else
UpdateAction := uaAbort;
end;
procedure TControllerMastData.DoCommand(Command: string; const args: string='');
begin
if Command = CMD_USE_LOCALDB then
TMastData.GetInstance.UseLocalData
else if Command = CMD_USE_REMOTEDB then
TMastData.GetInstance.UseRemoteData
else if Command = CMD_BACKORDERS then
begin
if args = 'on' then
TMastData.getInstance.PartsSource.Dataset := TMastData.getInstance.Parts
else
try
TMastData.getInstance.PartsQuery.Close;
TMastData.getInstance.PartsQuery.Open;
TMastData.getInstance.PartsSource.Dataset := TMastData.getInstance.PartsQuery;
except
TMastData.getInstance.PartsSource.Dataset := TMastData.getInstance.Parts;
raise;
end;
end;
end;
initialization
ControlCenter.RegController(TControllerMastData.Create);
end.
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -