diff --git a/modules/graph-layers/scripts/pack-marker-images.ts b/modules/graph-layers/scripts/pack-marker-images.ts new file mode 100644 index 00000000..bfdc61bd --- /dev/null +++ b/modules/graph-layers/scripts/pack-marker-images.ts @@ -0,0 +1,123 @@ +/** + * Generate texture atlas from images + * ``` + * # default input/output + * npx tsx ./pack-marker-images.ts + * + * # custom input/output directories + * npx tsx ./pack-marker-images.ts [inputDir] [outputDir] + * ``` + */ + +import {readFile, writeFile, readdir} from 'node:fs/promises'; +import {dirname, join, resolve} from 'node:path'; +import { fileURLToPath } from 'node:url'; +import ndarray, {NdArray} from 'ndarray'; +import {getPixels, savePixels} from 'ndarray-pixels'; +import pack from 'bin-pack'; +import Datauri from 'datauri'; + +const __dirname = dirname(fileURLToPath(import.meta.url)); +const packageRoot = join(__dirname, '..'); + +const inputDir = process.argv[2] || 'src/layers/common-layers/marker-layer/markers/'; +const outputDir = process.argv[3] || 'src/layers/common-layers/marker-layer/'; + +const INPUT_DIR = resolve(packageRoot, inputDir); +const OUTPUT_IMAGE = resolve(packageRoot, outputDir, 'marker-atlas.png'); +const OUTPUT_MAPPING = resolve(packageRoot, outputDir, `marker-mapping.ts`); +const OUTPUT_DATA_URL = resolve(packageRoot, outputDir, 'atlas-data-url.ts'); +const OUTPUT_LIST = resolve(packageRoot, outputDir, 'marker-list.ts'); +const IMAGE_PATTERN = /\.(png|jpg|jpeg|gif|bmp|tiff)$/i; + +// Get all images in the input path +const fileNames = (await readdir(INPUT_DIR)).filter((name) => IMAGE_PATTERN.test(name)); + +Promise.all(fileNames.map((name: string) => readImage(resolve(INPUT_DIR, name)))).then(async (images) => { + // Images are loaded + const nodes = images.map((pixels: NdArray, index: number) => ({ + name: fileNames[index], + pixels, + width: pixels.shape[0], + height: pixels.shape[1] + })); + + // Bin pack + const result = pack(nodes); + // console.log(result.items.length + ' items packed.'); + + // Convert to texture atlas + const outputJSON = {}; + const outputImage = createImage(result.width, result.height); + result.items.forEach((item) => { + outputJSON[item.item.name.replace(IMAGE_PATTERN, '')] = { + x: item.x, + y: item.y, + width: item.width, + height: item.height, + mask: true + }; + copyPixels(item.item.pixels, outputImage, item.x, item.y); + }); + + // Write to disk + await writeMapping(OUTPUT_MAPPING, outputJSON); + await writeImage(OUTPUT_IMAGE, outputImage, () => writeDataURL(OUTPUT_IMAGE, OUTPUT_DATA_URL)); + await writeList(OUTPUT_LIST, outputJSON); +}); + +/* Utils */ + +function copyPixels(fromImage: NdArray, toImage: NdArray, x: number, y: number): void { + const width = fromImage.shape[0]; + const height = fromImage.shape[1]; + const channels = fromImage.shape[2]; + + for (let i = 0; i < width; i++) { + for (let j = 0; j < height; j++) { + for (let k = 0; k < channels; k++) { + const value = fromImage.get(i, j, k); + toImage.set(i + x, j + y, k, value); + } + } + } +} + +async function writeMapping(filePath: string, content: unknown): Promise { + await exportJSFile(filePath, 'MarkerMapping', JSON.stringify(content, null, 2)); +} + +function createImage(width: number, height: number): NdArray { + return ndarray(new Uint8ClampedArray(width * height * 4), [width, height, 4]); +} + +async function writeDataURL(imagePath: string, outputFilePath: string): Promise { + const content = {dataURL: await Datauri(imagePath)}; + await exportJSFile(outputFilePath, 'AtlasDataURL', JSON.stringify(content, null, 2)); +} + +async function writeList(filePath: string, content: object): Promise { + const markers = Object.keys(content); + const markerMap = markers.reduce((res, marker) => { + res[marker] = marker; + return res; + }, {}); + const contentStr = JSON.stringify(markerMap, null, 2); + await exportJSFile(filePath, 'MarkerList', contentStr); +} + +async function exportJSFile(filePath: string, exportName: string, contentStr: string): Promise { + await writeFile( + filePath, + `/* eslint-disable */\nexport const ${exportName} = ${contentStr};\n/* eslint-enable */\n` + ); +} + +async function readImage(filePath: string): Promise { + return readFile(filePath).then((buffer) => getPixels(buffer, 'image/png')); +} + +async function writeImage(filePath: string, pixelArr: NdArray, createDataURL: () => void): Promise { + await writeFile(filePath, await savePixels(pixelArr, 'image/png')); + createDataURL(); +} diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/atlas-data-url.ts b/modules/graph-layers/src/layers/common-layers/marker-layer/atlas-data-url.ts index 35b856af..23b677f2 100644 --- a/modules/graph-layers/src/layers/common-layers/marker-layer/atlas-data-url.ts +++ b/modules/graph-layers/src/layers/common-layers/marker-layer/atlas-data-url.ts @@ -1,6 +1,5 @@ /* eslint-disable */ export const AtlasDataURL = { - dataURL: - '' + "dataURL": "" }; /* eslint-enable */ diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-atlas.png b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-atlas.png index 0cd0486e..06877ea0 100644 Binary files a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-atlas.png and b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-atlas.png differ diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-layer.ts b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-layer.ts index 77d8de1c..d979aff4 100644 --- a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-layer.ts +++ b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-layer.ts @@ -1,4 +1,3 @@ -// @noflow import {CompositeLayer} from '@deck.gl/core'; import {IconLayer} from '@deck.gl/layers'; diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-list.ts b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-list.ts index 46f32caa..34e27246 100644 --- a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-list.ts +++ b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-list.ts @@ -1,64 +1,64 @@ /* eslint-disable */ export const MarkerList = { - 'location-marker-filled': 'location-marker-filled', - 'bell-filled': 'bell-filled', - 'bookmark-filled': 'bookmark-filled', - bookmark: 'bookmark', - 'cd-filled': 'cd-filled', - cd: 'cd', - checkmark: 'checkmark', - 'circle-check-filled': 'circle-check-filled', - 'circle-check': 'circle-check', - 'circle-filled': 'circle-filled', - 'circle-i-filled': 'circle-i-filled', - 'circle-i': 'circle-i', - 'circle-minus-filled': 'circle-minus-filled', - 'circle-minus': 'circle-minus', - 'circle-plus-filled': 'circle-plus-filled', - 'circle-plus': 'circle-plus', - 'circle-questionmark-filled': 'circle-questionmark-filled', - 'circle-questionmark': 'circle-questionmark', - 'circle-slash-filled': 'circle-slash-filled', - 'circle-slash': 'circle-slash', - 'circle-x-filled': 'circle-x-filled', - 'circle-x': 'circle-x', - circle: 'circle', - 'diamond-filled': 'diamond-filled', - diamond: 'diamond', - 'flag-filled': 'flag-filled', - flag: 'flag', - gear: 'gear', - 'heart-filled': 'heart-filled', - heart: 'heart', - bell: 'bell', - 'location-marker': 'location-marker', - 'octagonal-star-filled': 'octagonal-star-filled', - 'octagonal-star': 'octagonal-star', - 'person-filled': 'person-filled', - person: 'person', - 'pin-filled': 'pin-filled', - pin: 'pin', - 'plus-small': 'plus-small', - plus: 'plus', - 'rectangle-filled': 'rectangle-filled', - rectangle: 'rectangle', - 'star-filled': 'star-filled', - star: 'star', - 'tag-filled': 'tag-filled', - tag: 'tag', - 'thumb-down-filled': 'thumb-down-filled', - 'thumb-down': 'thumb-down', - 'thumb-up': 'thumb-up', - 'thumb_up-filled': 'thumb_up-filled', - 'triangle-down-filled': 'triangle-down-filled', - 'triangle-down': 'triangle-down', - 'triangle-left-filled': 'triangle-left-filled', - 'triangle-left': 'triangle-left', - 'triangle-right-filled': 'triangle-right-filled', - 'triangle-right': 'triangle-right', - 'triangle-up-filled': 'triangle-up-filled', - 'triangle-up': 'triangle-up', - 'x-small': 'x-small', - x: 'x' + "bell-filled": "bell-filled", + "bell": "bell", + "bookmark-filled": "bookmark-filled", + "bookmark": "bookmark", + "cd-filled": "cd-filled", + "cd": "cd", + "checkmark": "checkmark", + "circle-check-filled": "circle-check-filled", + "circle-check": "circle-check", + "circle-filled": "circle-filled", + "circle-i-filled": "circle-i-filled", + "circle-i": "circle-i", + "circle-minus-filled": "circle-minus-filled", + "circle-minus": "circle-minus", + "circle-plus-filled": "circle-plus-filled", + "circle-plus": "circle-plus", + "circle-questionmark-filled": "circle-questionmark-filled", + "circle-questionmark": "circle-questionmark", + "circle-slash-filled": "circle-slash-filled", + "circle-slash": "circle-slash", + "circle-x-filled": "circle-x-filled", + "circle-x": "circle-x", + "circle": "circle", + "diamond-filled": "diamond-filled", + "diamond": "diamond", + "flag-filled": "flag-filled", + "flag": "flag", + "gear": "gear", + "heart-filled": "heart-filled", + "heart": "heart", + "location-marker-filled": "location-marker-filled", + "location-marker": "location-marker", + "octagonal-star-filled": "octagonal-star-filled", + "octagonal-star": "octagonal-star", + "person-filled": "person-filled", + "person": "person", + "pin-filled": "pin-filled", + "pin": "pin", + "plus-small": "plus-small", + "plus": "plus", + "rectangle-filled": "rectangle-filled", + "rectangle": "rectangle", + "star-filled": "star-filled", + "star": "star", + "tag-filled": "tag-filled", + "tag": "tag", + "thumb-down-filled": "thumb-down-filled", + "thumb-down": "thumb-down", + "thumb-up": "thumb-up", + "thumb_up-filled": "thumb_up-filled", + "triangle-down-filled": "triangle-down-filled", + "triangle-down": "triangle-down", + "triangle-left-filled": "triangle-left-filled", + "triangle-left": "triangle-left", + "triangle-right-filled": "triangle-right-filled", + "triangle-right": "triangle-right", + "triangle-up-filled": "triangle-up-filled", + "triangle-up": "triangle-up", + "x-small": "x-small", + "x": "x" }; /* eslint-enable */ diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-mapping.ts b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-mapping.ts index 7bc6f16a..f91ac2cf 100644 --- a/modules/graph-layers/src/layers/common-layers/marker-layer/marker-mapping.ts +++ b/modules/graph-layers/src/layers/common-layers/marker-layer/marker-mapping.ts @@ -1,424 +1,424 @@ /* eslint-disable */ export const MarkerMapping = { - 'location-marker-filled': { - x: 0, - y: 0, - width: 32, - height: 32, - mask: true - }, - 'bell-filled': { - x: 32, - y: 0, - width: 32, - height: 32, - mask: true - }, - 'bookmark-filled': { - x: 0, - y: 32, - width: 32, - height: 32, - mask: true - }, - bookmark: { - x: 32, - y: 32, - width: 32, - height: 32, - mask: true - }, - 'cd-filled': { - x: 64, - y: 0, - width: 32, - height: 32, - mask: true - }, - cd: { - x: 64, - y: 32, - width: 32, - height: 32, - mask: true - }, - checkmark: { - x: 0, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'circle-check-filled': { - x: 32, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'circle-check': { - x: 64, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'circle-filled': { - x: 96, - y: 0, - width: 32, - height: 32, - mask: true - }, - 'circle-i-filled': { - x: 96, - y: 32, - width: 32, - height: 32, - mask: true - }, - 'circle-i': { - x: 96, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'circle-minus-filled': { - x: 0, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'circle-minus': { - x: 32, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'circle-plus-filled': { - x: 64, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'circle-plus': { - x: 96, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'circle-questionmark-filled': { - x: 128, - y: 0, - width: 32, - height: 32, - mask: true - }, - 'circle-questionmark': { - x: 128, - y: 32, - width: 32, - height: 32, - mask: true - }, - 'circle-slash-filled': { - x: 128, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'circle-slash': { - x: 128, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'circle-x-filled': { - x: 0, - y: 128, - width: 32, - height: 32, - mask: true - }, - 'circle-x': { - x: 32, - y: 128, - width: 32, - height: 32, - mask: true - }, - circle: { - x: 64, - y: 128, - width: 32, - height: 32, - mask: true - }, - 'diamond-filled': { - x: 96, - y: 128, - width: 32, - height: 32, - mask: true - }, - diamond: { - x: 128, - y: 128, - width: 32, - height: 32, - mask: true - }, - 'flag-filled': { - x: 160, - y: 0, - width: 32, - height: 32, - mask: true - }, - flag: { - x: 160, - y: 32, - width: 32, - height: 32, - mask: true - }, - gear: { - x: 160, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'heart-filled': { - x: 160, - y: 96, - width: 32, - height: 32, - mask: true - }, - heart: { - x: 160, - y: 128, - width: 32, - height: 32, - mask: true - }, - bell: { - x: 0, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'location-marker': { - x: 32, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'octagonal-star-filled': { - x: 64, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'octagonal-star': { - x: 96, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'person-filled': { - x: 128, - y: 160, - width: 32, - height: 32, - mask: true - }, - person: { - x: 160, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'pin-filled': { - x: 192, - y: 0, - width: 32, - height: 32, - mask: true - }, - pin: { - x: 192, - y: 32, - width: 32, - height: 32, - mask: true - }, - 'plus-small': { - x: 192, - y: 64, - width: 32, - height: 32, - mask: true - }, - plus: { - x: 192, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'rectangle-filled': { - x: 192, - y: 128, - width: 32, - height: 32, - mask: true - }, - rectangle: { - x: 192, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'star-filled': { - x: 0, - y: 192, - width: 32, - height: 32, - mask: true - }, - star: { - x: 32, - y: 192, - width: 32, - height: 32, - mask: true - }, - 'tag-filled': { - x: 64, - y: 192, - width: 32, - height: 32, - mask: true - }, - tag: { - x: 96, - y: 192, - width: 32, - height: 32, - mask: true - }, - 'thumb-down-filled': { - x: 128, - y: 192, - width: 32, - height: 32, - mask: true - }, - 'thumb-down': { - x: 160, - y: 192, - width: 32, - height: 32, - mask: true - }, - 'thumb-up': { - x: 192, - y: 192, - width: 32, - height: 32, - mask: true - }, - 'thumb_up-filled': { - x: 224, - y: 0, - width: 32, - height: 32, - mask: true - }, - 'triangle-down-filled': { - x: 224, - y: 32, - width: 32, - height: 32, - mask: true - }, - 'triangle-down': { - x: 224, - y: 64, - width: 32, - height: 32, - mask: true - }, - 'triangle-left-filled': { - x: 224, - y: 96, - width: 32, - height: 32, - mask: true - }, - 'triangle-left': { - x: 224, - y: 128, - width: 32, - height: 32, - mask: true - }, - 'triangle-right-filled': { - x: 224, - y: 160, - width: 32, - height: 32, - mask: true - }, - 'triangle-right': { - x: 224, - y: 192, - width: 32, - height: 32, - mask: true - }, - 'triangle-up-filled': { - x: 0, - y: 224, - width: 32, - height: 32, - mask: true - }, - 'triangle-up': { - x: 32, - y: 224, - width: 32, - height: 32, - mask: true - }, - 'x-small': { - x: 64, - y: 224, - width: 32, - height: 32, - mask: true - }, - x: { - x: 96, - y: 224, - width: 32, - height: 32, - mask: true + "bell-filled": { + "x": 0, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "bell": { + "x": 32, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "bookmark-filled": { + "x": 0, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "bookmark": { + "x": 32, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "cd-filled": { + "x": 64, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "cd": { + "x": 64, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "checkmark": { + "x": 0, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "circle-check-filled": { + "x": 32, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "circle-check": { + "x": 64, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "circle-filled": { + "x": 96, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "circle-i-filled": { + "x": 96, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "circle-i": { + "x": 96, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "circle-minus-filled": { + "x": 0, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "circle-minus": { + "x": 32, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "circle-plus-filled": { + "x": 64, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "circle-plus": { + "x": 96, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "circle-questionmark-filled": { + "x": 128, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "circle-questionmark": { + "x": 128, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "circle-slash-filled": { + "x": 128, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "circle-slash": { + "x": 128, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "circle-x-filled": { + "x": 0, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "circle-x": { + "x": 32, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "circle": { + "x": 64, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "diamond-filled": { + "x": 96, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "diamond": { + "x": 128, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "flag-filled": { + "x": 160, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "flag": { + "x": 160, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "gear": { + "x": 160, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "heart-filled": { + "x": 160, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "heart": { + "x": 160, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "location-marker-filled": { + "x": 0, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "location-marker": { + "x": 32, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "octagonal-star-filled": { + "x": 64, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "octagonal-star": { + "x": 96, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "person-filled": { + "x": 128, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "person": { + "x": 160, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "pin-filled": { + "x": 192, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "pin": { + "x": 192, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "plus-small": { + "x": 192, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "plus": { + "x": 192, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "rectangle-filled": { + "x": 192, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "rectangle": { + "x": 192, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "star-filled": { + "x": 0, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "star": { + "x": 32, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "tag-filled": { + "x": 64, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "tag": { + "x": 96, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "thumb-down-filled": { + "x": 128, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "thumb-down": { + "x": 160, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "thumb-up": { + "x": 192, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "thumb_up-filled": { + "x": 224, + "y": 0, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-down-filled": { + "x": 224, + "y": 32, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-down": { + "x": 224, + "y": 64, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-left-filled": { + "x": 224, + "y": 96, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-left": { + "x": 224, + "y": 128, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-right-filled": { + "x": 224, + "y": 160, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-right": { + "x": 224, + "y": 192, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-up-filled": { + "x": 0, + "y": 224, + "width": 32, + "height": 32, + "mask": true + }, + "triangle-up": { + "x": 32, + "y": 224, + "width": 32, + "height": 32, + "mask": true + }, + "x-small": { + "x": 64, + "y": 224, + "width": 32, + "height": 32, + "mask": true + }, + "x": { + "x": 96, + "y": 224, + "width": 32, + "height": 32, + "mask": true } }; /* eslint-enable */ diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/scripts/pack-images.ts b/modules/graph-layers/src/layers/common-layers/marker-layer/scripts/pack-images.ts deleted file mode 100644 index d8e6dd55..00000000 --- a/modules/graph-layers/src/layers/common-layers/marker-layer/scripts/pack-images.ts +++ /dev/null @@ -1,127 +0,0 @@ -// @noflow -/** - * Generate texture atlas from images - * ``` - * npm install bin-pack ndarry get-pixels save-pixels datauri - * node pack-images - * ``` - */ - -const INPUT_DIR = process.argv[2]; -const OUTPUT_IMAGE = `${process.argv[3]}marker-atlas.png`; -const OUTPUT_MAPPING = `${process.argv[3]}marker-mapping.js`; -const OUTPUT_DATA_URL = `${process.argv[3]}atlas-data-url.js`; -const OUTPUT_LIST = `${process.argv[3]}marker-list.js`; -const IMAGE_PATTERN = /\.(png|jpg|jpeg|gif|bmp|tiff)$/i; - -const fs = require('fs'); -const path = require('path'); -const ndarray = require('ndarray'); -const savePixels = require('save-pixels'); -const getPixels = require('get-pixels'); -const pack = require('bin-pack'); -const Datauri = require('datauri'); - -// Get all images in the input path -const fileNames = fs.readdirSync(INPUT_DIR).filter((name) => IMAGE_PATTERN.test(name)); - -Promise.all(fileNames.map((name) => readImage(path.resolve(INPUT_DIR, name)))).then((images) => { - // Images are loaded - const nodes = images.map((pixels, index) => ({ - name: fileNames[index], - pixels, - width: pixels.shape[0], - height: pixels.shape[1] - })); - - // Bin pack - const result = pack(nodes); - // console.log(result.items.length + ' items packed.'); - - // Convert to texture atlas - const outputJSON = {}; - const outputImage = createImage(result.width, result.height); - result.items.forEach((item) => { - outputJSON[item.item.name.replace(IMAGE_PATTERN, '')] = { - x: item.x, - y: item.y, - width: item.width, - height: item.height, - mask: true - }; - copyPixels(item.item.pixels, outputImage, item.x, item.y); - }); - - // Write to disk - writeMapping(OUTPUT_MAPPING, outputJSON); - writeImage(OUTPUT_IMAGE, outputImage, () => writeDataURL(OUTPUT_IMAGE, OUTPUT_DATA_URL)); - writeList(OUTPUT_LIST, outputJSON); -}); - -/* Utils */ - -function copyPixels(fromImage, toImage, x, y) { - const width = fromImage.shape[0]; - const height = fromImage.shape[1]; - const channels = fromImage.shape[2]; - - for (let i = 0; i < width; i++) { - for (let j = 0; j < height; j++) { - for (let k = 0; k < channels; k++) { - const value = fromImage.get(i, j, k); - toImage.set(i + x, j + y, k, value); - } - } - } -} - -function writeMapping(filePath, content) { - const contentStr = JSON.stringify(content, null, 2); - exportJSFile(filePath, contentStr); -} - -function createImage(width, height) { - return ndarray(new Uint8ClampedArray(width * height * 4), [width, height, 4]); -} - -function writeDataURL(imagePath, outputFilePath) { - const datauri = new Datauri(imagePath); - const content = {dataURL: datauri.content}; - const contentStr = JSON.stringify(content, null, 2); - exportJSFile(outputFilePath, contentStr); -} - -function writeList(filePath, content) { - const markers = Object.keys(content); - const markerMap = markers.reduce((res, marker) => { - res[marker] = marker; - return res; - }, {}); - const contentStr = JSON.stringify(markerMap, null, 2); - exportJSFile(filePath, contentStr); -} - -function exportJSFile(filePath, contentStr) { - fs.writeFileSync( - filePath, - `/* eslint-disable */\nexport ${contentStr};\n/* eslint-enable */\n` - ); -} - -function readImage(filePath) { - return new Promise(function (resolve, reject) { - getPixels(filePath, function (err, pixels) { - if (err) { - resolve(null); - } else { - resolve(pixels); - } - }); - }); -} - -function writeImage(filePath, pixelArr, createDataURL) { - const file = fs.createWriteStream(filePath); - savePixels(pixelArr, 'png').pipe(file); - file.on('finish', createDataURL); -} diff --git a/modules/graph-layers/src/layers/common-layers/marker-layer/scripts/packing.sh b/modules/graph-layers/src/layers/common-layers/marker-layer/scripts/packing.sh deleted file mode 100644 index 7e6b8898..00000000 --- a/modules/graph-layers/src/layers/common-layers/marker-layer/scripts/packing.sh +++ /dev/null @@ -1,2 +0,0 @@ -# npm install --save-dev bin-pack ndarray get-pixels save-pixels datauri -node pack-images ../markers ../ diff --git a/modules/react-graph-layers/test/utils/hocs/fetch-s3-graph-hoc.tsx b/modules/react-graph-layers/test/utils/hocs/fetch-s3-graph-hoc.tsx index dbed09dd..dfc521aa 100644 --- a/modules/react-graph-layers/test/utils/hocs/fetch-s3-graph-hoc.tsx +++ b/modules/react-graph-layers/test/utils/hocs/fetch-s3-graph-hoc.tsx @@ -1,4 +1,3 @@ -// @noflow import React, {Component} from 'react'; import {fetchJSONFromS3} from '@deck.gl-community/test-utils'; import {JSONLoader} from 'deck-graph-layers'; diff --git a/modules/react-graph-layers/test/utils/hocs/sample-dataset-selector-hoc.tsx b/modules/react-graph-layers/test/utils/hocs/sample-dataset-selector-hoc.tsx index e7e45a37..41f30070 100644 --- a/modules/react-graph-layers/test/utils/hocs/sample-dataset-selector-hoc.tsx +++ b/modules/react-graph-layers/test/utils/hocs/sample-dataset-selector-hoc.tsx @@ -1,4 +1,3 @@ -// @noflow import React from 'react'; import {select} from '@storybook/addon-knobs'; import SAMPLE_GRAPH_DATASETS from '../data/sample-datasets'; diff --git a/package.json b/package.json index 97e1feca..243bcdca 100644 --- a/package.json +++ b/package.json @@ -36,13 +36,18 @@ "@probe.gl/test-utils": "^4.0.4", "@testing-library/jest-dom": "^6.4.2", "@testing-library/react": "^14.3.0", + "@types/bin-pack": "^1.0.3", "@vitejs/plugin-react": "^4.2.1", + "bin-pack": "^1.0.2", + "datauri": "^4.1.0", "esbuild": "^0.20.0", "eslint-plugin-jest": "^27.2.1", "jest": "^29.2.2", "jest-environment-jsdom": "^29.3.0", "jsdom": "^24.0.0", "lerna": "^8.1.2", + "ndarray": "^1.0.19", + "ndarray-pixels": "^3.1.0", "ocular-dev-tools": "2.0.0-alpha.29", "pre-commit": "^1.2.2", "pre-push": "^0.1.4", diff --git a/yarn.lock b/yarn.lock index 905c0236..725ef907 100644 --- a/yarn.lock +++ b/yarn.lock @@ -4708,6 +4708,11 @@ dependencies: "@babel/types" "^7.20.7" +"@types/bin-pack@^1.0.3": + version "1.0.3" + resolved "https://registry.yarnpkg.com/@types/bin-pack/-/bin-pack-1.0.3.tgz#311f0f44baeb2a9f5c9889ccceb33d6a442bcd24" + integrity sha512-FhdqIms08QvRuPvKiBW5Xpbq5dpMtmM4sOzvqMtaudftU4hCKpV3u6TRmGwMC1l6O324uXlwlrZWZnT7tBuh/w== + "@types/brotli@^1.3.0": version "1.3.4" resolved "https://registry.yarnpkg.com/@types/brotli/-/brotli-1.3.4.tgz#3eefc5493218a99141771f351142dd640efde5d8" @@ -4845,6 +4850,11 @@ resolved "https://registry.yarnpkg.com/@types/minimist/-/minimist-1.2.5.tgz#ec10755e871497bcd83efe927e43ec46e8c0747e" integrity sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag== +"@types/ndarray@^1.0.11": + version "1.0.14" + resolved "https://registry.yarnpkg.com/@types/ndarray/-/ndarray-1.0.14.tgz#96b28c09a3587a76de380243f87bb7a2d63b4b23" + integrity sha512-oANmFZMnFQvb219SSBIhI1Ih/r4CvHDOzkWyJS/XRqkMrGH5/kaPSA1hQhdIBzouaE+5KpE/f5ylI9cujmckQg== + "@types/node@*", "@types/node@>= 8": version "20.11.16" resolved "https://registry.yarnpkg.com/@types/node/-/node-20.11.16.tgz#4411f79411514eb8e2926f036c86c9f0e4ec6708" @@ -5860,6 +5870,32 @@ balanced-match@^1.0.0: resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== +bare-events@^2.0.0, bare-events@^2.2.0: + version "2.2.2" + resolved "https://registry.yarnpkg.com/bare-events/-/bare-events-2.2.2.tgz#a98a41841f98b2efe7ecc5c5468814469b018078" + integrity sha512-h7z00dWdG0PYOQEvChhOSWvOfkIKsdZGkWr083FgN/HyoQuebSew/cgirYqh9SCuy/hRvxc5Vy6Fw8xAmYHLkQ== + +bare-fs@^2.1.1: + version "2.2.3" + resolved "https://registry.yarnpkg.com/bare-fs/-/bare-fs-2.2.3.tgz#34f8b81b8c79de7ef043383c05e57d4a10392a68" + integrity sha512-amG72llr9pstfXOBOHve1WjiuKKAMnebcmMbPWDZ7BCevAoJLpugjuAPRsDINEyjT0a6tbaVx3DctkXIRbLuJw== + dependencies: + bare-events "^2.0.0" + bare-path "^2.0.0" + streamx "^2.13.0" + +bare-os@^2.1.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/bare-os/-/bare-os-2.2.1.tgz#c94a258c7a408ca6766399e44675136c0964913d" + integrity sha512-OwPyHgBBMkhC29Hl3O4/YfxW9n7mdTr2+SsO29XBWKKJsbgj3mnorDB80r5TiCQgQstgE5ga1qNYrpes6NvX2w== + +bare-path@^2.0.0, bare-path@^2.1.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/bare-path/-/bare-path-2.1.1.tgz#111db5bf2db0aed40081aa4ba38b8dfc2bb782eb" + integrity sha512-OHM+iwRDRMDBsSW7kl3dO62JyHdBKO3B25FB9vNQBPcGHMo4+eA8Yj41Lfbk3pS/seDY+siNge0LdRTulAau/A== + dependencies: + bare-os "^2.1.0" + base64-js@^1.1.2, base64-js@^1.3.1: version "1.5.1" resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" @@ -5895,6 +5931,11 @@ before-after-hook@^2.0.0, before-after-hook@^2.2.0: resolved "https://registry.yarnpkg.com/before-after-hook/-/before-after-hook-2.2.3.tgz#c51e809c81a4e354084422b9b26bad88249c517c" integrity sha512-NzUnlZexiaH/46WDhANlyR2bXRopNg4F/zuSA3OpZnllCUgRaOF2znDioDWrmbNVsuZk6l9pMquQB38cfBZwkQ== +bin-pack@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/bin-pack/-/bin-pack-1.0.2.tgz#c2a014edbf0bed70a3292062ed46577b96120679" + integrity sha512-aOk0SxEon5LF9cMxQFViSKb4qccG6rs7XKyMXIb1J8f8LA2acTIWnHdT0IOTe4gYBbqgjdbuTZ5f+UP+vlh4Mw== + binary-extensions@^2.0.0: version "2.2.0" resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" @@ -7071,6 +7112,13 @@ currently-unhandled@^0.4.1: dependencies: array-find-index "^1.0.1" +cwise-compiler@^1.0.0: + version "1.1.3" + resolved "https://registry.yarnpkg.com/cwise-compiler/-/cwise-compiler-1.1.3.tgz#f4d667410e850d3a313a7d2db7b1e505bb034cc5" + integrity sha512-WXlK/m+Di8DMMcCjcWr4i+XzcQra9eCdXIJrgh4TUgh0pIS/yJduLxS9JgefsHJ/YVLdgPtXm9r62W92MvanEQ== + dependencies: + uniq "^1.0.0" + cyclist@^1.0.1: version "1.0.2" resolved "https://registry.yarnpkg.com/cyclist/-/cyclist-1.0.2.tgz#673b5f233bf34d8e602b949429f8171d9121bea3" @@ -7383,6 +7431,14 @@ data-urls@^5.0.0: whatwg-mimetype "^4.0.0" whatwg-url "^14.0.0" +datauri@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/datauri/-/datauri-4.1.0.tgz#19e6e4fadc38ff8cea724d8ff59dcaad832520e4" + integrity sha512-y17kh32+I82G+ED9MNWFkZiP/Cq/vO1hN9+tSZsT9C9qn3NrvcBnh7crSepg0AQPge1hXx2Ca44s1FRdv0gFWA== + dependencies: + image-size "1.0.0" + mimer "^2.0.2" + dateformat@^3.0.0, dateformat@^3.0.3: version "3.0.3" resolved "https://registry.yarnpkg.com/dateformat/-/dateformat-3.0.3.tgz#a6e37499a4d9a9cf85ef5872044d62901c9889ae" @@ -7644,6 +7700,11 @@ detect-libc@^2.0.0: resolved "https://registry.yarnpkg.com/detect-libc/-/detect-libc-2.0.2.tgz#8ccf2ba9315350e1241b88d0ac3b0e1fbd99605d" integrity sha512-UX6sGumvvqSaXgdKGUsgZWqcUyIXZ/vZTrlRT/iobiKhGL0zL4d3osHj3uqllWJK+i+sixDS/3COVEOFbupFyw== +detect-libc@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/detect-libc/-/detect-libc-2.0.3.tgz#f0cd503b40f9939b894697d19ad50895e30cf700" + integrity sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw== + detect-newline@^3.0.0: version "3.1.0" resolved "https://registry.yarnpkg.com/detect-newline/-/detect-newline-3.1.0.tgz#576f5dfc63ae1a192ff192d8ad3af6308991b651" @@ -9806,6 +9867,13 @@ ignore@^5.0.4, ignore@^5.2.0, ignore@^5.2.4: resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.1.tgz#5073e554cd42c5b33b394375f538b8593e34d4ef" integrity sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw== +image-size@1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/image-size/-/image-size-1.0.0.tgz#58b31fe4743b1cec0a0ac26f5c914d3c5b2f0750" + integrity sha512-JLJ6OwBfO1KcA+TvJT+v8gbE6iWbj24LyDNFgFEN0lzegn6cC6a/p3NIDaepMsJjQjlUWqIC7wJv8lBFxPNjcw== + dependencies: + queue "6.0.2" + image-size@^0.7.4: version "0.7.5" resolved "https://registry.yarnpkg.com/image-size/-/image-size-0.7.5.tgz#269f357cf5797cb44683dfa99790e54c705ead04" @@ -9981,6 +10049,11 @@ invariant@^2.2.4: dependencies: loose-envify "^1.0.0" +iota-array@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/iota-array/-/iota-array-1.0.0.tgz#81ef57fe5d05814cd58c2483632a99c30a0e8087" + integrity sha512-pZ2xT+LOHckCatGQ3DcG/a+QuEqvoxqkiL7tvE8nn3uuu+f6i1TtpB5/FtWFbxUuVr5PZCx8KskuGatbJDXOWA== + ip@1.1.5: version "1.1.5" resolved "https://registry.yarnpkg.com/ip/-/ip-1.1.5.tgz#bdded70114290828c0a039e72ef25f5aaec4354a" @@ -10072,7 +10145,7 @@ is-boolean-object@^1.1.0: call-bind "^1.0.2" has-tostringtag "^1.0.0" -is-buffer@^1.1.5, is-buffer@~1.1.6: +is-buffer@^1.0.2, is-buffer@^1.1.5, is-buffer@~1.1.6: version "1.1.6" resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be" integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w== @@ -11912,6 +11985,11 @@ mime-types@^2.1.12, mime-types@~2.1.19: dependencies: mime-db "1.52.0" +mimer@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/mimer/-/mimer-2.0.2.tgz#941da26070e80bb485aed8a1ef4a5a325cbbfa96" + integrity sha512-izxvjsB7Ur5HrTbPu6VKTrzxSMBFBqyZQc6dWlZNQ4/wAvf886fD4lrjtFd8IQ8/WmZKdxKjUtqFFNaj3hQ52g== + mimic-fn@^1.0.0: version "1.2.0" resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-1.2.0.tgz#820c86a39334640e99516928bd03fca88057d022" @@ -12320,6 +12398,31 @@ natural-compare@^1.4.0: resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== +ndarray-ops@^1.2.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/ndarray-ops/-/ndarray-ops-1.2.2.tgz#59e88d2c32a7eebcb1bc690fae141579557a614e" + integrity sha512-BppWAFRjMYF7N/r6Ie51q6D4fs0iiGmeXIACKY66fLpnwIui3Wc3CXiD/30mgLbDjPpSLrsqcp3Z62+IcHZsDw== + dependencies: + cwise-compiler "^1.0.0" + +ndarray-pixels@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/ndarray-pixels/-/ndarray-pixels-3.1.0.tgz#0e8ab0f0c04adab2f50f62f4a70442ba0c103d1b" + integrity sha512-icw7L/kbxlHVoBgFWc0fCl1sYcatYSoZrAudD6KiQ1WXy84hRXIfqel5CLkh/SeLhb02Q2NW1ud9lE5G84V0zQ== + dependencies: + "@types/ndarray" "^1.0.11" + ndarray "^1.0.19" + ndarray-ops "^1.2.2" + sharp "^0.32.6" + +ndarray@^1.0.19: + version "1.0.19" + resolved "https://registry.yarnpkg.com/ndarray/-/ndarray-1.0.19.tgz#6785b5f5dfa58b83e31ae5b2a058cfd1ab3f694e" + integrity sha512-B4JHA4vdyZU30ELBw3g7/p9bZupyew5a7tX1Y/gGeF2hafrPaQZhgrGQfsvgfYbgdFZjYwuEcnaobeM/WMW+HQ== + dependencies: + iota-array "^1.0.0" + is-buffer "^1.0.2" + negotiator@^0.6.3: version "0.6.3" resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.3.tgz#58e323a72fedc0d6f9cd4d31fe49f51479590ccd" @@ -12398,6 +12501,11 @@ node-abi@^3.3.0: dependencies: semver "^7.3.5" +node-addon-api@^6.1.0: + version "6.1.0" + resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-6.1.0.tgz#ac8470034e58e67d0c6f1204a18ae6995d9c0d76" + integrity sha512-+eawOlIgy680F0kBzPUNFhMZGtJ1YmqM6l4+Crf4IkImjYrO/mqPwRMh352g23uIaQKFItcQ64I7KMaJxHgAVA== + node-fetch-npm@^2.0.2: version "2.0.4" resolved "https://registry.yarnpkg.com/node-fetch-npm/-/node-fetch-npm-2.0.4.tgz#6507d0e17a9ec0be3bec516958a497cec54bf5a4" @@ -13954,6 +14062,13 @@ queue-tick@^1.0.1: resolved "https://registry.yarnpkg.com/queue-tick/-/queue-tick-1.0.1.tgz#f6f07ac82c1fd60f82e098b417a80e52f1f4c142" integrity sha512-kJt5qhMxoszgU/62PLP1CJytzd2NKetjSRnyuj31fDd3Rlcz3fzlFdFLD1SItunPwyqEOkca6GbV612BWfaBag== +queue@6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/queue/-/queue-6.0.2.tgz#b91525283e2315c7553d2efa18d83e76432fed65" + integrity sha512-iHZWu+q3IdFZFX36ro/lKBkSvfkztY5Y7HMiPlOUjhupPcG2JMfst2KKEpu5XndviX/3UhFbRngUPNKtgvtZiA== + dependencies: + inherits "~2.0.3" + quick-lru@^1.0.0: version "1.1.0" resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-1.1.0.tgz#4360b17c61136ad38078397ff11416e186dcfbb8" @@ -14836,6 +14951,20 @@ shallowequal@1.1.0, shallowequal@^1.1.0: resolved "https://registry.yarnpkg.com/shallowequal/-/shallowequal-1.1.0.tgz#188d521de95b9087404fd4dcb68b13df0ae4e7f8" integrity sha512-y0m1JoUZSlPAjXVtPPW70aZWfIL/dSP7AFkRnniLCrK/8MDKog3TySTBmckD+RObVxH0v4Tox67+F14PdED2oQ== +sharp@^0.32.6: + version "0.32.6" + resolved "https://registry.yarnpkg.com/sharp/-/sharp-0.32.6.tgz#6ad30c0b7cd910df65d5f355f774aa4fce45732a" + integrity sha512-KyLTWwgcR9Oe4d9HwCwNM2l7+J0dUQwn/yf7S0EnTtb0eVS4RxO0eUSvxPtzT4F3SY+C4K6fqdv/DO27sJ/v/w== + dependencies: + color "^4.2.3" + detect-libc "^2.0.2" + node-addon-api "^6.1.0" + prebuild-install "^7.1.1" + semver "^7.5.4" + simple-get "^4.0.1" + tar-fs "^3.0.4" + tunnel-agent "^0.6.0" + shebang-command@^1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-1.2.0.tgz#44aac65b695b03398968c39f363fee5deafdf1ea" @@ -14912,7 +15041,7 @@ simple-concat@^1.0.0: resolved "https://registry.yarnpkg.com/simple-concat/-/simple-concat-1.0.1.tgz#f46976082ba35c2263f1c8ab5edfe26c41c9552f" integrity sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q== -simple-get@^4.0.0: +simple-get@^4.0.0, simple-get@^4.0.1: version "4.0.1" resolved "https://registry.yarnpkg.com/simple-get/-/simple-get-4.0.1.tgz#4a39db549287c979d352112fa03fd99fd6bc3543" integrity sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA== @@ -15251,6 +15380,16 @@ stream-shift@^1.0.0: resolved "https://registry.yarnpkg.com/stream-shift/-/stream-shift-1.0.3.tgz#85b8fab4d71010fc3ba8772e8046cc49b8a3864b" integrity sha512-76ORR0DO1o1hlKwTbi/DM3EXWGf3ZJYO8cXX5RJwnul2DEg2oyoZyjLNoQM8WsvZiFKCRfC1O0J7iCvie3RZmQ== +streamx@^2.13.0: + version "2.16.1" + resolved "https://registry.yarnpkg.com/streamx/-/streamx-2.16.1.tgz#2b311bd34832f08aa6bb4d6a80297c9caef89614" + integrity sha512-m9QYj6WygWyWa3H1YY69amr4nVgy61xfjys7xO7kviL5rfIEc2naf+ewFiOA+aEJD7y0JO3h2GoiUv4TDwEGzQ== + dependencies: + fast-fifo "^1.1.0" + queue-tick "^1.0.1" + optionalDependencies: + bare-events "^2.2.0" + streamx@^2.15.0: version "2.15.6" resolved "https://registry.yarnpkg.com/streamx/-/streamx-2.15.6.tgz#28bf36997ebc7bf6c08f9eba958735231b833887" @@ -15653,6 +15792,17 @@ tar-fs@^2.0.0: pump "^3.0.0" tar-stream "^2.1.4" +tar-fs@^3.0.4: + version "3.0.5" + resolved "https://registry.yarnpkg.com/tar-fs/-/tar-fs-3.0.5.tgz#f954d77767e4e6edf973384e1eb95f8f81d64ed9" + integrity sha512-JOgGAmZyMgbqpLwct7ZV8VzkEB6pxXFBVErLtb+XCOqzc6w1xiWKI9GVd6bwk68EX7eJ4DWmfXVmq8K2ziZTGg== + dependencies: + pump "^3.0.0" + tar-stream "^3.1.5" + optionalDependencies: + bare-fs "^2.1.1" + bare-path "^2.1.0" + tar-stream@^2.1.4, tar-stream@~2.2.0: version "2.2.0" resolved "https://registry.yarnpkg.com/tar-stream/-/tar-stream-2.2.0.tgz#acad84c284136b060dc3faa64474aa9aebd77287" @@ -16238,6 +16388,11 @@ union-value@^1.0.0: is-extendable "^0.1.1" set-value "^2.0.1" +uniq@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/uniq/-/uniq-1.0.1.tgz#b31c5ae8254844a3a8281541ce2b04b865a734ff" + integrity sha512-Gw+zz50YNKPDKXs+9d+aKAjVwpjNwqzvNpLigIruT4HA9lMZNdMqs9x07kKHB/L9WRzqp4+DlTU5s4wG2esdoA== + unique-filename@^1.1.1: version "1.1.1" resolved "https://registry.yarnpkg.com/unique-filename/-/unique-filename-1.1.1.tgz#1d69769369ada0583103a1e6ae87681b56573230"