.
This commit is contained in:
parent
6157f994a9
commit
449196b491
@ -1,7 +1,7 @@
|
|||||||
import "./style.scss";
|
import "./style.scss";
|
||||||
import { createElement, createElementInit } from "../../functions";
|
import { createElement, createElementInit } from "../../functions";
|
||||||
import { r } from "../../utility/lgres";
|
import { r } from "../../utility/lgres";
|
||||||
import { isEmail, isPhone } from "../../utility";
|
import { formatUrl, isEmail, isPhone } from "../../utility";
|
||||||
import { setTooltip } from "../../ui/tooltip";
|
import { setTooltip } from "../../ui/tooltip";
|
||||||
import { createIcon } from "../../ui/icon";
|
import { createIcon } from "../../ui/icon";
|
||||||
import { createCheckbox } from "../../ui/checkbox";
|
import { createCheckbox } from "../../ui/checkbox";
|
||||||
@ -195,32 +195,153 @@ class CustomerCommunication {
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
const message = createElement('div');
|
const message = createElement('div', 'list-bar');
|
||||||
this.#message = message;
|
this.#message = message;
|
||||||
container.appendChild(message);
|
container.appendChild(message);
|
||||||
return this.#container = container;
|
return this.#container = container;
|
||||||
}
|
}
|
||||||
|
|
||||||
load(data, contacts) {
|
load(data, contacts, followers) {
|
||||||
const children = [];
|
const children = [];
|
||||||
if (data?.length > 0) {
|
if (data?.length > 0) {
|
||||||
for (let comm of data) {
|
for (let comm of data) {
|
||||||
const div = document.createElement('div', 'txtdiv');
|
const div = createElement('div', 'item-div');
|
||||||
let name;
|
let name;
|
||||||
if (comm.IsReply) {
|
if (comm.IsReply) {
|
||||||
const email = isEmail(comm.Sender);
|
const c = isEmail(comm.Sender) ?
|
||||||
const c = contacts.find(c => email ?
|
contacts.find(c => c.Email === comm.Sender) :
|
||||||
c.Email === comm.Sender :
|
contacts.find(c => c.MobilePhone === comm.Sender);
|
||||||
c.MobilePhone === comm.Sender);
|
name = c?.Name;
|
||||||
if (c != null) {
|
}
|
||||||
name = c.Name;
|
name ??= comm.IsReply && String(comm.FormatSender) !== '' ? comm.FormatSender : comm.Sender;
|
||||||
|
let sendto = '';
|
||||||
|
if (!comm.IsReply && comm.OriPhoneNumbers?.length > 0) {
|
||||||
|
for (let oriph of comm.OriPhoneNumbers) {
|
||||||
|
let cname;
|
||||||
|
const email = isEmail(oriph);
|
||||||
|
if (contacts?.length > 0) {
|
||||||
|
let c = email ?
|
||||||
|
contacts.find(c => c.Email === oriph) :
|
||||||
|
contacts.find(c => c.MobilePhone === oriph);
|
||||||
|
if (c != null) {
|
||||||
|
cname = `${email ? c.Email : c.MobilePhone} - ${c.Name}`;
|
||||||
|
} else if (followers?.length > 0) {
|
||||||
|
c = email ?
|
||||||
|
followers.find(f => f.Email === oriph) :
|
||||||
|
followers.find(f => f.MobilePhone === oriph);
|
||||||
|
if (c != null) {
|
||||||
|
cname = `${email ? c.Email : c.MobilePhone} - ${c.Name}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sendto += (cname ?? oriph) + '\n';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (sendto !== '') {
|
||||||
|
sendto = r('sendToColon', 'Send To :') + `\n${sendto}`;
|
||||||
|
}
|
||||||
|
div.appendChild(createElementInit('div', div => {
|
||||||
|
div.className = 'item-poster';
|
||||||
|
div.innerText = name;
|
||||||
|
if (!comm.IsReply && sendto?.length > 0) {
|
||||||
|
setTooltip(div, sendto);
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
const content = createElement('div', 'item-content');
|
||||||
|
if (/https?:\/\//i.test(comm.Message)) {
|
||||||
|
content.innerHTML = formatUrl(comm.Message);
|
||||||
|
} else {
|
||||||
|
content.innerText = comm.Message;
|
||||||
|
}
|
||||||
|
if (comm.IsReply) {
|
||||||
|
div.classList.add('item-other');
|
||||||
|
} else {
|
||||||
|
const [status, statusmsg] = this.#getMessageStatus(comm);
|
||||||
|
if (status !== -100) {
|
||||||
|
let statustext;
|
||||||
|
switch (status) {
|
||||||
|
case 0:
|
||||||
|
statustext = r('pending', 'Pending');
|
||||||
|
content.style.backgroundColor = '#ffc107';
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
statustext = r('sent', 'Sent');
|
||||||
|
break;
|
||||||
|
case 9:
|
||||||
|
statustext = r('failed', 'Failed');
|
||||||
|
content.style.backgroundColor = '#ffc107';
|
||||||
|
break;
|
||||||
|
case 10:
|
||||||
|
statustext = r('optOut', 'Opt-Out');
|
||||||
|
content.style.backgroundColor = '#ffc107';
|
||||||
|
break;
|
||||||
|
case 412:
|
||||||
|
statustext = r('landline', 'Landline');
|
||||||
|
content.style.backgroundColor = '#ffc107';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
statustext = r('undelivered', 'Undelivered');
|
||||||
|
content.style.backgroundColor = '#ffc107';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
const divstatus = createElementInit('div', div => {
|
||||||
|
div.className = 'item-status';
|
||||||
|
div.innerText = statustext;
|
||||||
|
if (status == -10) {
|
||||||
|
setTooltip(div, statusmsg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
content.appendChild(divstatus);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
div.append(
|
||||||
|
content,
|
||||||
|
createElementInit('div', div => {
|
||||||
|
div.className = 'item-time';
|
||||||
|
div.innerText = comm.TimeStr;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
children.push(div);
|
||||||
}
|
}
|
||||||
children[0].style.marginTop = '0';
|
children[0].style.marginTop = '0';
|
||||||
this.#message.append(...children);
|
this.#message.append(...children);
|
||||||
|
this.#message.scrollTop = this.#message.scrollHeight
|
||||||
|
// setTimeout(() => this.#message.scrollTop = this.#message.scrollHeight, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#getMessageStatus(comm) {
|
||||||
|
let status = -100; // 没有状态,页面上不显示
|
||||||
|
const ls = [];
|
||||||
|
let statusmsg = '';
|
||||||
|
if (!comm.StatusIncorrect && comm.Participator?.length > 0) {
|
||||||
|
for (let p of comm.Participator) {
|
||||||
|
if (!isEmail(p.CustomerNumber)) {
|
||||||
|
if (ls.indexOf(p.Status) < 0) {
|
||||||
|
ls.push(p.Status);
|
||||||
|
}
|
||||||
|
if (statusmsg.length > 0) {
|
||||||
|
statusmsg += '\n';
|
||||||
|
}
|
||||||
|
statusmsg += `${p.CustomerNumber}: `;
|
||||||
|
const st = ({
|
||||||
|
0: r('undelivered', 'Undelivered'),
|
||||||
|
1: r('sent', 'Sent'),
|
||||||
|
9: r('failed', 'Failed')
|
||||||
|
})[p.Status];
|
||||||
|
if (st != null) {
|
||||||
|
statusmsg += st;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (ls.length === 1) {
|
||||||
|
status = ls[0];
|
||||||
|
} else if (ls.length > 1) {
|
||||||
|
status = -10; // 多种状态
|
||||||
|
}
|
||||||
|
return [status, statusmsg];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default CustomerCommunication;
|
export default CustomerCommunication;
|
@ -183,59 +183,87 @@
|
|||||||
>div {
|
>div {
|
||||||
padding: 0 10px 10px;
|
padding: 0 10px 10px;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&+div {
|
.list-bar {
|
||||||
flex: 1 1 auto;
|
flex: 1 1 auto;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
|
|
||||||
.msgdiv {
|
.item-div {
|
||||||
margin-top: 5px;
|
margin-top: 5px;
|
||||||
border-bottom: solid 1px lightgray;
|
border-bottom: solid 1px lightgray;
|
||||||
padding: 3px 10px 5px;
|
padding: 3px 10px 5px;
|
||||||
line-height: 1.5rem;
|
line-height: 1.5rem;
|
||||||
white-space: normal;
|
white-space: normal;
|
||||||
word-break: break-word;
|
word-break: break-word;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
font-size: .8125rem;
|
font-size: .8125rem;
|
||||||
color: #333;
|
color: #333;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
&:last-child {
|
&:last-child {
|
||||||
border-bottom: none;
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-poster {
|
||||||
|
font-weight: bold;
|
||||||
|
align-self: flex-end;
|
||||||
|
|
||||||
|
.tooltip-wrapper>.tooltip-content {
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-content {
|
||||||
|
line-height: 1.2rem;
|
||||||
|
padding: 8px 20px;
|
||||||
|
border-radius: 5px;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
word-break: break-word;
|
||||||
|
max-width: 240px;
|
||||||
|
margin-right: 10px;
|
||||||
|
background-color: #9eea6a;
|
||||||
|
align-self: flex-end;
|
||||||
|
|
||||||
|
a>svg {
|
||||||
|
width: 13px;
|
||||||
|
height: 13px;
|
||||||
|
fill: #2140fb;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
border-bottom: 1px solid;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.msgposter {
|
.item-status {
|
||||||
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;
|
text-align: right;
|
||||||
color: #aaa;
|
margin-top: 3px;
|
||||||
font-size: .7rem;
|
font-weight: bold;
|
||||||
|
margin-right: -12px;
|
||||||
|
font-size: 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-time {
|
||||||
|
align-self: flex-end;
|
||||||
|
color: #aaa;
|
||||||
|
font-size: .7rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.item-other {
|
||||||
|
|
||||||
|
.item-poster,
|
||||||
|
.item-content {
|
||||||
|
align-self: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-content {
|
||||||
|
background-color: rgb(244, 244, 244);
|
||||||
|
margin-right: unset;
|
||||||
|
margin-left: 10px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { getCookie, setCookie, deleteCookie } from "./utility/cookie";
|
import { getCookie, setCookie, deleteCookie } from "./utility/cookie";
|
||||||
import { init, r, lang } from "./utility/lgres";
|
import { init, r, lang } from "./utility/lgres";
|
||||||
import { get, post, upload } from "./utility/request";
|
import { get, post, upload } from "./utility/request";
|
||||||
import { nullOrEmpty, contains, endsWith, padStart } from "./utility/strings";
|
import { nullOrEmpty, contains, endsWith, padStart, formatUrl } from "./utility/strings";
|
||||||
|
|
||||||
let g = typeof globalThis !== 'undefined' ? globalThis : self;
|
let g = typeof globalThis !== 'undefined' ? globalThis : self;
|
||||||
|
|
||||||
@ -57,6 +57,7 @@ export {
|
|||||||
contains,
|
contains,
|
||||||
endsWith,
|
endsWith,
|
||||||
padStart,
|
padStart,
|
||||||
|
formatUrl,
|
||||||
// variables
|
// variables
|
||||||
g as global,
|
g as global,
|
||||||
isPositive,
|
isPositive,
|
||||||
|
3
lib/utility/strings.d.ts
vendored
3
lib/utility/strings.d.ts
vendored
@ -1,4 +1,5 @@
|
|||||||
export function nullOrEmpty(s?: string | any | null): boolean
|
export function nullOrEmpty(s?: string | any | null): boolean
|
||||||
export function contains(s: string, key: string | any, ignoreCase?: boolean): boolean
|
export function contains(s: string, key: string | any, ignoreCase?: boolean): boolean
|
||||||
export function endsWith(s: string, suffix: string): boolean
|
export function endsWith(s: string, suffix: string): boolean
|
||||||
export function padStart(s: string, num: Number, char: string): boolean
|
export function padStart(s: string, num: Number, char: string): boolean
|
||||||
|
export function formatUrl(msg: string): string
|
@ -48,6 +48,12 @@
|
|||||||
<p>
|
<p>
|
||||||
用此字符串填充,使得字符串对齐,默认为 ' '
|
用此字符串填充,使得字符串对齐,默认为 ' '
|
||||||
</p>
|
</p>
|
||||||
|
<h2>formatUrl</h2>
|
||||||
|
<code>function formatUrl(msg: string): string</code>
|
||||||
|
<h3>msg: string</h3>
|
||||||
|
<p>
|
||||||
|
把超链接解析替换为图标
|
||||||
|
</p>
|
||||||
<hr />
|
<hr />
|
||||||
<h2>用法</h2>
|
<h2>用法</h2>
|
||||||
<pre>const util = window["lib-utility"];
|
<pre>const util = window["lib-utility"];
|
||||||
|
@ -29,9 +29,37 @@ function padStart(s, num, char) {
|
|||||||
return (char ?? ' ').repeat(num - s.length);
|
return (char ?? ' ').repeat(num - s.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function formatUrl(msg) {
|
||||||
|
//const urlReg = /(https?:\/\/)?([\da-z\.-]+)\.([a-z\.]{2,6})([\/\w \.-]*)*\/?/ig;
|
||||||
|
//const urlArrray = str.match(urlReg);
|
||||||
|
const p = /(http|ftp|https):\/\/.+?(\s|\r\n|\r|\n|\"|\'|\*|$)/g;
|
||||||
|
const r = msg.match(p);
|
||||||
|
msg = htmlencode(msg);
|
||||||
|
|
||||||
|
if (r?.length > 0) {
|
||||||
|
const rs = [];
|
||||||
|
for (let t of r) {
|
||||||
|
t = t.replace(/["'\r\n ]/g, '');
|
||||||
|
if (rs.indexOf(t) < 0) {
|
||||||
|
rs.push(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let r of rs) {
|
||||||
|
msg = msg.replaceAll(r, '<a target="_blank" href="' + r + '"><svg><use xlink:href="' + ((typeof consts !== 'undefined' && consts.path) ?? '') + 'fonts/fa-regular.svg#link"></use></svg></a>');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return msg
|
||||||
|
.replaceAll('\r\n', '<br/>')
|
||||||
|
.replaceAll('\n', '<br/>')
|
||||||
|
.replaceAll(' ', ' ');
|
||||||
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
nullOrEmpty,
|
nullOrEmpty,
|
||||||
contains,
|
contains,
|
||||||
endsWith,
|
endsWith,
|
||||||
padStart
|
padStart,
|
||||||
|
formatUrl
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user