mirror of
https://github.com/livebook-dev/livebook.git
synced 2024-11-11 01:28:47 +08:00
5c8e117800
* Add initial user config modal * Assign user ids * Update session data to hold user ids * Get users list for specific ids * Render user avatar * User update * Refactor user changes * Subscribe to individual user updates * Show users in side panel * Add sidebar to homepage * Don't generate the same color twice in a row * Add documentation notes * Fix tests * Add tests * Keep users in session data * Rename color to hex_color
98 lines
3.3 KiB
Elixir
98 lines
3.3 KiB
Elixir
defmodule LivebookWeb.UserComponent do
|
|
use LivebookWeb, :live_component
|
|
|
|
import LivebookWeb.UserHelpers
|
|
|
|
alias Livebook.Users.User
|
|
|
|
@impl true
|
|
def update(assigns, socket) do
|
|
socket = assign(socket, assigns)
|
|
user = socket.assigns.user
|
|
|
|
{:ok, assign(socket, data: user_to_data(user), valid: true, preview_user: user)}
|
|
end
|
|
|
|
defp user_to_data(user) do
|
|
%{"name" => user.name || "", "hex_color" => user.hex_color}
|
|
end
|
|
|
|
@impl true
|
|
def render(assigns) do
|
|
~L"""
|
|
<div class="p-6 flex flex-col space-y-5">
|
|
<h3 class="text-2xl font-semibold text-gray-800">
|
|
User profile
|
|
</h3>
|
|
<div class="flex justify-center">
|
|
<%= render_user_avatar(@preview_user, class: "h-20 w-20", text_class: "text-3xl") %>
|
|
</div>
|
|
<%= f = form_for :data, "#",
|
|
id: "user_form",
|
|
phx_target: @myself,
|
|
phx_submit: "save",
|
|
phx_change: "validate",
|
|
phx_hook: "UserForm" %>
|
|
<div class="flex flex-col space-y-5">
|
|
<div>
|
|
<div class="input-label">Display name</div>
|
|
<%= text_input f, :name, value: @data["name"], class: "input", spellcheck: "false" %>
|
|
</div>
|
|
<div>
|
|
<div class="input-label">Cursor color</div>
|
|
<div class="flex space-x-4 items-center">
|
|
<div class="border-[3px] rounded-lg p-1 flex justify-center items-center"
|
|
style="border-color: <%= @preview_user.hex_color %>">
|
|
<div class="rounded h-5 w-5"
|
|
style="background-color: <%= @preview_user.hex_color %>">
|
|
</div>
|
|
</div>
|
|
<div class="relative flex-grow">
|
|
<%= text_input f, :hex_color, value: @data["hex_color"], class: "input", spellcheck: "false", maxlength: 7 %>
|
|
<%= tag :button, class: "icon-button absolute right-2 top-1",
|
|
type: "button",
|
|
phx_click: "randomize_color",
|
|
phx_target: @myself %>
|
|
<%= remix_icon("refresh-line", class: "text-xl") %>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<%= tag :button, class: "button button-blue flex space-x-1 justify-center items-center",
|
|
type: "submit",
|
|
disabled: not @valid %>
|
|
<%= remix_icon("save-line") %>
|
|
<span>Save</span>
|
|
</button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
"""
|
|
end
|
|
|
|
@impl true
|
|
def handle_event("randomize_color", %{}, socket) do
|
|
data = %{
|
|
socket.assigns.data
|
|
| "hex_color" => User.random_hex_color(except: [socket.assigns.preview_user.hex_color])
|
|
}
|
|
|
|
handle_event("validate", %{"data" => data}, socket)
|
|
end
|
|
|
|
def handle_event("validate", %{"data" => data}, socket) do
|
|
{valid, user} =
|
|
case User.change(socket.assigns.user, data) do
|
|
{:ok, user} -> {true, user}
|
|
{:error, _errors, user} -> {false, user}
|
|
end
|
|
|
|
{:noreply, assign(socket, data: data, valid: valid, preview_user: user)}
|
|
end
|
|
|
|
def handle_event("save", %{"data" => data}, socket) do
|
|
{:ok, user} = User.change(socket.assigns.user, data)
|
|
Livebook.Users.broadcast_change(user)
|
|
{:noreply, push_patch(socket, to: socket.assigns.return_to)}
|
|
end
|
|
end
|