1
0
mirror of https://github.com/algora-io/tv.git synced 2025-04-07 06:49:52 +02:00
algora-tv/lib/algora_web/live/chat_live.ex
2024-05-20 16:16:47 +03:00

257 lines
7.6 KiB
Elixir

defmodule AlgoraWeb.ChatLive do
use AlgoraWeb, :live_view
require Logger
alias Algora.{Accounts, Library, Storage, Chat}
alias AlgoraWeb.{LayoutComponent, Presence}
def render(assigns) do
assigns = assigns |> assign(tabs: [:chat])
~H"""
<aside id="side-panel" class="w-[400px] rounded ring-1 ring-purple-300 m-1 overflow-hidden">
<div>
<div :if={@channel.solving_challenge} class="bg-black px-4 py-4 rounded text-center">
<div class="font-medium text-base">
<.link
href="https://console.algora.io/challenges/tsperf"
class="font-semibold text-green-300 hover:underline"
>
Solving the $15,000 TSPerf Challenge
</.link>
</div>
<div class="pt-1.5 font-medium text-sm">
sponsored by
</div>
<div class="pt-2.5 mx-auto grid max-w-6xl gap-4 text-center grid-cols-3">
<a
target="_blank"
rel="noopener"
class="flex h-full flex-1 flex-col items-center text-white no-underline hover:no-underline"
href="https://unkey.com"
>
<img
src="https://console.algora.io/banners/unkey.png"
alt="Unkey"
class="-mt-1 h-8 w-auto saturate-0"
/>
</a>
<a
target="_blank"
rel="noopener"
class="flex h-full flex-1 flex-col items-center text-white no-underline hover:no-underline"
href="https://scalar.com"
>
<img
src="https://console.algora.io/banners/scalar.png"
alt="Scalar"
class="h-6 w-auto saturate-0"
/>
</a>
<a
target="_blank"
rel="noopener"
class="flex h-full flex-1 flex-col items-center text-white no-underline hover:no-underline"
href="https://tigrisdata.com"
>
<img
src="https://assets-global.website-files.com/657988158c7fb30f4d9ef37b/657990b61fd3a5d674cf2298_tigris-logo.svg"
alt="Tigris"
class="mt-1 h-6 w-auto saturate-0"
/>
</a>
</div>
</div>
<div>
<div
:for={{tab, i} <- Enum.with_index(@tabs)}
id={"side-panel-content-#{tab}"}
class={["side-panel-content", i != 0 && "hidden"]}
>
<div>
<div
id="chat-messages"
phx-hook="Chat"
phx-update="stream"
class="text-sm break-words flex-1 scrollbar-thin overflow-y-auto inset-0 h-[400px] py-4"
>
<div :for={{id, message} <- @streams.messages} id={id} class="px-4">
<span class={"font-semibold #{if(system_message?(message), do: "text-emerald-400", else: "text-indigo-400")}"}>
<%= message.sender_handle %>:
</span>
<span class="font-medium text-gray-100">
<%= message.body %>
</span>
<.live_component
module={AlgoraWeb.ChatMessageTimestampComponent}
id={"timestamp-#{message.id}"}
message={message}
/>
</div>
</div>
</div>
</div>
</div>
</div>
</aside>
"""
end
def mount(%{"channel_handle" => channel_handle, "video_id" => video_id}, _session, socket) do
channel =
Accounts.get_user_by!(handle: channel_handle)
|> Library.get_channel!()
video = Library.get_video!(video_id)
if connected?(socket) do
Library.subscribe_to_livestreams()
Library.subscribe_to_channel(channel)
Chat.subscribe_to_room(video)
Presence.subscribe(channel_handle)
schedule_timestamp_tick()
end
videos = Library.list_channel_videos(channel, 50)
subtitles = Library.list_subtitles(%Library.Video{id: video_id})
data = %{}
{:ok, encoded_subtitles} =
subtitles
|> Enum.map(&%{id: &1.id, start: &1.start, end: &1.end, body: &1.body})
|> Jason.encode(pretty: true)
types = %{subtitles: :string}
params = %{subtitles: encoded_subtitles}
changeset =
{data, types}
|> Ecto.Changeset.cast(params, Map.keys(types))
socket =
socket
|> assign(
channel: channel,
videos_count: Enum.count(videos),
video: video,
subtitles: subtitles
)
|> assign_form(changeset)
|> stream(:videos, videos)
|> stream(:messages, Chat.list_messages(video))
|> stream(:presences, Presence.list_online_users(channel_handle))
if connected?(socket), do: send(self(), {:play, video})
{:ok, socket}
end
def handle_params(params, _url, socket) do
LayoutComponent.hide_modal()
{:noreply, socket |> apply_action(socket.assigns.live_action, params)}
end
def handle_info({:play, video}, socket) do
socket = socket |> push_event("join_chat", %{id: video.id})
{:noreply, socket}
end
def handle_info({Presence, {:join, presence}}, socket) do
{:noreply, stream_insert(socket, :presences, presence)}
end
def handle_info({Presence, {:leave, presence}}, socket) do
if presence.metas == [] do
{:noreply, stream_delete(socket, :presences, presence)}
else
{:noreply, stream_insert(socket, :presences, presence)}
end
end
def handle_info(
{Storage, %Library.Events.ThumbnailsGenerated{video: video}},
socket
) do
{:noreply,
if video.user_id == socket.assigns.channel.user_id do
socket
|> stream_insert(:videos, video, at: 0)
else
socket
end}
end
def handle_info(
{Library, %Library.Events.LivestreamStarted{video: video}},
socket
) do
%{channel: channel} = socket.assigns
{:noreply,
if video.user_id == channel.user_id do
socket
|> assign(channel: %{channel | is_live: true})
|> stream_insert(:videos, video, at: 0)
else
socket
end}
end
def handle_info(
{Library, %Library.Events.LivestreamEnded{video: video}},
socket
) do
%{channel: channel} = socket.assigns
{:noreply,
if video.user_id == channel.user_id do
socket
|> assign(channel: %{channel | is_live: false})
|> stream_insert(:videos, video)
else
socket
end}
end
def handle_info(
{Chat, %Chat.Events.MessageDeleted{message: message}},
socket
) do
{:noreply, socket |> stream_delete(:messages, message)}
end
def handle_info({Chat, %Chat.Events.MessageSent{message: message}}, socket) do
{:noreply, socket |> stream_insert(:messages, message)}
end
def handle_info({Library, _}, socket), do: {:noreply, socket}
def handle_info(:tick, socket) do
send_update(AlgoraWeb.ChatMessageTimestampComponent, %{})
schedule_timestamp_tick()
{:noreply, socket}
end
defp schedule_timestamp_tick() do
Process.send_after(self(), :tick, :timer.minutes(1))
end
defp system_message?(%Chat.Message{} = message) do
message.sender_handle == "algora"
end
defp assign_form(socket, %Ecto.Changeset{} = changeset) do
assign(socket, :form, to_form(changeset, as: :data))
end
defp apply_action(socket, :show, params) do
socket
|> assign(:page_title, socket.assigns.channel.name || params["channel_handle"])
|> assign(:page_description, socket.assigns.video.title)
|> assign(:page_image, Library.get_og_image_url(socket.assigns.video))
end
end