view client/src/components/usermanagement/Usermanagement.vue @ 5509:36cbf14b878a deactivate-users

Client: Add ability to list only active users * Adjust the "UIBoxHeader.vue" component to accept "checkbox" object which displays checkbox element in the header if it passed from the parent components. * Filter users according to the checked value from the checkbox. * Checkbox for hiding inactive users is only visible if there is at least one deactivated user.
author Fadi Abbud <fadi.abbud@intevation.de>
date Fri, 24 Sep 2021 13:10:25 +0200
parents 279900b28b1b
children b7792e8d5c62
line wrap: on
line source

<template>
  <div class="main d-flex flex-row" style="position: relative;">
    <Spacer />
    <div class="d-flex content py-2">
      <div :class="userlistStyle">
        <div class="card shadow-xs">
          <UIBoxHeader
            icon="users-cog"
            :title="usersLabel"
            :checkBox="checkboxObject"
          />
          <UITableHeader
            :columns="[
              { id: 'role', title: `${roleForColumLabel}`, class: 'col-1' },
              { id: 'user', title: `${usernameLabel}`, class: 'col-4' },
              { id: 'country', title: `${countryLabel}`, class: 'col-1' },
              { id: 'email', title: `${emailLabel}`, class: 'col-3' },
              { id: 'reports', title: `${reportsLabel}`, class: 'col-1' }
            ]"
          />
          <UITableBody
            :data="
              usersForTable
                | sortTable(sortColumn, sortDirection, page, pageSize)
            "
            :isActive="item => item === currentUser"
            maxHeight="47rem"
          >
            <template v-slot:row="{ item: user }">
              <div
                class="table-cell center col-1"
                :style="{ opacity: user.active ? '1' : '0.7' }"
                @click="selectUser(user.user)"
              >
                <font-awesome-icon
                  v-tooltip="roleLabel(user.role)"
                  :icon="roleIcon(user.role)"
                  class="fa-lg"
                />
              </div>
              <div
                class="table-cell col-4"
                @click="selectUser(user.user)"
                :style="{ opacity: user.active ? '1' : '0.7' }"
              >
                {{ user.user }}
              </div>
              <div
                :style="{ opacity: user.active ? '1' : '0.7' }"
                class="table-cell center col-1"
                @click="selectUser(user.user)"
              >
                {{ user.country }}
              </div>
              <div
                class="table-cell col-3"
                @click="selectUser(user.user)"
                :style="{ opacity: user.active ? '1' : '0.7' }"
              >
                {{ user.email }}
              </div>
              <div class="table-cell center col-1">
                <toggle-button
                  :value="user.reports"
                  v-model="user.reports"
                  class="pt-1"
                  :sync="true"
                  :speed="100"
                  @change="toggleReport(user)"
                  v-tooltip="receivesReportLabel"
                  :width="40"
                  :disabled="!user.active"
                  :height="20"
                />
              </div>
              <div class="table-cell col text-right justify-content-end">
                <button
                  @click="sendTestMail(user.user)"
                  class="btn btn-xs btn-dark mr-1"
                  v-tooltip="sendMailLabel"
                  v-if="user.email"
                  :disabled="!user.active"
                  :style="{ cursor: user.active ? 'pointer' : 'default' }"
                >
                  <font-awesome-icon icon="paper-plane" fixed-width />
                </button>
                <button
                  @click="deleteUser(user.user)"
                  class="btn btn-xs btn-dark"
                  v-tooltip="deleteUserLabel"
                  :style="{ cursor: user.active ? 'pointer' : 'default' }"
                  :disabled="!user.active"
                >
                  <font-awesome-icon icon="trash" fixed-width />
                </button>
              </div>
            </template>
          </UITableBody>
          <div class="p-3 border-top d-flex justify-content-between">
            <div></div>
            <div>
              <button
                @click="prevPage"
                v-if="this.page !== 1"
                class="mr-2 btn btn-sm btn-light align-self-center"
              >
                <font-awesome-icon icon="angle-left" />
              </button>
              {{ this.page }} / {{ this.pages }}
              <button
                @click="nextPage"
                v-if="this.page !== this.pages"
                class="ml-2 btn btn-sm btn-light align-self-center"
              >
                <font-awesome-icon icon="angle-right" />
              </button>
            </div>
            <button @click="addUser" class="btn btn-info addbutton shadow-sm">
              <translate>Add User</translate>
            </button>
          </div>
        </div>
      </div>
      <Userdetail :reportToggled="reportToggled" v-if="isUserDetailsVisible" />
    </div>
  </div>
