Files
Laca-City/backend/node_modules/webpack/lib/util/TupleSet.js
PhongPham c65cc97a33 🎯 MapView v2.0 - Global Deployment Ready
 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
2025-07-20 19:52:16 +07:00

181 lines
4.0 KiB
JavaScript

/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
/**
* @template K
* @template V
* @typedef {Map<K, InnerMap<K, V> | Set<V>>} InnerMap
*/
/**
* @template T
* @template V
*/
class TupleSet {
/**
* @param {Iterable<[T, V, ...EXPECTED_ANY]>=} init init
*/
constructor(init) {
/** @type {InnerMap<T, V>} */
this._map = new Map();
this.size = 0;
if (init) {
for (const tuple of init) {
this.add(...tuple);
}
}
}
/**
* @param {[T, V, ...EXPECTED_ANY]} args tuple
* @returns {void}
*/
add(...args) {
let map = this._map;
for (let i = 0; i < args.length - 2; i++) {
const arg = args[i];
const innerMap = map.get(arg);
if (innerMap === undefined) {
map.set(arg, (map = new Map()));
} else {
map = /** @type {InnerMap<T, V>} */ (innerMap);
}
}
const beforeLast = args[args.length - 2];
let set = /** @type {Set<V>} */ (map.get(beforeLast));
if (set === undefined) {
map.set(beforeLast, (set = new Set()));
}
const last = args[args.length - 1];
this.size -= set.size;
set.add(last);
this.size += set.size;
}
/**
* @param {[T, V, ...EXPECTED_ANY]} args tuple
* @returns {boolean} true, if the tuple is in the Set
*/
has(...args) {
let map = this._map;
for (let i = 0; i < args.length - 2; i++) {
const arg = args[i];
map = /** @type {InnerMap<T, V>} */ (map.get(arg));
if (map === undefined) {
return false;
}
}
const beforeLast = args[args.length - 2];
const set = map.get(beforeLast);
if (set === undefined) {
return false;
}
const last = args[args.length - 1];
return set.has(last);
}
/**
* @param {[T, V, ...EXPECTED_ANY]} args tuple
* @returns {void}
*/
delete(...args) {
let map = this._map;
for (let i = 0; i < args.length - 2; i++) {
const arg = args[i];
map = /** @type {InnerMap<T, V>} */ (map.get(arg));
if (map === undefined) {
return;
}
}
const beforeLast = args[args.length - 2];
const set = map.get(beforeLast);
if (set === undefined) {
return;
}
const last = args[args.length - 1];
this.size -= set.size;
set.delete(last);
this.size += set.size;
}
/**
* @returns {Iterator<[T, V, ...EXPECTED_ANY]>} iterator
*/
[Symbol.iterator]() {
// This is difficult to type because we can have a map inside a map inside a map, etc. where the end is a set (each key is an argument)
// But in basic use we only have 2 arguments in our methods, so we have `Map<K, Set<V>>`
/** @type {MapIterator<[T, InnerMap<T, V> | Set<V>]>[]} */
const iteratorStack = [];
/** @type {[T?, V?, ...EXPECTED_ANY]} */
const tuple = [];
/** @type {SetIterator<V> | undefined} */
let currentSetIterator;
/**
* @param {MapIterator<[T, InnerMap<T, V> | Set<V>]>} it iterator
* @returns {boolean} result
*/
const next = it => {
const result = it.next();
if (result.done) {
if (iteratorStack.length === 0) return false;
tuple.pop();
return next(
/** @type {MapIterator<[T, InnerMap<T, V> | Set<V>]>} */
(iteratorStack.pop())
);
}
const [key, value] = result.value;
iteratorStack.push(it);
tuple.push(key);
if (value instanceof Set) {
currentSetIterator = value[Symbol.iterator]();
return true;
}
return next(value[Symbol.iterator]());
};
next(this._map[Symbol.iterator]());
return {
next() {
while (currentSetIterator) {
const result = currentSetIterator.next();
if (result.done) {
tuple.pop();
if (
!next(
/** @type {MapIterator<[T, InnerMap<T, V> | Set<V>]>} */
(iteratorStack.pop())
)
) {
currentSetIterator = undefined;
}
} else {
return {
done: false,
value:
/* eslint-disable unicorn/prefer-spread */
/** @type {[T, V, ...EXPECTED_ANY]} */
(tuple.concat(result.value))
};
}
}
return { done: true, value: undefined };
}
};
}
}
module.exports = TupleSet;