182 lines
3.9 KiB
TypeScript
Raw Normal View History

2017-06-19 14:49:24 +00:00
import { Application } from "./Application"
2017-06-21 16:44:20 +00:00
import { Diff } from "./Diff"
2017-06-20 10:41:26 +00:00
import { findAll } from "./utils"
import * as actions from "./actions"
2017-06-19 14:49:24 +00:00
export class AnimeNotifier {
app: Application
2017-06-24 14:17:38 +00:00
visibilityObserver: IntersectionObserver
2017-06-19 14:49:24 +00:00
constructor(app: Application) {
this.app = app
if("IntersectionObserver" in window) {
// Enable lazy load
this.visibilityObserver = new IntersectionObserver(
entries => {
for(let entry of entries) {
if(entry.intersectionRatio > 0) {
entry.target["became visible"]()
this.visibilityObserver.unobserve(entry.target)
}
2017-06-24 14:17:38 +00:00
}
},
{}
)
} else {
// Disable lazy load feature
this.visibilityObserver = {
disconnect: () => {},
observe: (elem: HTMLElement) => {
elem["became visible"]()
},
unobserve: (elem: HTMLElement) => {}
} as IntersectionObserver
}
2017-06-19 14:49:24 +00:00
}
2017-06-19 15:45:27 +00:00
onReadyStateChange() {
if(document.readyState !== "interactive") {
return
}
this.run()
}
2017-06-19 14:49:24 +00:00
run() {
this.app.content = this.app.find("content")
this.app.loading = this.app.find("loading")
this.app.run()
}
2017-06-19 15:45:27 +00:00
2017-06-24 14:17:38 +00:00
onContentLoaded() {
this.visibilityObserver.disconnect()
// Update each of these asynchronously
Promise.resolve().then(() => this.updateMountables())
Promise.resolve().then(() => this.updateActions())
Promise.resolve().then(() => this.lazyLoadImages())
}
2017-06-21 16:44:20 +00:00
reloadContent() {
return fetch("/_" + this.app.currentPath, {
credentials: "same-origin"
})
.then(response => response.text())
.then(html => Diff.innerHTML(this.app.content, html))
2017-06-24 00:10:04 +00:00
.then(() => this.app.emit("DOMContentLoaded"))
2017-06-21 16:44:20 +00:00
}
2017-06-20 10:41:26 +00:00
loading(isLoading: boolean) {
if(isLoading) {
this.app.loading.classList.remove(this.app.fadeOutClass)
} else {
this.app.loading.classList.add(this.app.fadeOutClass)
}
}
2017-06-21 12:00:52 +00:00
updateActions() {
2017-06-21 13:29:06 +00:00
for(let element of findAll("action")) {
2017-06-24 00:10:04 +00:00
if(element["action assigned"]) {
continue
}
2017-06-20 10:41:26 +00:00
let actionName = element.dataset.action
2017-06-20 20:54:45 +00:00
element.addEventListener(element.dataset.trigger, e => {
actions[actionName](this, element, e)
})
2017-06-24 00:10:04 +00:00
element["action assigned"] = true
2017-06-20 10:41:26 +00:00
}
2017-06-19 15:45:27 +00:00
}
2017-06-24 14:17:38 +00:00
lazyLoadImages() {
2017-06-21 13:29:06 +00:00
for(let element of findAll("user-image")) {
2017-06-24 14:17:38 +00:00
this.lazyLoadImage(element as HTMLImageElement)
}
for(let element of findAll("anime-cover-image")) {
this.lazyLoadImage(element as HTMLImageElement)
}
}
lazyLoadImage(img: HTMLImageElement) {
// Prevent browser from loading it instantly
img["original source"] = img.src
img.src = ""
// Once the image becomes visible, load it
img["became visible"] = () => {
img.src = img["original source"]
2017-06-19 15:45:27 +00:00
if(img.naturalWidth === 0) {
img.onload = function() {
this.classList.add("user-image-found")
}
img.onerror = function() {
this.classList.add("user-image-not-found")
}
} else {
img.classList.add("user-image-found")
}
}
2017-06-24 14:17:38 +00:00
this.visibilityObserver.observe(img)
2017-06-19 15:45:27 +00:00
}
2017-06-20 12:16:23 +00:00
2017-06-20 18:13:04 +00:00
updateMountables() {
const delay = 20
const maxDelay = 1000
let time = 0
2017-06-21 13:29:06 +00:00
for(let element of findAll("mountable")) {
2017-06-20 18:13:04 +00:00
setTimeout(() => {
window.requestAnimationFrame(() => element.classList.add("mounted"))
}, time)
time += delay
if(time > maxDelay) {
time = maxDelay
}
}
}
2017-06-21 12:00:52 +00:00
onPopState(e: PopStateEvent) {
if(e.state) {
this.app.load(e.state, {
addToHistory: false
})
} else if(this.app.currentPath !== this.app.originalPath) {
this.app.load(this.app.originalPath, {
addToHistory: false
})
}
}
onKeyDown(e: KeyboardEvent) {
2017-06-24 14:31:54 +00:00
// F = Search
if(e.keyCode == 70) {
2017-06-21 12:00:52 +00:00
let search = this.app.find("search") as HTMLInputElement
search.focus()
search.select()
e.preventDefault()
e.stopPropagation()
}
}
2017-06-20 12:16:23 +00:00
// onResize(e: UIEvent) {
// let hasScrollbar = this.app.content.clientHeight === this.app.content.scrollHeight
// if(hasScrollbar) {
// this.app.content.classList.add("has-scrollbar")
// } else {
// this.app.content.classList.remove("has-scrollbar")
// }
// }
2017-06-19 14:49:24 +00:00
}