mirror of
https://github.com/UofCBaja/BajaUofCWebsite.git
synced 2025-06-15 21:34:17 -06:00
Merge branch 'gallery' of github.com:UofCBaja/BajaUofCWebsite into ClubMemEvent
This commit is contained in:
commit
e4d34021d2
11162
package-lock.json
generated
11162
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -9,9 +9,10 @@
|
||||
"@testing-library/user-event": "^13.5.0",
|
||||
"react": "^18.2.0",
|
||||
"react-dom": "^18.2.0",
|
||||
"react-modal": "^3.16.1",
|
||||
"react-native": "^0.72.6",
|
||||
"react-router-dom": "^6.17.0",
|
||||
"react-scripts": "5.0.1",
|
||||
"react-router-dom": "^6.20.0",
|
||||
"react-scripts": "^5.0.1",
|
||||
"web-vitals": "^2.1.4"
|
||||
},
|
||||
"scripts": {
|
||||
|
@ -1,5 +1,16 @@
|
||||
import React from 'react';
|
||||
import ImageSlider from '../Images/ImageSlider'; // Adjust the import path based on your directory structure
|
||||
|
||||
|
||||
export default function AboutsUs() {
|
||||
// return (
|
||||
// <p>About Us</p>
|
||||
// );
|
||||
|
||||
return (
|
||||
<p>About Us</p>
|
||||
<div>
|
||||
<h1>Image Slider</h1>
|
||||
<ImageSlider category="Car" />
|
||||
</div>
|
||||
);
|
||||
};
|
52
src/Gallery/Gallery.css
Normal file
52
src/Gallery/Gallery.css
Normal file
@ -0,0 +1,52 @@
|
||||
.gallery {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.galleryItem {
|
||||
min-width: 10vw;
|
||||
min-height: 10vh;
|
||||
max-width: 60vw;
|
||||
max-height: 60vh;
|
||||
object-fit: cover;
|
||||
transition: transform 0.2s ease-in-out;
|
||||
}
|
||||
|
||||
.galleryItem:hover {
|
||||
transform: scale(1.05);
|
||||
}
|
||||
|
||||
.container {
|
||||
position: relative;
|
||||
width: fit-content;
|
||||
height: fit-content;
|
||||
}
|
||||
|
||||
.modalImage {
|
||||
min-width: 20vw;
|
||||
min-height: 20vh;
|
||||
max-width: 70vw;
|
||||
max-height: 70vh;
|
||||
}
|
||||
|
||||
.modalButton {
|
||||
position: absolute;
|
||||
top: 10%;
|
||||
left: 90%;
|
||||
transform: translate(-50%, -50%);
|
||||
-ms-transform: translate(-50%, -50%);
|
||||
opacity: 0.5;
|
||||
background-color: #555;
|
||||
color: white;
|
||||
font-size: 15px;
|
||||
padding: 15px 20px;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
border-radius: 50%;
|
||||
}
|
||||
|
||||
.modalButton:hover {
|
||||
background-color: whitesmoke;
|
||||
color: black;
|
||||
}
|
@ -1,5 +1,65 @@
|
||||
import Modal from 'react-modal';
|
||||
import { useState } from "react";
|
||||
import './Gallery.css';
|
||||
import { ImageFinder } from '../Images/ImageFinder.js';
|
||||
|
||||
/**
|
||||
* @param {null} null - requires onthing
|
||||
* @returns {JSX.Element} JSX - HTML tags and JS functionality
|
||||
* @description Gallery Page
|
||||
* @author Sarim <sheikhsarim20@gmail.com>
|
||||
* @todo Seperate the Modal section into its own seperate function, and add comments
|
||||
*/
|
||||
export default function Gallery() {
|
||||
const [open, setOpen] = useState(false)
|
||||
const [display, setDisplay] = useState('')
|
||||
const [altDisplay, setAltDisplay] =useState('')
|
||||
const allImages = ImageFinder('all')
|
||||
|
||||
const handleClick = (image, alt) => {
|
||||
setDisplay(image)
|
||||
setAltDisplay(alt)
|
||||
setOpen(true)
|
||||
console.log(open)
|
||||
}
|
||||
|
||||
return (
|
||||
<p>Gallery</p>
|
||||
<div>
|
||||
<div className="gallery">
|
||||
{allImages.map((image, index) => (
|
||||
<img
|
||||
key={index}
|
||||
className="galleryItem"
|
||||
src={image.src}
|
||||
alt={image.alt}
|
||||
onClick={() => handleClick(image.src, image.alt)}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
<div>
|
||||
<Modal
|
||||
isOpen={open}
|
||||
onRequestClose={() => setOpen(false)}
|
||||
style={{
|
||||
overlay: {
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
justifyContent: 'center',
|
||||
},
|
||||
content: {
|
||||
position: 'initial',
|
||||
border: 'none',
|
||||
borderRadius: '0px',
|
||||
padding: '0px'
|
||||
}
|
||||
}}
|
||||
>
|
||||
<div className="container">
|
||||
<button className="modalButton" onClick={() => setOpen(false)}>x</button>
|
||||
<img className="modalImage" src={display} alt={altDisplay} onClick={() => setOpen(false)} />
|
||||
</div>
|
||||
</Modal>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
}
|
11
src/Images/ImageFinder.js
Normal file
11
src/Images/ImageFinder.js
Normal file
@ -0,0 +1,11 @@
|
||||
import imagesData from './Images.json';
|
||||
|
||||
// Function to search and filter images based on the provided category
|
||||
export const ImageFinder = (category) => {
|
||||
if (category === 'all') {
|
||||
return imagesData;
|
||||
} else {
|
||||
const filteredImages = imagesData.filter(image => image.category === category);
|
||||
return filteredImages;
|
||||
}
|
||||
};
|
57
src/Images/ImageSlider.js
Normal file
57
src/Images/ImageSlider.js
Normal file
@ -0,0 +1,57 @@
|
||||
import React, { useState, useEffect } from 'react';
|
||||
import { ImageFinder } from './ImageFinder.js';
|
||||
import RenderImage from './RenderImage.js';
|
||||
|
||||
const ImageSlider = ({ category }) => {
|
||||
// Time Interval before the ImageSlider moves to the next image
|
||||
const timeInterval = 5000;
|
||||
|
||||
// State to hold the images of the specified category and the current index
|
||||
const [categoryImages, setCategoryImages] = useState([]);
|
||||
const [currentIndex, setCurrentIndex] = useState(0);
|
||||
|
||||
// Effect to filter images based on the provided category when it changes
|
||||
useEffect(() => {
|
||||
const filteredImages = ImageFinder(category);
|
||||
setCategoryImages(filteredImages);
|
||||
setCurrentIndex(0);
|
||||
}, [category]);
|
||||
|
||||
// Effect to automatically move to the next image after a certain time interval
|
||||
useEffect(() => {
|
||||
const intervalId = setInterval(() => {handleNext();}, timeInterval);
|
||||
|
||||
// Cleanup: Clear the interval on component unmount or when currentIndex or categoryImages changes
|
||||
return () => clearInterval(intervalId);
|
||||
}, [currentIndex, categoryImages]);
|
||||
|
||||
// Function to handle moving to the next image
|
||||
const handleNext = () => {
|
||||
setCurrentIndex((prevIndex) => (prevIndex + 1) % categoryImages.length);
|
||||
};
|
||||
|
||||
// Function to handle moving to the previous image
|
||||
const handlePrev = () => {
|
||||
setCurrentIndex((prevIndex) => (prevIndex - 1 + categoryImages.length) % categoryImages.length);
|
||||
};
|
||||
|
||||
return (
|
||||
<div style={{ position: 'relative', overflow: 'hidden' }}>
|
||||
{/* Container to display previous, current, and next images */}
|
||||
<div style={{ display: 'flex' }}>
|
||||
{/* Button to move to the previous image */}
|
||||
<button onClick={handlePrev}>Previous</button>
|
||||
{/* Render the previous image */}
|
||||
{RenderImage((currentIndex - 1 + categoryImages.length) % categoryImages.length, categoryImages)}
|
||||
{/* Render the current image */}
|
||||
{RenderImage(currentIndex, categoryImages)}
|
||||
{/* Render the next image */}
|
||||
{RenderImage((currentIndex + 1) % categoryImages.length, categoryImages)}
|
||||
{/* Button to move to the next image */}
|
||||
<button onClick={handleNext}>Next</button>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default ImageSlider;
|
51
src/Images/Images.json
Normal file
51
src/Images/Images.json
Normal file
@ -0,0 +1,51 @@
|
||||
[
|
||||
{
|
||||
"id": 1,
|
||||
"src": "https://img.freepik.com/free-photo/luxurious-car-parked-highway-with-illuminated-headlight-sunset_181624-60607.jpg?size=626&ext=jpg&ga=GA1.1.87170709.1707264000&semt=sph",
|
||||
"alt": "Image 1",
|
||||
"category": "Car"
|
||||
},
|
||||
{
|
||||
"id": 2,
|
||||
"src": "https://media.architecturaldigest.com/photos/63079fc7b4858efb76814bd2/16:9/w_4000,h_2250,c_limit/9.%20DeLorean-Alpha-5%20%5BDeLorean%5D.jpg",
|
||||
"alt": "Image 2",
|
||||
"category": "Car"
|
||||
},
|
||||
{
|
||||
"id": 3,
|
||||
"src": "https://cdn.motor1.com/images/mgl/g440ng/s3/rimac-nevera.jpg",
|
||||
"alt": "Image 3",
|
||||
"category": "Car"
|
||||
},
|
||||
{
|
||||
"id": 4,
|
||||
"src": "https://www.shutterstock.com/image-vector/realistic-vector-red-car-gradients-260nw-2271536365.jpg",
|
||||
"alt": "Image 4",
|
||||
"category": "Car"
|
||||
},
|
||||
{
|
||||
"id": 5,
|
||||
"src": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcTJNuga_vmLPm7BLwbqQmsuNklxIsqXm_tTMg&usqp=CAU",
|
||||
"alt": "Image 5",
|
||||
"category": "Members"
|
||||
},
|
||||
{
|
||||
"id": 6,
|
||||
"src": "",
|
||||
"alt": "Image 6",
|
||||
"category": "Members"
|
||||
},
|
||||
{
|
||||
"id": 7,
|
||||
"src": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcRMdEL6UJgg3rFpj41o1bcjX4z2b_OEaYYNJg&usqp=CAU",
|
||||
"alt": "Image 7",
|
||||
"category": "Members"
|
||||
},
|
||||
{
|
||||
"id": 8,
|
||||
"src": "https://png.pngtree.com/png-vector/20220331/ourmid/pngtree-group-of-people-silhouette-png-image_4520545.png",
|
||||
"alt": "Image 8",
|
||||
"category": "Members"
|
||||
}
|
||||
]
|
||||
|
26
src/Images/RenderImage.js
Normal file
26
src/Images/RenderImage.js
Normal file
@ -0,0 +1,26 @@
|
||||
import React from "react";
|
||||
|
||||
// Function to render an image based on the provided index
|
||||
const RenderImage = (index, categoryImages) => {
|
||||
// Check if categoryImages is not empty before accessing its properties
|
||||
if (categoryImages.length > 0 && categoryImages[index]) {
|
||||
return (
|
||||
<img
|
||||
key={index}
|
||||
src={categoryImages[index].src}
|
||||
alt={categoryImages[index].alt}
|
||||
// onClick={handleImageClick}
|
||||
style={{ maxWidth: '100%', maxHeight: '100%' }}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
return null; // or handle this case as needed
|
||||
}
|
||||
}
|
||||
|
||||
export default RenderImage;
|
||||
|
||||
// // Function to handle custom click actions on the image
|
||||
// const handleImageClick = () => {
|
||||
// // Will be worked on later
|
||||
// };
|
51
src/OurSponsors/OurSponsors.css
Normal file
51
src/OurSponsors/OurSponsors.css
Normal file
@ -0,0 +1,51 @@
|
||||
#OurSponsors {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
background-color: gray;
|
||||
padding-left: 10%;
|
||||
padding-right: 10%;
|
||||
}
|
||||
|
||||
#OurSponsors div div {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
background-color: white;
|
||||
}
|
||||
|
||||
#OurSponsors div div div div img {
|
||||
display: flex;
|
||||
justify-content: left;
|
||||
}
|
||||
|
||||
#OurSponsors div div p {
|
||||
display: flex;
|
||||
justify-content: left;
|
||||
}
|
||||
|
||||
#OurSponsors div div div div {
|
||||
padding: 0%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: left;
|
||||
}
|
||||
|
||||
#OurSponsors div div div {
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
justify-content: end;
|
||||
padding: 0%;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
#OurSponsors h2 {
|
||||
background-color: aquamarine;
|
||||
text-align: center;
|
||||
margin: 0px;
|
||||
}
|
||||
|
||||
#OurSponsors h3 {
|
||||
text-align: left;
|
||||
}
|
||||
|
@ -1,5 +1,165 @@
|
||||
export default function OurSponsors() {
|
||||
return (
|
||||
<p>Our Sponsors</p>
|
||||
);
|
||||
};
|
||||
import { useEffect } from 'react';
|
||||
import { useState } from 'react';
|
||||
import './OurSponsors.css'
|
||||
//can be removed later
|
||||
import fakeDelay from '../TestingTools/fakeDelay';
|
||||
import { useNavigate } from 'react-router-dom';
|
||||
|
||||
/*
|
||||
things to do
|
||||
*add link to become a sponsor at top, and thank you message
|
||||
*/
|
||||
|
||||
const OurSponsors = () => {
|
||||
/*
|
||||
OurSponsors Page
|
||||
REQUIRES:
|
||||
Nothing
|
||||
PROMISES:
|
||||
HTML tags and functionality
|
||||
Develop in part by: Brock
|
||||
Contact: darkicewolf50@gmail.com
|
||||
*/
|
||||
const [sponsorsDict, setSponsorsDict] = useState(); //variable states for the dictionary of sponsors
|
||||
|
||||
useEffect(() => {
|
||||
getSponsors(); //get sponsors on startup of page
|
||||
}, []);
|
||||
|
||||
const OpenPage = (arg) => {
|
||||
const navigate = useNavigate();
|
||||
navigate(arg);
|
||||
console.log(arg);
|
||||
};
|
||||
|
||||
const getSponsors = async () => {
|
||||
/*
|
||||
Gets the list of sponsors from the synology drive (not implemented), converts the json file into a dictionary
|
||||
REQUIRES:
|
||||
constant html link to synology drive
|
||||
PROMISES:
|
||||
returns Dictionary list of all our sponsors in this format of json file
|
||||
Develop in part by: Brock
|
||||
Contact: darkicewolf50@gmail.com
|
||||
*/
|
||||
const tempListOfSponsors = {
|
||||
"Main Tier": {
|
||||
1:{
|
||||
"Name":"Hello",
|
||||
"LogoUrl":"",
|
||||
"Url":"",
|
||||
"DescriptionAboutSponsor":"this has text i dont care",
|
||||
"DecriptionOnHelp":"this has text i dont care"
|
||||
}
|
||||
},
|
||||
"Platnum Tier": {
|
||||
1:{
|
||||
"Name":"Schulich School of Engineering",
|
||||
"LogoUrl":"https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcQj9L3ZGK6WtOmJbzxmCzRxwLXYKGC5SDcAKHb0ScfbUmbtG0IujQt6eQDaI_Pm9g4DZvc&usqp=CAU",
|
||||
"Url":"https://schulich.ucalgary.ca/",
|
||||
"DescriptionAboutSponsor":"Lorem ipsum dolor sit amet, consectetur adipiscing elit. Fusce eu magna in diam consectetur rhoncus vel nec turpis. Sed finibus mi eu sem varius faucibus. Donec semper erat et bibendum pharetra. Suspendisse cursus lorem sed nisi semper, a rutrum nunc luctus. Nunc ullamcorper enim id orci interdum ultrices. Donec vestibulum nunc vel nisl pretium tempus. Morbi quis ante et ligula eleifend eleifend. Proin bibendum maximus elit vitae congue. Vivamus egestas, ex in tempor posuere, ligula nunc iaculis massa, in imperdiet dui justo eu dolor. Nullam placerat velit quis sem mattis, laoreet pharetra elit tempor. ",
|
||||
"DecriptionOnHelp":"this has text i dont care"
|
||||
},
|
||||
2:{
|
||||
"Name":"",
|
||||
"LogoUrl":"",
|
||||
"Url":"",
|
||||
"DescriptionAboutSponsor":"this has text i dont care",
|
||||
"DecriptionOnHelp":"this has text i dont care"
|
||||
}
|
||||
},
|
||||
"Gold Tier":{
|
||||
1:{
|
||||
"Name":"",
|
||||
"LogoUrl":"",
|
||||
"Url":"",
|
||||
"DescriptionAboutSponsor":"this has text i dont care",
|
||||
"DecriptionOnHelp":"this has text i dont care"
|
||||
},
|
||||
2:{
|
||||
"Name":"",
|
||||
"LogoUrl":"",
|
||||
"Url":"",
|
||||
"DescriptionAboutSponsor":"1this has text i dont care",
|
||||
"DecriptionOnHelp":"2this has text i dont care"
|
||||
}
|
||||
},
|
||||
"Silver Tier": {
|
||||
1:{
|
||||
"Name":"Father",
|
||||
"LogoUrl":"",
|
||||
"Url":""
|
||||
//"DescriptionAboutSponsor":"", //Dont get this
|
||||
//"DecriptionOnHelp":"" //Dont get this
|
||||
},
|
||||
2:{
|
||||
"Name":"Help",
|
||||
"LogoUrl":"",
|
||||
"Url":""
|
||||
//"DescriptionAboutSponsor":"", //Dont get this
|
||||
//"DecriptionOnHelp":"" //Dont get this
|
||||
}
|
||||
},
|
||||
"Bronze Tier":{
|
||||
1:{
|
||||
"Name":"I'm",
|
||||
//"LogoUrl":"", //Dont get this
|
||||
"Url":""
|
||||
//"DescriptionAboutSponsor":"", //Dont get this
|
||||
//"DecriptionOnHelp":"" //Dont get this
|
||||
},
|
||||
2:{
|
||||
"Name":"Stuck to a cactus",
|
||||
//"LogoUrl":"", //Dont get this
|
||||
"Url":""
|
||||
//"DescriptionAboutSponsor":"", //Dont get this
|
||||
//"DecriptionOnHelp":"" //Dont get this
|
||||
}
|
||||
}
|
||||
};
|
||||
try {
|
||||
await fakeDelay(1000);
|
||||
console.log("It ran");
|
||||
let res = tempListOfSponsors;
|
||||
setSponsorsDict(res);
|
||||
} catch (error) { //error checking
|
||||
console.error('Error sending data to server:', error);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if (!sponsorsDict) { //awaiting for a resposne from the backend
|
||||
return <p>Loading...</p>
|
||||
}
|
||||
if(sponsorsDict) { //maps out the dictionary and displays the content
|
||||
return (<div id='OurSponsors'>
|
||||
<div id='BecomeASponsors'>
|
||||
<button onClick={() => OpenPage('/BecomeASponsor')}>Become A Sponsor</button>
|
||||
</div>
|
||||
{Object.keys(sponsorsDict).map((sponsorTier) => (
|
||||
<div key={sponsorTier} className={sponsorTier}>
|
||||
<h2>{sponsorTier}</h2>
|
||||
{Object.keys(sponsorsDict[sponsorTier]).map((sponsorKey) => {
|
||||
const sponsor = sponsorsDict[sponsorTier][sponsorKey];
|
||||
return (
|
||||
<div key={sponsorKey}>
|
||||
<div>
|
||||
<div>
|
||||
{sponsor.Name && <h3>{sponsor.Name}</h3>}
|
||||
{sponsor.LogoUrl && <a href={sponsor.Url}><img src={sponsor.LogoUrl} alt="Sponsor Logo" /></a>}
|
||||
</div>
|
||||
{(sponsor.DescriptionAboutSponsor !== undefined && sponsor.DecriptionOnHelp !== undefined) && <p>Another Element</p>}
|
||||
</div>
|
||||
{sponsor.DescriptionAboutSponsor !== undefined && <p>{sponsor.DescriptionAboutSponsor}</p>}
|
||||
{sponsor.DecriptionOnHelp !== undefined && <p>{sponsor.DecriptionOnHelp}</p>}
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
))}
|
||||
</div>);
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
export default OurSponsors;
|
@ -1,5 +1,15 @@
|
||||
import React from 'react';
|
||||
import ImageSlider from '../Images/ImageSlider'; // Adjust the import path based on your directory structure
|
||||
|
||||
export default function Teams() {
|
||||
// return (
|
||||
// <p>Teams</p>
|
||||
// );
|
||||
|
||||
return (
|
||||
<p>Teams</p>
|
||||
<div>
|
||||
<h1>Image Slider</h1>
|
||||
<ImageSlider category="Members" />
|
||||
</div>
|
||||
);
|
||||
};
|
18
src/TestingTools/fakeDelay.js
Normal file
18
src/TestingTools/fakeDelay.js
Normal file
@ -0,0 +1,18 @@
|
||||
|
||||
function fakeDelay(ms) {
|
||||
/*
|
||||
useful for testing allows for a fake response, allow us to see what the program does while waiting for data from the backend
|
||||
REQUIRES:
|
||||
time in milliseconds
|
||||
|
||||
needs a line like this to run
|
||||
await fakeDelay(5000);
|
||||
PROMISES:
|
||||
nothing
|
||||
Develop in part by: Brock
|
||||
Contact: darkicewolf50@gmial.com
|
||||
*/
|
||||
return new Promise(resolve => setTimeout(resolve, ms));
|
||||
}
|
||||
|
||||
export default fakeDelay;
|
Loading…
x
Reference in New Issue
Block a user