Add semantic token highlighting for 'that' variable and comment file references in VS Code extension Add Phone_Text_Input and Currency_Input components with formatting utilities Implement client widgets, form standardization, and soft delete functionality Add modal scroll lock and update documentation Implement comprehensive modal system with form integration and validation Fix modal component instantiation using jQuery plugin API Implement modal system with responsive sizing, queuing, and validation support Implement form submission with validation, error handling, and loading states Implement country/state selectors with dynamic data loading and Bootstrap styling Revert Rsx::Route() highlighting in Blade/PHP files Target specific PHP scopes for Rsx::Route() highlighting in Blade Expand injection selector for Rsx::Route() highlighting Add custom syntax highlighting for Rsx::Route() and Rsx.Route() calls Update jqhtml packages to v2.2.165 Add bundle path validation for common mistakes (development mode only) Create Ajax_Select_Input widget and Rsx_Reference_Data controller Create Country_Select_Input widget with default country support Initialize Tom Select on Select_Input widgets Add Tom Select bundle for enhanced select dropdowns Implement ISO 3166 geographic data system for country/region selection Implement widget-based form system with disabled state support 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
351 lines
9.1 KiB
JavaScript
Executable File
351 lines
9.1 KiB
JavaScript
Executable File
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Tobias Koppers @sokra
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
const util = require("util");
|
|
|
|
/** @type {Map<string, () => void>} */
|
|
const deprecationCache = new Map();
|
|
|
|
/**
|
|
* @typedef {object} FakeHookMarker
|
|
* @property {true} _fakeHook it's a fake hook
|
|
*/
|
|
|
|
/**
|
|
* @template T
|
|
* @typedef {T & FakeHookMarker} FakeHook<T>
|
|
*/
|
|
|
|
/**
|
|
* @param {string} message deprecation message
|
|
* @param {string} code deprecation code
|
|
* @returns {() => void} function to trigger deprecation
|
|
*/
|
|
const createDeprecation = (message, code) => {
|
|
const cached = deprecationCache.get(message);
|
|
if (cached !== undefined) return cached;
|
|
const fn = util.deprecate(
|
|
() => {},
|
|
message,
|
|
`DEP_WEBPACK_DEPRECATION_${code}`
|
|
);
|
|
deprecationCache.set(message, fn);
|
|
return fn;
|
|
};
|
|
|
|
/** @typedef {"concat" | "entry" | "filter" | "find" | "findIndex" | "includes" | "indexOf" | "join" | "lastIndexOf" | "map" | "reduce" | "reduceRight" | "slice" | "some"} COPY_METHODS_NAMES */
|
|
|
|
/** @type {COPY_METHODS_NAMES[]} */
|
|
const COPY_METHODS = [
|
|
"concat",
|
|
"entry",
|
|
"filter",
|
|
"find",
|
|
"findIndex",
|
|
"includes",
|
|
"indexOf",
|
|
"join",
|
|
"lastIndexOf",
|
|
"map",
|
|
"reduce",
|
|
"reduceRight",
|
|
"slice",
|
|
"some"
|
|
];
|
|
|
|
/** @typedef {"copyWithin" | "entries" | "fill" | "keys" | "pop" | "reverse" | "shift" | "splice" | "sort" | "unshift"} DISABLED_METHODS_NAMES */
|
|
|
|
/** @type {DISABLED_METHODS_NAMES[]} */
|
|
const DISABLED_METHODS = [
|
|
"copyWithin",
|
|
"entries",
|
|
"fill",
|
|
"keys",
|
|
"pop",
|
|
"reverse",
|
|
"shift",
|
|
"splice",
|
|
"sort",
|
|
"unshift"
|
|
];
|
|
|
|
/**
|
|
* @template T
|
|
* @typedef {Set<T> & { [Symbol.isConcatSpreadable]: boolean } & { push: (...items: T[]) => void, length?: number } & { [P in DISABLED_METHODS_NAMES]: () => void } & { [P in COPY_METHODS_NAMES]: P extends keyof Array<T> ? () => Pick<Array<T>, P> : never }} SetWithDeprecatedArrayMethods
|
|
*/
|
|
|
|
/**
|
|
* @template T
|
|
* @param {Set<T>} set new set
|
|
* @param {string} name property name
|
|
* @returns {void}
|
|
*/
|
|
module.exports.arrayToSetDeprecation = (set, name) => {
|
|
for (const method of COPY_METHODS) {
|
|
if (/** @type {SetWithDeprecatedArrayMethods<T>} */ (set)[method]) continue;
|
|
const d = createDeprecation(
|
|
`${name} was changed from Array to Set (using Array method '${method}' is deprecated)`,
|
|
"ARRAY_TO_SET"
|
|
);
|
|
/** @type {EXPECTED_ANY} */
|
|
(set)[method] =
|
|
// eslint-disable-next-line func-names
|
|
function () {
|
|
d();
|
|
// eslint-disable-next-line unicorn/prefer-spread
|
|
const array = Array.from(this);
|
|
return Array.prototype[
|
|
/** @type {keyof COPY_METHODS} */ (method)
|
|
].apply(
|
|
array,
|
|
// eslint-disable-next-line prefer-rest-params
|
|
arguments
|
|
);
|
|
};
|
|
}
|
|
const dPush = createDeprecation(
|
|
`${name} was changed from Array to Set (using Array method 'push' is deprecated)`,
|
|
"ARRAY_TO_SET_PUSH"
|
|
);
|
|
const dLength = createDeprecation(
|
|
`${name} was changed from Array to Set (using Array property 'length' is deprecated)`,
|
|
"ARRAY_TO_SET_LENGTH"
|
|
);
|
|
const dIndexer = createDeprecation(
|
|
`${name} was changed from Array to Set (indexing Array is deprecated)`,
|
|
"ARRAY_TO_SET_INDEXER"
|
|
);
|
|
/** @type {SetWithDeprecatedArrayMethods<T>} */
|
|
(set).push = function push() {
|
|
dPush();
|
|
// eslint-disable-next-line prefer-rest-params, unicorn/prefer-spread
|
|
for (const item of Array.from(arguments)) {
|
|
this.add(item);
|
|
}
|
|
return this.size;
|
|
};
|
|
for (const method of DISABLED_METHODS) {
|
|
if (/** @type {SetWithDeprecatedArrayMethods<T>} */ (set)[method]) continue;
|
|
|
|
/** @type {SetWithDeprecatedArrayMethods<T>} */
|
|
(set)[method] = () => {
|
|
throw new Error(
|
|
`${name} was changed from Array to Set (using Array method '${method}' is not possible)`
|
|
);
|
|
};
|
|
}
|
|
/**
|
|
* @param {number} index index
|
|
* @returns {() => T | undefined} value
|
|
*/
|
|
const createIndexGetter = (index) => {
|
|
/**
|
|
* @this {Set<T>} a Set
|
|
* @returns {T | undefined} the value at this location
|
|
*/
|
|
// eslint-disable-next-line func-style
|
|
const fn = function () {
|
|
dIndexer();
|
|
let i = 0;
|
|
for (const item of this) {
|
|
if (i++ === index) return item;
|
|
}
|
|
};
|
|
return fn;
|
|
};
|
|
/**
|
|
* @param {number} index index
|
|
*/
|
|
const defineIndexGetter = (index) => {
|
|
Object.defineProperty(set, index, {
|
|
get: createIndexGetter(index),
|
|
set(value) {
|
|
throw new Error(
|
|
`${name} was changed from Array to Set (indexing Array with write is not possible)`
|
|
);
|
|
}
|
|
});
|
|
};
|
|
defineIndexGetter(0);
|
|
let indexerDefined = 1;
|
|
Object.defineProperty(set, "length", {
|
|
get() {
|
|
dLength();
|
|
const length = this.size;
|
|
for (indexerDefined; indexerDefined < length + 1; indexerDefined++) {
|
|
defineIndexGetter(indexerDefined);
|
|
}
|
|
return length;
|
|
},
|
|
set(value) {
|
|
throw new Error(
|
|
`${name} was changed from Array to Set (writing to Array property 'length' is not possible)`
|
|
);
|
|
}
|
|
});
|
|
/** @type {SetWithDeprecatedArrayMethods<T>} */
|
|
(set)[Symbol.isConcatSpreadable] = true;
|
|
};
|
|
|
|
/**
|
|
* @template T
|
|
* @param {string} name name
|
|
* @returns {{ new <T = any>(values?: ReadonlyArray<T> | null): SetDeprecatedArray<T> }} SetDeprecatedArray
|
|
*/
|
|
module.exports.createArrayToSetDeprecationSet = (name) => {
|
|
let initialized = false;
|
|
|
|
/**
|
|
* @template T
|
|
*/
|
|
class SetDeprecatedArray extends Set {
|
|
/**
|
|
* @param {ReadonlyArray<T> | null=} items items
|
|
*/
|
|
constructor(items) {
|
|
super(items);
|
|
if (!initialized) {
|
|
initialized = true;
|
|
module.exports.arrayToSetDeprecation(
|
|
/** @type {SetWithDeprecatedArrayMethods<T>} */
|
|
(SetDeprecatedArray.prototype),
|
|
name
|
|
);
|
|
}
|
|
}
|
|
}
|
|
return SetDeprecatedArray;
|
|
};
|
|
|
|
/**
|
|
* @template {object} T
|
|
* @param {T} fakeHook fake hook implementation
|
|
* @param {string=} message deprecation message (not deprecated when unset)
|
|
* @param {string=} code deprecation code (not deprecated when unset)
|
|
* @returns {FakeHook<T>} fake hook which redirects
|
|
*/
|
|
module.exports.createFakeHook = (fakeHook, message, code) => {
|
|
if (message && code) {
|
|
fakeHook = deprecateAllProperties(fakeHook, message, code);
|
|
}
|
|
return Object.freeze(
|
|
Object.assign(fakeHook, { _fakeHook: /** @type {true} */ (true) })
|
|
);
|
|
};
|
|
|
|
/**
|
|
* @template T
|
|
* @param {T} obj object
|
|
* @param {string} message deprecation message
|
|
* @param {string} code deprecation code
|
|
* @returns {T} object with property access deprecated
|
|
*/
|
|
const deprecateAllProperties = (obj, message, code) => {
|
|
const newObj = {};
|
|
const descriptors = Object.getOwnPropertyDescriptors(obj);
|
|
for (const name of Object.keys(descriptors)) {
|
|
const descriptor = descriptors[name];
|
|
if (typeof descriptor.value === "function") {
|
|
Object.defineProperty(newObj, name, {
|
|
...descriptor,
|
|
value: util.deprecate(descriptor.value, message, code)
|
|
});
|
|
} else if (descriptor.get || descriptor.set) {
|
|
Object.defineProperty(newObj, name, {
|
|
...descriptor,
|
|
get: descriptor.get && util.deprecate(descriptor.get, message, code),
|
|
set: descriptor.set && util.deprecate(descriptor.set, message, code)
|
|
});
|
|
} else {
|
|
let value = descriptor.value;
|
|
Object.defineProperty(newObj, name, {
|
|
configurable: descriptor.configurable,
|
|
enumerable: descriptor.enumerable,
|
|
get: util.deprecate(() => value, message, code),
|
|
set: descriptor.writable
|
|
? util.deprecate(
|
|
/**
|
|
* @template T
|
|
* @param {T} v value
|
|
* @returns {T} result
|
|
*/
|
|
(v) => (value = v),
|
|
message,
|
|
code
|
|
)
|
|
: undefined
|
|
});
|
|
}
|
|
}
|
|
return /** @type {T} */ (newObj);
|
|
};
|
|
|
|
module.exports.deprecateAllProperties = deprecateAllProperties;
|
|
|
|
/**
|
|
* @template {object} T
|
|
* @param {T} obj object
|
|
* @param {string} name property name
|
|
* @param {string} code deprecation code
|
|
* @param {string} note additional note
|
|
* @returns {T} frozen object with deprecation when modifying
|
|
*/
|
|
module.exports.soonFrozenObjectDeprecation = (obj, name, code, note = "") => {
|
|
const message = `${name} will be frozen in future, all modifications are deprecated.${
|
|
note && `\n${note}`
|
|
}`;
|
|
return /** @type {T} */ (
|
|
new Proxy(obj, {
|
|
set: util.deprecate(
|
|
/**
|
|
* @param {object} target target
|
|
* @param {string | symbol} property property
|
|
* @param {EXPECTED_ANY} value value
|
|
* @param {EXPECTED_ANY} receiver receiver
|
|
* @returns {boolean} result
|
|
*/
|
|
(target, property, value, receiver) =>
|
|
Reflect.set(target, property, value, receiver),
|
|
message,
|
|
code
|
|
),
|
|
defineProperty: util.deprecate(
|
|
/**
|
|
* @param {object} target target
|
|
* @param {string | symbol} property property
|
|
* @param {PropertyDescriptor} descriptor descriptor
|
|
* @returns {boolean} result
|
|
*/
|
|
(target, property, descriptor) =>
|
|
Reflect.defineProperty(target, property, descriptor),
|
|
message,
|
|
code
|
|
),
|
|
deleteProperty: util.deprecate(
|
|
/**
|
|
* @param {object} target target
|
|
* @param {string | symbol} property property
|
|
* @returns {boolean} result
|
|
*/
|
|
(target, property) => Reflect.deleteProperty(target, property),
|
|
message,
|
|
code
|
|
),
|
|
setPrototypeOf: util.deprecate(
|
|
/**
|
|
* @param {object} target target
|
|
* @param {EXPECTED_OBJECT | null} proto proto
|
|
* @returns {boolean} result
|
|
*/
|
|
(target, proto) => Reflect.setPrototypeOf(target, proto),
|
|
message,
|
|
code
|
|
)
|
|
})
|
|
);
|
|
};
|