mirror of
https://gitea.smigz.com/smiggiddy/odin-codeprojects.git
synced 2025-04-05 03:10:57 -04:00
some code and stuff
This commit is contained in:
parent
75b83e30ea
commit
17d664eaeb
19 changed files with 762 additions and 4 deletions
84
inventory/public/css/style.css
Normal file
84
inventory/public/css/style.css
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
/* 1. Use a more-intuitive box-sizing model */
|
||||||
|
*,
|
||||||
|
*::before,
|
||||||
|
*::after {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 2. Remove default margin */
|
||||||
|
* {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
/* 3. Add accessible line-height */
|
||||||
|
line-height: 1.5;
|
||||||
|
/* 4. Improve text rendering */
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 5. Improve media defaults */
|
||||||
|
img,
|
||||||
|
picture,
|
||||||
|
video,
|
||||||
|
canvas,
|
||||||
|
svg {
|
||||||
|
display: block;
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 6. Inherit fonts for form controls */
|
||||||
|
input,
|
||||||
|
button,
|
||||||
|
textarea,
|
||||||
|
select {
|
||||||
|
font: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 7. Avoid text overflows */
|
||||||
|
p,
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
overflow-wrap: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 8. Improve line wrapping */
|
||||||
|
p {
|
||||||
|
text-wrap: pretty;
|
||||||
|
}
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
text-wrap: balance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
9. Create a root stacking context
|
||||||
|
*/
|
||||||
|
#root,
|
||||||
|
#__next {
|
||||||
|
isolation: isolate;
|
||||||
|
}
|
||||||
|
|
||||||
|
li {
|
||||||
|
list-style-type: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-around;
|
||||||
|
max-width: 50vw;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.flex {
|
||||||
|
display: flex;
|
||||||
|
}
|
58
inventory/public/js/components/modal.js
Normal file
58
inventory/public/js/components/modal.js
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
import { categoryPicker, storePicker } from "./selectComponent.js";
|
||||||
|
|
||||||
|
function formInput(props) {
|
||||||
|
const formItemDiv = document.createElement("div");
|
||||||
|
formItemDiv.classList.add("form-item");
|
||||||
|
const input = document.createElement("input");
|
||||||
|
input.name = props.name;
|
||||||
|
const label = document.createElement("label");
|
||||||
|
label.textContent = props.labelText;
|
||||||
|
formItemDiv.append(label, input);
|
||||||
|
|
||||||
|
return formItemDiv;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function modal() {
|
||||||
|
const div = document.createElement("div");
|
||||||
|
div.classList.add("modal");
|
||||||
|
div.style = `
|
||||||
|
display: none;
|
||||||
|
positino: fixed;
|
||||||
|
z-index: 1;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 50vh;
|
||||||
|
overflow: auto;
|
||||||
|
background-color: rgb(0,0,0);
|
||||||
|
background-color: rgba(0,0,0,0.4);
|
||||||
|
`;
|
||||||
|
|
||||||
|
const form = document.createElement("form");
|
||||||
|
form.method = "POST";
|
||||||
|
form.action = "/item";
|
||||||
|
const name = formInput({ labelText: "Name", name: "name" });
|
||||||
|
const price = formInput({ labelText: "Price", name: "price" });
|
||||||
|
const qty = formInput({ labelText: "QTY", name: "qty" });
|
||||||
|
|
||||||
|
const categories = await categoryPicker();
|
||||||
|
const stores = await storePicker();
|
||||||
|
|
||||||
|
const submitButton = document.createElement("button");
|
||||||
|
submitButton.type = "submit";
|
||||||
|
submitButton.classList.add("btn");
|
||||||
|
submitButton.textContent = "Add Item";
|
||||||
|
|
||||||
|
const closeButton = document.createElement("button");
|
||||||
|
closeButton.textContent = "Close";
|
||||||
|
closeButton.addEventListener("click", () => {
|
||||||
|
div.style.display = "none";
|
||||||
|
});
|
||||||
|
|
||||||
|
form.append(name, price, qty, categories, stores, submitButton, closeButton);
|
||||||
|
div.append(form);
|
||||||
|
|
||||||
|
return div;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { modal };
|
59
inventory/public/js/components/selectComponent.js
Normal file
59
inventory/public/js/components/selectComponent.js
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
async function getData(path) {
|
||||||
|
const categories = await fetch(`/${path}`);
|
||||||
|
return await categories.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectComponent(data, select) {
|
||||||
|
data.forEach((opt) => {
|
||||||
|
const option = document.createElement("option");
|
||||||
|
option.value = opt.name;
|
||||||
|
option.textContent = opt.name;
|
||||||
|
select.append(option);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function categoryPicker() {
|
||||||
|
const selectQuery = document.querySelector(".categories-list-modal");
|
||||||
|
if (selectQuery) document.body.remove(selectQuery);
|
||||||
|
|
||||||
|
const categories = await getData("categories");
|
||||||
|
|
||||||
|
const formItemDiv = document.createElement("div");
|
||||||
|
formItemDiv.classList.add("form-item");
|
||||||
|
|
||||||
|
const formLabel = document.createElement("label");
|
||||||
|
formLabel.textContent = "Categories";
|
||||||
|
|
||||||
|
const select = document.createElement("select");
|
||||||
|
select.classList.add("categories-list-modal");
|
||||||
|
select.name = "category";
|
||||||
|
|
||||||
|
selectComponent(categories, select);
|
||||||
|
|
||||||
|
formItemDiv.append(formLabel, select);
|
||||||
|
return formItemDiv;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function storePicker() {
|
||||||
|
const selectQuery = document.querySelector(".store-list-modal");
|
||||||
|
if (selectQuery) document.body.remove(selectQuery);
|
||||||
|
|
||||||
|
const stores = await getData("stores");
|
||||||
|
|
||||||
|
const formItemDiv = document.createElement("div");
|
||||||
|
formItemDiv.classList.add("form-item");
|
||||||
|
|
||||||
|
const formLabel = document.createElement("label");
|
||||||
|
formLabel.textContent = "stores";
|
||||||
|
|
||||||
|
const select = document.createElement("select");
|
||||||
|
select.classList.add("stores-list-modal");
|
||||||
|
select.name = "store";
|
||||||
|
|
||||||
|
selectComponent(stores, select);
|
||||||
|
|
||||||
|
formItemDiv.append(formLabel, select);
|
||||||
|
return formItemDiv;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { categoryPicker, storePicker };
|
63
inventory/public/js/script.js
Normal file
63
inventory/public/js/script.js
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
import { modal } from "./components/modal.js";
|
||||||
|
|
||||||
|
const newItemLink = document.querySelector("a[href='/add']");
|
||||||
|
//
|
||||||
|
//if (newItemLink && window.location !== "/item/") {
|
||||||
|
const modalElement = await modal();
|
||||||
|
let modalToggle = false;
|
||||||
|
async function handleDelete(e) {
|
||||||
|
console.log(e.target);
|
||||||
|
}
|
||||||
|
|
||||||
|
document.querySelectorAll(".delete-button").forEach((button) =>
|
||||||
|
button.addEventListener("click", async (e) => {
|
||||||
|
const itemName = e.target.dataset.id;
|
||||||
|
const data = new URLSearchParams(new FormData());
|
||||||
|
data.append("name", itemName);
|
||||||
|
|
||||||
|
const res = await fetch("/item", {
|
||||||
|
method: "DELETE",
|
||||||
|
body: data,
|
||||||
|
});
|
||||||
|
window.location.href = "/";
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
document.querySelectorAll(".edit-button").forEach((button) => {
|
||||||
|
button.addEventListener("click", async (e) => {
|
||||||
|
const itemId = e.target.parentNode.dataset.itemId;
|
||||||
|
|
||||||
|
window.location.href = `/item/${itemId}`;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
document.querySelectorAll(".delete-btn").forEach((button) => {
|
||||||
|
button.addEventListener("click", async (e) => {
|
||||||
|
const itemId = e.target.dataset.id;
|
||||||
|
const data = new URLSearchParams(new FormData());
|
||||||
|
data.append("id", itemId);
|
||||||
|
|
||||||
|
const res = await fetch("/category", {
|
||||||
|
method: "DELETE",
|
||||||
|
body: data,
|
||||||
|
});
|
||||||
|
const message = await res.text();
|
||||||
|
|
||||||
|
if (message.includes("error")) {
|
||||||
|
alert("unable to delete as item is using category");
|
||||||
|
} else {
|
||||||
|
window.location.href = "/category";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
newItemLink.addEventListener("click", (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
modalToggle
|
||||||
|
? (modalElement.style.display = "none")
|
||||||
|
: (modalElement.style.display = "block");
|
||||||
|
modalToggle = !modalToggle;
|
||||||
|
});
|
||||||
|
|
||||||
|
document.body.append(modalElement);
|
||||||
|
//}
|
|
@ -7,8 +7,9 @@ const port = 8080;
|
||||||
const { indexRouter } = require("./routes/indexRouter");
|
const { indexRouter } = require("./routes/indexRouter");
|
||||||
const assetsPath = path.join(path.dirname(__dirname), "public");
|
const assetsPath = path.join(path.dirname(__dirname), "public");
|
||||||
|
|
||||||
|
app.set("view engine", "ejs");
|
||||||
app.set("views", path.join(__dirname, "views"));
|
app.set("views", path.join(__dirname, "views"));
|
||||||
app.set(express.static(assetsPath));
|
app.use(express.static(assetsPath));
|
||||||
|
|
||||||
app.use(express.urlencoded({ extended: true }));
|
app.use(express.urlencoded({ extended: true }));
|
||||||
|
|
||||||
|
|
6
inventory/src/config.js
Normal file
6
inventory/src/config.js
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
exports.links = [
|
||||||
|
{ name: "home", href: "/" },
|
||||||
|
{ name: "store", href: "/store" },
|
||||||
|
{ name: "categories", href: "/category" },
|
||||||
|
{ name: "new item", href: "/add" },
|
||||||
|
];
|
150
inventory/src/controllers/indexController.js
Normal file
150
inventory/src/controllers/indexController.js
Normal file
|
@ -0,0 +1,150 @@
|
||||||
|
const db = require("../db/queries");
|
||||||
|
const { links } = require("../config");
|
||||||
|
|
||||||
|
async function indexGet(req, res) {
|
||||||
|
const items = await db.getAllItemsWithRelationships();
|
||||||
|
|
||||||
|
res.render("base", {
|
||||||
|
links: links,
|
||||||
|
pageTitle: "Iventory Home",
|
||||||
|
items: items,
|
||||||
|
page: "",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function storeGet(req, res) {
|
||||||
|
const stores = await db.getStores();
|
||||||
|
|
||||||
|
res.render("base", {
|
||||||
|
page: "add",
|
||||||
|
links: links,
|
||||||
|
pageTitle: "Inventory | Stores",
|
||||||
|
type: "store",
|
||||||
|
data: stores,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
async function storesGetAll(req, res) {
|
||||||
|
const stores = await db.getStores();
|
||||||
|
res.json(stores);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function storePost(req, res) {
|
||||||
|
const r = await handlePostRequets({ reqBody: req.body, dbName: "store" });
|
||||||
|
res.redirect("/store");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function categoryGet(req, res) {
|
||||||
|
const categories = await db.getCategories();
|
||||||
|
res.render("base", {
|
||||||
|
page: "add",
|
||||||
|
links: links,
|
||||||
|
pageTitle: "Inventory | Categories",
|
||||||
|
type: "category",
|
||||||
|
data: categories,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
async function categoryGetAll(req, res) {
|
||||||
|
const categories = await db.getCategories();
|
||||||
|
res.json(categories);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function categoryPost(req, res) {
|
||||||
|
const r = await handlePostRequets({ reqBody: req.body, dbName: "category" });
|
||||||
|
if (r) {
|
||||||
|
console.log("something went wrong");
|
||||||
|
}
|
||||||
|
res.redirect("/category");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handlePostRequets(data) {
|
||||||
|
const { dbName } = data;
|
||||||
|
const { name } = data.reqBody;
|
||||||
|
let res;
|
||||||
|
|
||||||
|
if (dbName === "store") {
|
||||||
|
res = db.insertStore(name);
|
||||||
|
} else if (dbName === "category") {
|
||||||
|
res = db.insertCategory(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function itemPost(req, res) {
|
||||||
|
const categoryId = await db.getCategoryId({ name: req.body.category });
|
||||||
|
const storeId = await db.getStoreId({ name: req.body.store });
|
||||||
|
|
||||||
|
let formData = req.body;
|
||||||
|
const data = {
|
||||||
|
name: formData.name,
|
||||||
|
price: formData.price,
|
||||||
|
qty: formData.qty,
|
||||||
|
store_id: storeId.store_id,
|
||||||
|
category_id: categoryId.category_id,
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const r = await db.insertItem(data);
|
||||||
|
if (r) throw new Error(r);
|
||||||
|
res.redirect("/");
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
res.send({ error: e });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function itemEditGet(req, res, next) {
|
||||||
|
const itemId = req.params.id;
|
||||||
|
|
||||||
|
const items = await db.getAllItemsWithRelationships();
|
||||||
|
const item = await db.getItemById(Number(itemId));
|
||||||
|
const categories = await db.getCategories();
|
||||||
|
const stores = await db.getStores();
|
||||||
|
|
||||||
|
if (item)
|
||||||
|
res.render("base", {
|
||||||
|
item: item,
|
||||||
|
categories: categories,
|
||||||
|
stores: stores,
|
||||||
|
pageTitle: `Inventory | Editing Item: ${item.name}`,
|
||||||
|
page: "edit",
|
||||||
|
links: links,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function itemEditPost(req, res, next) {
|
||||||
|
const itemId = req.params.id;
|
||||||
|
console.log(itemId);
|
||||||
|
res.redirect("/");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function itemDelete(req, res) {
|
||||||
|
console.log(req.body.name);
|
||||||
|
const itemId = await db.getItemByName(req.body.name);
|
||||||
|
if (itemId) db.deleteItem(itemId);
|
||||||
|
console.log(`Delted item: ${itemId}`);
|
||||||
|
res.redirect("deleted item: ${item}");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function categoryDelete(req, res) {
|
||||||
|
const query = await db.deleteCategory(req.body.id);
|
||||||
|
if (query) {
|
||||||
|
res.send(`${query}`);
|
||||||
|
}
|
||||||
|
res.send("category deleted");
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
indexGet,
|
||||||
|
itemPost,
|
||||||
|
itemEditGet,
|
||||||
|
itemEditPost,
|
||||||
|
itemDelete,
|
||||||
|
storeGet,
|
||||||
|
storePost,
|
||||||
|
storesGetAll,
|
||||||
|
categoryGet,
|
||||||
|
categoryPost,
|
||||||
|
categoryGetAll,
|
||||||
|
categoryDelete,
|
||||||
|
};
|
11
inventory/src/db/pool.js
Normal file
11
inventory/src/db/pool.js
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
const { Pool } = require("pg");
|
||||||
|
|
||||||
|
// Todo make this env variables
|
||||||
|
module.exports = new Pool({
|
||||||
|
host: "smig-ca04.lab.smig.tech",
|
||||||
|
user: "odin",
|
||||||
|
database: "odin",
|
||||||
|
password: "odinproject",
|
||||||
|
port: 32343,
|
||||||
|
});
|
||||||
|
// 32343;
|
120
inventory/src/db/queries.js
Normal file
120
inventory/src/db/queries.js
Normal file
|
@ -0,0 +1,120 @@
|
||||||
|
const pool = require("./pool");
|
||||||
|
|
||||||
|
async function getAllItems() {
|
||||||
|
const { rows } = await pool.query("SELECT * FROM items");
|
||||||
|
return rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getItemByName(name) {
|
||||||
|
const { rows } = await pool.query("SELECT * FROM items WHERE name = $1", [
|
||||||
|
name,
|
||||||
|
]);
|
||||||
|
return rows[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getItemById(id) {
|
||||||
|
const { rows } = await pool.query(
|
||||||
|
"SELECT id, items.name, price, qty, category.name AS category, store.name AS store FROM items LEFT JOIN category on items.category_id = category.category_id LEFT JOIN store ON items.store_id = store.store_id WHERE items.id = $1 ",
|
||||||
|
[id],
|
||||||
|
);
|
||||||
|
return rows[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
async function insertItem(data) {
|
||||||
|
const { name, price, qty, category_id, store_id } = data;
|
||||||
|
try {
|
||||||
|
await pool.query(
|
||||||
|
`INSERT INTO items (name, price, qty, category_id, store_id) VALUES ($1, $2, $3, $4, $5)`,
|
||||||
|
[name, price, qty, category_id, store_id],
|
||||||
|
);
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deleteItem(item) {
|
||||||
|
try {
|
||||||
|
const res = await pool.query("DELETE FROM items WHERE id = $1", [item.id]);
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getAllItemsWithRelationships() {
|
||||||
|
const { rows } = await pool.query(
|
||||||
|
`SELECT items.id, items.name, items.price, items.qty, category.name AS category_name, store.name AS store_name FROM items
|
||||||
|
LEFT JOIN category ON items.category_id = category.category_id
|
||||||
|
LEFT JOIN store ON items.store_id = store.store_id;`,
|
||||||
|
);
|
||||||
|
|
||||||
|
return rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getCategories() {
|
||||||
|
const { rows } = await pool.query("SELECT * FROM category");
|
||||||
|
return rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getCategoryId(item) {
|
||||||
|
const { rows } = await pool.query("SELECT * FROM category WHERE name = $1", [
|
||||||
|
item.name,
|
||||||
|
]);
|
||||||
|
return rows[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
async function insertCategory(data) {
|
||||||
|
try {
|
||||||
|
await pool.query("INSERT INTO category (name) VALUES ($1)", [data]);
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function deleteCategory(itemId) {
|
||||||
|
try {
|
||||||
|
await pool.query("DELETE FROM category WHERE category_id = $1", [itemId]);
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getStores() {
|
||||||
|
const { rows } = await pool.query("SELECT * FROM store");
|
||||||
|
return rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getStoreId(item) {
|
||||||
|
const { rows } = await pool.query("SELECT * FROM store WHERE name = $1", [
|
||||||
|
item.name,
|
||||||
|
]);
|
||||||
|
return rows[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
async function insertStore(data) {
|
||||||
|
try {
|
||||||
|
await pool.query("INSERT INTO store (name) VALUES ($1)", [data]);
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
insertItem,
|
||||||
|
getItemById,
|
||||||
|
getAllItems,
|
||||||
|
getItemByName,
|
||||||
|
deleteItem,
|
||||||
|
getCategories,
|
||||||
|
getCategoryId,
|
||||||
|
getStores,
|
||||||
|
getStoreId,
|
||||||
|
insertStore,
|
||||||
|
insertCategory,
|
||||||
|
deleteCategory,
|
||||||
|
getAllItemsWithRelationships,
|
||||||
|
};
|
64
inventory/src/db/setupDb.js
Normal file
64
inventory/src/db/setupDb.js
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
const { Client } = require("pg");
|
||||||
|
|
||||||
|
const CREATE_DB = `
|
||||||
|
CREATE DATABASE inventory_app;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const SQL = `
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS store (
|
||||||
|
store_id INTEGER PRIMARY KEY GENERATED BY DEFAULT AS IDENTITY,
|
||||||
|
name varchar(80) NOT NULL UNIQUE
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS category (
|
||||||
|
category_id INTEGER PRIMARY KEY GENERATED BY DEFAULT AS IDENTITY,
|
||||||
|
name varchar(40) NOT NULL UNIQUE
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS items (
|
||||||
|
id INTEGER PRIMARY KEY GENERATED BY DEFAULT AS IDENTITY,
|
||||||
|
name VARCHAR(80) NOT NULL,
|
||||||
|
price INTEGER,
|
||||||
|
qty INTEGER,
|
||||||
|
store_id INTEGER REFERENCES store (store_id),
|
||||||
|
category_id INTEGER REFERENCES category (category_id)
|
||||||
|
);
|
||||||
|
`;
|
||||||
|
|
||||||
|
const POPULATE = `
|
||||||
|
INSERT INTO category (name)
|
||||||
|
VALUES ('grocery'), ('clothing'), ('household'), ('cleaning'), ('misc');
|
||||||
|
|
||||||
|
INSERT INTO store (name)
|
||||||
|
VALUES ('target'), ('wegmans'), ('giant');
|
||||||
|
|
||||||
|
INSERT INTO items (name, price, qty, store_id, category_id)
|
||||||
|
VALUES ('paper towels', '399', '3', '2','3' );
|
||||||
|
`;
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
console.log("DB Creation script");
|
||||||
|
|
||||||
|
const client = new Client({
|
||||||
|
host: "smig-ca05.lab.smig.tech",
|
||||||
|
user: "odin",
|
||||||
|
database: "odin",
|
||||||
|
password: "odinproject",
|
||||||
|
port: 32343,
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
await client.connect();
|
||||||
|
await client.query(POPULATE);
|
||||||
|
await client.end();
|
||||||
|
} catch (e) {
|
||||||
|
console.log(`ERROR: ${e}`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
console.log("Done");
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
|
@ -1,9 +1,20 @@
|
||||||
const { Router } = require("express");
|
const { Router } = require("express");
|
||||||
|
const indexController = require("../controllers/indexController");
|
||||||
|
|
||||||
const indexRouter = Router();
|
const indexRouter = Router();
|
||||||
|
|
||||||
indexRouter.get("/", (req, res) => {
|
indexRouter.get("/", indexController.indexGet);
|
||||||
res.send("index");
|
//indexRouter.post("/add")
|
||||||
});
|
indexRouter.get("/store", indexController.storeGet);
|
||||||
|
indexRouter.get("/stores", indexController.storesGetAll);
|
||||||
|
indexRouter.post("/store", indexController.storePost);
|
||||||
|
indexRouter.get("/category", indexController.categoryGet);
|
||||||
|
indexRouter.get("/categories", indexController.categoryGetAll);
|
||||||
|
indexRouter.post("/category", indexController.categoryPost);
|
||||||
|
indexRouter.delete("/category", indexController.categoryDelete);
|
||||||
|
indexRouter.post("/item", indexController.itemPost);
|
||||||
|
indexRouter.get("/item/:id", indexController.itemEditGet);
|
||||||
|
indexRouter.post("/item/:id", indexController.itemEditPost);
|
||||||
|
indexRouter.delete("/item", indexController.itemDelete);
|
||||||
|
|
||||||
module.exports = { indexRouter };
|
module.exports = { indexRouter };
|
||||||
|
|
16
inventory/src/views/addForm.ejs
Normal file
16
inventory/src/views/addForm.ejs
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
<h1><%= type %></h1>
|
||||||
|
<form method="POST" action="/<%= type %>">
|
||||||
|
<div class="container">
|
||||||
|
<div class="form-item">
|
||||||
|
<label for="name">Name</label>
|
||||||
|
<input type="text" name="name" placeholder="" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-item btn-row">
|
||||||
|
<button type="submit">Submit</button>
|
||||||
|
<button type="button">Cancel</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<%- include("card", {items: data}) %>
|
10
inventory/src/views/base.ejs
Normal file
10
inventory/src/views/base.ejs
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
<%- include('header', {links: links}); %>
|
||||||
|
|
||||||
|
<% if (page === "add") { %>
|
||||||
|
<%- include('addForm', {type: type}); %>
|
||||||
|
<% } else if (page === "edit") { %>
|
||||||
|
<%- include('editForm', {item: item, categories: categories, stores: stores}); %>
|
||||||
|
<% } else { %>
|
||||||
|
<%- include('itemTable', {items: items}); %>
|
||||||
|
<% }; %>
|
||||||
|
<%- include('footer'); %>
|
14
inventory/src/views/card.ejs
Normal file
14
inventory/src/views/card.ejs
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
<div class="container">
|
||||||
|
<% items.forEach(item => { %>
|
||||||
|
<div class="container flex">
|
||||||
|
<p class="title">
|
||||||
|
<%= item.name %>
|
||||||
|
</p>
|
||||||
|
<button>Edit</button>
|
||||||
|
<% let itemId; %>
|
||||||
|
<% if (item.category_id) {itemId = item.category_id} else { itemId = item.store_id } %>
|
||||||
|
<button class="delete-btn" data-id="<%= itemId %>">Delete</button>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<% }); %>
|
||||||
|
</div>
|
34
inventory/src/views/editForm.ejs
Normal file
34
inventory/src/views/editForm.ejs
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
|
||||||
|
<form method="POST" action="/item/id">
|
||||||
|
<div class="form-item"><label>Name</label><input name="name" value="<%= item.name %>" /></div>
|
||||||
|
<div class="form-item"><label>Price</label><input name="price" value=<%= item.price %> /></div>
|
||||||
|
<div class="form-item"><label>QTY</label><input name="qty" value=<%= item.qty %> /></div>
|
||||||
|
<div class="form-item">
|
||||||
|
<label>Categories</label>
|
||||||
|
<select class="categories-list" name="category">
|
||||||
|
<% categories.forEach((cat) => { %>
|
||||||
|
<% if (cat.name === item.category) { %>
|
||||||
|
<option name=<%= cat.value %> data-cat-id=<%= cat.category_id %> selected><%= cat.name %></option>
|
||||||
|
<% } else { %>
|
||||||
|
<option name=<%= cat.value %> data-cat-id=<%= cat.category_id%>><%= cat.name %></option>
|
||||||
|
<% }; %>
|
||||||
|
<% }); %>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="form-item">
|
||||||
|
<label>stores</label>
|
||||||
|
|
||||||
|
<select class="stores-list" name="store">
|
||||||
|
<% stores.forEach((cat) => { %>
|
||||||
|
<% if (cat.name === item.store) { %>
|
||||||
|
<option name=<%= cat.name %> data-cat-id=<%= cat.store_id %> selected><%= cat.name %></option>
|
||||||
|
<% } else { %>
|
||||||
|
<option name=<%= cat.name %> data-cat-id=<%= cat.store_id %>><%= cat.name %></option>
|
||||||
|
|
||||||
|
<% } %>
|
||||||
|
<% }); %>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<button type="submit" class="btn">Add Item</button><button>Close</button>
|
||||||
|
</form>
|
||||||
|
|
3
inventory/src/views/footer.ejs
Normal file
3
inventory/src/views/footer.ejs
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
<script type="module" src="/js/script.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
18
inventory/src/views/header.ejs
Normal file
18
inventory/src/views/header.ejs
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
<title><%= pageTitle %></title>
|
||||||
|
<link href="/css/style.css" rel="stylesheet">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<nav class="nav">
|
||||||
|
<ul>
|
||||||
|
|
||||||
|
<% links.forEach(link => { %>
|
||||||
|
<li><a href="<%= link.href %>"><%= link.name %></a></li>
|
||||||
|
|
||||||
|
<% }); %>
|
||||||
|
</ul>
|
||||||
|
</nav>
|
10
inventory/src/views/index.html
Normal file
10
inventory/src/views/index.html
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
<title></title>
|
||||||
|
<link href="/css/style.css" rel="stylesheet" />
|
||||||
|
</head>
|
||||||
|
<body></body>
|
||||||
|
</html>
|
26
inventory/src/views/itemTable.ejs
Normal file
26
inventory/src/views/itemTable.ejs
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
<table class="items">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th scope="col">Name</th>
|
||||||
|
<th scope="col">Price</th>
|
||||||
|
<th scope="col">QTY</th>
|
||||||
|
<th scope="col">Store</th>
|
||||||
|
<th scope="col">Category</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
|
||||||
|
<% items.forEach(item => { %>
|
||||||
|
<tr data-item-id="<%=item.id %>">
|
||||||
|
<td><%= item.name %></td>
|
||||||
|
<td><%= item.price %></td>
|
||||||
|
<td><%= item.qty %></td>
|
||||||
|
<td><%= item.store_name %></td>
|
||||||
|
<td><%= item.category_name %></td>
|
||||||
|
<td class="edit-button">Edit</td>
|
||||||
|
<td class="delete-button" data-id="<%= item.name %>">Delete</td>
|
||||||
|
</tr>
|
||||||
|
<% }); %>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
Loading…
Add table
Reference in a new issue