mirror of
https://github.com/marcogll/passkit-generator.git
synced 2026-03-16 02:25:32 +00:00
Splitted them in a folder and renamed them to have the same name of typescript type; Added more SemanticKeys; Removed Schema resolution via string;
190 lines
4.6 KiB
TypeScript
190 lines
4.6 KiB
TypeScript
import { EOL } from "os";
|
|
import type * as Schemas from "./schemas";
|
|
import { sep } from "path";
|
|
|
|
/**
|
|
* Checks if an rgb value is compliant with CSS-like syntax
|
|
*
|
|
* @function isValidRGB
|
|
* @params {String} value - string to analyze
|
|
* @returns {Boolean} True if valid rgb, false otherwise
|
|
*/
|
|
|
|
export function isValidRGB(value?: string): boolean {
|
|
if (!value || typeof value !== "string") {
|
|
return false;
|
|
}
|
|
|
|
const rgb = value.match(
|
|
/^rgb\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*,\s*(\d{1,3})\s*\)/,
|
|
);
|
|
|
|
if (!rgb) {
|
|
return false;
|
|
}
|
|
|
|
return rgb.slice(1, 4).every((v) => Math.abs(Number(v)) <= 255);
|
|
}
|
|
|
|
/**
|
|
* Converts a date to W3C Standard format
|
|
*
|
|
* @function dateToW3Cstring
|
|
* @params date - The date to be parsed
|
|
* @returns - The parsed string if the parameter is valid,
|
|
* undefined otherwise
|
|
*/
|
|
|
|
export function dateToW3CString(date: Date) {
|
|
if (!(date instanceof Date)) {
|
|
return "";
|
|
}
|
|
|
|
// if it is NaN, it is "Invalid Date"
|
|
if (isNaN(Number(date))) {
|
|
return undefined;
|
|
}
|
|
|
|
const paddedMonth = padMeTwo(date.getMonth() + 1);
|
|
const paddedDay = padMeTwo(date.getDate());
|
|
const paddedHour = padMeTwo(date.getHours());
|
|
const paddedMinutes = padMeTwo(date.getMinutes());
|
|
const paddedSeconds = padMeTwo(date.getSeconds());
|
|
|
|
/**
|
|
* Date.prototype.getTimezoneOffset returns the timezone UTC offset in
|
|
* minutes of the local machine.
|
|
*
|
|
* That value should then be used to calculate the effective timezone as
|
|
* string, but still that would be related to the machine and not to the
|
|
* specified date.
|
|
*
|
|
* For this reason we are completing date with "Z" TimeZoneDesignator (TZD)
|
|
* to say it to use local timezone.
|
|
*
|
|
* In the future we might think to integrate another parameter to represent
|
|
* a custom timezone.
|
|
*
|
|
* @see https://www.w3.org/TR/NOTE-datetime
|
|
*/
|
|
|
|
return `${date.getFullYear()}-${paddedMonth}-${paddedDay}T${paddedHour}:${paddedMinutes}:${paddedSeconds}Z`;
|
|
}
|
|
|
|
function padMeTwo(original: string | number) {
|
|
return String(original).padStart(2, "0");
|
|
}
|
|
|
|
/**
|
|
* Apply a filter to arg0 to remove hidden files names (starting with dot)
|
|
*
|
|
* @function removeHidden
|
|
* @params {String[]} from - list of file names
|
|
* @return {String[]}
|
|
*/
|
|
|
|
export function removeHidden(from: Array<string>): Array<string> {
|
|
return from.filter((e) => e.charAt(0) !== ".");
|
|
}
|
|
|
|
/**
|
|
* Creates a buffer of translations in Apple .strings format
|
|
*
|
|
* @function generateStringFile
|
|
* @params {Object} lang - structure containing related to ISO 3166 alpha-2 code for the language
|
|
* @returns {Buffer} Buffer to be written in pass.strings for language in lang
|
|
* @see https://apple.co/2M9LWVu - String Resources
|
|
*/
|
|
|
|
export function generateStringFile(lang: { [index: string]: string }): Buffer {
|
|
if (!Object.keys(lang).length) {
|
|
return Buffer.from("", "utf8");
|
|
}
|
|
|
|
// Pass.strings format is the following one for each row:
|
|
// "key" = "value";
|
|
|
|
const strings = Object.keys(lang).map(
|
|
(key) => `"${key}" = "${lang[key].replace(/"/g, '"')}";`,
|
|
);
|
|
|
|
return Buffer.from(strings.join(EOL), "utf8");
|
|
}
|
|
|
|
/**
|
|
* Applies a partition to split one bundle
|
|
* to two
|
|
* @param origin
|
|
*/
|
|
|
|
type PartitionedBundleElements = [
|
|
Schemas.PartitionedBundle["l10nBundle"],
|
|
Schemas.PartitionedBundle["bundle"],
|
|
];
|
|
|
|
export function splitBufferBundle(
|
|
origin: Schemas.BundleUnit,
|
|
): PartitionedBundleElements {
|
|
const initialValue: PartitionedBundleElements = [{}, {}];
|
|
|
|
if (!origin) {
|
|
return initialValue;
|
|
}
|
|
|
|
return Object.entries(origin).reduce<PartitionedBundleElements>(
|
|
([l10n, bundle], [key, buffer]) => {
|
|
if (!key.includes(".lproj")) {
|
|
return [
|
|
l10n,
|
|
{
|
|
...bundle,
|
|
[key]: buffer,
|
|
},
|
|
];
|
|
}
|
|
|
|
const pathComponents = key.split(sep);
|
|
const lang = pathComponents[0];
|
|
const file = pathComponents.slice(1).join("/");
|
|
|
|
(l10n[lang] || (l10n[lang] = {}))[file] = buffer;
|
|
|
|
return [l10n, bundle];
|
|
},
|
|
initialValue,
|
|
);
|
|
}
|
|
|
|
type StringSearchMode = "includes" | "startsWith" | "endsWith";
|
|
|
|
export function getAllFilesWithName(
|
|
name: string,
|
|
source: string[],
|
|
mode: StringSearchMode = "includes",
|
|
forceLowerCase: boolean = false,
|
|
): string[] {
|
|
return source.filter((file) =>
|
|
((forceLowerCase && file.toLowerCase()) || file)[mode](name),
|
|
);
|
|
}
|
|
|
|
export function hasFilesWithName(
|
|
name: string,
|
|
source: string[],
|
|
mode: StringSearchMode = "includes",
|
|
forceLowerCase: boolean = false,
|
|
): boolean {
|
|
return source.some((file) =>
|
|
((forceLowerCase && file.toLowerCase()) || file)[mode](name),
|
|
);
|
|
}
|
|
|
|
export function deletePersonalization(
|
|
source: Schemas.BundleUnit,
|
|
logosNames: string[] = [],
|
|
): void {
|
|
[...logosNames, "personalization.json"].forEach(
|
|
(file) => delete source[file],
|
|
);
|
|
}
|