@salla.sa/twilight-components
Version:
Salla Web Component
284 lines (275 loc) • 18.6 kB
JavaScript
/*!
* Crafted with ❤ by Salla
*/
import { r as registerInstance, h, H as Host, a as getElement } from './index-BQQ2x3w_.js';
import { A as ArrowDown } from './keyboard_arrow_down-DCZbpt2a.js';
import { B as BellRing } from './bell-ring-D3mWkc-3.js';
import { P as PendingOrdersIcon } from './cart-DY4LZmNP.js';
import { W as WishListIcon } from './star-ZT7ehBBk.js';
import { C as Cancel } from './cancel-BsLF_HK7.js';
import { S as Star } from './star2-D4oPi1Ov.js';
import { l as loyaltyProgramIcon } from './gift-BChI23pG.js';
var OrderIcon = `<!-- Generated by IcoMoon.io -->
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 32 32">
<title>box-bankers</title>
<path d="M28 1.333h-24c-2.205 0-4 1.795-4 4v5.333c0 0.736 0.597 1.333 1.333 1.333v14.667c0 2.205 1.795 4 4 4h21.333c2.205 0 4-1.795 4-4v-14.667c0.736 0 1.333-0.597 1.333-1.333v-5.333c0-2.205-1.795-4-4-4zM28 26.667c0 0.735-0.599 1.333-1.333 1.333h-21.333c-0.735 0-1.333-0.599-1.333-1.333v-14.667h5.333v2.667c0 1.471 1.196 2.667 2.667 2.667h8c1.471 0 2.667-1.196 2.667-2.667v-2.667h5.333zM12 14.667v-2.667h8v2.667zM29.333 9.333h-26.667v-4c0-0.735 0.599-1.333 1.333-1.333h24c0.735 0 1.333 0.599 1.333 1.333z"></path>
</svg>
`;
var UserCircle = `<!-- Generated by IcoMoon.io -->
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 32 32">
<title>user-circle</title>
<path d="M16 22.964c-4.525 0-8.447 1.713-9.993 4.365-0.371 0.636-0.156 1.452 0.48 1.823s1.453 0.156 1.823-0.48c0.855-1.465 3.624-3.041 7.689-3.041s6.835 1.576 7.689 3.041c0.249 0.424 0.696 0.661 1.155 0.661 0.228 0 0.459-0.059 0.669-0.181 0.637-0.371 0.852-1.187 0.48-1.823-1.545-2.652-5.467-4.365-9.992-4.365zM22.667 13.631c0-3.676-2.991-6.667-6.667-6.667s-6.667 2.991-6.667 6.667 2.991 6.667 6.667 6.667 6.667-2.992 6.667-6.667zM12 13.631c0-2.205 1.795-4 4-4s4 1.795 4 4-1.795 4-4 4-4-1.795-4-4zM16 0.297c-8.823 0-16 7.177-16 16 0 2.941 0.821 5.831 2.373 8.357 0.252 0.411 0.689 0.636 1.137 0.636 0.239 0 0.479-0.064 0.696-0.197 0.628-0.385 0.824-1.207 0.439-1.833-1.295-2.108-1.979-4.516-1.979-6.963 0-7.352 5.981-13.333 13.333-13.333s13.333 5.981 13.333 13.333c0 2.448-0.684 4.856-1.979 6.961-0.385 0.628-0.189 1.448 0.437 1.835 0.627 0.384 1.448 0.189 1.835-0.437 1.553-2.527 2.373-5.416 2.373-8.359 0-8.823-7.177-16-16-16z"></path>
</svg>
`;
var LogoutIcon = `<!-- Generated by IcoMoon.io -->
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 32 32">
<title>send-out</title>
<path d="M16 26.667h-9.333c-0.736 0-1.333-0.597-1.333-1.333v-18.667c0-0.736 0.597-1.333 1.333-1.333h9.333c0.737 0 1.333-0.596 1.333-1.333s-0.596-1.333-1.333-1.333h-9.333c-2.205 0-4 1.795-4 4v18.667c0 2.205 1.795 4 4 4h9.333c0.737 0 1.333-0.596 1.333-1.333s-0.596-1.333-1.333-1.333zM29.231 15.491c-0.068-0.164-0.167-0.312-0.289-0.436l-5.332-5.332c-0.521-0.521-1.364-0.521-1.885 0s-0.521 1.364 0 1.885l3.057 3.059h-12.781c-0.737 0-1.333 0.596-1.333 1.333s0.596 1.333 1.333 1.333h12.781l-3.057 3.057c-0.521 0.521-0.521 1.364 0 1.885 0.26 0.26 0.601 0.391 0.943 0.391s0.683-0.131 0.943-0.391l5.332-5.332c0.124-0.123 0.221-0.271 0.289-0.435 0.135-0.325 0.135-0.693 0-1.019z"></path>
</svg>
`;
var WalletIcon = `<!-- Generated by IcoMoon.io -->
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 32 32">
<title>wallet</title>
<path d="M28 6.667h-24c-0.736 0-1.333-0.597-1.333-1.333s0.597-1.333 1.333-1.333h24c0.737 0 1.333-0.596 1.333-1.333s-0.596-1.333-1.333-1.333h-24c-2.204 0-3.999 1.793-4 3.997v18.669c0 3.676 2.991 6.667 6.667 6.667h21.333c2.205 0 4-1.795 4-4v-16c0-2.205-1.795-4-4-4zM29.333 26.667c0 0.736-0.597 1.333-1.333 1.333h-21.333c-2.205 0-4-1.795-4-4v-14.895c0.416 0.147 0.865 0.228 1.333 0.228h24c0.736 0 1.333 0.597 1.333 1.333zM22.667 14.667c-2.205 0-4 1.795-4 4s1.795 4 4 4 4-1.795 4-4-1.795-4-4-4zM22.667 20c-0.736 0-1.333-0.597-1.333-1.333s0.597-1.333 1.333-1.333 1.333 0.597 1.333 1.333-0.597 1.333-1.333 1.333z"></path>
</svg>
`;
var SettingstIcon = `<!-- Generated by IcoMoon.io -->
<svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="32" height="32" viewBox="0 0 32 32">
<title>settings</title>
<path d="M29.717 19.037l-1.817-1.499c0.067-0.512 0.1-1.028 0.1-1.539s-0.033-1.027-0.1-1.54l1.817-1.499c1.177-0.969 1.479-2.633 0.716-3.956l-1.159-2.009c-0.761-1.324-2.349-1.895-3.784-1.359l-2.212 0.829c-0.817-0.627-1.704-1.139-2.649-1.535l-0.388-2.332c-0.251-1.507-1.54-2.6-3.067-2.6h-2.319c-1.525 0-2.815 1.093-3.065 2.6l-0.388 2.332c-0.945 0.396-1.833 0.909-2.649 1.535l-2.212-0.829c-1.428-0.533-3.021 0.036-3.783 1.359l-1.16 2.011c-0.763 1.323-0.46 2.985 0.716 3.955l1.817 1.499c-0.067 0.513-0.1 1.029-0.1 1.54s0.033 1.027 0.1 1.539l-1.819 1.5c-1.176 0.971-1.476 2.635-0.715 3.955l1.159 2.011c0.763 1.324 2.356 1.899 3.784 1.36l2.212-0.831c0.816 0.625 1.703 1.139 2.649 1.535l0.388 2.332c0.251 1.507 1.54 2.6 3.067 2.6h2.319c1.527 0 2.816-1.093 3.065-2.6l0.389-2.332c0.947-0.396 1.833-0.909 2.649-1.535l2.212 0.831c1.432 0.539 3.023-0.035 3.783-1.36l1.16-2.011c0.76-1.32 0.459-2.984-0.717-3.956zM28.123 21.663l-1.16 2.011c-0.108 0.188-0.336 0.267-0.536 0.193l-2.931-1.1c-0.459-0.173-0.979-0.079-1.349 0.248-0.923 0.811-1.979 1.423-3.139 1.819-0.464 0.157-0.804 0.559-0.885 1.043l-0.515 3.087c-0.033 0.215-0.216 0.371-0.433 0.371h-2.319c-0.217 0-0.4-0.156-0.436-0.371l-0.513-3.087c-0.080-0.484-0.42-0.884-0.885-1.043-1.16-0.396-2.216-1.008-3.139-1.819-0.248-0.217-0.561-0.332-0.88-0.332-0.157 0-0.316 0.028-0.469 0.084l-2.931 1.1c-0.199 0.079-0.428-0.004-0.537-0.193l-1.159-2.011c-0.109-0.191-0.067-0.428 0.101-0.568l2.409-1.985c0.379-0.312 0.555-0.807 0.46-1.288-0.119-0.605-0.179-1.219-0.179-1.821 0-0.604 0.060-1.217 0.181-1.823 0.095-0.481-0.081-0.976-0.46-1.288l-2.409-1.985c-0.168-0.139-0.211-0.376-0.101-0.567l1.16-2.011c0.107-0.188 0.332-0.268 0.536-0.193l2.931 1.1c0.459 0.172 0.979 0.079 1.348-0.247 0.924-0.812 1.98-1.424 3.139-1.817 0.465-0.159 0.805-0.559 0.885-1.044l0.513-3.088c0.036-0.215 0.219-0.371 0.436-0.371h2.319c0.217 0 0.4 0.156 0.436 0.371l0.515 3.088c0.081 0.484 0.42 0.885 0.885 1.044 1.157 0.393 2.213 1.005 3.139 1.817 0.371 0.325 0.889 0.42 1.349 0.247l2.929-1.099c0.203-0.076 0.428 0.005 0.537 0.193l1.159 2.009c0.109 0.191 0.067 0.428-0.101 0.567l-2.409 1.985c-0.379 0.312-0.555 0.807-0.46 1.288 0.119 0.605 0.179 1.219 0.179 1.823 0 0.603-0.060 1.216-0.181 1.821-0.095 0.48 0.081 0.975 0.46 1.288l2.408 1.984c0.169 0.141 0.212 0.379 0.103 0.569zM16.016 10.667c-2.941 0-5.333 2.392-5.333 5.333s2.392 5.333 5.333 5.333 5.333-2.392 5.333-5.333-2.392-5.333-5.333-5.333zM16.016 18.667c-1.471 0-2.667-1.196-2.667-2.667s1.196-2.667 2.667-2.667 2.667 1.196 2.667 2.667-1.196 2.667-2.667 2.667z"></path>
</svg>
`;
const sallaUserMenuCss = "";
const SallaUserMenu = class {
constructor(hostRef) {
registerInstance(this, hostRef);
this.accountLoading = false;
this.opened = false;
this.notifications = salla.lang.get('common.titles.notifications');
this.orders = salla.lang.get('common.titles.orders');
this.pending_orders = salla.lang.get('common.titles.pending_orders');
this.wishlist = salla.lang.get('common.titles.wishlist');
this.profile = salla.lang.get('common.titles.profile');
this.rating = salla.lang.get('common.titles.rating');
this.wallet = salla.lang.get('common.titles.wallet');
this.settings = salla.lang.get('common.titles.settings');
this.loyalty_program = salla.lang.get('pages.loyalty_program.loyalty_points');
this.logout = salla.lang.get('blocks.header.logout');
this.hello = salla.lang.get('pages.checkout.hello');
this.first_name = salla.storage.get('user.first_name') || '';
this.last_name = salla.storage.get('user.last_name') || '';
this.avatar = salla.storage.get('user.avatar') || salla.url.cdn('images/avatar.png', 40, 40);
this.badges = {
notifications: salla.helpers.number(salla.storage.get('user.notifications') || 0),
pending_orders: salla.helpers.number(salla.storage.get('user.pending_orders') || 0),
};
this.sallaAccountEnabled = false;
this.hasBadges = Number(salla.storage.get('user.pending_orders')) > 0 ||
Number(salla.storage.get('user.notifications')) > 0;
/**
* To display only the list without the dropdown functionality
*/
this.inline = false;
/**
* To display the trigger as an avatar only
*/
this.avatarOnly = false;
/**
* To display the dropdown header in mobile sheet
*/
this.showHeader = false;
/**
* To Make the dropdown menu relative to parent element or not
*/
this.relativeDropdown = false;
/**
* To show the trigger button or not
*/
this.showTrigger = false;
this.onClickOutside = () => {
this.opened = false;
};
this.OrderUpdate = 0;
this.items = {
notifications: BellRing,
orders: OrderIcon,
pending_orders: PendingOrdersIcon,
wishlist: WishListIcon,
wallet: WalletIcon,
loyalty_program: loyaltyProgramIcon,
profile: UserCircle
};
}
async componentWillLoad() {
await salla.onReady();
await salla.lang.onLoaded();
this.loadTranslations();
this.initiate();
Salla.event.on('api::token.injected', (token) => this.profileUrl = this.buildProfileUrl(token));
this.sallaAccountEnabled = Salla.config.get('store.features')?.includes('salla-account');
this.items = this.sallaAccountEnabled ? { ...this.items, settings: SettingstIcon } : this.items;
//we need it only in theme-y
if (this.host.hasAttribute('with-rating')) {
this.items.rating = Star;
}
//we need it to be the last item
this.items.logout = LogoutIcon;
let token = Salla.storage.get('token');
if (!token) {
token = '';
}
this.profileUrl = this.buildProfileUrl(token);
const trigger = this.host.querySelector('[slot="trigger"]');
this.triggerSlot =
'<div class="s-user-menu-trigger"><div class="s-user-menu-avatar-wrap"><img class="s-user-menu-trigger-avatar" src="{avatar}" alt="{first_name}{last_name}" /></div><div class="s-user-menu-trigger-content"><span class="s-user-menu-trigger-hello">{hello}</span><p class="s-user-menu-trigger-name">{first_name} {last_name}</p></div> <i class="s-user-menu-trigger-icon">{icon}</i></div>';
if (!trigger) {
return;
}
this.triggerSlot = trigger.innerHTML;
trigger.innerHTML = this.replaceParams(trigger.innerHTML);
}
loadTranslations() {
this.notifications = Salla.lang.get('common.titles.notifications');
this.orders = Salla.lang.get('common.titles.orders');
this.pending_orders = Salla.lang.get('common.titles.pending_orders');
this.wishlist = Salla.lang.get('common.titles.wishlist');
this.profile = Salla.lang.get('common.titles.profile');
this.hello = Salla.lang.get('pages.checkout.hello');
this.rating = Salla.lang.get('common.titles.rating');
this.wallet = Salla.lang.get('common.titles.wallet');
this.settings = Salla.lang.get('common.titles.settings');
this.loyalty_program = Salla.lang.get('pages.loyalty_program.loyalty_points');
this.logout = Salla.lang.get('blocks.header.logout');
}
initiate() {
if (Salla.config.isGuest()) {
return this.autoMountLoginModal();
}
this.is_loggedIn = true;
/**
* Get Fresh Notifications In These Cases:
* - is notification page, if user already changed the status of his orders (to reset notification badge)
* - is pending orders page, if user already changed the status of his orders (to reset orders badge)
* - is profile page, in case user changed his name or avatar, we need to update it
* - half hour is passed from the last user data fetched
*
* //todo:: update the data in the storage in customer pages
* //todo:: cover two requests in customer pages
* //todo:: make sure to run this only after token is set
*/
const shouldFetchProfile = !this.inline &&
(salla.url.is_page('customer.notifications') ||
salla.url.is_page('customer.orders.index.pending') ||
salla.url.is_page('customer.profile') ||
(Date.now() - (salla.storage.get('user.fetched_at') || 0)) / 1000 / 60 > 30);
if (shouldFetchProfile) {
return this.fetchFreshProfile();
}
salla.event.on('profile::info.fetched', res => {
this.updateProfileState(res);
});
}
autoMountLoginModal() {
if (!this.showTrigger) {
return;
}
if (document.querySelector('salla-login-modal')) {
return;
}
Salla.hooks.mount('body:end', document.createElement('salla-login-modal'));
}
fetchFreshProfile() {
//don't request fetchFreshProfile unless token is injected into the api
if (!salla.api.token) {
salla.log('trying to fetchFreshProfile before injected the token!!');
return;
}
salla.profile.api.info().then(res => {
this.updateProfileState(res);
});
}
updateProfileState(res) {
this.badges = {
notifications: salla.helpers.number(res.data.notifications || 0),
pending_orders: salla.helpers.number(res.data.pending_orders || 0),
};
this.hasBadges = Number(res.data.pending_orders) > 0 || Number(res.data.notifications) > 0;
this.first_name = res.data.first_name;
this.last_name = res.data.last_name;
this.avatar = res.data.avatar || salla.url.cdn('images/avatar.png', 40, 40);
}
async open(e) {
this.opened = !this.opened;
e.stopPropagation();
if (this.opened) {
window.addEventListener('click', this.onClickOutside);
}
}
menuItemClicked(event, itemKey) {
if (itemKey !== 'logout') {
return;
}
event.preventDefault();
salla.auth.logout('sall-user-menu');
}
replaceParams(body) {
return body
.replace(/\{hello\}/g, this.hello)
.replace(/\{first_name\}/g, this.first_name)
.replace(/\{last_name\}/g, this.last_name)
.replace(/\{avatar\}/g, this.avatar)
.replace(/\{icon\}/g, ArrowDown);
}
getTheHeader() {
return (h("div", { class: {
's-user-menu-trigger-slot': true,
's-user-menu-red-dot': this.hasBadges,
's-user-menu-trigger-avatar-only': this.avatarOnly,
}, id: "trigger-slot", onClick: e => this.open(e), onKeyUp: e => this.open(e), innerHTML: this.replaceParams(this.triggerSlot) }));
}
getItemAnchorLinkAttrs(itemKey) {
if (itemKey === 'profile' && this.profileUrl && this.sallaAccountEnabled) {
return { href: this.profileUrl, target: '_blank', rel: 'noopener noreferrer' };
}
// Fix loyalty program URL mapping from master
const urlKey = itemKey === 'loyalty_program' ? 'loyalty' : itemKey;
return { href: Salla.url.get(urlKey) };
}
getMenuItem([itemKey, itemValue], i) {
//todo:: enhancement support slot here
if (itemKey === 'wallet' && !window.can_access_wallet)
return;
if (itemKey === 'loyalty_program' && !Salla.config.get('store.features').includes('loyalty-system'))
return;
return (h("li", { class: {
's-user-menu-dropdown-item': true,
's-user-menu-dropdown-item-logout': i + 1 === Object.entries(this.items).length,
} }, h("a", { ...this.getItemAnchorLinkAttrs(itemKey), class: "s-user-menu-dropdown-item-link", onClick: event => this.menuItemClicked(event, itemKey) }, h("i", { class: "s-user-menu-dropdown-item-prefix", innerHTML: itemValue }), h("span", { class: "s-user-menu-dropdown-item-title" }, this[itemKey]), !['٠', '0', undefined].includes(this.badges[itemKey]) ? (h("span", { class: "s-user-menu-dropdown-item-badge" }, this.badges[itemKey])) : (''))));
}
componentShouldUpdate() {
if (!this.opened) {
window.removeEventListener('click', this.onClickOutside);
}
}
render() {
if (!this.is_loggedIn) {
return (h(Host, null, h("slot", { name: "login-btn" }, h("button", { type: "button", class: "s-user-menu-login-btn", onClick: () => salla.event.dispatch('login::open'), innerHTML: UserCircle }))));
}
if (this.inline) {
return (h(Host, null, h("ul", { class: "s-user-menu-inline" }, Object.entries(this.items).map((item, i) => this.getMenuItem(item, i)))));
}
return (h(Host, null, h("div", { class: {
's-user-menu-wrapper': true,
's-user-menu-relative-dropdown': this.relativeDropdown,
} }, this.getTheHeader(), h("div", { class: { 's-user-menu-toggler': true, opened: this.opened } }, h("div", { class: "s-user-menu-dropdown", onClick: e => e.stopPropagation(), onKeyUp: e => e.stopPropagation() }, this.showHeader ? (h("div", { class: "s-user-menu-dropdown-header" }, h("img", { src: this.avatar, alt: `${this.first_name} ${this.last_name}` }), h("div", { class: "s-user-menu-dropdown-header-content" }, h("span", null, this.hello), h("p", null, this.first_name, " ", this.last_name)), h("button", { type: 'button', class: "s-user-menu-dropdown-header-close", innerHTML: Cancel, onClick: () => { this.opened = false; } }))) : (''), h("ul", { class: "s-user-menu-dropdown-list" }, Object.entries(this.items).map((item, i) => this.getMenuItem(item, i))))))));
}
componentDidLoad() {
document.lazyLoadInstance?.update(this.host.querySelectorAll('.lazy'));
}
buildProfileUrl(token) {
return `${Salla.config.get('account.url', 'https://accounts.salla.com')}/${salla.config.get('user.language_code')}/user?store=${Salla.config.get('store.id')}&info=${encodeURIComponent(token)}`;
}
get host() { return getElement(this); }
};
SallaUserMenu.style = sallaUserMenuCss;
export { SallaUserMenu as salla_user_menu };