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