文档章节

.NET中WebBrowser控件调用HTML页面中Frame内部的JavaScript代码

北风其凉
 北风其凉
发布于 2016/04/29 18:29
字数 1041
阅读 318
收藏 1

我的操作系统版本为Win7,VS版本为VS2012。

之前的一篇博文中提到使用.NET中的WebBrowser控件,调用加载的HTML页面中的JS函数,博文地址在:

http://my.oschina.net/Tsybius2014/blog/643909

不过,最近又遇到了一个棘手的问题,就是如果JS代码在Frame内的HTML页面中,上述博文中调用JS的方法就不灵了。

因此,我今天琢磨了一个新的办法,调用这些JS代码,先贴一组HTML页面的代码:

1、test_page.html

<html>
  <head>
    <title>Frame测试页面</title>
  </head>
  <script>
    // TODO - 这里输入JS代码
  </script>
  <frameset cols="40%,60%" rows="2*,*">
    <frame name="top_left" src="top_left.html">
    <frame name="top_right" src="top_right.html">
    <frame name="bottom_left" src="bottom_left.html">
    <frame name="bottom_right" src="bottom_right.html">
  </frameset>
</html>

2、其下有四个Frame,分居页面的左上、右上、左下、右下四端,以top_left.html为例,代码为:

<html>
  <head>
    <title>测试子页面</title>
  </head>
  <body>
    <div id="bulletin">你好</div>
    <script>
      // TODO - 这里输入JS代码
    </script>
  </body>
</html>

我的Winform工程是用C#语言建立的,窗体结构如下:


代码结构如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace WebBrowserFrameTest
{
    [System.Runtime.InteropServices.ComVisible(true)]
    public partial class FormMain : Form
    {
        public FormMain()
        {
            InitializeComponent();
        }

        private void FormMain_Load(object sender, EventArgs e)
        {
            try
            {
                string path = @"C:\Users\Tsybius\Desktop\测试页面\test_page.html";
                webBrowser.Navigate(path);
                webBrowser.ObjectForScripting = this;
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.ToString());
            }
        }
    }
}

我想到了三个方法调用Frame内部的JS函数,现在对它们进行逐一说明:

方法1:给定Frame的name和函数名,调用指定Frame中页面内指定名称的JS函数

C#程序为按钮btnTopLeft1定义单击事件代码:

private void btnTopLeft1_Click(object sender, EventArgs e)
{
    webBrowser.Document.InvokeScript("changeBulletin", new object[] { "top_left" });
}

在test_page.html的script标签下写入如下代码:

function changeBulletin(frameName) {
  window.frames[frameName].changeBulletin();
}

在top_left.html的script标签下写入如下代码:

function changeBulletin() {
  document.getElementById("bulletin").innerHTML = "TOP_LEFT";
}

此时单击按钮btnTopLeft1,就可以看到效果了(文字有变化)。

方法2:以参数的形式给定Frame的name和函数名,调用指定Frame中页面内指定名称的JS函数

方法1的问题很明显,方法1中,主页面的changeBulletin函数是写死的。在实际程序编写中,如果每个JS函数调用,我们都要在父页面实现一个同名的函数,那真是要累死人了。

优化的办法是,为C#程序为按钮btnTopLeft2定义单击事件代码:

private void btnTopLeft2_Click(object sender, EventArgs e)
{
    webBrowser.Document.InvokeScript("runJsInFrame", new object[] { "top_left", "changeBulletin" });
}

在test_page.html的script标签下写入如下代码:

function runJsInFrame(frameName, funcName) {
  window.frames[frameName][funcName]();
}

这样做达到的效果和方法1是一样的,不过好处是只定义一个函数就好了。

方法3:除以参数形式给定Frame的name和函数名外,还要支持调用函数的参数和返回值

方法2的缺点是,虽然实现了传入Frame的name和函数名灵活调用JS函数,但不能为JS函数添加参数并接收返回值。不能实现这一点,那这个方法就是个残缺的方法,因此我又想了方法3去修复这一问题。

为C#程序为按钮btnTopLeft3定义单击事件代码:

private void btnTopLeft3_Click(object sender, EventArgs e)
{
    object obj = webBrowser.Document.InvokeScript("runJsInFrame2", new object[] { "top_left", "addParams", "1", "2", "3" });
    if (obj != null)
    {
        MessageBox.Show(obj.ToString());
    }
    else
    {
        MessageBox.Show("null");
    }
}

