pinafore/routes/_components/compose/ComposeBox.html

254 lines
8.4 KiB
HTML
Raw Normal View History

{{#if realm === 'home'}}
<h1 class="sr-only">Compose toot</h1>
{{/if}}
2018-03-30 08:06:17 +00:00
<div class="{{computedClassName}} {{hideAndFadeIn}}">
<ComposeAuthor />
2018-03-03 23:44:43 +00:00
{{#if contentWarningShown}}
2018-03-04 00:25:22 +00:00
<div class="compose-content-warning-wrapper"
transition:slide="{duration: 333}">
<ComposeContentWarning :realm :contentWarning />
</div>
2018-03-03 23:44:43 +00:00
{{/if}}
2018-04-12 03:00:43 +00:00
<ComposeInput :realm :text :autoFocus on:postAction="doPostStatus()" />
2018-03-04 00:12:48 +00:00
<ComposeLengthGauge :length :overLimit />
2018-03-25 01:04:54 +00:00
<ComposeToolbar :realm :postPrivacy :media :contentWarningShown :text />
2018-03-04 00:12:48 +00:00
<ComposeLengthIndicator :length :overLimit />
<ComposeMedia :realm :media :mediaDescriptions />
2018-02-26 00:26:43 +00:00
</div>
<div class="compose-box-button-sentinel {{hideAndFadeIn}}" ref:sentinel></div>
2018-03-30 08:06:17 +00:00
<div class="compose-box-button-wrapper {{realm === 'home' ? 'compose-button-sticky' : ''}} {{hideAndFadeIn}}" >
2018-03-27 07:02:55 +00:00
<ComposeButton :length :overLimit :sticky on:click="onClickPostButton()" />
</div>
2018-03-30 08:06:17 +00:00
{{#if !hideBottomBorder}}
<div class="compose-box-border-bottom {{hideAndFadeIn}}"></div>
{{/if}}
2018-02-26 00:26:43 +00:00
<style>
2018-02-27 05:54:21 +00:00
.compose-box {
2018-02-26 00:26:43 +00:00
border-radius: 4px;
2018-03-27 07:02:55 +00:00
padding: 20px 20px 0 20px;
2018-02-26 00:26:43 +00:00
display: grid;
align-items: flex-start;
grid-template-areas:
2018-03-03 23:44:43 +00:00
"avatar name handle handle"
"avatar cw cw cw"
"avatar input input input"
"avatar gauge gauge gauge"
"avatar toolbar toolbar length"
2018-03-27 07:02:55 +00:00
"avatar media media media";
grid-template-columns: min-content minmax(0, max-content) 1fr 1fr;
2018-02-26 00:26:43 +00:00
width: 560px;
max-width: calc(100vw - 40px);
}
2018-04-05 06:03:26 +00:00
.compose-box.direct-reply {
background-color: var(--status-direct-background);
}
2018-03-30 08:06:17 +00:00
.compose-box.slim-size {
2018-03-27 07:02:55 +00:00
width: 540px;
2018-03-28 16:05:22 +00:00
max-width: calc(100vw - 60px);
2018-03-27 07:02:55 +00:00
}
.compose-box-fade-in {
transition: opacity 0.2s linear; /* main page reveal */
}
2018-03-27 07:02:55 +00:00
.compose-box-border-bottom {
height: 1px;
background: var(--main-border);
width: 100%;
}
.compose-box-button-wrapper {
/*
* We want pointer-events only for the sticky button, so use fit-content so that
* the element doesn't take up the full width, and then set its left margin to
* auto so that it sticks to the right. fit-content doesn't work in Edge, but
* that just means that content that is level with the button is not clickable.
*/
width: -moz-fit-content;
width: fit-content;
margin-left: auto;
2018-03-27 07:02:55 +00:00
display: flex;
justify-content: flex-end;
2018-03-30 08:06:17 +00:00
}
.compose-box-button-wrapper.compose-button-sticky {
2018-03-27 07:02:55 +00:00
position: -webkit-sticky;
position: sticky;
top: 10px;
z-index: 5000;
2018-03-27 07:02:55 +00:00
}
2018-03-04 00:25:22 +00:00
.compose-content-warning-wrapper {
grid-area: cw;
}
2018-02-26 00:26:43 +00:00
@media (max-width: 767px) {
2018-02-27 05:54:21 +00:00
.compose-box {
2018-03-27 07:02:55 +00:00
padding: 10px 10px 0 10px;
2018-02-26 00:26:43 +00:00
max-width: calc(100vw - 20px);
width: 580px;
}
2018-03-30 08:06:17 +00:00
.compose-box.slim-size {
2018-03-27 07:02:55 +00:00
width: 560px;
max-width: calc(100vw - 40px);
}
2018-04-13 02:37:37 +00:00
.compose-box-button-wrapper.compose-button-sticky {
2018-03-27 07:02:55 +00:00
top: 5px;
}
2018-02-26 00:26:43 +00:00
}
</style>
<script>
2018-02-27 05:50:03 +00:00
import ComposeToolbar from './ComposeToolbar.html'
import ComposeLengthGauge from './ComposeLengthGauge.html'
import ComposeLengthIndicator from './ComposeLengthIndicator.html'
2018-02-27 05:54:21 +00:00
import ComposeAuthor from './ComposeAuthor.html'
2018-02-27 06:22:56 +00:00
import ComposeInput from './ComposeInput.html'
import ComposeButton from './ComposeButton.html'
2018-03-02 05:21:49 +00:00
import ComposeMedia from './ComposeMedia.html'
2018-03-03 23:44:43 +00:00
import ComposeContentWarning from './ComposeContentWarning.html'
2018-03-03 22:51:48 +00:00
import { measureText } from '../../_utils/measureText'
import { CHAR_LIMIT, POST_PRIVACY_OPTIONS } from '../../_static/statuses'
import { store } from '../../_store/store'
2018-03-04 00:25:22 +00:00
import { slide } from 'svelte-transitions'
2018-04-04 00:50:48 +00:00
import { postStatus, insertHandleForReply, setReplySpoiler, setReplyVisibility } from '../../_actions/compose'
2018-03-27 07:02:55 +00:00
import { importDialogs } from '../../_utils/asyncModules'
import { classname } from '../../_utils/classname'
2018-02-26 01:21:17 +00:00
2018-02-26 00:26:43 +00:00
export default {
2018-03-08 02:04:20 +00:00
oncreate() {
let { realm } = this.get()
2018-03-27 07:02:55 +00:00
if (realm === 'home') {
this.setupStickyObserver()
2018-03-27 07:02:55 +00:00
} else if (realm !== 'dialog') {
2018-03-09 16:45:12 +00:00
// if this is a reply, populate the handle immediately
insertHandleForReply(realm)
2018-03-27 07:02:55 +00:00
}
2018-03-09 16:45:12 +00:00
let { replySpoiler } = this.get()
if (replySpoiler) {
// default spoiler is same as the replied-to status
setReplySpoiler(realm, replySpoiler)
}
let { replyVisibility } = this.get()
2018-04-04 00:50:48 +00:00
if (replyVisibility) {
// make sure the visibility is consistent with the replied-to status
setReplyVisibility(realm, replyVisibility)
}
2018-03-27 07:02:55 +00:00
},
ondestroy() {
this.teardownStickyObserver()
2018-03-08 02:04:20 +00:00
},
2018-02-26 00:26:43 +00:00
components: {
2018-02-27 05:54:21 +00:00
ComposeAuthor,
2018-02-27 05:50:03 +00:00
ComposeToolbar,
ComposeLengthGauge,
2018-02-27 06:22:56 +00:00
ComposeLengthIndicator,
ComposeInput,
2018-03-02 05:21:49 +00:00
ComposeButton,
2018-03-03 23:44:43 +00:00
ComposeMedia,
ComposeContentWarning
2018-03-03 22:51:48 +00:00
},
store: () => store,
computed: {
2018-04-05 06:03:26 +00:00
computedClassName: (overLimit, realm, size, postPrivacyKey, isReply) => {
return classname(
'compose-box',
overLimit && 'over-char-limit',
2018-04-05 06:03:26 +00:00
size === 'slim' && 'slim-size',
isReply && postPrivacyKey === 'direct' && 'direct-reply'
)
},
hideAndFadeIn: (hidden) => {
return classname(
'compose-box-fade-in',
hidden && 'hidden'
)
},
2018-03-03 22:51:48 +00:00
composeData: ($currentComposeData, realm) => $currentComposeData[realm] || {},
text: (composeData) => composeData.text || '',
media: (composeData) => composeData.media || [],
postPrivacy: (postPrivacyKey) => POST_PRIVACY_OPTIONS.find(_ => _.key === postPrivacyKey),
2018-04-04 00:50:48 +00:00
defaultPostPrivacyKey: ($currentVerifyCredentials) => $currentVerifyCredentials.source.privacy,
2018-03-03 22:51:48 +00:00
postPrivacyKey: (composeData, defaultPostPrivacyKey) => composeData.postPrivacy || defaultPostPrivacyKey,
textLength: (text) => measureText(text),
2018-03-04 00:12:48 +00:00
contentWarningLength: (contentWarning) => measureText(contentWarning),
length: (textLength, contentWarningLength, contentWarningShown) => {
return textLength + (contentWarningShown ? contentWarningLength : 0)
},
overLimit: (length) => length > CHAR_LIMIT,
2018-03-03 23:44:43 +00:00
contentWarningShown: (composeData) => composeData.contentWarningShown,
2018-03-08 02:04:20 +00:00
contentWarning: (composeData) => composeData.contentWarning || '',
timelineInitialized: ($timelineInitialized) => $timelineInitialized,
mediaDescriptions: (composeData) => composeData.mediaDescriptions || []
2018-03-04 00:25:22 +00:00
},
transitions: {
slide
2018-03-05 00:27:15 +00:00
},
methods: {
2018-03-27 07:02:55 +00:00
async onClickPostButton() {
let { sticky } = this.get()
if (sticky) {
2018-03-27 07:02:55 +00:00
// when the button is sticky, we're scrolled down the home timeline,
// so we should launch a new compose dialog
let dialogs = await importDialogs()
dialogs.showComposeDialog()
} else {
// else we're actually posting a new toot
2018-04-12 03:00:43 +00:00
this.doPostStatus();
}
},
2018-04-12 03:00:43 +00:00
doPostStatus() {
let {
text,
media,
postPrivacyKey,
contentWarning,
realm,
overLimit,
mediaDescriptions,
inReplyToUuid
} = this.get()
let sensitive = media.length && !!contentWarning
let mediaIds = media.map(_ => _.data.id)
let inReplyTo = (realm === 'home' || realm === 'dialog') ? null : realm
2018-03-08 02:04:20 +00:00
if (!text || overLimit) {
return // do nothing if invalid
2018-03-27 07:02:55 +00:00
}
/* no await */
postStatus(realm, text, inReplyTo, mediaIds,
sensitive, contentWarning, postPrivacyKey,
mediaDescriptions, inReplyToUuid)
},
setupStickyObserver() {
this.__stickyObserver = new IntersectionObserver(entries => {
this.set({sticky: !entries[0].isIntersecting})
})
this.__stickyObserver.observe(this.refs.sentinel)
// also create a one-shot observer for the $timelineInitialized event,
// due to a bug in Firefox where when the scrollTop is set
// manually, the other observer doesn't necessarily fire
this.observe('timelineInitialized', timelineInitialized => {
if (timelineInitialized) {
let observer = new IntersectionObserver(entries => {
this.set({sticky: !entries[0].isIntersecting})
observer.disconnect()
})
observer.observe(this.refs.sentinel)
}
}, {init: false})
},
teardownStickyObserver() {
if (this.__stickyObserver) {
this.__stickyObserver.disconnect()
}
2018-03-05 00:27:15 +00:00
}
2018-02-26 00:26:43 +00:00
}
}
2018-02-26 08:24:28 +00:00
</script>