Getting Started

Steps for getting started with FlexSheet in AngularJS applications:

  1. Add references to AngularJS, Wijmo, and Wijmo's AngularJS directives.
  2. Add a component to provide data and logic.
  3. Add a FlexGrid to the page and bind it to the data.
  4. Add a Sheet to the FlexSheet.
  5. Add some CSS to customize the grid's appearance.
<!DOCTYPE html> <html> <head> <link rel="stylesheet" type="text/css" href="css/bootstrap.css" /> <link rel="stylesheet" type="text/css" href="css/wijmo.css" /> <link rel="stylesheet" type="text/css" href="css/themes/wijmo.theme.office.css" /> <link rel="stylesheet" type="text/css" href="css/app.css"/> <!-- JSZip library --> <script src="http://cdnjs.cloudflare.com/ajax/libs/jszip/2.5.0/jszip.min.js"></script> <!-- Wijmo --> <script src="scripts/wijmo.js" type="text/javascript"/></script/> <script src="scripts/wijmo.input.js" type="text/javascript"/></script/> <script src="scripts/wijmo.grid.js" type="text/javascript"/></script/> <script src="scripts/wijmo.grid.filter.js" type="text/javascript"></script> <script src="scripts/wijmo.grid.sheet.js" type="text/javascript"></script> <script src="scripts/wijmo.grid.xlsx.js" type="text/javascript"></script> <script src="scripts/wijmo.xlsx.js" type="text/javascript"></script> <!-- Wijmo for Angular 2 --> <script src="scripts/wijmo.angular2.js" type="text/javascript"/></script/> <!-- Load the root application module --> <script> System.import('./src/app'); </script> </head> <body> <!-- Add root application component --> <app-cmp> <wj-flex-sheet #flexSheetIntro (initialized)="flexSheetInit(flexSheetIntro)"> <wj-sheet name="Country"></wj-sheet> <wj-sheet name="Empty Sheet"></wj-sheet> </wj-flex-sheet> </app-cmp> </body> </html>
import { Injectable } from '@angular/core'; // Common data service @Injectable() export class DataSvc { private _products = ['Widget', 'Gadget', 'Doohickey']; private _countries = ['US', 'Germany', 'UK', 'Japan', 'Italy', 'Greece']; get products(): string[] { return this._products; } get countries(): string[] { return this._countries; } // get matches for a search term getData(count: number): any[] { var data = [], i = 0, countryId, productId; for (var i = 0; i < count; i++) { countryId = Math.floor(Math.random() * this._countries.length); productId = Math.floor(Math.random() * this._products.length); data.push({ id: i, countryId: countryId, productId: productId, date: new Date(2014, i % 12, i % 28), amount: Math.random() * 10000, active: i % 4 === 0 }); } return data; } } ---------------------Component------------------- // Angular import * as wjcGridSheet from 'wijmo/wijmo.grid.sheet'; import * as wjcInput from 'wijmo/wijmo.input'; import * as wjcGrid from 'wijmo/wijmo.grid'; import * as wjcCore from 'wijmo/wijmo'; import { Component, EventEmitter, Inject, enableProdMode, ViewChild, NgModule } from '@angular/core'; import { FormsModule } from '@angular/forms'; import { CommonModule } from '@angular/common'; import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; import { BrowserModule } from '@angular/platform-browser'; import { WjGridSheetModule } from 'wijmo/wijmo.angular2.grid.sheet'; import { WjInputModule } from 'wijmo/wijmo.angular2.input'; import { TabsModule } from './components/AppTab'; import { DataSvc } from './services/DataSvc'; 'use strict'; // The Explorer application root component. @Component({ selector: 'app-cmp', templateUrl: 'src/app.html' }) export class AppCmp { protected dataSvc: DataSvc; data: any[]; // references FlexSheet named 'flexSheet' in the view @ViewChild('flexSheetIntro') flexSheetIntro: wjcGridSheet.FlexSheet; constructor(@Inject(DataSvc) dataSvc: DataSvc) { this.dataSvc = dataSvc; this.data = dataSvc.getData(50); } flexSheetInit(flexSheetIntro: wjcGridSheet.FlexSheet) { var self = this; if (flexSheetIntro) { flexSheetIntro.deferUpdate(() => { flexSheetIntro.selectedSheetIndex = 0; flexSheetIntro.selectedSheet.itemsSource = this.data; self._initDataMapForBindingSheet(flexSheetIntro); }); } } // initialize the dataMap for the bound sheet. private _initDataMapForBindingSheet(flexSheet) { var column; if (flexSheet) { column = flexSheet.columns.getColumn('countryId'); if (column && !column.dataMap) { column.dataMap = this._buildDataMap(this.dataSvc.countries); } column = flexSheet.columns.getColumn('productId'); if (column && !column.dataMap) { column.dataMap = this._buildDataMap(this.dataSvc.products); } } } // build a data map from a string array using the indices as keys private _buildDataMap(items) { var map = []; for (var i = 0; i < items.length; i++) { map.push({ key: i, value: items[i] }); } return new wjcGrid.DataMap(map, 'key', 'value'); } } @NgModule({ imports: [WjInputModule, WjGridSheetModule, BrowserModule, FormsModule, TabsModule], declarations: [AppCmp], providers: [DataSvc], bootstrap: [AppCmp] }) export class AppModule { } enableProdMode(); // Bootstrap application with hash style navigation and global services. platformBrowserDynamic().bootstrapModule(AppModule);
/* set default grid height and some shadow */ .wj-flexsheet { height: 400px; background-color: white; box-shadow: 4px 4px 10px 0px rgba(50, 50, 50, 0.75); margin-bottom: 12px; }

