วิธีสร้างฟอร์ม“ ไม่เปิดใช้งาน” ใน Firemonkey


147

ใน XCode โดยการเพิ่มวิธีการเหล่านี้ในคลาสย่อย NSView ของคุณสามารถป้องกันไม่ให้หน้าต่างทำงานเมื่อคลิกที่:

- (BOOL)shouldDelayWindowOrderingForEvent:(NSEvent )theEvent {
    return YES;
}
- (BOOL)acceptsFirstMouse:(NSEvent )theEvent {
    return YES; 
}
- (void)mouseDown:(NSEvent )theEvent {
    [[[NSApp]] preventWindowOrdering]; 
}

ในแพลตฟอร์ม Windows มันทำโดยรหัสง่ายๆนี้:

HWND hWnd = FindWindowW((String("FM") + fmxForm->ClassName()).c_str(), 
    fmxForm->Caption.c_str());

SetWindowLong(hWnd, GWL_EXSTYLE,
    GetWindowLong(hWnd, GWL_EXSTYLE) | WS_EX_NOACTIVATE);

ฉันจะ subclass NSView เพื่อป้องกันไม่ให้ FMX TForm ของฉันทำงานเมื่อคลิกที่มันได้อย่างไร?

ฉันจะสร้างฟอร์ม" ไม่เปิดใช้งาน " ในfiremonkey ได้อย่างไร


3
ไม่แน่ใจว่ามันใช้กับ Firemonkey เช่นกันหรือถ้าตอบคำถามของคุณถูกต้อง แต่คุณอาจต้องการดูตัวอย่างนี้: delphi.about.com/od/delphitips2008/qt/ex_noactivate.htm
TildalWave

ขอบคุณ แต่มันเป็นเพียงสำหรับ Windows และวิธีที่ง่ายกว่าคือโซลูชันของฉันที่อธิบายไว้ข้างต้นโดย "SetWindowLong" คำถามเกี่ยวกับ MacOS
mh taqia


Devon: ลิงค์นี้จะช่วยฉันได้อย่างไร
mh taqia

ขอบคุณ WBAR มันเป็นรางวัลที่สอง!
mh taqia

คำตอบ:


13

เป็นไปได้ที่จะใช้NSPanelกับธงNSNonactivatingPanelMask รูปแบบ NSView ของ fmx ควรเป็นรายการย่อยของ NSPanel ฉันได้เขียนคลาสตัวช่วยซึ่งใช้ได้กับทั้งแพลตฟอร์ม Windows และ Mac ( ทำงานบน XE4 ):

unit NoActivateForm;

interface

uses Fmx.Forms, Fmx.Types
{$IFDEF POSIX}
    , Macapi.AppKit
{$ENDIF}
    ;

type TNoActivateForm = class
private
    form: TForm;
{$IFDEF POSIX}
    panel: NSPanel;
    timer: TTimer;  // for simulating mouse hover event
{$ENDIF}
    procedure SetPosition(const x, y: Integer);
    procedure GetPosition(var x, y: Integer);
    procedure SetDimensions(const width, height: Integer);
    procedure SetLeft(const Value: Integer);
    procedure SetTop(const Value: Integer);
    procedure SetHeight(const Value: Integer);
    procedure SetWidth(const Value: Integer);
    procedure SetVisible(const Value: Boolean);
    function GetLeft: Integer;
    function GetTop: Integer;
    function GetHeight: Integer;
    function GetWidth: Integer;
    function GetVisible: Boolean;
{$IFDEF POSIX}
    procedure OnTimer(Sender: TObject);
{$ENDIF}
public
    constructor Create(AForm: TForm);
    destructor Destroy; override;
    property Left: Integer read GetLeft write SetLeft;
    property Top: Integer read GetTop write SetTop;
    property Height: Integer read GetHeight write SetHeight;
    property Width: Integer read GetWidth write SetWidth;
    property Visible: Boolean read GetVisible write SetVisible;
end;

implementation
uses
    Classes, System.Types
{$IFDEF MSWINDOWS}
    , Winapi.Windows;
{$ELSE}
    , Macapi.CocoaTypes, FMX.Platform.Mac, Macapi.CoreGraphics, Macapi.CoreFoundation;
{$ENDIF}

constructor TNoActivateForm.Create(AForm: TForm);
{$IFDEF POSIX}
var
    rect: NSRect;
    bounds: CGRect;
    window: NSWindow;
    style: integer;
    panelCount: integer;
