-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlayer.js
261 lines (237 loc) · 5.23 KB
/
layer.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
import $ from 'jquery';
import Events from '@flexui/events';
import * as Utils from '@flexui/utils';
import { getZIndex } from '@flexui/z-index';
import { BACKDROP } from './lib/backdrop.js';
import { FOCUS_SHIM } from './lib/focus-shim.js';
// 得到焦点类名
var LAYER_CLASS_FOCUS = '-focus';
/**
* Layer
*
* @constructor
* @export
*/
export default function Layer() {
var context = this;
context.destroyed = false;
context.node = document.createElement('div');
context.__node = $(context.node)
// 设定 tab 索引
.attr('tabindex', '-1')
// 得到焦点
.on('focusin', function() {
if (context !== Layer.active) {
context.focus();
}
});
}
// 当前得到焦点的实例
Layer.active = null;
// 锁定 tab 焦点在浮层内
Utils.doc.on('focusin', function(e) {
var target = e.target;
var active = Layer.active;
var anchor = BACKDROP.anchor;
if (anchor) {
// 锁定焦点
switch (target) {
case BACKDROP.node[0]:
if (active) {
active.focus();
} else {
anchor.focus();
}
break;
case FOCUS_SHIM.node[0]:
anchor.focus();
break;
}
} else if (active &&
target !== active.node &&
!active.node.contains(target)) {
// 焦点不在浮层让浮层失焦
active.blur();
}
});
// 原型方法
Utils.inherits(Layer, Events, {
/**
* 浮层 DOM 元素节点
*
* @public
* @readonly
*/
node: null,
/**
* 判断对话框是否删除
*
* @public
* @readonly
*/
destroyed: true,
/**
* 判断对话框是否显示
*
* @public
* @readonly
*/
open: false,
/**
* 是否是模态窗口
*
* @public
* @readonly
*/
modal: false,
/**
* 内部的 HTML 字符串
*
* @public
* @property
*/
innerHTML: '',
/**
* CSS 类名
* 只在浮层未初始化前可设置,之后不能更改
*
* @public
* @property
*/
className: 'ui-layer',
/**
* 构造函数
*
* @public
* @readonly
*/
constructor: Layer,
/**
* 让浮层获取焦点
*
* @public
*/
focus: function() {
var context = this;
// 销毁,未打开和已经得到焦点不做处理
if (context.destroyed || !context.open) {
return context;
}
// 激活实例
var active = Layer.active;
// 先让上一个激活实例失去焦点
if (active && active !== context) {
active.blur();
}
// 浮层
var node = context.node;
var layer = context.__node;
var focused = context.__getActive();
// 检查焦点是否在浮层里面
if (node !== focused && !node.contains(focused)) {
// 自动聚焦
context.__focus(layer.find('[autofocus]')[0] || node);
}
// 非激活状态刷新浮层状态
if (Layer.active !== context) {
var index = context.zIndex = getZIndex(true);
// 刷新遮罩
if (context.modal && context !== BACKDROP.anchor) {
// 刷新遮罩位置
context.__backdrop('show');
// 刷新遮罩层级
context.__backdrop('z-index', index);
}
// 设置浮层层级
layer.css('zIndex', index);
// 添加激活类名
layer.addClass(context.className + LAYER_CLASS_FOCUS);
// 触发事件
context.emit('focus');
// 保存当前激活实例
Layer.active = context;
}
return context;
},
/**
* 让浮层失去焦点。将焦点退还给之前的元素,照顾视力障碍用户
*
* @public
*/
blur: function() {
var context = this;
// 销毁和未打开不做处理
if (context.destroyed || !context.open) {
return context;
}
if (context === Layer.active) {
// 清理激活状态
Layer.active = null;
// 移除类名
context.__node.removeClass(context.className + LAYER_CLASS_FOCUS);
// 触发失去焦点事件
context.emit('blur');
}
return context;
},
/**
* 对元素安全聚焦
*
* @private
* @param {HTMLElement} element
*/
__focus: function(element) {
// 防止 iframe 跨域无权限报错
// 防止 IE 不可见元素报错
try {
// ie11 bug: iframe 页面点击会跳到顶部
if (!/^iframe$/i.test(element.nodeName)) {
element.focus();
}
} catch (e) {
// error
}
},
/**
* 获取当前焦点的元素
*
* @private
*/
__getActive: function() {
try {
// try: ie8~9, iframe #26
var activeElement = document.activeElement;
var contentDocument = activeElement.contentDocument;
var element = contentDocument && contentDocument.activeElement || activeElement;
return element;
} catch (e) {
// error
}
},
/**
* 智能遮罩操作方法
*
* @private
* @param {String} method
* @param {any} value
*/
__backdrop: function(method, value) {
var context = this;
switch (method) {
case 'show':
case 'hide':
// 遮罩层
if (context.modal) {
BACKDROP[method](context);
}
// 焦点锁定层
FOCUS_SHIM[method](context);
break;
case 'z-index':
if (context.modal) {
BACKDROP.zIndex(value);
}
break;
}
}
});