SaladUI – Implement avatar component for Phoenix LiveView

RMAG news

This post is written after I implement Avatar component for SaladUI component library for Phoenix Liveview.

An avatar component is quite simple, but I want to enhance it with fallback avatar text. The template structure is as following:

<.avatar>
<.avatar_image src=“./my-profile-img.jpg”></.avatar_image>
<.avatar_fallback class=“bg-primary text-white”>CN</.avatar_fallback>
<.avatar>

Here is an implementation of avatar component in html

<span class=“relative rounded-full overflow-hidden w-10 h-10”>
<img class=“aspect-square w-full h-full” src=“https://github.com/shadcn.png”>
<span class=“flex rounded-full bg-primary text-white items-center justify-center w-full h-full”>CN</span>
</span>

It’s the happy case, the image exist, and the fallback element is push down and hidden due to class overflow-hidden.

But when the image doesn’t exist, the broken image is displayed.

Hmm, the image should be hidden if it does not exist or got error while loading. Fortunately, img provide onerror event.

<span class=“relative rounded-full overflow-hidden w-10 h-10”>
<img class=“aspect-square w-full h-full” src=“badimage.png” onerror=“this.style.display=’none'”>
<span class=“flex rounded-full bg-primary text-white items-center justify-center w-full h-full”>CN</span>
</span>

Guest what. Nothing changed, the broken image is still visible.
It took me a while to discover the reason. onerror=”this.style.display=’none'” change attribute on client side, when Phoenix LiveView update, it patch the html and remove the display style value. So just add phx-update=”ignore” and you got the error image hidden.

<span class=“relative rounded-full overflow-hidden w-10 h-10”>
<img class=“aspect-square w-full h-full” src=“https://github.com/shadcn.png” onerror=“this.style.display=’none'” phx-update=“ignore”>
<span class=“flex rounded-full bg-primary text-white items-center justify-center w-full h-full”>CN</span>
</span>

And it works as expected

Now wrap up the component

defmodule SaladUI.Avatar do
@moduledoc false
use Phoenix.Component

attr(:class, :string, default: nil)
attr(:rest, :global)

def avatar(assigns) do
~H””
<span class={classes([“
relative h10 w10 overflowhidden roundedfull“, @class])} {@rest}>
<%= render_slot(@inner_block) %>
</span>
“””

end

attr(:class, :string, default: nil)
attr(:rest, :global)

def avatar_image(assigns) do
~H””
<img
class={classes([“
aspectsquare hfull wfull“, @class])}
{@rest}
phx-update=”
ignore
style=”
display:none
onload=”
this.style.display=
/>
“””

end

attr(:class, :string, default: nil)
attr(:rest, :global)
slot(:inner_block, required: false)

def avatar_fallback(assigns) do
~H””
<span
class={
classes([“
flex hfull wfull itemscenter justifycenter roundedfull bgmuted“, @class])
}
{@rest}
>
<%= render_slot(@inner_block) %>
</span>
“””

end
end

You may notice that I use onload event instead of onerror:

style=”display:none”
onload=”this.style.display=””

Using onerror browser waits until the image loading complete to decide if there is any error then trigger the event. This cause the white avatar while loading image. Using onload event to show image when the loading process complete, so by default if the image loading is slow, the fallback avatar still displays.

If you want to learn more, then visit my github repo https://github.com/bluzky/salad_ui.

Thanks for reading.

Please follow and like us:
Pin Share