From ba6a05484698e80bb57d4dc1f6020c401a78f59c Mon Sep 17 00:00:00 2001 From: Chris Lindelof Date: Tue, 20 Aug 2024 15:43:13 -0400 Subject: [PATCH] Add Redux. - Create ./store - Create authentication slice to manage authentication state globally. Add tailwindcss for styling Skeleton out the app logic to have authentication. Skeleton out the login component --- front-end/package-lock.json | 127 +++++++++++++++++- front-end/package.json | 8 ++ front-end/postcss.config.js | 6 + front-end/src/App.css | 38 ------ front-end/src/App.js | 34 ++--- front-end/src/App.test.js | 8 -- front-end/src/components/FleetManager.js | 5 + front-end/src/components/Login.js | 50 +++++++ front-end/src/index.css | 13 -- front-end/src/index.js | 8 +- front-end/src/logo.svg | 1 - front-end/src/main.css | 3 + .../src/stateSlicers/authenticationSlice.js | 29 ++++ front-end/src/store.js | 7 + front-end/tailwind.config.js | 9 ++ 15 files changed, 260 insertions(+), 86 deletions(-) create mode 100644 front-end/postcss.config.js delete mode 100644 front-end/src/App.css delete mode 100644 front-end/src/App.test.js create mode 100644 front-end/src/components/FleetManager.js create mode 100644 front-end/src/components/Login.js delete mode 100644 front-end/src/index.css delete mode 100644 front-end/src/logo.svg create mode 100644 front-end/src/main.css create mode 100644 front-end/src/stateSlicers/authenticationSlice.js create mode 100644 front-end/src/store.js create mode 100644 front-end/tailwind.config.js diff --git a/front-end/package-lock.json b/front-end/package-lock.json index aad6dcb..3f83781 100644 --- a/front-end/package-lock.json +++ b/front-end/package-lock.json @@ -8,13 +8,21 @@ "name": "front-end", "version": "0.1.0", "dependencies": { + "@reduxjs/toolkit": "^2.2.7", "@testing-library/jest-dom": "^5.17.0", "@testing-library/react": "^13.4.0", "@testing-library/user-event": "^13.5.0", "react": "^18.3.1", "react-dom": "^18.3.1", + "react-redux": "^9.1.2", "react-scripts": "5.0.1", "web-vitals": "^2.1.4" + }, + "devDependencies": { + "@babel/plugin-proposal-private-property-in-object": "^7.21.11", + "autoprefixer": "^10.4.20", + "postcss": "^8.4.41", + "tailwindcss": "^3.4.10" } }, "node_modules/@adobe/css-tools": { @@ -674,10 +682,18 @@ } }, "node_modules/@babel/plugin-proposal-private-property-in-object": { - "version": "7.21.0-placeholder-for-preset-env.2", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.0-placeholder-for-preset-env.2.tgz", - "integrity": "sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w==", + "version": "7.21.11", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.11.tgz", + "integrity": "sha512-0QZ8qP/3RLDVBwBFoWAwCtgcDZJVwA5LUJRZU8x2YFfKNuFq161wK3cuGrALu5yiPu+vzwTAg/sMWVNeWeNyaw==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-private-property-in-object instead.", + "dev": true, "license": "MIT", + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.18.6", + "@babel/helper-create-class-features-plugin": "^7.21.0", + "@babel/helper-plugin-utils": "^7.20.2", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5" + }, "engines": { "node": ">=6.9.0" }, @@ -2018,6 +2034,18 @@ "@babel/core": "^7.0.0-0" } }, + "node_modules/@babel/preset-env/node_modules/@babel/plugin-proposal-private-property-in-object": { + "version": "7.21.0-placeholder-for-preset-env.2", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.0-placeholder-for-preset-env.2.tgz", + "integrity": "sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w==", + "license": "MIT", + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, "node_modules/@babel/preset-env/node_modules/semver": { "version": "6.3.1", "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", @@ -3540,6 +3568,40 @@ } } }, + "node_modules/@reduxjs/toolkit": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/@reduxjs/toolkit/-/toolkit-2.2.7.tgz", + "integrity": "sha512-faI3cZbSdFb8yv9dhDTmGwclW0vk0z5o1cia+kf7gCbaCwHI5e+7tP57mJUv22pNcNbeA62GSrPpfrUfdXcQ6g==", + "license": "MIT", + "dependencies": { + "immer": "^10.0.3", + "redux": "^5.0.1", + "redux-thunk": "^3.1.0", + "reselect": "^5.1.0" + }, + "peerDependencies": { + "react": "^16.9.0 || ^17.0.0 || ^18", + "react-redux": "^7.2.1 || ^8.1.3 || ^9.0.0" + }, + "peerDependenciesMeta": { + "react": { + "optional": true + }, + "react-redux": { + "optional": true + } + } + }, + "node_modules/@reduxjs/toolkit/node_modules/immer": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/immer/-/immer-10.1.1.tgz", + "integrity": "sha512-s2MPrmjovJcoMaHtx6K11Ra7oD05NT97w1IC5zpMkT6Atjr7H8LjaDd81iIxUYpMKSRRNMJE703M1Fhr/TctHw==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/immer" + } + }, "node_modules/@rollup/plugin-babel": { "version": "5.3.1", "resolved": "https://registry.npmjs.org/@rollup/plugin-babel/-/plugin-babel-5.3.1.tgz", @@ -4836,6 +4898,12 @@ "integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==", "license": "MIT" }, + "node_modules/@types/use-sync-external-store": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.3.tgz", + "integrity": "sha512-EwmlvuaxPNej9+T4v5AuBPJa2x2UOJVdjCtDHgcDqitUeOtjnJKJ+apYjVcAoBEMjKW1VVFGZLUb5+qqa09XFA==", + "license": "MIT" + }, "node_modules/@types/ws": { "version": "8.5.12", "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.12.tgz", @@ -16206,6 +16274,29 @@ "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", "license": "MIT" }, + "node_modules/react-redux": { + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-9.1.2.tgz", + "integrity": "sha512-0OA4dhM1W48l3uzmv6B7TXPCGmokUU4p1M44DGN2/D9a1FjVPukVjER1PcPX97jIg6aUeLq1XJo1IpfbgULn0w==", + "license": "MIT", + "dependencies": { + "@types/use-sync-external-store": "^0.0.3", + "use-sync-external-store": "^1.0.0" + }, + "peerDependencies": { + "@types/react": "^18.2.25", + "react": "^18.0", + "redux": "^5.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "redux": { + "optional": true + } + } + }, "node_modules/react-refresh": { "version": "0.11.0", "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.11.0.tgz", @@ -16348,6 +16439,21 @@ "node": ">=8" } }, + "node_modules/redux": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/redux/-/redux-5.0.1.tgz", + "integrity": "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w==", + "license": "MIT" + }, + "node_modules/redux-thunk": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/redux-thunk/-/redux-thunk-3.1.0.tgz", + "integrity": "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw==", + "license": "MIT", + "peerDependencies": { + "redux": "^5.0.0" + } + }, "node_modules/reflect.getprototypeof": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.6.tgz", @@ -16509,6 +16615,12 @@ "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==", "license": "MIT" }, + "node_modules/reselect": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/reselect/-/reselect-5.1.1.tgz", + "integrity": "sha512-K/BG6eIky/SBpzfHZv/dd+9JBFiS4SWV7FIujVyJRux6e45+73RaUHXLmIR1f7WOMaQ0U1km6qwklRQxpJJY0w==", + "license": "MIT" + }, "node_modules/resolve": { "version": "1.22.8", "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", @@ -18694,6 +18806,15 @@ "requires-port": "^1.0.0" } }, + "node_modules/use-sync-external-store": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.2.tgz", + "integrity": "sha512-PElTlVMwpblvbNqQ82d2n6RjStvdSoNe9FG28kNfz3WiXilJm4DdNkEzRhCZuIDwY8U08WVihhGR5iRqAwfDiw==", + "license": "MIT", + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0" + } + }, "node_modules/util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", diff --git a/front-end/package.json b/front-end/package.json index 77252fc..5436421 100644 --- a/front-end/package.json +++ b/front-end/package.json @@ -3,11 +3,13 @@ "version": "0.1.0", "private": true, "dependencies": { + "@reduxjs/toolkit": "^2.2.7", "@testing-library/jest-dom": "^5.17.0", "@testing-library/react": "^13.4.0", "@testing-library/user-event": "^13.5.0", "react": "^18.3.1", "react-dom": "^18.3.1", + "react-redux": "^9.1.2", "react-scripts": "5.0.1", "web-vitals": "^2.1.4" }, @@ -34,5 +36,11 @@ "last 1 firefox version", "last 1 safari version" ] + }, + "devDependencies": { + "@babel/plugin-proposal-private-property-in-object": "^7.21.11", + "autoprefixer": "^10.4.20", + "postcss": "^8.4.41", + "tailwindcss": "^3.4.10" } } diff --git a/front-end/postcss.config.js b/front-end/postcss.config.js new file mode 100644 index 0000000..787a42d --- /dev/null +++ b/front-end/postcss.config.js @@ -0,0 +1,6 @@ +module.exports = { + plugins: { + tailwindcss: {}, + autoprefixer: {} + } +} \ No newline at end of file diff --git a/front-end/src/App.css b/front-end/src/App.css deleted file mode 100644 index 74b5e05..0000000 --- a/front-end/src/App.css +++ /dev/null @@ -1,38 +0,0 @@ -.App { - text-align: center; -} - -.App-logo { - height: 40vmin; - pointer-events: none; -} - -@media (prefers-reduced-motion: no-preference) { - .App-logo { - animation: App-logo-spin infinite 20s linear; - } -} - -.App-header { - background-color: #282c34; - min-height: 100vh; - display: flex; - flex-direction: column; - align-items: center; - justify-content: center; - font-size: calc(10px + 2vmin); - color: white; -} - -.App-link { - color: #61dafb; -} - -@keyframes App-logo-spin { - from { - transform: rotate(0deg); - } - to { - transform: rotate(360deg); - } -} diff --git a/front-end/src/App.js b/front-end/src/App.js index 3784575..92beefb 100644 --- a/front-end/src/App.js +++ b/front-end/src/App.js @@ -1,25 +1,19 @@ -import logo from './logo.svg'; -import './App.css'; +import React from "react"; +import './main.css' -function App() { +import { useSelector } from "react-redux"; +import { selectAuthenticated } from "./stateSlicers/authenticationSlice"; + +import FleetManager from "./components/FleetManager"; +import Login from "./components/Login"; + +function App () { + const authed = useSelector(selectAuthenticated); return ( -
-
- logo -

