mirror of
				https://github.com/MHSanaei/3x-ui.git
				synced 2025-10-26 18:14:50 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			163 lines
		
	
	
		
			No EOL
		
	
	
		
			6 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			163 lines
		
	
	
		
			No EOL
		
	
	
		
			6 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
| <!DOCTYPE html>
 | |
| <html lang="en">
 | |
| {{template "head" .}}
 | |
| <style>
 | |
| 
 | |
|     #app {
 | |
|         padding-top: 100px;
 | |
|     }
 | |
| 
 | |
|     h1 {
 | |
|         text-align: center;
 | |
|         color: #fff;
 | |
|         margin: 20px 0 50px 0;
 | |
|     }
 | |
| 
 | |
|     .ant-btn, .ant-input {
 | |
|         height: 50px;
 | |
|         border-radius: 30px;
 | |
|     }
 | |
| 
 | |
|     .ant-input-group-addon {
 | |
|         border-radius: 0 30px 30px 0;
 | |
|         width: 50px;
 | |
|         font-size: 18px;
 | |
|     }
 | |
| 
 | |
|     .ant-input-affix-wrapper .ant-input-prefix {
 | |
|         left: 23px;
 | |
|     }
 | |
| 
 | |
|     .ant-input-affix-wrapper .ant-input:not(:first-child) {
 | |
|         padding-left: 50px;
 | |
|     }
 | |
| 
 | |
|     .centered {
 | |
|         display: flex;
 | |
|         text-align: center;
 | |
|         align-items: center;
 | |
|         justify-content: center;
 | |
|     }
 | |
| 
 | |
|     .title {
 | |
|         font-size: 32px;
 | |
|         font-weight: bold;
 | |
|     }
 | |
| 
 | |
| </style>
 | |
| <body>
 | |
| <a-layout id="app" v-cloak :class="themeSwitcher.darkClass">
 | |
|     <transition name="list" appear>
 | |
|         <a-layout-content>
 | |
|             <a-row type="flex" justify="center">
 | |
|                 <a-col :xs="22" :sm="20" :md="16" :lg="12" :xl="8">
 | |
|                     <h1 class="title">{{ i18n "pages.login.title" }}</h1>
 | |
|                 </a-col>
 | |
|             </a-row>
 | |
|             <a-row type="flex" justify="center">
 | |
|                 <a-col :xs="22" :sm="20" :md="16" :lg="12" :xl="8">
 | |
|                     <a-form>
 | |
|                         <a-form-item>
 | |
|                             <a-input v-model.trim="user.username" placeholder='{{ i18n "username" }}'
 | |
|                                      @keydown.enter.native="login" autofocus>
 | |
|                                 <a-icon slot="prefix" type="user" :style="'font-size: 16px;' + themeSwitcher.textStyle" />
 | |
|                             </a-input>
 | |
|                         </a-form-item>
 | |
|                         <a-form-item>
 | |
|                             <password-input icon="lock" v-model.trim="user.password"
 | |
|                                             placeholder='{{ i18n "password" }}' @keydown.enter.native="login">
 | |
|                             </password-input>
 | |
|                         </a-form-item>
 | |
|                         <a-form-item v-if="secretEnable">
 | |
|                             <password-input icon="key" v-model.trim="user.loginSecret"
 | |
|                                             placeholder='{{ i18n "secretToken" }}' @keydown.enter.native="login">
 | |
|                             </password-input>
 | |
|                         </a-input>
 | |
|                         </a-form-item>
 | |
|                         <a-form-item>
 | |
|                             <a-row justify="center" class="centered">
 | |
|                                 <a-button type="primary" :loading="loading" @click="login" :icon="loading ? 'poweroff' : undefined"
 | |
|                                           :style="loading ? { width: '50px' } : { display: 'block', width: '100%' }">
 | |
|                                     [[ loading ? '' : '{{ i18n "login" }}' ]]
 | |
|                                 </a-button>
 | |
|                             </a-row>
 | |
|                         </a-form-item>
 | |
|                         <a-form-item>
 | |
|                             <a-row justify="center" class="centered">
 | |
|                                 <a-col :span="12">
 | |
|                                     <a-select ref="selectLang" v-model="lang" @change="setLang(lang)" :dropdown-class-name="themeSwitcher.darkCardClass">
 | |
|                                         <a-select-option :value="l.value" label="English" v-for="l in supportLangs">
 | |
|                                             <span role="img" aria-label="l.name" v-text="l.icon"></span>
 | |
|                                               <span v-text="l.name"></span>
 | |
|                                         </a-select-option>
 | |
|                                     </a-select>
 | |
|                                 </a-col>
 | |
|                             </a-row>
 | |
|                         </a-form-item>
 | |
|                         <a-form-item>
 | |
|                             <a-row justify="center" class="centered">
 | |
|                                 <theme-switch />
 | |
|                             </a-row>
 | |
|                         </a-form-item>
 | |
|                     </a-form>
 | |
|                 </a-col>
 | |
|             </a-row>
 | |
|         </a-layout-content>
 | |
|     </transition>
 | |
| </a-layout>
 | |
| {{template "js" .}}
 | |
| {{template "component/themeSwitcher" .}}
 | |
| {{template "component/password" .}}
 | |
| <script>
 | |
| 
 | |
|     const app = new Vue({
 | |
|         delimiters: ['[[', ']]'],
 | |
|         el: '#app',
 | |
|         data: {
 | |
|             themeSwitcher,
 | |
|             loading: false,
 | |
|             user: new User(),
 | |
|             secretEnable: false,
 | |
|             lang: ""
 | |
|         },
 | |
|         async created() {
 | |
|             this.updateBackground();
 | |
|             this.lang = getLang();
 | |
|             this.secretEnable = await this.getSecretStatus();
 | |
|         },
 | |
|         methods: {
 | |
|             async login() {
 | |
|                 this.loading = true;
 | |
|                 const msg = await HttpUtil.post('/login', this.user);
 | |
|                 this.loading = false;
 | |
|                 if (msg.success) {
 | |
|                     location.href = basePath + 'xui/';
 | |
|                 }
 | |
|             },
 | |
|             async getSecretStatus() {
 | |
|                 this.loading = true;
 | |
|                 const msg = await HttpUtil.post('/getSecretStatus');
 | |
|                 this.loading = false;
 | |
|                 if (msg.success) {
 | |
|                     this.secretEnable = msg.obj;
 | |
|                     return msg.obj;
 | |
|                 }
 | |
|             },
 | |
|             updateBackground() {
 | |
|                 const leftColor = RandomUtil.randomIntRange(0x222222, 0xFFFFFF / 2).toString(16);
 | |
|                 const rightColor = RandomUtil.randomIntRange(0xFFFFFF / 2, 0xDDDDDD).toString(16);
 | |
|                 const deg = RandomUtil.randomIntRange(0, 360);
 | |
|                 const background = `linear-gradient(${deg}deg, #${leftColor} 10%, #${rightColor} 100%)`;
 | |
|                 document.querySelector('#app').style.background = this.themeSwitcher.isDarkTheme ? colors.dark.bg : background;
 | |
|             },
 | |
|         },
 | |
|         watch: {
 | |
|             'themeSwitcher.isDarkTheme'(newVal, oldVal) {
 | |
|                 this.updateBackground();
 | |
|             },
 | |
|         },
 | |
|     });
 | |
| 
 | |
| </script>
 | |
| </body>
 | |
| </html> | 
