add import & export and format code

This commit is contained in:
RainSun 2020-09-02 13:18:23 +08:00
parent 52845ed311
commit f53442594f
17 changed files with 2773 additions and 2380 deletions

View File

@ -1,45 +1,47 @@
<template>
<div id="app">
<icon class="bg" name="canary"></icon>
<router-view />
</div>
<div id="app">
<icon class="bg" name="canary"></icon>
<router-view />
</div>
</template>
<script>
import { mapActions } from 'vuex';
import { mapActions } from "vuex";
export default {
name: 'App',
mounted() {
window.addEventListener('beforeunload', e => this.beforeunloadHandler(e))
window.addEventListener('unload', e => this.unloadHandler(e))
},
destroyed() {
window.removeEventListener('beforeunload', e => this.beforeunloadHandler(e))
window.removeEventListener('unload', e => this.unloadHandler(e))
},
methods: {
...mapActions(['setRowPwd']),
beforeunloadHandler(){
this._beforeUnload_time=new Date().getTime();
},
unloadHandler(e){
this._gap_time=new Date().getTime()-this._beforeUnload_time;
//
if(this._gap_time<=5){
// debugger
console.log('退出程序清除主密码')
this.setRowPwd(['', this]);
// debugger
}
}
}
name: "App",
mounted() {
window.addEventListener("beforeunload", (e) => this.beforeunloadHandler(e));
window.addEventListener("unload", (e) => this.unloadHandler(e));
},
destroyed() {
window.removeEventListener("beforeunload", (e) =>
this.beforeunloadHandler(e)
);
window.removeEventListener("unload", (e) => this.unloadHandler(e));
},
methods: {
...mapActions(["setRowPwd"]),
beforeunloadHandler() {
this._beforeUnload_time = new Date().getTime();
},
unloadHandler(e) {
this._gap_time = new Date().getTime() - this._beforeUnload_time;
//
if (this._gap_time <= 5) {
// debugger
console.log("退出程序清除主密码");
this.setRowPwd(["", this]);
// debugger
}
},
},
};
</script>
<style lang="scss">
@import './style/main';
@import './style/font';
@import '~vue-material/dist/theme/engine';
@import '~vue-material/dist/theme/all';
@import "./style/main";
@import "./style/font";
@import "~vue-material/dist/theme/engine";
@import "~vue-material/dist/theme/all";
body,
h1,
h2,
@ -65,15 +67,15 @@ input,
textarea,
th,
td {
margin: 0;
padding: 0;
margin: 0;
padding: 0;
}
body,
button,
input,
select,
textarea {
font: 12px/1.5tahoma, arial, \5b8b\4f53;
font: 12px/1.5tahoma, arial, \5b8b\4f53;
}
h1,
h2,
@ -81,113 +83,114 @@ h3,
h4,
h5,
h6 {
font-size: 100%;
font-size: 100%;
}
address,
cite,
dfn,
em,
var {
font-style: normal;
font-style: normal;
}
code,
kbd,
pre,
samp {
font-family: couriernew, courier, monospace;
font-family: couriernew, courier, monospace;
}
small {
font-size: 12px;
font-size: 12px;
}
ul,
ol {
list-style: none;
list-style: none;
}
a {
text-decoration: none;
text-decoration: none;
}
a:hover {
text-decoration: underline;
text-decoration: underline;
}
sup {
vertical-align: text-top;
vertical-align: text-top;
}
sub {
vertical-align: text-bottom;
vertical-align: text-bottom;
}
legend {
color: #000;
color: #000;
}
fieldset,
img {
border: 0;
border: 0;
}
button,
input,
select,
textarea {
font-size: 100%;
font-size: 100%;
}
table {
border-collapse: collapse;
border-spacing: 0;
border-collapse: collapse;
border-spacing: 0;
}
button {
margin: 0;
padding: 0;
background: none;
border: none;
outline: none;
margin: 0;
padding: 0;
background: none;
border: none;
outline: none;
}
html[data-theme='dark'] {
@include set-theme-dark();
@import '~vue-material/dist/theme/all';
.md-caption {
color: rgba(255, 255, 255, 0.7);
}
html[data-theme="dark"] {
@include set-theme-dark();
@import "~vue-material/dist/theme/all";
.md-caption {
color: rgba(255, 255, 255, 0.7);
}
}
html[data-theme='light'] {
@include set-theme-light();
@import '~vue-material/dist/theme/all';
html[data-theme="light"] {
@include set-theme-light();
@import "~vue-material/dist/theme/all";
}
#app {
font-family: 'Roboto', 'Helvetica Neue', Helvetica, 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', '微软雅黑', Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
/* text-align: center; */
// color: #2c3e50;
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
-moz-user-select: none;
-webkit-user-select: none;
-ms-user-select: none;
-khtml-user-select: none;
user-select: none;
font-family: "Roboto", "Helvetica Neue", Helvetica, "PingFang SC",
"Hiragino Sans GB", "Microsoft YaHei", "微软雅黑", Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
/* text-align: center; */
// color: #2c3e50;
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
-moz-user-select: none;
-webkit-user-select: none;
-ms-user-select: none;
-khtml-user-select: none;
user-select: none;
}
.bg {
position: absolute;
top: 50%;
left: 50%;
z-index: -100;
height: 5rem;
width: 5rem;
margin-top: -2.5rem;
margin-left: -2.5rem;
position: absolute;
top: 50%;
left: 50%;
z-index: -100;
height: 5rem;
width: 5rem;
margin-top: -2.5rem;
margin-left: -2.5rem;
}
@media screen and (max-width: 500px) {
.md-autocomplete-box-content {
left: 0.2rem !important;
}
.md-autocomplete-box-content {
left: 0.2rem !important;
}
}
#__vconsole {
display: none;
}
.vconsole-show {
display: block !important;
display: block !important;
}
</style>

View File

@ -5,7 +5,7 @@ var CryptoJS = require("crypto-js");
// 更新本地 传参 data => cid pwd
export function syncLocal(data) {
var sign = CryptoJS.MD5(JSON.stringify(data).replace(/\"/g,"'")).toString().toUpperCase();
var sign = CryptoJS.MD5(JSON.stringify(data).replace(/\"/g, "'")).toString().toUpperCase();
data.sign = sign
let params = new URLSearchParams();
params.append('data', JSON.stringify(data));
@ -16,7 +16,7 @@ export function syncLocal(data) {
export function syncCloud(data) {
let data_cache = Object.assign({}, data)
delete data_cache.content
var sign = CryptoJS.MD5(JSON.stringify(data_cache).replace(/\"/g,"'")).toString().toUpperCase();
var sign = CryptoJS.MD5(JSON.stringify(data_cache).replace(/\"/g, "'")).toString().toUpperCase();
data.sign = sign
let params = new URLSearchParams();
params.append('data', JSON.stringify(data));

View File

@ -10,7 +10,7 @@ export const api = axios.create({
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
'Accept': 'application/json',
'Access-Control-Allow-Origin': '*'
'Access-Control-Allow-Origin': '*'
},
timeout: 10 * 1000
})

View File

@ -3,7 +3,7 @@ import App from './App.vue'
import './registerServiceWorker'
import router from './router'
import store from './store'
import { setHtmlFontSize } from './utils/px2rem.js'
import { setHtmlFontSize } from './utils/px2rem.js'
// icon-loader
import Icon from 'vue-svg-icon/Icon.vue'
Vue.component('icon', Icon);
@ -25,7 +25,7 @@ Vue.use(VueClipboard);
// 手势控件
var VueTouch = require('vue-touch')
Vue.use(VueTouch, {name: 'v-touch'})
Vue.use(VueTouch, { name: 'v-touch' })
// vconsole
import Vconsole from 'vconsole';

View File

@ -4,29 +4,29 @@ import { register } from 'register-service-worker'
if (process.env.NODE_ENV === 'production') {
register(`${process.env.BASE_URL}service-worker.js`, {
ready () {
ready() {
console.log(
'App is being served from cache by a service worker.\n' +
'For more details, visit https://goo.gl/AFskqB'
)
},
registered () {
registered() {
console.log('Service worker has been registered.')
},
cached () {
cached() {
console.log('Content has been cached for offline use.')
},
updatefound () {
updatefound() {
console.log('New content is downloading.')
},
updated () {
updated() {
console.log('New content is available; please refresh.')
window.location.reload(true)
},
offline () {
offline() {
console.log('No internet connection found. App is running in offline mode.')
},
error (error) {
error(error) {
console.error('Error during service worker registration:', error)
}
})

View File

@ -4,37 +4,37 @@ import CRC32 from 'crc-32'
// aes加密
export function encrypt(code, json_row) {
// 循环冗余校验
let aesKey = CRC32.str(code)
let aesKey = CRC32.str(code)
// 取八位
aesKey = aesKey.toString().slice(0,8)
aesKey = aesKey.toString().slice(0, 8)
// 字符串化
let json_str = JSON.stringify(json_row)
let json_str = JSON.stringify(json_row)
// 加密
return CryptoJS.AES.encrypt(json_str, CryptoJS.enc.Utf8.parse(aesKey), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
}).toString();
return CryptoJS.AES.encrypt(json_str, CryptoJS.enc.Utf8.parse(aesKey), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
}).toString();
}
// aes解密
export function decrypt(code, encrypt_str) {
// 循环冗余校验
let aesKey = CRC32.str(code)
let aesKey = CRC32.str(code)
// 取八位
aesKey = aesKey.toString().slice(0,8)
aesKey = aesKey.toString().slice(0, 8)
// 解密
return CryptoJS.AES.decrypt(encrypt_str, CryptoJS.enc.Utf8.parse(aesKey), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
}).toString(CryptoJS.enc.Utf8);
return CryptoJS.AES.decrypt(encrypt_str, CryptoJS.enc.Utf8.parse(aesKey), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
}).toString(CryptoJS.enc.Utf8);
}
// aes加密用户密码
export function encryptMainCode(code) {
let default_key = 'e08b44a351a3'
return CryptoJS.AES.encrypt(code, CryptoJS.enc.Utf8.parse(default_key), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
}).toString();
}
@ -42,7 +42,7 @@ export function encryptMainCode(code) {
export function decryptMainCode(row_code) {
let default_key = 'e08b44a351a3'
return CryptoJS.AES.decrypt(row_code, CryptoJS.enc.Utf8.parse(default_key), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.Pkcs7
}).toString(CryptoJS.enc.Utf8);
}

View File

@ -43,9 +43,10 @@ export function lang() {
settings: {
CHS: {
title: '设置',
subheader: ['重置', '选项'],
subheader: ['重置', '选项', '导入 / 导出'],
reset_list: ['账户', '密码本', '应用', '主密码'],
option_list: ['语言', '黑暗模式', '密码超时时间'],
option_list: ['语言', '黑暗模式', '密码超时时间'],
import_export_list: ['导入密码本', '导出密码本'],
expired_time: '无限',
dialog: {
title: '输入新的主密码',
@ -53,12 +54,24 @@ export function lang() {
confirm: '完成',
cancel: '取消'
},
copy: {
successful: '恭喜!已成功导出至剪切板',
failed: '抱歉,导出失败。夸克等浏览器导出成功也会报错,请试着粘贴看看'
},
import_dialog: {
title: '输入导入内容',
placeholder: '不要乱输入东西不然会坏掉',
confirm: '导入',
cancel: '取消'
},
snakebar_msg: {
reset_account: '恭喜! 账户重置完成',
reset_codebook: '恭喜! 密码本重置完成',
reset_settings: '恭喜! 个性化设置成功',
reset_pwd_failed: '密码不能为空',
reset_pwd_successful: '恭喜! 新的密码已经应用'
reset_pwd_successful: '恭喜! 新的密码已经应用',
reset_row_data_failed: '密码本内容不能为空',
reset_row_data_successful: '恭喜! 新的密码本已经应用',
},
reset_dialog: {
title: '警告',
@ -69,22 +82,35 @@ export function lang() {
},
EN: {
title: 'Settings',
subheader: ['Reset', 'Option'],
subheader: ['Reset', 'Option', 'Import / Export'],
reset_list: ['Account', 'Codebook', 'Application', 'Main Password'],
option_list: ['Language', 'Dark Mode', 'Expired Time'],
expired_time: 'Infinite',
expired_time: 'Infinite',
import_export_list: ['Import Codebook', 'Export Codebook'],
dialog: {
title: 'Enter new password',
placeholder: 'Enter here...',
confirm: 'Done',
cancel: 'Cancel'
},
copy: {
successful: 'congratulations! Successfully exported to the clipboard',
failed: 'Failed to export, but failed in some cases. Try to paste'
},
import_dialog: {
title: 'Enter new codebook content',
placeholder: 'Enter here...',
confirm: 'Done',
cancel: 'Cancel'
},
snakebar_msg: {
reset_account: 'Congratulations! Reset account completed',
reset_codebook: 'Congratulations! Reset codebook completed',
reset_settings: 'Congratulations! Reset settings completed',
reset_pwd_failed: 'Password can not be none',
reset_pwd_successful: 'Congratulations! New password set up successfully'
reset_pwd_successful: 'Congratulations! New password set up successfully',
reset_row_data_failed: 'Codebook Content can not be none',
reset_row_data_successful: 'Congratulations! New Codebook Content set up successfully'
},
reset_dialog: {
title: 'Warning',
@ -135,11 +161,11 @@ export function lang() {
generator: {
title: '生成密码',
subheader: ['结果', '设置'],
length_setter:{
length_setter: {
label: '密码长度',
options:['6 位', '12 位', '18 位', '24 位', '32 位']
options: ['6 位', '12 位', '18 位', '24 位', '32 位']
},
settings:['包含大写', '包含小写', '包含数字', '包含符号'],
settings: ['包含大写', '包含小写', '包含数字', '包含符号'],
actions: ['生成', '复制', '关闭'],
code_res_empty: '未生成',
copy_successful: '(复制成功)',
@ -154,12 +180,12 @@ export function lang() {
generator: {
title: 'Generate Password',
subheader: ['Results (click to copy)', 'Settings'],
length_setter:{
length_setter: {
label: 'Password Length',
options:['6 Digit', '12 Digit', '18 Digit', '24 Digit', '32 Digit']
options: ['6 Digit', '12 Digit', '18 Digit', '24 Digit', '32 Digit']
},
settings:['Contain Uppercase', 'Contain Lowercase', 'Contain Number', 'Contain Symbol'],
actions: ['Generate', 'Copy','Close'],
settings: ['Contain Uppercase', 'Contain Lowercase', 'Contain Number', 'Contain Symbol'],
actions: ['Generate', 'Copy', 'Close'],
code_res_empty: 'Not generated',
copy_successful: '(Copy successful)',
copy_failed: '(Failed to copy)',
@ -175,7 +201,7 @@ export function lang() {
submit: '登录'
},
account: {
label: ['云端信息最后修改时间','云端信息最后修改设备名', '同步本地密码本至云端', '同步云端密码本至本地'],
label: ['云端信息最后修改时间', '云端信息最后修改设备名', '同步本地密码本至云端', '同步云端密码本至本地'],
logout: '退出登录'
},
cid_errmsg: ['教务账号不能为空'],
@ -197,7 +223,7 @@ export function lang() {
submit: 'Login'
},
account: {
label: ['Cloud last modified time','Cloud last modified drivce', 'Sync local codebook to cloud', 'Sync cloud codebook to local'],
label: ['Cloud last modified time', 'Cloud last modified drivce', 'Sync local codebook to cloud', 'Sync cloud codebook to local'],
logout: 'Logout'
},
cid_errmsg: ['Mail addr can not be none.'],
@ -214,9 +240,9 @@ export function lang() {
},
unlock: {
CHS: {
title: ['创建主密码','解锁'],
title: ['创建主密码', '解锁'],
subheader: ['创建主密码', '解锁Canary Codebook'],
pwd_label: ['主密码','解锁密码'],
pwd_label: ['主密码', '解锁密码'],
repeat_pwd: '再次输入主密码',
drivce_label: '请输入本设备名(多端同步用)',
enter: '提交',
@ -232,9 +258,9 @@ export function lang() {
drivce_error: '设备名不能为空'
},
EN: {
title: ['Create Password','Unlock'],
title: ['Create Password', 'Unlock'],
subheader: ['Create New Password', 'Unlock Canary Codebook'],
pwd_label: ['New Password','Unlock Password'],
pwd_label: ['New Password', 'Unlock Password'],
repeat_pwd: 'Repeat Password',
drivce_label: 'pleace enter drivce name',
enter: 'Enter',
@ -251,28 +277,28 @@ export function lang() {
}
},
generator: {
CHS:{
CHS: {
title: '生成密码',
subheader: ['结果(点击复制)', '设置'],
length_setter:{
length_setter: {
label: '密码长度',
options:['6 位', '12 位', '18 位', '24 位', '32 位']
options: ['6 位', '12 位', '18 位', '24 位', '32 位']
},
settings:['包含大写', '包含小写', '包含数字', '包含符号'],
settings: ['包含大写', '包含小写', '包含数字', '包含符号'],
submit: '生成',
generate_successful: '生成成功',
copy_successful: '复制成功',
copy_failed: '抱歉,复制失败。夸克等浏览器复制成功也会报错,请试着粘贴看看',
code_res_empty: '未生成'
},
EN:{
EN: {
title: 'Generate Password',
subheader: ['Results (click to copy)', 'Settings'],
length_setter:{
length_setter: {
label: 'Password Length',
options:['6 Digit', '12 Digit', '18 Digit', '24 Digit', '32 Digit']
options: ['6 Digit', '12 Digit', '18 Digit', '24 Digit', '32 Digit']
},
settings:['Contain Uppercase', 'Contain Lowercase', 'Contain Number', 'Contain Symbol'],
settings: ['Contain Uppercase', 'Contain Lowercase', 'Contain Number', 'Contain Symbol'],
submit: 'Generate',
generate_successful: 'Generated successfully',
copy_successful: 'Copy successful',
@ -281,12 +307,20 @@ export function lang() {
}
},
update_log: {
CHS:{
CHS: {
title: '更新日志',
timeline: [
{
label: '新增导入导出功能',
tag: '功能更新',
content: [
'为方便老用户转移数据或者自行备份数据,新增导入导出功能',
'2020-09-02'
]
},
{
label: '修复密码本为空的情况下解锁页面变为创建主密码',
tag:'bug修复',
tag: 'bug修复',
content: [
'密码本为空的情况下,解锁的页面会变成创建主密码,会进行不必要的输入,已修复',
'2020-09-02'
@ -294,7 +328,7 @@ export function lang() {
},
{
label: '长理专版密码本改造完成',
tag:'功能更新',
tag: '功能更新',
content: [
'获取了操作长理网盘的接口文档为了用户信息安全性将用户加密完的密码存放至网盘canary/codebook.txt',
'2020-09-01'
@ -302,7 +336,7 @@ export function lang() {
},
{
label: '修复搜索后关闭搜索框密码本内容消失',
tag:'bug修复',
tag: 'bug修复',
content: [
'进行搜索后关闭搜索框由于执行顺序问题,空缓存覆盖掉了要显示的信息导致密码本“视觉上”的消失,已修复',
'2020-03-24'
@ -310,7 +344,7 @@ export function lang() {
},
{
label: '切换页面之后保存主页状态',
tag:'bug修复',
tag: 'bug修复',
content: [
'正常情况下不保存主页状态进行页面跳转的时候会导致每次回到主页都是默认回顶的状态,已修复',
'2020-03-15'
@ -318,7 +352,7 @@ export function lang() {
},
{
label: '设置中的重置功能优化',
tag:'功能更新',
tag: '功能更新',
content: [
'所有重置操作之前会进行弹窗提醒,防止误操作',
'2020-03-08'
@ -326,7 +360,7 @@ export function lang() {
},
{
label: '设置中的重置主密码功能启用',
tag:'功能更新',
tag: '功能更新',
content: [
'设置中的重置主密码功能启用,重置后即可使用新密码解锁密码本,另改进了解锁页的密码错误判断逻辑',
'2020-03-08'
@ -334,7 +368,7 @@ export function lang() {
},
{
label: '新建密码记录生成的密码支持复制',
tag:'功能更新',
tag: '功能更新',
content: [
'在新建密码页生成密码支持一键复制',
'2020-03-07'
@ -342,7 +376,7 @@ export function lang() {
},
{
label: '修复云端最后修改时间不更新的问题',
tag:'bug修复',
tag: 'bug修复',
content: [
'之前同步密码本至本地后云端最后修改时间不会同时更新,现已修复',
'2020-03-06'
@ -350,7 +384,7 @@ export function lang() {
},
{
label: '新增页面返回手势',
tag:'功能更新',
tag: '功能更新',
content: [
'所有子页面支持右滑返回上一页面',
'2020-03-06'
@ -358,7 +392,7 @@ export function lang() {
},
{
label: '新建密码记录时支持生成密码',
tag:'功能更新',
tag: '功能更新',
content: [
'新建密码记录页的的密码输入框右侧支持打开生成密码页',
'2020-03-05'
@ -366,7 +400,7 @@ export function lang() {
},
{
label: '修复黑暗模式下主页分割线显示问题',
tag:'bug修复',
tag: 'bug修复',
content: [
'之前黑暗模式下主页密码分割线过亮,造成强烈的视觉撕裂感,现已修复',
'2020-03-04'
@ -374,7 +408,7 @@ export function lang() {
},
{
label: '新增控制台vConsole',
tag:'功能更新',
tag: '功能更新',
content: [
'主页呼出菜单栏点击头像十次即可唤出或隐藏控制台',
'2020-03-04'
@ -382,7 +416,7 @@ export function lang() {
},
{
label: '扩大一键回顶点击面积',
tag:'功能更新',
tag: '功能更新',
content: [
'一键回顶点击范围从标题文字延展到整个导航栏非功能区',
'2020-03-03'
@ -390,7 +424,7 @@ export function lang() {
},
{
label: '新增更新日志页',
tag:'功能更新',
tag: '功能更新',
content: [
'主页呼出菜单栏即可发现功能入口',
'2020-03-03'
@ -398,7 +432,7 @@ export function lang() {
},
{
label: '新增一键回顶',
tag:'功能更新',
tag: '功能更新',
content: [
'在主页点击导航栏页面标题即可一键回顶',
'2020-03-02'
@ -406,7 +440,7 @@ export function lang() {
},
{
label: '新增密码生成页',
tag:'功能更新',
tag: '功能更新',
content: [
'主页呼出菜单栏即可发现功能入口',
'2020-02-29'
@ -414,7 +448,7 @@ export function lang() {
},
{
label: '新增呼出菜单手势',
tag:'功能更新',
tag: '功能更新',
content: [
'在主页非导航栏任意处右滑即可呼出菜单栏',
'2020-02-26'
@ -422,12 +456,20 @@ export function lang() {
}
]
},
EN:{
EN: {
title: 'Update Log',
timeline: [
{
label: 'New import and export function',
tag: 'Feature update',
content: [
'In order to facilitate old users to transfer data or backup data by themselves, new import and export function is added',
'2020-09-02'
]
},
{
label: 'Fix that the unlock page becomes create master password when the password book is empty',
tag:'Bug fix',
tag: 'Bug fix',
content: [
'When the password book is empty, the unlocked page will become the creation of the master password, and unnecessary input will be made, which is fixed',
'2020-09-02'
@ -435,7 +477,7 @@ export function lang() {
},
{
label: 'Changli special edition codebook transformation completed',
tag:'Feature update',
tag: 'Feature update',
content: [
'Obtained the interface document for operating the Changli network disk, and stored the encrypted password of the user in the network disk for the security of user information canary/codebook.txt',
'2020-09-01'
@ -443,7 +485,7 @@ export function lang() {
},
{
label: 'Fixed search box password content disappeared after searching',
tag:'Bug fix',
tag: 'Bug fix',
content: [
'Close the search box after searching. Due to the execution order problem, the empty cache has overwritten the information to be displayed, causing the codebook to "visually" disappear. It has been fixed.',
'2020-03-24'
@ -451,7 +493,7 @@ export function lang() {
},
{
label: 'Save homepage status after switching pages',
tag:'Bug fix',
tag: 'Bug fix',
content: [
'Normally, when you do nt save the homepage state, the page jump will result in the default backing state every time you return to the homepage.',
'2020-03-15'
@ -459,7 +501,7 @@ export function lang() {
},
{
label: 'Optimization of reset function in settings',
tag:'Feature update',
tag: 'Feature update',
content: [
'Pop-up window reminder before all reset operations to prevent misoperation',
'2020-03-08'
@ -467,7 +509,7 @@ export function lang() {
},
{
label: 'The reset master password function is enabled in the settings',
tag:'Feature update',
tag: 'Feature update',
content: [
'The reset master password function is enabled in the settings. After resetting, you can use the new password to unlock the password book, and improve the password error judgment logic on the unlock page',
'2020-03-08'
@ -475,7 +517,7 @@ export function lang() {
},
{
label: 'Generated passwords support replication',
tag:'Feature update',
tag: 'Feature update',
content: [
'Generate a password on the new password page',
'2020-03-07'
@ -483,7 +525,7 @@ export function lang() {
},
{
label: 'Fix the problem that the last modification time of the cloud is not updated',
tag:'Bug fix',
tag: 'Bug fix',
content: [
'The last modified time in the cloud will not be updated at the same time after syncing the password book to the local, it is now fixed',
'2020-03-06'
@ -491,7 +533,7 @@ export function lang() {
},
{
label: 'Add page back gesture',
tag:'Feature update',
tag: 'Feature update',
content: [
'All subpages support swiping right to return to the previous page',
'2020-03-06'
@ -499,7 +541,7 @@ export function lang() {
},
{
label: 'Support password generation when creating new password records',
tag:'Feature update',
tag: 'Feature update',
content: [
'The right side of the password input box of the new password record page supports opening a generate password page',
'2020-03-05'
@ -507,7 +549,7 @@ export function lang() {
},
{
label: 'Fixed display of homepage split line in dark mode',
tag:'Bug fix',
tag: 'Bug fix',
content: [
'The homepage password dividing line was too bright in the dark mode, causing a strong visual tearing.',
'2020-03-04'
@ -515,7 +557,7 @@ export function lang() {
},
{
label: 'Add vConsole',
tag:'Feature update',
tag: 'Feature update',
content: [
'Home callout menu bar Click the avatar ten times to call up or hide the console',
'2020-03-04'
@ -523,7 +565,7 @@ export function lang() {
},
{
label: 'Expand one click back to top click area',
tag:'Feature update',
tag: 'Feature update',
content: [
'The one-click back to top click extends from the title text to the non-functional area of the entire navigation bar',
'2020-03-03'
@ -531,7 +573,7 @@ export function lang() {
},
{
label: 'New update log page',
tag:'Feature update',
tag: 'Feature update',
content: [
'Call out the menu bar on the homepage to find the function entry',
'2020-03-03'
@ -539,7 +581,7 @@ export function lang() {
},
{
label: 'Added one-click back to top',
tag:'Feature update',
tag: 'Feature update',
content: [
'Click the navigation page title on the homepage to return to the top with one click',
'2020-03-02'
@ -547,7 +589,7 @@ export function lang() {
},
{
label: 'New password generation page',
tag:'Feature update',
tag: 'Feature update',
content: [
'Call out the menu bar on the homepage to find the function entry',
'2020-02-29'
@ -555,7 +597,7 @@ export function lang() {
},
{
label: 'Added callout menu gesture',
tag:'Feature update',
tag: 'Feature update',
content: [
'Swipe right anywhere on the homepage other than the navigation bar to bring up the menu bar。',
'2020-02-26'
@ -570,10 +612,10 @@ export function lang() {
/**
* 引入
import { lang } from '@/utils/language.js'
* data 里边加入lang
data -> lang
* create() 初始化语言系统防止报错
this.lang = lang().home.CHS
console.log('临时语言系统加载完成')

View File

@ -1,431 +1,505 @@
<template>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()"><md-icon>arrow_back</md-icon></md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()">
<md-icon>arrow_back</md-icon>
</md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<icon class="logo" name="canary"></icon>
<template v-if="page_type == 'login'">
<md-field :class="cid_verify ? '' : 'md-invalid'">
<label>{{ lang.login.input_placeholder[0] }}</label>
<md-input v-model="cid" required></md-input>
<span class="md-error">{{ cid_errmsg }}</span>
</md-field>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<icon class="logo" name="canary"></icon>
<template v-if="page_type == 'login'">
<md-field :class="cid_verify ? '' : 'md-invalid'">
<label>{{ lang.login.input_placeholder[0] }}</label>
<md-input v-model="cid" required></md-input>
<span class="md-error">{{ cid_errmsg }}</span>
</md-field>
<md-field :class="password_verify ? '' : 'md-invalid'">
<label>{{ lang.login.input_placeholder[1] }}</label>
<md-input v-model="password" required type="password"></md-input>
<span class="md-error">{{ lang.login.password_errmsg }}</span>
</md-field>
<md-button class="md-raised md-primary expand" @click="judgeLogin()" :disabled="login_loading">
{{ lang.login.submit }}
<div class="loading-box"><md-progress-spinner v-if="login_loading" :md-diameter="22" :md-stroke="3" md-mode="indeterminate"></md-progress-spinner></div>
</md-button>
</template>
<md-field :class="password_verify ? '' : 'md-invalid'">
<label>{{ lang.login.input_placeholder[1] }}</label>
<md-input v-model="password" required type="password"></md-input>
<span class="md-error">{{ lang.login.password_errmsg }}</span>
</md-field>
<md-button
class="md-raised md-primary expand"
@click="judgeLogin()"
:disabled="login_loading"
>
{{ lang.login.submit }}
<div class="loading-box">
<md-progress-spinner
v-if="login_loading"
:md-diameter="22"
:md-stroke="3"
md-mode="indeterminate"
></md-progress-spinner>
</div>
</md-button>
</template>
<template v-if="page_type == 'account'">
<md-list>
<md-list-item>
<span class="md-list-item-text">{{ user_infos.cid }} | {{ user_infos.user_name }}</span>
</md-list-item>
<template v-if="page_type == 'account'">
<md-list>
<md-list-item>
<span class="md-list-item-text">{{ user_infos.cid }} | {{ user_infos.user_name }}</span>
</md-list-item>
<md-divider></md-divider>
<md-divider></md-divider>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[0] }}</span>
<span class="time-content">{{ update_time }}</span>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[0] }}</span>
<span class="time-content">{{ update_time }}</span>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[1] }}</span>
<span class="time-content">{{ user_infos.cloud_drivce }}</span>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[1] }}</span>
<span class="time-content">{{ user_infos.cloud_drivce }}</span>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[2] }}</span>
<md-button @click="syncCloudStart()" v-if="!sync_cloud_loading" class="md-icon-button md-list-action"><md-icon>cloud_upload</md-icon></md-button>
<md-progress-spinner v-else :md-diameter="22" :md-stroke="3" md-mode="indeterminate"></md-progress-spinner>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[2] }}</span>
<md-button
@click="syncCloudStart()"
v-if="!sync_cloud_loading"
class="md-icon-button md-list-action"
>
<md-icon>cloud_upload</md-icon>
</md-button>
<md-progress-spinner
v-else
:md-diameter="22"
:md-stroke="3"
md-mode="indeterminate"
></md-progress-spinner>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[3] }}</span>
<md-button @click="syncLocalStart()" v-if="!sync_local_loading" class="md-icon-button md-list-action"><md-icon>cloud_download</md-icon></md-button>
<md-progress-spinner v-else :md-diameter="22" :md-stroke="3" md-mode="indeterminate"></md-progress-spinner>
</md-list-item>
<md-list-item>
<span class="md-list-item-text">{{ lang.account.label[3] }}</span>
<md-button
@click="syncLocalStart()"
v-if="!sync_local_loading"
class="md-icon-button md-list-action"
>
<md-icon>cloud_download</md-icon>
</md-button>
<md-progress-spinner
v-else
:md-diameter="22"
:md-stroke="3"
md-mode="indeterminate"
></md-progress-spinner>
</md-list-item>
<md-button @click="startLogout()" class="md-raised md-primary expand" :disabled="sync_local_loading || sync_cloud_loading">{{ lang.account.logout }}</md-button>
</md-list>
</template>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
<md-button
@click="startLogout()"
class="md-raised md-primary expand"
:disabled="sync_local_loading || sync_cloud_loading"
>{{ lang.account.logout }}</md-button>
</md-list>
</template>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
</template>
<script>
// @ is an alias to /src
import { mapState, mapActions } from 'vuex';
import { encrypt, decrypt, encryptMainCode, decryptMainCode } from '@/utils/aes.js';
import { syncLocal, syncCloud } from '@/axios/api.js';
import { lang } from '@/utils/language.js';
import { setHtmlFontSize } from '@/utils/px2rem.js';
import { mapState, mapActions } from "vuex";
import {
encrypt,
decrypt,
encryptMainCode,
decryptMainCode,
} from "@/utils/aes.js";
import { syncLocal, syncCloud } from "@/axios/api.js";
import { lang } from "@/utils/language.js";
import { setHtmlFontSize } from "@/utils/px2rem.js";
export default {
name: 'Add',
data() {
return {
page_type: 'login', // login account
cid: '',
cid_verify: true,
cid_errmsg: '',
password: '',
password_verify: true,
sync_local_loading: false,
sync_cloud_loading: false,
login_loading: false,
show_snackbar: false,
snakebar_msg: '',
lang: '',
clientHeight: ''
};
},
computed: {
...mapState(['user_infos', 'row_data', 'row_pwd', 'settings']),
update_time: function() {
return this.formatDateTime(new Date(parseInt(this.user_infos.update_time)));
}
},
methods: {
...mapActions(['setUserInfo', 'setRowData', 'setRowPwd']),
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName('md-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.style.minHeight = clientHeight - 32 + 'px';
window.document.documentElement.setAttribute('data-theme', this.settings.is_dark_mode ? 'dark' : 'light');
},
//
init() {
// vuex
this.$store.replaceState(Object.assign(this.$store.state, JSON.parse(localStorage.getItem('storeState'))));
this.initLanguage();
//
if(this.user_infos && this.user_infos.cid && this.user_infos.row_login_pwd) this.page_type = 'account'
else this.page_type = 'login'
console.log('当前用户状态为' + this.page_type);
},
name: "Add",
data() {
return {
page_type: "login", // login account
cid: "",
cid_verify: true,
cid_errmsg: "",
password: "",
password_verify: true,
sync_local_loading: false,
sync_cloud_loading: false,
login_loading: false,
show_snackbar: false,
snakebar_msg: "",
lang: "",
clientHeight: "",
};
},
computed: {
...mapState(["user_infos", "row_data", "row_pwd", "settings"]),
update_time: function () {
return this.formatDateTime(
new Date(parseInt(this.user_infos.update_time))
);
},
},
methods: {
...mapActions(["setUserInfo", "setRowData", "setRowPwd"]),
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName('md-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight =
clientHeight + "px";
this.$refs.list_placeholder.parentNode.style.minHeight =
clientHeight - 32 + "px";
window.document.documentElement.setAttribute(
"data-theme",
this.settings.is_dark_mode ? "dark" : "light"
);
},
//
init() {
// vuex
this.$store.replaceState(
Object.assign(
this.$store.state,
JSON.parse(localStorage.getItem("storeState"))
)
);
this.initLanguage();
//
if (
this.user_infos &&
this.user_infos.cid &&
this.user_infos.row_login_pwd
)
this.page_type = "account";
else this.page_type = "login";
console.log("当前用户状态为" + this.page_type);
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().account.CHS;
} else {
this.lang = lang().account.EN;
}
console.log('语言配置完成');
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().account.CHS;
} else {
this.lang = lang().account.EN;
}
console.log("语言配置完成");
},
//
back() {
this.$router.go(-1);
},
//
back() {
this.$router.go(-1);
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace('/');
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace("/");
},
//
judgeLogin() {
// flag
let can_continue = true;
//
judgeLogin() {
// flag
let can_continue = true;
// mail_addr
if (!(this.cid = this.cid.trim())) {
can_continue = false;
this.cid_verify = false;
this.cid_errmsg = this.lang.cid_errmsg[0];
}
// password
if (!(this.password = this.password.trim())) {
can_continue = false;
this.password_verify = false;
}
// flag
if (can_continue) this.startLogin();
},
// mail_addr
if (!(this.cid = this.cid.trim())) {
can_continue = false;
this.cid_verify = false;
this.cid_errmsg = this.lang.cid_errmsg[0];
}
// password
if (!(this.password = this.password.trim())) {
can_continue = false;
this.password_verify = false;
}
// flag
if (can_continue) this.startLogin();
},
//
startLogin() {
this.login_loading = true;
let data = {
cid: this.cid,
pwd: this.password,
};
syncLocal(data)
.then(res => {
// console.log(res.data)
this.manageLoginRes(res.data);
})
.catch(err => {
console.log(err);
this.login_loading = false;
if(!err.response){
this.snakebar_msg = this.lang.snakebar_msg[0];
} else if(err.response.status == 512) {
this.snakebar_msg = this.lang.snakebar_msg[1];
} else if(err.response.status == 510) {
this.snakebar_msg = this.lang.snakebar_msg[2];
} else {
this.snakebar_msg = this.lang.snakebar_msg[3] + err.response.status;
}
this.show_snackbar = true;
});
},
//
startLogin() {
this.login_loading = true;
let data = {
cid: this.cid,
pwd: this.password,
};
syncLocal(data)
.then((res) => {
// console.log(res.data)
this.manageLoginRes(res.data);
})
.catch((err) => {
console.log(err);
this.login_loading = false;
if (!err.response) {
this.snakebar_msg = this.lang.snakebar_msg[0];
} else if (err.response.status == 512) {
this.snakebar_msg = this.lang.snakebar_msg[1];
} else if (err.response.status == 510) {
this.snakebar_msg = this.lang.snakebar_msg[2];
} else {
this.snakebar_msg = this.lang.snakebar_msg[3] + err.response.status;
}
this.show_snackbar = true;
});
},
//
manageLoginRes(data) {
//
let user_infos = {
...this.user_infos,
cid: this.cid,
row_login_pwd: encryptMainCode(this.password),
user_name: data.username,
update_time: data.modified,
cloud_drivce: this.settings.is_chinese ? '暂无' : 'unknown'
};
this.setUserInfo([user_infos, this]);
console.log('正常用户登录,用户信息覆写完成');
this.snakebar_msg = this.lang.snakebar_msg[4];
this.show_snackbar = true;
this.page_type = 'account';
this.login_loading = false;
},
//
manageLoginRes(data) {
//
let user_infos = {
...this.user_infos,
cid: this.cid,
row_login_pwd: encryptMainCode(this.password),
user_name: data.username,
update_time: data.modified,
cloud_drivce: this.settings.is_chinese ? "暂无" : "unknown",
};
this.setUserInfo([user_infos, this]);
console.log("正常用户登录,用户信息覆写完成");
this.snakebar_msg = this.lang.snakebar_msg[4];
this.show_snackbar = true;
this.page_type = "account";
this.login_loading = false;
},
//
formatDateTime(inputTime) {
var date = new Date(inputTime);
var y = date.getFullYear();
var m = date.getMonth() + 1;
m = m < 10 ? '0' + m : m;
var d = date.getDate();
d = d < 10 ? '0' + d : d;
var h = date.getHours();
h = h < 10 ? '0' + h : h;
var minute = date.getMinutes();
var second = date.getSeconds();
minute = minute < 10 ? '0' + minute : minute;
second = second < 10 ? '0' + second : second;
return y + '-' + m + '-' + d + ' ' + h + ':' + minute;
// return y + "-" + m + "-" + d + " " + h + ":" + minute + ":" + second;
// return y + "-" + m + "-" + d;
},
//
formatDateTime(inputTime) {
var date = new Date(inputTime);
var y = date.getFullYear();
var m = date.getMonth() + 1;
m = m < 10 ? "0" + m : m;
var d = date.getDate();
d = d < 10 ? "0" + d : d;
var h = date.getHours();
h = h < 10 ? "0" + h : h;
var minute = date.getMinutes();
var second = date.getSeconds();
minute = minute < 10 ? "0" + minute : minute;
second = second < 10 ? "0" + second : second;
return y + "-" + m + "-" + d + " " + h + ":" + minute;
// return y + "-" + m + "-" + d + " " + h + ":" + minute + ":" + second;
// return y + "-" + m + "-" + d;
},
//
syncLocalStart() {
this.sync_local_loading = true;
let data = {
cid: this.user_infos.cid,
pwd: decryptMainCode(this.user_infos.row_login_pwd)
};
syncLocal(data)
.then(res => {
let data = res.data
//
let content = ''
try{
content = JSON.parse(data.content)
if(typeof content == 'string') throw new Error()
if(!content || !content.codebook){
//
content = {
codebook: '',
drivce: content.drivce ? content.drivce : (this.settings.is_chinese ? '暂无' : 'unknown')
}
}
if(content.drivce && content.drivce != '暂无' && content.drivce != 'unknown') content.drivce = decodeURIComponent(content.drivce)
else content.drivce = this.settings.is_chinese ? '暂无' : 'unknown'
this.setRowData([content.codebook, this]);
console.log('数据获取成功,密码本覆写成功');
let user_infos = this.user_infos;
user_infos.update_time = data.modified - 1000 * 60 * 4;
user_infos.user_name = data.username;
user_infos.cloud_drivce = content.drivce
this.setUserInfo([user_infos, this]);
console.log('数据获取成功,用户信息覆写成功');
this.snakebar_msg = this.lang.snakebar_msg[6];
this.show_snackbar = true;
} catch(e) {
console.log(e)
console.log('数据获取成功,内容有误')
this.snakebar_msg = this.lang.snakebar_msg[5];
this.show_snackbar = true;
} finally {
setTimeout(
function() {
this.sync_local_loading = false;
}.bind(this),
1000
);
}
})
.catch(err => {
console.log(err);
this.sync_local_loading = false;
if(!err.response){
this.snakebar_msg = this.lang.snakebar_msg[0];
} else if(err.response.status == 512) {
this.snakebar_msg = this.lang.snakebar_msg[1];
} else if(err.response.status == 510) {
this.snakebar_msg = this.lang.snakebar_msg[2];
} else {
this.snakebar_msg = this.lang.snakebar_msg[3] + err.response.status;
}
this.show_snackbar = true;
});
},
//
syncLocalStart() {
this.sync_local_loading = true;
let data = {
cid: this.user_infos.cid,
pwd: decryptMainCode(this.user_infos.row_login_pwd),
};
syncLocal(data)
.then((res) => {
let data = res.data;
//
let content = "";
try {
content = JSON.parse(data.content);
if (typeof content == "string") throw new Error();
if (!content || !content.codebook) {
//
content = {
codebook: "",
drivce: content.drivce
? content.drivce
: this.settings.is_chinese
? "暂无"
: "unknown",
};
}
if (
content.drivce &&
content.drivce != "暂无" &&
content.drivce != "unknown"
)
content.drivce = decodeURIComponent(content.drivce);
else content.drivce = this.settings.is_chinese ? "暂无" : "unknown";
this.setRowData([content.codebook, this]);
console.log("数据获取成功,密码本覆写成功");
let user_infos = this.user_infos;
user_infos.update_time = data.modified - 1000 * 60 * 4;
user_infos.user_name = data.username;
user_infos.cloud_drivce = content.drivce;
this.setUserInfo([user_infos, this]);
console.log("数据获取成功,用户信息覆写成功");
this.snakebar_msg = this.lang.snakebar_msg[6];
this.show_snackbar = true;
} catch (e) {
console.log(e);
console.log("数据获取成功,内容有误");
this.snakebar_msg = this.lang.snakebar_msg[5];
this.show_snackbar = true;
} finally {
setTimeout(
function () {
this.sync_local_loading = false;
}.bind(this),
1000
);
}
})
.catch((err) => {
console.log(err);
this.sync_local_loading = false;
if (!err.response) {
this.snakebar_msg = this.lang.snakebar_msg[0];
} else if (err.response.status == 512) {
this.snakebar_msg = this.lang.snakebar_msg[1];
} else if (err.response.status == 510) {
this.snakebar_msg = this.lang.snakebar_msg[2];
} else {
this.snakebar_msg = this.lang.snakebar_msg[3] + err.response.status;
}
this.show_snackbar = true;
});
},
//
syncCloudStart() {
this.sync_cloud_loading = true;
let content = {
codebook: this.row_data,
drivce: encodeURIComponent(this.user_infos.drivce)
}
let data = {
cid: this.user_infos.cid,
pwd: decryptMainCode(this.user_infos.row_login_pwd),
content: JSON.stringify(content)
};
console.log(data)
syncCloud(data)
.then(res => {
let user_infos = this.user_infos;
user_infos.update_time = new Date().getTime();
user_infos.cloud_drivce = user_infos.drivce ? user_infos.drivce : (this.settings.is_chinese ? '暂无' : 'unknown')
this.setUserInfo([user_infos, this]);
console.log('数据上传成功,用户信息覆写成功');
this.snakebar_msg = this.lang.snakebar_msg[7];
this.show_snackbar = true;
setTimeout(
function() {
this.sync_cloud_loading = false;
}.bind(this),
1000
);
})
.catch(err => {
console.log(err);
this.sync_cloud_loading = false;
if(!err.response){
this.snakebar_msg = this.lang.snakebar_msg[0];
} else if(err.response.status == 512) {
this.snakebar_msg = this.lang.snakebar_msg[1];
} else if(err.response.status == 510) {
this.snakebar_msg = this.lang.snakebar_msg[2];
} else {
this.snakebar_msg = this.lang.snakebar_msg[3] + err.response.status;
}
this.show_snackbar = true;
});
},
//
syncCloudStart() {
this.sync_cloud_loading = true;
let content = {
codebook: this.row_data,
drivce: encodeURIComponent(this.user_infos.drivce),
};
let data = {
cid: this.user_infos.cid,
pwd: decryptMainCode(this.user_infos.row_login_pwd),
content: JSON.stringify(content),
};
console.log(data);
syncCloud(data)
.then((res) => {
let user_infos = this.user_infos;
user_infos.update_time = new Date().getTime();
user_infos.cloud_drivce = user_infos.drivce
? user_infos.drivce
: this.settings.is_chinese
? "暂无"
: "unknown";
this.setUserInfo([user_infos, this]);
console.log("数据上传成功,用户信息覆写成功");
this.snakebar_msg = this.lang.snakebar_msg[7];
this.show_snackbar = true;
setTimeout(
function () {
this.sync_cloud_loading = false;
}.bind(this),
1000
);
})
.catch((err) => {
console.log(err);
this.sync_cloud_loading = false;
if (!err.response) {
this.snakebar_msg = this.lang.snakebar_msg[0];
} else if (err.response.status == 512) {
this.snakebar_msg = this.lang.snakebar_msg[1];
} else if (err.response.status == 510) {
this.snakebar_msg = this.lang.snakebar_msg[2];
} else {
this.snakebar_msg = this.lang.snakebar_msg[3] + err.response.status;
}
this.show_snackbar = true;
});
},
// 退
startLogout() {
//
let user_infos = {
...this.user_infos,
user_name: 'A Little Canary',
cid: 'Codebook',
row_login_pwd: '',
cloud_drivce: this.settings.is_chinese ? '暂无' : 'unknown',
update_time: new Date().getTime()
};
this.setUserInfo([user_infos, this]);
console.log('用户信息覆写完成');
this.back();
}
},
created() {
this.lang = lang().account.CHS;
console.log('临时语言系统加载完成');
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function() {
this.changeFixed(this.clientHeight);
},
// `cid`
cid: function() {
this.cid_verify = true;
},
// `password`
password: function() {
this.password_verify = true;
},
// `activation_code`
activation_code: function() {
this.activation_code_verify = true;
}
},
beforeDestroy() {},
components: {}
// 退
startLogout() {
//
let user_infos = {
...this.user_infos,
user_name: "A Little Canary",
cid: "Codebook",
row_login_pwd: "",
cloud_drivce: this.settings.is_chinese ? "暂无" : "unknown",
update_time: new Date().getTime(),
};
this.setUserInfo([user_infos, this]);
console.log("用户信息覆写完成");
this.back();
},
},
created() {
this.lang = lang().account.CHS;
console.log("临时语言系统加载完成");
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function () {
this.changeFixed(this.clientHeight);
},
// `cid`
cid: function () {
this.cid_verify = true;
},
// `password`
password: function () {
this.password_verify = true;
},
// `activation_code`
activation_code: function () {
this.activation_code_verify = true;
},
},
beforeDestroy() {},
components: {},
};
</script>
<style scoped lang="scss" type="text/scss">
@import '../../style/main';
@import "../../style/main";
.account {
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.logo {
margin: 0 auto;
display: block;
height: 5rem;
width: 5rem;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.logo {
margin: 0 auto;
display: block;
height: 5rem;
width: 5rem;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
}
</style>

View File

@ -115,7 +115,7 @@ import {
encrypt,
decrypt,
encryptMainCode,
decryptMainCode
decryptMainCode,
} from "@/utils/aes.js";
import { lang } from "@/utils/language.js";
import { setHtmlFontSize } from "@/utils/px2rem.js";
@ -148,11 +148,11 @@ export default {
//
code_length: 12,
//
copy_state: ""
copy_state: "",
};
},
computed: {
...mapState(["row_data", "row_pwd", "settings"])
...mapState(["row_data", "row_pwd", "settings"]),
},
methods: {
...mapActions(["setRowData", "setRowPwd"]),
@ -260,7 +260,7 @@ export default {
user_name: this.user_name,
password: this.password,
node: this.node,
web_address: this.web_address
web_address: this.web_address,
};
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
@ -292,7 +292,7 @@ export default {
user_name: this.user_name,
password: this.password,
node: this.node,
web_address: this.web_address
web_address: this.web_address,
};
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
@ -363,7 +363,7 @@ export default {
setTimeout(() => {
this.copy_state = "";
}, 2000);
}
},
},
created() {
this.lang = lang().add.CHS;
@ -381,12 +381,12 @@ export default {
},
watch: {
// `clientHeight`
clientHeight: function() {
clientHeight: function () {
this.changeFixed(this.clientHeight);
}
},
},
beforeDestroy() {},
components: {}
components: {},
};
</script>

View File

@ -1,313 +1,359 @@
<template>
<div class="detail">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()"><md-icon>arrow_back</md-icon></md-button>
<h3 class="md-title" style="flex: 1">{{ content.title }}</h3>
</div>
<div class="md-toolbar-section-end">
<md-button class="md-icon-button" @click="turnToEdit()"><md-icon>edit</md-icon></md-button>
<md-button class="md-icon-button" @click="openDelDialog()"><md-icon>delete</md-icon></md-button>
</div>
</md-app-toolbar>
<div class="detail">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()">
<md-icon>arrow_back</md-icon>
</md-button>
<h3 class="md-title" style="flex: 1">{{ content.title }}</h3>
</div>
<div class="md-toolbar-section-end">
<md-button class="md-icon-button" @click="turnToEdit()">
<md-icon>edit</md-icon>
</md-button>
<md-button class="md-icon-button" @click="openDelDialog()">
<md-icon>delete</md-icon>
</md-button>
</div>
</md-app-toolbar>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-list class="md-double-line">
<md-list-item>
<label class="md-caption">{{ lang.label[0] }}</label>
<div class="msg-box">
<p>{{ content.user_name }}</p>
<md-button class="md-icon-button" v-clipboard:copy="content.user_name" v-clipboard:success="onCopyUrl" v-clipboard:error="onErrorUrl">
<icon class="icon" name="file_copy" :class="settings.is_dark_mode ? 'dark-theme-icon' : ''"></icon>
</md-button>
</div>
</md-list-item>
<md-list-item>
<label class="md-caption">{{ lang.label[1] }}</label>
<div class="msg-box">
<p class="password">{{ show_password ? content.password : doitPassword }}</p>
<md-button class="md-icon-button" @click="show_password = !show_password"><md-icon>remove_red_eye</md-icon></md-button>
<md-button class="md-icon-button" v-clipboard:copy="content.password" v-clipboard:success="onCopyUrl" v-clipboard:error="onErrorUrl">
<icon class="icon" name="file_copy" :class="settings.is_dark_mode ? 'dark-theme-icon' : ''"></icon>
</md-button>
</div>
</md-list-item>
<md-list-item>
<label class="md-caption">{{ lang.label[2] }}</label>
<div class="msg-box">
<p>{{ content.web_address ? content.web_address : lang.empty_placeholder }}</p>
<md-button class="md-icon-button" @click="openUrl()"><md-icon>open_in_new</md-icon></md-button>
</div>
</md-list-item>
<md-list-item>
<label class="md-caption">{{ lang.label[3] }}</label>
<div class="msg-box">
<p class="node">{{ content.node ? content.node : lang.empty_placeholder }}</p>
</div>
</md-list-item>
</md-list>
<md-dialog-confirm
:md-active.sync="show_dialog"
:md-title="lang.dialog.title"
:md-content="lang.dialog.content"
:md-confirm-text="lang.dialog.confirm"
:md-cancel-text="lang.dialog.cancel"
@md-cancel="onCancel"
@md-confirm="delCode"
/>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-list class="md-double-line">
<md-list-item>
<label class="md-caption">{{ lang.label[0] }}</label>
<div class="msg-box">
<p>{{ content.user_name }}</p>
<md-button
class="md-icon-button"
v-clipboard:copy="content.user_name"
v-clipboard:success="onCopyUrl"
v-clipboard:error="onErrorUrl"
>
<icon
class="icon"
name="file_copy"
:class="settings.is_dark_mode ? 'dark-theme-icon' : ''"
></icon>
</md-button>
</div>
</md-list-item>
<md-list-item>
<label class="md-caption">{{ lang.label[1] }}</label>
<div class="msg-box">
<p class="password">{{ show_password ? content.password : doitPassword }}</p>
<md-button class="md-icon-button" @click="show_password = !show_password">
<md-icon>remove_red_eye</md-icon>
</md-button>
<md-button
class="md-icon-button"
v-clipboard:copy="content.password"
v-clipboard:success="onCopyUrl"
v-clipboard:error="onErrorUrl"
>
<icon
class="icon"
name="file_copy"
:class="settings.is_dark_mode ? 'dark-theme-icon' : ''"
></icon>
</md-button>
</div>
</md-list-item>
<md-list-item>
<label class="md-caption">{{ lang.label[2] }}</label>
<div class="msg-box">
<p>{{ content.web_address ? content.web_address : lang.empty_placeholder }}</p>
<md-button class="md-icon-button" @click="openUrl()">
<md-icon>open_in_new</md-icon>
</md-button>
</div>
</md-list-item>
<md-list-item>
<label class="md-caption">{{ lang.label[3] }}</label>
<div class="msg-box">
<p class="node">{{ content.node ? content.node : lang.empty_placeholder }}</p>
</div>
</md-list-item>
</md-list>
<md-dialog-confirm
:md-active.sync="show_dialog"
:md-title="lang.dialog.title"
:md-content="lang.dialog.content"
:md-confirm-text="lang.dialog.confirm"
:md-cancel-text="lang.dialog.cancel"
@md-cancel="onCancel"
@md-confirm="delCode"
/>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
</template>
<script>
// @ is an alias to /src
import { mapState, mapActions } from 'vuex';
import { encrypt, decrypt, encryptMainCode, decryptMainCode } from '@/utils/aes.js';
import { lang } from '@/utils/language.js';
import { setHtmlFontSize } from '@/utils/px2rem.js';
import { mapState, mapActions } from "vuex";
import {
encrypt,
decrypt,
encryptMainCode,
decryptMainCode,
} from "@/utils/aes.js";
import { lang } from "@/utils/language.js";
import { setHtmlFontSize } from "@/utils/px2rem.js";
export default {
name: 'Add',
data() {
return {
show_snackbar: false,
snakebar_msg: '',
content: {},
show_password: false,
show_dialog: false,
lang: '',
clientHeight: ''
};
},
computed: {
...mapState(['row_data', 'row_pwd', 'settings']),
doitPassword() {
if (this.content.password) {
let doit = '';
for (let i = 0; i < this.content.password.length; i++) {
doit += '*';
}
return doit;
}
}
},
methods: {
...mapActions(['setRowData', 'setRowPwd']),
// md-app
changeFixed(clientHeight) {
//
// window.document.getElementsByClassName('md-app-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.style.minHeight = clientHeight - 32 + 'px';
window.document.documentElement.setAttribute('data-theme', this.settings.is_dark_mode ? 'dark' : 'light');
},
//
init() {
// vuex
this.$store.replaceState(Object.assign(this.$store.state, JSON.parse(localStorage.getItem('storeState'))));
//
if (this.$route.params.code_content) {
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.initLanguage();
this.content = this.$route.params.code_content;
this.addCount();
} else {
//
this.turnToHome('密码超时');
}
} else {
//
this.turnToHome('无密码');
}
} else {
//
this.turnToHome('无参跳转');
}
},
name: "Add",
data() {
return {
show_snackbar: false,
snakebar_msg: "",
content: {},
show_password: false,
show_dialog: false,
lang: "",
clientHeight: "",
};
},
computed: {
...mapState(["row_data", "row_pwd", "settings"]),
doitPassword() {
if (this.content.password) {
let doit = "";
for (let i = 0; i < this.content.password.length; i++) {
doit += "*";
}
return doit;
}
},
},
methods: {
...mapActions(["setRowData", "setRowPwd"]),
// md-app
changeFixed(clientHeight) {
//
// window.document.getElementsByClassName('md-app-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight =
clientHeight + "px";
this.$refs.list_placeholder.parentNode.style.minHeight =
clientHeight - 32 + "px";
window.document.documentElement.setAttribute(
"data-theme",
this.settings.is_dark_mode ? "dark" : "light"
);
},
//
init() {
// vuex
this.$store.replaceState(
Object.assign(
this.$store.state,
JSON.parse(localStorage.getItem("storeState"))
)
);
//
if (this.$route.params.code_content) {
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.initLanguage();
this.content = this.$route.params.code_content;
this.addCount();
} else {
//
this.turnToHome("密码超时");
}
} else {
//
this.turnToHome("无密码");
}
} else {
//
this.turnToHome("无参跳转");
}
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().detail.CHS;
} else {
this.lang = lang().detail.EN;
}
console.log('语言配置完成');
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().detail.CHS;
} else {
this.lang = lang().detail.EN;
}
console.log("语言配置完成");
},
//
back() {
this.$router.go(-1);
},
//
back() {
this.$router.go(-1);
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace('/');
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace("/");
},
//
turnToEdit() {
this.$router.push({ name: 'Add', params: { modify_content: this.content } });
},
//
turnToEdit() {
this.$router.push({
name: "Add",
params: { modify_content: this.content },
});
},
//
onCopyUrl(e) {
this.snakebar_msg = this.lang.copy.successful;
this.show_snackbar = true;
},
//
onCopyUrl(e) {
this.snakebar_msg = this.lang.copy.successful;
this.show_snackbar = true;
},
//
onErrorUrl(e) {
this.snakebar_msg = this.lang.copy.failed;
this.show_snackbar = true;
},
//
onErrorUrl(e) {
this.snakebar_msg = this.lang.copy.failed;
this.show_snackbar = true;
},
//
openUrl() {
if (this.content.web_address) {
window.open(this.content.web_address);
} else {
this.snakebar_msg = this.lang.snakebar_msg_empty;
this.show_snackbar = true;
}
},
//
addCount() {
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
//
let data_decrypt = decrypt(main_code_decrpt, this.row_data);
// json
let data_list = JSON.parse(data_decrypt);
//
for (let i in data_list) {
//
if (data_list[i].id == this.content.id) {
//
data_list[i].open_count++;
}
}
let data_list_aes = encrypt(main_code_decrpt, data_list);
this.setRowData([data_list_aes, this]);
console.log('新加密信息覆写成功,增加点击完成');
},
//
openUrl() {
if (this.content.web_address) {
window.open(this.content.web_address);
} else {
this.snakebar_msg = this.lang.snakebar_msg_empty;
this.show_snackbar = true;
}
},
//
addCount() {
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
//
let data_decrypt = decrypt(main_code_decrpt, this.row_data);
// json
let data_list = JSON.parse(data_decrypt);
//
for (let i in data_list) {
//
if (data_list[i].id == this.content.id) {
//
data_list[i].open_count++;
}
}
let data_list_aes = encrypt(main_code_decrpt, data_list);
this.setRowData([data_list_aes, this]);
console.log("新加密信息覆写成功,增加点击完成");
},
//
onCancel() {},
//
onCancel() {},
//
openDelDialog() {
this.show_dialog = true;
},
//
delCode() {
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
//
let data_decrypt = decrypt(main_code_decrpt, this.row_data);
// json
let data_list = JSON.parse(data_decrypt);
//
let data_cache = [];
//
for (let i in data_list) {
//
if (data_list[i].id != this.content.id) {
//
data_cache.push(data_list[i]);
}
}
let data_list_aes = encrypt(main_code_decrpt, data_cache);
this.setRowData([data_list_aes, this]);
console.log('新加密信息覆写成功,密码删除完成');
this.back();
}
},
created() {
this.lang = lang().detail.CHS;
console.log('临时语言系统加载完成');
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function() {
this.changeFixed(this.clientHeight);
}
},
beforeDestroy() {},
components: {}
//
openDelDialog() {
this.show_dialog = true;
},
//
delCode() {
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
//
let data_decrypt = decrypt(main_code_decrpt, this.row_data);
// json
let data_list = JSON.parse(data_decrypt);
//
let data_cache = [];
//
for (let i in data_list) {
//
if (data_list[i].id != this.content.id) {
//
data_cache.push(data_list[i]);
}
}
let data_list_aes = encrypt(main_code_decrpt, data_cache);
this.setRowData([data_list_aes, this]);
console.log("新加密信息覆写成功,密码删除完成");
this.back();
},
},
created() {
this.lang = lang().detail.CHS;
console.log("临时语言系统加载完成");
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function () {
this.changeFixed(this.clientHeight);
},
},
beforeDestroy() {},
components: {},
};
</script>
<style scoped lang="scss" type="text/scss">
@import '../../style/main';
@import "../../style/main";
.detail {
width: 100%;
min-height: 100%;
background: #fff;
.md-toolbar {
position: fixed !important;
}
.md-list-item-content {
position: relative;
label {
position: absolute;
top: 0.2rem;
left: 16px;
}
.msg-box {
margin-top: 0.5rem;
display: flex;
align-items: center;
justify-content: space-between;
width: 100%;
border-bottom: 1px #eee solid;
p {
word-wrap: break-word;
word-break: normal;
white-space: normal;
width: 80%;
margin: 0.2rem 0;
}
.icon {
height: 24px;
width: 24px;
color: gray;
}
.dark-theme-icon {
color: #fff !important;
}
.node {
width: 100%;
}
.password {
width: 70%;
}
}
}
width: 100%;
min-height: 100%;
background: #fff;
.md-toolbar {
position: fixed !important;
}
.md-list-item-content {
position: relative;
label {
position: absolute;
top: 0.2rem;
left: 16px;
}
.msg-box {
margin-top: 0.5rem;
display: flex;
align-items: center;
justify-content: space-between;
width: 100%;
border-bottom: 1px #eee solid;
p {
word-wrap: break-word;
word-break: normal;
white-space: normal;
width: 80%;
margin: 0.2rem 0;
}
.icon {
height: 24px;
width: 24px;
color: gray;
}
.dark-theme-icon {
color: #fff !important;
}
.node {
width: 100%;
}
.password {
width: 70%;
}
}
}
}
</style>

View File

@ -86,7 +86,7 @@
<script>
// @ is an alias to /src
import { mapState, mapActions } from "vuex";
import { setHtmlFontSize } from '@/utils/px2rem.js'
import { setHtmlFontSize } from "@/utils/px2rem.js";
export default {
name: "Home",
@ -97,35 +97,38 @@ export default {
clientHeight: "",
test_data: [
{
open_count:0,
title:'QQ',
user_name:'1144131090',
password:'test',
node:'测试用',
},{
open_count:0,
title:'微信',
user_name:'15143211127',
password:'test',
node:'测试用',
},{
open_count:0,
title:'QQ',
user_name:'1144131090',
password:'test',
node:'测试用',
},{
open_count:0,
title:'QQ',
user_name:'1144131090',
password:'test',
node:'测试用',
open_count: 0,
title: "QQ",
user_name: "1144131090",
password: "test",
node: "测试用",
},
]
{
open_count: 0,
title: "微信",
user_name: "15143211127",
password: "test",
node: "测试用",
},
{
open_count: 0,
title: "QQ",
user_name: "1144131090",
password: "test",
node: "测试用",
},
{
open_count: 0,
title: "QQ",
user_name: "1144131090",
password: "test",
node: "测试用",
},
],
};
},
computed: {
...mapState([])
...mapState([]),
},
methods: {
...mapActions([]),
@ -133,7 +136,7 @@ export default {
//
// console.log(clientHeight);
this.$refs.home.children[0].style.minHeight = clientHeight + "px";
}
},
},
created() {},
mounted() {
@ -147,12 +150,12 @@ export default {
},
watch: {
// `clientHeight`
clientHeight: function() {
clientHeight: function () {
this.changeFixed(this.clientHeight);
}
},
},
beforeDestroy() {},
components: {}
components: {},
};
</script>
@ -178,7 +181,7 @@ export default {
}
}
.code-card {
padding: .3rem 0;
padding: 0.3rem 0;
border-bottom: 1px #eee solid;
}
}

View File

@ -1,227 +1,252 @@
<template>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()"><md-icon>arrow_back</md-icon></md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()">
<md-icon>arrow_back</md-icon>
</md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-list>
<!-- 生成结果 -->
<md-subheader class="md-primary">{{ lang.subheader[0] }}</md-subheader>
<md-list-item>
<p v-clipboard:copy="code_res" v-clipboard:success="onCopyCode" v-clipboard:error="onErrorCode">{{ code_res ? code_res : lang.code_res_empty }}</p>
</md-list-item>
<!-- settings -->
<md-subheader class="md-primary">{{ lang.subheader[1] }}</md-subheader>
<!-- 长度设置 -->
<md-list-item>
<md-field>
<label for="expired_time">{{ lang.length_setter.label }}</label>
<md-select v-model="code_length" name="code_length" id="code_length">
<md-option value="6">{{ lang.length_setter.options[0] }}</md-option>
<md-option value="12">{{ lang.length_setter.options[1] }}</md-option>
<md-option value="18">{{ lang.length_setter.options[2] }}</md-option>
<md-option value="24">{{ lang.length_setter.options[3] }}</md-option>
<md-option value="32">{{ lang.length_setter.options[4] }}</md-option>
</md-select>
</md-field>
</md-list-item>
<!-- 大写字母 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[0] }}</span>
<md-switch v-model="have_upper" class="md-primary"></md-switch>
</md-list-item>
<!-- 小写字母 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[1] }}</span>
<md-switch v-model="have_lower" class="md-primary"></md-switch>
</md-list-item>
<!-- 数字 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[2] }}</span>
<md-switch v-model="have_num" class="md-primary"></md-switch>
</md-list-item>
<!-- 符号 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[3] }}</span>
<md-switch v-model="have_symbol" class="md-primary"></md-switch>
</md-list-item>
<md-button class="md-raised md-primary expand" @click="generateStart()">{{ lang.submit }}</md-button>
</md-list>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-list>
<!-- 生成结果 -->
<md-subheader class="md-primary">{{ lang.subheader[0] }}</md-subheader>
<md-list-item>
<p
v-clipboard:copy="code_res"
v-clipboard:success="onCopyCode"
v-clipboard:error="onErrorCode"
>{{ code_res ? code_res : lang.code_res_empty }}</p>
</md-list-item>
<!-- settings -->
<md-subheader class="md-primary">{{ lang.subheader[1] }}</md-subheader>
<!-- 长度设置 -->
<md-list-item>
<md-field>
<label for="expired_time">{{ lang.length_setter.label }}</label>
<md-select v-model="code_length" name="code_length" id="code_length">
<md-option value="6">{{ lang.length_setter.options[0] }}</md-option>
<md-option value="12">{{ lang.length_setter.options[1] }}</md-option>
<md-option value="18">{{ lang.length_setter.options[2] }}</md-option>
<md-option value="24">{{ lang.length_setter.options[3] }}</md-option>
<md-option value="32">{{ lang.length_setter.options[4] }}</md-option>
</md-select>
</md-field>
</md-list-item>
<!-- 大写字母 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[0] }}</span>
<md-switch v-model="have_upper" class="md-primary"></md-switch>
</md-list-item>
<!-- 小写字母 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[1] }}</span>
<md-switch v-model="have_lower" class="md-primary"></md-switch>
</md-list-item>
<!-- 数字 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[2] }}</span>
<md-switch v-model="have_num" class="md-primary"></md-switch>
</md-list-item>
<!-- 符号 -->
<md-list-item>
<span class="md-list-item-text">{{ lang.settings[3] }}</span>
<md-switch v-model="have_symbol" class="md-primary"></md-switch>
</md-list-item>
<md-button
class="md-raised md-primary expand"
@click="generateStart()"
>{{ lang.submit }}</md-button>
</md-list>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
</template>
<script>
// @ is an alias to /src
import { mapState, mapActions } from 'vuex';
import { lang } from '@/utils/language.js';
import { generatePassword } from '@/utils/generator.js';
import { setHtmlFontSize } from '@/utils/px2rem.js';
import { mapState, mapActions } from "vuex";
import { lang } from "@/utils/language.js";
import { generatePassword } from "@/utils/generator.js";
import { setHtmlFontSize } from "@/utils/px2rem.js";
export default {
name: 'Add',
data() {
return {
// snackbar
show_snackbar: false,
snakebar_msg: '',
//
lang: '',
//
clientHeight: '',
//
have_upper: true,
//
have_lower: true,
//
have_num: true,
//
have_symbol: false,
//
code_length: 12,
//
code_res: ''
};
},
computed: {
...mapState(['user_infos', 'row_data', 'row_pwd', 'settings'])
},
methods: {
...mapActions(['setUserInfo', 'setRowData', 'setRowPwd', 'setSettings']),
name: "Add",
data() {
return {
// snackbar
show_snackbar: false,
snakebar_msg: "",
//
lang: "",
//
clientHeight: "",
//
have_upper: true,
//
have_lower: true,
//
have_num: true,
//
have_symbol: false,
//
code_length: 12,
//
code_res: "",
};
},
computed: {
...mapState(["user_infos", "row_data", "row_pwd", "settings"]),
},
methods: {
...mapActions(["setUserInfo", "setRowData", "setRowPwd", "setSettings"]),
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName("md-content")[0].style.minHeight =
// clientHeight + "px";
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.style.minHeight = clientHeight - 32 + 'px';
window.document.documentElement.setAttribute('data-theme', this.settings.is_dark_mode ? 'dark' : 'light');
},
//
init() {
// vuex
this.$store.replaceState(Object.assign(this.$store.state, JSON.parse(localStorage.getItem('storeState'))));
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.initLanguage();
} else {
//
this.turnToHome('密码超时');
}
} else {
//
this.turnToHome('无密码');
}
},
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName("md-content")[0].style.minHeight =
// clientHeight + "px";
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight =
clientHeight + "px";
this.$refs.list_placeholder.parentNode.style.minHeight =
clientHeight - 32 + "px";
window.document.documentElement.setAttribute(
"data-theme",
this.settings.is_dark_mode ? "dark" : "light"
);
},
//
init() {
// vuex
this.$store.replaceState(
Object.assign(
this.$store.state,
JSON.parse(localStorage.getItem("storeState"))
)
);
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.initLanguage();
} else {
//
this.turnToHome("密码超时");
}
} else {
//
this.turnToHome("无密码");
}
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().generator.CHS;
} else {
this.lang = lang().generator.EN;
}
console.log('语言配置完成');
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().generator.CHS;
} else {
this.lang = lang().generator.EN;
}
console.log("语言配置完成");
},
//
back() {
this.$router.go(-1);
},
//
back() {
this.$router.go(-1);
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace('/');
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace("/");
},
//
generateStart() {
this.code_res = generatePassword(this.code_length, this.have_lower, this.have_upper, this.have_num, this.have_symbol);
this.snakebar_msg = this.lang.generate_successful;
this.show_snackbar = true;
},
//
generateStart() {
this.code_res = generatePassword(
this.code_length,
this.have_lower,
this.have_upper,
this.have_num,
this.have_symbol
);
this.snakebar_msg = this.lang.generate_successful;
this.show_snackbar = true;
},
//
onCopyCode() {
this.snakebar_msg = this.lang.copy_successful;
this.show_snackbar = true;
},
//
onCopyCode() {
this.snakebar_msg = this.lang.copy_successful;
this.show_snackbar = true;
},
//
onErrorCode() {
this.snakebar_msg = this.lang.copy_failed;
this.show_snackbar = true;
}
},
created() {
this.lang = lang().generator.CHS;
console.log('临时语言系统加载完成');
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function() {
this.changeFixed(this.clientHeight);
}
},
beforeDestroy() {},
components: {}
//
onErrorCode() {
this.snakebar_msg = this.lang.copy_failed;
this.show_snackbar = true;
},
},
created() {
this.lang = lang().generator.CHS;
console.log("临时语言系统加载完成");
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function () {
this.changeFixed(this.clientHeight);
},
},
beforeDestroy() {},
components: {},
};
</script>
<style scoped lang="scss" type="text/scss">
@import '../../style/main';
@import "../../style/main";
.account {
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
}
</style>

File diff suppressed because it is too large Load Diff

View File

@ -1,422 +1,516 @@
<template>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()"><md-icon>arrow_back</md-icon></md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()">
<md-icon>arrow_back</md-icon>
</md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-list>
<md-subheader class="md-primary">{{ lang.subheader[0] }}</md-subheader>
<md-list-item @click="openResetDialog('account')">
<md-icon>person</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[0] }}</span>
<md-progress-spinner v-if="account_loading" :md-diameter="22" :md-stroke="3" md-mode="indeterminate"></md-progress-spinner>
</md-list-item>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-list>
<md-subheader class="md-primary">{{ lang.subheader[0] }}</md-subheader>
<md-list-item @click="openResetDialog('account')">
<md-icon>person</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[0] }}</span>
<md-progress-spinner
v-if="account_loading"
:md-diameter="22"
:md-stroke="3"
md-mode="indeterminate"
></md-progress-spinner>
</md-list-item>
<md-list-item @click="openResetDialog('codebook')">
<md-icon>format_align_left</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[1] }}</span>
<md-progress-spinner v-if="codebook_loading" :md-diameter="22" :md-stroke="3" md-mode="indeterminate"></md-progress-spinner>
</md-list-item>
<md-list-item @click="openResetDialog('codebook')">
<md-icon>format_align_left</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[1] }}</span>
<md-progress-spinner
v-if="codebook_loading"
:md-diameter="22"
:md-stroke="3"
md-mode="indeterminate"
></md-progress-spinner>
</md-list-item>
<md-list-item @click="openResetDialog('application')">
<md-icon>layers</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[2] }}</span>
<md-progress-spinner v-if="application_loading" :md-diameter="22" :md-stroke="3" md-mode="indeterminate"></md-progress-spinner>
</md-list-item>
<md-list-item @click="openResetDialog('application')">
<md-icon>layers</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[2] }}</span>
<md-progress-spinner
v-if="application_loading"
:md-diameter="22"
:md-stroke="3"
md-mode="indeterminate"
></md-progress-spinner>
</md-list-item>
<md-list-item @click="open_dialog">
<md-icon>vpn_key</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[3] }}</span>
</md-list-item>
<md-list-item @click="open_dialog">
<md-icon>vpn_key</md-icon>
<span class="md-list-item-text">{{ lang.reset_list[3] }}</span>
</md-list-item>
<md-divider></md-divider>
<md-divider></md-divider>
<md-subheader class="md-primary">{{ lang.subheader[1] }}</md-subheader>
<md-list-item>
<md-icon>format_color_text</md-icon>
<span class="md-list-item-text">{{ lang.option_list[0] }}</span>
<md-switch v-model="is_chinese" class="md-primary"></md-switch>
</md-list-item>
<md-list-item>
<md-icon>brightness_6</md-icon>
<span class="md-list-item-text">{{ lang.option_list[1] }}</span>
<md-switch v-model="is_dark_mode" class="md-primary"></md-switch>
</md-list-item>
<md-subheader class="md-primary">{{ lang.subheader[1] }}</md-subheader>
<md-list-item>
<md-icon>format_color_text</md-icon>
<span class="md-list-item-text">{{ lang.option_list[0] }}</span>
<md-switch v-model="is_chinese" class="md-primary"></md-switch>
</md-list-item>
<md-list-item>
<md-icon>brightness_6</md-icon>
<span class="md-list-item-text">{{ lang.option_list[1] }}</span>
<md-switch v-model="is_dark_mode" class="md-primary"></md-switch>
</md-list-item>
<md-list-item>
<md-icon>update</md-icon>
<md-field>
<label for="expired_time">{{ lang.option_list[2] }}</label>
<md-select v-model="expired_time" name="expired_time" id="expired_time">
<md-option value="300000">5 min</md-option>
<md-option value="600000">10 min</md-option>
<md-option value="900000">15 min</md-option>
<md-option :value="Number.MAX_SAFE_INTEGER">{{ lang.expired_time }}</md-option>
</md-select>
</md-field>
</md-list-item>
</md-list>
<md-list-item>
<md-icon>update</md-icon>
<md-field>
<label for="expired_time">{{ lang.option_list[2] }}</label>
<md-select v-model="expired_time" name="expired_time" id="expired_time">
<md-option value="300000">5 min</md-option>
<md-option value="600000">10 min</md-option>
<md-option value="900000">15 min</md-option>
<md-option :value="Number.MAX_SAFE_INTEGER">{{ lang.expired_time }}</md-option>
</md-select>
</md-field>
</md-list-item>
<md-dialog-prompt
:md-active.sync="reset_main_pwd_active"
v-model="new_main_pwd"
:md-title="lang.dialog.title"
md-input-maxlength="30"
:md-input-placeholder="lang.dialog.placeholder"
:md-confirm-text="lang.dialog.confirm"
:md-cancel-text="lang.dialog.cancel"
@md-confirm="resetPwd"
/>
<md-dialog-confirm
:md-active.sync="reset_confirm"
:md-title="lang.reset_dialog.title"
:md-content="lang.reset_dialog.content"
:md-confirm-text="lang.reset_dialog.confirm"
:md-cancel-text="lang.reset_dialog.cancel"
@md-cancel="reset_confirm = false"
@md-confirm="resetStart"
/>
<md-divider></md-divider>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
<md-subheader class="md-primary">{{ lang.subheader[2] }}</md-subheader>
<md-list-item style="list-style-type:none;" @click="reset_row_data_active = true">
<md-icon>call_merge</md-icon>
<span class="md-list-item-text">{{ lang.import_export_list[0] }}</span>
</md-list-item>
<md-list-item
style="list-style-type:none;"
v-clipboard:copy="copy_data"
v-clipboard:success="onCopyCodebook"
v-clipboard:error="onErrorCodebook"
>
<md-icon>call_split</md-icon>
<span class="md-list-item-text">{{ lang.import_export_list[1] }}</span>
</md-list-item>
</md-list>
<md-dialog-prompt
:md-active.sync="reset_row_data_active"
v-model="new_row_data"
:md-title="lang.import_dialog.title"
:md-input-placeholder="lang.import_dialog.placeholder"
:md-confirm-text="lang.import_dialog.confirm"
:md-cancel-text="lang.import_dialog.cancel"
@md-confirm="resetRowData"
/>
<md-dialog-prompt
:md-active.sync="reset_main_pwd_active"
v-model="new_main_pwd"
:md-title="lang.dialog.title"
md-input-maxlength="30"
:md-input-placeholder="lang.dialog.placeholder"
:md-confirm-text="lang.dialog.confirm"
:md-cancel-text="lang.dialog.cancel"
@md-confirm="resetPwd"
/>
<md-dialog-confirm
:md-active.sync="reset_confirm"
:md-title="lang.reset_dialog.title"
:md-content="lang.reset_dialog.content"
:md-confirm-text="lang.reset_dialog.confirm"
:md-cancel-text="lang.reset_dialog.cancel"
@md-cancel="reset_confirm = false"
@md-confirm="resetStart"
/>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</v-touch>
</md-app-content>
</md-app>
</div>
</template>
<script>
// @ is an alias to /src
import { mapState, mapActions } from 'vuex';
import { encrypt, decrypt, encryptMainCode, decryptMainCode } from '@/utils/aes.js';
import { login, activation, syncLocal, syncCloud } from '@/axios/api.js';
import { lang } from '@/utils/language.js';
import { setHtmlFontSize } from '@/utils/px2rem.js';
import { mapState, mapActions } from "vuex";
import {
encrypt,
decrypt,
encryptMainCode,
decryptMainCode,
} from "@/utils/aes.js";
import { login, activation, syncLocal, syncCloud } from "@/axios/api.js";
import { lang } from "@/utils/language.js";
import { setHtmlFontSize } from "@/utils/px2rem.js";
export default {
name: 'Add',
data() {
return {
// loading
account_loading: false,
codebook_loading: false,
application_loading: false,
reset_main_pwd_active: false,
new_main_pwd: '',
reset_active: false,
//
is_chinese: false,
is_dark_mode: false,
// 300000 5
expired_time: 300000,
// snackbar
show_snackbar: false,
snakebar_msg: '',
//
has_init: false,
lang: '',
clientHeight: '',
//
reset_confirm: false,
reset_type: ''
};
},
computed: {
...mapState(['user_infos', 'row_data', 'row_pwd', 'settings'])
},
methods: {
...mapActions(['setUserInfo', 'setRowData', 'setRowPwd', 'setSettings']),
name: "Add",
data() {
return {
// loading
account_loading: false,
codebook_loading: false,
application_loading: false,
reset_main_pwd_active: false,
new_row_data: "",
reset_row_data_active: false,
new_main_pwd: "",
reset_active: false,
//
is_chinese: false,
is_dark_mode: false,
// 300000 5
expired_time: 300000,
// snackbar
show_snackbar: false,
snakebar_msg: "",
//
has_init: false,
lang: "",
clientHeight: "",
//
reset_confirm: false,
reset_type: "",
};
},
computed: {
...mapState(["user_infos", "row_data", "row_pwd", "settings"]),
copy_data: function () {
if (!this.row_data) return "无内容";
return this.row_data;
},
},
methods: {
...mapActions(["setUserInfo", "setRowData", "setRowPwd", "setSettings"]),
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName('md-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.style.minHeight = clientHeight - 32 + 'px';
window.document.documentElement.setAttribute('data-theme', this.settings.is_dark_mode ? 'dark' : 'light');
},
//
init() {
// vuex
this.$store.replaceState(Object.assign(this.$store.state, JSON.parse(localStorage.getItem('storeState'))));
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.is_chinese = this.settings.is_chinese;
this.is_dark_mode = this.settings.is_dark_mode;
this.expired_time = this.settings.expired_time;
this.initLanguage();
// flag
setTimeout(
function() {
this.has_init = true;
console.log('本页信息覆写完成');
}.bind(this),
10
);
} else {
//
this.turnToHome('密码超时');
}
} else {
//
this.turnToHome('无密码');
}
},
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName('md-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight =
clientHeight + "px";
this.$refs.list_placeholder.parentNode.style.minHeight =
clientHeight - 32 + "px";
window.document.documentElement.setAttribute(
"data-theme",
this.settings.is_dark_mode ? "dark" : "light"
);
},
//
init() {
// vuex
this.$store.replaceState(
Object.assign(
this.$store.state,
JSON.parse(localStorage.getItem("storeState"))
)
);
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.is_chinese = this.settings.is_chinese;
this.is_dark_mode = this.settings.is_dark_mode;
this.expired_time = this.settings.expired_time;
this.initLanguage();
// flag
setTimeout(
function () {
this.has_init = true;
console.log("本页信息覆写完成");
}.bind(this),
10
);
} else {
//
this.turnToHome("密码超时");
}
} else {
//
this.turnToHome("无密码");
}
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().settings.CHS;
} else {
this.lang = lang().settings.EN;
}
console.log('语言配置完成');
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().settings.CHS;
} else {
this.lang = lang().settings.EN;
}
console.log("语言配置完成");
},
//
back() {
//
if (this.reset_active) {
return;
}
this.$router.go(-1);
},
//
back() {
//
if (this.reset_active) {
return;
}
this.$router.go(-1);
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace('/');
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace("/");
},
//
resetAccount(need_loading = true, need_reset_drvice = false) {
this.reset_active = true;
if (need_loading) {
this.account_loading = true;
setTimeout(
function() {
this.account_loading = false;
this.snakebar_msg = this.lang.snakebar_msg.reset_account;
this.show_snackbar = true;
}.bind(this),
1000
);
}
//
let user_infos = {
user_name: 'A Little Canary',
cid: 'Codebook',
row_login_pwd: '',
drivce: need_reset_drvice ? '' : this.user_infos.drivce,
cloud_drivce: this.settings.is_chinese ? '暂无' : 'unknown',
update_time: new Date().getTime()
};
this.setUserInfo([user_infos, this]);
console.log('用户信息覆写完成');
this.reset_active = false;
},
//
resetAccount(need_loading = true, need_reset_drvice = false) {
this.reset_active = true;
if (need_loading) {
this.account_loading = true;
setTimeout(
function () {
this.account_loading = false;
this.snakebar_msg = this.lang.snakebar_msg.reset_account;
this.show_snackbar = true;
}.bind(this),
1000
);
}
//
let user_infos = {
user_name: "A Little Canary",
cid: "Codebook",
row_login_pwd: "",
drivce: need_reset_drvice ? "" : this.user_infos.drivce,
cloud_drivce: this.settings.is_chinese ? "暂无" : "unknown",
update_time: new Date().getTime(),
};
this.setUserInfo([user_infos, this]);
console.log("用户信息覆写完成");
this.reset_active = false;
},
//
resetCodebook(need_loading = true) {
this.reset_active = true;
if (need_loading) {
this.codebook_loading = true;
setTimeout(
function() {
this.codebook_loading = false;
this.snakebar_msg = this.lang.snakebar_msg.reset_codebook;
this.show_snackbar = true;
}.bind(this),
1000
);
}
this.setRowData(['', this]);
console.log('密码本覆写完成');
this.reset_active = false;
},
//
resetCodebook(need_loading = true) {
this.reset_active = true;
if (need_loading) {
this.codebook_loading = true;
setTimeout(
function () {
this.codebook_loading = false;
this.snakebar_msg = this.lang.snakebar_msg.reset_codebook;
this.show_snackbar = true;
}.bind(this),
1000
);
}
this.setRowData(["", this]);
console.log("密码本覆写完成");
this.reset_active = false;
},
//
resetApplication() {
this.application_loading = true;
this.reset_active = true;
this.resetAccount(false, true);
this.resetCodebook(false);
this.setRowPwd(['', this]);
console.log('主密码覆写完成');
let settings = {
is_chinese: true,
is_dark_mode: false,
expired_time: 300000
};
this.setSettings([settings, this]);
console.log('个性化设置覆写完成');
this.reset_active = false;
this.application_loading = false;
this.turnToHome('项目重置完成');
},
//
resetApplication() {
this.application_loading = true;
this.reset_active = true;
this.resetAccount(false, true);
this.resetCodebook(false);
this.setRowPwd(["", this]);
console.log("主密码覆写完成");
let settings = {
is_chinese: true,
is_dark_mode: false,
expired_time: 300000,
};
this.setSettings([settings, this]);
console.log("个性化设置覆写完成");
this.reset_active = false;
this.application_loading = false;
this.turnToHome("项目重置完成");
},
//
resetSettings() {
let settings = {
is_chinese: this.is_chinese,
is_dark_mode: this.is_dark_mode,
expired_time: this.expired_time
};
this.setSettings([settings, this]);
console.log('个性化设置覆写完成');
if (this.settings.is_chinese) {
this.lang = lang().settings.CHS;
} else {
this.lang = lang().settings.EN;
}
window.document.documentElement.setAttribute('data-theme', this.settings.is_dark_mode ? 'dark' : 'light');
console.log('语言配置完成');
this.snakebar_msg = this.lang.snakebar_msg.reset_settings;
this.show_snackbar = true;
},
//
resetSettings() {
let settings = {
is_chinese: this.is_chinese,
is_dark_mode: this.is_dark_mode,
expired_time: this.expired_time,
};
this.setSettings([settings, this]);
console.log("个性化设置覆写完成");
if (this.settings.is_chinese) {
this.lang = lang().settings.CHS;
} else {
this.lang = lang().settings.EN;
}
window.document.documentElement.setAttribute(
"data-theme",
this.settings.is_dark_mode ? "dark" : "light"
);
console.log("语言配置完成");
this.snakebar_msg = this.lang.snakebar_msg.reset_settings;
this.show_snackbar = true;
},
open_dialog() {
this.reset_main_pwd_active = true;
},
open_dialog() {
this.reset_main_pwd_active = true;
},
resetPwd(new_main_pwd) {
//
if (!new_main_pwd.trim()) {
this.snakebar_msg = this.lang.snakebar_msg.reset_pwd_failed;
this.show_snackbar = true;
}
//
if (this.row_data) {
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
let data_decrypt;
//
try {
data_decrypt = decrypt(main_code_decrpt, this.row_data);
} catch (e) {
//
//
console.log(e);
this.turnToHome('密码错误');
return;
}
let data_list = JSON.parse(data_decrypt);
let data_list_aes = encrypt(new_main_pwd, data_list);
this.setRowData([data_list_aes, this]);
console.log('密码本密码修改成功,覆写完成')
}
let main_code_aes = encryptMainCode(new_main_pwd);
let row_pwd = {
main_code: main_code_aes,
create_time: new Date().getTime()
};
this.setRowPwd([row_pwd, this]);
this.snakebar_msg = this.lang.snakebar_msg.reset_pwd_successful;
this.show_snackbar = true;
},
resetRowData(new_row_data) {
//
if (!new_row_data.trim()) {
this.snakebar_msg = this.lang.snakebar_msg.reset_row_data_failed;
this.show_snackbar = true;
return;
}
this.setRowData([new_row_data, this]);
this.snakebar_msg = this.lang.snakebar_msg.reset_row_data_successful;
this.show_snackbar = true;
},
//
openResetDialog(type) {
this.reset_confirm = true;
this.reset_type = type
},
resetPwd(new_main_pwd) {
//
if (!new_main_pwd.trim()) {
this.snakebar_msg = this.lang.snakebar_msg.reset_pwd_failed;
this.show_snackbar = true;
}
//
if (this.row_data) {
//
let main_code_decrpt = decryptMainCode(this.row_pwd.main_code);
let data_decrypt;
//
try {
data_decrypt = decrypt(main_code_decrpt, this.row_data);
} catch (e) {
//
//
console.log(e);
this.turnToHome("密码错误");
return;
}
let data_list = JSON.parse(data_decrypt);
let data_list_aes = encrypt(new_main_pwd, data_list);
this.setRowData([data_list_aes, this]);
console.log("密码本密码修改成功,覆写完成");
}
let main_code_aes = encryptMainCode(new_main_pwd);
let row_pwd = {
main_code: main_code_aes,
create_time: new Date().getTime(),
};
this.setRowPwd([row_pwd, this]);
this.snakebar_msg = this.lang.snakebar_msg.reset_pwd_successful;
this.show_snackbar = true;
},
//
resetStart() {
console.log('用户点击继续,重置开始')
switch(this.reset_type) {
case 'account':
this.resetAccount();
break
case 'codebook':
this.resetCodebook();
break
case 'application':
this.resetApplication();
break
default:
return
}
this.reset_type = ''
}
},
created() {
this.lang = lang().settings.CHS;
console.log('临时语言系统加载完成');
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function() {
this.changeFixed(this.clientHeight);
},
// `is_chinese`
is_chinese: function() {
if (this.has_init) this.resetSettings();
},
// `is_dark_mode`
is_dark_mode: function() {
if (this.has_init) this.resetSettings();
},
// `expired_time`
expired_time: function() {
if (this.has_init) this.resetSettings();
}
},
beforeDestroy() {},
components: {}
//
openResetDialog(type) {
this.reset_confirm = true;
this.reset_type = type;
},
//
resetStart() {
console.log("用户点击继续,重置开始");
switch (this.reset_type) {
case "account":
this.resetAccount();
break;
case "codebook":
this.resetCodebook();
break;
case "application":
this.resetApplication();
break;
default:
return;
}
this.reset_type = "";
},
//
onCopyCodebook(e) {
this.snakebar_msg = this.lang.copy.successful;
this.show_snackbar = true;
},
//
onErrorCodebook(e) {
this.snakebar_msg = this.lang.copy.failed;
this.show_snackbar = true;
},
},
created() {
this.lang = lang().settings.CHS;
console.log("临时语言系统加载完成");
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function () {
this.changeFixed(this.clientHeight);
},
// `is_chinese`
is_chinese: function () {
if (this.has_init) this.resetSettings();
},
// `is_dark_mode`
is_dark_mode: function () {
if (this.has_init) this.resetSettings();
},
// `expired_time`
expired_time: function () {
if (this.has_init) this.resetSettings();
},
},
beforeDestroy() {},
components: {},
};
</script>
<style scoped lang="scss" type="text/scss">
@import '../../style/main';
@import "../../style/main";
.account {
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
}
</style>
<!--

View File

@ -1,253 +1,266 @@
<template>
<div class="unlock" :class="settings.is_dark_mode ? 'dark-theme-unlock' : ''">
<!-- <md-toolbar class="md-primary">
<div class="unlock" :class="settings.is_dark_mode ? 'dark-theme-unlock' : ''">
<!-- <md-toolbar class="md-primary">
<h3 class="md-title" style="flex: 1">{{ is_create ? lang.title[0] : lang.title[1] }}</h3>
</md-toolbar> -->
</md-toolbar>-->
<icon class="logo" name="canary"></icon>
<icon class="logo" name="canary"></icon>
<p class="md-title center">{{ is_create ? lang.subheader[0] : lang.subheader[1] }}</p>
<p class="md-title center">{{ is_create ? lang.subheader[0] : lang.subheader[1] }}</p>
<md-field class="input-box" :class="messageClass">
<label>{{ is_create ? lang.pwd_label[0] : lang.pwd_label[1] }}</label>
<md-input v-model="pwd" type="password"></md-input>
<span class="md-error">{{ err_msg }}</span>
</md-field>
<md-field class="input-box" :class="messageClass">
<label>{{ is_create ? lang.pwd_label[0] : lang.pwd_label[1] }}</label>
<md-input v-model="pwd" type="password"></md-input>
<span class="md-error">{{ err_msg }}</span>
</md-field>
<md-field class="input-box" :class="messageClass" v-if="is_create">
<label>{{ lang.repeat_pwd }}</label>
<md-input v-model="repeat" type="password"></md-input>
<span class="md-error">{{ err_msg }}</span>
</md-field>
<md-field class="input-box" :class="messageClass" v-if="is_create">
<label>{{ lang.repeat_pwd }}</label>
<md-input v-model="repeat" type="password"></md-input>
<span class="md-error">{{ err_msg }}</span>
</md-field>
<md-field class="input-box" :class="drivceMessageClass" v-if="is_create">
<label>{{ lang.drivce_label }}</label>
<md-input v-model="drivce"></md-input>
<span class="md-error">{{ drivce_err_msg }}</span>
</md-field>
<md-field class="input-box" :class="drivceMessageClass" v-if="is_create">
<label>{{ lang.drivce_label }}</label>
<md-input v-model="drivce"></md-input>
<span class="md-error">{{ drivce_err_msg }}</span>
</md-field>
<md-button class="md-raised md-primary center" @click="judgePwd()">{{ lang.enter }}</md-button>
<md-button class="md-raised md-primary center" @click="judgePwd()">{{ lang.enter }}</md-button>
<p class="center tips">此密码本为长理专版未开启同步密码将加密后存放本地开启同步密码将加密后存放在您的长理网盘中请放心使用 <span class="qq-link" @click="openQQ()">点我加入用户群</span></p>
<p class="center tips">
此密码本为长理专版未开启同步密码将加密后存放本地开启同步密码将加密后存放在您的长理网盘中请放心使用
<span class="qq-link" @click="openQQ()">点我加入用户群</span>
</p>
<md-dialog-confirm
:md-active.sync="show_dialog"
:md-title="lang.dialog.title"
:md-content="lang.dialog.content"
:md-confirm-text="lang.dialog.confirm"
:md-cancel-text="lang.dialog.cancel"
@md-cancel="onCancel"
@md-confirm="submit"
/>
<md-dialog-confirm
:md-active.sync="show_dialog"
:md-title="lang.dialog.title"
:md-content="lang.dialog.content"
:md-confirm-text="lang.dialog.confirm"
:md-cancel-text="lang.dialog.cancel"
@md-cancel="onCancel"
@md-confirm="submit"
/>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</div>
<md-snackbar md-position="center" :md-active.sync="show_snackbar" md-persistent>
<span>{{ snakebar_msg }}</span>
</md-snackbar>
</div>
</template>
<script>
// @ is an alias to /src
import { mapState, mapActions } from 'vuex';
import { encrypt, decrypt, encryptMainCode, decryptMainCode } from '@/utils/aes.js';
import { lang } from '@/utils/language.js';
import { mapState, mapActions } from "vuex";
import {
encrypt,
decrypt,
encryptMainCode,
decryptMainCode,
} from "@/utils/aes.js";
import { lang } from "@/utils/language.js";
export default {
name: 'Unlock',
data() {
return {
pwd: '',
repeat: '',
drivce: '',
is_err: false,
is_drivce_err: false,
err_msg: '',
drivce_err_msg:'',
show_dialog: false,
is_create: false,
show_snackbar: false,
snakebar_msg: '',
lang: ''
};
},
computed: {
...mapState(['row_data', 'settings', 'user_infos']),
messageClass() {
return {
'md-invalid': this.is_err
};
},
drivceMessageClass() {
return {
'md-invalid': this.is_drivce_err
}
}
},
methods: {
...mapActions(['setRowPwd', 'setUserInfo']),
name: "Unlock",
data() {
return {
pwd: "",
repeat: "",
drivce: "",
is_err: false,
is_drivce_err: false,
err_msg: "",
drivce_err_msg: "",
show_dialog: false,
is_create: false,
show_snackbar: false,
snakebar_msg: "",
lang: "",
};
},
computed: {
...mapState(["row_data", "settings", "user_infos"]),
messageClass() {
return {
"md-invalid": this.is_err,
};
},
drivceMessageClass() {
return {
"md-invalid": this.is_drivce_err,
};
},
},
methods: {
...mapActions(["setRowPwd", "setUserInfo"]),
//
init() {
// vuex
this.$store.replaceState(Object.assign(this.$store.state, JSON.parse(localStorage.getItem('storeState'))));
this.initLanguage();
//
this.is_create = !this.row_data && !this.user_infos.drivce;
},
//
init() {
// vuex
this.$store.replaceState(
Object.assign(
this.$store.state,
JSON.parse(localStorage.getItem("storeState"))
)
);
this.initLanguage();
//
this.is_create = !this.row_data && !this.user_infos.drivce;
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().unlock.CHS;
} else {
this.lang = lang().unlock.EN;
}
console.log('语言配置完成');
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().unlock.CHS;
} else {
this.lang = lang().unlock.EN;
}
console.log("语言配置完成");
},
//
judgePwd() {
//
if (this.pwd.trim().length == 0) {
this.is_err = true;
this.err_msg = this.lang.empty_pwd_error;
console.log('密码为空拦截');
return;
}
//
if (!this.is_create) {
//
//
let test_decode
try {
test_decode = decrypt(this.pwd, this.row_data);
} catch (e) {
//
this.is_err = true;
this.err_msg = this.lang.wrong_pwd_error;
console.log('密码输入错误拦截');
return
}
// ,locolstorage
this.submit();
} else {
if (this.drivce.trim().length == 0) {
this.is_drivce_err = true;
this.drivce_err_msg = this.lang.drivce_error;
console.log('设备名为空拦截');
return;
}
//
if (this.pwd == this.repeat) {
//
this.show_dialog = true;
} else {
//
this.is_err = true;
this.err_msg = this.lang.match_pwd_error;
console.log('密码两次输入不一致拦截');
}
}
},
//
judgePwd() {
//
if (this.pwd.trim().length == 0) {
this.is_err = true;
this.err_msg = this.lang.empty_pwd_error;
console.log("密码为空拦截");
return;
}
//
if (!this.is_create) {
//
//
let test_decode;
try {
test_decode = decrypt(this.pwd, this.row_data);
} catch (e) {
//
this.is_err = true;
this.err_msg = this.lang.wrong_pwd_error;
console.log("密码输入错误拦截");
return;
}
// ,locolstorage
this.submit();
} else {
if (this.drivce.trim().length == 0) {
this.is_drivce_err = true;
this.drivce_err_msg = this.lang.drivce_error;
console.log("设备名为空拦截");
return;
}
//
if (this.pwd == this.repeat) {
//
this.show_dialog = true;
} else {
//
this.is_err = true;
this.err_msg = this.lang.match_pwd_error;
console.log("密码两次输入不一致拦截");
}
}
},
//
onCancel() {
console.log('用户点击取消');
},
//
onCancel() {
console.log("用户点击取消");
},
//
submit() {
let main_code_aes = encryptMainCode(this.pwd);
let row_pwd = {
main_code: main_code_aes,
create_time: new Date().getTime()
};
let user_infos = this.user_infos
user_infos.drivce = this.drivce
this.setUserInfo([user_infos, this])
this.setRowPwd([row_pwd, this]);
console.log('主密码设置完成返回Home');
this.$router.replace('/');
},
//
submit() {
let main_code_aes = encryptMainCode(this.pwd);
let row_pwd = {
main_code: main_code_aes,
create_time: new Date().getTime(),
};
let user_infos = this.user_infos;
user_infos.drivce = this.drivce;
this.setUserInfo([user_infos, this]);
this.setRowPwd([row_pwd, this]);
console.log("主密码设置完成返回Home");
this.$router.replace("/");
},
openQQ() {
window.open('https://jq.qq.com/?_wv=1027&k=KIKWmqys')
}
},
created() {
this.lang = lang().unlock.CHS;
console.log('临时语言系统加载完成');
this.init();
},
mounted() {
// SnakeBar
if (this.$route.params.msg) {
//
this.snakebar_msg = this.$route.params.msg;
this.show_snackbar = true;
}
},
watch: {
// `pwd`
pwd: function() {
this.is_err = false;
this.err_msg = '';
},
// `repeat`
repeat: function() {
this.is_err = false;
this.err_msg = '';
},
// `drivce`
drivce: function() {
this.is_drivce_err = false;
this.drivce_err_msg = ''
}
},
beforeDestroy() {},
components: {}
openQQ() {
window.open("https://jq.qq.com/?_wv=1027&k=KIKWmqys");
},
},
created() {
this.lang = lang().unlock.CHS;
console.log("临时语言系统加载完成");
this.init();
},
mounted() {
// SnakeBar
if (this.$route.params.msg) {
//
this.snakebar_msg = this.$route.params.msg;
this.show_snackbar = true;
}
},
watch: {
// `pwd`
pwd: function () {
this.is_err = false;
this.err_msg = "";
},
// `repeat`
repeat: function () {
this.is_err = false;
this.err_msg = "";
},
// `drivce`
drivce: function () {
this.is_drivce_err = false;
this.drivce_err_msg = "";
},
},
beforeDestroy() {},
components: {},
};
</script>
<style scoped lang="scss" type="text/scss">
@import '../../style/main';
@import "../../style/main";
.unlock {
width: 100%;
min-height: 100%;
background: #fff;
.logo {
margin: 0 auto;
display: block;
height: 5rem;
width: 5rem;
}
.center {
margin: 0 auto;
text-align: center;
display: block;
}
.input-box {
width: 70%;
max-width: 350px;
margin: 0.5rem auto;
}
.md-raised {
margin-top: 1rem;
}
width: 100%;
min-height: 100%;
background: #fff;
.logo {
margin: 0 auto;
display: block;
height: 5rem;
width: 5rem;
}
.center {
margin: 0 auto;
text-align: center;
display: block;
}
.input-box {
width: 70%;
max-width: 350px;
margin: 0.5rem auto;
}
.md-raised {
margin-top: 1rem;
}
}
.dark-theme-unlock {
background: #424242;
.center {
color: #fff !important;
}
background: #424242;
.center {
color: #fff !important;
}
}
.tips {
font-size: .3rem;
max-width: 70%;
padding: 1rem 0;
color: #989898;
font-size: 0.3rem;
max-width: 70%;
padding: 1rem 0;
color: #989898;
}
.qq-link {
color: #448aff;
color: #448aff;
}
</style>

View File

@ -1,149 +1,167 @@
<template>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()"><md-icon>arrow_back</md-icon></md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<div class="account">
<md-app md-waterfall md-mode="fixed">
<md-app-toolbar class="md-primary toolbar">
<div class="md-toolbar-section-start">
<md-button class="md-icon-button" @click="back()">
<md-icon>arrow_back</md-icon>
</md-button>
<h3 class="md-title" style="flex: 1">{{ lang.title }}</h3>
</div>
</md-app-toolbar>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-steppers md-vertical>
<md-step v-for="(item, index) in lang.timeline" :key="index" :id="index.toString()" :md-label="item.label" :md-description="item.tag">
<p v-for="(content, content_index) in item.content" :key="content_index">{{ content }}</p>
</md-step>
</md-steppers>
</v-touch>
</md-app-content>
</md-app>
</div>
<md-app-content>
<v-touch @swiperight="back()" :swipe-options="{ direction: 'horizontal' }">
<div ref="list_placeholder" style="height: 54px;"></div>
<md-steppers md-vertical>
<md-step
v-for="(item, index) in lang.timeline"
:key="index"
:id="index.toString()"
:md-label="item.label"
:md-description="item.tag"
>
<p v-for="(content, content_index) in item.content" :key="content_index">{{ content }}</p>
</md-step>
</md-steppers>
</v-touch>
</md-app-content>
</md-app>
</div>
</template>
<script>
// @ is an alias to /src
import { mapState } from 'vuex';
import { lang } from '@/utils/language.js';
import { setHtmlFontSize } from '@/utils/px2rem.js';
import { mapState } from "vuex";
import { lang } from "@/utils/language.js";
import { setHtmlFontSize } from "@/utils/px2rem.js";
export default {
name: 'UpdateLog',
data() {
return {
//
lang: '',
//
clientHeight: ''
};
},
computed: {
...mapState(['row_pwd', 'settings'])
},
methods: {
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName('md-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.style.minHeight = clientHeight - 32 + 'px';
window.document.documentElement.setAttribute('data-theme', this.settings.is_dark_mode ? 'dark' : 'light');
},
name: "UpdateLog",
data() {
return {
//
lang: "",
//
clientHeight: "",
};
},
computed: {
...mapState(["row_pwd", "settings"]),
},
methods: {
// md-app
changeFixed(clientHeight) {
//
// console.log(clientHeight);
// window.document.getElementsByClassName('md-content')[0].style.minHeight = clientHeight + 'px';
this.$refs.list_placeholder.parentNode.parentNode.style.maxHeight =
clientHeight + "px";
this.$refs.list_placeholder.parentNode.style.minHeight =
clientHeight - 32 + "px";
window.document.documentElement.setAttribute(
"data-theme",
this.settings.is_dark_mode ? "dark" : "light"
);
},
//
init() {
// vuex
this.$store.replaceState(Object.assign(this.$store.state, JSON.parse(localStorage.getItem('storeState'))));
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.initLanguage();
} else {
//
this.turnToHome('密码超时');
}
} else {
//
this.turnToHome('无密码');
}
},
//
init() {
// vuex
this.$store.replaceState(
Object.assign(
this.$store.state,
JSON.parse(localStorage.getItem("storeState"))
)
);
//
if (Object.keys(this.row_pwd).length != 0) {
//
let now = new Date().getTime();
if (now - this.row_pwd.create_time < this.settings.expired_time) {
//
//
this.initLanguage();
} else {
//
this.turnToHome("密码超时");
}
} else {
//
this.turnToHome("无密码");
}
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().update_log.CHS;
} else {
this.lang = lang().update_log.EN;
}
console.log('语言配置完成');
},
//
initLanguage() {
if (this.settings.is_chinese) {
this.lang = lang().update_log.CHS;
} else {
this.lang = lang().update_log.EN;
}
console.log("语言配置完成");
},
//
back() {
this.$router.go(-1);
},
//
back() {
this.$router.go(-1);
},
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace('/');
}
},
created() {
this.lang = lang().update_log.CHS;
console.log('临时语言系统加载完成');
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function() {
this.changeFixed(this.clientHeight);
}
},
beforeDestroy() {},
components: {}
// Home
turnToHome(msg) {
console.log(msg);
this.$router.replace("/");
},
},
created() {
this.lang = lang().update_log.CHS;
console.log("临时语言系统加载完成");
this.init();
},
mounted() {
//
this.clientHeight = `${document.documentElement.clientHeight}`;
//document.body.clientWidth;
window.onresize = function temp() {
this.clientHeight = `${document.documentElement.clientHeight}`;
setHtmlFontSize();
}.bind(this);
},
watch: {
// `clientHeight`
clientHeight: function () {
this.changeFixed(this.clientHeight);
},
},
beforeDestroy() {},
components: {},
};
</script>
<style scoped lang="scss" type="text/scss">
@import '../../style/main';
@import "../../style/main";
.account {
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
width: 100%;
min-height: 100%;
background: #fff;
.expand {
width: 100%;
margin: 0 !important;
min-height: 1.2rem !important;
margin-top: 0.5rem !important;
}
.time-content {
text-align: right !important;
}
.loading-box {
position: absolute;
right: 50%;
top: 50%;
margin-right: -11px;
margin-top: -11px;
}
.toolbar {
position: fixed !important;
}
}
</style>

View File

@ -1,31 +1,31 @@
this.show_list = [{
open_count: 1,
title: 'QQ',
user_name: '1144131090',
password: 'test',
node: '测试用'
},
{
open_count: 2,
title: '微信',
user_name: '15143211127',
password: 'test',
node: '测试用'
},
{
open_count: 3,
title: 'WIFI',
user_name: 'admin',
password: 'test',
node: '测试用'
},
{
open_count: 4,
title: 'WeGame',
user_name: '1144131090',
password: 'test',
node: '测试用'
}
open_count: 1,
title: 'QQ',
user_name: '1144131090',
password: 'test',
node: '测试用'
},
{
open_count: 2,
title: '微信',
user_name: '15143211127',
password: 'test',
node: '测试用'
},
{
open_count: 3,
title: 'WIFI',
user_name: 'admin',
password: 'test',
node: '测试用'
},
{
open_count: 4,
title: 'WeGame',
user_name: '1144131090',
password: 'test',
node: '测试用'
}
];
function test() {
@ -44,4 +44,4 @@ let test_encode = encrypt('10aeff', '这是一段测试文本');
console.log('加密后', test_encode)
let test_decode = decrypt('10aefs', test_encode)
console.log('解密后', test_decode)
console.log('类型', typeof(test_decode)) // 密码错误就是空字符串
console.log('类型', typeof (test_decode)) // 密码错误就是空字符串