Result (live):

Sorting

FlexSheet can be sorted by any of its columns.

The SortManager helps FlexSheet to manage the sort process. The following example uses SortManager to specify the order of the sorting, add or remove sort columns, and change the order of the sort columns.

<wj-flex-sheet #sortSheet (initialized)="sortSheetInit(sortSheet)"> <wj-sheet name="Country"></wj-sheet> </wj-flex-sheet> <table class="table table-bordered"> <thead> <tr> <th class="text-center">Column</th> <th class="text-center">Order</th> </tr> </thead> <tbody *ngIf="sortManager"> <tr *ngFor="let sortItem of sortManager.sortDescriptions.items" (click)="sortManager.sortDescriptions.moveCurrentTo(sortItem)" [ngClass]="{success: sortItem === sortManager.sortDescriptions.currentItem}"> <td> <select class="form-control" (change)="applySortColumnIndex($event, sortItem)"> <option value=-1></option> <option *ngFor="let column of columns; let i = index" [selected]="i === sortItem.columnIndex" value={​{i}​}> {​{column}​} </option> </select> </td> <td> <select class="form-control" (change)="applySortAscending($event, sortItem)"> <option [value]="true" [selected]="sortItem.ascending">Ascending</option> <option [value]="false" [selected]="!sortItem.ascending">Descending</option> </select> </td> </tr> </tbody> </table> <div class="btn-group"> <button type="button" class="btn btn-default" (click)="addSortLevel()"> Add Level </button> <button type="button" class="btn btn-default" (click)="deleteSortLevel()"> Delete Level </button> <button type="button" class="btn btn-default" (click)="copySortLevel()"> Copy Level </button> </div> <div class="btn-group"> <button id="moveup" type="button" class="btn btn-default" [disabled]="sortManager && sortManager.sortDescriptions.currentPosition <= 0" (click)="moveSortLevel(-1)"> <span class="glyphicon glyphicon-arrow-up"></span> </button> <button id="movedown" type="button" class="btn btn-default" [disabled]="sortManager && sortManager.sortDescriptions.currentPosition >= sortManager.sortDescriptions.itemCount - 1" (click)="moveSortLevel(1)"> <span class="glyphicon glyphicon-arrow-down"></span> </button> </div> <div class="btn-group"> <button type="button" class="btn btn-default" (click)="commitSort()">OK</button> <button type="button" class="btn btn-default" (click)="cancelSort()">Cancel</button> </div>
// references FlexSheet named 'sortSheet' in the view @ViewChild('sortSheet') sortSheet: wjcGridSheet.FlexSheet; sortSheetInit(sortSheet: wjcGridSheet.FlexSheet) { var self = this; if (sortSheet) { sortSheet.deferUpdate(() => { sortSheet.selectedSheetIndex = 0; sortSheet.selectedSheet.itemsSource = this.data; self._initDataMapForBindingSheet(sortSheet); self.sortManager = sortSheet.sortManager; self.columns = self._getColumns(sortSheet); }); sortSheet.selectedSheetChanged.addHandler(() => { self.columns = self._getColumns(sortSheet); if (!self.sortManager) { self.sortManager = sortSheet.sortManager; } }); } } // commit the sorts commitSort() { this.sortManager.commitSort(); }; // cancel the sorts cancelSort() { this.sortManager.cancelSort(); }; // add new sort level addSortLevel() { this.sortManager.addSortLevel(); }; // delete current sort level deleteSortLevel() { this.sortManager.deleteSortLevel(); }; // copy a new sort level by current sort level setting. copySortLevel() { this.sortManager.copySortLevel(); }; // move the sort level moveSortLevel(offset) { this.sortManager.moveSortLevel(offset); }; // apply column index property for sort item applySortColumnIndex(e, sortItem) { sortItem.columnIndex = +e.target.value; } // apply asceding property for sort item applySortAscending(e, sortItem) { if (e.target.value === 'true') { sortItem.ascending = true; } else { sortItem.ascending = false; } } private _getColumns(flexSheet: wjcGridSheet.FlexSheet): string[] { var columns = [], i = 0; if (flexSheet) { for (; i < flexSheet.columns.length; i++) { columns.push('Column ' + wjcGridSheet.FlexSheet.convertNumberToAlpha(i)); } } return columns; }

