1
0
Fork 0
mirror of https://github.com/FrankerFaceZ/FrankerFaceZ.git synced 2025-08-03 08:28:31 +00:00
FrankerFaceZ/src/modules/main_menu/components/main-menu.vue
SirStendec 7428781614 4.20.77
* Changed: The FFZ Control Center's home page now displays a list of new settings when you have new settings, as well as recently updated add-ons. A GitHub link was added as well. The Twitter embed will properly use light theme when appropriate.
* Changed: [Add-Ons](~add_ons) now supports sorting by update time, as well as name.
* Changed: The FFZ Control Center will now appear larger depending on your window size.
* Fixed: The [Add-Ons](~add_ons) list not being properly searchable in the FFZ Control Center.
* Fixed: Appearance of unread counters in the FFZ Control Center's navigation.
* Fixed: Messages not highlighting correctly when a viewer card is open.
2021-03-12 20:16:39 -05:00

307 lines
No EOL
7.8 KiB
Vue

<template lang="html">
<div
:class="{ maximized: maximized || exclusive, exclusive, faded }"
class="ffz-dialog ffz-main-menu tw-elevation-3 tw-c-background-alt tw-c-text-base tw-border tw-flex tw-flex-nowrap tw-flex-column"
>
<header ref="header" class="tw-c-background-base tw-full-width tw-align-items-center tw-flex tw-flex-nowrap" @dblclick="maybeResize($event)">
<h3 class="ffz-i-zreknarf ffz-i-pd-1">
FrankerFaceZ
</h3>
<div class="tw-flex-grow-1 tw-pd-x-2">
<div class="tw-search-input">
<label for="ffz-main-menu.search" class="tw-hide-accessible">{{ t('main-menu.search', 'Search Settings') }}</label>
<div class="tw-relative">
<div class="tw-absolute tw-align-items-center tw-c-text-alt-2 tw-flex tw-full-height ffz-input__icon tw-justify-content-center tw-left-0 tw-top-0 tw-z-default">
<figure class="ffz-i-search" />
</div>
<input
id="ffz-main-menu.search"
v-model="query"
:placeholder="t('main-menu.search', 'Search Settings')"
type="search"
class="tw-block tw-border-radius-medium tw-font-size-6 tw-full-width ffz-input tw-pd-l-3 tw-pd-r-1 tw-pd-y-05"
autocapitalize="off"
autocorrect="off"
autocomplete="off"
spellcheck="false"
>
</div>
</div>
</div>
<button v-if="!maximized && !exclusive" class="tw-button-icon tw-mg-x-05" @click="faded = ! faded">
<span class="tw-button-icon__icon">
<figure :class="faded ? 'ffz-i-eye-off' : 'ffz-i-eye'" />
</span>
</button>
<button v-if="!exclusive" class="tw-button-icon tw-mg-x-05 tw-relative tw-tooltip__container" @click="popout">
<span class="tw-button-icon__icon">
<figure class="ffz-i-link-ext" />
</span>
<div class="tw-tooltip tw-tooltip--down tw-tooltip--align-center">
{{ t('main-menu.popout', 'Open Settings in a New Window') }}
</div>
</button>
<button v-if="!exclusive" class="tw-button-icon tw-mg-x-05" @click="resize">
<span class="tw-button-icon__icon">
<figure :class="{'ffz-i-window-maximize': !maximized, 'ffz-i-window-restore': maximized}" />
</span>
</button>
<button v-if="!exclusive" class="tw-button-icon tw-mg-x-05" @click="close">
<span class="tw-button-icon__icon">
<figure class="ffz-i-window-close" />
</span>
</button>
</header>
<section class="tw-border-t tw-full-height tw-full-width tw-flex tw-flex-nowrap tw-overflow-hidden">
<nav class="ffz-vertical-nav tw-c-background-alt-2 tw-border-r tw-full-height tw-flex tw-flex-column tw-flex-shrink-0 tw-flex-nowrap">
<header class="tw-border-b tw-pd-1">
<profile-selector
:context="context"
@navigate="navigate"
/>
</header>
<div class="tw-full-width tw-full-height tw-overflow-hidden tw-flex tw-flex-nowrap tw-relative">
<simplebar classes="ffz-vertical-nav__items tw-full-width tw-flex-grow-1">
<header v-if="has_unseen" class="tw-border-b tw-pd-1">
<button
class="tw-button ffz-button--hollow tw-full-width"
@click="allRead"
>
<span class="tw-button__text">
{{ t('main-menu.mark-all-seen', 'Mark All Seen') }}
</span>
</button>
</header>
<menu-tree
:current-item="currentItem"
:modal="nav"
:filter="filter"
@change-item="changeItem"
@mark-seen="markSeen"
@mark-expanded="markExpanded"
@navigate="navigate"
/>
</simplebar>
</div>
<footer class="tw-c-text-alt tw-border-t tw-pd-1">
<div>
{{ t('main-menu.version', 'Version {version}', {version: version.toString()}) }}
</div>
<div class="tw-c-text-alt-2">
<a
v-if="version.commit"
:href="`https://www.github.com/FrankerFaceZ/FrankerFaceZ/commit/${version.commit}`"
class="tw-link tw-link--inherit"
target="_blank"
rel="noopener noreferrer"
>
{{ version.commit.slice(0,7) }}
</a>
<span v-else>
{{ version.build }}
</span>
</div>
</footer>
</nav>
<simplebar classes="tw-flex-grow-1">
<menu-page
v-if="currentItem"
ref="page"
:context="context"
:item="currentItem"
:filter="filter"
:nav_keys="nav_keys"
@change-item="changeItem"
@mark-seen="markSeen"
@navigate="navigate"
/>
</simplebar>
</section>
</div>
</template>
<script>
import displace from 'displacejs';
export default {
data() {
return this.$vnode.data;
},
computed: {
filter() {
return this.query.toLowerCase()
}
},
watch: {
maximized() {
this.updateDrag();
}
},
created() {
this.context.context._add_user();
},
destroyed() {
this.context.context._remove_user();
},
mounted() {
this.updateDrag();
this._on_resize = this.handleResize.bind(this);
window.addEventListener('resize', this._on_resize);
},
beforeDestroy() {
this.destroyDrag();
if ( this._on_resize ) {
window.removeEventListener('resize', this._on_resize);
this._on_resize = null;
}
},
methods: {
allRead() {
this.markAllSeen(this.nav);
this.has_unseen = false;
},
maybeResize(event) {
if ( event.target !== this.$refs.header )
return;
this.resize(event);
},
changeProfile() {
const new_id = this.$refs.profiles.value,
new_profile = this.context.profiles[new_id];
if ( new_profile )
this.context.currentProfile = new_profile;
},
changeItem(item) {
if ( this.$refs.page && this.$refs.page.onBeforeChange ) {
if ( this.$refs.page.onBeforeChange(this.currentItem, item) === false )
return;
}
this.markSeen(item);
if ( item.redirect )
return this.navigate(Array.isArray(item.redirect) ? item.redirect : item.redirect.split(/\./g));
this.currentItem = item;
this.restoredItem = true;
let url;
if ( this.exclusive ) {
url = new URL(location.href);
url.searchParams.set('ffz-settings', item.full_key);
url = url.toString();
}
try {
window.history.replaceState({
...window.history.state,
ffzcc: item.full_key
}, document.title, url);
} catch(err) {
/* no-op */
}
let current = item;
while(current = current.parent) // eslint-disable-line no-cond-assign
current.expanded = true;
},
updateDrag() {
if ( this.maximized )
this.destroyDrag();
else
this.createDrag();
},
destroyDrag() {
if ( this.displace ) {
this.displace.destroy();
this.displace = null;
}
},
createDrag() {
this.$nextTick(() => {
if ( ! this.maximized )
this.displace = displace(this.$el, {
handle: this.$el.querySelector('header'),
highlightInputs: true,
constrain: true,
ignoreFn(event) {
return event.target.closest('button') != null
}
});
})
},
handleResize() {
if ( this.displace )
this.displace.reinit();
},
navigate(...keys) {
let item;
for(const key of keys) {
item = this.nav_keys[key];
if ( item )
break;
}
const tabs = [];
while(item && item.page) {
if ( item.tab && item.parent?.tabs )
tabs.push([item.parent, item.parent.tabs.indexOf(item)]);
item = item.parent;
}
if ( ! item )
return;
if ( this.$refs.page && this.$refs.page.onBeforeChange ) {
if ( this.$refs.page.onBeforeChange(this.currentItem, item) === false )
return;
}
this.changeItem(item);
// Asynchronously walk down the tab tree, so that
// we can switch every tab correctly.
if ( tabs.length ) {
const bits = () => {
const latest = tabs.pop();
if ( latest?.[0]?._component )
latest[0]._component.select(latest[1]);
if ( tabs.length )
this.$nextTick(bits);
}
this.$nextTick(bits);
}
this.$nextTick(() => {
const el = this.$el.querySelector(`.ffz--menu-tree li[data-key="${item.full_key}"]`);
if ( el )
el.scrollIntoView();
});
}
}
}
</script>