当前位置:主页>仓库管理软件> 列表

webbrowser中如何实现鼠标拖曳后打开被拖的链接? 找东莞速达进销存软件

记账软件版1楼: 我想实现myie的url连接拖放功能(就是在url上按住鼠标一拖就可以在新页面中打开)

有帖:http://www.delphibbs.com/delphibbs/dispq.asp?lid=2754837

2楼: 到盒子上搜索一下,有拖放控件可以使用的 如速达软件大管家12

3楼: 不要控件的。
请大家帮助下啊。

4楼: 拖放(DragDrop)是Windows提供的一种快捷的操作方式。作为基于Windows的开发工具,Delphi同样支持拖放操作,而且开发应用系统的拖放功能十分方便,真正体现了Delphi的强大功能和方便性。
  Delphi提供的所有控件(Control,即能获得输入焦点的部件)都支持拖放操作,并有相应的拖放属性、拖放事件和拖放方法。下面我们先介绍控件的拖放支持,而后再给出开发拖放操作的一般步骤和应用实例。 
9.1 控件的拖放支持 
  拖放操作中控件可以分为源控件和目标控件两类。绝大部分控件既可以作为源控件也可以作为目标控件。但也有一部分控件只能支持其中的一种。 
9.1.1 拖放属性 
  拖放属性主要有两个:
  ● DragMode : 拖动模式
  ● DragCursor : 拖动光标 
  它们都是在拖放的源控件中设置。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; 
9.2 开发拖放功能的一般步骤 
  拖放作为Windows提供的一种方便操作对象的功能,在Delphi中可以很容易地开发出来。根据拖放操作的过程可以把开发步骤划分为四个阶段,即:


  ● 开始拖动操作
  ● 接收拖动项目
  ● 放下拖动项目
  ● 终止拖动操作 
  在介绍过程中我们将结合一个TabSet(标签集)的拖放操作实例。界面设计如图。在运行时当用户把一个标签拖动到另一个标签的位置时,该标签将移动到该位置并引起标签集的重新布置。
9.2.1 开始拖动操作 
  当拖动模式(DragMode)设置为dmAutomatic时,用户在源控件上按下鼠标时拖动自动开始;当设置为dmManual时通过处理鼠标事件来决定拖动是否开始。如果想开始拖动调用BeginDrag方法。
  在TabSet拖放中,我们用下面的MouseDown事件处理过程来开始一个标签的拖动。首先判断按下的是否是左键,而后再判断项目是否合法。 
procedure TForm1.TabSet1MouseDown(Sender: TObject; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer);
var
DragItem: Integer;
begin
if Button = mbLeft then
begin
DragItem := TabSet1.ItemAtPos(Point(X, Y));
if (DragItem > -1) and (DragItem < TabSet1.Tabs.Count) then
TabSet1.BeginDrag(False);
end;
end; 
9.2.2 接收拖动项目 
  一个控件能否接收拖动项目是由该控件的OnDragOver事件决定的。在TabSet拖动中,主要是利用鼠标的位置进行判断。  
procedure TForm1.TabSet1DragOver(Sender, Source: TObject; X, Y: Integer;
State: TDragState; var Accept: Boolean);
var
DropPos: Integer;
begin
if Source = TabSet1 then
begin
DropPos := TabSet1.ItemAtPos(Point(X, Y));
Accept := (DropPos > -1) and (DropPos <> TabSet1.TabIndex) and
(DropPos < TabSet1.Tabs.Count);
end;
else
Accept := False;
end; 
9.2.3 放下拖动项目 
  当OnDragOver事件处理过程返回的Accept为True且项目被放下时,由OnDragDrop事件处理过程来完成拖动放下后的响应。在TabSet拖放实例中是改变标签的位置。 
