运用javascript的成员访问特性来实现通用版的兼容所有浏览器的打开对话框功能
打开网页对话框,一般有三种方法:window.open、window.showModalDialog、window.showModelessDialog,每一种都有它的优点与不足。第一种方法:window.open是大家常用的方法,它兼容所有的浏览器,打开一个新网页窗口很简单,但这种方法打开的“对话框”它默认的情况下是不支持交互(比如:获得返回值)与挂起操作(即:打开后若没有关闭,则其它网页窗口均无法操作,处于挂起等待状态);第二、第三种方法:window.showModalDialog、window.showModelessDialog支持交互操作且若使用showModalDialog还支持模式对话框,能够实现其它网页窗口被挂起,符合对话框的标准,但由于这两种方法只支持IE浏览器,所以除非是要求用户指定在IE下浏览操作网页,否则不建议使用。对话框其实在实际的网站类系统中必不可少,为了解决对话框的兼容与交互功能,我针对javascript的成员访问特性(所有的成员均以键值对的形式保存,可以通过成员访问符.来访问,如:window.alert,也可以通过键来访问,如:window["alert"])来实现可交互的通用对话框。
一、先来定义对话框的页面内容,代码如下:
WinDialog.html
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<title></title>
</head>
<body>
<table id="optionstable">
<thead>
<tr>
<th>工 号</th>
<th>姓 名</th>
<th>性 别</th>
<th>学 历</th>
</tr>
</thead>
<tbody>
<tr>
<td>0001</td>
<td>张三</td>
<td>男</td>
<td>本科</td>
</tr>
<tr>
<td>0002</td>
<td>张巧</td>
<td>女</td>
<td>本科</td>
</tr>
<tr>
<td>0003</td>
<td>李四</td>
<td>男</td>
<td>专科</td>
</tr>
<tr>
<td>0004</td>
<td>赵六</td>
<td>男</td>
<td>博士</td>
</tr>
</tbody>
</table>
<script type="text/javascript">
var request = getRequest();
var cancel = true; window.onload = function () {
var tbody = document.getElementById("optionstable").getElementsByTagName("tbody").item(0);
for (var i = 0; i < tbody.rows.length; i++) {
var row = tbody.rows[i];
row.onmouseover = function () {
this.style.backgroundColor = "red";
}
row.onmouseout = function () {
this.style.backgroundColor = "";
}
row.onclick = function () {
if (request["complete"]) {
window.opener[request["complete"]](getSelectValues(this)); //回调父窗口的方法
window.cancel = false;
window.close();
}
}
}
} window.onunload = function () {
if (cancel && request["cancel"]) {
window.opener[request["cancel"]](); //回调父窗口的方法
}
} //获取URL参数
function getRequest() {
var url = location.search; //获取url中"?"符后的字串
var theRequest = new Object();
if (url.indexOf("?") != -1) {
var str = url.substr(1);
strs = str.split("&");
for (var i = 0; i < strs.length; i++) {
theRequest[strs[i].split("=")[0]] = (strs[i].split("=")[1]);
}
}
return theRequest;
} //获取当前选中行的单元格的内容数组
function getSelectValues(row) {
var values = [];
for (var i = 0; i < row.cells.length; i++) {
values[i] = row.cells[i].innerHTML;
}
return values;
}
</script>
</body>
</html>
上述代码中关键的地方就是回调父窗口的方法语句:window.opener[request["complete"]](getSelectValues(this));及 window.opener[request["cancel"]]();也可以使用eval的形式来动态调用。
二、接下来看父窗口(即:调用窗口)如何来弹出该对话框并实现交互,代码如下:
WinOpner.html
<!DOCTYPE html>
<html>
<head>
<title>父窗口</title>
</head>
<body>
<p>
<input id="Text1" type="text" /><input id="Button1" type="button" value="打开子窗口" />
</p>
<script type="text/javascript">
window.onload = function () {
document.getElementById("Button1").onclick = function () {
var WinDialog = window.open("WinDialog.html?complete=dialog_complete&cancel=dialog_cancel",//参数中指定回调方法
"_blank","toolbar=no,menubar=no,scrollbars=auto,resizable=no,location=no,status=no,depended=yes,width=600,height=500,left=" + (screen.width - 600) / 2 + ",top=" + (screen.height - 500) / 2);
};
} function dialog_complete(result) { //完成时回调方法
alert(result);
} function dialog_cancel() {//对话框关闭时回调方法
alert("cancel!");
}
</script>
</body>
</html>
上述代码中除了使用window.open方法来打开新窗口外,特别需要注意的是URL中需包含WinDialog.html对话框网页中约定的回调方法参数:complete=dialog_complete&cancel=dialog_cancel,这样在对话框打开后执行选择就会自动回调dialog_complete方法,若取消关闭就会回调dialog_cancel方法。
先来看一下效果截图:
打开对话框:

