mirror of
https://gitea.smigz.com/smiggiddy/odin-codeprojects.git
synced 2025-04-05 03:10:57 -04:00
feat: basic crud complete
This commit is contained in:
parent
6fb88b315a
commit
77a93ae9b1
10 changed files with 105 additions and 27 deletions
|
@ -68,6 +68,21 @@ h6 {
|
||||||
isolation: isolate;
|
isolation: isolate;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* remove dumb stuff
|
||||||
|
*/
|
||||||
|
/* Chrome, Safari, Edge, Opera */
|
||||||
|
input::-webkit-outer-spin-button,
|
||||||
|
input::-webkit-inner-spin-button {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Firefox */
|
||||||
|
input[type="number"] {
|
||||||
|
-moz-appearance: textfield;
|
||||||
|
}
|
||||||
|
|
||||||
li {
|
li {
|
||||||
list-style-type: none;
|
list-style-type: none;
|
||||||
}
|
}
|
||||||
|
@ -106,6 +121,10 @@ nav {
|
||||||
font-size: 1.5rem;
|
font-size: 1.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 800px;
|
max-width: 800px;
|
||||||
|
@ -118,7 +137,7 @@ nav {
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
padding: 0.5em 1em;
|
padding: 0.5em 1em;
|
||||||
margin: 1em;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.container.flex {
|
.container.flex {
|
||||||
|
@ -137,10 +156,12 @@ nav {
|
||||||
.items-table {
|
.items-table {
|
||||||
width: 80%;
|
width: 80%;
|
||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
|
table-layout: fixed;
|
||||||
}
|
}
|
||||||
|
|
||||||
.form-item {
|
.form-item {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
.form-item.btn-row {
|
.form-item.btn-row {
|
||||||
|
|
|
@ -1,14 +1,17 @@
|
||||||
function addForm() {
|
function addForm() {
|
||||||
let buttonClicked = true;
|
let buttonClicked = true;
|
||||||
const addForm = document.querySelector(".add-form");
|
const addFormClass = document.querySelector(".add-form");
|
||||||
|
|
||||||
|
if (addFormClass != null) {
|
||||||
const button = document.querySelector(".add-btn");
|
const button = document.querySelector(".add-btn");
|
||||||
|
|
||||||
button.addEventListener("click", () => {
|
button.addEventListener("click", () => {
|
||||||
buttonClicked
|
buttonClicked
|
||||||
? (addForm.style.display = " block")
|
? (addFormClass.style.display = " block")
|
||||||
: (addForm.style.display = "none");
|
: (addFormClass.style.display = "none");
|
||||||
buttonClicked = !buttonClicked;
|
buttonClicked = !buttonClicked;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export { addForm };
|
export { addForm };
|
||||||
|
|
|
@ -5,6 +5,9 @@ function formInput(props) {
|
||||||
formItemDiv.classList.add("form-item");
|
formItemDiv.classList.add("form-item");
|
||||||
const input = document.createElement("input");
|
const input = document.createElement("input");
|
||||||
input.name = props.name;
|
input.name = props.name;
|
||||||
|
input.required = true;
|
||||||
|
input.type = props.type;
|
||||||
|
if (props.type === "number") input.step = "0.01";
|
||||||
const label = document.createElement("label");
|
const label = document.createElement("label");
|
||||||
label.textContent = props.labelText;
|
label.textContent = props.labelText;
|
||||||
formItemDiv.append(label, input);
|
formItemDiv.append(label, input);
|
||||||
|
@ -17,23 +20,30 @@ async function modal() {
|
||||||
div.classList.add("modal");
|
div.classList.add("modal");
|
||||||
div.style = `
|
div.style = `
|
||||||
display: none;
|
display: none;
|
||||||
positino: fixed;
|
position: fixed;
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: 0;
|
top: 0;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 50vh;
|
height: 100vh;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
background-color: rgb(0,0,0);
|
background-color: rgb(0,0,0);
|
||||||
background-color: rgba(0,0,0,0.4);
|
background-color: rgba(0,0,0,0.4);
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
backdrop-filter: blur(15px);
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const form = document.createElement("form");
|
const form = document.createElement("form");
|
||||||
form.method = "POST";
|
form.method = "POST";
|
||||||
form.action = "/item";
|
form.action = "/item";
|
||||||
const name = formInput({ labelText: "Name", name: "name" });
|
const name = formInput({ labelText: "name", name: "name", type: "text" });
|
||||||
const price = formInput({ labelText: "Price", name: "price" });
|
const price = formInput({
|
||||||
const qty = formInput({ labelText: "QTY", name: "qty" });
|
labelText: "price",
|
||||||
|
name: "price",
|
||||||
|
type: "number",
|
||||||
|
});
|
||||||
|
const qty = formInput({ labelText: "qty", name: "qty", type: "number" });
|
||||||
|
|
||||||
const categories = await categoryPicker();
|
const categories = await categoryPicker();
|
||||||
const stores = await storePicker();
|
const stores = await storePicker();
|
||||||
|
@ -44,12 +54,19 @@ async function modal() {
|
||||||
submitButton.textContent = "Add Item";
|
submitButton.textContent = "Add Item";
|
||||||
|
|
||||||
const closeButton = document.createElement("button");
|
const closeButton = document.createElement("button");
|
||||||
|
closeButton.classList.add("btn");
|
||||||
closeButton.textContent = "Close";
|
closeButton.textContent = "Close";
|
||||||
|
closeButton.type = "button";
|
||||||
closeButton.addEventListener("click", () => {
|
closeButton.addEventListener("click", () => {
|
||||||
div.style.display = "none";
|
div.style.display = "none";
|
||||||
});
|
});
|
||||||
|
|
||||||
form.append(name, price, qty, categories, stores, submitButton, closeButton);
|
const btns = document.createElement("div");
|
||||||
|
btns.classList.add("form-item");
|
||||||
|
btns.style.flexDirection = "row";
|
||||||
|
btns.append(submitButton, closeButton);
|
||||||
|
|
||||||
|
form.append(name, price, qty, categories, stores, btns);
|
||||||
div.append(form);
|
div.append(form);
|
||||||
|
|
||||||
return div;
|
return div;
|
||||||
|
|
|
@ -22,7 +22,7 @@ async function categoryPicker() {
|
||||||
formItemDiv.classList.add("form-item");
|
formItemDiv.classList.add("form-item");
|
||||||
|
|
||||||
const formLabel = document.createElement("label");
|
const formLabel = document.createElement("label");
|
||||||
formLabel.textContent = "Categories";
|
formLabel.textContent = "categories";
|
||||||
|
|
||||||
const select = document.createElement("select");
|
const select = document.createElement("select");
|
||||||
select.classList.add("categories-list-modal");
|
select.classList.add("categories-list-modal");
|
||||||
|
|
|
@ -4,6 +4,7 @@ import { addForm } from "./components/addForm.js";
|
||||||
const newItemLink = document.querySelector("a[href='/add']");
|
const newItemLink = document.querySelector("a[href='/add']");
|
||||||
const modalElement = await modal();
|
const modalElement = await modal();
|
||||||
let modalToggle = false;
|
let modalToggle = false;
|
||||||
|
|
||||||
async function handleDelete(e) {
|
async function handleDelete(e) {
|
||||||
console.log(e.target);
|
console.log(e.target);
|
||||||
}
|
}
|
||||||
|
@ -55,7 +56,13 @@ newItemLink.addEventListener("click", (e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
modalToggle
|
modalToggle
|
||||||
? (modalElement.style.display = "none")
|
? (modalElement.style.display = "none")
|
||||||
: (modalElement.style.display = "block");
|
: (modalElement.style.display = "flex");
|
||||||
|
modalToggle = !modalToggle;
|
||||||
|
});
|
||||||
|
|
||||||
|
modalElement.addEventListener("click", (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
if (e.target === e.currentTarget) modalElement.style.display = "none";
|
||||||
modalToggle = !modalToggle;
|
modalToggle = !modalToggle;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -122,13 +122,25 @@ async function itemEditGet(req, res, next) {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function itemEditPost(req, res, next) {
|
async function itemEditPost(req, res, next) {
|
||||||
const itemId = req.params.id;
|
const { id, name, qty, price, category, store } = req.body;
|
||||||
console.log(itemId);
|
const category_id = await db.getCategoryId({ name: category });
|
||||||
|
const store_id = await db.getStoreId({ name: store });
|
||||||
|
|
||||||
|
const update = {
|
||||||
|
id: id,
|
||||||
|
name: name,
|
||||||
|
qty: qty,
|
||||||
|
price: utils.convertToInteger(price),
|
||||||
|
category_id: category_id.category_id,
|
||||||
|
store_id: store_id.store_id,
|
||||||
|
};
|
||||||
|
|
||||||
|
const dbRes = await db.updateItem(update);
|
||||||
|
|
||||||
res.redirect("/");
|
res.redirect("/");
|
||||||
}
|
}
|
||||||
|
|
||||||
async function itemDelete(req, res) {
|
async function itemDelete(req, res) {
|
||||||
console.log(req.body.name);
|
|
||||||
const itemId = await db.getItemByName(req.body.name);
|
const itemId = await db.getItemByName(req.body.name);
|
||||||
if (itemId) db.deleteItem(itemId);
|
if (itemId) db.deleteItem(itemId);
|
||||||
console.log(`Delted item: ${itemId}`);
|
console.log(`Delted item: ${itemId}`);
|
||||||
|
|
|
@ -33,6 +33,22 @@ async function insertItem(data) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function updateItem(item) {
|
||||||
|
const { id, name, price, qty, category_id, store_id } = item;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await pool.query(
|
||||||
|
`UPDATE items
|
||||||
|
SET name=$1, price=$2, qty=$3, category_id=$4, store_id=$5
|
||||||
|
WHERE id=$6`,
|
||||||
|
[name, price, qty, category_id, store_id, id],
|
||||||
|
);
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function deleteItem(item) {
|
async function deleteItem(item) {
|
||||||
try {
|
try {
|
||||||
const res = await pool.query("DELETE FROM items WHERE id = $1", [item.id]);
|
const res = await pool.query("DELETE FROM items WHERE id = $1", [item.id]);
|
||||||
|
@ -46,7 +62,7 @@ async function getAllItemsWithRelationships() {
|
||||||
const { rows } = await pool.query(
|
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
|
`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 category ON items.category_id = category.category_id
|
||||||
LEFT JOIN store ON items.store_id = store.store_id;`,
|
LEFT JOIN store ON items.store_id = store.store_id ORDER BY items.id;`,
|
||||||
);
|
);
|
||||||
|
|
||||||
return rows;
|
return rows;
|
||||||
|
@ -117,6 +133,7 @@ module.exports = {
|
||||||
getAllItems,
|
getAllItems,
|
||||||
getItemByName,
|
getItemByName,
|
||||||
deleteItem,
|
deleteItem,
|
||||||
|
updateItem,
|
||||||
getCategories,
|
getCategories,
|
||||||
getCategoryId,
|
getCategoryId,
|
||||||
getStores,
|
getStores,
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
<h1 class="heading"><%= type %></h1>
|
<h1 class="heading"><%= type %></h1>
|
||||||
|
|
||||||
|
<div class="container">
|
||||||
<%- include("card", {items: data}) %>
|
<%- include("card", {items: data}) %>
|
||||||
<button class="btn add-btn">New <%= type %></button>
|
<button class="btn add-btn">New <%= type %></button>
|
||||||
|
</div>
|
||||||
<form method="POST" action="/<%= type %>" class="add-form">
|
<form method="POST" action="/<%= type %>" class="add-form">
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<div class="form-item">
|
<div class="form-item">
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
<div class="container">
|
|
||||||
<% items.forEach(item => { %>
|
<% items.forEach(item => { %>
|
||||||
<div class="container flex">
|
<div class="container flex">
|
||||||
<p class="title">
|
<p class="title">
|
||||||
|
@ -11,4 +10,3 @@
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<% }); %>
|
<% }); %>
|
||||||
</div>
|
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
|
|
||||||
<form method="POST" action="/item/id">
|
<form method="POST" action="/item/id">
|
||||||
<div class="form-item"><label>Name</label><input name="name" value="<%= item.name %>" /></div>
|
<input type="hidden" value=<%= item.id %> name="id">
|
||||||
<div class="form-item"><label>Price</label><input name="price" value=<%= item.price %> /></div>
|
<div class="form-item"><label>Name</label><input name="name" value="<%= item.name %>" required /></div>
|
||||||
<div class="form-item"><label>QTY</label><input name="qty" value=<%= item.qty %> /></div>
|
<div class="form-item"><label>Price</label><input name="price" value=<%= (item.price/100).toFixed(2) %> type="number" step="0.01" required /></div>
|
||||||
|
<div class="form-item"><label>QTY</label><input name="qty" value=<%= item.qty %> type="number" required /></div>
|
||||||
<div class="form-item">
|
<div class="form-item">
|
||||||
<label>Categories</label>
|
<label>Categories</label>
|
||||||
<select class="categories-list" name="category">
|
<select class="categories-list" name="category">
|
||||||
|
@ -29,6 +30,6 @@
|
||||||
<% }); %>
|
<% }); %>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<button type="submit" class="btn">Add Item</button><button>Close</button>
|
<button type="submit" class="btn">UPDATE ITEM</button><button class="btn">RETURN</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue