Image loading and auth

This commit is contained in:
wisplite
2025-11-23 02:27:09 -06:00
parent b61bd2f8d6
commit 2b7268b3f6
4 changed files with 158 additions and 5 deletions
+38
View File
@@ -36,4 +36,42 @@ func RegisterMediaRoutes(rg *gin.RouterGroup) {
} }
c.JSON(http.StatusOK, gin.H{"media": media}) c.JSON(http.StatusOK, gin.H{"media": media})
}) })
media.GET("/getAllMediaInAlbum", func(c *gin.Context) {
accessToken := c.GetHeader("Authorization")
albumID := c.Query("albumId")
media, err := services.GetAllMediaInAlbum(albumID, accessToken)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"media": media})
})
media.GET("/:albumId/:mediaId", func(c *gin.Context) {
albumID := c.Param("albumId")
mediaID := c.Param("mediaId")
if albumID == "root" {
albumID = ""
}
accessToken := c.GetHeader("Authorization")
userID, err := services.ValidateAccessToken(accessToken)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
accessLevel, err := services.CheckUserAlbumAccess(userID, albumID)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
if accessLevel < 0 {
c.JSON(http.StatusForbidden, gin.H{"error": "user does not have permission to view media in this album"})
return
}
mediaData, err := services.GetMedia(albumID, mediaID)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.File(mediaData.Path)
})
} }
+29
View File
@@ -38,3 +38,32 @@ func UploadMedia(file *multipart.FileHeader, albumID string, accessToken string)
} }
return media, nil return media, nil
} }
func GetAllMediaInAlbum(albumID string, accessToken string) ([]models.Media, error) {
userID, err := ValidateAccessToken(accessToken)
if err != nil {
return []models.Media{}, err
}
accessLevel, err := CheckUserAlbumAccess(userID, albumID)
if err != nil {
return []models.Media{}, err
}
if accessLevel < 0 {
return []models.Media{}, fmt.Errorf("user does not have permission to view media in this album")
}
media := []models.Media{}
result := db.GetDB().Where("album_id = ?", albumID).Find(&media)
if result.Error != nil {
return []models.Media{}, result.Error
}
return media, nil
}
func GetMedia(albumID string, mediaID string) (models.Media, error) {
media := models.Media{}
result := db.GetDB().First(&media, "album_id = ? AND id = ?", albumID, mediaID)
if result.Error != nil {
return models.Media{}, result.Error
}
return media, nil
}
+65
View File
@@ -0,0 +1,65 @@
import { useState, useEffect } from 'react'
export default function AuthImage({ src, token, alt, className, ...props }) {
const [imageSrc, setImageSrc] = useState(null)
const [loading, setLoading] = useState(true)
const [error, setError] = useState(false)
useEffect(() => {
let objectUrl = null
let active = true
const fetchImage = async () => {
setLoading(true)
setError(false)
try {
const response = await fetch(src, {
headers: {
'Authorization': token
}
})
if (!response.ok) {
throw new Error('Failed to load image')
}
const blob = await response.blob()
if (active) {
objectUrl = URL.createObjectURL(blob)
setImageSrc(objectUrl)
setLoading(false)
}
} catch (err) {
if (active) {
console.error("Error loading image:", err)
setError(true)
setLoading(false)
}
}
}
if (src && token) {
fetchImage()
} else {
setLoading(false)
}
return () => {
active = false
if (objectUrl) {
URL.revokeObjectURL(objectUrl)
}
}
}, [src, token])
if (loading) {
return <div className={`bg-gray-800 animate-pulse ${className}`} />
}
if (error || !imageSrc) {
return <div className={`bg-gray-800 flex items-center justify-center text-gray-500 ${className}`}>Error</div>
}
return <img src={imageSrc} alt={alt} className={className} {...props} />
}
+26 -5
View File
@@ -1,5 +1,6 @@
import { Upload } from 'lucide-react' import { Upload } from 'lucide-react'
import MediaUploadModal from './MediaUploadModal' import MediaUploadModal from './MediaUploadModal'
import AuthImage from '../../components/AuthImage'
import { useState, useEffect } from 'react' import { useState, useEffect } from 'react'
import { getServerUrl } from '../../hooks/getConstants' import { getServerUrl } from '../../hooks/getConstants'
import { useAccount } from '../../contexts/useAccount' import { useAccount } from '../../contexts/useAccount'
@@ -14,12 +15,23 @@ export default function MediaList({ albumId, albumName }) {
useEffect(() => { useEffect(() => {
let ignore = false; let ignore = false;
const getMedia = async () => { const getMedia = async () => {
if (!albumId) return
// TODO: Implement media fetching from API // TODO: Implement media fetching from API
// const response = await fetch(...) // const response = await fetch(...)
const response = await fetch(`${getServerUrl()}/api/media/getAllMediaInAlbum?albumId=${albumId}`, {
method: 'GET',
headers: {
'Authorization': getAccessToken(),
},
})
const data = await response.json()
if (ignore) return
if (data.error) {
showError(data.error)
} else {
console.log("data.media", data.media)
setMedia(data.media)
}
} }
getMedia() getMedia()
return () => { ignore = true; } return () => { ignore = true; }
}, [albumId]) }, [albumId])
@@ -31,12 +43,21 @@ export default function MediaList({ albumId, albumName }) {
<Upload className="w-6 h-6 cursor-pointer" color="white" onClick={() => setOpen(true)} /> <Upload className="w-6 h-6 cursor-pointer" color="white" onClick={() => setOpen(true)} />
</div> </div>
{/* Media Grid Placeholder */} {/* Media Grid */}
<div className="flex flex-row items-center justify-start gap-2 w-full px-6 flex-wrap"> <div className="flex flex-row items-center justify-start gap-2 w-full px-6 flex-wrap">
{media.length === 0 && ( {media.length === 0 && (
<p className="text-gray-500 red-hat-text">No media in this album</p> <p className="text-gray-500 red-hat-text">No media in this album</p>
)} )}
{/* Render media items here */} {media.map((media) => (
<div key={media.id} className="flex flex-col items-center justify-center w-24 h-24 bg-[#1A1A1A] rounded-md border border-[#2B2B2B]">
<AuthImage
src={`${getServerUrl()}/api/media/${albumId ? albumId : 'root'}/${media.ID}`}
token={getAccessToken()}
alt={media.Title}
className="w-full h-full object-cover"
/>
</div>
))}
</div> </div>
<MediaUploadModal <MediaUploadModal