new onepager draft for v3 #8
7
threefold_v3_onepager/README.md
Normal file
@ -0,0 +1,7 @@
|
||||
# Deploy Locally
|
||||
|
||||
```
|
||||
git clone https://git.ourworld.tf/tfgrid/www_projectmycelium_io/
|
||||
cd www_projectmycelium_io/threefold_v3_onepager
|
||||
xdg-open index.html
|
||||
```
|
BIN
threefold_v3_onepager/img/1.png
Normal file
After Width: | Height: | Size: 39 KiB |
BIN
threefold_v3_onepager/img/10.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
threefold_v3_onepager/img/11.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
threefold_v3_onepager/img/12.png
Normal file
After Width: | Height: | Size: 71 KiB |
BIN
threefold_v3_onepager/img/13.png
Normal file
After Width: | Height: | Size: 71 KiB |
BIN
threefold_v3_onepager/img/14.png
Normal file
After Width: | Height: | Size: 39 KiB |
BIN
threefold_v3_onepager/img/15.png
Normal file
After Width: | Height: | Size: 33 KiB |
BIN
threefold_v3_onepager/img/16.png
Normal file
After Width: | Height: | Size: 13 KiB |
BIN
threefold_v3_onepager/img/17.png
Normal file
After Width: | Height: | Size: 39 KiB |
BIN
threefold_v3_onepager/img/2.png
Normal file
After Width: | Height: | Size: 23 KiB |
BIN
threefold_v3_onepager/img/3.png
Normal file
After Width: | Height: | Size: 44 KiB |
BIN
threefold_v3_onepager/img/4.png
Normal file
After Width: | Height: | Size: 27 KiB |
BIN
threefold_v3_onepager/img/5.png
Normal file
After Width: | Height: | Size: 120 KiB |
BIN
threefold_v3_onepager/img/6.png
Normal file
After Width: | Height: | Size: 132 KiB |
BIN
threefold_v3_onepager/img/7.png
Normal file
After Width: | Height: | Size: 15 KiB |
BIN
threefold_v3_onepager/img/8.png
Normal file
After Width: | Height: | Size: 95 KiB |
BIN
threefold_v3_onepager/img/9.png
Normal file
After Width: | Height: | Size: 28 KiB |
BIN
threefold_v3_onepager/img/myceliumdate_blackbg.gif
Normal file
After Width: | Height: | Size: 9.1 KiB |
341
threefold_v3_onepager/index.html
Normal file
@ -0,0 +1,341 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>Project Mycelium</title>
|
||||
<style>
|
||||
* {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
body {
|
||||
background-color: #222222;
|
||||
color: #e2e8f0;
|
||||
font-family: system-ui, -apple-system, sans-serif;
|
||||
line-height: 1.6;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.content-container {
|
||||
height: 100vh;
|
||||
overflow: hidden;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.scroll-content {
|
||||
position: relative;
|
||||
transition: transform 1s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
.landing-section {
|
||||
height: 100vh;
|
||||
width: 100vw;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.header-container {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.header-image {
|
||||
width: 100%;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.gif-container {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.gif-image {
|
||||
max-width: 100%;
|
||||
max-height: 100%;
|
||||
object-fit: contain;
|
||||
}
|
||||
|
||||
.image-section {
|
||||
position: relative;
|
||||
height: 100vh;
|
||||
width: 100vw;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
opacity: 0;
|
||||
transform: translateY(40px);
|
||||
transition: all 1.5s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
/* Special handling for footer section */
|
||||
.image-section:last-child {
|
||||
align-items: flex-end;
|
||||
justify-content: flex-end;
|
||||
}
|
||||
|
||||
.image-section:last-child .image-container {
|
||||
height: auto;
|
||||
}
|
||||
|
||||
.image-section.visible {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
|
||||
.image-section.entering-up {
|
||||
opacity: 0;
|
||||
transform: translateY(-40px);
|
||||
}
|
||||
|
||||
.image-section.entering-down {
|
||||
opacity: 0;
|
||||
transform: translateY(40px);
|
||||
}
|
||||
|
||||
.image-section.leaving-up {
|
||||
opacity: 0;
|
||||
transform: translateY(-40px);
|
||||
}
|
||||
|
||||
.image-section.leaving-down {
|
||||
opacity: 0;
|
||||
transform: translateY(40px);
|
||||
}
|
||||
|
||||
.image-container {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.scroll-image {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
object-fit: contain;
|
||||
}
|
||||
|
||||
.image-section:last-child .scroll-image {
|
||||
object-fit: cover;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.progress-bar {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 4px;
|
||||
background: #1e293b;
|
||||
z-index: 1000;
|
||||
}
|
||||
|
||||
.progress {
|
||||
height: 100%;
|
||||
width: 0%;
|
||||
background: linear-gradient(90deg, #60a5fa, #93c5fd);
|
||||
transition: width 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="progress-bar">
|
||||
<div class="progress"></div>
|
||||
</div>
|
||||
|
||||
<div class="content-container">
|
||||
<div class="scroll-content">
|
||||
<!-- Landing section with header and GIF -->
|
||||
<div class="landing-section">
|
||||
<div class="header-container">
|
||||
<img src="./img/1.png" alt="Header" class="header-image">
|
||||
</div>
|
||||
<div class="gif-container">
|
||||
<img src="./img/myceliumdate_blackbg.gif" alt="GIF" class="gif-image">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Images 3-17 will be inserted here via JavaScript -->
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
const content = document.querySelector('.scroll-content');
|
||||
const landingSection = document.querySelector('.landing-section');
|
||||
|
||||
// Create image sections starting from image 3
|
||||
for (let i = 3; i <= 17; i++) {
|
||||
const section = document.createElement('div');
|
||||
section.className = 'image-section';
|
||||
|
||||
const imageContainer = document.createElement('div');
|
||||
imageContainer.className = 'image-container';
|
||||
|
||||
const img = document.createElement('img');
|
||||
img.src = `./img/${i}.png`;
|
||||
img.alt = `Image ${i}`;
|
||||
img.className = 'scroll-image';
|
||||
|
||||
imageContainer.appendChild(img);
|
||||
section.appendChild(imageContainer);
|
||||
content.appendChild(section);
|
||||
}
|
||||
|
||||
const sections = document.querySelectorAll('.image-section');
|
||||
const progressBar = document.querySelector('.progress');
|
||||
|
||||
let currentSection = 0;
|
||||
let isScrolling = false;
|
||||
let lastScrollTime = Date.now();
|
||||
const scrollCooldown = 1000;
|
||||
|
||||
function updateProgress() {
|
||||
const progress = (currentSection / sections.length) * 100;
|
||||
progressBar.style.width = `${progress}%`;
|
||||
}
|
||||
|
||||
function showSection(index, direction) {
|
||||
// For index 0, show the landing section
|
||||
if (index === 0) {
|
||||
content.style.transform = 'translateY(0)';
|
||||
} else {
|
||||
// For other indices, offset by the height of one section
|
||||
// and subtract 1 from index since we're counting from the sections after landing
|
||||
content.style.transform = `translateY(-${index * 100}vh)`;
|
||||
}
|
||||
|
||||
// Reset all sections
|
||||
sections.forEach(section => {
|
||||
section.classList.remove('visible', 'entering-up', 'entering-down', 'leaving-up', 'leaving-down');
|
||||
});
|
||||
|
||||
if (index > 0) {
|
||||
// Current section (subtract 1 since sections array starts after landing)
|
||||
const currentSection = sections[index - 1];
|
||||
currentSection.classList.remove(direction === 'up' ? 'entering-up' : 'entering-down');
|
||||
currentSection.classList.add('visible');
|
||||
|
||||
// Previous section
|
||||
if (index > 1 && direction === 'down') {
|
||||
sections[index - 2].classList.add('leaving-up');
|
||||
}
|
||||
|
||||
// Next section
|
||||
if (index < sections.length && direction === 'up') {
|
||||
sections[index].classList.add('leaving-down');
|
||||
}
|
||||
|
||||
// Prepare next section for entry
|
||||
if (index < sections.length) {
|
||||
if (sections[index]) sections[index].classList.add('entering-down');
|
||||
}
|
||||
|
||||
// Prepare previous section for entry
|
||||
if (index > 1) {
|
||||
sections[index - 2].classList.add('entering-up');
|
||||
}
|
||||
}
|
||||
|
||||
updateProgress();
|
||||
}
|
||||
|
||||
function handleScroll(event) {
|
||||
event.preventDefault();
|
||||
|
||||
const now = Date.now();
|
||||
if (isScrolling || now - lastScrollTime < scrollCooldown) return;
|
||||
|
||||
const delta = event.deltaY;
|
||||
|
||||
if (delta > 0 && currentSection < sections.length) {
|
||||
isScrolling = true;
|
||||
currentSection++;
|
||||
showSection(currentSection, 'down');
|
||||
} else if (delta < 0 && currentSection > 0) {
|
||||
isScrolling = true;
|
||||
currentSection--;
|
||||
showSection(currentSection, 'up');
|
||||
}
|
||||
|
||||
lastScrollTime = now;
|
||||
setTimeout(() => {
|
||||
isScrolling = false;
|
||||
}, scrollCooldown);
|
||||
}
|
||||
|
||||
function handleKeydown(event) {
|
||||
const now = Date.now();
|
||||
if (isScrolling || now - lastScrollTime < scrollCooldown) return;
|
||||
|
||||
if (event.key === 'ArrowDown' && currentSection < sections.length) {
|
||||
event.preventDefault();
|
||||
isScrolling = true;
|
||||
currentSection++;
|
||||
showSection(currentSection, 'down');
|
||||
} else if (event.key === 'ArrowUp' && currentSection > 0) {
|
||||
event.preventDefault();
|
||||
isScrolling = true;
|
||||
currentSection--;
|
||||
showSection(currentSection, 'up');
|
||||
}
|
||||
|
||||
lastScrollTime = now;
|
||||
setTimeout(() => {
|
||||
isScrolling = false;
|
||||
}, scrollCooldown);
|
||||
}
|
||||
|
||||
// Initialize
|
||||
showSection(0, 'down');
|
||||
|
||||
// Event listeners
|
||||
window.addEventListener('wheel', handleScroll, { passive: false });
|
||||
window.addEventListener('keydown', handleKeydown);
|
||||
|
||||
// Touch events for mobile
|
||||
let touchStartY = 0;
|
||||
let touchEndY = 0;
|
||||
|
||||
window.addEventListener('touchstart', (e) => {
|
||||
touchStartY = e.touches[0].clientY;
|
||||
}, { passive: false });
|
||||
|
||||
window.addEventListener('touchmove', (e) => {
|
||||
e.preventDefault();
|
||||
}, { passive: false });
|
||||
|
||||
window.addEventListener('touchend', (e) => {
|
||||
touchEndY = e.changedTouches[0].clientY;
|
||||
const now = Date.now();
|
||||
|
||||
if (isScrolling || now - lastScrollTime < scrollCooldown) return;
|
||||
|
||||
const delta = touchStartY - touchEndY;
|
||||
|
||||
if (Math.abs(delta) > 50) {
|
||||
if (delta > 0 && currentSection < sections.length) {
|
||||
isScrolling = true;
|
||||
currentSection++;
|
||||
showSection(currentSection, 'down');
|
||||
} else if (delta < 0 && currentSection > 0) {
|
||||
isScrolling = true;
|
||||
currentSection--;
|
||||
showSection(currentSection, 'up');
|
||||
}
|
||||
|
||||
lastScrollTime = now;
|
||||
setTimeout(() => {
|
||||
isScrolling = false;
|
||||
}, scrollCooldown);
|
||||
}
|
||||
}, { passive: false });
|
||||
</script>
|
||||
</body>
|
||||
</html>
|