Compare commits

...

26 commits

Author SHA1 Message Date
3a23f60488 Merge branch 'main' into feature/settings-page 2026-02-07 12:04:11 -05:00
40db6ec877 Merge branch 'feature/settings-page' of https://github.com/Doble-Technologies/ShiftSync-Website into feature/settings-page 2025-12-01 22:55:55 -05:00
941110abc1 Settings add 2025-12-01 22:55:54 -05:00
4ba9e91eee Update 2025-08-09 15:49:15 -04:00
a0bb4bef4a Init API Side 2025-06-23 17:55:42 -04:00
16eb49d41e Merge branch 'main' into feature/settings-page 2025-06-15 22:32:05 -04:00
bb8a672b98 Merge branch 'main' into feature/settings-page 2025-06-12 16:33:18 -04:00
6389eb8d6b Merge branch 'main' into feature/settings-page 2025-06-12 16:25:53 -04:00
4e49222e2b Merge branch 'main' into feature/settings-page 2025-06-12 16:24:08 -04:00
3d274363e7 Merge branch 'main' into feature/settings-page 2025-06-12 16:23:11 -04:00
ec58d344dc Merge branch 'main' into feature/settings-page 2025-06-12 16:19:20 -04:00
e60de19e02 Merge branch 'main' into feature/settings-page 2025-06-12 16:18:22 -04:00
6806cdabfb Merge branch 'main' into feature/settings-page 2025-06-12 16:16:04 -04:00
79c3acb6be Merge branch 'main' into feature/settings-page 2025-06-12 16:12:27 -04:00
0010ba3a4f Merge branch 'main' into feature/settings-page 2025-06-12 16:09:27 -04:00
1b7d1e54e2 Merge branch 'main' into feature/settings-page 2025-06-12 16:07:48 -04:00
1d00dd528f Remove Echo 2025-06-12 15:44:30 -04:00
b0670cc3a3 Update Settings.jsx 2025-06-12 15:41:28 -04:00
f4766ed7e1 Merge branch 'main' into feature/settings-page 2025-06-12 15:41:24 -04:00
bdfe409891 Integrity 1 2025-06-11 10:53:22 -04:00
20c088e9e7 Update Dockerfile 2025-06-11 10:42:19 -04:00
375059e3ee Update Dockerfile 2025-06-11 10:39:55 -04:00
21ac649bba Merge branch 'main' into feature/settings-page 2025-06-11 10:22:36 -04:00
3a7a6d3e2c Update package.json 2025-06-11 10:22:31 -04:00
fa56c3a785 Update docker-compose.yaml 2025-06-10 23:17:33 -04:00
49c5e4026c Update docker-compose.yaml 2025-06-10 23:10:05 -04:00
13 changed files with 4654 additions and 656 deletions

6
api/router/rest/index.js Normal file
View file

@ -0,0 +1,6 @@
import express from 'express';
import { userDataRouter } from './userData/index.js';
export const restRouter = express.Router();
restRouter.use('/userData', userDataRouter);

View file

@ -0,0 +1,17 @@
import express from 'express';
import { databaseServices } from '../../../services/index.js';
export const userDataRouter = express.Router();
userDataRouter.get('/', async (req, res) => {
let data;
try {
data = await databaseServices.getUsers(req.query);
res.status(200);
} catch (err) {
data = { Error: err?.message };
res.status(500);
} finally {
res.send(data);
}
})

6
api/router/routes.js Normal file
View file

@ -0,0 +1,6 @@
import express from 'express';
import { restRouter } from './rest/index.js';
export const routes = express.Router();
routes.use('/rest', restRouter);

View file

@ -0,0 +1,27 @@
import { Pool } from 'pg';
import 'dotenv/config';
const poolCreds = {
host: process.env.POSTGRES_HOST,
database: process.env.POSTGRES_DB,
port: process.env.POSTGRES_PORT,
user: process.env.POSTGRES_USER,
password: process.env.POSTGRES_PASSWORD,
ssl:
process.env.APP_ENV === 'local'
? null
: { require: true, rejectUnauthorized }
};
const pool = new Pool(poolCreds);
export const runQuery = async (query, params = []) => {
const client = await pool.connect();
try {
const pgQueryResponse = await client.query(query, params);
return pgQueryResponse?.rows || [];
} finally {
client.release();
}
}

View file

@ -0,0 +1,14 @@
import express from 'express';
const router = express.Router();
// GET /api/departments
router.get('/', (req, res) => {
res.json([{ id: 1, name: 'Fire Department' }]);
});
// POST /api/departments
router.post('/', (req, res) => {
res.status(201).json({ message: 'Department created' });
});
export default router;

