Push JS Commands to the client

I’d like to do sth in the line of

JS.push_to_client( socket, JS.set_attribute({"some-attr", "some-value"}, to: "#some-id")} ) 

What would be the best way to accomplish that?

Background

I try to get webcomponents to work with LV.
One of the major pain points are attributes that are changed by the components, most common example is sth like @open for a drawer.

So it’s not ideal to do

~H""" <wc-drawer open={@is_open} ... /> """ 

because

  1. If the the drawer component changes the open-state itself, it will be overwritten with the next LV-update. This can be fixed in onBeforeElUpdated for example but it gets messy
  2. @is-open would have to be in the assigns, but it’s not really application state. It’s rather the result of the imperative drawer.open(). Also there would have to be an @is_open for each WC used.

So I think the best way would be to just command the drawer to open and forget about it afterwards.

There is push_event/3 which you can use either with a hook or a global event listener.

If you use a hook, you can use the handleEvent function to set the attribute.

const AttributeHook = { mounted() { this.handleEvent("set_some_attr", data => { this.el.setAttribute(data.attr_name, data.value) } } } 
def mount(socket) do socket = push_event(socket, "set_some_attr", %{ attr_name: "foo", value: "bar" }) {:ok, socket} end 
def render(assigns) do ~H""" <wc-drawer phx-hook="AttributeHook" /> """ end 

Or if you just want to do it on mount and not on a server event:

def render(assigns) do ~H""" <wc-drawer phx-mounted={JS.set_attribute({"some-attr", "some-value"}, to: "#some-id")} /> """ end 

Why does is_open needs to be a server state?

1 Like

Thanks, I knew that actually. I did to much JS the last days, my head hurts. :dizzy_face:

It does not. But if I set it in a function-component it has to be in assigns.
Say sth is happening in the backend and I want to open a drawer in response.
How should I do that if not set the open-state for a specific drawer in the assigns of the LV?

push_event as @benlime showed. Letting Phoenix and some front-end code share control of an attribute is not the happy path imo.

sure, that refers to the question why is_open needs to be in the state (if I do not use push, put set an attribute in a function component)

Still, I’d use push_event/3 for that. You would have a handle_event or handle_info callback in your LiveView and then call push_event so the JS hook can set the attribute accordingly (or call drawer.open() or whatever code is needed to open the drawer client-side).