img tags lose props on nested components
Describe the bug
I have some some images with like:
<img
class="flex-shrink-0 mx-auto rounded-full w-32 h-32"
is="img-cache"
expire="86400"
url="https://images.unsplash.com/photo-1727713274972-d1d138ea0569?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=4&w=256&h=256&q=60"
width="100"
height="100"
alt="" />
one script I use which will run as custom element and is included globaly via index.tsx change the prop url to src which will be the cached image as database64 string like:
<img
class="flex-shrink-0 mx-auto rounded-full w-32 h-32"
is="img-cache"
expire="86400"
url="https://images.unsplash.com/photo-1727713274972-d1d138ea0569?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjEyMDd9&auto=format&fit=facearea&facepad=4&w=256&h=256&q=60"
width="100"
height="100"
alt=""
src="">
Also I have this in use:
declare module "solid-js" {
namespace JSX {
interface ImgHTMLAttributes<T> {
is?: string;
expire?: string | number;
url?: string;
lazy?: boolean;
}
}
}
Once i use it in the "main" component that is directly called from the Router > Route I get the url prop and also the src attached from my script. But once I use it in some child components, the url will never rendered.
output will be like
https://..... URL OF IMAGE....
<img is="img-cache" expire="86400" width="100" height="100" class="rounded-md w-16 h-16" alt="user avatar">
origin code to debug like:
<Show when={user.data}>
{(user) => (
<div class="flex flex-col items-center gap-3">
<h3 class="font-bold">Hi {user().displayName}!</h3>
<Show when={user().photoURL}>
{user().photoURL}
<img is="img-cache" expire="86400" url={user().photoURL} width="100" height="100" class="rounded-md w-16 h-16" alt="user avatar" />
</Show>
<p>Your userID is {user().uid}</p>
<Logout />
</div>
)}
</Show>
I have checked it multiple times, the variable is there to fill the url prop and as it works on the "main" component called from the route directly, it should be a Bug, when it is not working on nested components.
Your Example Website or App
Steps to Reproduce the Bug or Issue
all described already in "Describe the bug" block
Expected behavior
the <img/> tag should be rendered the same way as in the main component.
Screenshots or Videos
Does not work in nested Components:
working: direct called component via route:
Platform
- OS: [e.g. macOS, Windows, Linux] all three
- Browser: [e.g. Chrome, Safari, Firefox] chrome, chomium, firefox
- Version: [e.g. 91.1] latest currently 129.0.6668.60 from chome
Additional context
https://github.com/solidjs/solid/discussions/2317
EDIT: 2024-10-02 10:36 CEST
Also the Props data-*="xxx" will get removes on child components but will in output on "main" component called via route.
EDIT 2024-10-02 11:03 CEST
It looks only for custom <img is="img-cache".../> to be the problem
EDIT: 2024-10-02 12:46 CEST
Add demo app build with sourcemap demoapp.zip
It looks like in Profile135415457278656946752.js it sets the values, but because it is not set via setAttribute("url", ... is does not rendering here in the nested component. r.url=e.e=t would not put the content.
But while the other templates are rendered differently when called directly it is only on the nested components.
Update 2024-10-02 13:06 CEST build without minify enabled: demoapp2.zip
Here my TypeScript for the Image Cache:
Sorry if the script is based for mobile app building with capacitor. But it works as well while developing on the normal browser.
import { Directory, Encoding, Filesystem } from '@capacitor/filesystem';
import { Preferences } from '@capacitor/preferences';
import path from 'path';
const CACHE_KEY = 'cached-images';
let observer;
let cacheData;
let saveRequired;
class ImageCache extends HTMLImageElement {
constructor() {
super();
if (this.hasAttribute('clear')) {
clearCacheData();
}
if (this.hasAttribute('lazy')) {
observer.observe(this);
}
else {
setSource(this);
}
}
}
(function () {
observer = new IntersectionObserver(intersectionCallback);
window.customElements.define('img-cache', ImageCache, { extends: 'img' });
requestIdleCallback();
})();
function intersectionCallback(entries, observer) {
entries.forEach(entry => {
console.log(entry);
if (entry.isIntersecting /*&& !entry.target.src*/ && entry.target.url) {
setSource(entry.target);
}
});
}
function requestIdleCallback() {
window.requestIdleCallback(saveCacheData, { timeout: 3000 });
}
function setSource(node) {
let url = node.getAttribute('url');
let mySrc = node.getAttribute('src');
let expire = node.getAttribute('expire');
console.log("setSource", url, mySrc, expire)
if (!url) {
url = mySrc;
}
console.log("setSource", url, expire)
makeLink(url, expire).then(link => node.src = link);
}
async function clearCacheData() {
await Preferences.remove({ key: CACHE_KEY });
cacheData = [];
}
async function loadCacheData() {
let obj = await Preferences.get({ key: CACHE_KEY });
cacheData = (obj.value ? JSON.parse(obj.value) : []);
}
async function saveCacheData() {
if (saveRequired) {
saveRequired = false;
await Preferences.set({ key: CACHE_KEY, value: JSON.stringify(cacheData) });
}
requestIdleCallback();
}
async function makeLink(url, expiration) {
if (!cacheData) {
await loadCacheData();
}
let index = findIndex(url);
let isCached = index != -1;
if (isCached && isExpired(cacheData[index].expiration)) {
await deleteFile(cacheData[index].file);
removeFromList(index);
isCached = false;
}
try {
if (!isCached) {
throw new Error();
}
return await readFile(cacheData[index].file);
}
catch (error) {
return await downloadImage(url, expiration);
}
}
function findIndex(url) {
return cacheData.findIndex(entry => entry.url == url);
}
function isExpired(expiration) {
return expiration && Date.now() > (new Date(expiration)).getTime();
}
async function readFile(filename) {
let contents = await Filesystem.readFile({ path: filename, directory: Directory.Cache, encoding: Encoding.UTF8 });
return contents.data;
}
async function writeFile(filename, data) {
return await Filesystem.writeFile({ path: filename, data: data, directory: Directory.Cache, encoding: Encoding.UTF8 });
}
async function deleteFile(filename) {
return await Filesystem.deleteFile({ path: filename, directory: Directory.Cache });
}
function removeFromList(index) {
cacheData.splice(index, 1);
saveRequired = true;
}
function addToList(url, filename, expiration) {
let date = (expiration ? (new Date(Date.now() + expiration * 60000)).toISOString() : null);
cacheData[cacheData.length] = { file: filename, expiration: date, url: url };
saveRequired = true;
}
async function downloadImage(url, expiration) {
let response = await fetch(url);
let blob = await response.blob();
let link = await blobToBase64(blob);
let filename = uid() + path.extname(url);
await writeFile(filename, link);
addToList(url, filename, expiration);
return link;
}
function blobToBase64(blob) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onloadend = () => resolve(reader.result);
reader.readAsDataURL(blob);
});
}
function uid() {
return Math.random().toString().substr(-8) + Date.now().toString().substr(-12);
}
When in need to control if something is a prop or an attribute, you can use any of the following: https://docs.solidjs.com/reference/jsx-attributes/attr https://docs.solidjs.com/reference/jsx-attributes/prop https://docs.solidjs.com/reference/jsx-attributes/bool
Trying to guess, I suppose url is set as a prop instead of as an attribute. Id try changing the bits of const url = img.getAttribute(...) for const url = img.url
A playground repro would be welcome https://playground.solidjs.com/
@titoBouzout thank you, I have now created a own Image Tag, that catch the informations via
import { makeLink } from "@lib/imgCache";
import { ComponentProps, createEffect, createSignal, JSX, onMount, ParentComponent, Show, splitProps } from "solid-js";
const ImgCached: ParentComponent<
ComponentProps<"img"> & {
is?: string | undefined;
expire?: string | number | undefined;
url?: string | undefined;
lazy?: boolean | undefined;
}
> = (props) => {
const [isImageLoaded, setImageLoaded] = createSignal(false);
const [local, attrs] = splitProps(props, ['is', 'expire', 'url', 'lazy']);
let url: string;
let returnURL: string;
let expire: any;
onMount(async () => {
if (local.url) {
url = local.url;
} else if (attrs.src) {
url = attrs.src;
}
if (local.expire) {
expire = local.expire;
}
await makeLink(url, expire).then(link => {
attrs.src = returnURL = link as string;
setImageLoaded(true);
}).catch((e) => console.error(e));
})
return <>
<Show when={isImageLoaded()} >
<img {...attrs} {...local} src={returnURL} />
</Show>
</>
;
};
export {
ImgCached
};
this works for now, but I still have problems with the attr:mynewAttribute=120 because its not rendered or in my setup it dont find the types... I'm not sure, I use vite with rollup and typescript.
I'm missing maybe as its my first time with solidjs some examples that are showing how it works, the ones I have checked and tried to update, where all 2+ years old and broken...
this works for now, but I still have problems with the attr:mynewAttribute=120 because its not rendered or in my setup it dont find the types... I'm not sure, I use vite with rollup and typescript.
Attributes are case-insensitive, mynewAttribute will become mynewattribute.
I cannot seem to be able to reproduce, the attributes are rendered https://playground.solidjs.com/anonymous/8cd98070-c00f-42ca-ac2a-b46a8d92fb20
Included in the link an example on how to type attr: and theres a PR for adding is to HTMLAttributes typings https://github.com/ryansolid/dom-expressions/pull/363/files
@Dexus Is this still an issue?
Closed as stale as above it appears it can't be reproduced. Reopen if still an issue.