Util工具方法
# 概述
Leaflet 框架中的 Util
模块,提供了许多实用工具函数,用于简化开发、处理常见任务(如对象操作、函数绑定、字符串处理等),并支持跨浏览器兼容性.
# 源码分析
# 源码实现如下
export function extend(dest) {
var i, j, len, src;
for (j = 1, len = arguments.length; j < len; j++) {
src = arguments[j];
for (i in src) {
dest[i] = src[i];
}
}
return dest;
}
export var create =
Object.create ||
(function () {
function F() {}
return function (proto) {
F.prototype = proto;
return new F();
};
})();
export function bind(fn, obj) {
var slice = Array.prototype.slice;
if (fn.bind) {
return fn.bind.apply(fn, slice.call(arguments, 1));
}
var args = slice.call(arguments, 2);
return function () {
return fn.apply(
obj,
args.length ? args.concat(slice.call(arguments)) : arguments
);
};
}
export var lasted = 0;
export function stamp(obj) {
if (!("_leaflet_id" in obj)) {
obj["_leaflet_id"] = ++lasted;
}
return obj._leaflet_id;
}
export function throttle(fn, time, context) {
var lock, args, wrapperFn, later;
later = function () {
lock = false;
if (args) {
wrapperFn.apply(context, args);
args = false;
}
};
wrapperFn = function () {
if (lock) {
args = arguments;
} else {
fn.apply(context, arguments);
setTimeout(later, time);
lock = true;
}
};
return wrapperFn;
}
export function wrapNum(x, range, includeMax) {
var max = range[1],
min = range[0],
d = max - min;
return x === max && includeMax ? x : ((((x - min) % d) + d) % d) + min;
}
export function falseFn() {
return false;
}
export function formatNum(num, precision) {
if (precision === false) {
return num;
}
var pow = Math.pow(10, precision === undefined ? 6 : precision);
return Math.round(num * pow) / pow;
}
export function trim(str) {
return str.trim ? str.trim() : str.replace(/^\s+|\s+$/g, "");
}
export function splitWords(str) {
return trim(str).split(/\s+/);
}
export function setOptions(obj, options) {
if (!Object.prototype.hasOwnProperty.call(obj, "options")) {
obj.options = obj.options ? create(obj.options) : {};
}
for (var i in options) {
obj.options[i] = options[i];
}
return obj.options;
}
export function getParamString(obj, existingUrl, uppercase) {
var params = [];
for (var i in obj) {
params.push(
encodeURIComponent(uppercase ? i.toUpperCase() : i) +
"=" +
encodeURIComponent(obj[i])
);
}
return (
(!existingUrl || existingUrl.indexOf("?") === -1 ? "?" : "&") +
params.join("&")
);
}
var templateRe = /\{*([\w_-]+)*\}/g;
export function template(str, data) {
return str.replace(templateRe, function (str, key) {
var value = data[key];
if (value === undefined) {
throw new Error("No value provided for variable " + str);
} else if (typeof value === "function") {
value = value(data);
}
return value;
});
}
export var isArray =
Array.isArray ||
function (obj) {
return Object.prototype.toString.call(obj) === "[object Array]";
};
export function indexOf(arr, el) {
for (var i = 0, len = arr.length; i < len; i++) {
if (arr[i] === el) {
return i;
}
}
return -1;
}
export var emptyImageUrl =
"";
function getPrefixed(name) {
return window["webkit" + name] || window["moz" + name] || window["ms" + name];
}
var lastTime = 0;
function timeoutDefer(fn) {
var time = +new Date(),
timeToCall = Math.max(0, 16 - (time - lastTime));
lastTime = time + timeToCall;
return window.setTimeout(fn, timeToCall);
}
export var requestFn =
window.requestAnimationFrame ||
getPrefixed("RequestAnimationFrame") ||
timeoutDefer;
export var cancelFn =
window.cancelAnimationFrame ||
getPrefixed("CancelAnimationFrame") ||
getPrefixed("CancelRequestAnimationFrame") ||
function (id) {
window.clearTimeout(id);
};
export function requestAnimFrame(fn, context, immediate) {
if (immediate && requestFn === timeoutDefer) {
fn.call(context);
} else {
return requestFn.call(window, bind(fn, context));
}
}
export function cancelAnimFrame(id) {
if (id) {
cancelFn.call(window, id);
}
}
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
# 函数详细介绍
extend(dest,...src)
功能:将多个对象的属性合并到目标对象中.
参数:
dest
:目标对象src
:源对象(可传入多个)
返回值:合并后的目标对象
示例
const obj = { a: 1 };
Util.extend(obj, { b: 2 }, { c: 3 });
console.log(obj); // { a: 1, b: 2, c: 3 }
2
3
create(proto)
功能:创建一个新对象,并继承指定原型.
参数:
proto
:新对象的原型
返回值:新创建的对象.
示例
const obj = { a: 1 };
const newObj = Util.create(obj);
console.log(newObj.__proto__ === obj); // true
2
3
bind(fn,obj,...args)
功能:创建一个新函数,该函数的
this
值被绑定到指定的对象,同时可以传入额外的参数.参数:
fn
:要绑定的函数obj
:绑定的对象args
:额外的参数(可选)
返回值:绑定后的函数
示例
const obj = { value: 10 };
const fn = function (a, b) { return this.value + a + b; };
const boundFn = Util.bind(fn, obj, 1);
console.log(boundFn(2)); // 13
2
3
4
- stamp(obj)
功能:为对象生成唯一标识符(ID),如果对象已有标识符则直接返回.
参数:
obj
:需要生成标识符的对象
返回值: 对象的唯一标识符
示例
const obj = {};
console.log(Util.stamp(obj)); // 1
console.log(Util.stamp(obj)); // 1 (相同对象返回相同 ID)
2
3
- throttle(fn,time,context)
功能:节流函数,限制函数在指定时间间隔内最多只执行一次.
参数:
fn
:需要节流的函数time
:时间间隔(毫秒)context
:函数的上下文(this
指向)
返回值:节流后的函数
示例
const throttledFn = Util.throttle(() => console.log('Throttled!'), 1000);
window.addEventListener('resize', throttledFn);
2
- wrapNum(x,range,includeMax)
功能:将数值限制在指定范围内,支持循环(例如,经度范围在
-180
到180
)参数
x
:需要处理的数值range
:范围数组,如[min,max]
includeMax
:是否包含最大值,默认为false
返回值:处理后的数值
示例
console.log(Util.wrapNum(190, [-180, 180])); // -170
- falseFn()
功能:返回
false
的函数返回值:
false
示例
console.log(Util.falseFn()); // false
- formatNum(num,precision)
功能:格式化数字,保留指定精度
参数
num
:需要格式化的数字precision
:小数位数(可选,默认为6
)
返回值:格式化后的数值
示例
console.log(Util.formatNum(3.1415926535, 2)); // 3.14
- trim(str)
功能:去除字符串两端的空白字符。
参数:
str
:需要处理的字符串。
返回值:处理后的字符串。
示例:
console.log(Util.trim(' Hello, Leaflet! ')); // 'Hello, Leaflet!'
- splitWords(str)
功能:将字符串按空白字符分割为单词数组。
参数:
str
:需要处理的字符串。
返回值:单词数组。
示例:
console.log(Util.splitWords('Hello World')); // ['Hello', 'World']
- setOptions(obj, options)
- 功能:为对象设置选项(
options
),如果options
不存在则创建。 - 参数:
obj
:目标对象。options
:需要设置的选项。
- 返回值:设置后的
options
对象。
- 示例:
const obj = {};
Util.setOptions(obj, { a: 1 });
console.log(obj.options); // { a: 1 }
2
3
- getParamString(obj, existingUrl, uppercase)
功能:将对象转换为
URL
参数字符串。 参数:
obj
:需要转换的对象。existingUrl
:已存在的URL
(可选)。uppercase
:是否将参数名转换为大写(可选)。返回值:
URL
参数字符串。 示例:
console.log(Util.getParamString({ a: 1, b: 2 })); // '?a=1&b=2'
- template(str, data)
- 功能:将字符串中的占位符替换为数据。
- 参数:
str
:模板字符串。data
:数据对象。
- 返回值:替换后的字符串。
- 示例:
console.log(Util.template('Hello, {name}!', { name: 'Leaflet' })); // 'Hello, Leaflet!'
- isArray(obj)
- 功能:检查对象是否为数组。
- 参数:
obj
:需要检查的对象。
- 返回值:布尔值。
- 示例:
console.log(Util.isArray([1, 2, 3])); // true
- indexOf(arr, el)
功能:查找元素在数组中的索引。
参数:
arr
:数组。el
:需要查找的元素。 返回值:元素的索引(如果未找到则返回 -1)。
示例:
console.log(Util.indexOf([1, 2, 3], 2)); // 1
- requestAnimFrame(fn, context, immediate)
功能:请求动画帧,支持立即执行。
参数:
fn
:需要执行的函数。context
:函数的上下文(this
指向)。immediate
:是否立即执行(可选)。
返回值:动画帧 ID。
示例:
Util.requestAnimFrame(() => console.log('Frame!'));
- cancelAnimFrame(id)
- 功能:取消动画帧。
- 参数:
id
:动画帧 ID。 - 示例:
const id = Util.requestAnimFrame(() => console.log('Frame!'));
Util.cancelAnimFrame(id);
2
# 总结
Util
模块是 Leaflet 的核心工具模块,提供了许多实用函数,用于简化开发、处理常见任务(如对象操作、函数绑定、字符串处理等),并支持跨浏览器兼容性。这些函数在 Leaflet 的内部实现中被广泛使用,同时也可以被开发者直接调用