From c4c81fa6231bce95c6f104e76643a59716cb1991 Mon Sep 17 00:00:00 2001 From: WebView2 Github Bot Date: Wed, 12 Jun 2024 18:02:15 +0000 Subject: [PATCH 1/2] Updates for Win32, WPF, WinForms, UWP and WinUI3 sample apps from 128.0.2646.0 --- SampleApps/WebView2APISample/AppWindow.cpp | 12 + .../ScenarioFileSystemHandleShare.cpp | 30 +-- .../ScenarioFileTypePolicy.cpp | 116 ++++++++ .../ScenarioFileTypePolicy.h | 28 ++ .../ScenarioScreenCapture.cpp | 248 ++++++++++++++++++ .../WebView2APISample/ScenarioScreenCapture.h | 22 ++ .../WebView2APISample/WebView2APISample.rc | 2 + .../WebView2APISample.vcxproj | 12 +- .../assets/LandingPageForFindDemo.html | 70 +++++ .../assets/SecnarioFileTypePolicy.html | 30 +++ SampleApps/WebView2APISample/resource.h | 4 +- SampleApps/WebView2WpfBrowser/MainWindow.xaml | 9 + .../WebView2WpfBrowser/MainWindow.xaml.cs | 133 +++++++++- 13 files changed, 694 insertions(+), 22 deletions(-) create mode 100644 SampleApps/WebView2APISample/ScenarioFileTypePolicy.cpp create mode 100644 SampleApps/WebView2APISample/ScenarioFileTypePolicy.h create mode 100644 SampleApps/WebView2APISample/assets/LandingPageForFindDemo.html create mode 100644 SampleApps/WebView2APISample/assets/SecnarioFileTypePolicy.html diff --git a/SampleApps/WebView2APISample/AppWindow.cpp b/SampleApps/WebView2APISample/AppWindow.cpp index 283de40..3b062e7 100644 --- a/SampleApps/WebView2APISample/AppWindow.cpp +++ b/SampleApps/WebView2APISample/AppWindow.cpp @@ -45,8 +45,10 @@ #include "ScenarioNotificationReceived.h" #include "ScenarioPermissionManagement.h" #include "ScenarioSaveAs.h" +#include "ScenarioScreenCapture.h" #include "ScenarioSharedBuffer.h" #include "ScenarioSharedWorkerWRR.h" +#include "ScenarioFileTypePolicy.h" #include "ScenarioVirtualHostMappingForPopUpWindow.h" #include "ScenarioVirtualHostMappingForSW.h" #include "ScenarioWebMessage.h" @@ -665,6 +667,16 @@ bool AppWindow::ExecuteWebViewCommands(WPARAM wParam, LPARAM lParam) NewComponent(this); return true; } + case IDM_SCENARIO_SCREEN_CAPTURE: + { + NewComponent(this); + return true; + } + case IDM_SCENARIO_FILE_TYPE_POLICY: + { + NewComponent(this); + return true; + } } return false; } diff --git a/SampleApps/WebView2APISample/ScenarioFileSystemHandleShare.cpp b/SampleApps/WebView2APISample/ScenarioFileSystemHandleShare.cpp index aa53438..16604e9 100644 --- a/SampleApps/WebView2APISample/ScenarioFileSystemHandleShare.cpp +++ b/SampleApps/WebView2APISample/ScenarioFileSystemHandleShare.cpp @@ -16,6 +16,7 @@ static constexpr WCHAR c_samplePath[] = L"ScenarioFileSystemHandleShare.html"; extern wil::unique_bstr GetDomainOfUri(PWSTR uri); +//! [PostWebMessageWithAdditionalObjects] ScenarioFileSystemHandleShare::ScenarioFileSystemHandleShare(AppWindow* appWindow) : m_appWindow(appWindow) { @@ -29,25 +30,23 @@ ScenarioFileSystemHandleShare::ScenarioFileSystemHandleShare(AppWindow* appWindo ICoreWebView2* sender, ICoreWebView2NavigationCompletedEventArgs* args) -> HRESULT { - wil::com_ptr webview24 = - m_webView.try_query(); - CHECK_FEATURE_RETURN_HRESULT(webview24); + wil::com_ptr webview23 = + m_webView.try_query(); + CHECK_FEATURE_RETURN_HRESULT(webview23); wil::com_ptr environment = appWindow->GetWebViewEnvironment(); - wil::com_ptr - environment_experimental14 = - environment.try_query(); - CHECK_FEATURE_RETURN_HRESULT(environment_experimental14); - wil::com_ptr rootHandle; - CHECK_FAILURE(environment_experimental14->CreateWebFileSystemDirectoryHandle( + wil::com_ptr + environment14 = + environment.try_query(); + CHECK_FEATURE_RETURN_HRESULT(environment14); + wil::com_ptr rootHandle; + CHECK_FAILURE(environment14->CreateWebFileSystemDirectoryHandle( L"C:\\", COREWEBVIEW2_FILE_SYSTEM_HANDLE_PERMISSION_READ_ONLY, &rootHandle)); - wil::com_ptr webObjectCollection; + wil::com_ptr webObjectCollection; IUnknown* webObjects[] = {rootHandle.get()}; - CHECK_FAILURE(environment_experimental14->CreateObjectCollection( + CHECK_FAILURE(environment14->CreateObjectCollection( ARRAYSIZE(webObjects), webObjects, &webObjectCollection)); - wil::com_ptr webObjectCollectionView = - webObjectCollection.try_query(); wil::unique_cotaskmem_string source; CHECK_FAILURE(m_webView->get_Source(&source)); @@ -57,9 +56,9 @@ ScenarioFileSystemHandleShare::ScenarioFileSystemHandleShare(AppWindow* appWindo // Check the source to ensure the message is sent to the correct target content. if (std::wstring(expectedDomain) == sourceDomain.get()) { - CHECK_FAILURE(webview24->PostWebMessageAsJsonWithAdditionalObjects( + CHECK_FAILURE(webview23->PostWebMessageAsJsonWithAdditionalObjects( L"{ \"messageType\" : \"RootDirectoryHandle\" }", - webObjectCollectionView.get())); + webObjectCollection.get())); } return S_OK; @@ -67,6 +66,7 @@ ScenarioFileSystemHandleShare::ScenarioFileSystemHandleShare(AppWindow* appWindo .Get(), &m_navigationCompletedToken)); } +//! [PostWebMessageWithAdditionalObjects] ScenarioFileSystemHandleShare::~ScenarioFileSystemHandleShare() { diff --git a/SampleApps/WebView2APISample/ScenarioFileTypePolicy.cpp b/SampleApps/WebView2APISample/ScenarioFileTypePolicy.cpp new file mode 100644 index 0000000..debcb19 --- /dev/null +++ b/SampleApps/WebView2APISample/ScenarioFileTypePolicy.cpp @@ -0,0 +1,116 @@ +// Copyright (C) Microsoft Corporation. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "stdafx.h" + +#include "AppWindow.h" +#include "CheckFailure.h" +#include "ScenarioFileTypePolicy.h" + +using namespace Microsoft::WRL; + +static constexpr WCHAR c_samplePath[] = L"SecnarioFileTypePolicy.html"; + +ScenarioFileTypePolicy::ScenarioFileTypePolicy(AppWindow* appWindow) + : m_appWindow(appWindow), m_webView2(appWindow->GetWebView()) +{ + if (m_webView2) + { + m_webView2Experimental27 = m_webView2.try_query(); + m_webView2_2 = m_webView2.try_query(); + + m_sampleUri = m_appWindow->GetLocalUri(c_samplePath); + CHECK_FAILURE(m_webView2->Navigate(m_sampleUri.c_str())); + SuppressPolicyForExtension(); + + // Turn off this scenario if we navigate away from the demo page. + CHECK_FAILURE(m_webView2_2->add_DOMContentLoaded( + Callback( + [this](ICoreWebView2* sender, ICoreWebView2DOMContentLoadedEventArgs* args) + -> HRESULT + { + wil::unique_cotaskmem_string uri; + sender->get_Source(&uri); + if (uri.get() != m_sampleUri) + m_appWindow->DeleteComponent(this); + return S_OK; + }) + .Get(), + &m_DOMcontentLoadedToken)); + } +} + +//! [SuppressPolicyForExtension] +// This example will register the event with two custom rules. +// 1. Suppressing file type policy, security dialog, and allows saving ".eml" files +// directly. +// 2. When the URI is trusted.- Showing customized warning UI when saving ".iso" +// files. It allows to block the saving directly. +bool ScenarioFileTypePolicy::SuppressPolicyForExtension() +{ + if (!m_webView2Experimental27) + return false; + m_webView2Experimental27->add_SaveFileSecurityCheckStarting( + Callback( + [this]( + ICoreWebView2* sender, + ICoreWebView2ExperimentalSaveFileSecurityCheckStartingEventArgs* args) + -> HRESULT + { + // Get the file extension for file to be saved. + // And convert the extension to lower case for a + // case-insensitive comparasion. + wil::unique_cotaskmem_string extension; + CHECK_FAILURE(args->get_FileExtension(&extension)); + std::wstring extension_lower = extension.get(); + std::transform( + extension_lower.begin(), extension_lower.end(), extension_lower.begin(), + ::towlower); + + // Suppress default policy for ".eml" file. + if (wcscmp(extension_lower.c_str(), L".eml") == 0) + { + CHECK_FAILURE(args->put_SuppressDefaultPolicy(TRUE)); + } + + // Cancel save/download for ".iso" file. + if (wcscmp(extension_lower.c_str(), L".iso") == 0) + { + wil::com_ptr deferral; + CHECK_FAILURE(args->GetDeferral(&deferral)); + + m_appWindow->RunAsync( + [this, args = wil::make_com_ptr(args), deferral]() + { + // With the deferral, the cancel decision and + // message box can be replaced with a customized UI. + CHECK_FAILURE(args->put_CancelSave(TRUE)); + MessageBox( + m_appWindow->GetMainWindow(), L"The saving has been blocked", + L"Info", MB_OK); + CHECK_FAILURE(deferral->Complete()); + }); + } + return S_OK; + }) + .Get(), + &m_saveFileSecurityCheckStartingToken); + + MessageBox( + m_appWindow->GetMainWindow(), + (L"Example rules of Dangerous File Security Policy has been applied in this demo page"), + L"Info", MB_OK); + return true; +} +//! [SuppressPolicyForExtension] + +ScenarioFileTypePolicy::~ScenarioFileTypePolicy() +{ + if (m_webView2Experimental27) + { + CHECK_FAILURE(m_webView2Experimental27->remove_SaveFileSecurityCheckStarting( + m_saveFileSecurityCheckStartingToken)); + } + CHECK_FAILURE(m_webView2_2->remove_DOMContentLoaded(m_DOMcontentLoadedToken)); +} \ No newline at end of file diff --git a/SampleApps/WebView2APISample/ScenarioFileTypePolicy.h b/SampleApps/WebView2APISample/ScenarioFileTypePolicy.h new file mode 100644 index 0000000..4933fa5 --- /dev/null +++ b/SampleApps/WebView2APISample/ScenarioFileTypePolicy.h @@ -0,0 +1,28 @@ +// Copyright (C) Microsoft Corporation. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "stdafx.h" + +#include + +#include "AppWindow.h" +#include "ComponentBase.h" + +class ScenarioFileTypePolicy : public ComponentBase +{ +public: + ScenarioFileTypePolicy(AppWindow* appWindow); + ~ScenarioFileTypePolicy(); + +private: + bool SuppressPolicyForExtension(); + + AppWindow* m_appWindow; + wil::com_ptr m_webView2; + wil::com_ptr m_webView2_2; + wil::com_ptr m_webView2Experimental27; + EventRegistrationToken m_saveFileSecurityCheckStartingToken = {}; + EventRegistrationToken m_DOMcontentLoadedToken = {}; + std::wstring m_sampleUri; +}; diff --git a/SampleApps/WebView2APISample/ScenarioScreenCapture.cpp b/SampleApps/WebView2APISample/ScenarioScreenCapture.cpp index 837590f..bd54afb 100644 --- a/SampleApps/WebView2APISample/ScenarioScreenCapture.cpp +++ b/SampleApps/WebView2APISample/ScenarioScreenCapture.cpp @@ -10,3 +10,251 @@ #include "CheckFailure.h" using namespace Microsoft::WRL; + +static constexpr WCHAR c_samplePath[] = L"ScenarioScreenCapture.html"; + +ScenarioScreenCapture::ScenarioScreenCapture(AppWindow* appWindow) + : m_appWindow(appWindow), m_webView(appWindow->GetWebView()) +{ + m_sampleUri = m_appWindow->GetLocalUri(c_samplePath); + + //! [ScreenCaptureStarting0] + m_webViewExperimental26 = m_webView.try_query(); + if (m_webViewExperimental26) + { + m_webViewExperimental26->add_ScreenCaptureStarting( + Callback( + [this]( + ICoreWebView2* sender, + ICoreWebView2ExperimentalScreenCaptureStartingEventArgs* args) -> HRESULT + { + // Get Frame Info + wil::com_ptr frameInfo; + CHECK_FAILURE(args->get_OriginalSourceFrameInfo(&frameInfo)); + + wil::com_ptr frameInfo2; + CHECK_FAILURE(frameInfo->QueryInterface(IID_PPV_ARGS(&frameInfo2))); + + UINT32 source_frameId; + CHECK_FAILURE(frameInfo2->get_FrameId(&source_frameId)); + + BOOL cancel = m_mainFramePermission == FALSE; + CHECK_FAILURE(args->put_Cancel(cancel)); + return S_OK; + }) + .Get(), + &m_screenCaptureStartingToken); + } + //! [ScreenCaptureStarting0] + + //! [ScreenCaptureStarting1] + m_webView4 = m_webView.try_query(); + if (m_webView4) + { + CHECK_FAILURE(m_webView4->add_FrameCreated( + Callback( + [this]( + ICoreWebView2* sender, ICoreWebView2FrameCreatedEventArgs* args) -> HRESULT + { + wil::com_ptr webviewFrame; + CHECK_FAILURE(args->get_Frame(&webviewFrame)); + + auto frame5 = webviewFrame.try_query(); + + UINT32 frameId; + frame5->get_FrameId(&frameId); + + m_screenCaptureFrameIdPermission[frameId] = TRUE; + + wil::com_ptr webviewFrame2 = + webviewFrame.try_query(); + if (!webviewFrame2) + { + return S_OK; + } + + bool cancel_on_frame = false; + + CHECK_FAILURE(webviewFrame2->add_WebMessageReceived( + Microsoft::WRL::Callback< + ICoreWebView2FrameWebMessageReceivedEventHandler>( + [this, frameId]( + ICoreWebView2Frame* sender, + ICoreWebView2WebMessageReceivedEventArgs* args) -> HRESULT + { + BOOL cancel = false; + + wil::unique_cotaskmem_string messageRaw; + HRESULT hr = args->TryGetWebMessageAsString(&messageRaw); + if (hr == E_INVALIDARG) + { + // Was not a string message. Ignore. + return hr; + } + // Any other problems are fatal. + CHECK_FAILURE(hr); + std::wstring message = messageRaw.get(); + + if (message == L"EnableScreenCapture") + { + cancel = false; + } + else if (message == L"DisableScreenCapture") + { + cancel = true; + } + else + { + // Ignore unrecognized messages, but log for further + // investigation since it suggests a mismatch between the + // web content and the host. + OutputDebugString( + std::wstring( + L"Unexpected message from main page:" + message) + .c_str()); + } + + m_screenCaptureFrameIdPermission[frameId] = (cancel == FALSE); + + return S_OK; + }) + .Get(), + nullptr)); + + m_experimentalFrame6 = + webviewFrame.try_query(); + + m_experimentalFrame6->add_ScreenCaptureStarting( + Callback< + ICoreWebView2ExperimentalFrameScreenCaptureStartingEventHandler>( + [this]( + ICoreWebView2Frame* sender, + ICoreWebView2ExperimentalScreenCaptureStartingEventArgs* args) + -> HRESULT + { + args->put_Handled(TRUE); + + bool cancel = FALSE; + + // Get Frame Info + wil::com_ptr frameInfo; + CHECK_FAILURE(args->get_OriginalSourceFrameInfo(&frameInfo)); + + wil::com_ptr frameInfo2; + CHECK_FAILURE( + frameInfo->QueryInterface(IID_PPV_ARGS(&frameInfo2))); + + // Frame Source + wil::unique_cotaskmem_string frameSource; + CHECK_FAILURE(frameInfo->get_Source(&frameSource)); + + UINT32 source_frameId; + CHECK_FAILURE(frameInfo2->get_FrameId(&source_frameId)); + + cancel = + (m_screenCaptureFrameIdPermission[source_frameId] == FALSE); + + CHECK_FAILURE(args->put_Cancel(cancel)); + return S_OK; + }) + .Get(), + &m_frameScreenCaptureStartingToken); + + return S_OK; + }) + .Get(), + &m_frameCreatedToken)); + } + else + { + FeatureNotAvailable(); + } + //! [ScreenCaptureStarting1] + + // Turn off this scenario if we navigate away from the sample page + CHECK_FAILURE(m_webView->add_ContentLoading( + Callback( + [this](ICoreWebView2* sender, ICoreWebView2ContentLoadingEventArgs* args) -> HRESULT + { + wil::unique_cotaskmem_string uri; + sender->get_Source(&uri); + if (uri.get() != m_sampleUri) + { + m_appWindow->DeleteComponent(this); + } + return S_OK; + }) + .Get(), + &m_contentLoadingToken)); + + CHECK_FAILURE(m_webView->add_WebMessageReceived( + Microsoft::WRL::Callback( + [this](ICoreWebView2* sender, ICoreWebView2WebMessageReceivedEventArgs* args) + -> HRESULT + { + BOOL cancel = FALSE; + wil::unique_cotaskmem_string uri; + CHECK_FAILURE(args->get_Source(&uri)); + + // Always validate that the origin of the message is what you + // expect. + if (uri.get() != m_sampleUri) + { + // Ignore messages from untrusted sources. + return E_INVALIDARG; + } + wil::unique_cotaskmem_string messageRaw; + HRESULT hr = args->TryGetWebMessageAsString(&messageRaw); + if (hr == E_INVALIDARG) + { + // Was not a string message. Ignore. + return hr; + } + // Any other problems are fatal. + CHECK_FAILURE(hr); + std::wstring message = messageRaw.get(); + + if (message == L"EnableScreenCapture") + { + cancel = FALSE; + } + else if (message == L"DisableScreenCapture") + { + cancel = TRUE; + } + else + { + // Ignore unrecognized messages, but log for further + // investigation since it suggests a mismatch between the + // web content and the host. + OutputDebugString( + std::wstring(L"Unexpected message from main page:" + message).c_str()); + } + m_mainFramePermission = (cancel == FALSE); + return S_OK; + }) + .Get(), + &m_webMessageReceivedToken)); + + CHECK_FAILURE(m_webView->Navigate(m_sampleUri.c_str())); +} + +ScenarioScreenCapture::~ScenarioScreenCapture() +{ + m_webView->remove_ContentLoading(m_contentLoadingToken); + m_webView->remove_WebMessageReceived(m_webMessageReceivedToken); + if (m_webViewExperimental26) + { + CHECK_FAILURE(m_webViewExperimental26->remove_ScreenCaptureStarting( + m_screenCaptureStartingToken)); + } + if (m_experimentalFrame6) + { + CHECK_FAILURE(m_experimentalFrame6->remove_ScreenCaptureStarting( + m_frameScreenCaptureStartingToken)); + } + if (m_webView4) + { + CHECK_FAILURE(m_webView4->remove_FrameCreated(m_frameCreatedToken)); + } +} diff --git a/SampleApps/WebView2APISample/ScenarioScreenCapture.h b/SampleApps/WebView2APISample/ScenarioScreenCapture.h index 273b918..529fc49 100644 --- a/SampleApps/WebView2APISample/ScenarioScreenCapture.h +++ b/SampleApps/WebView2APISample/ScenarioScreenCapture.h @@ -9,3 +9,25 @@ #include "AppWindow.h" #include "ComponentBase.h" + +class ScenarioScreenCapture : public ComponentBase +{ +public: + ScenarioScreenCapture(AppWindow* appWindow); + ~ScenarioScreenCapture() override; + +private: + AppWindow* m_appWindow = nullptr; + wil::com_ptr m_webView; + wil::com_ptr m_webView4; + wil::com_ptr m_webViewExperimental26; + wil::com_ptr m_experimentalFrame6; + std::wstring m_sampleUri; + std::map m_screenCaptureFrameIdPermission; + BOOL m_mainFramePermission = TRUE; + EventRegistrationToken m_contentLoadingToken = {}; + EventRegistrationToken m_webMessageReceivedToken = {}; + EventRegistrationToken m_frameCreatedToken = {}; + EventRegistrationToken m_screenCaptureStartingToken = {}; + EventRegistrationToken m_frameScreenCaptureStartingToken = {}; +}; diff --git a/SampleApps/WebView2APISample/WebView2APISample.rc b/SampleApps/WebView2APISample/WebView2APISample.rc index 07d1026..760bc30 100644 --- a/SampleApps/WebView2APISample/WebView2APISample.rc +++ b/SampleApps/WebView2APISample/WebView2APISample.rc @@ -302,6 +302,8 @@ BEGIN MENUITEM "Programmatic Save As", IDM_SCENARIO_SAVE_AS_PROGRAMMATIC END MENUITEM "Accelerator Key Pressed", IDM_SCENARIO_ACCELERATOR_KEY_PRESSED + MENUITEM "Screen Capture", IDM_SCENARIO_SCREEN_CAPTURE + MENUITEM "File Type Policy", IDM_SCENARIO_FILE_TYPE_POLICY END POPUP "&Audio" BEGIN diff --git a/SampleApps/WebView2APISample/WebView2APISample.vcxproj b/SampleApps/WebView2APISample/WebView2APISample.vcxproj index cc97767..e2b71d1 100644 --- a/SampleApps/WebView2APISample/WebView2APISample.vcxproj +++ b/SampleApps/WebView2APISample/WebView2APISample.vcxproj @@ -1,4 +1,4 @@ - + @@ -238,6 +238,7 @@ + @@ -290,6 +291,7 @@ + @@ -347,6 +349,9 @@ $(OutDir)\assets + + $(OutDir)\assets + $(OutDir)\assets @@ -407,6 +412,9 @@ $(OutDir)\assets + + $(OutDir)\assets + $(OutDir)\assets @@ -481,4 +489,4 @@ - + \ No newline at end of file diff --git a/SampleApps/WebView2APISample/assets/LandingPageForFindDemo.html b/SampleApps/WebView2APISample/assets/LandingPageForFindDemo.html new file mode 100644 index 0000000..121420b --- /dev/null +++ b/SampleApps/WebView2APISample/assets/LandingPageForFindDemo.html @@ -0,0 +1,70 @@ + + + + + Demo: Programmatic Find API + + + + +
+

