半年ほど前に、
「Edge IEモードをVBAから自動操作できた!」という記事をUPしましたが、これを閲覧された方からソースコードを教えてほしいとのご要望がありました。
なので、今回は私が作ったソースコードを公開したいと思います。
と言いつつ、私自身もネットを探し回ってソースコードをかき集めてきたクチなので、コードに統一性が無いのはご容赦ください。
トレパクしたコードを一部流用させて頂いた部分も結構あります。
Option Explicit
Private Declare Function EnumChildWindows Lib "user32" (ByVal hwndParent As Long, ByVal lpEnumFunc As Long, lParam As Long) As Long
Private Declare Function GetClassName Lib "user32" Alias "GetClassNameA" (ByVal hwnd As Long, ByVal lpClassName As String, ByVal nMaxCount As Long) As Long
Private Declare Function GetNextWindow Lib "user32" Alias "GetWindow" (ByVal hwnd As Long, ByVal wFlag As Long) As Long
Private Declare Function GetParent Lib "user32" (ByVal hwnd As Long) As Long
Private Declare Function GetTopWindow Lib "user32" (ByVal hwnd As Long) As Long
Private Declare Function GetWindowThreadProcessId Lib "user32" (ByVal hwnd As Long, lpdwProcessId As Long) As Long
Private Declare Function IIDFromString Lib "ole32" (lpsz As Any, lpiid As Any) As Long
Private Declare Function ObjectFromLresult Lib "oleacc" (ByVal lResult As Long, riid As Any, ByVal wParam As Long, ppvObject As Object) As Long
Private Declare Function RegisterWindowMessage Lib "user32" Alias "RegisterWindowMessageA" (ByVal lpString As String) As Long
Private Declare Function SendMessageTimeout Lib "user32" Alias "SendMessageTimeoutA" (ByVal hwnd As Long, ByVal msg As Long, ByVal wParam As Long, ByVal lParam As Long, ByVal fuFlags As Long, ByVal uTimeout As Long, lpdwResult As Long) As Long
Private Declare Function GetWindowText Lib "user32" Alias "GetWindowTextA" (ByVal hwnd As Long, ByVal lpString As String, ByVal cch As Long) As Long
Private Declare Function GetWindowTextLength Lib "user32" Alias "GetWindowTextLengthA" (ByVal hwnd As Long) As Long
Private Declare Function EnumWindows Lib "user32" (ByVal lpEnumFunc As Long, ByVal lParam As Long) As Long
Private Declare Function SetForegroundWindow Lib "user32" (ByVal hwnd As Long) As Long
Private Declare Function IsWindowVisible Lib "user32" (ByVal hwnd As Long) As Long
Private Const GW_HWNDNEXT = &H2
Dim m_WinNum As Integer 'ウィンドウハンドル要素数
Dim hWnds() As Long 'ウィンドウハンドル配列
Private hIES As Long
'Edgeオブジェクト取得
Public Function GetEdgeObject(ByVal title As String)
Dim hwnd As Long
hIES = 0 '初期化
hwnd = displayForeground(title) '指定したタイトルのウィンドウハンドルを取得
'IEモードのウィンドウハンドルを探索(Edgeの子ハンドルとして存在しているため)
Do
If GetParent(hwnd) = 0 Then
'Edgeの子ウィンドウ列挙
EnumChildWindows hwnd, AddressOf EnumChildProcIES, 0
If hIES <> 0 Then Exit Do
End If
hwnd = GetNextWindow(hwnd, GW_HWNDNEXT)
Loop While hwnd <> 0
If hIES = 0 Then
Set GetEdgeObject = Nothing
Exit Function
End If
Set GetEdgeObject = GetHTMLDocumentFromIES(hIES)
End Function
'Internet Explorer_Serverクラス(= IEモード)のウィンドウハンドルを取得
Private Function EnumChildProcIES(ByVal hwnd As Long, ByVal lParam As Long) As Long
Dim buf As String * 255
Dim ClassName As String
GetClassName hwnd, buf, Len(buf)
ClassName = Left(buf, InStr(buf, vbNullChar) - 1)
If ClassName = "Internet Explorer_Server" Then
hIES = hwnd
EnumChildProcIES = False
Exit Function
End If
EnumChildProcIES = True
End Function
'Internet Explorer_Serverクラスのウィンドウハンドルから、HTMLDocumentオブジェクトに変換する
Private Function GetHTMLDocumentFromIES(ByVal hwnd As Long) As Object
Dim msg As Long, res As Long
Dim iid(0 To 3) As Long
Dim ret As Object, obj As Object
Const SMTO_ABORTIFHUNG = &H2
Const IID_IHTMLDocument2 = "{332C4425-26CB-11D0-B483-00C04FD90119}"
Set ret = Nothing '初期化
msg = RegisterWindowMessage("WM_HTML_GETOBJECT")
SendMessageTimeout hwnd, msg, 0, 0, SMTO_ABORTIFHUNG, 1000, res
If res Then
IIDFromString StrPtr(IID_IHTMLDocument2), iid(0)
If ObjectFromLresult(res, iid(0), 0, obj) = 0 Then Set ret = obj
End If
Set GetHTMLDocumentFromIES = ret
End Function
'可視の全ウィンドウハンドルを取得
Function EnumWinProc(ByVal hWndX As Long, ByVal lParam As Long) As Boolean
If IsWindowVisible(hWndX) Then
m_WinNum = m_WinNum + 1
ReDim Preserve hWnds(0 To m_WinNum)
hWnds(m_WinNum) = hWndX
End If
EnumWinProc = True
End Function
'ウィンドウ名取得
Function GetWinText(hWndTarget As Long) As String
Dim l As Long
Dim s As String
Dim dum As Boolean
l = GetWindowTextLength(hWndTarget) + 1
s = String(l, 0)
dum = GetWindowText(hWndTarget, s, l)
GetWinText = s
End Function
'指定画面のウィンドウハンドル取得&最前面に表示
Function displayForeground(ByVal title As String) As Long
Dim dumb As Boolean
Dim i As Integer
Dim windowName As String
ReDim hWnds(0)
m_WinNum = -1
Call EnumWindows(AddressOf EnumWinProc, 0&)
For i = 0 To UBound(hWnds)
windowName = GetWinText(hWnds(i))
If InStr(windowName, title) > 0 Then
'指定画面有り
SetForegroundWindow hWnds(i) '最前面に表示
displayForeground = hWnds(i)
Exit Function
End If
Next i
displayForeground = 0 '指定画面無し
End Function
以上のコードを標準モジュールあたりにそのまま貼り付けてください。
あとはメイン処理側で、
Dim objIE As Object
Set objIE = GetEdgeObject("hogehoge")
っていうコードを書くだけで、objIEにはHTMLDocumentオブジェクトが入ってきますので、その後は今まで通りのDOM操作が可能となります。
「hogehoge」の部分は、操作対象ウィンドウのタイトルを指定して下さい。
例えばこの記事のウィンドウを操作したい場合は、「36歳中小ベンダーSE」とか「Edge IEモードをVBAから操作する」などと指定します。
(タイトルに指定した文言を含むウィンドウのオブジェクトを取得します)
注意点は、以前の記事でも触れたように、
objIE.document.getElementById("hoge").Focus
というふうに書いていたところを、
objIE.getElementById("hoge").Focus
というように書き換える必要があること。
(objIEの時点でdocumentまで含んでいるため)
さらに、画面表示完了を待機する際に使っていた
objIE.Busy
が使えなくなったこと。
objIE.ReadyStateは引き続き使えるので、ReadyStateが"complete"かどうかという点のみで、画面表示完了を待機することになります。
(= 残念ながら、若干安定性が低下)
ここまでが、Edge IEモードを操作する際の基本的なコードになります。
あとは対象システムの特性に応じて、色々と肉付けしていって下さい。
以上、ご参考までに。
それにしても、今回は超久々にエンジニアっぽい内容を書いたな。(笑)
還元率の高いポイントサイトで、ハイペースでポイントが貯まります