forked from AkkomaGang/akkoma-fe
587 lines
15 KiB
Vue
587 lines
15 KiB
Vue
<template>
|
|
<div class="status-el" v-if="!hideReply && !deleted" :class="[{ 'status-el_focused': isFocused }, { 'status-conversation': inlineExpanded }]">
|
|
<template v-if="muted && !noReplyLinks">
|
|
<div class="media status container muted">
|
|
<small>
|
|
<router-link :to="userProfileLink(status.user.id, status.user.screen_name)">
|
|
{{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" :class="[repeaterClass, { highlighted: repeaterStyle }]" :style="[repeaterStyle]" class="media container retweet-info">
|
|
<StillImage v-if="retweet" class='avatar' :class='{ "better-shadow": betterShadow }' :src="statusoid.user.profile_image_url_original"/>
|
|
<div class="media-body faint">
|
|
<a v-if="retweeterHtml" :href="statusoid.user.statusnet_profile_url" class="user-name" :title="'@'+statusoid.user.screen_name" v-html="retweeterHtml"></a>
|
|
<a v-else :href="statusoid.user.statusnet_profile_url" class="user-name" :title="'@'+statusoid.user.screen_name">{{retweeter}}</a>
|
|
<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 }]" :style="[ userStyle ]" class="media status">
|
|
<div v-if="!noHeading" class="media-left">
|
|
<a :href="status.user.statusnet_profile_url" @click.stop.prevent.capture="toggleUserExpanded">
|
|
<StillImage class='avatar' :class="{'avatar-compact': compact, 'better-shadow': betterShadow}" :src="status.user.profile_image_url_original"/>
|
|
</a>
|
|
</div>
|
|
<div class="status-body">
|
|
<div class="usercard media-body" v-if="userExpanded">
|
|
<user-card-content :user="status.user" :switcher="false"></user-card-content>
|
|
</div>
|
|
<div v-if="!noHeading" class="media-body container media-heading">
|
|
<div class="media-heading-left">
|
|
<div class="name-and-links">
|
|
<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>
|
|
<span class="links">
|
|
<router-link :to="userProfileLink(status.user.id, status.user.screen_name)">
|
|
{{status.user.screen_name}}
|
|
</router-link>
|
|
<span v-if="status.in_reply_to_screen_name" class="faint reply-info">
|
|
<i class="icon-right-open"></i>
|
|
<router-link :to="userProfileLink(status.in_reply_to_user_id, status.in_reply_to_screen_name)">
|
|
{{status.in_reply_to_screen_name}}
|
|
</router-link>
|
|
</span>
|
|
<a v-if="isReply && !noReplyLinks" href="#" @click.prevent="gotoOriginal(status.in_reply_to_status_id)" :title="$t('tool_tip.reply')">
|
|
<i class="button-icon icon-reply" @mouseenter="replyEnter(status.in_reply_to_status_id, $event)" @mouseout="replyLeave()"></i>
|
|
</a>
|
|
</span>
|
|
</div>
|
|
<h4 class="replies" v-if="inConversation && !noReplyLinks">
|
|
<small v-if="replies.length">Replies:</small>
|
|
<small class="reply-link" v-for="reply in replies">
|
|
<a href="#" @click.prevent="gotoOriginal(reply.id)" @mouseenter="replyEnter(reply.id, $event)" @mouseout="replyLeave()">{{reply.name}} </a>
|
|
</small>
|
|
</h4>
|
|
</div>
|
|
<div class="media-heading-right">
|
|
<router-link class="timeago" :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" class="source_url" title="Source">
|
|
<i class="button-icon icon-link-ext-alt"></i>
|
|
</a>
|
|
<template v-if="expandable">
|
|
<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>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="showPreview" class="status-preview-container">
|
|
<status class="status-preview" v-if="preview" :noReplyLinks="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="{'tall-status': hideTallStatus}" class="status-content-wrapper">
|
|
<a class="tall-status-hider" :class="{ 'tall-status-hider_focused': isFocused }" v-if="hideTallStatus" href="#" @click.prevent="toggleShowMore">Show more</a>
|
|
<div @click.prevent="linkClicked" class="status-content media-body" v-html="status.statusnet_html" 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">Show more</a>
|
|
<a v-if="showingMore" href="#" class="status-unhider" @click.prevent="toggleShowMore">Show less</a>
|
|
</div>
|
|
|
|
<div v-if='status.attachments && !hideSubjectStatus' class='attachments media-body'>
|
|
<attachment :size="attachmentSize" :status-id="status.id" :nsfw="nsfwClickthrough" :attachment="attachment" v-for="attachment in status.attachments" :key="attachment.id">
|
|
</attachment>
|
|
</div>
|
|
|
|
<div v-if="!noHeading && !noReplyLinks" class='status-actions media-body'>
|
|
<div v-if="loggedIn">
|
|
<a href="#" v-on:click.prevent="toggleReplying" :title="$t('tool_tip.reply')">
|
|
<i class="button-icon icon-reply" :class="{'icon-reply-active': replying}"></i>
|
|
</a>
|
|
</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-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;
|
|
}
|
|
}
|
|
|
|
.status-el {
|
|
hyphens: auto;
|
|
overflow-wrap: break-word;
|
|
word-wrap: break-word;
|
|
word-break: break-word;
|
|
border-left-width: 0px;
|
|
line-height: 18px;
|
|
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;
|
|
margin: 0 0 0.25em 0.8em;
|
|
}
|
|
|
|
.usercard {
|
|
margin-bottom: .7em
|
|
}
|
|
|
|
.media-heading {
|
|
flex-wrap: nowrap;
|
|
line-height: 18px;
|
|
}
|
|
|
|
.media-heading-left {
|
|
padding: 0;
|
|
vertical-align: bottom;
|
|
flex-basis: 100%;
|
|
|
|
small {
|
|
font-weight: lighter;
|
|
}
|
|
h4 {
|
|
white-space: nowrap;
|
|
font-size: 14px;
|
|
margin-right: 0.25em;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
.name-and-links {
|
|
padding: 0;
|
|
flex: 1 0;
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
align-items: baseline;
|
|
|
|
.user-name {
|
|
margin-right: .45em;
|
|
|
|
img {
|
|
width: 14px;
|
|
height: 14px;
|
|
vertical-align: middle;
|
|
object-fit: contain
|
|
}
|
|
}
|
|
}
|
|
|
|
.links {
|
|
display: flex;
|
|
font-size: 12px;
|
|
color: $fallback--link;
|
|
color: var(--link, $fallback--link);
|
|
max-width: 100%;
|
|
a {
|
|
max-width: 100%;
|
|
text-overflow: ellipsis;
|
|
overflow: hidden;
|
|
white-space: nowrap;
|
|
}
|
|
}
|
|
.reply-info {
|
|
display: flex;
|
|
}
|
|
.replies {
|
|
line-height: 16px;
|
|
}
|
|
.reply-link {
|
|
margin-right: 0.2em;
|
|
}
|
|
}
|
|
|
|
.media-heading-right {
|
|
display: inline-flex;
|
|
flex-shrink: 0;
|
|
flex-wrap: nowrap;
|
|
margin-left: .25em;
|
|
align-self: baseline;
|
|
|
|
.timeago {
|
|
margin-right: 0.2em;
|
|
font-size: 12px;
|
|
align-self: last baseline;
|
|
}
|
|
|
|
> * {
|
|
margin-left: 0.2em;
|
|
}
|
|
a:hover i {
|
|
color: $fallback--text;
|
|
color: var(--text, $fallback--text);
|
|
}
|
|
}
|
|
|
|
a {
|
|
display: inline-block;
|
|
word-break: break-all;
|
|
}
|
|
|
|
.tall-status {
|
|
position: relative;
|
|
height: 220px;
|
|
overflow-x: hidden;
|
|
overflow-y: hidden;
|
|
}
|
|
|
|
.tall-status-hider {
|
|
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;
|
|
}
|
|
|
|
.status-content {
|
|
margin-right: 0.5em;
|
|
font-family: var(--postFont, sans-serif);
|
|
|
|
img, video {
|
|
max-width: 100%;
|
|
max-height: 400px;
|
|
vertical-align: middle;
|
|
object-fit: contain;
|
|
}
|
|
|
|
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;
|
|
margin-top: 0.2em;
|
|
margin-bottom: 0.5em;
|
|
}
|
|
|
|
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 0.6em 0 0.6em;
|
|
margin: 0;
|
|
|
|
.avatar {
|
|
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;
|
|
|
|
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;
|
|
|
|
div, favorite-button {
|
|
padding-top: 0.25em;
|
|
max-width: 6em;
|
|
flex: 1;
|
|
}
|
|
}
|
|
|
|
.icon-reply:hover {
|
|
color: $fallback--cBlue;
|
|
color: var(--cBlue, $fallback--cBlue);
|
|
}
|
|
|
|
.icon-reply.icon-reply-active {
|
|
color: $fallback--cBlue;
|
|
color: var(--cBlue, $fallback--cBlue);
|
|
}
|
|
|
|
.status .avatar-compact {
|
|
width: 32px;
|
|
height: 32px;
|
|
box-shadow: var(--avatarStatusShadow);
|
|
border-radius: $fallback--avatarAltRadius;
|
|
border-radius: var(--avatarAltRadius, $fallback--avatarAltRadius);
|
|
|
|
&.better-shadow {
|
|
box-shadow: var(--avatarStatusShadowInset);
|
|
filter: var(--avatarStatusShadowFilter)
|
|
}
|
|
}
|
|
|
|
.avatar.still-image {
|
|
width: 48px;
|
|
height: 48px;
|
|
box-shadow: var(--avatarStatusShadow);
|
|
border-radius: $fallback--avatarRadius;
|
|
border-radius: var(--avatarRadius, $fallback--avatarRadius);
|
|
overflow: hidden;
|
|
position: relative;
|
|
|
|
&.better-shadow {
|
|
box-shadow: var(--avatarStatusShadowInset);
|
|
filter: var(--avatarStatusShadowFilter)
|
|
}
|
|
|
|
img {
|
|
width: 100%;
|
|
height: 100%;
|
|
}
|
|
|
|
&.animated::before {
|
|
display: none;
|
|
}
|
|
|
|
&.retweeted {
|
|
}
|
|
}
|
|
|
|
.status:hover .animated.avatar {
|
|
canvas {
|
|
display: none;
|
|
}
|
|
img {
|
|
visibility: visible;
|
|
}
|
|
}
|
|
|
|
.status {
|
|
display: flex;
|
|
padding: 0.6em;
|
|
&.is-retweet {
|
|
padding-top: 0.1em;
|
|
}
|
|
}
|
|
|
|
.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-bottom-radius: 0 0 $fallback--panelRadius $fallback--panelRadius;;
|
|
border-radius: 0 0 var(--panelRadius, $fallback--panelRadius) var(--panelRadius, $fallback--panelRadius);
|
|
border-bottom: none;
|
|
}
|
|
}
|
|
|
|
@media all and (max-width: 960px) {
|
|
.status-el {
|
|
.retweet-info {
|
|
.avatar {
|
|
margin-left: 20px;
|
|
}
|
|
}
|
|
}
|
|
.status {
|
|
max-width: 100%;
|
|
}
|
|
|
|
.status .avatar {
|
|
width: 40px;
|
|
height: 40px;
|
|
}
|
|
|
|
.status .avatar-compact {
|
|
width: 32px;
|
|
height: 32px;
|
|
}
|
|
}
|
|
|
|
</style>
|