lazarus/lcl/interfaces/cocoa/cocoaprivate.pas

1146 lines
31 KiB
ObjectPascal

{ $Id: $}
{ --------------------------------------------
cocoaprivate.pp - Cocoa internal classes
--------------------------------------------
This unit contains the private classhierarchy for the Cocoa implemetations
*****************************************************************************
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.
*****************************************************************************
}
unit CocoaPrivate;
{$mode objfpc}{$H+}
{$modeswitch objectivec1}
{$modeswitch objectivec2}
{$interfaces corba}
{$include cocoadefines.inc}
{.$DEFINE COCOA_DEBUG_SETBOUNDS}
{.$DEFINE COCOA_SPIN_DEBUG}
{.$DEFINE COCOA_SPINEDIT_INSIDE_CONTAINER}
{.$DEFINE COCOA_SUPERVIEW_HEIGHT}
interface
uses
Types, Classes, SysUtils, LCLType, Forms, LazUTF8,
MacOSAll, CocoaAll, CocoaCallback, CocoaCursor, cocoa_extra, CocoaUtils;
const
SPINEDIT_DEFAULT_STEPPER_WIDTH = 15;
SPINEDIT_EDIT_SPACING_FOR_SELECTION = 4;
STATUSBAR_DEFAULT_HEIGHT = 18;
type
// Some components might be using CocoaPrivate for use of LCLObjCBoolean
// Thus this declaration needs to be here.
LCLObjCBoolean = cocoa_extra.LCLObjCBoolean;
{ LCLObjectExtension }
LCLObjectExtension = objccategory(NSObject)
function lclIsEnabled: Boolean; message 'lclIsEnabled';
procedure lclSetEnabled(AEnabled: Boolean); message 'lclSetEnabled:';
function lclIsVisible: Boolean; message 'lclIsVisible';
procedure lclSetVisible(AVisible: Boolean); message 'lclSetVisible:';
function lclWindowState: Integer; message 'lclWindowState';
procedure lclInvalidateRect(const r: TRect); message 'lclInvalidateRect:';
procedure lclInvalidate; message 'lclInvalidate';
procedure lclUpdate; message 'lclUpdate';
// Returns the position of the view or window, in the immediate
// parent (view or screen), relative to its client coordinates system
// Left and Top are always returned in LCL coordinate system.
procedure lclLocalToScreen(var X, Y: Integer); message 'lclLocalToScreen::';
procedure lclScreenToLocal(var X, Y: Integer); message 'lclScreenToLocal::';
function lclParent: id; message 'lclParent';
function lclFrame: TRect; message 'lclFrame';
procedure lclSetFrame(const r: TRect); message 'lclSetFrame:';
// returns rectangle describing deltas to get "Layout" rectangle from "Frame" rectangle
// left, top - return offsets from top-left corner of the control (not reversed as in Cocoa coordinates)
// (values are typically positive)
// right, bottom - offsets for bottom-right corner
// (typically negative)
function lclGetFrameToLayoutDelta: TRect; message 'lclGetFrameToLayoutDelta';
function lclClientFrame: TRect; message 'lclClientFrame';
function lclGetCallback: ICommonCallback; message 'lclGetCallback';
procedure lclClearCallback; message 'lclClearCallback';
function lclGetPropStorage: TStringList; message 'lclGetPropStorage';
function lclGetTarget: TObject; message 'lclGetTarget';
function lclDeliverMessage(Msg: Cardinal; WParam: WParam; LParam: LParam): LResult; message 'lclDeliverMessage:::';
function lclContentView: NSView; message 'lclContentView';
procedure lclOffsetMousePos(var Point: NSPoint); message 'lclOffsetMousePos:';
procedure lclExpectedKeys(var wantTabs, wantArrows, wantReturn, wantAll: Boolean); message 'lclExpectedKeys::::';
function lclIsMouseInAuxArea(Event: NSEvent): Boolean; message 'lclMouseInAuxArea:';
end;
{ LCLViewExtension }
LCLViewExtension = objccategory(NSView)
function lclInitWithCreateParams(const AParams: TCreateParams): id; message 'lclInitWithCreateParams:';
function lclIsEnabled: Boolean; message 'lclIsEnabled'; reintroduce;
procedure lclSetEnabled(AEnabled: Boolean); message 'lclSetEnabled:'; reintroduce;
function lclIsVisible: Boolean; message 'lclIsVisible'; reintroduce;
procedure lclSetVisible(AVisible: Boolean); message 'lclSetVisible:'; reintroduce;
function lclIsPainting: Boolean; message 'lclIsPainting';
procedure lclInvalidateRect(const r: TRect); message 'lclInvalidateRect:'; reintroduce;
procedure lclInvalidate; message 'lclInvalidate'; reintroduce;
procedure lclUpdate; message 'lclUpdate'; reintroduce;
procedure lclLocalToScreen(var X, Y: Integer); message 'lclLocalToScreen::'; reintroduce;
procedure lclScreenToLocal(var X, Y: Integer); message 'lclScreenToLocal::'; reintroduce;
function lclParent: id; message 'lclParent'; reintroduce;
function lclFrame: TRect; message 'lclFrame'; reintroduce;
procedure lclSetFrame(const r: TRect); message 'lclSetFrame:'; reintroduce;
function lclClientFrame: TRect; message 'lclClientFrame'; reintroduce;
function lclContentView: NSView; message 'lclContentView'; reintroduce;
procedure lclOffsetMousePos(var Point: NSPoint); message 'lclOffsetMousePos:'; reintroduce;
end;
{ LCLControlExtension }
LCLControlExtension = objccategory(NSControl)
function lclIsEnabled: Boolean; message 'lclIsEnabled'; reintroduce;
procedure lclSetEnabled(AEnabled: Boolean); message 'lclSetEnabled:'; reintroduce;
end;
{ TCocoaGroupBox }
TCocoaGroupBox = objcclass(NSBox)
public
callback: ICommonCallback;
function acceptsFirstResponder: LCLObjCBoolean; override;
function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override;
function lclClientFrame: TRect; override;
function lclContentView: NSView; override;
function lclGetFrameToLayoutDelta: TRect; override;
end;
const
PROGRESS_REG_HEIGHT = 16; // no longer applies on later macOS version
PROGRESS_SMALL_HEIGHT = 10;
type
{ TCocoaProgressIndicator }
TCocoaProgressIndicator = objcclass(NSProgressIndicator)
callback: ICommonCallback;
function acceptsFirstResponder: LCLObjCBoolean; override;
function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override;
function lclGetFrameToLayoutDelta: TRect; override;
procedure lclSetFrame(const r: TRect); override;
// mouse
function acceptsFirstMouse(event: NSEvent): LCLObjCBoolean; override;
procedure mouseDown(event: NSEvent); override;
procedure mouseUp(event: NSEvent); override;
procedure rightMouseDown(event: NSEvent); override;
procedure rightMouseUp(event: NSEvent); override;
procedure rightMouseDragged(event: NSEvent); override;
procedure otherMouseDown(event: NSEvent); override;
procedure otherMouseUp(event: NSEvent); override;
procedure otherMouseDragged(event: NSEvent); override;
procedure mouseDragged(event: NSEvent); override;
procedure mouseMoved(event: NSEvent); override;
procedure scrollWheel(event: NSEvent); override;
end;
{ TManualTicks }
TManualTicks = class(TObject)
count : integer;
//todo: keep sorted and do binary search
ticks : array of Integer;
draw : Boolean;
function AddTick(atick: integer): Boolean;
end;
{ TCocoaSlider }
TCocoaSlider = objcclass(NSSlider)
callback : ICommonCallback;
intval : Integer;
man : TManualTicks;
procedure dealloc; override;
procedure drawRect(dirtyRect: NSRect); override;
function acceptsFirstResponder: LCLObjCBoolean; override;
function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override;
//
procedure SnapToInteger(AExtraFactor: Integer = 0); message 'SnapToInteger:';
procedure sliderAction(sender: id); message 'sliderAction:';
// mouse
function acceptsFirstMouse(event: NSEvent): LCLObjCBoolean; override;
procedure mouseDown(event: NSEvent); override;
procedure mouseUp(event: NSEvent); override;
procedure rightMouseDown(event: NSEvent); override;
procedure rightMouseUp(event: NSEvent); override;
procedure rightMouseDragged(event: NSEvent); override;
procedure otherMouseDown(event: NSEvent); override;
procedure otherMouseUp(event: NSEvent); override;
procedure otherMouseDragged(event: NSEvent); override;
procedure mouseDragged(event: NSEvent); override;
procedure mouseMoved(event: NSEvent); override;
procedure scrollWheel(event: NSEvent); override;
procedure lclAddManTick(atick : integer); message 'lclAddManTick:';
procedure lclSetManTickDraw(adraw: Boolean); message 'lclSetManTickDraw:';
procedure lclExpectedKeys(var wantTabs, wantArrows, wantReturn, wantAll: Boolean); override;
end;
TCocoaSliderCell = objcclass(NSSliderCell)
end;
procedure SetViewDefaults(AView: NSView);
function CheckMainThread: Boolean;
function GetNSViewSuperViewHeight(view: NSView): CGFloat;
procedure SetNSControlSize(ctrl: NSView; newHeight, miniHeight, smallHeight: Integer; AutoChangeFont: Boolean);
var
// todo: this should be a threadvar
TrackedControl : NSObject = nil;
{$ifdef COCOALOOPHIJACK}
// The flag is set to true once hi-jacked loop is finished (at the end of app)
// The flag is checked in Menus to avoid "double" Cmd+Q menu
LoopHiJackEnded : Boolean = false;
{$endif}
function isCallbackForSameObject(cb1, cb2: ICommonCallback): Boolean;
function NSViewIsLCLEnabled(v: NSView): Boolean;
function NSObjectIsLCLEnabled(obj: NSObject): Boolean;
function NSViewCanFocus(v: NSView): Boolean;
implementation
function NSObjectIsLCLEnabled(obj: NSObject): Boolean;
begin
if obj.isKindOfClass(NSView) then
Result := NSViewIsLCLEnabled(NSView(obj))
else
Result := obj.lclIsEnabled;
end;
function NSViewIsLCLEnabled(v: NSView): Boolean;
begin
Result := true;
while Assigned(v) do
begin
if not v.lclIsEnabled then begin
Result := false;
break;
end;
v:=v.superview;
end;
end;
function NSViewCanFocus(v: NSView): Boolean;
var
cb: ICommonCallback;
begin
if Assigned(v) then
begin
cb := v.lclGetCallback;
if Assigned(cb) then
Result := cb.CanFocus
else
Result := true;
end
else
Result := false;
end;
function isCallbackForSameObject(cb1, cb2: ICommonCallback): Boolean;
begin
Result := Assigned(cb1) and Assigned(cb2);
if Result then
Result := (cb1 = cb2) or (cb1.GetTarget = cb2.GetTarget);
end;
procedure SetViewDefaults(AView: NSView);
var
mask: NSUInteger;
begin
if not Assigned(AView) then Exit;
if Assigned(AView.superview) and AView.superview.isFlipped then
mask:= NSViewMaxYMargin or NSViewMaxXMargin
else
mask:= NSViewMinYMargin or NSViewMaxXMargin;
AView.setAutoresizingMask(mask);
end;
function CheckMainThread: Boolean;
begin
Result := NSThread.currentThread.isMainThread;
end;
function GetNSViewSuperViewHeight(view: NSView): CGFloat;
begin
Result := -1;
if not Assigned(view) then Exit;
if not Assigned(view.superview) then Exit;
//if view.superview.isKindOfClass_(TCocoaTabPageView) then
//Result := TCocoaTabPageView(view.superview).tabview.contentRect.size.height
//else
Result := view.superview.frame.size.height;
{$IFDEF COCOA_SUPERVIEW_HEIGHT}
WriteLn(Format('GetNSViewSuperViewHeight Result=%f', [Result]));
{$ENDIF}
end;
{ TManualTicks }
function TManualTicks.AddTick(atick: integer): Boolean;
var
i : integer;
begin
//todo: must be a binary search
for i:=0 to length(ticks)-1 do
if ticks[i]=atick then begin
Result:=false;
Exit;
end;
// adding new tick
if length(ticks)=count then begin
if count=0 then SetLength(ticks, 8)
else SetLength(ticks, count * 2);
end;
ticks[count]:=atick;
inc(count);
Result := true;
end;
{ TCocoaGroupBox }
function TCocoaGroupBox.lclClientFrame: TRect;
var
v : NSView;
begin
v:=contentView;
if not Assigned(v) then
Result := inherited lclClientFrame
else
if v.isFlipped then
Result := NSRectToRect( v.frame )
else
NSToLCLRect(v.frame, frame.size.height, Result);
end;
function TCocoaGroupBox.lclContentView: NSView;
begin
Result := NSView(contentView);
end;
function TCocoaGroupBox.lclGetFrameToLayoutDelta: TRect;
begin
Result.Left := 3;
Result.Right := -3;
Result.Top := 0;
Result.Bottom := -4;
end;
function TCocoaGroupBox.acceptsFirstResponder: LCLObjCBoolean;
begin
Result := True;
end;
function TCocoaGroupBox.lclGetCallback: ICommonCallback;
begin
Result := callback;
end;
procedure TCocoaGroupBox.lclClearCallback;
begin
callback := nil;
end;
{ LCLObjectExtension }
function LCLObjectExtension.lclIsEnabled: Boolean;
begin
Result := False;
end;
procedure LCLObjectExtension.lclSetEnabled(AEnabled: Boolean);
begin
end;
function LCLObjectExtension.lclIsVisible: Boolean;
begin
Result := False;
end;
procedure LCLObjectExtension.lclSetVisible(AVisible: Boolean);
begin
end;
function LCLObjectExtension.lclWindowState: Integer;
begin
Result := SIZE_RESTORED;
end;
procedure LCLObjectExtension.lclInvalidateRect(const r: TRect);
begin
end;
procedure LCLObjectExtension.lclInvalidate;
begin
end;
procedure LCLObjectExtension.lclUpdate;
begin
end;
procedure LCLObjectExtension.lclLocalToScreen(var X,Y: Integer);
begin
end;
procedure LCLObjectExtension.lclScreenToLocal(var X, Y: Integer);
begin
end;
function LCLObjectExtension.lclParent:id;
begin
Result:=nil;
end;
function LCLObjectExtension.lclFrame:TRect;
begin
FillChar(Result, sizeof(Result), 0);
end;
procedure LCLObjectExtension.lclSetFrame(const r:TRect);
begin
end;
function LCLObjectExtension.lclGetFrameToLayoutDelta: TRect;
begin
Result.Top := 0;
Result.Left := 0;
Result.Right := 0;
Result.Bottom := 0;
end;
function LCLObjectExtension.lclClientFrame:TRect;
begin
FillChar(Result, sizeof(Result), 0);
end;
function LCLObjectExtension.lclGetCallback: ICommonCallback;
begin
Result := nil;
end;
procedure LCLObjectExtension.lclClearCallback;
begin
end;
function LCLObjectExtension.lclGetPropStorage: TStringList;
var
Callback: ICommonCallback;
begin
Callback := lclGetCallback;
if Assigned(Callback) then
Result := Callback.GetPropStorage
else
Result := nil;
end;
function LCLObjectExtension.lclGetTarget: TObject;
var
Callback: ICommonCallback;
begin
Callback := lclGetCallback;
if Assigned(Callback) then
Result := Callback.GetTarget
else
Result := nil;
end;
function LCLObjectExtension.lclDeliverMessage(Msg: Cardinal; WParam: WParam; LParam: LParam): LResult;
var
Callback: ICommonCallback;
begin
Callback := lclGetCallback;
if Assigned(Callback) then
Result := Callback.DeliverMessage(Msg, WParam, LParam)
else
Result := 0;
end;
function LCLObjectExtension.lclContentView: NSView;
begin
Result := nil;
end;
procedure LCLObjectExtension.lclOffsetMousePos(var Point: NSPoint);
begin
end;
procedure LCLObjectExtension.lclExpectedKeys(var wantTabs, wantArrows,
wantReturn, wantAll: Boolean);
begin
wantTabs := false;
wantArrows := false;
wantReturn := false;
wantAll := false;
end;
{ The method should return TRUE, if mouse is located above an auxilary area
of a (composited) control, and thus MOUSE MOVE event should not be propagated
to LCL. For example, controls with Scrollbars should not report mouse events
if mouse cursor is above ScrollBar and scroll bar is visible. (ScrollBar = Auxillary area)
By default, the whole area is considered to be non-auxillary and must be
reported to LCL.
}
function LCLObjectExtension.lclIsMouseInAuxArea(Event: NSEvent): Boolean;
begin
Result := false;
end;
{ LCLControlExtension }
function RectToViewCoord(view: NSView; const r: TRect): NSRect;
var
b: NSRect;
begin
b := view.bounds;
Result.origin.x := r.Left;
Result.size.width := r.Right - r.Left;
Result.size.height := r.Bottom - r.Top;
if Assigned(view) and (view.isFlipped) then
Result.origin.y := r.Top
else
Result.origin.y := b.size.height - r.Bottom;
end;
function LCLControlExtension.lclIsEnabled:Boolean;
begin
Result := IsEnabled;
end;
procedure LCLControlExtension.lclSetEnabled(AEnabled:Boolean);
begin
{$ifdef BOOLFIX}
SetEnabled_( Ord(AEnabled and NSViewIsLCLEnabled(self.superview) ));
{$else}
SetEnabled( AEnabled and NSViewIsLCLEnabled(self.superview) );
{$endif}
inherited lclSetEnabled(AEnabled);
end;
function LCLViewExtension.lclInitWithCreateParams(const AParams: TCreateParams): id;
var
p: NSView;
ns: NSRect;
{$IFDEF COCOA_DEBUG_SETBOUNDS}
pstr: string;
{$ENDIF}
begin
p := nil;
if (AParams.WndParent <> 0) then
p := NSView(AParams.WndParent).lclContentView;
if Assigned(p) and p.isFlipped then
LCLToNSRect(Types.Bounds(AParams.X, AParams.Y, AParams.Width, AParams.Height),
p.frame.size.height, ns)
else
ns := GetNSRect(AParams.X, AParams.Y, AParams.Width, AParams.Height);
{$IFDEF COCOA_DEBUG_SETBOUNDS}
if Assigned(p) then
begin
pstr := NSStringToString(p.className);
if NSStringToString(NSObject(AParams.WndParent).className) = 'TCocoaTabPage' then
pstr := pstr + ' ' + NSStringToString(TCocoaTabPage(AParams.WndParent).label_);
end
else
pstr := '';
WriteLn(Format('[LCLViewExtension.lclInitWithCreateParams] Class=%s Caption=%s ParentClass=%s ParentClassView=%s rect=%d %d %d %d Visible=%d',
[NSStringToString(Self.className), AParams.Caption,
NSStringToString(NSObject(AParams.WndParent).className), pstr,
Round(ns.Origin.x), Round(ns.Origin.y), Round(ns.size.width), Round(ns.size.height),
AParams.Style and WS_VISIBLE]));
{$ENDIF}
Result := initWithFrame(ns);
if not Assigned(Result) then
Exit;
{$ifdef BOOLFIX}
setHidden_(Ord(AParams.Style and WS_VISIBLE = 0));
{$else}
setHidden(AParams.Style and WS_VISIBLE = 0);
{$endif}
if Assigned(p) then
p.lclContentView.addSubview(Result);
SetViewDefaults(Result);
end;
function LCLViewExtension.lclIsEnabled: Boolean;
begin
Result := true;
end;
procedure LCLViewExtension.lclSetEnabled(AEnabled: Boolean);
var
cb : ICommonCallback;
obj : NSObject;
begin
for obj in subviews do begin
cb := obj.lclGetCallback;
obj.lclSetEnabled(AEnabled and ((not Assigned(cb)) or cb.GetShouldBeEnabled) );
end;
end;
function LCLViewExtension.lclIsVisible: Boolean;
begin
Result := not isHidden;
end;
procedure LCLViewExtension.lclSetVisible(AVisible: Boolean);
begin
{$ifdef BOOLFIX}
setHidden_(Ord(not AVisible));
{$else}
setHidden(not AVisible);
{$endif}
{$IFDEF COCOA_DEBUG_SETBOUNDS}
WriteLn(Format('LCLViewExtension.lclSetVisible: %s AVisible=%d',
[NSStringToString(Self.ClassName), Integer(AVisible)]));
{$ENDIF}
end;
function LCLViewExtension.lclIsPainting: Boolean;
begin
Result := Assigned(lclGetCallback) and Assigned(lclGetCallback.GetContext);
end;
procedure LCLViewExtension.lclInvalidateRect(const r:TRect);
var
view : NSView;
begin
view:=lclContentView;
if Assigned(view) then
view.setNeedsDisplayInRect(RectToViewCoord(view, r))
else
self.setNeedsDisplayInRect(RectToViewCoord(Self, r));
//todo: it might be necessary to always invalidate self
// just need to get offset of the contentView relative for self
end;
procedure LCLViewExtension.lclInvalidate;
begin
{$ifdef BOOLFIX}
setNeedsDisplay__(Ord(True));
{$else}
setNeedsDisplay_(True);
{$endif}
end;
procedure LCLViewExtension.lclUpdate;
begin
{$ifdef BOOLFIX}
setNeedsDisplay__(Ord(True));
{$else}
setNeedsDisplay_(True);
{$endif}
//display;
end;
procedure LCLViewExtension.lclLocalToScreen(var X, Y:Integer);
var
P: NSPoint;
scrollView: NSScrollView;
begin
// 1. convert to window base
// Convert from View-lcl to View-cocoa
P.x := X;
if isFlipped then
p.y := Y
else
P.y := frame.size.height-y; // convert to Cocoa system
scrollView:= self.enclosingScrollView;
if Assigned(scrollView) and (scrollView.documentView=self) then begin
P.y:= P.y + Round(scrollView.documentVisibleRect.origin.y);
end;
// Convert from View-cocoa to Window-cocoa
P := convertPoint_ToView(P, nil);
// Convert from Window-cocoa to Window-lcl
X := Round(P.X);
Y := Round(window.frame.size.height-P.Y); // convert to LCL system
// 2. convert window to screen
// Use window function to convert fomr Window-lcl to Screen-lcl
window.lclLocalToScreen(X, Y);
end;
procedure LCLViewExtension.lclScreenToLocal(var X, Y: Integer);
var
P: NSPoint;
begin
// 1. convert from screen to window
// use window function to onvert from Screen-lcl to Window-lcl
window.lclScreenToLocal(X, Y);
// Convert from Window-lcl to Window-cocoa
P.x := X;
P.y := Round(window.frame.size.height-Y); // convert to Cocoa system
// 2. convert from window to local
// Convert from Window-cocoa to View-cocoa
P := convertPoint_FromView(P, nil);
// Convert from View-cocoa to View-lcl
X := Round(P.x);
if isFlipped then
Y := Round(p.y)
else
Y := Round(frame.size.height-P.y); // convert to Cocoa system
end;
function LCLViewExtension.lclParent:id;
begin
Result := superView;
end;
function LCLViewExtension.lclFrame: TRect;
var
v: NSView;
begin
v := superview;
if Assigned(v) and not v.isFlipped then
NSToLCLRect(frame, v.frame.size.height, Result)
else
Result := NSRectToRect(frame);
AddLayoutToFrame( lclGetFrameToLayoutDelta, Result);
end;
procedure LCLViewExtension.lclSetFrame(const r: TRect);
var
ns: NSRect;
svHeight: CGFloat;
rr : TRect;
begin
rr := r;
SubLayoutFromFrame( lclGetFrameToLayoutDelta, rr);
svHeight := GetNSViewSuperViewHeight(Self);
if Assigned(superview) and not superview.isFlipped then
begin
LCLToNSRect(rr, svHeight, ns)
end
else
ns := RectToNSRect(rr);
if ns.size.width<1 then ns.size.width:=1;
if ns.size.height<1 then ns.size.height:=1;
{$IFDEF COCOA_DEBUG_SETBOUNDS}
WriteLn(Format('LCLViewExtension.lclSetFrame: %s Bounds=%s height=%d ns_pos=%d %d ns_size=%d %d',
[NSStringToString(Self.ClassName), dbgs(r), Round(svHeight),
Round(ns.origin.x), Round(ns.origin.y), Round(ns.size.width), Round(ns.size.height)]));
{$ENDIF}
setFrame(ns);
end;
function LCLViewExtension.lclClientFrame: TRect;
begin
Result := lclFrame;
Types.OffsetRect(Result, -Result.Left, -Result.Top);
end;
function LCLViewExtension.lclContentView: NSView;
begin
Result := self;
end;
procedure LCLViewExtension.lclOffsetMousePos(var Point: NSPoint);
var
es : NSScrollView;
r : NSRect;
dlt : TRect;
begin
Point := convertPoint_fromView(Point, nil);
es := enclosingScrollView;
if es.documentView <> self then es := nil;
if not isFlipped then
Point.y := bounds.size.height - Point.y;
if Assigned(es) then
begin
r := es.documentVisibleRect;
if isFlipped then
Point.y := Point.y - r.origin.y
else
Point.y := Point.y - (es.documentView.frame.size.height - r.size.height - r.origin.y);
Point.X := Point.X - r.origin.x;
end;
dlt := lclGetFrameToLayoutDelta;
Point.X := Point.X - dlt.Left;
Point.Y := Point.Y - dlt.Top;
end;
{ TCocoaProgressIndicator }
function TCocoaProgressIndicator.acceptsFirstResponder: LCLObjCBoolean;
begin
Result:=True;
end;
function TCocoaProgressIndicator.lclGetCallback: ICommonCallback;
begin
Result:=callback;
end;
procedure TCocoaProgressIndicator.lclClearCallback;
begin
callback:=nil;
end;
function TCocoaProgressIndicator.lclGetFrameToLayoutDelta: TRect;
begin
case controlSize of
NSSmallControlSize, NSMiniControlSize:
begin
Result.Left := 1;
Result.Right := -1;
Result.Top := 0;
Result.Bottom := -2;
end;
else
Result.Left := 2;
Result.Right := -2;
Result.Top := 0;
Result.Bottom := -4;
end;
end;
procedure TCocoaProgressIndicator.lclSetFrame(const r: TRect);
begin
SetNSControlSize(self, r.Bottom - r.Top, 0, PROGRESS_SMALL_HEIGHT, true);
inherited lclSetFrame(r);
end;
function TCocoaProgressIndicator.acceptsFirstMouse(event: NSEvent): LCLObjCBoolean;
begin
Result:=true;
end;
procedure TCocoaProgressIndicator.mouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
begin
inherited mouseDown(event);
callback.MouseUpDownEvent(event, true);
end;
end;
procedure TCocoaProgressIndicator.mouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited mouseUp(event);
end;
procedure TCocoaProgressIndicator.rightMouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseDown(event);
end;
procedure TCocoaProgressIndicator.rightMouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseUp(event);
end;
procedure TCocoaProgressIndicator.rightMouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseDragged(event);
end;
procedure TCocoaProgressIndicator.otherMouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited otherMouseDown(event);
end;
procedure TCocoaProgressIndicator.otherMouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited otherMouseUp(event);
end;
procedure TCocoaProgressIndicator.otherMouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited otherMouseDragged(event);
end;
procedure TCocoaProgressIndicator.mouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited mouseDragged(event);
end;
procedure TCocoaProgressIndicator.mouseMoved(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited mouseMoved(event);
end;
procedure TCocoaProgressIndicator.scrollWheel(event: NSEvent);
begin
if not Assigned(callback) or not callback.scrollWheel(event) then
inherited scrollWheel(event);
end;
{ TCocoaSlider }
function GetManTicks(slider: TCocoaSlider): TManualTicks;
begin
if not Assigned(slider.man) then
slider.man := TManualTicks.Create;
Result := slider.man;
end;
procedure TCocoaSlider.dealloc;
begin
man.Free;
inherited dealloc;
end;
procedure TCocoaSlider.drawRect(dirtyRect: NSRect);
var
i : integer;
nr : NSRect;
xr : NSRect;
dr : NSRect;
nm : integer;
ctx : NSGraphicsContext;
pth : NSBezierPath;
begin
if not Assigned(man) or (not man.draw) then begin
inherited drawRect(dirtyRect);
Exit;
end;
nm := round(maxValue - minValue);
if nm = 0 then Exit;
if numberOfTickMarks < 2 then Exit;
nr := rectOfTickMarkAtIndex(0);
xr := rectOfTickMarkAtIndex(1);
ctx := NSGraphicsContext.currentContext;
pth := NSBezierPath.bezierPath;
NSColor.controlShadowColor.setFill;
dr:=nr;
dr.origin.y := dr.origin.y + 1;
dr.size.height := dr.size.height - 1;
for i := 0 to man.count - 1 do begin
dr.origin.x := round(nr.origin.x + (xr.origin.x - nr.origin.x) * (man.ticks[i] - minValue) / nm);
pth.fillRect(dr);
end;
inherited drawRect(dirtyRect);
end;
function TCocoaSlider.acceptsFirstResponder: LCLObjCBoolean;
begin
Result := True;
end;
function TCocoaSlider.lclGetCallback: ICommonCallback;
begin
Result:=callback;
end;
procedure TCocoaSlider.lclClearCallback;
begin
callback := nil;
end;
procedure TCocoaSlider.SnapToInteger(AExtraFactor: Integer);
begin
setIntValue(Round(doubleValue() + AExtraFactor));
end;
procedure TCocoaSlider.sliderAction(sender: id);
var
newval: Integer;
begin
SnapToInteger();
newval := intValue;
if newval <> intval then begin
intval := newval;
// OnChange event
if callback <> nil then
callback.SendOnChange();
end;
end;
function TCocoaSlider.acceptsFirstMouse(event: NSEvent): LCLObjCBoolean;
begin
Result:=true;
end;
procedure TCocoaSlider.mouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
begin
inherited mouseDown(event);
callback.MouseUpDownEvent(event, true);
end;
end;
procedure TCocoaSlider.mouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited mouseUp(event);
end;
procedure TCocoaSlider.rightMouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseDown(event);
end;
procedure TCocoaSlider.rightMouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseUp(event);
end;
procedure TCocoaSlider.rightMouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseDragged(event);
end;
procedure TCocoaSlider.otherMouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited otherMouseDown(event);
end;
procedure TCocoaSlider.otherMouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited otherMouseUp(event);
end;
procedure TCocoaSlider.otherMouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited otherMouseDragged(event);
end;
procedure TCocoaSlider.mouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited mouseDragged(event);
end;
procedure TCocoaSlider.mouseMoved(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited mouseMoved(event);
end;
procedure TCocoaSlider.scrollWheel(event: NSEvent);
begin
if not Assigned(callback) or not callback.scrollWheel(event) then
inherited scrollWheel(event);
end;
procedure TCocoaSlider.lclAddManTick(atick: integer);
var
mn : TManualTicks;
begin
mn := GetManTicks(self);
if mn.AddTick(atick) then
begin
if mn.draw then self.setNeedsDisplay_(true);
end;
end;
procedure TCocoaSlider.lclSetManTickDraw(adraw: Boolean);
var
mn : TManualTicks;
begin
mn := GetManTicks(self);
if mn.draw=adraw then Exit;
mn.draw:=adraw;
self.setNeedsDisplay_(true);
end;
procedure TCocoaSlider.lclExpectedKeys(var wantTabs, wantArrows, wantReturn,
wantAll: Boolean);
begin
wantTabs := false;
wantArrows := true;
wantReturn := false;
wantAll := false;
end;
type
NSViewControlSizeExt = objccategory external (NSView)
function controlSize: Integer; message 'controlSize';
procedure setControlSize(ASize: Integer); message 'setControlSize:';
function cell: id; message 'cell';
procedure setFont(afont: NSFont); message 'setFont:';
end;
procedure SetNSControlSize(ctrl: NSView; newHeight, miniHeight, smallHeight: Integer; AutoChangeFont: Boolean);
var
sz : NSControlSize;
begin
if (miniHeight>0) and (newHeight<=miniHeight) then
sz:=NSMiniControlSize
else if (smallHeight>0) and (newHeight<=smallHeight) then
sz:=NSSmallControlSize
else
sz:=NSRegularControlSize;
if ctrl.respondsToSelector(ObjCSelector('setControlSize:')) then
ctrl.setControlSize(sz)
else if ctrl.respondsToSelector(ObjCSelector('cell')) then
begin
if NSCell(ctrl.cell).controlSize<>sz then
NSCell(ctrl.cell).setControlSize(sz);
end;
if AutoChangeFont and (ctrl.respondsToSelector(ObjCSelector('setFont:'))) then
ctrl.setFont(NSFont.systemFontOfSize(NSFont.systemFontSizeForControlSize(sz)));
end;
end.