Allow links/images to be loaded from browser /src (#1932)

* Allow links/images to be loaded from browser /src

* Adds a changeset

* typo
This commit is contained in:
Matthew Phillips 2021-11-19 15:25:05 -05:00 committed by GitHub
parent 74624d9697
commit a0d4d99a9a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 15 additions and 7 deletions

View file

@ -0,0 +1,5 @@
---
'astro': patch
---
Allow using /src in URL for CSS / images

View file

@ -25,13 +25,13 @@ const ASTRO_EMPTY = '@astro-empty';
const tagsWithSrcSet = new Set(['img', 'source']); const tagsWithSrcSet = new Set(['img', 'source']);
const isAstroInjectedLink = (node: parse5.Element) => isStylesheetLink(node) && getAttribute(node, 'data-astro-injected') === ''; const isAstroInjectedLink = (node: parse5.Element) => isStylesheetLink(node) && getAttribute(node, 'data-astro-injected') === '';
const isBuildableLink = (node: parse5.Element, srcRoot: string) => { const isBuildableLink = (node: parse5.Element, srcRoot: string, srcRootWeb: string) => {
if (isAstroInjectedLink(node)) return true; if (isAstroInjectedLink(node)) return true;
const href = getAttribute(node, 'href'); const href = getAttribute(node, 'href');
if (typeof href !== 'string' || !href.length) return false; if (typeof href !== 'string' || !href.length) return false;
return href.startsWith(srcRoot) || `/${href}`.startsWith(srcRoot); // Windows fix: some paths are missing leading "/" return href.startsWith(srcRoot) || href.startsWith(srcRootWeb) || `/${href}`.startsWith(srcRoot); // Windows fix: some paths are missing leading "/"
}; };
const isBuildableImage = (node: parse5.Element, srcRoot: string) => getTagName(node) === 'img' && getAttribute(node, 'src')?.startsWith(srcRoot); const isBuildableImage = (node: parse5.Element, srcRoot: string, srcRootWeb: string) => getTagName(node) === 'img' && (getAttribute(node, 'src')?.startsWith(srcRoot) || getAttribute(node, 'src')?.startsWith(srcRootWeb));
const hasSrcSet = (node: parse5.Element) => tagsWithSrcSet.has(getTagName(node)) && !!getAttribute(node, 'srcset'); const hasSrcSet = (node: parse5.Element) => tagsWithSrcSet.has(getTagName(node)) && !!getAttribute(node, 'srcset');
const isHoistedScript = (node: parse5.Element) => getTagName(node) === 'script' && hasAttribute(node, 'hoist'); const isHoistedScript = (node: parse5.Element) => getTagName(node) === 'script' && hasAttribute(node, 'hoist');
@ -52,7 +52,10 @@ interface PluginOptions {
export function rollupPluginAstroBuildHTML(options: PluginOptions): VitePlugin { export function rollupPluginAstroBuildHTML(options: PluginOptions): VitePlugin {
const { astroConfig, astroStyleMap, astroPageStyleMap, chunkToReferenceIdMap, pureCSSChunks, logging, origin, allPages, routeCache, viteServer, pageNames } = options; const { astroConfig, astroStyleMap, astroPageStyleMap, chunkToReferenceIdMap, pureCSSChunks, logging, origin, allPages, routeCache, viteServer, pageNames } = options;
// The filepath root of the src folder
const srcRoot = astroConfig.src.pathname; const srcRoot = astroConfig.src.pathname;
// The web path of the src folter
const srcRootWeb = srcRoot.substr(astroConfig.projectRoot.pathname.length - 1);
// A map of pages to rendered HTML // A map of pages to rendered HTML
const renderedPageMap = new Map<string, string>(); const renderedPageMap = new Map<string, string>();
@ -135,12 +138,12 @@ export function rollupPluginAstroBuildHTML(options: PluginOptions): VitePlugin {
const assetImports = []; const assetImports = [];
for (let node of findAssets(document)) { for (let node of findAssets(document)) {
if (isBuildableLink(node, srcRoot)) { if (isBuildableLink(node, srcRoot, srcRootWeb)) {
const href = getAttribute(node, 'href')!; const href = getAttribute(node, 'href')!;
assetImports.push(href); assetImports.push(href);
} }
if (isBuildableImage(node, srcRoot)) { if (isBuildableImage(node, srcRoot, srcRootWeb)) {
const src = getAttribute(node, 'src'); const src = getAttribute(node, 'src');
if (src?.startsWith(srcRoot) && !astroAssetMap.has(src)) { if (src?.startsWith(srcRoot) && !astroAssetMap.has(src)) {
astroAssetMap.set(src, fs.readFile(new URL(`file://${src}`))); astroAssetMap.set(src, fs.readFile(new URL(`file://${src}`)));
@ -356,7 +359,7 @@ export function rollupPluginAstroBuildHTML(options: PluginOptions): VitePlugin {
const styleId = getAstroPageStyleId(pathname); const styleId = getAstroPageStyleId(pathname);
let pageCSSAdded = false; let pageCSSAdded = false;
for (const node of findAssets(document)) { for (const node of findAssets(document)) {
if (isBuildableLink(node, srcRoot)) { if (isBuildableLink(node, srcRoot, srcRootWeb)) {
const rel = getAttribute(node, 'rel'); const rel = getAttribute(node, 'rel');
switch (rel) { switch (rel) {
case 'stylesheet': { case 'stylesheet': {
@ -378,7 +381,7 @@ export function rollupPluginAstroBuildHTML(options: PluginOptions): VitePlugin {
} }
} }
if (isBuildableImage(node, srcRoot)) { if (isBuildableImage(node, srcRoot, srcRootWeb)) {
const src = getAttribute(node, 'src')!; const src = getAttribute(node, 'src')!;
const referenceId = assetIdMap.get(src); const referenceId = assetIdMap.get(src);
if (referenceId) { if (referenceId) {