added 3d model page

main
Antonio De Lucreziis 8 months ago
parent c0a941fac6
commit edcf04bf31

@ -5,8 +5,10 @@
"name": "aulastud-website",
"dependencies": {
"@astrojs/preact": "^4.1.0",
"@types/three": "^0.180.0",
"astro": "^5.13.5",
"preact": "^10.27.1",
"three": "^0.180.0",
},
},
},
@ -67,6 +69,8 @@
"@capsizecss/unpack": ["@capsizecss/unpack@2.4.0", "", { "dependencies": { "blob-to-buffer": "^1.2.8", "cross-fetch": "^3.0.4", "fontkit": "^2.0.2" } }, "sha512-GrSU71meACqcmIUxPYOJvGKF0yryjN/L1aCuE9DViCTJI7bfkjgYDPD1zbNDcINJwSSP6UaBZY9GAbYDO7re0Q=="],
"@dimforge/rapier3d-compat": ["@dimforge/rapier3d-compat@0.12.0", "", {}, "sha512-uekIGetywIgopfD97oDL5PfeezkFpNhwlzlaEYNOA0N6ghdsOvh/HYjSMek5Q2O1PYvRSDFcqFVJl4r4ZBwOow=="],
"@emnapi/runtime": ["@emnapi/runtime@1.5.0", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-97/BJ3iXHww3djw6hYIfErCZFee7qCtrneuLa20UXFCOTCfBM2cvQHjWJ2EG0s0MtdNwInarqCTz35i4wWXHsQ=="],
"@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.25.9", "", { "os": "aix", "cpu": "ppc64" }, "sha512-OaGtL73Jck6pBKjNIe24BnFE6agGl+6KxDtTfHhy1HmhthfKouEcOhqpSL64K4/0WCtbKFLOdzD/44cJ4k9opA=="],
@ -245,6 +249,8 @@
"@swc/helpers": ["@swc/helpers@0.5.17", "", { "dependencies": { "tslib": "^2.8.0" } }, "sha512-5IKx/Y13RsYd+sauPb2x+U/xZikHjolzfuDgTAl/Tdf3Q8rslRvC19NKDLgAJQ6wsqADk10ntlv08nPFw/gO/A=="],
"@tweenjs/tween.js": ["@tweenjs/tween.js@23.1.3", "", {}, "sha512-vJmvvwFxYuGnF2axRtPYocag6Clbb5YS7kLL+SO/TeVFzHqDIWrNKYtcsPMibjDx9O+bu+psAy9NKfWklassUA=="],
"@types/debug": ["@types/debug@4.1.12", "", { "dependencies": { "@types/ms": "*" } }, "sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ=="],
"@types/estree": ["@types/estree@1.0.8", "", {}, "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w=="],
@ -261,10 +267,18 @@
"@types/node": ["@types/node@24.3.1", "", { "dependencies": { "undici-types": "~7.10.0" } }, "sha512-3vXmQDXy+woz+gnrTvuvNrPzekOi+Ds0ReMxw0LzBiK3a+1k0kQn9f2NWk+lgD4rJehFUmYy2gMhJ2ZI+7YP9g=="],
"@types/stats.js": ["@types/stats.js@0.17.4", "", {}, "sha512-jIBvWWShCvlBqBNIZt0KAshWpvSjhkwkEu4ZUcASoAvhmrgAUI2t1dXrjSL4xXVLB4FznPrIsX3nKXFl/Dt4vA=="],
"@types/three": ["@types/three@0.180.0", "", { "dependencies": { "@dimforge/rapier3d-compat": "~0.12.0", "@tweenjs/tween.js": "~23.1.3", "@types/stats.js": "*", "@types/webxr": "*", "@webgpu/types": "*", "fflate": "~0.8.2", "meshoptimizer": "~0.22.0" } }, "sha512-ykFtgCqNnY0IPvDro7h+9ZeLY+qjgUWv+qEvUt84grhenO60Hqd4hScHE7VTB9nOQ/3QM8lkbNE+4vKjEpUxKg=="],
"@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="],
"@types/webxr": ["@types/webxr@0.5.23", "", {}, "sha512-GPe4AsfOSpqWd3xA/0gwoKod13ChcfV67trvxaW2krUbgb9gxQjnCx8zGshzMl8LSHZlNH5gQ8LNScsDuc7nGQ=="],
"@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="],
"@webgpu/types": ["@webgpu/types@0.1.65", "", {}, "sha512-cYrHab4d6wuVvDW5tdsfI6/o6vcLMDe6w2Citd1oS51Xxu2ycLCnVo4fqwujfKWijrZMInTJIKcXxteoy21nVA=="],
"acorn": ["acorn@8.15.0", "", { "bin": { "acorn": "bin/acorn" } }, "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg=="],
"ansi-align": ["ansi-align@3.0.1", "", { "dependencies": { "string-width": "^4.1.0" } }, "sha512-IOfwwBF5iczOjp/WeY4YxyjqAFMQoZufdQWDd19SEExbVLNXqvpzSJ/M7Za4/sCPmQ0+GRquoA7bGcINcxew6w=="],
@ -415,6 +429,8 @@
"fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="],
"fflate": ["fflate@0.8.2", "", {}, "sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A=="],
"flattie": ["flattie@1.1.1", "", {}, "sha512-9UbaD6XdAL97+k/n+N7JwX46K/M6Zc6KcFYskrYL8wbBV/Uyk0CTAMY0VT+qiK5PM7AIc9aTWYtq65U7T+aCNQ=="],
"fontace": ["fontace@0.3.0", "", { "dependencies": { "@types/fontkit": "^2.0.8", "fontkit": "^2.0.4" } }, "sha512-czoqATrcnxgWb/nAkfyIrRp6Q8biYj7nGnL6zfhTcX+JKKpWHFBnb8uNMw/kZr7u++3Y3wYSYoZgHkCcsuBpBg=="],
@ -525,6 +541,8 @@
"mdn-data": ["mdn-data@2.12.2", "", {}, "sha512-IEn+pegP1aManZuckezWCO+XZQDplx1366JoVhTpMpBB1sPey/SbveZQUosKiKiGYjg1wH4pMlNgXbCiYgihQA=="],
"meshoptimizer": ["meshoptimizer@0.22.0", "", {}, "sha512-IebiK79sqIy+E4EgOr+CAw+Ke8hAspXKzBd0JdgEmPHiAwmvEj2S4h1rfvo+o/BnfEYd/jAOg5IeeIjzlzSnDg=="],
"micromark": ["micromark@4.0.2", "", { "dependencies": { "@types/debug": "^4.0.0", "debug": "^4.0.0", "decode-named-character-reference": "^1.0.0", "devlop": "^1.0.0", "micromark-core-commonmark": "^2.0.0", "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-chunked": "^2.0.0", "micromark-util-combine-extensions": "^2.0.0", "micromark-util-decode-numeric-character-reference": "^2.0.0", "micromark-util-encode": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0", "micromark-util-resolve-all": "^2.0.0", "micromark-util-sanitize-uri": "^2.0.0", "micromark-util-subtokenize": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-zpe98Q6kvavpCr1NPVSCMebCKfD7CA2NqZ+rykeNhONIJBpc1tFKt9hucLGwha3jNTNI8lHpctWJWoimVF4PfA=="],
"micromark-core-commonmark": ["micromark-core-commonmark@2.0.3", "", { "dependencies": { "decode-named-character-reference": "^1.0.0", "devlop": "^1.0.0", "micromark-factory-destination": "^2.0.0", "micromark-factory-label": "^2.0.0", "micromark-factory-space": "^2.0.0", "micromark-factory-title": "^2.0.0", "micromark-factory-whitespace": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-chunked": "^2.0.0", "micromark-util-classify-character": "^2.0.0", "micromark-util-html-tag-name": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0", "micromark-util-resolve-all": "^2.0.0", "micromark-util-subtokenize": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-RDBrHEMSxVFLg6xvnXmb1Ayr2WzLAWjeSATAoxwKYJV94TeNavgoIdA0a9ytzDSVzBy2YKFK+emCPOEibLeCrg=="],
@ -711,6 +729,8 @@
"strip-ansi": ["strip-ansi@7.1.0", "", { "dependencies": { "ansi-regex": "^6.0.1" } }, "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ=="],
"three": ["three@0.180.0", "", {}, "sha512-o+qycAMZrh+TsE01GqWUxUIKR1AL0S8pq7zDkYOQw8GqfX8b8VoCKYUoHbhiX5j+7hr8XsuHDVU6+gkQJQKg9w=="],
"tiny-inflate": ["tiny-inflate@1.0.3", "", {}, "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw=="],
"tinyexec": ["tinyexec@0.3.2", "", {}, "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA=="],

@ -10,7 +10,9 @@
},
"dependencies": {
"@astrojs/preact": "^4.1.0",
"@types/three": "^0.180.0",
"astro": "^5.13.5",
"preact": "^10.27.1"
"preact": "^10.27.1",
"three": "^0.180.0"
}
}

@ -0,0 +1,191 @@
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
font-family: system-ui, sans-serif;
background: #334;
}
.page-header {
background: #334;
text-align: center;
padding: 2rem 1rem 1rem;
z-index: 100;
}
.page-title {
color: white;
font-size: 2.5rem;
font-weight: bold;
margin: 0;
text-shadow: 0 0 8px rgba(255, 255, 255, 0.5);
}
@media (max-width: 768px) {
.page-title {
font-size: 2rem;
}
.page-header {
padding: 1.5rem 1rem 0.5rem;
}
}
@media (max-width: 480px) {
.page-title {
font-size: 1.5rem;
}
}
.masonry-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
grid-template-rows: repeat(auto, 10px);
grid-auto-flow: row dense;
padding: 0.25rem;
gap: 0.25rem;
width: 100vw;
padding-bottom: 30vh;
justify-content: center;
}
.masonry-item {
cursor: pointer;
transition: opacity 0.2s ease;
overflow: hidden;
grid-row-end: span var(--img-height);
}
.masonry-item:hover {
opacity: 0.8;
}
.masonry-item img {
width: 100%;
height: 100%;
object-fit: cover;
}
/* Modal styles */
.modal {
position: fixed;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
background: rgba(0, 0, 0, 0.95);
display: none;
z-index: 1000;
justify-content: center;
align-items: center;
}
.modal.active {
display: grid;
grid-template-rows: 1fr auto;
padding: 2rem 1rem;
}
.modal-content {
position: relative;
/* max-width: 90vw; */
max-height: none;
height: 100%;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.modal img {
max-width: 100%;
max-height: 80vh;
object-fit: contain;
}
.modal-info {
color: white;
text-align: center;
margin-top: 1rem;
display: flex;
flex-direction: column;
gap: 1rem;
}
.modal-title {
font-size: 1.2rem;
font-family: monospace;
}
.modal-actions {
position: static;
transform: none;
display: flex;
flex-direction: column;
gap: 1rem;
margin-top: 1rem;
align-self: end;
justify-self: center;
}
.modal-actions-row {
display: flex;
gap: 1rem;
justify-content: center;
}
.btn {
padding: 0.5rem 1rem;
background: white;
color: black;
border: none;
cursor: pointer;
font-family: inherit;
text-decoration: none;
display: inline-block;
transition: background 0.2s ease;
}
.btn:hover {
background: #f0f0f0;
}
#imageModal {
padding: 4rem 0;
}
@media (max-width: 1400px) {
.masonry-grid {
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
}
}
@media (max-width: 1000px) {
.masonry-grid {
grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
}
}
@media (max-width: 768px) {
.masonry-grid {
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
}
}
@media (max-width: 480px) {
.masonry-grid {
grid-template-columns: 1fr 1fr;
gap: 2px;
padding: 2px;
}
/* Mobile-specific adjustments */
.modal img {
max-height: 70vh;
}
}

@ -0,0 +1,32 @@
---
type Props = {
title?: string
}
const { title } = Astro.props
---
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" type="image/svg+xml" href="/frigo_icon.png" />
<meta name="viewport" content="width=device-width" />
<meta name="generator" content={Astro.generator} />
<meta name="robots" content="noindex" />
<!-- OpenGraph Tags -->
<meta property="og:title" content="Meme AulaStud" />
<meta
property="og:description"
content="Galleria con gli scan dei meme appesi sulle pareti dell'AulaStud di Matematica dell'Università di Pisa."
/>
<meta property="og:image" content="/frigo_icon.png" />
<meta property="og:type" content="website" />
<meta property="og:url" content="https://meme.phc.dm.unipi.it" />
<title>{title ? `${title} | Meme AulaStud` : 'Meme AulaStud'}</title>
</head>
<body>
<slot />
</body>
</html>

@ -0,0 +1,308 @@
---
import Base from '@/layout/Base.astro'
---
<Base>
<div class="header">
<div class="loading">
<div class="loading-text">Loading Model <span id="progress-label">0%</span>...</div>
</div>
<div class="timeline">
<div class="event current">
<div class="event-dot"></div>
<div class="event-label">Luglio 2025</div>
</div>
<div class="event">
<div class="event-dot"></div>
<div class="event-label">???</div>
</div>
<div class="line"></div>
</div>
</div>
<canvas id="canvas"></canvas>
<div class="help">
<p>Use mouse to navigate the 3D room</p>
<p><strong>Left Click + Drag</strong>: Move view</p>
<p><strong>Right Click + Drag</strong>: Rotate view</p>
</div>
<style>
* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
html,
body {
height: 100%;
width: 100%;
}
body {
margin: 0;
overflow: hidden;
display: grid;
place-items: center;
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans',
'Helvetica Neue', sans-serif;
font-size: 15px;
background-color: #555;
color: white;
position: relative;
user-select: none;
line-height: 1;
}
canvas {
display: block;
position: absolute;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
}
.header {
position: absolute;
top: 0;
left: 50%;
transform: translateX(-50%);
z-index: 10;
display: grid;
place-content: center;
place-items: center;
align-content: start;
.loading {
background: linear-gradient(-15deg, #222, #333);
border: 1px solid #222;
border-top: none;
padding: 0.5rem 1rem;
border-bottom-left-radius: 0.5rem;
border-bottom-right-radius: 0.5rem;
box-shadow: 0 0 1rem rgba(0, 0, 0, 0.1);
.loading-text {
padding: 0.5rem 0;
}
}
.timeline {
margin-top: 0.5rem;
background: linear-gradient(-15deg, #222, #333);
border: 1px solid #222;
border-top: none;
padding: 1rem;
border-radius: 0.5rem;
box-shadow: 0 0 1rem rgba(0, 0, 0, 0.1);
position: relative;
min-width: 50vw;
max-width: 100vw;
display: grid;
grid-template-rows: auto auto;
grid-auto-columns: auto;
grid-auto-flow: dense;
place-items: center;
> .line {
position: absolute;
top: 20.5px;
left: 0;
right: 0;
width: 100%;
height: 2px;
background: linear-gradient(to right, transparent, #888, transparent);
z-index: 1;
}
> .event {
grid-row: 1 / -1;
grid-column: span 1;
z-index: 2;
display: grid;
grid-template-rows: subgrid;
grid-template-columns: auto;
place-items: center;
gap: 0.25rem;
min-width: 4rem;
> .event-dot {
grid-row: 1 / 2;
grid-column: 1 / 2;
width: 12px;
height: 12px;
background: linear-gradient(165deg, #fff, #888);
border-radius: 50%;
margin-bottom: 0.25rem;
}
> .event-label {
grid-row: 2 / 3;
grid-column: 1 / 2;
font-size: 15px;
}
cursor: pointer;
&:hover {
> .event-dot {
box-shadow: 0 0 0.75rem #ddd;
}
}
&.current {
> .event-dot {
background: linear-gradient(165deg, #c0ffbe, #196b16);
box-shadow: 0 0 0.75rem #080;
}
> .event-label {
font-weight: bold;
}
}
}
}
}
.help {
position: absolute;
bottom: 0;
left: 50%;
transform: translateX(-50%);
background: linear-gradient(-15deg, #222, #333);
border: 1px solid #222;
border-bottom: none;
z-index: 10;
padding: 0.5rem 1rem;
border-top-left-radius: 0.5rem;
border-top-right-radius: 0.5rem;
box-shadow: 0 0 1rem rgba(0, 0, 0, 0.1);
text-align: center;
padding: 1rem;
display: grid;
grid-auto-flow: row;
grid-auto-rows: auto;
gap: 0.5rem;
}
</style>
<script>
// @ts-nocheck
import * as THREE from 'three'
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader.js'
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls.js'
import { FlyControls } from 'three/examples/jsm/controls/FlyControls.js'
import { MapControls } from 'three/examples/jsm/controls/MapControls.js'
// GLB 3D room scene
document.addEventListener('DOMContentLoaded', () => {
const $canvas = document.getElementById('canvas')
const $progressLabel = document.getElementById('progress-label')
const scene = new THREE.Scene()
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000)
const renderer = new THREE.WebGLRenderer({
antialias: true,
canvas: $canvas,
})
const SCALE_FACTOR = 1
renderer.setSize($canvas.clientWidth * SCALE_FACTOR, $canvas.clientHeight * SCALE_FACTOR)
renderer.setClearColor(0x333333)
renderer.shadowMap.enabled = true
renderer.shadowMap.type = THREE.PCFSoftShadowMap
// Add lighting
const ambientLight = new THREE.AmbientLight(0x404040, 100)
scene.add(ambientLight)
// Set up camera position
camera.position.set(0, 0, 0)
// Set up controls
const controls = new MapControls(camera, renderer.domElement)
controls.enableDamping = true
controls.dampingFactor = 0.1
controls.enableZoom = false
// Load GLB model
const loader = new GLTFLoader()
loader.load(
'https://static.phc.dm.unipi.it/aulastud-2025-07.glb',
gltf => {
const model = gltf.scene
scene.add(model)
// Center the model and adjust camera
const box = new THREE.Box3().setFromObject(model)
const center = box.getCenter(new THREE.Vector3())
model.position.sub(center)
const size = box.getSize(new THREE.Vector3())
const maxDim = Math.max(size.x, size.y, size.z)
const distance = maxDim * 1.5
const s = distance * 0.1
camera.position.set(2 * s, s * 0.05, 0)
// controls.target.set(s, s * 0.1, 0)
camera.lookAt(0, 0, 0)
controls.update()
document.querySelector('.loading').style.display = 'none'
},
progress => {
console.log('Loading progress:', (progress.loaded / progress.total) * 100 + '%')
const percent = Math.round((progress.loaded / progress.total) * 100)
$progressLabel.textContent = percent + '%'
},
error => {
console.error('Error loading GLB model:', error)
}
)
// Handle window resize
window.addEventListener('resize', () => {
camera.aspect = window.innerWidth / window.innerHeight
camera.updateProjectionMatrix()
renderer.setSize(window.innerWidth, window.innerHeight)
})
function animate() {
requestAnimationFrame(animate)
controls.update()
renderer.render(scene, camera)
}
animate()
})
</script>
</Base>

@ -1,6 +1,9 @@
---
import { Image } from 'astro:assets'
import '@/homepage.css'
import Base from '@/layout/Base.astro'
// Import all JPG images from the assets/images folder
const images = import.meta.glob('@/assets/images/*.jpg', { eager: true })
const imageList = Object.entries(images).map(([path, module]) => {
@ -13,390 +16,175 @@ const imageList = Object.entries(images).map(([path, module]) => {
})
---
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" type="image/svg+xml" href="/frigo_icon.png" />
<meta name="viewport" content="width=device-width" />
<meta name="generator" content={Astro.generator} />
<meta name="robots" content="noindex">
<!-- OpenGraph Tags -->
<meta property="og:title" content="Meme AulaStud" />
<meta
property="og:description"
content="Galleria con gli scan dei meme appesi sulle pareti dell'AulaStud di Matematica dell'Università di Pisa."
/>
<meta property="og:image" content="/frigo_icon.png" />
<meta property="og:type" content="website" />
<meta property="og:url" content="https://meme.phc.dm.unipi.it" />
<title>Meme AulaStud</title>
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
font-family: system-ui, sans-serif;
background: #334;
}
.page-header {
background: #334;
text-align: center;
padding: 2rem 1rem 1rem;
z-index: 100;
}
.page-title {
color: white;
font-size: 2.5rem;
font-weight: bold;
margin: 0;
text-shadow: 0 0 8px rgba(255, 255, 255, 0.5);
}
@media (max-width: 768px) {
.page-title {
font-size: 2rem;
}
.page-header {
padding: 1.5rem 1rem 0.5rem;
}
}
@media (max-width: 480px) {
.page-title {
font-size: 1.5rem;
}
}
.masonry-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
grid-template-rows: repeat(auto, 10px);
grid-auto-flow: row dense;
padding: 0.25rem;
gap: 0.25rem;
width: 100vw;
padding-bottom: 30vh;
justify-content: center;
}
.masonry-item {
cursor: pointer;
transition: opacity 0.2s ease;
overflow: hidden;
grid-row-end: span var(--img-height);
}
.masonry-item:hover {
opacity: 0.8;
}
.masonry-item img {
width: 100%;
height: 100%;
object-fit: cover;
}
/* Modal styles */
.modal {
position: fixed;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
background: rgba(0, 0, 0, 0.95);
display: none;
z-index: 1000;
justify-content: center;
align-items: center;
}
.modal.active {
display: grid;
grid-template-rows: 1fr auto;
padding: 2rem 1rem;
}
.modal-content {
position: relative;
/* max-width: 90vw; */
max-height: none;
height: 100%;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.modal img {
max-width: 100%;
max-height: 80vh;
object-fit: contain;
}
.modal-info {
color: white;
text-align: center;
margin-top: 1rem;
display: flex;
flex-direction: column;
gap: 1rem;
}
.modal-title {
font-size: 1.2rem;
font-family: monospace;
}
.modal-actions {
position: static;
transform: none;
display: flex;
flex-direction: column;
gap: 1rem;
margin-top: 1rem;
align-self: end;
justify-self: center;
}
.modal-actions-row {
display: flex;
gap: 1rem;
justify-content: center;
}
.btn {
padding: 0.5rem 1rem;
background: white;
color: black;
border: none;
cursor: pointer;
font-family: inherit;
text-decoration: none;
display: inline-block;
transition: background 0.2s ease;
}
.btn:hover {
background: #f0f0f0;
}
#imageModal {
padding: 4rem 0;
}
@media (max-width: 1400px) {
.masonry-grid {
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
}
}
@media (max-width: 1000px) {
.masonry-grid {
grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
}
}
@media (max-width: 768px) {
.masonry-grid {
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
}
}
@media (max-width: 480px) {
.masonry-grid {
grid-template-columns: 1fr 1fr;
gap: 2px;
padding: 2px;
}
/* Mobile-specific adjustments */
.modal img {
max-height: 70vh;
}
}
</style>
</head>
<body>
<header class="page-header">
<h1 class="page-title">Meme AulaStud</h1>
</header>
<!-- Modal -->
<div class="modal" id="imageModal">
<div class="modal-content">
<img id="modalImage" src="" alt="" />
<div class="modal-info">
<div class="modal-title" id="modalTitle"></div>
</div>
</div>
<div class="modal-actions">
<div class="modal-actions-row">
<a id="downloadBtn" class="btn" download>Download</a>
<button class="btn" onclick="closeModal()">Close</button>
</div>
<div class="modal-actions-row">
<button class="btn" onclick="previousImage()">Previous</button>
<button class="btn" onclick="nextImage()">Next</button>
</div>
<Base>
<header class="page-header">
<h1 class="page-title">Meme AulaStud</h1>
</header>
<!-- Modal -->
<div class="modal" id="imageModal">
<div class="modal-content">
<img id="modalImage" src="" alt="" />
<div class="modal-info">
<div class="modal-title" id="modalTitle"></div>
</div>
</div>
<div class="masonry-grid">
{
imageList.map((image, index) => {
// Calculate the height span based on image aspect ratio
// Assuming 250px base width (from minmax), calculate proportional height
const aspectRatio = image.src.width / image.src.height
const baseWidth = 300
const calculatedHeight = Math.ceil(baseWidth / aspectRatio + 2) // +2 for gap
return (
<div
class="masonry-item"
data-index={index}
data-src={image.src.src}
data-name={image.name}
style={`--img-height: ${Math.floor(calculatedHeight / 10)}`}
>
<Image src={image.src} alt={image.name} format="webp" quality={90} loading="lazy" />
</div>
)
})
}
<div class="modal-actions">
<div class="modal-actions-row">
<a id="downloadBtn" class="btn" download>Download</a>
<button class="btn" onclick="closeModal()">Close</button>
</div>
<div class="modal-actions-row">
<button class="btn" onclick="previousImage()">Previous</button>
<button class="btn" onclick="nextImage()">Next</button>
</div>
</div>
<script is:inline>
let currentImageIndex = 0
let allItems = []
function openModal(src, name) {
const modal = document.getElementById('imageModal')
const modalImage = document.getElementById('modalImage')
const modalTitle = document.getElementById('modalTitle')
const downloadBtn = document.getElementById('downloadBtn')
if (modal && modalImage && modalTitle && downloadBtn) {
modalImage.src = src
modalImage.alt = name
modalTitle.textContent = name
downloadBtn.href = src
downloadBtn.download = name + '.jpg'
modal.classList.add('active')
document.body.style.overflow = 'hidden'
}
}
function closeModal() {
const modal = document.getElementById('imageModal')
const modalImage = document.getElementById('modalImage')
if (modal) {
modal.classList.remove('active')
document.body.style.overflow = 'auto'
// Remove the modal image element directly
if (modalImage) {
modalImage.remove()
// Create a new modal image element for next use
const newModalImage = document.createElement('img')
newModalImage.id = 'modalImage'
newModalImage.src = ''
newModalImage.alt = ''
// Insert it back into the modal content
const modalContent = modal.querySelector('.modal-content')
const modalInfo = modal.querySelector('.modal-info')
if (modalContent && modalInfo) {
modalContent.insertBefore(newModalImage, modalInfo)
}
</div>
<div class="masonry-grid">
{
imageList.map((image, index) => {
// Calculate the height span based on image aspect ratio
// Assuming 250px base width (from minmax), calculate proportional height
const aspectRatio = image.src.width / image.src.height
const baseWidth = 300
const calculatedHeight = Math.ceil(baseWidth / aspectRatio + 2) // +2 for gap
return (
<div
class="masonry-item"
data-index={index}
data-src={image.src.src}
data-name={image.name}
style={`--img-height: ${Math.floor(calculatedHeight / 10)}`}
>
<Image src={image.src} alt={image.name} format="webp" quality={90} loading="lazy" />
</div>
)
})
}
</div>
<script is:inline>
let currentImageIndex = 0
let allItems = []
function openModal(src, name) {
const modal = document.getElementById('imageModal')
const modalImage = document.getElementById('modalImage')
const modalTitle = document.getElementById('modalTitle')
const downloadBtn = document.getElementById('downloadBtn')
if (modal && modalImage && modalTitle && downloadBtn) {
modalImage.src = src
modalImage.alt = name
modalTitle.textContent = name
downloadBtn.href = src
downloadBtn.download = name + '.jpg'
modal.classList.add('active')
document.body.style.overflow = 'hidden'
}
}
function closeModal() {
const modal = document.getElementById('imageModal')
const modalImage = document.getElementById('modalImage')
if (modal) {
modal.classList.remove('active')
document.body.style.overflow = 'auto'
// Remove the modal image element directly
if (modalImage) {
modalImage.remove()
// Create a new modal image element for next use
const newModalImage = document.createElement('img')
newModalImage.id = 'modalImage'
newModalImage.src = ''
newModalImage.alt = ''
// Insert it back into the modal content
const modalContent = modal.querySelector('.modal-content')
const modalInfo = modal.querySelector('.modal-info')
if (modalContent && modalInfo) {
modalContent.insertBefore(newModalImage, modalInfo)
}
}
}
}
function previousImage() {
if (allItems.length === 0) return
function previousImage() {
if (allItems.length === 0) return
currentImageIndex = (currentImageIndex - 1 + allItems.length) % allItems.length
const item = allItems[currentImageIndex]
const src = item.dataset.src
const name = item.dataset.name
currentImageIndex = (currentImageIndex - 1 + allItems.length) % allItems.length
const item = allItems[currentImageIndex]
const src = item.dataset.src
const name = item.dataset.name
if (src && name) {
openModal(src, name)
}
if (src && name) {
openModal(src, name)
}
}
function nextImage() {
if (allItems.length === 0) return
function nextImage() {
if (allItems.length === 0) return
currentImageIndex = (currentImageIndex + 1) % allItems.length
const item = allItems[currentImageIndex]
const src = item.dataset.src
const name = item.dataset.name
currentImageIndex = (currentImageIndex + 1) % allItems.length
const item = allItems[currentImageIndex]
const src = item.dataset.src
const name = item.dataset.name
if (src && name) {
openModal(src, name)
}
if (src && name) {
openModal(src, name)
}
}
// Add click listeners to all masonry items
document.addEventListener('DOMContentLoaded', function () {
const items = document.querySelectorAll('.masonry-item')
allItems = Array.from(items)
// Add click listeners to all masonry items
document.addEventListener('DOMContentLoaded', function () {
const items = document.querySelectorAll('.masonry-item')
allItems = Array.from(items)
// Set up click listeners
items.forEach(function (item, index) {
item.addEventListener('click', function () {
const src = item.dataset.src
const name = item.dataset.name
if (src && name) {
currentImageIndex = index
openModal(src, name)
}
})
// Set up click listeners
items.forEach(function (item, index) {
item.addEventListener('click', function () {
const src = item.dataset.src
const name = item.dataset.name
if (src && name) {
currentImageIndex = index
openModal(src, name)
}
})
})
// Close modal on background click
const modal = document.getElementById('imageModal')
if (modal) {
modal.addEventListener('click', function (e) {
if (e.target === modal) {
closeModal()
}
})
}
// Close modal on Escape key and add navigation
document.addEventListener('keydown', function (e) {
const modal = document.getElementById('imageModal')
const isModalOpen = modal && modal.classList.contains('active')
if (!isModalOpen) return
if (e.key === 'Escape') {
// Close modal on background click
const modal = document.getElementById('imageModal')
if (modal) {
modal.addEventListener('click', function (e) {
if (e.target === modal) {
closeModal()
} else if (e.key === 'ArrowLeft') {
e.preventDefault()
previousImage()
} else if (e.key === 'ArrowRight') {
e.preventDefault()
nextImage()
}
})
}
// Close modal on Escape key and add navigation
document.addEventListener('keydown', function (e) {
const modal = document.getElementById('imageModal')
const isModalOpen = modal && modal.classList.contains('active')
if (!isModalOpen) return
if (e.key === 'Escape') {
closeModal()
} else if (e.key === 'ArrowLeft') {
e.preventDefault()
previousImage()
} else if (e.key === 'ArrowRight') {
e.preventDefault()
nextImage()
}
})
</script>
</body>
</html>
})
</script>
</Base>

Loading…
Cancel
Save