Programmatic Find API Demo

+
+ +

Introduction

+

Welcome to this demo showcasing the capabilities of the Programmatic Find API with WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 WebView2 . We have designed this feature to make it easier for developers to programmatically access and utilize WebView2's finding capabilities.

+ +

Why Programmatic Find API?

+

Programmatic Find API offers a way to embed WebView2's finding functionality directly into your applications. This enhances the overall user experience by allowing for customized behavior and extended functionalities.

+ +

Features

+
    +
  • Programmatic Access
  • +
  • Automated Testing Support
  • +
  • Real-time Results with WebView2
  • +
  • Enhanced Customization
  • +
+ +

Use Cases

+

Some common use cases include text highlighting, real-time search features, and quality assurance testing. This ensures that applications built with WebView2 can offer a more dynamic and interactive experience.

+ +

Technical Aspects

+

The API is designed to be platform-agnostic, allowing developers to implement WebView2-based functionalities across a variety of platforms. It is also built to be backward-compatible, ensuring a smooth transition for applications relying on older versions of WebView2.

+ +

Conclusion

+

We hope this demo helps you understand the advantages of using the Programmatic Find API in your WebView2-based applications. The aim is to make your development process smoother while enhancing the user experience.

+ +

FAQs

+
    +
  1. Is this feature limited to any specific programming languages?
  2. +
  3. How can I customize the search experience in WebView2?
  4. +
  5. What platforms currently support this WebView2 feature?
  6. +
