@vendasta/store
Version:
Components and data for Store
184 lines • 34.2 kB
JavaScript
import { Component, Input, Output, EventEmitter } from '@angular/core';
import { BehaviorSubject, combineLatest } from 'rxjs';
import { debounceTime, distinctUntilChanged, map, startWith, filter, take, skip } from 'rxjs/operators';
import { TranslateService } from '@ngx-translate/core';
import { Filters, FilterSection, TabbedFilterField, CheckboxFilterField, FilterService, } from '@vendasta/uikit';
import { ProductAnalyticsService } from '@vendasta/product-analytics';
import { LMI_CATEGORIES } from '../lmi-categories';
import { Categories, ShoppingCartItemType } from '@vendasta/marketplace-packages';
import * as i0 from "@angular/core";
import * as i1 from "@vendasta/uikit";
import * as i2 from "@vendasta/product-analytics";
import * as i3 from "@ngx-translate/core";
import * as i4 from "../store.component";
import * as i5 from "@angular/common";
export const STOREFRONT_FILTER_NAME = 'Store Filters';
export class StorefrontComponent {
constructor(filterService, snowplowService, translateService) {
this.filterService = filterService;
this.snowplowService = snowplowService;
this.translateService = translateService;
this.useCustomCase = false;
this.showFilters = true;
this.initialFilteredStoreItems = null;
this.showAllPrices = false;
this.filtersDisabled = false;
this.categorySelected = new EventEmitter();
this.productSelected = new EventEmitter();
this.getDataForCategory = new EventEmitter();
this.categories$$ = new BehaviorSubject([]);
this.packagesMap$$ = new BehaviorSubject(new Map());
this.searchTerm$$ = new BehaviorSubject('');
this.selectedCateoryId$$ = new BehaviorSubject('');
this.filterByInitialStoreItemIds$$ = new BehaviorSubject(false);
}
set categories(value) {
this.categories$$.next(value || []);
}
set categoryIdToStoreItemListMap(value) {
this.packagesMap$$.next(value || new Map());
}
ngOnInit() {
this.filterByInitialStoreItemIds$$.next(!!this.initialFilteredStoreItems && !!this.initialFilteredStoreItems.storeItemIds
? this.initialFilteredStoreItems.storeItemIds.length > 0
: false);
const lmiCategoryCheckboxList = LMI_CATEGORIES.map((category) => {
return new CheckboxFilterField({
name: this.translateService.instant(category.i18nKey),
id: category.categoryId,
value: false,
condition: (item) => {
let lmiCategories = [];
if (item.lmiCategories) {
lmiCategories = item.lmiCategories.map((cat) => {
return Categories[cat].toLowerCase();
});
}
return lmiCategories.indexOf(category.categoryId) > -1;
},
});
});
const lmiCategoryFilterSection = new FilterSection({
title: 'FRONTEND.STORE.STOREFRONT_FILTERS.LMI_CATEGORY',
type: 'or',
fields: lmiCategoryCheckboxList,
});
this.lmiCategoryFilter = new Filters(STOREFRONT_FILTER_NAME, [lmiCategoryFilterSection]);
const packageIdsCheckbox = new CheckboxFilterField({
name: !!this.initialFilteredStoreItems ? this.initialFilteredStoreItems.filterName : '',
id: 'package_id_checkbox',
value: !!this.initialFilteredStoreItems && !!this.initialFilteredStoreItems.storeItemIds
? this.initialFilteredStoreItems.storeItemIds.length > 0
: false,
});
const packageIdsFilterSection = new FilterSection({
title: 'Store item IDs',
type: 'and',
fields: [packageIdsCheckbox],
});
packageIdsFilterSection.hideSelectors = true;
this.filters$ = this.categories$$.pipe(map((categories) => {
const tabbedFilterField = new TabbedFilterField({
name: '',
value: null,
id: 'categorytab',
options: categories,
});
const filters = new Filters('FRONTEND.STORE.STOREFRONT_FILTERS.TITLE', [
new FilterSection({
title: 'FRONTEND.STORE.STOREFRONT_FILTERS.CATEGORIES',
type: 'or',
fields: [tabbedFilterField],
}),
lmiCategoryFilterSection,
packageIdsFilterSection,
]);
filters.id = STOREFRONT_FILTER_NAME.toLowerCase().replace(' ', '_');
return this.showFilters ? filters : null;
}));
this.filterService.setFilters(this.lmiCategoryFilter);
this.searchTermSubscription = this.searchTerm$$
.asObservable()
.pipe(debounceTime(300), distinctUntilChanged(), skip(1))
.subscribe((searchTerm) => this.snowplowService.trackClick('store', 'search-input', searchTerm));
const selectedLMIFilters$ = this.filterService.filters.fieldValuesObservable.pipe(startWith([]), map(() => {
return this.lmiCategoryFilter.fields.filter((item) => item.value);
}));
this.showLMIFilter$ = selectedLMIFilters$.pipe(filter((selectedLMIFilters) => !!selectedLMIFilters), take(1), map((selectedLMIFilters) => selectedLMIFilters.length > 0));
this.filteredItems$ = combineLatest([
this.packagesMap$$,
this.searchTerm$$,
this.selectedCateoryId$$,
selectedLMIFilters$,
this.filterByInitialStoreItemIds$$,
]).pipe(map(([packages, searchTerm, selectedCategoryId, selectedLMIFilters, filterByInitialStoreItemIds]) => {
let storeItems = [];
if (packages.size) {
storeItems = selectedCategoryId
? packages.get(selectedCategoryId)
: packages.get(Array.from(packages.keys())[0]);
}
storeItems = storeItems ? this.lmiCategoryFilter.applyFilters(storeItems) : [];
if (filterByInitialStoreItemIds) {
storeItems = storeItems.filter((storeItem) => this.initialFilteredStoreItems.storeItemIds.includes(storeItem.id));
}
return searchTerm
? storeItems.filter((item) => item?.name?.toLowerCase().indexOf(searchTerm.toLowerCase()) !== -1)
: storeItems;
}));
}
ngOnDestroy() {
this.searchTermSubscription.unsubscribe();
}
onSearchTermChanged(term) {
this.searchTerm$$.next(term);
}
onFieldValueChanged(filterField) {
this.snowplowService.trackClick('store', 'tab-filter', filterField.value.id);
if (filterField.id === 'package_id_checkbox') {
this.filterByInitialStoreItemIds$$.next(filterField.value);
return;
}
if (!(filterField instanceof TabbedFilterField)) {
return;
}
this.selectedCateoryId$$.next(filterField.value.id);
this.categorySelected.emit(filterField.value);
const currentStoreItemsForSelectCategory = this.packagesMap$$.getValue().get(filterField.value.id);
if (!(currentStoreItemsForSelectCategory && currentStoreItemsForSelectCategory.length)) {
this.getDataForCategory.emit(filterField.value);
}
}
storeItemClicked(item) {
const label = item.type === ShoppingCartItemType.SHOPPING_CART_ITEM_TYPE_PRODUCT ? 'view-app' : 'view-package';
this.snowplowService.trackClick('store', label, item.id);
this.productSelected.emit({ id: item.id, type: item.type });
}
}
StorefrontComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.2", ngImport: i0, type: StorefrontComponent, deps: [{ token: i1.FilterService }, { token: i2.ProductAnalyticsService }, { token: i3.TranslateService }], target: i0.ɵɵFactoryTarget.Component });
StorefrontComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.0.2", type: StorefrontComponent, selector: "app-storefront", inputs: { categories: "categories", categoryIdToStoreItemListMap: "categoryIdToStoreItemListMap", useCustomCase: "useCustomCase", showFilters: "showFilters", initialFilteredStoreItems: "initialFilteredStoreItems", showAllPrices: "showAllPrices", filtersDisabled: "filtersDisabled" }, outputs: { categorySelected: "categorySelected", productSelected: "productSelected", getDataForCategory: "getDataForCategory" }, ngImport: i0, template: "<va-filter-container\n [filters]=\"filters$ | async\"\n (searchTermChanged)=\"onSearchTermChanged($event)\"\n (fieldValueChanged)=\"onFieldValueChanged($event)\"\n [theme]=\"'plain'\"\n [filterIsOpen]=\"showLMIFilter$ | async\"\n [useCustomCase]=\"useCustomCase\"\n [filtersDisabled]=\"filtersDisabled\"\n>\n <app-store\n [items]=\"filteredItems$ | async\"\n [searchable]=\"false\"\n (itemClicked)=\"storeItemClicked($event)\"\n [showAllPrices]=\"showAllPrices\"\n content\n ></app-store>\n</va-filter-container>\n", styles: [".public-store-container{padding:20px}app-store{margin-top:20px;display:block}:host ::ng-deep .filter-wrapper:not(.hide)+.content-container{padding-left:20px}::ng-deep .toolbar ::ng-deep div.table-controls-row .filter-button.disabled{display:none}@media only screen and (device-width: 375px) and (-webkit-device-pixel-ratio: 2){::ng-deep va-filter-tabbed{width:280px}::ng-deep .filter-container .toolbar.toolbar{background:none}}@media only screen and (device-width: 414px) and (-webkit-device-pixel-ratio: 3){::ng-deep va-filter-tabbed{width:280px}::ng-deep .filter-container .toolbar.toolbar{background:none}}\n"], components: [{ type: i1.FilterContainerComponent, selector: "va-filter-container", inputs: ["showFilterSharing", "searchable", "toolbarText", "showToolbar", "searchDisabled", "filtersDisabled", "filterIsOpen", "filters", "savedFilters", "theme", "useCustomCase", "enabledSaveSearchTerm", "contextKey"], outputs: ["searchTermChanged", "fieldValueChanged", "savedFilterSelected", "removeSavedFilter", "saveCurrentFilter", "exportFilterDialog", "importFilterDialog"] }, { type: i4.StoreComponent, selector: "app-store", inputs: ["items", "showAllPrices", "searchable"], outputs: ["itemClicked"] }], pipes: { "async": i5.AsyncPipe } });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.2", ngImport: i0, type: StorefrontComponent, decorators: [{
type: Component,
args: [{ selector: 'app-storefront', template: "<va-filter-container\n [filters]=\"filters$ | async\"\n (searchTermChanged)=\"onSearchTermChanged($event)\"\n (fieldValueChanged)=\"onFieldValueChanged($event)\"\n [theme]=\"'plain'\"\n [filterIsOpen]=\"showLMIFilter$ | async\"\n [useCustomCase]=\"useCustomCase\"\n [filtersDisabled]=\"filtersDisabled\"\n>\n <app-store\n [items]=\"filteredItems$ | async\"\n [searchable]=\"false\"\n (itemClicked)=\"storeItemClicked($event)\"\n [showAllPrices]=\"showAllPrices\"\n content\n ></app-store>\n</va-filter-container>\n", styles: [".public-store-container{padding:20px}app-store{margin-top:20px;display:block}:host ::ng-deep .filter-wrapper:not(.hide)+.content-container{padding-left:20px}::ng-deep .toolbar ::ng-deep div.table-controls-row .filter-button.disabled{display:none}@media only screen and (device-width: 375px) and (-webkit-device-pixel-ratio: 2){::ng-deep va-filter-tabbed{width:280px}::ng-deep .filter-container .toolbar.toolbar{background:none}}@media only screen and (device-width: 414px) and (-webkit-device-pixel-ratio: 3){::ng-deep va-filter-tabbed{width:280px}::ng-deep .filter-container .toolbar.toolbar{background:none}}\n"] }]
}], ctorParameters: function () { return [{ type: i1.FilterService }, { type: i2.ProductAnalyticsService }, { type: i3.TranslateService }]; }, propDecorators: { categories: [{
type: Input
}], categoryIdToStoreItemListMap: [{
type: Input
}], useCustomCase: [{
type: Input
}], showFilters: [{
type: Input
}], initialFilteredStoreItems: [{
type: Input
}], showAllPrices: [{
type: Input
}], filtersDisabled: [{
type: Input
}], categorySelected: [{
type: Output
}], productSelected: [{
type: Output
}], getDataForCategory: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storefront.component.js","sourceRoot":"","sources":["../../../../../../libs/store/src/lib/storefront/storefront.component.ts","../../../../../../libs/store/src/lib/storefront/storefront.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, Input, Output, EventEmitter, OnDestroy } from '@angular/core';\nimport { BehaviorSubject, Observable, combineLatest, Subscription } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, map, startWith, filter, take, skip } from 'rxjs/operators';\nimport { TranslateService } from '@ngx-translate/core';\n\nimport {\n  Filters,\n  FilterSection,\n  TabbedFilterField,\n  CheckboxFilterField,\n  FilterService,\n  FilterField,\n} from '@vendasta/uikit';\nimport { ProductAnalyticsService } from '@vendasta/product-analytics';\nimport { StoreItem, CategoryToItemListMap } from '../store-item';\nimport { LMI_CATEGORIES } from '../lmi-categories';\nimport { Categories, ShoppingCartItemType } from '@vendasta/marketplace-packages';\n\nexport const STOREFRONT_FILTER_NAME = 'Store Filters';\n\nexport interface StorefrontCategory {\n  id: string;\n  name: string;\n  packageIds: string[];\n}\n\nexport interface InitialFilteredStoreItems {\n  storeItemIds: string[];\n  filterName: string;\n}\n\n@Component({\n  selector: 'app-storefront',\n  templateUrl: './storefront.component.html',\n  styleUrls: ['./storefront.component.scss'],\n})\nexport class StorefrontComponent implements OnInit, OnDestroy {\n  @Input()\n  set categories(value: StorefrontCategory[]) {\n    this.categories$$.next(value || []);\n  }\n\n  @Input()\n  set categoryIdToStoreItemListMap(value: CategoryToItemListMap) {\n    this.packagesMap$$.next(value || new Map<string, StoreItem[]>());\n  }\n\n  @Input() useCustomCase = false;\n  @Input() showFilters = true;\n  @Input() initialFilteredStoreItems: InitialFilteredStoreItems = null;\n  @Input() showAllPrices = false;\n  @Input() filtersDisabled = false;\n  @Output() categorySelected = new EventEmitter<StorefrontCategory>();\n  @Output() productSelected = new EventEmitter<{ id: string; type: ShoppingCartItemType }>();\n  @Output() getDataForCategory = new EventEmitter<StorefrontCategory>();\n\n  private categories$$: BehaviorSubject<StorefrontCategory[]> = new BehaviorSubject<StorefrontCategory[]>([]);\n  private packagesMap$$: BehaviorSubject<CategoryToItemListMap> = new BehaviorSubject<CategoryToItemListMap>(\n    new Map<string, StoreItem[]>(),\n  );\n  private searchTerm$$: BehaviorSubject<string> = new BehaviorSubject<string>('');\n  private selectedCateoryId$$: BehaviorSubject<string> = new BehaviorSubject<string>('');\n  private lmiCategoryFilter: Filters;\n  filters$: Observable<Filters>;\n  filteredItems$: Observable<StoreItem[]>;\n  searchTermSubscription: Subscription;\n  showLMIFilter$: Observable<boolean>;\n\n  private filterByInitialStoreItemIds$$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n\n  constructor(\n    private filterService: FilterService,\n    private snowplowService: ProductAnalyticsService,\n    private translateService: TranslateService,\n  ) {}\n\n  ngOnInit(): void {\n    this.filterByInitialStoreItemIds$$.next(\n      !!this.initialFilteredStoreItems && !!this.initialFilteredStoreItems.storeItemIds\n        ? this.initialFilteredStoreItems.storeItemIds.length > 0\n        : false,\n    );\n\n    const lmiCategoryCheckboxList = LMI_CATEGORIES.map((category) => {\n      return new CheckboxFilterField({\n        name: this.translateService.instant(category.i18nKey),\n        id: category.categoryId,\n        value: false,\n        condition: (item: { lmiCategories: string[] }): boolean => {\n          let lmiCategories: string[] = [];\n          if (item.lmiCategories) {\n            lmiCategories = item.lmiCategories.map((cat) => {\n              return Categories[cat].toLowerCase();\n            });\n          }\n          return lmiCategories.indexOf(category.categoryId) > -1;\n        },\n      });\n    });\n\n    const lmiCategoryFilterSection = new FilterSection({\n      title: 'FRONTEND.STORE.STOREFRONT_FILTERS.LMI_CATEGORY',\n      type: 'or',\n      fields: lmiCategoryCheckboxList,\n    });\n\n    this.lmiCategoryFilter = new Filters(STOREFRONT_FILTER_NAME, [lmiCategoryFilterSection]);\n\n    const packageIdsCheckbox = new CheckboxFilterField({\n      name: !!this.initialFilteredStoreItems ? this.initialFilteredStoreItems.filterName : '',\n      id: 'package_id_checkbox',\n      value:\n        !!this.initialFilteredStoreItems && !!this.initialFilteredStoreItems.storeItemIds\n          ? this.initialFilteredStoreItems.storeItemIds.length > 0\n          : false,\n    });\n\n    const packageIdsFilterSection = new FilterSection({\n      title: 'Store item IDs', // Shouldn't be seen\n      type: 'and',\n      fields: [packageIdsCheckbox],\n    });\n    packageIdsFilterSection.hideSelectors = true;\n\n    this.filters$ = this.categories$$.pipe(\n      map((categories) => {\n        const tabbedFilterField = new TabbedFilterField({\n          name: '',\n          value: null,\n          id: 'categorytab',\n          options: categories,\n        });\n\n        const filters = new Filters('FRONTEND.STORE.STOREFRONT_FILTERS.TITLE', [\n          new FilterSection({\n            title: 'FRONTEND.STORE.STOREFRONT_FILTERS.CATEGORIES',\n            type: 'or',\n            fields: [tabbedFilterField],\n          }),\n          lmiCategoryFilterSection,\n          packageIdsFilterSection,\n        ]);\n        filters.id = STOREFRONT_FILTER_NAME.toLowerCase().replace(' ', '_');\n\n        return this.showFilters ? filters : null;\n      }),\n    );\n\n    this.filterService.setFilters(this.lmiCategoryFilter);\n\n    this.searchTermSubscription = this.searchTerm$$\n      .asObservable()\n      .pipe(debounceTime(300), distinctUntilChanged(), skip(1))\n      .subscribe((searchTerm) => this.snowplowService.trackClick('store', 'search-input', searchTerm));\n\n    const selectedLMIFilters$ = this.filterService.filters.fieldValuesObservable.pipe(\n      startWith([]),\n      map(() => {\n        return this.lmiCategoryFilter.fields.filter((item: CheckboxFilterField) => item.value);\n      }),\n    );\n\n    this.showLMIFilter$ = selectedLMIFilters$.pipe(\n      filter((selectedLMIFilters) => !!selectedLMIFilters),\n      take(1),\n      map((selectedLMIFilters) => selectedLMIFilters.length > 0),\n    );\n\n    this.filteredItems$ = combineLatest([\n      this.packagesMap$$,\n      this.searchTerm$$,\n      this.selectedCateoryId$$,\n      selectedLMIFilters$,\n      this.filterByInitialStoreItemIds$$,\n    ]).pipe(\n      map(([packages, searchTerm, selectedCategoryId, selectedLMIFilters, filterByInitialStoreItemIds]) => {\n        let storeItems: StoreItem[] = [];\n        if (packages.size) {\n          storeItems = selectedCategoryId\n            ? packages.get(selectedCategoryId)\n            : packages.get(Array.from(packages.keys())[0]);\n        }\n        storeItems = storeItems ? this.lmiCategoryFilter.applyFilters(storeItems) : [];\n\n        if (filterByInitialStoreItemIds) {\n          storeItems = storeItems.filter((storeItem) =>\n            this.initialFilteredStoreItems.storeItemIds.includes(storeItem.id),\n          );\n        }\n\n        return searchTerm\n          ? storeItems.filter((item) => item?.name?.toLowerCase().indexOf(searchTerm.toLowerCase()) !== -1)\n          : storeItems;\n      }),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.searchTermSubscription.unsubscribe();\n  }\n\n  public onSearchTermChanged(term: string): void {\n    this.searchTerm$$.next(term);\n  }\n\n  public onFieldValueChanged(filterField: FilterField<any>): void {\n    this.snowplowService.trackClick('store', 'tab-filter', filterField.value.id);\n    if (filterField.id === 'package_id_checkbox') {\n      this.filterByInitialStoreItemIds$$.next(filterField.value);\n      return;\n    }\n    if (!(filterField instanceof TabbedFilterField)) {\n      return;\n    }\n\n    this.selectedCateoryId$$.next(filterField.value.id);\n    this.categorySelected.emit(filterField.value as StorefrontCategory);\n\n    const currentStoreItemsForSelectCategory = this.packagesMap$$.getValue().get(filterField.value.id);\n    if (!(currentStoreItemsForSelectCategory && currentStoreItemsForSelectCategory.length)) {\n      this.getDataForCategory.emit(filterField.value as StorefrontCategory);\n    }\n  }\n\n  public storeItemClicked(item: StoreItem): void {\n    const label = item.type === ShoppingCartItemType.SHOPPING_CART_ITEM_TYPE_PRODUCT ? 'view-app' : 'view-package';\n    this.snowplowService.trackClick('store', label, item.id);\n    this.productSelected.emit({ id: item.id, type: item.type });\n  }\n}\n","<va-filter-container\n  [filters]=\"filters$ | async\"\n  (searchTermChanged)=\"onSearchTermChanged($event)\"\n  (fieldValueChanged)=\"onFieldValueChanged($event)\"\n  [theme]=\"'plain'\"\n  [filterIsOpen]=\"showLMIFilter$ | async\"\n  [useCustomCase]=\"useCustomCase\"\n  [filtersDisabled]=\"filtersDisabled\"\n>\n  <app-store\n    [items]=\"filteredItems$ | async\"\n    [searchable]=\"false\"\n    (itemClicked)=\"storeItemClicked($event)\"\n    [showAllPrices]=\"showAllPrices\"\n    content\n  ></app-store>\n</va-filter-container>\n"]}