begin
    form := AForm;
    form.Visible := false;
    bounds := CGDisplayBounds(CGMainDisplayID);
    rect := MakeNSRect(form.Left, bounds.size.height - form.Top - form.Height,
        form.ClientWidth, form.ClientHeight);
    style := NSNonactivatingPanelMask;
    style := style or NSHUDWindowMask;
    panel := TNSPanel.Wrap(
        TNSPanel.Alloc.initWithContentRect(rect, style, NSBackingStoreBuffered,
        true));
    panel.setFloatingPanel(true);
    //panel.setHasShadow(false); optional
    window := WindowHandleToPlatform(form.Handle).Wnd;

    panel.setContentView(TNSView.Wrap(window.contentView));
    TNSView.Wrap(window.contentView).retain;

    timer := TTimer.Create(form.Owner);
    timer.OnTimer := OnTimer;
    timer.Interval := 50;
end;
{$ELSE}
var hWin: HWND;
begin
    form := AForm;
    form.TopMost := true;
    hWin := FindWindow(PWideChar('FM' + form.ClassName), PWideChar(form.Caption));
    if hWin <> 0 then
        SetWindowLong(hWin, GWL_EXSTYLE,
            GetWindowLong(hWin, GWL_EXSTYLE) or WS_EX_NOACTIVATE);
end;
{$ENDIF}

destructor TNoActivateForm.Destroy;
{$IFDEF POSIX}
begin
    panel.release;
end;
{$ELSE}
begin
end;
{$ENDIF}

procedure TNoActivateForm.SetPosition(const x, y: Integer);
{$IFDEF POSIX}
var point: NSPoint;
    screen: CGRect;
begin
    screen := CGDisplayBounds(CGMainDisplayID);
    point.x := x;
    point.y := round(screen.size.height) - y - form.height;
    panel.setFrameOrigin(point);
end;
{$ELSE}
begin
    form.Left := x;
    form.Top := y;
end;
{$ENDIF}

procedure TNoActivateForm.GetPosition(var x, y: Integer);
{$IFDEF POSIX}
var screen: CGRect;
begin
    screen := CGDisplayBounds(CGMainDisplayID);
    x := round(panel.frame.origin.x);
    y := round(screen.size.height - panel.frame.origin.y - panel.frame.size.height);
end;
{$ELSE}
begin
    x := form.Left;
    y := form.Top;
end;
{$ENDIF}

procedure TNoActivateForm.SetDimensions(const width, height: Integer);
{$IFDEF POSIX}
var size: NSSize;
begin
    size.width := width;
    size.height := height;
    panel.setContentSize(size);
end;
{$ELSE}
begin
    form.width := width;
    form.height := height;
end;
{$ENDIF}

procedure TNoActivateForm.SetLeft(const Value: Integer);
begin
    SetPosition(Value, Top);
end;

procedure TNoActivateForm.SetTop(const Value: Integer);
begin
    SetPosition(Left, Value);
end;

procedure TNoActivateForm.SetHeight(const Value: Integer);
begin
    SetDimensions(Width, Value);
end;

procedure TNoActivateForm.SetWidth(const Value: Integer);
begin
    SetDimensions(Value, Height);
end;

procedure TNoActivateForm.SetVisible(const Value: Boolean);
begin
{$IFDEF POSIX}
    panel.setIsVisible(Value);
{$ELSE}
    form.visible := Value;
{$ENDIF}
end;

function TNoActivateForm.GetLeft: Integer;
var x, y: Integer;
begin
    GetPosition(x, y);
    result := x;
end;

function TNoActivateForm.GetTop: Integer;
var x, y: Integer;
begin
    GetPosition(x, y);
    result := y;
end;

function TNoActivateForm.GetHeight: Integer;
begin
{$IFDEF POSIX}
    result := round(panel.frame.size.height);
{$ELSE}
    result := form.Height;
{$ENDIF}
end;

function TNoActivateForm.GetWidth: Integer;
begin
{$IFDEF POSIX}
    result := round(panel.frame.size.width);
{$ELSE}
    result := form.Width;
{$ENDIF}
end;

function TNoActivateForm.GetVisible: Boolean;
begin
{$IFDEF POSIX}
    result := panel.isVisible();
{$ELSE}
    result := form.visible;
{$ENDIF}
end;

{$IFDEF POSIX}
procedure TNoActivateForm.OnTimer(Sender: TObject);
var event: CGEventRef;
    point: CGPoint;
    form_rect: TRectF;
    client_point, mouse_loc: TPointF;
    shift: TShiftState;
begin
    event := CGEventCreate(nil);
    point := CGEventGetLocation(event);
    CFRelease(event);
    mouse_loc.SetLocation(point.x, point.y);
    if Visible = true then
    begin
        form_rect := RectF(0, 0, form.Width, form.Height);
        client_point.X := mouse_loc.X - Left;
        client_point.Y := mouse_loc.y - Top;
        if PtInRect(form_rect, client_point) then
            form.MouseMove(shift, client_point.x, client_point.y)
        else
            form.MouseLeave();
    end;
end;
{$ENDIF}

end.

การใช้งานของหน่วยข้างต้น:

