Advertisement
ClaudWatari

T.S Component

Apr 5th, 2025
276
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
TypeScript 20.43 KB | Source Code | 0 0
  1. import { Component, OnInit, Input } from "@angular/core";
  2. import {
  3.   UntypedFormGroup,
  4.   FormGroup,
  5.   UntypedFormBuilder,
  6.   Validators,
  7.   UntypedFormArray,
  8.   UntypedFormControl,
  9.   FormControl,
  10.   FormArray,
  11. } from "@angular/forms";
  12. import { MatDialog } from "@angular/material/dialog";
  13. import { animate, state, style, transition, trigger } from "@angular/animations";
  14. import { TooltipPosition } from "@angular/material/tooltip";
  15.  
  16. /** Custom Components */
  17. import { FormDialogComponent } from "app/shared/form-dialog/form-dialog.component";
  18. import { DepositProductIncentiveFormDialogComponent } from "app/products/deposit-product-incentive-form-dialog/deposit-product-incentive-form-dialog.component";
  19. import { DeleteDialogComponent } from "app/shared/delete-dialog/delete-dialog.component";
  20.  
  21. /** Dialog Components */
  22. import { FormfieldBase } from "app/shared/form-dialog/formfield/model/formfield-base";
  23. import { SelectBase } from "app/shared/form-dialog/formfield/model/select-base";
  24. import { InputBase } from "app/shared/form-dialog/formfield/model/input-base";
  25.  
  26. /** Custom Services */
  27. import { SettingsService } from "app/settings/settings.service";
  28. import { Dates } from "app/core/utils/dates";
  29.  
  30. @Component({
  31.   selector: "mifosx-fixed-deposit-product-interest-rate-chart-step",
  32.   templateUrl: "./fixed-deposit-product-interest-rate-chart-step.component.html",
  33.   styleUrls: ["./fixed-deposit-product-interest-rate-chart-step.component.scss"],
  34.   animations: [
  35.     trigger("expandChartSlab", [
  36.       state("collapsed", style({ height: "0px", minHeight: "0" })),
  37.       state("expanded", style({ height: "*" })),
  38.       transition("expanded <=> collapsed", animate("225ms cubic-bezier(0.4, 0.0, 0.2, 1)")),
  39.     ]),
  40.   ],
  41. })
  42. export class FixedDepositProductInterestRateChartStepComponent implements OnInit {
  43.   @Input() fixedDepositProductsTemplate: any;
  44.  
  45.   fixedDepositProductInterestRateChartForm: UntypedFormGroup;
  46.  
  47.   periodTypeData: any;
  48.   entityTypeData: any;
  49.   attributeNameData: any;
  50.   conditionTypeData: any;
  51.   genderData: any;
  52.   clientTypeData: any;
  53.   clientClassificationData: any;
  54.   incentiveTypeData: any;
  55.  
  56.   chartSlabsDisplayedColumns: { [key: number]: string[] } = {};
  57.   chartSlabsIncentivesDisplayedColumns: string[] = ["incentives"];
  58.   incentivesDisplayedColumns: string[] = [
  59.     "entityType",
  60.     "attributeName",
  61.     "conditionType",
  62.     "attributeValue",
  63.     "incentiveType",
  64.     "amount",
  65.     "actions",
  66.   ];
  67.  
  68.   minDate = new Date(2000, 0, 1);
  69.   maxDate = new Date(new Date().setFullYear(new Date().getFullYear() + 10));
  70.  
  71.   expandChartSlabIndex: number[] = [];
  72.   chartDetailData: any = [];
  73.   chartsDetail: any[] = [];
  74.   deletedSlabs: any[] = [];
  75.   loading: boolean = true;
  76.  
  77.   /**
  78.    * @param {FormBuilder} formBuilder Form Builder.
  79.    * @param {MatDialog} dialog Dialog reference.
  80.    * @param {Dates} dateUtils Date Utils.
  81.    * @param {SettingsService} settingsService Settings Service.
  82.    */
  83.  
  84.   constructor(
  85.     private readonly formBuilder: UntypedFormBuilder,
  86.     public dialog: MatDialog,
  87.     private readonly dateUtils: Dates,
  88.     private readonly settingsService: SettingsService,
  89.   ) {
  90.     this.createFixedDepositProductInterestRateChartForm();
  91.   }
  92.  
  93.   ngOnInit() {
  94.     this.periodTypeData = this.fixedDepositProductsTemplate.chartTemplate.periodTypes;
  95.     this.entityTypeData = this.fixedDepositProductsTemplate.chartTemplate.entityTypeOptions;
  96.     this.attributeNameData = this.fixedDepositProductsTemplate.chartTemplate.attributeNameOptions;
  97.     this.conditionTypeData = this.fixedDepositProductsTemplate.chartTemplate.conditionTypeOptions;
  98.     this.genderData = this.fixedDepositProductsTemplate.chartTemplate.genderOptions;
  99.     this.clientTypeData = this.fixedDepositProductsTemplate.chartTemplate.clientTypeOptions;
  100.     this.clientClassificationData = this.fixedDepositProductsTemplate.chartTemplate.clientClassificationOptions;
  101.     this.incentiveTypeData = this.fixedDepositProductsTemplate.chartTemplate.incentiveTypeOptions;
  102.  
  103.     //console.clear();
  104.     console.log(this.fixedDepositProductsTemplate);
  105.     //if (this.chartDetailData.length === 0 || !this.chartDetailData[0]?.name) {
  106.     //return;
  107.     //}
  108.     if (!(this.fixedDepositProductsTemplate === undefined)) {
  109.       setTimeout(() => {
  110.         this.assignFormData();
  111.         this.loading = false;
  112.       }, 2000);
  113.     }
  114.   }
  115.  
  116.   assignFormData() {
  117.     this.addChart();
  118.     if (this.fixedDepositProductsTemplate?.interestRateCharts) {
  119.       this.chartDetailData.push(this.fixedDepositProductsTemplate.interestRateCharts);
  120.     }
  121.     this.getChartsDetailsData();
  122.  
  123.     // Iterates for every chart in charts
  124.     this.charts.controls.forEach((chartDetailControl: UntypedFormGroup, i: number) => {
  125.       if (!this.chartsDetail[i]) {
  126.         return;
  127.       }
  128.  
  129.       // Iterate for every chartSlab in chart
  130.       this.chartsDetail[i].chartSlabs.forEach((chartSlabDetail: any, j: number) => {
  131.         const chartSlabInfo = this.formBuilder.group({
  132.           amountRangeFrom: [chartSlabDetail.amountRangeFrom || ""],
  133.           amountRangeTo: [chartSlabDetail.amountRangeTo || ""],
  134.           annualInterestRate: [chartSlabDetail.annualInterestRate, Validators.required],
  135.           description: [chartSlabDetail.description, Validators.required],
  136.           fromPeriod: [chartSlabDetail.fromPeriod, Validators.required],
  137.           toPeriod: [chartSlabDetail.toPeriod || ""],
  138.           periodType: [chartSlabDetail.periodType, Validators.required],
  139.           incentives: this.formBuilder.array([]),
  140.         });
  141.         const formArray = chartDetailControl.controls["chartSlabs"] as UntypedFormArray;
  142.         formArray.push(chartSlabInfo);
  143.  
  144.         // Iterate for every slab in chartSlab
  145.         const chartSlabs = chartDetailControl.get("chartSlabs") as UntypedFormArray;
  146.         const chartIncentiveControl = chartSlabs.controls[j];
  147.  
  148.         // Iterate to input all the incentive for particular chart slab
  149.         this.chartsDetail[i].chartSlabs[j].incentives.forEach((chartIncentiveDetail: any) => {
  150.           const incentiveInfo = this.formBuilder.group({
  151.             amount: [chartIncentiveDetail.amount, Validators.required],
  152.             attributeName: [chartIncentiveDetail.attributeName, Validators.required],
  153.             attributeValue: [chartIncentiveDetail.attributeValue, Validators.required],
  154.             conditionType: [chartIncentiveDetail.conditionType, Validators.required],
  155.             entityType: [chartIncentiveDetail.entityType, Validators.required],
  156.             incentiveType: [chartIncentiveDetail.incentiveType, Validators.required],
  157.           });
  158.           const newFormArray = chartIncentiveControl?.get("incentives") as UntypedFormArray; // Get the 'incentives' array
  159.           newFormArray.push(incentiveInfo);
  160.         });
  161.       });
  162.     });
  163.   }
  164.  
  165.   getChartsDetailsData() {
  166.     const flattenedData = this.chartDetailData.flat();
  167.     console.log("chartData", this.chartDetailData);
  168.     flattenedData.forEach((chartData: any) => {
  169.       const chart = {
  170.         endDate: chartData.endDate ? new Date(chartData.endDate) : "",
  171.         fromDate: chartData.fromDate ? new Date(chartData.fromDate) : "",
  172.         isPrimaryGroupingByAmount: chartData.isPrimaryGroupingByAmount,
  173.         name: chartData.name,
  174.         description: chartData.description,
  175.         chartSlabs: this.getChartSlabsData(chartData.chartSlabs),
  176.       };
  177.       const chartsArray = this.fixedDepositProductInterestRateChartForm.get("charts") as UntypedFormArray;
  178.  
  179.       this.chartsDetail.push(chart);
  180.  
  181.       setTimeout(() => {
  182.         // Ensure the FormArray has the correct number of controls
  183.         while (chartsArray.length < this.chartsDetail.length) {
  184.           const formGroup = this.formBuilder.group({});
  185.           // Dynamically add controls based on object keys
  186.           Object.keys(this.chartsDetail[0]).forEach((key) => {
  187.             formGroup.addControl(key, new FormControl(chartData[key]));
  188.           });
  189.           chartsArray.push(formGroup);
  190.         }
  191.       }, 500);
  192.  
  193.       setTimeout(() => {
  194.         console.clear();
  195.         console.log("chartSlabsDisplayedColumns", this.chartSlabsDisplayedColumns);
  196.         this.fixedDepositProductInterestRateChartForm.setControl(
  197.           "charts",
  198.           this.createChartsFormArray(this.chartsDetail),
  199.         );
  200.       }, 1000);
  201.     });
  202.   }
  203.  
  204.   createChartsFormArray(chartsData: any[]): UntypedFormArray {
  205.     const formArray = this.formBuilder.array([]);
  206.  
  207.     chartsData.forEach((chart) => {
  208.       formArray.push(
  209.         this.formBuilder.group({
  210.           endDate: [chart.endDate],
  211.           fromDate: [chart.fromDate],
  212.           isPrimaryGroupingByAmount: [chart.isPrimaryGroupingByAmount],
  213.           name: [chart.name],
  214.           description: [chart.description],
  215.           chartSlabs: this.createChartSlabsFormArray(chart.chartSlabs),
  216.         }),
  217.       );
  218.     });
  219.  
  220.     return formArray;
  221.   }
  222.   createChartSlabsFormArray(chartSlabsData: any[]): UntypedFormArray {
  223.     const formArray = this.formBuilder.array([]);
  224.  
  225.     chartSlabsData.forEach((slab) => {
  226.       formArray.push(
  227.         this.formBuilder.group({
  228.           periodType: [slab.periodType?.id ?? null],
  229.           amountRangeFrom: [slab.amountRangeFrom ?? 0],
  230.           amountRangeTo: [slab.amountRangeTo ?? 0],
  231.           annualInterestRate: [slab.annualInterestRate ?? 0],
  232.           description: [slab.description || ""],
  233.           fromPeriod: [slab.fromPeriod ?? null],
  234.           toPeriod: [slab.toPeriod ?? null],
  235.           incentives: this.createIncentivesFormArray(slab.incentives),
  236.         }),
  237.       );
  238.     });
  239.  
  240.     return formArray;
  241.   }
  242.  
  243.   createIncentivesFormArray(incentivesData: any[]): UntypedFormArray {
  244.     const formArray = this.formBuilder.array([]);
  245.  
  246.     incentivesData.forEach((incentive) => {
  247.       formArray.push(
  248.         this.formBuilder.group({
  249.           amount: [incentive?.amount ?? 0],
  250.           attributeName: [incentive?.attributeName ?? ""],
  251.           attributeValue: [incentive?.attributeValue ?? ""],
  252.           conditionType: [incentive?.conditionType ?? ""],
  253.           entityType: [incentive?.entityType ?? ""],
  254.           incentiveType: [incentive?.incentiveType ?? ""],
  255.         }),
  256.       );
  257.     });
  258.  
  259.     return formArray;
  260.   }
  261.  
  262.   getChartSlabsData(chartData: any): any[] {
  263.     const chartSlabs: any[] = [];
  264.  
  265.     chartData.forEach((dataItem: any) => {
  266.       const slabData = Array.isArray(dataItem.chartSlabs) ? dataItem.chartSlabs : [dataItem];
  267.  
  268.       slabData.forEach((eachChartSlabData: any) => {
  269.         chartSlabs.push({
  270.           periodType: eachChartSlabData.periodType?.id ?? null,
  271.           amountRangeFrom: eachChartSlabData.amountRangeFrom ?? 0,
  272.           amountRangeTo: eachChartSlabData.amountRangeTo ?? 0,
  273.           annualInterestRate: eachChartSlabData.annualInterestRate ?? 0,
  274.           description: eachChartSlabData.description || "",
  275.           fromPeriod: eachChartSlabData.fromPeriod ?? null,
  276.           toPeriod: eachChartSlabData.toPeriod ?? null,
  277.           incentives: this.getIncentivesData(eachChartSlabData),
  278.         });
  279.       });
  280.     });
  281.  
  282.     return chartSlabs;
  283.   }
  284.  
  285.   getIncentivesData(chartSlabData: any) {
  286.     let incentives: any[] = [
  287.       {
  288.         amount: 0,
  289.         attributeName: "",
  290.         attributeValue: "",
  291.         conditionType: "",
  292.         entityType: "",
  293.         incentiveType: "",
  294.       },
  295.     ];
  296.     let incentiveDatas: any[] = [];
  297.     if (chartSlabData.incentives) {
  298.       incentives = [];
  299.       const isChartIncentiveArray = Array.isArray(chartSlabData.incentives);
  300.       if (!isChartIncentiveArray) {
  301.         incentiveDatas.push(chartSlabData.incentives);
  302.       } else {
  303.         incentiveDatas = chartSlabData.incentives;
  304.       }
  305.       incentiveDatas.forEach((incentiveData: any) => {
  306.         const incentive = {
  307.           amount: incentiveData.amount,
  308.           attributeName: incentiveData.attributeName,
  309.           attributeValue: incentiveData.attributeValue,
  310.           conditionType: incentiveData.conditionType,
  311.           entityType: incentiveData.entityType,
  312.           incentiveType: incentiveData.incentiveType,
  313.         };
  314.         incentives.push(incentive);
  315.       });
  316.     }
  317.     return incentives;
  318.   }
  319.  
  320.   createFixedDepositProductInterestRateChartForm() {
  321.     this.fixedDepositProductInterestRateChartForm = this.formBuilder.group({
  322.       charts: this.formBuilder.array([]),
  323.     });
  324.   }
  325.  
  326.   get charts(): UntypedFormArray {
  327.     return this.fixedDepositProductInterestRateChartForm.get("charts") as UntypedFormArray;
  328.   }
  329.  
  330.   createChartForm(): UntypedFormGroup {
  331.     return this.formBuilder.group({
  332.       name: [""],
  333.       description: [""],
  334.       fromDate: ["", Validators.required],
  335.       endDate: [""],
  336.       isPrimaryGroupingByAmount: [false],
  337.       chartSlabs: this.formBuilder.array([], Validators.required),
  338.     });
  339.   }
  340.  
  341.   addChart() {
  342.     this.charts.push(this.createChartForm());
  343.     this.setConditionalControls(this.charts.length - 1);
  344.   }
  345.  
  346.   setConditionalControls(chartIndex: number): string[] {
  347.     if (!this.chartSlabsDisplayedColumns[chartIndex]) {
  348.       this.chartSlabsDisplayedColumns[chartIndex] = [
  349.         "period",
  350.         "amountRange",
  351.         "annualInterestRate",
  352.         "description",
  353.         "actions",
  354.       ];
  355.     }
  356.     this.charts
  357.       .at(chartIndex)
  358.       .get("isPrimaryGroupingByAmount")
  359.       .valueChanges.subscribe((isPrimaryGroupingByAmount: boolean) => {
  360.         this.chartSlabsDisplayedColumns[chartIndex] = isPrimaryGroupingByAmount
  361.           ? ["amountRange", "period"]
  362.           : ["period", "amountRange"];
  363.         this.chartSlabsDisplayedColumns[chartIndex].push("annualInterestRate", "description", "actions");
  364.       });
  365.     return this.chartSlabsDisplayedColumns[chartIndex];
  366.   }
  367.  
  368.   // getIncentives(chartSlabs: UntypedFormArray, chartSlabIndex: number): UntypedFormArray {
  369.   //   console.log("Is FormArray:", chartSlabs instanceof UntypedFormArray);
  370.   //   // return chartSlabs.at(chartSlabIndex).get("incentives") as UntypedFormArray;
  371.   //   return chartSlabs.controls[chartSlabIndex].get("incentives") as UntypedFormArray;
  372.   // }
  373.  
  374.   getIncentives(chartSlabs: any, chartSlabIndex: number): UntypedFormArray {
  375.     console.clear();
  376.  
  377.     if (!(chartSlabs?.controls instanceof Array)) {
  378.       console.warn("chartSlabs was not a FormArray. Converting it now.");
  379.       chartSlabs = new UntypedFormArray(
  380.         (Array.isArray(chartSlabs) ? chartSlabs : []).map((slab) =>
  381.           this.formBuilder.group({
  382.             incentives: this.formBuilder.array(slab?.incentives || []),
  383.           }),
  384.         ),
  385.       );
  386.     }
  387.  
  388.     if (chartSlabIndex < 0 || (chartSlabs?.value?.length !== 0 && chartSlabIndex >= chartSlabs?.value?.length)) {
  389.       console.error(`Invalid chartSlabIndex: ${chartSlabIndex}. Index out of range.`);
  390.       return new UntypedFormArray([]);
  391.     }
  392.  
  393.     const chartSlab = chartSlabs?.at(chartSlabIndex) as FormGroup | undefined;
  394.     if (!chartSlab) {
  395.       console.error(`Chart slab at index ${chartSlabIndex} is undefined.`);
  396.       return new UntypedFormArray([]);
  397.     }
  398.  
  399.     const incentives = chartSlab?.get("incentives") as UntypedFormArray | undefined;
  400.     console.log({ chartSlab, incentives });
  401.     if (!(incentives instanceof UntypedFormArray)) {
  402.       console.error(`Incentives field is missing or not a FormArray.`);
  403.       return new UntypedFormArray([]);
  404.     }
  405.  
  406.     return incentives;
  407.   }
  408.  
  409.   addChartSlab(chartSlabs: UntypedFormArray) {
  410.     const data = { ...this.getData("Slab") };
  411.     const dialogRef = this.dialog.open(FormDialogComponent, { data });
  412.     dialogRef.afterClosed().subscribe((response: any) => {
  413.       if (response.data) {
  414.         response.data.addControl("incentives", this.formBuilder.array([]));
  415.         chartSlabs.push(response.data);
  416.       }
  417.     });
  418.   }
  419.  
  420.   addIncentive(incentives: UntypedFormArray) {
  421.     const data = { ...this.getData("Incentive"), entityType: this.entityTypeData[0].id };
  422.     const dialogRef = this.dialog.open(DepositProductIncentiveFormDialogComponent, { data });
  423.     dialogRef.afterClosed().subscribe((response: any) => {
  424.       if (response.data) {
  425.         console.clear();
  426.         incentives.push(response.data);
  427.         console.log({ incentives });
  428.       }
  429.     });
  430.   }
  431.  
  432.   editChartSlab(chartSlabs: UntypedFormArray, chartSlabIndex: number) {
  433.     const data = { ...this.getData("Slab", chartSlabs.at(chartSlabIndex).value), layout: { addButtonText: "Edit" } };
  434.     console.log(data);
  435.     const dialogRef = this.dialog.open(FormDialogComponent, { data });
  436.     dialogRef.afterClosed().subscribe((response: any) => {
  437.       if (response.data) {
  438.         chartSlabs.at(chartSlabIndex).patchValue(response.data.value);
  439.       }
  440.     });
  441.   }
  442.  
  443.   editIncentive(incentives: UntypedFormArray, incentiveIndex: number) {
  444.     console.clear();
  445.     console.log(incentives);
  446.     const data = {
  447.       ...this.getData("Incentive", incentives?.at(incentiveIndex)?.value),
  448.       layout: { addButtonText: "Edit" },
  449.     };
  450.     console.log(data);
  451.     const dialogRef = this.dialog.open(DepositProductIncentiveFormDialogComponent, { data });
  452.     dialogRef.afterClosed().subscribe((response: any) => {
  453.       if (response.data) {
  454.         setTimeout(() => {
  455.           console.clear();
  456.         }, 1000);
  457.         incentives?.at(incentiveIndex)?.patchValue(response.data.value);
  458.       }
  459.     });
  460.   }
  461.  
  462.   delete(formArray: UntypedFormArray, index: number) {
  463.     const dialogRef = this.dialog.open(DeleteDialogComponent, {
  464.       data: { deleteContext: `this slab` },
  465.     });
  466.     dialogRef.afterClosed().subscribe((response: any) => {
  467.       if (response.delete) {
  468.         const chartSlab = formArray.at(index).value;
  469.         chartSlab.deleted = true;
  470.  
  471.         this.deletedSlabs.push(chartSlab);
  472.         localStorage.setItem("deletedSlabs", JSON.stringify(this.deletedSlabs));
  473.         formArray.removeAt(index);
  474.       }
  475.     });
  476.   }
  477.  
  478.   getData(formType: string, values?: any) {
  479.     switch (formType) {
  480.       case "Slab":
  481.         return { title: "Slab", formfields: this.getSlabFormfields(values) };
  482.       case "Incentive":
  483.         return { values, chartTemplate: this.fixedDepositProductsTemplate.chartTemplate };
  484.     }
  485.   }
  486.  
  487.   getSlabFormfields(values?: any) {
  488.     const formfields: FormfieldBase[] = [
  489.       new SelectBase({
  490.         controlName: "periodType",
  491.         label: "Period Type",
  492.         value: values ? values.periodType : this.periodTypeData[0].id,
  493.         options: { label: "value", value: "id", data: this.periodTypeData },
  494.         required: true,
  495.         order: 1,
  496.       }),
  497.       new InputBase({
  498.         controlName: "fromPeriod",
  499.         label: "Period From",
  500.         value: values ? values.fromPeriod : undefined,
  501.         type: "number",
  502.         required: true,
  503.         order: 2,
  504.       }),
  505.       new InputBase({
  506.         controlName: "toPeriod",
  507.         label: "Period To",
  508.         value: values ? values.toPeriod : undefined,
  509.         type: "number",
  510.         order: 3,
  511.       }),
  512.       new InputBase({
  513.         controlName: "amountRangeFrom",
  514.         label: "Amount Range From",
  515.         value: values ? values.amountRangeFrom : undefined,
  516.         type: "number",
  517.         order: 4,
  518.       }),
  519.       new InputBase({
  520.         controlName: "amountRangeTo",
  521.         label: "Amount Range To",
  522.         value: values ? values.amountRangeTo : undefined,
  523.         type: "number",
  524.         order: 5,
  525.       }),
  526.       new InputBase({
  527.         controlName: "annualInterestRate",
  528.         label: "Interest",
  529.         value: values ? values.annualInterestRate : undefined,
  530.         type: "number",
  531.         required: true,
  532.         order: 6,
  533.       }),
  534.       new InputBase({
  535.         controlName: "description",
  536.         label: "Description",
  537.         value: values ? values.description : undefined,
  538.         required: true,
  539.         order: 7,
  540.       }),
  541.     ];
  542.     return formfields;
  543.   }
  544.  
  545.   get fixedDepositProductInterestRateChart() {
  546.     // TODO: Update once language and date settings are setup
  547.     const locale = this.settingsService.language.code;
  548.     const dateFormat = this.settingsService.dateFormat;
  549.     const fixedDepositProductInterestRateChart = this.fixedDepositProductInterestRateChartForm.value;
  550.     for (const chart of fixedDepositProductInterestRateChart.charts) {
  551.       chart.dateFormat = dateFormat;
  552.       chart.locale = locale;
  553.       chart.fromDate = this.dateUtils.formatDate(chart.fromDate, dateFormat) || "";
  554.       chart.endDate = this.dateUtils.formatDate(chart.endDate, dateFormat) || "";
  555.       if (chart.endDate === "") {
  556.         delete chart.endDate;
  557.       }
  558.     }
  559.     return fixedDepositProductInterestRateChart;
  560.   }
  561. }
  562.  
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement