iceshrimp-legacy/src/client/app/common/views/components/dialog.vue

289 lines
6.1 KiB
Vue
Raw Normal View History

2018-11-14 08:30:58 +01:00
<template>
2018-11-14 16:01:49 +01:00
<div class="felqjxyj" :class="{ splash }">
2018-11-14 08:30:58 +01:00
<div class="bg" ref="bg" @click="onBgClick"></div>
2019-04-29 08:20:37 +02:00
<div class="main" ref="main" :class="{ round: $store.state.device.roundedCorners }">
2019-02-25 12:08:56 +01:00
<template v-if="type == 'signin'">
<mk-signin/>
</template>
<template v-else>
2019-04-18 12:40:23 +02:00
<div class="icon" v-if="icon">
<fa :icon="icon"/>
</div>
<div class="icon" v-else-if="!input && !select && !user" :class="type">
<fa icon="check" v-if="type === 'success'"/>
<fa :icon="faTimesCircle" v-if="type === 'error'"/>
<fa icon="exclamation-triangle" v-if="type === 'warning'"/>
<fa icon="info-circle" v-if="type === 'info'"/>
<fa :icon="faQuestionCircle" v-if="type === 'question'"/>
<fa icon="spinner" pulse v-if="type === 'waiting'"/>
</div>
2019-02-25 12:08:56 +01:00
<header v-if="title" v-html="title"></header>
2019-05-18 13:36:33 +02:00
<header v-if="title == null && user">{{ $t('@.enter-username') }}</header>
2019-02-25 12:08:56 +01:00
<div class="body" v-if="text" v-html="text"></div>
<ui-input v-if="input" v-model="inputValue" autofocus :type="input.type || 'text'" :placeholder="input.placeholder" @keydown="onInputKeydown"></ui-input>
<ui-input v-if="user" v-model="userInputValue" autofocus @keydown="onInputKeydown"><template #prefix>@</template></ui-input>
2019-02-25 12:18:15 +01:00
<ui-select v-if="select" v-model="selectedValue" autofocus>
<template v-if="select.items">
<option v-for="item in select.items" :value="item.value">{{ item.text }}</option>
</template>
<template v-else>
<optgroup v-for="groupedItem in select.groupedItems" :label="groupedItem.label">
<option v-for="item in groupedItem.items" :value="item.value">{{ item.text }}</option>
</optgroup>
</template>
2019-02-25 12:08:56 +01:00
</ui-select>
2019-04-18 12:40:23 +02:00
<ui-horizon-group no-grow class="buttons fit-bottom" v-if="!splash && (showOkButton || showCancelButton)">
2019-07-04 15:59:52 +02:00
<ui-button @click="ok" v-if="showOkButton" primary :autofocus="!input && !select && !user" :disabled="!canOk">{{ (showCancelButton || input || select || user) ? $t('@.ok') : $t('@.got-it') }}</ui-button>
2019-02-25 12:08:56 +01:00
<ui-button @click="cancel" v-if="showCancelButton || input || select || user">{{ $t('@.cancel') }}</ui-button>
</ui-horizon-group>
</template>
2018-11-14 08:30:58 +01:00
</div>
</div>
</template>
<script lang="ts">
import Vue from 'vue';
2019-01-18 05:06:11 +01:00
import anime from 'animejs';
2018-11-14 08:30:58 +01:00
import { faTimesCircle, faQuestionCircle } from '@fortawesome/free-regular-svg-icons';
2018-12-02 12:10:53 +01:00
import parseAcct from "../../../../../misc/acct/parse";
2019-02-25 12:08:56 +01:00
import i18n from '../../../i18n';
2018-11-14 08:30:58 +01:00
export default Vue.extend({
2019-02-25 12:08:56 +01:00
i18n: i18n(),
2018-11-14 08:30:58 +01:00
props: {
type: {
type: String,
required: false,
default: 'info'
},
title: {
type: String,
required: false
},
text: {
type: String,
2018-11-14 16:01:49 +01:00
required: false
2018-11-14 08:30:58 +01:00
},
2018-12-02 12:10:53 +01:00
input: {
required: false
},
2018-12-02 07:26:56 +01:00
select: {
required: false
},
2018-12-02 12:10:53 +01:00
user: {
required: false
},
2019-04-18 12:40:23 +02:00
icon: {
required: false
},
showOkButton: {
type: Boolean,
default: true
},
2018-11-14 08:30:58 +01:00
showCancelButton: {
type: Boolean,
default: false
},
2019-04-18 12:40:23 +02:00
cancelableByBgClick: {
type: Boolean,
default: true
},
2018-11-14 16:01:49 +01:00
splash: {
2018-11-14 08:30:58 +01:00
type: Boolean,
2018-11-14 16:01:49 +01:00
default: false
2018-11-14 08:30:58 +01:00
}
},
2018-12-02 07:26:56 +01:00
data() {
return {
2018-12-02 12:10:53 +01:00
inputValue: this.input && this.input.default ? this.input.default : null,
userInputValue: null,
selectedValue: this.select ? this.select.default ? this.select.default : this.select.items ? this.select.items[0].value : this.select.groupedItems[0].items[0].value : null,
2019-07-04 15:59:52 +02:00
canOk: true,
2019-04-18 12:40:23 +02:00
faTimesCircle, faQuestionCircle
2018-12-02 07:26:56 +01:00
};
},
2019-07-04 15:59:52 +02:00
watch: {
userInputValue() {
if (this.user) {
this.$root.api('users/show', parseAcct(this.userInputValue)).then(u => {
this.canOk = u != null;
}).catch(() => {
this.canOk = false;
});
}
}
},
2018-11-14 08:30:58 +01:00
mounted() {
2019-07-04 15:59:52 +02:00
if (this.user) this.canOk = false;
2018-11-14 08:30:58 +01:00
this.$nextTick(() => {
(this.$refs.bg as any).style.pointerEvents = 'auto';
anime({
targets: this.$refs.bg,
opacity: 1,
duration: 100,
easing: 'linear'
});
anime({
targets: this.$refs.main,
opacity: 1,
scale: [1.2, 1],
duration: 300,
2019-01-18 05:06:11 +01:00
easing: 'cubicBezier(0, 0.5, 0.5, 1)'
2018-11-14 08:30:58 +01:00
});
2018-11-14 16:01:49 +01:00
if (this.splash) {
setTimeout(() => {
this.close();
}, 1000);
}
2018-11-14 08:30:58 +01:00
});
},
methods: {
2018-12-02 12:10:53 +01:00
async ok() {
2019-07-04 15:59:52 +02:00
if (!this.canOk) return;
2019-04-18 12:40:23 +02:00
if (!this.showOkButton) return;
2018-12-02 12:10:53 +01:00
if (this.user) {
const user = await this.$root.api('users/show', parseAcct(this.userInputValue));
if (user) {
this.$emit('ok', user);
this.close();
}
} else {
const result =
this.input ? this.inputValue :
this.select ? this.selectedValue :
true;
this.$emit('ok', result);
this.close();
}
2018-11-14 08:30:58 +01:00
},
cancel() {
this.$emit('cancel');
this.close();
},
close() {
2019-02-20 21:53:10 +01:00
this.$el.style.pointerEvents = 'none';
2018-11-14 08:30:58 +01:00
(this.$refs.bg as any).style.pointerEvents = 'none';
2019-02-20 21:53:10 +01:00
(this.$refs.main as any).style.pointerEvents = 'none';
2018-11-14 08:30:58 +01:00
anime({
targets: this.$refs.bg,
opacity: 0,
duration: 300,
easing: 'linear'
});
anime({
targets: this.$refs.main,
opacity: 0,
scale: 0.8,
duration: 300,
2019-01-18 05:06:11 +01:00
easing: 'cubicBezier(0, 0.5, 0.5, 1)',
2018-11-14 08:30:58 +01:00
complete: () => this.destroyDom()
});
},
onBgClick() {
2019-04-18 12:40:23 +02:00
if (this.cancelableByBgClick) {
this.cancel();
}
2018-12-02 12:10:53 +01:00
},
onInputKeydown(e) {
if (e.which == 13) { // Enter
e.preventDefault();
e.stopPropagation();
this.ok();
}
2018-11-14 08:30:58 +01:00
}
}
});
</script>
<style lang="stylus" scoped>
.felqjxyj
display flex
align-items center
justify-content center
position fixed
z-index 30000
top 0
left 0
width 100%
height 100%
2018-11-14 16:01:49 +01:00
&.splash
> .main
min-width 0
width initial
2018-11-14 08:30:58 +01:00
> .bg
display block
position fixed
top 0
left 0
width 100%
height 100%
background rgba(#000, 0.7)
opacity 0
pointer-events none
> .main
display block
position fixed
margin auto
2018-11-14 12:23:51 +01:00
padding 32px
2018-11-14 08:30:58 +01:00
min-width 320px
max-width 480px
width calc(100% - 32px)
text-align center
background var(--face)
color var(--faceText)
opacity 0
2019-04-29 08:20:37 +02:00
&.round
border-radius 8px
2018-11-14 08:30:58 +01:00
> .icon
font-size 32px
&.success
color #85da5a
2018-11-14 08:30:58 +01:00
&.error
color #ec4137
&.warning
color #ecb637
> *
display block
margin 0 auto
2018-12-02 07:26:56 +01:00
& + header
margin-top 16px
2018-11-14 12:17:12 +01:00
> header
2018-12-02 07:26:56 +01:00
margin 0 0 8px 0
2018-11-14 08:30:58 +01:00
font-weight bold
2018-11-14 12:17:12 +01:00
font-size 20px
2018-11-14 08:30:58 +01:00
2018-11-14 10:00:23 +01:00
& + .body
margin-top 8px
2018-11-14 08:30:58 +01:00
> .body
2019-04-18 12:40:23 +02:00
margin 16px 0 0 0
2018-11-14 08:30:58 +01:00
2018-11-14 12:17:12 +01:00
> .buttons
margin-top 16px
2018-11-14 08:30:58 +01:00
</style>