View file

@ -0,0 +1,15 @@
import { runQuery } from '../connection.js';
const getUsers = async (args) => {
try {
const dataResp = await runQuery('SELECT * FROM users');
return dataResp;
} catch (err) {
console.log('GET USERS ERROR: ', err);
throw err;
}
};
export const userDataOperations = {
getUsers
}

View file

@ -0,0 +1,16 @@
import express from 'express';
const router = express.Router();
// GET /api/users
router.get('/', (req, res) => {
res.json([{ id: 1, name: 'John Doe' }]);
});
// POST /api/users
router.post('/', (req, res) => {
res.status(201).json({ message: 'User created' });
});
// Add more routes like PUT, DELETE here later
export default router;

View file

@ -194,7 +194,8 @@ export const TransferBox = ({ fields, leftGroup, rightGroup, onSave, user={} })
</CenterButton> </CenterButton>
<CenterButton <CenterButton
onClick={() => { onClick={() => {
return onSave(rightItems?.map((item) => { return item?.id })) onSave(rightItems?.map((item) => { return item?.id }));
setItemSelected({});
}} }}
color='#00B33C' color='#00B33C'
buttonEnabled={hasChanges} buttonEnabled={hasChanges}

View file

@ -5,4 +5,6 @@ export const useLocalStore = create((set) => ({
setUser: (user) => set({ user }), setUser: (user) => set({ user }),
department: null, department: null,
setDepartment: (department) => set({ department }), setDepartment: (department) => set({ department }),
access: null,
setAccess: (access) => set({ access }),
})); }));

3856
web/package-lock.json generated Normal file

File diff suppressed because it is too large Load diff

View file

