diff --git a/src/helpers/ui/paste_interactive.ts b/src/helpers/ui/paste_interactive.ts index ffa23ad37d..62b9d53d5c 100644 --- a/src/helpers/ui/paste_interactive.ts +++ b/src/helpers/ui/paste_interactive.ts @@ -1,4 +1,4 @@ -import { CURRENT_VERSION } from "../../migrations/data"; +import { getCurrentVersion } from "../../migrations/data"; import { _t } from "../../translation"; import { ClipboardPasteOptions, @@ -75,7 +75,7 @@ export async function interactivePasteFromOS( } catch (error) { const parsedSpreadsheetContent = parsedClipboardContent.data; - if (parsedSpreadsheetContent?.version !== CURRENT_VERSION) { + if (parsedSpreadsheetContent?.version !== getCurrentVersion()) { env.raiseError( _t( "An unexpected error occurred while pasting content.\ diff --git a/src/migrations/data.ts b/src/migrations/data.ts index e801f77970..576b8528a2 100644 --- a/src/migrations/data.ts +++ b/src/migrations/data.ts @@ -15,11 +15,18 @@ import { XlsxReader } from "../xlsx/xlsx_reader"; import { migrationStepRegistry } from "./migration_steps"; /** - * This is the current state version number. It should be incremented each time - * a breaking change is made in the way the state is handled, and an upgrade - * function should be defined + * Represents the current version of the exported JSON data. + * A new version must be created whenever a breaking change is introduced in the export format. + * To define a new version, add an upgrade function to `migrationStepRegistry`. */ -export const CURRENT_VERSION = 25; +export function getCurrentVersion() { + return getSortedVersions().at(-1)!; +} + +function getSortedVersions() { + return migrationStepRegistry.getKeys().sort(compareVersions); +} + const INITIAL_SHEET_ID = "Sheet1"; /** @@ -47,7 +54,10 @@ export function load(data?: any, verboseImport?: boolean): WorkbookData { // apply migrations, if needed if ("version" in data) { - if (data.version < CURRENT_VERSION) { + if (isLegacyVersioning(data)) { + data.version = LEGACY_VERSION_MAPPING[data.version]; + } + if (data.version !== getCurrentVersion()) { console.debug("Migrating data from version", data.version); data = migrate(data); } @@ -58,6 +68,44 @@ export function load(data?: any, verboseImport?: boolean): WorkbookData { return data; } +const LEGACY_VERSION_MAPPING = { + 25: "18.2", + 24: "18.1.1", + 23: "18.1", + 22: "18.0.4", + 21: "18.0.3", + 20: "18.0.2", + 19: "18.0.1", + 18: "18.0", + 17: "17.4", + 16: "17.3", + 15: "17.2", + 14: "16.4", + 13: "16.3", + 12: "15.4", + + // not accurate starting at this point + 11: "0.10", + 10: "0.9", + 9: "0.8", + 8: "0.7", + 7: "0.6", + 6: "0.5", + 5: "0.4", + 4: "0.3", + 3: "0.2", + 2: "0.1", + 1: "0", +}; + +/** + * Versions used to be an incremented integer. + * This was later changed to match release versions (matching Odoo release names). + */ +function isLegacyVersioning(data: { version: number | string }): boolean { + return typeof data.version === "number"; +} + // ----------------------------------------------------------------------------- // Migrations // ----------------------------------------------------------------------------- @@ -83,12 +131,11 @@ function compareVersions(v1: string, v2: string): number { function migrate(data: any): WorkbookData { const start = performance.now(); - const steps = migrationStepRegistry - .getAll() - .sort((a, b) => compareVersions(a.versionFrom, b.versionFrom)); - const index = steps.findIndex((step) => step.versionFrom === data.version.toString()); - for (let i = index; i < steps.length; i++) { - data = steps[i].migrate(data); + const versions = getSortedVersions(); + const index = versions.findIndex((v) => v === data.version); + for (let i = index + 1; i < versions.length; i++) { + const nextVersion = versions[i]; + data = migrationStepRegistry.get(nextVersion).migrate(data); } console.debug("Data migrated in", performance.now() - start, "ms"); return data; @@ -131,7 +178,7 @@ function forceUnicityOfFigure(data: Partial): Partial): Partial { const data: WorkbookData = Object.assign(createEmptyWorkbookData(), partialData, { - version: CURRENT_VERSION, + version: getCurrentVersion(), }); data.sheets = data.sheets ? data.sheets.map((s, i) => @@ -291,7 +338,7 @@ export function createEmptySheet(sheetId: UID, name: string): SheetData { export function createEmptyWorkbookData(sheetName = "Sheet1"): WorkbookData { return { - version: CURRENT_VERSION, + version: getCurrentVersion(), sheets: [createEmptySheet(INITIAL_SHEET_ID, sheetName)], styles: {}, formats: {}, diff --git a/src/migrations/migration_steps.ts b/src/migrations/migration_steps.ts index 0418ca18a5..7be8cd4f5b 100644 --- a/src/migrations/migration_steps.ts +++ b/src/migrations/migration_steps.ts @@ -10,16 +10,14 @@ import { normalizeV9 } from "./legacy_tools"; import { WEEK_START } from "./locale"; export interface MigrationStep { - versionFrom: string; migrate: (data: any) => any; } export const migrationStepRegistry = new Registry(); migrationStepRegistry - .add("migration_1", { + .add("0.1", { // add the `activeSheet` field on data - versionFrom: "1", migrate(data: any): any { if (data.sheets && data.sheets[0]) { data.activeSheet = data.sheets[0].name; @@ -27,9 +25,8 @@ migrationStepRegistry return data; }, }) - .add("migration_2", { + .add("0.2", { // add an id field in each sheet - versionFrom: "2", migrate(data: any): any { if (data.sheets && data.sheets.length) { for (let sheet of data.sheets) { @@ -39,9 +36,8 @@ migrationStepRegistry return data; }, }) - .add("migration_3", { + .add("0.3", { // activeSheet is now an id, not the name of a sheet - versionFrom: "3", migrate(data: any): any { if (data.sheets && data.activeSheet) { const activeSheet = data.sheets.find((s) => s.name === data.activeSheet); @@ -50,9 +46,8 @@ migrationStepRegistry return data; }, }) - .add("migration_4", { + .add("0.4", { // add figures object in each sheets - versionFrom: "4", migrate(data: any): any { for (let sheet of data.sheets || []) { sheet.figures = sheet.figures || []; @@ -60,9 +55,8 @@ migrationStepRegistry return data; }, }) - .add("migration_5", { + .add("0.5", { // normalize the content of the cell if it is a formula to avoid parsing all the formula that vary only by the cells they use - versionFrom: "5", migrate(data: any): any { for (let sheet of data.sheets || []) { for (let xc in sheet.cells || []) { @@ -75,9 +69,8 @@ migrationStepRegistry return data; }, }) - .add("migration_6", { + .add("0.6", { // transform chart data structure - versionFrom: "6", migrate(data: any): any { for (let sheet of data.sheets || []) { for (let f in sheet.figures || []) { @@ -100,9 +93,8 @@ migrationStepRegistry return data; }, }) - .add("migration_7", { + .add("0.7", { // remove single quotes in sheet names - versionFrom: "7", migrate(data: any): any { const namesTaken: string[] = []; for (let sheet of data.sheets || []) { @@ -165,9 +157,8 @@ migrationStepRegistry return data; }, }) - .add("migration_8", { + .add("0.8", { // transform chart data structure with design attributes - versionFrom: "8", migrate(data: any): any { for (const sheet of data.sheets || []) { for (const chart of sheet.figures || []) { @@ -180,9 +171,8 @@ migrationStepRegistry return data; }, }) - .add("migration_9", { + .add("0.9", { // de-normalize formula to reduce exported json size (~30%) - versionFrom: "9", migrate(data: any): any { for (let sheet of data.sheets || []) { for (let xc in sheet.cells || []) { @@ -201,9 +191,8 @@ migrationStepRegistry return data; }, }) - .add("migration_10", { + .add("0.10", { // normalize the formats of the cells - versionFrom: "10", migrate(data: any): any { const formats: { [formatId: number]: Format } = {}; for (let sheet of data.sheets || []) { @@ -218,9 +207,8 @@ migrationStepRegistry return data; }, }) - .add("migration_11", { + .add("15.4", { // Add isVisible to sheets - versionFrom: "11", migrate(data: any): any { for (let sheet of data.sheets || []) { sheet.isVisible = true; @@ -228,16 +216,14 @@ migrationStepRegistry return data; }, }) - .add("migration_12", { + .add("15.4.1", { // Fix data filter duplication - versionFrom: "12", migrate(data: any): any { return fixOverlappingFilters(data); }, }) - .add("migration_12_5", { + .add("16.3", { // Change Border description structure - versionFrom: "12.5", migrate(data: any): any { for (const borderId in data.borders) { const border = data.borders[borderId]; @@ -253,9 +239,8 @@ migrationStepRegistry return data; }, }) - .add("migration_13", { + .add("16.4", { // Add locale to spreadsheet settings - versionFrom: "13", migrate(data: any): any { if (!data.settings) { data.settings = {}; @@ -266,16 +251,14 @@ migrationStepRegistry return data; }, }) - .add("migration_14", { + .add("16.4.1", { // Fix datafilter duplication (post saas-17.1) - versionFrom: "14", migrate(data: any): any { return fixOverlappingFilters(data); }, }) - .add("migration_14_5", { + .add("17.2", { // Rename filterTable to tables - versionFrom: "14.5", migrate(data: any): any { for (const sheetData of data.sheets || []) { sheetData.tables = sheetData.tables || sheetData.filterTables || []; @@ -284,9 +267,8 @@ migrationStepRegistry return data; }, }) - .add("migration_15", { + .add("17.3", { // Add pivots - versionFrom: "15", migrate(data: any): any { if (!data.pivots) { data.pivots = {}; @@ -297,9 +279,8 @@ migrationStepRegistry return data; }, }) - .add("migration_16", { + .add("17.4", { // transform chart data structure (2) - versionFrom: "16", migrate(data: any): any { for (const sheet of data.sheets || []) { for (const f in sheet.figures || []) { @@ -320,18 +301,16 @@ migrationStepRegistry return data; }, }) - .add("migration_17", { + .add("18.0", { // Empty migration to allow external modules to add their own migration steps // before this version - versionFrom: "17", migrate(data: any): any { return data; }, }) - .add("migration_18", { + .add("18.0.1", { // Change measures and dimensions `name` to `fieldName` // Add id to measures - versionFrom: "18", migrate(data: any): any { interface PivotCoreMeasureV17 { name: string; @@ -362,9 +341,8 @@ migrationStepRegistry return data; }, }) - .add("migration_19", { + .add("18.0.2", { // "Add weekStart to locale", - versionFrom: "19", migrate(data: any): any { const locale = data.settings?.locale; if (locale) { @@ -374,9 +352,8 @@ migrationStepRegistry return data; }, }) - .add("migration_20", { + .add("18.0.3", { // group style and format into zones, - versionFrom: "20", migrate(data: any): any { for (const sheet of data.sheets || []) { sheet.styles = {}; @@ -394,9 +371,8 @@ migrationStepRegistry return data; }, }) - .add("migration_21", { + .add("18.0.4", { // "Add operator in gauge inflection points", - versionFrom: "21", migrate(data: WorkbookData): any { for (const sheet of data.sheets || []) { for (const figure of sheet.figures || []) { @@ -415,9 +391,8 @@ migrationStepRegistry return data; }, }) - .add("migration_22", { + .add("18.1", { // "tables are no longer inserted with filters by default", - versionFrom: "22", migrate(data: WorkbookData): any { for (const sheet of data.sheets || []) { for (const table of sheet.tables || []) { @@ -429,9 +404,8 @@ migrationStepRegistry return data; }, }) - .add("migration_23", { + .add("18.1.1", { // Flatten cell content: { content: "value" } -> "value" - versionFrom: "23", migrate(data: WorkbookData): any { for (const sheet of data.sheets || []) { for (const xc in sheet.cells) { @@ -444,12 +418,16 @@ migrationStepRegistry return data; }, }) - .add("migration_24", { + .add("18.2", { // Empty migration to allow odoo migrate pivot custom sorting. - versionFrom: "24", migrate(data: WorkbookData): any { return data; }, + }) + .add("18.3", { + migrate(data) { + return data; + }, }); function fixOverlappingFilters(data: any): any { diff --git a/src/plugins/ui_stateful/clipboard.ts b/src/plugins/ui_stateful/clipboard.ts index 0d9b4d29c2..e33cae363c 100644 --- a/src/plugins/ui_stateful/clipboard.ts +++ b/src/plugins/ui_stateful/clipboard.ts @@ -6,7 +6,7 @@ import { SELECTION_BORDER_COLOR } from "../../constants"; import { getClipboardDataPositions } from "../../helpers/clipboard/clipboard_helpers"; import { getMaxFigureSize } from "../../helpers/figures/figure/figure"; import { UuidGenerator, isZoneValid, union } from "../../helpers/index"; -import { CURRENT_VERSION } from "../../migrations/data"; +import { getCurrentVersion } from "../../migrations/data"; import { _t } from "../../translation"; import { ClipboardData, @@ -47,7 +47,7 @@ type MinimalClipboardData = { }; export interface SpreadsheetClipboardData extends MinimalClipboardData { - version?: number; + version?: string; clipboardId?: string; } /** @@ -526,7 +526,7 @@ export class ClipboardPlugin extends UIPlugin { private getSheetData(): SpreadsheetClipboardData { const data = { - version: CURRENT_VERSION, + version: getCurrentVersion(), clipboardId: this.clipboardId, }; if (this.copiedData && "figureId" in this.copiedData) { diff --git a/src/types/workbook_data.ts b/src/types/workbook_data.ts index 311f23dc88..0f474c89d1 100644 --- a/src/types/workbook_data.ts +++ b/src/types/workbook_data.ts @@ -57,7 +57,7 @@ interface WorkbookSettings { type PivotData = { formulaId: string } & PivotCoreDefinition; export interface WorkbookData { - version: number; + version: string; sheets: SheetData[]; styles: { [key: number]: Style }; formats: { [key: number]: Format }; diff --git a/src/xlsx/xlsx_reader.ts b/src/xlsx/xlsx_reader.ts index b9bb14bdaa..beb068b6c8 100644 --- a/src/xlsx/xlsx_reader.ts +++ b/src/xlsx/xlsx_reader.ts @@ -26,7 +26,7 @@ import { getXLSXFilesOfType } from "./helpers/xlsx_helper"; import { XLSXImportWarningManager } from "./helpers/xlsx_parser_error_manager"; import { escapeTagNamespaces, parseXML } from "./helpers/xml_helpers"; -const EXCEL_IMPORT_VERSION = 24; +const EXCEL_IMPORT_VERSION = "18.3"; export class XlsxReader { warningManager: XLSXImportWarningManager; diff --git a/tests/model/data.test.ts b/tests/model/data.test.ts index 25f419e769..02feef33bc 100644 --- a/tests/model/data.test.ts +++ b/tests/model/data.test.ts @@ -1,12 +1,12 @@ import { DEFAULT_REVISION_ID } from "../../src/constants"; -import { CURRENT_VERSION, load } from "../../src/migrations/data"; +import { getCurrentVersion, load } from "../../src/migrations/data"; import { DEFAULT_LOCALE } from "../../src/types"; describe("load data", () => { test("create empty workbookdata when loading nothing", () => { const emptyWorkbook = load({}); expect(emptyWorkbook).toMatchObject({ - version: CURRENT_VERSION, + version: getCurrentVersion(), borders: {}, styles: {}, formats: {}, @@ -41,7 +41,7 @@ describe("load data", () => { sheets: [{ id: "asdf", merges: ["A1:B2"] }], }) ).toMatchObject({ - version: CURRENT_VERSION, + version: getCurrentVersion(), borders: {}, styles: {}, formats: {}, @@ -73,7 +73,7 @@ describe("load data", () => { sheets: [{ name: "Sheet1", merges: ["A1:B2"] }], }) ).toMatchObject({ - version: CURRENT_VERSION, + version: getCurrentVersion(), borders: {}, styles: {}, formats: {}, @@ -154,7 +154,7 @@ describe("load data", () => { sheets: [{ merges: ["A1:B2"] }], }) ).toMatchObject({ - version: CURRENT_VERSION, + version: getCurrentVersion(), borders: {}, styles: {}, formats: {}, @@ -205,7 +205,7 @@ describe("load data", () => { ], }) ).toMatchObject({ - version: CURRENT_VERSION, + version: getCurrentVersion(), sheets: [ { figures: [ diff --git a/tests/model/model_import_export.test.ts b/tests/model/model_import_export.test.ts index d1e0f6ec13..21a8272fd8 100644 --- a/tests/model/model_import_export.test.ts +++ b/tests/model/model_import_export.test.ts @@ -8,7 +8,7 @@ import { } from "../../src/constants"; import { toCartesian, toZone } from "../../src/helpers"; import { DEFAULT_TABLE_CONFIG } from "../../src/helpers/table_presets"; -import { CURRENT_VERSION } from "../../src/migrations/data"; +import { getCurrentVersion } from "../../src/migrations/data"; import { BorderDescr, ColorScaleRule, @@ -59,7 +59,7 @@ describe("Migrations", () => { ], }); const data = model.exportData(); - expect(data.version).toBe(CURRENT_VERSION); + expect(data.version).toBe(getCurrentVersion()); expect(data.sheets[0].id).toBeDefined(); expect(data.sheets[0].figures).toBeDefined(); expect(data.sheets[0].cells.A1).toBe("=A1"); @@ -82,7 +82,7 @@ describe("Migrations", () => { }); const data = model.exportData(); const cells = data.sheets[0].cells; - expect(data.version).toBe(CURRENT_VERSION); + expect(data.version).toBe(getCurrentVersion()); // formulas are de-normalized with version 9 expect(cells.A1).toBe("=A1"); expect(cells.A2).toBe("=1"); @@ -450,8 +450,8 @@ describe("Migrations", () => { ], }); const data = model.exportData(); - expect(data.version).toEqual(CURRENT_VERSION); - expect(CURRENT_VERSION).toBeGreaterThanOrEqual(14.5); + expect(data.version).toEqual(getCurrentVersion()); + expect(Number(getCurrentVersion())).toBeGreaterThanOrEqual(14.5); expect(data.sheets[0].tables).toEqual([ { range: "A1:C2", @@ -473,7 +473,7 @@ describe("Migrations", () => { }); expect(model.getters.getTables("1")).toMatchObject([{ range: { zone: toZone("A1:B2") } }]); let data = model.exportData(); - expect(data.version).toBe(CURRENT_VERSION); + expect(data.version).toBe(getCurrentVersion()); expect(data.sheets[0].tables).toEqual([ { range: "A1:B2", @@ -504,7 +504,7 @@ describe("Migrations", () => { expect(getCell(model, "A1")?.style).toEqual(style); expect(getBorder(model, "A1")).toEqual(border); const data = model.exportData(); - expect(data.version).toBe(CURRENT_VERSION); + expect(data.version).toBe(getCurrentVersion()); expect(data.sheets[0].cells).toEqual({ A1: "hi" }); expect(data.sheets[0].formats).toEqual({ A1: 1 }); expect(data.sheets[0].styles).toEqual({ A1: 1 }); @@ -565,7 +565,7 @@ describe("Migrations", () => { }); expect(model.getters.getTables("1")).toMatchObject([{ range: { zone: toZone("A1:B2") } }]); const data = model.exportData(); - expect(data.version).toBe(CURRENT_VERSION); + expect(data.version).toBe(getCurrentVersion()); expect(data.sheets[0].tables).toEqual([ { range: "A1:B2", @@ -715,7 +715,7 @@ describe("Export", () => { test("complete import, then export", () => { const modelData = { - version: CURRENT_VERSION, + version: getCurrentVersion(), revisionId: DEFAULT_REVISION_ID, sheets: [ { @@ -810,7 +810,7 @@ test("complete import, then export", () => { test("can import cells outside sheet size", () => { const sheetId = "someuuid"; const modelData = { - version: CURRENT_VERSION, + version: getCurrentVersion(), sheets: [ { id: sheetId, @@ -846,7 +846,7 @@ test("Data of a duplicate sheet are correctly duplicated", () => { test("import then export (figures)", () => { const modelData = { - version: CURRENT_VERSION, + version: getCurrentVersion(), revisionId: DEFAULT_REVISION_ID, sheets: [ { diff --git a/tests/pivots/pivot_data.ts b/tests/pivots/pivot_data.ts index 897ba5bd2f..f944152b8b 100644 --- a/tests/pivots/pivot_data.ts +++ b/tests/pivots/pivot_data.ts @@ -2,555 +2,188 @@ import { toZone } from "../../src/helpers/index"; export const pivotModelData = function (xc: string) { return { - version: 19, + version: 25, sheets: [ { id: "pivot", name: "Pivot", colNumber: 26, rowNumber: 100, - rows: {}, - cols: {}, - merges: [], cells: { - A1: { - content: "Created on", - }, - A2: { - content: "04/03/2024", - }, - A3: { - content: "03/03/2024", - }, - A4: { - content: "02/02/2024", - }, - A5: { - content: "04/02/2024", - }, - A6: { - content: "03/28/2024", - }, - A7: { - content: "04/02/2024", - }, - A8: { - content: "04/02/2024", - }, - A9: { - content: "02/27/2024", - }, - A10: { - content: "04/01/2024", - }, - A11: { - content: "04/02/2024", - }, - A12: { - content: "04/03/2024", - }, - A13: { - content: "02/03/2024", - }, - A14: { - content: "03/03/2024", - }, - A15: { - content: "01/26/2024", - }, - A16: { - content: "03/27/2024", - }, - A17: { - content: "03/27/2024", - }, - A18: { - content: "03/31/2024", - }, - A19: { - content: "01/31/2024", - }, - A20: { - content: "04/02/2024", - }, - A21: { - content: "04/05/2024", - }, - A22: { - content: "19/05/2024", - }, - B1: { - content: "Opportunity", - }, - B2: { - content: "my opportunity", - }, - B3: { - content: "test opportunity", - }, - B4: { - content: "interested in tables", - }, - B5: { - content: "Interest in your products", - }, - B6: { - content: "Open Space Design", - }, - B7: { - content: "Modern Open Space", - }, - B8: { - content: "Office Design and Architecture", - }, - B9: { - content: "Distributor Contract", - }, - B10: { - content: "Furnitures", - }, - B11: { - content: "Office Design Project", - }, - B12: { - content: "abc opportunity", - }, - B13: { - content: "Quote for 600 Chairs", - }, - B14: { - content: "Devis pour 150 tapis", - }, - B15: { - content: "5 VP Chairs", - }, - B16: { - content: "Customizable Desk", - }, - B17: { - content: "DeltaPC: 10 Computer Desks", - }, - B18: { - content: "Potential Distributor", - }, - B19: { - content: "Info about services", - }, - B20: { - content: "Quote for 12 Tables", - }, - B21: { - content: "Need 20 Desks", - }, - B22: { - content: "Access to Online Catalog", - }, - C1: { - content: "Contact Name", - }, - C2: { - content: "Michel", - }, - C4: { - content: "Alice", - }, - C7: { - content: "Bob", - }, - C9: { - content: "Charlie", - }, - C10: { - content: "Robin", - }, - C13: { - content: "Erik", - }, - C14: { - content: "Erik", - }, - C15: { - content: "Carlos", - }, - C16: { - content: "Charles", - }, - C17: { - content: "Roger", - }, - C18: { - content: "Mich-Mich", - }, - C20: { - content: "Will", - }, - D1: { - content: "Email", - }, - D2: { - content: "admin@yourcompany.example.com", - }, - D3: { - content: "info@yourcompany.example.com", - }, - D4: { - content: "adam@example.com", - }, - D5: { - content: "info@example.com", - }, - D6: { - content: "info@example.com", - }, - D7: { - content: "henry@exemple.com", - }, - D8: { - content: "info@my.example.com", - }, - D9: { - content: "john@tech.info", - }, - D10: { - content: "info@my.example.com", - }, - D11: { - content: "info@exemple.com", - }, - D12: { - content: "info@yourcompany.example.com", - }, - D13: { - content: "erik@blop.com", - }, - D14: { - content: "erik@blop.com", - }, - D15: { - content: "info@yourcompany.example.com", - }, - D16: { - content: "info@yourcompany.example.com", - }, - D17: { - content: "info@my.example.com", - }, - D18: { - content: "Carlos@inc.sa", - }, - D19: { - content: "info@mycompany.com", - }, - D20: { - content: "Will@example.com", - }, - D21: { - content: "info@mycompany.net", - }, - D22: { - content: "mich-mich@example.com", - }, - E1: { - content: "Salesperson", - }, - E2: { - content: "Kevin", - }, - E3: { - content: "Kevin", - }, - E4: { - content: "Kevin", - }, - E5: { - content: "Eden", - }, - E6: { - content: "Eden", - }, - E7: { - content: "Kevin", - }, - E8: { - content: "Kevin", - }, - E9: { - content: "Kevin", - }, - E10: { - content: "Kevin", - }, - E11: { - content: "Eden", - }, - E12: { - content: "Kevin", - }, - E13: { - content: "Kevin", - }, - E14: { - content: "Kevin", - }, - E15: { - content: "Kevin", - }, - E16: { - content: "Eden", - }, - E17: { - content: "Eden", - }, - E18: { - content: "Eden", - }, - E19: { - content: "Kevin", - }, - E20: { - content: "Kevin", - }, - E21: { - content: "Kevin", - }, - E22: { - content: "Eden", - }, - F1: { - content: "Expected Revenue", - }, - F2: { - format: 1, - content: "2000", - }, - F3: { - format: 1, - content: "11000", - }, - F4: { - format: 1, - content: "4500", - }, - F5: { - format: 1, - }, - F6: { - format: 1, - }, - F7: { - format: 1, - }, - F8: { - format: 1, - content: "9000", - }, - F9: { - format: 1, - content: "19800", - }, - F10: { - format: 1, - content: "3800", - }, - F11: { - format: 1, - content: "24000", - }, - F12: { - format: 1, - }, - F13: { - format: 1, - content: "22500", - }, - F14: { - format: 1, - content: "40000", - }, - F15: { - format: 1, - content: "5600", - }, - F16: { - format: 1, - content: "15000", - }, - F17: { - format: 1, - content: "35000", - }, - F18: { - format: 1, - content: "1000", - }, - F19: { - format: 1, - content: "25000", - }, - F20: { - format: 1, - content: "40000", - }, - F21: { - format: 1, - content: "60000", - }, - F22: { - format: 1, - content: "2000", - }, - G1: { - content: "Expected MRR", - }, - G7: { - content: "333.33", - }, - H1: { - content: "Stage", - }, - H2: { - content: "New", - }, - H3: { - content: "New", - }, - H4: { - content: "New", - }, - H5: { - content: "Won", - }, - H6: { - content: "Proposition", - }, - H7: { - content: "Won", - }, - H8: { - content: "Proposition", - }, - H9: { - content: "Won", - }, - H10: { - content: "Qualified", - }, - H11: { - content: "New", - }, - H12: { - content: "New", - }, - H13: { - content: "Qualified", - }, - H14: { - content: "New", - }, - H15: { - content: "Proposition", - }, - H16: { - content: "Proposition", - }, - H17: { - content: "Qualified", - }, - H18: { - content: "Qualified", - }, - H19: { - content: "Qualified", - }, - H20: { - content: "New", - }, - H21: { - content: "Proposition", - }, - H22: { - content: "Won", - }, - I1: { - content: "Active", - }, - I2: { - content: "TRUE", - }, - I3: { - content: "FALSE", - }, - I4: { - content: "TRUE", - }, - I5: { - content: "TRUE", - }, - I6: { - content: "TRUE", - }, - I7: { - content: "TRUE", - }, - I8: { - content: "TRUE", - }, - I9: { - content: "TRUE", - }, - I10: { - content: "TRUE", - }, - I11: { - content: "TRUE", - }, - I12: { - content: "FALSE", - }, - I13: { - content: "FALSE", - }, - I14: { - content: "FALSE", - }, - I15: { - content: "FALSE", - }, - I16: { - content: "FALSE", - }, - I17: { - content: "FALSE", - }, - I18: { - content: "FALSE", - }, - I19: { - content: "FALSE", - }, - I20: { - content: "FALSE", - }, - I21: { - content: "FALSE", - }, - I22: { - content: "FALSE", - }, + A1: "Created on", + A2: "04/03/2024", + A3: "03/03/2024", + A4: "02/02/2024", + A5: "04/02/2024", + A6: "03/28/2024", + A7: "04/02/2024", + A8: "04/02/2024", + A9: "02/27/2024", + A10: "04/01/2024", + A11: "04/02/2024", + A12: "04/03/2024", + A13: "02/03/2024", + A14: "03/03/2024", + A15: "01/26/2024", + A16: "03/27/2024", + A17: "03/27/2024", + A18: "03/31/2024", + A19: "01/31/2024", + A20: "04/02/2024", + A21: "04/05/2024", + A22: "19/05/2024", + B1: "Opportunity", + B2: "my opportunity", + B3: "test opportunity", + B4: "interested in tables", + B5: "Interest in your products", + B6: "Open Space Design", + B7: "Modern Open Space", + B8: "Office Design and Architecture", + B9: "Distributor Contract", + B10: "Furnitures", + B11: "Office Design Project", + B12: "abc opportunity", + B13: "Quote for 600 Chairs", + B14: "Devis pour 150 tapis", + B15: "5 VP Chairs", + B16: "Customizable Desk", + B17: "DeltaPC: 10 Computer Desks", + B18: "Potential Distributor", + B19: "Info about services", + B20: "Quote for 12 Tables", + B21: "Need 20 Desks", + B22: "Access to Online Catalog", + C1: "Contact Name", + C2: "Michel", + C4: "Alice", + C7: "Bob", + C9: "Charlie", + C10: "Robin", + C13: "Erik", + C14: "Erik", + C15: "Carlos", + C16: "Charles", + C17: "Roger", + C18: "Mich-Mich", + C20: "Will", + D1: "Email", + D2: "admin@yourcompany.example.com", + D3: "info@yourcompany.example.com", + D4: "adam@example.com", + D5: "info@example.com", + D6: "info@example.com", + D7: "henry@exemple.com", + D8: "info@my.example.com", + D9: "john@tech.info", + D10: "info@my.example.com", + D11: "info@exemple.com", + D12: "info@yourcompany.example.com", + D13: "erik@blop.com", + D14: "erik@blop.com", + D15: "info@yourcompany.example.com", + D16: "info@yourcompany.example.com", + D17: "info@my.example.com", + D18: "Carlos@inc.sa", + D19: "info@mycompany.com", + D20: "Will@example.com", + D21: "info@mycompany.net", + D22: "mich-mich@example.com", + E1: "Salesperson", + E2: "Kevin", + E3: "Kevin", + E4: "Kevin", + E5: "Eden", + E6: "Eden", + E7: "Kevin", + E8: "Kevin", + E9: "Kevin", + E10: "Kevin", + E11: "Eden", + E12: "Kevin", + E13: "Kevin", + E14: "Kevin", + E15: "Kevin", + E16: "Eden", + E17: "Eden", + E18: "Eden", + E19: "Kevin", + E20: "Kevin", + E21: "Kevin", + E22: "Eden", + F1: "Expected Revenue", + F2: "2000", + F3: "11000", + F4: "4500", + F8: "9000", + F9: "19800", + F10: "3800", + F11: "24000", + F13: "22500", + F14: "40000", + F15: "5600", + F16: "15000", + F17: "35000", + F18: "1000", + F19: "25000", + F20: "40000", + F21: "60000", + F22: "2000", + G1: "Expected MRR", + G7: "333.33", + H1: "Stage", + H2: "New", + H3: "New", + H4: "New", + H5: "Won", + H6: "Proposition", + H7: "Won", + H8: "Proposition", + H9: "Won", + H10: "Qualified", + H11: "New", + H12: "New", + H13: "Qualified", + H14: "New", + H15: "Proposition", + H16: "Proposition", + H17: "Qualified", + H18: "Qualified", + H19: "Qualified", + H20: "New", + H21: "Proposition", + H22: "Won", + I1: "Active", + I2: "TRUE", + I3: "FALSE", + I4: "TRUE", + I5: "TRUE", + I6: "TRUE", + I7: "TRUE", + I8: "TRUE", + I9: "TRUE", + I10: "TRUE", + I11: "TRUE", + I12: "FALSE", + I13: "FALSE", + I14: "FALSE", + I15: "FALSE", + I16: "FALSE", + I17: "FALSE", + I18: "FALSE", + I19: "FALSE", + I20: "FALSE", + I21: "FALSE", + I22: "FALSE", }, - conditionalFormats: [], - figures: [], - tables: [], - areGridLinesVisible: true, - isVisible: true, - headerGroups: { - ROW: [], - COL: [], + styles: {}, + formats: { + "F2:F22": 1, }, - dataValidationRules: [], + isVisible: true, + areGridLinesVisible: true, }, ], - styles: {}, formats: { "1": "[$$]#,##0.00", }, @@ -594,5 +227,6 @@ export const pivotModelData = function (xc: string) { }, }, pivotNextId: 2, + customTableStyles: {}, }; };