html
Switzerland Alps SAINT ANTONIEN Tucked away in the Switzerland Alps, Saint Antönien offers an idyllic retreat for those seeking tranquility and adventure alike. It's a hidden gem for backcountry skiing in winter and boasts lush trails for hiking and mountain biking during the warmer months. Switzerland Alps SAINT ANTONIEN Tucked away in the Switzerland Alps, Saint Antönien offers an idyllic retreat for those seeking tranquility and adventure alike. It's a hidden gem for backcountry skiing in winter and boasts lush trails for hiking and mountain biking during the warmer months.
css
@import url("https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&family=Oswald:wght@500&display=swap"); $primary-color: #ecad29; $text-color: #FFFFFFDD; body { margin: 0; background-color: #1a1a1a; color: $text-color; position: relative; overflow: hidden; font-family: "Inter", sans-serif; } .card { position: absolute; left: 0; top: 0; background-position: center; background-size: cover; box-shadow: 6px 6px 10px 2px rgba(0, 0, 0, 0.6); } #btn { position: absolute; top: 690px; left: 16px; z-index: 99; } .card-content { position: absolute; left: 0; top: 0; color: $text-color; padding-left: 16px; } .content-place { margin-top: 6px; font-size: 13px; font-weight: 500; } .content-place { font-weight: 500; } .content-title-1, .content-title-2 { font-weight: 600; font-size: 20px; font-family: "Oswald", sans-serif; } .content-start { width: 30px; height: 5px; border-radius: 99px; background-color: $text-color; } .details { z-index: 22; position: absolute; top: 240px; left: 60px; .place-box { height: 46px; overflow: hidden; .text { padding-top: 16px; font-size: 20px; &:before { top: 0; left: 0; position: absolute; content: ""; width: 30px; height: 4px; border-radius: 99px; background-color: white; } } } .title-1, .title-2 { font-weight: 600; font-size: 72px; font-family: "Oswald", sans-serif; } .title-box-1, .title-box-2 { margin-top: 2px; height: 100px; overflow: hidden; // background-color: blue; } > .desc { margin-top: 16px; width: 500px; } > .cta { width: 500px; margin-top: 24px; display: flex; align-items: center; > .bookmark { border: none; background-color: $primary-color; width: 36px; height: 36px; border-radius: 99px; color: white; display: grid; place-items: center; svg { width: 20px; height: 20px; } } > .discover { border: 1px solid #ffffff; background-color: transparent; height: 36px; border-radius: 99px; color: #ffffff; padding: 4px 24px; font-size: 12px; margin-left: 16px; text-transform: uppercase; } } } nav { position: fixed; left: 0; top: 0; right: 0; z-index: 50; display: flex; align-items: center; justify-content: space-between; padding: 20px 36px; font-weight: 500; svg { width: 20px; height: 20px; } .svg-container { width: 20px; height: 20px; } > div { display: inline-flex; align-items: center; text-transform: uppercase; font-size: 14px; &:first-child { gap: 10px; } &:last-child { gap: 24px; > .active { position: relative; &:after { bottom: -8px; left: 0; right: 0; position: absolute; content: ""; height: 3px; border-radius: 99px; background-color: $primary-color; } } } } } .indicator { position: fixed; left: 0; right: 0; top: 0; height: 5px; z-index: 60; background-color: $primary-color; } .pagination { position: absolute; left: 0px; top: 0px; display: inline-flex; > .arrow { z-index: 60; width: 50px; height: 50px; border-radius: 999px; border: 2px solid #ffffff55; display: grid; place-items: center; &:nth-child(2){ margin-left: 20px; } svg { width: 24px; height: 24px; stroke-width: 2; color: #ffffff99; } } .progress-sub-container{ margin-left: 24px; z-index: 60; width: 500px; height: 50px; display: flex; align-items: center; .progress-sub-background{ width: 500px; height: 3px; background-color: #ffffff33; .progress-sub-foreground{ height: 3px; background-color: $primary-color; } } } .slide-numbers{ width: 50px; height: 50px; overflow: hidden; // background-color: #111111; z-index: 60; position: relative; .item{ width: 50px; height: 50px; position: absolute; // background: rgb(31, 31, 41); color: white; top: 0; left: 0; display: grid; place-items: center; font-size: 32px; font-weight: bold; } } } .cover{ position: absolute; left: 0; top: 0; width: 100vw; height: 100vh; background-color: #fff; z-index: 100; }
js
const data = [ { place:'Switzerland Alps', title:'SAINT', title2:'ANTONIEN', description:'Tucked away in the Switzerland Alps, Saint Antönien offers an idyllic retreat for those seeking tranquility and adventure alike. It\'s a hidden gem for backcountry skiing in winter and boasts lush trails for hiking and mountain biking during the warmer months.', image:'https://assets.codepen.io/3685267/timed-cards-1.jpg' }, { place:'Japan Alps', title:'NANGANO', title2:'PREFECTURE', description:'Nagano Prefecture, set within the majestic Japan Alps, is a cultural treasure trove with its historic shrines and temples, particularly the famous Zenkō-ji. The region is also a hotspot for skiing and snowboarding, offering some of the country\'s best powder.', image:'https://assets.codepen.io/3685267/timed-cards-2.jpg' }, { place:'Sahara Desert - Morocco', title:'MARRAKECH', title2:'MEROUGA', description:'The journey from the vibrant souks and palaces of Marrakech to the tranquil, starlit sands of Merzouga showcases the diverse splendor of Morocco. Camel treks and desert camps offer an unforgettable immersion into the nomadic way of life.', image:'https://assets.codepen.io/3685267/timed-cards-3.jpg' }, { place:'Sierra Nevada - USA', title:'YOSEMITE', title2:'NATIONAL PARAK', description:'Yosemite National Park is a showcase of the American wilderness, revered for its towering granite monoliths, ancient giant sequoias, and thundering waterfalls. The park offers year-round recreational activities, from rock climbing to serene valley walks.', image:'https://assets.codepen.io/3685267/timed-cards-4.jpg' }, { place:'Tarifa - Spain', title:'LOS LANCES', title2:'BEACH', description:'Los Lances Beach in Tarifa is a coastal paradise known for its consistent winds, making it a world-renowned spot for kitesurfing and windsurfing. The beach\'s long, sandy shores provide ample space for relaxation and sunbathing, with a vibrant atmosphere of beach bars and cafes.', image:'https://assets.codepen.io/3685267/timed-cards-5.jpg' }, { place:'Cappadocia - Turkey', title:'Göreme', title2:'Valley', description:'Göreme Valley in Cappadocia is a historical marvel set against a unique geological backdrop, where centuries of wind and water have sculpted the landscape into whimsical formations. The valley is also famous for its open-air museums, underground cities, and the enchanting experience of hot air ballooning.', image:'https://assets.codepen.io/3685267/timed-cards-6.jpg' }, ] const _ = (id)=>document.getElementById(id) const cards = data.map((i, index)=>``).join('') const cardContents = data.map((i, index)=>` ${i.place} ${i.title} ${i.title2} `).join('') const sildeNumbers = data.map((_, index)=>`${index+1}`).join('') _('demo').innerHTML = cards + cardContents _('slide-numbers').innerHTML = sildeNumbers const range = (n) => Array(n) .fill(0) .map((i, j) => i + j); const set = gsap.set; function getCard(index) { return `#card${index}`; } function getCardContent(index) { return `#card-content-${index}`; } function getSliderItem(index) { return `#slide-item-${index}`; } function animate(target, duration, properties) { return new Promise((resolve) => { gsap.to(target, { ...properties, duration: duration, onComplete: resolve, }); }); } let order = [0, 1, 2, 3, 4, 5]; let detailsEven = true; let offsetTop = 200; let offsetLeft = 700; let cardWidth = 200; let cardHeight = 300; let gap = 40; let numberSize = 50; const ease = "sine.inOut"; function init() { const [active, ...rest] = order; const detailsActive = detailsEven ? "#details-even" : "#details-odd"; const detailsInactive = detailsEven ? "#details-odd" : "#details-even"; const { innerHeight: height, innerWidth: width } = window; offsetTop = height - 430; offsetLeft = width - 830; gsap.set("#pagination", { top: offsetTop + 330, left: offsetLeft, y: 200, opacity: 0, zIndex: 60, }); gsap.set("nav", { y: -200, opacity: 0 }); gsap.set(getCard(active), { x: 0, y: 0, width: window.innerWidth, height: window.innerHeight, }); gsap.set(getCardContent(active), { x: 0, y: 0, opacity: 0 }); gsap.set(detailsActive, { opacity: 0, zIndex: 22, x: -200 }); gsap.set(detailsInactive, { opacity: 0, zIndex: 12 }); gsap.set(`${detailsInactive} .text`, { y: 100 }); gsap.set(`${detailsInactive} .title-1`, { y: 100 }); gsap.set(`${detailsInactive} .title-2`, { y: 100 }); gsap.set(`${detailsInactive} .desc`, { y: 50 }); gsap.set(`${detailsInactive} .cta`, { y: 60 }); gsap.set(".progress-sub-foreground", { width: 500 * (1 / order.length) * (active + 1), }); rest.forEach((i, index) => { gsap.set(getCard(i), { x: offsetLeft + 400 + index * (cardWidth + gap), y: offsetTop, width: cardWidth, height: cardHeight, zIndex: 30, borderRadius: 10, }); gsap.set(getCardContent(i), { x: offsetLeft + 400 + index * (cardWidth + gap), zIndex: 40, y: offsetTop + cardHeight - 100, }); gsap.set(getSliderItem(i), { x: (index + 1) * numberSize }); }); gsap.set(".indicator", { x: -window.innerWidth }); const startDelay = 0.6; gsap.to(".cover", { x: width + 400, delay: 0.5, ease, onComplete: () => { setTimeout(() => { loop(); }, 500); }, }); rest.forEach((i, index) => { gsap.to(getCard(i), { x: offsetLeft + index * (cardWidth + gap), zIndex: 30, delay: 0.05 * index, ease, delay: startDelay, }); gsap.to(getCardContent(i), { x: offsetLeft + index * (cardWidth + gap), zIndex: 40, delay: 0.05 * index, ease, delay: startDelay, }); }); gsap.to("#pagination", { y: 0, opacity: 1, ease, delay: startDelay }); gsap.to("nav", { y: 0, opacity: 1, ease, delay: startDelay }); gsap.to(detailsActive, { opacity: 1, x: 0, ease, delay: startDelay }); } let clicks = 0; function step() { return new Promise((resolve) => { order.push(order.shift()); detailsEven = !detailsEven; const detailsActive = detailsEven ? "#details-even" : "#details-odd"; const detailsInactive = detailsEven ? "#details-odd" : "#details-even"; document.querySelector(`${detailsActive} .place-box .text`).textContent = data[order[0]].place; document.querySelector(`${detailsActive} .title-1`).textContent = data[order[0]].title; document.querySelector(`${detailsActive} .title-2`).textContent = data[order[0]].title2; document.querySelector(`${detailsActive} .desc`).textContent = data[order[0]].description; gsap.set(detailsActive, { zIndex: 22 }); gsap.to(detailsActive, { opacity: 1, delay: 0.4, ease }); gsap.to(`${detailsActive} .text`, { y: 0, delay: 0.1, duration: 0.7, ease, }); gsap.to(`${detailsActive} .title-1`, { y: 0, delay: 0.15, duration: 0.7, ease, }); gsap.to(`${detailsActive} .title-2`, { y: 0, delay: 0.15, duration: 0.7, ease, }); gsap.to(`${detailsActive} .desc`, { y: 0, delay: 0.3, duration: 0.4, ease, }); gsap.to(`${detailsActive} .cta`, { y: 0, delay: 0.35, duration: 0.4, onComplete: resolve, ease, }); gsap.set(detailsInactive, { zIndex: 12 }); const [active, ...rest] = order; const prv = rest[rest.length - 1]; gsap.set(getCard(prv), { zIndex: 10 }); gsap.set(getCard(active), { zIndex: 20 }); gsap.to(getCard(prv), { scale: 1.5, ease }); gsap.to(getCardContent(active), { y: offsetTop + cardHeight - 10, opacity: 0, duration: 0.3, ease, }); gsap.to(getSliderItem(active), { x: 0, ease }); gsap.to(getSliderItem(prv), { x: -numberSize, ease }); gsap.to(".progress-sub-foreground", { width: 500 * (1 / order.length) * (active + 1), ease, }); gsap.to(getCard(active), { x: 0, y: 0, ease, width: window.innerWidth, height: window.innerHeight, borderRadius: 0, onComplete: () => { const xNew = offsetLeft + (rest.length - 1) * (cardWidth + gap); gsap.set(getCard(prv), { x: xNew, y: offsetTop, width: cardWidth, height: cardHeight, zIndex: 30, borderRadius: 10, scale: 1, }); gsap.set(getCardContent(prv), { x: xNew, y: offsetTop + cardHeight - 100, opacity: 1, zIndex: 40, }); gsap.set(getSliderItem(prv), { x: rest.length * numberSize }); gsap.set(detailsInactive, { opacity: 0 }); gsap.set(`${detailsInactive} .text`, { y: 100 }); gsap.set(`${detailsInactive} .title-1`, { y: 100 }); gsap.set(`${detailsInactive} .title-2`, { y: 100 }); gsap.set(`${detailsInactive} .desc`, { y: 50 }); gsap.set(`${detailsInactive} .cta`, { y: 60 }); clicks -= 1; if (clicks > 0) { step(); } }, }); rest.forEach((i, index) => { if (i !== prv) { const xNew = offsetLeft + index * (cardWidth + gap); gsap.set(getCard(i), { zIndex: 30 }); gsap.to(getCard(i), { x: xNew, y: offsetTop, width: cardWidth, height: cardHeight, ease, delay: 0.1 * (index + 1), }); gsap.to(getCardContent(i), { x: xNew, y: offsetTop + cardHeight - 100, opacity: 1, zIndex: 40, ease, delay: 0.1 * (index + 1), }); gsap.to(getSliderItem(i), { x: (index + 1) * numberSize, ease }); } }); }); } async function loop() { await animate(".indicator", 2, { x: 0 }); await animate(".indicator", 0.8, { x: window.innerWidth, delay: 0.3 }); set(".indicator", { x: -window.innerWidth }); await step(); loop(); } async function loadImage(src) { return new Promise((resolve, reject) => { let img = new Image(); img.onload = () => resolve(img); img.onerror = reject; img.src = src; }); } async function loadImages() { const promises = data.map(({ image }) => loadImage(image)); return Promise.all(promises); } async function start() { try { await loadImages(); init(); } catch (error) { console.error("One or more images failed to load", error); } } start()
还没有评论,来说两句吧...