Skip to content

buildingu/bu-front

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

85 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

bu-front

Welcome interns! This is the official repository for building-U's new website(web app), here I'll explain the guidelines you'll be following to ensure we stay organized and consistent across the project.

Technologies

This is a React application built using Vite and utilizes CSS modules for styling. NPM is our package manager, and react-router-dom is also used for routing.

Vite

We are using Vite instead of create-react-app. I don't need to go into to much detail about Vite because you wouldn't need to worry about it, but Vite is a modern build tool that makes the development process faster and more efficient. It provides quicker build times and faster updates when you make changes to your code through Vite's fast hot module replacement (HMR), which all helps to improve the development experience.

Styling

CSS Modules vs CSS

CSS Modules might be new to many of you, so here's a brief explanation. CSS Modules provide a way to 'encapsulate' a CSS file so that the styles defined in it are scoped locally to the component that imports it. This ensures there aren't any style conflicts throughout the app, as CSS classes from one module won't affect those in another module. This way, even if two CSS Modules define the same class name, they won't interfere with each other.

How to use CSS Modules

// CSS Modules
import s from "./styles.module.css";

const Component = () => {
  return <div className={s.container}>Component</div>;
};

//-------------------------------------------------

// CSS (Global CSS)
import "./styles.css";

const Component = () => {
  return <div className="container">Component</div>;
};

Use of Global CSS in the Project

The global CSS is defined in the index.css file at the base of the src directory. Here, we'll define our default CSS properties and CSS variables, such as colors and typography, etc. You'll use these variables in your CSS Module files to maintain consistency across the project. If you find a property value used across multiple files(not just your page files), add it as a CSS variable in index.css.

Here's an example of how to use CSS variables:

import s from "./styles.module.css";

const Component = () => {
  return (
    <div className={s.container}>
      <h2>Heading</h2>
    </div>
  );
};
/* styles.module.css */

.container {
  /* properties... */
}
.container h2 {
  font-family: var(--heading-font1); /* Heading Font 1 */
  color: var(--c-heading-w); /* White Heading */
}

NPM

NPM (Node Package Manager) is a tool that helps you manage packages(also called dependencies) and run scripts for your project. The package.json file lists the packages your project depends on and includes script commands like dev that you can run. The package-lock.json file keeps track of the exact versions of the packages installed.

Whenever you need to add a package or run a command, you'll work in the package.json file. You generally don't need to worry about the package-lock.json file; it just ensures that everyone working on the project has the same package versions installed.

You can go to https://www.npmjs.com/, to search for other packages from developers if you're curious.

react-router-dom and SSR

Our router is react-router-dom, which helps us set up our routes(pages, e.g., /let-talk) and interact with the router. For a better look, here is the official documentation.

Our Custom History Object

In the react-router-dom documentation, you'll find a hook called useNavigate used for when you want to navigate to a page programmatically. Instead of using this hook, we use our custom history object, located here. This utility allows us to navigate both inside and outside of components. For consistency, use this custom history object in your pages/components and everywhere else.

How you would normally navigate with react-router-dom:

import { useNavigate } from "react-router-dom";

const Whatever = () => {
  const navigate = useNavigate()

  useEffect(() => {
    // Navigates to /error on page load.
    navigate("/error")
  }, [])
  
  return (
    <div>Whatever</div>
  );
};

How we navigate with our custom history object:

import { history } from "../../utils/History";

const Whatever = () => {
  useEffect(() => {
    // Navigates to /error on page load.
    history.push("/error")
  }, [])
  
  return (
    <div>Whatever</div>
  );
};

You can find more information about useNavigate in the docs, but you won't be using it: https://reactrouter.com/en/main/hooks/use-navigate

Link Component

When we want to navigate programmatically(with a function), we use the custom history object as I mentioned above. For link-based navigation, you might think to use the <a> tag, but this will refresh the page, as it treats the link as a traditional HTML link(to another HTML file). Instead, we use the <Link> component to navigate without causing a refresh!

We'll use links like this:

import { Link } from "react-router-dom";

const Whatever = () => {
  return (
    <div>
      <Link to="/error">Example</Link>
    </div>
  );
};

Link component in the docs: https://reactrouter.com/en/main/components/link

Server Side Rendering (SSR)

In our project, the server, which is server.js, renders the JSX on the server side, not on the client side. FYI, you don't need to know all of this, but you should be familiar with it. So, in a normal React project, they're single-page applications (SPAs), where there is just one HTML file, and the entire React project is in a script tag. That isn't ideal for this project, since this is a real-world project on the internet, we should care about Search Engine optimization (SEO). SSR helps improve SEO because the initial HTML rendered by the server is more accessible to search engines, this would make search engines and their bots and crawlers or whatever actually understand our site.

Our project still uses one HTML file like a SPA, but since the server pre-renders the JSX and includes it in the initial HTML (the index.html file), all pages are shown dynamically in the HTML.

The main things you have to worry about is ensuring that the server-rendered HTML matches the client-rendered HTML on load to prevent differences. If there are discrepancies, it will cause hydration issues, which defeats the purpose of SSR. React will throw an error and tell us it switched to client-side rendering from now on, which is what we want to avoid. All you need to know is below.

A common example of how a difference between server and client HTML can occur:

import { useState } from "react";

const Whatever = () => {
  const [show, setShow] = useState(true)

  return (
    <div>
      <button onClick={() => setShow(false)}>Hide</button>
      {show && <div>Showing</div>}
    </div>
  );
};

