Thursday, March 12, 2020
Get the Url of a Hyperlink in a TWebBrowser Document
Get the Url of a Hyperlink in a TWebBrowser Document The TWebBrowser Delphi component provides access to the Web browser functionality from your Delphi applications. In most situations you use the TWebBrowser to display HTML documents to the user - thus creating your own version of the (Internet Explorer) Web browser. Note that the TWebBrowser can also display Word documents, for example. A very nice feature of a Browser is to display link information, for example, in the status bar, when the mouse hovers over a link in a document. The TWebBrowser does not expose an event like OnMouseMove. Even if such an event would exist it would be fired for the TWebBrowser component - NOT for the document being displayed inside the TWebBrowser. In order to provide such information (and much more, as you will see in a moment) in your Delphi application using the TWebBrowser component, a technique called events sinking must be implemeted. WebBrowser Event Sink To navigate to a web page using the TWebBrowser component you call the Navigate method. The Document property of the TWebBrowser returns an IHTMLDocument2 value (for web documents). This interface is used to retrieve information about a document, to examine and modify the HTML elements and text within the document, and to process related events. To get the href attribute (link) of an a tag inside a document, while the mouse hovers over a document, you need to react on the onmousemove event of the IHTMLDocument2. Here are the steps to sink events for the currently loaded document: Sink the WebBrowser controls events in the DocumentComplete event raised by the TWebBrowser. This event is fired when the document is fully loaded into the Web Browser.Inside DocumentComplete, retrieve the WebBrowsers document object and sink the HtmlDocumentEvents interface.Handle the event you are interested in.Clear the sink in the in BeforeNavigate2 - that is when the new document is loaded in the Web Browser. HTML Document OnMouseMove Since we are interested in the HREF attribute of an A element - in order to show the URL of a link the mouse is over, we will sink the onmousemove event. The procedure to get the tag (and its attributes) below the mouse can be defined as: var à à htmlDoc : IHTMLDocument2; ... procedure TForm1.Document_OnMouseOver; var à à element : IHTMLElement; begin à à if htmlDoc nil then Exit; à à element : htmlDoc.parentWindow.event.srcElement; à à elementInfo.Clear; à à if LowerCase(element.tagName) a then à à begin à à à à ShowMessage(Link, HREF : element.getAttribute(href,0)]) ; à à end à à else if LowerCase(element.tagName) img then à à begin à à à à ShowMessage(IMAGE, SRC : element.getAttribute(src,0)]) ; à à end à à else à à begin à à à à elementInfo.Lines.Add(Format(TAG : %s,[element.tagName])) ; à à end; end; (*Document_OnMouseOver*) As explained above, we attach to the onmousemove event of a document in the OnDocumentComplete event of a TWebBrowser: procedure TForm1.WebBrowser1DocumentComplete( à à ASender: TObject; à à const pDisp: IDispatch; à à var URL: OleVariant) ; begin à à if Assigned(WebBrowser1.Document) then à à begin à à à à htmlDoc : WebBrowser1.Document as IHTMLDocument2; à à à à htmlDoc.onmouseover : (TEventObject.Create(Document_OnMouseOver) as IDispatch) ; à à end; end; (*WebBrowser1DocumentComplete*) And this is where the problems arise! As you might guess the onmousemove event is *not* a usual event - as are those we are used to work with in Delphi. The onmousemove expects a pointer to a variable of type VARIANT of type VT_DISPATCH that receives the IDispatch interface of an object with a default method that is invoked when the event occurs. In order to attach a Delphi procedure to onmousemove you need to create a wrapper that implements IDispatch and raises your event in its Invoke method. Heres the TEventObject interface: TEventObject class(TInterfacedObject, IDispatch) private à à FOnEvent: TObjectProcedure; protected à à function GetTypeInfoCount(out Count: Integer): HResult; stdcall; à à function GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): HResult; stdcall; à à function GetIDsOfNames(const IID: TGUID; Names: Pointer; NameCount, LocaleID: Integer; DispIDs: Pointer): HResult; stdcall; à à function Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer; Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): HResult; stdcall; public à à constructor Create(const OnEvent: TObjectProcedure) ; à à property OnEvent: TObjectProcedure read FOnEvent write FOnEvent; à à end; Heres how to implement event sinking for a document displayed by the TWebBrowser component - and get the info of a HTML element below the mouse. TWebBrowser Document Event Sinking Example Download Drop a TWebBrowser (WebBrowser1) on a Form (Form1). Add a TMemo (elementInfo)... unit Unit1;interfaceusesà à Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,à à Dialogs, OleCtrls, SHDocVw, MSHTML, ActiveX, StdCtrls;typeà à TObjectProcedure procedure of object;à à TEventObject class(TInterfacedObject, IDispatch)à à privateà à à à FOnEvent: TObjectProcedure;à à protectedà à à à function GetTypeInfoCount(out Count: Integer): HResult; stdcall;à à à à function GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): HResult; stdcall;à à à à function GetIDsOfNames(const IID: TGUID; Names: Pointer; NameCount, LocaleID: Integer; DispIDs: Pointer): HResult; stdcall;à à à à function Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer; Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): HResult; stdcall;à à publicà à à à constructor Create(const OnEvent: TObjectProcedure) ;à à à à property OnEvent: TObjectProcedure read FOnEvent writ e FOnEvent;à à end;à à TForm1 class(TForm)à à à à WebBrowser1: TWebBrowser;à à à à elementInfo: TMemo;à à à à procedure WebBrowser1BeforeNavigate2(ASender: TObject; const pDisp: IDispatch; var URL, Flags, TargetFrameName, PostData, Headers: OleVariant; var Cancel: WordBool) ;à à à à procedure WebBrowser1DocumentComplete(ASender: TObject; const pDisp: IDispatch; var URL: OleVariant) ;à à à à procedure FormCreate(Sender: TObject) ;à à privateà à à à procedure Document_OnMouseOver;à à publicà à à à { Public declarations }à à end;varà à Form1: TForm1;à à htmlDoc : IHTMLDocument2;implementation{$R *.dfm}procedure TForm1.Document_OnMouseOver;varà à element : IHTMLElement;beginà à if htmlDoc nil then Exit;à à element : htmlDoc.parentWindow.event.srcElement;à à elementInfo.Clear;à à if LowerCase(element.tagName) a thenà à beginà à à à elementInfo.Lines. Add(LINK info...) ;à à à à elementInfo.Lines.Add(Format(HREF : %s,[element.getAttribute(href,0)])) ;à à endà à else if LowerCase(element.tagName) img thenà à beginà à à à elementInfo.Lines.Add(IMAGE info...) ;à à à à elementInfo.Lines.Add(Format(SRC : %s,[element.getAttribute(src,0)])) ;à à endà à elseà à beginà à à à elementInfo.Lines.Add(Format(TAG : %s,[element.tagName])) ;à à end;end; (*Document_OnMouseOver*)procedure TForm1.FormCreate(Sender: TObject) ;beginà à WebBrowser1.Navigate(http://delphi.about.com) ;à à elementInfo.Clear;à à elementInfo.Lines.Add(Move your mouse over the document...) ;end; (*FormCreate*)procedure TForm1.WebBrowser1BeforeNavigate2(ASender: TObject; const pDisp: IDispatch; var URL, Flags, TargetFrameName, PostData, Headers: OleVariant; var Cancel: WordBool) ;beginà à htmlDoc : nil;end; (*WebBrowser1BeforeNavigate2*)procedure TForm1.WebBrowser1DocumentComplete(ASend er: TObject; const pDisp: IDispatch; var URL: OleVariant) ;beginà à if Assigned(WebBrowser1.Document) thenà à beginà à à à htmlDoc : WebBrowser1.Document as IHTMLDocument2;à à à à htmlDoc.onmouseover : (TEventObject.Create(Document_OnMouseOver) as IDispatch) ;à à end;end; (*WebBrowser1DocumentComplete*){ TEventObject }constructor TEventObject.Create(const OnEvent: TObjectProcedure) ;beginà à inherited Create;à à FOnEvent : OnEvent;end;function TEventObject.GetIDsOfNames(const IID: TGUID; Names: Pointer; NameCount, LocaleID: Integer; DispIDs: Pointer): HResult;beginà à Result : E_NOTIMPL;end;function TEventObject.GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): HResult;beginà à Result : E_NOTIMPL;end;function TEventObject.GetTypeInfoCount(out Count: Integer): HResult;beginà à Result : E_NOTIMPL;end;function TEventObject.Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer; Flags: Word; var Params; VarResult, ExcepIn fo, ArgErr: Pointer): HResult;beginà à if (DispID DISPID_VALUE) thenà à beginà à à à if Assigned(FOnEvent) then FOnEvent;à à à à Result : S_OK;à à endà à else Result : E_NOTIMPL;end;end.
Subscribe to:
Posts (Atom)