view client/src/components/admin/usermanagement/Usermanagement.vue @ 1416:3af7ad9717e2

Client: add a set fo marked translation * add some marked transaltions in the templates. and update the corresponding .po files. * remove trailing whitespace and improve the formating in some files
author Fadi Abbud <fadi.abbud@intevation.de>
date Thu, 29 Nov 2018 11:08:02 +0100
parents ca33ad696594
children ef388dcbb946
line wrap: on
line source

<template>
  <div class="main d-flex flex-row">
    <div :class="spacerStyle"></div>
    <div class="d-flex content flex-column">
      <div class="d-flex flex-row">
        <div :class="userlistStyle">
          <div class="card">
            <div class="card-header shadow-sm text-white bg-info mb-3">
              <translate>Users</translate>
            </div>
            <div class="card-body">
              <table id="datatable" :class="tableStyle">
                <thead>
                  <tr>
                    <th scope="col" @click="sortBy('user')">
                      <span>Username&nbsp;
                        <font-awesome-icon v-if="sortCriterion=='user'" icon="angle-down"></font-awesome-icon>
                      </span>
                    </th>
                    <th scope="col" @click="sortBy('country')">
                      <span>Country&nbsp;
                        <font-awesome-icon v-if="sortCriterion=='country'" icon="angle-down"></font-awesome-icon>
                      </span>
                    </th>
                    <th scope="col" @click="sortBy('email')">
                      <span>Email&nbsp;
                        <font-awesome-icon v-if="sortCriterion=='email'" icon="angle-down"></font-awesome-icon>
                      </span>
                    </th>
                    <th scope="col" @click="sortBy('role')">
                      <span>Role&nbsp;
                        <font-awesome-icon v-if="sortCriterion=='role'" icon="angle-down"></font-awesome-icon>
                      </span>
                    </th>
                    <th scope="col"></th>
                  </tr>
                </thead>
                <tbody>
                  <tr v-for="user in users" :key="user.user" @click="selectUser(user.user)">
                    <td>{{ user.user }}</td>
                    <td>{{ user.country }}</td>
                    <td>{{ user.email}}</td>
                    <td>
                      <font-awesome-icon :icon="roleIcon(user.role)" @click="deleteUser(user.user)"></font-awesome-icon>
                    </td>
                    <td>
                      <font-awesome-icon icon="trash" @click="deleteUser(user.user)"></font-awesome-icon>
                    </td>
                  </tr>
                </tbody>
              </table>
            </div>
            <div class="d-flex mx-auto align-items-center">
              <button
                @click="prevPage"
                v-if="this.currentPage !== 1"
                class="mr-2 btn btn-sm btn-light align-self-center"
              >
                <font-awesome-icon icon="angle-left"></font-awesome-icon>
              </button>
              {{this.currentPage}} / {{this.pages}}
              <button
                @click="nextPage"
                v-if="this.currentPage !== this.pages"
                class="ml-2 btn btn-sm btn-light align-self-center"
              >
                <font-awesome-icon icon="angle-right"></font-awesome-icon>
              </button>
            </div>
            <div class="mr-3 pb-3">
              <button @click="addUser " class="btn btn-info pull-right shadow-sm">
                <translate>Add User</translate>
              </button>
            </div>
          </div>
        </div>
        <Userdetail v-if="isUserDetailsVisible "></Userdetail>
      </div>
    </div>
  </div>
</template>

<style scoped lang="sass">
@import "../../../assets/tooltip.sass"

.spacer
  height: 100vh

.spacer-collapsed
  min-width: $icon-width + $offset
  transition: $transition-fast

@media screen and (min-width: 600px)
  .spacer-expanded
    min-width: $icon-width + $offset

@media screen and (max-width: 1650px)
  .spacer-expanded
    min-width: $sidebar-width + $offset

.main
  height: 100vh

@media screen and (min-width: 600px)
  .content
    margin-left: $sidebar-width
    margin-right: auto

@media screen and (min-width: 1650px)
  .content
    margin-left: $sidebar-width
    margin-right: auto

