| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466 |
- unit fpobserver;
- {$mode objfpc}{$H+}
- {$interfaces corba}
- interface
- uses
- Classes, SysUtils, typinfo, contnrs;
- Type
- TObservedHook = Class(TObject,IFPObserved)
- Protected
- FObservers : TFPList;
- FSender : TObject;
- Public
- // ASender will be the default sender.
- Constructor CreateSender(ASender : TObject);
- Destructor Destroy; override;
- Procedure FPOAttachObserver(AObserver : TObject);
- Procedure FPODetachObserver(AObserver : TObject);
- Procedure Changed;
- Procedure AddItem(AItem : TObject);
- Procedure DeleteItem(AItem : TObject);
- Procedure CustomNotify(Data : Pointer = Nil);
- Procedure FPONotifyObservers(ASender : TObject; AOperation : TFPObservedOperation; Data : Pointer);
- Property Sender : TObject Read FSender;
- end;
- // EObserver = Class(Exception);
- { TBaseMediator }
- TMediatingEvent = Procedure(Sender : TObject; var Handled : Boolean) of object;
- TBaseMediator = Class(TComponent,IFPObserver)
- private
- FActive: Boolean;
- FOnObjectToView: TMediatingEvent;
- FOnViewToObject: TMediatingEvent;
- FReadOnly: Boolean;
- FTransferring : Boolean;
- FSubjectPropertyName: String;
- FSubject: TObject;
- FValueList: TObjectList;
- FViewPropertyName: String;
- procedure SetReadOnly(const AValue: Boolean);
- procedure SetValueList(const AValue: TObjectList);
- procedure SetViewPropertyName(const AValue: String); Virtual;
- Protected
- // Should return true (Default) if ViewPropertyName is published
- Class Function PublishedViewProperty : Boolean; virtual;
- // Should return true (Default) if SubjectPropertyName is published
- Class Function PublishedSubjectProperty : Boolean; virtual;
- // Set active. Descendents (such as list mediators) can override this.
- procedure SetActive(const AValue: Boolean); virtual;
- // set subject. Attaches observer and calls MaybeObjectToView
- procedure SetSubject(const AValue: TObject); virtual;
- // set subjectpropertyname. Checks if it exists, and calls MaybeObjectToView
- procedure SetSubjectPropertyName(const AValue: String); virtual;
- // Can be used in descendents to respond to onchange events
- Procedure ViewChangedHandler(Sender : TObject); virtual;
- // Check if APropertyName is published property of AObject.
- // Only performed if both parameters are not empty.
- procedure CheckPropertyName(AObject: TObject; const APropertyName: String);
- // If all CheckObjectSubject and Active are true, call ObjectToView.
- Procedure MaybeObjectToView;
- // If all CheckObjectSubject and Active are true, call ViewToObject.
- Procedure MaybeViewToObject;
- // Check if Subject/View and property names are set up correctly.
- Function CheckViewSubject : Boolean;
- // Override next two for custom behaviour.
- // Copies Subject.SubjectPropertyName to View.ViewPropertyName.
- Procedure DoObjectToView; virtual;
- // Copies View.ViewPropertyName to Subject.SubjectPropertyName
- Procedure DoViewToObject; virtual;
- // Override these, and call inherited at the end.
- // Get View component. Typically a TCustomEdit instance.
- function GetView : TObject; virtual;
- // Descendents should call this when the view changed.
- procedure ViewChanged; virtual;
- // Descendents should override this to handle changes in the value list
- procedure ValuelistChanged; virtual;
- // IFPObserver. Will call the necessary events.
- Procedure FPOObservedChanged(ASender : TObject; Operation : TFPObservedOperation; Data : Pointer);
- // Raise an error which shows more information about the control, subject and fieldname.
- Procedure RaiseMediatorError(Const Msg : String); overload;
- // Format version
- Procedure RaiseMediatorError(Const Fmt : String; Args : Array of const); overload;
- // View property that will be set by default
- Property ViewPropertyName : String Read FViewPropertyName Write SetViewPropertyName;
- // Is a copy operation View <-> Subject in progress ?
- Property Transferring : Boolean Read FTransferring;
- Public
- Destructor Destroy; override;
- // Copy subject to view. No check is done to see if all is well.
- Procedure ObjectToView;
- // Copy view to subject. No check is done to see if all is well.
- Procedure ViewToObject;
- // Minimum class that View must have to be handled by this mediator.
- class function ViewClass: TClass; virtual;
- // Composite mediator or not ?
- class function CompositeMediator : Boolean; virtual;
- // Subject. Must have IFPObserved interface
- Property Subject : TObject Read FSubject Write SetSubject;
- // View. Must have ViewPropertyName, if in use.
- Property View : TObject Read GetView;
- // Value list. To be used in mediators that use a dynamical value list
- // such as Listbox, combobox, groupbox.
- Property Valuelist : TObjectList Read FValueList Write SetValueList;
- Published
- // Property that will be copied to view.
- Property SubjectPropertyName : String Read FSubjectPropertyName Write SetSubjectPropertyName;
- // If not active, no copying is being done either way.
- Property Active : Boolean Read FActive Write SetActive;
- // If ReadOnly, only ObjectToView is used
- Property ReadOnly : Boolean Read FReadOnly Write SetReadOnly;
- // Can be used to copy data from control (view) to subject manually
- Property OnViewToObject : TMediatingEvent Read FOnViewToObject Write FOnViewToObject;
- // Can be used to copy data from control (view) to subject manually
- Property OnObjectToView : TMediatingEvent Read FOnObjectToView Write FOnObjectToView;
- end;
- TMediatorClass = Class of TBaseMediator;
- // Forward definitions
- TBaseListMediator = Class;
- { TComponentMediator }
- { General-purpose of Mediating views. Can be used on any form/component }
- TComponentMediator = Class(TBaseMediator)
- FViewComponent : TComponent;
- Protected
- function GetView : TObject; override;
- procedure SetComponent(const AValue: TComponent);
- Public
- procedure Notification(AComponent: TComponent; Operation: TOperation); override;
- Published
- // General component which can be set in Object Inspector
- Property ViewComponent : TComponent Read FViewComponent Write SetComponent;
- // Punlish property so it can be set in Object Inspector
- Property ViewPropertyName;
- end;
- { Event object used for OnBeforeSetupField event. Is used to allow formatting
- of fields before written to listview Caption or Items. }
- TOnBeforeSetupField = procedure(AObject: TObject; const AFieldName: string; var AValue: string) of object;
- { TListItemMediator }
- TListItemMediator = class(TObject, IFPObserver)
- private
- FSubject: TObject;
- FOnBeforeSetupField: TOnBeforeSetupField;
- FListMediator : TBaseListMediator;
- Function GetActive : Boolean;
- protected
- procedure SetSubject(const AValue: TObject); virtual;
- Procedure FPOObservedChanged(ASender : TObject; Operation : TFPObservedOperation; Data : Pointer); virtual;
- Procedure ObjectToView; virtual;
- Procedure ViewToObject; virtual;
- public
- destructor Destroy; override;
- procedure MaybeObjectToView;
- property OnBeforeSetupField: TOnBeforeSetupField read FOnBeforeSetupField write FOnBeforeSetupField;
- property Subject : TObject read FSubject write SetSubject;
- property Active : Boolean read GetActive;
- end;
- { TBaseListMediator - Base mediator that handles lists of objects.
- Needs a TList as subject. Items in list must have IFPObserved
- interface. It will create one (and use as subject) if passed a normal
- list or a collection.
- }
- TBaseListMediator = class(TBaseMediator)
- private
- FOnBeforeSetupField: TOnBeforeSetupField;
- FMediatorList: TFPObjectList;
- FListChanged : Boolean;
- procedure SetOnBeforeSetupField(const Value: TOnBeforeSetupField);
- protected
- // This needs to return false
- Class Function PublishedViewProperty : Boolean; override;
- // Descendents can override;
- Function AddObject(AObject: TObject; AIndex: Integer) : TListItemMediator; virtual;
- // Set all descendents to active
- procedure SetActive(const AValue: Boolean); override;
- // Must be overridden in descendents, and should return selected object
- function GetSelectedObject: TObject; virtual;
- // Must be overridden in descendents, and should set selected object
- procedure SetSelectedObject(const AValue: TObject); virtual;
- // Must be overridden in descendents to create an item mediator and add it to GUI control
- // Subject will be set after this call.
- Function CreateItemMediator(AData: TObject; ARow : integer) : TListItemMediator; virtual; abstract;
- // This frees the mediator. Descendents can override to additionally update the GUI control
- procedure DoDeleteItemMediator(AIndex : Integer; AMediator : TListItemMediator); virtual;
- // Creates a mediator for all items in the list. List Item Mediators are re-used (subject is set)
- procedure CreateSubMediators; virtual;
- // Does nothing
- procedure DoViewToObject; override;
- // Calls CreateSubMediators. Override for additional GUI setup.
- procedure DoObjectToView; override;
- // Additional checks on subject.
- procedure SetSubject(const AValue: TObject); override;
- Function FindObjectMediator(AObject : TObject; out AtIndex : Integer) : TListItemMediator;
- property MediatorList: TFPObjectList read FMediatorList;
- public
- constructor Create(AOwner : TComponent); override;
- destructor Destroy; override;
- class function CompositeMediator: Boolean; override;
- // This should handle additional additem/deleteitem events
- Procedure ObservedChanged(ASender : TObject; Operation : TFPObservedOperation);
- // Selected item in the list.
- property SelectedObject: TObject read GetSelectedObject write SetSelectedObject;
- published
- // Event to setup fields in item mediators.
- property OnBeforeSetupField: TOnBeforeSetupField read FOnBeforeSetupField write SetOnBeforeSetupField;
- end;
- { TMediatorFieldInfo - Describe a column in a columnar list display }
- TMediatorFieldInfo = class(TCollectionItem)
- private
- FWidth: integer;
- FCaption: string;
- FPropName: string;
- FAlign: TAlignment;
- function GetCaption: string;
- procedure SetAlign(const AValue: TAlignment);
- procedure SetCaption(const AValue: string);
- procedure SetPropName(const AValue: string);
- procedure SetWidth(const AValue: Integer);
- protected
- function GetAsString: string; virtual;
- procedure SetAsString(const AValue: string); virtual;
- Procedure Change;
- public
- procedure Assign(Source: TPersistent); override;
- // Setting this will parse everything.
- property AsString: string read GetAsString write SetAsString;
- published
- // Property Caption to be used for column head.
- property Caption: string read GetCaption write SetCaption;
- // Property Name to be displayed in column
- property PropertyName: string read FPropName write SetPropName;
- // Width of column
- property Width: Integer read FWidth write SetWidth;
- // Alignment of column
- property Alignment: TAlignment read FAlign write SetAlign default taLeftJustify;
- end;
- TColumnsListMediator = Class;
- { TMediatorFieldInfoList - Collection describing the columns in a columnar list display }
- TMediatorFieldInfoList = class(TCollection)
- private
- FMediator : TColumnsListMediator;
- function GetAsString: string;
- function GetI(Index: integer): TMediatorFieldInfo;
- procedure SetI(Index: integer; const AValue: TMediatorFieldInfo);
- protected
- procedure Notify(Item: TCollectionItem;Action: TCollectionNotification); override;
- Property Mediator : TColumnsListMediator read FMediator;
- public
- // Adding items to the collection.
- function AddFieldInfo: TMediatorFieldInfo; overload;
- function AddFieldInfo (Const APropName : String; AFieldWidth : Integer) : TMediatorFieldInfo; overload;
- function AddFieldInfo (Const APropName,ACaption : String; AFieldWidth : Integer) : TMediatorFieldInfo; overload;
- function AddFieldInfo (Const APropName,ACaption : String; AFieldWidth : Integer; AAlignment : TAlignment) : TMediatorFieldInfo; overload;
- property FieldInfo[Index: integer]: TMediatorFieldInfo read GetI write SetI; default;
- property AsString: string read GetAsString;
- end;
- { TColumnsListItemMediator - List item mediator that can handle multiple columns }
- TColumnsListItemMediator = class(TListItemMediator)
- Private
- Function GetFieldsInfo: TMediatorFieldInfoList;
- Published
- property FieldsInfo: TMediatorFieldInfoList read GetFieldsInfo;
- end;
- { TColumnsListMediator - List mediator that handles multiple columns }
- TColumnsListMediator = class(TBaseListMediator)
- Private
- FFieldsInfo: TMediatorFieldInfoList;
- procedure SetFieldsInfo(const AValue: TMediatorFieldInfoList);
- function GetDisplayNames: string;
- procedure SetDisplayNames(const AValue: string);
- procedure FieldInfoChanged(Item: TMediatorFieldInfo; Action: TCollectionNotification); virtual;
- Protected
- Class Function PublishedSubjectProperty : Boolean; override;
- procedure ParseDisplayNames(const AValue: string);
- // Called by DoViewToObject prior to creating item mediators
- procedure CreateColumns; virtual;
- // Calls CreateColumns and CreateSubMediators. Override for additional GUI setup.
- procedure DoObjectToView; override;
- Public
- Constructor Create(AOwner: TComponent); override;
- Destructor Destroy; override;
- // Can be used to set the column properties in 1 statement.
- property DisplayNames: string read GetDisplayNames write SetDisplayNames;
- Published
- // How to display the columns in the list.
- property FieldsInfo: TMediatorFieldInfoList read FFieldsInfo write SetFieldsInfo;
- end;
- { TMediatorDef - Mediator Definition Storage for MediatorManager }
- TMediatorDef = class(TCollectionItem)
- private
- FMC: TMediatorClass;
- FMSC: TClass;
- FPN: string;
- FPT: TTypeKinds;
- public
- // Return True if this definition handles the Subject,Gui,APropinfo trio
- function Handles(ASubject: TObject; AGui: TComponent; APropInfo: PPropInfo): Boolean;
- // Return True if this definition matches 'closer' than M.
- // Note that both current and M must have Handles() returned true for this to be useful.
- function BetterMatch(M: TMediatorDef): Boolean;
- // Definition
- property MediatorClass: TMediatorClass read FMC write FMC;
- property MinSubjectClass: TClass read FMSC write FMSC;
- property PropertyTypes: TTypeKinds read FPT write FPT;
- property PropertyName: string read FPN write FPN;
- end;
- TMediatorDefs = class(TCollection)
- private
- function GetDef(Index: integer): TMediatorDef;
- procedure SetDef(Index: integer; const AValue: TMediatorDef);
- public
- function AddDef: TMediatorDef;
- property Defs[Index: integer]: TMediatorDef read GetDef write SetDef; default;
- end;
- TMediatorManager = class(TObject)
- private
- FDefs: TMediatorDefs;
- public
- constructor Create;
- destructor Destroy; override;
- // If APropName is empty or APropInfo is Nil, a composite mediator will be searched.
- function FindDefFor(ASubject: TObject; AGui: TComponent): TMediatorDef; overload;
- function FindDefFor(ASubject: TObject; AGui: TComponent; const APropName: string): TMediatorDef; overload;
- function FindDefFor(ASubject: TObject; AGui: TComponent; APropInfo: PPropInfo): TMediatorDef; overload;
- function RegisterMediator(MediatorClass: TMediatorClass; MinSubjectClass: TClass): TMediatorDef; overload;
- function RegisterMediator(MediatorClass: TMediatorClass; MinSubjectClass: TClass; PropertyName: string): TMediatorDef; overload;
- function RegisterMediator(MediatorClass: TMediatorClass; MinSubjectClass: TClass; PropertyTypes: TTypeKinds): TMediatorDef; overload;
- property Defs: TMediatorDefs read FDefs;
- end;
- EMediator = class(Exception);
- function MediatorManager: TMediatorManager;
- Procedure MediatorError(Sender : TObject; Const Msg : String); overload;
- Procedure MediatorError(Sender : TObject; Const Fmt : String; Args : Array of const); overload;
- implementation
- Resourcestring
- SErrNotObserver = 'Instance of class %s is not an observer.';
- SErrInvalidPropertyName = '%s is not a valid published property of class %s';
- SErrObjectCannotBeObserved = 'Cannot observe an instance of class %d';
- sErrInvalidFieldName = 'No fieldname specified for column %d';
- sErrInvalidAlignmentChar = 'Invalid alignment character "%s" specified for column %d';
- sErrInvalidWidthSpecifier = 'Invalid with "%s" specified for column %d';
- sErrNotListObject = '%s is not a TObjectList';
- sErrCompositeNeedsList = '%s needs a TObjectList class but is registered with %s';
- SErrActive = 'Operation not allowed while the mediator is active';
- SErrNoGuiFieldName = 'no gui fieldname set';
- SErrNoSubjectFieldName = 'no subject fieldname set';
- { ---------------------------------------------------------------------
- Mediator global routines
- ---------------------------------------------------------------------}
- Procedure MediatorError(Sender : TObject; Const Msg : String); overload;
- Var
- M : TBaseMediator;
- C : TComponent;
- V,S : TObject;
- CN,SN,Err : String;
- begin
- if (Sender=Nil) then
- Err:=Msg
- else If Sender is TBaseMediator then
- begin
- M:=TBaseMediator(Sender);
- V:=M.View;
- S:=M.Subject;
- CN:='';
- If Assigned(V) then
- begin
- if (V is TComponent) then
- begin
- C:=TComponent(V);
- CN:=C.Name;
- end;
- If (CN='') then
- CN:=C.ClassName+' instance';
- end
- else
- CN:='Nil';
- If Assigned(S) then
- SN:=S.ClassName
- else
- SN:='Nil';
- Err:=Format('Mediator %s (%s,%s,%s) : %s',[M.ClassName,SN,CN,M.SubjectPropertyName,Msg]);
- end
- else if (Sender is TComponent) and (TComponent(Sender).Name<>'') then
- Err:=Format('%s : %s',[TComponent(Sender).Name,Msg])
- else
- Err:=Format('%s : %s',[Sender.ClassName,Msg]);
- Raise EMediator.Create(Err);
- end;
- Procedure MediatorError(Sender : TObject; const Fmt : String; Args : Array of const); overload;
- begin
- MediatorError(Sender,Format(Fmt,Args));
- end;
- Var
- MM : TMediatorManager;
- function MediatorManager: TMediatorManager;
- begin
- if (MM = nil) then
- MM := TMediatorManager.Create;
- Result := MM;
- end;
- { TObservedHook }
- constructor TObservedHook.CreateSender(ASender: TObject);
- begin
- FSender:=ASender;
- If FSender=Nil then
- FSender:=Self;
- end;
- destructor TObservedHook.Destroy;
- begin
- If Assigned(FObservers) then
- begin
- FPONotifyObservers(FSender,ooFree,Nil);
- FreeAndNil(FObservers);
- end;
- inherited Destroy;
- end;
- procedure TObservedHook.FPOAttachObserver(AObserver: TObject);
- Var
- I : IFPObserver;
- begin
- If Not AObserver.GetInterface(SGUIDObserver,I) then
- Raise EObserver.CreateFmt(SErrNotObserver,[AObserver.ClassName]);
- If not Assigned(FObservers) then
- FObservers:=TFPList.Create;
- FObservers.Add(I);
- end;
- procedure TObservedHook.FPODetachObserver(AObserver: TObject);
- Var
- I : IFPObserver;
- begin
- If Not AObserver.GetInterface(SGUIDObserver,I) then
- Raise EObserver.CreateFmt(SErrNotObserver,[AObserver.ClassName]);
- If Assigned(FObservers) then
- begin
- FObservers.Remove(I);
- If (FObservers.Count=0) then
- FreeAndNil(FObservers);
- end;
- end;
- procedure TObservedHook.Changed;
- begin
- FPONotifyObservers(Sender,ooChange,Nil)
- end;
- procedure TObservedHook.AddItem(AItem: TObject);
- begin
- FPONotifyObservers(FSender,ooAddItem,AItem);
- end;
- procedure TObservedHook.DeleteItem(AItem: TObject);
- begin
- FPONotifyObservers(FSender,ooDeleteItem,AItem);
- end;
- procedure TObservedHook.CustomNotify(Data : Pointer = Nil);
- begin
- FPONotifyObservers(FSender,ooCustom,Data);
- end;
- procedure TObservedHook.FPONotifyObservers(ASender: TObject; AOperation: TFPObservedOperation; Data : Pointer);
- Var
- O : TObject;
- I : Integer;
- Obs : IFPObserver;
- begin
- If Assigned(FObservers) then
- For I:=FObservers.Count-1 downto 0 do
- begin
- Obs:=IFPObserver(FObservers[i]);
- Obs.FPOObservedChanged(ASender,AOperation,Data);
- end;
- end;
- { TBaseMediator }
- function TBaseMediator.GetView: TObject;
- begin
- Result:=Nil;
- end;
- procedure TBaseMediator.ViewChanged;
- begin
- If PublishedViewProperty then
- CheckPropertyName(View,ViewPropertyName);
- MaybeObjectToView
- end;
- procedure TBaseMediator.ValuelistChanged;
- begin
- // Do nothing
- end;
- procedure TBaseMediator.SetActive(const AValue: Boolean);
- begin
- if FActive=AValue then exit;
- FActive:=AValue;
- MaybeObjectToView;
- end;
- procedure TBaseMediator.SetReadOnly(const AValue: Boolean);
- begin
- if FReadOnly=AValue then exit;
- FReadOnly:=AValue;
- MaybeObjectToView;
- end;
- procedure TBaseMediator.SetValueList(const AValue: TObjectList);
- Var
- I : IFPObserved;
- begin
- if FValueList=AValue then exit;
- If FValueList<>Nil then
- I.FPODetachObserver(Self);
- If Assigned(AValue) then
- begin
- FValueList:=AValue;
- If Assigned(AValue) then
- AValue.FPOAttachObserver(Self);
- end;
- FValueList:=AValue;
- ValueListChanged;
- end;
- procedure TBaseMediator.CheckPropertyName(AObject : TObject; const APropertyName : String);
- begin
- If Assigned(AObject) and (APropertyName<>'') then
- If Not IsPublishedProp(AObject,APropertyName) then
- Raise EObserver.CreateFmt(SErrInvalidPropertyName,[APropertyName,AObject.ClassName]);
- end;
- procedure TBaseMediator.MaybeObjectToView;
- begin
- If FActive and CheckViewSubject then
- ObjectToView
- end;
- procedure TBaseMediator.MaybeViewToObject;
- begin
- If FActive and (Not ReadOnly) and CheckViewSubject then
- ViewToObject;
- end;
- function TBaseMediator.CheckViewSubject: Boolean;
- Var
- O : TObject;
- begin
- O:=GetView;
- Result:=Assigned(FSubject)
- and Assigned(O)
- and (ViewPropertyName<>'')
- and (SubjectPropertyName<>'');
- end;
- procedure TBaseMediator.SetSubjectPropertyName(const AValue: String);
- begin
- if FSubjectPropertyName=AValue then exit;
- If PublishedSubjectProperty then
- CheckPropertyName(FSubject,AValue);
- FSubjectPropertyName:=AValue;
- MaybeObjectToView;
- end;
- procedure TBaseMediator.SetSubject(const AValue: TObject);
- Var
- I : IFPObserved;
- begin
- if FSubject=AValue then exit;
- If PublishedSubjectProperty then
- CheckPropertyName(AValue,FSubjectPropertyName);
- If FSubject<>Nil then
- If FSubject.GetInterface(SGUIDObserved,I) then
- I.FPODetachObserver(Self);
- If (AValue<>Nil) then
- begin
- If not AValue.GetInterface(SGUIDObserved,I) then
- Raise EObserver.CreateFmt(SErrObjectCannotBeObserved,[AValue.ClassName]);
- FSubject:=AValue;
- I.FPOAttachObserver(Self);
- end
- else
- FSubject:=AValue;
- MaybeObjectToView;
- end;
- procedure TBaseMediator.SetViewPropertyName(const AValue: String);
- begin
- if FViewPropertyName=AValue then exit;
- If PublishedViewProperty then
- CheckPropertyName(GetView,AValue);
- FViewPropertyName:=AValue;
- MaybeObjectToView;
- end;
- class function TBaseMediator.PublishedViewProperty: Boolean;
- begin
- Result:=True;
- end;
- class function TBaseMediator.PublishedSubjectProperty: Boolean;
- begin
- Result:=True;
- end;
- procedure TBaseMediator.ViewChangedHandler(Sender: TObject);
- begin
- MaybeViewToObject;
- end;
- procedure TBaseMediator.FPOObservedChanged(ASender: TObject;
- Operation: TFPObservedOperation; Data : Pointer);
- begin
- If (ASender=FSubject) then
- begin
- If Operation=ooChange then
- MaybeObjectToView
- else if Operation=ooFree then
- FSubject:=Nil;
- end
- else if (ASender=FValueList) then
- begin
- If Operation=ooChange then
- ValueListChanged
- else if Operation=ooFree then
- FValueList:=Nil;
- end;
- end;
- procedure TBaseMediator.RaiseMediatorError(const Msg: String);
- begin
- MediatorError(Self,Msg);
- end;
- procedure TBaseMediator.RaiseMediatorError(const Fmt: String;
- Args: array of const);
- begin
- RaiseMediatorError(Format(FMT,Args));
- end;
- destructor TBaseMediator.Destroy;
- begin
- Subject:=Nil;
- ValueList:=Nil;
- inherited Destroy;
- end;
- procedure TBaseMediator.DoObjectToView;
- begin
- SetPropValue(GetView,ViewPropertyName,GetPropValue(FSubject,FSubjectPropertyName));
- end;
- procedure TBaseMediator.DoViewToObject;
- begin
- SetPropValue(FSubject,FSubjectPropertyName,GetPropValue(GetView,ViewPropertyName));
- end;
- procedure TBaseMediator.ObjectToView;
- Var
- B : Boolean;
- begin
- If Not FTransferring then
- begin
- FTransferring:=True;
- try
- B:=False;
- If Assigned(FOnObjectToView) then
- FOnObjectToView(Self,B);
- If not B then
- DoObjectToView;
- finally
- FTransferring:=False;
- end;
- end;
- end;
- procedure TBaseMediator.ViewToObject;
- Var
- B : Boolean;
- begin
- If Not FTransferring then
- begin
- FTransferring:=True;
- try
- B:=False;
- If Assigned(FONViewToObject) then
- FONViewToObject(Self,B);
- If not B then
- DoViewToObject;
- finally
- FTransferring:=False;
- end;
- end;
- end;
- class function TBaseMediator.ViewClass: TClass;
- begin
- Result:=TObject;
- end;
- class function TBaseMediator.CompositeMediator: Boolean;
- begin
- Result:=False;
- end;
- { TComponentMediator }
- function TComponentMediator.GetView: TObject;
- begin
- Result:=FViewComponent;
- end;
- procedure TComponentMediator.SetComponent(const AValue: TComponent);
- begin
- If (Avalue=FViewComponent) then
- Exit;
- If Assigned(FViewComponent) then
- FViewComponent.RemoveFreeNotification(Self);
- FViewComponent:=AValue;
- If Assigned(FViewComponent) then
- FViewComponent.FreeNotification(Self);
- ViewChanged;
- end;
- procedure TComponentMediator.Notification(AComponent: TComponent;
- Operation: TOperation);
- begin
- inherited Notification(AComponent, Operation);
- If (Operation=opRemove) and (AComponent=FViewComponent) then
- begin
- FViewComponent:=Nil;
- ViewChanged;
- end;
- end;
- { TMediatorDef }
- function TMediatorDef.Handles(ASubject: TObject; AGui: TComponent; APropInfo: PPropInfo): Boolean;
- var
- N: string;
- begin
- if (APropInfo = nil) then
- Result := FMC.CompositeMediator
- else
- begin
- N := APropInfo^.Name;
- Result := True;
- end;
- if not Result then
- Exit; // ==>
- // At least the classes must match
- Result := AGui.InheritsFrom(FMC.ViewClass) and ASubject.InheritsFrom(FMSC);
- if Result and not FMC.CompositeMediator then
- if (PropertyName <> '') then
- Result := (CompareText(N, PropertyName) = 0)
- else // Property kind should match. Note that property MUST be set to something.
- Result := (APropInfo^.PropType^.Kind in PropertyTypes); // If PropertyName is set, it must match
- end;
- function TMediatorDef.BetterMatch(M: TMediatorDef): Boolean;
- begin
- Result := (M = nil);
- if not Result then
- begin
- Result := (FMC.CompositeMediator = M.MediatorClass.CompositeMediator);
- if Result then
- begin
- Result := (PropertyName <> '') and (M.PropertyName = '');
- if not Result then
- begin
- // M's property matches closer
- Result := not ((M.PropertyName <> '') and (PropertyName = ''));
- if Result then
- begin
- // Properties are on equal level. Check GUI class.
- // Closer GUI class ?
- Result := FMC.ViewClass.InheritsFrom(M.MediatorClass.ViewClass);
- if not Result then
- begin
- // M's GUI class matches closer ?
- Result := not (M.MediatorClass.ViewClass.InheritsFrom(FMC.ViewClass));
- if Result then
- begin
- // GUI classes are on equal level (different branches in tree). Check subject class.
- // Closer Subject class ?
- Result := FMSC.InheritsFrom(M.FMSC);
- if not Result then
- // M's subject class matches closer ?
- Result := not M.FMSC.InheritsFrom(FMSC);
- end;
- end;
- end;
- end;
- end;
- end;
- end;
- { TMediatorDefs }
- function TMediatorDefs.GetDef(Index: integer): TMediatorDef;
- begin
- Result := TMediatorDef(Items[Index]);
- end;
- procedure TMediatorDefs.SetDef(Index: integer; const AValue: TMediatorDef);
- begin
- Items[Index] := AValue;
- end;
- function TMediatorDefs.AddDef: TMediatorDef;
- begin
- Result := Add as TMediatorDef;
- end;
- { TMediatorManager }
- constructor TMediatorManager.Create;
- begin
- FDefs := TMediatorDefs.Create(TMediatorDef);
- end;
- destructor TMediatorManager.Destroy;
- begin
- FreeAndNil(FDefs);
- inherited Destroy;
- end;
- function TMediatorManager.FindDefFor(ASubject: TObject; AGui: TComponent): TMediatorDef;
- begin
- Result := FindDefFor(ASubject, AGUI, PPropInfo(nil));
- end;
- function TMediatorManager.FindDefFor(ASubject: TObject; AGui: TComponent; const APropName: string): TMediatorDef;
- var
- propinfo: PPropInfo;
- begin
- propinfo := GetPropInfo(ASubject, APropName);
- Result := FindDefFor(ASubject, AGUI, propinfo);
- end;
- function TMediatorManager.FindDefFor(ASubject: TObject; AGui: TComponent; APropInfo: PPropInfo): TMediatorDef;
- var
- D: TMediatorDef;
- I: integer;
- begin
- Result := nil;
- for I := 0 to FDefs.Count - 1 do
- begin
- D := FDefs[I];
- if D.Handles(ASubject, AGUI, APropInfo) then
- if (D.BetterMatch(Result)) then
- Result := D;
- end;
- end;
- function TMediatorManager.RegisterMediator(MediatorClass: TMediatorClass; MinSubjectClass: TClass): TMediatorDef;
- begin
- Result := FDefs.AddDef;
- Result.MediatorClass := MediatorClass;
- Result.FMSC := MinSubjectClass;
- Result.FPN := '';
- Result.FPT := tkProperties - [tkClass, tkInterface, tkDynArray, tkObject, tkInterfaceRaw];
- end;
- function TMediatorManager.RegisterMediator(MediatorClass: TMediatorClass; MinSubjectClass: TClass; PropertyName: string): TMediatorDef;
- begin
- Result := FDefs.AddDef;
- Result.MediatorClass := MediatorClass;
- Result.FMSC := MinSubjectClass;
- Result.FPN := PropertyName;
- Result.FPT := [];
- end;
- function TMediatorManager.RegisterMediator(MediatorClass: TMediatorClass; MinSubjectClass: TClass; PropertyTypes: TTypeKinds): TMediatorDef;
- begin
- Result := FDefs.AddDef;
- Result.MediatorClass := MediatorClass;
- Result.FMSC := MinSubjectClass;
- Result.FPN := '';
- Result.FPT := PropertyTypes;
- end;
- { TListItemMediator }
- function TListItemMediator.GetActive: Boolean;
- begin
- Result:=False;
- If Assigned(FListMediator) then
- Result:=FListMediator.Active;
- end;
- procedure TListItemMediator.SetSubject(const AValue: TObject);
- Var
- I : IFPObserved;
- begin
- if Avalue=FSubject then
- Exit;
- If FSubject<>Nil then
- If FSubject.GetInterface(SGUIDObserved,I) then
- I.FPODetachObserver(Self);
- FSubject:=AValue;
- If (FSubject<>Nil) then
- begin
- If not FSubject.GetInterface(SGUIDObserved,I) then
- Raise EObserver.CreateFmt(SErrObjectCannotBeObserved,[FSubject.ClassName]);
- I.FPOAttachObserver(Self);
- end;
- MaybeObjectToView
- end;
- procedure TListItemMediator.FPOObservedChanged(ASender: TObject;
- Operation: TFPObservedOperation; Data : Pointer);
- begin
- If Operation=ooFree then
- FSubject:=Nil
- else
- MaybeObjectToView;
- end;
- procedure TListItemMediator.ObjectToView;
- begin
- // Do nothing
- end;
- procedure TListItemMediator.ViewToObject;
- begin
- // Do nothing
- end;
- destructor TListItemMediator.Destroy;
- begin
- Subject:=Nil;
- inherited Destroy;
- end;
- procedure TListItemMediator.MaybeObjectToView;
- begin
- If Assigned(FSubject) and Active then
- ObjectToView;
- end;
- { TMediatorFieldInfo }
- procedure TMediatorFieldInfo.Change;
- begin
- FPONotifyObservers(Self,ooChange,Nil);
- end;
- function TMediatorFieldInfo.GetCaption: string;
- begin
- Result:=FCaption;
- If (Result='') then
- Result:=FPropName;
- end;
- procedure TMediatorFieldInfo.SetAlign(const AValue: TAlignment);
- begin
- If AValue=fAlign then Exit;
- FAlign:=AValue;
- Change;
- end;
- procedure TMediatorFieldInfo.SetCaption(const AValue: string);
- begin
- If AValue=Caption then Exit;
- FCaption:=AValue;
- Change;
- end;
- procedure TMediatorFieldInfo.SetPropName(const AValue: string);
- begin
- If AValue=FPropName then Exit;
- FPropName:=AValue;
- Change;
- end;
- procedure TMediatorFieldInfo.SetWidth(const AValue: Integer);
- begin
- If (FWidth=AValue) then Exit;
- FWidth:=AValue;
- Change;
- end;
- const
- AlignChars: array[TAlignMent] of char = ('l', 'r', 'c');
- function TMediatorFieldInfo.GetAsString: string;
- begin
- Result := Format('%s|%s|%d|%s', [PropertyName, AlignChars[Alignment], Width, Caption]);
- end;
- procedure TMediatorFieldInfo.SetAsString(const AValue: string);
- Function GetToken(Var S : String) : String;
- Var
- P : Integer;
- begin
- P:=Pos('|',S);
- If P=0 then P:=Length(S)+1;
- Result:=Copy(S,1,P-1);
- Delete(S,1,P);
- end;
- var
- V,S: string;
- A: TAlignment;
- I: integer;
- begin
- V:=S;
- I := 0;
- PropertyName:=GetToken(V);
- if (PropertyName = '') then
- MediatorError(Self,SErrInvalidFieldName, [Index + 1]);
- Alignment:=taLeftJustify;
- Width:=50;
- S:=GetToken(V);
- if (S<>'') then
- begin
- if (length(S)<>1) then
- MediatorError(Self,SErrInvalidAlignmentChar, [S,Index+1]);
- for A := Low(Talignment) to High(TAlignment) do
- if (Upcase(AlignChars[A])=Upcase(S[1])) then
- Alignment := A;
- S:=GetToken(V);
- if (S<>'') then
- begin
- if not TryStrToInt(S,i) then
- MediatorError(Self,SErrInvalidWidthSpecifier,[S]);
- Width:=I;
- S:=getToken(V);
- if (S<>'') then
- Caption := S;
- end;
- end;
- end;
- procedure TMediatorFieldInfo.Assign(Source: TPersistent);
- Var
- M : TMediatorFieldInfo;
- begin
- if (Source is TMediatorFieldInfo) then
- begin
- M:=Source as TMediatorFieldInfo;
- FWidth:=M.FWidth;
- FCaption:=M.FCaption;
- FPropName:=M.FPropname;
- FAlign:=M.FAlign;
- end
- else
- inherited Assign(Source);
- end;
- { TColumnsListItemMediator }
- function TColumnsListItemMediator.GetFieldsInfo: TMediatorFieldInfoList;
- begin
- If Assigned(FListmediator) and (FListMediator is TColumnsListMediator) then
- Result:=TColumnsListMediator(FListMediator).FFieldsInfo;
- end;
- { TBaseListMediator }
- procedure TBaseListMediator.SetOnBeforeSetupField(
- const Value: TOnBeforeSetupField);
- var
- I: integer;
- begin
- FOnBeforeSetupField := Value;
- for I := 0 to FMediatorList.Count - 1 do
- TListItemMediator(FMediatorList[i]).OnBeforeSetupField := Value;
- end;
- class function TBaseListMediator.PublishedViewProperty: Boolean;
- begin
- Result:=False;
- end;
- procedure TBaseListMediator.SetActive(const AValue: Boolean);
- Var
- i : Integer;
- begin
- inherited SetActive(AValue);
- If AValue then
- For I:=0 to MediatorList.Count-1 do
- TListItemMediator(MediatorList[i]).MaybeObjectToView;
- end;
- function TBaseListMediator.GetSelectedObject: TObject;
- begin
- Result := nil;
- end;
- procedure TBaseListMediator.SetSelectedObject(const AValue: TObject);
- begin
- // Do nothing
- end;
- procedure TBaseListMediator.DoDeleteItemMediator(AIndex: Integer;
- AMediator: TListItemMediator);
- begin
- MediatorList.Delete(AIndex);
- end;
- Function TBaseListMediator.AddObject(AObject : TObject; AIndex : Integer) : TListItemMediator;
- begin
- Result:=CreateItemMediator(AObject,AIndex);
- If (Result<>Nil) then
- begin
- Result.FListMediator:=Self;
- Result.Subject:=AObject;
- MediatorList.Add(Result);
- end;
- end;
- procedure TBaseListMediator.CreateSubMediators;
- var
- I : integer;
- Model : TObjectList;
-
- begin
- Model:=Subject as TObjectList;
- for i := 0 to Model.Count - 1 do
- begin
- if i < MediatorList.Count then
- TListItemMediator(MediatorList[i]).Subject := Model[i]
- else
- AddObject(Model[i], i);
- end;
- for i := MediatorList.Count-1 downto Model.Count do
- DoDeleteItemMediator(I,TListItemMediator(MediatorList[i]));
- FListChanged:=False;
- end;
- procedure TBaseListMediator.DoViewToObject;
- begin
- // Do nothing
- end;
- procedure TBaseListMediator.DoObjectToView;
- begin
- CreateSubMediators;
- end;
- procedure TBaseListMediator.SetSubject(const AValue: TObject);
- Var
- V : TOBject;
- begin
- if (AValue <> nil) then
- begin
- V:=Nil;
- if (AValue is TObjectList) then
- V:=AValue
- else If (AValue is TList) then
- V:=AValue
- else If (AValue is TCollection) then
- V:=AValue;
- if (V=Nil) then
- RaiseMediatorError(SErrNotListObject, [AValue.ClassName]);
- end;
- FListChanged:=True;
- inherited SetSubject(AValue)
- end;
- function TBaseListMediator.FindObjectMediator(AObject: TObject; out
- AtIndex: Integer): TListItemMediator;
- begin
- AtIndex:=FMediatorList.Count-1;
- While (AtIndex>=0) and (TListItemMediator(FMediatorList[AtIndex]).Subject<>AObject) do
- Dec(AtIndex);
- If (AtIndex=-1) then
- Result:=Nil
- else
- Result:=TListItemMediator(FMediatorList[AtIndex]);
- end;
- constructor TBaseListMediator.Create(AOwner: TComponent);
- begin
- inherited Create(AOwner);
- FMediatorList := TFPObjectList.Create;
- Active := False;
- ViewPropertyName:='Caption';
- end;
- destructor TBaseListMediator.Destroy;
- begin
- FreeAndNil(FMediatorList);
- inherited Destroy;
- end;
- class function TBaseListMediator.CompositeMediator: Boolean;
- begin
- Result:=True;
- end;
- procedure TBaseListMediator.ObservedChanged(ASender: TObject;
- Operation: TFPObservedOperation);
- var
- M : TListItemMediator;
- Model : TObjectList;
- I : Integer;
- begin
- // Do not call inherited, it will rebuild the list !!
- Case Operation of
- ooAddItem : AddObject(ASender,TObjectList(Subject).Count-1); // always at the end...
- ooDeleteItem : begin
- M:=FindObjectMediator(ASender,I);
- if M<>nil then
- DoDeleteItemMediator(I,M);
- end;
- ooChange : begin
- Model:=(Subject as TObjectList);
- if FListChanged or (TObjectList(Model).Count<>MediatorList.Count) or (Model.Count=0) then // Safety measure
- MaybeObjectToView;
- end;
- end;
- end;
- { TColumnsListMediator }
- procedure TColumnsListMediator.SetFieldsInfo(
- const AValue: TMediatorFieldInfoList);
- begin
- FFieldsInfo.Assign(AValue);
- end;
- function TColumnsListMediator.GetDisplayNames: string;
- begin
- Result := FFieldsInfo.AsString;
- end;
- procedure TColumnsListMediator.SetDisplayNames(const AValue: string);
- begin
- SubjectPropertyName:=AValue;
- ParseDisplayNames(AValue);
- end;
- procedure TColumnsListMediator.FieldInfoChanged(Item: TMediatorFieldInfo;
- Action: TCollectionNotification);
- begin
- If Active then
- RaiseMediatorError(SErrActive);
- end;
- class function TColumnsListMediator.PublishedSubjectProperty: Boolean;
- begin
- Result:=False;
- end;
- procedure TColumnsListMediator.ParseDisplayNames(const AValue: string);
- Function GetToken(Var S : String) : String;
- Var
- P : Integer;
- begin
- P:=Pos(';',S);
- If P=0 then P:=Length(S)+1;
- Result:=Copy(S,1,P-1);
- Delete(S,1,P);
- end;
- var
- I : integer;
- lField : string;
- MFI : TMediatorFieldInfo;
- A,S : String;
- begin
- FFieldsInfo.Clear;
- A:=AValue;
- Repeat
- S:=GetToken(A);
- If (S<>'') then
- begin
- MFI:=FFieldsInfo.AddFieldInfo;
- MFI.AsString:=S;
- end;
- until (S='');
- end;
- procedure TColumnsListMediator.CreateColumns;
- begin
- // Do nothing. Must be implemented by descendent objects.
- end;
- procedure TColumnsListMediator.DoObjectToView;
- begin
- CreateColumns;
- inherited DoObjectToView;
- end;
- constructor TColumnsListMediator.Create(AOwner: TComponent);
- begin
- inherited Create(AOwner);
- FFieldsInfo:=TMediatorFieldInfoList.create(TMediatorFieldInfo);
- SubjectPropertyName:='Caption';
- end;
- destructor TColumnsListMediator.Destroy;
- begin
- FreeAndNil(FFieldsInfo);
- inherited Destroy;
- end;
- { TMediatorFieldInfoList }
- function TMediatorFieldInfoList.GetAsString: string;
- var
- I: integer;
- begin
- Result := '';
- for I := 0 to Count - 1 do
- begin
- if (Result <> '') then
- Result := Result + ';';
- Result := Result + FieldInfo[i].AsString;
- end;
- end;
- function TMediatorFieldInfoList.GetI(Index: integer): TMediatorFieldInfo;
- begin
- Result := TMediatorFieldInfo(Items[Index]);
- end;
- procedure TMediatorFieldInfoList.SetI(Index: integer;
- const AValue: TMediatorFieldInfo);
- begin
- Items[Index] := AValue;
- end;
- procedure TMediatorFieldInfoList.Notify(Item: TCollectionItem;
- Action: TCollectionNotification);
- begin
- inherited Notify(Item, Action);
- If Assigned(FMediator) then
- FMediator.FieldInfoChanged(Item as TMediatorFieldInfo,Action)
- end;
- function TMediatorFieldInfoList.AddFieldInfo: TMediatorFieldInfo;
- begin
- Result := Add as TMediatorFieldInfo;
- end;
- function TMediatorFieldInfoList.AddFieldInfo(const APropName: String;
- AFieldWidth: Integer): TMediatorFieldInfo;
- begin
- Result:=AddFieldInfo();
- Result.PropertyName:=APropName;
- Result.Width:=AFieldWidth;
- end;
- function TMediatorFieldInfoList.AddFieldInfo(const APropName, ACaption: String;
- AFieldWidth: Integer): TMediatorFieldInfo;
- begin
- Result:=AddFieldInfo(APropName,AFieldWidth);
- Result.Caption:=ACaption;
- end;
- function TMediatorFieldInfoList.AddFieldInfo(const APropName, ACaption: String;
- AFieldWidth: Integer; AAlignment: TAlignment): TMediatorFieldInfo;
- begin
- Result:=AddFieldInfo(APropName,ACaption,AFieldWidth);
- Result.Alignment:=AAlignment;
- end;
- end.
|