From d702197a3f08723521087fd4c648edaa908f10d7 Mon Sep 17 00:00:00 2001
From: Tsanie Lily
Date: Thu, 13 Apr 2023 17:36:42 +0800
Subject: [PATCH] communication fix
---
css/grid.scss | 2 +-
css/tooltip.scss | 2 +-
lib/app/communications/contact.js | 95 ++++---
lib/app/communications/customer.js | 384 +++++++++++++++++------------
lib/app/communications/follower.js | 2 +-
lib/app/communications/internal.js | 25 +-
lib/app/communications/style.scss | 4 +-
lib/ui/grid.js | 6 +-
lib/ui/tooltip.d.ts | 2 +-
lib/ui/tooltip.html | 6 +-
lib/ui/tooltip.js | 55 +++--
11 files changed, 365 insertions(+), 218 deletions(-)
diff --git a/css/grid.scss b/css/grid.scss
index 440167f..6ed68c0 100644
--- a/css/grid.scss
+++ b/css/grid.scss
@@ -3,7 +3,7 @@
box-sizing: border-box;
display: flex;
flex-direction: column;
- overflow-x: hidden;
+ overflow: visible;
& {
--hover-bg-color: lightyellow;
diff --git a/css/tooltip.scss b/css/tooltip.scss
index a100dbf..b0cee11 100644
--- a/css/tooltip.scss
+++ b/css/tooltip.scss
@@ -9,7 +9,7 @@
}
.tooltip-wrapper {
- position: fixed;
+ position: absolute;
word-wrap: break-word;
height: auto;
text-align: left;
diff --git a/lib/app/communications/contact.js b/lib/app/communications/contact.js
index 4977dfe..1ec6d82 100644
--- a/lib/app/communications/contact.js
+++ b/lib/app/communications/contact.js
@@ -45,6 +45,38 @@ class Contact {
txt.maxLength = 2000;
txt.style.height = '100px';
});
+ const buttons = [];
+ if (this.#option.company) {
+ buttons.push({
+ text: c == null ? r('addContactRecord', 'Add Contact Record') : r('editContactRecord', 'Edit Contact Record'),
+ trigger: () => {
+ const item = this.prepare();
+ if (item == null) {
+ return false;
+ }
+ item.SaveToCustomer = 1;
+ if (typeof this.#option.onSave === 'function') {
+ return this.#option.onSave.call(this, item, c == null);
+ }
+ }
+ });
+ }
+ buttons.push(
+ {
+ text: r('workOrderOnly', 'Work Order Only'),
+ trigger: () => {
+ const item = this.prepare();
+ if (item == null) {
+ return false;
+ }
+ item.SaveToCustomer = 0;
+ if (typeof this.#option.onSave === 'function') {
+ return this.#option.onSave.call(this, item, c == null);
+ }
+ }
+ },
+ { text: r('cancel', 'Cancel') }
+ );
const popup = createPopup(
c == null ? r('addContact', 'Add Contact') : r('editContact', 'Edit Contact'),
createElement('div', wrapper => {
@@ -76,33 +108,7 @@ class Contact {
contactNotes
)
),
- {
- text: c == null ? r('addContactRecord', 'Add Contact Record') : r('editContactRecord', 'Edit Contact Record'),
- trigger: () => {
- const item = this.prepare();
- if (item == null) {
- return false;
- }
- item.SaveToCustomer = 1;
- if (typeof this.#option.onSave === 'function') {
- return this.#option.onSave.call(this, item, c == null);
- }
- }
- },
- {
- text: r('workOrderOnly', 'Work Order Only'),
- trigger: () => {
- const item = this.prepare();
- if (item == null) {
- return false;
- }
- item.SaveToCustomer = 0;
- if (typeof this.#option.onSave === 'function') {
- return this.#option.onSave.call(this, item, c == null);
- }
- }
- },
- { text: r('cancel', 'Cancel') }
+ ...buttons
)
if (c != null) {
contactName.value = c.Name;
@@ -128,35 +134,42 @@ class Contact {
}
prepare() {
- const item = {
- 'Id': this.#option.contact?.Id,
- 'Name': this.#refs.contactName.value,
- 'ContactPreference': this.#refs.preferences.selected.value,
- 'Email': this.#refs.contactEmail.value,
- 'MobilePhone': this.#refs.contactMobile.value,
- 'OptOut': this.#refs.checkOpt.querySelector('input').checked,
- 'Notes': this.#refs.contactNotes.value
- };
+ const name = this.#refs.contactName.value;
+ const pref = this.#refs.preferences.selected.value;
+ const email = this.#refs.contactEmail.value;
+ const phone = this.#refs.contactMobile.value;
+ const opt = this.#refs.checkOpt.querySelector('input').checked;
+ const notes = this.#refs.contactNotes.value;
const title = this.#option.contact == null ? r('addContact', 'Add Contact') : r('editContact', 'Edit Contact');
- if (nullOrEmpty(item.Name)) {
+ if (nullOrEmpty(name)) {
showAlert(title, r('contactNameRequired', 'Contact Name cannot be empty.'), 'warn')
.then(() => this.#refs.contactName.focus());
return null;
}
- if (nullOrEmpty(item.Email) && nullOrEmpty(item.MobilePhone)) {
+ if (nullOrEmpty(email) && nullOrEmpty(phone)) {
showAlert(title, r('contactEmailPhoneRequired', 'Email and Mobile Phone cannot both be empty.'), 'warn')
- .then(() => nullOrEmpty(item.Email) ?
+ .then(() => nullOrEmpty(email) ?
this.#refs.contactEmail.focus() :
this.#refs.contactMobile.focus());
return null;
}
- if (!nullOrEmpty(item.Email) && !isEmail(item.Email)) {
+ if (!nullOrEmpty(email) && !isEmail(email)) {
showAlert(title, r('contactEmailInvalid', 'The email address is invalid.'), 'warn')
.then(() => this.#refs.contactEmail.focus());
return null;
}
- return item;
+ let contact = this.#option.contact;
+ if (contact == null) {
+ contact = {};
+ }
+ contact.Name = name;
+ contact.ContactPreference = pref;
+ contact.Email = email;
+ contact.MobilePhone = phone;
+ contact.OptOut = opt;
+ contact.Notes = notes;
+ return contact;
}
}
diff --git a/lib/app/communications/customer.js b/lib/app/communications/customer.js
index 4428a42..b3ed7e4 100644
--- a/lib/app/communications/customer.js
+++ b/lib/app/communications/customer.js
@@ -1,4 +1,4 @@
-import { Grid, createElement, setTooltip, createIcon, createCheckbox, createRadiobox, createPopup, showAlert, showConfirm } from "../../ui";
+import { Grid, createElement, setTooltip, setTooltipNext, createIcon, createCheckbox, createRadiobox, createPopup, showAlert, showConfirm } from "../../ui";
import { r, nullOrEmpty, formatUrl, isEmail, isPhone } from "../../utility";
import { createBox } from "./lib";
import Contact from "./contact";
@@ -13,11 +13,11 @@ class NoteCol extends Grid.GridColumn {
return wrapper;
}
- static setValue(element, _val, item) {
+ static setValue(element, _val, item, _col, grid) {
const name = element.querySelector('.contact-name');
name.innerText = item.Name;
if (name.scrollWidth > name.offsetWidth) {
- setTooltip(name, item.Name);
+ setTooltip(name, item.Name, false, grid.element);
}
element.querySelector('.contact-note').innerText = item.Notes;
}
@@ -28,6 +28,7 @@ class CustomerCommunication {
#option;
#contacts;
#followers;
+ #buttonFollower;
#enter;
#message;
#data = {};
@@ -115,20 +116,42 @@ class CustomerCommunication {
const mp = String(c.MobilePhone).trim();
const email = String(c.Email).trim();
const pref = String(c.ContactPreference);
- if ((pref === '0' || pref === '2') && !isPhone(mp) ||
+ if ((pref !== '1') && !isPhone(mp) ||
pref === '1' && !isEmail(email)) {
continue;
}
- const to = pref === '0' || pref === '2' ? mp : email;
+ const to = pref === '1' ? email : mp;
+ let icon;
+ let method;
+ switch (pref) {
+ case '0':
+ icon = 'comment-lines';
+ method = r('textsToColon', 'Texts to:');
+ break;
+ case '2':
+ icon = 'mobile';
+ method = r('callsToColon', 'Calls to:');
+ break;
+ default:
+ icon = 'envelope';
+ method = r('emailsToColon', 'Emails to:');
+ break;
+ }
+ const span = createElement('span', span => {
+ span.dataset.to = to;
+ span.dataset.name = c.Name;
+ span.innerText = c.Name;
+ });
const item = createElement('div', 'contact-item',
- createIcon('fa-light', pref === '0' ? 'comment-lines' : pref === '2' ? 'mobile' : 'envelope'),
- setTooltip(createElement('span', span => {
- span.dataset.to = to;
- span.dataset.name = c.Name;
- span.innerText = nullOrEmpty(to) ? c.Name : to;
- }), to, true)
+ createIcon('fa-light', icon),
+ span
);
this.#contacts.appendChild(item);
+ let tip = `${method} ${to}`;
+ if (span.scrollWidth > span.offsetWidth) {
+ tip = r('nameColon', 'Name:') + ` ${c.Name}\n${tip}`;
+ }
+ setTooltip(span, tip);
}
this.#message.scrollTop = this.#message.scrollHeight
}
@@ -145,6 +168,7 @@ class CustomerCommunication {
this.#container.querySelector('.button-edit-contacts').style.display = flag === true ? 'none' : '';
this.#container.querySelector('.button-edit-followers').style.display = flag === true ? 'none' : '';
this.#enter.disabled = flag === true;
+ this.#container.querySelector('.button-send-message').style.display = flag === true ? 'none' : '';
}
/**
@@ -156,6 +180,21 @@ class CustomerCommunication {
return;
}
this.#container.querySelector('.button-edit-contacts').style.display = flag === true ? 'none' : '';
+ this.#container.querySelector('.button-edit-followers').style.display = flag === true ? 'none' : '';
+ }
+
+ /**
+ * @param {String} name
+ */
+ set companyName(name) {
+ this.#option.companyName = name;
+ const div = this.#container.querySelector('.title-company');
+ if (nullOrEmpty(name)) {
+ div.style.display = 'none';
+ } else {
+ div.innerText = name;
+ div.style.display = '';
+ }
}
get followers() {
@@ -169,6 +208,7 @@ class CustomerCommunication {
this.#followers.replaceChildren();
if (followers?.length > 0) {
this.#container.querySelector('.follower-bar').style.display = '';
+ this.#container.querySelector('.follower-bar>.bar-list').appendChild(this.#buttonFollower);
for (let f of followers) {
if (f.OptOut) {
continue;
@@ -177,28 +217,40 @@ class CustomerCommunication {
const email = String(f.Email).trim();
const tips = [];
if (f.SendEmail) {
- tips.push(r('emailColon', 'Email:') + ` ${email}`);
+ tips.push(r('emailsToColon', 'Emails to:') + ` ${email}`);
}
if (f.SendText) {
- tips.push(r('phoneColon', 'Mobile Phone:' + ` ${mp}`));
+ tips.push(r('textsToColon', 'Texts to:' + ` ${mp}`));
}
+ let icon;
+ if (f.SendText && f.SendEmail) {
+ icon = 'at';
+ } else {
+ icon = f.SendText ? 'comment-lines' : 'envelope';
+ }
+ const span = createElement('span', span => {
+ if (f.SendEmail) {
+ span.dataset.email = email;
+ }
+ if (f.SendText) {
+ span.dataset.mp = mp;
+ }
+ span.dataset.name = f.Name;
+ span.innerText = f.Name;
+ });
const item = createElement('div', 'contact-item',
- createIcon('fa-light', f.SendText ? 'comment-lines' : 'envelope'),
- setTooltip(createElement('span', span => {
- if (f.SendEmail) {
- span.dataset.email = email;
- }
- if (f.SendText) {
- span.dataset.mp = mp;
- }
- span.dataset.name = f.Name;
- span.innerText = f.SendText ? mp : email;
- }), tips.join('\n'), true)
+ createIcon('fa-light', icon),
+ span
);
this.#followers.appendChild(item);
+ if (span.scrollWidth > span.offsetWidth) {
+ tips.splice(0, 0, r('nameColon', 'Name:') + ` ${c.Name}`);
+ }
+ setTooltip(span, tips.join('\n'));
}
} else {
this.#container.querySelector('.follower-bar').style.display = 'none';
+ this.#container.querySelector('.button-edit-contacts').insertAdjacentElement('beforebegin', this.#buttonFollower)
}
this.#message.scrollTop = this.#message.scrollHeight
}
@@ -251,6 +303,7 @@ class CustomerCommunication {
);
// contacts
const readonly = option.readonly;
+ const recordReadonly = option.recordReadonly;
const contacts = createElement('div');
container.append(
createElement('div', 'contact-bar',
@@ -264,7 +317,7 @@ class CustomerCommunication {
createElement('button', button => {
button.className = 'roundbtn button-edit-contacts';
button.style.backgroundColor = 'rgb(1, 199, 172)';
- if (readonly === true || option.recordReadonly) {
+ if (readonly === true) {
button.style.display = 'none';
}
button.appendChild(createIcon('fa-solid', 'user-edit'));
@@ -273,6 +326,7 @@ class CustomerCommunication {
const pop = createPopup(
createElement('div', div => {
div.style.display = 'flex';
+ div.style.alignItems = 'center';
div.append(
createElement('div', div => {
div.className = 'popup-move';
@@ -293,12 +347,16 @@ class CustomerCommunication {
button.style.backgroundColor = 'rgb(1, 199, 172)';
button.style.marginRight = '10px';
button.className = 'roundbtn button-add-contact';
+ if (recordReadonly) {
+ button.style.display = 'none';
+ }
button.appendChild(createIcon('fa-solid', 'user-plus', {
width: '16px',
height: '16px'
}));
button.addEventListener('click', () => {
const add = new Contact({
+ company: !nullOrEmpty(this.#data.companyCode),
onSave: (item) => {
const exists = this.#gridContact.source.some(s => s.Name === item.Name && s.MobilePhone === item.MobilePhone);
if (exists) {
@@ -326,10 +384,16 @@ class CustomerCommunication {
}),
createElement('div', null,
createElement('div', div => {
+ if (nullOrEmpty(this.#data.companyCode)) {
+ div.style.display = 'none';
+ }
div.style.fontWeight = 'bold';
div.innerText = r('contactFromRecord', 'Contacts from Customer Record');
}),
createElement('div', div => {
+ if (nullOrEmpty(this.#data.companyCode)) {
+ div.style.display = 'none';
+ }
div.className = 'contacts-record';
div.style.maxHeight = '400px';
div.style.width = '660px';
@@ -377,6 +441,7 @@ class CustomerCommunication {
const buttonCol = {
type: Grid.ColumnTypes.Icon,
width: 40,
+ visible: !recordReadonly,
align: 'center',
iconType: 'fa-light'
};
@@ -390,6 +455,7 @@ class CustomerCommunication {
onclick: function () {
const edit = new Contact({
contact: this,
+ company: !nullOrEmpty(This.#data.companyCode),
onSave: item => {
const exists =
This.#gridContact.source.some(s => s !== this && s.Name === item.Name && s.MobilePhone === item.MobilePhone) ||
@@ -427,7 +493,7 @@ class CustomerCommunication {
nameCol,
{ key: 'Email', width: 180 },
{ key: 'MobilePhone', width: 130 },
- createEditCol(grid),
+ createEditCol(this),
{
key: 'delete',
...buttonCol,
@@ -490,6 +556,11 @@ class CustomerCommunication {
});
grid.extraRows = customerRecords.filter(c => !nullOrEmpty(c.Notes)).length;
grid.source = customerRecords;
+ grid.selectedRowChanged = index => {
+ if (index >= 0 && this.#gridWo.selectedIndexes?.length > 0) {
+ this.#gridWo.selectedIndexes = [];
+ }
+ };
this.#gridContact = grid;
// contacts from work order only
@@ -503,14 +574,14 @@ class CustomerCommunication {
nameCol,
{ key: 'Email', width: 180 },
{ key: 'MobilePhone', width: 130 },
- createEditCol(gridWo),
+ createEditCol(this),
{
key: 'delete',
...buttonCol,
text: 'times',
tooltip: r('delete', 'Delete'),
events: {
- onclick: () => {
+ onclick: function () {
showConfirm(r('remoteContact', 'Remove Contact'), r('removeFromWorkorder', 'You are removing {name} from work order.\n\nDo you want to Continue?').replace('{name}', this.Name), [
{ key: 'continue', text: r('continue', 'Continue') },
{ key: 'cancel', text: r('cancel', 'Cancel') }
@@ -544,6 +615,11 @@ class CustomerCommunication {
});
gridWo.extraRows = workOrderOnly.filter(c => !nullOrEmpty(c.Notes)).length;
gridWo.source = workOrderOnly;
+ gridWo.selectedRowChanged = index => {
+ if (index >= 0 && this.#gridContact.selectedIndexes?.length > 0) {
+ this.#gridContact.selectedIndexes = [];
+ }
+ };
this.#gridWo = gridWo;
});
});
@@ -554,6 +630,131 @@ class CustomerCommunication {
this.#contacts = contacts;
// followers
const followers = createElement('div');
+ const buttonEditFollower = createElement('button', button => {
+ button.className = 'roundbtn button-edit-followers';
+ button.style.backgroundColor = 'rgb(48, 107, 255)';
+ if (readonly === true || recordReadonly) {
+ button.style.display = 'none';
+ }
+ button.appendChild(createIcon('fa-solid', 'pen'));
+ setTooltip(button, r('editFollower', 'Edit Followers'));
+ button.addEventListener('click', () => {
+ const pop = createPopup(
+ createElement('div', div => {
+ div.style.display = 'flex';
+ div.style.alignItems = 'center';
+ div.append(
+ createElement('div', div => {
+ div.className = 'popup-move';
+ div.style.flex = '1 1 auto';
+ div.innerText = r('editContacts', 'Edit Contacts') + '\n' + r('followers', 'Followers');
+ }),
+ createElement('button', button => {
+ button.style.flex = '0 0 auto';
+ button.style.backgroundColor = 'rgb(1, 199, 172)';
+ button.style.marginRight = '10px';
+ button.className = 'roundbtn button-add-follower';
+ button.appendChild(createIcon('fa-solid', 'user-plus', {
+ width: '16px',
+ height: '16px'
+ }));
+ button.addEventListener('click', () => {
+ if (typeof this.#option.onInitFollower === 'function') {
+ this.#option.onInitFollower().then(data => {
+ if (typeof data === 'string') {
+ showAlert(r('customerRecord', 'Customer Record'), data, 'warn');
+ return;
+ }
+ const add = new Follower({
+ followers: data,
+ onOk: list => {
+ if (typeof this.#option.onAddFollower === 'function') {
+ const result = this.#option.onAddFollower(list);
+ if (typeof result?.then === 'function') {
+ return result.then(r => {
+ this.#gridFollower.source = r;
+ return r;
+ });
+ }
+ return false;
+ }
+ }
+ });
+ add.show(container);
+ })
+ }
+ });
+ setTooltip(button, r('addFollower', 'Add Follower'))
+ })
+ )
+ }),
+ createElement('div', null,
+ createElement('div', div => {
+ div.style.fontWeight = 'bold';
+ div.innerText = r('contactFromRecord', 'Contacts from Customer Record');
+ }),
+ createElement('div', div => {
+ div.className = 'followers-record';
+ div.style.maxHeight = '400px';
+ div.style.width = '660px';
+ })
+ )
+ );
+ pop.show(container).then(() => {
+ const grid = new Grid();
+ grid.height = 0;
+ grid.allowHtml = true;
+ grid.headerVisible = false;
+ grid.columns = [
+ {
+ key: 'type',
+ type: Grid.ColumnTypes.Icon,
+ width: 50,
+ filter: c => c.SendText ? 'comment-lines' : 'envelope',
+ className: 'icon-contact-type',
+ iconType: 'fa-light'
+ },
+ { key: 'Name', width: 160 },
+ { key: 'Email', width: 180 },
+ { key: 'MobilePhone', width: 130 },
+ {
+ key: 'delete',
+ type: Grid.ColumnTypes.Icon,
+ width: 40,
+ align: 'center',
+ iconType: 'fa-light',
+ text: 'times',
+ tooltip: r('delete', 'Delete'),
+ events: {
+ onclick: function () {
+ showConfirm(
+ r('deleteFollower', 'Delete Follower'),
+ r('promptDeleteFollower', 'Do you want to delete this follower?'))
+ .then(result => {
+ if (result?.key === 'yes') {
+ if (typeof option.onDeleteFollower === 'function') {
+ option.onDeleteFollower(result.key, this);
+ }
+ const index = grid.source.indexOf(this);
+ if (index >= 0) {
+ const source = grid.source;
+ source.splice(index, 1);
+ grid.extraRows = source.filter(c => !nullOrEmpty(c.Notes)).length;
+ grid.source = source;
+ }
+ }
+ });
+ }
+ }
+ }
+ ];
+ grid.init(pop.container.querySelector('.followers-record'));
+ grid.source = this.#data.followers;
+ this.#gridFollower = grid;
+ });
+ });
+ });
+ this.#buttonFollower = buttonEditFollower;
container.append(
createElement('div', div => {
div.className = 'contact-bar follower-bar';
@@ -570,129 +771,7 @@ class CustomerCommunication {
), r('copied', 'Copied')),
createElement('div', 'bar-list',
followers,
- createElement('button', button => {
- button.className = 'roundbtn button-edit-followers';
- button.style.backgroundColor = 'rgb(48, 107, 255)';
- if (readonly === true) {
- button.style.display = 'none';
- }
- button.appendChild(createIcon('fa-solid', 'pen'));
- setTooltip(button, r('editFollower', 'Edit Followers'));
- button.addEventListener('click', () => {
- const pop = createPopup(
- createElement('div', div => {
- div.style.display = 'flex';
- div.style.alignItems = 'center';
- div.append(
- createElement('div', div => {
- div.className = 'popup-move';
- div.style.flex = '1 1 auto';
- div.innerText = r('editContacts', 'Edit Contacts') + '\n' + r('followers', 'Followers');
- }),
- createElement('button', button => {
- button.style.flex = '0 0 auto';
- button.style.backgroundColor = 'rgb(1, 199, 172)';
- button.style.marginRight = '10px';
- button.className = 'roundbtn button-add-follower';
- button.appendChild(createIcon('fa-solid', 'user-plus', {
- width: '16px',
- height: '16px'
- }));
- button.addEventListener('click', () => {
- if (typeof this.#option.onInitFollower === 'function') {
- this.#option.onInitFollower().then(data => {
- if (typeof data === 'string') {
- showAlert(r('customerRecord', 'Customer Record'), data, 'warn');
- return;
- }
- const add = new Follower({
- followers: data,
- onOk: list => {
- if (typeof this.#option.onAddFollower === 'function') {
- const result = this.#option.onAddFollower(list);
- if (typeof result?.then === 'function') {
- return result.then(r => {
- this.#gridFollower.source = r;
- return r;
- });
- }
- return false;
- }
- }
- });
- add.show(container);
- })
- }
- });
- setTooltip(button, r('addFollower', 'Add Follower'))
- })
- )
- }),
- createElement('div', null,
- createElement('div', div => {
- div.style.fontWeight = 'bold';
- div.innerText = r('contactFromRecord', 'Contacts from Customer Record');
- }),
- createElement('div', div => {
- div.className = 'followers-record';
- div.style.maxHeight = '400px';
- div.style.width = '660px';
- })
- )
- );
- pop.show(container).then(() => {
- const grid = new Grid();
- grid.height = 0;
- grid.allowHtml = true;
- grid.headerVisible = false;
- grid.columns = [
- {
- key: 'type',
- type: Grid.ColumnTypes.Icon,
- width: 50,
- filter: c => c.SendText ? 'comment-lines' : 'envelope',
- className: 'icon-contact-type',
- iconType: 'fa-light'
- },
- { key: 'Name', width: 160 },
- { key: 'Email', width: 180 },
- { key: 'MobilePhone', width: 130 },
- {
- key: 'delete',
- type: Grid.ColumnTypes.Icon,
- width: 40,
- align: 'center',
- iconType: 'fa-light',
- text: 'times',
- tooltip: r('delete', 'Delete'),
- events: {
- onclick: function () {
- showConfirm(
- r('deleteFollower', 'Delete Follower'),
- r('promptDeleteFollower', 'Do you want to delete this follower?')).then(result => {
- if (result?.key === 'yes') {
- if (typeof option.onDeleteFollower === 'function') {
- option.onDeleteFollower(result.key, this);
- }
- const index = grid.source.indexOf(this);
- if (index >= 0) {
- const source = grid.source;
- source.splice(index, 1);
- grid.extraRows = source.filter(c => !nullOrEmpty(c.Notes)).length;
- grid.source = source;
- }
- }
- });
- }
- }
- }
- ];
- grid.init(pop.container.querySelector('.followers-record'));
- grid.source = this.#data.followers;
- this.#gridFollower = grid;
- });
- });
- })
+ buttonEditFollower
)
)
);
@@ -718,6 +797,9 @@ class CustomerCommunication {
createElement('button', button => {
button.className = 'roundbtn button-send-message';
button.style.backgroundColor = 'rgb(19, 150, 204)';
+ if (readonly === true) {
+ button.style.display = 'none';
+ }
button.appendChild(createIcon('fa-solid', 'paper-plane'));
setTooltip(button, r('sendMessage', 'Send Message'));
button.addEventListener('click', () => {
diff --git a/lib/app/communications/follower.js b/lib/app/communications/follower.js
index 299950f..0c0eabf 100644
--- a/lib/app/communications/follower.js
+++ b/lib/app/communications/follower.js
@@ -22,7 +22,7 @@ class Follower {
if (nullOrEmpty(key)) {
this.#grid.source = this.#option.followers;
} else {
- this.#grid.source = this.#option.followers.filter(f => contains(f.DisplayName, key, true));
+ this.#grid.source = this.#option.followers.filter(f => f.Text || f.Email || contains(f.DisplayName, key, true));
}
});
}),
diff --git a/lib/app/communications/internal.js b/lib/app/communications/internal.js
index 6f6ac17..09667de 100644
--- a/lib/app/communications/internal.js
+++ b/lib/app/communications/internal.js
@@ -22,12 +22,26 @@ class InternalComment {
}
}
+ /**
+ * @param {boolean} flag
+ */
+ set readonly(flag) {
+ this.#option.readonly = flag;
+ if (this.#container == null) {
+ return;
+ }
+ this.#enter.disabled = flag === true;
+ this.#container.querySelector('.button-send-message').style.display = flag === true ? 'none' : '';
+ this.#container.querySelector('.button-post-note').style.display = flag === true ? 'none' : '';
+ }
+
create() {
const container = createBox(
createElement('div', null,
createElement('div', div => div.innerText = r('internalComments', 'Internal Comments'))
), []
);
+ const readonly = this.#option.readonly;
// enter box
const enter = createElement('textarea');
enter.placeholder = r('typeComment', 'Enter Comment Here');
@@ -37,6 +51,9 @@ class InternalComment {
const s = String(nullOrEmpty(val) ? 0 : val.length) + '/3000';
this.#container.querySelector('.message-bar .prompt-count').innerText = s;
});
+ if (readonly === true) {
+ enter.disabled = true;
+ }
this.#enter = enter;
container.appendChild(
createElement('div', 'message-bar',
@@ -46,6 +63,9 @@ class InternalComment {
createElement('button', button => {
button.className = 'roundbtn button-send-message';
button.style.backgroundColor = 'rgb(19, 150, 204)';
+ if (readonly === true) {
+ button.style.display = 'none';
+ }
button.appendChild(createIcon('fa-solid', 'paper-plane'));
setTooltip(button, r('sendMessage', 'Send Message'));
button.addEventListener('click', () => {
@@ -58,6 +78,9 @@ class InternalComment {
button.className = 'roundbtn button-post-note';
button.style.border = '1px solid rgb(19, 150, 204)';
button.style.fill = 'rgb(19, 150, 204)';
+ if (readonly === true) {
+ button.style.display = 'none';
+ }
button.appendChild(createIcon('fa-solid', 'comment-alt-lines'));
setTooltip(button, r('postNote', 'Post Note'));
button.addEventListener('click', () => {
@@ -89,7 +112,7 @@ class InternalComment {
div.innerText = comment.UserName;
}));
const content = createElement('div', 'item-content');
- content.appendChild(createElement('span', span => span.innerText = escapeHtml(comment.Comment)));
+ content.appendChild(createElement('span', span => span.innerHTML = escapeHtml(comment.Comment)));
if (comment.FollowUp?.length > 0) {
div.classList.add('item-sent');
const sendto = r('sendToColon', 'Send To :') + '\r\n' + comment.FollowUp.split(';').join('\r\n');
diff --git a/lib/app/communications/style.scss b/lib/app/communications/style.scss
index 9030361..36bea08 100644
--- a/lib/app/communications/style.scss
+++ b/lib/app/communications/style.scss
@@ -176,9 +176,10 @@
}
>span {
- flex: 1 1 auto;
+ // flex: 1 1 auto;
color: var(--strong-color);
font-size: var(--font-larger-size);
+ white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
padding-right: 10px;
@@ -319,7 +320,6 @@
.grid {
height: 100%;
min-height: 120px;
- overflow: hidden;
>.grid-body .grid-body-content>.grid-row>td {
vertical-align: top;
diff --git a/lib/ui/grid.js b/lib/ui/grid.js
index ff9b606..f34db13 100644
--- a/lib/ui/grid.js
+++ b/lib/ui/grid.js
@@ -200,7 +200,7 @@ class GridCheckboxColumn extends GridColumn {
class GridIconColumn extends GridColumn {
static create() { return createElement('span', 'col-icon') }
- static setValue(element, val, item, col) {
+ static setValue(element, val, item, col, grid) {
let className = col.className;
if (typeof className === 'function') {
className = className.call(col, item);
@@ -219,7 +219,7 @@ class GridIconColumn extends GridColumn {
const icon = createIcon(type, val);
// const layer = element.children[0];
element.replaceChildren(icon);
- !nullOrEmpty(col.tooltip) && setTooltip(element, col.tooltip);
+ !nullOrEmpty(col.tooltip) && setTooltip(element, col.tooltip, false, grid.element);
element.dataset.type = type;
element.dataset.icon = val;
}
@@ -312,6 +312,8 @@ class Grid {
this.#parent = container;
}
+ get element() { return this.#el }
+
get source() { return this.#source?.map(s => s.values) }
set source(list) {
if (this.#el == null) {
diff --git a/lib/ui/tooltip.d.ts b/lib/ui/tooltip.d.ts
index 1e0a69d..a095451 100644
--- a/lib/ui/tooltip.d.ts
+++ b/lib/ui/tooltip.d.ts
@@ -1,2 +1,2 @@
-export function setTooltip(container: HTMLElement, content: string | HTMLElement, flag?: boolean): HTMLElement
+export function setTooltip(container: HTMLElement, content: string | HTMLElement, flag?: boolean, parent?: HTMLElement): HTMLElement
export function resolveTooltip(container?: HTMLElement): HTMLElement
\ No newline at end of file
diff --git a/lib/ui/tooltip.html b/lib/ui/tooltip.html
index ecb66c0..0288cff 100644
--- a/lib/ui/tooltip.html
+++ b/lib/ui/tooltip.html
@@ -5,7 +5,7 @@
给某个元素或者页面上含有 title 属性的元素设置一个统一样式的 tooltip。
setTooltip
- function setTooltip(container: HTMLElement, content: string | HTMLElement, flag?: boolean): void
+ function setTooltip(container: HTMLElement, content: string | HTMLElement, flag?: boolean, parent?: HTMLElement): void
container: HTMLElement
要设置 tooltip 的元素
@@ -18,6 +18,10 @@
是否启用严格模式,只有显示不完整时才显示 tooltip
+ parent?: HTMLElement
+
+ 创建在哪个元素内,默认创建在目标元素之内
+
resolveTooltip
function resolveTooltip(container?: HTMLElement): HTMLElement
container?: HTMLElement
diff --git a/lib/ui/tooltip.js b/lib/ui/tooltip.js
index c57d5c4..d295629 100644
--- a/lib/ui/tooltip.js
+++ b/lib/ui/tooltip.js
@@ -1,14 +1,22 @@
import { createElement } from "../functions";
+// import { global } from "../utility";
-function setTooltip(container, content, flag = false) {
- const tip = container.querySelector('.tooltip-wrapper');
- if (tip != null) {
- tip.remove();
+function setTooltip(container, content, flag = false, parent = null) {
+ const isParent = parent instanceof HTMLElement;
+ if (isParent) {
+ const tipid = container.dataset.tipId;
+ const tip = parent.querySelector(`.tooltip-wrapper[data-tip-id="${tipid}"]`);
+ tip?.remove();
+ } else {
+ const tip = container.querySelector('.tooltip-wrapper');
+ tip?.remove();
}
const wrapper = createElement('div', wrapper => {
wrapper.className = 'tooltip-wrapper tooltip-color';
wrapper.style.visibility = 'hidden';
wrapper.style.opacity = 0;
+ wrapper.style.top = '0';
+ wrapper.style.left = '0';
},
createElement('div', 'tooltip-pointer tooltip-color'),
createElement('div', 'tooltip-curtain tooltip-color'),
@@ -22,7 +30,14 @@ function setTooltip(container, content, flag = false) {
})
);
// container.insertAdjacentElement('afterend', wrapper);
- container.appendChild(wrapper);
+ if (isParent) {
+ const tipId = String(Math.random()).substring(2);
+ container.dataset.tipId = tipId;
+ wrapper.dataset.tipId = tipId;
+ parent.appendChild(wrapper);
+ } else {
+ container.appendChild(wrapper);
+ }
let tid;
container.addEventListener('mouseenter', () => {
@@ -36,19 +51,27 @@ function setTooltip(container, content, flag = false) {
}
if (!flag || c.scrollWidth > c.offsetWidth) {
tid = setTimeout(() => {
- let parent = c;
- let left = c.offsetLeft;
- let top = c.offsetTop;
- while ((parent = parent.offsetParent) != null) {
- left += parent.offsetLeft;
- top += parent.offsetTop;
+ let p;
+ let left;
+ let top;
+ left = c.offsetLeft;
+ top = c.offsetTop;
+ if (isParent) {
+ p = c.offsetParent;
+ while (p != null && p !== parent) {
+ left += p.offsetLeft;
+ top += p.offsetTop;
+ p = p.offsetParent;
+ }
}
- parent = c;
- while ((parent = parent.parentElement) != null) {
- left -= parent.scrollLeft;
- top -= parent.scrollTop;
+ p = c.parentElement;
+ const offsetParent = c.offsetParent;
+ while (p != null && p !== (isParent ? parent : offsetParent)) {
+ left -= p.scrollLeft;
+ top -= p.scrollTop;
+ p = p.parentElement;
}
- left -= wrapper.offsetWidth / 2 - c.offsetWidth / 2;
+ left += (c.offsetWidth - wrapper.offsetWidth) / 2;
top -= wrapper.offsetHeight + 14;
wrapper.style.left = `${left}px`;
wrapper.style.top = `${top}px`;