FullScreen全屏控件源码分析
# 概述
本文主要介绍 Openlayers 中FullScreen
全屏控件的实现原理,关于全屏 API 可以参考HTML5 全屏讲解 (opens new window)。可以想象 Openlayers 中的全屏也是基于requestFullscreen
实现的,关键点是要处理好地图视图在全屏和非全屏两种状态下切换的处理。
# 源码分析
FullScreen
类继承于Control
类,关于Control
类,可以参考Control基类介绍 (opens new window)。
# FullScreen
类的实现如下
class FullScreen extends Control {
constructor(options) {
options = options ? options : {};
super({
element: document.createElement("div"),
target: options.target,
});
this.on;
this.once;
this.un;
this.keys_ = options.keys !== undefined ? options.keys : false;
this.source_ = options.source;
this.isInFullscreen_ = false;
this.boundHandleMapTargetChange_ = this.handleMapTargetChange_.bind(this);
this.cssClassName_ =
options.className !== undefined ? options.className : "ol-full-screen";
this.documentListeners_ = [];
this.activeClassName_ =
options.activeClassName !== undefined
? options.activeClassName.split(" ")
: [this.cssClassName_ + "-true"];
this.inactiveClassName_ =
options.inactiveClassName !== undefined
? options.inactiveClassName.split(" ")
: [this.cssClassName_ + "-false"];
const label = options.label !== undefined ? options.label : "\u2922";
this.labelNode_ =
typeof label === "string" ? document.createTextNode(label) : label;
const labelActive =
options.labelActive !== undefined ? options.labelActive : "\u00d7";
this.labelActiveNode_ =
typeof labelActive === "string"
? document.createTextNode(labelActive)
: labelActive;
const tipLabel = options.tipLabel ? options.tipLabel : "Toggle full-screen";
this.button_ = document.createElement("button");
this.button_.title = tipLabel;
this.button_.setAttribute("type", "button");
this.button_.appendChild(this.labelNode_);
this.button_.addEventListener(
EventType.CLICK,
this.handleClick_.bind(this),
false
);
this.setClassName_(this.button_, this.isInFullscreen_);
this.element.className = `${this.cssClassName_} ${CLASS_UNSELECTABLE} ${CLASS_CONTROL}`;
this.element.appendChild(this.button_);
}
handleClick_(event) {
event.preventDefault();
this.handleFullScreen_();
}
handleFullScreen_() {
const map = this.getMap();
if (!map) {
return;
}
const doc = map.getOwnerDocument();
if (!isFullScreenSupported(doc)) {
return;
}
if (isFullScreen(doc)) {
exitFullScreen(doc);
} else {
let element;
if (this.source_) {
element =
typeof this.source_ === "string"
? doc.getElementById(this.source_)
: this.source_;
} else {
element = map.getTargetElement();
}
if (this.keys_) {
requestFullScreenWithKeys(element);
} else {
requestFullScreen(element);
}
}
}
handleFullScreenChange_() {
const map = this.getMap();
if (!map) {
return;
}
const wasInFullscreen = this.isInFullscreen_;
this.isInFullscreen_ = isFullScreen(map.getOwnerDocument());
if (wasInFullscreen !== this.isInFullscreen_) {
this.setClassName_(this.button_, this.isInFullscreen_);
if (this.isInFullscreen_) {
replaceNode(this.labelActiveNode_, this.labelNode_);
this.dispatchEvent(FullScreenEventType.ENTERFULLSCREEN);
} else {
replaceNode(this.labelNode_, this.labelActiveNode_);
this.dispatchEvent(FullScreenEventType.LEAVEFULLSCREEN);
}
map.updateSize();
}
}
setClassName_(element, fullscreen) {
if (fullscreen) {
element.classList.remove(...this.inactiveClassName_);
element.classList.add(...this.activeClassName_);
} else {
element.classList.remove(...this.activeClassName_);
element.classList.add(...this.inactiveClassName_);
}
}
setMap(map) {
const oldMap = this.getMap();
if (oldMap) {
oldMap.removeChangeListener(
MapProperty.TARGET,
this.boundHandleMapTargetChange_
);
}
super.setMap(map);
this.handleMapTargetChange_();
if (map) {
map.addChangeListener(
MapProperty.TARGET,
this.boundHandleMapTargetChange_
);
}
}
handleMapTargetChange_() {
const listeners = this.documentListeners_;
for (let i = 0, ii = listeners.length; i < ii; ++i) {
unlistenByKey(listeners[i]);
}
listeners.length = 0;
const map = this.getMap();
if (map) {
const doc = map.getOwnerDocument();
if (isFullScreenSupported(doc)) {
this.element.classList.remove(CLASS_UNSUPPORTED);
} else {
this.element.classList.add(CLASS_UNSUPPORTED);
}
for (let i = 0, ii = events.length; i < ii; ++i) {
listeners.push(
listen(doc, events[i], this.handleFullScreenChange_, this)
);
}
this.handleFullScreenChange_();
}
}
}
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
# FullScreen
类的构造函数
FullScreen
类的构造函数就是创建了一个按钮,然后监听按钮的点击事件,进行全屏和推出全屏的状态切换,以及按钮状态的切换;构造函数的参数可能有如下属性:
target
:控件的容器,默认为空,将会添加到overlay container
中keys
:全屏key
,默认false
source
:控件源,默认undefined
className
:控件类名,默认ol-full-screen
activeClassName
:控件全屏状态时的类名ol-full-screen-true
inactiveClassName
:控件非全屏状态时的类名ol-full-screen-false
label
:控件标签labelActive
:控件全屏(激活状态)标签tipLabel
:控件hover
时显示
除了定义变量接受参数或有默认值外,构造函数还定义了其它一些变量如下
this.isInFullscreen_
:默认为false
,表示当前是否处于全屏状态this.documentListeners_
:默认为[]
,
控件按钮监听的点击事件是handleClick
# FullScreen
中全屏函数
isFullScreenSupported
函数:判断浏览器是否支持全屏 APIrequestFullscreen
以及全屏是否可用fullscreenEnabled
function isFullScreenSupported(doc) {
const body = doc.body;
return !!(
body["webkitRequestFullscreen"] ||
(body.requestFullscreen && doc.fullscreenEnabled)
);
}
2
3
4
5
6
7
isFullScreen
函数:判断当前是否处于全屏状态
function isFullScreen(doc) {
return !!(doc["webkitIsFullScreen"] || doc.fullscreenElement);
}
2
3
requestFullScreen
函数:全屏
function requestFullScreen(element) {
if (element.requestFullscreen) {
element.requestFullscreen();
} else if (element["webkitRequestFullscreen"]) {
element["webkitRequestFullscreen"]();
}
}
2
3
4
5
6
7
requestFullScreenWithKeys
函数:全屏
function requestFullScreenWithKeys(element) {
if (element["webkitRequestFullscreen"]) {
element["webkitRequestFullscreen"]();
} else {
requestFullScreen(element);
}
}
2
3
4
5
6
7
exitFullScreen
函数:调用element.exitFullscreen
退出全屏
function exitFullScreen(doc) {
if (doc.exitFullscreen) {
doc.exitFullscreen();
} else if (doc["webkitExitFullscreen"]) {
doc["webkitExitFullscreen"]();
}
}
2
3
4
5
6
7
# FullScreen
类的主要方法
浏览器的标准规范规定全屏 API 需要用户手动才能触发,不能默认开启全屏。因此默认情况下,只有用户点击按钮交互时,才能操作全屏的相关 API。在上面的构造函数中提到点击按钮会触发handleClick_
方法,该方法内部就是防止了默认事件以及调用了handleFullScreen_
方法。
handleFullScreen_
方法:handleFullScreen
方法先是调用getMap
判断map
是否存在,若不存在则返回;然后调用map.getOwnerDocument()
获取doc
元素,该方法会返回地图容器target
,若target
不存在,则返回document
;然后判断doc
元素上是否支持全屏api
,若不支持,则返回;然后判断当前状态,若是全屏状态,则调用exitFullScreen
退出全屏;否则,判断this.source_
是否有值,若有值,则将其赋值给element
;否则调用map.getTargetElement
获取地图容器元素,最后根据this.keys
调用requestFullScreenWithKeys
或者requestFullScreen
方法使element
全屏。setClassName_
方法:根据当前地图是否是全屏状态设置控件的类名setMap
方法:setMap
方法在Map
类中初始化控件时或者调用addControl
方法添加控件后会调用;FullScreen
类中该方法会先调用this.getMap
获取之前的oldMap
,若存在,则移除target
类型的boundHandleMapTargetChange_
事件,然后调用父类的setMap
方法,调用this.handleMapTargetChange_
方法,判断参数map
是否存在,若存在,则注册target
类型的boundHandleMapTargetChange_
监听事件,boundHandleMapTargetChange_
实际上就是``handleMapTargetChange_
方法:处理地图容器target
发生改变时的方法;该方法内部会先遍历this.documentListeners
中的监听,然后将其置空;再就是调用this.getMap()
获取map
,然后判断map
,若map
存在,则判断是否支持全屏,以此设置全屏控件的样式;然后遍历events
,['fullscreenchange','webkitfullscreenchange','MSFullscreenChange']
添加它们的监听事件handleFullScreenChange_
,并将listen
方法返回的key
值保存在this.documentListeners
中,最后调用一次this.handleFullScreenChange_
方法handleFullScreenChange_
方法:handleFullScreenChange_
方法就是判断全屏状态是否发生改变,若发生变化,则调用setClassName_
方法修改全屏控件的类名,以及调用dispatchEvent
派发enterfullscreen
或者leavefullscreen
类型的事件;最后调用map.updateSize()
方法更新地图的大小。
# 总结
本文主要介绍了 Openlayers 中全屏控件FullScreen
的实现,该控件就是基于HTML5
全屏 API实现的,在元素全屏状态切换时更新地图的大小。