2024-11-21 16:20:54 +08:00
|
|
|
|
#ifndef WINXP
|
2024-06-05 02:27:47 +08:00
|
|
|
|
|
|
|
|
|
#include <wrl.h>
|
|
|
|
|
#include <wil/com.h>
|
|
|
|
|
#include <wil/result.h>
|
|
|
|
|
#include <wil/com.h>
|
|
|
|
|
#include <wrl/implements.h>
|
|
|
|
|
using namespace Microsoft::WRL;
|
|
|
|
|
#include <WebView2.h>
|
2025-01-06 17:01:27 +08:00
|
|
|
|
#else
|
|
|
|
|
typedef int COREWEBVIEW2_PREFERRED_COLOR_SCHEME;
|
|
|
|
|
typedef int EventRegistrationToken;
|
2025-01-07 20:18:35 +08:00
|
|
|
|
typedef int ICoreWebView2Controller;
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#endif
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void set_transparent_background(ICoreWebView2Controller *m_host)
|
2024-11-12 02:57:11 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2024-06-18 17:44:02 +08:00
|
|
|
|
COREWEBVIEW2_COLOR color;
|
2024-11-12 02:57:11 +08:00
|
|
|
|
ZeroMemory(&color, sizeof(color));
|
2025-01-07 20:18:35 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
2024-06-18 17:44:02 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller2> coreWebView2 =
|
2024-11-12 02:57:11 +08:00
|
|
|
|
m_controller.try_query<ICoreWebView2Controller2>();
|
|
|
|
|
if (coreWebView2)
|
|
|
|
|
{
|
2024-06-18 17:44:02 +08:00
|
|
|
|
coreWebView2->put_DefaultBackgroundColor(color);
|
|
|
|
|
}
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#endif
|
2024-06-18 17:44:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void put_PreferredColorScheme(ICoreWebView2Controller *m_host, COREWEBVIEW2_PREFERRED_COLOR_SCHEME scheme)
|
2024-06-05 02:27:47 +08:00
|
|
|
|
{
|
2025-01-06 17:01:27 +08:00
|
|
|
|
#ifndef WINXP
|
2025-01-07 20:18:35 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
2024-06-17 17:01:01 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2> coreWebView2;
|
2024-11-12 02:57:11 +08:00
|
|
|
|
[&]()
|
2024-06-17 17:01:01 +08:00
|
|
|
|
{
|
2024-11-12 02:57:11 +08:00
|
|
|
|
CHECK_FAILURE(m_controller->get_CoreWebView2(&coreWebView2));
|
|
|
|
|
auto webView2_13 = coreWebView2.try_query<ICoreWebView2_13>();
|
|
|
|
|
if (webView2_13)
|
|
|
|
|
{
|
|
|
|
|
wil::com_ptr<ICoreWebView2Profile> profile;
|
|
|
|
|
CHECK_FAILURE(webView2_13->get_Profile(&profile));
|
|
|
|
|
CHECK_FAILURE(profile->put_PreferredColorScheme(scheme));
|
|
|
|
|
}
|
|
|
|
|
return S_OK;
|
|
|
|
|
}();
|
2024-11-15 03:34:15 +08:00
|
|
|
|
#endif
|
2025-01-06 17:01:27 +08:00
|
|
|
|
}
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void *add_ZoomFactorChanged(ICoreWebView2Controller *m_host, void (*signal)(double))
|
2024-06-17 17:01:01 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2024-06-05 02:27:47 +08:00
|
|
|
|
EventRegistrationToken *m_zoomFactorChangedToken = new EventRegistrationToken;
|
|
|
|
|
// Register a handler for the ZoomFactorChanged event.
|
|
|
|
|
// This handler just announces the new level of zoom on the window's title bar.
|
2025-01-07 20:18:35 +08:00
|
|
|
|
m_host->add_ZoomFactorChanged(
|
2024-06-05 02:27:47 +08:00
|
|
|
|
Callback<ICoreWebView2ZoomFactorChangedEventHandler>(
|
|
|
|
|
[signal](ICoreWebView2Controller *sender, IUnknown *args) -> HRESULT
|
|
|
|
|
{
|
|
|
|
|
double zoomFactor;
|
|
|
|
|
sender->get_ZoomFactor(&zoomFactor);
|
|
|
|
|
signal(zoomFactor);
|
|
|
|
|
// std::wstring message = L"WebView2APISample (Zoom: " +
|
|
|
|
|
// std::to_wstring(int(zoomFactor * 100)) + L"%)";
|
|
|
|
|
// SetWindowText(m_appWindow->GetMainWindow(), message.c_str());
|
|
|
|
|
return S_OK;
|
|
|
|
|
})
|
|
|
|
|
.Get(),
|
|
|
|
|
m_zoomFactorChangedToken);
|
|
|
|
|
return m_zoomFactorChangedToken;
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#else
|
|
|
|
|
return NULL;
|
|
|
|
|
#endif
|
2024-06-05 02:27:47 +08:00
|
|
|
|
}
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void remove_ZoomFactorChanged(ICoreWebView2Controller *m_host, EventRegistrationToken *token)
|
2024-06-05 02:27:47 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2025-01-07 20:18:35 +08:00
|
|
|
|
m_host->remove_ZoomFactorChanged(*token);
|
2024-11-12 02:57:11 +08:00
|
|
|
|
delete token;
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#endif
|
2024-06-05 02:27:47 +08:00
|
|
|
|
}
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API double get_ZoomFactor(ICoreWebView2Controller *m_host)
|
2024-06-05 02:27:47 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2024-06-05 02:27:47 +08:00
|
|
|
|
double zoomFactor;
|
2025-01-07 20:18:35 +08:00
|
|
|
|
m_host->get_ZoomFactor(&zoomFactor);
|
2024-06-05 02:27:47 +08:00
|
|
|
|
return zoomFactor;
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#else
|
|
|
|
|
return 1;
|
|
|
|
|
#endif
|
2024-06-05 02:27:47 +08:00
|
|
|
|
}
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void put_ZoomFactor(ICoreWebView2Controller *m_host, double zoomFactor)
|
2024-06-05 02:27:47 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2025-01-07 20:18:35 +08:00
|
|
|
|
m_host->put_ZoomFactor(zoomFactor);
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#endif
|
2024-11-12 02:57:11 +08:00
|
|
|
|
}
|
|
|
|
|
// https://github.com/MicrosoftEdge/WebView2Feedback/blob/main/specs/WebMessageObjects.md
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void remove_WebMessageReceived(ICoreWebView2Controller *m_host, EventRegistrationToken *token)
|
2024-11-12 02:57:11 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2025-01-07 20:18:35 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
2024-11-12 02:57:11 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2> m_webView;
|
|
|
|
|
[&]()
|
|
|
|
|
{
|
|
|
|
|
CHECK_FAILURE(m_controller->get_CoreWebView2(&m_webView));
|
|
|
|
|
CHECK_FAILURE(m_webView->remove_WebMessageReceived(*token));
|
|
|
|
|
return S_OK;
|
|
|
|
|
}();
|
|
|
|
|
delete token;
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#endif
|
2024-11-12 02:57:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void *add_WebMessageReceived(ICoreWebView2Controller *m_host, void (*callback)(const wchar_t *))
|
2024-11-12 02:57:11 +08:00
|
|
|
|
{
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#ifndef WINXP
|
2025-01-07 20:18:35 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
2024-11-12 02:57:11 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller4> coreWebView4 =
|
|
|
|
|
m_controller.try_query<ICoreWebView2Controller4>();
|
|
|
|
|
if (coreWebView4)
|
|
|
|
|
{
|
|
|
|
|
coreWebView4->put_AllowExternalDrop(true);
|
|
|
|
|
}
|
|
|
|
|
wil::com_ptr<ICoreWebView2> m_webView;
|
|
|
|
|
EventRegistrationToken *m_webMessageReceivedToken = new EventRegistrationToken;
|
|
|
|
|
[&]()
|
|
|
|
|
{
|
|
|
|
|
CHECK_FAILURE(m_controller->get_CoreWebView2(&m_webView));
|
|
|
|
|
CHECK_FAILURE(m_webView->add_WebMessageReceived(
|
|
|
|
|
Callback<ICoreWebView2WebMessageReceivedEventHandler>(
|
|
|
|
|
[=](ICoreWebView2 *sender, ICoreWebView2WebMessageReceivedEventArgs *args) noexcept
|
|
|
|
|
{
|
|
|
|
|
wil::unique_cotaskmem_string message;
|
|
|
|
|
CHECK_FAILURE(args->TryGetWebMessageAsString(&message));
|
|
|
|
|
if (std::wstring(L"FilesDropped") == message.get())
|
|
|
|
|
{
|
|
|
|
|
wil::com_ptr<ICoreWebView2WebMessageReceivedEventArgs2> args2 =
|
|
|
|
|
wil::com_ptr<ICoreWebView2WebMessageReceivedEventArgs>(args)
|
|
|
|
|
.query<ICoreWebView2WebMessageReceivedEventArgs2>();
|
|
|
|
|
if (args2)
|
|
|
|
|
{
|
|
|
|
|
wil::com_ptr<ICoreWebView2ObjectCollectionView>
|
|
|
|
|
objectsCollection;
|
|
|
|
|
CHECK_FAILURE(args2->get_AdditionalObjects(&objectsCollection));
|
|
|
|
|
unsigned int length;
|
|
|
|
|
CHECK_FAILURE(objectsCollection->get_Count(&length));
|
|
|
|
|
std::vector<std::wstring> paths;
|
|
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < length; i++)
|
|
|
|
|
{
|
|
|
|
|
wil::com_ptr<IUnknown> object;
|
|
|
|
|
CHECK_FAILURE(objectsCollection->GetValueAtIndex(i, &object));
|
|
|
|
|
// Note that objects can be null.
|
|
|
|
|
if (object)
|
|
|
|
|
{
|
|
|
|
|
wil::com_ptr<ICoreWebView2File> file =
|
|
|
|
|
object.query<ICoreWebView2File>();
|
|
|
|
|
if (file)
|
|
|
|
|
{
|
|
|
|
|
// Add the file to message to be sent back to webview
|
|
|
|
|
wil::unique_cotaskmem_string path;
|
|
|
|
|
CHECK_FAILURE(file->get_Path(&path));
|
|
|
|
|
paths.push_back(path.get());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// ProcessPaths(paths);
|
|
|
|
|
if (paths.size())
|
|
|
|
|
{
|
|
|
|
|
callback(paths[0].c_str());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return S_OK;
|
|
|
|
|
})
|
|
|
|
|
.Get(),
|
|
|
|
|
m_webMessageReceivedToken));
|
|
|
|
|
return S_OK;
|
|
|
|
|
}();
|
|
|
|
|
return m_webMessageReceivedToken;
|
2024-11-15 03:30:21 +08:00
|
|
|
|
#else
|
|
|
|
|
return NULL;
|
|
|
|
|
#endif
|
2024-11-15 03:21:30 +08:00
|
|
|
|
}
|
2024-11-20 20:58:36 +08:00
|
|
|
|
|
2024-11-20 20:53:01 +08:00
|
|
|
|
struct contextcallbackdatas
|
|
|
|
|
{
|
|
|
|
|
EventRegistrationToken contextMenuRequestedToken;
|
2025-01-06 16:39:48 +08:00
|
|
|
|
std::vector<std::pair<std::wstring, void (*)(const wchar_t *)>> menus;
|
2024-11-20 20:53:01 +08:00
|
|
|
|
};
|
|
|
|
|
// https://learn.microsoft.com/zh-cn/microsoft-edge/webview2/how-to/context-menus?tabs=cpp
|
|
|
|
|
// https://learn.microsoft.com/en-us/microsoft-edge/webview2/reference/win32/icorewebview2_11?view=webview2-1.0.2849.39
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void add_menu_list(contextcallbackdatas *ptr, int index, const wchar_t *label, void (*callback)(const wchar_t *))
|
2025-01-06 16:39:48 +08:00
|
|
|
|
{
|
|
|
|
|
if (!ptr)
|
|
|
|
|
return;
|
2025-01-07 20:18:35 +08:00
|
|
|
|
ptr->menus.insert(ptr->menus.begin() + index, std::make_pair(label, callback));
|
2025-01-06 16:39:48 +08:00
|
|
|
|
}
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void *add_ContextMenuRequested(ICoreWebView2Controller *m_host)
|
2024-11-20 20:53:01 +08:00
|
|
|
|
{
|
|
|
|
|
#ifndef WINXP
|
|
|
|
|
contextcallbackdatas *data = new contextcallbackdatas;
|
|
|
|
|
[=]()
|
|
|
|
|
{
|
2025-01-07 20:18:35 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
2024-11-20 20:53:01 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2> m_webView;
|
|
|
|
|
CHECK_FAILURE(m_controller->get_CoreWebView2(&m_webView));
|
|
|
|
|
auto m_webView2_11 = m_webView.try_query<ICoreWebView2_11>();
|
|
|
|
|
if (!m_webView2_11)
|
|
|
|
|
return S_OK;
|
|
|
|
|
m_webView2_11->add_ContextMenuRequested(
|
|
|
|
|
Callback<ICoreWebView2ContextMenuRequestedEventHandler>(
|
|
|
|
|
[=](
|
|
|
|
|
ICoreWebView2 *sender,
|
|
|
|
|
ICoreWebView2ContextMenuRequestedEventArgs *args)
|
|
|
|
|
{
|
|
|
|
|
wil::com_ptr<ICoreWebView2ContextMenuTarget> target;
|
|
|
|
|
CHECK_FAILURE(args->get_ContextMenuTarget(&target));
|
|
|
|
|
COREWEBVIEW2_CONTEXT_MENU_TARGET_KIND targetKind;
|
|
|
|
|
BOOL hasselection;
|
|
|
|
|
CHECK_FAILURE(target->get_Kind(&targetKind));
|
|
|
|
|
CHECK_FAILURE(target->get_HasSelection(&hasselection));
|
|
|
|
|
if (!(hasselection && (targetKind ==
|
|
|
|
|
COREWEBVIEW2_CONTEXT_MENU_TARGET_KIND_SELECTED_TEXT)))
|
|
|
|
|
return S_OK;
|
|
|
|
|
wil::com_ptr<ICoreWebView2_11> m_webView2_11;
|
|
|
|
|
CHECK_FAILURE(sender->QueryInterface(IID_PPV_ARGS(&m_webView2_11)));
|
|
|
|
|
|
|
|
|
|
wil::com_ptr<ICoreWebView2Environment> webviewEnvironment;
|
|
|
|
|
CHECK_FAILURE(m_webView2_11->get_Environment(&webviewEnvironment));
|
|
|
|
|
auto webviewEnvironment_5 = webviewEnvironment.try_query<ICoreWebView2Environment9>();
|
|
|
|
|
if (!webviewEnvironment_5)
|
|
|
|
|
return S_OK;
|
|
|
|
|
wil::com_ptr<ICoreWebView2ContextMenuItemCollection> items;
|
|
|
|
|
CHECK_FAILURE(args->get_MenuItems(&items));
|
|
|
|
|
UINT32 itemsCount;
|
|
|
|
|
CHECK_FAILURE(items->get_Count(&itemsCount));
|
|
|
|
|
// Adding a custom context menu item for the page that will display the page's URI.
|
2025-01-06 16:39:48 +08:00
|
|
|
|
UINT idx = 0;
|
|
|
|
|
for (auto &&[label, callback] : data->menus)
|
2024-11-29 22:58:22 +08:00
|
|
|
|
{
|
2025-01-06 16:39:48 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2ContextMenuItem> newMenuItem;
|
|
|
|
|
if (label.size())
|
|
|
|
|
{
|
|
|
|
|
CHECK_FAILURE(webviewEnvironment_5->CreateContextMenuItem(
|
|
|
|
|
label.c_str(),
|
|
|
|
|
nullptr,
|
|
|
|
|
COREWEBVIEW2_CONTEXT_MENU_ITEM_KIND_COMMAND, &newMenuItem));
|
|
|
|
|
newMenuItem->add_CustomItemSelected(
|
|
|
|
|
Callback<ICoreWebView2CustomItemSelectedEventHandler>(
|
|
|
|
|
[=, &callback](
|
|
|
|
|
ICoreWebView2ContextMenuItem *sender,
|
|
|
|
|
IUnknown *args)
|
|
|
|
|
{
|
|
|
|
|
wil::unique_cotaskmem_string selecttext;
|
|
|
|
|
CHECK_FAILURE(target->get_SelectionText(&selecttext));
|
|
|
|
|
callback(selecttext.get());
|
|
|
|
|
return S_OK;
|
|
|
|
|
})
|
|
|
|
|
.Get(),
|
|
|
|
|
nullptr);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
CHECK_FAILURE(webviewEnvironment_5->CreateContextMenuItem(
|
|
|
|
|
L"",
|
|
|
|
|
nullptr,
|
|
|
|
|
COREWEBVIEW2_CONTEXT_MENU_ITEM_KIND_SEPARATOR, &newMenuItem));
|
|
|
|
|
}
|
|
|
|
|
CHECK_FAILURE(items->InsertValueAtIndex(idx++, newMenuItem.get()));
|
2024-11-29 22:58:22 +08:00
|
|
|
|
}
|
2025-01-06 16:39:48 +08:00
|
|
|
|
|
2024-11-20 20:53:01 +08:00
|
|
|
|
return S_OK;
|
|
|
|
|
})
|
|
|
|
|
.Get(),
|
|
|
|
|
&data->contextMenuRequestedToken);
|
|
|
|
|
return S_OK;
|
|
|
|
|
}();
|
|
|
|
|
return data;
|
|
|
|
|
#else
|
|
|
|
|
return NULL;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2025-01-07 20:18:35 +08:00
|
|
|
|
DECLARE_API void remove_ContextMenuRequested(ICoreWebView2Controller *m_host, contextcallbackdatas *data)
|
2024-11-20 20:53:01 +08:00
|
|
|
|
{
|
|
|
|
|
#ifndef WINXP
|
2025-01-07 20:18:35 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
2024-11-20 20:53:01 +08:00
|
|
|
|
wil::com_ptr<ICoreWebView2> m_webView;
|
|
|
|
|
[&]()
|
|
|
|
|
{
|
|
|
|
|
CHECK_FAILURE(m_controller->get_CoreWebView2(&m_webView));
|
|
|
|
|
auto m_webView2_11 = m_webView.try_query<ICoreWebView2_11>();
|
|
|
|
|
if (!m_webView2_11)
|
|
|
|
|
return S_OK;
|
2025-01-06 17:01:27 +08:00
|
|
|
|
CHECK_FAILURE(m_webView2_11->remove_ContextMenuRequested(data->contextMenuRequestedToken));
|
2024-11-20 20:53:01 +08:00
|
|
|
|
return S_OK;
|
|
|
|
|
}();
|
2025-01-06 17:01:27 +08:00
|
|
|
|
delete data;
|
2024-11-20 20:53:01 +08:00
|
|
|
|
#endif
|
2025-01-07 20:18:35 +08:00
|
|
|
|
}
|
|
|
|
|
DECLARE_API void get_root_html(ICoreWebView2Controller *m_host, void (*cb)(LPCWSTR))
|
|
|
|
|
{
|
|
|
|
|
#ifndef WINXP
|
|
|
|
|
wil::com_ptr<ICoreWebView2Controller> m_controller(m_host);
|
|
|
|
|
wil::com_ptr<ICoreWebView2> m_webView;
|
2025-01-12 18:27:49 +08:00
|
|
|
|
wil::unique_handle asyncMethodCompleteEvent(CreateEvent(nullptr, false, false, nullptr));
|
2025-01-07 20:18:35 +08:00
|
|
|
|
|
|
|
|
|
CHECK_FAILURE_NORET(m_controller->get_CoreWebView2(&m_webView));
|
2025-01-12 18:27:49 +08:00
|
|
|
|
CHECK_FAILURE_NORET(
|
|
|
|
|
m_webView->ExecuteScript(
|
|
|
|
|
L"document.documentElement.outerHTML",
|
|
|
|
|
Callback<ICoreWebView2ExecuteScriptCompletedHandler>(
|
|
|
|
|
[=, asyncMethodCompleteEventHandle = asyncMethodCompleteEvent.get()](HRESULT errorCode, LPCWSTR resultObjectAsJson)
|
|
|
|
|
{
|
|
|
|
|
SetEvent(asyncMethodCompleteEventHandle);
|
|
|
|
|
cb(resultObjectAsJson);
|
|
|
|
|
return S_OK;
|
|
|
|
|
})
|
|
|
|
|
.Get()));
|
|
|
|
|
DWORD handleIndex = 0;
|
|
|
|
|
CoWaitForMultipleHandles(COWAIT_DISPATCH_WINDOW_MESSAGES | COWAIT_DISPATCH_CALLS | COWAIT_INPUTAVAILABLE,
|
|
|
|
|
INFINITE, 1, asyncMethodCompleteEvent.addressof(), &handleIndex);
|
2025-01-07 20:18:35 +08:00
|
|
|
|
#endif
|
|
|
|
|
}
|