Result (live):

Column Order

Format Cells

FlexSheet allows you to set format for each cell. This includes setting font style, data format of cell value (Date/Number format), cell's fill color and horizontal alignment.

<wj-flex-sheet #formatSheet (initialized)="formatSheetInit(formatSheet)"> <wj-sheet name="Number" [rowCount]="20" [columnCount]="8"></wj-sheet> <wj-sheet name="Date" [rowCount]="20" [columnCount]="8"></wj-sheet> </wj-flex-sheet> <wj-color-picker #colorPicker style="display:none;position:fixed;z-index:100"> (initialized)="colorPickerInit(colorPicker)"</wj-color-picker> <div class="well well-lg"> <div> Format: <wj-menu header="Format" [(value)]="format"> <wj-menu-item value="0">Decimal Format</wj-menu-item> <wj-menu-item value="n2">Number Format</wj-menu-item> <wj-menu-item value="p">Percentage Format</wj-menu-item> <wj-menu-item value="c2">Currency Format</wj-menu-item> <wj-menu-separator></wj-menu-separator> <wj-menu-item value="d">Short Date</wj-menu-item> <wj-menu-item value="D">Long Date</wj-menu-item> <wj-menu-item value="f">Full Date/TIme (short time)</wj-menu-item> <wj-menu-item value="F">Full Date/TIme (long time)</wj-menu-item> </wj-menu> </div> <div> Font: <wj-combo-box #cboFontName style="width:120px" [itemsSource]="fonts" [selectedIndex]="10" [displayMemberPath]="'name'" [selectedValuePath]="'value'" [isEditable]="false" (initialized)="cboFontNameInit(cboFontName)"> </wj-combo-box> <wj-combo-box #cboFontSize style="width:80px" [itemsSource]="fontSizeList" [selectedIndex]="5" [displayMemberPath]="'name'" [selectedValuePath]="'value'" [isEditable]="false" (initialized)="cboFontSizeInit(cboFontSize)"> </wj-combo-box> <div class="btn-group"> <button type="button" class="btn btn-default {​{selectionFormatState.isBold ? 'active' : ''}​}" (click)="applyBoldStyle()">Bold</button> <button type="button" class="btn btn-default {​{selectionFormatState.isItalic ? 'active' : ''}​}" (click)="applyItalicStyle()">Italic</button> <button type="button" class="btn btn-default {​{selectionFormatState.isUnderline ? 'active' : ''}​}" (click)="applyUnderlineStyle()">Underline</button> </div> </div> <div> Color: <div class="btn-group"> <button type="button" class="btn btn-default" (click)="showColorPicker($event, false)">Fore Color</button> <button type="button" class="btn btn-default" (click)="showColorPicker($event, true)">Fill Color</button> </div> Alignment: <div class="btn-group"> <button type="button" class="btn btn-default {​{selectionFormatState.textAlign === 'left' ? 'active' : ''}​}" (click)="applyCellTextAlign('left')">Left</button> <button type="button" class="btn btn-default {​{selectionFormatState.textAlign === 'center' ? 'active' : ''}​}" (click)="applyCellTextAlign('center')">Center</button> <button type="button" class="btn btn-default {​{selectionFormatState.textAlign === 'right' ? 'active' : ''}​}" (click)="applyCellTextAlign('right')">Right</button> </div> </div> </div>
// references FlexSheet named 'formatSheet' in the view @ViewChild('formatSheet') formatSheet: wjcGridSheet.FlexSheet; // references Combobox named 'cboFontName' in the view @ViewChild('cboFontName') cboFontName: wjcInput.ComboBox; // references Combobox named 'cboFontSize' in the view @ViewChild('cboFontSize') cboFontSize: wjcInput.ComboBox; // references Combobox named 'cboFontSize' in the view @ViewChild('colorPicker') colorPicker: wjcInput.ColorPicker; // Gets or sets _format for the formatSheet. get format(): string { return this._format; } set format(value: string) { if (this._format !== value) { this._format = value; this.formatSheet.applyCellsStyle({ format: this._format }); } } formatSheetInit(formatSheet: wjcGridSheet.FlexSheet) { var self = this; if (formatSheet) { formatSheet.deferUpdate(() => { var sheetIdx, sheetName, colIdx, rowIdx, date; for (sheetIdx = 0; sheetIdx < formatSheet.sheets.length; sheetIdx++) { formatSheet.selectedSheetIndex = sheetIdx; sheetName = formatSheet.selectedSheet.name; for (colIdx = 0; colIdx < formatSheet.columns.length; colIdx++) { for (rowIdx = 0; rowIdx < formatSheet.rows.length; rowIdx++) { if (sheetName === 'Number') { formatSheet.setCellData(rowIdx, colIdx, colIdx + rowIdx); } else { date = new Date(2015, colIdx, rowIdx + 1); formatSheet.setCellData(rowIdx, colIdx, date); } } } } formatSheet.selectedSheetIndex = 0; self._updateSelection(formatSheet, formatSheet.selection); }); formatSheet.selectionChanged.addHandler((sender: any, args: wjcGrid.CellRangeEventArgs) => { self._updateSelection(formatSheet, args.range); self.selectionFormatState = formatSheet.getSelectionFormatState(); }); } } cboFontNameInit(cboFontName: wjcInput.ComboBox) { var self = this; if (cboFontName) { cboFontName.selectedIndexChanged.addHandler(() => { // apply the font family for the selected cells if (!self._updatingSelection) { self.formatSheet.applyCellsStyle({ fontFamily: cboFontName.selectedItem.value }); } }); } } cboFontSizeInit(cboFontSize: wjcInput.ComboBox) { var self = this; if (cboFontSize) { cboFontSize.selectedIndexChanged.addHandler(() => { // apply the font size for the selected cells if (!self._updatingSelection) { self.formatSheet.applyCellsStyle({ fontSize: cboFontSize.selectedItem.value }); } }); } } colorPickerInit(colorPicker: wjcInput.ColorPicker) { var self = this, ua = window.navigator.userAgent, blurEvt: string; if (colorPicker) { // if the browser is firefox, we should bind the blur event. (TFS #124387) // if the browser is IE, we should bind the focusout event. (TFS #124500) blurEvt = /firefox/i.test(ua) ? 'blur' : 'focusout'; // Hide the color picker control when it lost the focus. colorPicker.hostElement.addEventListener(blurEvt, () => { setTimeout(() => { if (!colorPicker.containsFocus()) { self._applyFillColor = false; colorPicker.hostElement.style.display = 'none'; } }, 0); }); // Initialize the value changed event handler for the color picker control. colorPicker.valueChanged.addHandler(() => { if (self._applyFillColor) { self.formatSheet.applyCellsStyle({ backgroundColor: colorPicker.value }); } else { self.formatSheet.applyCellsStyle({ color: colorPicker.value }); } }); } } // apply the text alignment for the selected cells applyCellTextAlign(textAlign) { if (this.formatSheet) { this.formatSheet.applyCellsStyle({ textAlign: textAlign }); this.selectionFormatState.textAlign = textAlign; } } // apply the bold font weight for the selected cells applyBoldStyle() { if (this.formatSheet) { this.formatSheet.applyCellsStyle({ fontWeight: this.selectionFormatState.isBold ? 'none' : 'bold' }); this.selectionFormatState.isBold = !this.selectionFormatState.isBold; } } // apply the underline text decoration for the selected cells applyUnderlineStyle() { if (this.formatSheet) { this.formatSheet.applyCellsStyle({ textDecoration: this.selectionFormatState.isUnderline ? 'none' : 'underline' }); this.selectionFormatState.isUnderline = !this.selectionFormatState.isUnderline; } } // apply the italic font style for the selected cells applyItalicStyle() { if (this.formatSheet) { this.formatSheet.applyCellsStyle({ fontStyle: this.selectionFormatState.isItalic ? 'none' : 'italic' }); this.selectionFormatState.isItalic = !this.selectionFormatState.isItalic; } }