@ -136,6 +136,7 @@ const InnerCardRowLabel = styled('label')`
width: 60%; width: 60%;
padding-right: 10px; padding-right: 10px;
font-size: 14px; font-size: 14px;
align-items: center;
`; `;
const InnerCardRowInput = styled('input')` const InnerCardRowInput = styled('input')`
@ -152,6 +153,7 @@ const InnerCardRadioInput = styled('input')`
const InnerCardRadioLabel = styled('label')` const InnerCardRadioLabel = styled('label')`
font-size: 14px; font-size: 14px;
padding-left: 5px; padding-left: 5px;
align-items: center;
`; `;
const InnerCardRowRadioDiv = styled('div')` const InnerCardRowRadioDiv = styled('div')`
@ -451,7 +453,9 @@ export const Settings = () => {
fields={field} fields={field}
leftGroup={department[field?.origList]} leftGroup={department[field?.origList]}
rightGroup={department[field?.id]} rightGroup={department[field?.id]}
onSave={(t) => console.log(t)} onSave={(t) => {
return onSubmit({ [field.id]: t });
}}
/> />
} }
return field?.type !== 'header' && ( return field?.type !== 'header' && (

View file

@ -3,7 +3,7 @@ import { Routes, Route } from 'react-router-dom';
import { Home, Profile, Schedule, Settings } from '@src/pages'; import { Home, Profile, Schedule, Settings } from '@src/pages';
import { Shell } from '@components'; import { Shell } from '@components';
import { useLocalStore } from '@components'; import { useLocalStore } from '@components';
import axios from "axios"; import { fetchAPI } from './axios.js';
const dept = { const dept = {
id: 1, id: 1,
@ -31,6 +31,7 @@ const users = [
first_name: 'ShiftSync-Administrator', first_name: 'ShiftSync-Administrator',
last_name: 'Test-User', last_name: 'Test-User',
email: 'testuserA@shift-sync.com', email: 'testuserA@shift-sync.com',
accessLevel: 150,
is_ss_admin: false is_ss_admin: false
}, },
{ {
@ -38,6 +39,7 @@ const users = [
first_name: 'ShiftSync-Manager', first_name: 'ShiftSync-Manager',
last_name: 'Test-User', last_name: 'Test-User',
email: 'testuserM@shift-sync.com', email: 'testuserM@shift-sync.com',
accessLevel: 100,
is_ss_admin: false is_ss_admin: false
}, },
{ {
@ -45,6 +47,7 @@ const users = [
first_name: 'ShiftSync-Scheduler', first_name: 'ShiftSync-Scheduler',
last_name: 'Test-User', last_name: 'Test-User',
email: 'testuserS@shift-sync.com', email: 'testuserS@shift-sync.com',
accessLevel: 50,
is_ss_admin: false is_ss_admin: false
}, },
{ {
@ -52,74 +55,78 @@ const users = [
first_name: 'ShiftSync-User', first_name: 'ShiftSync-User',
last_name: 'Test-User', last_name: 'Test-User',
email: 'testuserU@shift-sync.com', email: 'testuserU@shift-sync.com',
accessLevel: 1,
is_ss_admin: false is_ss_admin: false
}, },
] ];
const AppRouter = () => { const AppRouter = () => {
const { user, setUser, setDepartment } = useLocalStore(); const { user, setUser, setDepartment } = useLocalStore();
const [userChanged, setUserChanged] = useState(false); const [userChanged, setUserChanged] = useState(false);
const isDev = true; // change for it.
const fetchAPI = async () => {
const location = window.location;
const uri = `${location?.protocol}//${location.hostname}/api`;
const response = await axios.get(uri);
console.log(response.data.fruits);
};
useEffect(() => { useEffect(() => {
const localVersion = localStorage.getItem("APP_VERSION"); const init = async () => {
const currentVersion = window.APP_VERSION; const localVersion = localStorage.getItem("APP_VERSION");
const currentVersion = window.APP_VERSION;
if (localVersion && localVersion !== currentVersion) { if (localVersion && localVersion !== currentVersion) {
console.log("Version changed, forcing reload"); console.log("Version changed, forcing reload");
localStorage.setItem("APP_VERSION", currentVersion); localStorage.setItem("APP_VERSION", currentVersion);
window.location.reload(true); // force full page reload window.location.reload(true);
} else { return;
localStorage.setItem("APP_VERSION", currentVersion); } else {
} localStorage.setItem("APP_VERSION", currentVersion);
}
fetchAPI(); const data = await fetchAPI('userData', 'get');
// await call for getting the count of employees and any other calls to db. console.log('data:', data);
const employee_count = 1;
const subs_expiration = '10/22/2025'; // TODO: Replace this with real data from your API
setUser({ // const users = data?.users || []; // Example fix
...users[0], // const dept = data?.dept || {}; // Example fix
scheduler: dept?.schedulers?.includes(1),
manager: dept?.managers?.includes(1), const employee_count = 1;
administrator: dept?.administrators?.includes(1) const subs_expiration = '10/22/2025';
});
const newAdministrators = dept?.administrators?.map((admin) => { setUser({
const user = users?.find((user) => { ...users[0],
return user?.id === admin; scheduler: dept?.schedulers?.includes(1),
manager: dept?.managers?.includes(1),
administrator: dept?.administrators?.includes(1)
}); });
return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` };
}); const newAdministrators = dept?.administrators?.map((admin) => {
const newManagers = dept?.managers?.map((manager) => { const user = users?.find((user) => user?.id === admin);
const user = users?.find((user) => { return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` };
return user?.id === manager;
}); });
return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` };
}); const newManagers = dept?.managers?.map((manager) => {
const newSchedulers = dept?.schedulers?.map((scheduler) => { const user = users?.find((user) => user?.id === manager);
const user = users?.find((user) => { return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` };
return user?.id === scheduler;
}); });
return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` };
}); const newSchedulers = dept?.schedulers?.map((scheduler) => {
const newUsers = users?.map((user) => { const user = users?.find((user) => user?.id === scheduler);
return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` }; return { id: user?.id, value: `${user?.last_name}, ${user?.first_name}` };
}); });
setDepartment({
...dept, const newUsers = users?.map((user) => ({
users: newUsers, id: user?.id,
schedulers: newSchedulers, value: `${user?.last_name}, ${user?.first_name}`
managers: newManagers, }));
administrators: newAdministrators,
employee_count, setDepartment({
subs_expiration ...dept,
}); users: newUsers,
schedulers: newSchedulers,
managers: newManagers,
administrators: newAdministrators,
employee_count,
subs_expiration
});
};
init();
}, []); }, []);
useEffect(() => { useEffect(() => {

27
web/src/router/axios.js Normal file
View file

@ -0,0 +1,27 @@
import axios from 'axios';
export const fetchAPI = async (endpoint, type, body = null) => {
const location = window.location;
const url = `${location?.protocol}//${location.hostname}${location?.hostname === 'localhost' ? ':5172' : ''}/api/rest/${endpoint}`;
const requestOptions = {
headers: {
'Content-Type': 'application/json'
}
};
try {
let response;
if (type === 'post') {
response = await axios.post(url, body, requestOptions);
} else if (type === 'get') {
response = await axios.get(url, requestOptions);
} else {
console.warn('No proper type given: ', type);
return;
}
return response.data;
} catch (err) {
console.error('PG Rest Query Error: ', err);
}
};