@vendasta/store
Version:
Components and data for Store
181 lines • 21.6 kB
JavaScript
/**
* @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"]}