From 5f63b2277779ef117ab0c495ce52b8ffaeec2aa7 Mon Sep 17 00:00:00 2001 From: Patrick Nelson Date: Mon, 4 Dec 2023 20:20:13 -0800 Subject: [PATCH] =?UTF-8?q?Issue=20#34:=20Demo=20site:=20Consolidating=20t?= =?UTF-8?q?he=20"import"=20spam=20by=20using=20a=20new=20and=20experimenta?= =?UTF-8?q?l=20"autoDefine()"=20function=20=20(due=20to=20large=20number?= =?UTF-8?q?=20of=20components).=20Undocumented=20for=20a=20reason...=20ple?= =?UTF-8?q?ase=20do=20not=20use=20in=20production!=20=F0=9F=98=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- demo/vercel/src/main.js | 201 +++++++++++++--------------------------- 1 file changed, 64 insertions(+), 137 deletions(-) diff --git a/demo/vercel/src/main.js b/demo/vercel/src/main.js index af2cd78..ad09955 100644 --- a/demo/vercel/src/main.js +++ b/demo/vercel/src/main.js @@ -1,15 +1,6 @@ import svelteRetag from '../../../index.js'; import './app.css'; -import App from './App.svelte'; -import Intro from './lib/Intro.svelte'; -import Counter from './lib/Counter.svelte'; -import ExampleTag from './lib/ExampleTag.svelte'; -import { TabsWrapper, TabList, TabPanel, TabButton } from './lib/tabs'; -import TabsDemo from './lib/TabsDemo.svelte'; -import FeaturesInfo from './lib/FeaturesInfo.svelte'; -import ExamplesInfo from './lib/ExamplesInfo.svelte'; -import TabsInfo from './lib/TabsInfo.svelte'; /** @@ -22,136 +13,72 @@ const shadow = !!params.get('shadow'); const shadowStylesheet = ''; // TODO: ISSUE-6: Find good solution for defining this. For now, hack is to inject style[data-vite-dev-id] into shadowRoot. -/** - * The definitions below are intentionally redundant and kept simple just to demonstrate syntax. - * - * This could easily be consolidated by wrapping your own wrapper function. For example, I might normally wrap this in - * a function like "defineTag(component, tag, attribs)". - */ - -svelteRetag({ - component: App, - tagname: 'app-tag', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: Intro, - tagname: 'intro-tag', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: FeaturesInfo, - tagname: 'features-info', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: ExamplesInfo, - tagname: 'examples-info', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: TabsInfo, - tagname: 'tabs-info', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: Counter, - tagname: 'counter-tag', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: ExampleTag, - tagname: 'example-tag', - attributes: true, - debugMode, - hydratable, - shadow, - href: shadowStylesheet, -}); -svelteRetag({ - component: TabPanel, - tagname: 'tab-panel', - attributes: true, - hydratable, - debugMode, - shadow, - href: shadowStylesheet, +// Since we have a large number of components, we'll take advantage of Vite's glob import feature to automatically find +// and then define each svelteTag() component and derive the tagname from the filename (see autoDefine() function below). +// NOTE: This example uses eager loading, but you can also use lazy loading by removing the { eager: true } option. +autoDefine(import.meta.glob('./**/*.svelte', { eager: true }), { + // In case we encounter components that are only a single word, we must have a prefix or suffix to add. + suffix: 'tag', +}, () => { + // Extend with additional options, if desired. + return { + attributes: true, + debugMode, + hydratable, + shadow, + href: shadowStylesheet, + }; }); -svelteRetag({ - component: TabButton, - tagname: 'tab-button', - attributes: true, - hydratable, - debugMode, - shadow, - href: shadowStylesheet, -}); -svelteRetag({ - component: TabsWrapper, - tagname: 'tabs-wrapper', - attributes: true, - hydratable, - debugMode, - shadow, - href: shadowStylesheet, -}); - -svelteRetag({ - component: TabList, - tagname: 'tab-list', - attributes: true, - hydratable, - debugMode, - shadow, - href: shadowStylesheet, -}); - - -/** - * Note: This is here for local testing and debugging only. This is just a demonstration of pure svelte-only context - * (without maintaining that context through svelte-retag). - */ - -svelteRetag({ - component: TabsDemo, - tagname: 'tabs-demo', - attributes: true, - hydratable, - debugMode, - shadow, - href: shadowStylesheet, -}); +// Usage: autoDefine(import.meta.glob('./**/*.svelte', { eager: true }), { prefix: 'my', suffix: 'tag' }, (tagname, component) => { return { attributes: true, shadow: false }; }); +// WARNING: This autoDefine is experimental, and is not recommended for use in production yet. Especially for dynamic imports (i.e. without the eager option set). +// This is for demonstration purposes only, for now. Use at your own risk! + +async function autoDefine(modules, { + prefix = '', + suffix = '' +} = {}, optionsCallback) { + // Force prefix to alphanumeric chars in case user accidentally included dashes or other characters. + prefix = prefix.toLowerCase().replace(/[^a-z0-9]/g, ''); + for (const path in modules) { + let module = modules[path]; + + // Determine if this module was fetched asynchronously (returned as a callable function) or eagerly. + let component; + if (module instanceof Function) { + // TODO: Can be optimized by parallelling this execution. Right now this is called in order (one blocking the other). + // TODO: Alternatively, could + component = (await module()).default; + } else { + component = module.default; + } + + // Break name up into words based on capitalized letters and join words with dashes and convert to lowercase so we can + // convert it to a tag name. + const name = path.match(/([^/]+)\.svelte$/)[1]; + let words = name.match(/([A-Z][a-z]+)/g); + + // If not using camel case, try to split on dashes or underscores and then failover to a default prefix (if still only one word). + if (!words) { + words = name.split(/[-_]/); + } + if (words.length === 1) { + words = [prefix, words[0], suffix].filter(Boolean); // Filter out empty strings. + } + const tagname = words.join('-').toLowerCase(); + + // Take additional options from callback. + const options = optionsCallback ? optionsCallback(tagname, component) : {}; + + // Go ahead and use svelte-retag to define tag now with inferred options (component and tag) and combine with customized options. + svelteRetag({ + component, + tagname, + ...options, + }); + } + +}