Windows Phone开发-Webbrowser使用技巧

5月份开发了脸萌WP版,其中需要使用web技术来绘制图像,于是就使用了原生webbrowser控件。在使用webbrowser control的过程中,发现了一些坑,也总结的几个小技巧,分享给大家。

首先我们需要了解下webbrowser control,与桌面windows平台上的webbrowser一样,都是基于IE内核封装的web展示控件,用法与wpf基本一致,同时也是非托管资源,但wp平台上没有实现IDispose接口,带了了内存控制的挑战。此外,webbrowser本身支持有限的js script和c#代码交互,带来了更多的可操作性。

 

1.c#调用webbrowser function

这里主要是指使用本地代码(C#)和webbrowser加载的Javascript代码的交互行为,利用此方法可以很容易的实现web项目对wp原生项目的移植,这对web开发者来讲是很方便的。这里主要给大家介绍2个API:分别是InvokeScript()方法以及ScriptNotify事件。

InvokeScript方法可以传参并执行webbrowser加载的javascript代码中的函数,实现对应的效果,用法如下:

webbrowser.InvokeScript("initPerson");
webbrowser.InvokeScript("initPerson", new string[] {"0","2001","1" });

第一段就是简单的调用js function,第二行则是传递参数并调用js function。这两种方法使用起来很灵活,足以来处理很多的交互行为。以脸萌WP版为例,我在C#中处理各种用户行为,并将获得的参数传入webbrowser,js 脚本响应后会动态执行,使用web技术绘制出我想要的头像,这非常灵活。

 同时,对js 函数适当的封装可以提高交互效率,因为InvokeScript可传入的参数类型只能为 string[],js函数不恰当的封装会导致交互的困难,甚至是无法交互,下面我举例说明:

复制代码
 var personA = {
            'eye': {
                'id': 0
            },
            'hair': {
                'id': 0
            }
        };

        function initPerson(person) {
            a = person.eye.id;
            b = person.hair.id;
        }
复制代码

 

这里的initPerson函数需要直接传入javascript对象,这是InvokeScript难以直接传入的,但我们可以进行简单的封装,就可以顺利的实现意图,如下:

 

复制代码
var personA = {
            'eye': {
                'id': 0
            },
            'hair': {
                'id': 0
            }
        };

        function initPerson(person) {
            a = person.eye.id;
            b = person.hair.id;
        }
        function drawPerson(eyeId, hairId) {
            personA.eye.id = eyeId;
            personA.hair.id = hairId;
            initPerson(personA);
        }
复制代码

 这样C#就直接可以写为:

webbrowser.InvokeScript("drawPerson", new string[] {"0","2001" });

 

能够顺利的实现交互,以上只是方法之一,大家还可以直接传入一个字符串,然后再在javascript中进行处理,如下:

function initPerson(person) {
            if (typeof person == 'string')
                var person = eval(person);
            a = person.eye.id;
            b = person.hair.id;
        }

 

C#:

webbrowser.InvokeScript("initPerson", "{ 'eye': {'id': 0 }, 'hair': { 'id': 0}}");

借助神奇的eval()函数,同样可以实现我们想要的效果。MVP郝冠军告诉我,使用eval并不恰当,可以使用Native json API来替代,大家可以自行搜索下用法,使用方式和eval一致。

2.webbrowser调用C#脚本

这里使用的API主要是ScriptNotify事件,可以实现javascript函数调用c#脚本的效果,能够动态的改变整个应用的行为,这能够使得基于web技术的app调用更高级的原生功能,包括位置,推送,动态磁贴等等,是不是觉得很棒呀,下面我们来看简单的实现方式。

首先webbrowser需要注册ScriptNotify事件,以接收来自javascript的信息

 

webbrowser.ScriptNotify += (e, g) =>
            {
                //响应代码
                var result = g.Value;
            };

 

同时,javascript发送数据方式如下:

 

 

window.external.notify("参数");


如此即可实现webbrowser与整个app的动态交互,这是非常便利的。

 

3.webbrowser的内存控制

由于webbrowser是非托管资源,同时wp上未实现IDispose接口,这给app的内存带来了很大压力,每次导航都可以明显发现有20M左右内存没有释放,经过多次检查,发现是webbrowser的资源没有释放,同时也发现强制GC也没有任何效果,最终经过一番学习,通过js来进行释放,大幅减小了内存的消耗,但仍不彻底,诸位大牛如果有好的办法可以指点下我。下面是我的方法:

js中将所有使用的资源都做空处理:

 

function memoryRelease() {
            personA = null;
            window = null;
            CollectGarbage();
        }

C#每次在离开当前页面时执行它:

 

protected override void OnNavigatedFrom(NavigationEventArgs e)
        {
            webbrowser.InvokeScript("memoryRelease");
        }

 

CollectGarbage()是IE的一个特有属性,用于释放内存的,使用方法应该是,将该变量或引用对象,设置为null或delete,然后在进行释放动作。

通过这种方式可以有效的减少内存泄漏,仅做抛砖引玉只用。

 

 

最后,silverlight架构下的webbrowser控件虽然坑不少,但仍是一个非常灵活的控件,这对于使用html5开发跨平台app的开发者来讲,是个非常棒的方式。

posted @   msp的昌伟哥哥  阅读(1420)  评论(4编辑  收藏  举报
编辑推荐:
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?
点击右上角即可分享
微信分享提示