UNPKG

@salla.sa/twilight-components

Version:
284 lines (275 loc) 18.6 kB
/*! * 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 };