WPF中AllowsTransparency和WebBrowser相容性問題

来源:http://www.cnblogs.com/greed/archive/2016/02/24/5213235.html
-Advertisement-
Play Games

最近項目中需要在WPF中加入WebBrowser控制項,發現與AllowsTransparency="True" WindowStyle="None"有衝突,在網上搜索了一下,找到的解決方案跟我的項目有衝突,只好自己解決。 產生原因:引自http://www.cnblogs.com/SkyD/arch


最近項目中需要在WPF中加入WebBrowser控制項,發現與AllowsTransparency="True" WindowStyle="None"有衝突,在網上搜索了一下,找到的解決方案跟我的項目有衝突,只好自己解決。

產生原因:引自http://www.cnblogs.com/SkyD/archive/2009/12/16/1625216.html一段回覆

原因在於 WebBrowser 控制項是GDI負責呈現的。而WPF是用dx繪製的,但AllowsTransparency="True" WindowStyle="None" 時原先由gdi繪製的窗體就沒有了。完全用dx繪製。 WebBrowser 也就顯示不出來,但能響應事件。

暫時想到最直接簡單的解決辦法就是做一個無邊框不透明的窗體,在這個窗體上放置WebBrowser控制項,將這個窗體置於需要WebBroswer的窗體之上(Owner),為了方便使用,封裝成控制項,包含兩部分:

1、包含WebBrowser的窗體

2、自定義用戶控制項

 

1、窗體的xaml很簡單,Web.xaml

<Window x:Class="ZControls.WebBrowserEx.Web"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <Grid x:Name="grid"></Grid>
</Window>

b code,Web.xaml.cs

namespace ZControls.WebBrowserEx
{
    /// <summary>
    /// Web.xaml 的交互邏輯
    /// </summary>
    public partial class Web : Window
    {
        public System.Windows.Forms.WebBrowser WebBrowser { get { return web_browser; } }
        private System.Windows.Forms.Integration.WindowsFormsHost forms_host;
        private System.Windows.Forms.WebBrowser web_browser;
        public Web()
        {
            InitializeComponent();
            this.WindowStyle = System.Windows.WindowStyle.None;
            this.ResizeMode = System.Windows.ResizeMode.NoResize;
            this.ShowInTaskbar = false;
            forms_host = new System.Windows.Forms.Integration.WindowsFormsHost();
            web_browser = new System.Windows.Forms.WebBrowser();
            forms_host.Child = web_browser;
            this.grid.Children.Add(forms_host);
        }
    }
}

 

2、用戶控制項,WebBrowserExControl.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Interop;
namespace ZControls.WebBrowserEx
{
    public class WebBrowserExControl : Canvas
    {
        public System.Windows.Forms.WebBrowser WebBrowser
        {
            get
            {
                return web == null ? null : web.WebBrowser;
            }
        }
        public bool IsShow { get { return show; } }
        private bool show = false;
        private Web web = null;
        private HwndSourceHook hook;
        private HwndSource hwndSource;
        private Window window;
        private const int WM_MOVE = 0x0003;
        private const int WM_MOVING = 0x0216;
        private const int WM_SIZE = 0x0005;
        private const int WM_SIZING = 0x0214;
        private const int WM_ENTERSIZEMOVE = 0x231;
        private const int WM_EXITSIZEMOVE = 0x232;
        private const int WM_PAINT = 0x000F;
        private const int WM_GETMINMAXINFO = 0x0024;//此消息發送給視窗當它將要改變大小或位置
        private const int WM_WINDOWPOSCHANGING = 0x0046;//發送此消息給那個視窗的大小和位置將要被改變時,來調用setwindowpos函數或其它視窗管理函數
        private const int WM_NCCALCSIZE = 0x0083;//當某個視窗的客戶區域必須被核算時發送此消息
        private const int WM_WINDOWPOSCHANGED = 0x0047;//發送此消息給那個視窗的大小和位置已經被改變時,來調用setwindowpos函數或其它視窗管理函數
        private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
        {
            switch (msg)
            {
                //case WM_MOVE:
                //case WM_MOVING:
                //case WM_SIZE:
                //case WM_SIZING:
                //case WM_ENTERSIZEMOVE:
                //case WM_EXITSIZEMOVE:
                //case WM_GETMINMAXINFO:
                //case WM_WINDOWPOSCHANGING:
                //case WM_NCCALCSIZE:
                //case WM_WINDOWPOSCHANGED:
                case 49849: //case 0xc2b9: 這是一個很神奇的消息,可能是WPF專用               
                case WM_PAINT:
                    render_web();
                    break;
            }
            //Console.WriteLine(msg);
            return IntPtr.Zero;
        }
        public WebBrowserExControl()
        {
            //this.Text = "WebBrowserExControl";
        }
        ~WebBrowserExControl()
        {
            Hide();
        }
        public void Show()
        {
            if (!show)
            {
                web = new Web();
                window = Window.GetWindow(this);
                hwndSource = PresentationSource.FromVisual(this) as HwndSource;
                if (hwndSource != null)
                {
                    hook = new HwndSourceHook(WndProc);
                    hwndSource.AddHook(hook);
                }
                web.Owner = window;
                web.Show();
                show = true;
                render_web();
            }
        }
        public void Hide()
        {
            if (hwndSource != null)
            {
                hwndSource.RemoveHook(hook);
            }
            if (web != null)
            {
                //try
                //{
                web.WebBrowser.Dispose();
                web.Close();
                //}
                //catch { }
                web = null;
            }
            show = false;
        }
        private void render_web()
        {
            if (show)
            {
                Point point = this.TransformToAncestor(window).Transform(new Point(0, 0));
                //point = this.PointToScreen(point);
                web.Left = window.Left + point.X;// -LeftEx;
                web.Top = window.Top + point.Y;// -TopEx;
                //point = this.PointToScreen(point);
                //web.Left = point.X;// -LeftEx;
                //web.Top = point.Y;// -TopEx;
                web.Width = this.ActualWidth;
                web.Height = this.ActualHeight;
            }
        }
    }
}

 

最後,直接在需要的地方引用這個控制項

<WebBrowserEx:WebBrowserExControl Background="Aquamarine" x:Name="my_web"/>

b code

private void Button_Click(object sender, RoutedEventArgs e)
 {
            var url = txt_url.Text;
            my_web.Show();
            my_web.WebBrowser.Navigate(url);
 }

在hook窗體的過程中,需要找一個窗體move與size改變的消息,嘗試了半個多小時,找到消息 49849 (0xc2b9),能比較好的滿足需求,猜想這個可能是WPF定義的消息。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 最近寫了兩個小程式都要調用Windows自帶的命令行程式,一個是調用Openfiles.exe查詢正在編輯的共用文檔,一個是調用DiskPart.exe查詢硬碟狀態。兩種命令行程式調用有點不同,記錄一下。 1.用ProcessStartInfo配置參數調用。 這種是在CMD里直接帶參數輸入的,要註意
  • Atitit.Java exe bat 作為windows系統服務程式運行 1. 使用SC命令+srvany.exe (不錯,推薦)+net start1 1.1. First 創建一個java的運行bat1 1.2. 配置srvany 做serv wrapper1 1.3. 使用sc 創建/del
  • 實現科學上網,同時讓git和svn也使用shadowsocks來順暢的獲取源碼。
  • 本文先引入給讀者一個自己研究的機會,下次深入說明一下: 廢話不多說,直接上圖 新建一個mvc的項目 在視圖裡面添加一個移動端視圖 正常訪問一下 Bootstrap自帶的響應式的方式(頁面代碼並沒有改變) 我們來模擬一下移動端訪問: 谷歌或者火狐藉助:user-agent switcher 深入的部分
  • 首先,吐槽一下金數據的API文檔 http://help.jinshuju.net/articles/api-intro.html 寫的很粗糙啊...反正我是沒太看明白 拿表單api舉例,只告訴你了個地址https://jinshuju.net/api/v1/forms/ex27t2,然後呢,然後沒
  • 方法: 1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Web; 5 using System.Web.Mvc; 6 namespace PaiXie.Pos.Admin
  • 主要是想把日期和其它因素考慮進來。 使用RNGCryptoServiceProvider類創建唯一的最多8位數字元串。 private static string GetUniqueKey() { int maxSize = 8; int minSize = 5; char[] chars = ne
  • 泛型通常用在集合和集合上運行的方法中; 泛型NET Framework2.0提供一個新的命名空間System。Collections。Generic。 早期集合缺點: System.Collections.ArrayList list = new System.Collections.ArrayLi
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...