122 lines
4.9 KiB
JavaScript
122 lines
4.9 KiB
JavaScript
var controller = new AbortController()
|
|
|
|
// 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);
|
|
}
|
|
|
|
}
|
|
|
|
function showNetworkLoading(img) {
|
|
|
|
var wrapper = document.createElement("div");
|
|
wrapper.classList.add("activityIndicatorWrap");
|
|
img.parentNode.insertBefore(wrapper, img);
|
|
wrapper.appendChild(img);
|
|
|
|
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);
|
|
}
|
|
|
|
function hideNetworkLoading(img) {
|
|
var wrapper = img.parentNode;
|
|
var wrapperParent = wrapper.parentNode;
|
|
wrapperParent.insertBefore(img, wrapper);
|
|
wrapperParent.removeChild(wrapper);
|
|
}
|
|
|
|
// Used to animate the transition to a fullscreen image
|
|
function hideClickedImage() {
|
|
var img = document.querySelector('.nnwClicked')
|
|
img.style.opacity = 0
|
|
}
|
|
|
|
// 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);
|
|
}
|
|
}
|
|
}
|
|
|
|
// 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() {
|
|
document.querySelectorAll("video").forEach(element => {
|
|
element.setAttribute("playsinline", true)
|
|
});
|
|
}
|
|
|
|
function postRenderProcessing() {
|
|
cancelImageLoad();
|
|
imageClicks();
|
|
inlineVideos();
|
|
}
|
|
|
|
const activityIndicator = "";
|