UNPKG

vue3-sfc-loader

Version:
371 lines (370 loc) 12 kB
import { ParserPlugin as babel_ParserPlugin } from '@babel/parser'; /** * @internal */ export type ValueFactoryApi = { preventCache(): void; }; /** * @internal */ export type ValueFactory = (api: ValueFactoryApi) => Promise<any>; export type Cache = { get(key: string): Promise<string | undefined>; set(key: string, value: string): Promise<void>; }; export type ModuleCacheId = string; /** * An abstract way to specify a path. It could be a simple string or a object like an URL. An AbstractPath must always be convertible to a string. */ export type AbstractPath = { toString(): string; }; /** * A PathContext represents a path (relPath) relative to an abolute path (refPath) * Note that relPath is not necessary relative, but when it is, relPath is relative to refPath. */ export type PathContext = { /** reference path */ refPath: AbstractPath | undefined; /** relative to @refPath */ relPath: AbstractPath; }; /** relative to absolute module path resolution */ export type PathResolve = (pathCx: PathContext, options: Options) => AbstractPath; /** * Used by the library when it needs to handle a does not know how to handle a given file type (eg. `.json` files). * @param type The type of the file. It can be anything, but must be '.vue', '.js' or '.mjs' for vue, js and esm files. * @param getContentData The method to get the content data of a file (text or binary). see [[ File['getContentData'] ]] * @param path The path of the file * @param options The options * * * **example:** * * ```javascript * ... * ... * ``` */ export type ModuleHandler = (type: string, getContentData: File['getContentData'], path: AbstractPath, options: Options) => Promise<ModuleExport | null>; export type ContentData = string | ArrayBuffer; /** * Represents a file content and the extension name. */ export type File = { /** The content data accessor (request data as text of binary)*/ getContentData: (asBinary: Boolean) => Promise<ContentData>; /** The content type (file extension name, eg. '.svg' ) */ type: string; }; /** * Represents a resource. */ export type Resource = { /** * 'abstract' unique id of the resource. * This id is used as the key of the [[Options.moduleCache]] */ id: ModuleCacheId; /** file path of the resource */ path: AbstractPath; /** asynchronously get the content of the resource. Once you got the content, you can asynchronously get the data through the getContentData(asBinary) method. */ getContent: () => Promise<File>; }; /** * CustomBlockCallback function type */ export type CustomBlockCallback = (component: ModuleExport) => void; /** * A custom block */ export type CustomBlock = { type: string; content: string; attrs: Record<string, string | true>; }; /** * This just represents a loaded js module exports */ export type ModuleExport = {} | null; /** * This just represents a loaded js module */ export type Module = { exports: ModuleExport; }; /** * @internal */ export type LoadingType<T> = { promise: Promise<T>; }; export type Options = { /** * Initial cache that will contain resolved dependencies. All new modules go here. * `vue` must initially be contained in this object. * [[moduleCache]] is mandatory and should be shared between options objects used for you application (note that you can also pass the same options object through multiple loadModule calls) * It is recommended to provide a prototype-less object (`Object.create(null)`) to avoid potential conflict with `Object` properties (constructor, __proto__, hasOwnProperty, ...). ​ * The library take the ownership of [[moduleCache]] when [[loadModule]] is called. * See also [[options.loadModule]]. * * **example:** * ```javascript * ... * moduleCache: Object.assign(Object.create(null), { * vue: Vue, * }), * ... * ``` * */ moduleCache: Record<ModuleCacheId, LoadingType<ModuleExport> | ModuleExport>; /** * Called by the library when it needs a file. * @param path The path of the file * @returns a Promise of the file content or an accessor to the file content that handles text or binary data * * **example:** * ```javascript * ... * async getFile(url) { * * const res = await fetch(url); * * if ( !res.ok ) * throw Object.assign(new Error(url+' '+res.statusText), { res }); * * return { * getContentData: asBinary => asBinary ? res.arrayBuffer() : res.text(), * } * * return await res.text(); * }, * ... * ``` */ getFile(path: AbstractPath): Promise<File | ContentData>; /** * Called by the library when CSS style must be added in the HTML document. * @param style The CSS style chunk * @param scopeId The scope ID of the CSS style chunk * @return * * **example:** * ```javascript * ... * addStyle(styleStr) { * * const style = document.createElement('style'); * style.textContent = styleStr; * const ref = document.head.getElementsByTagName('style')[0] || null; * document.head.insertBefore(style, ref); * }, * ... * ``` */ addStyle(style: string, scopeId: string | undefined): void; /** * Sets the delimiters used for text interpolation within the template. * Typically this is used to avoid conflicting with server-side frameworks that also use mustache syntax. * * ```javascript * ... * <script> * * // <!-- * const vueContent = ` * <template> Hello [[[[ who ]]]] !</template> * <script> * export default { * data() { * return { * who: 'world' * } * } * } * </script> * `; * // --> * * const options = { * moduleCache: { vue: Vue }, * getFile: () => vueContent, * addStyle: () => {}, * delimiters: ['[[[[', ']]]]'], * } * * const app = Vue.createApp(Vue.defineAsyncComponent(() => window['vue3-sfc-loader'].loadModule('file.vue', options))); * app.mount(document.body); * * </script> * ... * ``` */ delimiters?: [string, string]; /** * Whitespace handling strategy * see https://github.com/vuejs/vue/tree/dev/packages/vue-template-compiler#options */ whitespace?: 'preserve' | 'condense'; /** * Specifies a check method to recognize native custom elements. * * see. https://vuejs.org/api/application.html#app-config-compileroptions-iscustomelement * note: this option has no effect on vue2 */ isCustomElement: ((tag: string) => boolean) | undefined; /** * Additional babel parser plugins. [TBD] * * ```javascript * ... * ... * ``` */ additionalBabelParserPlugins?: babel_ParserPlugin[]; /** * Additional babel plugins. [TBD] * * ```javascript * ... * ... * ``` */ additionalBabelPlugins?: Record<string, any>; /** * Handle additional module types (eg. '.svg', '.json' ). see [[ModuleHandler]] * */ handleModule?: ModuleHandler; /** * [[get]]() and [[set]]() functions of this object are called when the lib needs to save or load already compiled code. get and set functions must return a `Promise` (or can be `async`). * Since compilation consume a lot of CPU, is is always a good idea to provide this object. * * **example:** * * In the following example, we cache the compiled code in the browser's local storage. Note that local storage is a limited place (usually 5MB). * Here we handle space limitation in a very basic way. * Maybe (not tested), the following libraries may help you to gain more space [pako](https://github.com/nodeca/pako), [lz-string](https://github.com/pieroxy/lz-string/) * ```javascript * ... * compiledCache: { * set(key, str) { * * // naive storage space management * for (;;) { * * try { * * // doc: https://developer.mozilla.org/en-US/docs/Web/API/Storage * window.localStorage.setItem(key, str); * break; * } catch(ex) { * // here we handle DOMException: Failed to execute 'setItem' on 'Storage': Setting the value of 'XXX' exceeded the quota * * window.localStorage.removeItem(window.localStorage.key(0)); * } * } * }, * get(key) { * * return window.localStorage.getItem(key); * }, * }, * ... * ``` */ compiledCache?: Cache; /** * Called by the library when there is somthing to log (eg. scripts compilation errors, template compilation errors, template compilation tips, style compilation errors, ...) * @param type the type of the notification, it respects console property names (error, warn, info). * @param args the values to log * @return * * ```javascript * ... * log(type, ...args) { * * console.log(type, ...args); * }, * ... * ``` */ log?(type: string, ...data: any[]): void; /** * Called when the lib requires a module. Do return `undefined` to let the library handle this. * @param path The path of the module. * @param options The options object. * @returns A Promise of the module or undefined * * [[moduleCache]] and [[Options.loadModule]] are strongly related, in the sense that the result of [[options.loadModule]] is stored in [[moduleCache]]. * However, [[options.loadModule]] is asynchronous and may help you to handle modules or components that are conditionally required (optional features, current languages, plugins, ...). * ```javascript * ... * loadModule(path, options) { * * if ( path === 'vue' ) * return Vue; * }, * ... * ``` */ loadModule?(path: AbstractPath, options: Options): Promise<ModuleExport | undefined>; /** * creates a CommonJS module from JS source string. * *(optional)* */ createCJSModule(refPath: AbstractPath, source: string, options: Options): Module; /** * Abstact path handling * *(optional)* */ pathResolve: PathResolve; /** * by default, remove the search string * in situation where you need to keep the path intact, use `getPathname: path => path` * @param path a path that may contains extra components like search params or hash (eg. ./mydir/mycomponent.vue?v=123) * @returns the pathname component of the path withouy any orher component (eg. ./mydir/mycomponent.vue) */ getPathname: (path: string) => string; /** * Abstact resource handling * *(optional)* * */ getResource(pathCx: PathContext, options: Options): Resource; /** * Called for each custom block. * @returns A Promise of the module or undefined * * ```javascript * ... * customBlockHandler(block, filename, options) { * * if ( block.type !== 'i18n' ) * return; * * return (component) => { * * component.i18n = JSON.parse(block.content); * } * } * ... * ``` */ customBlockHandler?(block: CustomBlock, filename: AbstractPath, options: Options): Promise<CustomBlockCallback | undefined>; /** * Set development mode * prevent minification, allow debugger statement, */ devMode?: boolean; /** * * @param srcRaw * @param lang * @param filename */ processStyles(srcRaw: string, lang: string | undefined, filename: AbstractPath, options: Options): Promise<string>; }; export type LangProcessor = (source: string, preprocessOptions?: any) => Promise<string> | string;