</template>

<style lang="sass" scoped>
.content
  width: 100%

.userdetails
  width: 50%

.main
  height: 100%

.icon
  font-size: large

.userlist
  min-width: 520px
  height: 100%

.userlistsmall
  width: 100%

.userlistextended
  width: 100%
</style>

<script>
/* This is Free Software under GNU Affero General Public License v >= 3.0
 * without warranty, see README.md and license for details.
 *
 * SPDX-License-Identifier: AGPL-3.0-or-later
 * License-Filename: LICENSES/AGPL-3.0.txt
 *
 * Copyright (C) 2018 by via donau
 *   – Österreichische Wasserstraßen-Gesellschaft mbH
 * Software engineering by Intevation GmbH
 *
 * Author(s):
 * Thomas Junk <thomas.junk@intevation.de>
 */
import store from "@/store";
import { mapGetters, mapState } from "vuex";
import { displayError, displayInfo } from "@/lib/errors";
import { HTTP } from "@/lib/http";
import { sortTable } from "@/lib/mixins";

export default {
  name: "userview",
  mixins: [sortTable],
  data() {
    return {
      sortColumn: "user", // overriding the sortTable mixin's empty default value
      reportToggled: false,
      usersForTable: [],
      areSomeUsersHidden: false
    };
  },
  components: {
    Userdetail: () => import("./Userdetail"),
    Spacer: () => import("@/components/Spacer")
  },
  computed: {
    ...mapGetters("usermanagement", ["isUserDetailsVisible", "users"]),
    ...mapState("application", ["showSidebar"]),
    ...mapState("usermanagement", ["currentUser"]),
    usersLabel() {
      return this.$gettext("Users");
    },
    reportsLabel() {
      return this.$gettext("DQL Report");
    },
    receivesReportLabel() {
      return this.$gettext("User receives Data Quality Report");
    },
    sendMailLabel() {
      return this.$gettext("Send testmail");
    },
    deleteUserLabel() {
      return this.$gettext("Delete user");
    },
    roleForColumLabel() {
      return this.$gettext("Role");
    },
    usernameLabel() {
      return this.$gettext("Username");
    },
    countryLabel() {
      return this.$gettext("Country");
    },
    emailLabel() {
      return this.$gettext("Email");
    },
    pages() {
      return Math.ceil(this.usersForTable.length / this.pageSize);
    },
    tableStyle() {
      return {
        table: true,
        "table-hover": true,
        "table-sm": this.isUserDetailsVisible,
        fadeIn: true,
        animated: true
      };
    },
    userlistStyle() {
      return [
        "userlist mr-2",
        {
          userlistsmall: this.isUserDetailsVisible,
          userlistextended: !this.isUserDetailsVisible
        }
      ];
    },
    checkboxObject() {
      // Hide checkbox in case there are no deactivated users
      if (this.users.some(u => !u.active)) {
        return {
          value: this.areSomeUsersHidden,
          label: "Hide inactive users",
          callback: () => {
            this.changeDisplayingState();
          }
        };
      } else {
        return undefined;
      }
    }
  },
  watch: {
    users() {
      this.filterUsers();
    }
  },
  mounted() {
    this.usersForTable = this.users;
  },
  methods: {
    changeDisplayingState() {
      this.areSomeUsersHidden = !this.areSomeUsersHidden;
      this.filterUsers();
    },
    filterUsers() {
      if (this.areSomeUsersHidden) {
        this.usersForTable = this.users.filter(u => u.active);
      } else {
        this.usersForTable = this.users;
      }
    },
    toggleReport(user) {
      HTTP.patch(
        `/users/${user.user}`,
        {
          reports: user.reports
        },
        {
          headers: {
            "X-Gemma-Auth": localStorage.getItem("token"),
            "Content-type": "application/json; charset=UTF-8"
          }
        }
      )
        .then(() => {
          if (this.currentUser && this.currentUser.user === user.user) {
            this.reportToggled = !this.reportToggled;
          }
        })
        .catch(error => {
          let message = "Backend not reachable";
          if (error.response) {
            const { status, data } = error.response;
            message = `${status}: ${data.message || data}`;
          }
          displayError({
            title: this.$gettext("Backend Error"),
            message: message
          });
          user.reports = !user.reports;
        });
    },
    sendTestMail(user) {
      HTTP.get("/testmail/" + encodeURIComponent(user), {
        headers: {
          "X-Gemma-Auth": localStorage.getItem("token"),
          "Content-type": "text/xml; charset=UTF-8"
        }
      })
        .then(() => {
          displayInfo({
            message: this.$gettext("Testmail sent")
          });
        })
        .catch(error => {
          this.loginFailed = true;
          let message = "Backend not reachable";
          if (error.response) {
            const { status, data } = error.response;
            message = `${status}: ${data.message || data}`;
          }
          displayError({
            title: this.$gettext("Backend Error"),
            message: message
          });
        });
    },
    nextPage() {
      if (this.page < this.pages) {
        this.page += 1;
      }
    },
    prevPage() {
      if (this.page > 0) {
        this.page -= 1;
      }
    },
    deleteUser(name) {
      this.$store.commit("application/popup", {
        icon: "trash",
        title: this.$gettext("Delete User"),
        content:
          this.$gettext(
            "Do you really want to delete the following user account:"
          ) +
          `<br>
        <b>${name}</b>`,
        confirm: {
          label: this.$gettext("Delete"),
          icon: "trash",
          callback: () => {
            this.$store
              .dispatch("usermanagement/deleteUser", { name })
              .then(response => {
                displayInfo({
                  message:
                    name +
                    // Exclude whitespaces from the string passed to "gettext" function
                    " " +
                    this.$gettext("user account") +
                    " " +
                    response.data.action +
                    " " +
                    this.$gettext("successfully")
                });
                this.$store
                  .dispatch("usermanagement/loadUsers")
                  .catch(error => {
                    let message = "Backend not reachable";
                    if (error.response) {
                      const { status, data } = error.response;
                      message = `${status}: ${data.message || data}`;
                    }
                    displayError({
                      title: this.$gettext("Backend Error"),
                      message: message
                    });
                  });
              })
              .catch(error => {
                let message = "Backend not reachable";
                if (error.response) {
                  const { status, data } = error.response;
                  message = `${status}: ${data.message || data}`;
                }
                displayError({
                  title: this.$gettext("Backend Error"),
                  message: message
                });
              });
          }
        },
        cancel: {
          label: this.$gettext("Cancel"),
          icon: "times"
        }
      });
    },
    addUser() {
      this.$store.commit("usermanagement/clearCurrentUser");
      this.$store.commit("usermanagement/setUserDetailsVisible");
    },
    selectUser(name) {
      const user = this.$store.getters["usermanagement/getUserByName"](name);
      this.$store.commit("usermanagement/setCurrentUser", user);
    },
    roleIcon(role) {
      if (role === "sys_admin") return "star";
      if (role === "waterway_admin") return ["fab", "adn"];
      return "user";
    },
    roleLabel(role) {
      const labels = {
        sys_admin: this.$gettext("System-Administrator"),
        waterway_admin: this.$gettext("Waterway Admin"),
        waterway_user: this.$gettext("Waterway User")
      };
      return labels[role];
    }
  },
  beforeRouteEnter(to, from, next) {
    store
      .dispatch("usermanagement/loadUsers")
      .then(next)
      .catch(error => {
        let message = "Backend not reachable";
        if (error.response) {
          const { status, data } = error.response;
          message = `${status}: ${data.message || data}`;
        }
        displayError({
          title: this.$gettext("Backend Error"),
          message: message
        });
      });
  },
  beforeRouteLeave(to, from, next) {
    store.commit("usermanagement/clearCurrentUser");
    store.commit("usermanagement/setUserDetailsInvisible");
    next();
  }
};
</script>