Chromeのタブドラッグ機能をエミュレートしようとしています。ユーザーがタブをタブストリップの新しい場所にドラッグしたり、アプリケーションの外にドロップして新しいウィンドウを作成したりできるようにしたいと思います。アプリケーション内でのドラッグは簡単ですが、ユーザーがアプリ以外の場所にドロップしたことを検出するにはどうすればよいですか?
本質的に、私は「ティアオフ」タブを実装しようとしています。
Chromeのタブドラッグ機能をエミュレートしようとしています。ユーザーがタブをタブストリップの新しい場所にドラッグしたり、アプリケーションの外にドロップして新しいウィンドウを作成したりできるようにしたいと思います。アプリケーション内でのドラッグは簡単ですが、ユーザーがアプリ以外の場所にドロップしたことを検出するにはどうすればよいですか?
本質的に、私は「ティアオフ」タブを実装しようとしています。
OnEndDrag
マウスはドラッグ操作中にキャプチャされるため、アプリケーションのフォーム外であっても、ハンドラーでドラッグ操作がいつ終了したかを検出することに問題はありません。「ターゲット」オブジェクトをテストすることで、ドロップが受け入れられるかどうかを判断できます。ドロップが受け入れられない場合は、マウスの位置をテストすることで、ドロップがアプリケーションの外部にあるかどうかを判断できます。
ただし、このアプローチにはまだ問題があります。「Esc」キーを押してドラッグがキャンセルされたかどうかはわかりません。フォームの外側ではコントロールが呼び出されないため、ドラッグ カーソルを「承認済み」に設定できないという問題もOnDragOver
あります。
これらの問題は、作成したドラッグ オブジェクトを使用してドラッグ操作の動作を変更することで解決できます。以下に一例を示します。
unit Unit1;
interface
uses
Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics,
Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.ComCtrls;
type
TForm1 = class(TForm)
PageControl1: TPageControl;
TabSheet1: TTabSheet;
TabSheet2: TTabSheet;
TabSheet3: TTabSheet;
procedure FormCreate(Sender: TObject);
procedure PageControl1MouseDown(Sender: TObject; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer);
procedure PageControl1StartDrag(Sender: TObject;
var DragObject: TDragObject);
procedure PageControl1EndDrag(Sender, Target: TObject; X, Y: Integer);
procedure PageControl1DragOver(Sender, Source: TObject; X, Y: Integer;
State: TDragState; var Accept: Boolean);
end;
var
Form1: TForm1;
implementation
{$R *.dfm}
procedure TForm1.FormCreate(Sender: TObject);
begin
PageControl1.DragMode := dmManual;
end;
type
TDragFloatSheet = class(TDragControlObjectEx)
private
class var
FDragSheet: TTabSheet;
FDragPos: TPoint;
FCancelled: Boolean;
protected
procedure WndProc(var Msg: TMessage); override;
end;
procedure TDragFloatSheet.WndProc(var Msg: TMessage);
begin
if (Msg.Msg = CN_KEYDOWN) and (Msg.WParam = VK_ESCAPE) then
FCancelled := True;
FDragPos := DragPos;
inherited;
if (Msg.Msg = WM_MOUSEMOVE) and
(not Assigned(FindVCLWindow(SmallPointToPoint(TWMMouse(Msg).Pos)))) then
Winapi.Windows.SetCursor(Screen.Cursors[GetDragCursor(True, 0, 0)]);
end;
//-------------------
procedure TForm1.PageControl1MouseDown(Sender: TObject; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer);
begin
TDragFloatSheet.FDragSheet :=
(Sender as TPageControl).Pages[TPageControl(Sender).IndexOfTabAt(X, Y)];
PageControl1.BeginDrag(False);
end;
procedure TForm1.PageControl1StartDrag(Sender: TObject;
var DragObject: TDragObject);
begin
DragObject := TDragFloatSheet.Create(Sender as TPageControl);
end;
procedure TForm1.PageControl1DragOver(Sender, Source: TObject; X, Y: Integer;
State: TDragState; var Accept: Boolean);
var
TargetSheet: TTabSheet;
begin
TargetSheet :=
(Sender as TPageControl).Pages[TPageControl(Sender).IndexOfTabAt(X, Y)];
Accept := Assigned(TargetSheet) and (TargetSheet <> TDragFloatSheet.FDragSheet);
end;
procedure TForm1.PageControl1EndDrag(Sender, Target: TObject; X, Y: Integer);
begin
if Assigned(Target) then begin
// normal processing, f.i. find the target tab as in OnDragOver
// and switch positions with TDragFloatSheet.FDragSheet
end else begin
if not TDragFloatSheet.FCancelled then begin
if not Assigned(FindVCLWindow(TDragFloatSheet.FDragPos)) then begin
// drop TDragFloatSheet.FDragSheet at TDragFloatSheet.FDragPos
end;
end;
end;
end;
end.