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 = "";