From f0ebe3d309c1c6480e8c7aab35513cecb3956c92 Mon Sep 17 00:00:00 2001 From: Robert Lucas <100799838+Robert-M-Lucas@users.noreply.github.com> Date: Sat, 27 Apr 2024 17:05:34 +0100 Subject: [PATCH 01/10] Added JM's changes --- src/pages/transactions/TransactionPage.tsx | 86 +++++++++++++++ src/pages/transactions/transactionsTable.css | 71 ++++++++++++ .../transactions/transactionsTable.css.map | 1 + src/pages/transactions/transactionsTable.scss | 69 ++++++++++++ src/router.tsx | 104 +----------------- src/utils/transaction_utils.ts | 4 +- 6 files changed, 232 insertions(+), 103 deletions(-) create mode 100644 src/pages/transactions/TransactionPage.tsx create mode 100644 src/pages/transactions/transactionsTable.css create mode 100644 src/pages/transactions/transactionsTable.css.map create mode 100644 src/pages/transactions/transactionsTable.scss diff --git a/src/pages/transactions/TransactionPage.tsx b/src/pages/transactions/TransactionPage.tsx new file mode 100644 index 0000000..be50316 --- /dev/null +++ b/src/pages/transactions/TransactionPage.tsx @@ -0,0 +1,86 @@ +import { useEffect, useState } from "react"; + +// CSS +import "./transactionsTable.scss"; +import {getTransactionsPage} from "../../utils/transaction_utils.ts"; +import {auth} from "../../utils/firebase.ts"; +import {getTransactions, Transaction} from "../../utils/transaction.ts"; + +export function TransactionPage() { + const [transactions, setTransactions] = useState([]); + const [currentPage, setCurrentPage] = useState(1); + const itemsPerPage = 10; + + console.log("A"); + + // adjusted dylan.s code to use getTransactionPage instead of getTransactions + useEffect(() => { + if (!auth.currentUser) return; + + getTransactionsPage(auth.currentUser, itemsPerPage, currentPage) + .then((pageTransactions) => { + console.log("Fetched transactions:"); + console.log(pageTransactions); + setTransactions(pageTransactions); + }); + }, [currentPage]); + + console.log("B"); + + if (transactions.length === 0) { + return
Fetching transactions
; + } + + console.log("C"); + + return ( +
+ + + + + + + + + + + + + {transactions.map((transaction) => ( + + // maps every transaction as a row in the table + ))} + +
NameCategoryEmojiDateAmountNotes
+
+ {/* conditional previous page button, only displayed if page number > 1 */} + {currentPage > 1 && ( + + )} + {/* spacers to push buttons to the edges */} +
+ Page {currentPage} +
+ +
+
+ ); +} + +function TransactionItem({data}: { data: Transaction }) { + return ( + + {data.name} + {data.category} + {data.emoji} + {data.dateTime} + {data.amount} + {data.notes} + + ); +} \ No newline at end of file diff --git a/src/pages/transactions/transactionsTable.css b/src/pages/transactions/transactionsTable.css new file mode 100644 index 0000000..c974e3c --- /dev/null +++ b/src/pages/transactions/transactionsTable.css @@ -0,0 +1,71 @@ +/* page background */ +body { + background-color: #e6f2ff; +} + +/* table */ +table { + width: 100%; + border-collapse: separate; + border-spacing: 0; + border-radius: 10px; /* rounded borders */ + background-color: #ffffff; + overflow: hidden; /* hide overflow from rounded borders */ +} + +th { + padding: 12px 15px; + background-color: #f2f2f2; + border-bottom: 1px solid transparent; /* transparent border for rows */ + border-top-left-radius: 10px; /* rounded top-left corner */ + border-top-right-radius: 10px; /* rounded top-right corner */ +} + +td { + padding: 12px 15px; + text-align: left; + border-bottom: 1px solid transparent; /* transparent border for rows */ +} + +/* highlight on hover for table rows */ +tbody tr:hover { + background-color: #f5f5f5; +} + +/* Pagination */ +.pagination { + margin-top: 20px; + display: flex; + align-items: center; /* align items vertically */ +} + +.pagination button { + padding: 8px 12px; + border: 1px solid #007bff; + background-color: #007bff; + color: white; + cursor: pointer; + border-radius: 5px; +} + +.pagination .spacer { + flex-grow: 1; /* grow to fill available space */ +} + +.page-counter { + margin: 0 10px; /* add margin to separate from buttons */ +} + +/* hover effect for buttons */ +.pagination button:hover { + background-color: #0056b3; +} + +/* disabled state for buttons */ +.pagination button:disabled { + background-color: #cccccc; + border-color: #cccccc; + cursor: not-allowed; +} + +/*# sourceMappingURL=transactionsTable.css.map */ diff --git a/src/pages/transactions/transactionsTable.css.map b/src/pages/transactions/transactionsTable.css.map new file mode 100644 index 0000000..f8787c1 --- /dev/null +++ b/src/pages/transactions/transactionsTable.css.map @@ -0,0 +1 @@ +{"version":3,"sourceRoot":"","sources":["transactionsTable.scss"],"names":[],"mappings":"AAAA;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA;EACA;EACA;EACA;;;AAGF;EACE;EACA;EACA;EACA;EACA;;;AAGF;EACE;EACA;EACA;;;AAGF;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA;;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;;;AAGF;EACE;;;AAGF;EACE;;;AAGF;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA","file":"transactionsTable.css"} \ No newline at end of file diff --git a/src/pages/transactions/transactionsTable.scss b/src/pages/transactions/transactionsTable.scss new file mode 100644 index 0000000..9a8704f --- /dev/null +++ b/src/pages/transactions/transactionsTable.scss @@ -0,0 +1,69 @@ +/* page background */ +body { + background-color: #e6f2ff; +} + +/* table */ +table { + width: 100%; + border-collapse: separate; + border-spacing: 0; + border-radius: 10px; /* rounded borders */ + background-color: #ffffff; + overflow: hidden; /* hide overflow from rounded borders */ +} + +th { + padding: 12px 15px; + background-color: #f2f2f2; + border-bottom: 1px solid transparent; /* transparent border for rows */ + border-top-left-radius: 10px; /* rounded top-left corner */ + border-top-right-radius: 10px; /* rounded top-right corner */ +} + +td { + padding: 12px 15px; + text-align: left; + border-bottom: 1px solid transparent; /* transparent border for rows */ +} + +/* highlight on hover for table rows */ +tbody tr:hover { + background-color: #f5f5f5; +} + +/* Pagination */ +.pagination { + margin-top: 20px; + display: flex; + align-items: center; /* align items vertically */ +} + +.pagination button { + padding: 8px 12px; + border: 1px solid #007bff; + background-color: #007bff; + color: white; + cursor: pointer; + border-radius: 5px; +} + +.pagination .spacer { + flex-grow: 1; /* grow to fill available space */ +} + +.page-counter { + margin: 0 10px; /* add margin to separate from buttons */ +} + +/* hover effect for buttons */ +.pagination button:hover { + background-color: #0056b3; +} + +/* disabled state for buttons */ +.pagination button:disabled { + background-color: #cccccc; + border-color: #cccccc; + cursor: not-allowed; +} \ No newline at end of file diff --git a/src/router.tsx b/src/router.tsx index 283b71b..9b24d6f 100644 --- a/src/router.tsx +++ b/src/router.tsx @@ -7,93 +7,10 @@ import {SampleSidebarHeader} from "./pages/samples/sidebar_header/SampleSidebarH import {SampleModal} from "./pages/samples/modal/SampleModal.tsx"; import GraphDashboard from "./pages/dashboard/DashboardPage.tsx"; import {TestFirestorePage} from "./pages/test firestore/TestFirestore.tsx"; +import {TransactionPage} from "./pages/transactions/TransactionPage.tsx"; // ? Routing - see https://reactrouter.com/en/main -// const tileset_defult: Array<{ text: string; rows: number; cols: number }> = [ -// { text: "Tile 1", cols: 1, rows: 1 }, -// { text: "Tile 2", cols: 1, rows: 1 }, -// { text: "Tile 3", cols: 2, rows: 2 }, -// { text: "Tile 4", cols: 2, rows: 2 }, -// { text: "Tile 5", cols: 1, rows: 1 }, -// { text: "Tile 6", cols: 1, rows: 1 }, -// { text: "Tile 7", cols: 1, rows: 1 }, -// { text: "Tile 8", cols: 1, rows: 1 }, -// { text: "Tile 9", cols: 2, rows: 1 }, -// ]; - -const tileset_column: Array<{ text: string; rows: number; cols: number }> = [ - { text: "Tile 1", cols: 1, rows: 1 }, - { text: "Tile 2", cols: 1, rows: 2 }, - { text: "Tile 3", cols: 2, rows: 2 }, - { text: "Tile 4", cols: 2, rows: 2 }, - { text: "Tile 5", cols: 1, rows: 1 }, - { text: "Tile 6", cols: 1, rows: 1 }, - { text: "Tile 7", cols: 1, rows: 1 }, - { text: "Tile 8", cols: 1, rows: 1 }, - { text: "Tile 9", cols: 2, rows: 1 }, -]; - -const tileset_many: Array<{ text: string; rows: number; cols: number }> = [ - { text: "Tile 1", cols: 1, rows: 1 }, - { text: "Tile 2", cols: 2, rows: 1 }, - { text: "Tile 3", cols: 2, rows: 2 }, - { text: "Tile 4", cols: 2, rows: 2 }, - { text: "Tile 5", cols: 1, rows: 1 }, - { text: "Tile 6", cols: 1, rows: 1 }, - { text: "Tile 7", cols: 1, rows: 1 }, - { text: "Tile 8", cols: 1, rows: 1 }, - { text: "Tile 9", cols: 2, rows: 1 }, - { text: "Tile 11", cols: 1, rows: 1 }, - { text: "Tile 12", cols: 2, rows: 1 }, - { text: "Tile 13", cols: 2, rows: 2 }, - { text: "Tile 14", cols: 2, rows: 2 }, - { text: "Tile 15", cols: 1, rows: 1 }, - {text: "Tile 16", cols: 1, rows:1 }, - { text: "Tile 17", cols: 1, rows: 1 }, - { text: "Tile 18", cols: 1, rows: 1 }, - { text: "Tile 19", cols: 2, rows: 1 }, - { text: "Tile 21", cols: 1, rows: 1 }, - { text: "Tile 22", cols: 2, rows: 1 }, - { text: "Tile 23", cols: 2, rows: 2 }, - { text: "Tile 24", cols: 2, rows: 2 }, - { text: "Tile 25", cols: 1, rows: 1 }, - { text: "Tile 26", cols: 1, rows: 1 }, - { text: "Tile 27", cols: 1, rows: 1 }, - { text: "Tile 28", cols: 1, rows: 1 }, - { text: "Tile 29", cols: 2, rows: 1 }, - { text: "Tile 31", cols: 1, rows: 1 }, - { text: "Tile 32", cols: 2, rows: 1 }, - { text: "Tile 33", cols: 2, rows: 2 }, - { text: "Tile 34", cols: 2, rows: 2 }, - { text: "Tile 35", cols: 1, rows: 1 }, - { text: "Tile 36", cols: 1, rows: 1 }, - { text: "Tile 37", cols: 1, rows: 1 }, - { text: "Tile 38", cols: 1, rows: 1 }, - { text: "Tile 39", cols: 2, rows: 1 }, -]; - -const tileset_weird: Array<{ text: string; rows: number; cols: number }> = [ - { text: "Tile 1", cols: 1, rows: 3 }, - { text: "Tile 2", cols: 2, rows: 3 }, - { text: "Tile 3", cols: 3, rows: 2 }, - { text: "Tile 4", cols: 3, rows: 2 }, - { text: "Tile 5", cols: 3, rows: 1 }, - { text: "Tile 6", cols: 1, rows: 1 }, - { text: "Tile 7", cols: 1, rows: 1 }, - { text: "Tile 8", cols: 1, rows: 1 }, - { text: "Tile 9", cols: 2, rows: 1 }, - { text: "Tile 11", cols: 1, rows: 1 }, - { text: "Tile 12", cols: 2, rows: 1 }, - { text: "Tile 13", cols: 2, rows: 2 }, - { text: "Tile 14", cols: 2, rows: 2 }, - { text: "Tile 15", cols: 1, rows: 1 }, - { text: "Tile 16", cols: 1, rows: 1 }, - { text: "Tile 17", cols: 3, rows: 3 }, - { text: "Tile 18", cols: 1, rows: 1 }, - { text: "Tile 19", cols: 2, rows: 1 }, -]; - export const router = createBrowserRouter([ { path: "/", @@ -102,15 +19,12 @@ export const router = createBrowserRouter([ }, { path: "/dash", - element: , + element: , }, { path: "/transactions", - element: , + element: , }, - // --> - - { path: "/user-test", element: , @@ -120,18 +34,6 @@ export const router = createBrowserRouter([ path: "/graphs", element: , }, - { - path: "/dash-2", - element: , - }, - { - path: "/dash-3", - element: , - }, - { - path: "/dash-4", - element: , - }, { path: "/sample_sidebar", element: , diff --git a/src/utils/transaction_utils.ts b/src/utils/transaction_utils.ts index 53dd236..47dc12f 100644 --- a/src/utils/transaction_utils.ts +++ b/src/utils/transaction_utils.ts @@ -1,6 +1,6 @@ import {User} from "firebase/auth"; import {getTransactions, getTransactionsFilterOrderBy, Transaction} from "./transaction.ts"; -import {limit, startAfter, where} from "firebase/firestore"; +import {limit, orderBy, startAfter, where} from "firebase/firestore"; export async function getCurrentBalance(user: User): Promise { const transactions = await getTransactions(user); @@ -28,5 +28,5 @@ export async function getLastDayTransaction(user: User): Promise // Returns `pageSize` transactions for the given `user` with the `docName` attribute set export async function getTransactionsPage(user: User, pageSize: number, page: number): Promise { - return await getTransactionsFilterOrderBy(user, startAfter(page * pageSize), limit(pageSize)); + return await getTransactionsFilterOrderBy(user, orderBy("dateTime", "desc"), startAfter(page * pageSize), limit(pageSize)); } \ No newline at end of file From 5e7d5963bae2e07aa5b8c3951c319c3026266754 Mon Sep 17 00:00:00 2001 From: Robert Lucas <100799838+Robert-M-Lucas@users.noreply.github.com> Date: Sat, 27 Apr 2024 17:39:00 +0100 Subject: [PATCH 02/10] Got transaction page working. Removed getTransactionsPage function as it was fundamentally broken --- src/pages/transactions/TransactionPage.tsx | 127 ++++++++++++++---- src/pages/transactions/transactionsTable.css | 2 - .../transactions/transactionsTable.css.map | 2 +- src/pages/transactions/transactionsTable.scss | 4 +- src/utils/transaction_utils.ts | 7 +- 5 files changed, 106 insertions(+), 36 deletions(-) diff --git a/src/pages/transactions/TransactionPage.tsx b/src/pages/transactions/TransactionPage.tsx index be50316..b554d91 100644 --- a/src/pages/transactions/TransactionPage.tsx +++ b/src/pages/transactions/TransactionPage.tsx @@ -1,39 +1,100 @@ -import { useEffect, useState } from "react"; +import {useState} from "react"; +import {auth} from "../../utils/firebase.ts"; +import {getTransactionsFilterOrderBy, Transaction} from "../../utils/transaction.ts"; +import {FullscreenCenter} from "../../components/FullscreenCenter.tsx"; +import {User} from "firebase/auth"; +import {Header} from "../../components/Header.tsx"; +import {signInWithGoogle} from "../../utils/authentication.ts"; +import {limit, orderBy, startAfter} from "firebase/firestore"; // CSS import "./transactionsTable.scss"; -import {getTransactionsPage} from "../../utils/transaction_utils.ts"; -import {auth} from "../../utils/firebase.ts"; -import {getTransactions, Transaction} from "../../utils/transaction.ts"; export function TransactionPage() { - const [transactions, setTransactions] = useState([]); - const [currentPage, setCurrentPage] = useState(1); + const [transactions, setTransactions] = useState(null); + const [currentPage, setCurrentPage] = useState(0); + const [authResolved, setAuthResolved] = useState(false); + const [update, setUpdate] = useState(0); + const [pageStarts, setPageStarts] = useState([Infinity]); const itemsPerPage = 10; - console.log("A"); - // adjusted dylan.s code to use getTransactionPage instead of getTransactions - useEffect(() => { - if (!auth.currentUser) return; + if (!authResolved) { + auth.authStateReady().then(() => setAuthResolved(true)); + return <> + +
+

Waiting for Auth

+
+
+ ; + } - getTransactionsPage(auth.currentUser, itemsPerPage, currentPage) + if (auth.currentUser === null) { + auth.onAuthStateChanged((new_user: User | null) => { + if (new_user !== null) { + setUpdate(update + 1); + } + }); + return <> +
+ +
+

Not Logged In

+ +
+
+ ; + } + + if (!transactions) { + getTransactionsFilterOrderBy(auth.currentUser, orderBy("dateTime", "desc"), limit(itemsPerPage), startAfter(pageStarts[pageStarts.length - 1])) .then((pageTransactions) => { - console.log("Fetched transactions:"); - console.log(pageTransactions); setTransactions(pageTransactions); }); - }, [currentPage]); - - console.log("B"); - if (transactions.length === 0) { - return
Fetching transactions
; + return <> +
+ + + + + + + + + + + + + + + + + + + + + +
NameCategoryEmojiDateAmountNotes
Fetching...Fetching...Fetching...Fetching...Fetching...Fetching...
+ ; } - console.log("C"); - return ( + // adjusted dylan.s code to use getTransactionPage instead of getTransactions + // useEffect(() => { + // getTransactionsPage(auth.currentUser, itemsPerPage, currentPage) + // .then((pageTransactions) => { + // console.log("Fetched transactions:"); + // console.log(pageTransactions); + // setTransactions(pageTransactions); + // }); + // }, [currentPage]); + + return <> +
@@ -53,10 +114,18 @@ export function TransactionPage() { ))}
-
+ { transactions.length == 0 && +

[No More Data]

+ } +
{/* conditional previous page button, only displayed if page number > 1 */} - {currentPage > 1 && ( - )} @@ -64,12 +133,20 @@ export function TransactionPage() {
Page {currentPage}
- + }
+
- ); + ; } function TransactionItem({data}: { data: Transaction }) { diff --git a/src/pages/transactions/transactionsTable.css b/src/pages/transactions/transactionsTable.css index c974e3c..2eb9216 100644 --- a/src/pages/transactions/transactionsTable.css +++ b/src/pages/transactions/transactionsTable.css @@ -17,8 +17,6 @@ th { padding: 12px 15px; background-color: #f2f2f2; border-bottom: 1px solid transparent; /* transparent border for rows */ - border-top-left-radius: 10px; /* rounded top-left corner */ - border-top-right-radius: 10px; /* rounded top-right corner */ } td { diff --git a/src/pages/transactions/transactionsTable.css.map b/src/pages/transactions/transactionsTable.css.map index f8787c1..30fecee 100644 --- a/src/pages/transactions/transactionsTable.css.map +++ b/src/pages/transactions/transactionsTable.css.map @@ -1 +1 @@ -{"version":3,"sourceRoot":"","sources":["transactionsTable.scss"],"names":[],"mappings":"AAAA;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA;EACA;EACA;EACA;;;AAGF;EACE;EACA;EACA;EACA;EACA;;;AAGF;EACE;EACA;EACA;;;AAGF;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA;;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;;;AAGF;EACE;;;AAGF;EACE;;;AAGF;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA","file":"transactionsTable.css"} \ No newline at end of file +{"version":3,"sourceRoot":"","sources":["transactionsTable.scss"],"names":[],"mappings":"AAAA;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA;EACA;EACA;EACA;;;AAGF;EACE;EACA;EACA;;;AAKF;EACE;EACA;EACA;;;AAGF;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA;;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;;;AAGF;EACE;;;AAGF;EACE;;;AAGF;AACA;EACE;;;AAGF;AACA;EACE;EACA;EACA","file":"transactionsTable.css"} \ No newline at end of file diff --git a/src/pages/transactions/transactionsTable.scss b/src/pages/transactions/transactionsTable.scss index 9a8704f..b8e0aeb 100644 --- a/src/pages/transactions/transactionsTable.scss +++ b/src/pages/transactions/transactionsTable.scss @@ -17,8 +17,8 @@ th { padding: 12px 15px; background-color: #f2f2f2; border-bottom: 1px solid transparent; /* transparent border for rows */ - border-top-left-radius: 10px; /* rounded top-left corner */ - border-top-right-radius: 10px; /* rounded top-right corner */ + //border-top-left-radius: 10px; /* rounded top-left corner */ + //border-top-right-radius: 10px; /* rounded top-right corner */ } td { diff --git a/src/utils/transaction_utils.ts b/src/utils/transaction_utils.ts index 47dc12f..69fa01d 100644 --- a/src/utils/transaction_utils.ts +++ b/src/utils/transaction_utils.ts @@ -1,6 +1,6 @@ import {User} from "firebase/auth"; import {getTransactions, getTransactionsFilterOrderBy, Transaction} from "./transaction.ts"; -import {limit, orderBy, startAfter, where} from "firebase/firestore"; +import {where} from "firebase/firestore"; export async function getCurrentBalance(user: User): Promise { const transactions = await getTransactions(user); @@ -24,9 +24,4 @@ const DAY_MILLIS = 8.64e+7; export async function getLastDayTransaction(user: User): Promise { return await getTransactionsFilterOrderBy(user, where("dateTime", ">", Date.now() - DAY_MILLIS)); -} - -// Returns `pageSize` transactions for the given `user` with the `docName` attribute set -export async function getTransactionsPage(user: User, pageSize: number, page: number): Promise { - return await getTransactionsFilterOrderBy(user, orderBy("dateTime", "desc"), startAfter(page * pageSize), limit(pageSize)); } \ No newline at end of file From e219ac9e5d15195864020aaf8601519abbf79d14 Mon Sep 17 00:00:00 2001 From: Robert Lucas <100799838+Robert-M-Lucas@users.noreply.github.com> Date: Sat, 27 Apr 2024 18:10:45 +0100 Subject: [PATCH 03/10] Made tweaks to transaction page --- package-lock.json | 17 ++++++++++++++++- package.json | 4 +++- src/pages/transactions/TransactionPage.tsx | 18 ++++++++++++++---- 3 files changed, 33 insertions(+), 6 deletions(-) diff --git a/package-lock.json b/package-lock.json index aa0a8a9..c65656a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,6 +9,7 @@ "version": "0.0.0", "dependencies": { "@types/papaparse": "^5.3.14", + "@types/strftime": "^0.9.8", "bootstrap": "^5.3.3", "firebase": "^10.11.0", "lodash": "^4.17.21", @@ -19,7 +20,8 @@ "react-router-dom": "^6.23.0", "react-tiles-dnd": "^0.1.2", "recharts": "^2.12.5", - "sass": "^1.72.0" + "sass": "^1.72.0", + "strftime": "^0.10.2" }, "devDependencies": { "@faker-js/faker": "^8.4.1", @@ -2584,6 +2586,11 @@ "@types/send": "*" } }, + "node_modules/@types/strftime": { + "version": "0.9.8", + "resolved": "https://registry.npmjs.org/@types/strftime/-/strftime-0.9.8.tgz", + "integrity": "sha512-QIvDlGAKyF3YJbT3QZnfC+RIvV5noyDbi+ZJ5rkaSRqxCGrYJefgXm3leZAjtoQOutZe1hCXbAg+p89/Vj4HlQ==" + }, "node_modules/@types/tough-cookie": { "version": "4.0.5", "resolved": "https://registry.npmjs.org/@types/tough-cookie/-/tough-cookie-4.0.5.tgz", @@ -6705,6 +6712,14 @@ "dev": true, "optional": true }, + "node_modules/strftime": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/strftime/-/strftime-0.10.2.tgz", + "integrity": "sha512-Y6IZaTVM80chcMe7j65Gl/0nmlNdtt+KWPle5YeCAjmsBfw+id2qdaJ5MDrxUq+OmHKab+jHe7mUjU/aNMSZZg==", + "engines": { + "node": ">=0.2.0" + } + }, "node_modules/string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", diff --git a/package.json b/package.json index caaae65..4dce34f 100644 --- a/package.json +++ b/package.json @@ -13,6 +13,7 @@ }, "dependencies": { "@types/papaparse": "^5.3.14", + "@types/strftime": "^0.9.8", "bootstrap": "^5.3.3", "firebase": "^10.11.0", "lodash": "^4.17.21", @@ -23,7 +24,8 @@ "react-router-dom": "^6.23.0", "react-tiles-dnd": "^0.1.2", "recharts": "^2.12.5", - "sass": "^1.72.0" + "sass": "^1.72.0", + "strftime": "^0.10.2" }, "devDependencies": { "@faker-js/faker": "^8.4.1", diff --git a/src/pages/transactions/TransactionPage.tsx b/src/pages/transactions/TransactionPage.tsx index b554d91..52618a2 100644 --- a/src/pages/transactions/TransactionPage.tsx +++ b/src/pages/transactions/TransactionPage.tsx @@ -6,6 +6,7 @@ import {User} from "firebase/auth"; import {Header} from "../../components/Header.tsx"; import {signInWithGoogle} from "../../utils/authentication.ts"; import {limit, orderBy, startAfter} from "firebase/firestore"; +import strftime from "strftime"; // CSS import "./transactionsTable.scss"; @@ -16,7 +17,7 @@ export function TransactionPage() { const [authResolved, setAuthResolved] = useState(false); const [update, setUpdate] = useState(0); const [pageStarts, setPageStarts] = useState([Infinity]); - const itemsPerPage = 10; + const itemsPerPage = 14; if (!authResolved) { @@ -131,7 +132,7 @@ export function TransactionPage() { )} {/* spacers to push buttons to the edges */}
- Page {currentPage} + Page {currentPage + 1}
{transactions.length === itemsPerPage && +
+ + ; + } + if (!transactionPoints) { return <> +

Fetching

@@ -68,18 +80,18 @@ export default function Dashboard() { ; } - const transactionTiles = [ - {d: TileElement.newGraph(transactionPoints.raw), cols:5, rows:2}, - {d: TileElement.newGraph(transactionPoints.in), cols:5, rows:2}, - {d: TileElement.newGraph(transactionPoints.out), cols:5, rows:2}, - {d: TileElement.newTSX(test), cols:1, rows:1} + const transactionTiles: TileElement[] = [ + TileElement.newTSX(() => totalTile(transactions), 1, 1, columns), + TileElement.newGraph(transactionPoints.raw, 3, 2, columns), + TileElement.newGraph(transactionPoints.in, 3, 2, columns), + TileElement.newGraph(transactionPoints.out, 3, 2, columns), ]; - const renderFirebase: RenderTileFunction = ({ data, isDragging }) => ( + const renderTile: RenderTileFunction = ({ data, isDragging }) => (
- {data.d.isGraph() ? : data.d.forceGetTSX()()} + {data.isGraph() ? : data.forceGetTSX()()}
); @@ -87,23 +99,23 @@ export default function Dashboard() { return (
-
- - + {/*
*/} + {/* */} + {/* */} - - -
+ {/* */} + {/* */} + {/*
*/}
- - {balance} + {/**/} + {/*{balance}*/} diff --git a/src/pages/dashboard/TileUtils.ts b/src/pages/dashboard/TileUtils.ts index 792d6ad..3fbee2b 100644 --- a/src/pages/dashboard/TileUtils.ts +++ b/src/pages/dashboard/TileUtils.ts @@ -1,22 +1,27 @@ import {ReactNode} from "react"; import {transactionPoint} from "./GraphUtils.ts"; +import {min} from "lodash"; type tsxContents = ReactNode; export class TileElement { private readonly graph?: transactionPoint[]; private readonly TSX?: () => tsxContents; + public readonly cols: number; + public readonly rows: number; - constructor(graph: transactionPoint[] | undefined, TSX: (() => tsxContents) | undefined) { + private constructor(graph: transactionPoint[] | undefined, TSX: (() => tsxContents) | undefined, cols: number, rows: number) { this.graph = graph; this.TSX = TSX; + this.cols = cols; + this.rows = rows; } - static newGraph(graph: transactionPoint[]): TileElement { - return new TileElement(graph, undefined); + static newGraph(graph: transactionPoint[], cols: number, rows: number, maxCol: number): TileElement { + return new TileElement(graph, undefined, min([cols, maxCol])!, rows); } - static newTSX(TSX: () => tsxContents): TileElement { - return new TileElement(undefined, TSX); + static newTSX(TSX: () => tsxContents, cols: number, rows: number, maxCol: number): TileElement { + return new TileElement(undefined, TSX, min([cols, maxCol])!, rows); } isGraph(): boolean { @@ -29,4 +34,10 @@ export class TileElement { forceGetTSX(): () => tsxContents { return this.TSX!; } +} + +export type tileData = { d: TileElement; rows: number; cols: number }; + +export function getTileSize (tile: TileElement) { + return {colSpan: tile.cols, rowSpan: tile.rows}; } \ No newline at end of file diff --git a/src/pages/dashboard/Tiles.tsx b/src/pages/dashboard/Tiles.tsx new file mode 100644 index 0000000..0155cff --- /dev/null +++ b/src/pages/dashboard/Tiles.tsx @@ -0,0 +1,8 @@ +import {ReactNode} from "react"; +import { Transaction } from "../../utils/transaction"; + +export default function totalTile(transactions: Transaction[]): ReactNode { + return <> +

Total: {transactions.reduce((prev, curr): number => prev + curr.amount, 0).toFixed(2)}

+ ; +} \ No newline at end of file diff --git a/src/pages/dashboard/test.tsx b/src/pages/dashboard/test.tsx deleted file mode 100644 index 13cba03..0000000 --- a/src/pages/dashboard/test.tsx +++ /dev/null @@ -1,7 +0,0 @@ -export default function test() { - return ( - <> - Hi - - ); -} \ No newline at end of file diff --git a/src/router.tsx b/src/router.tsx index 9b24d6f..f5b2c09 100644 --- a/src/router.tsx +++ b/src/router.tsx @@ -26,7 +26,7 @@ export const router = createBrowserRouter([ element: , }, { - path: "/user-test", + path: "/user-tiles", element: , errorElement:<_404Page/> }, @@ -47,7 +47,7 @@ export const router = createBrowserRouter([ element: , }, { - path: "/test", + path: "/tiles", element: , }, ]); \ No newline at end of file From 6e5d46b35c529f4e80127d148f3e62a86a4386cb Mon Sep 17 00:00:00 2001 From: Robert Lucas <100799838+Robert-M-Lucas@users.noreply.github.com> Date: Sat, 27 Apr 2024 19:43:57 +0100 Subject: [PATCH 08/10] Added total tile --- src/pages/dashboard/DashboardPage.tsx | 2 +- src/pages/dashboard/Tiles.tsx | 36 ++++++++++++++++++++++++--- 2 files changed, 34 insertions(+), 4 deletions(-) diff --git a/src/pages/dashboard/DashboardPage.tsx b/src/pages/dashboard/DashboardPage.tsx index b07d7d8..7be0ea7 100644 --- a/src/pages/dashboard/DashboardPage.tsx +++ b/src/pages/dashboard/DashboardPage.tsx @@ -81,7 +81,7 @@ export default function Dashboard() { } const transactionTiles: TileElement[] = [ - TileElement.newTSX(() => totalTile(transactions), 1, 1, columns), + TileElement.newTSX(() => totalTile(transactions), 2, 1, columns), TileElement.newGraph(transactionPoints.raw, 3, 2, columns), TileElement.newGraph(transactionPoints.in, 3, 2, columns), TileElement.newGraph(transactionPoints.out, 3, 2, columns), diff --git a/src/pages/dashboard/Tiles.tsx b/src/pages/dashboard/Tiles.tsx index 0155cff..6bad195 100644 --- a/src/pages/dashboard/Tiles.tsx +++ b/src/pages/dashboard/Tiles.tsx @@ -1,8 +1,38 @@ import {ReactNode} from "react"; import { Transaction } from "../../utils/transaction"; +import {max, min} from "lodash"; export default function totalTile(transactions: Transaction[]): ReactNode { - return <> -

Total: {transactions.reduce((prev, curr): number => prev + curr.amount, 0).toFixed(2)}

- ; + const balance = transactions.reduce((prev, curr): number => prev + curr.amount, 0); + const income = transactions.reduce((prev, curr): number => prev + max([curr.amount, 0])!, 0); + const expenses = transactions.reduce((prev, curr): number => prev + min([curr.amount, 0])!, 0); + + return
    +
  • +
    +
    Balance:
    + {balance > 0 ? <> +
    £
    +
    {balance.toFixed(2)}
    + : <> +
    £
    +
    {balance.toFixed(2)}
    + } +
    +
  • +
  • +
    +
    Income:
    +
    £
    +
    {income.toFixed(2)}
    +
    +
  • +
  • +
    +
    Expenses:
    +
    £
    +
    {expenses.toFixed(2)}
    +
    +
  • +
; } \ No newline at end of file From f08ffbc080d53defe97d2339ab8f0f1a02ac845a Mon Sep 17 00:00:00 2001 From: Robert Lucas <100799838+Robert-M-Lucas@users.noreply.github.com> Date: Sat, 27 Apr 2024 21:00:02 +0100 Subject: [PATCH 09/10] Added final UserPrefs --- src/pages/test firestore/TestFirestore.tsx | 12 ++++-- src/utils/user_prefs.ts | 46 +++++++++++++++++++--- 2 files changed, 49 insertions(+), 9 deletions(-) diff --git a/src/pages/test firestore/TestFirestore.tsx b/src/pages/test firestore/TestFirestore.tsx index 2655ca2..899c941 100644 --- a/src/pages/test firestore/TestFirestore.tsx +++ b/src/pages/test firestore/TestFirestore.tsx @@ -15,6 +15,7 @@ import {Header} from "../../components/Header.tsx"; import { orderBy } from "firebase/firestore"; import {User} from "firebase/auth"; import {getUserPrefs, setUserPrefs, UserPrefs} from "../../utils/user_prefs.ts"; +import {round} from "lodash"; function writeSampleData() { if (auth.currentUser === null) { @@ -95,11 +96,14 @@ export function TestFirestorePage() {

UserPrefs

{ userPrefs ? <> -

Goal: {userPrefs.goal}

+

Goal: {userPrefs.getNeedsBudget()} | {userPrefs.getWantsBudget()} | {userPrefs.getSavingsBudget()}

:

Loading

} diff --git a/src/utils/user_prefs.ts b/src/utils/user_prefs.ts index 2aa65b5..24ebba6 100644 --- a/src/utils/user_prefs.ts +++ b/src/utils/user_prefs.ts @@ -1,17 +1,52 @@ import {collection, doc, DocumentSnapshot, getDoc, setDoc, SnapshotOptions} from "firebase/firestore"; import {User} from "firebase/auth"; import {db} from "./firebase.ts"; +import {round} from "lodash"; export class UserPrefs { - public goal: number; + private readonly needsBudget: number; + private readonly wantsBudget: number; - constructor(goal: number) { - this.goal = goal; + private constructor(needsBudget: number, wantsBudget: number) { + if (needsBudget > 1) { + needsBudget = 1; + } + + if (needsBudget + wantsBudget > 1) { + wantsBudget = 1 - needsBudget; + } + + this.needsBudget = round(needsBudget, 2); + this.wantsBudget = round(wantsBudget, 2); + } + + static newChecked(needsBudget: number, wantsBudget: number): UserPrefs | Error { + if (needsBudget > 1) { + return new Error("needsBudget > 1!"); + } + + if (needsBudget + wantsBudget > 1) { + return new Error("needsBudget + wantsBudget > 1!"); + } + + return new UserPrefs(needsBudget, wantsBudget); } static default(): UserPrefs { - return new UserPrefs(100); + return new UserPrefs(0.5, 0.3); + } + + getNeedsBudget(): number { + return this.needsBudget; + } + + getWantsBudget(): number { + return this.wantsBudget; + } + + getSavingsBudget(): number { + return round(1 - this.wantsBudget - this.needsBudget, 2); } // Utility method for creating `Transactions` @@ -20,7 +55,8 @@ export class UserPrefs { if (!data) { throw Error("No data returned for snapshot!"); } - return new UserPrefs(data.goal); + + return new UserPrefs(round(data.needsBudget, 2), round(data.wantsBudget, 2)); } toSendObject(): object { From 5c2834e0710abc04909db46d1b80a1686e259d5e Mon Sep 17 00:00:00 2001 From: Robert Lucas <100799838+Robert-M-Lucas@users.noreply.github.com> Date: Sat, 27 Apr 2024 21:13:22 +0100 Subject: [PATCH 10/10] Added basic goal tile --- src/pages/dashboard/DashboardPage.tsx | 23 +++++++------------ src/pages/dashboard/GoalsTile.tsx | 7 ++++++ .../dashboard/{Tiles.tsx => TotalTile.tsx} | 0 3 files changed, 15 insertions(+), 15 deletions(-) create mode 100644 src/pages/dashboard/GoalsTile.tsx rename src/pages/dashboard/{Tiles.tsx => TotalTile.tsx} (100%) diff --git a/src/pages/dashboard/DashboardPage.tsx b/src/pages/dashboard/DashboardPage.tsx index 7be0ea7..7004cbf 100644 --- a/src/pages/dashboard/DashboardPage.tsx +++ b/src/pages/dashboard/DashboardPage.tsx @@ -7,19 +7,22 @@ import {useEffect, useState} from "react"; import {Transaction, getTransactionsFilterOrderBy } from "../../utils/transaction.ts" import {auth} from "../../utils/firebase.ts"; import {orderBy} from "firebase/firestore"; -import { User } from "firebase/auth"; import {FullscreenCenter} from "../../components/FullscreenCenter.tsx"; import Graphs from "./Graphs.tsx" import {getTileSize, TileElement} from "./TileUtils.ts"; import {finalGraphData, readTransactions} from "./GraphUtils.ts"; import {signInWithGoogle} from "../../utils/authentication.ts"; -import totalTile from "./Tiles.tsx"; +import totalTile from "./TotalTile.tsx"; +import {getUserPrefs, UserPrefs} from "../../utils/user_prefs.ts"; +import {User} from "firebase/auth"; +import goalsTile from "./GoalsTile.tsx"; export default function Dashboard() { // const [balance, setBalance] = useState(0); const [transactionPoints, setPoints] = useState(null); const [transactions, setTransactions] = useState([]); const [authResolved, setAuthResolved] = useState(false); + const [userPrefs, setUserPrefs] = useState(null); // const [showCSVModal, setShowCSVModal] = useState(false); // const [showTransactionModal, setShowTransactionModal] = useState(false); const [update, setUpdate] = useState(0) @@ -37,6 +40,7 @@ export default function Dashboard() { useEffect(() => { if (auth.currentUser !== null) { fetchTransactions(auth.currentUser).then(); + getUserPrefs(auth.currentUser).then((prefs) => setUserPrefs(prefs)); } },[auth.currentUser]) @@ -69,7 +73,7 @@ export default function Dashboard() { ; } - if (!transactionPoints) { + if (!transactionPoints || !userPrefs) { return <>
@@ -82,6 +86,7 @@ export default function Dashboard() { const transactionTiles: TileElement[] = [ TileElement.newTSX(() => totalTile(transactions), 2, 1, columns), + TileElement.newTSX(() => (goalsTile(userPrefs)), 2, 1, columns), TileElement.newGraph(transactionPoints.raw, 3, 2, columns), TileElement.newGraph(transactionPoints.in, 3, 2, columns), TileElement.newGraph(transactionPoints.out, 3, 2, columns), @@ -99,19 +104,7 @@ export default function Dashboard() { return (
- {/*
*/} - {/* */} - {/* */} - - {/* */} - {/* */} - {/*
*/}
- {/**/} - {/*{balance}*/} Goal: {userPrefs.getNeedsBudget()} | {userPrefs.getWantsBudget()} | {userPrefs.getSavingsBudget()}

; +} \ No newline at end of file diff --git a/src/pages/dashboard/Tiles.tsx b/src/pages/dashboard/TotalTile.tsx similarity index 100% rename from src/pages/dashboard/Tiles.tsx rename to src/pages/dashboard/TotalTile.tsx