More robust page loader

This commit is contained in:
2018-04-02 22:38:00 +02:00
parent bb555457c6
commit 0d74291075

View File

@ -14,6 +14,7 @@ export default class Application {
currentPath: string currentPath: string
originalPath: string originalPath: string
lastRequest: XMLHttpRequest | null lastRequest: XMLHttpRequest | null
contentInvisible: boolean
onError: (err: Error) => void onError: (err: Error) => void
constructor() { constructor() {
@ -104,40 +105,73 @@ export default class Application {
// Mark active links // Mark active links
this.markActiveLinks() this.markActiveLinks()
let onTransitionEnd = (e: Event) => { let consume = async () => {
// Ignore transitions of child elements. let html = await request
// We only care about the transition event on the content element.
if(e.target !== this.content) {
return
}
// Outdated response.
if(this.currentPath !== url) { if(this.currentPath !== url) {
return return
} }
// Remove listener after we finally got the correct event. // Set content
this.content.removeEventListener("transitionend", onTransitionEnd) this.setContent(html)
this.scrollToTop()
// Wait for the network request to end. // Fade in listener
request.then(html => { let onFadedIn: EventListener = (e: Event) => {
// Set content // Ignore transitions of child elements.
this.setContent(html) // We only care about the transition event on the content element.
this.scrollToTop() if(e.target !== this.content) {
return
}
// Fade animations // Reset the transition ended flag
this.content.classList.remove(this.fadeOutClass) this.contentInvisible = false
this.loading.classList.add(this.fadeOutClass)
// Send DOMContentLoaded Event // Remove listener after we finally got the correct event.
this.emit("DOMContentLoaded") this.content.removeEventListener("transitionend", onFadedIn)
}) }
this.content.addEventListener("transitionend", onFadedIn)
// Fade animations
this.content.classList.remove(this.fadeOutClass)
this.loading.classList.add(this.fadeOutClass)
// Send DOMContentLoaded Event
this.emit("DOMContentLoaded")
} }
this.content.addEventListener("transitionend", onTransitionEnd) if(this.contentInvisible) {
consume()
} else {
// Fade out listener
let onFadedOut: EventListener = (e: Event) => {
// Ignore transitions of child elements.
// We only care about the transition event on the content element.
if(e.target !== this.content) {
return
}
this.content.classList.add(this.fadeOutClass) this.contentInvisible = true
this.loading.classList.remove(this.fadeOutClass)
// Remove listener after we finally got the correct event.
this.content.removeEventListener("transitionend", onFadedOut)
// Outdated response.
if(this.currentPath !== url) {
return
}
// Wait for the network request to end.
consume()
}
this.content.addEventListener("transitionend", onFadedOut)
// Add fade out class
this.content.classList.add(this.fadeOutClass)
this.loading.classList.remove(this.fadeOutClass)
}
return request return request
} }