Result (live):

Format: Decimal Format Number Format Percentage Format Currency Format Short Date Long Date Full Date/TIme (short time) Full Date/TIme (long time)
Font:
Color:
Alignment:

Cell Merging

FlexSheet supports merging the selected cells into one by invoking the mergeRange method.

If the selected cells contain merged cell, the mergeRange method will un-merge the merged cell. Otherwise, it will merge the selected cells into one cell.

FlexSheet allows merging of cells that contain any data. This is different from FlexGrid, which supports content-driven cell merging.

<wj-flex-sheet #mergeCellSheet (initialized)="mergeCellSheetInit(mergeCellSheet)"> <wj-sheet name="Unbound" [rowCount]="20" [columnCount]></wj-sheet> </wj-flex-sheet> <button type="button" class="btn btn-default" (click)="mergeCells()">{​{mergeState.isMergedCell ? 'UnMerge' : 'Merge'}​}</button>
// references FlexSheet named 'mergeCellSheet' in the view @ViewChild('mergeCellSheet') mergeCellSheet: wjcGridSheet.FlexSheet; mergeCellSheetInit(mergeCellSheet: wjcGridSheet.FlexSheet) { var self = this; if (mergeCellSheet) { mergeCellSheet.selectionChanged.addHandler(function () { self.mergeState = mergeCellSheet.getSelectionFormatState(); }); } } mergeCells() { if (this.mergeCellSheet) { this.mergeCellSheet.mergeRange(); this.mergeState = this.mergeCellSheet.getSelectionFormatState(); } }

