✨ 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
106 lines
2.8 KiB
JavaScript
106 lines
2.8 KiB
JavaScript
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Tobias Koppers @sokra
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
/**
|
|
* @template T
|
|
* @typedef {Record<string, string | string[] | T>} Item
|
|
*/
|
|
|
|
/**
|
|
* @template T
|
|
* @typedef {(string | Item<T>)[] | Item<T>} ContainerOptionsFormat
|
|
*/
|
|
|
|
/**
|
|
* @template T
|
|
* @template N
|
|
* @param {ContainerOptionsFormat<T>} options options passed by the user
|
|
* @param {(item: string | string[], itemOrKey: string) => N} normalizeSimple normalize a simple item
|
|
* @param {(value: T, key: string) => N} normalizeOptions normalize a complex item
|
|
* @param {(item: string, normalized: N) => void} fn processing function
|
|
* @returns {void}
|
|
*/
|
|
const process = (options, normalizeSimple, normalizeOptions, fn) => {
|
|
/**
|
|
* @param {(string | Item<T>)[]} items items
|
|
*/
|
|
const array = items => {
|
|
for (const item of items) {
|
|
if (typeof item === "string") {
|
|
fn(item, normalizeSimple(item, item));
|
|
} else if (item && typeof item === "object") {
|
|
object(item);
|
|
} else {
|
|
throw new Error("Unexpected options format");
|
|
}
|
|
}
|
|
};
|
|
/**
|
|
* @param {Item<T>} obj an object
|
|
*/
|
|
const object = obj => {
|
|
for (const [key, value] of Object.entries(obj)) {
|
|
if (typeof value === "string" || Array.isArray(value)) {
|
|
fn(key, normalizeSimple(value, key));
|
|
} else {
|
|
fn(key, normalizeOptions(value, key));
|
|
}
|
|
}
|
|
};
|
|
if (!options) {
|
|
// Do nothing
|
|
} else if (Array.isArray(options)) {
|
|
array(options);
|
|
} else if (typeof options === "object") {
|
|
object(options);
|
|
} else {
|
|
throw new Error("Unexpected options format");
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @template T
|
|
* @template R
|
|
* @param {ContainerOptionsFormat<T>} options options passed by the user
|
|
* @param {(item: string | string[], itemOrKey: string) => R} normalizeSimple normalize a simple item
|
|
* @param {(value: T, key: string) => R} normalizeOptions normalize a complex item
|
|
* @returns {[string, R][]} parsed options
|
|
*/
|
|
const parseOptions = (options, normalizeSimple, normalizeOptions) => {
|
|
/** @type {[string, R][]} */
|
|
const items = [];
|
|
process(options, normalizeSimple, normalizeOptions, (key, value) => {
|
|
items.push([key, value]);
|
|
});
|
|
return items;
|
|
};
|
|
|
|
/**
|
|
* @template T
|
|
* @param {string} scope scope name
|
|
* @param {ContainerOptionsFormat<T>} options options passed by the user
|
|
* @returns {Record<string, string | string[] | T>} options to spread or pass
|
|
*/
|
|
const scope = (scope, options) => {
|
|
/** @type {Record<string, string | string[] | T>} */
|
|
const obj = {};
|
|
process(
|
|
options,
|
|
item => /** @type {string | string[] | T} */ (item),
|
|
item => /** @type {string | string[] | T} */ (item),
|
|
(key, value) => {
|
|
obj[
|
|
key.startsWith("./") ? `${scope}${key.slice(1)}` : `${scope}/${key}`
|
|
] = value;
|
|
}
|
|
);
|
|
return obj;
|
|
};
|
|
|
|
module.exports.parseOptions = parseOptions;
|
|
module.exports.scope = scope;
|