Menkir

Menkir

Trigger CSS transitions from server side

Hi guys.
I’m currently working with the new LV JS API and it’s pretty cool. However i have a case where i need to trigger transitions of html elements from the server side. e.g.: A Notification that pops up after the last action was verified by the backend. Something that happens quite often.

I came up with a solution with a hook that triggers the transition by clicking on a hidden button.

Here a minimal example in surface:

LV template

...
<Notification id="notify-fail" 
type="error" 
title={@notitication} 
message={@message}/>

<Submit/>
...

LV callback

def handle_event("submit", params, socket) do
    socket = if valid?(params) do
         redirect(socket, to: "/foo")
    else
        socket 
        |> assign(title: "Something went wrong", message: "Pls check your foo")
        |> Notification.open("notify-fail")
    end
    {:noreply, socket}
end

Notification Component

defmodule Notification do
    use Surface.LiveComponent
    alias Phoenix.LiveView.JS

    prop message, :string
    prop title, :string

    def render(assigns) do
        ~F"""
            <div id={@id} class="..." :hook="Notification">
                <button id={@id <> "-trigger"} type="button" hidden :on-click={__MODULE__.show(@id)}></button>
               {!-- some template stuff  with @message and @string --}
            </div>
        """
    end

    # clientside open
    def show(id, js \\ %JS{}),
    do: JS.show(js, to: "##{id}", transition: {"ease-out duration-300", "opacity-0", "opacity-100"})

    # serverside open
    def open(socket, id) do
        socket |> push_event("open", %{id: id})
    end
    ...
end

Notification Hook

...
mounted() {
    this.handleEvent("open", ({id}) => {
        if(id == this.el.id){
            this.el.querySelector(`#${id}-trigger`).click()
        }
    })
}
...

Imo this solution seems like a dirty workaround because it created the necessity of writing an extra hook as well as providing two additional functions in the module doing basically the same thing . Will there be a feature handling this behaviour? Or do you know a better way of solving this issue?

greets

Most Liked

DecoyLexi

DecoyLexi

I had a similar problem and managed to solve it without using hooks. In addition to pushing events to hooks, LiveView also dispatches events sent with push_event/3 to the window with a phx: prefix.

live_helpers.ex:

def push_js(socket, js \\ %JS{}) do
  cmd = Jason.encode!(js.ops) # LiveView expects this to be a string when we call `execJS` on the client

  socket
  |> push_event("exec_js", %{id: socket.id, cmd: cmd})
end

app.js:

window.addEventListener("phx:exec_js", (e) => {
  let el = document.getElementById(e.detail.id)

  liveSocket.execJS(el, e.detail.cmd)
})

Then in your LiveView, you can just do

def open(socket, id) do  
  socket
  |> push_js(show(id))
end

There might be a better way to do this, but I haven’t found one yet.

riebeekn

riebeekn

I think hooks are currently the only option when wanting to call javascript from the server. I’d probably just put the necessary plain javascript in the hook however instead of going with the hidden button technique.

CherryPoppins

CherryPoppins

You can checkout Apline.js to handle stuff like this. I’ve found it works pretty good with typical controllers/templates, but in my experience it can be buggy in liveviews and I’ve only had limited success there before giving up on it. It’s been a few months though since i tried it out so maybe they have worked out some of the liveview issues where you seemingly lose your Alpine instance on DOM updates.

Where Next?

Popular in Questions Top

vertexbuffer
Hello, can anybody help here..? I have a list of players and I what to delete an element, but every for loop the list is reverting to ori...
New
siddhant3030
Hi, I have to write a raw query for one of my project. But till now I have used ecto queries and don’t have much experience writing raw ...
New
greenz1
I have a phoenix application from which a user can download multiple(5-6) files of size 1MB. I couldn’t find anything related to sending ...
New
electic
Hi, I am new to Elixir. I am trying to use the DateTime component to insert a date into MySQL however the there seems to be no way to fo...
New
Kurisu
For example for a current url like http://localhost:4000/cosmetic/products?_utf8=✓&amp;query=perfume&amp;page=2, I would like to get: ...
New
stefanluptak
Hello everybody, usually, I use a 29" ultra-wide monitor for VSCode which can easily accomodate explorer (files panel) + file with code ...
New
vonH
When I run the Plug and I recompile I wind up having to use Ctrl C to quit iex and start again. Witht the help of rlwrap I can use the cu...
New
aalberti333
As the title describes, I’m trying to run Enum.map() over a list of key/value pairs, where the value is a map. My data looks like this: ...
New
Qqwy
Original source of discussion: This topic on the Pragmatic Programmers' Functional Web Development with Elixir, OTP, and Phoenix forum. ...
New
romenigld
I am trying to run a deploy with docker and I successfully runned with this command: docker build -t romenigld/blog-prod . but when I t...
New

Other popular topics Top

sorentwo
Hello! tl;dr Announcing Oban, an Ecto based job processing library with a focus on reliability and historical observability. After spen...
985 42842 311
New
Harrisonl
We have an ECS cluster with 4 services, where each task joins a single cluster, via discovery ECS discovery service. Currently when I de...
New
mcarvalho
What is the difference between System.get_env and Application.get_env? For example, what are best practices to use one versus another.
New
skosch
To my knowledge, put_in, Map.update etc. all have the one limitation of not automatically creating intermediate keys when needed (for exa...
New
chrismccord
Phoenix 1.4.0 released Phoenix 1.4 is out! This release ships with exciting new features, most notably with HTTP2 support, improved deve...
688 30840 112
New
chrismccord
As promised, the first release candidate of Phoenix 1.3.0 is out! This release focuses on code generators with improved project structure...
New
stefanchrobot
What’s the safe way to decode a JSON string into a struct? I want to avoid calling String.to_atom. Jason.decode can give me a map with st...
New
vrod
I am using the Starship cross-shell prompt – it seems pretty nice, but I get some errors: [WARN] - (starship::utils): Executing command ...
New
bsollish-terakeet
Credo is smart enough to check for (something like) this: assert length(the_list) == 0 with this response: Checking if an enum is empt...
New
vonH
In asking this question I am more interested about the expressiveness of the language itself and less concerned about the availability of...
New

We're in Beta

About us Mission Statement