UNPKG

@vendasta/store

Version:

Components and data for Store

181 lines 21.6 kB
/** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { Component, EventEmitter, Input, Output } from '@angular/core'; import { BehaviorSubject, ReplaySubject, combineLatest, merge as observableMerge } from 'rxjs'; import { Package } from './package'; import { Addon } from '@vendasta/core/shared'; import { skipWhile, shareReplay, map, take } from 'rxjs/operators'; import { orderProductsAndAddons } from './utils'; export class VaPackageDetailsComponent { constructor() { this.showActionButton = true; this.showPricing = true; this.actionLabel = 'Get It Now'; this.getItNow = new EventEmitter(); this.standAloneProducts$$ = new BehaviorSubject([]); this.products$ = new BehaviorSubject([]); this.addons$ = new BehaviorSubject([]); this.pkg$ = new ReplaySubject(1); this.itemSelectedEvent$ = new ReplaySubject(1); this.combinedItemsList$ = combineLatest(this.products$, this.addons$, (products, addons) => { return (/** @type {?} */ ([].concat(products || []).concat(addons || []))); }); this.orderedItemsList$ = combineLatest(this.combinedItemsList$, this.pkg$, (combined, pkg) => { return orderProductsAndAddons(combined, pkg ? pkg.productOrder || [] : []); }); /** @type {?} */ const nonEmptyCombinedItemsList = this.combinedItemsList$.pipe(skipWhile(items => !items || items.length === 0), shareReplay(1)); // Need to map to an index because that's how product selection works. /** @type {?} */ const initialSelectedItem$ = nonEmptyCombinedItemsList.pipe(map((_) => 0), take(1)); /** @type {?} */ const mergedSelection = observableMerge(initialSelectedItem$, this.itemSelectedEvent$); this.selectedItem$ = combineLatest(mergedSelection, this.orderedItemsList$).pipe(map(([index, items]) => { return items[index]; })); this.lmiCategories$ = this.products$.pipe(skipWhile(products => !products || products.length === 0), map(products => { return VaPackageDetailsComponent.getLmiCategories(products); })); this.productNameForSingleAddon$ = nonEmptyCombinedItemsList.pipe(map(items => { return VaPackageDetailsComponent.getProductNameForSingleAddon(items); })); this.displayTagName$ = combineLatest(this.pkg$, this.combinedItemsList$).pipe(map(([pkg, items]) => { return VaPackageDetailsComponent.getDisplayTagline(pkg, items); })); } /** * @param {?} standAloneProducts * @return {?} */ set standAloneProducts(standAloneProducts) { this.standAloneProducts$$.next(standAloneProducts); } /** * @param {?} products * @return {?} */ set products(products) { this.products$.next(products); } /** * @param {?} addons * @return {?} */ set addons(addons) { this.addons$.next(addons); } /** * @param {?} pkg * @return {?} */ set pkg(pkg) { this.pkg$.next(pkg); } /** * @param {?} products * @return {?} */ static getLmiCategories(products) { /** @type {?} */ let lmiCategories = []; if (products) { products.forEach(product => { lmiCategories = lmiCategories.concat(product.getLmiCategoryNames()); }); } return new Set(lmiCategories); } /** * @param {?} pkg * @param {?} items * @return {?} */ static getDisplayTagline(pkg, items) { if (pkg && pkg.tagline) { return pkg.tagline; } else if (items && items.length === 1) { // Single product / addon package. return items[0].tagline || ''; } return ''; } /** * @param {?} items * @return {?} */ static getProductNameForSingleAddon(items) { if (items && items.length === 1 && items[0] instanceof Addon) { /** @type {?} */ const addon = (/** @type {?} */ (items[0])); return addon.productName; } } /** * @param {?} i * @return {?} */ selectItem(i) { this.itemSelectedEvent$.next(i); } /** * @return {?} */ emitGetItNow() { this.getItNow.emit(); } } VaPackageDetailsComponent.decorators = [ { type: Component, args: [{ selector: 'va-package-details', template: "<div *ngIf=\"pkg$ | async as pkg\" class=\"page\">\n <va-header-container [iconUrl]=\"pkg.icon\"\n [title]=\"pkg.name\"\n [tagline]=\"displayTagName$ | async\"\n [prerequisite]=\"productNameForSingleAddon$ | async\"\n [chipLabels]=\"lmiCategories$ | async\"\n [pricing]=\"pkg.pricing\"\n [pricingLabel]=\"'Pricing'\"\n [showAction]=\"showActionButton\"\n [showPricing]=\"showPricing\"\n [actionLabel]=\"actionLabel\"\n [actionEnabled]=true\n (actionSelected)=\"emitGetItNow()\"\n ></va-header-container>\n <ng-container *ngIf=\"pkg?.content\">\n <va-selling-info [description]=\"pkg.content\"></va-selling-info>\n </ng-container>\n\n <va-products-nav [items]=\"orderedItemsList$ | async\"\n [hideItemsNav]=\"pkg?.hide_product_icons_and_names\"\n (itemSelected)=\"selectItem($event)\"></va-products-nav>\n\n <ng-container *ngIf=\"selectedItem$ | async as selectedItem\">\n <div *ngIf=\"!pkg?.hide_product_details\" class=\"package-content\">\n <div class=\"left-column\">\n <section\n *ngIf=\"(selectedItem.endUserMarketing?.description || selectedItem.endUserMarketing?.keySellingPoints) ||\n (selectedItem.description || selectedItem.keySellingPoints)\">\n <va-selling-info [description]=\"selectedItem.endUserMarketing?.description || selectedItem.description\"\n [keySellingPoints]=\"selectedItem.endUserMarketing?.keySellingPoints || selectedItem.keySellingPoints\"></va-selling-info>\n </section>\n\n <section *ngIf=\"(selectedItem.endUserMarketing?.faqs?.length > 0\n && selectedItem.endUserMarketing?.faqs[0].question) || selectedItem.faqs?.length > 0\">\n <h2 class=\"va-component-title\">FAQs</h2>\n <va-faqs [faqs]=\"selectedItem.endUserMarketing?.faqs || selectedItem.faqs\"></va-faqs>\n </section>\n </div>\n\n <div class=\"right-column\">\n <section *ngIf=\"selectedItem.screenshotUrls?.length > 0 || selectedItem.screenshots?.length > 0\">\n <h2 class=\"va-component-title\">Gallery</h2>\n <va-image-gallery [imageUrls]=\"selectedItem.screenshotUrls || selectedItem.screenshots\"></va-image-gallery>\n </section>\n\n <section *ngIf=\"selectedItem.endUserMarketing?.files?.length > 0 || selectedItem.files?.length > 0\">\n <h2 class=\"va-component-title\">Files</h2>\n <va-files [files]=\"selectedItem.endUserMarketingFiles || selectedItem.files\"></va-files>\n </section>\n </div>\n </div>\n </ng-container>\n</div>\n", styles: [":host{display:block;font-size:14px;line-height:1.4}:host *{box-sizing:border-box}.va-component-title{font-size:24px;font-weight:300;margin-top:0}@media screen and (min-width:600px){.va-component-title{font-size:32px}}img{max-width:100%}va-icon{display:inline-block}.page{position:relative;margin:0 auto 20px;background:#fff;color:#212121;box-shadow:0 3px 10px rgba(33,33,33,.3);overflow:hidden;font-size:16px}.page .item-amount{color:#9e9e9e;font-size:14px;text-align:right;margin:0 24px 8px}section{padding:24px}.left-column{width:66%}@media screen and (max-width:600px){.left-column{width:100%}}.right-column{width:34%}.package-content{display:flex}@media screen and (max-width:600px){.right-column{width:100%}.package-content{flex-direction:column}}"] }] } ]; /** @nocollapse */ VaPackageDetailsComponent.ctorParameters = () => []; VaPackageDetailsComponent.propDecorators = { showActionButton: [{ type: Input }], showPricing: [{ type: Input }], actionLabel: [{ type: Input }], getItNow: [{ type: Output }], standAloneProducts: [{ type: Input }], products: [{ type: Input }], addons: [{ type: Input }], pkg: [{ type: Input }] }; if (false) { /** @type {?} */ VaPackageDetailsComponent.prototype.showActionButton; /** @type {?} */ VaPackageDetailsComponent.prototype.showPricing; /** @type {?} */ VaPackageDetailsComponent.prototype.actionLabel; /** @type {?} */ VaPackageDetailsComponent.prototype.getItNow; /** @type {?} */ VaPackageDetailsComponent.prototype.standAloneProducts$$; /** @type {?} */ VaPackageDetailsComponent.prototype.products$; /** @type {?} */ VaPackageDetailsComponent.prototype.addons$; /** @type {?} */ VaPackageDetailsComponent.prototype.pkg$; /** @type {?} */ VaPackageDetailsComponent.prototype.itemSelectedEvent$; /** @type {?} */ VaPackageDetailsComponent.prototype.selectedItem$; /** @type {?} */ VaPackageDetailsComponent.prototype.combinedItemsList$; /** @type {?} */ VaPackageDetailsComponent.prototype.orderedItemsList$; /** @type {?} */ VaPackageDetailsComponent.prototype.lmiCategories$; /** @type {?} */ VaPackageDetailsComponent.prototype.productNameForSingleAddon$; /** @type {?} */ VaPackageDetailsComponent.prototype.displayTagName$; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"package-details.component.js","sourceRoot":"ng://@vendasta/store/","sources":["lib/package-details/package-details.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { BehaviorSubject, Observable, ReplaySubject, combineLatest, merge as observableMerge } from 'rxjs';\nimport { Package } from './package';\nimport { Addon } from '@vendasta/core/shared';\nimport { Product } from '../shared/product';\nimport { skipWhile, shareReplay, map, take } from 'rxjs/operators';\nimport { orderProductsAndAddons } from './utils';\n\n@Component({\n  selector: 'va-package-details',\n  templateUrl: 'package-details.component.html',\n  styleUrls: ['./package-details.component.scss']\n})\nexport class VaPackageDetailsComponent {\n  @Input() showActionButton = true;\n  @Input() showPricing = true;\n  @Input() actionLabel = 'Get It Now';\n\n  @Output() getItNow = new EventEmitter();\n\n  standAloneProducts$$: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([]);\n  products$: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([]);\n  addons$: BehaviorSubject<Addon[]> = new BehaviorSubject<Addon[]>([]);\n  pkg$: ReplaySubject<Package> = new ReplaySubject<Package>(1);\n  itemSelectedEvent$: ReplaySubject<number> = new ReplaySubject<number>(1);\n\n  selectedItem$: Observable<Product | Addon>;\n  combinedItemsList$: Observable<(Product | Addon)[]>;\n  orderedItemsList$: Observable<(Product | Addon)[]>;\n  lmiCategories$: Observable<Set<string>>;\n  productNameForSingleAddon$: Observable<string>;\n  displayTagName$: Observable<string>;\n\n  @Input()\n  set standAloneProducts(standAloneProducts: Product[]) {\n    this.standAloneProducts$$.next(standAloneProducts);\n  }\n\n  @Input()\n  set products(products: Product[]) {\n    this.products$.next(products);\n  }\n\n  @Input()\n  set addons(addons: Addon[]) {\n    this.addons$.next(addons);\n  }\n\n  @Input()\n  set pkg(pkg: Package) {\n    this.pkg$.next(pkg);\n  }\n\n  static getLmiCategories(products: Product[]): Set<string> {\n    let lmiCategories: string[] = [];\n    if (products) {\n      products.forEach(product => {\n        lmiCategories = lmiCategories.concat(product.getLmiCategoryNames());\n      });\n    }\n    return new Set(lmiCategories);\n  }\n\n  static getDisplayTagline(pkg: Package, items: (Product | Addon)[]): string {\n    if (pkg && pkg.tagline) {\n      return pkg.tagline;\n    } else if (items && items.length === 1) {\n      // Single product / addon package.\n      return items[0].tagline || '';\n    }\n    return '';\n  }\n\n  static getProductNameForSingleAddon(items: (Product | Addon)[]): string {\n    if (items && items.length === 1 && items[0] instanceof Addon) {\n      const addon = items[0] as Addon;\n      return addon.productName;\n    }\n  }\n\n  constructor() {\n    this.combinedItemsList$ = combineLatest(this.products$, this.addons$, (products: Product[], addons: Addon[]) => {\n      return [].concat(products || []).concat(addons || []) as (Product | Addon)[];\n    });\n\n    this.orderedItemsList$ = combineLatest(this.combinedItemsList$, this.pkg$, (combined: (Product | Addon)[], pkg: Package) => {\n      return orderProductsAndAddons(combined, pkg ? pkg.productOrder || [] : [])\n    });\n\n    const nonEmptyCombinedItemsList = this.combinedItemsList$.pipe(\n      skipWhile(items => !items || items.length === 0),\n      shareReplay(1)\n    );\n\n    // Need to map to an index because that's how product selection works.\n    const initialSelectedItem$ = nonEmptyCombinedItemsList.pipe(\n      map((_: any) => 0),\n      take(1)\n    );\n\n    const mergedSelection = observableMerge(initialSelectedItem$, this.itemSelectedEvent$);\n\n    this.selectedItem$ = combineLatest(mergedSelection, this.orderedItemsList$).pipe(\n      map(([index, items]) => {\n        return items[index];\n      })\n    );\n\n    this.lmiCategories$ = this.products$.pipe(\n      skipWhile(products => !products || products.length === 0),\n      map(products => {\n        return VaPackageDetailsComponent.getLmiCategories(products);\n      })\n    );\n\n    this.productNameForSingleAddon$ = nonEmptyCombinedItemsList.pipe(\n      map(items => {\n        return VaPackageDetailsComponent.getProductNameForSingleAddon(items);\n      })\n    );\n\n    this.displayTagName$ = combineLatest(this.pkg$, this.combinedItemsList$).pipe(\n      map(([pkg, items]) => {\n        return VaPackageDetailsComponent.getDisplayTagline(pkg, items);\n      })\n    );\n  }\n\n  selectItem(i: number): void {\n    this.itemSelectedEvent$.next(i);\n  }\n\n  emitGetItNow(): void {\n    this.getItNow.emit();\n  }\n}\n"]}