260 lines
9.9 KiB
Plaintext
260 lines
9.9 KiB
Plaintext
<div class="flex space-x-3 w-full neomorph is-nxxsmall rounded-xl @CssContainer">
|
|
|
|
<div class="flex flex-col py-3 pl-3 md:py-4 md:pl-4 flex-none rounded-l-xl bg-cover bg-no-repeat bg-center"
|
|
style="background-image:linear-gradient(to left, var(--background), transparent), url('@(Message.User?.BackgroundUrl)');">
|
|
|
|
<a class="block h-12 w-12 md:h-16 md:w-16" href="@Message.User.ProfileUrl" title="@Message.User.UserName">
|
|
<img alt="@Message.User.UserName" class="h-12 w-12 md:h-16 md:w-16 object-cover rounded-full neomorph is-nxxsmall" src="@(Message.User.PictureUrl ?? "/imgs/icon-192.png")" />
|
|
</a>
|
|
|
|
</div>
|
|
|
|
<div class="flex flex-col space-y-3 flex-1 py-3 pr-3 md:py-4 md:pr-4 min-w-0">
|
|
|
|
<div class="flex flex-col space-y-1 flex-1 min-w-0">
|
|
@if (Message.BoostingUser != null)
|
|
{
|
|
<p class="inline-flex flex-1 space-x-2 min-w-0 text-xs text-xs">
|
|
<a class="font-bold" href="@Message.BoostingUser.ProfileUrl" title="@Message.BoostingUser.UserName">@Message.BoostingUser.DisplayName</a>
|
|
<i class="ion-md-repeat has-text-info" aria-hidden="true"></i>
|
|
<span>@CascadingState.Localizer["boosted"]</span>
|
|
</p>
|
|
}
|
|
<p class="inline-flex flex-1 space-x-2 min-w-0 justify-between text-xs md:text-sm">
|
|
<span class="inline-flex space-x-2 min-w-0">
|
|
<b class="shrink truncate max-w-[80%]" title="@Message.User.DisplayName">
|
|
@Message.User.DisplayName
|
|
</b>
|
|
<a class="underline flex-1 min-w-6 opacity-50 truncate self-center text-xs" href="@Message.User.ProfileUrl" title="@Message.User.UserName">
|
|
@Message.User.UserName
|
|
</a>
|
|
</span>
|
|
<span class="flex-none inline-flex space-x-1 items-center">
|
|
<span title="@Message.CreatedAt.ToLocalTime().ToString("📅dd/MM/yy 🕒HH:mm")">
|
|
@Message.CreatedAt.GetPassedTime(CascadingState.Localizer)
|
|
</span>
|
|
<i aria-hidden="true" class="@Message.MessageType.GetMessageTypeIcon() text-md"
|
|
title="@CascadingState.Localizer[Message.MessageType.ToString()]">
|
|
</i>
|
|
</span>
|
|
</p>
|
|
@if (Message.Title is { Length: > 0 })
|
|
{
|
|
<p class="text-sm md:text-base font-bold break-all">@Message.Title</p>
|
|
}
|
|
|
|
@if (Message.Content is { Length: > 0 })
|
|
{
|
|
<div class="text-sm md:text-base break-all">
|
|
@((MarkupString)Message.Content)
|
|
</div>
|
|
}
|
|
|
|
@if (Message.Medias.Count != 0)
|
|
{
|
|
<div class="grid gap-4 auto-cols-auto grid-rows-1 grid-flow-col-dense">
|
|
@foreach (var media in Message.Medias)
|
|
{
|
|
if (media.ContentType.StartsWith("image"))
|
|
{
|
|
<a class="w-auto" href="@media.Url">
|
|
<img alt="@media.AltText" class="w-full rounded-lg @SUtility.IfTrueThen(Message.Medias.Count > 1, "max-h-[30vh]")" src="@media.Url" title="@media.AltText">
|
|
</a>
|
|
}
|
|
else if (media.ContentType.StartsWith("video"))
|
|
{
|
|
<video class="w-full max-h-[50vh] aspect-video rounded-lg mx-auto neomorphInset is-nxxsmall" controls="controls" playsinline="playsinline" preload="metadata" title="@media.FileName">
|
|
<source src="@media.Url" type="@media.ContentType" />
|
|
</video>
|
|
}
|
|
else if (media.ContentType.StartsWith("audio"))
|
|
{
|
|
<audio class="w-full max-h-8" controls="controls" preload="metadata" title="@media.FileName">
|
|
<source src="@media.Url" type="@media.ContentType" />
|
|
</audio>
|
|
}
|
|
else
|
|
{
|
|
<div class="flex items-center space-x-3 align-center rounded-lg p-3 md:p-4 neomorph is-nxxsmall">
|
|
<span>
|
|
<i class="text-2xl ion-md-document"></i>
|
|
</span>
|
|
<div class="flex flex-col w-full space-y-1">
|
|
<p class="text-xs md:text-sm break-all">
|
|
<b>@media.FileName</b>
|
|
</p>
|
|
<p class="text-xs break-all">
|
|
<i class="ion-md-code"></i> @media.ContentType
|
|
</p>
|
|
</div>
|
|
<button class="button is-small is-rounded neoBtnSmall" @onclick="async () => await OnMessageMediaDownload.InvokeAsync(media)" type="button">
|
|
<span class="icon">
|
|
<i class="ion-md-download text-base"></i>
|
|
</span>
|
|
</button>
|
|
</div>
|
|
}
|
|
}
|
|
</div>
|
|
}
|
|
|
|
</div>
|
|
|
|
<div class="flex space-x-3 mt-3 justify-between">
|
|
<div class="flex space-x-3">
|
|
@if (OnMessageReply.HasDelegate)
|
|
{
|
|
<button class="button is-small is-rounded @(isAnswering ? "neoBtnSmallInsetPlain" : "neoBtnSmall")" @onclick="Reply"
|
|
title="@CascadingState.Localizer[isAnswering ? "Close" : "Reply"]">
|
|
<span class="icon">
|
|
<i aria-hidden="true" class="@SUtility.IfTrueThen(isAnswering, "ion-md-close-circle", "ion-md-text") text-lg has-text-success"></i>
|
|
</span>
|
|
</button>
|
|
}
|
|
@if (OnMessageBoost.HasDelegate)
|
|
{
|
|
<button class="button is-small is-rounded @SUtility.IfTrueThen(Message.IsBoostedByCurrentUser,"neoBtnSmallInsetPlain","neoBtnSmall") @SUtility.IfTrueThen(Message.BoostsCounter != 0, "has-icons-left")" @onclick="() => OnMessageBoost.InvokeAsync(Message)"
|
|
title="@CascadingState.Localizer["Boost"]">
|
|
<span class="icon is-left">
|
|
<i aria-hidden="true" class="ion-md-repeat @SUtility.IfTrueThen(Message.BoostsCounter != 0,null,"text-lg") has-text-info"></i>
|
|
</span>
|
|
@if (Message.BoostsCounter != 0)
|
|
{
|
|
<span>@Message.BoostsCounter</span>
|
|
}
|
|
</button>
|
|
}
|
|
@if (OnMessageFavourite.HasDelegate)
|
|
{
|
|
<button class="button is-small is-rounded @SUtility.IfTrueThen(Message.IsFavourite, "neoBtnSmallInsetPlain", "neoBtnSmall")" @onclick="() => OnMessageFavourite.InvokeAsync(Message)"
|
|
title="@CascadingState.Localizer["Favourite"]">
|
|
<span class="icon">
|
|
<i aria-hidden="true" class="@SUtility.IfTrueThen(Message.IsFavourite, "ion-md-heart-dislike", "ion-md-heart") text-lg text-pink-300"></i>
|
|
</span>
|
|
</button>
|
|
}
|
|
</div>
|
|
|
|
<div class="flex space-x-3">
|
|
<DropdownButton IsOpen="Message.IsOptionsOpen">
|
|
<DropdownTrigger>
|
|
<button class="button is-small is-rounded neoBtnSmall" @onclick="() => Message.IsOptionsOpen = !Message.IsOptionsOpen"
|
|
title="@CascadingState.Localizer["Other"]">
|
|
<span class="icon">
|
|
<i aria-hidden="true" class="ion-md-more text-lg"></i>
|
|
</span>
|
|
</button>
|
|
</DropdownTrigger>
|
|
<DropdownContent>
|
|
@if (OnUserDirectMessage.HasDelegate)
|
|
{
|
|
<div class="dropdown-item">
|
|
<button class="button is-small is-rounded has-icons-left neoBtnSmall" @onclick="() => OnUserDirectMessage.InvokeAsync(Message)">
|
|
<span class="icon is-left">
|
|
<i aria-hidden="true" class="ion-md-paper-plane text-base has-text-success"></i>
|
|
</span>
|
|
<span>@CascadingState.Localizer["Direct message"]</span>
|
|
</button>
|
|
</div>
|
|
}
|
|
@if (OnUserSilence.HasDelegate)
|
|
{
|
|
<div class="dropdown-item">
|
|
<button class="button is-small is-rounded has-icons-left neoBtnSmall" @onclick="() => OnUserSilence.InvokeAsync(Message.User)">
|
|
<span class="icon is-left">
|
|
<i aria-hidden="true" class="ion-md-eye-off text-base drop-shadow has-text-warning"></i>
|
|
</span>
|
|
<span>@CascadingState.Localizer["Mute"]</span>
|
|
</button>
|
|
</div>
|
|
}
|
|
@if (OnUserBlock.HasDelegate)
|
|
{
|
|
<div class="dropdown-item">
|
|
<button class="button is-small is-rounded has-icons-left neoBtnSmall" @onclick="() => OnUserBlock.InvokeAsync(Message.User)">
|
|
<span class="icon is-left">
|
|
<i aria-hidden="true" class="ion-md-remove-circle text-base has-text-danger"></i>
|
|
</span>
|
|
<span>@CascadingState.Localizer["Block"]</span>
|
|
</button>
|
|
</div>
|
|
}
|
|
@if (@*Message.User.UserName == CurrentUserName &&*@ OnMessageDelete.HasDelegate)
|
|
{
|
|
<div class="dropdown-item">
|
|
<button class="button is-small has-icons-left is-rounded neoBtnSmall" @onclick="DeleteMessage"
|
|
title="@CascadingState.Localizer["Delete"]">
|
|
<span class="icon is-left">
|
|
<i aria-hidden="true" class="ion-md-trash text-lg has-text-danger"></i>
|
|
</span>
|
|
<span>@CascadingState.Localizer["Delete"]</span>
|
|
</button>
|
|
</div>
|
|
}
|
|
</DropdownContent>
|
|
</DropdownButton>
|
|
@if (IncludeExpand)
|
|
{
|
|
<NavLink class="button is-small is-rounded neoBtnSmall" href="@($"expand/{Message.MessageId}")"
|
|
title="@CascadingState.Localizer["Expand"]">
|
|
<span class="icon">
|
|
<i aria-hidden="true" class="ion-md-expand text-lg"></i>
|
|
</span>
|
|
</NavLink>
|
|
}
|
|
</div>
|
|
</div>
|
|
|
|
@if (isAnswering)
|
|
{
|
|
<MessageUpsertForm AnsweringMessage="Message" OnMessageSubmit="SubmitReply"></MessageUpsertForm>
|
|
}
|
|
</div>
|
|
|
|
</div>
|
|
|
|
@code {
|
|
[CascadingParameter] Task<AuthenticationState> AuthState { get; set; }
|
|
[CascadingParameter] CascadingState CascadingState { get; set; }
|
|
[Parameter] public Message Message { get; set; } = new();
|
|
[Parameter] public EventCallback<MessageForm> OnMessageReply { get; set; }
|
|
[Parameter] public EventCallback<Message> OnMessageBoost { get; set; }
|
|
[Parameter] public EventCallback<Message> OnMessageFavourite { get; set; }
|
|
[Parameter] public EventCallback<Message> OnMessageDelete { get; set; }
|
|
[Parameter] public EventCallback<Message> OnUserDirectMessage { get; set; }
|
|
[Parameter] public EventCallback<Media> OnMessageMediaDownload { get; set; }
|
|
[Parameter] public EventCallback<MessageUser> OnUserBlock { get; set; }
|
|
[Parameter] public EventCallback<MessageUser> OnUserSilence { get; set; }
|
|
[Parameter] public string CssContainer { get; set; }
|
|
[Parameter] public bool IncludeExpand { get; set; } = true;
|
|
|
|
bool isAnswering { get; set; } = false;
|
|
|
|
string CurrentUserName
|
|
{
|
|
get
|
|
{
|
|
return AuthState.Result.User.Identity?.Name;
|
|
}
|
|
}
|
|
|
|
async Task DeleteMessage()
|
|
{
|
|
isAnswering = false;
|
|
await OnMessageDelete.InvokeAsync(Message);
|
|
}
|
|
|
|
async Task SubmitReply(MessageForm messageForm)
|
|
{
|
|
isAnswering = false;
|
|
await OnMessageReply.InvokeAsync(messageForm);
|
|
}
|
|
|
|
async Task Reply()
|
|
{
|
|
await Task.Run(() =>
|
|
{
|
|
});
|
|
isAnswering = !isAnswering;
|
|
}
|
|
} |