Spaces:
Running
Running
<html> | |
<head> | |
<script type="module" src="https://unpkg.com/@google/model-viewer/dist/model-viewer.min.js"></script> | |
<style> | |
html, body { | |
width: 100vw; | |
height: 100vh; | |
margin: 0; | |
padding: 0; | |
font-family: Arial, sans-serif; | |
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); | |
} | |
.container { | |
width: 100vw; | |
height: 100vh; | |
display: flex; | |
flex-direction: column; | |
align-items: center; | |
justify-content: center; | |
} | |
.selector-panel { | |
background: rgba(255, 255, 255, 0.95); | |
padding: 2rem; | |
border-radius: 15px; | |
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.3); | |
text-align: center; | |
backdrop-filter: blur(10px); | |
max-width: 500px; | |
margin: 20px; | |
} | |
.selector-panel h1 { | |
color: #333; | |
margin-bottom: 1.5rem; | |
font-size: 1.8rem; | |
} | |
.file-option { | |
display: block; | |
width: 100%; | |
padding: 1rem 1.5rem; | |
margin: 1rem 0; | |
background: linear-gradient(45deg, #4facfe 0%, #00f2fe 100%); | |
color: white; | |
border: none; | |
border-radius: 25px; | |
cursor: pointer; | |
font-size: 1.1rem; | |
font-weight: bold; | |
transition: all 0.3s ease; | |
text-decoration: none; | |
box-shadow: 0 4px 15px rgba(79, 172, 254, 0.4); | |
} | |
.file-option:hover { | |
transform: translateY(-2px); | |
box-shadow: 0 6px 20px rgba(79, 172, 254, 0.6); | |
} | |
.file-option:active { | |
transform: translateY(0); | |
} | |
.file-option.secondary { | |
background: linear-gradient(45deg, #fa709a 0%, #fee140 100%); | |
box-shadow: 0 4px 15px rgba(250, 112, 154, 0.4); | |
} | |
.file-option.secondary:hover { | |
box-shadow: 0 6px 20px rgba(250, 112, 154, 0.6); | |
} | |
model-viewer { | |
width: 100vw; | |
height: 100vh; | |
display: none; | |
} | |
model-viewer.active { | |
display: block; | |
} | |
.back-button { | |
position: fixed; | |
top: 20px; | |
left: 20px; | |
background: rgba(255, 255, 255, 0.9); | |
border: none; | |
padding: 10px 20px; | |
border-radius: 25px; | |
cursor: pointer; | |
font-weight: bold; | |
box-shadow: 0 4px 15px rgba(0, 0, 0, 0.2); | |
transition: all 0.3s ease; | |
z-index: 1000; | |
display: none; | |
align-items: center; | |
gap: 8px; | |
} | |
.back-button:hover { | |
background: white; | |
transform: translateY(-1px); | |
box-shadow: 0 6px 20px rgba(0, 0, 0, 0.3); | |
} | |
.back-button.active { | |
display: flex; | |
} | |
.back-arrow { | |
display: inline-block; | |
width: 0; | |
height: 0; | |
border-top: 6px solid transparent; | |
border-bottom: 6px solid transparent; | |
border-right: 10px solid #333; | |
margin-right: 5px; | |
} | |
.file-description { | |
font-size: 0.9rem; | |
color: #666; | |
margin-top: 0.5rem; | |
} | |
.loading-overlay { | |
position: fixed; | |
top: 0; | |
left: 0; | |
width: 100vw; | |
height: 100vh; | |
background: rgba(0, 0, 0, 0.8); | |
display: none; | |
flex-direction: column; | |
align-items: center; | |
justify-content: center; | |
z-index: 2000; | |
color: white; | |
} | |
.loading-spinner { | |
width: 50px; | |
height: 50px; | |
border: 4px solid rgba(255, 255, 255, 0.3); | |
border-top: 4px solid white; | |
border-radius: 50%; | |
animation: spin 1s linear infinite; | |
margin-bottom: 20px; | |
} | |
.loading-text { | |
font-size: 1.2rem; | |
font-weight: bold; | |
} | |
@keyframes spin { | |
0% { transform: rotate(0deg); } | |
100% { transform: rotate(360deg); } | |
} | |
</style> | |
</head> | |
<body> | |
<div class="container"> | |
<div class="selector-panel" id="selector"> | |
<h1>Choose a 3D Model to View</h1> | |
<button class="file-option" onclick="loadModel('nordkapp.glb')"> | |
Nordkapp | |
<div class="file-description">Machine room</div> | |
</button> | |
<button class="file-option secondary" onclick="loadModel('polarlys.glb')"> | |
Polarlys | |
<div class="file-description">Machine room</div> | |
</button> | |
</div> | |
</div> | |
<button class="back-button" id="backButton" onclick="showSelector()"> | |
<span class="back-arrow"></span> | |
Back to Selection | |
</button> | |
<model-viewer id="modelViewer" auto-rotate camera-controls></model-viewer> | |
<div class="loading-overlay" id="loadingOverlay"> | |
<div class="loading-spinner"></div> | |
<div class="loading-text">Loading 3D Model...</div> | |
</div> | |
<script> | |
function loadModel(filename) { | |
const modelViewer = document.getElementById('modelViewer'); | |
const selector = document.getElementById('selector'); | |
const backButton = document.getElementById('backButton'); | |
const loadingOverlay = document.getElementById('loadingOverlay'); | |
// Show loading overlay | |
loadingOverlay.style.display = 'flex'; | |
// Hide selector and show model viewer | |
selector.style.display = 'none'; | |
modelViewer.classList.add('active'); | |
backButton.classList.add('active'); | |
// Load the selected model | |
modelViewer.src = filename; | |
// Listen for load events | |
modelViewer.addEventListener('load', function() { | |
console.log('Model loaded successfully:', filename); | |
loadingOverlay.style.display = 'none'; | |
}, { once: true }); | |
modelViewer.addEventListener('error', function(event) { | |
console.error('Error loading model:', filename, event); | |
loadingOverlay.style.display = 'none'; | |
alert('Error loading model: ' + filename + '\nPlease check if the file exists and is valid.'); | |
}, { once: true }); | |
// Timeout after 240 seconds | |
setTimeout(() => { | |
if (loadingOverlay.style.display !== 'none') { | |
console.warn('Model loading timeout:', filename); | |
loadingOverlay.style.display = 'none'; | |
alert('Loading timeout. The model file might be too large or there might be a network issue.'); | |
} | |
}, 240000); | |
} | |
function showSelector() { | |
const modelViewer = document.getElementById('modelViewer'); | |
const selector = document.getElementById('selector'); | |
const backButton = document.getElementById('backButton'); | |
// Show selector and hide model viewer | |
selector.style.display = 'block'; | |
modelViewer.classList.remove('active'); | |
backButton.classList.remove('active'); | |
// Clear the model source | |
modelViewer.src = ''; | |
} | |
</script> | |
</body> | |
</html> |