hosts-switch/usr/lib/hosts-switch/webapp/lib/ui/layer/index.js

328 lines
11 KiB
JavaScript

import { css, html, Component, bind, styleMap } from "wkit";
import "../form/input.js";
let uniqueInstance = null;
let toastInstance = null;
const LANG_TITLE = "\u63D0\u793A";
const LANG_BTNS = ["\u53D6\u6D88", "\u786E\u5B9A"];
const UNIQUE_TYPES = ["alert", "confirm", "prompt"];
const BUILDIN_TYPES = UNIQUE_TYPES.concat(["toast"]);
class Layer extends Component {
static animation = {};
static props = {
type: {
type: String,
default: null,
observer(v) {
this.#wrapped = !BUILDIN_TYPES.includes(v);
}
},
left: { type: String, attribute: false },
right: { type: String, attribute: false },
top: { type: String, attribute: false },
bottom: { type: String, attribute: false },
background: { type: String, attribute: false },
maskColor: { type: String, attribute: false },
mask: false,
maskClose: false,
title: { type: String, default: "", attribute: false },
content: { type: String, default: "", attribute: false },
btns: []
};
static styles = [
css`:host{display:none;justify-content:center;align-items:center;position:fixed;z-index:65534;left:0;top:0;width:100%;height:0}:host([type]){display:flex}:host([type=toast]) .layer,:host([type=common]) .layer{position:absolute}.noselect{-webkit-touch-callout:none;user-select:none}.noselect img,.noselect a{-webkit-user-drag:none}`,
css`.layer{flex:0 auto;position:relative;z-index:65535;border-radius:3px;color:#666;font-size:14px;background:rgba(255,255,255,.8);box-shadow:0 5px 20px rgba(0,0,0,.3);transition:opacity .2s ease-in-out,left .2s ease-in-out,right .2s ease-in-out,top .2s ease-in-out,bottom .2s ease-in-out}.layer:active{z-index:65536}`,
/* 弹层样式 */
css`.layer__title{display:flex;justify-content:space-between;align-items:center;width:100%;height:60px;padding:15px;font-size:16px;color:var(--color-dark-2)}.layer__title wc-icon{--wc-icon-size: 14px}.layer__title wc-icon:hover{color:var(--color-red-1)}.layer__content{display:flex;position:relative;width:100%;height:auto;min-height:50px;word-break:break-all;word-wrap:break-word}::slotted(.layer__content__input){flex:1}::slotted(.layer__content__toast){flex-shrink:0;flex:1;display:flex;align-items:center;width:300px;min-height:40px;margin-bottom:15px !important;padding:0 10px !important;border-radius:3px;font-weight:normal;text-indent:8px;--wc-icon-size: 16px;color:var(--color-dark-1);box-shadow:0 2px 12px rgba(0,0,0,.1)}::slotted(.layer__content__toast+.layer__content__toast){margin-top:30px}::slotted(.layer__content__toast.style-info){border:1px solid #ebeef5;background:#edf2fc;color:var(--color-grey-3)}::slotted(.layer__content__toast.style-success){border:1px solid #e1f3d8;background:#f0f9eb;color:var(--color-green-3)}::slotted(.layer__content__toast.style-warning){border:1px solid #faebb4;background:#faecd8;color:var(--color-red-1)}::slotted(.layer__content__toast.style-error){border:1px solid #f5c4c4;background:#fde2e2;color:var(--color-red-1)}.layer__ctrl{display:flex;justify-content:flex-end;width:100%;height:60px;padding:15px;line-height:30px;font-size:14px;color:#454545;text-align:right}.layer__ctrl button{min-width:64px;height:30px;padding:0 10px;margin:0 5px;border:1px solid var(--color-plain-3);border-radius:3px;white-space:nowrap;background:#fff;font-size:inherit;font-family:inherit;outline:none;color:inherit}.layer__ctrl button:hover{background:var(--color-plain-1)}.layer__ctrl button:active{border-color:var(--color-grey-1)}.layer__ctrl button:focus{box-shadow:0 0 0 2px var(--color-plain-a)}.layer__ctrl button:last-child{color:#fff;background:var(--color-teal-2);border-color:rgba(0,0,0,0)}.layer__ctrl button:last-child:hover{background:var(--color-teal-1)}.layer__ctrl button:last-child:active{background:var(--color-teal-3)}.layer__ctrl button:last-child:focus{box-shadow:0 0 0 2px var(--color-teal-a)}.layer__ctrl button::-moz-focus-inner{border:none}`,
css`:host([mask]){height:100%;background:rgba(0,0,0,.2)}:host([type=alert]) .layer,:host([type=confirm]) .layer,:host([type=prompt]) .layer{max-width:600px;min-width:300px;background:#fff}:host([type=alert]) .layer__content,:host([type=confirm]) .layer__content,:host([type=prompt]) .layer__content{padding:0 15px}:host([type=toast]) .layer{box-shadow:none;background:none}:host([type=toast]) .layer__content{flex-direction:column;min-height:40px}:host([blurry]) .layer{backdrop-filter:blur(5px)}`
];
#wrapped = false;
#resolve = null;
#reject = null;
constructor() {
super();
this.promise = new Promise((resolve, reject) => {
this.#resolve = resolve;
this.#reject = reject;
});
this.promise.host = this;
}
#intercept(value) {
if (this.intercept) {
this.intercept(value, (_) => {
delete this.intercept;
this.#resolve(value);
this.$animate(true);
this.$refs.box.$animate(true).then((_2) => this.close());
});
} else {
this.#resolve(value);
this.$animate(true);
this.$refs.box.$animate(true).then((_) => this.close());
}
}
#play() {
switch (this.type) {
case "toast":
let elem = this.lastElementChild;
elem._anim = elem.animate(
[
{ marginTop: "-30px", opacity: 0 },
{ marginTop: "0", opacity: 1 }
],
{
duration: 200,
fill: "forwards"
}
);
setTimeout(() => {
elem._anim.reverse();
elem._anim.onfinish = (_) => {
elem.remove();
if (this.children.length === 0) {
this.close();
toastInstance = null;
}
};
}, 3e3);
break;
default:
this.$animate();
this.$refs.box.$animate();
break;
}
}
mounted() {
if (this.type === "prompt") {
this.$refs.input = this.firstElementChild;
bind(this.$refs.input, "submit", (ev) => {
this.#intercept(ev.target.value);
});
} else if (this.type === "toast") {
this.style.display = "";
}
if (this.mask) {
this.$on("click", (ev) => {
let path = ev.composedPath();
if (path[0] === ev.currentTarget) {
if (UNIQUE_TYPES.includes(this.type)) {
return;
}
if (this.maskClose) {
if (this.#wrapped === false) {
this.#reject();
}
this.$refs.box.$animate(true).then((_) => this.close());
}
}
});
if (this.maskColor) {
this.style.backgroundColor = this.maskColor;
}
}
if (this.background) {
this.$refs.box.style.backgroundColor = this.background;
}
this.#play();
}
updated() {
this.#play();
}
moveTo(obj = {}) {
var css2 = "";
for (var k in obj) {
css2 += `${k}:${obj[k]};`;
}
this.$refs.box.style.cssText += css2;
}
show() {
if (this.#wrapped) {
this.type = "common";
}
}
/**
* 关闭实例
*/
close() {
if (this.#wrapped) {
this.type = null;
this.$emit("close");
} else {
if (UNIQUE_TYPES.includes(this.type)) {
uniqueInstance = null;
}
this.$emit("close");
this.remove();
}
}
// 按钮的点击事件
handleBtnClick(ev) {
if (ev.target.tagName === "BUTTON") {
let idx = +ev.target.dataset.idx || 0;
switch (this.type) {
case "alert":
this.#intercept(null);
break;
case "confirm":
case "prompt":
if (idx === 0) {
this.#reject();
this.$animate(true);
this.$refs.box.$animate(true).then((_) => this.close());
} else {
let value = this.type === "prompt" ? this.$refs.input.value : null;
this.#intercept(value);
}
break;
default:
this.#intercept(idx);
break;
}
}
}
render() {
let { type, mask, left, right, top, bottom } = this;
let styles = "";
if (type === "common" && mask === false) {
top = top || 0;
}
styles = styleMap({ left, right, top, bottom });
return html`
<div
ref="box"
class="layer"
#animation=${{ type: "micro-bounce" }}
style=${styles}
>
<div
class="layer__title noselect"
style=${styleMap({ display: !!this.title ? "" : "none" })}
>
${this.title}
</div>
<div class="layer__content">
<slot></slot>
</div>
<div
class="layer__ctrl noselect"
style=${styleMap({ display: this.btns.length ? "" : "none" })}
@click=${this.handleBtnClick}
>
${this.btns.map((s, i) => html`<button data-idx=${i}>${s}</button>`)}
</div>
</div>
`;
}
}
function layer(opt = {}) {
let { type = "common", content = "" } = opt;
let layDom = type === "toast" ? toastInstance || document.createElement("wc-layer") : document.createElement("wc-layer");
let alreadyInTree = type === "toast" && !!toastInstance;
layDom.type = opt.type;
if (type === "toast") {
toastInstance = layDom;
layDom.top = "20px";
} else {
if (opt.btns && opt.btns.length) {
layDom.btns = opt.btns;
}
if (opt.intercept && typeof opt.intercept === "function") {
layDom.intercept = opt.intercept;
}
layDom.mask = opt.mask;
layDom.title = opt.title;
if (UNIQUE_TYPES.includes(type)) {
if (uniqueInstance) {
uniqueInstance.$animate(true).then((_) => {
uniqueInstance.close();
uniqueInstance = layDom;
});
} else {
uniqueInstance = layDom;
}
}
}
if (alreadyInTree) {
let tmp = document.createElement("template");
tmp.innerHTML = content;
layDom.appendChild(tmp.content.cloneNode(true));
layDom.updated();
} else {
layDom.innerHTML = content;
document.body.appendChild(layDom);
}
return layDom.promise;
}
layer.alert = function(content, title = LANG_TITLE, btns = LANG_BTNS.slice(1)) {
if (typeof title === "object") {
btns = title;
title = LANG_TITLE;
}
return this({
type: "alert",
title,
content,
mask: true,
btns
});
};
layer.confirm = function(content, title = LANG_TITLE, btns = LANG_BTNS.concat()) {
if (typeof title === "object") {
btns = title;
title = LANG_TITLE;
}
return this({
type: "confirm",
title,
content,
mask: true,
btns
});
};
layer.prompt = function(title = LANG_TITLE, defaultValue = "", intercept) {
if (typeof defaultValue === "function") {
intercept = defaultValue;
defaultValue = "";
}
if (!intercept) {
intercept = function(val, done) {
if (val) {
done();
}
};
}
return this({
type: "prompt",
title,
content: `<wc-input autofocus class="layer__content__input" value="${defaultValue}"></wc-input>`,
mask: true,
intercept,
btns: LANG_BTNS.concat()
});
};
layer.toast = function(txt, type = "info") {
var ico = type;
switch (type) {
case "info":
case "warning":
break;
case "error":
ico = "deny";
break;
case "success":
ico = "get";
break;
default:
ico = "info";
}
return this({
content: `
<div class="layer__content__toast style-${type}">
<wc-icon name="${ico}"></wc-icon>
<span class="toast-txt">${txt}</span>
</div>`,
type: "toast"
});
};
Layer.reg("layer");
window.layer = layer;
伪域名解析工具。
JavaScript 77.1%
CSS 10.9%
Python 6.2%
HTML 3%
Shell 2.8%