Result (live):

Drag & Drop

FlexSheet supports dragging and dropping the columns or rows into other columns or rows.

FlexSheet not only copies or moves the data of the cells, but also copies or moves the style of the cells.

When you drag and drop columns/rows without pressing any key, it will move the selected columns or rows into the target columns or rows.

When you drag and drop columns/rows with 'Ctrl' key pressed, it will copy the selected columns or rows into the target columns or rows.

When you drag and drop columns/rows with 'Shift' key pressed, it will change the position of the selected columns or rows with the target columns or rows.

<wj-flex-sheet #dragDropSheet (initialized)="dragDropSheetInit(dragDropSheet)"> <wj-sheet name="Unbound" [rowCount]="12" [columnCount]="8"></wj-sheet> </wj-flex-sheet>

Result (live):

Frozen Cells

FlexSheet allows you to freeze the rows and columns of the selected cell by the freezeAtCursor method.

<wj-flex-sheet #frozenSheet (initialized)="frozenSheetInit(frozenSheet)"> <wj-sheet name="Unbound" [rowCount]="20" [columnCount]="8"></wj-sheet> </wj-flex-sheet> <button type="button" class="btn btn-default" (click)="freezeCells()">{​{isFrozen ? 'UnFreeze' : 'Freeze'}​}</button>
// references FlexSheet named 'frozenSheet' in the view @ViewChild('frozenSheet') frozenSheet: wjcGridSheet.FlexSheet; frozenSheetInit(frozenSheet: wjcGridSheet.FlexSheet) { var self = this; if (frozenSheet) { frozenSheet.selectedSheetChanged.addHandler(() => { if (frozenSheet.frozenColumns > 0 || frozenSheet.frozenRows > 0) { self.isFrozen = true; } else { self.isFrozen = false; } }); } } freezeCells() { if (this.frozenSheet) { this.frozenSheet.freezeAtCursor(); if (this.frozenSheet.frozenColumns > 0 || this.frozenSheet.frozenRows > 0) { this.isFrozen = true; } else { this.isFrozen = false; } } }

Result (live):

Undo / Redo

The FlexSheet control enables you to undo/redo following operations:

  1. Editing cells
  2. Resizing row/column
  3. Adding/Removing rows/columns
  4. Changing cell style
  5. Merging cells
  6. Sorting
  7. Dragging and Dropping rows/columns
<wj-flex-sheet #undoSheet (initialized)="undoSheetInit(undoSheet)"> <wj-sheet name="Unbound" [rowCount]="20" [columnCount]="8"></wj-sheet> </wj-flex-sheet> <button type="button" class="btn btn-default" [disabled]="!(undoStack && undoStack.canUndo)" (click)="undo()">Undo</button> <button type="button" class="btn btn-default" [disabled]="!(undoStack && undoStack.canRedo)" (click)="redo()">Redo</button>
// references FlexSheet named 'undoSheet' in the view @ViewChild('undoSheet') undoSheet: wjcGridSheet.FlexSheet; // Excutes undo command. undo () { this.undoSheet.undo(); } // Excutes redo command. redo () { this.undoSheet.redo(); }