.icon
  font-size: large

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

.userlistsmall
  width: 30vw

.userlistextended
  width: 70vw

.table
  width: 90% !important
  margin: auto

.table th
  cursor: pointer

.table th,
td
  font-size: $smaller
  border-top: 0px !important
  text-align: left
  padding: $small-offset !important

.table td
  font-size: $smaller
  cursor: pointer

tr span
  display: flex
</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 Userdetail from "./Userdetail";
import store from "../../../store";
import { mapGetters, mapState } from "vuex";
import { displayError } from "../../../lib/errors.js";

export default {
  name: "userview",
  data() {
    return {
      sortCriterion: "user",
      pageSize: 10,
      currentPage: 1
    };
  },
  components: {
    Userdetail
  },
  computed: {
    ...mapGetters("usermanagement", ["isUserDetailsVisible"]),
    ...mapState("application", ["showSidebar", "showUsermenu"]),
    spacerStyle() {
      return [
        "spacer ml-3",
        {
          "spacer-expanded": this.showUsermenu && this.showSidebar,
          "spacer-collapsed": !this.showUsermenu && this.showSidebar
        }
      ];
    },
    users() {
      let users = [...this.$store.getters["usermanagement/users"]];
      users.sort((a, b) => {
        if (
          a[this.sortCriterion].toLowerCase() <
          b[this.sortCriterion].toLowerCase()
        )
          return -1;
        if (
          a[this.sortCriterion].toLowerCase() >
          b[this.sortCriterion].toLowerCase()
        )
          return 1;
        return 0;
      });
      const start = (this.currentPage - 1) * this.pageSize;
      return users.slice(start, start + this.pageSize);
    },
    pages() {
      let users = [...this.$store.getters["usermanagement/users"]];
      return Math.ceil(users.length / this.pageSize);
    },
    tableStyle() {
      return {
        table: true,
        "table-hover": true,
        "table-sm": this.isUserDetailsVisible,
        fadeIn: true,
        animated: true
      };
    },
    userlistStyle() {
      return [
        "userlist mt-3 mr-3 shadow",
        {
          userlistsmall: this.isUserDetailsVisible,
          userlistextended: !this.isUserDetailsVisible
        }
      ];
    }
  },
  methods: {
    tween() {},
    nextPage() {
      if (this.currentPage < this.pages) {
        document.querySelector("#datatable").classList.add("fadeOut");
        setTimeout(() => {
          document.querySelector("#datatable").classList.remove("fadeOut");
          this.currentPage += 1;
        }, 10);
      }
      return;
    },
    prevPage() {
      if (this.currentPage > 0) {
        document.querySelector("#datatable").classList.add("fadeOut");
        setTimeout(() => {
          document.querySelector("#datatable").classList.remove("fadeOut");
          this.currentPage -= 1;
        }, 10);
      }
      return;
    },
    sortBy(criterion) {
      this.sortCriterion = criterion;
    },
    deleteUser(name) {
      this.$store
        .dispatch("usermanagement/deleteUser", { name: name })
        .then(() => {
          this.submitted = false;
          this.$store.dispatch("usermanagement/loadUsers").catch(error => {
            const { status, data } = error.response;
            displayError({
              title: "Backend Error",
              message: `${status}: ${data.message || data}`
            });
          });
        })
        .catch(error => {
          const { status, data } = error.response;
          displayError({
            title: "Backend Error",
            message: `${status}: ${data.message || data}`
          });
        });
    },
    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";
    }
  },
  beforeRouteEnter(to, from, next) {
    store
      .dispatch("usermanagement/loadUsers")
      .then(next)
      .catch(error => {
        const { status, data } = error.response;
        displayError({
          title: "Backend Error",
          message: `${status}: ${data}`
        });
      });
  },
  beforeRouteLeave(to, from, next) {
    store.commit("usermanagement/clearCurrentUser");
    store.commit("usermanagement/setUserDetailsInvisible");
    next();
  }
};
</script>