在test_page.html的script标签下写入如下代码:

function runJsInFrame2(frameName, funcName) {
  var args = "";
  if (arguments.length > 2) {
    for (var i = 2; i < arguments.length; i++) {
      args += "\"" + arguments[i] + "\"";
      if (i < arguments.length - 1) {
        args += ",";
      }
    }
  }
  var command = "window.frames[\"" + frameName + "\"][\"" + funcName + "\"](" + args + ")";
  var result = eval(command);
  return result;
}

在top_left.html的script标签下写入如下代码:

function addParams() {
  var sum = 0;
  for (var i = 0; i < arguments.length; i++) {
    sum += parseInt(arguments[i]);
  }
  return sum;
}

这里调用了top_left.html中的addParams函数,将所有的输入参数做加法运算,并返回加法运算的和。

这里使用到了JS的可变参数,runJsInFrame2函数中,传入的第一个参数是Frame的name,传入的第二个参数是函数名,后面任意多个参数可根据实际使用自行添加。

这段代码的运行结果就是,返回一个6(1+2+3=6)。

END

© 著作权归作者所有

北风其凉

北风其凉

粉丝 120
博文 497
码字总数 462305
作品 4
朝阳
程序员
私信 提问
在 Windows Phone 中集成 HTML5 以及 JavaScript

这年头,到处都在讨论HTML5,我自己也在 Danish Developer Conference 2012 on Windows Phone and HTML5 大会上提到了它。我们也看到了 HTML5 的能力与优点。但是对于移动应用来说,HTML5 对...

junwong
2012/03/20
3.9K
0
WebBrowser(WebView) API (浏览器控件WebView)

一、WP8中的WebBrowser Windows Phone 提供基于桌面浏览器的 WebBrowser 控件。Windows Phone OS 7.1 的 WebBrowser 控件基于 Internet Explorer 9,且 Windows Phone 8 的 WebBrowser 控件基......

失足处男的倒霉孩子
2013/12/27
7.7K
0
WP7,让WebBrowser从独立存储中加载css/js文件以及图片文件

介绍: 在我们的Windows Phone 7项目中,有可能需要动态生成一些HTML字符串或者从WebService上获取一段HTML字符串,然后将这些HTML字符串在WebBrowser控件上显示,那么一般在这种情况下你可能...

junwong
2012/02/21
3.1K
3
delphi中Webbrowser

1.获得网页中变量值 htm中<script> var currID=123</script> 程序中可以这么调用 id := Form1.WebBrowser1.OleObject.Document.script.currID 值得说明的是,变量可以是javascript定义的,也可......

任远
2015/05/08
191
0
javascript与c#相互调用

首先建立网页: 二、建立Windows应用程序 创建Windows应用程序项目 在Form1窗体中添加WebBrowser控件 在Form1类的上方添加 [System.Runtime.InteropServices.ComVisibleAttribute(true)] 这是...

crossmix
2015/04/16
196
0

没有更多内容

加载失败,请刷新页面

加载更多

Activity启动模式二

上篇文章Activity启动模式一主要介绍了Activity的四种启动模式,这些启动模式都是在AndroidManifest中进行配置的。除此之外,Android系统还通过Intent类提供了一些标志位,同样可以指定Activ...

ltlovezh
32分钟前
4
0
三原色还原

1、Color Filter Array — CFA 随着数码相机、手机的普及,CCD/CMOS 图像传感器近年来得到广泛的关注和应用。 图像传感器一般都采用一定的模式来采集图像数据,常用的有 BGR 模式和 CFA 模式...

天王盖地虎626
43分钟前
3
0
kubernetes pod exec接口调用

正文 一般生产环境上由于网络安全策略,大多数端口是不能为集群外部访问的。多个集群之间一般都是通过k8s的ApiServer组件提供的接口通信,如https://192.168.1.101:6443。所以在做云平台时,...

码农实战
今天
8
0
3_数组

3_数组

行者终成事
今天
8
0
经典系统设计面试题解析:如何设计TinyURL(二)

原文链接:https://www.educative.io/courses/grokking-the-system-design-interview/m2ygV4E81AR 编者注:本文以一道经典的系统设计面试题:《如何设计TinyURL》的参考答案和解析为例,帮助...

APEMESH
今天
7
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部