lazarus/lcl/interfaces/cocoa/cocoaprivate.pas
2020-08-10 01:40:07 +00:00

1664 lines
46 KiB
ObjectPascal
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

{ $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
// rtl+ftl
Types, Classes, SysUtils,
CGGeometry,
// Libs
MacOSAll, CocoaAll, CocoaUtils, CocoaGDIObjects,
cocoa_extra,
// LCL
LCLType;
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;
{ ICommonCallback }
ICommonCallback = interface
// mouse events
function MouseUpDownEvent(Event: NSEvent; AForceAsMouseUp: Boolean = False; AOverrideBlock: Boolean = False): Boolean;
procedure MouseClick;
function MouseMove(Event: NSEvent): Boolean;
// KeyEvXXX methods were introduced to allow a better control
// over when Cocoa keys processing is being called.
// (The initial KeyEvent() replicates Carbon implementation, and it's not
// suitable for Cocoa, due to the use of OOP and the extual "inherited Key..."needs to be called
// where for Carbon there's a special fucntion to call the "next event handler" present)
//
// The desired use is as following:
// Call KeyEvPrepare and pass NSEvent object
// after that call KeyEvBefore and pass a flag if AllowCocoaHandle
//
// The call would populate the flag. If it's "True" you should call "inherited" method (to let Cocoa handle the key).
// If the flag returned "False", you should not call inherited.
//
// No matter what the flag value was you should call KeyEvAfter.
procedure KeyEvBefore(Event: NSEvent; out AllowCocoaHandle: boolean);
procedure KeyEvAfter;
procedure KeyEvAfterDown(out AllowCocoaHandle: boolean);
procedure KeyEvHandled;
procedure SetTabSuppress(ASuppress: Boolean);
function scrollWheel(Event: NSEvent): Boolean;
function CanFocus: Boolean;
// size, pos events
procedure frameDidChange(sender: id);
procedure boundsDidChange(sender: id);
// misc events
procedure Draw(ctx: NSGraphicsContext; const bounds, dirty: NSRect);
procedure DrawBackground(ctx: NSGraphicsContext; const bounds, dirty: NSRect);
procedure DrawOverlay(ctx: NSGraphicsContext; const bounds, dirty: NSRect);
function ResetCursorRects: Boolean;
procedure BecomeFirstResponder;
procedure ResignFirstResponder;
procedure DidBecomeKeyNotification;
procedure DidResignKeyNotification;
procedure SendOnChange;
procedure SendOnTextChanged;
procedure scroll(isVert: Boolean; Pos: Integer; AScrollPart: NSScrollerPart = NSScrollerNoPart);
// non event methods
function DeliverMessage(Msg: Cardinal; WParam: WParam; LParam: LParam): LResult;
function GetPropStorage: TStringList;
function GetContext: TCocoaContext;
function GetTarget: TObject;
function GetHasCaret: Boolean;
function GetCallbackObject: TObject;
procedure SetHasCaret(AValue: Boolean);
function GetIsOpaque: Boolean;
procedure SetIsOpaque(AValue: Boolean);
function GetShouldBeEnabled: Boolean;
// the method is called, when handle is being destroyed.
// the callback object to stay alive a little longer than LCL object (Target)
// thus it needs to know that LCL object has been destroyed.
// After this called has been removed, any Cocoa events should not be
// forwarded to LCL target
procedure RemoveTarget;
procedure InputClientInsertText(const utf8: string);
// properties
property HasCaret: Boolean read GetHasCaret write SetHasCaret;
property IsOpaque: Boolean read GetIsOpaque write SetIsOpaque;
end;
{ 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 lclRelativePos(var Left, Top: Integer); message 'lclRelativePos::';
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 lclRelativePos(var Left, Top: Integer); message 'lclRelativePos::'; 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;
{ TCocoaCustomControl }
TCocoaCustomControl = objcclass(NSControl, NSTextInputClientProtocol)
private
fstr : NSString;
isdrawing : integer;
faileddraw : Boolean;
public
callback: ICommonCallback;
auxMouseByParent: Boolean;
procedure dealloc; override;
function acceptsFirstResponder: LCLObjCBoolean; override;
procedure drawRect(dirtyRect: NSRect); override;
function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override;
function lclIsMouseInAuxArea(Event: NSevent): Boolean; 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 mouseEntered(event: NSEvent); override;
procedure mouseExited(event: NSEvent); override;
procedure mouseMoved(event: NSEvent); override;
procedure scrollWheel(event: NSEvent); override;
// nsview
procedure setFrame(aframe: NSRect); override;
// other
procedure resetCursorRects; override;
// value
procedure setStringValue(avalue: NSString); override;
function stringValue: NSString; override;
procedure addSubView(aview: NSView); override;
// this is parts of
procedure insertText_replacementRange (aString: id; replacementRange: NSRange);
procedure doCommandBySelector (aSelector: SEL); override;
procedure setMarkedText_selectedRange_replacementRange (aString: id; selectedRange: NSRange; replacementRange: NSRange);
procedure unmarkText;
function selectedRange: NSRange;
function markedRange: NSRange;
function hasMarkedText: LCLObjCBoolean;
function attributedSubstringForProposedRange_actualRange (aRange: NSRange; actualRange: NSRangePointer): NSAttributedString;
function validAttributesForMarkedText: NSArray;
function firstRectForCharacterRange_actualRange (aRange: NSRange; actualRange: NSRangePointer): NSRect;
function characterIndexForPoint (aPoint: NSPoint): NSUInteger;
end;
TStatusItemData = record
Text : NSString;
Width : Integer;
Align : TAlignment;
end;
TStatusItemDataArray = array of TStatusItemData;
{ TCocoaStatusBar }
IStatusBarCallback = interface {(ICommonCallback) // not needed to inherit from ICommonCallback}
function GetBarsCount: Integer;
//todo: consider the use Cocoa native types, instead of FPC TAlignment
function GetBarItem(idx: Integer; var txt: String;
var width: Integer; var align: TAlignment): Boolean;
procedure DrawPanel(idx: Integer; const r: TRect);
end;
TCocoaStatusBar = objcclass(TCocoaCustomControl)
public
//StatusBar : TStatusBar;
barcallback : IStatusBarCallback;
panelCell : NSCell;
procedure drawRect(dirtyRect: NSRect); override;
procedure dealloc; override;
end;
{ TCocoaGroupBox }
TCocoaGroupBox = objcclass(NSBox)
public
callback: ICommonCallback;
function acceptsFirstResponder: LCLObjCBoolean; override;
function lclGetCallback: ICommonCallback; override;
procedure lclClearCallback; override;
procedure resetCursorRects; 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;
procedure resetCursorRects; 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 resetCursorRects; 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);
// these constants are missing from CocoaAll for some reason
const
NSTextAlignmentLeft = 0;
NSTextAlignmentRight = 1; // it's 2 for iOS and family
NSTextAlignmentCenter = 2; // it's 1 for iOS and family
NSTextAlignmentJustified = 3;
NSTextAlignmentNatural = 4;
var
// todo: this should be a threadvar
TrackedControl : NSObject = nil;
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);
begin
if not Assigned(AView) then Exit;
AView.setAutoresizingMask(NSViewMinYMargin or NSViewMaxXMargin);
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;
procedure TCocoaGroupBox.resetCursorRects;
begin
if not Assigned(callback) or not callback.resetCursorRects then
inherited resetCursorRects;
end;
{ TCocoaCustomControl }
procedure TCocoaCustomControl.setStringValue(avalue: NSString);
begin
if Assigned(fstr) then fstr.release;
if ASsigned(avalue) then
fstr:=avalue.copyWithZone(nil)
else
fstr:=nil;
inherited setStringValue(avalue);
end;
function TCocoaCustomControl.stringValue: NSString;
begin
Result:=fstr;
end;
procedure TCocoaCustomControl.addSubView(aview: NSView);
begin
inherited addSubView(aview);
if Assigned(aview) then
begin
// forcing LCL compatible "auto-move" mode. Sticking to left/top corner
if not autoresizesSubviews then
{$ifdef BOOLFIX}
setAutoresizesSubviews_(Ord(true));
{$else}
setAutoresizesSubviews(true);
{$endif}
aview.setAutoresizingMask(NSViewMaxXMargin or NSViewMinYMargin);
end;
end;
procedure TCocoaCustomControl.insertText_replacementRange(aString: id;
replacementRange: NSRange);
begin
lclGetCallback.InputClientInsertText(NSStringToString(NSString(astring)));
end;
procedure TCocoaCustomControl.doCommandBySelector(aSelector: SEL);
begin
inherited doCommandBySelector(ASelector);
end;
procedure TCocoaCustomControl.setMarkedText_selectedRange_replacementRange(
aString: id; selectedRange: NSRange; replacementRange: NSRange);
begin
end;
procedure TCocoaCustomControl.unmarkText;
begin
end;
function TCocoaCustomControl.selectedRange: NSRange;
begin
Result := NSMakeRange(0,0);
end;
function TCocoaCustomControl.markedRange: NSRange;
begin
Result := NSMakeRange(0,0);
end;
function TCocoaCustomControl.hasMarkedText: LCLObjCBoolean;
begin
Result := false;
end;
function TCocoaCustomControl.attributedSubstringForProposedRange_actualRange(
aRange: NSRange; actualRange: NSRangePointer): NSAttributedString;
begin
Result := nil;
end;
function TCocoaCustomControl.validAttributesForMarkedText: NSArray;
begin
Result := nil;
end;
function TCocoaCustomControl.firstRectForCharacterRange_actualRange(
aRange: NSRange; actualRange: NSRangePointer): NSRect;
begin
Result := NSMakeRect(0,0,0,0);
end;
function TCocoaCustomControl.characterIndexForPoint(aPoint: NSPoint
): NSUInteger;
begin
Result := 0;
end;
procedure TCocoaCustomControl.dealloc;
begin
if Assigned(fstr) then fstr.release;
inherited dealloc;
end;
function TCocoaCustomControl.acceptsFirstResponder: LCLObjCBoolean;
begin
Result := True;
end;
function TCocoaCustomControl.acceptsFirstMouse(event: NSEvent): LCLObjCBoolean;
begin
// By default, a mouse-down event in a window that isnt the key window
// simply brings the window forward and makes it key; the event isnt sent
// to the NSView object over which the mouse click occurs. The NSView can
// claim an initial mouse-down event, however, by overriding acceptsFirstMouse: to return YES.
// see bug #33034
Result:=true;
end;
procedure TCocoaCustomControl.drawRect(dirtyRect: NSRect);
begin
if isdrawing=0 then faileddraw:=false;
inc(isdrawing);
inherited drawRect(dirtyRect);
// Implement Color property
if Assigned(callback) then
callback.DrawBackground(NSGraphicsContext.currentContext, bounds, dirtyRect);
if CheckMainThread and Assigned(callback) then
callback.Draw(NSGraphicsContext.currentContext, bounds, dirtyRect);
dec(isdrawing);
if (isdrawing=0) and (faileddraw) then
begin
// Similar to Carbon. Cocoa doesn't welcome changing a framerects during paint event
// If such thing happens, the results are pretty much inpredicatable. #32970
// TreeView tries to updatedScrollBars during paint event. That sometimes is causing
// the frame to be changed (i.e. scroll bar showed or hidden, resized the client rect)
// as a result, the final image is shown up-side-down.
//
// Below is an attempt to prevent graphical artifacts and to redraw
// the control again.
inherited drawRect(dirtyRect);
if Assigned(callback) then
callback.DrawBackground(NSGraphicsContext.currentContext, bounds, dirtyRect);
if CheckMainThread and Assigned(callback) then
callback.Draw(NSGraphicsContext.currentContext, bounds, dirtyRect);
end;
end;
function TCocoaCustomControl.lclGetCallback: ICommonCallback;
begin
Result := callback;
end;
procedure TCocoaCustomControl.lclClearCallback;
begin
callback := nil;
end;
function TCocoaCustomControl.lclIsMouseInAuxArea(Event: NSevent): Boolean;
begin
if auxMouseByParent and Assigned(superview) then
Result := superview.lclIsMouseInAuxArea(Event)
else
Result := false;
end;
procedure TCocoaCustomControl.mouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited mouseDown(event);
end;
procedure TCocoaCustomControl.mouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
// calling inherited causes the drag event to be passed to the
// parent controls
//inherited mouseDragged(event);
;
end;
procedure TCocoaCustomControl.mouseEntered(event: NSEvent);
begin
inherited mouseEntered(event);
end;
procedure TCocoaCustomControl.mouseExited(event: NSEvent);
begin
inherited mouseExited(event);
end;
procedure TCocoaCustomControl.mouseMoved(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited mouseMoved(event);
end;
procedure TCocoaCustomControl.scrollWheel(event: NSEvent);
begin
if not Assigned(callback) or not callback.scrollWheel(event) then
inherited scrollWheel(event);
end;
procedure TCocoaCustomControl.setFrame(aframe: NSRect);
begin
if NSEqualRects(aframe, frame) then Exit;
if isdrawing>0 then
faileddraw := true;
inherited setFrame(aframe);
// it actually should come from a notifcation
if Assigned(callback) then callback.frameDidChange(self);
end;
procedure TCocoaCustomControl.mouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited mouseUp(event);
end;
procedure TCocoaCustomControl.rightMouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseDown(event);
end;
procedure TCocoaCustomControl.rightMouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited rightMouseUp(event);
end;
procedure TCocoaCustomControl.rightMouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited rightMouseDragged(event);
end;
procedure TCocoaCustomControl.otherMouseDown(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited otherMouseDown(event);
end;
procedure TCocoaCustomControl.otherMouseUp(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseUpDownEvent(event) then
inherited otherMouseUp(event);
end;
procedure TCocoaCustomControl.otherMouseDragged(event: NSEvent);
begin
if not Assigned(callback) or not callback.MouseMove(event) then
inherited otherMouseDragged(event);
end;
procedure TCocoaCustomControl.resetCursorRects;
begin
if not Assigned(callback) or not callback.resetCursorRects then
inherited resetCursorRects;
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.lclRelativePos(var Left,Top: Integer);
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) 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.lclRelativePos(var Left, Top: Integer);
var
sv : NSView;
fr : NSRect;
begin
Left := Round(frame.origin.x);
sv := superview;
if Assigned(sv) and (not sv.isFlipped) then
begin
fr := frame;
Top := Round(sv.frame.size.height - fr.origin.y - fr.size.height);
end
else
Top := Round(frame.origin.y);
end;
procedure LCLViewExtension.lclLocalToScreen(var X, Y:Integer);
var
P: NSPoint;
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
// 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);
{$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;
{ TCocoaStatusBar }
procedure TCocoaStatusBar.drawRect(dirtyRect: NSRect);
var
R : TRect;
i : Integer;
cs : NSString;
nr : NSRect;
dr : NSRect;
al : TAlignment;
x : Integer;
txt : string;
cnt : Integer;
w : Integer;
const
CocoaAlign: array [TAlignment] of Integer = (NSNaturalTextAlignment, NSRightTextAlignment, NSCenterTextAlignment);
begin
if not Assigned(barcallback) then Exit;
if not Assigned(panelCell) then Exit;
panelCell.setControlView(Self);
r := lclClientFrame();
nr.origin.y := 0;
nr.size.height := self.lclFrame.Height;
x:=0;
cnt := barcallback.GetBarsCount;
for i:=0 to cnt - 1 do begin
txt := '';
w := 0;
al := taLeftJustify;
if not barcallback.GetBarItem(i, txt, w, al) then Continue;
if i = cnt - 1 then w := r.Right - x;
nr.size.width := w;
nr.origin.x := x;
// dr - draw rect. should be 1 pixel wider
// and 1 pixel taller, than the actual rect.
// to produce a better visual effect
dr := nr;
dr.size.width := dr.size.width + 1;
dr.size.height := dr.size.height + 1;
dr.origin.y := dr.origin.y-1;
cs := NSStringUtf8(txt);
panelCell.setTitle(cs);
panelCell.setAlignment(CocoaAlign[al]);
panelCell.drawWithFrame_inView(dr, Self);
cs.release;
barcallback.DrawPanel(i, NSRectToRect(nr));
inc(x, w);
if x > r.Right then break; // no place left
end;
end;
procedure TCocoaStatusBar.dealloc;
begin
if Assigned(panelCell) then panelCell.release;
inherited;
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;
procedure TCocoaProgressIndicator.resetCursorRects;
begin
if not callback.resetCursorRects then
inherited resetCursorRects;
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.resetCursorRects;
begin
if not callback.resetCursorRects then
inherited resetCursorRects;
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.