Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

added item pagination with 10 items per page and 4 buttons to select from along with arrows #35

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 2 additions & 3 deletions packages/web/src/components/Home/Home.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -142,8 +142,7 @@ export default function Home() {
e.preventDefault();
try {
await axios.patch(
`${
import.meta.env.VITE_REACT_APP_AWS_BACKEND_URL
`${import.meta.env.VITE_REACT_APP_AWS_BACKEND_URL
}/leaderboard/changeSubscription`,
{
email: user.email,
Expand Down Expand Up @@ -564,7 +563,7 @@ export default function Home() {
colorScheme="#74a2fa"
color="#5f85cf"
spacing={3}
// boxShadow="5px 2px 9px rgba(0, 0, 0, 0.2);"
// boxShadow="5px 2px 9px rgba(0, 0, 0, 0.2);"
>
<Button
backgroundColor="white"
Expand Down
50 changes: 25 additions & 25 deletions packages/web/src/components/Map/Map.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -132,26 +132,28 @@ export default function Map({
);

const markersData = results.length > 0 ? results : data;
const allMarkers = markersData.filter(filterItemCallback).map((item) => {
return (
<Marker
key={item.key}
position={item.location}
eventHandlers={{
click: () => {
onOpen();
setItemData(item);
setFocusLocation(item.location);
},
}}
icon={
item.isresolved
? iconsMap["resolved"][item.islost]
: (iconsMap[item.type] || iconsMap["others"])[item.islost]
}
></Marker>
);
});
const allMarkers = useMemo(() => {
return markersData.filter(filterItemCallback).map((item) => {
return (
<Marker
key={item.key}
position={item.location}
eventHandlers={{
click: () => {
onOpen();
setItemData(item);
setFocusLocation(item.location);
},
}}
icon={
item.isresolved
? iconsMap["resolved"][item.islost]
: (iconsMap[item.type] || iconsMap["others"])[item.islost]
}
></Marker>
);
});
}, [markersData, filterItemCallback]);

// moves map when focusLocation state changes
function MapFocusLocation({ location }) {
Expand All @@ -178,10 +180,10 @@ export default function Map({
}),
[setPosition]
);
async function handleSubmit() {
const handleSubmit = useCallback(async () => {
const date = new Date();

if (!token) {
if (!token || !newAddedItem) {
return;
}
axios
Expand Down Expand Up @@ -264,7 +266,7 @@ export default function Map({
.catch((err) => console.log(err));

setLoading(true);
}
}, [token, newAddedItem]);

const toggleDraggable = () => {
if (!bounds.contains(position)) {
Expand Down Expand Up @@ -300,7 +302,6 @@ export default function Map({
</>
);
}

const NewItemMarker = () => {
useMapEvents({
click(event) {
Expand Down Expand Up @@ -348,7 +349,6 @@ export default function Map({
{!isEdit && (
<MapFocusLocation location={focusLocation} search={search} />
)}
{!isEdit}
{!isEdit && allMarkers}

{isEdit && <NewItemMarker />}
Expand Down
150 changes: 127 additions & 23 deletions packages/web/src/components/ResultsBar/ResultsBar.jsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
import { useContext, useCallback, useState } from "react";
import { useContext, useCallback, useState, useEffect, useRef } from "react";
import "./ResultsBar.css";
import ResultCard from "../ResultCard/ResultCard";
import { Box, Flex, Text, Button } from "@chakra-ui/react";
import { Box, Flex, Text, Button, IconButton } from "@chakra-ui/react";
import { ChevronRightIcon, ChevronLeftIcon } from "@chakra-ui/icons";
import DataContext from "../../context/DataContext";
import { UserAuth } from "../../context/AuthContext";
import Fuse from "fuse.js";
Expand All @@ -18,8 +19,6 @@ export default function ResultsBar({
const { data, setData } = useContext(DataContext);
const { user } = UserAuth();

const [itemsonScreenLimit, setItemsOnScreenLimit] = useState(10);

const filterItemCallback = useCallback(
(item) => filterItem(item, findFilter, user),
[findFilter, user]
Expand Down Expand Up @@ -60,33 +59,131 @@ export default function ResultsBar({
? data.filter(filterItemCallback).map(mapItem)
: results.filter(filterItemCallback).map(mapItem);

// Callback function that increases the number of items displayed on the screen by 10
const handleLoadMore = useCallback(() => {
setItemsOnScreenLimit(itemsonScreenLimit + 10);
}, [itemsonScreenLimit]);

const loadMoreButton = (

// Display only the 10 items on the screen and only 4 buttons (the last one is always there)
const itemsOnScreenLimit = 10;
const buttonLimit = 3;

const [selectedPageNumber, setSelectedPageNumber] = useState(1);
const [paginationButtons, setPaginationButtons] = useState([])
const [viewableResults, setViewableResults] = useState([])
const resultsBarRef = useRef(null);

const changeViewToNewPage = useCallback((newSelectedPage) => {
const pageStart = (newSelectedPage * itemsOnScreenLimit) - itemsOnScreenLimit
const pageEnd = Math.min((newSelectedPage * itemsOnScreenLimit), allResults.length)

setViewableResults(allResults.slice(
pageStart,
pageEnd
));
resultsBarRef.current.scrollTo({
top: 0,
});
}, [allResults, allResults.length]);

const updatePaginationButtons = useCallback(
(newSelectedPage) => {
changeViewToNewPage(newSelectedPage);
setSelectedPageNumber(newSelectedPage);

const maxIndex = Math.floor(allResults.length / itemsOnScreenLimit) + 1;
if (newSelectedPage >= maxIndex - buttonLimit) {
setPaginationButtons(createPaginationButtons(maxIndex - buttonLimit, newSelectedPage));
}
else {
const startPage = newSelectedPage - ((newSelectedPage - 1) % 3);
setPaginationButtons(createPaginationButtons(startPage, newSelectedPage));
}
},
[allResults, allResults.length, changeViewToNewPage]
)
const PageButton = useCallback(({ keyProp, selected }) => (
<Button
onClick={handleLoadMore}
variant="outline"
onClick={() => updatePaginationButtons(keyProp)}
key={keyProp}
variant={selected ? "solid" : "outline"}

colorScheme="blue"
width="100%"
height={"80px"}
width="40px"
height={"40px"}
marginTop="10px"
marginBottom="10px"
fontSize={"xl"}
>
Load More
{keyProp}
</Button>
);
), [updatePaginationButtons]);

// Retrieve all items that meet the filter criteria
function createPaginationButtons(startPageNumber, selectedPageNumber) {
const buttons_array = [];
for (let i = startPageNumber; i < startPageNumber + buttonLimit; i++) {
buttons_array.push(<PageButton keyProp={i} selected={i == selectedPageNumber} />);
}
return buttons_array;
}


const handlePageArrowClick = useCallback((direction) => {
const maxIndex = Math.floor(allResults.length / itemsOnScreenLimit) + 1

let startPage = 0;
let selected = 0;
if (direction == "right" && selectedPageNumber < maxIndex) {
if (selectedPageNumber >= maxIndex - buttonLimit) {
startPage = maxIndex - buttonLimit;
selected = maxIndex;
}
else {
startPage = selectedPageNumber + (3 - ((selectedPageNumber - 1) % 3));
selected = selectedPageNumber + 3;
if (startPage > maxIndex - buttonLimit)
startPage = maxIndex - buttonLimit;
}

setSelectedPageNumber(selected);
changeViewToNewPage(selected);
setPaginationButtons(createPaginationButtons(startPage, selected));
}
else if (direction == "left" && selectedPageNumber > 0) {
if (selectedPageNumber <= 3) {
startPage = 1;
selected = 1;
}
else if (selectedPageNumber == maxIndex) {
startPage = maxIndex - 4;
selected = maxIndex - 4;
}
else {
startPage = selectedPageNumber - (3 + (selectedPageNumber - 1) % 3);
selected = selectedPageNumber - 3;
}

setSelectedPageNumber(selected);
changeViewToNewPage(selected);
setPaginationButtons(createPaginationButtons(startPage, selected));
}
}, [allResults, allResults.length, selectedPageNumber])



const firstRender = useRef(true);
useEffect(() => {
if (!firstRender.current || allResults.length === 0)
return;

firstRender.current = false;
const pageStart = (selectedPageNumber * itemsOnScreenLimit) - itemsOnScreenLimit
const pageEnd = Math.min((selectedPageNumber * itemsOnScreenLimit), allResults.length)

setViewableResults(allResults.slice(
pageStart,
pageEnd
));
setPaginationButtons(createPaginationButtons(1, 1));
}, [allResults])

// Display only the first 10 items on the screen, all items if there are less than 10 items left to be loaded
const viewableResults = allResults.slice(
0,
Math.min(itemsonScreenLimit, allResults.length)
);

// Define JSX for empty results bar (no result cards)
const noResults = (
Expand All @@ -100,13 +197,20 @@ export default function ResultsBar({
return (
<Box
paddingX="5px"
width={{ base: "90vw", md: "21vw" }}
paddingY="20px"
width={{ base: "30vw", md: "21vw" }}
height="80vh"
overflowY="scroll"
overflowX="hidden"
ref={resultsBarRef}
>
{allResults.length > 0 ? viewableResults : noResults}
{viewableResults.length < allResults.length && loadMoreButton}
<Box width="100%" display="flex" justifyContent={"space-between"}>
<IconButton icon={<ChevronLeftIcon />} height="40px" width="10px" marginTop="10px" marginBottom="10px" onClick={() => handlePageArrowClick("left")} />
{paginationButtons}
<PageButton keyProp={Math.floor(allResults.length / 10) + 1} selected={Math.floor(allResults.length / 10) + 1 == selectedPageNumber} />
<IconButton icon={<ChevronRightIcon />} height="40px" width="10px" marginTop="10px" marginBottom="10px" onClick={() => handlePageArrowClick("right")} />
</Box>
</Box>
);
}
1 change: 0 additions & 1 deletion packages/web/src/utils/Utils.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
// Define callback function to return filtered items (filtered according to search bar and filter markers)
const filterItem = (item, findFilter, user) => {
console.log(item);
return (
((findFilter.islost && item.islost) ||
(findFilter.isFound && !item.islost)) &&
Expand Down