+ +
+

For more information, please visit the official WebView2 documentation. https://learn.microsoft.com/en-us/microsoft-edge/webview2/?form=MA13LH

+
+ + + diff --git a/SampleApps/WebView2APISample/assets/SecnarioFileTypePolicy.html b/SampleApps/WebView2APISample/assets/SecnarioFileTypePolicy.html new file mode 100644 index 0000000..91af992 --- /dev/null +++ b/SampleApps/WebView2APISample/assets/SecnarioFileTypePolicy.html @@ -0,0 +1,30 @@ + + + + ScenarioFileTypePolicy + + + +

File Type Policy API Demo Page

+

Two customized example rules in this demo:

+

1. Smoothly save *.eml file without file extension warning

+

2. Intentionally block save *.iso file

+

Please enter a file extension: +

+
+ + diff --git a/SampleApps/WebView2APISample/resource.h b/SampleApps/WebView2APISample/resource.h index 74ac115..33093c4 100644 --- a/SampleApps/WebView2APISample/resource.h +++ b/SampleApps/WebView2APISample/resource.h @@ -186,6 +186,7 @@ #define IDM_SCENARIO_SAVE_AS_TOGGLE_SILENT 2040 #define IDM_SCENARIO_SAVE_AS_PROGRAMMATIC 2041 #define IDM_SCENARIO_ACCELERATOR_KEY_PRESSED 2042 +#define IDM_SCENARIO_FILE_TYPE_POLICY 2049 #define IDM_CREATION_MODE_WINDOWED 3000 #define IDM_CREATION_MODE_VISUAL_DCOMP 3001 #define IDM_CREATION_MODE_TARGET_DCOMP 3002 @@ -224,6 +225,7 @@ #define IDC_CHECK_USE_OS_REGION 32803 #define ID_CUSTOM_DATA_PARTITION 32804 #define ID_SETTINGS_NON_CLIENT_REGION_SUPPORT_ENABLED 32805 +#define IDM_SCENARIO_SCREEN_CAPTURE 32809 #define IDC_STATIC -1 // Next default values for new objects // @@ -231,7 +233,7 @@ #ifndef APSTUDIO_READONLY_SYMBOLS #define _APS_NO_MFC 1 #define _APS_NEXT_RESOURCE_VALUE 245 -#define _APS_NEXT_COMMAND_VALUE 32806 +#define _APS_NEXT_COMMAND_VALUE 32810 #define _APS_NEXT_CONTROL_VALUE 1015 #define _APS_NEXT_SYMED_VALUE 110 #endif diff --git a/SampleApps/WebView2WpfBrowser/MainWindow.xaml b/SampleApps/WebView2WpfBrowser/MainWindow.xaml index f557cfe..a963646 100644 --- a/SampleApps/WebView2WpfBrowser/MainWindow.xaml +++ b/SampleApps/WebView2WpfBrowser/MainWindow.xaml @@ -97,8 +97,13 @@ found in the LICENSE file. + + + + @@ -165,6 +170,7 @@ found in the LICENSE file. + @@ -225,8 +231,11 @@ found in the LICENSE file. + + + diff --git a/SampleApps/WebView2WpfBrowser/MainWindow.xaml.cs b/SampleApps/WebView2WpfBrowser/MainWindow.xaml.cs index 020fe8b..6427fd2 100644 --- a/SampleApps/WebView2WpfBrowser/MainWindow.xaml.cs +++ b/SampleApps/WebView2WpfBrowser/MainWindow.xaml.cs @@ -119,6 +119,8 @@ public partial class MainWindow : Window public static RoutedCommand ToggleSilentCommand = new RoutedCommand(); public static RoutedCommand ThrottlingControlCommand = new RoutedCommand(); public static RoutedCommand FileExplorerCommand = new RoutedCommand(); + public static RoutedCommand ToggleScreenCaptureEnableCommand = new RoutedCommand(); + public static RoutedCommand FileTypePolicyCommand = new RoutedCommand(); #endregion commands @@ -1304,16 +1306,54 @@ void DeleteProfileExecuted(object target, ExecutedRoutedEventArgs e) void NonClientRegionSupportCmdExecuted(object target, ExecutedRoutedEventArgs e) { + try + { + // + WebViewSettings.IsNonClientRegionSupportEnabled = + !WebViewSettings.IsNonClientRegionSupportEnabled; + // + MessageBox.Show("Non-client region support will be " + + (WebViewSettings.IsNonClientRegionSupportEnabled ? + "enabled " : "disabled ") + "after the next navigation."); + } + catch (NotImplementedException exception) + { + MessageBox.Show(this, "Toggle Non-client region support failed: " + + exception.Message, "Non-client Support"); + } } + void NonClientRegionSupportEnabledCmdExecuted(object target, ExecutedRoutedEventArgs e) { + webView.CoreWebView2.NavigationStarting += WebView_NavigationStartingNonClientRegionSupport; + webView.CoreWebView2.DOMContentLoaded += WebView_DOMContentLoadedNonClientRegionSupport; + + webView.Source = new Uri("https://appassets.example/ScenarioNonClientRegionSupport.html"); } + void WebView_DOMContentLoadedNonClientRegionSupport(object sender, CoreWebView2DOMContentLoadedEventArgs e) { + if (webView.CoreWebView2.Source != "https://appassets.example/ScenarioNonClientRegionSupport.html") + { + webView.CoreWebView2.NavigationStarting -= WebView_NavigationStartingNonClientRegionSupport; + webView.CoreWebView2.DOMContentLoaded -= WebView_DOMContentLoadedNonClientRegionSupport; + } } + void WebView_NavigationStartingNonClientRegionSupport(object sender, CoreWebView2NavigationStartingEventArgs e) { + if (e.Uri == "https://appassets.example/ScenarioNonClientRegionSupport.html" + && !WebViewSettings.IsNonClientRegionSupportEnabled) + { + WebViewSettings.IsNonClientRegionSupportEnabled = true; + } + else if (e.Uri != "https://appassets.example/ScenarioNonClientRegionSupport.html" + && WebViewSettings.IsNonClientRegionSupportEnabled) + { + WebViewSettings.IsNonClientRegionSupportEnabled = false; + } } + void PdfToolbarSaveCmdExecuted(object target, ExecutedRoutedEventArgs e) { // @@ -2022,6 +2062,10 @@ void WebView_CoreWebView2InitializationCompleted(object sender, CoreWebView2Init webView.CoreWebView2.DOMContentLoaded += WebView_PermissionManager_DOMContentLoaded; webView.CoreWebView2.WebMessageReceived += WebView_PermissionManager_WebMessageReceived; +#if USE_WEBVIEW2_EXPERIMENTAL + webView.CoreWebView2.ScreenCaptureStarting += WebView_ScreenCaptureStarting; +#endif + // The CoreWebView2Environment instance is reused when re-assigning CoreWebView2CreationProperties // to the replacement control. We don't need to re-attach the event handlers unless the environment // instance has changed. @@ -3397,13 +3441,12 @@ string GetJSONStringField(string jsonMessage, string fieldName) void FileExplorerExecuted(object target, ExecutedRoutedEventArgs e) { -#if USE_WEBVIEW2_EXPERIMENTAL webView.CoreWebView2.NavigationCompleted += delegate ( object webview2, CoreWebView2NavigationCompletedEventArgs args) { if (args.IsSuccess && webView.CoreWebView2.Source.Equals("https://appassets.example/ScenarioFileSystemHandleShare.html")) { - webView.CoreWebView2.PostWebMessageAsJsonWithAdditionalObjects("{ \"messageType\" : \"RootDirectoryHandle\" }", new List() + webView.CoreWebView2.PostWebMessageAsJson("{ \"messageType\" : \"RootDirectoryHandle\" }", new List() { webView.CoreWebView2.Environment.CreateWebFileSystemDirectoryHandle( "C:\\", @@ -3411,12 +3454,94 @@ void FileExplorerExecuted(object target, ExecutedRoutedEventArgs e) }); } }; - webView.Source = new Uri("https://appassets.example/ScenarioFileSystemHandleShare.html"); -#endif + webView.Source = new Uri("https://appassets.example/ScenarioFileSystemHandleShare.html"); } void ThrottlingControlExecuted(object target, ExecutedRoutedEventArgs e) { } + // +#if USE_WEBVIEW2_EXPERIMENTAL + private bool isScreenCaptureEnabled = true; + + void WebView_ScreenCaptureStarting(object sender, CoreWebView2ScreenCaptureStartingEventArgs args) + { + // Get Frame Info + CoreWebView2FrameInfo frameInfo; + frameInfo = args.OriginalSourceFrameInfo; + + // Frame Source + string frameSource; + frameSource = frameInfo.Source; + + args.Cancel = !isScreenCaptureEnabled; + } +#endif + + void TogglScreenCaptureEnabledCmdExecuted(object target, ExecutedRoutedEventArgs e) + { +#if USE_WEBVIEW2_EXPERIMENTAL + isScreenCaptureEnabled = !isScreenCaptureEnabled; + MessageBox.Show(isScreenCaptureEnabled ? "Screen Capture Enabled" : "Screen Capture Disabled", "Info"); +#endif + } + + // + // + void FileTypePolicyExecuted(object target, ExecutedRoutedEventArgs e) + { +#if USE_WEBVIEW2_EXPERIMENTAL + webView.CoreWebView2.SaveFileSecurityCheckStarting += WebView_SaveFileSecurityCheckStarting; + webView.CoreWebView2.DOMContentLoaded += WebView_FileTypePolicy_DOMContentLoaded; + webView.CoreWebView2.SetVirtualHostNameToFolderMapping( + "appassets.example", "assets", CoreWebView2HostResourceAccessKind.DenyCors); + webView.Source = new Uri("https://appassets.example/SecnarioFileTypePolicy.html"); + MessageBox.Show("Example rules of Dangerous File Security Policy has been applied in this demo page", + "Info"); +#endif + } + // + +#if USE_WEBVIEW2_EXPERIMENTAL + // + void WebView_SaveFileSecurityCheckStarting(object sender, CoreWebView2SaveFileSecurityCheckStartingEventArgs args) + { + + // Suppress default policy for ".eml" file. + if (string.Equals(args.FileExtension, ".eml", StringComparison.OrdinalIgnoreCase)) + { + args.SuppressDefaultPolicy = true; + } + + // Cancel save/download for ".iso" file. + if (args.FileExtension == ".iso") + { + CoreWebView2Deferral deferral = args.GetDeferral(); + System.Threading.SynchronizationContext.Current.Post((_) => + { + using (deferral) + { + // With the deferral, the cancel decision and + // message box can be replaced with a customized UI. + args.CancelSave = true; + MessageBox.Show("The saving has been blocked", "Info"); + } + }, null); + } + } + // + + void WebView_FileTypePolicy_DOMContentLoaded(object sender, CoreWebView2DOMContentLoadedEventArgs e) + { + // Turn off this scenario if we navigate away from the demo page. + if (webView.CoreWebView2.Source != "https://appassets.example/SecnarioFileTypePolicy.html") + { + webView.CoreWebView2.SaveFileSecurityCheckStarting -= WebView_SaveFileSecurityCheckStarting; + webView.CoreWebView2.DOMContentLoaded -= WebView_FileTypePolicy_DOMContentLoaded; + } + + } +#endif + } } From e09d6d46aa522a3bc8105d04a94664718ca52cd3 Mon Sep 17 00:00:00 2001 From: WebView2 Github Bot Date: Wed, 12 Jun 2024 18:02:56 +0000 Subject: [PATCH 2/2] Updated package version for Win32, WPF and WinForms sample apps to 1.0.2646-prerelease --- SampleApps/WebView2APISample/WebView2APISample.vcxproj | 8 ++++---- SampleApps/WebView2APISample/packages.config | 2 +- .../WebView2WindowsFormsBrowser.csproj | 2 +- SampleApps/WebView2WpfBrowser/WebView2WpfBrowser.csproj | 2 +- 4 files changed, 7 insertions(+), 7 deletions(-) diff --git a/SampleApps/WebView2APISample/WebView2APISample.vcxproj b/SampleApps/WebView2APISample/WebView2APISample.vcxproj index e2b71d1..b1d682f 100644 --- a/SampleApps/WebView2APISample/WebView2APISample.vcxproj +++ b/SampleApps/WebView2APISample/WebView2APISample.vcxproj @@ -1,4 +1,4 @@ - + @@ -480,13 +480,13 @@ - + This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. - + - \ No newline at end of file + diff --git a/SampleApps/WebView2APISample/packages.config b/SampleApps/WebView2APISample/packages.config index 8f9266a..7620352 100644 --- a/SampleApps/WebView2APISample/packages.config +++ b/SampleApps/WebView2APISample/packages.config @@ -1,5 +1,5 @@  - + \ No newline at end of file diff --git a/SampleApps/WebView2WindowsFormsBrowser/WebView2WindowsFormsBrowser.csproj b/SampleApps/WebView2WindowsFormsBrowser/WebView2WindowsFormsBrowser.csproj index baf91be..4541477 100644 --- a/SampleApps/WebView2WindowsFormsBrowser/WebView2WindowsFormsBrowser.csproj +++ b/SampleApps/WebView2WindowsFormsBrowser/WebView2WindowsFormsBrowser.csproj @@ -25,7 +25,7 @@ AnyCPU - + diff --git a/SampleApps/WebView2WpfBrowser/WebView2WpfBrowser.csproj b/SampleApps/WebView2WpfBrowser/WebView2WpfBrowser.csproj index 741d041..4e3ba86 100644 --- a/SampleApps/WebView2WpfBrowser/WebView2WpfBrowser.csproj +++ b/SampleApps/WebView2WpfBrowser/WebView2WpfBrowser.csproj @@ -61,7 +61,7 @@ - +