In the example above a difference error will happen since useState is only a client-side hook. The server doesn't run JavaScript like the useState, it just renders the initial HTML. So, the server wouldn't know about the conditional rendering of the <div> element based on the show state, since the initial state is true, thus causing a difference in the server-rendered HTML and the client-rendered HTML.

How to fix this difference:

import { useState, useEffect } from "react";

const Whatever = () => {
  const [show, setShow] = useState(false)

  useEffect(() => {
    setShow(true);
  }, [])
  
  return (
    <div>
      <button onClick={() => setShow(false)}>Hide</button>
      {show && <div>Showing</div>}
    </div>
  );
};

In the example above, the difference error is fixed by using a useEffect with an empty dependency array [], the useEffect will only run when the client loads. As a result, the useEffect runs after the server renders, making the server and client HTML match since the <div> is shown only on the client.

Other important considerations:
Since our server is rendering our JSX initially, we have to be cautious when using client-specific things like the document, window, etc, on page load.

const Whatever = () => {
  const button = document.querySelector("button") // Will cause an error since document is undefined on the server.

  return (
    <div>
      <button>Hide</button>
    </div>
  );
};

How to fix the undefined error:

import { useRef } from "react";

const Whatever = () => {
  const buttonRef = useRef(null); // Use a useRef since it is a part of the react life-cycle.

  const someFunc = () => {
    // You can use it just like a querySelector or getElementById, etc. But you have to use .current...
    buttonRef.current.style.color = "red";
  };

  return (
    <div>
      <button ref={buttonRef} onClick={someFunc}>Hide</button>
    </div>
  );
};

By using a useRef, you can safely interact with DOM elements without causing errors during server-side rendering.

Project Architecture

In this project, we will be using a reusable architecture for all directories/files. This means creating a new component only if it is used in more than one page directory or component. The same rule applies to utils, hooks, constants, and other directories. This approach ensures that everything stays modular and organized.

So, for components, utilities, etc, that are only used within a single page directory, define them as a file within that page directory or within the page file itself.

In your learning lessons, you might not have used ES Modules (ESM); instead, you probably used CommonJS, which is the default module format in Node.js. ESM is the modern JavaScript module format and has several advantages over CommonJS.

CommonJs vs ESM

// logger.js (CommonJS)
const logger = (message) => {
  console.log(message);
};

module.exports = logger;

//-------------------------------------------------

// main.js (CommonJS)
const logger = require('./logger');

logger('Hello, World!');
// logger.js (ESM)
export const logger = (message) => {
  console.log(message);
};

//-------------------------------------------------

// main.js (ESM)
import { logger } from './logger.js';

logger('Hello, World!');

Directory Structure

...TODO

Getting Started

Prerequisites

  • Nodejs version 20 or greater.
  • NPM version 10 or greater.

Steps

  1. To first get started, install all the packages listed in the package.json by doing an npm install.
  2. While still in the main branch, create a file called .env.development at the base of the project(the same level as package.json). We need this file because our SSR server depends on certain environment variables, like NODE_ENV to determine the environment mode. For development, we'll use .env.development. So, there is an example.env.development file to show you what variables to set. Just copy and paste the contents in the example.env.development and put that in the .env.development you created. The variable PORT=, after the equals, set it to 3000 and there you go!
  3. Now you'll have to create your team's branch. Since we are working by pages, you'll create your branch by page, for example, page/home. When you create your branch, make sure you're on your branch. Then run the dev command to start the development environment.
$ npm install
$ npm run dev

Running npm run dev would run the project in development mode.

$ npm run dev

installing a package (you most likely don't have to do this):

$ npm install <package>

Your Page and Branch

...TODO

If you provide nothing to the Layout, it will default to the 'default' orange header and the showFooter prop defaults to true.

The Layout Component How to:

import { Layout } from "../../components/layout";

import s from "./styles.module.css";

const DifferentHeader = () => {
  return (
    <header>DifferentHeader</div>
  )
}


const YourPage = () => {
  return (
    <Layout Header={DifferentHeader}>
      <main className={s.container}>
        {/* Content */}
      </main>
    </Layout>
  );
};

export default Home;

//-------------------------------------------------
// If you have props to give to your header for whatever reason.

const YourPage = () => {
  return (
    <Layout Header={() => <DifferentHeader someProp={something} />}>
      <main className={s.container}>
        {/* Content */}
      </main>
    </Layout>
  );
};

//-------------------------------------------------
// If your page doesn't show a footer or you just don't want to show it.

const YourPage = () => {
  return (
    <Layout Header={DifferentHeader} showFooter={false}>
      <main className={s.container}>
        {/* Content */}
      </main>
    </Layout>
  );
};

Pushing Code

You'll always be working in your page/<page name> branch. When you have completed a good amount or feel that this part of the code is ready, you can push it to the develop branch. The develop branch is where all the current code in development is located. But before you push your code to develop, set up a meeting with me, David Bishop, so I can check it out :)

Pulling New Changes From Main (Mostly likely if I something)

First move to the main branch, pull the changes. Then go back to your page branch and do git rebase main, sometimes you might have to do git rebase origin/main(mostly likely when it's your first time doing it) and then you should have all the changes in your page branch.

Helpful Resources

https://www.freecodecamp.org/news/intro-to-react-components/ - React vs Static HTML

https://www.semrush.com/blog/semantic-html5-guide/ - How to be Semantic With Your HTML(JSX)

https://www.freecodecamp.org/news/full-guide-to-react-hooks/ - Common Hook Guide (You don't need to use all of them shown there)

https://www.dreamhost.com/blog/environment-variables/ - What are .env environment variables?

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Contributors 4

  •  
  •  
  •  
  •