solid icon indicating copy to clipboard operation
solid copied to clipboard

img tags lose props on nested components

Open Dexus opened this issue 1 year ago • 4 comments

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&amp;ixid=eyJhcHBfaWQiOjEyMDd9&amp;auto=format&amp;fit=facearea&amp;facepad=4&amp;w=256&amp;h=256&amp;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: image image

working: direct called component via route: image

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

Dexus avatar Oct 02 '24 07:10 Dexus

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);
}

Dexus avatar Oct 02 '24 10:10 Dexus

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 avatar Oct 02 '24 16:10 titoBouzout

@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...

Dexus avatar Oct 02 '24 19:10 Dexus

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

titoBouzout avatar Oct 04 '24 15:10 titoBouzout

@Dexus Is this still an issue?

titoBouzout avatar Oct 21 '24 00:10 titoBouzout

Closed as stale as above it appears it can't be reproduced. Reopen if still an issue.

ryansolid avatar Apr 30 '25 20:04 ryansolid