TNoActivateForm *naKeyboard; // global scope    
void __fastcall TfrmKeyboard::TfrmKeyboard(TObject *Sender)
{
    naKeyboard = new TNoActivateForm(frmKeyboard); // frmKeyboard is a normal fmx form
    naKeyboard->Visible = true;
}

ถ้า frmKeyboard เป็นแบบฟอร์มหลักของคุณอย่าใส่รหัสข้างบนลงใน form constructor ขอแนะนำให้ใส่ไว้ใน OnShow

ป้อนคำอธิบายรูปภาพที่นี่

หมายเหตุ : WindowHandleToPlatform ดูเหมือนจะไม่มีอยู่ใน XE3 เพื่อให้สามารถแทนที่บรรทัดนั้นได้

window := NSWindow(NSWindowFromObjC(FmxHandleToObjC(Form.Handle)));

1
ขอบคุณสำหรับโซลูชันที่ยอดเยี่ยม - windowhandletoplatform ดูเหมือนจะไม่มีอยู่ใน XE3 ดังนั้นบรรทัดนั้นสามารถแทนที่ด้วย window: = NSWindow (NSWindowFromObjC (FmxHandleToObjC (Form.Handle)))
David Peters

2

คุณสามารถปิดการจัดการเมาส์ของฟอร์มเพื่อป้องกันการโฟกัส สมมติว่าฟอร์มของคุณเรียกว่า myform:

uses fmx.platform.mac, macapi.appkit;
.
.
Var nswin:nswindow;
.
.  
NSWin:= NSWindow(NSWindowFromObjC(FmxHandleToObjC(myform.Handle))); { get the NSWindow }
NSWin.setIgnoresMouseEvents(true);                                 { ignore mouse events }
NSWin.setAcceptsMouseMovedEvents(false);

มีปัญหาเล็กน้อยคือมันไม่หยุดคลิกเมาส์ขวา หากเป็นปัญหาคุณจะต้องตอบกลับเหตุการณ์ที่เกิดขึ้นในแบบฟอร์มและโทรไปที่แบบฟอร์มหลักที่มีการวางเมาส์ไว้เพื่อไม่ให้เหตุการณ์เมาส์เสีย เนื่องจากเมาส์ขวาลงจากนั้นจะจับกิจกรรมเมาส์คุณจึงต้องตอบสนองต่อการเลื่อนเมาส์และเลื่อนกิจกรรมขึ้นด้วยเช่นกัน - ส่งต่อไปยังฟอร์มหลักของคุณ แม้ว่ามันจะจับเมาส์ด้วยการคลิกขวา แต่มันก็ยังไม่ได้โฟกัสที่แบบฟอร์ม

ซอฟต์แวร์ Dave Peters DP


ไม่ถูกต้องไม่ทำงาน รูปแบบการเปลี่ยนแปลงแป้นพิมพ์มุ่งเน้นไปที่การคลิก
mh taqia

มันไม่ได้รับการโฟกัส แต่สิ่งที่เกิดขึ้นคือการคลิกเมาส์ใด ๆ ก็ตามที่อยู่ใต้ฟอร์ม หากคุณสามารถจัดเรียงว่าแบบฟอร์มที่ไม่มุ่งเน้นมีชุดคุณสมบัติ TopMost และมีเพียงส่วนที่ว่างเปล่าของแบบฟอร์มหลักของคุณเท่านั้นที่อยู่ด้านล่างมันจะทำงานได้ หากคุณมีการควบคุมรูปแบบหลักใด ๆ ภายใต้หน้าต่างจากนั้นพวกเขาจะได้รับการมุ่งเน้นเมื่อคุณคลิกเมาส์เป็นหน้าต่างที่ไม่ใช่การมุ่งเน้นการทำงานเช่นมันไม่ได้มี หากหน้าต่างถูกวางไว้เหนือเดสก์ท็อปเดสก์ท็อปจะได้รับการคลิกเมาส์และแอปพลิเคชันของคุณขาดโฟกัส
David Peters

โปรดทราบว่าฉันต้องการกิจกรรมเมาส์ ฉันไม่สามารถเพิกเฉยต่อเหตุการณ์เมาส์ได้ ฉันต้องการคลิกที่ปุ่มและฉันต้องการให้มีภาพเคลื่อนไหว firemonkey เมื่อตัวชี้เมาส์เข้าสู่การควบคุม สมมติว่าฉันต้องการสร้างแป้นพิมพ์เสมือนจริงแอปพลิเคชันพื้นหน้าคือ (ตัวอย่าง) TextEdit เมื่อฉันคลิกที่ปุ่มบนรูปแบบ fmx ของฉันเหตุการณ์แป้นพิมพ์จะถูกสร้างขึ้นและตัวละครจะถูกพิมพ์
mh taqia
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.