UNPKG

@vendasta/store

Version:

Components and data for Store

115 lines 30.4 kB
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'; import * as i0 from "@angular/core"; import * as i1 from "../header-container/header-container.component"; import * as i2 from "../selling-info/selling-info.component"; import * as i3 from "../products-nav/products-nav.component"; import * as i4 from "../faqs/faqs.component"; import * as i5 from "@vendasta/uikit"; import * as i6 from "../files/files.component"; import * as i7 from "@angular/common"; 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 [].concat(products || []).concat(addons || []); }); this.orderedItemsList$ = combineLatest(this.combinedItemsList$, this.pkg$, (combined, pkg) => { return orderProductsAndAddons(combined, pkg ? pkg.productOrder || [] : []); }); 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. const initialSelectedItem$ = nonEmptyCombinedItemsList.pipe(map((_) => 0), take(1)); 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); })); } set standAloneProducts(standAloneProducts) { this.standAloneProducts$$.next(standAloneProducts); } set products(products) { this.products$.next(products); } set addons(addons) { this.addons$.next(addons); } set pkg(pkg) { this.pkg$.next(pkg); } static getLmiCategories(products) { let lmiCategories = []; if (products) { products.forEach((product) => { lmiCategories = lmiCategories.concat(product.getLmiCategoryNames()); }); } return new Set(lmiCategories); } 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 ''; } static getProductNameForSingleAddon(items) { if (items && items.length === 1 && items[0] instanceof Addon) { const addon = items[0]; return addon.productName; } } selectItem(i) { this.itemSelectedEvent$.next(i); } emitGetItNow() { this.getItNow.emit(); } } VaPackageDetailsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.2", ngImport: i0, type: VaPackageDetailsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); VaPackageDetailsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.0.2", type: VaPackageDetailsComponent, selector: "app-store-package-details", inputs: { showActionButton: "showActionButton", showPricing: "showPricing", actionLabel: "actionLabel", standAloneProducts: "standAloneProducts", products: "products", addons: "addons", pkg: "pkg" }, outputs: { getItNow: "getItNow" }, ngImport: i0, template: "<div *ngIf=\"pkg$ | async as pkg\" class=\"page\">\n <app-header-container\n [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 ></app-header-container>\n <ng-container *ngIf=\"pkg?.content\">\n <app-selling-info [description]=\"pkg.content\"></app-selling-info>\n </ng-container>\n\n <app-products-nav\n [items]=\"orderedItemsList$ | async\"\n [hideItemsNav]=\"pkg?.hide_product_icons_and_names\"\n (itemSelected)=\"selectItem($event)\"\n ></app-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=\"\n selectedItem.endUserMarketing?.description ||\n selectedItem.endUserMarketing?.keySellingPoints ||\n selectedItem.description ||\n selectedItem.keySellingPoints\n \"\n >\n <app-selling-info\n [description]=\"\n selectedItem.endUserMarketing?.description ||\n selectedItem.description\n \"\n [keySellingPoints]=\"\n selectedItem.endUserMarketing?.keySellingPoints ||\n selectedItem.keySellingPoints\n \"\n ></app-selling-info>\n </section>\n\n <section\n *ngIf=\"\n (selectedItem.endUserMarketing?.faqs?.length > 0 &&\n selectedItem.endUserMarketing?.faqs[0].question) ||\n selectedItem.faqs?.length > 0\n \"\n >\n <h2 class=\"va-component-title\">FAQs</h2>\n <app-faqs\n [faqs]=\"selectedItem.endUserMarketing?.faqs || selectedItem.faqs\"\n ></app-faqs>\n </section>\n </div>\n\n <div class=\"right-column\">\n <section\n *ngIf=\"\n selectedItem.screenshotUrls?.length > 0 ||\n selectedItem.screenshots?.length > 0\n \"\n >\n <h2 class=\"va-component-title\">Gallery</h2>\n <va-image-gallery\n [imageUrls]=\"\n selectedItem.screenshotUrls || selectedItem.screenshots\n \"\n ></va-image-gallery>\n </section>\n\n <section\n *ngIf=\"\n selectedItem.endUserMarketing?.files?.length > 0 ||\n selectedItem.files?.length > 0\n \"\n >\n <h2 class=\"va-component-title\">Files</h2>\n <app-store-files\n [files]=\"selectedItem.endUserMarketingFiles || selectedItem.files\"\n ></app-store-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:#ffffff;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%}@media screen and (max-width: 600px){.right-column{width:100%}}.package-content{display:flex}@media screen and (max-width: 600px){.package-content{flex-direction:column}}\n"], components: [{ type: i1.VaHeaderContainerComponent, selector: "app-header-container", inputs: ["iconUrl", "title", "tagline", "prerequisite", "chipLabels", "pricing", "billedProduct", "loaded", "hasVerifiedContract", "pricingLabel", "primaryPricingActionLabel", "pricingActionLabel", "pricingActionEnabled", "actionEnabled", "actionLabel", "showAction", "showPricing", "showEnableAddon", "prerequisiteLabel"], outputs: ["prerequisiteSelected", "actionSelected", "primaryPricingActionSelected", "pricingActionSelected"] }, { type: i2.VaSellingInfoComponent, selector: "app-selling-info", inputs: ["description", "keySellingPoints"] }, { type: i3.VaProductsNavComponent, selector: "app-products-nav", inputs: ["items", "hideItemsNav"], outputs: ["itemSelected"] }, { type: i4.VaFaqsComponent, selector: "app-faqs", inputs: ["faqs"] }, { type: i5.VaImageGalleryComponent, selector: "va-image-gallery", inputs: ["imageUrls", "clickable"] }, { type: i6.VaFilesComponent, selector: "app-store-files", inputs: ["files"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], pipes: { "async": i7.AsyncPipe } }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.2", ngImport: i0, type: VaPackageDetailsComponent, decorators: [{ type: Component, args: [{ selector: 'app-store-package-details', template: "<div *ngIf=\"pkg$ | async as pkg\" class=\"page\">\n <app-header-container\n [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 ></app-header-container>\n <ng-container *ngIf=\"pkg?.content\">\n <app-selling-info [description]=\"pkg.content\"></app-selling-info>\n </ng-container>\n\n <app-products-nav\n [items]=\"orderedItemsList$ | async\"\n [hideItemsNav]=\"pkg?.hide_product_icons_and_names\"\n (itemSelected)=\"selectItem($event)\"\n ></app-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=\"\n selectedItem.endUserMarketing?.description ||\n selectedItem.endUserMarketing?.keySellingPoints ||\n selectedItem.description ||\n selectedItem.keySellingPoints\n \"\n >\n <app-selling-info\n [description]=\"\n selectedItem.endUserMarketing?.description ||\n selectedItem.description\n \"\n [keySellingPoints]=\"\n selectedItem.endUserMarketing?.keySellingPoints ||\n selectedItem.keySellingPoints\n \"\n ></app-selling-info>\n </section>\n\n <section\n *ngIf=\"\n (selectedItem.endUserMarketing?.faqs?.length > 0 &&\n selectedItem.endUserMarketing?.faqs[0].question) ||\n selectedItem.faqs?.length > 0\n \"\n >\n <h2 class=\"va-component-title\">FAQs</h2>\n <app-faqs\n [faqs]=\"selectedItem.endUserMarketing?.faqs || selectedItem.faqs\"\n ></app-faqs>\n </section>\n </div>\n\n <div class=\"right-column\">\n <section\n *ngIf=\"\n selectedItem.screenshotUrls?.length > 0 ||\n selectedItem.screenshots?.length > 0\n \"\n >\n <h2 class=\"va-component-title\">Gallery</h2>\n <va-image-gallery\n [imageUrls]=\"\n selectedItem.screenshotUrls || selectedItem.screenshots\n \"\n ></va-image-gallery>\n </section>\n\n <section\n *ngIf=\"\n selectedItem.endUserMarketing?.files?.length > 0 ||\n selectedItem.files?.length > 0\n \"\n >\n <h2 class=\"va-component-title\">Files</h2>\n <app-store-files\n [files]=\"selectedItem.endUserMarketingFiles || selectedItem.files\"\n ></app-store-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:#ffffff;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%}@media screen and (max-width: 600px){.right-column{width:100%}}.package-content{display:flex}@media screen and (max-width: 600px){.package-content{flex-direction:column}}\n"] }] }], ctorParameters: function () { return []; }, 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 }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"package-details.component.js","sourceRoot":"","sources":["../../../../../../libs/store/src/lib/package-details/package-details.component.ts","../../../../../../libs/store/src/lib/package-details/package-details.component.html"],"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: 'app-store-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(\n      this.combinedItemsList$,\n      this.pkg$,\n      (combined: (Product | Addon)[], pkg: Package) => {\n        return orderProductsAndAddons(combined, pkg ? pkg.productOrder || [] : []);\n      },\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","<div *ngIf=\"pkg$ | async as pkg\" class=\"page\">\n  <app-header-container\n    [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  ></app-header-container>\n  <ng-container *ngIf=\"pkg?.content\">\n    <app-selling-info [description]=\"pkg.content\"></app-selling-info>\n  </ng-container>\n\n  <app-products-nav\n    [items]=\"orderedItemsList$ | async\"\n    [hideItemsNav]=\"pkg?.hide_product_icons_and_names\"\n    (itemSelected)=\"selectItem($event)\"\n  ></app-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=\"\n            selectedItem.endUserMarketing?.description ||\n            selectedItem.endUserMarketing?.keySellingPoints ||\n            selectedItem.description ||\n            selectedItem.keySellingPoints\n          \"\n        >\n          <app-selling-info\n            [description]=\"\n              selectedItem.endUserMarketing?.description ||\n              selectedItem.description\n            \"\n            [keySellingPoints]=\"\n              selectedItem.endUserMarketing?.keySellingPoints ||\n              selectedItem.keySellingPoints\n            \"\n          ></app-selling-info>\n        </section>\n\n        <section\n          *ngIf=\"\n            (selectedItem.endUserMarketing?.faqs?.length > 0 &&\n              selectedItem.endUserMarketing?.faqs[0].question) ||\n            selectedItem.faqs?.length > 0\n          \"\n        >\n          <h2 class=\"va-component-title\">FAQs</h2>\n          <app-faqs\n            [faqs]=\"selectedItem.endUserMarketing?.faqs || selectedItem.faqs\"\n          ></app-faqs>\n        </section>\n      </div>\n\n      <div class=\"right-column\">\n        <section\n          *ngIf=\"\n            selectedItem.screenshotUrls?.length > 0 ||\n            selectedItem.screenshots?.length > 0\n          \"\n        >\n          <h2 class=\"va-component-title\">Gallery</h2>\n          <va-image-gallery\n            [imageUrls]=\"\n              selectedItem.screenshotUrls || selectedItem.screenshots\n            \"\n          ></va-image-gallery>\n        </section>\n\n        <section\n          *ngIf=\"\n            selectedItem.endUserMarketing?.files?.length > 0 ||\n            selectedItem.files?.length > 0\n          \"\n        >\n          <h2 class=\"va-component-title\">Files</h2>\n          <app-store-files\n            [files]=\"selectedItem.endUserMarketingFiles || selectedItem.files\"\n          ></app-store-files>\n        </section>\n      </div>\n    </div>\n  </ng-container>\n</div>\n"]}