请教窗体拖放问题 ????? 1.如何将一个无边界的窗体拖着移动2.如果将其它的内容(比如IE的文字)拖放到此窗体上,并进行相应处理 解决方案 » 免费领取超大流量手机卡,每月29元包185G流量+100分钟通话, 中国电信官方发货 最简单的→实现拖动无标题栏窗体:ReleaseCapture;SendMessage(Handle, WM_NCLBUTTONDOWN, HTCAPTION, 0);把上述两行代码加到窗体的鼠标按下事件里面。 http://blog.csdn.net/cureSHY/拖放(DragDrop)是Windows提供的一种快捷的操作方式。作为基于Windows的开发工具,Delphi同样支持拖放操作,而且开发应用系统的拖放功能十分方便,真正体现了Delphi 的强大功能和方便性。 Delphi提供的所有控件(Control,即能获得输入焦点的部件)都支持拖放操作,并有相应的拖放属性、拖放事件和拖放方法。下面我们先介绍控件的拖放支持,而后再给出开发拖放操作的一般步骤和应用实例。 9.1 控件的拖放支持 拖放操作中控件可以分为源控件和目标控件两类。绝大部分控件既可以作为源控件也可以作为目标控件。但也有一部分控件只能支持其中的一种。 9.1.1拖放属性 拖放属性主要有两个: ●DragMode:拖动模式 它们都是在拖放的源控件中设置。DragMode控制用户在运行时间内当在控件上按下鼠标时控件如何反应。如果DragMode置为dmAutomatic,那么当用户在控件上按下鼠标时拖动自动开始;如果DragMode置为dmManual(这是缺省值),则将通过处理鼠标事件来判断一个拖动是否可以开始。 DragCursor用于选择拖动时显示的光标,缺省值是CrDrag,一般不要去修改它。在程序设计过程中通用的界面规范应该得到开发者的尊重。但有时候为了特定的目的,开发者也可以把自己设计的光标赋给DragCursor。 9.1.2拖放事件 拖放事件主要有三个: ●OnDragOver:拖动经过时激发 ●OnDragDrop:拖动放下时激发 ●OnEndDrop:拖动结束时激发 前两个事件由目标控件响应,后一个事件由源控件响应。 OnDragOver事件最主要的功能是确定当用户就地放下拖动时控件是否可以接受。它的参数包括: Source:TObject; {源控件} X,Y:Integer; {光标位置} State:TDragState;{拖动状态} var Accept:Boolean {能否接受} TDragState是一个枚举类型,表示拖放项目与目标控件的关系。 type TDragState = (dsDragEnter, dsDragLeave, dsDragMove); 不同取值的意义如下表: 表9.1 DragState的取值与意义 ━━━━━━━━━━━━━━━━━━━━━━━━━━━ 取值意义 ─────────────────────────── dsDragEnter拖动对象进入一个允许拖动对象放的控件中。为缺省状态。 dsDragLeave拖动对象离开一个允许拖动对象放下的控件。 dsDragMove拖动对象在一个允许拖动对象放下的控件内移动。 ━━━━━━━━━━━━━━━━━━━━━━━━━━━ 用户可以利用提供的参数来确定放下的拖动是否可被接受,如: ●判断源控件类型: Accept := Source is TLabel; ●判断源控件对象: Accept := (Source = TabSet1); ●判断光标位置: 见(9.2),(9.3)中的例程。 ●判断拖动状态: If (Source is TLabel) and (State = dsDragMove) then begin source.DragIcon := ' New.Ico '; Accept := True; end else Accept := False; 当Accept=True时,目标控件可以响应OnDragDrop事件,用于确定拖动被放下后程序如何进行处理。 OnDragDrop事件处理过程的参数包括源控件和光标位置。这些信息可用于处理方式的确定。 OnEndDrag事件是在拖动操作结束后由源控件来进行响应的,用于源控件进行相应的处理。拖动操作结束既包括拖动放下被接受,也包括用户在一个不能接受放下的控件上释放了鼠标。该事件处理过程的参数包括目标控件(Target)和放下位置的坐标。如果Target=nil, 表示拖动项目没有被任何控件接受。 在第3节将介绍的文件拖放移动、拖放拷贝操作中,如果操作成功,则文件列表框应更新显示内容。下面这段程序用于实现这一功能。procedure TFMForm.FileListEndDrag(Sender, Target: TObject; X, Y: Integer); begin if Target <> nil then FileList.Update; end; 除以上介绍的三个事件外,还有一个事件OnMouseDown也常用于拖放操作的响应。 OnMouseDown虽然不是一个专门的拖放事件,但在人工模式下拖动的开始是在这一事件的处理过程中实现的。 9.1.3拖放方法 拖放方法有三个: ●BeginDrag:人工方式下开始一个拖动 ●EndDrag: 结束一个拖动 ●Dragging:判断一个控件是否正被拖动 这三个方法都被源控件使用。 当DragMode置为dmManual时,拖动必须调用控件的BeginDrag方法才能开始。 BeginDrag有一个布尔参数Immediate。如果输入参数为True,拖动立即开始,光标改变到DragCursor的设置。如果输入参数为False,直到用户将光标移动了一定的距离(5个象素点)后才改变光标,开始拖动。这就允许控件接受一个OnClick事件而并不开始拖动操作。 EndDrag方法中止一个对象的被拖动状态。它有一个布尔参数Drop。如果Drop设置为True,被拖动的对象在当前位置放下(能否被接受由目标控件决定);如果Drop设置为False,则拖动就地被取消。 下面一段程序表明当拖动进入一控制面板时拖动被取消。procedure TForm1.Panel1DragOver(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean); begin Accept := False; if (Source is TLabel) and (State = dsDragEnter) then (Source as TLabel).EndDrag(False); end; Draging方法判断一个控件是否正被拖动。在下面的例子中当用户拖动不同的检查框时窗口改变为不同的颜色。procedure TForm1.FormActivate(Sender: TObject); begin CheckBox1.DragMode := dmAutomatic; CheckBox2.DragMode := dmAutomatic; CheckBox3.DragMode := dmAutomatic; end; procedure TForm1.FormDragOver(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean); begin if CheckBox1.Dragging then Color := clAqua; if CheckBox2.Dragging then Color := clYellow; if CheckBox3.Dragging then Color := clLime; end; type TForm1 = class(TForm) procedure MyMsg(var msg: TWMNCHitTest); message WM_NCHITTEST; end;var Form1: TForm1;implementation{$R *.dfm}procedure TForm1.MyMsg(var msg: TWMNCHitTest);begin Inherited; if msg.Result = HTCLIENT then msg.Result := HTCAPTION;end;end. 第一个问题已经ok了procedure TFormIcon.Image1MouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer);begin if Shift = [ssLeft] then begin releasecapture; perform(WM_SYSCOMMAND,$f012,0); end;end;第二个问题,2.如果将其它的内容(比如IE的文字)拖放到我自己的Form上,并进行相应处理,如何处理 谁说不行?procedure TForm1.FormMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);beginReleaseCapture;SendMessage(Handle, WM_NCLBUTTONDOWN, HTCAPTION, 0);end;绝对可以。不要放在MouseMove事件里面。放在按下事件里面! ReleaseCapture; //拖动窗体的代码 SendMessage(Handle,WM_SysCommand,$F012,0); ReleaseCapture; //拖动不规则窗体的代码 SendMessage(Handle,WM_SysCommand,$F012,0); 密码修改界面 一个SQL语句中如何实现表中一个字段多次赋值 在Richedit中怎么实际URL,谢谢了。 Delphi怎样做成安装文件 文本编辑器,小问题。。。 记录定义中含备注字段的问题 用rave生成报表时,怎么处理字段的换行? 生日,散分~! project文件(*.dpr)与普通的unit(*.pas)文件在结构上有什么区别? 窗体变量的释放问题 WM_XXXX消息问题,高手进来帮帮忙吧 如何在oracle中取SQ_SPEC_GETLOGNO将其设置成其他表格的ID
SendMessage(Handle, WM_NCLBUTTONDOWN, HTCAPTION, 0);把上述两行代码加到窗体的鼠标按下事件里面。
具,Delphi同样支持拖放操作,而且开发应用系统的拖放功能十分方便,真正体现了
Delphi 的强大功能和方便性。 Delphi提供的所有控件(Control,即能获得输入焦点的部件)都支持拖放操作,并有
相应的拖放属性、拖放事件和拖放方法。下面我们先介绍控件的拖放支持,而后再给出开
发拖放操作的一般步骤和应用实例。 9.1 控件的拖放支持 拖放操作中控件可以分为源控件和目标控件两类。绝大部分控件既可以作为源控件
也可以作为目标控件。但也有一部分控件只能支持其中的一种。 9.1.1拖放属性 拖放属性主要有两个: ●DragMode:拖动模式 它们都是在拖放的源控件中设置。DragMode控制用户在运行时间内当在控件上按
下鼠标时控件如何反应。如果DragMode置为dmAutomatic,那么当用户在控件上按下鼠
标时拖动自动开始;如果DragMode置为dmManual(这是缺省值),则将通过处理鼠标事件
来判断一个拖动是否可以开始。 DragCursor用于选择拖动时显示的光标,缺省值是CrDrag,一般不要去修改它。
在程序设计过程中通用的界面规范应该得到开发者的尊重。但有时候为了特定的目的,
开发者也可以把自己设计的光标赋给DragCursor。 9.1.2拖放事件 拖放事件主要有三个: ●OnDragOver:拖动经过时激发 ●OnDragDrop:拖动放下时激发 ●OnEndDrop:拖动结束时激发 前两个事件由目标控件响应,后一个事件由源控件响应。 OnDragOver事件最主要的功能是确定当用户就地放下拖动时控件是否可以接受。
它的参数包括: Source:TObject; {源控件} X,Y:Integer; {光标位置}
State:TDragState;{拖动状态} var Accept:Boolean {能否接受} TDragState是一个枚举类型,表示拖放项目与目标控件的关系。 type TDragState = (dsDragEnter, dsDragLeave, dsDragMove); 不同取值的意义如下表: 表9.1 DragState的取值与意义 ━━━━━━━━━━━━━━━━━━━━━━━━━━━ 取值意义 ─────────────────────────── dsDragEnter拖动对象进入一个允许拖动对象放的控件中。为缺省状态。 dsDragLeave拖动对象离开一个允许拖动对象放下的控件。 dsDragMove拖动对象在一个允许拖动对象放下的控件内移动。 ━━━━━━━━━━━━━━━━━━━━━━━━━━━ 用户可以利用提供的参数来确定放下的拖动是否可被接受,如: ●判断源控件类型: Accept := Source is TLabel; ●判断源控件对象: Accept := (Source = TabSet1); ●判断光标位置: 见(9.2),(9.3)中的例程。 ●判断拖动状态: If (Source is TLabel) and (State = dsDragMove) then begin source.DragIcon := ' New.Ico '; Accept := True; end else Accept := False; 当Accept=True时,目标控件可以响应OnDragDrop事件,用于确定拖动被放下后程序
如何进行处理。 OnDragDrop事件处理过程的参数包括源控件和光标位置。这些信息可用于处理方式
的确定。 OnEndDrag事件是在拖动操作结束后由源控件来进行响应的,用于源控件进行相应的
处理。拖动操作结束既包括拖动放下被接受,也包括用户在一个不能接受放下的控件上释
放了鼠标。该事件处理过程的参数包括目标控件(Target)和放下位置的坐标。如果
Target=nil, 表示拖动项目没有被任何控件接受。 在第3节将介绍的文件拖放移动、拖放拷贝操作中,如果操作成功,则文件列表框
应更新显示内容。下面这段程序用于实现这一功能。
procedure TFMForm.FileListEndDrag(Sender, Target: TObject; X, Y: Integer); begin if Target <> nil then FileList.Update; end;
除以上介绍的三个事件外,还有一个事件OnMouseDown也常用于拖放操作的响应。 OnMouseDown虽然不是一个专门的拖放事件,但在人工模式下拖动的开始是在这一
事件的处理过程中实现的。 9.1.3拖放方法 拖放方法有三个: ●BeginDrag:人工方式下开始一个拖动 ●EndDrag: 结束一个拖动 ●Dragging:判断一个控件是否正被拖动 这三个方法都被源控件使用。 当DragMode置为dmManual时,拖动必须调用控件的BeginDrag方法才能开始。 BeginDrag有一个布尔参数Immediate。如果输入参数为True,拖动立即开始,光标
改变到DragCursor的设置。如果输入参数为False,直到用户将光标移动了一定的距离
(5个象素点)后才改变光标,开始拖动。这就允许控件接受一个OnClick事件而并不开始
拖动操作。 EndDrag方法中止一个对象的被拖动状态。它有一个布尔参数Drop。如果Drop设置
为True,被拖动的对象在当前位置放下(能否被接受由目标控件决定);如果Drop设置
为False,则拖动就地被取消。 下面一段程序表明当拖动进入一控制面板时拖动被取消。
procedure TForm1.Panel1DragOver(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean); begin Accept := False; if (Source is TLabel) and (State = dsDragEnter) then (Source as TLabel).EndDrag(False); end;
Draging方法判断一个控件是否正被拖动。在下面的例子中当用户拖动不同的检查框
时窗口改变为不同的颜色。
procedure TForm1.FormActivate(Sender: TObject); begin CheckBox1.DragMode := dmAutomatic; CheckBox2.DragMode := dmAutomatic; CheckBox3.DragMode := dmAutomatic; end;
procedure TForm1.FormDragOver(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean); begin if CheckBox1.Dragging then Color := clAqua; if CheckBox2.Dragging then
Color := clYellow; if CheckBox3.Dragging then Color := clLime; end;
type
TForm1 = class(TForm)
procedure MyMsg(var msg: TWMNCHitTest); message WM_NCHITTEST;
end;var
Form1: TForm1;implementation{$R *.dfm}procedure TForm1.MyMsg(var msg: TWMNCHitTest);
begin
Inherited;
if msg.Result = HTCLIENT then
msg.Result := HTCAPTION;
end;end.
procedure TFormIcon.Image1MouseMove(Sender: TObject; Shift: TShiftState; X,
Y: Integer);
begin
if Shift = [ssLeft] then
begin
releasecapture;
perform(WM_SYSCOMMAND,$f012,0);
end;
end;
第二个问题,2.如果将其它的内容(比如IE的文字)拖放到我自己的Form上,并进行相应处理,如何处理
Shift: TShiftState; X, Y: Integer);
begin
ReleaseCapture;
SendMessage(Handle, WM_NCLBUTTONDOWN, HTCAPTION, 0);
end;绝对可以。不要放在MouseMove事件里面。放在按下事件里面!
SendMessage(Handle,WM_SysCommand,$F012,0);
SendMessage(Handle,WM_SysCommand,$F012,0);