@vendasta/store
Version:
Components and data for Store
176 lines • 29.4 kB
JavaScript
import { Component, Input } from '@angular/core';
import { TranslateService } from '@ngx-translate/core';
import { DurationPeriod, RevenuePeriod } from '@vendasta/sales-orders';
import { of } from 'rxjs';
import { getPeriodTranslationKey } from '../../shared/conversion-utils';
import { OrderSummaryService } from '../order-summary.service';
import * as i0 from "@angular/core";
import * as i1 from "@ngx-translate/core";
import * as i2 from "@angular/common";
import * as i3 from "@vendasta/galaxy/pipes";
export class ContractEstimatedTotalComponent {
constructor(translateService) {
this.translateService = translateService;
}
set orderItems(orderItems) {
const summaryItems = OrderSummaryService.getPriceSummary(orderItems);
this.summaryItems = summaryItems.items;
this.includesUndefinedPricing = summaryItems.includesUndefinedPricing;
}
getDailyTotal(revenueComponent) {
const periodTotal = this.calculateTotalPrice(revenueComponent);
switch (revenueComponent.period) {
case RevenuePeriod.YEARLY:
return periodTotal / 365;
case RevenuePeriod.MONTHLY:
// Average number of days in a month
return periodTotal / 30.42;
case RevenuePeriod.BIWEEKLY:
return periodTotal / 14;
case RevenuePeriod.WEEKLY:
return periodTotal / 7;
case RevenuePeriod.DAILY:
return periodTotal;
default:
return periodTotal;
}
}
getWeeklyTotal(revenueComponent) {
const periodTotal = this.calculateTotalPrice(revenueComponent);
switch (revenueComponent.period) {
case RevenuePeriod.YEARLY:
return periodTotal / 52;
case RevenuePeriod.MONTHLY:
// Average number of weeks in a month
return periodTotal / 4.333;
case RevenuePeriod.BIWEEKLY:
return periodTotal / 2;
case RevenuePeriod.WEEKLY:
return periodTotal;
case RevenuePeriod.DAILY:
return periodTotal * 7;
default:
return periodTotal;
}
}
getMonthlyTotal(revenueComponent) {
const periodTotal = this.calculateTotalPrice(revenueComponent);
switch (revenueComponent.period) {
case RevenuePeriod.YEARLY:
return periodTotal / 12;
case RevenuePeriod.MONTHLY:
return periodTotal;
case RevenuePeriod.BIWEEKLY:
return periodTotal * 2;
case RevenuePeriod.WEEKLY:
return periodTotal * 4;
case RevenuePeriod.DAILY:
return periodTotal * 30;
default:
return periodTotal;
}
}
getYearlyTotal(revenueComponent) {
const periodTotal = this.calculateTotalPrice(revenueComponent);
switch (revenueComponent.period) {
case RevenuePeriod.YEARLY:
return periodTotal;
case RevenuePeriod.MONTHLY:
return periodTotal * 12;
case RevenuePeriod.BIWEEKLY:
return periodTotal * 26;
case RevenuePeriod.WEEKLY:
return periodTotal * 52;
case RevenuePeriod.DAILY:
return periodTotal * 365;
default:
return periodTotal;
}
}
getRevenuePeriod(period) {
return getPeriodTranslationKey(period);
}
getDurationDescription() {
if (!this.duration) {
return of('');
}
switch (this.duration.duration) {
case DurationPeriod.DAY:
return this.translateService.stream('FRONTEND.STORE.DURATION.DAY_CONTRACT', { value: this.duration.value });
case DurationPeriod.WEEK:
return this.translateService.stream('FRONTEND.STORE.DURATION.WEEK_CONTRACT', { value: this.duration.value });
case DurationPeriod.MONTH:
return this.translateService.stream('FRONTEND.STORE.DURATION.MONTH_CONTRACT', { value: this.duration.value });
case DurationPeriod.YEAR:
return this.translateService.stream('FRONTEND.STORE.DURATION.YEAR_CONTRACT', { value: this.duration.value });
default:
return of('');
}
}
calculatePriceForDuration() {
const results = [];
if (!this.duration) {
return results;
}
this.summaryItems.forEach((item) => {
let startingPrice = false;
const total = item.revenueComponents.reduce((all, revenueComponent) => {
if (revenueComponent.isStartingRevenue) {
startingPrice = true;
}
if (revenueComponent.period === RevenuePeriod.ONETIME) {
return all + this.calculateTotalPrice(revenueComponent);
}
if (this.duration.duration === DurationPeriod.DAY) {
return all + this.getDailyTotal(revenueComponent) * this.duration.value;
}
if (this.duration.duration === DurationPeriod.WEEK) {
return all + this.getWeeklyTotal(revenueComponent) * this.duration.value;
}
if (this.duration.duration === DurationPeriod.MONTH) {
return all + this.getMonthlyTotal(revenueComponent) * this.duration.value;
}
if (this.duration.duration === DurationPeriod.YEAR) {
return all + this.getYearlyTotal(revenueComponent) * this.duration.value;
}
return all + this.calculateTotalPrice(revenueComponent);
}, 0);
results.push({
total: total / 100,
currencyCode: item.currencyCode,
startingPrice: startingPrice,
});
});
return results;
}
calculateTotalPrice(revenueComponent) {
if (revenueComponent.value === null) {
return null;
}
if (!this.taxOptions) {
return revenueComponent.value;
}
const taxTotal = this.taxOptions.reduce((all, tax) => all + tax.percentageMultiplier * revenueComponent.value, 0);
return revenueComponent.value + taxTotal;
}
isFree() {
const summaryTotal = this.summaryItems.reduce((all, item) => {
const totalPrices = item.revenueComponents.reduce((acc, revenueComponent) => acc + revenueComponent.value, 0);
return all + totalPrices;
}, 0);
return summaryTotal === 0;
}
}
ContractEstimatedTotalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.2", ngImport: i0, type: ContractEstimatedTotalComponent, deps: [{ token: i1.TranslateService }], target: i0.ɵɵFactoryTarget.Component });
ContractEstimatedTotalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.0.2", type: ContractEstimatedTotalComponent, selector: "app-contract-estimated-total", inputs: { orderItems: "orderItems", taxOptions: "taxOptions", duration: "duration" }, ngImport: i0, template: "<div class=\"contract-duration\">\n <div class=\"duration-description\">\n {{ getDurationDescription() | async }}\n </div>\n <ng-container *ngIf=\"!includesUndefinedPricing && !isFree()\">\n <div class=\"duration-total\">\n <span class=\"total-label\">\n {{ 'FRONTEND.STORE.ESTIMATED_TOTAL' | translate }}\n </span>\n <div class=\"duration-col\">\n <ng-container *ngFor=\"let durationPrice of calculatePriceForDuration()\">\n <ng-container *ngIf=\"durationPrice.total > 0\">\n <div class=\"duration-row\">\n <ng-container *ngIf=\"durationPrice.startingPrice\">\n <strong>{{ 'FRONTEND.STORE.STARTING_AT' | translate }}</strong>\n </ng-container>\n <strong>\n {{\n durationPrice.total | glxyCurrency: durationPrice.currencyCode\n }}\n </strong>\n </div>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </ng-container>\n</div>\n", styles: [".contract-duration{display:flex;flex-direction:column;justify-content:flex-end}.contract-duration .duration-description{display:flex;flex-direction:row;justify-content:flex-end;font-size:14px;color:#616161}.contract-duration .duration-total{display:flex;flex-direction:row;justify-content:flex-end;margin-top:12px}.contract-duration .duration-total .duration-col{display:flex;flex-direction:column}.contract-duration .duration-total .duration-row{display:flex;flex-direction:row;justify-content:flex-end}.contract-duration .duration-total .total-label{margin-right:40px;color:#9e9e9e}\n"], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i2.AsyncPipe, "translate": i1.TranslatePipe, "glxyCurrency": i3.GalaxyCurrencyPipe } });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.2", ngImport: i0, type: ContractEstimatedTotalComponent, decorators: [{
type: Component,
args: [{ selector: 'app-contract-estimated-total', template: "<div class=\"contract-duration\">\n <div class=\"duration-description\">\n {{ getDurationDescription() | async }}\n </div>\n <ng-container *ngIf=\"!includesUndefinedPricing && !isFree()\">\n <div class=\"duration-total\">\n <span class=\"total-label\">\n {{ 'FRONTEND.STORE.ESTIMATED_TOTAL' | translate }}\n </span>\n <div class=\"duration-col\">\n <ng-container *ngFor=\"let durationPrice of calculatePriceForDuration()\">\n <ng-container *ngIf=\"durationPrice.total > 0\">\n <div class=\"duration-row\">\n <ng-container *ngIf=\"durationPrice.startingPrice\">\n <strong>{{ 'FRONTEND.STORE.STARTING_AT' | translate }}</strong>\n </ng-container>\n <strong>\n {{\n durationPrice.total | glxyCurrency: durationPrice.currencyCode\n }}\n </strong>\n </div>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </ng-container>\n</div>\n", styles: [".contract-duration{display:flex;flex-direction:column;justify-content:flex-end}.contract-duration .duration-description{display:flex;flex-direction:row;justify-content:flex-end;font-size:14px;color:#616161}.contract-duration .duration-total{display:flex;flex-direction:row;justify-content:flex-end;margin-top:12px}.contract-duration .duration-total .duration-col{display:flex;flex-direction:column}.contract-duration .duration-total .duration-row{display:flex;flex-direction:row;justify-content:flex-end}.contract-duration .duration-total .total-label{margin-right:40px;color:#9e9e9e}\n"] }]
}], ctorParameters: function () { return [{ type: i1.TranslateService }]; }, propDecorators: { orderItems: [{
type: Input
}], taxOptions: [{
type: Input
}], duration: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"contract-estimated-total.component.js","sourceRoot":"","sources":["../../../../../../../libs/store/src/lib/order-summary/contract-estimated-total/contract-estimated-total.component.ts","../../../../../../../libs/store/src/lib/order-summary/contract-estimated-total/contract-estimated-total.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from '@angular/core';\nimport { TranslateService } from '@ngx-translate/core';\nimport { DurationInterface, DurationPeriod, RevenueComponent, RevenuePeriod } from '@vendasta/sales-orders';\nimport { Observable, of } from 'rxjs';\nimport { DurationTotal, SummaryItem, TaxOption, UILineItem } from '../../order-form/order-item-list/interface';\nimport { getPeriodTranslationKey } from '../../shared/conversion-utils';\nimport { OrderSummaryService } from '../order-summary.service';\n\n@Component({\n  selector: 'app-contract-estimated-total',\n  templateUrl: './contract-estimated-total.component.html',\n  styleUrls: ['./contract-estimated-total.component.scss'],\n})\nexport class ContractEstimatedTotalComponent {\n  @Input()\n  set orderItems(orderItems: UILineItem[]) {\n    const summaryItems = OrderSummaryService.getPriceSummary(orderItems);\n    this.summaryItems = summaryItems.items;\n    this.includesUndefinedPricing = summaryItems.includesUndefinedPricing;\n  }\n\n  @Input() taxOptions: TaxOption[];\n  @Input() duration: DurationInterface;\n\n  private summaryItems: SummaryItem[];\n  includesUndefinedPricing: boolean;\n\n  constructor(private translateService: TranslateService) {}\n\n  getDailyTotal(revenueComponent: RevenueComponent): number {\n    const periodTotal = this.calculateTotalPrice(revenueComponent);\n    switch (revenueComponent.period) {\n      case RevenuePeriod.YEARLY:\n        return periodTotal / 365;\n      case RevenuePeriod.MONTHLY:\n        // Average number of days in a month\n        return periodTotal / 30.42;\n      case RevenuePeriod.BIWEEKLY:\n        return periodTotal / 14;\n      case RevenuePeriod.WEEKLY:\n        return periodTotal / 7;\n      case RevenuePeriod.DAILY:\n        return periodTotal;\n      default:\n        return periodTotal;\n    }\n  }\n\n  getWeeklyTotal(revenueComponent: RevenueComponent): number {\n    const periodTotal = this.calculateTotalPrice(revenueComponent);\n    switch (revenueComponent.period) {\n      case RevenuePeriod.YEARLY:\n        return periodTotal / 52;\n      case RevenuePeriod.MONTHLY:\n        // Average number of weeks in a month\n        return periodTotal / 4.333;\n      case RevenuePeriod.BIWEEKLY:\n        return periodTotal / 2;\n      case RevenuePeriod.WEEKLY:\n        return periodTotal;\n      case RevenuePeriod.DAILY:\n        return periodTotal * 7;\n      default:\n        return periodTotal;\n    }\n  }\n\n  getMonthlyTotal(revenueComponent: RevenueComponent): number {\n    const periodTotal = this.calculateTotalPrice(revenueComponent);\n    switch (revenueComponent.period) {\n      case RevenuePeriod.YEARLY:\n        return periodTotal / 12;\n      case RevenuePeriod.MONTHLY:\n        return periodTotal;\n      case RevenuePeriod.BIWEEKLY:\n        return periodTotal * 2;\n      case RevenuePeriod.WEEKLY:\n        return periodTotal * 4;\n      case RevenuePeriod.DAILY:\n        return periodTotal * 30;\n      default:\n        return periodTotal;\n    }\n  }\n\n  getYearlyTotal(revenueComponent: RevenueComponent): number {\n    const periodTotal = this.calculateTotalPrice(revenueComponent);\n    switch (revenueComponent.period) {\n      case RevenuePeriod.YEARLY:\n        return periodTotal;\n      case RevenuePeriod.MONTHLY:\n        return periodTotal * 12;\n      case RevenuePeriod.BIWEEKLY:\n        return periodTotal * 26;\n      case RevenuePeriod.WEEKLY:\n        return periodTotal * 52;\n      case RevenuePeriod.DAILY:\n        return periodTotal * 365;\n      default:\n        return periodTotal;\n    }\n  }\n\n  getRevenuePeriod(period: RevenuePeriod): string {\n    return getPeriodTranslationKey(period);\n  }\n\n  public getDurationDescription(): Observable<string> {\n    if (!this.duration) {\n      return of('');\n    }\n    switch (this.duration.duration) {\n      case DurationPeriod.DAY:\n        return this.translateService.stream('FRONTEND.STORE.DURATION.DAY_CONTRACT', { value: this.duration.value });\n      case DurationPeriod.WEEK:\n        return this.translateService.stream('FRONTEND.STORE.DURATION.WEEK_CONTRACT', { value: this.duration.value });\n      case DurationPeriod.MONTH:\n        return this.translateService.stream('FRONTEND.STORE.DURATION.MONTH_CONTRACT', { value: this.duration.value });\n      case DurationPeriod.YEAR:\n        return this.translateService.stream('FRONTEND.STORE.DURATION.YEAR_CONTRACT', { value: this.duration.value });\n      default:\n        return of('');\n    }\n  }\n\n  public calculatePriceForDuration(): DurationTotal[] {\n    const results: DurationTotal[] = [];\n    if (!this.duration) {\n      return results;\n    }\n    this.summaryItems.forEach((item) => {\n      let startingPrice = false;\n      const total = item.revenueComponents.reduce((all, revenueComponent) => {\n        if (revenueComponent.isStartingRevenue) {\n          startingPrice = true;\n        }\n\n        if (revenueComponent.period === RevenuePeriod.ONETIME) {\n          return all + this.calculateTotalPrice(revenueComponent);\n        }\n\n        if (this.duration.duration === DurationPeriod.DAY) {\n          return all + this.getDailyTotal(revenueComponent) * this.duration.value;\n        }\n\n        if (this.duration.duration === DurationPeriod.WEEK) {\n          return all + this.getWeeklyTotal(revenueComponent) * this.duration.value;\n        }\n\n        if (this.duration.duration === DurationPeriod.MONTH) {\n          return all + this.getMonthlyTotal(revenueComponent) * this.duration.value;\n        }\n\n        if (this.duration.duration === DurationPeriod.YEAR) {\n          return all + this.getYearlyTotal(revenueComponent) * this.duration.value;\n        }\n\n        return all + this.calculateTotalPrice(revenueComponent);\n      }, 0);\n      results.push({\n        total: total / 100,\n        currencyCode: item.currencyCode,\n        startingPrice: startingPrice,\n      });\n    });\n    return results;\n  }\n\n  public calculateTotalPrice(revenueComponent: RevenueComponent): number {\n    if (revenueComponent.value === null) {\n      return null;\n    }\n    if (!this.taxOptions) {\n      return revenueComponent.value;\n    }\n    const taxTotal = this.taxOptions.reduce((all, tax) => all + tax.percentageMultiplier * revenueComponent.value, 0);\n    return revenueComponent.value + taxTotal;\n  }\n\n  public isFree(): boolean {\n    const summaryTotal = this.summaryItems.reduce((all, item) => {\n      const totalPrices = item.revenueComponents.reduce((acc, revenueComponent) => acc + revenueComponent.value, 0);\n      return all + totalPrices;\n    }, 0);\n    return summaryTotal === 0;\n  }\n}\n","<div class=\"contract-duration\">\n  <div class=\"duration-description\">\n    {{ getDurationDescription() | async }}\n  </div>\n  <ng-container *ngIf=\"!includesUndefinedPricing && !isFree()\">\n    <div class=\"duration-total\">\n      <span class=\"total-label\">\n        {{ 'FRONTEND.STORE.ESTIMATED_TOTAL' | translate }}\n      </span>\n      <div class=\"duration-col\">\n        <ng-container *ngFor=\"let durationPrice of calculatePriceForDuration()\">\n          <ng-container *ngIf=\"durationPrice.total > 0\">\n            <div class=\"duration-row\">\n              <ng-container *ngIf=\"durationPrice.startingPrice\">\n                <strong>{{ 'FRONTEND.STORE.STARTING_AT' | translate }}</strong>\n              </ng-container>\n              <strong>\n                {{\n                  durationPrice.total | glxyCurrency: durationPrice.currencyCode\n                }}\n              </strong>\n            </div>\n          </ng-container>\n        </ng-container>\n      </div>\n    </div>\n  </ng-container>\n</div>\n"]}