Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/nextcloud/server.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'apps/user_status/src/UserStatus.vue')
-rw-r--r--apps/user_status/src/UserStatus.vue219
1 files changed, 219 insertions, 0 deletions
diff --git a/apps/user_status/src/UserStatus.vue b/apps/user_status/src/UserStatus.vue
new file mode 100644
index 00000000000..51761582595
--- /dev/null
+++ b/apps/user_status/src/UserStatus.vue
@@ -0,0 +1,219 @@
+<!--
+ - @copyright Copyright (c) 2020 Georg Ehrke <oc.list@georgehrke.com>
+ - @author Georg Ehrke <oc.list@georgehrke.com>
+ -
+ - @license GNU AGPL version 3 or any later version
+ -
+ - This program is free software: you can redistribute it and/or modify
+ - it under the terms of the GNU Affero General Public License as
+ - published by the Free Software Foundation, either version 3 of the
+ - License, or (at your option) any later version.
+ -
+ - This program is distributed in the hope that it will be useful,
+ - but WITHOUT ANY WARRANTY; without even the implied warranty of
+ - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ - GNU Affero General Public License for more details.
+ -
+ - You should have received a copy of the GNU Affero General Public License
+ - along with this program. If not, see <http://www.gnu.org/licenses/>.
+ -
+ -->
+
+<template>
+ <li>
+ <div class="user-status-menu-item">
+ <!-- Username display -->
+ <span
+ v-if="!inline"
+ class="user-status-menu-item__header"
+ :title="displayName">
+ {{ displayName }}
+ </span>
+
+ <!-- Status modal toggle -->
+ <toggle :is="inline ? 'button' : 'a'"
+ :class="{'user-status-menu-item__toggle--inline': inline}"
+ class="user-status-menu-item__toggle"
+ href="#"
+ @click.prevent.stop="openModal">
+ <span :class="statusIcon" class="user-status-menu-item__toggle-icon" />
+ {{ visibleMessage }}
+ </toggle>
+ </div>
+
+ <!-- Status management modal -->
+ <SetStatusModal
+ v-if="isModalOpen"
+ @close="closeModal" />
+ </li>
+</template>
+
+<script>
+import { getCurrentUser } from '@nextcloud/auth'
+import debounce from 'debounce'
+
+import { sendHeartbeat } from './services/heartbeatService'
+import OnlineStatusMixin from './mixins/OnlineStatusMixin'
+
+export default {
+ name: 'UserStatus',
+
+ components: {
+ SetStatusModal: () => import(/* webpackChunkName: 'user-status-modal' */'./components/SetStatusModal'),
+ },
+ mixins: [OnlineStatusMixin],
+
+ props: {
+ inline: {
+ type: Boolean,
+ default: false,
+ },
+ },
+
+ data() {
+ return {
+ isModalOpen: false,
+ heartbeatInterval: null,
+ setAwayTimeout: null,
+ mouseMoveListener: null,
+ isAway: false,
+ }
+ },
+ computed: {
+ /**
+ * The display-name of the current user
+ *
+ * @returns {String}
+ */
+ displayName() {
+ return getCurrentUser().displayName
+ },
+ },
+
+ /**
+ * Loads the current user's status from initial state
+ * and stores it in Vuex
+ */
+ mounted() {
+ this.$store.dispatch('loadStatusFromInitialState')
+
+ if (OC.config.session_keepalive) {
+ // Send the latest status to the server every 5 minutes
+ this.heartbeatInterval = setInterval(this._backgroundHeartbeat.bind(this), 1000 * 60 * 5)
+ this.setAwayTimeout = () => {
+ this.isAway = true
+ }
+ // Catch mouse movements, but debounce to once every 30 seconds
+ this.mouseMoveListener = debounce(() => {
+ const wasAway = this.isAway
+ this.isAway = false
+ // Reset the two minute counter
+ clearTimeout(this.setAwayTimeout)
+ // If the user did not move the mouse within two minutes,
+ // mark them as away
+ setTimeout(this.setAwayTimeout, 1000 * 60 * 2)
+
+ if (wasAway) {
+ this._backgroundHeartbeat()
+ }
+ }, 1000 * 2, true)
+ window.addEventListener('mousemove', this.mouseMoveListener, {
+ capture: true,
+ passive: true,
+ })
+
+ this._backgroundHeartbeat()
+ }
+ },
+
+ /**
+ * Some housekeeping before destroying the component
+ */
+ beforeDestroy() {
+ window.removeEventListener('mouseMove', this.mouseMoveListener)
+ clearInterval(this.heartbeatInterval)
+ },
+
+ methods: {
+ /**
+ * Opens the modal to set a custom status
+ */
+ openModal() {
+ this.isModalOpen = true
+ },
+ /**
+ * Closes the modal
+ */
+ closeModal() {
+ this.isModalOpen = false
+ },
+
+ /**
+ * Sends the status heartbeat to the server
+ *
+ * @returns {Promise<void>}
+ * @private
+ */
+ async _backgroundHeartbeat() {
+ await sendHeartbeat(this.isAway)
+ await this.$store.dispatch('reFetchStatusFromServer')
+ },
+ },
+}
+</script>
+
+<style lang="scss">
+$max-width-user-status: 200px;
+
+.user-status-menu-item {
+ &__header {
+ display: block;
+ overflow: hidden;
+ box-sizing: border-box;
+ max-width: $max-width-user-status;
+ padding: 10px 12px 5px 38px;
+ text-align: left;
+ white-space: nowrap;
+ text-overflow: ellipsis;
+ opacity: 1;
+ color: var(--color-text-maxcontrast);
+ }
+
+ &__toggle {
+ &-icon {
+ width: 16px;
+ height: 16px;
+ margin-right: 10px;
+ opacity: 1 !important;
+ background-size: 16px;
+ }
+
+ // In dashboard
+ &--inline {
+ width: auto;
+ min-width: 44px;
+ height: 44px;
+ margin: 0;
+ border: 0;
+ border-radius: var(--border-radius-pill);
+ background-color: var(--color-background-translucent);
+ font-size: inherit;
+ font-weight: normal;
+
+ -webkit-backdrop-filter: var(--background-blur);
+ backdrop-filter: var(--background-blur);
+
+ &:active,
+ &:hover,
+ &:focus {
+ background-color: var(--color-background-hover);
+ }
+ }
+ }
+}
+
+li {
+ list-style-type: none;
+}
+
+</style>