diff --git a/php/get.php b/php/get.php new file mode 100644 index 0000000..d73ebc0 --- /dev/null +++ b/php/get.php @@ -0,0 +1,52 @@ + + diff --git a/src/App.css b/src/App.css index 74b5e05..6529d74 100644 --- a/src/App.css +++ b/src/App.css @@ -1,38 +1,65 @@ -.App { - text-align: center; -} - -.App-logo { - height: 40vmin; - pointer-events: none; -} - -@media (prefers-reduced-motion: no-preference) { - .App-logo { - animation: App-logo-spin infinite 20s linear; - } -} - -.App-header { - background-color: #282c34; +body { min-height: 100vh; + background-color: #282c34; + color: #B9B8D6; +} + +.Header { + display: flex; + flex-direction: row; + + margin: 1em; + padding: 0; +} + +.Contents { display: flex; flex-direction: column; align-items: center; - justify-content: center; - font-size: calc(10px + 2vmin); - color: white; + white-space: pre-wrap; } -.App-link { - color: #61dafb; +.GalleryItem { + display: flex; + flex-direction: column; + text-align: center; + + background-color: #3B3E49; + + border-radius: 10px 10px 0px 0px; + margin: 0 0 1em 0; + padding: 0em; + max-width: calc(100vw - 2em); } -@keyframes App-logo-spin { - from { - transform: rotate(0deg); - } - to { - transform: rotate(360deg); - } +.GalleryItem > p { + margin: 0.5em; } + +.GalleryItem > img, video { + max-height: 80vh; +} + +.DirectoryItem > button { + background-color: #44655D; + color: #B9B8D6; + border: none; + padding: 10px 20px; + text-align: center; + text-decoration: none; /* No underline for text */ + display: inline-block; + font-size: 16px; + margin: 0px 4px 0px 0px; + cursor: pointer; + border-radius: 10px; /* Slightly rounded corners for a modern feel */ + transition: background-color 0.3s; /* Smooth transition for hover effect */ +} + +.DirectoryItem > button:hover { + background-color: #0056b3; +} + +.ParentDirectoryItem > button { + background-color: #4D4F5D; +} + diff --git a/src/App.tsx b/src/App.tsx index a53698a..7962027 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -1,26 +1,152 @@ import React from 'react'; -import logo from './logo.svg'; +//import logo from './logo.svg'; import './App.css'; +import { useState, useEffect } from 'react'; function App() { + const [data, setData] = useState(''); + const [category, setCategory] = useState(''); + + useEffect(() => { + post(category, (json: string) => { + setData(json); + }); + }); + return (
-
- logo -

- Edit src/App.tsx and save to reload. -

- - Learn React - -
+
+ {getContents(data, true, category, setCategory)} +
+
+ {getContents(data, false, category, setCategory)} +
); } +interface galleryItemInfo { + is_dir: boolean; + url: string; +} + +function getContents( + data: string, + getDir: boolean, + category: string, + setCategory: (category: string) => void +) { + if (data === '') return; + + let obj = null; + try { + obj = JSON.parse(data); + } catch (error: unknown) { + if (!(error instanceof SyntaxError)) { + throw new Error(error as unknown as undefined); + } + return data; + } + + return obj + .sort((a: galleryItemInfo, b: galleryItemInfo) => { + if (a.is_dir && b.is_dir) return 0; + if (a.is_dir && !b.is_dir) return -1; + if (!a.is_dir && b.is_dir) return 1; + if (!a.is_dir && !b.is_dir) return 0; + }) + .filter((item: galleryItemInfo) => { + return getDir == item.is_dir; + }) + .map((item: galleryItemInfo, index: number) => { + const url = item.url; + let onClick = () => {}; + if (item.is_dir) { + onClick = () => { + const subCategory = getFileName(url); + + if (category !== '') category = `${category}/${subCategory}`; + else category = subCategory; + if (subCategory === '..') + category = category.split('/').slice(0, -2).join('/'); + + setCategory(category); + }; + return ; + } + if (url.endsWith('.mp4')) { + return ; + } + return ; + }); +} + +interface galleryItem { + url: string; + onClick: () => void; +} + +function ImageItem({ url }: galleryItem) { + return ( +
+

{getFileName(url)}

+ +
+ ); +} + +function VideoItem({ url }: galleryItem) { + return ( +
+

{getFileName(url)}

+ +
+ ); +} + +function DirectoryItem({ url, onClick }: galleryItem) { + let buttonText = getFileName(url); + const isBackButton = buttonText === '..'; + buttonText = isBackButton ? 'Back' : `Category: ${buttonText}`; + const backButtonClass = isBackButton ? 'ParentDirectoryItem' : ''; + return ( +
+ +
+ ); +} + +function getFileName(string: string): string { + return string.split('/').at(-1) as unknown as string; +} + +function post(category: string, callback: (text: string) => void) { + fetch('http://localhost/react-photo-viewer/php/get.php', { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify({ + category: category, + }), + }) + .then((response) => { + if (!response.ok) { + throw new Error('Network response was not ok'); + } + return response; + }) + .then((response) => response.text()) + .then((data) => { + callback(data as unknown as string); + }) + .catch((error) => { + console.error('There was a problem with the fetch operation:', error); + }); +} + export default App;