distefam

distefam

Updating individual stream items

I have a table view of items that I am rendering using Streams in a Live Component. The component fetches the list of items and updates the socket Streams in update/2 and everything works as expected.

However, I want to allow users to select multiple items in the table view and update the styling of the row when selected (adding a checkmark and changing the background color, for example).

I’ve attempted to do this two ways:

  1. Maintain a separate Map in assigns called selected_items that looks like %{item_id: true} if an item is selected. I then simply use :if={Map.get(@selected_items, item.id, false) to hide/show some element.

The problem with this approach, of course, is that updating selected_items will not result in updating the table row since the Stream is not updated. So, while this renders correctly on initial load it is not responsive to user input.

  1. Use Enum.map to add a selected field to the Stream item’s map. This works great, however, the issue is updating…

To update an individual Stream item you must call stream_insert, which requires that I have the full item to insert. Since selecting or deselecting an individual row will be a common action and one done in rapid succession, it doesn’t feel like a great idea to repeatedly have to query and update the database just to toggle a selected state, especially since this will not be committed to the database until the user clicks “save”.

So, is there any way to handle this use case using streams? Or, do I need to use a regular assigns with manual pagination, etc.?

Marked As Solved

distefam

distefam

To solve this, I had to use both JS.push and style children based on parent classes.

Here’s a simplified version of the table:

<tbody id="items" phx-update="stream">
  <tr 
    :for={{dom_id, item} <- @streams.items}
    id={dom_id}
    phx-click={select_item(@myself)}
    phx-value-id={item.id}
    class={"group #{if MapSet.member?(@selected_items, item), do: "selected"} ..."}
  >
    <td>
      <.icon class="hidden group-[.selected]:block" name="hero-check-circle" />
    </td>
  </tr>
</tbody> 

note: selected is a custom Tailwind class I created in app.css, which adds relevant background color and other styles to selected table rows.

Then, the private select_item/1 function that is invoked by the phx-click action on the table row above:

defp select_item(target) do
  JS.push("select-item", target: target)
  |> JS.toggle_class("selected")
end

This was the crucial thing: having the click event trigger a local function that performs the client-side JS and triggers the server-side event that will be handled by handle_event/3 rather than simply using handle_event/3 directly.

Finally, the event handler:

def handle_event("select-item", params, socket) do
  %{"id" => id} = params
  socket =
    update(socket, :selected_items, fn selected_items ->
      if MapSet.member?(selected_items, id),
        do: MapSet.delete(selected_items, id),
        else: MapSet.put(selected_items, id)
    end)
end

Where Next?

Popular in Questions Top

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
pmjoe
I have a relationship of love and hate with Elixir. Lots of things are just absolutely right, but there are some things that are kind of ...
New
jononomo
I am trying to figure out how Mix knows whether the environment is test, dev, or prod – where is this set? Thanks.
New
Emily
I have VueJS GUIs with the project generated using Webpack. I have Elixir modules that will need to be used by the VueJS GUIs. I forese...
New
LegitStack
I’m trying to make a websocket server in Phoenix or raw Elixir. I heard about gun, I think I could use cowboy, but since I’m not that sma...
New
dblack
I’ve got an issue with an app and I’ve no idea of how to troubleshoot it. I’m hoping someone here might have seen something similar. I p...
New
nsuchy
Hi. I’ve noticed that Windows Powershell has it’s own IEX command and you cannot access Elixir’s IEX due to the conflict. This isn’t a cr...
New
shijith.k
I am trying to start a new phoenix project with elixir 1.9, but mix phx.new does not work. It says that ** (Mix) The task "phx.new" could...
New
jononomo
For some reason my phoenix channels are working for me in my local dev environment, but as soon as I deploy via Docker, I get a 403 error...
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

Other popular topics Top

JakeBecker
TL;DR: I’ve just released an implementation of Microsoft’s IDE-independent Language Server Protocol for Elixir. It adds language support ...
1144 53690 245
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
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
ovidiubadita
Hey all, I discovered Elixir and I love it. I always wanted to learn a functional programming and I intended to go for Haskell, but afte...
New
johnnyicon
Hi all, I’ve just started learning Elixir and Phoenix Framework, so please pardon my n00bness at this stage. I’m trying to use Postgres...
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
gausby
I asked this very same question on twitter and got some interesting feedback, but I thought it would be a good question to ask here as we...
1207 39297 209
New
saif
Hello everyone, Long time lurker first time poster here. I’ve recently begun working on Elixir full-time again! :raised_hands: It’s been...
New
marick
I had some trouble figuring out how to make many-to-many associations work. Once I got it working, I wrote a blog post. Because I’m a nov...
New
openscript
Hello! Sorry for this astonishing simple question, but I’m really stuck. I try to set up the intellij-elixir plugin, but I don’t know ho...
New

We're in Beta

About us Mission Statement