2020-01-09 20:16:46 +01:00
|
|
|
var activeImageViewer = null;
|
2019-10-30 17:06:55 +01:00
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
class ImageViewer {
|
|
|
|
constructor(img) {
|
|
|
|
this.img = img;
|
|
|
|
this.loadingInterval = null;
|
|
|
|
this.activityIndicator = "";
|
|
|
|
}
|
2019-10-19 23:25:06 +02:00
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
isLoaded() {
|
|
|
|
return this.img.classList.contains("nnwLoaded");
|
|
|
|
}
|
2019-10-14 02:41:34 +02:00
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
clicked() {
|
|
|
|
this.showLoadingIndicator();
|
|
|
|
if (this.isLoaded()) {
|
|
|
|
this.showViewer();
|
|
|
|
} else {
|
|
|
|
var callback = () => {
|
|
|
|
if (this.isLoaded()) {
|
|
|
|
clearInterval(this.loadingInterval);
|
|
|
|
this.showViewer();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.loadingInterval = setInterval(callback, 100);
|
2019-10-30 17:06:55 +01:00
|
|
|
}
|
2020-01-09 20:16:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
cancel() {
|
|
|
|
clearInterval(this.loadingInterval);
|
|
|
|
this.hideLoadingIndicator();
|
|
|
|
}
|
2019-10-30 17:06:55 +01:00
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
showViewer() {
|
|
|
|
this.hideLoadingIndicator();
|
|
|
|
|
|
|
|
var canvas = document.createElement("canvas");
|
|
|
|
canvas.width = this.img.naturalWidth;
|
|
|
|
canvas.height = this.img.naturalHeight;
|
|
|
|
canvas.getContext("2d").drawImage(this.img, 0, 0);
|
|
|
|
|
|
|
|
const rect = this.img.getBoundingClientRect();
|
|
|
|
const message = {
|
|
|
|
x: rect.x,
|
|
|
|
y: rect.y,
|
|
|
|
width: rect.width,
|
|
|
|
height: rect.height,
|
|
|
|
imageTitle: this.img.title,
|
|
|
|
imageURL: canvas.toDataURL(),
|
|
|
|
};
|
|
|
|
|
|
|
|
var jsonMessage = JSON.stringify(message);
|
|
|
|
window.webkit.messageHandlers.imageWasClicked.postMessage(jsonMessage);
|
|
|
|
}
|
2019-10-30 17:06:55 +01:00
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
hideImage() {
|
|
|
|
this.img.style.opacity = 0;
|
2019-10-14 02:41:34 +02:00
|
|
|
}
|
2019-10-13 22:47:11 +02:00
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
showImage() {
|
|
|
|
this.img.style.opacity = 1
|
|
|
|
}
|
|
|
|
|
|
|
|
showLoadingIndicator() {
|
|
|
|
var wrapper = document.createElement("div");
|
|
|
|
wrapper.classList.add("activityIndicatorWrap");
|
|
|
|
this.img.parentNode.insertBefore(wrapper, this.img);
|
|
|
|
wrapper.appendChild(this.img);
|
|
|
|
|
|
|
|
var activityIndicatorImg = document.createElement("img");
|
|
|
|
activityIndicatorImg.classList.add("activityIndicator");
|
|
|
|
activityIndicatorImg.style.opacity = 0;
|
|
|
|
activityIndicatorImg.src = this.activityIndicator;
|
|
|
|
wrapper.appendChild(activityIndicatorImg);
|
|
|
|
|
2019-10-19 23:25:06 +02:00
|
|
|
activityIndicatorImg.style.opacity = 1;
|
|
|
|
}
|
2020-01-09 20:16:46 +01:00
|
|
|
|
|
|
|
hideLoadingIndicator() {
|
|
|
|
var wrapper = this.img.parentNode;
|
|
|
|
if (wrapper.classList.contains("activityIndicatorWrap")) {
|
|
|
|
var wrapperParent = wrapper.parentNode;
|
|
|
|
wrapperParent.insertBefore(this.img, wrapper);
|
|
|
|
wrapperParent.removeChild(wrapper);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static init() {
|
|
|
|
cancelImageLoad();
|
|
|
|
|
|
|
|
// keep track of when an image has finished downloading for ImageViewer
|
|
|
|
document.querySelectorAll("img").forEach(element => {
|
|
|
|
element.onload = function() {
|
|
|
|
this.classList.add("nnwLoaded");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Add the click listener for images
|
|
|
|
window.onclick = function(event) {
|
|
|
|
if (event.target.matches("img")) {
|
|
|
|
if (activeImageViewer && activeImageViewer.img === event.target) {
|
|
|
|
cancelImageLoad();
|
|
|
|
} else {
|
|
|
|
cancelImageLoad();
|
|
|
|
activeImageViewer = new ImageViewer(event.target);
|
|
|
|
activeImageViewer.clicked();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-10-19 23:25:06 +02:00
|
|
|
}
|
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
function cancelImageLoad() {
|
|
|
|
if (activeImageViewer) {
|
|
|
|
activeImageViewer.cancel();
|
|
|
|
activeImageViewer = null;
|
|
|
|
}
|
2019-10-19 23:25:06 +02:00
|
|
|
}
|
|
|
|
|
2019-10-16 23:40:49 +02:00
|
|
|
function hideClickedImage() {
|
2020-01-09 20:16:46 +01:00
|
|
|
if (activeImageViewer) {
|
|
|
|
activeImageViewer.hideImage();
|
|
|
|
}
|
2019-10-16 23:40:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Used to animate the transition from a fullscreen image
|
|
|
|
function showClickedImage() {
|
2020-01-09 20:16:46 +01:00
|
|
|
if (activeImageViewer) {
|
|
|
|
activeImageViewer.showImage();
|
2019-10-15 16:04:21 +02:00
|
|
|
}
|
2020-01-16 02:09:47 +01:00
|
|
|
window.webkit.messageHandlers.imageWasShown.postMessage("");
|
2019-10-13 22:47:11 +02:00
|
|
|
}
|
|
|
|
|
2020-01-09 20:16:46 +01:00
|
|
|
// Add the playsinline attribute to any HTML5 videos that don"t have it.
|
2019-10-13 22:47:11 +02:00
|
|
|
// Without this attribute videos may autoplay and take over the whole screen
|
|
|
|
// on an iphone when viewing an article.
|
|
|
|
function inlineVideos() {
|
|
|
|
document.querySelectorAll("video").forEach(element => {
|
|
|
|
element.setAttribute("playsinline", true)
|
2020-01-17 20:17:38 +01:00
|
|
|
element.setAttribute("controls", true)
|
2019-10-13 22:47:11 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function postRenderProcessing() {
|
2020-01-09 20:16:46 +01:00
|
|
|
ImageViewer.init();
|
2019-10-30 17:06:55 +01:00
|
|
|
inlineVideos();
|
2019-10-13 22:47:11 +02:00
|
|
|
}
|