Можно ли клонировать изображение после загрузки в другие части страницы без загрузки?
Я столкнулся с простой проблемой. Пусть говорит, что мой пользователь загружает около 150 изображений с помощью простого <img data-image='1' src="myimg1.jpg">
<img data-image=2' src="myimg2.jpg">
.. и т.д.
Когда пользователь наводится над одним из изображений. Я хочу отобразить этот myimg-thisimage.jpg
в маленьком меню внизу экрана. На данный момент я изменяю атрибут src в своем меню как:
$('#info-poster').attr("src","myimage-thisimage.jpg");
Примечание. myimage-thisimage.jpg
- текущее зависание над изображением.
Но, когда я это делаю. Браузер перезагружает изображение (потому что есть небольшая задержка). Есть ли способ обойти эту загрузку, так как пользователь уже загрузил изображение, используя умный способ клонирования элемента DOM, возможно?
PS: Кэш изображений браузера включен. Поэтому кеш не является проблемой.
Изменить: Я знаю, что один способ - создать 300 элементов изображения и скрыть остальные 150 из них. Но в сценарии (определенно возможно), где есть около 500 изображений, мне пришлось бы создать около 1000 элементов DOM, что было бы большой проблемой производительности.
Ответы
Ответ 1
Вы можете использовать элемент canvas, чтобы показать эскиз, таким образом изображение копируется и масштабируется локально. В следующем фрагменте я добавил два холста, в первом изображение масштабируется при сохранении пропорций (при необходимости я использую методы Letterboxing и Pillarboxing); во втором изображение растянуто. Я также добавил другое изображение внизу, которое игнорируется, так как оно не имеет атрибута data-image.
Важно не использовать алгоритм масштабирования drawImage, поскольку он дает негладкие результаты, когда вы значительно уменьшаете изображение. Чтобы достичь этого, установите логический размер холста в соответствии с естественным размером изображения. Затем скопируйте изображение на холст, вызвав метод drawImage. Наконец, установите размер экрана холста на желаемый. Таким образом, браузер использует лучший алгоритм масштабирования изображения.
Ниже приведены некоторые выдающиеся цитаты из спецификации метода drawImage()
- Вы можете быть уверены, что изображение не будет перезагружено и что вы должны использовать естественный размер изображения, чтобы избежать масштабирования с помощью drawImage:
Если исходные данные изображения являются растровым изображением, значение, нарисованное в точке в прямоугольнике назначения, вычисляется путем фильтрации исходных данных изображения.
- Браузер решает, какой алгоритм масштабирования использовать. На момент написания этого: Edge, Chrome и Firefox не используют ничего лучше, чем алгоритмы билинейного или ближайшего соседа. Это может измениться в будущем:
Пользовательский агент может использовать любой алгоритм фильтрации (например, билинейную интерполяцию или ближайший сосед).
function initCanvas(id,image,naturalWidth,naturalHeight){
var canvas = document.getElementById(id);
var ctx = canvas.getContext("2d");
// Set the logical size of the canvas to match the
// natural size of the image, this way we don't use
// the scaling algorithm of drawImage (It isn't good
// for reducing big images as it produces unsmooth results).
$(canvas).attr("width",naturalWidth) ;
$(canvas).attr("height",naturalHeight) ;
// Copy the image:
ctx.drawImage(image,0,0,naturalWidth,naturalHeight);
return canvas ;
}
function clearCanvas(id){
var canvas = document.getElementById(id);
var ctx = canvas.getContext("2d");
ctx.clearRect(0,0,canvas.width,canvas.height);
}
$(window).on("load", function( ){
var images = $("img").filter(function(){
var dataImage = $(this).data("image") ;
if( typeof dataImage != "number" ) return false ;
var number = parseInt(dataImage,10) ;
return number > 0 && dataImage === number ;
}) ;
images.on("mouseenter", function( ){
var naturalWidth = $(this).prop("naturalWidth") ;
var naturalHeight = $(this).prop("naturalHeight") ;
// Scaled thumbnail:
// Copy the image to canvas-scaled and get a reference to it:
var scaledCanvas = initCanvas("canvas-scaled",this,naturalWidth,naturalHeight);
// Calculate the display size of the canvas:
var hwfactor = naturalHeight/naturalWidth ;
var whfactor = naturalWidth/naturalHeight ;
var scaledWidth, scaledHeight ;
if( hwfactor >= 1 ){ // Pillarboxing
scaledHeight = "100px" ;
scaledWidth = (100*whfactor)+"px" ;
}
else{ // Letterboxing
scaledWidth = "100px" ;
scaledHeight = (100*hwfactor)+"px" ;
}
// Now we change the display size of the canvas.
// A better scaling algorithm will be used.
$(scaledCanvas).css("width",scaledWidth);
$(scaledCanvas).css("height",scaledHeight);
// Stretched thumbnail:
// Copy the image to canvas-stretched. The display size
// of canvas-stretched is already set in the style section.
initCanvas("canvas-stretched",this,naturalWidth,naturalHeight);
});
images.on("mouseleave", function( ){
clearCanvas("canvas-scaled");
clearCanvas("canvas-stretched");
});
});
body{
background: #000;
}
.wrapper img{
width: 100px ;
height: auto ;
}
#banner{
display: block ;
width: 100% ;
height: 40px ;
padding-top: 1pt ;
}
#canvas-stretched{
width: 100px ;
height: 100px ;
}
.canvas-wrapper{
display: -webkit-inline-flex ;
display: inline-flex ;
-webkit-justify-content: space-around ;
justify-content: space-around ;
-webkit-align-items: center ;
align-items: center ;
vertical-align: bottom ;
border: 1px solid #888 ;
width: 100px ;
height: 100px ;
overflow: hidden ;
}
.viewer{
display: inline-block ;
}
.viewer span{
color: #ddd ;
font-family: sans-serif ;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.0.0/jquery.min.js"></script>
<span class="wrapper">
<img data-image="1" src="https://upload.wikimedia.org/wikipedia/commons/thumb/e/ec/Mona_Lisa%2C_by_Leonardo_da_Vinci%2C_from_C2RMF_retouched.jpg/550px-Mona_Lisa%2C_by_Leonardo_da_Vinci%2C_from_C2RMF_retouched.jpg"/>
<img data-image="2" src="https://upload.wikimedia.org/wikipedia/en/8/81/Megadrive_another_world.png"/>
<img data-image="3" src="https://upload.wikimedia.org/wikipedia/en/e/ee/TheKlingonHamlet.jpg"/>
</span>
<span class="viewer">
<span>scaled</span><br>
<div class="canvas-wrapper">
<canvas id="canvas-scaled"></canvas>
</div>
</span>
<span class="viewer">
<span>stretched</span><br>
<div class="canvas-wrapper">
<canvas id="canvas-stretched"></canvas>
</div>
</span>
<img id="banner" src="https://upload.wikimedia.org/wikipedia/commons/thumb/0/05/The_southern_plane_of_the_Milky_Way_from_the_ATLASGAL_survey.jpg/320px-The_southern_plane_of_the_Milky_Way_from_the_ATLASGAL_survey.jpg"/>
Ответ 2
Эта строка является проблемой:
$('#info-poster').attr("src","myimage-thisimage.jpg");
Браузер перезагружает изображение, потому что вы повторно используете (плохую практику) атрибут "src".
Вместо этого вы можете использовать опции CSS для отображения/скрытия "myimage-thisimage.jpg".
Поскольку вы используете jQuery, мы можем использовать методы: hide/show.
Вы упомянули "клон", я не думаю, что вы имеете в виду клонирование HTML-элементов.
Пример: (жить на JS Bin)
<img id="dummy" width="200" height="150" data-image='1' src="http://europunkt.ro/wp-content/uploads/2016/06/romania.jpg">
<!-- Hidden by default -->
<img style="display:none" id="info-poster" width="200" height="150">
<script src="https://code.jquery.com/jquery-2.2.4.js"></script>
<script>
var $dummy = $("#dummy");
var $infoPoster = $("#info-poster");
var infoPosterHasLoaded = false;
$dummy.on("mouseenter", function() {
// add the src attribute ONLY once
if(infoPosterHasLoaded === false){
$infoPoster.attr("src", "http://www.ilovemaramures.com/wp-content/uploads/2012/05/Pasul-Prislop.jpg")
infoPosterHasLoaded = true;
}
$infoPoster.show();
});
$dummy.on("mouseleave", function() {
$infoPoster.hide();
});
</script>
Для более причудливого "скрыть/показать" вы можете проверить jQuery Effects.
Изменить - после того, как я прочитал ваш комментарий
Если вы хотите использовать атрибут "data-image" из зависающего элемента, проверьте эти объекты: event.target, event.currentTarget, this
Новая версия JS Bin.
Ответ 3
Вы можете конвертировать запросы изображений на сервере, чтобы ответить на строку base64, которую вы можете сохранить в своем кеше.
Пример кода ниже:
HTML
<img id="image1Id" src="" />
<input type="button" onclick='javascript:loadSomeThing("image1", "", "image1Id");' value="Load Image1" />
Script
var imageArray = [];
function loadSomeThing(key, someUrl, elementId) {
var imageData = imageArray[key];
if (!imageData) {
imageData = ajaxGetImageData(someUrl);
imageArray[key] = imageData;
}
document.getElementById(elementId).src = imageData;
}
function ajaxGetImageData(url) {
//Code to get base64 image string
return "";
}
Demo
jsFiddle
Ответ 4
Я считаю, что то, что вы хотите, возможно с помощью функции jQuery .clone()
и .append()
. См. Пример ниже.
jQuery.clone()
jQuery.append()
$(function() {
$('img#ToClone').click(function(){
var imageClone = $('#ToClone').clone();
var cloneDestination = $('#CloneTo');
cloneDestination.append(imageClone);
});
});
div
{
padding:2px;
background:blue;
}
div#CloneTo
{
background:green;
}
img{
height:50px;
width:50px;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<div>
<p>Click on the red square to clone it below.</p>
<img id="ToClone" src="" />
</div>
<div id="CloneTo">
<p>Clone should appear here.</p>
<!-- The cloned image should appear here. -->
</div>
Ответ 5
Вы должны позволить браузеру выполнять обработку кеша.
Я предлагаю вам иметь <img id="info-poster" src="myimage-thisimage.jpg" class="hide-on-load"/>
, поэтому, если ваш браузер захочет загрузить новую копию изображения, он сделает это до того, как пользователь наметит ваши другие изображения. (если это небольшое/приемлемое изображение, которое пользователь может загрузить для каждой загрузки страницы)
Затем вы можете просто привязать $("img.bind-event").on("mouseenter", function() { $("#info-poster").show(); });
и $("img.bind-event").on("mouseleave", function() { $("#info-poster").hide(); });
Ответ 6
IDEA
начальная разметка
<img data-src='myimg1.jpg' data-image='1' src='placeholder.jpg'>
после того, как myimg1.jpg загружен динамически (*)
<img data-image='1' src='blob:asdfasdfasdfasdfadfa'>
Затем в 'mouseenter'
infoPosterEl.src = thisImageEl.src
// update image src to an object url(e.g. "blob:") will not bother http comm.
(*)
// Fetch acutal image as blob
// Create object url for the blob
// Update this <img> src to the object url
Ответ 7
Вы можете сохранить путь к каждому изображению в массиве, итерации массива с помощью Array.prototype.forEach()
, установить background
каждого элемента <img>
, используя url("/path/to/image")
; в mouseover
каждого <img>
установите background-size
элемента меню на 100% 100%
по индексу зависающего элемента <img>
в коллекции, используя Array.prototype.slice()
, Array.prototype.splice()
. Этот подход должен запрашивать каждое изображение с сервера не более одного раза, переключая изображение, отображаемое в элементе меню, в соответствии с зависающим изображением.
var urls = ["http://placehold.it/100x100?text=1"
, "http://placehold.it/100x100?text=2"
, "http://placehold.it/100x100?text=3"
, "http://placehold.it/100x100?text=4"
, "http://placehold.it/100x100?text=5"]
, sources = []
, sizes = []
, imgs = document.querySelectorAll(".img")
, menu = document.querySelector(".menu");
function toggleImage(index) {
this.onmouseover = function() {
var curr = sizes.slice(0);
curr.splice(index, 1, "100% 100%");
menu.style.backgroundSize = curr.join(",");
}
}
urls.forEach(function(path, index) {
sources.push("url(" + path + ")");
sizes.push("0% 0%");
imgs[index].style.background = sources[index];
toggleImage.call(imgs[index], index);
});
menu.style.background = sources.join(",");
menu.style.backgroundSize = sizes.join(",");
.menu {
left: calc(100vw / 2 - 50px);
position: relative;
width: 75px;
height: 75px;
display: block;
}
<img class="img" width="100" height="100" src="" /><img class="img" width="100" height="100" /><img class="img" width="100" height="100" /><img class="img" width="100" height="100" /><img class="img" width="100" height="100" />
<div class="menu">
</div>