web 3d图形渲染器
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

377 lines
10 KiB

  1. /**
  2. * lodash (Custom Build) <https://lodash.com/>
  3. * Build: `lodash modularize exports="npm" -o ./`
  4. * Copyright jQuery Foundation and other contributors <https://jquery.org/>
  5. * Released under MIT license <https://lodash.com/license>
  6. * Based on Underscore.js 1.8.3 <http://underscorejs.org/LICENSE>
  7. * Copyright Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
  8. */
  9. /** Used as the `TypeError` message for "Functions" methods. */
  10. var FUNC_ERROR_TEXT = 'Expected a function';
  11. /** Used as references for various `Number` constants. */
  12. var NAN = 0 / 0;
  13. /** `Object#toString` result references. */
  14. var symbolTag = '[object Symbol]';
  15. /** Used to match leading and trailing whitespace. */
  16. var reTrim = /^\s+|\s+$/g;
  17. /** Used to detect bad signed hexadecimal string values. */
  18. var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;
  19. /** Used to detect binary string values. */
  20. var reIsBinary = /^0b[01]+$/i;
  21. /** Used to detect octal string values. */
  22. var reIsOctal = /^0o[0-7]+$/i;
  23. /** Built-in method references without a dependency on `root`. */
  24. var freeParseInt = parseInt;
  25. /** Detect free variable `global` from Node.js. */
  26. var freeGlobal = typeof global == 'object' && global && global.Object === Object && global;
  27. /** Detect free variable `self`. */
  28. var freeSelf = typeof self == 'object' && self && self.Object === Object && self;
  29. /** Used as a reference to the global object. */
  30. var root = freeGlobal || freeSelf || Function('return this')();
  31. /** Used for built-in method references. */
  32. var objectProto = Object.prototype;
  33. /**
  34. * Used to resolve the
  35. * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
  36. * of values.
  37. */
  38. var objectToString = objectProto.toString;
  39. /* Built-in method references for those with the same name as other `lodash` methods. */
  40. var nativeMax = Math.max,
  41. nativeMin = Math.min;
  42. /**
  43. * Gets the timestamp of the number of milliseconds that have elapsed since
  44. * the Unix epoch (1 January 1970 00:00:00 UTC).
  45. *
  46. * @static
  47. * @memberOf _
  48. * @since 2.4.0
  49. * @category Date
  50. * @returns {number} Returns the timestamp.
  51. * @example
  52. *
  53. * _.defer(function(stamp) {
  54. * console.log(_.now() - stamp);
  55. * }, _.now());
  56. * // => Logs the number of milliseconds it took for the deferred invocation.
  57. */
  58. var now = function() {
  59. return root.Date.now();
  60. };
  61. /**
  62. * Creates a debounced function that delays invoking `func` until after `wait`
  63. * milliseconds have elapsed since the last time the debounced function was
  64. * invoked. The debounced function comes with a `cancel` method to cancel
  65. * delayed `func` invocations and a `flush` method to immediately invoke them.
  66. * Provide `options` to indicate whether `func` should be invoked on the
  67. * leading and/or trailing edge of the `wait` timeout. The `func` is invoked
  68. * with the last arguments provided to the debounced function. Subsequent
  69. * calls to the debounced function return the result of the last `func`
  70. * invocation.
  71. *
  72. * **Note:** If `leading` and `trailing` options are `true`, `func` is
  73. * invoked on the trailing edge of the timeout only if the debounced function
  74. * is invoked more than once during the `wait` timeout.
  75. *
  76. * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
  77. * until to the next tick, similar to `setTimeout` with a timeout of `0`.
  78. *
  79. * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
  80. * for details over the differences between `_.debounce` and `_.throttle`.
  81. *
  82. * @static
  83. * @memberOf _
  84. * @since 0.1.0
  85. * @category Function
  86. * @param {Function} func The function to debounce.
  87. * @param {number} [wait=0] The number of milliseconds to delay.
  88. * @param {Object} [options={}] The options object.
  89. * @param {boolean} [options.leading=false]
  90. * Specify invoking on the leading edge of the timeout.
  91. * @param {number} [options.maxWait]
  92. * The maximum time `func` is allowed to be delayed before it's invoked.
  93. * @param {boolean} [options.trailing=true]
  94. * Specify invoking on the trailing edge of the timeout.
  95. * @returns {Function} Returns the new debounced function.
  96. * @example
  97. *
  98. * // Avoid costly calculations while the window size is in flux.
  99. * jQuery(window).on('resize', _.debounce(calculateLayout, 150));
  100. *
  101. * // Invoke `sendMail` when clicked, debouncing subsequent calls.
  102. * jQuery(element).on('click', _.debounce(sendMail, 300, {
  103. * 'leading': true,
  104. * 'trailing': false
  105. * }));
  106. *
  107. * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
  108. * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
  109. * var source = new EventSource('/stream');
  110. * jQuery(source).on('message', debounced);
  111. *
  112. * // Cancel the trailing debounced invocation.
  113. * jQuery(window).on('popstate', debounced.cancel);
  114. */
  115. function debounce(func, wait, options) {
  116. var lastArgs,
  117. lastThis,
  118. maxWait,
  119. result,
  120. timerId,
  121. lastCallTime,
  122. lastInvokeTime = 0,
  123. leading = false,
  124. maxing = false,
  125. trailing = true;
  126. if (typeof func != 'function') {
  127. throw new TypeError(FUNC_ERROR_TEXT);
  128. }
  129. wait = toNumber(wait) || 0;
  130. if (isObject(options)) {
  131. leading = !!options.leading;
  132. maxing = 'maxWait' in options;
  133. maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;
  134. trailing = 'trailing' in options ? !!options.trailing : trailing;
  135. }
  136. function invokeFunc(time) {
  137. var args = lastArgs,
  138. thisArg = lastThis;
  139. lastArgs = lastThis = undefined;
  140. lastInvokeTime = time;
  141. result = func.apply(thisArg, args);
  142. return result;
  143. }
  144. function leadingEdge(time) {
  145. // Reset any `maxWait` timer.
  146. lastInvokeTime = time;
  147. // Start the timer for the trailing edge.
  148. timerId = setTimeout(timerExpired, wait);
  149. // Invoke the leading edge.
  150. return leading ? invokeFunc(time) : result;
  151. }
  152. function remainingWait(time) {
  153. var timeSinceLastCall = time - lastCallTime,
  154. timeSinceLastInvoke = time - lastInvokeTime,
  155. result = wait - timeSinceLastCall;
  156. return maxing ? nativeMin(result, maxWait - timeSinceLastInvoke) : result;
  157. }
  158. function shouldInvoke(time) {
  159. var timeSinceLastCall = time - lastCallTime,
  160. timeSinceLastInvoke = time - lastInvokeTime;
  161. // Either this is the first call, activity has stopped and we're at the
  162. // trailing edge, the system time has gone backwards and we're treating
  163. // it as the trailing edge, or we've hit the `maxWait` limit.
  164. return (lastCallTime === undefined || (timeSinceLastCall >= wait) ||
  165. (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait));
  166. }
  167. function timerExpired() {
  168. var time = now();
  169. if (shouldInvoke(time)) {
  170. return trailingEdge(time);
  171. }
  172. // Restart the timer.
  173. timerId = setTimeout(timerExpired, remainingWait(time));
  174. }
  175. function trailingEdge(time) {
  176. timerId = undefined;
  177. // Only invoke if we have `lastArgs` which means `func` has been
  178. // debounced at least once.
  179. if (trailing && lastArgs) {
  180. return invokeFunc(time);
  181. }
  182. lastArgs = lastThis = undefined;
  183. return result;
  184. }
  185. function cancel() {
  186. if (timerId !== undefined) {
  187. clearTimeout(timerId);
  188. }
  189. lastInvokeTime = 0;
  190. lastArgs = lastCallTime = lastThis = timerId = undefined;
  191. }
  192. function flush() {
  193. return timerId === undefined ? result : trailingEdge(now());
  194. }
  195. function debounced() {
  196. var time = now(),
  197. isInvoking = shouldInvoke(time);
  198. lastArgs = arguments;
  199. lastThis = this;
  200. lastCallTime = time;
  201. if (isInvoking) {
  202. if (timerId === undefined) {
  203. return leadingEdge(lastCallTime);
  204. }
  205. if (maxing) {
  206. // Handle invocations in a tight loop.
  207. timerId = setTimeout(timerExpired, wait);
  208. return invokeFunc(lastCallTime);
  209. }
  210. }
  211. if (timerId === undefined) {
  212. timerId = setTimeout(timerExpired, wait);
  213. }
  214. return result;
  215. }
  216. debounced.cancel = cancel;
  217. debounced.flush = flush;
  218. return debounced;
  219. }
  220. /**
  221. * Checks if `value` is the
  222. * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
  223. * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
  224. *
  225. * @static
  226. * @memberOf _
  227. * @since 0.1.0
  228. * @category Lang
  229. * @param {*} value The value to check.
  230. * @returns {boolean} Returns `true` if `value` is an object, else `false`.
  231. * @example
  232. *
  233. * _.isObject({});
  234. * // => true
  235. *
  236. * _.isObject([1, 2, 3]);
  237. * // => true
  238. *
  239. * _.isObject(_.noop);
  240. * // => true
  241. *
  242. * _.isObject(null);
  243. * // => false
  244. */
  245. function isObject(value) {
  246. var type = typeof value;
  247. return !!value && (type == 'object' || type == 'function');
  248. }
  249. /**
  250. * Checks if `value` is object-like. A value is object-like if it's not `null`
  251. * and has a `typeof` result of "object".
  252. *
  253. * @static
  254. * @memberOf _
  255. * @since 4.0.0
  256. * @category Lang
  257. * @param {*} value The value to check.
  258. * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
  259. * @example
  260. *
  261. * _.isObjectLike({});
  262. * // => true
  263. *
  264. * _.isObjectLike([1, 2, 3]);
  265. * // => true
  266. *
  267. * _.isObjectLike(_.noop);
  268. * // => false
  269. *
  270. * _.isObjectLike(null);
  271. * // => false
  272. */
  273. function isObjectLike(value) {
  274. return !!value && typeof value == 'object';
  275. }
  276. /**
  277. * Checks if `value` is classified as a `Symbol` primitive or object.
  278. *
  279. * @static
  280. * @memberOf _
  281. * @since 4.0.0
  282. * @category Lang
  283. * @param {*} value The value to check.
  284. * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
  285. * @example
  286. *
  287. * _.isSymbol(Symbol.iterator);
  288. * // => true
  289. *
  290. * _.isSymbol('abc');
  291. * // => false
  292. */
  293. function isSymbol(value) {
  294. return typeof value == 'symbol' ||
  295. (isObjectLike(value) && objectToString.call(value) == symbolTag);
  296. }
  297. /**
  298. * Converts `value` to a number.
  299. *
  300. * @static
  301. * @memberOf _
  302. * @since 4.0.0
  303. * @category Lang
  304. * @param {*} value The value to process.
  305. * @returns {number} Returns the number.
  306. * @example
  307. *
  308. * _.toNumber(3.2);
  309. * // => 3.2
  310. *
  311. * _.toNumber(Number.MIN_VALUE);
  312. * // => 5e-324
  313. *
  314. * _.toNumber(Infinity);
  315. * // => Infinity
  316. *
  317. * _.toNumber('3.2');
  318. * // => 3.2
  319. */
  320. function toNumber(value) {
  321. if (typeof value == 'number') {
  322. return value;
  323. }
  324. if (isSymbol(value)) {
  325. return NAN;
  326. }
  327. if (isObject(value)) {
  328. var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
  329. value = isObject(other) ? (other + '') : other;
  330. }
  331. if (typeof value != 'string') {
  332. return value === 0 ? value : +value;
  333. }
  334. value = value.replace(reTrim, '');
  335. var isBinary = reIsBinary.test(value);
  336. return (isBinary || reIsOctal.test(value))
  337. ? freeParseInt(value.slice(2), isBinary ? 2 : 8)
  338. : (reIsBadHex.test(value) ? NAN : +value);
  339. }
  340. module.exports = debounce;