added 3d model page
parent
c0a941fac6
commit
edcf04bf31
@ -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>
|
||||||
Loading…
Reference in New Issue