选择一行(单击)数据,回调了父窗口中的dialog_complete方法,弹出选择的数据,当然你可以将数据写入到文本框中:

若不选择直接关闭(即:取消),回调了父窗口中的dialog_cancel:

看到这些,有没有觉得不错,我觉得还是可以的,当然也有人可能会说上面的URL需包含回调方法参数,若参数名或方法名写错了,就会造成回调不成功,这是个问题,还有每个调用页面要重复写一堆的Window.open参数很不好,能不能封装成一个对象?当然,这些我都想到了,而且也实现了,请看下面的封装代码:
WebDialog.js中定义成WebDialog对象:
var WebDialog = function (url, width, height, resizeable) {
this.left = (screen.width - width) / 2;
this.top = (screen.height - height) / 2;
this.complete = function (result) { };
this.cancel = function () { };
var _self = this;
this.show = function () {
var _features = "toolbar=no,menubar=no,scrollbars=auto,resizable=" + resizeable + ",location=no,status=no,depended=yes,width=" + width + ",height=" + height + ",left=" + _self.left + ",top=" + _self.top;
var winDialog = window.open(url, "WebDialogWindow", _features);
winDialog.dialog_complete = _self.complete;
winDialog.dialog_cancel = _self.cancel;
}
}
上述代码中将window.open返回的窗口对象赋值给winDialog变量,然后为其指定dialog_complete及dialog_cancel方法
WinDialog2.html 定义对话框显示的内容:
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<title></title>
</head>
<body>
<table id="optionstable">
<thead>
<tr>
<th>工 号</th>
<th>姓 名</th>
<th>性 别</th>
<th>学 历</th>
</tr>
</thead>
<tbody>
<tr>
<td>0001</td>
<td>张三</td>
<td>男</td>
<td>本科</td>
</tr>
<tr>
<td>0002</td>
<td>张巧</td>
<td>女</td>
<td>本科</td>
</tr>
<tr>
<td>0003</td>
<td>李四</td>
<td>男</td>
<td>专科</td>
</tr>
<tr>
<td>0004</td>
<td>赵六</td>
<td>男</td>
<td>博士</td>
</tr>
</tbody>
</table>
<script type="text/javascript">
var cancel = true; window.onload = function () {
var tbody = document.getElementById("optionstable").getElementsByTagName("tbody").item(0);
for (var i = 0; i < tbody.rows.length; i++) {
var row = tbody.rows[i];
row.onmouseover = function () {
this.style.backgroundColor = "red";
}
row.onmouseout = function () {
this.style.backgroundColor = "";
}
row.onclick = function () {
var dialog_complete = window["dialog_complete"];//获取回调方法
if (dialog_complete && typeof dialog_complete != "undefined") {
dialog_complete(getSelectValues(this));
window.cancel = false;
window.close();
}
}
}
} window.onunload = function () {
var dialog_cancel = window["dialog_cancel"]; //获取回调方法
if (cancel && dialog_cancel && typeof dialog_cancel != "undefined") {
dialog_cancel();
}
} //获取当前选中行的单元格的内容数组
function getSelectValues(row) {
var values = [];
for (var i = 0; i < row.cells.length; i++) {
values[i] = row.cells[i].innerHTML;
}
return values;
} </script>
</body>
</html>
上述代码中,通过window["dialog_complete"]、window["dialog_cancel"]来获取回调方法,然后通过判断是否有获取成功,若获取成功则直接调用,调用的方法与前面封装的指定的回调方法签名必需保持一致。
最后再来看如何通过WebDialog对象调用完成相同的交互功能,代码如下:
WinOpner2.html
<!DOCTYPE html>
<html>
<head>
<title>父窗口</title>
<script type="text/javascript" src="WebDialog.js?v1"></script>
</head>
<body>
<p>
<input id="Text1" type="text" /><input id="Button1" type="button" value="打开子窗口" />
</p>
<script type="text/javascript">
document.getElementById("Button1").onclick=function(){
var dlg = new WebDialog("WinDialog2.html", 600, 500, "yes");
dlg.complete=function(result){//完成时回调方法
alert(result);
};
dlg.cancel=function() {//取消时回调方法
alert("cancel!");
};
dlg.show();
} </script>
</body>
</html>
最终实现的效果与没有封装前的方法相同,在此就不重复截图了。
如果说嫌回调方法太麻烦了,不想每个父窗口中都写回调方法,只需要简单的打开对话框,并且把选中的值赋给指定的文本框即可,行不行?当然可以,为WebDialog增加一个refillInputId属性用于存入回填文本框的ID,如下:
//部份代码:
this.refillInputId = null;//回填文框ID
var _self = this;
this.show = function () {
var _features = "toolbar=no,menubar=no,scrollbars=auto,resizable=" + resizeable + ",location=no,status=no,depended=yes,width=" + width + ",height=" + height + ",left=" + _self.left + ",top=" + _self.top;
var winDialog = window.open(url, "WebDialogWindow", _features);
winDialog.dialog_complete = _self.complete;
winDialog.dialog_cancel = _self.cancel;
winDialog.refillInputId = _self.refillInputId;
}
然后在对话框内容页面中加入选择时回填值即可,如下:
row.onclick = function () {
var dialog_complete = window["dialog_complete"]; //获取回调方法
if (dialog_complete && typeof dialog_complete != "undefined") {
dialog_complete(getSelectValues(this));
}
var refillInputId = window["refillInputId"]; //回填文本框
if (typeof refillInputId == "string") {
window.opener.document.getElementById(refillInputId).value = getSelectValues(this)[0];
}
window.cancel = false;
window.close();
这样在父窗口中调用的时候,你若需要自行处理选择的结果,你可以指定回调方法,否则省略掉,直接指定回填文本框即可,如下:
document.getElementById("Button1").onclick = function () {
var dlg = new WebDialog("WinDialog2.html", 600, 500, "yes");
// dlg.complete=function(result){//完成时回调方法
// alert(result);
// };
// dlg.cancel=function() {//取消时回调方法
// alert("cancel!");
// };
dlg.refillInputId = "Text1";
dlg.show();
}
最终的效果如下:

好了,对话框研究得差不多了,我所说的方法不一定最好,但一定可以满足基本要求,关于文中一开头提到的除了交互,还有就是挂起,这个目前还没有想到办法,应该是浏览器限制的因素,因为如果有人恶意制作无数个挂起窗口,那电脑就会崩溃掉的,当然如果有更好的方法,欢迎交流,谢谢!
运用javascript的成员访问特性来实现通用版的兼容所有浏览器的打开对话框功能的更多相关文章
- JavaScript 数据属性和访问器属性
在JavaScript中对象被定义为"无序属性的集合,其属性可以包含基本值.对象或函数."通俗点讲,我们可以把对象理解为一组一组的名值对,其中值可以是数据或函数. 创建自定义对象通 ...
- 理解Javascript的动态语言特性
原文:理解Javascript的动态语言特性 理解Javascript的动态语言特性 Javascript是一种解释性语言,而并非编译性,它不能编译成二进制文件. 理解动态执行与闭包的概念 动态执行: ...
- java基础-类成员访问权限控制
一 前言 这篇文章是很基础的一文,没多大深度,对于开发人员必然是熟练于心.本篇文章的主题是为什么java要设置类成员访问级别?其原因也很简单,就是为了面向对象的封装特性:将类成员使用不同的访问级别控制 ...
- Java基础进阶:多态与接口重点摘要,类和接口,接口特点,接口详解,多态详解,多态中的成员访问特点,多态的好处和弊端,多态的转型,多态存在的问题,附重难点,代码实现源码,课堂笔记,课后扩展及答案
多态与接口重点摘要 接口特点: 接口用interface修饰 interface 接口名{} 类实现接口用implements表示 class 类名 implements接口名{} 接口不能实例化,可 ...
- Java基础详解 (一)Java的类成员访问权限修饰词(以及类访问权限)
在一个类的内部,其成员(包括成员变量和成员函数)能否被其他类所访问,取决于该成员的修饰词.Java的类成员访问权限修饰词有四类:private,无(默认情况下),protected和public.其权 ...
- Java 声明和访问控制(三) finalize方法 成员访问修饰符
finalize()方法是Object类的一个方法,在垃圾回收器执行的时候,会调用被回收对象的此方法,可以覆盖此方法提供垃圾收集时的其他资源的回收,例如文件关闭等. 成员访问修饰符: 默认访问:包访问 ...
- JavaScript对象属性访问的两种方式
JavaScript对象属性访问的两种方式 object.attribute object["attribute"] 例如: var employees = [ { "f ...
- c/c++ 重载运算符 关系,下标,递增减,成员访问的重载
重载运算符 关系,下标,递增减,成员访问的重载 为了演示关系,下标,递增减,成员访问的重载,创建了下面2个类. 1,类StrBlob重载了关系,下标运算符 2,类StrBlobPtr重载了递增,抵减, ...
- 【转】C++对成员访问运算符->的重载
运算符->的重载比较特别,它只能是非静态的成员函数形式,而且没有参数. 1.如果返回值是一个原始指针,那么就将运算符的右操作数当作这个原始指针所指向类型的成员进行访问: 2.如果返回值是另一个类 ...
随机推荐
- SQL入门经典(八) 之存储过程
存储过程(stored procedure)有时候称为sproc,它是真正的脚本-或者更准确的说,他是批处理(batch)-它存储于数据库中,而不是淡出的文件中.无论如何,这个比较并不是很确定.存储过 ...
- SQL语言分类
SQL语言共分为四大类:数据查询语言DQL,数据操纵语言DML, 数据定义语言DDL,数据控制语言DCL. 1 数据查询语言DQL数据查询语言DQL基本结构是由SELECT子句,FROM子句,WHER ...
- java 多线程(synchronized)
package com.example; public class App { public static void main(String[] args) { doRunable dr = new ...
- Linux 网络编程(IO模型)
针对linux 操作系统的5类IO模型,阻塞式.非阻塞式.多路复用.信号驱动和异步IO进行整理,参考<linux网络编程>及相关网络资料. 阻塞模式 在socket编程(如下图)中调用如下 ...
- 团队项目——站立会议 DAY11
团队项目--站立会议 DAY11 团队成员介绍(5人):张靖颜.何玥.钟灵毓秀.赵莹.王梓萱 今日(2016/5/20),站立会议已进行了两周时间,将这一周所遇到的问题和心 ...
- 深入理解MVVM模式中Silverlight的Trigger、Action和Behavior及Silverlight的继承机制
接触Silverlight已经有两三个月了,开始一直感觉他和Winform很相似,拖拖控件就行了,所以一直把经历放在了研究后台和服务器交互和性能优化上面,很少去仔细研究Silverlight的页面.前 ...
- 从Windows中卸载Apache
在重装Apache或者妳不再需要它的时候,这时就需要将它卸载. 下面是步骤: 打开开始菜单(win8中ÿ+X)或者我的电脑(废话) 找到并打开Apache的安装目录(Program Files\Apa ...
- 据说每个大牛、小牛都应该有自己的库——JavaScript原生对象拓展
在据说每个大牛.小牛都应该有自己的库——框架篇中我扬言要做个小牛,没想到一天没更新,小伙儿伴们就戏谑的问我,油哥是不是要太监了?其实事情是这个样子的,这不是太监的节奏,一是,关于写个自己的库的想法由来 ...
- 230行实现一个简单的MVVM
作者:mirone链接:https://zhuanlan.zhihu.com/p/24451202来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. MVVM这两年在前端届 ...
- 说说设计模式~门面模式(Facade)
返回目录 门面模式(Facade)属于结构型模式的一种,它符合面向对象的封装原则,但又不符合开闭原则,呵呵,今天我们主要说它的优点,不谈缺点. 定义 门面模式,是指提供一个统一的接口去访问多个子系统的 ...