procedure TForm1.TabSet1DragDrop(Sender, Source: TObject; X, Y: Integer);
var
OldPos: Integer;
NewPos: Integer;
begin
if Source = TabSet1 then
begin
OldPos := TabSet1.TabIndex;
NewPos := TabSet1.ItemAtPos(Point(X, Y));
if (NewPos > -1) and (NewPos <> OldPos) then
TabSet1.Tabs.Move(OldPos, NewPos);
end;
end; 
9.2.4 结束拖动操作 
  结束拖动操作的方式有两种:或者是用户释放了鼠标键或者是程序用EndDrag方法强行中止拖动。结束拖动操作的后果有两种:放下被接受或放下被忽略。
  拖动操作结束后源控件都要收到一条消息响应拖动结束事件OnEndDrag。 
9.3  拖放应用实例:文件管理器的拖放支持 
  在第六章最后开发的文件管理器应用实例,虽然功能上已初具规模,但在操作上与Windows的文件管理器相比还有很大不足。其中最大的缺陷是它不支持文件的拖放移动和拖放拷贝。在这一章结束的时候,我们可以来弥补这一缺陷了。
  文件拖放移动指的是当用户把一个文件拖动到目录树下的某一目录并放下时,文件将自动移动到该目录中;文件拖放拷贝指的是当用户把一个文件拖动到某个驱动器标签上并放下时,文件将自动拷贝到该驱动器的当前目录下。作为源控件的文件列表框和作为目标控件的目录树、驱动器标签可以位于不同的子窗口。驱动器的当前目录是任一子窗口的最新操作结果,而不论这一子窗口与拖动源、拖动目标是否有关系。
  为了实现上述功能,有两个问题必须首先解决:
  1.如何记录每一驱动器的当前目录?
  为此我们定义了一个全局变量: 
  var
CurentDirList: Array[0...25] of string[70]; 
在DirectoryOutline的OnChange事件中: 
procedure TFMForm.DirectoryOutlineChange(Sender: TObject);
begin
CreateCaption;
FileList.clear;
FileList.Directory := DirectoryOutline.Directory;
FileList.Update;
CurrentDirList[DriveTabSet.TabIndex] := DirectoryOutline.Directory;
FileManager.DirectoryPanel.Caption := DirectoryOutline.Directory;
end;  
  由于DriveTabSet在响应OnDragDrop事件前先响应OnClick事件,并由该事件激发DirectoryOutline的Onchange事件,因而可保证在任何时候OnDragDrop事件中用到的CurrentDirList数组项不为空字符串。
 2.如何保证移动、拷贝与子窗口的无关性?
  在这里一个关键问题是我们判断源控件时是用is操作符进行类型检查: 
If Source is TFileList then

  如果我们用下面的语句: 
  If Source = FileList then
   …
  则移动、拷贝操作将限制在本子窗口范围内。
  当解决了上述问题后我们的工作就只是遵循拖放的一般开发步骤,按步就班来完成了。
  1.FileList开始拖动操作 
procedure TFMForm.FileListMouseDown(Sender: TObject; Button: TMouseButton;
Shift: TShiftState; X, Y: Integer);
begin
if Button = mbLeft then
with Sender as TFileListBox do
begin
if ItemAtPos(Point(X, Y), True) >= 0 then
BeginDrag(False);
end;
end;
  ItemAtPos用来检查当前是否有文件存在。而BeginDrag方法传递参数False, 允许FileList单独处理鼠标事件而并不开始拖动。事实上这种情况是大量存在的。 
  2.DirectoryOutline、DriveTabSet决定是否能接受拖动的就地放下。  
procedure TFMForm.DirectoryOutlineDragOver(Sender, Source: TObject; X,
Y: Integer; State: TDragState; var Accept: Boolean);
begin
if Source is TFileListBox then
Accept := True;
end; 
procedure TFMForm.DriveTabSetDragOver(Sender, Source: TObject; X,
Y: Integer; State: TDragState; var Accept: Boolean);
var
PropPos: Integer;
begin
if Source is TFileListBox then
with DriveTabSet do
begin
PropPos := ItemAtPos(Point(X,Y));
Accept := (PropPos > -1) and (PropPos < Tabs.Count);
end;
end;
  DirectoryOutline是无条件的接受,而DriveTabSet需检查是否是合法的标签。 
  3.拖动放下的响应
  DirectoryOutline的拖动放下用于实现文件移动功能。程序中调用ConfirmChange事件处理过程,目标路径由DirctoryOutline.Items[GetItem(X,Y)].FullPath来得到。  
