Menu

[r4348]: / trunk / Src / FmBase.pas  Maximize  Restore  History

Download this file

295 lines (262 with data), 10.3 kB

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
{
* This Source Code Form is subject to the terms of the Mozilla Public License,
* v. 2.0. If a copy of the MPL was not distributed with this file, You can
* obtain one at http://mozilla.org/MPL/2.0/
*
* Copyright (C) 2005-2013, Peter Johnson (www.delphidabbler.com).
*
* $Rev$
* $Date$
*
* Implements a form that provides the ancestor of all forms in the application.
* Provides default names for form window classes along with various operations
* that are common to all forms in application.
}
unit FmBase;
interface
uses
// Delphi
Classes, Forms, Controls, Messages,
// Project
IntfAligner, UControlStateMgr;
type
/// <summary>Base class for all forms in application.</summary>
/// <remarks>Sets a unique window class name for all derived forms and
/// provides various operations that are common to all forms in application.
/// </remarks>
TBaseForm = class(TForm)
/// <summary>Handles form's OnDestroy event. Frees owned objects.</summary>
procedure FormDestroy(Sender: TObject);
/// <summary>Handles form's OnShow event. Aligns form on screen using
/// aligner object. Calls virtual methods that sub-classes override to
/// perform pre- and post- alignment initialisation.</summary>
procedure FormShow(Sender: TObject);
/// <summary>Handles form's OnCreate event to perform initialisations
/// required for every form.</summary>
/// <remarks>In addition to creation of owned objects this method also
/// sets form's font in OS dependent way.</remarks>
procedure FormCreate(Sender: TObject);
/// <summary>Handles form's OnKeyDown event. Intercepts Alt+F10 key press
/// and displays any available context menu.</summary>
procedure FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
strict private
var
/// <summary>Object used to enable / disable all form's controls when.
/// form's enabled state changes.</summary>
fCtrlStateMgr: TControlStateMgr;
const
/// <summary>Custom message used to call AfterShow method after form
/// appears on screen.</summary>
WM_AFTERSHOW = WM_USER + 1; // Custom message used to call AfterShow
strict private
/// <summary>Aligns form on screen using an IAligner instance.</summary>
/// <remarks>Called from OnShow event after form is customised and before
/// it is initialised.</remarks>
procedure AlignForm;
/// <summary>Message handler that responds to changes in form's enabled
/// state by updating state of all controls according to whether form is
/// enabled or disabled.</summary>
procedure CMEnabledChanged(var Msg: TMessage); message CM_ENABLEDCHANGED;
/// <summary>Message handler that responds to custom message sent from
/// OnShow event handler that arrives after form has been displayed. Calls
/// virtual AfterShow method.</summary>
procedure WMAfterShow(var Msg: TMessage); message WM_AFTERSHOW;
/// <summary>Activates any context menu associated with the active control
/// or any of its parent controls.</summary>
procedure ActivateContextMenu;
strict protected
/// <summary>Overrides window creation parameters to set window class name
/// to that provided by virtual WindowClassName method.</summary>
procedure CreateParams(var Params: TCreateParams); override;
/// <summary>Returns a window class name comprised of company, program and
/// form class names.</summary>
function WindowClassName: string; virtual;
/// <summary>Returns new instance of aligner object used to align form to
/// owner.</summary>
/// <remarks>This implementation returns a null aligner. Subclasses that
/// require alignment should return a suitable IAligner instance.</remarks>
function GetAligner: IFormAligner; virtual;
/// <summary>Customises form.</summary>
/// <remarks>
/// <para>This method is called during the form's OnShow event before the
/// form is aligned.</para>
/// <para>In this implementation the method does nothing. Subclasses should
/// overrride to perform any customisation and to change the default size
/// of form if necessary instead handling the OnShow event.</para>
/// </remarks>
procedure CustomiseForm; virtual;
/// <summary>Initialises form content.</summary>
/// <remarks>
/// <para>This method is called during the form's OnShow event after the
/// form is aligned.</para>
/// <para>This implementation does nothing. Subclasses should override to
/// initialise the form instead of handling the OnShow event.</para>
/// <para>The form size should not be changed in this method since it will
/// interfere with the aligment.</para>
/// </remarks>
procedure InitForm; virtual;
/// <summary>Performs any actions needed after the form is visible on
/// screen.</summary>
/// <remarks>This implementation does nothing. Subclasses that need this
/// functionality should override this method.</remarks>
procedure AfterShowForm; virtual;
/// <summary>Protected constructor. Does nothing but call the inherited
/// constructor.</summary>
/// <remarks>
/// <para>This constructor is provided for use in derived form classes that
/// implement the INoPublicConstruct interface where the public Create
/// constructor can't be called.</para>
/// <para>Such classes must instantiate the form from a class method that
/// must call InternalCreate instead of Create.</para>
/// </remarks>
constructor InternalCreate(AOwner: TComponent); virtual;
public
/// <summary>Public constructor. Does nothing but call the inherited
/// constructor.</summary>
/// <remarks>
/// <para>This constructor can be called directly or from class methods in
/// a descendant class, providing that class does not support the
/// INoPublicConstruct interface.</para>
/// <para>In cases where INoPublicConstruct is supported the protected
/// InternalCreate constructor must be called instead.</para>
/// </remarks>
constructor Create(AOwner: TComponent); override;
end;
implementation
uses
// Delphi
SysUtils, Windows, Menus,
// Project
UAppInfo, UBaseObjects, UClassHelpers, UFontHelper, UKeysHelper, UMenus,
UNulFormAligner, UStrUtils;
{$R *.dfm}
{ TBaseForm }
procedure TBaseForm.ActivateContextMenu;
var
Ctrl: TControl; // active control or a parent that supports pop-up menu
MenuIntf: IPopupMenu; // interface reference to controls supporting IPopupMenu
PopupPos: TPoint; // pop-up position of menu in screen co-ords
begin
// search active control parents to try to find if pop-up menu supported
Ctrl := ActiveControl;
if not Assigned(Ctrl) then
Ctrl := Self;
while Assigned(Ctrl)
and (Ctrl <> Self)
and not Ctrl.HasPopupMenu
and not (Supports(Ctrl, IPopupMenu, MenuIntf) and MenuIntf.HasPopup) do
Ctrl := Ctrl.Parent;
if not Assigned(Ctrl) then
Exit;
// we use an arbitrary pop-up position: may be able to improve on this
PopupPos := Ctrl.ClientToScreen(
Point(40, 40)
);
// show pop-up menu, either via PopupMenu property or via IPopupMenu interface
if Ctrl.HasPopupMenu then
Ctrl.GetPopupMenu.Popup(PopupPos.X, PopupPos.Y)
else if Supports(Ctrl, IPopupMenu, MenuIntf) and MenuIntf.HasPopup then
MenuIntf.Popup(PopupPos);
end;
procedure TBaseForm.AfterShowForm;
begin
// Do nothing
end;
procedure TBaseForm.AlignForm;
begin
// Align the control. This does nothing by default, since default aligner is
// a do-nothing instance
GetAligner.AlignForm(Self);
end;
procedure TBaseForm.CMEnabledChanged(var Msg: TMessage);
begin
inherited;
// We update state of all controls, menu items and actions if possible
if Assigned(fCtrlStateMgr) then
fCtrlStateMgr.Update;
end;
constructor TBaseForm.Create(AOwner: TComponent);
begin
Assert(not Supports(Self, INoPublicConstruct),
ClassName + '.Create: Form''s public constructor can''t be called');
inherited;
end;
procedure TBaseForm.CreateParams(var Params: TCreateParams);
var
ClassName: string; // window class name
begin
inherited;
ClassName := WindowClassName;
if ClassName <> '' then
StrLCopy(
Params.WinClassName,
PChar(ClassName),
SizeOf(Params.WinClassName) div SizeOf(Char) - 1
);
end;
procedure TBaseForm.CustomiseForm;
begin
// Do nothing
end;
procedure TBaseForm.FormCreate(Sender: TObject);
begin
inherited;
fCtrlStateMgr := TControlStateMgr.Create(Self);
// Set form font to OS default font. This will be used by all descendants and
// controls that have ParentFont true.
TFontHelper.SetDefaultFont(Self.Font);
end;
procedure TBaseForm.FormDestroy(Sender: TObject);
begin
FreeAndNil(fCtrlStateMgr);
end;
procedure TBaseForm.FormKeyDown(Sender: TObject; var Key: Word;
Shift: TShiftState);
begin
inherited;
if (Key = VK_F10) and (ExtractShiftKeys(Shift) = [ssAlt]) then
ActivateContextMenu;
end;
procedure TBaseForm.FormShow(Sender: TObject);
begin
CustomiseForm;
AlignForm;
InitForm;
// Post message that causes AfterShowForm to be called after form has appeared
// on screen
PostMessage(Handle, WM_AFTERSHOW, 0, 0);
end;
function TBaseForm.GetAligner: IFormAligner;
begin
Result := TNulAligner.Create;
end;
procedure TBaseForm.InitForm;
begin
// Do nothing
end;
constructor TBaseForm.InternalCreate(AOwner: TComponent);
begin
Assert(Supports(Self, INoPublicConstruct), ClassName + '.InternalCreate: '
+ 'Form''s protected constructor can''t be called');
inherited Create(AOwner);
end;
function TBaseForm.WindowClassName: string;
var
PostfixName: string; // Postfix to name, based on form's class name
begin
// Calculate window class name postfix. This is form class name, stripped of
// any preceeding 'T'
if StrStartsStr('T', ClassName) then
PostfixName := StrSliceRight(ClassName, Length(ClassName) - 1)
else
PostfixName := ClassName;
// Build window class name
Result := TAppInfo.CompanyName
+ '.' + TAppInfo.ProgramName
+ '.' + PostfixName;
end;
procedure TBaseForm.WMAfterShow(var Msg: TMessage);
begin
AfterShowForm;
end;
end.
Want the latest updates on software, tech news, and AI?
Get latest updates about software, tech news, and AI from SourceForge directly in your inbox once a month.