mirror of
https://github.com/koel/koel
synced 2025-01-02 15:58:46 +00:00
378 lines
12 KiB
Vue
378 lines
12 KiB
Vue
<template>
|
|
<div id="usersWrapper">
|
|
<h1 class="heading">
|
|
<span>Users</span>
|
|
|
|
<div class="buttons">
|
|
<button class="btn-create" @click="creating = !creating">
|
|
<i class="fa fa-plus"></i>
|
|
Add</button>
|
|
</div>
|
|
</h1>
|
|
|
|
<div class="main-scroll-wrap">
|
|
<form class="user-create" v-show="creating" @submit.prevent="store">
|
|
<div class="input-col">
|
|
<label>Name</label>
|
|
<input type="text" v-model="newUser.name" required v-koel-focus="creating">
|
|
</div>
|
|
<div class="input-col">
|
|
<label>Email</label>
|
|
<input type="email" v-model="newUser.email" required>
|
|
</div>
|
|
<div class="input-col">
|
|
<label>Password</label>
|
|
<input type="password" v-model="newUser.password" required>
|
|
</div>
|
|
<div class="btn-col">
|
|
<button>Create</button>
|
|
<button class="cancel" @click.prevent="creating = false">Cancel</button>
|
|
</div>
|
|
</form>
|
|
|
|
<form class="user-edit" v-show="editedUser.id" @submit.prevent="update(editedUser)">
|
|
<div class="input-col">
|
|
<label>Name</label>
|
|
<input type="text" v-model="editedUser.name" required v-koel-focus="editedUser.id">
|
|
</div>
|
|
<div class="input-col">
|
|
<label>Email</label>
|
|
<input type="email" v-model="editedUser.email" required>
|
|
</div>
|
|
<div class="input-col">
|
|
<label>Password</label>
|
|
<input type="password" v-model="editedUser.password" placeholder="Leave blank for no changes">
|
|
</div>
|
|
<div class="btn-col">
|
|
<button>Update</button>
|
|
<button class="cancel" @click.prevent="cancelEdit">Cancel</button>
|
|
</div>
|
|
</form>
|
|
|
|
<div class="users">
|
|
<article v-for="user in state.users" class="user-item" :class="{ editing: editedUser === user }">
|
|
<img :src="user.avatar" width="128" height="128" alt="">
|
|
|
|
<div class="right">
|
|
<div class="info">
|
|
<h1>{{ user.name }}
|
|
<i v-if="user.id === state.current.id" class="you fa fa-check-circle"></i>
|
|
</h1>
|
|
|
|
<p>{{ user.email }}</p>
|
|
</div>
|
|
|
|
<div class="buttons" v-show="editedUser !== user">
|
|
<button class="edit" @click="edit(user)" v-show="deletedUser !== user">
|
|
{{ user.id === state.current.id ? 'Update Profile' : 'Edit' }}
|
|
</button>
|
|
<button v-if="user.id !== state.current.id && deletedUser !== user"
|
|
class="delete"
|
|
@click="confirmDelete(user)">Delete
|
|
</button>
|
|
<span v-show="deletedUser === user">
|
|
<button class="delete" @click="destroy(user)">Confirm</button>
|
|
<button @click="cancelDelete(user)">Cancel</button>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</article>
|
|
|
|
<article class="user-item"></article>
|
|
<article class="user-item"></article>
|
|
<article class="user-item"></article>
|
|
<article class="user-item"></article>
|
|
<article class="user-item"></article>
|
|
<article class="user-item"></article>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import _ from 'lodash';
|
|
|
|
import userStore from '../../../stores/user';
|
|
|
|
export default {
|
|
data() {
|
|
return {
|
|
state: userStore.state,
|
|
creating: false,
|
|
newUser: _.clone(userStore.stub),
|
|
editedUser: _.clone(userStore.stub),
|
|
deletedUser: _.clone(userStore.stub),
|
|
cached: null,
|
|
};
|
|
},
|
|
|
|
methods: {
|
|
/**
|
|
* Show the "Create User" form.
|
|
*/
|
|
create() {
|
|
this.$root.loadMainView('user-create');
|
|
},
|
|
|
|
/**
|
|
* Show the "Edit User" form.
|
|
*/
|
|
edit(user) {
|
|
if (user.id === this.state.current.id) {
|
|
this.$root.loadMainView('profile');
|
|
} else {
|
|
// Keep a cached version of the user for rollback.
|
|
this.cached = _.clone(user);
|
|
this.editedUser = user;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Cancel editing, or simply close the form after updating.
|
|
* @param bool rollback If true, editing was cancelled.
|
|
* The original state of the edited user must be restored.
|
|
* If false, editing was successfully committed.
|
|
*/
|
|
cancelEdit(rollback = true) {
|
|
if (rollback) {
|
|
// Find the original user and roll it back
|
|
_.extend(userStore.byId(this.editedUser.id), this.cached);
|
|
this.cached = null;
|
|
}
|
|
|
|
this.editedUser = _.clone(userStore.stub);
|
|
},
|
|
|
|
/**
|
|
* Store the newly created user.
|
|
*/
|
|
store() {
|
|
userStore.store(this.newUser.name, this.newUser.email, this.newUser.password, () => {
|
|
this.newUser = _.clone(userStore.stub);
|
|
this.creating = false;
|
|
// TODO: Scroll to bottom?
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Update the edited user.
|
|
*/
|
|
update() {
|
|
userStore.update(this.editedUser,
|
|
this.editedUser.name,
|
|
this.editedUser.email,
|
|
this.editedUser.password, () => {
|
|
this.cancelEdit(false);
|
|
// TODO: Scroll to the user?
|
|
}
|
|
);
|
|
},
|
|
|
|
/**
|
|
* Show the controls to really delete a user or cancel deleting.
|
|
*
|
|
* @param object user
|
|
*/
|
|
confirmDelete(user) {
|
|
this.deletedUser = user;
|
|
},
|
|
|
|
/**
|
|
* Cancel deleting. The confirmation will be closed.
|
|
*
|
|
* @param object user
|
|
*/
|
|
cancelDelete(user) {
|
|
this.deletedUser = _.clone(userStore.stub);
|
|
},
|
|
|
|
/**
|
|
* Delete a user.
|
|
*
|
|
* @param object user
|
|
*/
|
|
destroy(user) {
|
|
userStore.destroy(user, () => {
|
|
// Set our data to a stub to avoid any reference errors.
|
|
this.deletedUser = _.clone(userStore.stub);
|
|
});
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<style lang="sass">
|
|
@import "resources/assets/sass/partials/_vars.scss";
|
|
@import "resources/assets/sass/partials/_mixins.scss";
|
|
|
|
@keyframes barberpole {
|
|
from { background-position: 0 0; }
|
|
to { background-position: 60px 30px; }
|
|
}
|
|
|
|
#usersWrapper {
|
|
.users {
|
|
justify-content: space-between;
|
|
flex-wrap: wrap;
|
|
display: flex;
|
|
|
|
.user-item {
|
|
display: flex;
|
|
flex: 0 0 376px;
|
|
margin-bottom: 16px;
|
|
|
|
&.editing {
|
|
background-size: 30px 30px;
|
|
background-image: linear-gradient(
|
|
-45deg,
|
|
rgba(black, 0.3) 25%,
|
|
transparent 25%,
|
|
transparent 50%,
|
|
rgba(black, 0.3) 50%,
|
|
rgba(black, 0.3) 75%,
|
|
transparent 75%,
|
|
transparent
|
|
);
|
|
|
|
animation: barberpole 2s linear infinite;
|
|
}
|
|
|
|
img {
|
|
flex: 0 0 128px;
|
|
}
|
|
|
|
.right {
|
|
flex: 1;
|
|
padding: 16px;
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: space-between;
|
|
background-color: rgba(255, 255, 255, .02);
|
|
}
|
|
|
|
h1 {
|
|
font-size: 140%;
|
|
margin-bottom: 5px;
|
|
|
|
.you {
|
|
font-size: 14px;
|
|
color: $colorHighlight;
|
|
margin-left: 8px;
|
|
}
|
|
}
|
|
|
|
.buttons {
|
|
display: none;
|
|
}
|
|
|
|
&:hover {
|
|
.buttons {
|
|
display: block;
|
|
}
|
|
}
|
|
|
|
button {
|
|
font-size: 12px;
|
|
padding: 6px 14px;
|
|
background: transparent;
|
|
border: 1px solid rgba(255, 255, 255, .1);
|
|
|
|
&.edit:hover {
|
|
background-color: $colorBlue;
|
|
}
|
|
|
|
&.delete:hover {
|
|
background-color: $colorRed;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
.btn-create {
|
|
background: $colorGreen !important;
|
|
|
|
&:hover {
|
|
background: darken($colorGreen, 10%) !important;
|
|
}
|
|
}
|
|
|
|
form.user-create, form.user-edit {
|
|
display: flex;
|
|
align-items: flex-end;
|
|
margin-bottom: 32px;
|
|
padding-bottom: 32px;
|
|
border-bottom: 1px solid rgba(255, 255, 255, .1);
|
|
|
|
.input-col {
|
|
flex: 1;
|
|
padding-right: 8px;
|
|
|
|
input {
|
|
height: 32px;
|
|
width: 100%;
|
|
}
|
|
}
|
|
|
|
button {
|
|
height: 32px;
|
|
padding-top: 6px;
|
|
|
|
&.cancel {
|
|
background: $colorRed;
|
|
}
|
|
}
|
|
}
|
|
|
|
@media only screen
|
|
and (max-device-width : 667px)
|
|
and (orientation : portrait) {
|
|
form.user-create, form.user-edit {
|
|
flex-direction: column;
|
|
align-items: stretch;
|
|
|
|
.input-col {
|
|
padding-right: 0;
|
|
}
|
|
|
|
.input-col, .btn-col {
|
|
margin-top: 12px;
|
|
}
|
|
}
|
|
|
|
.users {
|
|
flex-direction: column;
|
|
|
|
.user-item {
|
|
flex: 1;
|
|
|
|
img {
|
|
display: none;
|
|
}
|
|
|
|
.buttons {
|
|
margin-top: 12px;
|
|
display: block;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@media only screen
|
|
and (min-device-width : 668px)
|
|
and (max-device-width : 768px)
|
|
and (orientation : portrait) {
|
|
.users {
|
|
flex-direction: column;
|
|
|
|
.user-item {
|
|
flex: 1;
|
|
}
|
|
|
|
.buttons {
|
|
margin-top: 12px;
|
|
display: block;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|