天天看點

JavaScript圖檔放大效果詳解

效果:

運作代碼框

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">

<html xmlns="http://www.w3.org/1999/xhtml">

<head>

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />

<title>Magnifier</title>

<style type="text/css">

#magnifier{

width:342px;

height:420px;

position:absolute;

top:100px;

left:250px;

font-size:0;

border:1px solid #000;

}

#img{

width:342px;

height:420px;

}

#Browser{

border:1px solid #000;

z-index:100;

position:absolute;

background:#555;

}

#mag{

border:1px solid #000;

overflow:hidden;

z-index:100;

}

</style>

<script type="text/javascript">

function getEventObject(W3CEvent) { //事件标準化函數

return W3CEvent || window.event;

}

function getPointerPosition(e) { //相容浏覽器的滑鼠x,y獲得函數

e = e || getEventObject(e);

var x = e.pageX || (e.clientX + (document.documentElement.scrollLeft || document.body.scrollLeft));

var y = e.pageY || (e.clientY + (document.documentElement.scrollTop || document.body.scrollTop));

return { 'x':x,'y':y };

}

function setOpacity(elem,level) { //相容浏覽器設定透明值

if(elem.filters) {

elem.style.filter = 'alpha(opacity=' + level * 100 + ')';

} else {

elem.style.opacity = level;

}

}

function css(elem,prop) { //css設定函數,可以友善設定css值,并且相容設定透明值

for(var i in prop) {

if(i == 'opacity') {

setOpacity(elem,prop[i]);

} else {

elem.style[i] = prop[i];

}

}

return elem;

}

var magnifier = {

m : null,

init:function(magni){

var m = this.m = magni || {

cont : null, //裝載原始圖像的div

img : null, //放大的圖像

mag : null, //放大框

scale : 15 //比例值,設定的值越大放大越大,但是這裡有個問題就是如果不可以整除時,會産生些很小的白邊,目前不知道如何解決

}

css(m.img,{

'position' : 'absolute',

'width' : (m.cont.clientWidth * m.scale) + 'px', //原始圖像的寬*比例值

'height' : (m.cont.clientHeight * m.scale) + 'px' //原始圖像的高*比例值

})

css(m.mag,{

'display' : 'none',

'width' : m.cont.clientWidth + 'px', //m.cont為原始圖像,與原始圖像等寬

'height' : m.cont.clientHeight + 'px',

'position' : 'absolute',

'left' : m.cont.offsetLeft + m.cont.offsetWidth + 10 + 'px', //放大框的位置為原始圖像的右方遠10px

'top' : m.cont.offsetTop + 'px'

})

var borderWid = m.cont.getElementsByTagName('div')[0].offsetWidth - m.cont.getElementsByTagName('div')[0].clientWidth; //擷取border的寬

css(m.cont.getElementsByTagName('div')[0],{ //m.cont.getElementsByTagName('div')[0]為浏覽框

'display' : 'none', //開始設定為不可見

'width' : m.cont.clientWidth / m.scale - borderWid + 'px', //原始圖檔的寬/比例值 - border的寬度

'height' : m.cont.clientHeight / m.scale - borderWid + 'px', //原始圖檔的高/比例值 - border的寬度

'opacity' : 0.5 //設定透明度

})

m.img.src = m.cont.getElementsByTagName('img')[0].src; //讓原始圖像的src值給予放大圖像

m.cont.style.cursor = 'crosshair';

m.cont.onmouseover = magnifier.start;

},

start:function(e){

if(document.all){ //隻在IE下執行,主要避免IE6的select無法覆寫

magnifier.createIframe(magnifier.m.img);

}

this.onmousemove = magnifier.move; //this指向m.cont

this.onmouseout = magnifier.end;

},

move:function(e){

var pos = getPointerPosition(e); //事件标準化

this.getElementsByTagName('div')[0].style.display = '';

css(this.getElementsByTagName('div')[0],{

'top' : Math.min(Math.max(pos.y - this.offsetTop - parseInt(this.getElementsByTagName('div')[0].style.height) / 2,0),this.clientHeight - this.getElementsByTagName('div')[0].offsetHeight) + 'px',

'left' : Math.min(Math.max(pos.x - this.offsetLeft - parseInt(this.getElementsByTagName('div')[0].style.width) / 2,0),this.clientWidth - this.getElementsByTagName('div')[0].offsetWidth) + 'px' //left=滑鼠x - this.offsetLeft - 浏覽框寬/2,Math.max和Math.min讓浏覽框不會超出圖像

})

magnifier.m.mag.style.display = '';

css(magnifier.m.img,{

'top' : - (parseInt(this.getElementsByTagName('div')[0].style.top) * magnifier.m.scale) + 'px',

'left' : - (parseInt(this.getElementsByTagName('div')[0].style.left) * magnifier.m.scale) + 'px'

})

},

end:function(e){

this.getElementsByTagName('div')[0].style.display = 'none';

magnifier.removeIframe(magnifier.m.img); //銷毀iframe

magnifier.m.mag.style.display = 'none';

},

createIframe:function(elem){

var layer = document.createElement('iframe');

layer.tabIndex = '-1';

layer.src = 'javascript:false;';

elem.parentNode.appendChild(layer);

layer.style.width = elem.offsetWidth + 'px';

layer.style.height = elem.offsetHeight + 'px';

},

removeIframe:function(elem){

var layers = elem.parentNode.getElementsByTagName('iframe');

while(layers.length >0){

layers[0].parentNode.removeChild(layers[0]);

}

}

}

