mirror of
https://github.com/neocities/neocities.git
synced 2025-05-28 08:29:59 +02:00
128 lines
3.9 KiB
JavaScript
128 lines
3.9 KiB
JavaScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
import { distinct } from './arrays.js';
|
|
import { Iterable } from './iterator.js';
|
|
import { generateUuid } from './uuid.js';
|
|
export function createStringDataTransferItem(stringOrPromise) {
|
|
return {
|
|
asString: async () => stringOrPromise,
|
|
asFile: () => undefined,
|
|
value: typeof stringOrPromise === 'string' ? stringOrPromise : undefined,
|
|
};
|
|
}
|
|
export function createFileDataTransferItem(fileName, uri, data) {
|
|
const file = { id: generateUuid(), name: fileName, uri, data };
|
|
return {
|
|
asString: async () => '',
|
|
asFile: () => file,
|
|
value: undefined,
|
|
};
|
|
}
|
|
export class VSDataTransfer {
|
|
constructor() {
|
|
this._entries = new Map();
|
|
}
|
|
get size() {
|
|
let size = 0;
|
|
for (const _ of this._entries) {
|
|
size++;
|
|
}
|
|
return size;
|
|
}
|
|
has(mimeType) {
|
|
return this._entries.has(this.toKey(mimeType));
|
|
}
|
|
matches(pattern) {
|
|
const mimes = [...this._entries.keys()];
|
|
if (Iterable.some(this, ([_, item]) => item.asFile())) {
|
|
mimes.push('files');
|
|
}
|
|
return matchesMimeType_normalized(normalizeMimeType(pattern), mimes);
|
|
}
|
|
get(mimeType) {
|
|
return this._entries.get(this.toKey(mimeType))?.[0];
|
|
}
|
|
/**
|
|
* Add a new entry to this data transfer.
|
|
*
|
|
* This does not replace existing entries for `mimeType`.
|
|
*/
|
|
append(mimeType, value) {
|
|
const existing = this._entries.get(mimeType);
|
|
if (existing) {
|
|
existing.push(value);
|
|
}
|
|
else {
|
|
this._entries.set(this.toKey(mimeType), [value]);
|
|
}
|
|
}
|
|
/**
|
|
* Set the entry for a given mime type.
|
|
*
|
|
* This replaces all existing entries for `mimeType`.
|
|
*/
|
|
replace(mimeType, value) {
|
|
this._entries.set(this.toKey(mimeType), [value]);
|
|
}
|
|
/**
|
|
* Remove all entries for `mimeType`.
|
|
*/
|
|
delete(mimeType) {
|
|
this._entries.delete(this.toKey(mimeType));
|
|
}
|
|
/**
|
|
* Iterate over all `[mime, item]` pairs in this data transfer.
|
|
*
|
|
* There may be multiple entries for each mime type.
|
|
*/
|
|
*[Symbol.iterator]() {
|
|
for (const [mine, items] of this._entries) {
|
|
for (const item of items) {
|
|
yield [mine, item];
|
|
}
|
|
}
|
|
}
|
|
toKey(mimeType) {
|
|
return normalizeMimeType(mimeType);
|
|
}
|
|
}
|
|
function normalizeMimeType(mimeType) {
|
|
return mimeType.toLowerCase();
|
|
}
|
|
export function matchesMimeType(pattern, mimeTypes) {
|
|
return matchesMimeType_normalized(normalizeMimeType(pattern), mimeTypes.map(normalizeMimeType));
|
|
}
|
|
function matchesMimeType_normalized(normalizedPattern, normalizedMimeTypes) {
|
|
// Anything wildcard
|
|
if (normalizedPattern === '*/*') {
|
|
return normalizedMimeTypes.length > 0;
|
|
}
|
|
// Exact match
|
|
if (normalizedMimeTypes.includes(normalizedPattern)) {
|
|
return true;
|
|
}
|
|
// Wildcard, such as `image/*`
|
|
const wildcard = normalizedPattern.match(/^([a-z]+)\/([a-z]+|\*)$/i);
|
|
if (!wildcard) {
|
|
return false;
|
|
}
|
|
const [_, type, subtype] = wildcard;
|
|
if (subtype === '*') {
|
|
return normalizedMimeTypes.some(mime => mime.startsWith(type + '/'));
|
|
}
|
|
return false;
|
|
}
|
|
export const UriList = Object.freeze({
|
|
// http://amundsen.com/hypermedia/urilist/
|
|
create: (entries) => {
|
|
return distinct(entries.map(x => x.toString())).join('\r\n');
|
|
},
|
|
split: (str) => {
|
|
return str.split('\r\n');
|
|
},
|
|
parse: (str) => {
|
|
return UriList.split(str).filter(value => !value.startsWith('#'));
|
|
}
|
|
});
|