Es geht auch ohne Javascript Framework
oder wie baut man eine Karusel mit (Vanilla) Javascript
Über mich
-
Robert Mittl
-
Stuttgart, Germany
-
selbständig seit 2012
-
Web Entwickler
-
JUG Stuttgart
E-Mail: info@mittl-medien.de
Twitter: @mittlmedien
JS Frameworks und Libraries
Drei Fraktionen:
-
Hey Cool: React, Vue, Angular 😎
-
Weinende: ich will nicht ohne jQuery 😫
-
Oh Sch..: Javscript 💩
Hello World!
ist mit Frameworks
- oft sehr einfach
-
bei größeren Projekten wird komplex 😬
Javascript lernen
-
scheint Anfangs schwierig
-
grundlegendes Lernen - Step bei Step
-
nicht "Wissen" stellt eine Hürde da
-
erlerntes Wissen hilft auch bei den Frameworks
Toolset
-
Editor
-
beschäftigt euch nicht Code einzurücken
-
benutzt Spellchecker
-
anständiger Browser
Debugging
-
Tastaturkürzel für console.log()
-
im Browser selbst
-
Chrome Debugging über Node
Chrome Debugging über Node
-
im Terminal
- in der Browserzeile chrome://inspect/
node --inspect-brk index.js
am Praxisbeispiel lernen 👨🎓👩🎓
-
Aufbau Step by Step
-
Verständis der Zusammenhänge einfacher
-
viele Wege führen nach Rom
-
Ziel, Grundverständis von Javascript zu vermitteln
-
kann dann nach belieben erweitert werden
-
wäre später einfach als "Custom Modul" in Joomla zu integrieren
Code Bilder
<div class="carousel__track-container">
<ul class="carousel__track">
<li class="carousel__slide is-selected">
<a href="#" style="background-image: url('images/carousel_image_1.jpg')"></a>
</li>
<li class="carousel__slide">
<a href="#" style="background-image: url('carousel_image_2.jpg')"></a>
</li>
<li class="carousel__slide">
<a href="#" style="background-image: url('carousel_image_3.jpg')"></a>
</li>
<li class="carousel__slide">
<a href="#" style="background-image: url('carousel_image_4.jpg')"></a>
</li>
</ul>
</div>
Bilder
absolut nebeneinander positioniert
Navigation / Status Joomla Logo
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 398.7 398.2">
<title>
Joomla!-Logo
</title>
<path id="j-green" d="M85.1,213.7l-7.5-7.4c-23.9-23.9-30.8-57.4-22.3-87"
class="joomla-logo-1 is-selected" transform="translate(-13.8 -13.3)" />
<path id="j-orange" d="M129.2,17"
class="joomla-logo-2" transform="translate(-13.8 -13.3)" />
<path id="j-red" d="M306.2,368.4c-3.2,368.4Z"
class="joomla-logo-3" transform="translate(-13.8 -13.3)" />
<path id="j-blue" d="M290.8,257....."
class="joomla-logo-4" transform="translate(-13.8 -13.3)" />
</svg>
Left und Right Button
<button class="carousel__button">
<svg >
......
</svg>
</button>
Positionierung der Bilder mit JS
const track = document.querySelector('.carousel__track')
const slides = Array.from(track.children)
console.log('slides', slides)
const rect = slides[0].getBoundingClientRect()
const slideWidth = rect.width
console.log('slideWidth', slideWidth)
Positionierung der Bilder mit JS
const slideWidth = slides[0].getBoundingClientRect().width
slides[0].style.left = slideWidth * 0 + 'px'
slides[1].style.left = slideWidth * 1 + 'px'
slides[2].style.left = slideWidth * 2 + 'px'
slides[3].style.left = slideWidth * 3 + 'px'
Positionierung der Bilder mit JS
slides.forEach((slide, index) => {
slide.style.left = slideWidth * index + 'px'
})
const slideWidth = slides[0].getBoundingClientRect().width
slides[0].style.left = slideWidth * 0 + 'px'
slides[1].style.left = slideWidth * 1 + 'px'
slides[2].style.left = slideWidth * 2 + 'px'
slides[3].style.left = slideWidth * 3 + 'px'
Click -> nächste Bild
nextButton.addEventListener('click', e => {
const currentSlide = track.querySelector('.is-selected')
const nextSlide = currentSlide.nextElementSibling
const movingWidth = nextSlide.style.left
console.log('movingWidth', movingWidth)
})
nextButton.addEventListener('click', e => {
//..
track.style.left = '-' + movingWidth
})
Click -> nächste Bilder
nextButton.addEventListener('click', e => {
//...
currentSlide.classList.remove('is-selected')
nextSlide.classList.add('is-selected')
})
nach dem letzten Slide
Abhilfe Ausblenden des Buttons
nextButton.addEventListener('click', e => {
//...
const isLastSlide = !nextSlide.nextElementSibling
if (isLastSlide) {
nextButton.classList.add('is-hidden')
}
})
.is-hidden {
display: none;
}
CSS
Update Joomla Logo
nextButton.addEventListener('click', e => {
//...
const isLastSlide = !nextSlide.nextElementSibling
if (isLastSlide) {
nextButton.classList.add('is-hidden')
}
})
.is-hidden {
display: none;
}
CSS
🥳 erster Schritt
nächster Schritt Code kopieren
Änderung des 'nextButton' to 'backButton'
const backButton = document.querySelector('.back')
backButton.addEventListener('click', e => {
const currentSlide = track.querySelector('.is-selected')
const backSlide = currentSlide.previousElementSibling
const movingWidth = backSlide.style.left
track.style.left = '-' + movingWidth
currentSlide.classList.remove('is-selected')
backSlide.classList.add('is-selected')
const isLastSlide = !backSlide.previousElementSibling
if (isLastSlide) {
backButton.classList.add('is-hidden')
}
const currentLogoPart = joomlaLogo.querySelector('.is-selected')
const backLogoPart = currentLogoPart.previousElementSibling
currentLogoPart.classList.remove('is-selected')
backLogoPart.classList.add('is-selected')
})
Joomla Rotation Logo
joomlaLogo.addEventListener('click', e => {
const clickPath = e.target.closest('path')
console.log('clickPath', clickPath)
})
Verwendung Event Delegation Pattern
- matches
- closest
Current Slide
joomlaLogo.addEventListener('click', e => {
const clickPath = e.target.closest('path')
if (clickPath) {
const currentPath = clickPath.querySelector('.is-selected')
const currentSlide = track.querySelector('.is_selected')
const targetPath = clickPath
}
})
Joomla Logo indexieren
for (let index = 0; index < paths.length; index++) {
if (paths[index] === targetPath) {
targetIndex = index - 1
}
}
const targetSlide = slides[targetIndex]
um das Zielslide zu herauszufinden
const paths = Array.from(joomlaLogo.children)
im joomlaLogo Listener
Slide ändern/bewegen
const targetSlide = slides[targetIndex]
const movingWidth = targetSlide.style.left
track.style.left = '-' + movingWidth
currentSlide.classList.remove('is-selected')
targetSlide.classList.add('is-selected')
currentPath.classList.remove('is-selected')
targetPath.classList.add('is-selected')
zusätzlich Select-Status ändern
Buttons ausblenden
if (targetIndex === 0) {
backButton.classList.add('is-hidden')
nextButton.classList.remove('is-hidden')
} else if (targetIndex === slides.length - 1) {
backButton.classList.remove('is-hidden')
nextButton.classList.add('is-hidden')
} else {
backButton.classList.remove('is-hidden')
nextButton.classList.remove('is-hidden')
} const targetSlide = slides[targetIndex]
}
Buttons zeigen
backButton.addEventListener('click', e => {
nextButton.classList.remove('is-hidden')
}
nextButton.addEventListener('click', e => {
backButton.classList.remove('is-hidden')
}
wenn wir auf den zurück oder nächste klicken
Slides animieren
track.style.left = '-' + movingWidth
ersetzen von
mit
track.style.transform = `translateX(-${movingWidth})`
CSS
.carousel__track {
transform: translateX(0);
transition: transform 0.3s ease-out;
}
💪 Läuft 🤩
Aber....🤔
es geht bestimmt besser...
setzte Position Slide:
const setSlidePostion = (slide, index) => {
slide.style.left = slideWidth * index + 'px'
}
slides.forEach(setSlidePostion)
Auslagern in Funktionen
Code vereinfachen
nextButton.addEventListener('click', e => {
//...
const movingWidth = nextSlide.style.left
track.style.transform = `translateX(-${movingWidth})`
currentSlide.classList.remove('is-selected')
nextSlide.classList.add('is-selected')
}
Code vereinfachen
const moveSlide = (track, targetSlide, currentSlide) => {
track.style.transform = `translateX(-${targetSlide.style.left})`
currentSlide.classList.remove('is-selected')
targetSlide.classList.add('is-selected')
}
nextButton.addEventListener('click', e => {
//...
moveSlide(track, nextSlide, currentSlide)
}
Early return - Verschachtelungen vermeiden
if (clickPath) {
}
if (!clickPath) return
Logo Farbe updaten
const updateLogoColor = (targetLogoPart, currentLogoPart) => {
currentLogoPart.classList.remove('is-selected')
targetLogoPart.classList.add('is-selected')
}
Buttons ein- und ausblenden
const showHideNextBackButtons = (targetIndex, backButton, nextButton) => {
if (targetIndex === 0) {
backButton.classList.add('is-hidden')
nextButton.classList.remove('is-hidden')
} else if (targetIndex === slides.length - 1) {
backButton.classList.remove('is-hidden')
nextButton.classList.add('is-hidden')
} else {
backButton.classList.remove('is-hidden')
nextButton.classList.remove('is-hidden')
}
}
for Schleife kürzen
let targetIndex
for (let index = 0; index < paths.length; index++) {
if (paths[index] === clickPath) {
targetIndex = index - 1
}
}
const targetIndex = paths.findIndex(path => path === clickPath) - 1
findIndex bietet neue Möglichkeiten
const currentSlide = track.querySelector('.is-selected')
const nextSlide = currentSlide.nextElementSibling
const currentIndexSlide = slides.findIndex(slide => slide.classList.contains('is-selected'))
Scope?
E-Mail: info@mittl-medien.de
Twitter: @mittlmedien
Fragen?
Zell Liew (Javascript Developer zellwk.com, Twitter @zellwk)
Niels Nübel für Fotos (https://www.flickr.com/photos/niels85/)
Danke👏
Es geht auch ohne JS Framework
By Robert Mittl
Es geht auch ohne JS Framework
- 953