datasource.inc 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676
  1. { ---------------------------------------------------------------------
  2. TDatalink
  3. ---------------------------------------------------------------------}
  4. Constructor TDataLink.Create;
  5. begin
  6. Inherited Create;
  7. FBufferCount:=1;
  8. FDataSource := nil;
  9. FDatasourceFixed:=False;
  10. end;
  11. Destructor TDataLink.Destroy;
  12. begin
  13. Factive:=False;
  14. FEditing:=False;
  15. FDataSourceFixed:=False;
  16. DataSource:=Nil;
  17. Inherited Destroy;
  18. end;
  19. Procedure TDataLink.ActiveChanged;
  20. begin
  21. end;
  22. Procedure TDataLink.CheckActiveAndEditing;
  23. Var
  24. B : Boolean;
  25. begin
  26. B:=Assigned(DataSource) and (DataSource.State<>dsInactive);
  27. If B<>FActive then
  28. begin
  29. FActive:=B;
  30. ActiveChanged;
  31. end;
  32. B:=Assigned(DataSource) and (DataSource.State in dsEditModes) and Not FReadOnly;
  33. If B<>FEditing Then
  34. begin
  35. FEditing:=B;
  36. EditingChanged;
  37. end;
  38. end;
  39. Procedure TDataLink.CheckBrowseMode;
  40. begin
  41. end;
  42. Function TDataLink.CalcFirstRecord(Index : Integer) : Integer;
  43. { recalculates index of first record in buffer,
  44. and returns number of record scrolled.}
  45. Var
  46. A, // Index of active record in buffer
  47. F, // Index of current first record in buffer
  48. L : Integer; // index of Last (for us) record in buffer;
  49. begin
  50. Result:=0;
  51. A:=DataSource.DataSet.FActiveRecord;
  52. F:=FFirstRecord+Index;
  53. L:=F+FBufferCount-1;
  54. If A>L Then
  55. Result:=A-L
  56. else If A<F Then
  57. Result:=A-F;
  58. FFirstRecord:=F+Result;
  59. {$IfDef dsDebug}
  60. WriteLn('CalcFirstRecord: FFirstRecord=',FFirstRecord,' Scrolled=',Result);
  61. {$Endif}
  62. end;
  63. Procedure TDataLink.CalcRange;
  64. var
  65. aMax, aMin: integer;
  66. begin
  67. aMin:= DataSet.FActiveRecord - FBufferCount + 1;
  68. If aMin < 0 Then aMin:= 0;
  69. aMax:= Dataset.FBufferCount - FBufferCount;
  70. If aMax < 0 then aMax:= 0;
  71. If aMax>DataSet.FActiveRecord Then aMax:=DataSet.FActiveRecord;
  72. If FFirstRecord < aMin Then FFirstRecord:= aMin;
  73. If FFirstrecord > aMax Then FFirstRecord:= aMax;
  74. If (FfirstRecord<>0) And
  75. (DataSet.FActiveRecord - FFirstRecord < FBufferCount -1) Then
  76. Dec(FFirstRecord, 1);
  77. end;
  78. Procedure TDataLink.DataEvent(Event: TDataEvent; Info: Ptrint);
  79. begin
  80. Case Event of
  81. deFieldChange, deRecordChange:
  82. If Not FUpdatingRecord then
  83. RecordChanged(TField(Info));
  84. deDataSetChange: begin
  85. SetActive(DataSource.DataSet.Active);
  86. CalcFirstRecord(Info);
  87. DatasetChanged;
  88. end;
  89. deDataSetScroll: DatasetScrolled(CalcFirstRecord(Info));
  90. deLayoutChange: begin
  91. CalcFirstRecord(Info);
  92. LayoutChanged;
  93. end;
  94. deUpdateRecord: UpdateRecord;
  95. deUpdateState: CheckActiveAndEditing;
  96. deCheckBrowseMode: CheckBrowseMode;
  97. deFocusControl: FocusControl(TFieldRef(Info));
  98. end;
  99. end;
  100. Procedure TDataLink.DataSetChanged;
  101. begin
  102. RecordChanged(Nil);
  103. end;
  104. Procedure TDataLink.DataSetScrolled(Distance: Integer);
  105. begin
  106. DataSetChanged;
  107. end;
  108. Procedure TDataLink.EditingChanged;
  109. begin
  110. end;
  111. Procedure TDataLink.FocusControl(Field: TFieldRef);
  112. begin
  113. end;
  114. Function TDataLink.GetActiveRecord: Integer;
  115. begin
  116. Result:=Dataset.FActiveRecord - FFirstRecord;
  117. end;
  118. Function TDatalink.GetDataSet : TDataset;
  119. begin
  120. If Assigned(Datasource) then
  121. Result:=DataSource.DataSet
  122. else
  123. Result:=Nil;
  124. end;
  125. Function TDataLink.GetBOF: Boolean;
  126. begin
  127. Result:=DataSet.BOF
  128. end;
  129. Function TDataLink.GetBufferCount: Integer;
  130. begin
  131. Result:=FBufferCount;
  132. end;
  133. Function TDataLink.GetEOF: Boolean;
  134. begin
  135. Result:=DataSet.EOF
  136. end;
  137. Function TDataLink.GetRecordCount: Integer;
  138. begin
  139. Result:=Dataset.FRecordCount;
  140. If Result>BufferCount then
  141. Result:=BufferCount;
  142. end;
  143. Procedure TDataLink.LayoutChanged;
  144. begin
  145. DataSetChanged;
  146. end;
  147. Function TDataLink.MoveBy(Distance: Integer): Integer;
  148. begin
  149. Result:=DataSet.MoveBy(Distance);
  150. end;
  151. Procedure TDataLink.RecordChanged(Field: TField);
  152. begin
  153. end;
  154. Procedure TDataLink.SetActiveRecord(Value: Integer);
  155. begin
  156. {$ifdef dsdebug}
  157. Writeln('Datalink. Setting active record to ',Value,' with firstrecord ',ffirstrecord);
  158. {$endif}
  159. Dataset.FActiveRecord:=Value + FFirstRecord;
  160. end;
  161. Procedure TDataLink.SetBufferCount(Value: Integer);
  162. begin
  163. If FBufferCount<>Value then
  164. begin
  165. FBufferCount:=Value;
  166. if Active then begin
  167. DataSet.RecalcBufListSize;
  168. CalcRange;
  169. end;
  170. end;
  171. end;
  172. procedure TDataLink.SetActive(AActive: Boolean);
  173. begin
  174. if Active <> AActive then
  175. begin
  176. FActive := AActive;
  177. // !!!: Set internal state
  178. ActiveChanged;
  179. end;
  180. end;
  181. Procedure TDataLink.SetDataSource(Value : TDatasource);
  182. begin
  183. if not FDataSourceFixed then
  184. begin
  185. if Assigned(DataSource) then
  186. Begin
  187. DataSource.UnregisterDatalink(Self);
  188. FDataSource := nil;
  189. CheckActiveAndEditing;
  190. End;
  191. FDataSource := Value;
  192. if Assigned(DataSource) then
  193. begin
  194. DataSource.RegisterDatalink(Self);
  195. CheckActiveAndEditing;
  196. End;
  197. end;
  198. end;
  199. Procedure TDatalink.SetReadOnly(Value : Boolean);
  200. begin
  201. If FReadOnly<>Value then
  202. begin
  203. FReadOnly:=Value;
  204. CheckActiveAndEditing;
  205. end;
  206. end;
  207. Procedure TDataLink.UpdateData;
  208. begin
  209. end;
  210. Function TDataLink.Edit: Boolean;
  211. begin
  212. If Not FReadOnly then
  213. DataSource.Edit;
  214. // Triggered event will set FEditing
  215. Result:=FEditing;
  216. end;
  217. Procedure TDataLink.UpdateRecord;
  218. begin
  219. FUpdatingRecord:=True;
  220. Try
  221. UpdateData;
  222. finally
  223. FUpdatingRecord:=False;
  224. end;
  225. end;
  226. { ---------------------------------------------------------------------
  227. TDetailDataLink
  228. ---------------------------------------------------------------------}
  229. Function TDetailDataLink.GetDetailDataSet: TDataSet;
  230. begin
  231. Result := nil;
  232. end;
  233. { ---------------------------------------------------------------------
  234. TMasterDataLink
  235. ---------------------------------------------------------------------}
  236. constructor TMasterDataLink.Create(ADataSet: TDataSet);
  237. begin
  238. inherited Create;
  239. FDetailDataSet:=ADataSet;
  240. FFields:=TList.Create;
  241. end;
  242. destructor TMasterDataLink.Destroy;
  243. begin
  244. FFields.Free;
  245. inherited Destroy;
  246. end;
  247. Procedure TMasterDataLink.ActiveChanged;
  248. begin
  249. FFields.Clear;
  250. if Active then
  251. try
  252. DataSet.GetFieldList(FFields, FFieldNames);
  253. except
  254. FFields.Clear;
  255. raise;
  256. end;
  257. if FDetailDataSet.Active and not (csDestroying in FDetailDataSet.ComponentState) then
  258. if Active and (FFields.Count > 0) then
  259. DoMasterChange
  260. else
  261. DoMasterDisable;
  262. end;
  263. Procedure TMasterDataLink.CheckBrowseMode;
  264. begin
  265. if FDetailDataSet.Active then FDetailDataSet.CheckBrowseMode;
  266. end;
  267. Function TMasterDataLink.GetDetailDataSet: TDataSet;
  268. begin
  269. Result := FDetailDataSet;
  270. end;
  271. Procedure TMasterDataLink.LayoutChanged;
  272. begin
  273. ActiveChanged;
  274. end;
  275. Procedure TMasterDataLink.RecordChanged(Field: TField);
  276. begin
  277. if (DataSource.State <> dsSetKey) and FDetailDataSet.Active and
  278. (FFields.Count > 0) and ((Field = nil) or
  279. (FFields.IndexOf(Field) >= 0)) then
  280. DoMasterChange;
  281. end;
  282. procedure TMasterDatalink.SetFieldNames(const Value: string);
  283. begin
  284. if FFieldNames <> Value then
  285. begin
  286. FFieldNames := Value;
  287. ActiveChanged;
  288. end;
  289. end;
  290. Procedure TMasterDataLink.DoMasterDisable;
  291. begin
  292. if Assigned(FOnMasterDisable) then
  293. FOnMasterDisable(Self);
  294. end;
  295. Procedure TMasterDataLink.DoMasterChange;
  296. begin
  297. If Assigned(FOnMasterChange) then
  298. FOnMasterChange(Self);
  299. end;
  300. { ---------------------------------------------------------------------
  301. TMasterDataLink
  302. ---------------------------------------------------------------------}
  303. constructor TMasterParamsDataLink.Create(ADataSet: TDataSet);
  304. Var
  305. P : TParams;
  306. begin
  307. inherited Create(ADataset);
  308. If (ADataset<>Nil) then
  309. begin
  310. P:=TParams(GetObjectProp(ADataset,'Params',TParams));
  311. if (P<>Nil) then
  312. Params:=P;
  313. end;
  314. end;
  315. Procedure TMasterParamsDataLink.SetParams(AVAlue : TParams);
  316. Var
  317. N : String;
  318. begin
  319. N:=DetailDataset.Name;
  320. FParams:=AValue;
  321. If (AValue<>Nil) then
  322. RefreshParamNames;
  323. end;
  324. Procedure TMasterParamsDataLink.RefreshParamNames;
  325. Var
  326. FN : String;
  327. DS : TDataset;
  328. F : TField;
  329. I : Integer;
  330. begin
  331. FN:='';
  332. DS:=Dataset;
  333. If Assigned(FParams) then
  334. begin
  335. F:=Nil;
  336. For I:=0 to FParams.Count-1 do
  337. begin
  338. If Assigned(DS) then
  339. F:=DS.FindField(FParams[i].Name);
  340. If (Not Assigned(DS)) or (F<>Nil) then
  341. begin
  342. If (FN<>'') then
  343. FN:=FN+';';
  344. FN:=FN+FParams[i].Name;
  345. end;
  346. end;
  347. end;
  348. FieldNames:=FN;
  349. end;
  350. Procedure TMasterParamsDataLink.CopyParamsFromMaster(CopyBound : Boolean);
  351. begin
  352. if Assigned(FParams) then
  353. FParams.CopyParamValuesFromDataset(Dataset,CopyBound);
  354. end;
  355. Procedure TMasterParamsDataLink.DoMasterDisable;
  356. begin
  357. Inherited;
  358. If Assigned(DetailDataset) and DetailDataset.Active then
  359. DetailDataset.Close;
  360. end;
  361. Procedure TMasterParamsDataLink.DoMasterChange;
  362. begin
  363. Inherited;
  364. if Assigned(Params) and Assigned(DetailDataset) and DetailDataset.Active then
  365. begin
  366. DetailDataset.Close;
  367. DetailDataset.Open;
  368. end;
  369. end;
  370. { ---------------------------------------------------------------------
  371. TDatasource
  372. ---------------------------------------------------------------------}
  373. Constructor TDataSource.Create(AOwner: TComponent);
  374. begin
  375. Inherited Create(AOwner);
  376. FDatalinks := TList.Create;
  377. FEnabled := True;
  378. FAutoEdit := True;
  379. end;
  380. Destructor TDataSource.Destroy;
  381. begin
  382. FOnStateCHange:=Nil;
  383. Dataset:=Nil;
  384. With FDataLinks do
  385. While Count>0 do
  386. TDatalink(Items[Count - 1]).DataSource:=Nil;
  387. FDatalinks.Free;
  388. inherited Destroy;
  389. end;
  390. Procedure TDatasource.Edit;
  391. begin
  392. If (State=dsBrowse) and AutoEdit Then
  393. Dataset.Edit;
  394. end;
  395. Function TDataSource.IsLinkedTo(ADataSet: TDataSet): Boolean;
  396. begin
  397. Result:=False;
  398. end;
  399. procedure TDatasource.DistributeEvent(Event: TDataEvent; Info: Ptrint);
  400. Var
  401. i : Longint;
  402. begin
  403. With FDatalinks do
  404. begin
  405. For I:=0 to Count-1 do
  406. With TDatalink(Items[i]) do
  407. If Not VisualControl Then
  408. DataEvent(Event,Info);
  409. For I:=0 to Count-1 do
  410. With TDatalink(Items[i]) do
  411. If VisualControl Then
  412. DataEvent(Event,Info);
  413. end;
  414. end;
  415. procedure TDatasource.RegisterDataLink(DataLink: TDataLink);
  416. begin
  417. FDatalinks.Add(DataLink);
  418. if Assigned(DataSet) then
  419. DataSet.RecalcBufListSize;
  420. end;
  421. procedure TDatasource.SetDataSet(ADataSet: TDataSet);
  422. begin
  423. If FDataset<>Nil Then
  424. Begin
  425. FDataset.UnRegisterDataSource(Self);
  426. FDataSet:=nil;
  427. ProcessEvent(deUpdateState,0);
  428. End;
  429. If ADataset<>Nil Then
  430. begin
  431. ADataset.RegisterDatasource(Self);
  432. FDataSet:=ADataset;
  433. ProcessEvent(deUpdateState,0);
  434. End;
  435. end;
  436. procedure TDatasource.SetEnabled(Value: Boolean);
  437. begin
  438. FEnabled:=Value;
  439. end;
  440. Procedure TDatasource.DoDataChange (Info : Pointer);
  441. begin
  442. If Assigned(OnDataChange) Then
  443. OnDataChange(Self,TField(Info));
  444. end;
  445. Procedure TDatasource.DoStateChange;
  446. begin
  447. If Assigned(OnStateChange) Then
  448. OnStateChange(Self);
  449. end;
  450. Procedure TDatasource.DoUpdateData;
  451. begin
  452. If Assigned(OnUpdateData) Then
  453. OnUpdateData(Self);
  454. end;
  455. procedure TDatasource.UnregisterDataLink(DataLink: TDataLink);
  456. begin
  457. FDatalinks.Remove(Datalink);
  458. If Dataset<>Nil then
  459. DataSet.RecalcBufListSize;
  460. //Dataset.SetBufListSize(DataLink.BufferCount);
  461. end;
  462. procedure TDataSource.ProcessEvent(Event : TDataEvent; Info : Ptrint);
  463. Const
  464. OnDataChangeEvents = [deRecordChange, deDataSetChange, deDataSetScroll,
  465. deLayoutChange,deUpdateState];
  466. Var
  467. NeedDataChange : Boolean;
  468. FLastState : TdataSetState;
  469. begin
  470. // Special UpdateState handling.
  471. If Event=deUpdateState then
  472. begin
  473. NeedDataChange:=(FState=dsInactive);
  474. FLastState:=FState;
  475. If Assigned(Dataset) then
  476. FState:=Dataset.State
  477. else
  478. FState:=dsInactive;
  479. // Don't do events if nothing changed.
  480. If FState=FlastState then
  481. exit;
  482. end
  483. else
  484. NeedDataChange:=True;
  485. DistributeEvent(Event,Info);
  486. // Extra handlers
  487. If Not (csDestroying in ComponentState) then
  488. begin
  489. If (Event=deUpdateState) then
  490. DoStateChange;
  491. If (Event in OnDataChangeEvents) and
  492. NeedDataChange Then
  493. DoDataChange(Nil);
  494. If (Event = deFieldChange) Then
  495. DoDataCHange(Pointer(Info));
  496. If (Event=deUpdateRecord) then
  497. DoUpdateData;
  498. end;
  499. end;