forked from AkkomaGang/akkoma-fe
723 lines
19 KiB
Vue
723 lines
19 KiB
Vue
<template>
|
|
<div class="status-el" v-if="!hideStatus" :class="[{ 'status-el_focused': isFocused }, { 'status-conversation': inlineExpanded }]">
|
|
<template v-if="muted && !isPreview">
|
|
<div class="media status container muted">
|
|
<small>
|
|
<router-link :to="userProfileLink">
|
|
{{status.user.screen_name}}
|
|
</router-link>
|
|
</small>
|
|
<small class="muteWords">{{muteWordHits.join(', ')}}</small>
|
|
<a href="#" class="unmute" @click.prevent="toggleMute"><i class="button-icon icon-eye-off"></i></a>
|
|
</div>
|
|
</template>
|
|
<template v-else>
|
|
<div v-if="retweet && !noHeading && !inConversation" :class="[repeaterClass, { highlighted: repeaterStyle }]" :style="[repeaterStyle]" class="media container retweet-info">
|
|
<UserAvatar class="media-left" v-if="retweet" :betterShadow="betterShadow" :src="statusoid.user.profile_image_url_original"/>
|
|
<div class="media-body faint">
|
|
<span class="user-name">
|
|
<router-link v-if="retweeterHtml" :to="retweeterProfileLink" v-html="retweeterHtml"/>
|
|
<router-link v-else :to="retweeterProfileLink">{{retweeter}}</router-link>
|
|
</span>
|
|
<i class='fa icon-retweet retweeted' :title="$t('tool_tip.repeat')"></i>
|
|
{{$t('timeline.repeated')}}
|
|
</div>
|
|
</div>
|
|
|
|
<div :class="[userClass, { highlighted: userStyle, 'is-retweet': retweet && !inConversation }]" :style="[ userStyle ]" class="media status">
|
|
<div v-if="!noHeading" class="media-left">
|
|
<router-link :to="userProfileLink" @click.stop.prevent.capture.native="toggleUserExpanded">
|
|
<UserAvatar :compact="compact" :betterShadow="betterShadow" :src="status.user.profile_image_url_original"/>
|
|
</router-link>
|
|
</div>
|
|
<div class="status-body">
|
|
<UserCard :user="status.user" :rounded="true" :bordered="true" class="status-usercard" v-if="userExpanded"/>
|
|
<div v-if="!noHeading" class="media-heading">
|
|
<div class="heading-name-row">
|
|
<div class="name-and-account-name">
|
|
<h4 class="user-name" v-if="status.user.name_html" v-html="status.user.name_html"></h4>
|
|
<h4 class="user-name" v-else>{{status.user.name}}</h4>
|
|
<router-link class="account-name" :to="userProfileLink">
|
|
{{status.user.screen_name}}
|
|
</router-link>
|
|
</div>
|
|
|
|
<span class="heading-right">
|
|
<router-link class="timeago faint-link" :to="{ name: 'conversation', params: { id: status.id } }">
|
|
<timeago :since="status.created_at" :auto-update="60"></timeago>
|
|
</router-link>
|
|
<div class="button-icon visibility-icon" v-if="status.visibility">
|
|
<i :class="visibilityIcon(status.visibility)" :title="status.visibility | capitalize"></i>
|
|
</div>
|
|
<a :href="status.external_url" target="_blank" v-if="!status.is_local && !isPreview" class="source_url" title="Source">
|
|
<i class="button-icon icon-link-ext-alt"></i>
|
|
</a>
|
|
<template v-if="expandable && !isPreview">
|
|
<a href="#" @click.prevent="toggleExpanded" title="Expand">
|
|
<i class="button-icon icon-plus-squared"></i>
|
|
</a>
|
|
</template>
|
|
<a href="#" @click.prevent="toggleMute" v-if="unmuted"><i class="button-icon icon-eye-off"></i></a>
|
|
</span>
|
|
</div>
|
|
|
|
<div class="heading-reply-row">
|
|
<div v-if="isReply" class="reply-to-and-accountname">
|
|
<a class="reply-to"
|
|
href="#" @click.prevent="gotoOriginal(status.in_reply_to_status_id)"
|
|
:aria-label="$t('tool_tip.reply')"
|
|
@mouseenter.prevent.stop="replyEnter(status.in_reply_to_status_id, $event)"
|
|
@mouseleave.prevent.stop="replyLeave()"
|
|
>
|
|
<i class="button-icon icon-reply" v-if="!isPreview"></i>
|
|
<span class="faint-link reply-to-text">{{$t('status.reply_to')}}</span>
|
|
</a>
|
|
<router-link :to="replyProfileLink">
|
|
{{replyToName}}
|
|
</router-link>
|
|
<span class="faint replies-separator" v-if="replies && replies.length">
|
|
-
|
|
</span>
|
|
</div>
|
|
<div class="replies" v-if="inConversation && !isPreview">
|
|
<span class="faint" v-if="replies && replies.length">{{$t('status.replies_list')}}</span>
|
|
<span class="reply-link faint" v-if="replies" v-for="reply in replies">
|
|
<a href="#" @click.prevent="gotoOriginal(reply.id)" @mouseenter="replyEnter(reply.id, $event)" @mouseout="replyLeave()">{{reply.name}}</a>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
<div v-if="showPreview" class="status-preview-container">
|
|
<status class="status-preview" v-if="preview" :isPreview="true" :statusoid="preview" :compact=true></status>
|
|
<div class="status-preview status-preview-loading" v-else>
|
|
<i class="icon-spin4 animate-spin"></i>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="status-content-wrapper" :class="{ 'tall-status': !showingLongSubject }" v-if="longSubject">
|
|
<a class="tall-status-hider" :class="{ 'tall-status-hider_focused': isFocused }" v-if="!showingLongSubject" href="#" @click.prevent="showingLongSubject=true">{{$t("general.show_more")}}</a>
|
|
<div @click.prevent="linkClicked" class="status-content media-body" v-html="contentHtml"></div>
|
|
<a v-if="showingLongSubject" href="#" class="status-unhider" @click.prevent="showingLongSubject=false">{{$t("general.show_less")}}</a>
|
|
</div>
|
|
<div :class="{'tall-status': hideTallStatus}" class="status-content-wrapper" v-else>
|
|
<a class="tall-status-hider" :class="{ 'tall-status-hider_focused': isFocused }" v-if="hideTallStatus" href="#" @click.prevent="toggleShowMore">{{$t("general.show_more")}}</a>
|
|
<div @click.prevent="linkClicked" class="status-content media-body" v-html="contentHtml" v-if="!hideSubjectStatus"></div>
|
|
<div @click.prevent="linkClicked" class="status-content media-body" v-html="status.summary_html" v-else></div>
|
|
<a v-if="hideSubjectStatus" href="#" class="cw-status-hider" @click.prevent="toggleShowMore">{{$t("general.show_more")}}</a>
|
|
<a v-if="showingMore" href="#" class="status-unhider" @click.prevent="toggleShowMore">{{$t("general.show_less")}}</a>
|
|
</div>
|
|
|
|
<div v-if="status.attachments && (!hideSubjectStatus || showingLongSubject)" class="attachments media-body">
|
|
<attachment
|
|
class="non-gallery"
|
|
v-for="attachment in nonGalleryAttachments"
|
|
:size="attachmentSize"
|
|
:nsfw="nsfwClickthrough"
|
|
:attachment="attachment"
|
|
:allowPlay="true"
|
|
:setMedia="setMedia()"
|
|
:key="attachment.id"
|
|
/>
|
|
<gallery
|
|
v-if="galleryAttachments.length > 0"
|
|
:nsfw="nsfwClickthrough"
|
|
:attachments="galleryAttachments"
|
|
:setMedia="setMedia()"
|
|
/>
|
|
</div>
|
|
|
|
<div v-if="status.card && !hideSubjectStatus && !noHeading" class="link-preview media-body">
|
|
<link-preview :card="status.card" :size="attachmentSize" :nsfw="nsfwClickthrough" />
|
|
</div>
|
|
|
|
<div class="favs-repeated-users" v-if="combinedFavsAndRepeatsAvatars.length > 0" :class="{ 'status-fadein': combinedFavsAndRepeatsAvatars.length > 0 }">
|
|
<ul class="stats">
|
|
<li class="stat-count" v-if="statusoid.rebloggedBy && statusoid.rebloggedBy.length > 0">
|
|
<a class="stat-title">{{ $t('settings.notification_visibility_repeats') }}</a>
|
|
<div class="stat-number">{{ statusoid.rebloggedBy.length }}</div>
|
|
</li>
|
|
<li class="stat-count" v-if="statusoid.favoritedBy && statusoid.favoritedBy.length > 0">
|
|
<a class="stat-title">{{ $t('user_card.favorites') }}</a>
|
|
<div class="stat-number">{{ statusoid.favoritedBy.length }}</div>
|
|
</li>
|
|
<li class="avatar-row">
|
|
<AvatarList :avatars='combinedFavsAndRepeatsAvatars'></AvatarList>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div v-if="!noHeading && !isPreview" class='status-actions media-body'>
|
|
<div v-if="loggedIn">
|
|
<i class="button-icon icon-reply" v-on:click.prevent="toggleReplying" :title="$t('tool_tip.reply')" :class="{'icon-reply-active': replying}"></i>
|
|
<span v-if="status.replies_count > 0">{{status.replies_count}}</span>
|
|
</div>
|
|
<retweet-button :visibility='status.visibility' :loggedIn='loggedIn' :status='status'></retweet-button>
|
|
<favorite-button :loggedIn='loggedIn' :status='status'></favorite-button>
|
|
<delete-button :status='status'></delete-button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="container" v-if="replying">
|
|
<div class="reply-left"/>
|
|
<post-status-form class="reply-body" :reply-to="status.id" :attentions="status.attentions" :repliedUser="status.user" :copy-message-scope="status.visibility" :subject="replySubject" v-on:posted="toggleReplying"/>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
|
|
<script src="./status.js" ></script>
|
|
<style lang="scss">
|
|
@import '../../_variables.scss';
|
|
|
|
$status-margin: 0.75em;
|
|
|
|
.status-body {
|
|
flex: 1;
|
|
min-width: 0;
|
|
}
|
|
|
|
.status-preview.status-el {
|
|
border-style: solid;
|
|
border-width: 1px;
|
|
border-color: $fallback--border;
|
|
border-color: var(--border, $fallback--border);
|
|
}
|
|
|
|
.status-preview-container {
|
|
position: relative;
|
|
max-width: 100%;
|
|
}
|
|
|
|
.status-preview {
|
|
position: absolute;
|
|
max-width: 95%;
|
|
display: flex;
|
|
background-color: $fallback--bg;
|
|
background-color: var(--bg, $fallback--bg);
|
|
border-color: $fallback--border;
|
|
border-color: var(--border, $fallback--border);
|
|
border-style: solid;
|
|
border-width: 1px;
|
|
border-radius: $fallback--tooltipRadius;
|
|
border-radius: var(--tooltipRadius, $fallback--tooltipRadius);
|
|
box-shadow: 2px 2px 3px rgba(0, 0, 0, 0.5);
|
|
box-shadow: var(--popupShadow);
|
|
margin-top: 0.25em;
|
|
margin-left: 0.5em;
|
|
z-index: 50;
|
|
|
|
.status {
|
|
flex: 1;
|
|
border: 0;
|
|
min-width: 15em;
|
|
}
|
|
}
|
|
|
|
.status-preview-loading {
|
|
display: block;
|
|
min-width: 15em;
|
|
padding: 1em;
|
|
text-align: center;
|
|
border-width: 1px;
|
|
border-style: solid;
|
|
|
|
i {
|
|
font-size: 2em;
|
|
}
|
|
}
|
|
|
|
.media-left {
|
|
margin-right: $status-margin;
|
|
}
|
|
|
|
.status-el {
|
|
hyphens: auto;
|
|
overflow-wrap: break-word;
|
|
word-wrap: break-word;
|
|
word-break: break-word;
|
|
border-left-width: 0px;
|
|
min-width: 0;
|
|
border-color: $fallback--border;
|
|
border-color: var(--border, $fallback--border);
|
|
|
|
border-left: 4px $fallback--cRed;
|
|
border-left: 4px var(--cRed, $fallback--cRed);
|
|
|
|
&_focused {
|
|
background-color: $fallback--lightBg;
|
|
background-color: var(--lightBg, $fallback--lightBg);
|
|
}
|
|
|
|
.timeline & {
|
|
border-bottom-width: 1px;
|
|
border-bottom-style: solid;
|
|
}
|
|
|
|
.media-body {
|
|
flex: 1;
|
|
padding: 0;
|
|
}
|
|
|
|
.status-usercard {
|
|
margin-bottom: $status-margin;
|
|
}
|
|
|
|
.user-name {
|
|
white-space: nowrap;
|
|
font-size: 14px;
|
|
overflow: hidden;
|
|
flex-shrink: 0;
|
|
max-width: 85%;
|
|
font-weight: bold;
|
|
|
|
img {
|
|
width: 14px;
|
|
height: 14px;
|
|
vertical-align: middle;
|
|
object-fit: contain
|
|
}
|
|
}
|
|
|
|
.media-heading {
|
|
padding: 0;
|
|
vertical-align: bottom;
|
|
flex-basis: 100%;
|
|
margin-bottom: 0.5em;
|
|
|
|
a {
|
|
display: inline-block;
|
|
word-break: break-all;
|
|
}
|
|
|
|
small {
|
|
font-weight: lighter;
|
|
}
|
|
|
|
.heading-name-row {
|
|
padding: 0;
|
|
display: flex;
|
|
justify-content: space-between;
|
|
line-height: 18px;
|
|
|
|
.name-and-account-name {
|
|
display: flex;
|
|
min-width: 0;
|
|
}
|
|
|
|
.user-name {
|
|
flex-shrink: 1;
|
|
margin-right: 0.4em;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
|
|
.account-name {
|
|
min-width: 1.6em;
|
|
margin-right: 0.4em;
|
|
white-space: nowrap;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
flex: 1 1 0;
|
|
}
|
|
}
|
|
|
|
.heading-right {
|
|
display: flex;
|
|
flex-shrink: 0;
|
|
}
|
|
|
|
.timeago {
|
|
margin-right: 0.2em;
|
|
}
|
|
|
|
.heading-reply-row {
|
|
align-content: baseline;
|
|
font-size: 12px;
|
|
line-height: 18px;
|
|
max-width: 100%;
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
align-items: stretch;
|
|
|
|
a {
|
|
max-width: 100%;
|
|
text-overflow: ellipsis;
|
|
overflow: hidden;
|
|
white-space: nowrap;
|
|
}
|
|
}
|
|
|
|
.reply-to-and-accountname {
|
|
display: flex;
|
|
height: 18px;
|
|
margin-right: 0.5em;
|
|
overflow: hidden;
|
|
max-width: 100%;
|
|
.icon-reply {
|
|
transform: scaleX(-1);
|
|
}
|
|
}
|
|
|
|
.reply-info {
|
|
display: flex;
|
|
}
|
|
|
|
.reply-to {
|
|
display: flex;
|
|
}
|
|
|
|
.reply-to-text {
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
margin: 0 0.4em 0 0.2em;
|
|
}
|
|
|
|
.replies-separator {
|
|
margin-left: 0.4em;
|
|
}
|
|
|
|
.replies {
|
|
line-height: 18px;
|
|
font-size: 12px;
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
& > * {
|
|
margin-right: 0.4em;
|
|
}
|
|
}
|
|
|
|
.reply-link {
|
|
height: 17px;
|
|
}
|
|
}
|
|
|
|
.tall-status {
|
|
position: relative;
|
|
height: 220px;
|
|
overflow-x: hidden;
|
|
overflow-y: hidden;
|
|
}
|
|
|
|
.tall-status-hider {
|
|
display: inline-block;
|
|
word-break: break-all;
|
|
position: absolute;
|
|
height: 70px;
|
|
margin-top: 150px;
|
|
width: 100%;
|
|
text-align: center;
|
|
line-height: 110px;
|
|
background: linear-gradient(to bottom, rgba(0, 0, 0, 0), $fallback--bg 80%);
|
|
background: linear-gradient(to bottom, rgba(0, 0, 0, 0), var(--bg, $fallback--bg) 80%);
|
|
&_focused {
|
|
background: linear-gradient(to bottom, rgba(0, 0, 0, 0), $fallback--lightBg 80%);
|
|
background: linear-gradient(to bottom, rgba(0, 0, 0, 0), var(--lightBg, $fallback--lightBg) 80%);
|
|
}
|
|
}
|
|
|
|
.status-unhider, .cw-status-hider {
|
|
width: 100%;
|
|
text-align: center;
|
|
display: inline-block;
|
|
word-break: break-all;
|
|
}
|
|
|
|
.status-content {
|
|
font-family: var(--postFont, sans-serif);
|
|
line-height: 1.4em;
|
|
|
|
img, video {
|
|
max-width: 100%;
|
|
max-height: 400px;
|
|
vertical-align: middle;
|
|
object-fit: contain;
|
|
|
|
&.emoji {
|
|
width: 32px;
|
|
height: 32px;
|
|
}
|
|
}
|
|
|
|
blockquote {
|
|
margin: 0.2em 0 0.2em 2em;
|
|
font-style: italic;
|
|
}
|
|
|
|
pre {
|
|
overflow: auto;
|
|
}
|
|
|
|
code, samp, kbd, var, pre {
|
|
font-family: var(--postCodeFont, monospace);
|
|
}
|
|
|
|
p {
|
|
margin: 0 0 1em 0;
|
|
}
|
|
|
|
p:last-child {
|
|
margin: 0 0 0 0;
|
|
}
|
|
|
|
h1 {
|
|
font-size: 1.1em;
|
|
line-height: 1.2em;
|
|
margin: 1.4em 0;
|
|
}
|
|
|
|
h2 {
|
|
font-size: 1.1em;
|
|
margin: 1.0em 0;
|
|
}
|
|
|
|
h3 {
|
|
font-size: 1em;
|
|
margin: 1.2em 0;
|
|
}
|
|
|
|
h4 {
|
|
margin: 1.1em 0;
|
|
}
|
|
}
|
|
|
|
.retweet-info {
|
|
padding: 0.4em $status-margin;
|
|
margin: 0;
|
|
|
|
.avatar.still-image {
|
|
border-radius: $fallback--avatarAltRadius;
|
|
border-radius: var(--avatarAltRadius, $fallback--avatarAltRadius);
|
|
margin-left: 28px;
|
|
width: 20px;
|
|
height: 20px;
|
|
}
|
|
|
|
.media-body {
|
|
font-size: 1em;
|
|
line-height: 22px;
|
|
|
|
display: flex;
|
|
align-content: center;
|
|
flex-wrap: wrap;
|
|
|
|
.user-name {
|
|
font-weight: bold;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
|
|
img {
|
|
width: 14px;
|
|
height: 14px;
|
|
vertical-align: middle;
|
|
object-fit: contain
|
|
}
|
|
}
|
|
|
|
i {
|
|
padding: 0 0.2em;
|
|
}
|
|
|
|
a {
|
|
max-width: 100%;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
.status-fadein {
|
|
animation-duration: 0.4s;
|
|
animation-name: fadein;
|
|
}
|
|
|
|
@keyframes fadein {
|
|
from {
|
|
opacity: 0;
|
|
}
|
|
to {
|
|
opacity: 1;
|
|
}
|
|
}
|
|
|
|
.greentext {
|
|
color: green;
|
|
}
|
|
|
|
.status-conversation {
|
|
border-left-style: solid;
|
|
}
|
|
|
|
.status-actions {
|
|
width: 100%;
|
|
display: flex;
|
|
margin-top: $status-margin;
|
|
|
|
div, favorite-button {
|
|
max-width: 4em;
|
|
flex: 1;
|
|
}
|
|
}
|
|
|
|
.icon-reply:hover {
|
|
color: $fallback--cBlue;
|
|
color: var(--cBlue, $fallback--cBlue);
|
|
cursor: pointer;
|
|
}
|
|
|
|
.icon-reply.icon-reply-active {
|
|
color: $fallback--cBlue;
|
|
color: var(--cBlue, $fallback--cBlue);
|
|
}
|
|
|
|
.status:hover .animated.avatar {
|
|
canvas {
|
|
display: none;
|
|
}
|
|
img {
|
|
visibility: visible;
|
|
}
|
|
}
|
|
|
|
.status {
|
|
display: flex;
|
|
padding: $status-margin;
|
|
&.is-retweet {
|
|
padding-top: 0;
|
|
}
|
|
}
|
|
|
|
.status-conversation:last-child {
|
|
border-bottom: none;
|
|
}
|
|
|
|
.muted {
|
|
padding: 0.25em 0.5em;
|
|
button {
|
|
margin-left: auto;
|
|
}
|
|
|
|
.muteWords {
|
|
margin-left: 10px;
|
|
}
|
|
}
|
|
|
|
a.unmute {
|
|
display: block;
|
|
margin-left: auto;
|
|
}
|
|
|
|
.reply-left {
|
|
flex: 0;
|
|
min-width: 48px;
|
|
}
|
|
|
|
.reply-body {
|
|
flex: 1;
|
|
}
|
|
|
|
.timeline > {
|
|
.status-el:last-child {
|
|
border-radius: 0 0 $fallback--panelRadius $fallback--panelRadius;
|
|
border-radius: 0 0 var(--panelRadius, $fallback--panelRadius) var(--panelRadius, $fallback--panelRadius);
|
|
border-bottom: none;
|
|
}
|
|
}
|
|
|
|
.favs-repeated-users {
|
|
margin-top: 0.75em;
|
|
|
|
.stats {
|
|
width: 100%;
|
|
margin: 0;
|
|
list-style: none;
|
|
overflow: hidden;
|
|
padding: 0;
|
|
display: flex;
|
|
line-height: 1em;
|
|
|
|
.stat-count {
|
|
margin-right: 10px;
|
|
|
|
.stat-title {
|
|
color: var(--faint, $fallback--faint);
|
|
font-size: 12px;
|
|
text-transform: uppercase;
|
|
position: relative;
|
|
margin-bottom: 3px;
|
|
cursor: pointer;
|
|
line-height: 1em;
|
|
|
|
&:hover,
|
|
&:focus {
|
|
text-decoration: underline;
|
|
}
|
|
}
|
|
|
|
.stat-number {
|
|
font-weight: bolder;
|
|
font-size: 16px;
|
|
line-height: 1em;
|
|
}
|
|
|
|
/* a {
|
|
cursor: pointer;
|
|
color: var(--faint, $fallback--faint);
|
|
|
|
strong {
|
|
color: var(--text, $fallback--text);
|
|
}
|
|
} */
|
|
}
|
|
|
|
.avatar-row {
|
|
flex: 1;
|
|
overflow: hidden;
|
|
position: relative;
|
|
|
|
&::before {
|
|
content: '';
|
|
position: absolute;
|
|
height: 100%;
|
|
width: 1px;
|
|
left: 0;
|
|
background-color: var(--faint, $fallback--faint);
|
|
}
|
|
}
|
|
|
|
li {
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: center;
|
|
}
|
|
}
|
|
}
|
|
|
|
@media all and (max-width: 800px) {
|
|
.status-el {
|
|
.retweet-info {
|
|
.avatar.still-image {
|
|
margin-left: 20px;
|
|
}
|
|
}
|
|
}
|
|
.status {
|
|
max-width: 100%;
|
|
}
|
|
|
|
.status .avatar.still-image {
|
|
width: 40px;
|
|
height: 40px;
|
|
|
|
&.avatar-compact {
|
|
width: 32px;
|
|
height: 32px;
|
|
}
|
|
}
|
|
}
|
|
|
|
</style>
|