✨ MAJOR FEATURES: • Auto-zoom intelligence với smart bounds fitting • Enhanced 3D GPS markers với pulsing effects • Professional route display với 6-layer rendering • Status-based parking icons với availability indicators • Production-ready build optimizations 🗺️ AUTO-ZOOM FEATURES: • Smart bounds fitting cho GPS + selected parking • Adaptive padding (50px) cho visual balance • Max zoom control (level 16) để tránh quá gần • Dynamic centering khi không có selection 🎨 ENHANCED VISUALS: • 3D GPS marker với multi-layer pulse effects • Advanced parking icons với status colors • Selection highlighting với animation • Dimming system cho non-selected items 🛣️ ROUTE SYSTEM: • OpenRouteService API integration • Multi-layer route rendering (glow, shadow, main, animated) • Real-time distance & duration calculation • Visual route info trong popup 📱 PRODUCTION READY: • SSR safe với dynamic imports • Build errors resolved • Global deployment via Vercel • Optimized performance 🌍 DEPLOYMENT: • Vercel: https://whatever-ctk2auuxr-phong12hexdockworks-projects.vercel.app • Bundle size: 22.8 kB optimized • Global CDN distribution • HTTPS enabled 💾 VERSION CONTROL: • MapView-v2.0.tsx backup created • MAPVIEW_VERSIONS.md documentation • Full version history tracking
105 lines
3.1 KiB
JavaScript
105 lines
3.1 KiB
JavaScript
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var test = require('tape');
|
|
var assert = require('assert');
|
|
|
|
var noop = function() {};
|
|
|
|
var mustCallChecks = [];
|
|
|
|
function runCallChecks(exitCode) {
|
|
if (exitCode !== 0) return;
|
|
|
|
var failed = filter(mustCallChecks, function(context) {
|
|
if ('minimum' in context) {
|
|
context.messageSegment = 'at least ' + context.minimum;
|
|
return context.actual < context.minimum;
|
|
} else {
|
|
context.messageSegment = 'exactly ' + context.exact;
|
|
return context.actual !== context.exact;
|
|
}
|
|
});
|
|
|
|
for (var i = 0; i < failed.length; i++) {
|
|
var context = failed[i];
|
|
console.log('Mismatched %s function calls. Expected %s, actual %d.',
|
|
context.name,
|
|
context.messageSegment,
|
|
context.actual);
|
|
// IE8 has no .stack
|
|
if (context.stack) console.log(context.stack.split('\n').slice(2).join('\n'));
|
|
}
|
|
|
|
assert.strictEqual(failed.length, 0);
|
|
}
|
|
|
|
exports.mustCall = function(fn, exact) {
|
|
return _mustCallInner(fn, exact, 'exact');
|
|
};
|
|
|
|
function _mustCallInner(fn, criteria, field) {
|
|
if (typeof criteria == 'undefined') criteria = 1;
|
|
|
|
if (typeof fn === 'number') {
|
|
criteria = fn;
|
|
fn = noop;
|
|
} else if (fn === undefined) {
|
|
fn = noop;
|
|
}
|
|
|
|
if (typeof criteria !== 'number')
|
|
throw new TypeError('Invalid ' + field + ' value: ' + criteria);
|
|
|
|
var context = {
|
|
actual: 0,
|
|
stack: (new Error()).stack,
|
|
name: fn.name || '<anonymous>'
|
|
};
|
|
|
|
context[field] = criteria;
|
|
|
|
// add the exit listener only once to avoid listener leak warnings
|
|
if (mustCallChecks.length === 0) test.onFinish(function() { runCallChecks(0); });
|
|
|
|
mustCallChecks.push(context);
|
|
|
|
return function() {
|
|
context.actual++;
|
|
return fn.apply(this, arguments);
|
|
};
|
|
}
|
|
|
|
exports.mustNotCall = function(msg) {
|
|
return function mustNotCall() {
|
|
assert.fail(msg || 'function should not have been called');
|
|
};
|
|
};
|
|
|
|
function filter(arr, fn) {
|
|
if (arr.filter) return arr.filter(fn);
|
|
var filtered = [];
|
|
for (var i = 0; i < arr.length; i++) {
|
|
if (fn(arr[i], i, arr)) filtered.push(arr[i]);
|
|
}
|
|
return filtered
|
|
}
|