window.onload = function(){

magnifier.init({

cont : document.getElementById('magnifier'),

img : document.getElementById('magnifierImg'),

mag : document.getElementById('mag'),

scale : 3

});

}

</script>

</head>

<body>

<div id="magnifier">

<img src="http://www.blueidea.com/articleimg/2009/10/7087/03.jpg" id="img" />

<div id="Browser"></div>

</div>

<div id="mag"><img id="magnifierImg" /></div>

<select style="position:absolute;top:200px;left:650px;width:100px;">

<option>select測試</option>

<option>是否能覆寫</option>

</select>

</body>

</html>

[Ctrl+A 全部選擇 提示:你可先修改部分代碼,再按運作]

以前寫過一個jQuery的圖檔放大效果,但是存在着一些小問題,然後最近有時間重寫了一遍,做了很詳盡的改進.改進了大部分bug,而且不采用jQuery.

程式說明

主要為magnifier類,裡面的主要方法有:

  • init:運作方法
  • start:則是滑鼠移入div的事件處理
  • move:則是滑鼠在div中移動的事件處理
  • end:滑鼠移出後的事件處理

程式介紹

主要思維:當滑鼠移入圖檔的時候,放大層的DIV出現,然後根據滑鼠移動狀況,改變放大層内圖像的top值和left值,使得2個地方保持一緻的現實。而2個圖像跟據比例進行設定,width和height值,使之産生放大的效果,下面進行詳細的解釋:

在init方法中,主要處理浏覽框div層的大小,放大框的大小和放大的圖像大小。

浏覽框div的width和height跟據,原始圖檔的大小/比例值可以獲得,見代碼:

css(m.cont.getElementsByTagName('div')[0],{        //m.cont.getElementsByTagName('div')[0]為浏覽框

    'display' : 'none',                //開始設定為不可見

    'width' : m.cont.clientWidth / m.scale - borderWid + 'px',    //原始圖檔的寬/比例值 - border的寬度

    'height' : m.cont.clientHeight / m.scale - borderWid + 'px',    原始圖檔的高/比例值 - border的寬度

    'opacity' : 0.5                    //設定透明度

})

放大框的大小則設定為于原始圖像相同大小,代碼如下:

css(m.mag,{

    'display' : 'none',

    'width' : m.cont.clientWidth + 'px',        //m.cont為原始圖像

    'height' : m.cont.clientHeight + 'px',

    'position' : 'absolute',

    'left' : m.cont.offsetLeft + m.cont.offsetWidth + 10 + 'px',    //放大框的位置為原始圖像的右方遠10px

    'top' : m.cont.offsetTop + 'px'

})

放大的圖像大小為,原始圖像大小*比例值,代碼如下:

css(m.img,{

    'position' : 'absolute',

    'width' : (m.cont.clientWidth * m.scale) + 'px',    //原始圖像的寬*比例值

    'height' : (m.cont.clientHeight * m.scale) + 'px'    //原始圖像的高*比例值

})

由于放大是根據比例進行放大,是以在浏覽框上和放大圖像上需要仔細計算,這也就是該程式的主要思維之一。

在第一次寫的程式裡,直接省去了onmouseover,因為直接使用onmousemove就可以滿足功能。而這次使用onmouseover是為了避免在使用過程中遇到select,在IE6下,select無法設定z-Index值,使得放大框的突然出現卻無法覆寫select。詳細下面在讨論,

在move方法中,最重要的就是如果做到滑鼠移動過程中,浏覽框随着滑鼠移動的同時,放大圖像也跟着運動,,使得放大圖像所顯示的範圍與浏覽框所在原始圖像位置一緻。

先說說浏覽框跟随滑鼠移動,主要代碼如下:

top:pos.y - this.offsetTop - parseInt(this.getElementsByTagName('div')[0].style.height) / 2

left:pos.x - this.offsetLeft - parseInt(this.getElementsByTagName('div')[0].style.width) / 2

由于是,對m.cont綁定事件,是以這個時候this指向m.cont。

JavaScript圖檔放大效果詳解

由圖像可以得知left=滑鼠x - this.offsetLeft - 浏覽框寬/2,是以跟據該幾何思想可以得出而代碼,而top的值也是根據一樣的道理所得,這裡就不做詳細解釋了。接下來就是在滑鼠運動的同時,放大圖像也要跟着改變top和left值,代碼如下:

css(magnifier.m.img,{

    'top' : - (parseInt(this.getElementsByTagName('div')[0].style.top) * magnifier.m.scale) + 'px',

    'left' : - (parseInt(this.getElementsByTagName('div')[0].style.left) * magnifier.m.scale) + 'px'

})

代碼很清晰的可以得出,隻需要在浏覽框的top和left值上*比例就可以了。而加上負号的原因是預設坐标為(0,0),而在移動過程中,始坐标隻會向負方向移動。

在該方法中有2個需要注意的地方:

1.

this.getElementsByTagName('div')[0].style.display = '';

應該放在設定this.getElementsByTagName('div')[0]的top與left之前,原因是如果display為none的話,無法擷取其寬和高。如果把display = ''放在設定top與left之後,會出現一個奇怪的現象,大家可以試下,該問題一直困擾了我很久,在多次嘗試中才發現問題再這上面。奇怪現象如下:

JavaScript圖檔放大效果詳解

2.

'top' : Math.min(Math.max(pos.y - this.offsetTop - parseInt(this.getElementsByTagName('div')[0].style.height) / 2,0),this.clientHeight - this.getElementsByTagName('div')[0].offsetHeight) + 'px';

這麼長的代碼可能讓人很困惑,我隻是用Math.max()和Math.min()去避免了采用if語句,自己偷了點懶,就是為了實作浏覽框不會超出原始圖像而已,仔細看看就清楚啦。

end方法很清晰,就是浏覽框和放大框進行隐藏。

[覆寫select]

在為了在IE6下可以覆寫select,我加入了2個放法createIframe和removeIframe。分别是在onmouseover事件裡建立一個iframe和在onmouseout裡銷毀iframe。

createIframe:function(elem){

    var layer = document.createElement('iframe');

    layer.tabIndex = '-1';

    layer.src = 'javascript:false;';

    elem.parentNode.appendChild(layer);

    layer.style.width = elem.offsetWidth + 'px';

    layer.style.height = elem.offsetHeight + 'px';

}

首先需要使用負的tabIndex值把iframe排除在tab序列之外,否則使用者可能會使用鍵盤導航到它,這就亂了套了,是以需要将tabIndex值設定為負的。另外,還要設定src,設定該值是為了避免在SSL頁面上出現問題.在IE中,沒有設定src的iframe将會自動裝載about:blank。IE将此視為不安全頁面,而且會産生一個警告對話框,内容是“該頁面包含安全和非安全的内容”。為了避免這個問題,可以将src設定為“javascript:false;”。(該段摘自<<JavaScript精髓>>)

而避免iframe在頁面所造成的混亂,是以在onmouseout中将iframe銷毀,而不對其進行隐藏。

使用說明

由于時間上的問題,是以沒有封裝的太好,主要是在CSS上,最好根據我所設定的那樣設定,感覺有些亂。希望大家能夠了解,而修改也不會太難.因為我自帶一個css()函數,隻要稍加設定就可以了。使用例子:

magnifier.init({

    cont : document.getElementById('magnifier'),

    img : document.getElementById('magnifierImg'),

    mag : document.getElementById('mag'),

    scale : 3

});

cont為container縮寫,指的是裝載原始圖像的div;

img則是放大的圖像;

mag則為magnifier縮寫,是指放大框;

scale為比例值,設定的值越大放大越大,但是這裡有個問題就是如果不可以整除時,會産生些很小的白邊,目前不知道如何解決;

至于浏覽框和原始圖像為m.cont.getElementsByTagName('img')[0]和m.cont.getElementsByTagName('div')[0],是以建議在裝載圖像的div中最好隻放一個div和img。

PS:剛剛給源碼也加了注釋了,有些沒有的,大概在下面的代碼說明會有.this指針搞的好混亂~希望大家能看的明白什麼回事..越用this越覺得不太好用啊..诶.>~搞了1個下午,累,休息下先哈。

繼續閱讀