mirror of
https://github.com/flarum/core.git
synced 2025-10-10 14:34:30 +02:00
* Update FontAwesome to v5.0.6 * Adapt DiscussionListItem-count icon to match FontAwesome 5 syntax * Change icon name to match FontAwesome 5.0.6 fas icon * Add font type prefix parameter to icon helper * Add Enable Icon Prefix to show icon in Extension Page * Fix invalid icon behavior * Change icon name to match FontAwesome 5.0.6 far icon * Use iconPrefix property on component * Use full icon class name * Update icon helper docblock * Full icon class syntax
531 lines
15 KiB
JavaScript
531 lines
15 KiB
JavaScript
import Component from 'flarum/Component';
|
|
import ItemList from 'flarum/utils/ItemList';
|
|
import ComposerButton from 'flarum/components/ComposerButton';
|
|
import listItems from 'flarum/helpers/listItems';
|
|
import classList from 'flarum/utils/classList';
|
|
|
|
/**
|
|
* The `Composer` component displays the composer. It can be loaded with a
|
|
* content component with `load` and then its position/state can be altered with
|
|
* `show`, `hide`, `close`, `minimize`, `fullScreen`, and `exitFullScreen`.
|
|
*/
|
|
class Composer extends Component {
|
|
init() {
|
|
/**
|
|
* The composer's current position.
|
|
*
|
|
* @type {Composer.PositionEnum}
|
|
*/
|
|
this.position = Composer.PositionEnum.HIDDEN;
|
|
|
|
/**
|
|
* The composer's intended height, which can be modified by the user
|
|
* (by dragging the composer handle).
|
|
*
|
|
* @type {Integer}
|
|
*/
|
|
this.height = null;
|
|
|
|
/**
|
|
* Whether or not the composer currently has focus.
|
|
*
|
|
* @type {Boolean}
|
|
*/
|
|
this.active = false;
|
|
}
|
|
|
|
view() {
|
|
const classes = {
|
|
'normal': this.position === Composer.PositionEnum.NORMAL,
|
|
'minimized': this.position === Composer.PositionEnum.MINIMIZED,
|
|
'fullScreen': this.position === Composer.PositionEnum.FULLSCREEN,
|
|
'active': this.active
|
|
};
|
|
classes.visible = classes.normal || classes.minimized || classes.fullScreen;
|
|
|
|
// If the composer is minimized, tell the composer's content component that
|
|
// it shouldn't let the user interact with it. Set up a handler so that if
|
|
// the content IS clicked, the composer will be shown.
|
|
if (this.component) this.component.props.disabled = classes.minimized;
|
|
|
|
const showIfMinimized = this.position === Composer.PositionEnum.MINIMIZED ? this.show.bind(this) : undefined;
|
|
|
|
return (
|
|
<div className={'Composer ' + classList(classes)}>
|
|
<div className="Composer-handle" config={this.configHandle.bind(this)}/>
|
|
<ul className="Composer-controls">{listItems(this.controlItems().toArray())}</ul>
|
|
<div className="Composer-content" onclick={showIfMinimized}>
|
|
{this.component ? this.component.render() : ''}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
config(isInitialized, context) {
|
|
// Set the height of the Composer element and its contents on each redraw,
|
|
// so that they do not lose it if their DOM elements are recreated.
|
|
this.updateHeight();
|
|
|
|
if (isInitialized) return;
|
|
|
|
// Since this component is a part of the global UI that persists between
|
|
// routes, we will flag the DOM to be retained across route changes.
|
|
context.retain = true;
|
|
|
|
this.initializeHeight();
|
|
this.$().hide().css('bottom', -this.computedHeight());
|
|
|
|
// Whenever any of the inputs inside the composer are have focus, we want to
|
|
// add a class to the composer to draw attention to it.
|
|
this.$().on('focus blur', ':input', e => {
|
|
this.active = e.type === 'focusin';
|
|
m.redraw();
|
|
});
|
|
|
|
// When the escape key is pressed on any inputs, close the composer.
|
|
this.$().on('keydown', ':input', 'esc', () => this.close());
|
|
|
|
// Don't let the user leave the page without first giving the composer's
|
|
// component a chance to scream at the user to make sure they don't
|
|
// unintentionally lose any contnet.
|
|
window.onbeforeunload = () => {
|
|
return (this.component && this.component.preventExit()) || undefined;
|
|
};
|
|
|
|
const handlers = {};
|
|
|
|
$(window).on('resize', handlers.onresize = this.updateHeight.bind(this)).resize();
|
|
|
|
$(document)
|
|
.on('mousemove', handlers.onmousemove = this.onmousemove.bind(this))
|
|
.on('mouseup', handlers.onmouseup = this.onmouseup.bind(this));
|
|
|
|
context.onunload = () => {
|
|
$(window).off('resize', handlers.onresize);
|
|
|
|
$(document)
|
|
.off('mousemove', handlers.onmousemove)
|
|
.off('mouseup', handlers.onmouseup);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Add the necessary event handlers to the composer's handle so that it can
|
|
* be used to resize the composer.
|
|
*
|
|
* @param {DOMElement} element
|
|
* @param {Boolean} isInitialized
|
|
*/
|
|
configHandle(element, isInitialized) {
|
|
if (isInitialized) return;
|
|
|
|
const composer = this;
|
|
|
|
$(element).css('cursor', 'row-resize')
|
|
.bind('dragstart mousedown', e => e.preventDefault())
|
|
.mousedown(function(e) {
|
|
composer.mouseStart = e.clientY;
|
|
composer.heightStart = composer.$().height();
|
|
composer.handle = $(this);
|
|
$('body').css('cursor', 'row-resize');
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Resize the composer according to mouse movement.
|
|
*
|
|
* @param {Event} e
|
|
*/
|
|
onmousemove(e) {
|
|
if (!this.handle) return;
|
|
|
|
// Work out how much the mouse has been moved, and set the height
|
|
// relative to the old one based on that. Then update the content's
|
|
// height so that it fills the height of the composer, and update the
|
|
// body's padding.
|
|
const deltaPixels = this.mouseStart - e.clientY;
|
|
this.changeHeight(this.heightStart + deltaPixels);
|
|
|
|
// Update the body's padding-bottom so that no content on the page will ever
|
|
// get permanently hidden behind the composer. If the user is already
|
|
// scrolled to the bottom of the page, then we will keep them scrolled to
|
|
// the bottom after the padding has been updated.
|
|
const scrollTop = $(window).scrollTop();
|
|
const anchorToBottom = scrollTop > 0 && scrollTop + $(window).height() >= $(document).height();
|
|
this.updateBodyPadding(anchorToBottom);
|
|
}
|
|
|
|
/**
|
|
* Finish resizing the composer when the mouse is released.
|
|
*/
|
|
onmouseup() {
|
|
if (!this.handle) return;
|
|
|
|
this.handle = null;
|
|
$('body').css('cursor', '');
|
|
}
|
|
|
|
/**
|
|
* Update the DOM to reflect the composer's current height. This involves
|
|
* setting the height of the composer's root element, and adjusting the height
|
|
* of any flexible elements inside the composer's body.
|
|
*/
|
|
updateHeight() {
|
|
const height = this.computedHeight();
|
|
const $flexible = this.$('.Composer-flexible');
|
|
|
|
this.$().height(height);
|
|
|
|
if ($flexible.length) {
|
|
const headerHeight = $flexible.offset().top - this.$().offset().top;
|
|
const paddingBottom = parseInt($flexible.css('padding-bottom'), 10);
|
|
const footerHeight = this.$('.Composer-footer').outerHeight(true);
|
|
|
|
$flexible.height(this.$().outerHeight() - headerHeight - paddingBottom - footerHeight);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update the amount of padding-bottom on the body so that the page's
|
|
* content will still be visible above the composer when the page is
|
|
* scrolled right to the bottom.
|
|
*/
|
|
updateBodyPadding() {
|
|
const visible = this.position !== Composer.PositionEnum.HIDDEN &&
|
|
this.position !== Composer.PositionEnum.MINIMIZED &&
|
|
this.$().css('position') !== 'absolute';
|
|
|
|
const paddingBottom = visible
|
|
? this.computedHeight() - parseInt($('#app').css('padding-bottom'), 10)
|
|
: 0;
|
|
|
|
$('#content').css({paddingBottom});
|
|
}
|
|
|
|
/**
|
|
* Determine whether or not the Composer is covering the screen.
|
|
*
|
|
* This will be true if the Composer is in full-screen mode on desktop, or
|
|
* if the Composer is positioned absolutely as on mobile devices.
|
|
*
|
|
* @return {Boolean}
|
|
* @public
|
|
*/
|
|
isFullScreen() {
|
|
return this.position === Composer.PositionEnum.FULLSCREEN || this.$().css('position') === 'absolute';
|
|
}
|
|
|
|
/**
|
|
* Confirm with the user that they want to close the composer and lose their
|
|
* content.
|
|
*
|
|
* @return {Boolean} Whether or not the exit was cancelled.
|
|
*/
|
|
preventExit() {
|
|
if (this.component) {
|
|
const preventExit = this.component.preventExit();
|
|
|
|
if (preventExit) {
|
|
return !confirm(preventExit);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Load a content component into the composer.
|
|
*
|
|
* @param {Component} component
|
|
* @public
|
|
*/
|
|
load(component) {
|
|
if (this.preventExit()) return;
|
|
|
|
// If we load a similar component into the composer, then Mithril will be
|
|
// able to diff the old/new contents and some DOM-related state from the
|
|
// old composer will remain. To prevent this from happening, we clear the
|
|
// component and force a redraw, so that the new component will be working
|
|
// on a blank slate.
|
|
if (this.component) {
|
|
this.clear();
|
|
m.redraw(true);
|
|
}
|
|
|
|
this.component = component;
|
|
}
|
|
|
|
/**
|
|
* Clear the composer's content component.
|
|
*
|
|
* @public
|
|
*/
|
|
clear() {
|
|
this.component = null;
|
|
}
|
|
|
|
/**
|
|
* Animate the Composer into the given position.
|
|
*
|
|
* @param {Composer.PositionEnum} position
|
|
*/
|
|
animateToPosition(position) {
|
|
// Before we redraw the composer to its new state, we need to save the
|
|
// current height of the composer, as well as the page's scroll position, so
|
|
// that we can smoothly transition from the old to the new state.
|
|
const oldPosition = this.position;
|
|
const $composer = this.$().stop(true);
|
|
const oldHeight = $composer.outerHeight();
|
|
const scrollTop = $(window).scrollTop();
|
|
|
|
this.position = position;
|
|
|
|
m.redraw(true);
|
|
|
|
// Now that we've redrawn and the composer's DOM has been updated, we want
|
|
// to update the composer's height. Once we've done that, we'll capture the
|
|
// real value to use as the end point for our animation later on.
|
|
$composer.show();
|
|
this.updateHeight();
|
|
|
|
const newHeight = $composer.outerHeight();
|
|
|
|
if (oldPosition === Composer.PositionEnum.HIDDEN) {
|
|
$composer.css({bottom: -newHeight, height: newHeight});
|
|
} else {
|
|
$composer.css({height: oldHeight});
|
|
}
|
|
|
|
$composer.animate({bottom: 0, height: newHeight}, 'fast', () => this.component.focus());
|
|
|
|
this.updateBodyPadding();
|
|
$(window).scrollTop(scrollTop);
|
|
}
|
|
|
|
/**
|
|
* Show the Composer backdrop.
|
|
*/
|
|
showBackdrop() {
|
|
this.$backdrop = $('<div/>')
|
|
.addClass('composer-backdrop')
|
|
.appendTo('body');
|
|
}
|
|
|
|
/**
|
|
* Hide the Composer backdrop.
|
|
*/
|
|
hideBackdrop() {
|
|
if (this.$backdrop) this.$backdrop.remove();
|
|
}
|
|
|
|
/**
|
|
* Show the composer.
|
|
*
|
|
* @public
|
|
*/
|
|
show() {
|
|
if (this.position === Composer.PositionEnum.NORMAL || this.position === Composer.PositionEnum.FULLSCREEN) {
|
|
return;
|
|
}
|
|
|
|
this.animateToPosition(Composer.PositionEnum.NORMAL);
|
|
|
|
if (this.isFullScreen()) {
|
|
this.$().css('top', $(window).scrollTop());
|
|
this.showBackdrop();
|
|
this.component.focus();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Close the composer.
|
|
*
|
|
* @public
|
|
*/
|
|
hide() {
|
|
const $composer = this.$();
|
|
|
|
// Animate the composer sliding down off the bottom edge of the viewport.
|
|
// Only when the animation is completed, update the Composer state flag and
|
|
// other elements on the page.
|
|
$composer.stop(true).animate({bottom: -$composer.height()}, 'fast', () => {
|
|
this.position = Composer.PositionEnum.HIDDEN;
|
|
this.clear();
|
|
m.redraw();
|
|
|
|
$composer.hide();
|
|
this.hideBackdrop();
|
|
this.updateBodyPadding();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Confirm with the user so they don't lose their content, then close the
|
|
* composer.
|
|
*
|
|
* @public
|
|
*/
|
|
close() {
|
|
if (!this.preventExit()) {
|
|
this.hide();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Minimize the composer. Has no effect if the composer is hidden.
|
|
*
|
|
* @public
|
|
*/
|
|
minimize() {
|
|
if (this.position === Composer.PositionEnum.HIDDEN) return;
|
|
|
|
this.animateToPosition(Composer.PositionEnum.MINIMIZED);
|
|
|
|
this.$().css('top', 'auto');
|
|
this.hideBackdrop();
|
|
}
|
|
|
|
/**
|
|
* Take the composer into fullscreen mode. Has no effect if the composer is
|
|
* hidden.
|
|
*
|
|
* @public
|
|
*/
|
|
fullScreen() {
|
|
if (this.position !== Composer.PositionEnum.HIDDEN) {
|
|
this.position = Composer.PositionEnum.FULLSCREEN;
|
|
m.redraw();
|
|
this.updateHeight();
|
|
this.component.focus();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Exit fullscreen mode.
|
|
*
|
|
* @public
|
|
*/
|
|
exitFullScreen() {
|
|
if (this.position === Composer.PositionEnum.FULLSCREEN) {
|
|
this.position = Composer.PositionEnum.NORMAL;
|
|
m.redraw();
|
|
this.updateHeight();
|
|
this.component.focus();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Build an item list for the composer's controls.
|
|
*
|
|
* @return {ItemList}
|
|
*/
|
|
controlItems() {
|
|
const items = new ItemList();
|
|
|
|
if (this.position === Composer.PositionEnum.FULLSCREEN) {
|
|
items.add('exitFullScreen', ComposerButton.component({
|
|
icon: 'fa fa-compress',
|
|
title: app.translator.trans('core.forum.composer.exit_full_screen_tooltip'),
|
|
onclick: this.exitFullScreen.bind(this)
|
|
}));
|
|
} else {
|
|
if (this.position !== Composer.PositionEnum.MINIMIZED) {
|
|
items.add('minimize', ComposerButton.component({
|
|
icon: 'fa fa-minus minimize',
|
|
title: app.translator.trans('core.forum.composer.minimize_tooltip'),
|
|
onclick: this.minimize.bind(this),
|
|
itemClassName: 'App-backControl'
|
|
}));
|
|
|
|
items.add('fullScreen', ComposerButton.component({
|
|
icon: 'fa fa-expand',
|
|
title: app.translator.trans('core.forum.composer.full_screen_tooltip'),
|
|
onclick: this.fullScreen.bind(this)
|
|
}));
|
|
}
|
|
|
|
items.add('close', ComposerButton.component({
|
|
icon: 'fa fa-times',
|
|
title: app.translator.trans('core.forum.composer.close_tooltip'),
|
|
onclick: this.close.bind(this)
|
|
}));
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
/**
|
|
* Initialize default Composer height.
|
|
*/
|
|
initializeHeight() {
|
|
this.height = localStorage.getItem('composerHeight');
|
|
|
|
if (!this.height) {
|
|
this.height = this.defaultHeight();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Default height of the Composer in case none is saved.
|
|
* @returns {Integer}
|
|
*/
|
|
defaultHeight() {
|
|
return this.$().height();
|
|
}
|
|
|
|
/**
|
|
* Minimum height of the Composer.
|
|
* @returns {Integer}
|
|
*/
|
|
minimumHeight() {
|
|
return 200;
|
|
}
|
|
|
|
/**
|
|
* Maxmimum height of the Composer.
|
|
* @returns {Integer}
|
|
*/
|
|
maximumHeight() {
|
|
return $(window).height() - $('#header').outerHeight();
|
|
}
|
|
|
|
/**
|
|
* Computed the composer's current height, based on the intended height, and
|
|
* the composer's current state. This will be applied to the composer's
|
|
* content's DOM element.
|
|
* @returns {Integer|String}
|
|
*/
|
|
computedHeight() {
|
|
// If the composer is minimized, then we don't want to set a height; we'll
|
|
// let the CSS decide how high it is. If it's fullscreen, then we need to
|
|
// make it as high as the window.
|
|
if (this.position === Composer.PositionEnum.MINIMIZED) {
|
|
return '';
|
|
} else if (this.position === Composer.PositionEnum.FULLSCREEN) {
|
|
return $(window).height();
|
|
}
|
|
|
|
// Otherwise, if it's normal or hidden, then we use the intended height.
|
|
// We don't let the composer get too small or too big, though.
|
|
return Math.max(this.minimumHeight(), Math.min(this.height, this.maximumHeight()));
|
|
}
|
|
|
|
/**
|
|
* Save a new Composer height and update the DOM.
|
|
* @param {Integer} height
|
|
*/
|
|
changeHeight(height) {
|
|
this.height = height;
|
|
this.updateHeight();
|
|
|
|
localStorage.setItem('composerHeight', this.height);
|
|
}
|
|
}
|
|
|
|
Composer.PositionEnum = {
|
|
HIDDEN: 'hidden',
|
|
NORMAL: 'normal',
|
|
MINIMIZED: 'minimized',
|
|
FULLSCREEN: 'fullScreen'
|
|
};
|
|
|
|
export default Composer;
|