It is hosted live here.
The app does not save any data or calculations; when you refresh, everything you typed will be gone.
- The spreadsheet was created with vanilla javascript (no frameworks, although inspiration was drawn from React!)
- The
index.html
file, as usual, anchors the page with the basic links and set up. Theindex.js
script injects the important HTML snippets to make the whole spreadsheet interactive, as well as adding all event listeners. This file has a good many dependencies, to perform the various bits of fucntionality: setting up the formula bar, setting up the sheet, translating Excel-style cell addresses into more machine-useable column-row coordinates, doing the calculations and managing the storage. - The app draws a 100 X 100 grid (configurable, up to 702 colums), with an extra row and column for the heading labels. The headings are sticky, meaning they scroll with their corresponding columns or rows, but stay on the top and left of the page.
- The current cell's formula is copied up to the formula bar and can be edited
there as well. When TAB or ENTER or the refresh button
=
is pressed, the result is calculated. - When you click a cell, the formula you enter (or raw number if that is all you entered) gets stored in one plane of a 3-dimensional array. The array dimensions correspond to columns, rows and formulae or styling. Calculated values are effectively stored in the screen grid.
- The refresh button (
=
) recalculates all formulae on the sheet and reloads the values - The spreadsheet can handle basic arithmetic formulae with cell references,
e.g.
=(A1+B2)*C5
; it distinguishes formula from values with the=
sign. If any of the referenced cells change, the formula's result changes too. - Three basic range functions have been included:
SUM()
,COUNT()
andAVERAGE()
. The spreadsheet is not case sensitve on function names, nor on cell references. The functions are listed in theconfig.json
and are structured as objects there, the idea being that in future, extra keys and values can be added to the functions (as well as extra functions) and the config could hold information about how they are to be proceed, e.g. a function likePI()
won't take any arguments, whereas a function likeSIN()
would take a single number,SUM()
, etc. take a range of values and a complex functions like=DSUM()
take a list of arguments that include ranges. - Functions can be nested and operated on.
- Finally, the spreadsheet does basic cell-level formatting, just
Bold
,Italic
andUnderline
.
- The choice was: use table widget or a flex-grid and and a long list of inputs. In the end I went for the flex-grid; it's more modern and the user's movement around the enclosed input boxes is more slick. It was tricky to get the column and row headings sticky.
- I made lots of small functions spread over a number of JS modules. Not having
any frameworks means not having any packages either -- no
npm install
to be done. However, there is one exception: I used Vite as a deploy solution. - Tests were written for the address-to-coorinates converter. Although other test files are on the system, they are empty. It takes a lot of time to write tests! In the real world, with a Production environment, this isn't an option.
- The overall aim was to keep the app composable and extendable, allowing new functionality to be 'plugged in', rather than requiring new code builds.
To build and run the app locally on your machine, clone this repo and cd
into
the working directory. Then npm run build
will have Vite install the
dependencies and build the code. To run it on your local machine, use
npm run dev
. To run the few tests that have been written, use npm run test
.
- Make the column width and row height adjustment work. Store the column width and row height in an array that is used to build the sheet grid up.
- Reduce the memory requirment (currently 300MB+) to under 100MB. Do this by reducing the number of elements on the page (e.g. the cell input text box's border can be highlighted rather than having extra, separate div elements--- 100 x 100 x 2 = 20,000 extra elements)
- Create a separate IAM service, to allow authentication and role assignment for app authorisation. The service will return JWTs for auth on any app.
- Use the IAM service to store the spreadsheet arrays in a cloud-based database.
- Alllow selection of multiple cells, whole columns and rows. Show the selection range on a widget on nthe formula bar.
- Build a generic function processor that take data from the config to calculate built-in function results, rather than having hard-coded function calculators.
- Create test using Selenium and a headless browser to test the page layout:
sheet.test.js