procedure TFMForm.DirectoryOutlineDragDrop(Sender, Source: TObject; X,
Y: Integer);
begin
if Source is TFileListBox then
with DirectoryOutline do
begin
ConfirmChange(''Move'',FileList.FileName, Items[GetItem(X, Y)].FullPath);
end;
end;
  DriveTabSet的拖动放下用于实现文件拷贝功能。程序中把当前位置转化为相应的驱动器号,目标路径由CurrentDirList[DriveTabSet.TabIndex]获得。 
procedure TFMForm.DriveTabSetDragDrop(Sender, Source: TObject; X,Y: Integer);
var
APoint: TPoint;
begin
APoint.X := X; APoint.Y := Y;
DriveTabSet.TabIndex := DriveTabSet.ItemAtPos(APoint);
if Source is TFileListBox then
with DriveTabSet do
begin
if CurrentDirList[TabIndex] <> '''' then
ConfirmChange(''Copy'',TheFilename,CurrentDirList[TabIndex]);
end;
end; 
4.FileList响应拖动结束,更新文件列表 
procedure TFMForm.FileListEndDrag(Sender, Target: TObject; X, Y: Integer);
begin
if Target <> nil then FileList.Update;
end; 

5楼: 自己实现的话要用到IDropTarget接口,去msdn查查吧.

6楼: 还得再U*P一下。


那个TDragDrop也最高支持到D6吧。
大家给个例子啊。

记账软件版7楼: 自己再U*P一下。

8楼: 告诉你的办法你试了吗?

9楼: 你告诉我的办法你试了吗?
发现DELPHIBBS上有许多人都是,别人提问题的时候,把稍微靠一点边的代码去别处复制来,大把大把的帖。从来不管是否得体和提问富翁的正确需求。

我再说一遍,我所需要的是:
“我想实现myie的url连接拖放功能(就是在url上按住鼠标一拖就可以在新页面中打开)”

请能够做到的同志多帮助。

10楼: 当我什么都没说.你是不是要我把实现代码贴出来你才高兴?告诉你相关技术自己去举一反三就不行?

11楼: 实话告诉你,你发的那大段代码根本就不可行。
因为我的需要是:
“我想实现myie的url连接拖放功能(就是在url上按住鼠标一拖就可以在新页面中打开)”
是在WebBrowser中拖,并在自己本身中释放鼠标的。
并不是你帖的那大段,
在这个里面DropDrop,在那个里面DropOver的。况且TWebBrowser中根本就没MouseDown什么的。

12楼: 看看我得分的问题是些什么问题吧.

我还不至于不知道你是要在twebbrowser

一开始提醒你用IDropTarget,你不理,后来贴个文章给你,你也不理.我就没办法了. 如进销存数据库管理

13楼: 是用IDropTarget,satanmonkey说的是对的。

记账软件版14楼: DragDrop在D7下使用也没问题

15楼: 来自:东兰梦舞, 时间:2006-3-15 12:37:04, ID:3382397


是用IDropTarget,satanmonkey说的是对的。

当然我也知道用IDropTarget能实现,但是在CSDN上也看过文章,最终还是没实现。

来自:rabbitlzx, 时间:2006-3-15 12:44:02, ID:3382400
DragDrop在D7下使用也没问题

这个在D7下当然没问题,我也没说有问题,但是到最后没实现的了。因为不是DropOver这个,DragDrop那个的。都是在WebBrowser中操作进行的。

16楼: 你不是看过这个帖子了吗?

http://www.delphibbs.com/delphibbs/dispq.asp?lid=2754837

有人回复
--------------------------------------------------
要在 TWebBrowser 上面增加 IDocHostUIHandler 接口。
然后控制 GetDropTarget。
--------------------------------------------------

另外那贴的帖主也放出了一些代码
你没有启发吗?
这个
TWebB = class(Twebbrowser,IDropTarget)

不要什么都等现成的.我一半回答问题只提供思路.

17楼: 压根就编译不出来。
我在程序中直接放入的WebBrowser,不是动态创建的。

18楼: 不动态创建url一拖就弹出个ie来.你不希望要这个效果吧.还是考虑动态创建吧?
因为你不搞动态创建,在onnewwindow2事件里就不能指定用那个wb打开,默认动作就是打开ie



你非要直接拖到form上也可以,自己继承TWebbrowser以后,注册到面版上就可以拖了.

19楼: 老兄,我的算服了你了。
能联系一下我吗?请教一下老兄。
QQ:3249136 MSN:login_free@163.com

20楼: 这里的事情这里讨论不就可以了.另外你自己的帖子自己都不订阅.别人回复了,你怎么会第一时间知道?

记账软件版21楼: 如果WebBrowser需要创建,才能实现拖曳的话,老兄可否给相关例子看看?
还有,请问一下,如果做多页面浏览器,在整体架构上,应该注意哪些?需要做哪些准备等等。比如,TWebBrowser需要自己创建等等,现在用的是将WebBrowser拉在一个框架中了,在创新新页面的时候,标签加上框架,就是这样的,请给一些意见和见议。谢谢。

22楼: 做这个东西建议你去下载个myie的代码看看.

或者去http://www.tomore.com/上搜代码

比如
http://www.tomore.com/1/26502.html 如东莞速达进销存软件

23楼: myie的代码以前下过,但是是VC++的,看不懂。

24楼: http://www.tomore.com/上有好几个多tab的浏览器的.自己去搜 ,关键字浏览器

25楼: 上面的早就看过了。
对了,看到你一前的一个帖子。里面说:

来自:satanmonkey, 时间:2004-11-17 21:02:26, ID:2896236
换EmbededWB,这个是Twebbrowser的一个bug,很多人碰到。
EmbededWB和Twebbrowser完全兼容。
http://www.euromind.com/iedelphi/

我将程序中的TWebBrowser替换成EmbededWB了,那这样用这个控件,能实现拖曳吗?

26楼: 晕,你那么害怕自己继承组件啊?

EmbededWB是Twebbrowser的加强版,拖曳还是要自己实现IDropTarget

27楼: 刚才好像发错了,应该是看到你的这个:

来自:satanmonkey, 时间:2004-11-19 9:05:55, ID:2898149
你只要拖个EmbedebWB上去,把名字换成原来的Twebbrowser的名字就可以了。
2个完全兼容的。不用改一行代码。如果是动态创建的,只要改改申明的类型就可以了。

看过这个之后才改的。
我啊,现在做的浏览器,界面大家一致认为不大好,可是我从来不愿意用外部的控件。比如外部的界面控件等。
用DELPHI默认的TWebBrowser就是如此,但看到你上面所说的完全兼容,换一下就行了,所以我就换了试试,果真可以。
但不知道这个比DELPHI默认的TWebBrowser控件优秀在哪里?也没见得有什么过人之处。无非是增加了,是否允许Script、ActiveX等等。。。

记账软件版28楼: 好处是多些事件.多些属性,改了twb的一些bug

这些东西本来是要你写很多代码才有的,他帮你封装好了.他实质也是TWebbrowser,他就是从Twebbrowser继承的,所以完全兼容.

29楼: 原来如此,谢谢解释。
但我以前用TWebBrowser已经好久了,比如像一直以来的不能回车了,与菜单中的(&F)冲突之类的,不能响应菜单中快捷键之类的,我都已经做好了,但不知道再加这个控件还有何用呢?

30楼: 你需要使用他多出来的属性和方法的话就用,不需要的话就不用.

31楼: OK,虽然那些都实现了,但是看来用用也无妨。
老兄知道如何让自己写的程序的菜单栏变颜色吗?比如变成红色,白色什么的。
前两天有朋友告诉我,用SetSysColors,目的是达到了,但是系统中所有的其他程序的菜单栏也变色了,这样肯定不行,他又说得画什么的。但是没解决了。如果能帮助就帮助一下啊。

32楼: http://www.delphibbs.com/delphibbs/dispq.asp?lid=3358718

网页事件,如鼠标点击onmousedown(分析鼠标当前位置网页元素),拖放ondragstart(做差差)等
HTMLDocument:= FWebBrowser.Document as IHTMLDocument2;
InterfaceConnect(HTMLDocument, HTMLDocumentEvents2, Self, FConnection);

HTMLDocumentEvents2 = dispinterface
[''{3050F613-98B5-11CF-BB82-00AA00BDCE0B}'']


function onhelp(const pEvtObj: IHTMLEventObj): WordBool; dispid -2147418102;
function onclick(const pEvtObj: IHTMLEventObj): WordBool; dispid -600;
function ondblclick(const pEvtObj: IHTMLEventObj): WordBool; dispid -601;
procedure onkeydown(const pEvtObj: IHTMLEventObj); dispid -602;
procedure onkeyup(const pEvtObj: IHTMLEventObj); dispid -604;
function onkeypress(const pEvtObj: IHTMLEventObj): WordBool; dispid -603;
procedure onmousedown(const pEvtObj: IHTMLEventObj); dispid -605;
procedure onmousemove(const pEvtObj: IHTMLEventObj); dispid -606;
procedure onmouseup(const pEvtObj: IHTMLEventObj); dispid -607;
procedure onmouseout(const pEvtObj: IHTMLEventObj); dispid -2147418103;
procedure onmouseover(const pEvtObj: IHTMLEventObj); dispid -2147418104;
procedure onreadystatechange(const pEvtObj: IHTMLEventObj); dispid -609;
function onbeforeupdate(const pEvtObj: IHTMLEventObj): WordBool; dispid -2147418108;
procedure onafterupdate(const pEvtObj: IHTMLEventObj); dispid -2147418107;
function onrowexit(const pEvtObj: IHTMLEventObj): WordBool; dispid -2147418106;
procedure onrowenter(const pEvtObj: IHTMLEventObj); dispid -2147418105;
function ondragstart(const pEvtObj: IHTMLEventObj): WordBool; dispid -2147418101;
function onselectstart(const pEvtObj: IHTMLEventObj): WordBool; dispid -2147418100;
function onerrorupdate(const pEvtObj: IHTMLEventObj): WordBool; dispid -2147418099;
function oncontextmenu(const pEvtObj: IHTMLEventObj): WordBool; dispid 1023;
function onstop(const pEvtObj: IHTMLEventObj): WordBool; dispid 1026;
procedure onrowsdelete(const pEvtObj: IHTMLEventObj); dispid -2147418080;
procedure onrowsinserted(const pEvtObj: IHTMLEventObj); dispid -2147418079;
procedure oncellchange(const pEvtObj: IHTMLEventObj); dispid -2147418078;
procedure onpropertychange(const pEvtObj: IHTMLEventObj); dispid -2147418093;
procedure ondatasetchanged(const pEvtObj: IHTMLEventObj); dispid -2147418098;
procedure ondataavailable(const pEvtObj: IHTMLEventObj); dispid -2147418097;
procedure ondatasetcomplete(const pEvtObj: IHTMLEventObj); dispid -2147418096;
procedure onbeforeeditfocus(const pEvtObj: IHTMLEventObj); dispid 1027;
end;

33楼: 汗,弄这么多声明啊?不懂。。

34楼: 结帖,谢谢satanmonkey,给92分。