- Edit src/App.js and save to reload. -

- - Learn React - -
-
- ); + (authed) + ? + : + ) } export default App; diff --git a/front-end/src/App.test.js b/front-end/src/App.test.js deleted file mode 100644 index 1f03afe..0000000 --- a/front-end/src/App.test.js +++ /dev/null @@ -1,8 +0,0 @@ -import { render, screen } from '@testing-library/react'; -import App from './App'; - -test('renders learn react link', () => { - render(); - const linkElement = screen.getByText(/learn react/i); - expect(linkElement).toBeInTheDocument(); -}); diff --git a/front-end/src/components/FleetManager.js b/front-end/src/components/FleetManager.js new file mode 100644 index 0000000..8d9d1cc --- /dev/null +++ b/front-end/src/components/FleetManager.js @@ -0,0 +1,5 @@ +export default function FleetManager() { + return ( +
+ ) +} \ No newline at end of file diff --git a/front-end/src/components/Login.js b/front-end/src/components/Login.js new file mode 100644 index 0000000..22e28e9 --- /dev/null +++ b/front-end/src/components/Login.js @@ -0,0 +1,50 @@ +import React, {Component} from 'react' + +export default class Login extends Component { + constructor() { + super() + + this.state = { + username: '', + password: '' + } + } + + updateField = (event) => { + this.setState({ + [event.target.id]: event.target.value + }) + } + + submit = () => { + + } + + + render() { + return ( +
+
+
Fleet Management
+
+ +
+ +
Username
+
+
+ +
Password
+
+
+
+ Login +
+
+
+
+
+ ) + } + +} \ No newline at end of file diff --git a/front-end/src/index.css b/front-end/src/index.css deleted file mode 100644 index ec2585e..0000000 --- a/front-end/src/index.css +++ /dev/null @@ -1,13 +0,0 @@ -body { - margin: 0; - font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', - 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', - sans-serif; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -code { - font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', - monospace; -} diff --git a/front-end/src/index.js b/front-end/src/index.js index d563c0f..ba4beca 100644 --- a/front-end/src/index.js +++ b/front-end/src/index.js @@ -1,14 +1,16 @@ import React from 'react'; import ReactDOM from 'react-dom/client'; -import './index.css'; import App from './App'; import reportWebVitals from './reportWebVitals'; +import { Provider } from 'react-redux' +import store from './store' + const root = ReactDOM.createRoot(document.getElementById('root')); root.render( - + - + ); // If you want to start measuring performance in your app, pass a function diff --git a/front-end/src/logo.svg b/front-end/src/logo.svg deleted file mode 100644 index 9dfc1c0..0000000 --- a/front-end/src/logo.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/front-end/src/main.css b/front-end/src/main.css new file mode 100644 index 0000000..b5c61c9 --- /dev/null +++ b/front-end/src/main.css @@ -0,0 +1,3 @@ +@tailwind base; +@tailwind components; +@tailwind utilities; diff --git a/front-end/src/stateSlicers/authenticationSlice.js b/front-end/src/stateSlicers/authenticationSlice.js new file mode 100644 index 0000000..49dbeb7 --- /dev/null +++ b/front-end/src/stateSlicers/authenticationSlice.js @@ -0,0 +1,29 @@ +import { createSlice } from "@reduxjs/toolkit"; + +export const authenticationSlice = createSlice({ + name: 'auth', + initialState: { + authenticated: false, + userId: null, + fullName: null + }, + reducers: { + authenticated: (state, action) => { + state.authenticated = true; + state.userId = action.payload.userId; + state.fullName = action.payload.fullName; + }, + logout: (state) => { + state = { + authenticated: false, + userId: null, + fullName: null + } + } + } +}) + +export const {authenticated, logout} = authenticationSlice.actions +export const selectAuthenticated = (state) => state.authenticated + +export default authenticationSlice.reducer \ No newline at end of file diff --git a/front-end/src/store.js b/front-end/src/store.js new file mode 100644 index 0000000..fdd6bdd --- /dev/null +++ b/front-end/src/store.js @@ -0,0 +1,7 @@ +import { configureStore } from '@reduxjs/toolkit' + +import authenticationReducer from './stateSlicers/authenticationSlice' + +export default configureStore({ + reducer: {authenticationReducer}, +}) \ No newline at end of file diff --git a/front-end/tailwind.config.js b/front-end/tailwind.config.js new file mode 100644 index 0000000..e5b64f3 --- /dev/null +++ b/front-end/tailwind.config.js @@ -0,0 +1,9 @@ +/** @type {import('tailwindcss').Config} */ +module.exports = { + content: ["./src/**/*.{html,js}"], + theme: { + extend: {}, + }, + plugins: [], +} +