Group Layout

TableSheet supports using a group layout, enabling you to define the header aggregations and footer subtotals.

Description
app.vue
index.html
Copy to CodeMine

Layout Mode

TableSheet supports three different Group Layouts, which can be set by groupLayout:

tableSheet.options.groupLayout = {
    mode: GC.Spread.Sheets.TableSheet.GroupLayoutMode.outline
};

Group Layout Mode:

  • Tabular
  • Outline
  • Condensed

Group Fields

Users can set up multiple group fields at the same time, or for each field.

Spacing

The customer could show the spacing to separate the root group level items with the spacing. Spacing is only useful in Outline and Condensed modes.

tableSheet.groupBy([
    {
        field: "priority",
        spacing: { row: 20 }
    }
]);

Multiple Group Fields

Multiple group headings can be added, with different levels of indentation for the content of the grouped items.

tableSheet.groupBy([
    {
        field: "priority"
    },
    {
        field: "duration"
    }
]);

Summary Fields Options

Users can set the content of each field.

tableSheet.groupBy([
    {
        field: "priority",
        summaryFields: [
            {  
                caption: "PRIORITY_FOOTER",
                formula: '="1st Summary Footer"',
                position: 'footer',
                relateTo: 'name',
                style: {
                    formatter: "$ #,##0.00",
                    backColor: 'green' 
                }
            },
        ]
    }
]);
Layout Mode TableSheet supports three different Group Layouts, which can be set by groupLayout: Group Layout Mode: Tabular Outline Condensed Group Fields Users can set up multiple group fields at the same time, or for each field. Spacing The customer could show the spacing to separate the root group level items with the spacing. Spacing is only useful in Outline and Condensed modes. Multiple Group Fields Multiple group headings can be added, with different levels of indentation for the content of the grouped items. Summary Fields Options Users can set the content of each field.
<template> <div id="split-view" class="sample-tutorial"> <gc-spread-sheets class="sample-spreadsheets split-content" @workbookInitialized="initSpread"> </gc-spread-sheets> <div class="split-panel"> <div class="split-tabs"> <button class="split-tab-link" :class="{ active: selectedTab === 0 }" @click="selectTab(0)"> TableSheet Panel </button> <button class="split-tab-link" :class="{ active: selectedTab === 1 }" @click="selectTab(1)"> Setting </button> </div> <div class="split-panels"> <div id="panel" class="split-tab-content container" :key="0" v-show="selectedTab === 0"></div> <div id="settings" class="split-tab-content options-container" :key="1" v-show="selectedTab === 1"> <fieldset> <legend>Group Layout Mode</legend> <select v-model.number="tableSheetGroupLayoutMode" @change="onTableSheetGroupLayoutMode"> <option :value="0">Tabular</option> <option :value="1">Outline</option> <option :value="2">Condensed</option> </select> </fieldset> <fieldset> <legend>Default Summary Position</legend> <select v-model.number="tableSheetGroupLayoutPosition" @change="onTableSheetGroupLayoutPosition"> <option value="header" selected>Header</option> <option value="footer">Footer</option> </select> </fieldset> </div> </div> </div> </div> </template> <script setup> import { ref, toRaw } from "vue"; import GC from "@mescius/spread-sheets"; import "@mescius/spread-sheets-vue"; import "@mescius/spread-sheets-tablesheet"; const spreadRef = ref(null); let tableSheet = ref(null); let selectedTab = ref(0); let tableSheetGroupLayoutMode = ref(1); let tableSheetGroupLayoutPosition = ref("header"); let OutlineAndCondensedGroup = [ { field: "Office", caption: "OFFICE", style: { foreColor: '#000000', backColor: '#ededed', fontSize: "20px", cellPadding: "16 0 0 0" }, headerStyle: { font: "10px Arial", foreColor: "#666", cellPadding: "2 0 0 0" }, summaryFields: [ { caption: "COUNT", formula: "=COUNTA([OrderID])", style: { foreColor: "rgb(185, 122, 87)", fontSize: "20px", cellPadding: "16 0 0 0" }, headerStyle: { font: "10px Arial", foreColor: "#666", cellPadding: "2 0 0 0" } }, { caption: "TOTAL AMOUNT", formula: '=SUM([Amount])', relateTo: "Amount", style: { formatter: "$ #,##0.00", cellPadding: "30 0 0 0" }, headerStyle: { cellPadding: "8 0 0 0" } }, { caption: "Σ(Quantity)", formula: "=SUM([Quantity])", style: { cellPadding: "30 0 0 0" }, headerStyle: { cellPadding: "8 0 0 0" } }, ] }, { field: "OrderDate", spacing: { row: 20 }, caption: "ORDER DATE", style: { foreColor: '#000000', backColor: '#f7f7f7', fontSize: "20px", cellPadding: "16 0 0 0" }, headerStyle: { font: "10px Arial", foreColor: "#666", cellPadding: "2 0 0 0" }, summaryFields: [ { caption: "SUBTOTAL AMOUNT", formula: '=SUM([Amount])', position: "footer", relateTo: "Amount", style: { formatter: "$ #,##0.00", cellPadding: "30 0 0 0" }, headerStyle: { cellPadding: "8 0 0 0" } }, { caption: "SUBTOTAL COUNT", formula: "=SUM([Quantity])", position: "footer", style: { cellPadding: "30 0 0 0" }, headerStyle: { cellPadding: "8 0 0 0" } } ] } ]; let TabularGroup = [ { field: "Office", caption: "OFFICE", style: { foreColor: '#000000', backColor: '#ededed', fontSize: "20px" }, headerStyle: { font: "10px Arial", foreColor: "#666" }, summaryFields: [ { caption: "COUNT", formula: "=COUNTA([OrderID])", style: { foreColor: "rgb(185, 122, 87)", fontSize: "20px" }, headerStyle: { font: "10px Arial", foreColor: "#666" } }, { caption: "TOTAL AMOUNT", formula: '=SUM([Amount])', relateTo: "Amount", style: { formatter: "$ #,##0.00" }, }, { caption: "Σ(Quantity)", formula: "=SUM([Quantity])", }, ] }, { field: "OrderDate", spacing: { row: 20 }, caption: "ORDER DATE", style: { foreColor: '#000000', backColor: '#f7f7f7', fontSize: "20px" }, headerStyle: { font: "10px Arial", foreColor: "#666" }, summaryFields: [ { caption: "SUBTOTAL AMOUNT", formula: '=SUM([Amount])', position: "footer", relateTo: "Amount", style: { formatter: "$ #,##0.00" }, }, { caption: "SUBTOTAL COUNT", formula: "=SUM([Quantity])", position: "footer", } ] } ]; function initSpread(spread) { spreadRef.value = spread; spread.suspendPaint(); spread.options.autoFitType = GC.Spread.Sheets.AutoFitType.cellWithHeader; spread.options.scrollByPixel = true; spread.options.scrollPixel = 5; //init a data manager var dataManager = spread.dataManager(); var ordersTable = dataManager.addTable("ordersTable", { data: orderDataSource, schema: { type: "csv", columns: { OrderDate: { dataType: "date" }, ShipDate: { dataType: "date" }, Quantity: { dataType: "number" }, Amount: { dataType: "number" }, }, }, }); //init a table sheet var sheet = spread.addSheetTab( 0, "TableSheet1", GC.Spread.Sheets.SheetType.tableSheet ); tableSheet.value = sheet; sheet.setDefaultRowHeight(40, GC.Spread.Sheets.SheetArea.colHeader); sheet.options.allowAddNew = false; //hide new row sheet.options.groupLayout.mode = GC.Spread.Sheets.TableSheet.GroupLayoutMode.outline; // apply the outline group layout mode sheet.options.showRowNumber = true; sheet.applyTableTheme(GC.Spread.Sheets.Tables.TableThemes.light1); sheet.options.alternatingRowOptions = { step: [1, 1], style: new GC.Spread.Sheets.Style("#ffffff"), }; //bind a view to the table sheet ordersTable.fetch().then(function () { var myView = ordersTable.addView("myView", [ { value: "OrderID", width: 180, style: { formatter: "O00000" } }, { value: "Product", width: 200 }, { value: "Quantity", width: 100 }, { value: "OrderDate", width: 100, visible: false }, { value: "ShipDate", width: 100, visible: false }, { value: "Office", width: 100 }, { value: "Category", width: 150 }, { value: "Amount", width: 120 }, ]); sheet.setDataView(myView); //init a table sheet panel var host = document.getElementById("panel"); var panel = new GC.Spread.Sheets.TableSheet.TableSheetPanel( "myPanel", sheet, host ); sheet.groupBy(OutlineAndCondensedGroup); }); spread.resumePaint(); initSplitView(spread); } function onTableSheetGroupLayoutMode() { resetGroupLayout(); } function onTableSheetGroupLayoutPosition() { resetGroupLayout(); } function resetGroupLayout() { tableSheet.value.options.groupLayout = { mode: tableSheetGroupLayoutMode.value, position: tableSheetGroupLayoutPosition.value }; if (tableSheetGroupLayoutMode.value === 0) { tableSheet.value.groupBy(TabularGroup); } else { tableSheet.value.groupBy(OutlineAndCondensedGroup); } } function selectTab(index) { selectedTab.value = index; } function initSplitView(spread) { var host = document.getElementById("split-view"); var content = host.getElementsByClassName("split-content")[0]; var panel = host.getElementsByClassName("split-panel")[0]; new SplitView({ host: host, content: content, panel: panel, refreshContent: function () { spread.refresh(); }, defaultPanelWidth: 320, allowResize: false, }); } function getBaseApiUrl() { return window.location.href.match(/http.+spreadjs\/demos\//)[0] + 'server/api'; } </script> <style scoped> #app { height: 100%; } .sample-tutorial { position: relative; height: 100%; overflow: hidden; } body { position: absolute; top: 0; bottom: 0; left: 0; right: 0; margin: 0; } .sample-spreadsheets { width: calc(100% - 303px); height: 100%; overflow: hidden; float: left; } .container { width: 300px; height: 100%; float: left; border: 1px solid lightgrey; } fieldset select { margin-top: 6px; padding: 4px 6px; width: 125px; } fieldset input { margin-top: 6px; padding: 4px 6px; width: 110px; } .split-panels { height: 535px; } .tab-container { width: 50%; margin: 0 auto; } .split-tabs { position: absolute; top: 12px; right: 5px; display: flex; justify-content: flex-start; flex-direction: row; align-items: center; } .split-tab-link { height: 30px; background-color: #f1f1f1; border: none; cursor: pointer; transition: background-color 0.3s; margin-bottom: 10px; } .split-tab-link:hover { background-color: #ddd; } .split-tab-link.active { background-color: #ccc; } .split-tab-content.active { display: block; } </style>
<!DOCTYPE html> <html style="height:100%;font-size:14px;"> <head> <meta charset="utf-8" /> <meta http-equiv="X-UA-Compatible" content="IE=edge" /> <title>SpreadJS VUE</title> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <link rel="stylesheet" type="text/css" href="$DEMOROOT$/en/vue3/node_modules/@mescius/spread-sheets/styles/gc.spread.sheets.excel2013white.css"> <link rel="stylesheet" type="text/css" href="$DEMOROOT$/spread/source/splitView/splitView.css"> <script src="$DEMOROOT$/spread/source/data/orderDataSource.js" type="text/javascript"></script> <script src="$DEMOROOT$/en/vue3/node_modules/systemjs/dist/system.src.js"></script> <script src="./systemjs.config.js"></script> <script src="./compiler.js" type="module"></script> <script src="$DEMOROOT$/spread/source/splitView/SplitView.js"></script> <script> var System = SystemJS; System.import("./src/app.js"); System.import('$DEMOROOT$/en/lib/vue3/license.js'); </script> </head> <body> <div id="app"></div> </body> </html>
(function (global) { SystemJS.config({ transpiler: 'plugin-babel', babelOptions: { es2015: true }, paths: { // paths serve as alias 'npm:': 'node_modules/' }, packageConfigPaths: [ './node_modules/*/package.json', "./node_modules/@mescius/*/package.json", "./node_modules/@babel/*/package.json", "./node_modules/@vue/*/package.json" ], map: { 'vue': "npm:vue/dist/vue.esm-browser.js", 'tiny-emitter': 'npm:tiny-emitter/index.js', 'plugin-babel': 'npm:systemjs-plugin-babel/plugin-babel.js', "systemjs-babel-build": "npm:systemjs-plugin-babel/systemjs-babel-browser.js", '@mescius/spread-sheets': 'npm:@mescius/spread-sheets/index.js', '@mescius/spread-sheets-vue': 'npm:@mescius/spread-sheets-vue/index.js', '@mescius/spread-sheets-tablesheet': 'npm:@mescius/spread-sheets-tablesheet/index.js' }, meta: { '*.css': { loader: 'systemjs-plugin-css' }, '*.vue': { loader: "../plugin-vue/index.js" } } }); })(this);