monaco-editor-core
Version:
A browser based code editor
129 lines (128 loc) • 4 kB
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('#'));
}
});