lazarus/lcl/interfaces/carbon/carbonprivatecontrol.inc

1799 lines
65 KiB
PHP

{%MainUnit carbonprivate.pp}
{
*****************************************************************************
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.
*****************************************************************************
}
// ==================================================================
// H A N D L E R S
// ==================================================================
{------------------------------------------------------------------------------
Name: CarbonControl_Hit
Handles click and LM_MOUSEUP events
------------------------------------------------------------------------------}
function CarbonControl_Hit(ANextHandler: EventHandlerCallRef;
AEvent: EventRef;
AWidget: TCarbonWidget): OSStatus; {$IFDEF darwin}mwpascal;{$ENDIF}
var
ControlPart: ControlPartCode;
LCLTarget: TWinControl;
begin
{$IFDEF VerboseControlEvent}
DebugLn('CarbonControl_Hit: ', DbgSName(AWidget.LCLObject));
{$ENDIF}
Result := CallNextEventHandler(ANextHandler, AEvent);
if OSError(
GetEventParameter(AEvent, kEventParamControlPart, typeControlPartCode, nil,
SizeOf(ControlPartCode), nil, @ControlPart), 'CarbonControl_Hit', SGetEvent,
SControlPart) then Exit;
// save LCL target now as the widget might be removed within hit
LCLTarget := AWidget.LCLObject;
(AWidget as TCarbonControl).Hit(ControlPart);
// send postponed mouse up event
DeliverMessage(LCLTarget, SavedMouseUpMsg);
NotifyApplicationUserInput(LCLTarget, SavedMouseUpMsg.Msg);
CarbonWidgetSet.SetCaptureWidget(0); // capture is released
end;
{------------------------------------------------------------------------------
Name: CarbonControl_ValueChanged
Handles value change
------------------------------------------------------------------------------}
function CarbonControl_ValueChanged(ANextHandler: EventHandlerCallRef;
AEvent: EventRef;
AWidget: TCarbonWidget): OSStatus; {$IFDEF darwin}mwpascal;{$ENDIF}
begin
{$IFDEF VerboseControlEvent}
DebugLn('CarbonControl_ValueChanged ', DbgSName(AWidget.LCLObject));
{$ENDIF}
Result := CallNextEventHandler(ANextHandler, AEvent);
(AWidget as TCarbonControl).ValueChanged;
end;
{------------------------------------------------------------------------------
Name: CarbonControl_IndicatorMoved
Handles indicator move
------------------------------------------------------------------------------}
function CarbonControl_IndicatorMoved(ANextHandler: EventHandlerCallRef;
AEvent: EventRef;
AWidget: TCarbonWidget): OSStatus; {$IFDEF darwin}mwpascal;{$ENDIF}
begin
{$IFDEF VerboseControlEvent}
DebugLn('CarbonControl_IndicatorMoved ', DbgSName(AWidget.LCLObject));
{$ENDIF}
Result := CallNextEventHandler(ANextHandler, AEvent);
(AWidget as TCarbonControl).IndicatorMoved;
end;
{------------------------------------------------------------------------------
Name: CarbonControl_CommandProcess
Handles copy/paste... commands
------------------------------------------------------------------------------}
function CarbonControl_CommandProcess(ANextHandler: EventHandlerCallRef;
AEvent: EventRef;
AWidget: TCarbonWidget): OSStatus; {$IFDEF darwin}mwpascal;{$ENDIF}
var
Command: HICommandExtended;
const SName = 'CarbonControl_CommandProcess';
begin
{$IFDEF VerboseControlEvent}
DebugLn('CarbonControl_CommandProcess ' + DbgSName(AWidget.LCLObject));
{$ENDIF}
if not OSError(
GetEventParameter(AEvent, kEventParamDirectObject,
typeHICommand, nil, SizeOf(HICommand), nil, @Command),
SName, 'GetEventParameter') then
begin
{$IFDEF VerboseControlEvent}
DebugLn('CarbonControl_CommandProcess ID: ' + DbgS(Command.commandID) + ' Target: ' +
DbgSName(AWidget.LCLObject));
{$ENDIF}
case Command.commandID of
kHICommandUndo:;
kHICommandRedo:;
kHICommandCut: SendSimpleMessage(AWidget.LCLObject, LM_CUT);
kHICommandCopy: SendSimpleMessage(AWidget.LCLObject, LM_COPY);
kHICommandPaste: SendSimpleMessage(AWidget.LCLObject, LM_PASTE);
kHICommandClear: SendSimpleMessage(AWidget.LCLObject, LM_CLEAR);
kHICommandSelectAll:;
end;
end;
Result := CallNextEventHandler(ANextHandler, AEvent);
end;
{------------------------------------------------------------------------------
Name: CarbonControl_Accessibility
Accessibility
The events for this handler are:
kEventAccessibleGetChildAtPoint
kEventAccessibleGetFocusedChild
kEventAccessibleGetAllAttributeNames
kEventAccessibleGetAllParameterizedAttributeNames
kEventAccessibleGetNamedAttribute
kEventAccessibleSetNamedAttribute
kEventAccessibleIsNamedAttributeSettable
kEventAccessibleGetAllActionNames
kEventAccessiblePerformNamedAction
kEventAccessibleGetNamedActionDescription
For kEventAccessibleGetNamedAttribute the named attributes are:
AXRole
AXRoleDescription
AXDescription
AXHelp
AXParent
AXChildren
AXWindow
AXTopLevelUIElement
AXEnabled
AXSize
AXPosition
And besides the ones listed here there are tons more. See which attributes each role
should implement here:
http://developer.apple.com/library/mac/#documentation/UserExperience/Reference/Accessibility_RoleAttribute_Ref/Role.html#//apple_ref/doc/uid/TP40007870-Roles-AXStaticText
Table explaining the events:
http://developer.apple.com/library/mac/#documentation/Accessibility/Reference/AccessibilityCarbonRef/Reference/reference.html
Documentation explaining in details which attributes are required for each Role:
http://developer.apple.com/library/mac/#documentation/UserExperience/Reference/Accessibility_RoleAttribute_Ref/Attributes.html
List of routines to handle a AXUIElementRef:
http://developer.apple.com/library/mac/#documentation/Accessibility/Reference/AccessibilityLowlevel/AXUIElement_h/CompositePage.html
------------------------------------------------------------------------------}
function CarbonControl_Accessibility(ANextHandler: EventHandlerCallRef;
AEvent: EventRef;
AWidget: TCarbonWidget): OSStatus; {$IFDEF darwin}mwpascal;{$ENDIF}
var
// Inputs
lAXRole, lInputStr: CFStringRef;
lInputAXObject: AXUIElementRef;
lInputID64: UInt64;
lInputAccessibleObject: TLazAccessibleObject;
lInputPasStr: string;
lInputMutableArray: CFMutableArrayRef;
lInputHIPoint: HIPoint;
lInputPoint: TPoint;
// Outputs
lOutputStr: CFStringRef;
lOutputBool: Boolean;
lOutputInt: SInt64;
lOutputNum: CFNumberRef;
lOutputValue: AXValueRef;
lOutputPoint: CGPoint;
lOutputSize: CGSize;
//
lLazControl: TControl;
lLazAXRole: TLazAccessibilityRole;
lCurAccessibleObject: TLazAccessibleObject;
EventKind: UInt32;
// array
lArray: CFMutableArrayRef;
lElement, lElement2: AXUIElementRef;
lCount: Integer;
i: Integer;
lAccessibleObj: TLazAccessibleObject;
lHandle: PtrInt;
lSelection: TLazAccessibleObject;
lChildControl: TControl;
const SName = 'CarbonControl_Accessibility';
begin
{$IF defined(VerboseControlEvent) or defined(VerboseAccessibilityEvent)}
DebugLn('CarbonControl_Accessibility LCLObject=', DbgSName(AWidget.LCLObject));
{$ENDIF}
Result := CallNextEventHandler(ANextHandler, AEvent); // Must be called at the event handling start
lLazControl := TControl((AWidget as TCarbonControl).LCLObject);
if lLazControl = nil then Exit;
GetEventParameter(AEvent, kEventParamAccessibleObject,
typeCFTypeRef, nil, SizeOf(AXUIElementRef), nil, @lInputAXObject);
// Check if this is an event to a child accessible object
AXUIElementGetIdentifier(lInputAXObject, lInputID64{%H-});
if (lLazControl is TCustomControl) and (lInputID64 <> 0) then
lInputAccessibleObject := TLazAccessibleObject(PtrInt(lInputID64))
else
begin
lInputAccessibleObject := lLazControl.GetAccessibleObject();
{// Store the AX Handle
if not lInputAccessibleObject.HandleAllocated then
lInputAccessibleObject.Handle := PtrInt(lInputAXObject);}
end;
lLazAXRole := lInputAccessibleObject.AccessibleRole;
EventKind := GetEventKind(AEvent);
case EventKind of
kEventAccessibleGetChildAtPoint:
begin
if not (lLazControl is TCustomControl) then Exit;
// For TCustomForm get out immediately, otherwise we disturbe the navigation of VoiceOver
if lLazControl is TCustomForm then Exit;
// The location in global coordinates.
GetEventParameter(AEvent, kEventParamMouseLocation,
typeHIPoint, nil, SizeOf(HIPoint), nil, @lInputHIPoint);
// <-- kEventParamAccessibleChild (out, typeCFTypeRef)
// On exit, contains the child of the accessible object at the
// specified point, in the form of an AXUIElementRef.
lInputPoint := Types.Point(Round(lInputHIPoint.x), Round(lInputHIPoint.y));
lInputPoint := TWinControl(lLazControl).ScreenToClient(lInputPoint);
if lInputAccessibleObject = nil then Exit;
lAccessibleObj := lInputAccessibleObject.GetChildAccessibleObjectAtPos(lInputPoint);
if (lAccessibleObj = nil) or (lAccessibleObj.Handle = 0) then Exit;
lHandle := lAccessibleObj.Handle;
SetEventParameter(AEvent, kEventParamAccessibleChild, typeCFTypeRef,
SizeOf(AXUIElementRef), @lHandle);
end;
kEventAccessibleGetAllAttributeNames:
begin
{$IF defined(VerboseControlEvent) or defined(VerboseAccessibilityEvent)}
DebugLn('CarbonControl_Accessibility kEventAccessibleGetAllAttributeNames');
{$ENDIF}
if not (lLazControl is TCustomControl) then Exit;
// For TCustomForm get out immediately, otherwise we disturbe the navigation of VoiceOver
if lLazControl is TCustomForm then Exit;
if OSError(
GetEventParameter(AEvent, kEventParamAccessibleAttributeNames,
typeCFMutableArrayRef, nil, SizeOf(CFMutableArrayRef), nil, @lInputMutableArray),
SName, 'GetEventParameter') then Exit;
{$IF defined(VerboseControlEvent) or defined(VerboseAccessibilityEvent)}
DebugLn('CarbonControl_Accessibility kEventAccessibleGetAllAttributeNames After GetEventParameter');
{$ENDIF}
// AXFocused interrests all focusable classes
lOutputStr := CFSTR('AXFocused');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
// AXChildren
lCount := lInputAccessibleObject.GetChildAccessibleObjectsCount;
if lCount > 0 then
begin
lOutputStr := CFSTR('AXChildren');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end;
// Now elements for each role
// AXStaticText
if lLazAXRole in larAXStaticTextRoles then
begin
lOutputStr := CFSTR('AXNumberOfCharacters');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end;
// Now elements for each role
// AXList
if lLazAXRole in larAXListRoles then
begin
lOutputStr := CFSTR('AXOrientation');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSelectedChildren');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXVisibleChildren');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end;
// Basic elements for non-windowed accessible objects
// Note that AXRole, AXRoleDescription, AXParent seam to be added by default
if lInputID64 <> 0 then
begin
lOutputStr := CFSTR('AXValue');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXPosition');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXTopLevelUIElement');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSize');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXWindow');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXEnabled');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end;
end; // kEventAccessibleGetAllAttributeNames
kEventAccessibleGetAllParameterizedAttributeNames:
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if OSError(
GetEventParameter(AEvent, kEventParamAccessibleAttributeNames,
typeCFMutableArrayRef, nil, SizeOf(CFMutableArrayRef), nil, @lInputMutableArray),
SName, 'GetEventParameter') then Exit;
// Now elements for each role
// AXStaticText
if lLazAXRole in larAXStaticTextRoles then
begin
if lLazAXRole = larResizeGrip then
begin
lOutputStr := CFSTR('AXValue');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end;
lOutputStr := CFSTR('AXStringForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXAttributedStringForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXBoundsForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRangeForIndex');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRangeForLine');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRangeForPosition');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRTFForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXStyleRangeForIndex');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end
else if lLazAXRole = larTextEditorMultiline then
begin
lOutputStr := CFSTR('AXAttributedStringForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXBoundsForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXInsertionPointLineNumber');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXLineForIndex');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXNumberOfCharacters');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRangeForIndex');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRangeForLine');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRangeForPosition');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXRTFForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSelectedText');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSelectedTextRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSelectedTextRanges');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSharedCharacterRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXSharedTextUIElements');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXStringForRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXStyleRangeForIndex');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
lOutputStr := CFSTR('AXVisibleCharacterRange');
CFArrayAppendValue(lInputMutableArray, lOutputStr);
end;
end;// kEventAccessibleGetAllParameterizedAttributeNames
kEventAccessibleGetNamedAttribute:
begin
if OSError(
GetEventParameter(AEvent, kEventParamAccessibleAttributeName,
typeCFStringRef, nil, SizeOf(CFStringRef), nil, @lInputStr),
SName, 'GetEventParameter') then Exit;
lInputPasStr := CFStringToStr(lInputStr);
{$IF defined(VerboseControlEvent) or defined(VerboseAccessibilityEvent)}
DebugLn('CarbonControl_Accessibility kEventAccessibleGetNamedAttribute kEventParamAccessibleAttributeName=' + lInputPasStr);
{$ENDIF}
// AXRole overrides TCustomControl and TCustomWindow values
if lInputPasStr = 'AXRole' then
begin
if not (lLazControl is TCustomControl) then Exit;
case lLazAXRole of
larAnimation: lAXRole := CFSTR('AXImage');
larButton: lAXRole := CFSTR('AXButton');
larCell: lAXRole := CFSTR('AXCell');
larChart: lAXRole := CFSTR('AXImage');
larCheckBox: lAXRole := CFSTR('AXCheckBox');
larClock: lAXRole := CFSTR('AXStaticText');
larColorPicker: lAXRole := CFSTR('AXColorWell');
larComboBox: lAXRole := CFSTR('AXComboBox');
larDateField: lAXRole := CFSTR('AXDateField');
larGrid: lAXRole := CFSTR('AXGrid');
larGroup: lAXRole := CFSTR('AXGroup');
larImage: lAXRole := CFSTR('AXImage');
larLabel: lAXRole := CFSTR('AXStaticText');
larListBox: lAXRole := CFSTR('AXList');
larListItem: lAXRole := CFSTR('AXStaticText');
larMenuBar: lAXRole := CFSTR('AXMenuBar');
larMenuItem: lAXRole := CFSTR('AXMenuItem');
larProgressIndicator: lAXRole := CFSTR('AXProgressIndicator');
larResizeGrip: lAXRole := CFSTR('AXStaticText'); // VoiceOver cannot handle AXHandle in Mac 10.6, so we fallback to AXStaticText
larScrollBar: lAXRole := CFSTR('AXScrollBar');
larSpinner: lAXRole := CFSTR('AXIncrementor');
larTabControl: lAXRole := CFSTR('AXTabGroup'); // Note that here we represent the tabsheet actually
larTextEditorMultiline: lAXRole := CFSTR('AXTextArea');
larTextEditorSingleline: lAXRole := CFSTR('AXTextField');
larTrackBar: lAXRole := CFSTR('AXSlider');
larTreeView: lAXRole := CFSTR('AXList');
larTreeItem: lAXRole := CFSTR('AXStaticText');
larWindow: lAXRole := CFSTR('AXUnknown'); // We report for the client are of the window, not the window itself
else
lAXRole := CFSTR('AXUnknown');
end;
//if OSError(
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFStringRef,
SizeOf(CFStringRef), @lAXRole);//, SName, SSetEvent, SControlAction) then Exit;
Result := noErr;
Exit;
end
// Specially only AXRoleDescription is allowed to override non-TCustomControl values
else if lInputPasStr = 'AXRoleDescription' then
begin
if lInputAccessibleObject.AccessibleDescription = '' then Exit;
CreateCFString(lInputAccessibleObject.AccessibleDescription, lOutputStr);
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFStringRef,
SizeOf(CFStringRef), @lOutputStr);
FreeCFString(lOutputStr);
Result := noErr;
Exit;
end
else if lInputPasStr = 'AXValue' then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if (lInputID64 = 0) and (not (lLazAXRole = larResizeGrip)) then Exit;
// Hack around the lack of support for AXHandle in VoiceOver
if lLazAXRole = larResizeGrip then
CreateCFString(lInputAccessibleObject.AccessibleDescription, lOutputStr)
else
CreateCFString(lInputAccessibleObject.AccessibleValue, lOutputStr);
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFStringRef,
SizeOf(CFStringRef), @lOutputStr);
FreeCFString(lOutputStr);
Result := noErr;
end
else if (lInputPasStr = 'AXNumberOfCharacters') then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
lOutputInt := UTF8Length(lInputAccessibleObject.AccessibleValue);
lOutputNum := CFNumberCreate(nil, kCFNumberSInt64Type, @lOutputInt);
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFNumberRef,
SizeOf(lOutputNum), @lOutputNum);
CFRelease(lOutputNum);
Result := noErr;
end
//
// Parameterized attributes
//
else if (lInputPasStr = 'AXStringForRange') then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if lInputAccessibleObject.AccessibleValue = '' then Exit;
CreateCFString(lInputAccessibleObject.AccessibleValue, lOutputStr);
SetEventParameter(AEvent, kEventParamAccessibleAttributeParameter, typeCFStringRef,
SizeOf(CFStringRef), @lOutputStr);
FreeCFString(lOutputStr);
Result := noErr;
end
else if (lInputPasStr = 'AXAttributedStringForRange') or
(lInputPasStr = 'AXBoundsForRange') or
(lInputPasStr = 'AXRangeForIndex') or
(lInputPasStr = 'AXRangeForLine') or
(lInputPasStr = 'AXRangeForPosition') or
(lInputPasStr = 'AXRTFForRange') or
(lInputPasStr = 'AXStyleRangeForIndex') then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
SetEventParameter(AEvent, kEventParamAccessibleAttributeParameter, typeCFTypeRef,
0, nil);
Result := noErr;
end
//if (CFStringCompare(lInputStr, kAXFocusedAttribute, 0) = kCFCompareEqualTo) then
else if lInputPasStr = 'AXFocused' then
begin
// if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if not (lLazControl is TWinControl) then lOutputBool := False
else if TWinControl(lLazControl).Focused then lOutputBool := True
else lOutputBool := False;
{$IF defined(VerboseControlEvent) or defined(VerboseAccessibilityEvent)}
DebugLn('CarbonControl_Accessibility AXFocused');
{$ENDIF}
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeBoolean,
SizeOf(Boolean), @lOutputBool);
Result := noErr;
end
//else if (CFStringCompare(lInputStr, CFSTR('AXChildren'), 0) = kCFCompareEqualTo) then // kAXChildrenAttribute
else if (lInputPasStr = 'AXChildren') or (lInputPasStr = 'AXVisibleChildren')
or (lInputPasStr = 'AXSelectedChildren') then
begin
if not (lLazControl is TCustomControl) then Exit;
if (lLazControl is TCustomForm) then Exit;
// Create and return an array of AXUIElements describing the children of this view.
if (lInputPasStr = 'AXSelectedChildren') then
begin
lSelection := lInputAccessibleObject.GetSelectedChildAccessibleObject();
if lSelection = nil then
lArray := CFArrayCreateMutable(kCFAllocatorDefault, 0, @kCFTypeArrayCallBacks)
else
begin
lArray := CFArrayCreateMutable(kCFAllocatorDefault, 1, @kCFTypeArrayCallBacks);
lElement := {%H-}AXUIElementRef(lSelection.Handle);
if lElement <> nil then
CFArrayAppendValue(lArray, lElement);
end;
end
else
begin
// Using GetEventParemeter seams to bring crashes, no idea why
//GetEventParameter(AEvent, kEventParamAccessibleAttributeValue,
// typeCFArrayRef, nil, SizeOf(CFArrayRef), nil, @lInputArray);
// lArray := CFArrayCreateMutableCopy(kCFAllocatorDefault, 0, lInputArray);
// DebugLn('AXChildren 1 lArray='+IntToStr(PtrInt(lArray)));
lArray := CFArrayCreateMutable(kCFAllocatorDefault, lCount, @kCFTypeArrayCallBacks);
// Add non-control children
for lCurAccessibleObject in lInputAccessibleObject do
begin
lElement := {%H-}AXUIElementRef(lCurAccessibleObject.Handle);
if lElement <> nil then
CFArrayAppendValue(lArray, lElement);
end;
// Add control children
//DebugLn('1');
for i := 0 to TWinControl(lLazControl).ControlCount-1 do
begin
//DebugLn('2');
lChildControl := TWinControl(lLazControl).Controls[i];
lElement := {%H-}AXUIElementRef(lChildControl.GetAccessibleObject().Handle);
if lElement <> nil then
CFArrayAppendValue(lArray, lElement);
end;
end;
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFTypeRef,
SizeOf(lArray), @lArray);
CFRelease(lArray);
Result := noErr;
end
else if lInputPasStr = 'AXOrientation' then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
{$ifdef CarbonUseCocoaAll}
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFStringRef,
SizeOf(CFStringRef), @NSAccessibilityVerticalOrientationValue);
Result := noErr;
{$endif}
end
// Basic elements for non-windowed accessible objects
// therefore only do something if lInputID <> 0
else if lInputPasStr = 'AXParent' then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if lInputID64 = 0 then Exit;
if lInputAccessibleObject.Parent = nil then Exit;
lElement := {%H-}AXUIElementRef(lInputAccessibleObject.Parent.Handle);
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFTypeRef,
SizeOf(AXUIElementRef), @lElement);
Result := noErr;
end
else if lInputPasStr = 'AXPosition' then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if lInputID64 = 0 then Exit;
lOutputPoint.x := lInputAccessibleObject.Position.X;
lOutputPoint.y := lInputAccessibleObject.Position.Y;
lOutputValue := AXValueCreate(kAXValueCGPointType, @lOutputPoint);
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFTypeRef,
SizeOf(lOutputValue), @lOutputValue);
Result := noErr;
end
else if (lInputPasStr = 'AXWindow') or (lInputPasStr = 'AXTopLevelUIElement') then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if lInputID64 = 0 then Exit;
// Just copy from our parent the AXWindow
if lInputAccessibleObject.Parent = nil then Exit;
lElement := {%H-}AXUIElementRef(lInputAccessibleObject.Parent.Handle);
AXUIElementCopyAttributeValue(lElement, CFSTR('AXWindow'), lElement2{%H-});
if lElement2 = nil then Exit;
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFTypeRef,
SizeOf(AXUIElementRef), @lElement2);
Result := noErr;
end
else if lInputPasStr = 'AXSize' then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if lInputID64 = 0 then Exit;
lOutputSize.width := lInputAccessibleObject.Size.cx;
lOutputSize.height := lInputAccessibleObject.Size.cy;
lOutputValue := AXValueCreate(kAXValueCGSizeType, @lOutputSize);
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeCFTypeRef,
SizeOf(lOutputValue), @lOutputValue);
Result := noErr;
end
else if lInputPasStr = 'AXEnabled' then
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if lInputID64 = 0 then Exit;
lOutputBool := True;
SetEventParameter(AEvent, kEventParamAccessibleAttributeValue, typeBoolean,
SizeOf(Boolean), @lOutputBool);
Result := noErr;
end;
end; // kEventAccessibleGetNamedAttribute
kEventAccessibleIsNamedAttributeSettable:
begin
if not (lLazControl is TCustomControl) then Exit;
if lLazControl is TCustomForm then Exit;
if OSError(
GetEventParameter(AEvent, kEventParamAccessibleAttributeName,
typeCFStringRef, nil, SizeOf(CFStringRef), nil, @lInputStr),
SName, 'GetEventParameter') then Exit;
lInputPasStr := CFStringToStr(lInputStr);
if (lInputPasStr = 'AXFocused') then
begin
lOutputBool := TCustomControl(lLazControl).TabStop;
SetEventParameter(AEvent, kEventParamAccessibleAttributeSettable, typeBoolean,
SizeOf(Boolean), @lOutputBool);
Result := noErr;
Exit;
end;
// For all other attributes simply default for not settable
if (lInputPasStr = 'AXStringForRange') or
(lInputPasStr = 'AXNumberOfCharacters') or (lInputPasStr = 'AXChildren') or
(lInputPasStr = 'AXSelectedChildren') or (lInputPasStr = 'AXVisibleChildren') or
(lInputPasStr = 'AXPosition') or (lInputPasStr = 'AXSize') or
(lInputPasStr = 'AXParent') or (lInputPasStr = 'AXTopLevelUIElement') or
(lInputPasStr = 'AXWindow') or (lInputPasStr = 'AXOrientation') or
(lInputPasStr = 'AXValue') or (lInputPasStr = 'AXEnabled') then
begin
lOutputBool := False;
SetEventParameter(AEvent, kEventParamAccessibleAttributeSettable, typeBoolean,
SizeOf(Boolean), @lOutputBool);
Result := noErr;
end;
end; // kEventAccessibleIsNamedAttributeSettable
end; // case EventKind of
end;
{ TCarbonToolBar }
procedure TCarbonToolBar.CreateWidget(const AParams: TCreateParams);
begin
CarbonWidgetFlag := cwdTToolBar;
inherited CreateWidget(AParams);
end;
{ TCarbonControl }
{------------------------------------------------------------------------------
Method: TCarbonControl.GetValidEvents
Returns: Set of events with installed handlers
------------------------------------------------------------------------------}
class function TCarbonControl.GetValidEvents: TCarbonControlEvents;
begin
Result := [];
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.Hit
Params: AControlPart - Hitted control part
Hit event handler
------------------------------------------------------------------------------}
procedure TCarbonControl.Hit(AControlPart: ControlPartCode);
begin
DebugLn('TCarbonControl.Hit is invalid ', ClassName, ' ',
LCLObject.Name, ': ', LCLObject.ClassName);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.Draw
Draw event handler
------------------------------------------------------------------------------}
procedure TCarbonControl.Draw;
begin
DebugLn('TCarbonControl.Draw is invalid ', ClassName, ' ',
LCLObject.Name, ': ', LCLObject.ClassName);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.ValueChanged
Value changed event handler
------------------------------------------------------------------------------}
procedure TCarbonControl.ValueChanged;
begin
DebugLn('TCarbonControl.ValueChanged is invalid ', ClassName, ' ',
LCLObject.Name, ': ', LCLObject.ClassName);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.IndicatorMoved
Indicator moved event handler
------------------------------------------------------------------------------}
procedure TCarbonControl.IndicatorMoved;
begin
DebugLn('TCarbonControl.IndicatorMoved is invalid ', ClassName, ' ',
LCLObject.Name, ': ', LCLObject.ClassName);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.DoAction
Params: AControlPart - Control part to perform the action
Action event handler
------------------------------------------------------------------------------}
procedure TCarbonControl.DoAction(AControlPart: ControlPartCode);
begin
DebugLn('TCarbonControl.DoAction is invalid ', ClassName, ' ',
LCLObject.Name, ': ', LCLObject.ClassName);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.AddToWidget
Params: AParent - Parent widget
Adds control to parent widget
------------------------------------------------------------------------------}
procedure TCarbonControl.AddToWidget(AParent: TCarbonWidget);
var
I: Integer;
begin
// add frame to parent content area
for I := 0 to GetFrameCount - 1 do
begin
OSError(HIViewSetVisible(Frames[I], LCLObject.HandleObjectShouldBeVisible), Self, 'AddToWidget', SViewVisible);
OSError(HIViewAddSubview(AParent.Content, Frames[I]), Self, 'AddToWidget',
SViewAddView);
end;
AParent.ControlAdded;
//DebugLn('TCarbonControl.AddToWidget ' + LCLObject.Name + ' ' + DbgS(LCLObject.Parent.ClientRect));
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.RegisterEvents
Registers event handlers for control
------------------------------------------------------------------------------}
procedure TCarbonControl.RegisterEvents;
var
TmpSpec: EventTypeSpec;
Events: TCarbonControlEvents;
AccessibilitySpec: array [0..5] of EventTypeSpec;
begin
Events := GetValidEvents;
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlDispose);
InstallControlEventHandler(Widget,
RegisterEventHandler(@CarbonCommon_Dispose),
1, @TmpSpec, Pointer(Self), nil);
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlDraw);
InstallControlEventHandler(Content,
RegisterEventHandler(@CarbonCommon_Draw),
1, @TmpSpec, Pointer(Self), nil);
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlBoundsChanged);
InstallControlEventHandler(Frames[0],
RegisterEventHandler(@CarbonCommon_BoundsChanged),
1, @TmpSpec, Pointer(Self), nil);
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlSetFocusPart);
InstallControlEventHandler(Content,
RegisterEventHandler(@CarbonCommon_SetFocusPart),
1, @TmpSpec, Pointer(Self), nil);
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlGetNextFocusCandidate);
InstallControlEventHandler(Content,
RegisterEventHandler(@CarbonCommon_GetNextFocusCandidate),
1, @TmpSpec, Pointer(Self), nil);
TmpSpec := MakeEventSpec(kEventClassCommand, kEventCommandProcess);
InstallControlEventHandler(Widget,
RegisterEventHandler(@CarbonControl_CommandProcess),
1, @TmpSpec, Pointer(Self), nil);
// cursor set
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlSetCursor);
InstallControlEventHandler(Content,
RegisterEventHandler(@CarbonCommon_SetCursor),
1, @TmpSpec, Pointer(Self), nil);
// user messages
TmpSpec := MakeEventSpec(LCLCarbonEventClass, LCLCarbonEventKindUser);
InstallControlEventHandler(Widget,
RegisterEventHandler(@CarbonCommon_User),
1, @TmpSpec, Pointer(Self), nil);
if cceHit in Events then
begin
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlHit);
InstallControlEventHandler(Widget,
RegisterEventHandler(@CarbonControl_Hit),
1, @TmpSpec, Pointer(Self), nil);
end;
if cceValueChanged in Events then
begin
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlValueFieldChanged);
InstallControlEventHandler(Widget,
RegisterEventHandler(@CarbonControl_ValueChanged),
1, @TmpSpec, Pointer(Self), nil);
end;
if cceIndicatorMoved in Events then
begin
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlIndicatorMoved);
InstallControlEventHandler(Widget,
RegisterEventHandler(@CarbonControl_IndicatorMoved),
1, @TmpSpec, Pointer(Self), nil);
end;
// Accessibility
AccessibilitySpec[0].eventClass := kEventClassAccessibility;
AccessibilitySpec[0].eventKind := kEventAccessibleGetChildAtPoint;
AccessibilitySpec[1].eventClass := kEventClassAccessibility;
AccessibilitySpec[1].eventKind := kEventAccessibleGetFocusedChild;
AccessibilitySpec[2].eventClass := kEventClassAccessibility;
AccessibilitySpec[2].eventKind := kEventAccessibleGetAllAttributeNames;
AccessibilitySpec[3].eventClass := kEventClassAccessibility;
AccessibilitySpec[3].eventKind := kEventAccessibleGetAllParameterizedAttributeNames;
AccessibilitySpec[4].eventClass := kEventClassAccessibility;
AccessibilitySpec[4].eventKind := kEventAccessibleGetNamedAttribute;
// kEventAccessibleSetNamedAttribute
AccessibilitySpec[5].eventClass := kEventClassAccessibility;
AccessibilitySpec[5].eventKind := kEventAccessibleIsNamedAttributeSettable;
// kEventAccessibleGetAllActionNames
// kEventAccessiblePerformNamedAction
// kEventAccessibleGetNamedActionDescription
InstallControlEventHandler(Content,
RegisterEventHandler(@CarbonControl_Accessibility),
6, @AccessibilitySpec[0], Pointer(Self), nil);
{$IFDEF VerboseControlEvent}
DebugLn('TCarbonControl.RegisterEvents ', ClassName, ' ',
LCLObject.Name, ': ', LCLObject.ClassName);
{$ENDIF}
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.CreateWidget
Params: AParams - Creation parameters
Override to provide Carbon control creation
------------------------------------------------------------------------------}
procedure TCarbonControl.CreateWidget(const AParams: TCreateParams);
begin
CarbonWidgetFlag := cwfNone;
AddControlPart(Widget);
if Content <> ControlRef(Widget) then AddControlPart(Content);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.DestroyWidget
Override to do some clean-up
------------------------------------------------------------------------------}
procedure TCarbonControl.DestroyWidget;
begin
DisposeControl(ControlRef(Widget));
Widget := nil;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.AddControlPart
Params: AControl - Control
Adds control part of composite control
------------------------------------------------------------------------------}
procedure TCarbonControl.AddControlPart(const AControl: ControlRef);
var
TmpSpec: EventTypeSpec;
begin
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlTrack);
InstallControlEventHandler(AControl,
RegisterEventHandler(@CarbonCommon_Track),
1, @TmpSpec, Pointer(Self), nil);
TmpSpec := MakeEventSpec(kEventClassControl, kEventControlContextualMenuClick);
InstallControlEventHandler(AControl,
RegisterEventHandler(@CarbonCommon_ContextualMenuClick),
1, @TmpSpec, Pointer(Self), nil);
OSError(
SetControlProperty(AControl, LAZARUS_FOURCC, WIDGETINFO_FOURCC, SizeOf(Self), @Self),
Self, 'AddControlPart', SSetControlProp);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetContent
Returns: Content area control
------------------------------------------------------------------------------}
function TCarbonControl.GetContent: ControlRef;
begin
Result := ControlRef(Widget);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetControlContentRect
Params: ARect - Content area rectangle
Returns: If the function succeeds
------------------------------------------------------------------------------}
function TCarbonControl.GetControlContentRect(var ARect: TRect): Boolean;
var
AClientRect: MacOSAll.Rect;
ClientRegion: MacOSAll.RgnHandle;
begin
Result := False;
ClientRegion := MacOSAll.NewRgn();
try
if OSError(GetControlRegion(ControlRef(Widget), kControlContentMetaPart, ClientRegion),
Self, 'GetControlContentRect', 'GetControlRegion') then Exit;
Result := GetRegionBounds(ClientRegion, AClientRect{%H-}) <> nil;
if Result then ARect := CarbonRectToRect(AClientRect);
finally
MacOSAll.DisposeRgn(ClientRegion);
end;
{$IFDEF VerboseBounds}
DebugLn('TCarbonControl.GetControlContentRect ' + LCLObject.Name + ' ' + DbgS(ARect) +
' ' + DbgS(Result));
{$ENDIF}
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetFrame
Params: Frame index
Returns: Frame area control
------------------------------------------------------------------------------}
function TCarbonControl.GetFrame(Index: Integer): ControlRef;
begin
Result := ControlRef(Widget);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetFrameCount
Returns: Count of control frames
------------------------------------------------------------------------------}
class function TCarbonControl.GetFrameCount: Integer;
begin
Result := 1;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetForceEmbedInScrollView
Returns: Whether use scroll view even if no scroll bars are needed
------------------------------------------------------------------------------}
function TCarbonControl.GetForceEmbedInScrollView: Boolean;
begin
Result := False;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.UpdateContentBounds
Updates bounds of content area
------------------------------------------------------------------------------}
function TCarbonControl.UpdateContentBounds: Boolean;
var
R: TRect;
begin
Result := False;
if not GetClientRect(R{%H-}) then
begin
DebugLn('TCarbonControl.UpdateContentBounds Error - unable to get client area!');
Exit;
end;
{$IFDEF VerboseBounds}
DebugLn('TCarbonControl.UpdateContentBounds ' + DbgS(R));
{$ENDIF}
if OSError(HIViewSetFrame(Content, RectToCGRect(R)),
Self, 'UpdateContentBounds', SViewFrame) then Exit;
Result := True;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.EmbedInScrollView
Params: AParams - Creation parameters
Returns: Scroll view
Should be called right after widget creation only
------------------------------------------------------------------------------}
function TCarbonControl.EmbedInScrollView(const AParams: TCreateParams): HIViewRef;
var
ScrollBars: TScrollStyle;
begin
case AParams.Style and (WS_VSCROLL or WS_HSCROLL) of
WS_VSCROLL: ScrollBars := ssAutoVertical;
WS_HSCROLL: ScrollBars := ssAutoHorizontal;
WS_VSCROLL or WS_HSCROLL: ScrollBars := ssAutoBoth;
else
ScrollBars := ssNone;
end;
Result := EmbedInScrollView(ScrollBars);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.EmbedInScrollView
Params: AScrollBars - Scroll style
Returns: Scroll view
Should be called right after widget creation only
------------------------------------------------------------------------------}
function TCarbonControl.EmbedInScrollView(AScrollBars: TScrollStyle): HIViewRef;
var
ScrollOptions: MacOSAll.OptionBits;
Bounds: HIRect;
s, d: HIRect;
cnt: Integer;
const
SName = 'EmbedInScrollView';
begin
Result := nil;
if (not GetForceEmbedInScrollView) and (AScrollBars = ssNone) then
begin
Result := Widget;
Exit;
end;
case AScrollBars of
ssNone, ssBoth, ssAutoBoth:
ScrollOptions := kHIScrollViewOptionsVertScroll or
kHIScrollViewOptionsHorizScroll;
ssVertical, ssAutoVertical:
ScrollOptions := kHIScrollViewOptionsVertScroll;
ssHorizontal, ssAutoHorizontal:
ScrollOptions := kHIScrollViewOptionsHorizScroll;
end;
if OSError(HIScrollViewCreate(ScrollOptions, Result), Self, SName,
'HIScrollViewCreate') then Exit;
// set scroll view bounds
OSError(HIViewGetFrame(Widget, Bounds{%H-}), Self, SName, 'HIViewGetFrame');
OSError(HIViewSetFrame(Result, Bounds), Self, SName, SViewFrame);
OSError(HIScrollViewSetScrollBarAutoHide(Result,
AScrollBars in [ssNone, ssAutoVertical, ssAutoHorizontal, ssAutoBoth]),
Self, SName, SViewSetScrollBarAutoHide);
HIViewGetFrame(Widget, s{%H-});
HIViewRemoveFromSuperview(Widget);
OSError(HIViewAddSubview(Result, Widget), Self, SName, SViewAddView);
HIViewGetFrame(Widget, d{%H-});
// hack: for some reason, sometimes HIViewAddSubview, changes the size
// of the widget by scrollsize width/height (15 pixels). #19425
cnt:=10;
while (cnt>0) and ((s.size.width<>d.size.width) or (s.size.height<>d.size.height)) do
begin
HIViewRemoveFromSuperview(Widget);
HIViewSetFrame(Widget, s);
HIViewSetFrame(Result, s);
HIViewAddSubview(Result, Widget);
HIViewGetFrame(Widget, d);
dec(cnt);
end;
OSError(HIViewSetVisible(Widget, True), Self, SName, SViewVisible);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.ChangeScrollBars
Params: AScrollView - Scroll view
AScrollBars - Actual scroll style
ANewValue - New scroll style
Changes scroll style of scroll view
------------------------------------------------------------------------------}
procedure TCarbonControl.ChangeScrollBars(AScrollView: HIViewRef;
var AScrollBars: TScrollStyle; ANewValue: TScrollStyle);
begin
if ANewValue <> AScrollBars then
begin
if (not GetForceEmbedInScrollView) and (ANewValue = ssNone) or
(AScrollBars = ssNone) then
begin
RecreateWnd(LCLObject);
Exit;
end;
if ((ANewValue in [ssNone, ssBoth, ssAutoBoth]) and
(AScrollBars in [ssNone, ssBoth, ssAutoBoth])) or
((ANewValue in [ssVertical, ssAutoVertical]) and
(AScrollBars in [ssVertical, ssAutoVertical])) or
((ANewValue in [ssHorizontal, ssAutoHorizontal]) and
(AScrollBars in [ssHorizontal, ssAutoHorizontal])) then
begin
AScrollBars := ANewValue;
OSError(HIScrollViewSetScrollBarAutoHide(AScrollView,
AScrollBars in [ssNone, ssAutoVertical, ssAutoHorizontal, ssAutoBoth]),
Self, 'ChangeScrollBars', SViewSetScrollBarAutoHide);
end
else
RecreateWnd(LCLObject);
end;
end;
procedure TCarbonCustomControl.SendScrollUpdate;
var
Event: EventRef;
after, before : TRect;
const
SName='SendScrollUpdate';
begin
if OSError(
CreateEvent(nil, kEventClassScrollable, kEventScrollableInfoChanged, 0,
kEventAttributeUserEvent, Event{%H-}),
Self, SName, 'CreateEvent') then Exit;
try
GetClientRect(before{%H-});
OSError(SendEventToEventTarget(Event, GetControlEventTarget(FScrollView)),
Self, SName, 'SendEventToEventTarget');
if Assigned(FPopupWin) then
InvalidPaint:=True; // Needed to get scrollbars updated
GetClientRect(after{%H-});
if not CompareRect(@before, @after) then
UpdateLCLClientRect;
finally
ReleaseEvent(Event);
end;
end;
procedure TCarbonCustomControl.UpdateLCLClientRect;
var
R: TRect;
ClientR: TRect;
LCLR: TRect;
LCLClientR: TRect;
RChanged: Boolean;
ClientChanged: Boolean;
begin
if not Resizing then
begin
GetBounds(R{%H-});
GetClientRect(ClientR{%H-});
LCLR:=LCLObject.BoundsRect;
LCLClientR:=LCLObject.ClientRect;
RChanged:=not CompareRect(@R,@LCLR);
ClientChanged:=not CompareRect(@ClientR,@LCLClientR);
if not RChanged and ((LCLObject.CachedClientWidth <> ClientR.Right) or
(LCLObject.CachedClientHeight <> ClientR.Bottom)) then
LCLObject.InvalidateClientRectCache(False);
if RChanged or ClientChanged then
LCLSendSizeMsg(LCLObject, R.Right - R.Left, R.Bottom - R.Top, Size_SourceIsInterface);
end;
end;
procedure TCarbonControl.AllowMenuProcess(MenuHotKey: AnsiChar; State: TShiftState; var AllowCommandProcess: Boolean);
begin
AllowCommandProcess:=True;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetValue
Returns: The value of Carbon control
------------------------------------------------------------------------------}
function TCarbonControl.GetValue: Integer;
begin
Result := GetControl32BitValue(ControlRef(Widget));
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetValue
Params: AValue - New control value
Sets the Carbon control value
------------------------------------------------------------------------------}
procedure TCarbonControl.SetValue(AValue: Integer);
begin
SetControl32BitValue(ControlRef(Widget), AValue);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetMinimum
Params: AValue - New control minimum
Sets the Carbon control minimum
------------------------------------------------------------------------------}
procedure TCarbonControl.SetMinimum(AValue: Integer);
begin
SetControl32BitMinimum(ControlRef(Widget), AValue);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetMaximum
Params: AValue - New control maximum
Sets the Carbon control maximum
------------------------------------------------------------------------------}
procedure TCarbonControl.SetMaximum(AValue: Integer);
begin
SetControl32BitMaximum(ControlRef(Widget), AValue);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetViewSize
Params: AValue - New control view size
Sets the Carbon control view size
------------------------------------------------------------------------------}
procedure TCarbonControl.SetViewSize(AValue: Integer);
begin
SetControlViewSize(ControlRef(Widget), AValue);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.Invalidate
Params: Rect - Pointer to rect (optional)
Invalidates the specified client rect or entire area of control
------------------------------------------------------------------------------}
procedure TCarbonControl.Invalidate(Rect: PRect);
var
I: Integer;
R: TRect;
begin
if Rect = nil then
begin
for I := 0 to GetFrameCount - 1 do
OSError(
HiViewSetNeedsDisplay(Frames[I], True), Self, SInvalidate, SViewNeedsDisplay);
end
else
begin
R := Rect^;
InflateRect(R, 1, 1);
OSError(
HiViewSetNeedsDisplayInRect(Content, RectToCGRect(R), True), Self,
SInvalidate, SViewNeedsDisplayRect);
end;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.IsEnabled
Returns: If control is enabled
------------------------------------------------------------------------------}
function TCarbonControl.IsEnabled: Boolean;
begin
Result := IsControlEnabled(Frames[0]);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.IsVisible
Returns: If control is visible
------------------------------------------------------------------------------}
function TCarbonControl.IsVisible: Boolean;
begin
Result := MacOSAll.IsControlVisible(Frames[0]);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.Enable
Params: AEnable - if enable
Returns: If control is enabled
Changes control enabled
------------------------------------------------------------------------------}
function TCarbonControl.Enable(AEnable: Boolean): Boolean;
var
I: Integer;
begin
Result := not MacOSAll.IsControlEnabled(Frames[0]);
if AEnable then
begin
for I := 0 to GetFrameCount - 1 do
OSError(MacOSAll.EnableControl(Frames[I]), Self, SEnable, SEnableControl);
end
else
for I := 0 to GetFrameCount - 1 do
OSError(MacOSAll.DisableControl(Frames[I]), Self, SEnable, SDisableControl);
Invalidate;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetFrameBounds
Params: ARect - TRect
Returns: If function succeeds
Returns the control bounding rectangle relative to the client origin of its
parent
------------------------------------------------------------------------------}
function TCarbonControl.GetFrameBounds(var ARect: TRect): Boolean;
var
BoundsRect: MacOSAll.Rect;
begin
Result := False;
if GetControlBounds(Frames[0], BoundsRect{%H-}) = nil then
begin
DebugLn('TCarbonControl.GetFrameBounds failed!');
Exit;
end;
if Assigned(FPopupWin) then
GetWindowBounds(FPopupWin, kWindowContentRgn, BoundsRect);
ARect := SortRect(CarbonRectToRect(BoundsRect));
Result := True;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetBounds
Params: ARect - Record for control coordinates
Returns: If function succeeds
Returns the control bounding rectangle relative to the client origin of its
parent
------------------------------------------------------------------------------}
function TCarbonControl.GetBounds(var ARect: TRect): Boolean;
begin
Result := GetFrameBounds(ARect);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetScreenBounds
Params: ARect - Record for control coordinates
Returns: If function succeeds
Returns the control bounding rectangle relative to the screen
------------------------------------------------------------------------------}
function TCarbonControl.GetScreenBounds(var ARect: TRect): Boolean;
var
BoundsRect: HIRect;
WindowRect: MacOSAll.Rect;
const
SName = 'GetScreenBounds';
begin
Result := False;
if not GetFrameBounds(ARect) then Exit;
BoundsRect := RectToCGRect(ARect);
BoundsRect.origin.x := 0;
BoundsRect.origin.y := 0;
if OSError(HIViewConvertRect(BoundsRect, Frames[0], nil), Self, SName,
'HIViewConvertRect') then Exit;
if OSError(GetWindowBounds(GetTopParentWindow, kWindowStructureRgn,
WindowRect{%H-}), Self, SName, SGetWindowBounds) then Exit;
ARect := CGRectToRect(BoundsRect);
OffsetRect(ARect, WindowRect.left, WindowRect.top);
Result := True;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetBounds
Params: ARect - Record for control coordinates
Returns: If function succeeds
Sets the control bounding rectangle relative to the client origin of its
parent
------------------------------------------------------------------------------}
function TCarbonControl.SetBounds(const ARect: TRect): Boolean;
var
B :CGRect;
T: TRect;
begin
Result := False;
if (IsDrawEvent > 0) then
if Assigned(FPopupWin) then
HIWindowGetBounds(FPopupWin, kWindowContentRgn, kHICoordSpaceWindow, B)
else
HIViewGetBounds(Frames[0], B{%H-});
T := ARect;
if Assigned(FPopupWin) then
begin
SetWindowBounds(FPopupWin, kWindowContentRgn, GetCarbonRect(ARect));
T := Classes.Rect(0, 0, ARect.Right - ARect.Left, ARect.Bottom - ARect.Top);
end;
if OSError(HIViewSetFrame(Frames[0], RectToCGRect(T)),
Self, SSetBounds, SViewFrame) then Exit;
// ensure bounds are send back to LCL once after creation
BoundsChanged;
if (IsDrawEvent > 0) and not Types.EqualRect( CGRectToRect(B), ARect) then
InvalidPaint := true;
Result := True;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetFocus
Sets the focus to control
------------------------------------------------------------------------------}
procedure TCarbonControl.SetFocus;
var
Window: WindowRef;
Control: ControlRef;
begin
Window := GetTopParentWindow;
OSError(
SetUserFocusWindow(Window), Self, SSetFocus, SSetUserFocusWindow);
OSError(GetKeyboardFocus(Window, Control{%H-}), Self, SSetFocus, SGetKeyboardFocus);
if Control <> ControlRef(Widget) then
OSError(SetKeyboardFocus(Window, ControlRef(Widget), kControlFocusNextPart),
Self, SSetFocus, 'SetKeyboardFocus');
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetColor
Params: AColor - New color
Sets the color of control (for edit like controls)
------------------------------------------------------------------------------}
procedure TCarbonControl.SetColor(const AColor: TColor);
var
FontStyle: ControlFontStyleRec;
Color: TColor;
begin
// get current font style preserve other font settings
OSError(GetControlData(ControlRef(Widget), kControlEntireControl,
kControlFontStyleTag, SizeOf(FontStyle), @FontStyle, nil), Self, SSetColor,
SGetData, SControlFont);
FontStyle.flags := FontStyle.flags or kControlUseBackColorMask;
Color := AColor;
if Color = clDefault then
Color := LCLObject.GetDefaultColor(dctBrush);
FontStyle.backColor := ColorToRGBColor(Color);
OSError(SetControlFontStyle(ControlRef(Widget), FontStyle), Self, SSetColor,
SSetFontStyle);
// invalidate control
Invalidate;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetFont
Params: AFont - New font
Sets the font of control
------------------------------------------------------------------------------}
procedure TCarbonControl.SetFont(const AFont: TFont);
var
FontStyle: ControlFontStyleRec;
ID: FontFamilyID;
const
SName = 'SetFont';
begin
// get current font style to preserve other font settings
OSError(GetControlData(ControlRef(Widget), kControlEntireControl,
kControlFontStyleTag, SizeOf(FontStyle), @FontStyle, nil), Self, SName,
SGetData, SControlFont);
if not FindQDFontFamilyID(AFont.Name, ID{%H-}) then ID:=0;
FontStyle.flags := FontStyle.flags or kControlUseFontMask or kControlUseSizeMask or
kControlUseFaceMask or kControlUseForeColorMask;
if ID = 0 then // use default font
FontStyle.flags := FontStyle.flags and not kControlUseFontMask;
if AFont.Size = 0 then // use default size
FontStyle.flags := FontStyle.flags and not kControlUseSizeMask;
FontStyle.font := SInt16(ID);
FontStyle.size := AFont.Size;
FontStyle.style := FontStyleToQDStyle(AFont.Style);
FontStyle.foreColor := ColorToRGBColor(AFont.Color);
OSError(SetControlFontStyle(ControlRef(Widget), FontStyle), Self, SName,
SSetFontStyle);
// invalidate control
Invalidate;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetZOrder
Params: AOrder - Order
ARefWidget - Reference widget
Sets the Z order of control
------------------------------------------------------------------------------}
procedure TCarbonControl.SetZOrder(AOrder: HIViewZOrderOp;
ARefWidget: TCarbonWidget);
var
RefView: HIViewRef;
const
SName = 'SetZOrder';
begin
if ARefWidget = nil then
RefView := nil
else
if AOrder = kHIViewZOrderBelow then
RefView := TCarbonControl(ARefWidget).Frames[0]
else
RefView := TCarbonControl(ARefWidget).Frames[GetFrameCount - 1];
OSError(HIViewSetZOrder(Frames[0], AOrder, RefView),
Self, SName, 'HIViewSetZOrder');
if GetFrameCount = 2 then // second frame is allways above first
OSError(HIViewSetZOrder(Frames[1], kHIViewZOrderAbove, Frames[0]),
Self, SName, 'HIViewSetZOrder', 'Frames[1]');
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.ShowHide
Params: AVisible - if show
Shows or hides control
------------------------------------------------------------------------------}
procedure TCarbonControl.ShowHide(AVisible: Boolean);
var
I: Integer;
v: Boolean;
begin
//DebugLn('TCarbonControl.ShowHide ' + DbgSName(LCLobject),' ', DbgS(AVisible));
v := IsVisible;
for I := 0 to GetFrameCount - 1 do
OSError(HIViewSetVisible(Frames[I], AVisible or (csDesigning in LCLobject.ComponentState)),
Self, 'ShowHide', SViewVisible);
if Assigned(FPopupWin) then
MacOSAll.ShowHide(FPopupWin, AVisible);
if (IsDrawEvent > 0) and (AVisible <> v) and (AVisible or (csDesigning in LCLobject.ComponentState)) and (GetFrameCount>0) then
InvalidPaint := true;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetText
Params: S - Text
Returns: If the function succeeds
Gets the text or caption of control
------------------------------------------------------------------------------}
function TCarbonControl.GetText(var S: String): Boolean;
begin
Result := False; // control caption is static, edit controls override this
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.SetText
Params: S - New text
Returns: If the function succeeds
Sets the text or caption of control
------------------------------------------------------------------------------}
function TCarbonControl.SetText(const S: String): Boolean;
var
CFString: CFStringRef;
T: String;
begin
Result := False;
T := S;
DeleteAmpersands(T);
CreateCFString(T, CFString);
try
if OSError(HIViewSetText(HIViewRef(Widget), CFString), Self, SSetText,
'HIViewSetText') then Exit;
Result := True;
finally
FreeCFString(CFString);
end;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.Update
Returns: If the function succeeds
Updates control
------------------------------------------------------------------------------}
function TCarbonControl.Update: Boolean;
var
I: Integer;
const
SName = 'Update';
begin
Result := True;
if IsDrawEvent>0 then Exit;
if Widget <> HIViewRef(Frames[0]) then
if OSError(HIViewRender(Widget), Self, SName, SViewRender) then Result := False;
if Widget <> Content then
if OSError(HIViewRender(Content), Self, SName, SViewRender) then Result := False;
for I := 0 to GetFrameCount - 1 do
if OSError(HIViewRender(HIViewRef(Frames[I])), Self, SName, SViewRender) then
Result := False;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.WidgetAtPos
Params: P
Returns: Retrieves the embedded Carbon control at the specified pos
------------------------------------------------------------------------------}
function TCarbonControl.WidgetAtPos(const P: TPoint): ControlRef;
begin
Result := nil;
OSError(HIViewGetSubviewHit(Frames[0], PointToHIPoint(P), True, Result),
Self, 'WidgetAtPos', 'HIViewGetSubviewHit');
if Result = nil then Result := Widget;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetTopParentWindow
Returns: Retrieves the parent window reference of the Carbon control
------------------------------------------------------------------------------}
function TCarbonControl.GetTopParentWindow: WindowRef;
var
Window: TControl;
begin
Window := LCLObject.GetTopParent;
if Assigned(FPopupWin) then
Result := FPopupWin
else
if (Window is TWinControl) and (Window.Parent = nil) and (TWinControl(Window).ParentWindow <> 0) then
Result := TCarbonControl(TWinControl(Window).ParentWindow).GetTopParentWindow
else
if Window is TCustomForm then
Result := TCarbonWindow((Window as TWinControl).Handle).GetTopParentWindow
else
Result := nil;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetThemeDrawState
Returns: The control widget actual theme draw state (active, ...)
------------------------------------------------------------------------------}
function TCarbonControl.GetThemeDrawState: ThemeDrawState;
begin
if IsControlActive(ControlRef(Widget)) then
begin
if IsControlHilited(ControlRef(Widget)) then Result := kThemeStatePressed
else Result := kThemeStateActive;
end
else Result := kThemeStateInactive;
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetMousePos
Returns: The position of mouse cursor in local coordinates
------------------------------------------------------------------------------}
function TCarbonControl.GetWindowRelativePos(winX, winY: Integer): TPoint;
var
MousePoint: HIPoint;
const
SName = 'GetMousePos';
begin
MousePoint.X := winX;
MousePoint.Y := winY;
OSError(HIViewConvertPoint(MousePoint, nil, Content), Self, SName, SViewConvert);
Result.X := Trunc(MousePoint.X);
Result.Y := Trunc(MousePoint.Y);
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetClientRect
Params: ARect - Record for client area coordinates
Returns: If the function succeeds
Returns the control client rectangle relative to the control origin
------------------------------------------------------------------------------}
function TCarbonControl.GetClientRect(var ARect: TRect): Boolean;
var
R: HIRect;
begin
Result := False;
// controls without content area have clientrect = boundsrect
if OSError(HIViewGetFrame(Content, R{%H-}),
Self, 'GetClientRect', 'HIViewGetFrame') then Exit;
ARect := CGrectToRect(R);
OffsetRect(ARect, -ARect.Left, -ARect.Top);
Result := True;
{$IFDEF VerboseBounds}
DebugLn('TCarbonControl.GetClientRect ' + LCLObject.Name + ' ' + DbgS(ARect) +
' ' + DbgS(Result));
{$ENDIF}
end;
{------------------------------------------------------------------------------
Method: TCarbonControl.GetPreferredSize
Returns: The prefered size of control for autosizing or (0, 0)
------------------------------------------------------------------------------}
function TCarbonControl.GetPreferredSize: TPoint;
var
R: MacOSAll.Rect;
S: SmallInt;
begin
Result.X := 0;
Result.Y := 0;
if (LCLObject is TWinControl) and (TWinControl(LCLObject).ControlCount>0) then
begin
end else begin
R := GetCarbonRect(0, 0, 0, 0);
if OSError(GetBestControlRect(ControlRef(Widget), R, S{%H-}), Self,
'GetPreferredSize', 'GetBestControlRect') then Exit;
Result.X := R.right - R.left;
Result.Y := R.bottom - R.top;
end;
end;