Result (live):

Formulas

The FlexSheet control has a built-in calculation engine that supports over 80 functions just like Microsoft Excel. (See complete list.)

<wj-flex-sheet #formulaSheet (initialized)="formulaSheetInit(formulaSheet)"> <wj-sheet name="Expence Report" [rowCount]="14" [columnCount]="6"></wj-sheet> </wj-flex-sheet> <div><b>Cell Content: </b>{​{currentCellData}​}</div>

Result (live):

Cell Content: {{currentCellData}}

Custom Function

Although the functions provided in FlexSheet should cover a vast majority of use scenarios, still there may be some cases where users may need additional functions.

FlexSheet provides two methods that allow you to add your own custom functions: addCustomFunction and unknownFunction.

The addCustomFunction method adds a custom function to the list of built-in functions.

The addCustomFunction method is usually the best way to add custom functions to the FlexSheet calculation engine. However, there are scenarios where the function names are variable or unknown ahead of time. For example, named ranges or value dictionaries.

In these situations, you can use the unknownFunction event to look up the value of a function dynamically. When the FlexSheet detects an unknown function name, it raises the unknownFunction event and provides parameters that contain the function name and parameters. The event handler then calculates the result and returns the value.

<wj-flex-sheet #customFuncSheet (initialized)="customFuncSheetInit(customFuncSheet)"> <wj-sheet name="Custom Function" [rowCount]="25" [columnCount]="6"></wj-sheet> </wj-flex-sheet>
// references FlexSheet named 'customFuncSheet' in the view @ViewChild('customFuncSheet') customFuncSheet: wjcGridSheet.FlexSheet; customFuncSheetInit(customFuncSheet: wjcGridSheet.FlexSheet) { var self = this; if (customFuncSheet) { customFuncSheet.addCustomFunction('customSumProduct', (range1, range2) => { var flexSheet = customFuncSheet, result = 0, val1, val2; if (range1.rowSpan === range2.rowSpan && range1.columnSpan === range2.columnSpan) { for (var rowIndex = 0; rowIndex < range1.rowSpan; rowIndex++) { for (var columnIndex = 0; columnIndex < range1.columnSpan; columnIndex++) { val1 = +flexSheet.getCellValue(range1.topRow + rowIndex, range1.leftCol + columnIndex, false); val2 = +flexSheet.getCellValue(range2.topRow + rowIndex, range2.leftCol + columnIndex, false); result += val1 * val2; } } } return result; }, 'Custom SumProduct Function', 2, 2); customFuncSheet.unknownFunction.addHandler((sender: any, e: wjcGridSheet.UnknownFunctionEventArgs) => { var result = ''; if (e.params) { for (var i = 0; i < e.params.length; i++) { result += e.params[i]; } } e.value = result; }); } }

Result (live):

Excel I/O

FlexSheet supports saving/loading xlsx file by using the client 'save' and 'load' method.

<wj-flex-sheet #excelIOSheet (initialized)="excelIOSheetInit(excelIOSheet)"> <wj-sheet name="Country"></wj-sheet> <wj-sheet name="Unbound" [rowCount]="20" [columnCount]="8"></wj-sheet> </wj-flex-sheet> <div class="form-inline well well-lg"> <input type="file" class="form-control" id="importFile" accept="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" /> <button class="btn btn-default" (click)="load()">Load</button> </div> <div class="form-inline well well-lg"> File Name: <input type="text" class="form-control" [(ngModel)]="fileName" /> <button class="btn btn-default" (click)="save()">Save</button> </div>
// references FlexSheet named 'excelIOSheet' in the view @ViewChild('excelIOSheet') excelIOSheet: wjcGridSheet.FlexSheet; // Load xlsx file to FlexSheet. load () { var flexSheet = this.excelIOSheet, fileInput = <HTMLInputElement>document.getElementById('importFile'); if (flexSheet && fileInput.files[0]) { flexSheet.load(fileInput.files[0]); } } // Save FlexSheet to xlsx file. save () { var flexSheet = this.excelIOSheet, fileName; if (flexSheet) { if (!!this.fileName) { fileName = this.fileName; } else { fileName = 'FlexSheet.xlsx'; } flexSheet.save(fileName); } }

Result (live):

File Name: