business logic, customer communication.
This commit is contained in:
parent
7c5a9214a1
commit
6157f994a9
@ -161,7 +161,7 @@ $listMaxHeight: 210px;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
>.dropdown-panel {
|
>.dropdown-box {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
visibility: hidden;
|
visibility: hidden;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
|
@ -64,7 +64,7 @@
|
|||||||
&,
|
&,
|
||||||
input[type="text"],
|
input[type="text"],
|
||||||
.dropdown-wrapper>.dropdown-header>.dropdown-text,
|
.dropdown-wrapper>.dropdown-header>.dropdown-text,
|
||||||
.dropdown-wrapper>.dropdown-panel>.dropdown-list {
|
.dropdown-wrapper>.dropdown-box>.dropdown-list {
|
||||||
font-size: var(--font-size);
|
font-size: var(--font-size);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -283,7 +283,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
>.dropdown-panel {
|
>.dropdown-box {
|
||||||
top: calc(var(--line-height) + 2px);
|
top: calc(var(--line-height) + 2px);
|
||||||
|
|
||||||
&.slide-up {
|
&.slide-up {
|
||||||
|
5
lib/app.js
Normal file
5
lib/app.js
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
import CustomerCommunication from "./app/communications/customer";
|
||||||
|
|
||||||
|
export {
|
||||||
|
CustomerCommunication
|
||||||
|
}
|
28
lib/app/communications/customer.d.ts
vendored
Normal file
28
lib/app/communications/customer.d.ts
vendored
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
interface CheckboxConfig {
|
||||||
|
checked: boolean
|
||||||
|
onchanged: (flag: boolean) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
interface InitConfig {
|
||||||
|
autoUpdates?: CheckboxConfig;
|
||||||
|
statusLink?: CheckboxConfig;
|
||||||
|
readonly?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class CustomerCommunication {
|
||||||
|
get autoUpdatesEnabled(): boolean;
|
||||||
|
set autoUpdatesEnabled(enabled: boolean);
|
||||||
|
get autoUpdates(): boolean;
|
||||||
|
set autoUpdates(checked: boolean);
|
||||||
|
|
||||||
|
get statusLinkEnabled(): boolean;
|
||||||
|
set statusLinkEnabled(enabled: boolean);
|
||||||
|
get statusLink(): boolean;
|
||||||
|
set statusLink(checked: boolean);
|
||||||
|
|
||||||
|
create(): HTMLElement;
|
||||||
|
}
|
||||||
|
|
||||||
|
declare var CustomerCommunication: {
|
||||||
|
new(opt: InitConfig): CustomerCommunication
|
||||||
|
}
|
226
lib/app/communications/customer.js
Normal file
226
lib/app/communications/customer.js
Normal file
@ -0,0 +1,226 @@
|
|||||||
|
import "./style.scss";
|
||||||
|
import { createElement, createElementInit } from "../../functions";
|
||||||
|
import { r } from "../../utility/lgres";
|
||||||
|
import { isEmail, isPhone } from "../../utility";
|
||||||
|
import { setTooltip } from "../../ui/tooltip";
|
||||||
|
import { createIcon } from "../../ui/icon";
|
||||||
|
import { createCheckbox } from "../../ui/checkbox";
|
||||||
|
import { createBox } from "./lib";
|
||||||
|
|
||||||
|
class CustomerCommunication {
|
||||||
|
#container;
|
||||||
|
#option;
|
||||||
|
#contacts;
|
||||||
|
#followers;
|
||||||
|
#enter;
|
||||||
|
#message;
|
||||||
|
|
||||||
|
constructor(opt) {
|
||||||
|
this.#option = opt ?? {};
|
||||||
|
}
|
||||||
|
|
||||||
|
get #autoUpdates() { return this.#container.querySelector('.check-auto-update>input') }
|
||||||
|
get autoUpdatesEnabled() { return this.#autoUpdates?.disabled !== true }
|
||||||
|
set autoUpdatesEnabled(flag) {
|
||||||
|
const element = this.#autoUpdates;
|
||||||
|
element != null && (element.disabled = flag === false);
|
||||||
|
}
|
||||||
|
get autoUpdates() { return this.#autoUpdates?.checked }
|
||||||
|
set autoUpdates(flag) {
|
||||||
|
const element = this.#autoUpdates;
|
||||||
|
element != null && (element.checked = flag);
|
||||||
|
}
|
||||||
|
|
||||||
|
get #statusLink() { return this.#container.querySelector('.check-status-link') }
|
||||||
|
get statusLinkEnabled() { return this.#statusLink?.disabled !== true }
|
||||||
|
set statusLinkEnabled(flag) {
|
||||||
|
const element = this.#statusLink;
|
||||||
|
element != null && (element.disabled = flag === false);
|
||||||
|
}
|
||||||
|
get statusLink() { return this.#statusLink?.checked }
|
||||||
|
set statusLink(flag) {
|
||||||
|
const element = this.#statusLink;
|
||||||
|
element != null && (element.checked = flag);
|
||||||
|
}
|
||||||
|
|
||||||
|
get text() { return this.#enter?.value }
|
||||||
|
set text(s) {
|
||||||
|
const element = this.#enter;
|
||||||
|
element != null && (element.value = s);
|
||||||
|
}
|
||||||
|
|
||||||
|
get contacts() {
|
||||||
|
return [...this.#contacts.children].map(el => {
|
||||||
|
const span = el.querySelector('span');
|
||||||
|
return { 'Key': span.innerText, 'Value': span.dataset.name };
|
||||||
|
});
|
||||||
|
}
|
||||||
|
set contacts(contacts) {
|
||||||
|
this.#contacts.replaceChildren();
|
||||||
|
if (contacts?.length > 0) {
|
||||||
|
for (let c of contacts) {
|
||||||
|
if (c.OptOut || c.OptOut_BC) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const mp = String(c.MobilePhone).trim();
|
||||||
|
const email = String(c.Email).trim();
|
||||||
|
if (c.ContactPreference === '0' && !isPhone(mp) ||
|
||||||
|
c.ContactPreference === '1' && !isEmail(email)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const to = c.ContactPreference === '0' ? mp : email;
|
||||||
|
this.#contacts.appendChild(
|
||||||
|
createElement('div', 'contact-item',
|
||||||
|
createIcon('fa-light', c.ContactPreference === '0' ? 'comment-lines' : 'envelope'),
|
||||||
|
createElementInit('span', span => {
|
||||||
|
span.dataset.name = c.Name;
|
||||||
|
span.innerText = to;
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
create() {
|
||||||
|
// functions
|
||||||
|
const checkAutoUpdate = createCheckbox({
|
||||||
|
className: 'check-auto-update',
|
||||||
|
checked: this.#option.autoUpdates?.checked,
|
||||||
|
enabled: this.#option.autoUpdates?.enabled,
|
||||||
|
checkedNode: createIcon('fa-regular', 'redo-alt'),
|
||||||
|
uncheckedNode: createIcon('fa-regular', 'ban'),
|
||||||
|
onchange: () => {
|
||||||
|
setTooltip(checkAutoUpdate, this.checked ?
|
||||||
|
r('autoUpdateEnabled', 'Auto Updates Enabled') :
|
||||||
|
r('autoUpdateDisabled', 'Auto Updates Disabled'));
|
||||||
|
if (typeof this.#option.autoUpdates?.onchanged === 'function') {
|
||||||
|
this.#option.autoUpdates.onchanged(this.checked);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const checkLink = createCheckbox({
|
||||||
|
className: 'check-status-link',
|
||||||
|
checked: this.#option.statusLink?.checked,
|
||||||
|
enabled: this.#option.statusLink?.enabled,
|
||||||
|
checkedNode: createIcon('fa-regular', 'link'),
|
||||||
|
uncheckedNode: createIcon('fa-regular', 'unlink'),
|
||||||
|
onchange: () => {
|
||||||
|
setTooltip(checkLink, this.checked ?
|
||||||
|
r('statusLinkIncluded', 'Status Link Included') :
|
||||||
|
r('statusLinkExcluded', 'Status Link Excluded'));
|
||||||
|
if (typeof this.#option.statusLink?.onchanged === 'function') {
|
||||||
|
this.#option.statusLink.onchanged(this.checked);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const container = createBox(
|
||||||
|
createElement('div', null,
|
||||||
|
createElementInit('div', div => div.innerText = r('messages', 'Customer Communication')),
|
||||||
|
createElementInit('div', div => div.innerText = consts.user?.companyName)),
|
||||||
|
[
|
||||||
|
setTooltip(checkAutoUpdate, r('autoUpdateDisabled', 'Auto Updates Disabled')),
|
||||||
|
setTooltip(checkLink, r('statusLinkExcluded', 'Status Link Excluded'))
|
||||||
|
]
|
||||||
|
);
|
||||||
|
// contacts
|
||||||
|
const contacts = createElement('div');
|
||||||
|
container.append(
|
||||||
|
createElement('div', 'contact-bar',
|
||||||
|
createIcon('fa-solid', 'user-circle', {
|
||||||
|
'fill': 'lightgray',
|
||||||
|
'flex': '0 0 auto'
|
||||||
|
}),
|
||||||
|
createElementInit('div', div => div.style.flex = '1 1 auto',
|
||||||
|
contacts,
|
||||||
|
createElementInit('button', button => {
|
||||||
|
button.className = 'roundbtn';
|
||||||
|
button.style.backgroundColor = 'rgb(1, 199, 172)';
|
||||||
|
button.appendChild(createIcon('fa-solid', 'user-edit'));
|
||||||
|
setTooltip(button, r('editContacts', 'Edit Contacts'));
|
||||||
|
button.addEventListener('click', () => {
|
||||||
|
// TODO:
|
||||||
|
});
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
this.#contacts = contacts;
|
||||||
|
// followers
|
||||||
|
const followers = createElement('div');
|
||||||
|
container.append(
|
||||||
|
createElement('div', 'contact-bar follower-bar',
|
||||||
|
createIcon('fa-solid', 'user-tag', {
|
||||||
|
'fill': '#fff',
|
||||||
|
'background-color': 'lightgray',
|
||||||
|
'box-sizing': 'border-box',
|
||||||
|
'border-radius': '15px',
|
||||||
|
'padding': '4px',
|
||||||
|
'flex': '0 0 auto'
|
||||||
|
}),
|
||||||
|
createElementInit('div', div => div.style.flex = '1 1 auto',
|
||||||
|
followers,
|
||||||
|
createElementInit('button', button => {
|
||||||
|
button.className = 'roundbtn';
|
||||||
|
button.style.backgroundColor = 'rgb(48, 107, 255)';
|
||||||
|
button.appendChild(createIcon('fa-solid', 'pen'));
|
||||||
|
setTooltip(button, r('editFollower', 'Edit Followers'));
|
||||||
|
button.addEventListener('click', () => {
|
||||||
|
// TODO:
|
||||||
|
});
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
this.#followers = followers;
|
||||||
|
// enter box
|
||||||
|
const enter = createElement('textarea');
|
||||||
|
enter.placeholder = r('typeComment', 'Enter Message Here');
|
||||||
|
enter.maxLength = 3000;
|
||||||
|
this.#enter = enter;
|
||||||
|
container.appendChild(
|
||||||
|
createElement('div', 'message-bar',
|
||||||
|
enter,
|
||||||
|
createElementInit('div', div => div.style.textAlign = 'right',
|
||||||
|
createElementInit('button', button => {
|
||||||
|
button.className = 'roundbtn';
|
||||||
|
button.style.backgroundColor = 'rgb(19, 150, 204)';
|
||||||
|
button.appendChild(createIcon('fa-solid', 'paper-plane'));
|
||||||
|
setTooltip(button, r('sendMessage', 'Send Message'));
|
||||||
|
button.addEventListener('click', () => {
|
||||||
|
// TODO: Add text
|
||||||
|
})
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
const message = createElement('div');
|
||||||
|
this.#message = message;
|
||||||
|
container.appendChild(message);
|
||||||
|
return this.#container = container;
|
||||||
|
}
|
||||||
|
|
||||||
|
load(data, contacts) {
|
||||||
|
const children = [];
|
||||||
|
if (data?.length > 0) {
|
||||||
|
for (let comm of data) {
|
||||||
|
const div = document.createElement('div', 'txtdiv');
|
||||||
|
let name;
|
||||||
|
if (comm.IsReply) {
|
||||||
|
const email = isEmail(comm.Sender);
|
||||||
|
const c = contacts.find(c => email ?
|
||||||
|
c.Email === comm.Sender :
|
||||||
|
c.MobilePhone === comm.Sender);
|
||||||
|
if (c != null) {
|
||||||
|
name = c.Name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
children[0].style.marginTop = '0';
|
||||||
|
this.#message.append(...children);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default CustomerCommunication;
|
1
lib/app/communications/lib.d.ts
vendored
Normal file
1
lib/app/communications/lib.d.ts
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
export function createBox(title: HTMLElement, functions: HTMLElement[]): HTMLElement
|
15
lib/app/communications/lib.js
Normal file
15
lib/app/communications/lib.js
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
import { createElement, createElementInit } from "../../functions";
|
||||||
|
|
||||||
|
function createBox(title, functions) {
|
||||||
|
const container = createElement('div', 'comm');
|
||||||
|
const header = createElement('div', 'title-bar',
|
||||||
|
title,
|
||||||
|
createElement('div', 'title-functions', ...functions)
|
||||||
|
);
|
||||||
|
container.appendChild(header);
|
||||||
|
return container;
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
createBox
|
||||||
|
}
|
242
lib/app/communications/style.scss
Normal file
242
lib/app/communications/style.scss
Normal file
@ -0,0 +1,242 @@
|
|||||||
|
:root {
|
||||||
|
--dark-fore-color: #fff;
|
||||||
|
--dark-fore-opacity-color: rgba(255, 255, 255, .6);
|
||||||
|
--title-color: #fff;
|
||||||
|
--title-bg-color: rgb(68, 114, 196);
|
||||||
|
--strong-color: #333;
|
||||||
|
--medium-font-size: .875rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.comm {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
width: 320px;
|
||||||
|
background-color: var(--dark-fore-color);
|
||||||
|
border: 1px solid var(--title-bg-color);
|
||||||
|
|
||||||
|
.roundbtn {
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
display: inline-block;
|
||||||
|
box-sizing: border-box;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 7px;
|
||||||
|
margin-left: 10px;
|
||||||
|
fill: var(--dark-fore-color);
|
||||||
|
border-radius: 15px;
|
||||||
|
border: none;
|
||||||
|
outline: none;
|
||||||
|
transition: background-color .2s;
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--dark-fore-opacity-color);
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
opacity: .6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.disabled {
|
||||||
|
fill: lightgray;
|
||||||
|
background-color: transparent !important;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
opacity: unset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
width: 13px;
|
||||||
|
height: 14px;
|
||||||
|
display: block;
|
||||||
|
transition: opacity .2s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-check-box {
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
>input[type="checkbox"] {
|
||||||
|
display: none;
|
||||||
|
|
||||||
|
&:checked~.unchecked {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:checked~.checked {
|
||||||
|
display: unset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
>.unchecked {
|
||||||
|
opacity: .5;
|
||||||
|
}
|
||||||
|
|
||||||
|
>.checked {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.title-bar {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
padding: 5px 0 5px 10px;
|
||||||
|
color: var(--title-color);
|
||||||
|
background-color: var(--title-bg-color);
|
||||||
|
line-height: 24px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
>div {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
>.title-functions {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
display: flex;
|
||||||
|
|
||||||
|
>label {
|
||||||
|
margin: 0 4px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
cursor: pointer;
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
background-color: var(--dark-fore-color);
|
||||||
|
border-radius: 15px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
transition: background-color .2s;
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
fill: var(--strong-color);
|
||||||
|
width: 14px;
|
||||||
|
height: 14px;
|
||||||
|
transition: opacity .2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--dark-fore-opacity-color);
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
opacity: .6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.contact-bar {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
padding: 4px 0;
|
||||||
|
display: flex;
|
||||||
|
border-bottom: 1px solid var(--title-bg-color);
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
margin: 0 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.contact-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
line-height: 22px;
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
margin-right: 6px;
|
||||||
|
fill: var(--strong-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
>span {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
color: var(--strong-color);
|
||||||
|
font-size: var(--medium-font-size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.roundbtn {
|
||||||
|
float: right;
|
||||||
|
margin: 4px 10px 10px;
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
width: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-bar {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
border-bottom: 1px solid var(--title-bg-color);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
>textarea {
|
||||||
|
padding: 10px 10px 0;
|
||||||
|
border: none;
|
||||||
|
height: 70px;
|
||||||
|
}
|
||||||
|
|
||||||
|
>div {
|
||||||
|
padding: 0 10px 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&+div {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
overflow: auto;
|
||||||
|
margin-top: 8px;
|
||||||
|
|
||||||
|
.msgdiv {
|
||||||
|
margin-top: 5px;
|
||||||
|
border-bottom: solid 1px lightgray;
|
||||||
|
padding: 3px 10px 5px;
|
||||||
|
line-height: 1.5rem;
|
||||||
|
white-space: normal;
|
||||||
|
word-break: break-word;
|
||||||
|
overflow: hidden;
|
||||||
|
font-size: .8125rem;
|
||||||
|
color: #333;
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.msgposter {
|
||||||
|
font-weight: bold;
|
||||||
|
|
||||||
|
&+div {
|
||||||
|
line-height: 1.2rem;
|
||||||
|
padding: 8px 20px;
|
||||||
|
background-color: rgb(244, 244, 244);
|
||||||
|
border-radius: 5px;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
text-align: left;
|
||||||
|
word-break: break-word;
|
||||||
|
|
||||||
|
&.txtself {
|
||||||
|
max-width: 240px;
|
||||||
|
margin-right: 10px;
|
||||||
|
background-color: #9eea6a;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.txtother {
|
||||||
|
max-width: 240px;
|
||||||
|
margin-left: 10px;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.msgtime {
|
||||||
|
text-align: right;
|
||||||
|
color: #aaa;
|
||||||
|
font-size: .7rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
2
lib/functions.d.ts
vendored
Normal file
2
lib/functions.d.ts
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
export function createElement<K extends keyof HTMLElementTagNameMap>(tagName: K, className?: string, ...children?: (Node | string)[]): HTMLElementTagNameMap[K];
|
||||||
|
export function createElementInit<K extends keyof HTMLElementTagNameMap>(tagName: K, init?: (element: HTMLElementTagNameMap[K]) => void, ...children?: (Node | string)[]): HTMLElementTagNameMap[K];
|
19
lib/functions.js
Normal file
19
lib/functions.js
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
function createElement(tagName, className, ...children) {
|
||||||
|
return createElementInit(tagName, className && (element => element.className = className), ...children);
|
||||||
|
}
|
||||||
|
|
||||||
|
function createElementInit(tagName, init, ...children) {
|
||||||
|
const element = document.createElement(tagName);
|
||||||
|
if (typeof init === 'function') {
|
||||||
|
init(element);
|
||||||
|
}
|
||||||
|
if (children.length > 0) {
|
||||||
|
element.append(...children);
|
||||||
|
}
|
||||||
|
return element;
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
createElement,
|
||||||
|
createElementInit
|
||||||
|
}
|
3
lib/ui/checkbox.d.ts
vendored
3
lib/ui/checkbox.d.ts
vendored
@ -1,8 +1,9 @@
|
|||||||
interface CheckboxOptions {
|
interface CheckboxOptions {
|
||||||
|
className?: string;
|
||||||
|
enabled?: boolean;
|
||||||
type?: string;
|
type?: string;
|
||||||
label?: string | HTMLElement;
|
label?: string | HTMLElement;
|
||||||
checked?: boolean;
|
checked?: boolean;
|
||||||
isImage?: boolean;
|
|
||||||
imageHeight?: Number;
|
imageHeight?: Number;
|
||||||
checkedNode?: HTMLElement;
|
checkedNode?: HTMLElement;
|
||||||
uncheckedNode?: HTMLElement;
|
uncheckedNode?: HTMLElement;
|
||||||
|
@ -14,7 +14,6 @@
|
|||||||
type?: string;
|
type?: string;
|
||||||
label?: string;
|
label?: string;
|
||||||
checked?: boolean;
|
checked?: boolean;
|
||||||
isImage?: boolean;
|
|
||||||
imageHeight?: Number;
|
imageHeight?: Number;
|
||||||
checkedNode?: HTMLElement;
|
checkedNode?: HTMLElement;
|
||||||
uncheckedNode?: HTMLElement;
|
uncheckedNode?: HTMLElement;
|
||||||
@ -22,6 +21,14 @@
|
|||||||
onchange?: (this: HTMLInputElement, ev: Event) => any;
|
onchange?: (this: HTMLInputElement, ev: Event) => any;
|
||||||
}</pre>
|
}</pre>
|
||||||
</p>
|
</p>
|
||||||
|
<h4>className?: string</h4>
|
||||||
|
<p>
|
||||||
|
复选框的自定义 className
|
||||||
|
</p>
|
||||||
|
<h4>enabled?: boolean</h4>
|
||||||
|
<p>
|
||||||
|
复选框默认是否可用
|
||||||
|
</p>
|
||||||
<h4>type?: string</h4>
|
<h4>type?: string</h4>
|
||||||
<p>
|
<p>
|
||||||
复选框图标的样式,可选值目前有 <code>fa-regular</code>、<code>fa-light</code>、<code>fa-solid</code>
|
复选框图标的样式,可选值目前有 <code>fa-regular</code>、<code>fa-light</code>、<code>fa-solid</code>
|
||||||
@ -34,10 +41,6 @@
|
|||||||
<p>
|
<p>
|
||||||
初始是否选中
|
初始是否选中
|
||||||
</p>
|
</p>
|
||||||
<h4>isImage?: boolean</h4>
|
|
||||||
<p>
|
|
||||||
是否为图片复选框
|
|
||||||
</p>
|
|
||||||
<h4>imageHeight?: Number</h4>
|
<h4>imageHeight?: Number</h4>
|
||||||
<p>
|
<p>
|
||||||
为图片复选框时的图片限制高度
|
为图片复选框时的图片限制高度
|
||||||
|
@ -1,54 +1,51 @@
|
|||||||
|
import { createElement, createElementInit } from "../functions";
|
||||||
import { createIcon } from "./icon";
|
import { createIcon } from "./icon";
|
||||||
|
|
||||||
function fillCheckbox(container, type, label) {
|
function fillCheckbox(container, type, label) {
|
||||||
const layer = document.createElement('layer');
|
container.appendChild(
|
||||||
layer.className = 'check-box-inner';
|
createElement('layer', 'check-box-inner', createIcon(type, 'check'))
|
||||||
layer.appendChild(createIcon(type, 'check'));
|
);
|
||||||
container.appendChild(layer);
|
|
||||||
if (label instanceof HTMLElement) {
|
if (label instanceof HTMLElement) {
|
||||||
container.appendChild(label);
|
container.appendChild(label);
|
||||||
} else if (label?.length > 0) {
|
} else if (label?.length > 0) {
|
||||||
const span = document.createElement('span');
|
container.appendChild(
|
||||||
span.innerText = label;
|
createElementInit('span', span => span.innerText = label)
|
||||||
container.appendChild(span);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function createCheckbox(opts = {}) {
|
function createCheckbox(opts = {}) {
|
||||||
const container = document.createElement('label');
|
const container = createElement('label', 'checkbox-wrapper',
|
||||||
container.className = 'checkbox-wrapper';
|
createElementInit('input', input => {
|
||||||
const input = document.createElement('input');
|
input.setAttribute('type', 'checkbox');
|
||||||
input.setAttribute('type', 'checkbox');
|
if (opts.checked === true) {
|
||||||
if (opts.checked === true) {
|
input.checked = true;
|
||||||
input.checked = true;
|
}
|
||||||
|
if (opts.enabled === false) {
|
||||||
|
input.disabled = true;
|
||||||
|
}
|
||||||
|
if (opts.customerAttributes != null) {
|
||||||
|
for (let entry of Object.entries(opts.customerAttributes)) {
|
||||||
|
input.setAttribute(entry[0], entry[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (typeof opts.onchange === 'function') {
|
||||||
|
input.addEventListener('change', opts.onchange);
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
if (opts.className) {
|
||||||
|
container.classList.add(opts.className);
|
||||||
}
|
}
|
||||||
if (opts.customerAttributes != null) {
|
if (opts.checkedNode != null && opts.uncheckedNode != null) {
|
||||||
for (let entry of Object.entries(opts.customerAttributes)) {
|
|
||||||
input.setAttribute(entry[0], entry[1]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (typeof opts.onchange === 'function') {
|
|
||||||
input.addEventListener('change', opts.onchange);
|
|
||||||
}
|
|
||||||
container.appendChild(input);
|
|
||||||
if (opts.isImage) {
|
|
||||||
container.classList.add('checkbox-image');
|
container.classList.add('checkbox-image');
|
||||||
let height = opts.imageHeight;
|
let height = opts.imageHeight;
|
||||||
if (isNaN(height) || height <= 0) {
|
if (isNaN(height) || height <= 0) {
|
||||||
height = 14;
|
height = 14;
|
||||||
}
|
}
|
||||||
if (opts.checkedNode != null) {
|
opts.checkedNode.classList.add('checked');
|
||||||
if (!opts.checkedNode.classList.contains('checked')) {
|
container.appendChild(opts.checkedNode);
|
||||||
opts.checkedNode.classList.add('checked');
|
opts.uncheckedNode.classList.add('unchecked');
|
||||||
}
|
container.appendChild(opts.uncheckedNode);
|
||||||
container.appendChild(opts.checkedNode);
|
|
||||||
}
|
|
||||||
if (opts.uncheckedNode != null) {
|
|
||||||
if (!opts.uncheckedNode.classList.contains('unchecked')) {
|
|
||||||
opts.uncheckedNode.classList.add('unchecked');
|
|
||||||
}
|
|
||||||
container.appendChild(opts.uncheckedNode);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
fillCheckbox(container, opts.type || 'fa-regular', opts.label);
|
fillCheckbox(container, opts.type || 'fa-regular', opts.label);
|
||||||
}
|
}
|
||||||
@ -91,7 +88,7 @@ function resolveCheckbox(container = document.body, legacy) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (label == null) {
|
if (label == null) {
|
||||||
label = document.createElement('label');
|
label = createElement('label');
|
||||||
chk.parentElement.insertBefore(label, chk);
|
chk.parentElement.insertBefore(label, chk);
|
||||||
} else {
|
} else {
|
||||||
text = label.innerText;
|
text = label.innerText;
|
||||||
@ -118,7 +115,7 @@ function resolveCheckbox(container = document.body, legacy) {
|
|||||||
box.removeAttribute('data-type');
|
box.removeAttribute('data-type');
|
||||||
box.removeAttribute('data-label');
|
box.removeAttribute('data-label');
|
||||||
}
|
}
|
||||||
const input = document.createElement('input');
|
const input = createElement('input');
|
||||||
const id = box.dataset.id;
|
const id = box.dataset.id;
|
||||||
if (id?.length > 0) {
|
if (id?.length > 0) {
|
||||||
input.id = id;
|
input.id = id;
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
import { r } from "../utility/lgres";
|
import { r } from "../utility/lgres";
|
||||||
import { contains, nullOrEmpty } from "../utility/strings";
|
import { contains, nullOrEmpty } from "../utility/strings";
|
||||||
import { global, isPositive } from "../utility";
|
import { global, isPositive } from "../utility";
|
||||||
|
import { createElement } from "../functions";
|
||||||
import { createCheckbox } from "./checkbox";
|
import { createCheckbox } from "./checkbox";
|
||||||
import { createIcon } from "./icon"
|
import { createIcon } from "./icon"
|
||||||
|
|
||||||
@ -19,7 +20,7 @@ if (dropdownGlobal == null) {
|
|||||||
configurable: false,
|
configurable: false,
|
||||||
enumerable: false,
|
enumerable: false,
|
||||||
value: function () {
|
value: function () {
|
||||||
const panel = document.querySelector('.dropdown-wrapper .dropdown-panel.active');
|
const panel = document.querySelector('.dropdown-wrapper .dropdown-box.active');
|
||||||
if (panel == null) {
|
if (panel == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -39,7 +40,7 @@ if (dropdownGlobal == null) {
|
|||||||
document.addEventListener('mousedown', e => {
|
document.addEventListener('mousedown', e => {
|
||||||
let parent = e.target;
|
let parent = e.target;
|
||||||
while (parent != null) {
|
while (parent != null) {
|
||||||
if (parent.classList.contains('dropdown-panel')) {
|
if (parent.classList.contains('dropdown-box')) {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -110,16 +111,14 @@ class Dropdown {
|
|||||||
const options = this.#options;
|
const options = this.#options;
|
||||||
|
|
||||||
// wrapper
|
// wrapper
|
||||||
const wrapper = document.createElement('div');
|
const wrapper = createElement('div', 'dropdown-wrapper');
|
||||||
const dropId = String(Math.random()).substring(2);
|
const dropId = String(Math.random()).substring(2);
|
||||||
wrapper.dataset.dropId = dropId;
|
wrapper.dataset.dropId = dropId;
|
||||||
wrapper.className = 'dropdown-wrapper';
|
|
||||||
dropdownGlobal[dropId] = this;
|
dropdownGlobal[dropId] = this;
|
||||||
this.#wrapper = wrapper;
|
this.#wrapper = wrapper;
|
||||||
|
|
||||||
// header
|
// header
|
||||||
const header = document.createElement('div');
|
const header = createElement('div', 'dropdown-header');
|
||||||
header.className = 'dropdown-header';
|
|
||||||
header.addEventListener('click', () => {
|
header.addEventListener('click', () => {
|
||||||
if (this.disabled) {
|
if (this.disabled) {
|
||||||
return;
|
return;
|
||||||
@ -137,8 +136,7 @@ class Dropdown {
|
|||||||
// label or input
|
// label or input
|
||||||
let label;
|
let label;
|
||||||
if (options.input) {
|
if (options.input) {
|
||||||
label = document.createElement('input');
|
label = createElement('input', 'dropdown-text');
|
||||||
label.className = 'dropdown-text';
|
|
||||||
label.setAttribute('type', 'text');
|
label.setAttribute('type', 'text');
|
||||||
options.placeholder && label.setAttribute('placeholder', options.placeholder);
|
options.placeholder && label.setAttribute('placeholder', options.placeholder);
|
||||||
isPositive(options.maxlength) && label.setAttribute('maxlength', options.maxlength);
|
isPositive(options.maxlength) && label.setAttribute('maxlength', options.maxlength);
|
||||||
@ -153,8 +151,7 @@ class Dropdown {
|
|||||||
label.addEventListener('mousedown', e => this.#expanded && e.stopPropagation());
|
label.addEventListener('mousedown', e => this.#expanded && e.stopPropagation());
|
||||||
} else {
|
} else {
|
||||||
isPositive(options.tabindex) && header.setAttribute('tabindex', options.tabindex);
|
isPositive(options.tabindex) && header.setAttribute('tabindex', options.tabindex);
|
||||||
label = document.createElement('label');
|
label = createElement('label', 'dropdown-text');
|
||||||
label.className = 'dropdown-text';
|
|
||||||
}
|
}
|
||||||
this.#label = label;
|
this.#label = label;
|
||||||
if (options.multiselect) {
|
if (options.multiselect) {
|
||||||
@ -167,10 +164,7 @@ class Dropdown {
|
|||||||
} else if (options.selected != null) {
|
} else if (options.selected != null) {
|
||||||
this.select(options.selected, true);
|
this.select(options.selected, true);
|
||||||
}
|
}
|
||||||
header.appendChild(label);
|
header.append(label, createElement('label', 'dropdown-caret'));
|
||||||
const caret = document.createElement('label');
|
|
||||||
caret.className = 'dropdown-caret';
|
|
||||||
header.appendChild(caret);
|
|
||||||
wrapper.appendChild(header);
|
wrapper.appendChild(header);
|
||||||
|
|
||||||
this.disabled = options.disabled || false;
|
this.disabled = options.disabled || false;
|
||||||
@ -285,16 +279,13 @@ class Dropdown {
|
|||||||
|
|
||||||
#dropdown(flag = true) {
|
#dropdown(flag = true) {
|
||||||
const options = this.#options;
|
const options = this.#options;
|
||||||
const textkey = options.textkey;
|
|
||||||
let panel = this.#container;
|
let panel = this.#container;
|
||||||
if (panel == null) {
|
if (panel == null) {
|
||||||
panel = document.createElement('div');
|
panel = createElement('div', 'dropdown-box');
|
||||||
panel.className = 'dropdown-panel';
|
|
||||||
// search box
|
// search box
|
||||||
if (!options.input && options.search) {
|
if (!options.input && options.search) {
|
||||||
const search = document.createElement('div');
|
const search = createElement('div', 'dropdown-search');
|
||||||
search.className = 'dropdown-search';
|
const input = createElement('input');
|
||||||
const input = document.createElement('input');
|
|
||||||
input.setAttribute('type', 'text');
|
input.setAttribute('type', 'text');
|
||||||
isPositive(options.tabindex) && input.setAttribute('tabindex', options.tabindex);
|
isPositive(options.tabindex) && input.setAttribute('tabindex', options.tabindex);
|
||||||
!nullOrEmpty(options.searchplaceholder) && input.setAttribute('placeholder', options.searchplaceholder);
|
!nullOrEmpty(options.searchplaceholder) && input.setAttribute('placeholder', options.searchplaceholder);
|
||||||
@ -303,13 +294,11 @@ class Dropdown {
|
|||||||
const source = filterSource(options.searchkeys, options.textkey, key, this.source);
|
const source = filterSource(options.searchkeys, options.textkey, key, this.source);
|
||||||
this.#filllist(source);
|
this.#filllist(source);
|
||||||
})
|
})
|
||||||
search.appendChild(input);
|
search.append(input, createIcon('fa-light', 'search'));
|
||||||
search.appendChild(createIcon('fa-light', 'search'));
|
|
||||||
panel.appendChild(search);
|
panel.appendChild(search);
|
||||||
}
|
}
|
||||||
// list
|
// list
|
||||||
const list = document.createElement('ul');
|
const list = createElement('ul', 'dropdown-list');
|
||||||
list.className = 'dropdown-list';
|
|
||||||
if (!this.multiselect) {
|
if (!this.multiselect) {
|
||||||
list.addEventListener('click', e => {
|
list.addEventListener('click', e => {
|
||||||
let li = e.target;
|
let li = e.target;
|
||||||
@ -364,15 +353,16 @@ class Dropdown {
|
|||||||
const multiselect = this.multiselect;
|
const multiselect = this.multiselect;
|
||||||
const allchecked = this.#allChecked;
|
const allchecked = this.#allChecked;
|
||||||
if (multiselect) {
|
if (multiselect) {
|
||||||
const liall = document.createElement('li');
|
list.appendChild(
|
||||||
const boxall = createCheckbox({
|
createElement('li', null,
|
||||||
label: r('allItem', '( All )'),
|
createCheckbox({
|
||||||
checked: allchecked,
|
label: r('allItem', '( All )'),
|
||||||
customerAttributes: { 'isall': '1' },
|
checked: allchecked,
|
||||||
onchange: e => this.#triggerselect(e.target)
|
customerAttributes: { 'isall': '1' },
|
||||||
});
|
onchange: e => this.#triggerselect(e.target)
|
||||||
liall.appendChild(boxall);
|
})
|
||||||
list.appendChild(liall);
|
)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
// TODO: virtual mode
|
// TODO: virtual mode
|
||||||
const valuekey = this.#options.valuekey;
|
const valuekey = this.#options.valuekey;
|
||||||
@ -383,7 +373,7 @@ class Dropdown {
|
|||||||
let scrolled;
|
let scrolled;
|
||||||
source.slice(0, 200).forEach((item, i) => {
|
source.slice(0, 200).forEach((item, i) => {
|
||||||
const val = item[valuekey];
|
const val = item[valuekey];
|
||||||
const li = document.createElement('li');
|
const li = createElement('li');
|
||||||
li.dataset.value = val;
|
li.dataset.value = val;
|
||||||
li.setAttribute('title', item[textkey]);
|
li.setAttribute('title', item[textkey]);
|
||||||
let label;
|
let label;
|
||||||
@ -394,7 +384,7 @@ class Dropdown {
|
|||||||
if (multiselect) {
|
if (multiselect) {
|
||||||
const selected = selectedlist.some(s => s[valuekey] === val);
|
const selected = selectedlist.some(s => s[valuekey] === val);
|
||||||
if (label == null) {
|
if (label == null) {
|
||||||
label = document.createElement('span');
|
label = createElement('span');
|
||||||
label.innerText = item[textkey];
|
label.innerText = item[textkey];
|
||||||
}
|
}
|
||||||
const box = createCheckbox({
|
const box = createCheckbox({
|
||||||
|
@ -170,8 +170,6 @@
|
|||||||
};
|
};
|
||||||
|
|
||||||
const iconCol = {
|
const iconCol = {
|
||||||
...Grid.GridColumn,
|
|
||||||
|
|
||||||
create() {
|
create() {
|
||||||
const a = document.createElement('a');
|
const a = document.createElement('a');
|
||||||
a.target = '_blank';
|
a.target = '_blank';
|
||||||
@ -186,7 +184,8 @@
|
|||||||
},
|
},
|
||||||
setEnabled(element, enabled) {
|
setEnabled(element, enabled) {
|
||||||
element.style.display = enabled === false ? 'none' : '';
|
element.style.display = enabled === false ? 'none' : '';
|
||||||
}
|
},
|
||||||
|
setStyle: Grid.GridColumn.setStyle
|
||||||
};
|
};
|
||||||
|
|
||||||
// grid.readonly = true;
|
// grid.readonly = true;
|
||||||
@ -205,7 +204,8 @@
|
|||||||
{ value: 'broken', text: 'Broken' },
|
{ value: 'broken', text: 'Broken' },
|
||||||
{ value: 'running', text: 'Running' }
|
{ value: 'running', text: 'Running' }
|
||||||
],
|
],
|
||||||
enabled: 'enabled'
|
enabled: 'enabled',
|
||||||
|
onchanged: (item, value) => console.log('dropdown changed', item, value)
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
key: 'c2b',
|
key: 'c2b',
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { global, isPositive, isMobile, throttle, truncate } from "../utility";
|
import { global, isPositive, isMobile, throttle, truncate } from "../utility";
|
||||||
import { nullOrEmpty } from "../utility/strings";
|
import { nullOrEmpty } from "../utility/strings";
|
||||||
import { r } from "../utility/lgres";
|
import { r } from "../utility/lgres";
|
||||||
|
import { createElement, createElementInit } from "../functions";
|
||||||
import { createIcon } from "./icon";
|
import { createIcon } from "./icon";
|
||||||
import { createCheckbox } from "./checkbox";
|
import { createCheckbox } from "./checkbox";
|
||||||
import { setTooltip } from "./tooltip";
|
import { setTooltip } from "./tooltip";
|
||||||
@ -41,7 +42,7 @@ function indexOfParent(target) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridColumn {
|
class GridColumn {
|
||||||
static create() { return document.createElement('span') }
|
static create() { return createElement('span') }
|
||||||
|
|
||||||
static setValue(element, val) { element.innerText = val }
|
static setValue(element, val) { element.innerText = val }
|
||||||
|
|
||||||
@ -54,7 +55,7 @@ class GridColumn {
|
|||||||
|
|
||||||
class GridInputColumn extends GridColumn {
|
class GridInputColumn extends GridColumn {
|
||||||
static createEdit(trigger) {
|
static createEdit(trigger) {
|
||||||
const input = document.createElement('input');
|
const input = createElement('input');
|
||||||
input.setAttribute('type', 'text');
|
input.setAttribute('type', 'text');
|
||||||
if (typeof trigger === 'function') {
|
if (typeof trigger === 'function') {
|
||||||
input.addEventListener('change', trigger);
|
input.addEventListener('change', trigger);
|
||||||
@ -157,12 +158,7 @@ class GridCheckboxColumn extends GridColumn {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GridIconColumn extends GridColumn {
|
class GridIconColumn extends GridColumn {
|
||||||
static create() {
|
static create() { return createElement('span', 'icon') }
|
||||||
const element = document.createElement('span');
|
|
||||||
element.className = 'icon';
|
|
||||||
// element.appendChild(document.createElement('layer'));
|
|
||||||
return element;
|
|
||||||
}
|
|
||||||
|
|
||||||
static setValue(element, val, item, col) {
|
static setValue(element, val, item, col) {
|
||||||
let type = col.iconType;
|
let type = col.iconType;
|
||||||
@ -351,8 +347,7 @@ class Grid {
|
|||||||
throw new Error('no specified parent.');
|
throw new Error('no specified parent.');
|
||||||
}
|
}
|
||||||
this.#parent = container;
|
this.#parent = container;
|
||||||
const grid = document.createElement('div');
|
const grid = createElement('div', 'grid');
|
||||||
grid.className = 'grid';
|
|
||||||
grid.setAttribute('tabindex', 0);
|
grid.setAttribute('tabindex', 0);
|
||||||
grid.addEventListener('keydown', e => {
|
grid.addEventListener('keydown', e => {
|
||||||
let index = this.selectedIndex;
|
let index = this.selectedIndex;
|
||||||
@ -382,8 +377,7 @@ class Grid {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
container.replaceChildren(grid);
|
container.replaceChildren(grid);
|
||||||
const sizer = document.createElement('span');
|
const sizer = createElement('span', 'grid-sizer');
|
||||||
sizer.className = 'grid-sizer';
|
|
||||||
grid.appendChild(sizer);
|
grid.appendChild(sizer);
|
||||||
this.#refs.sizer = sizer;
|
this.#refs.sizer = sizer;
|
||||||
|
|
||||||
@ -394,11 +388,9 @@ class Grid {
|
|||||||
grid.appendChild(body);
|
grid.appendChild(body);
|
||||||
|
|
||||||
// loading
|
// loading
|
||||||
const loading = document.createElement('div');
|
const loading = createElement('div', 'grid-loading',
|
||||||
loading.className = 'grid-loading';
|
createElement('div', null, createIcon('fa-regular', 'spinner-third'))
|
||||||
const loadingHolder = document.createElement('div');
|
);
|
||||||
loadingHolder.appendChild(createIcon('fa-regular', 'spinner-third'));
|
|
||||||
loading.appendChild(loadingHolder);
|
|
||||||
this.#refs.loading = loading;
|
this.#refs.loading = loading;
|
||||||
grid.appendChild(loading);
|
grid.appendChild(loading);
|
||||||
this.#el = grid;
|
this.#el = grid;
|
||||||
@ -548,14 +540,13 @@ class Grid {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#createHeader() {
|
#createHeader() {
|
||||||
const thead = document.createElement('table');
|
const thead = createElement('table', 'grid-header');
|
||||||
thead.className = 'grid-header';
|
const header = createElement('tr');
|
||||||
const header = document.createElement('tr');
|
|
||||||
thead.appendChild(header);
|
thead.appendChild(header);
|
||||||
const sizer = this.#refs.sizer;
|
const sizer = this.#refs.sizer;
|
||||||
for (let col of this.columns) {
|
for (let col of this.columns) {
|
||||||
if (col.visible === false) {
|
if (col.visible === false) {
|
||||||
const hidden = document.createElement('th');
|
const hidden = createElement('th');
|
||||||
hidden.style.display = 'none';
|
hidden.style.display = 'none';
|
||||||
if (col.sortable !== false) {
|
if (col.sortable !== false) {
|
||||||
hidden.dataset.key = col.key;
|
hidden.dataset.key = col.key;
|
||||||
@ -594,8 +585,7 @@ class Grid {
|
|||||||
};
|
};
|
||||||
this.#set(col.key, 'style', style);
|
this.#set(col.key, 'style', style);
|
||||||
// element
|
// element
|
||||||
const th = document.createElement('th');
|
const th = createElement('th', 'column');
|
||||||
th.className = 'column';
|
|
||||||
th.dataset.key = col.key;
|
th.dataset.key = col.key;
|
||||||
for (let css of Object.entries(style)) {
|
for (let css of Object.entries(style)) {
|
||||||
th.style.setProperty(css[0], css[1]);
|
th.style.setProperty(css[0], css[1]);
|
||||||
@ -608,7 +598,7 @@ class Grid {
|
|||||||
col.orderable = true;
|
col.orderable = true;
|
||||||
th.addEventListener('mousedown', e => this.#onDragStart(e, col));
|
th.addEventListener('mousedown', e => this.#onDragStart(e, col));
|
||||||
}
|
}
|
||||||
const wrapper = document.createElement('div');
|
const wrapper = createElement('div');
|
||||||
th.appendChild(wrapper);
|
th.appendChild(wrapper);
|
||||||
if (!this.readonly && col.enabled !== false && col.allcheck && isCheckbox) {
|
if (!this.readonly && col.enabled !== false && col.allcheck && isCheckbox) {
|
||||||
const check = createCheckbox({
|
const check = createCheckbox({
|
||||||
@ -616,7 +606,7 @@ class Grid {
|
|||||||
});
|
});
|
||||||
wrapper.appendChild(check);
|
wrapper.appendChild(check);
|
||||||
}
|
}
|
||||||
const caption = document.createElement('span');
|
const caption = createElement('span');
|
||||||
if (col.textStyle != null) {
|
if (col.textStyle != null) {
|
||||||
for (let css of Object.entries(col.textStyle)) {
|
for (let css of Object.entries(col.textStyle)) {
|
||||||
caption.style.setProperty(css[0], css[1]);
|
caption.style.setProperty(css[0], css[1]);
|
||||||
@ -626,9 +616,7 @@ class Grid {
|
|||||||
wrapper.appendChild(caption);
|
wrapper.appendChild(caption);
|
||||||
// order arrow
|
// order arrow
|
||||||
if (col.sortable) {
|
if (col.sortable) {
|
||||||
const arrow = document.createElement('layer');
|
th.appendChild(createElement('layer', 'arrow'));
|
||||||
arrow.className = 'arrow';
|
|
||||||
th.appendChild(arrow);
|
|
||||||
}
|
}
|
||||||
// filter
|
// filter
|
||||||
if (col.allowFilter) {
|
if (col.allowFilter) {
|
||||||
@ -636,8 +624,7 @@ class Grid {
|
|||||||
}
|
}
|
||||||
// resize spliter
|
// resize spliter
|
||||||
if (col.resizable !== false) {
|
if (col.resizable !== false) {
|
||||||
const spliter = document.createElement('layer');
|
const spliter = createElement('layer', 'spliter');
|
||||||
spliter.className = 'spliter';
|
|
||||||
spliter.addEventListener('mousedown', e => this.#onResizeStart(e, col));
|
spliter.addEventListener('mousedown', e => this.#onResizeStart(e, col));
|
||||||
spliter.addEventListener('dblclick', e => this.#onAutoResize(e, col));
|
spliter.addEventListener('dblclick', e => this.#onAutoResize(e, col));
|
||||||
th.appendChild(spliter);
|
th.appendChild(spliter);
|
||||||
@ -646,13 +633,9 @@ class Grid {
|
|||||||
// !nullOrEmpty(col.tooltip) && setTooltip(th, col.tooltip);
|
// !nullOrEmpty(col.tooltip) && setTooltip(th, col.tooltip);
|
||||||
header.appendChild(th);
|
header.appendChild(th);
|
||||||
}
|
}
|
||||||
const placeholder = document.createElement('th');
|
const dragger = createElement('div', 'dragger');
|
||||||
const dragger = document.createElement('div');
|
const draggerCursor = createElement('layer', 'dragger-cursor');
|
||||||
dragger.className = 'dragger';
|
header.appendChild(createElement('th', null, dragger, draggerCursor));
|
||||||
const draggerCursor = document.createElement('layer');
|
|
||||||
draggerCursor.className = 'dragger-cursor';
|
|
||||||
placeholder.append(dragger, draggerCursor);
|
|
||||||
header.appendChild(placeholder);
|
|
||||||
|
|
||||||
sizer.replaceChildren();
|
sizer.replaceChildren();
|
||||||
this.#refs.header = header;
|
this.#refs.header = header;
|
||||||
@ -662,8 +645,7 @@ class Grid {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#createBody() {
|
#createBody() {
|
||||||
const body = document.createElement('div');
|
const body = createElement('div', 'grid-body');
|
||||||
body.className = 'grid-body';
|
|
||||||
body.addEventListener('scroll', e => throttle(this.#onScroll, RefreshInterval, this, e), { passive: true });
|
body.addEventListener('scroll', e => throttle(this.#onScroll, RefreshInterval, this, e), { passive: true });
|
||||||
const cols = this.columns;
|
const cols = this.columns;
|
||||||
let width = 1;
|
let width = 1;
|
||||||
@ -673,7 +655,7 @@ class Grid {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// body container
|
// body container
|
||||||
const bodyContainer = document.createElement('div');
|
const bodyContainer = createElement('div');
|
||||||
bodyContainer.style.position = 'relative';
|
bodyContainer.style.position = 'relative';
|
||||||
bodyContainer.style.minWidth = '100%';
|
bodyContainer.style.minWidth = '100%';
|
||||||
bodyContainer.style.minHeight = '1px';
|
bodyContainer.style.minHeight = '1px';
|
||||||
@ -682,8 +664,7 @@ class Grid {
|
|||||||
}
|
}
|
||||||
body.appendChild(bodyContainer);
|
body.appendChild(bodyContainer);
|
||||||
// body content
|
// body content
|
||||||
const bodyContent = document.createElement('table');
|
const bodyContent = createElement('table', 'grid-body-content');
|
||||||
bodyContent.className = 'grid-body-content';
|
|
||||||
bodyContent.addEventListener('mousedown', e => {
|
bodyContent.addEventListener('mousedown', e => {
|
||||||
let [parent, target] = this.#getRowTarget(e.target);
|
let [parent, target] = this.#getRowTarget(e.target);
|
||||||
const rowIndex = indexOfParent(parent);
|
const rowIndex = indexOfParent(parent);
|
||||||
@ -698,8 +679,7 @@ class Grid {
|
|||||||
// this.#adjustRows();
|
// this.#adjustRows();
|
||||||
// events
|
// events
|
||||||
if (!this.holderDisabled) {
|
if (!this.holderDisabled) {
|
||||||
const holder = document.createElement('div');
|
const holder = createElement('div', 'grid-hover-holder');
|
||||||
holder.className = 'grid-hover-holder';
|
|
||||||
holder.addEventListener('mousedown', e => {
|
holder.addEventListener('mousedown', e => {
|
||||||
const keyid = e.currentTarget.keyid;
|
const keyid = e.currentTarget.keyid;
|
||||||
if (keyid == null) {
|
if (keyid == null) {
|
||||||
@ -730,12 +710,11 @@ class Grid {
|
|||||||
count -= exists;
|
count -= exists;
|
||||||
if (count > 0) {
|
if (count > 0) {
|
||||||
for (let i = 0; i < count; i += 1) {
|
for (let i = 0; i < count; i += 1) {
|
||||||
const row = document.createElement('tr');
|
const row = createElement('tr', 'grid-row');
|
||||||
row.className = 'grid-row';
|
|
||||||
// row.addEventListener('mousedown', e => this.#onRowClicked(e, exists + i));
|
// row.addEventListener('mousedown', e => this.#onRowClicked(e, exists + i));
|
||||||
// row.addEventListener('dblclick', e => this.#onRowDblClicked(e));
|
// row.addEventListener('dblclick', e => this.#onRowDblClicked(e));
|
||||||
cols.forEach((col, j) => {
|
cols.forEach((col, j) => {
|
||||||
const cell = document.createElement('td');
|
const cell = createElement('td');
|
||||||
if (col.visible !== false) {
|
if (col.visible !== false) {
|
||||||
cell.keyid = ((exists + i) << MaxColumnBit) | j;
|
cell.keyid = ((exists + i) << MaxColumnBit) | j;
|
||||||
const style = this.#get(col.key, 'style');
|
const style = this.#get(col.key, 'style');
|
||||||
@ -771,7 +750,7 @@ class Grid {
|
|||||||
}
|
}
|
||||||
row.appendChild(cell);
|
row.appendChild(cell);
|
||||||
});
|
});
|
||||||
row.appendChild(document.createElement('td'));
|
row.appendChild(createElement('td'));
|
||||||
content.appendChild(row);
|
content.appendChild(row);
|
||||||
}
|
}
|
||||||
} else if (count < 0) {
|
} else if (count < 0) {
|
||||||
|
2
lib/ui/icon.d.ts
vendored
2
lib/ui/icon.d.ts
vendored
@ -1,2 +1,2 @@
|
|||||||
export function createIcon(type: string, id: string): SVGSVGElement
|
export function createIcon(type: string, id: string, style?: { [key: string]: string }): SVGSVGElement
|
||||||
export function resolveIcon(container: HTMLElement): HTMLElement
|
export function resolveIcon(container: HTMLElement): HTMLElement
|
@ -6,7 +6,7 @@
|
|||||||
标签到指定的图标元素。
|
标签到指定的图标元素。
|
||||||
</p>
|
</p>
|
||||||
<h2>createIcon</h2>
|
<h2>createIcon</h2>
|
||||||
<code>function createIcon(type: string, id: string): SVGElement</code>
|
<code>function createIcon(type: string, id: string, style?: { [key: string]: string }): SVGElement</code>
|
||||||
<h3>type: string</h3>
|
<h3>type: string</h3>
|
||||||
<p>
|
<p>
|
||||||
图标类型,可选值目前有 <code>fa-regular</code>、<code>fa-light</code>、<code>fa-solid</code>
|
图标类型,可选值目前有 <code>fa-regular</code>、<code>fa-light</code>、<code>fa-solid</code>
|
||||||
@ -16,6 +16,10 @@
|
|||||||
图形 id,例如
|
图形 id,例如
|
||||||
<code>user-edit</code>、<code>address-card</code>、<code>frog</code>……
|
<code>user-edit</code>、<code>address-card</code>、<code>frog</code>……
|
||||||
</p>
|
</p>
|
||||||
|
<h3>style?: { [key: string]: string }</h3>
|
||||||
|
<p>
|
||||||
|
自定义样式的对象
|
||||||
|
</p>
|
||||||
<h2>resolveIcon</h2>
|
<h2>resolveIcon</h2>
|
||||||
<code>function resolveIcon(container: HTMLElement): HTMLElement</code>
|
<code>function resolveIcon(container: HTMLElement): HTMLElement</code>
|
||||||
<h3>container: HTMLElement</h3>
|
<h3>container: HTMLElement</h3>
|
||||||
|
@ -9,9 +9,14 @@ function createUse(type, id) {
|
|||||||
return use;
|
return use;
|
||||||
}
|
}
|
||||||
|
|
||||||
function createIcon(type, id) {
|
function createIcon(type, id, style) {
|
||||||
const svg = document.createElementNS(svgns, 'svg');
|
const svg = document.createElementNS(svgns, 'svg');
|
||||||
svg.appendChild(createUse(type, id));
|
svg.appendChild(createUse(type, id));
|
||||||
|
if (style != null) {
|
||||||
|
for (let css of Object.entries(style)) {
|
||||||
|
svg.style.setProperty(css[0], css[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
return svg;
|
return svg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
2
lib/ui/tooltip.d.ts
vendored
2
lib/ui/tooltip.d.ts
vendored
@ -1,2 +1,2 @@
|
|||||||
export function setTooltip(container: HTMLElement, content: string | HTMLElement): void
|
export function setTooltip(container: HTMLElement, content: string | HTMLElement): HTMLElement
|
||||||
export function resolveTooltip(container?: HTMLElement): HTMLElement
|
export function resolveTooltip(container?: HTMLElement): HTMLElement
|
@ -1,23 +1,28 @@
|
|||||||
function setTooltip(container, content) {
|
import { createElement, createElementInit } from "../functions";
|
||||||
const wrapper = document.createElement('div');
|
|
||||||
wrapper.className = 'tooltip-wrapper tooltip-color';
|
|
||||||
wrapper.style.visibility = 'hidden';
|
|
||||||
wrapper.style.opacity = 0;
|
|
||||||
const pointer = document.createElement('div');
|
|
||||||
pointer.className = 'tooltip-pointer tooltip-color';
|
|
||||||
const curtain = document.createElement('div');
|
|
||||||
curtain.className = 'tooltip-curtain tooltip-color';
|
|
||||||
wrapper.append(pointer, curtain);
|
|
||||||
|
|
||||||
const cnt = document.createElement('div');
|
function setTooltip(container, content) {
|
||||||
cnt.className = 'tooltip-content';
|
const tip = container.querySelector('.tooltip-wrapper');
|
||||||
if (content instanceof HTMLElement) {
|
if (tip != null) {
|
||||||
cnt.appendChild(content);
|
tip.remove();
|
||||||
} else {
|
|
||||||
cnt.innerText = content;
|
|
||||||
}
|
}
|
||||||
wrapper.appendChild(cnt);
|
const wrapper = createElementInit('div', wrapper => {
|
||||||
container.insertAdjacentElement('afterend', wrapper);
|
wrapper.className = 'tooltip-wrapper tooltip-color';
|
||||||
|
wrapper.style.visibility = 'hidden';
|
||||||
|
wrapper.style.opacity = 0;
|
||||||
|
},
|
||||||
|
createElement('div', 'tooltip-pointer tooltip-color'),
|
||||||
|
createElement('div', 'tooltip-curtain tooltip-color'),
|
||||||
|
createElementInit('div', cnt => {
|
||||||
|
cnt.className = 'tooltip-content';
|
||||||
|
if (content instanceof HTMLElement) {
|
||||||
|
cnt.appendChild(content);
|
||||||
|
} else {
|
||||||
|
cnt.innerText = content;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
// container.insertAdjacentElement('afterend', wrapper);
|
||||||
|
container.appendChild(wrapper);
|
||||||
|
|
||||||
let tid;
|
let tid;
|
||||||
container.addEventListener('mouseenter', () => {
|
container.addEventListener('mouseenter', () => {
|
||||||
@ -56,6 +61,7 @@ function setTooltip(container, content) {
|
|||||||
wrapper.style.opacity = 0;
|
wrapper.style.opacity = 0;
|
||||||
}, 300);
|
}, 300);
|
||||||
});
|
});
|
||||||
|
return container;
|
||||||
}
|
}
|
||||||
|
|
||||||
function resolveTooltip(container = document.body) {
|
function resolveTooltip(container = document.body) {
|
||||||
|
@ -31,6 +31,14 @@ function truncate(v) {
|
|||||||
return (v > 0 ? Math.floor : Math.ceil)(v);
|
return (v > 0 ? Math.floor : Math.ceil)(v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function isEmail(text) {
|
||||||
|
return /^\w[-\w.+]*@\w+([-.]\w+)*\.\w+([-.]\w+)*$/.test(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
function isPhone(text) {
|
||||||
|
return /^[1-9]\d{9,}$/.test(text);
|
||||||
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
// cookie
|
// cookie
|
||||||
getCookie,
|
getCookie,
|
||||||
@ -55,5 +63,7 @@ export {
|
|||||||
isMobile,
|
isMobile,
|
||||||
// functions
|
// functions
|
||||||
throttle,
|
throttle,
|
||||||
truncate
|
truncate,
|
||||||
|
isEmail,
|
||||||
|
isPhone
|
||||||
}
|
}
|
@ -16,6 +16,34 @@ const libraries = [
|
|||||||
formats: ['umd'],
|
formats: ['umd'],
|
||||||
fileName: (_format, name) => `${name}.min.js`
|
fileName: (_format, name) => `${name}.min.js`
|
||||||
},
|
},
|
||||||
|
rollupOptions: {
|
||||||
|
output: {
|
||||||
|
assetFileNames: info => info.name === 'style.css' ? 'ui.min.css' : info.name
|
||||||
|
}
|
||||||
|
},
|
||||||
|
sourcemap: true,
|
||||||
|
emptyOutDir: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
clearScreen: false,
|
||||||
|
css: {
|
||||||
|
postcss: {
|
||||||
|
plugins: [postcssPresetEnv()]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
build: {
|
||||||
|
lib: {
|
||||||
|
entry: './lib/app.js',
|
||||||
|
name: 'lib-app',
|
||||||
|
formats: ['umd'],
|
||||||
|
fileName: (_format, name) => `${name}.min.js`
|
||||||
|
},
|
||||||
|
rollupOptions: {
|
||||||
|
output: {
|
||||||
|
assetFileNames: info => info.name === 'style.css' ? 'app.min.css' : info.name
|
||||||
|
}
|
||||||
|
},
|
||||||
sourcemap: true,
|
sourcemap: true,
|
||||||
emptyOutDir: false
|
emptyOutDir: false
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user