diff options
| author | Pinapelz <yukais@pinapelz.com> | 2025-09-23 15:05:48 -0700 |
|---|---|---|
| committer | Pinapelz <yukais@pinapelz.com> | 2025-09-23 15:05:48 -0700 |
| commit | 6df78f254d98453520e8644996b917d673ee3978 (patch) | |
| tree | 9e2ed1a8cea6bbab145191e8f40b5b2c59b6c51c /frontend/src/components | |
| parent | abca372d8ef3d9ab0154c3706d88e0c3772bacc3 (diff) | |
initial mobile CSS style
Diffstat (limited to 'frontend/src/components')
| -rw-r--r-- | frontend/src/components/NavBar.tsx | 59 | ||||
| -rw-r--r-- | frontend/src/components/displays/GenericScoreDisplay.tsx | 43 | ||||
| -rw-r--r-- | frontend/src/components/modals/JsonUploadModal.tsx | 2 |
3 files changed, 80 insertions, 24 deletions
diff --git a/frontend/src/components/NavBar.tsx b/frontend/src/components/NavBar.tsx index 4be8607..ac70dfb 100644 --- a/frontend/src/components/NavBar.tsx +++ b/frontend/src/components/NavBar.tsx @@ -1,10 +1,13 @@ import { Link } from "react-router"; +import { useState } from "react"; export const NavBar = ({ currentPage, user, handleLogout }: { currentPage: string; user: { username: string }; handleLogout: () => void; }) => { + const [isMobileMenuOpen, setIsMobileMenuOpen] = useState(false); + const menuOptions = ( <> <Link @@ -14,6 +17,7 @@ export const NavBar = ({ currentPage, user, handleLogout }: { ? 'bg-violet-600 text-white shadow-lg shadow-violet-500/25' : 'text-slate-300 hover:text-white hover:bg-slate-800/50' }`} + onClick={() => setIsMobileMenuOpen(false)} > Home </Link> @@ -24,6 +28,7 @@ export const NavBar = ({ currentPage, user, handleLogout }: { ? 'bg-violet-600 text-white shadow-lg shadow-violet-500/25' : 'text-slate-300 hover:text-white hover:bg-slate-800/50' }`} + onClick={() => setIsMobileMenuOpen(false)} > Import Data </Link> @@ -34,6 +39,7 @@ export const NavBar = ({ currentPage, user, handleLogout }: { ? 'bg-violet-600 text-white shadow-lg shadow-violet-500/25' : 'text-slate-300 hover:text-white hover:bg-slate-800/50' }`} + onClick={() => setIsMobileMenuOpen(false)} > Community Scores </Link> @@ -44,13 +50,16 @@ export const NavBar = ({ currentPage, user, handleLogout }: { <nav className="border-b border-slate-800/50 bg-slate-950/95 backdrop-blur-sm sticky top-0 z-50"> <div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8"> <div className="flex items-center justify-between h-16"> + {/* Logo */} <div className="flex items-center space-x-3"> <div className="w-8 h-8 bg-gradient-to-r from-violet-600 to-violet-700 rounded-lg flex items-center justify-center shadow-lg"> <span className="text-white font-bold text-sm">M</span> </div> <span className="text-white font-semibold text-lg">Mirage</span> </div> - <div className="flex items-center space-x-4"> + + {/* Desktop Menu */} + <div className="hidden md:flex items-center space-x-4"> {menuOptions} <span className="text-slate-300 text-sm"> Welcome back,{" "} @@ -65,8 +74,54 @@ export const NavBar = ({ currentPage, user, handleLogout }: { Sign Out </button> </div> + + {/* Mobile menu button */} + <div className="flex md:hidden"> + <button + onClick={() => setIsMobileMenuOpen(!isMobileMenuOpen)} + className="inline-flex items-center justify-center p-2 rounded-md text-slate-300 hover:text-white hover:bg-slate-800/50 focus:outline-none focus:ring-2 focus:ring-inset focus:ring-violet-500" + aria-expanded={isMobileMenuOpen} + > + <span className="sr-only">Open main menu</span> + {isMobileMenuOpen ? ( + <svg className="block h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor"> + <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" /> + </svg> + ) : ( + <svg className="block h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor"> + <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M4 6h16M4 12h16M4 18h16" /> + </svg> + )} + </button> + </div> </div> + + {/* Mobile Menu */} + {isMobileMenuOpen && ( + <div className="md:hidden"> + <div className="px-2 pt-2 pb-3 space-y-1"> + {menuOptions} + <div className="border-t border-slate-800/50 mt-3 pt-3"> + <div className="text-slate-300 text-sm px-4 py-2"> + Welcome back,{" "} + <span className="text-violet-400 font-medium"> + {user.username} + </span> + </div> + <button + onClick={() => { + handleLogout(); + setIsMobileMenuOpen(false); + }} + className="w-full text-left text-slate-300 hover:text-white px-4 py-2 rounded-lg text-sm font-medium transition-all duration-200 hover:bg-slate-800/50" + > + Sign Out + </button> + </div> + </div> + </div> + )} </div> </nav> ); -}; +};
\ No newline at end of file diff --git a/frontend/src/components/displays/GenericScoreDisplay.tsx b/frontend/src/components/displays/GenericScoreDisplay.tsx index 4201d95..41a125c 100644 --- a/frontend/src/components/displays/GenericScoreDisplay.tsx +++ b/frontend/src/components/displays/GenericScoreDisplay.tsx @@ -269,7 +269,7 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ if (viewMode === "cards") { return ( - <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-2 xl:grid-cols-3 gap-6"> + <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-2 xl:grid-cols-3 gap-4 sm:gap-6"> {sortedScores.map((score, index) => { const chartIdHash = SHA1(`${internalGameName}${score.title}${score.artist}`).toString(); // eslint-disable-next-line @typescript-eslint/no-unused-vars @@ -279,18 +279,18 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ return ( <div key={score.id || index} - className="bg-slate-900/50 backdrop-blur-sm border border-slate-800/50 rounded-xl p-6 hover:border-violet-500/30 transition-all duration-300 hover:shadow-lg hover:shadow-violet-500/10" + className="bg-slate-900/50 backdrop-blur-sm border border-slate-800/50 rounded-lg sm:rounded-xl p-4 sm:p-6 hover:border-violet-500/30 transition-all duration-300 hover:shadow-lg hover:shadow-violet-500/10" > {/* Primary Info */} <div className="flex items-start justify-between mb-4"> <div className="flex-1 min-w-0"> {!hideTitleArtist && ( <Link to={`/chart?chartId=${chartIdHash}`}> - <h3 className="text-lg font-semibold text-white mb-1 break-words leading-tight"> + <h3 className="text-base sm:text-lg font-semibold text-white mb-1 break-words leading-tight"> {score.title || score.song || "Unknown Title"} </h3> {score.artist && ( - <p className="text-slate-400 text-sm break-words leading-tight"> + <p className="text-slate-400 text-xs sm:text-sm break-words leading-tight"> {score.artist} </p> )} @@ -306,13 +306,13 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ {/* Main Stats */} {mainStats.length > 0 && ( - <div className="grid grid-cols-2 gap-4 mb-4"> + <div className="grid grid-cols-2 gap-2 sm:gap-4 mb-4"> {mainStats.slice(0, 4).map(([key, value]) => ( - <div key={key} className="bg-slate-800/50 rounded-lg p-3"> - <p className="text-slate-400 text-xs uppercase tracking-wide mb-1"> + <div key={key} className="bg-slate-800/50 rounded-lg p-2 sm:p-3"> + <p className="text-slate-400 text-[10px] sm:text-xs uppercase tracking-wide mb-1"> {getDisplayName(key)} </p> - <p className="text-white font-semibold text-lg"> + <p className="text-white font-semibold text-sm sm:text-lg"> {renderValue(value, key)} </p> </div> @@ -335,7 +335,7 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ <div className="mb-4"> <div className="grid grid-cols-2 gap-2 text-sm"> {others.map(([key, value]) => ( - <div key={key} className="flex justify-between"> + <div key={key} className="flex justify-between text-xs sm:text-sm"> <span className="text-slate-400"> {getDisplayName(key)}: </span> @@ -350,7 +350,7 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ {/* Timestamp */} <div className="pt-4 border-t border-slate-800/50"> - <p className="text-slate-500 text-xs"> + <p className="text-slate-500 text-[10px] sm:text-xs"> {new Date( typeof timestamp === "number" ? timestamp : timestamp, ).toLocaleDateString()}{" "} @@ -372,21 +372,22 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ return ( <div className="bg-slate-900/50 backdrop-blur-sm border border-slate-800/50 rounded-xl overflow-hidden"> - <div className="overflow-x-auto"> - <table className="w-full text-sm min-w-[1000px]"> + <div className="overflow-x-auto relative"> + <div className="md:hidden absolute right-0 top-0 bottom-0 w-8 bg-gradient-to-l from-slate-900/80 to-transparent pointer-events-none z-10"></div> + <table className="w-full text-sm min-w-[800px] md:min-w-[1000px]"> <thead className="bg-slate-800/50 border-b border-slate-700/50"> <tr> {tableKeys.map((key) => ( <th key={key} - className="px-4 py-3 text-left text-slate-300 font-medium" + className="px-2 sm:px-4 py-2 sm:py-3 text-left text-slate-300 font-medium text-xs sm:text-sm" > {key === "judgements" ? ( <span>{getDisplayName(key)}</span> ) : ( <button onClick={() => onSort(key)} - className="flex items-center space-x-2 hover:text-white transition-colors" + className="flex items-center space-x-1 sm:space-x-2 hover:text-white transition-colors" > <span>{getDisplayName(key)}</span> <SortIcon field={key} /> @@ -395,7 +396,7 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ </th> ))} {showActions && ( - <th className="px-4 py-3 text-left text-slate-300 font-medium w-16"> + <th className="px-2 sm:px-4 py-2 sm:py-3 text-left text-slate-300 font-medium w-16 text-xs sm:text-sm"> Actions </th> )} @@ -408,10 +409,10 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ className="hover:bg-slate-800/30 transition-colors group" > {tableKeys.map((key) => ( - <td key={key} className="px-4 py-3"> + <td key={key} className="px-2 sm:px-4 py-2 sm:py-3 text-xs sm:text-sm"> {key === "lamp" || key === "diff_lamp" ? ( <div className="flex items-center space-x-2"> - <span className="inline-block bg-slate-800/50 text-slate-200 px-2 py-1 rounded text-xs border border-slate-600"> + <span className="inline-block bg-slate-800/50 text-slate-200 px-1 sm:px-2 py-0.5 sm:py-1 rounded text-[10px] sm:text-xs border border-slate-600 whitespace-nowrap"> {score[key] || "No Clear"} </span> </div> @@ -420,7 +421,7 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ {renderValue(score[key], key, true)} </div> ) : key === "timestamp" ? ( - <span className="text-slate-400 text-xs"> + <span className="text-slate-400 text-[10px] sm:text-xs whitespace-nowrap"> {new Date( typeof score[key] === "number" ? score[key] @@ -428,7 +429,7 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ ).toLocaleDateString()} </span> ) : key === "username" ? ( - <span className="text-violet-400 text-sm font-medium"> + <span className="text-violet-400 text-xs sm:text-sm font-medium"> {score[key] || "Unknown"} </span> ) : ( @@ -441,13 +442,13 @@ const ScoreDisplay: React.FC<ScoreDisplayProps> = ({ </td> ))} {showActions && ( - <td className="px-4 py-3"> + <td className="px-2 sm:px-4 py-2 sm:py-3"> <button onClick={() => onDelete(score.id)} className="text-red-400 hover:text-red-300 opacity-100 transition-opacity duration-200 p-1 rounded bg-red-500/10" title="Delete score" > - <svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"> + <svg className="w-3 h-3 sm:w-4 sm:h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"> <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" /> </svg> </button> diff --git a/frontend/src/components/modals/JsonUploadModal.tsx b/frontend/src/components/modals/JsonUploadModal.tsx index 7d8f838..3bb2b9b 100644 --- a/frontend/src/components/modals/JsonUploadModal.tsx +++ b/frontend/src/components/modals/JsonUploadModal.tsx @@ -77,7 +77,7 @@ const JsonUploadModal = ({ isOpen, onClose, onUpload, game }: JsonUploadModalPro /> {/* Modal */} - <div className="flex min-h-full items-center justify-center p-4"> + <div className="flex min-h-full items-center justify-center p-4 sm:p-6"> <div className="relative bg-slate-900 rounded-lg border border-slate-700 w-full max-w-md p-6 shadow-xl"> {/* Header */} <div className="mb-6"> |
