lazarus/lcl/include/screen.inc

1184 lines
33 KiB
PHP

{%MainUnit ../forms.pp}
{******************************************************************************
TScreen
******************************************************************************
*****************************************************************************
This file is part of the Lazarus Component Library (LCL)
See the file COPYING.modifiedLGPL.txt, included in this distribution,
for details about the license.
*****************************************************************************
}
const
MonitorSearchFlags: array[TMonitorDefaultTo] of DWord =
(
{ mdNearest } MONITOR_DEFAULTTONEAREST,
{ mdNull } MONITOR_DEFAULTTONULL,
{ mdPrimary } MONITOR_DEFAULTTOPRIMARY
);
function EnumMonitors(hMonitor: HMONITOR; hdcMonitor: HDC; lprcMonitor: PRect;
dwData: LPARAM): LongBool; extdecl;
var
Screen: TScreen absolute dwData;
Monitor: TMonitor;
begin
Monitor := TMonitor.Create;
Monitor.FHandle := hMonitor;
Monitor.FMonitorNum := Screen.FMonitors.Add(Monitor);
Result := True;
end;
function EnumFontsNoDups(var LogFont: TEnumLogFontEx; var Metric: TNewTextMetricEx;
FontType: Longint; Data: LParam): LongInt; extdecl;
var
L: TStringList;
S: String;
begin
L := TStringList(Data);
Assert(not L.Sorted, 'EnumFontsNoDups: List of fonts has Sorted=True.');
Assert(not L.UseLocale, 'EnumFontsNoDups: List of fonts has UseLocale=True.');
S := LogFont.elfLogFont.lfFaceName;
// Prevent consecutive duplicates. The list is alphabetically sorted in all platforms.
if (L.Count=0) or (L[L.Count-1] <> S) then begin
// There should be no duplicates elsewhere in the list.
Assert(L.IndexOf(S)<0,'EnumFontsNoDups: Unexpected duplicate font "'+S+'"');
L.Add(S);
end;
Result := 1;
end;
function ScreenGetSystemFont: HFont;
begin
Result := Screen.SystemFont.Reference.Handle;
end;
{------------------------------------------------------------------------------
Method: TScreen.Create
Params: AOwner: the owner of the class
Returns: Nothing
Constructor for the class.
------------------------------------------------------------------------------}
constructor TScreen.Create(AOwner : TComponent);
begin
inherited Create(AOwner);
FFonts := TStringList.Create; // FFonts will be sorted in GetFonts.
FFonts.UseLocale := False;
//FFonts.CaseSensitive := True;
FCursorMap := TMap.Create(its4, SizeOf(HCursor));
FMonitors := TMonitorList.Create;
FCustomForms := TFPList.Create;
FCustomFormsZOrdered := TFPList.Create;
FFormList := TFPList.Create;
FDataModuleList := TFPList.Create;
FPixelsPerInch := ScreenInfo.PixelsPerInchX;
FSaveFocusedList := TFPList.Create;
FMagnetManager := TWindowMagnetManager.Create;
AddDataModule := @DoAddDataModule;
RemoveDataModule := @DoRemoveDataModule;
OnGetSystemFont := @ScreenGetSystemFont;
end;
{------------------------------------------------------------------------------
Method: TScreen.Destroy
Params: None
Returns: Nothing
Destructor for the class.
------------------------------------------------------------------------------}
destructor TScreen.Destroy;
var
HandlerType: TScreenNotification;
begin
for HandlerType := Low(FScreenHandlers) to High(FScreenHandlers) do
FreeThenNil(FScreenHandlers[HandlerType]);
AddDataModule := Nil;
RemoveDataModule := Nil;
FreeThenNil(FHintFont);
FreeThenNil(FIconFont);
FreeThenNil(FMenuFont);
FreeThenNil(FSystemFont);
FreeThenNil(FDataModuleList);
FreeThenNil(FFormList);
FreeThenNil(FCustomForms);
FreeThenNil(FCustomFormsZOrdered);
FreeThenNil(FSaveFocusedList);
FreeThenNil(FFonts);
FreeThenNil(FMagnetManager);
// DestroyCursors; - free on widgetset free
FCursorMap.Free;
FMonitors.Free;
inherited Destroy;
end;
{------------------------------------------------------------------------------
function TScreen.CustomFormIndex(AForm: TCustomForm): integer;
------------------------------------------------------------------------------}
function TScreen.CustomFormIndex(AForm: TCustomForm): Integer;
begin
Result := FCustomForms.Count - 1;
while (Result >= 0) and (CustomForms[Result] <> AForm) do
Dec(Result);
end;
{------------------------------------------------------------------------------
function TScreen.FormIndex(AForm: TForm): integer;
------------------------------------------------------------------------------}
function TScreen.FormIndex(AForm: TForm): Integer;
begin
Result := FFormList.Count - 1;
while (Result >= 0) and (Forms[Result] <> AForm) do
Dec(Result);
end;
{------------------------------------------------------------------------------
function TScreen.CustomFormZIndex(AForm: TCustomForm): integer;
------------------------------------------------------------------------------}
function TScreen.CustomFormZIndex(AForm: TCustomForm): Integer;
begin
Result:=FCustomFormsZOrdered.Count-1;
while (Result >= 0) and (CustomFormsZOrdered[Result] <> AForm) do dec(Result);
end;
procedure TScreen.MoveFormToFocusFront(ACustomForm: TCustomForm);
begin
if (Self = nil) or (ACustomForm = nil) or
(csDestroying in ACustomForm.ComponentState) or
(FCustomForms.IndexOf(ACustomForm)<0)
then
LazTracer.RaiseGDBException('TScreen.MoveFormToFocusFront');
if (FCustomForms.Count=0) or (TObject(FCustomForms[0])<>ACustomForm) then
begin
FCustomForms.Remove(ACustomForm);
FCustomForms.Insert(0, ACustomForm);
end;
if ACustomForm is TForm then
begin
if (FFormList.Count=0) or (TObject(FFormList[0])<>ACustomForm) then
begin
FFormList.Remove(ACustomForm);
FFormList.Insert(0, ACustomForm);
end;
end;
MoveFormToZFront(ACustomForm);
end;
procedure TScreen.MoveFormToZFront(ACustomForm: TCustomForm);
begin
if (Self = nil) or (ACustomForm = nil) or
(csDestroying in ACustomForm.ComponentState) or
(FCustomForms.IndexOf(ACustomForm)<0)
then
LazTracer.RaiseGDBException('TScreen.MoveFormToZFront');
if (FCustomFormsZOrdered.Count = 0) or
(TObject(FCustomFormsZOrdered[0]) <> ACustomForm) then
begin
FCustomFormsZOrdered.Remove(ACustomForm);
FCustomFormsZOrdered.Insert(0, ACustomForm);
end;
end;
procedure TScreen.NewFormWasCreated(AForm: TCustomForm);
begin
NotifyScreenFormHandler(snNewFormCreated,AForm);
end;
function TScreen.GetCurrentModalForm: TCustomForm;
var
i: Integer;
begin
i := GetCurrentModalFormZIndex;
if (i >= 0) then
Result := CustomFormsZOrdered[i]
else
Result := nil;
end;
function TScreen.GetCurrentModalFormZIndex: Integer;
begin
Result := 0;
while (Result < CustomFormCount) and not (fsModal in CustomFormsZOrdered[Result].FormState) do
inc(Result);
if Result = CustomFormCount then Result := -1;
end;
function TScreen.CustomFormBelongsToActiveGroup(AForm: TCustomForm): Boolean;
var
CurForm: TCustomForm;
i: Integer;
begin
i := 0;
Result := False;
while (i < CustomFormCount) do
begin
CurForm := CustomFormsZOrdered[i];
if CurForm = AForm then
Result := True;
if fsModal in CurForm.FormState then exit;
inc(i);
end;
end;
function TScreen.FindNonDesignerForm(const FormName: string): TCustomForm;
var
i: Integer;
begin
for i := 0 to FCustomForms.Count - 1 do
begin
Result:=TCustomForm(FCustomForms[i]);
if (not (csDesigning in Result.ComponentState))
and (CompareText(Result.Name, FormName) = 0) then
exit;
end;
Result := nil;
end;
function TScreen.FindForm(const FormName: string): TCustomForm;
var
i: Integer;
begin
for i := 0 to FCustomForms.Count - 1 do
if CompareText(TCustomForm(FCustomForms[i]).Name, FormName) = 0 then
Exit(TCustomForm(FCustomForms[i]));
Result := nil;
end;
function TScreen.FindNonDesignerDataModule(const DataModuleName: string
): TDataModule;
var
i: Integer;
begin
for i:=0 to FDataModuleList.Count-1 do
begin
Result:=TDataModule(FDataModuleList[i]);
if (not (csDesigning in Result.ComponentState))
and (CompareText(Result.Name, DataModuleName) = 0) then
Exit;
end;
Result := nil;
end;
function TScreen.FindDataModule(const DataModuleName: string): TDataModule;
var
i: Integer;
begin
for i:=0 to FDataModuleList.Count-1 do
begin
Result:=TDataModule(FDataModuleList[i]);
if (CompareText(Result.Name, DataModuleName) = 0) then
Exit;
end;
Result := nil;
end;
procedure TScreen.UpdateScreen;
begin
FPixelsPerInch := ScreenInfo.PixelsPerInchX;
end;
procedure TScreen.RemoveAllHandlersOfObject(AnObject: TObject);
var
HandlerType: TScreenNotification;
begin
for HandlerType:=Low(TScreenNotification) to High(TScreenNotification) do
FScreenHandlers[HandlerType].RemoveAllMethodsOfObject(AnObject);
inherited RemoveAllHandlersOfObject(AnObject);
end;
procedure TScreen.AddHandlerNewFormCreated(OnNewFormCreated: TScreenFormEvent;
AsFirst: Boolean);
begin
AddHandler(snNewFormCreated,TMethod(OnNewFormCreated),AsFirst);
end;
procedure TScreen.RemoveHandlerNewFormCreated(OnNewFormCreated: TScreenFormEvent);
begin
RemoveHandler(snNewFormCreated,TMethod(OnNewFormCreated));
end;
procedure TScreen.AddHandlerFormAdded(OnFormAdded: TScreenFormEvent;
AsFirst: Boolean);
begin
AddHandler(snFormAdded,TMethod(OnFormAdded),AsFirst);
end;
procedure TScreen.RemoveHandlerFormAdded(OnFormAdded: TScreenFormEvent);
begin
RemoveHandler(snFormAdded,TMethod(OnFormAdded));
end;
procedure TScreen.AddHandlerRemoveForm(OnRemoveForm: TScreenFormEvent;
AsFirst: Boolean);
begin
AddHandler(snRemoveForm,TMethod(OnRemoveForm),AsFirst);
end;
procedure TScreen.BeginScreenCursor;
begin
BeginTempCursor(crScreen);
end;
procedure TScreen.BeginTempCursor(const aCursor: TCursor);
var
OldCursor: TCursor;
begin
OldCursor := RealCursor;
SetLength(FTempCursors, Length(FTempCursors)+1);
FTempCursors[High(FTempCursors)] := aCursor;
if OldCursor<>aCursor then
WidgetSet.SetCursor(Cursors[aCursor]);
end;
procedure TScreen.BeginWaitCursor;
begin
BeginTempCursor(crHourGlass);
end;
procedure TScreen.RemoveHandlerRemoveForm(OnRemoveForm: TScreenFormEvent);
begin
RemoveHandler(snRemoveForm,TMethod(OnRemoveForm));
end;
procedure TScreen.AddHandlerActiveControlChanged(
OnActiveControlChanged: TScreenControlEvent; AsFirst: Boolean);
begin
AddHandler(snActiveControlChanged,TMethod(OnActiveControlChanged),AsFirst);
end;
procedure TScreen.RemoveHandlerActiveControlChanged(
OnActiveControlChanged: TScreenControlEvent);
begin
RemoveHandler(snActiveControlChanged,TMethod(OnActiveControlChanged));
end;
procedure TScreen.AddHandlerActiveFormChanged(
OnActiveFormChanged: TScreenFormEvent; AsFirst: Boolean);
begin
AddHandler(snActiveFormChanged,TMethod(OnActiveFormChanged),AsFirst);
end;
procedure TScreen.RemoveHandlerActiveFormChanged(
OnActiveFormChanged: TScreenFormEvent);
begin
RemoveHandler(snActiveFormChanged,TMethod(OnActiveFormChanged));
end;
procedure TScreen.AddHandlerFormVisibleChanged(
OnFormVisibleChanged: TScreenFormEvent; AsFirst: Boolean);
begin
AddHandler(snFormVisibleChanged,TMethod(OnFormVisibleChanged),AsFirst);
end;
procedure TScreen.RemoveHandlerFormVisibleChanged(
OnFormVisibleChanged: TScreenFormEvent);
begin
RemoveHandler(snFormVisibleChanged,TMethod(OnFormVisibleChanged));
end;
{
Disable all forms except SkipForm. If DisabledList is available then add forms
to that list and return it, otherwise return new list. Used to show forms and
other dialogs modal
}
function TScreen.DisableForms(SkipForm: TCustomForm; DisabledList: TList = nil): TList;
procedure ProcessWindow(AWindow: HWND); inline;
begin
if IsWindowEnabled(AWindow) and IsWindowVisible(AWindow) then
begin
EnableWindow(AWindow, False);
Result.Add(Pointer(AWindow));
end;
end;
var
i: integer;
AForm: TCustomForm;
begin
Result := DisabledList;
if Result = nil then
Result := TList.Create;
for i := 0 to CustomFormCount - 1 do
begin
AForm := CustomForms[i];
// we cannot use AForm.Enabled := False; since it checks csDesigning
// but we need this for IDE too. We also cannot check AForm.Visible -
// it returns wrong info for the forms opened in the designer
if (AForm <> SkipForm) and AForm.HandleAllocated then
ProcessWindow(AForm.Handle);
end;
if WidgetSet.GetLCLCapability(lcApplicationWindow) = LCL_CAPABILITY_YES then
ProcessWindow(WidgetSet.AppHandle);
end;
{
Enable all forms from AFormList and destroy list. Used to restore state after
DisableForms
}
procedure TScreen.EnableForms(var AFormList: TList);
function WindowExists(AWindow: HWND): Boolean;
var
I: Integer;
begin
Result := (WidgetSet.GetLCLCapability(lcApplicationWindow) = LCL_CAPABILITY_YES) and
(WidgetSet.AppHandle = AWindow);
if not Result then
begin
I := FCustomForms.Count - 1;
while (I >= 0) and not (CustomForms[I].HandleAllocated and (CustomForms[I].Handle = AWindow)) do
Dec(I);
Result := I >= 0;
end;
end;
var
i: integer;
AWindow: HWND;
begin
if AFormList = nil then
Exit;
for i := AFormList.Count - 1 downto 0 do
begin
AWindow := HWND(AFormList[i]);
if WindowExists(AWindow) then
EnableWindow(AWindow, True);
end;
FreeAndNil(AFormList);
end;
procedure TScreen.EndScreenCursor;
begin
EndTempCursor(crScreen);
end;
procedure TScreen.EndTempCursor(const aCursor: TCursor);
var
I: Integer;
OldCursor: TCursor;
begin
OldCursor := RealCursor;
for I := High(FTempCursors) downto Low(FTempCursors) do
begin
if FTempCursors[I]=aCursor then
begin
Delete(FTempCursors,I,1);
if OldCursor<>RealCursor then
WidgetSet.SetCursor(Cursors[RealCursor]);
Exit;
end;
end;
raise Exception.CreateFmt('Unbalanced BeginTempCursor/EndTempCursor calls for cursor %d', [aCursor]);
end;
procedure TScreen.EndWaitCursor;
begin
EndTempCursor(crHourGlass);
end;
function TScreen.UpdatedMonitor(AHandle: HMONITOR; ADefault: TMonitorDefaultTo;
AErrorMsg: string): TMonitor;
var
i: Integer;
begin
Result := nil;
if (AHandle = 0) and (ADefault = mdNull) then
Exit;
// the user expects some monitor but handle wasn't found -> the monitor list has probably changed
UpdateMonitors;
for i := 0 to MonitorCount - 1 do
if Monitors[i].Handle = AHandle then
Exit(Monitors[i]);
LazTracer.RaiseGDBException(AErrorMsg); // internal error
end;
function TScreen.MonitorFromPoint(const Point: TPoint;
MonitorDefault: TMonitorDefaultTo): TMonitor;
var
MonitorHandle: HMONITOR;
i: integer;
R: TRect;
begin
MonitorHandle := WidgetSet.MonitorFromPoint(Point, MonitorSearchFlags[MonitorDefault]);
for i := 0 to MonitorCount - 1 do
if Monitors[i].Handle = MonitorHandle then
Exit(Monitors[i]);
if MonitorHandle = MONITOR_UNIMPL then
begin
R.TopLeft := Point;
R.BottomRight := Types.Point(Point.X + 1, Point.Y + 1);
Result := MonitorFromRect(R, MonitorDefault);
end
else
Result := UpdatedMonitor(MonitorHandle, MonitorDefault, 'TScreen.MonitorFromPoint');
end;
function TScreen.MonitorFromRect(const Rect: TRect;
MonitorDefault: TMonitorDefaultTo): TMonitor;
var
MonitorHandle: HMONITOR;
i, Square, Distance, BestSquare, BestDistance: integer;
MonitorRect, Intersection: TRect;
Nearest: TMonitor;
begin
MonitorHandle := WidgetSet.MonitorFromRect(@Rect, MonitorSearchFlags[MonitorDefault]);
for i := 0 to MonitorCount - 1 do
if Monitors[i].Handle = MonitorHandle then
Exit(Monitors[i]);
// we are here => interface does not support our search functions
if MonitorHandle = MONITOR_UNIMPL then
begin
Result := nil;
BestSquare := -1;
BestDistance := MaxInt;
Nearest := nil;
for i := 0 to MonitorCount - 1 do
begin
MonitorRect := Monitors[i].BoundsRect;
if IntersectRect(Intersection, Rect, MonitorRect) then
begin
Square := Abs(Intersection.Right - Intersection.Left)
* Abs(Intersection.Bottom - Intersection.Top);
if Square > BestSquare then
begin
BestSquare := Square;
Result := Monitors[i];
end
end;
if MonitorDefault = mdNearest then
begin
Distance := Min(Min(Abs(Rect.Left - MonitorRect.Right), Abs(Rect.Right - MonitorRect.Left)),
Min(Abs(Rect.Top - MonitorRect.Bottom), Abs(Rect.Bottom - MonitorRect.Top)));
if Distance < BestDistance then
begin
BestDistance := Distance;
Nearest := Monitors[i];
end;
end;
end;
if Result = nil then
case MonitorDefault of
mdPrimary: Result := PrimaryMonitor;
mdNull : Result := nil;
mdNearest: Result := Nearest;
end;
end
else
Result := UpdatedMonitor(MonitorHandle, MonitorDefault, 'TScreen.MonitorFromRect');
end;
function TScreen.MonitorFromWindow(const Handle: TLCLHandle;
MonitorDefault: TMonitorDefaultTo): TMonitor;
var
MonitorHandle: HMONITOR;
i: integer;
R: TRect;
begin
MonitorHandle := WidgetSet.MonitorFromWindow(Handle, MonitorSearchFlags[MonitorDefault]);
for i := 0 to MonitorCount - 1 do
if Monitors[i].Handle = MonitorHandle then
Exit(Monitors[i]);
if MonitorHandle = MONITOR_UNIMPL then
begin
GetWindowRect(Handle, R);
Result := MonitorFromRect(R, MonitorDefault);
end
else
Result := UpdatedMonitor(MonitorHandle, MonitorDefault, 'TScreen.MonitorFromWindow');
end;
{------------------------------------------------------------------------------
function TScreen.GetFonts : TStrings;
------------------------------------------------------------------------------}
function TScreen.GetFonts : TStrings;
var
lf: TLogFont;
DC: HDC;
begin
if FFonts.Count = 0 then
begin
lf.lfCharSet := DEFAULT_CHARSET;
lf.lfFaceName := '';
lf.lfPitchAndFamily := 0;
DC := GetDC(0);
try
EnumFontFamiliesEx(DC, @lf, @EnumFontsNoDups, PtrInt(FFonts), 0);
finally
ReleaseDC(0, DC);
end;
// Widgetset already provided a sorted list, but this enables fast binary search.
FFonts.Sorted:=True;
end;
Result := FFonts;
end;
{------------------------------------------------------------------------------
procedure TScreen.DeleteCursor(Index: Integer);
------------------------------------------------------------------------------}
procedure TScreen.DeleteCursor(AIndex: Integer);
var
ACursor: HCursor;
begin
if not FCursorMap.GetData(AIndex, ACursor) then Exit;
WidgetSet.DestroyCursor(ACursor);
FCursorMap.Delete(AIndex);
end;
{------------------------------------------------------------------------------
procedure TScreen.DestroyCursors;
------------------------------------------------------------------------------}
procedure TScreen.DestroyCursors;
var
Iterator: TMapIterator;
ACursor: HCURSOR;
begin
Iterator := TMapIterator.Create(FCursorMap);
Iterator.First;
while not Iterator.EOM do
begin
Iterator.GetData(ACursor);
WidgetSet.DestroyCursor(ACursor);
Iterator.Next;
end;
Iterator.Free;
FCursorMap.Clear;
end;
procedure TScreen.DestroyMonitors;
begin
FMonitors.Clear;
end;
{------------------------------------------------------------------------------
function TScreen.GetCursors(Index: Integer): HCURSOR;
------------------------------------------------------------------------------}
function TScreen.GetCursors(AIndex: Integer): HCURSOR;
var
CursorImage: TCursorImage;
begin
Result := 0;
//if AIndex = crNone then Exit;
if FCursorMap.GetData(AIndex, Result) then Exit;
Result := FDefaultCursor;
if AIndex > crHigh then Exit;
if AIndex < crLow then Exit;
// not yet loaded
Result := WidgetSet.CreateStandardCursor(AIndex);
if (Result = 0) and (AIndex <> crDefault) then
begin
CursorImage := TCursorImage.Create;
try
CursorImage.LoadFromResourceName(hInstance, 'cur_' + IntToStr(-AIndex));
Result := CursorImage.ReleaseHandle;
finally
CursorImage.Free;
end;
end;
if Result = 0 then
Exit;
FCursorMap.Add(AIndex, Result);
end;
{------------------------------------------------------------------------------
function TScreen.GetCustomFormCount: Integer;
------------------------------------------------------------------------------}
function TScreen.GetCustomFormCount: Integer;
begin
Result := FCustomForms.Count;
end;
function TScreen.GetCustomFormZOrderCount: Integer;
begin
Result := FCustomFormsZOrdered.Count;
end;
{------------------------------------------------------------------------------
function TScreen.GetCustomForms(Index: Integer): TCustomForm;
------------------------------------------------------------------------------}
function TScreen.GetCustomForms(Index: Integer): TCustomForm;
begin
Result := TCustomForm(FCustomForms[Index]);
end;
{------------------------------------------------------------------------------
function TScreen.GetCustomFormsZOrdered(Index: Integer): TCustomForm;
------------------------------------------------------------------------------}
function TScreen.GetCustomFormsZOrdered(Index: Integer): TCustomForm;
begin
Result := TCustomForm(FCustomFormsZOrdered[Index]);
end;
function TScreen.GetDataModuleCount: Integer;
begin
Result := FDataModuleList.Count;
end;
function TScreen.GetDataModules(AIndex: Integer): TDataModule;
begin
Result := TDataModule(FDataModuleList.Items[AIndex]);
end;
function TScreen.GetDesktopHeight: Integer;
begin
Result := GetSystemMetrics(SM_CYVIRTUALSCREEN);
end;
function TScreen.GetDesktopWidth: Integer;
begin
Result := GetSystemMetrics(SM_CXVIRTUALSCREEN);
end;
function TScreen.GetDesktopLeft: Integer;
begin
Result := GetSystemMetrics(SM_XVIRTUALSCREEN);
end;
function TScreen.GetDesktopRect: TRect;
begin
with Result do
begin
Left := DesktopLeft;
Top := DesktopTop;
Right := Left + DesktopWidth;
Bottom := Top + DesktopHeight;
end;
end;
function TScreen.GetDesktopTop: Integer;
begin
Result := GetSystemMetrics(SM_YVIRTUALSCREEN);
end;
function TScreen.GetWorkAreaLeft: Integer;
begin
Result := WorkAreaRect.Left;
end;
function TScreen.GetWorkAreaRect: TRect;
begin
SystemParametersInfo(SPI_GETWORKAREA, 0, @Result, 0);
end;
function TScreen.GetWorkAreaTop: Integer;
begin
Result := WorkAreaRect.Top;
end;
function TScreen.GetWorkAreaHeight: Integer;
begin
with WorkAreaRect do Result := Bottom - Top;
end;
function TScreen.GetWorkAreaWidth: Integer;
begin
with WorkAreaRect do Result := Right - Left;
end;
{------------------------------------------------------------------------------
Function: TScreen.AddForm
Params: FForm: The form to be added
Returns: Nothing
Do not use this procedure. This procedure is used by TScreen internally.
------------------------------------------------------------------------------}
procedure TScreen.AddForm(AForm: TCustomForm);
begin
FCustomForms.Add(AForm);
FCustomFormsZOrdered.Add(AForm);
if AForm is TForm then
begin
FFormList.Add(AForm);
Application.UpdateVisible;
end;
NotifyScreenFormHandler(snFormAdded,AForm);
end;
{------------------------------------------------------------------------------
Function: TScreen.GetFormCount
Params: none
Returns: The count of forms. (TODO: discribe this better; my English is not perfect)
Returns the count of forms. (TODO: discribe this better; my English is not perfect)
------------------------------------------------------------------------------}
function TScreen.GetFormCount: Integer;
begin
Result := FFormList.Count;
end;
{------------------------------------------------------------------------------
Function: TScreen.GetForms
Params: IIndex: The index of the form
Returns: A form stored in FFormList
This function is used by the Forms property.
------------------------------------------------------------------------------}
function TScreen.GetForms(IIndex: Integer): TForm;
begin
Result := TForm(FFormList.Items[IIndex]);
end;
{------------------------------------------------------------------------------
Method: TScreen.GetWidth
Params: none
Returns: Screen Width
returns the screen width
------------------------------------------------------------------------------}
function TScreen.GetWidth : Integer;
begin
Result := GetSystemMetrics(SM_CXSCREEN);
end;
{------------------------------------------------------------------------------
Method: TScreen.GetHeight
Params: none
Returns: Screen Height
Returns the Screen Height
------------------------------------------------------------------------------}
function TScreen.GetHeight : Integer;
begin
Result := GetSystemMetrics(SM_CYSCREEN);
end;
function TScreen.GetMonitor(Index: Integer): TMonitor;
begin
if FMonitors.Count = 0 then
UpdateMonitors;
Result := FMonitors[Index];
end;
function TScreen.GetMonitorCount: Integer;
begin
if FMonitors.Count = 0 then
UpdateMonitors;
Result := FMonitors.Count;
end;
function TScreen.GetPrimaryMonitor: TMonitor;
var
i: integer;
begin
for i := 0 to MonitorCount - 1 do
if Monitors[i].Primary then
Exit(Monitors[i]);
Result := nil;
end;
function TScreen.GetRealCursor: TCursor;
begin
if (Length(FTempCursors)>0) and (FTempCursors[High(FTempCursors)]<>crScreen) then
Result := FTempCursors[High(FTempCursors)]
else
if Cursor<>crScreen then
Result := Cursor
else // Screen.Cursor=crScreen - we have to use crDefault because crScreen is otherwise invalid
Result := crDefault;
end;
function TScreen.GetSystemFont: TFont;
begin
if (FSystemFont = nil) then
FSystemFont := TFont.Create;
if not WidgetSet.InitStockFont(FSystemFont, sfSystem) then
begin
FSystemFont.FontData := DefFontData;
FSystemFont.Color := {$ifdef UseCLDefault}clDefault{$else}clWindowText{$endif};
end;
Result := FSystemFont;
end;
procedure TScreen.Notification(AComponent: TComponent; Operation: TOperation);
begin
inherited Notification(AComponent, Operation);
if Operation=opRemove then
begin
if FLastActiveControl=AComponent then
FLastActiveControl:=nil;
if FLastActiveCustomForm=AComponent then
FLastActiveCustomForm:=nil;
end;
end;
function TScreen.GetHintFont: TFont;
begin
if (FHintFont = nil) then
begin
FHintFont := TFont.Create;
if not WidgetSet.InitStockFont(FHintFont, sfHint) then
begin
FHintFont.FontData := DefFontData;
FHintFont.Color := clInfoText;
end;
end;
Result := FHintFont;
end;
function TScreen.GetIconFont: TFont;
begin
if (FIconFont = nil) then
FIconFont := TFont.Create;
if not WidgetSet.InitStockFont(FIconFont, sfIcon) then
begin
FIconFont.FontData := DefFontData;
FIconFont.Color := {$ifdef UseCLDefault}clDefault{$else}clWindowText{$endif};
end;
Result := FIconFont;
end;
function TScreen.GetMenuFont: TFont;
begin
if (FMenuFont = nil) then
FMenuFont := TFont.Create;
if not WidgetSet.InitStockFont(FMenuFont, sfMenu) then
begin
FMenuFont.FontData := DefFontData;
FMenuFont.Color := clMenuText;
end;
Result := FMenuFont;
end;
{------------------------------------------------------------------------------
Function: TScreen.RemoveForm
Params: FForm: The form to be removed
Returns: Nothing
------------------------------------------------------------------------------}
procedure TScreen.RemoveForm(AForm: TCustomForm);
begin
NotifyScreenFormHandler(snRemoveForm,AForm);
FCustomForms.Remove(AForm);
FCustomFormsZOrdered.Remove(AForm);
FFormList.Remove(AForm);
if FFocusedForm=AForm then FFocusedForm:=nil;
if FLastActiveCustomForm=AForm then FLastActiveCustomForm:=nil;
if FActiveForm=AForm then FActiveForm:=nil;
if FActiveCustomForm=AForm then FActiveCustomForm:=nil;
Application.UpdateVisible;
end;
function TScreen.SetFocusedForm(AForm: TCustomForm): Boolean;
var
LastState: TFocusState;
begin
// result determins if focused state has changed during Activate/Deactivate events
// if so we should return False (since activate/deactivate failed)
Result := True;
if FFocusedForm <> AForm then
begin
// send deactivate to the previosly focused form
LastState := SaveFocusState;
if FFocusedForm <> nil then
FFocusedForm.Perform(CM_DEACTIVATE, 0, 0);
if SaveFocusState <> LastState then
begin
FFocusedForm := nil;
Exit(False);
end;
// send activate to the newly focused form
FFocusedForm := AForm;
LastState := SaveFocusState;
if FFocusedForm <> nil then
FFocusedForm.Perform(CM_ACTIVATE, 0, 0);
if SaveFocusState <> LastState then
Exit(False);
end;
end;
{------------------------------------------------------------------------------
procedure TScreen.SetCursor(const AValue: TCursor);
------------------------------------------------------------------------------}
procedure TScreen.SetCursor(const AValue: TCursor);
var
OldCursor: TCursor;
begin
if AValue <> Cursor then
begin
OldCursor := RealCursor;
FCursor := AValue;
if OldCursor<>RealCursor then
WidgetSet.SetCursor(Cursors[RealCursor]);
end;
end;
{------------------------------------------------------------------------------
procedure TScreen.SetCursors(Index: Integer; const AValue: HCURSOR);
------------------------------------------------------------------------------}
procedure TScreen.SetCursors(AIndex: Integer; const AValue: HCURSOR);
begin
case AIndex of
crDefault: begin
if (AValue = 0) and (WidgetSet <> nil) then
FDefaultCursor := WidgetSet.CreateStandardCursor(crDefault)
else
FDefaultCursor := AValue
end;
{ crNone: begin
end;}
else
DeleteCursor(AIndex);
if AValue <> 0 then
FCursorMap.Add(AIndex, AValue);
end;
end;
procedure TScreen.SetHintFont(const AValue: TFont);
begin
if (FHintFont = nil) then
FHintFont := TFont.Create;
FHintFont.Assign(AValue);
end;
procedure TScreen.SetIconFont(const AValue: TFont);
begin
if (FIconFont = nil) then
FIconFont := TFont.Create;
FIconFont.Assign(AValue);
end;
procedure TScreen.SetMenuFont(const AValue: TFont);
begin
if (FMenuFont = nil) then
FMenuFont := TFont.Create;
FMenuFont.Assign(AValue);
end;
procedure TScreen.SetSystemFont(const AValue: TFont);
begin
if (FSystemFont = nil) then
FSystemFont := TFont.Create;
FSystemFont.Assign(AValue);
end;
{------------------------------------------------------------------------------
procedure TScreen.UpdateLastActive;
------------------------------------------------------------------------------}
procedure TScreen.UpdateLastActive;
procedure NotifyOnActiveFormChanged;
begin
if Assigned(FOnActiveFormChange) then FOnActiveFormChange(Self);
NotifyScreenFormHandler(snActiveFormChanged,FLastActiveCustomForm);
end;
procedure NotifyOnActiveControlChanged;
var
i: Integer;
Handler: TScreenControlEvent;
begin
if Assigned(FOnActiveControlChange) then FOnActiveControlChange(Self);
i := FScreenHandlers[snActiveControlChanged].Count;
while FScreenHandlers[snActiveControlChanged].NextDownIndex(i) do
begin
Handler := TScreenControlEvent(FScreenHandlers[snActiveControlChanged][i]);
Handler(Self, FLastActiveControl);
end;
end;
begin
if FLastActiveCustomForm <> FActiveCustomForm then
begin
FLastActiveCustomForm := FActiveCustomForm;
NotifyOnActiveFormChanged;
end;
if FLastActiveControl <> FActiveControl then
begin
FLastActiveControl := FActiveControl;
if FLastActiveControl<>nil then
FreeNotification(FLastActiveControl);
NotifyOnActiveControlChanged;
end;
end;
procedure TScreen.UpdateMonitors;
begin
DestroyMonitors;
EnumDisplayMonitors(0, nil, @EnumMonitors, LParam(Self));
end;
procedure TScreen.RestoreLastActive;
function CanFocusForm(AControl: TWinControl): Boolean; inline;
var
AForm: TCustomForm;
begin
AForm := GetParentForm(AControl);
Result := (AForm <> nil) and AForm.CanFocus;
end;
begin
if (FLastActiveControl <> nil) and FLastActiveControl.CanFocus and
CanFocusForm(FLastActiveControl) then
begin
// when mdiform is restored it should raise it's active mdi child
// not focus itself.
if (FLastActiveControl = FLastActiveCustomForm) and
(FLastActiveCustomForm.FormStyle = fsMDIForm) and
(FLastActiveCustomForm.ActiveMDIChild <> nil) then
FLastActiveCustomForm.ActiveMDIChild.BringToFront
else
FLastActiveControl.SetFocus;
end else
if (FLastActiveCustomForm <> nil) and FLastActiveCustomForm.CanFocus then
FLastActiveCustomForm.SetFocus;
end;
{------------------------------------------------------------------------------
procedure TScreen.AddHandler(HandlerType: TScreenNotification;
const Handler: TMethod);
------------------------------------------------------------------------------}
procedure TScreen.AddHandler(HandlerType: TScreenNotification;
const Handler: TMethod; AsFirst: Boolean);
begin
if Handler.Code = nil then
LazTracer.RaiseGDBException('TScreen.AddHandler');
if FScreenHandlers[HandlerType] = nil then
FScreenHandlers[HandlerType] := TMethodList.Create;
FScreenHandlers[HandlerType].Add(Handler, not AsFirst);
end;
procedure TScreen.RemoveHandler(HandlerType: TScreenNotification; const Handler: TMethod);
begin
FScreenHandlers[HandlerType].Remove(Handler);
end;
procedure TScreen.DoAddDataModule(DataModule: TDataModule);
begin
FDataModuleList.Add(DataModule);
end;
procedure TScreen.DoRemoveDataModule(DataModule: TDataModule);
begin
FDataModuleList.Remove(DataModule);
end;
procedure TScreen.NotifyScreenFormHandler(HandlerType: TScreenNotification;
Form: TCustomForm);
var
i: integer;
Handler: TScreenFormEvent;
begin
i := FScreenHandlers[HandlerType].Count;
while FScreenHandlers[HandlerType].NextDownIndex(i) do
begin
Handler := TScreenFormEvent(FScreenHandlers[HandlerType][i]);
Handler(Self, Form);
end;
end;
// included by forms.pp