mirror of
https://gitea.smigz.com/smiggiddy/odin-codeprojects.git
synced 2025-04-05 03:10:57 -04:00
trying to get it right
This commit is contained in:
parent
3fd4ed9ba8
commit
2dca50ff7a
11 changed files with 206 additions and 26 deletions
|
@ -4,6 +4,9 @@
|
||||||
--desktop-line-height: 1;
|
--desktop-line-height: 1;
|
||||||
--mobile-heading-font-size: 4em;
|
--mobile-heading-font-size: 4em;
|
||||||
--mobile-line-height: 0.8;
|
--mobile-line-height: 0.8;
|
||||||
|
--border-radius-sm: 0.5em;
|
||||||
|
--border-radius-md: 0.7em;
|
||||||
|
--border-radius-lg: 1em;
|
||||||
|
|
||||||
/* colors */
|
/* colors */
|
||||||
|
|
||||||
|
@ -35,6 +38,9 @@ body {
|
||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
font-family: "Lato, serif";
|
font-family: "Lato, serif";
|
||||||
background: var(--100-blue);
|
background: var(--100-blue);
|
||||||
|
min-height: 100vh;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 5. Improve media defaults */
|
/* 5. Improve media defaults */
|
||||||
|
@ -99,6 +105,8 @@ nav {
|
||||||
|
|
||||||
nav > ul {
|
nav > ul {
|
||||||
list-style: none;
|
list-style: none;
|
||||||
|
text-align: center;
|
||||||
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
ul {
|
ul {
|
||||||
|
@ -127,12 +135,13 @@ input {
|
||||||
}
|
}
|
||||||
|
|
||||||
footer {
|
footer {
|
||||||
justify-content: center;
|
margin-top: auto;
|
||||||
|
padding: 2em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.full-page {
|
/*.full-page {*/
|
||||||
height: 90vh;
|
/* height: 90vh;*/
|
||||||
}
|
/*}*/
|
||||||
|
|
||||||
.divider {
|
.divider {
|
||||||
height: 0.25em;
|
height: 0.25em;
|
||||||
|
@ -290,6 +299,10 @@ footer {
|
||||||
|
|
||||||
.card__likes {
|
.card__likes {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timestamp {
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -342,10 +355,24 @@ footer {
|
||||||
transform-origin: bottom left;
|
transform-origin: bottom left;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.hidden {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.social-icons {
|
||||||
|
gap: 0.5em;
|
||||||
|
justify-content: baseline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.social-icons a {
|
||||||
|
font-size: 1.2em;
|
||||||
|
color: var(--800-blue);
|
||||||
|
}
|
||||||
|
|
||||||
@media only screen and (min-width: 750px) {
|
@media only screen and (min-width: 750px) {
|
||||||
ul {
|
/*ul {*/
|
||||||
max-width: 60vw;
|
/* max-width: 40vw;*/
|
||||||
}
|
/*}*/
|
||||||
.hero__heading {
|
.hero__heading {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-size: var(--desktop-heading-font-size);
|
font-size: var(--desktop-heading-font-size);
|
||||||
|
@ -381,6 +408,20 @@ footer {
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.nav__profile {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav__hover {
|
||||||
|
position: absolute;
|
||||||
|
background: var(--100-blue);
|
||||||
|
border: 2px solid black;
|
||||||
|
width: 100%;
|
||||||
|
max-width: 150px;
|
||||||
|
border-radius: var(--border-radius-sm);
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
/*.form-item > .btn {*/
|
/*.form-item > .btn {*/
|
||||||
/* max-width: 200px;*/
|
/* max-width: 200px;*/
|
||||||
/*}*/
|
/*}*/
|
||||||
|
|
|
@ -0,0 +1,109 @@
|
||||||
|
function usernameValidation(username) {
|
||||||
|
const errors = [];
|
||||||
|
if (username.length < 5) {
|
||||||
|
errors.push("*username must be at least 8 characters");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (username.includes(" ")) {
|
||||||
|
errors.push("*username must not contain any spaces");
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkStringForNumber(str) {
|
||||||
|
for (i = 0; i < str.length; i++) {
|
||||||
|
if (!isNaN(str[i])) return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkStringForChar(str) {
|
||||||
|
return /[!#%^,\.]/.test(str);
|
||||||
|
}
|
||||||
|
|
||||||
|
function passwordValidation(password) {
|
||||||
|
const errors = [];
|
||||||
|
|
||||||
|
if (password.length < 8) {
|
||||||
|
errors.push("*password must be at least 8 characters");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!checkStringForNumber(password)) {
|
||||||
|
errors.push("*password must include a number.");
|
||||||
|
}
|
||||||
|
if (!checkStringForChar(password)) {
|
||||||
|
errors.push("*password must include a special character. !#%^,.");
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
function formValidator() {
|
||||||
|
const username = document.querySelector('input[name="username"]');
|
||||||
|
const password = document.querySelector('input[name="password"]');
|
||||||
|
const passwordConfirmation = document.querySelector(
|
||||||
|
'input[name="password-confirmation"]',
|
||||||
|
);
|
||||||
|
|
||||||
|
if (username && password && passwordConfirmation) {
|
||||||
|
username.addEventListener("keyup", () => {
|
||||||
|
let errors = usernameValidation(username.value);
|
||||||
|
let errorDiv = username.nextElementSibling;
|
||||||
|
errorDiv.textContent = "";
|
||||||
|
|
||||||
|
if (errors.length > 0) {
|
||||||
|
errors.forEach((err) => {
|
||||||
|
const p = document.createElement("p");
|
||||||
|
p.textContent = err;
|
||||||
|
errorDiv.appendChild(p);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
errorDiv.textContent = "";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
password.addEventListener("keyup", (e) => {
|
||||||
|
let errorDiv = password.nextElementSibling;
|
||||||
|
errorDiv.textContent = "";
|
||||||
|
let errors = passwordValidation(e.target.value);
|
||||||
|
|
||||||
|
if (errors.length > 0) {
|
||||||
|
errors.forEach((err) => {
|
||||||
|
const p = document.createElement("p");
|
||||||
|
p.textContent = err;
|
||||||
|
errorDiv.appendChild(p);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
errorDiv.textContent = "";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
passwordConfirmation.addEventListener("keyup", (e) => {
|
||||||
|
let error = "*passwords do not match!";
|
||||||
|
let errorDiv = passwordConfirmation.nextSibling;
|
||||||
|
|
||||||
|
if (password.value !== e.target.value) {
|
||||||
|
errorDiv.textContent = error;
|
||||||
|
} else {
|
||||||
|
errorDiv.textContent = "";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function navProfileHover() {
|
||||||
|
const profile = document.querySelector(".nav__profile");
|
||||||
|
if (profile) {
|
||||||
|
const profileHover = document.querySelector(".nav__hover");
|
||||||
|
profile.addEventListener("mouseleave", () => {
|
||||||
|
profileHover.classList.add("hidden");
|
||||||
|
});
|
||||||
|
profile.addEventListener("mouseenter", () => {
|
||||||
|
profileHover.classList.remove("hidden");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
navProfileHover();
|
||||||
|
formValidator();
|
|
@ -66,11 +66,13 @@ function getProfile(req, res, next) {
|
||||||
|
|
||||||
if (res.locals.currentUser) {
|
if (res.locals.currentUser) {
|
||||||
if (userExists) {
|
if (userExists) {
|
||||||
const sumUserPosts = db.getAllNotesByUserId(userId);
|
const sumUserPosts = db.getSumNotesByUserId(userId);
|
||||||
|
const allPosts = db.getAllNotesByUserId(userId);
|
||||||
res.render("view-profile", {
|
res.render("view-profile", {
|
||||||
pageTitle: "InspiredCliches | Edit Profile",
|
pageTitle: "InspiredCliches | Edit Profile",
|
||||||
user: userExists,
|
user: userExists,
|
||||||
totalPosts: sumUserPosts,
|
totalPosts: sumUserPosts,
|
||||||
|
allPosts: allPosts,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
res.redirect("/");
|
res.redirect("/");
|
||||||
|
|
|
@ -31,7 +31,7 @@ INSERT OR IGNORE INTO users (username, password)
|
||||||
VALUES ("jim", "$2a$10$Bmjre5WSpSSAi.nWBfLZFOlhQhbIAoY/MM7ikJz3Ho9tqeXCExaB6"),
|
VALUES ("jim", "$2a$10$Bmjre5WSpSSAi.nWBfLZFOlhQhbIAoY/MM7ikJz3Ho9tqeXCExaB6"),
|
||||||
("demo", "$2a$10$Vo.XmsVQVx9gGojRIdewpOap5SnhrgZ21/Im5IxhH3PC4FycM5uwC");
|
("demo", "$2a$10$Vo.XmsVQVx9gGojRIdewpOap5SnhrgZ21/Im5IxhH3PC4FycM5uwC");
|
||||||
|
|
||||||
INSERT OR REPLACE INTO messages (message_id, message, media, date, user_id)
|
INSERT OR IGNORE INTO messages (message_id, message, media, date, user_id)
|
||||||
VALUES
|
VALUES
|
||||||
(1, "When life gives you lemons, make lemonade.", NULL, strftime('%s', 'now')*1000, 1),
|
(1, "When life gives you lemons, make lemonade.", NULL, strftime('%s', 'now')*1000, 1),
|
||||||
(2, "Rome wasn’t built in a day – take your time.", NULL, strftime('%s', 'now')*1000, 1),
|
(2, "Rome wasn’t built in a day – take your time.", NULL, strftime('%s', 'now')*1000, 1),
|
||||||
|
|
|
@ -108,12 +108,18 @@ function getUserById(userId) {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getAllNotesByUserId(userId) {
|
function getSumNotesByUserId(userId) {
|
||||||
const query = db.query(`SELECT COUNT(1) FROM messages WHERE user_id = $1`);
|
const query = db.query(`SELECT COUNT(1) FROM messages WHERE user_id = $1`);
|
||||||
const res = query.get({ $1: userId });
|
const res = query.get({ $1: userId });
|
||||||
|
|
||||||
return res["COUNT(1)"];
|
return res["COUNT(1)"];
|
||||||
}
|
}
|
||||||
|
function getAllNotesByUserId(userId) {
|
||||||
|
const query = db.query(`SELECT * FROM messages WHERE user_id = $1`);
|
||||||
|
const res = query.all({ $1: userId });
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
getEveryNote,
|
getEveryNote,
|
||||||
|
@ -127,5 +133,6 @@ module.exports = {
|
||||||
getLikesByUser,
|
getLikesByUser,
|
||||||
getTotalLikesByMessageId,
|
getTotalLikesByMessageId,
|
||||||
getUserById,
|
getUserById,
|
||||||
|
getSumNotesByUserId,
|
||||||
getAllNotesByUserId,
|
getAllNotesByUserId,
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,7 +1,4 @@
|
||||||
<%- include('header') %>
|
<%- include('header') %>
|
||||||
<h1 class="feed-heading">
|
|
||||||
<%= currentUser.username %>'s <span class="normal">feed of cliche's</span>
|
|
||||||
</h1>
|
|
||||||
<div class="grid">
|
<div class="grid">
|
||||||
<% keynotes.forEach((note) => { %>
|
<% keynotes.forEach((note) => { %>
|
||||||
<% const likedPost = userLikedPosts.find((post) => post.message_id == note.message_id) %>
|
<% const likedPost = userLikedPosts.find((post) => post.message_id == note.message_id) %>
|
||||||
|
@ -15,7 +12,7 @@
|
||||||
<div class="flex author__meta">
|
<div class="flex author__meta">
|
||||||
<p class="card__author"><a href="/profile?userId=<%= note.user_id %>"><%= note.user %></a></p>
|
<p class="card__author"><a href="/profile?userId=<%= note.user_id %>"><%= note.user %></a></p>
|
||||||
<p class="card__likes">•</p>
|
<p class="card__likes">•</p>
|
||||||
<p class="card__likes"><%= new Date(note.date).toLocaleString() %></p>
|
<p class="card__likes timestamp"><%= new Date(note.date).toLocaleString() %></p>
|
||||||
</div>
|
</div>
|
||||||
<% if (currentUser.user_id === note.user_id) { %>
|
<% if (currentUser.user_id === note.user_id) { %>
|
||||||
<a href="/delete?userId=<%= currentUser.user_id %>&messageId=<%= note.message_id %>">delete post</a>
|
<a href="/delete?userId=<%= currentUser.user_id %>&messageId=<%= note.message_id %>">delete post</a>
|
||||||
|
|
|
@ -1,5 +1,11 @@
|
||||||
<footer class="footer flex">
|
<footer class="container">
|
||||||
Test
|
<h2 class="foooter-heading">Inspired Cliches</h2>
|
||||||
|
<p>Making unsolicited advice inspiring.</p>
|
||||||
|
<div class="social-icons flex">
|
||||||
|
<a href="https://github.com/smiggiddy/odin-codeprojects" target="_blank"><i class="fa-brands fa-github"></i></a>
|
||||||
|
<a href="https://smig.tech" target="_blank"><i class="fa-solid fa-link"></i></a>
|
||||||
|
</div>
|
||||||
|
<p> © Copyright <%= new Date(Date.now()).getFullYear() %></p>
|
||||||
</footer>
|
</footer>
|
||||||
<script src="/js/script.js"></script>
|
<script src="/js/script.js"></script>
|
||||||
</body>
|
</body>
|
||||||
|
|
|
@ -18,9 +18,15 @@
|
||||||
<ul class="container flex">
|
<ul class="container flex">
|
||||||
<li><a class="hover__underline" href="/">Home</a></li>
|
<li><a class="hover__underline" href="/">Home</a></li>
|
||||||
<% if (currentUser) { %>
|
<% if (currentUser) { %>
|
||||||
<li><a class="hover__underline" href="/new" class="new-link">New Cliche</a></li>
|
<li class="nav__profile"><%= currentUser.username %>
|
||||||
<li><a class="hover__underline" href="/profile?userId=<%= currentUser.user_id %>">Profile</a></li>
|
<div class="hidden nav__hover">
|
||||||
<li><a class="hover__underline" href="/auth/logout">Logout</a></li>
|
<ul class="container flex col">
|
||||||
|
<li><a class="hover__underline" href="/profile?userId=<%= currentUser.user_id %>">view profile</a></li>
|
||||||
|
<li><a class="hover__underline" href="/new" class="new-link">new cliche</a></li>
|
||||||
|
<li><a class="hover__underline" href="/auth/logout">logout</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
<% } else { %>
|
<% } else { %>
|
||||||
<li><a class="hover__underline" href="/auth/login">Login</a></li>
|
<li><a class="hover__underline" href="/auth/login">Login</a></li>
|
||||||
<% }; %>
|
<% }; %>
|
||||||
|
|
|
@ -21,7 +21,9 @@
|
||||||
<div class="form-item-row flex">
|
<div class="form-item-row flex">
|
||||||
<button type="submit" class="btn btn__form">Login</button>
|
<button type="submit" class="btn btn__form">Login</button>
|
||||||
</div>
|
</div>
|
||||||
<a href="/auth/register">Need an account? Click here!</a>
|
<p>Need an account?
|
||||||
|
<a href="/auth/register">Click here!</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
</form>
|
</form>
|
||||||
</main>
|
</main>
|
||||||
|
|
|
@ -14,11 +14,12 @@
|
||||||
<div class="form-item flex">
|
<div class="form-item flex">
|
||||||
<label for="username">username</label>
|
<label for="username">username</label>
|
||||||
<input type="text" name="username" required/>
|
<input type="text" name="username" required/>
|
||||||
|
<div class="error"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-item flex">
|
<div class="form-item flex">
|
||||||
<label for="password">password</label>
|
<label for="password">password</label>
|
||||||
<input type="password" name="password" required/>
|
<input type="password" name="password" required/>
|
||||||
<div class="error"></div>
|
<div class="error flex col"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-item flex">
|
<div class="form-item flex">
|
||||||
<label for="password-confirmation">confirm password</label>
|
<label for="password-confirmation">confirm password</label>
|
||||||
|
|
|
@ -1,11 +1,20 @@
|
||||||
<%- include("header") %>
|
<%- include("header") %>
|
||||||
|
|
||||||
<div class="full-page container center-horizontal">
|
<div class="full-page container center-horizontal">
|
||||||
<h1 class="username"><%= user.username %></h1>
|
<div class="container">
|
||||||
<% if(currentUser.user_id === user.user_id ) { %>
|
<h1 class="username"><%= user.username %></h1>
|
||||||
<p>This is your profile page.</p>
|
<% if(currentUser.user_id === user.user_id ) { %>
|
||||||
<% }; %>
|
<p>This is your profile page.</p>
|
||||||
<p><%= totalPosts %> Posts</p>
|
<% }; %>
|
||||||
|
</div>
|
||||||
|
<div class="container">
|
||||||
|
<h2><%= user.username %>'s <%= totalPosts %> post(s): </h2>
|
||||||
|
<% allPosts.forEach((post) => { %>
|
||||||
|
<div class="container">
|
||||||
|
<p>"<%= post.message %>"</p>
|
||||||
|
</div>
|
||||||
|
<% }); %>
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<%- include("footer") %>
|
<%- include("footer") %>
|
||||||
|
|
Loading…
Add table
Reference in a new issue