123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263 |
- {{ template "page/head_start" .}}
- {{ template "page/head_end" .}}
- {{ template "page/body_start" .}}
- <a-layout id="app" v-cloak :class="themeSwitcher.currentTheme + ' login-app'">
- <transition name="list" appear>
- <a-layout-content class="under min-h-0">
- <div class="waves-header">
- <div class="waves-inner-header"></div>
- <svg class="waves" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"
- viewBox="0 24 150 28" preserveAspectRatio="none" shape-rendering="auto">
- <defs>
- <path id="gentle-wave" d="M-160 44c30 0 58-18 88-18s 58 18 88 18 58-18 88-18 58 18 88 18 v44h-352z" />
- </defs>
- <g class="parallax">
- <use xlink:href="#gentle-wave" x="48" y="0" fill="rgba(0, 135, 113, 0.08)" />
- <use xlink:href="#gentle-wave" x="48" y="3" fill="rgba(0, 135, 113, 0.08)" />
- <use xlink:href="#gentle-wave" x="48" y="5" fill="rgba(0, 135, 113, 0.08)" />
- <use xlink:href="#gentle-wave" x="48" y="7" fill="#c7ebe2" />
- </g>
- </svg>
- </div>
- <a-row type="flex" justify="center" align="middle" class="h-100 overflow-y-auto overflow-x-hidden">
- <a-col :xs="22" :sm="12" :md="10" :lg="8" :xl="6" :xxl="5" id="login" class="my-3rem">
- <template v-if="!loadingStates.fetched">
- <div class="text-center">
- <a-spin size="large" />
- </div>
- </template>
- <template v-else>
- <div class="setting-section">
- <a-popover :overlay-class-name="themeSwitcher.currentTheme" title='{{ i18n "menu.settings" }}'
- placement="bottomRight" trigger="click">
- <template slot="content">
- <a-space direction="vertical" :size="10">
- <a-theme-switch-login></a-theme-switch-login>
- <span>{{ i18n "pages.settings.language" }}</span>
- <a-select ref="selectLang" class="w-100" v-model="lang"
- @change="LanguageManager.setLanguage(lang)" :dropdown-class-name="themeSwitcher.currentTheme">
- <a-select-option :value="l.value" label="English" v-for="l in LanguageManager.supportedLanguages">
- <span role="img" aria-label="l.name" v-text="l.icon"></span>
- <span v-text="l.name"></span>
- </a-select-option>
- </a-select>
- </a-space>
- </template>
- <a-button shape="circle" icon="setting"></a-button>
- </a-popover>
- </div>
- <a-row type="flex" justify="center">
- <a-col :style="{ width: '100%' }">
- <h2 class="title headline zoom">
- <span class="words-wrapper">
- <b class="is-visible">{{ i18n "pages.login.hello" }}</b>
- <b>{{ i18n "pages.login.title" }}</b>
- </span>
- </h2>
- </a-col>
- </a-row>
- <a-row type="flex" justify="center">
- <a-col span="24">
- <a-form @submit.prevent="login">
- <a-space direction="vertical" size="middle">
- <a-form-item>
- <a-input autocomplete="username" name="username" v-model.trim="user.username"
- placeholder='{{ i18n "username" }}' autofocus required>
- <a-icon slot="prefix" type="user" class="fs-1rem"></a-icon>
- </a-input>
- </a-form-item>
- <a-form-item>
- <a-input-password autocomplete="password" name="password" v-model.trim="user.password"
- placeholder='{{ i18n "password" }}' required>
- <a-icon slot="prefix" type="lock" class="fs-1rem"></a-icon>
- </a-input-password>
- </a-form-item>
- <a-form-item v-if="twoFactorEnable">
- <a-input autocomplete="one-time-code" name="twoFactorCode" v-model.trim="user.twoFactorCode"
- placeholder='{{ i18n "twoFactorCode" }}' required>
- <a-icon slot="prefix" type="key" class="fs-1rem"></a-icon>
- </a-input>
- </a-form-item>
- <a-form-item>
- <a-row justify="center" class="centered">
- <div class="wave-btn-bg wave-btn-bg-cl h-50px mt-1rem" :style="loadingStates.spinning ? 'width: 52px' : 'display: inline-block'">
- <a-button class="ant-btn-primary-login" type="primary" :loading="loadingStates.spinning"
- :icon="loadingStates.spinning ? 'poweroff' : undefined" html-type="submit">
- [[ loadingStates.spinning ? '' : '{{ i18n "login" }}' ]]
- </a-button>
- </div>
- </a-row>
- </a-form-item>
- </a-space>
- </a-form>
- </a-col>
- </a-row>
- </template>
- </a-col>
- </a-row>
- </a-layout-content>
- </transition>
- </a-layout>
- {{template "page/body_scripts" .}}
- {{template "component/aThemeSwitch" .}}
- <script>
- const app = new Vue({
- delimiters: ['[[', ']]'],
- el: '#app',
- data: {
- themeSwitcher,
- loadingStates: {
- fetched: false,
- spinning: false
- },
- user: {
- username: "",
- password: "",
- twoFactorCode: ""
- },
- twoFactorEnable: false,
- lang: ""
- },
- async mounted() {
- this.lang = LanguageManager.getLanguage();
- this.twoFactorEnable = await this.getTwoFactorEnable();
- },
- methods: {
- async login() {
- this.loadingStates.spinning = true;
- const msg = await HttpUtil.post('/login', this.user);
- if (msg.success) {
- location.href = basePath + 'panel/';
- }
- this.loadingStates.spinning = false;
- },
- async getTwoFactorEnable() {
- const msg = await HttpUtil.post('/getTwoFactorEnable');
- if (msg.success) {
- this.twoFactorEnable = msg.obj;
- this.loadingStates.fetched = true;
- return msg.obj;
- }
- },
- },
- });
- document.addEventListener("DOMContentLoaded", function () {
- var animationDelay = 2000;
- initHeadline();
- function initHeadline() {
- animateHeadline(document.querySelectorAll('.headline'));
- }
- function animateHeadline(headlines) {
- var duration = animationDelay;
- headlines.forEach(function (headline) {
- setTimeout(function () {
- hideWord(headline.querySelector('.is-visible'));
- }, duration);
- });
- }
- function hideWord(word) {
- var nextWord = takeNext(word);
- switchWord(word, nextWord);
- setTimeout(function () {
- hideWord(nextWord);
- }, animationDelay);
- }
- function takeNext(word) {
- return word.nextElementSibling ? word.nextElementSibling : word.parentElement.firstElementChild;
- }
- function switchWord(oldWord, newWord) {
- oldWord.classList.remove('is-visible');
- oldWord.classList.add('is-hidden');
- newWord.classList.remove('is-hidden');
- newWord.classList.add('is-visible');
- }
- });
- const pm_input_selector = 'input.ant-input, textarea.ant-input';
- const pm_strip_props = [
- 'background',
- 'background-color',
- 'background-image',
- 'color'
- ];
- const pm_observed_forms = new WeakSet();
- function pm_strip_inline(el) {
- if (!el || el.nodeType !== 1 || !el.matches?.(pm_input_selector)) return;
- let did_change = false;
- for (const prop of pm_strip_props) {
- if (el.style.getPropertyValue(prop)) {
- el.style.removeProperty(prop);
- did_change = true;
- }
- }
- if (did_change && el.style.length === 0) {
- el.removeAttribute('style');
- }
- }
- function pm_attach_observer(form) {
- if (pm_observed_forms.has(form)) return;
- pm_observed_forms.add(form);
- form.querySelectorAll(pm_input_selector).forEach(pm_strip_inline);
- const pm_mo = new MutationObserver(mutations => {
- for (const m of mutations) {
- if (m.type === 'attributes') {
- pm_strip_inline(m.target);
- } else if (m.type === 'childList') {
- for (const n of m.addedNodes) {
- if (n.nodeType !== 1) continue;
- if (n.matches?.(pm_input_selector)) pm_strip_inline(n);
- n.querySelectorAll?.(pm_input_selector).forEach(pm_strip_inline);
- }
- }
- }
- });
- pm_mo.observe(form, {
- attributes: true,
- attributeFilter: ['style'],
- childList: true,
- subtree: true
- });
- }
- function pm_init() {
- document.querySelectorAll('form.ant-form').forEach(pm_attach_observer);
- const pm_host = document.getElementById('login') || document.body;
- const pm_wait_for_forms = new MutationObserver(mutations => {
- for (const m of mutations) {
- for (const n of m.addedNodes) {
- if (n.nodeType !== 1) continue;
- if (n.matches?.('form.ant-form')) pm_attach_observer(n);
- n.querySelectorAll?.('form.ant-form').forEach(pm_attach_observer);
- }
- }
- });
- pm_wait_for_forms.observe(pm_host, { childList: true, subtree: true });
- }
- if (document.readyState === 'loading') {
- document.addEventListener('DOMContentLoaded', pm_init, { once: true });
- } else {
- pm_init();
- }
- </script>
- {{ template "page/body_end" .}}
|