Merge pull request #1579 from benubois/image_viewer

Use existing image for image viewer.
This commit is contained in:
Maurice Parker 2020-01-09 11:16:46 -08:00 committed by GitHub
commit 563b61c61b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 115 additions and 91 deletions

View File

@ -1,109 +1,136 @@
var controller = new AbortController()
var activeImageViewer = null;
// Cancel any pending image loads (there might be none) and reset the controller
function cancelImageLoad() {
controller.abort();
controller = new AbortController();
}
// Used to pop a resizable image view
async function imageWasClicked(img) {
cancelImageLoad();
showNetworkLoading(img);
try {
const signal = controller.signal;
const response = await fetch(img.src, { signal: signal });
if (!response.ok) {
throw new Error('Network response was not ok.');
}
const imgBlob = await response.blob();
if (signal.aborted) {
throw new Error('Network response was aborted.');
}
hideNetworkLoading(img);
var reader = new FileReader();
reader.readAsDataURL(imgBlob);
reader.onloadend = function() {
img.classList.add("nnwClicked");
const rect = img.getBoundingClientRect();
var message = {
x: rect.x,
y: rect.y,
width: rect.width,
height: rect.height
};
message.imageURL = reader.result;
var jsonMessage = JSON.stringify(message);
window.webkit.messageHandlers.imageWasClicked.postMessage(jsonMessage);
}
} catch (error) {
hideNetworkLoading(img);
console.log('There has been a problem with your fetch operation: ', error.message);
class ImageViewer {
constructor(img) {
this.img = img;
this.loadingInterval = null;
this.activityIndicator = "";
}
}
function showNetworkLoading(img) {
isLoaded() {
return this.img.classList.contains("nnwLoaded");
}
var wrapper = document.createElement("div");
wrapper.classList.add("activityIndicatorWrap");
img.parentNode.insertBefore(wrapper, img);
wrapper.appendChild(img);
clicked() {
this.showLoadingIndicator();
if (this.isLoaded()) {
this.showViewer();
} else {
var callback = () => {
if (this.isLoaded()) {
clearInterval(this.loadingInterval);
this.showViewer();
}
}
this.loadingInterval = setInterval(callback, 100);
}
}
cancel() {
clearInterval(this.loadingInterval);
this.hideLoadingIndicator();
}
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,
imageURL: canvas.toDataURL(),
};
var jsonMessage = JSON.stringify(message);
window.webkit.messageHandlers.imageWasClicked.postMessage(jsonMessage);
}
hideImage() {
this.img.style.opacity = 0;
}
showImage() {
this.img.style.opacity = 1
window.webkit.messageHandlers.imageWasShown.postMessage("");
}
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);
var activityIndicatorImg = document.createElement("img");
activityIndicatorImg.classList.add("activityIndicator");
activityIndicatorImg.style.opacity = 0;
activityIndicatorImg.src = activityIndicator;
wrapper.appendChild(activityIndicatorImg);
// Wait a bit before showing the indicator
function showActivityIndicator() {
activityIndicatorImg.style.opacity = 1;
}
setTimeout(showActivityIndicator, 300);
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();
}
}
}
}
}
function hideNetworkLoading(img) {
var wrapper = img.parentNode;
var wrapperParent = wrapper.parentNode;
wrapperParent.insertBefore(img, wrapper);
wrapperParent.removeChild(wrapper);
function cancelImageLoad() {
if (activeImageViewer) {
activeImageViewer.cancel();
activeImageViewer = null;
}
}
// Used to animate the transition to a fullscreen image
function hideClickedImage() {
var img = document.querySelector('.nnwClicked')
img.style.opacity = 0
if (activeImageViewer) {
activeImageViewer.hideImage();
}
}
// Used to animate the transition from a fullscreen image
function showClickedImage() {
var img = document.querySelector('.nnwClicked')
img.classList.remove("nnwClicked");
img.style.opacity = 1
window.webkit.messageHandlers.imageWasShown.postMessage("");
}
// Add the click listener for images
function imageClicks() {
window.onclick = function(event) {
if (event.target.matches('img')) {
imageWasClicked(event.target);
}
if (activeImageViewer) {
activeImageViewer.showImage();
}
}
// Add the playsinline attribute to any HTML5 videos that don't have it.
// Add the playsinline attribute to any HTML5 videos that don"t have it.
// Without this attribute videos may autoplay and take over the whole screen
// on an iphone when viewing an article.
function inlineVideos() {
@ -113,9 +140,6 @@ function inlineVideos() {
}
function postRenderProcessing() {
cancelImageLoad();
imageClicks();
ImageViewer.init();
inlineVideos();
}
const activityIndicator = "";