JavaScript實現(xiàn)一個簡易的計算器實例代碼

自己期末復(fù)習(xí)的時候就一直想要寫一個計算器,閑暇的時候也在想具體怎么實現(xiàn),覺得應(yīng)該不難,但就是想寫。昨天終于可以開始動工,剛開始還以為實現(xiàn)出來需要一個周左右至少兩天的時間,想著實現(xiàn)完我就可以先回家兩天了。但沒想到整個實現(xiàn)過程算比較順利吧,昨天用了大概六個小時完成了從設(shè)計到具體實現(xiàn)。

創(chuàng)新互聯(lián)公司專注于企業(yè)成都營銷網(wǎng)站建設(shè)、網(wǎng)站重做改版、清水河網(wǎng)站定制設(shè)計、自適應(yīng)品牌網(wǎng)站建設(shè)、H5建站、商城網(wǎng)站定制開發(fā)、集團公司官網(wǎng)建設(shè)、外貿(mào)網(wǎng)站制作、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁設(shè)計等建站業(yè)務(wù),價格優(yōu)惠性價比高,為清水河等各大城市提供網(wǎng)站開發(fā)制作服務(wù)。

有大概一個月沒怎么寫代碼了,整個大腦都不適應(yīng),反應(yīng)也慢,一些基本的東西都有點模糊不清了。可能是原來就沒有太理解,再加上沒有其余練習(xí),導(dǎo)致效率有些低。

正文

html代碼:

<div class="errorHint" id="errorHint"><img src="https://github.com/crystalYY/calculator/blob/master/img/error.png?raw=true"></div>
	<table cellpadding="0">
		<tr>
			<th colspan="5">計算器</th>
		</tr>
		<tr>
			<td colspan="5">
				<input type="text" value="0" name="showResult">
			</td>
		</tr>
		<tr>
			<td><button>7</button></td>
			<td><button>8</button></td>
			<td><button>9</button></td>
			<td><button class="setChange" id="backSpace">退格</button></td>
			<td><button class="setChange" id="clearNum">C</button></td>
		</tr>
		<tr>
			<td><button>4</button></td>
			<td><button>5</button></td>
			<td><button>6</button></td>
			<td><button>+</button></td>
			<td><button>-</button></td>
		</tr>
		<tr>
			<td><button>1</button></td>
			<td><button>2</button></td>
			<td><button>3</button></td>
			<td><button>*</button></td>
			<td><button>/</button></td>
		</tr>
		<tr>
			<td><button>0</button></td>
			<td><button>.</button></td>
			<td><button>%</button></td>
			<td colspan="2"><button class="setChange" id="gainResult">Enter</button></td>
		</tr>
	</table>
	<script type="text/javascript" src='index.js'>		
	</script>

 CSS代碼:

*{margin: 0px; padding: 0px;}
		.errorHint{position: absolute; left: 130px; top:-282px;}
		.showError{border:1px solid red;}
		table{ border: 2px solid #996c33; width: 550px; padding: 10px; margin: 150px auto; background:url(https://github.com/crystalYY/calculator/blob/master/img/bg2.jpg?raw=trueg) left center no-repeat; border-radius: 10px;}
		table td{
			text-align: center;
			width: 100px;
			height: 40px;
			padding-left: 2px;
			padding-bottom: 2px;
		}
		table th{
			font-size: 18px;
			font-family:'楷體';
			color: 	#8B0000;
		}
		table td button{
			width: 98%;
			height: 98%;
			font-size: 16px;
			font-family: 'Microsoft yahei';
			background: none;
			color: 	#8B4726;
			outline:none;
			border:1px solid #000;
			border-radius: 5px;
			cursor: pointer;
		}
		table td input{
			width: 100%;
			margin: 10px 0;
			padding: 5px;
			border:1px solid #996c33;
			box-sizing: border-box;						
			text-align: right;
			font-size: 16px;
			font-family: 'Microsoft yahei';
		}

JS代碼:

var oinput=document.getElementsByTagName('input')[0];
		//獲取外部樣式
		function getStyle(obj, name)
		{
			if(obj.currentStyle)
			{
				return obj.currentStyle[name];
			}
			else
			{
				return getComputedStyle(obj, false)[name];
			}
		}
		//漸變動畫
		function move(obj,attr,tar){
			clearInterval(obj.timer);
			obj.timer=setInterval(function(){
				var cur=parseInt(getStyle(obj,attr));
				var itarget=parseInt(tar);
				var speed=(itarget-cur)/6;
				speed=speed>0?Math.ceil(speed):Math.floor(speed);
				obj.style[attr]=parseInt(getStyle(obj,attr))+speed+'px';
				if(speed==0){
					clearInterval(obj.timer);
				}
			},30);
		}
		//事件綁定函數(shù)
		function addEvent(obj,ev,fun){
			if(obj.attachEvent){
				obj.attachEvent('on'+ev,fun);
			}else{
				obj.addEventListener(ev,fun,false);
			}
		}
		//阻止默認(rèn)行為
		function stopEvent(ev){
			var e=ev||window.event;
			if(e.preventDefault){
				e.preventDefault();
			}
			else{
				e.returnValue=false;//ie
			}
		}
		//計算最終結(jié)果
		function getResult(){
			function evalResult(){
				var result=eval(oinput.value);
				return result;			
			}
			//捕獲異常
			try{
				var x=evalResult();
				return x;
			}
			catch (e){
				oinput.className='showError';
				var errorHint=document.getElementById('errorHint');
				move(errorHint,'top',0);
				setTimeout(function(){
					oinput.className='';
					move(errorHint,'top',-282);
				},2000);
				return oinput.value;
			}
		}
		//文本框獲取焦點,錯誤提示消失
		//按下回車得到結(jié)果
		function enterResult(ev){
			var e=ev||window.event;
			if(e.keyCode==13){
				stopEvent(ev);//阻止enter鍵的默認(rèn)行為
				var result=getResult();
				oinput.value=result;
			}
		}
		//綁定點擊事件
		function init(){
			var otable=document.getElementsByTagName('table')[0];
			addEvent(otable,'keydown',function(ev){
				enterResult(ev);
			});
			addEvent(otable,'click',function(ev){
				stopEvent(ev);
				var e=ev||window.event;
				var itat=e.target||e.srcElement;
				var obtns=document.getElementsByTagName('button');
				if(itat.nodeName.toLowerCase()=='button'){
					for(var i=0;i<obtns.length;i++){
						obtns[i].style.borderColor='#000';
					}
					itat.style.borderColor='white';
					if(itat.className!='setChange'){
						if(oinput.value=='0'){
							oinput.value='';
							oinput.value+=itat.innerHTML;
						}
						else{
							oinput.value+=itat.innerHTML;
						}
					}else{
						if(itat.id=='backSpace'){
							oinput.value=oinput.value.toString().slice(0,-1);
						}
						else if(itat.id=='clearNum'){
							oinput.value='0';
						}else{
							var result=getResult();
							oinput.value=result;
						}
					}
				}
			});
		}
		init();

正常顯示界面

JavaScript實現(xiàn)一個簡易的計算器實例代碼

錯誤提示界面

JavaScript實現(xiàn)一個簡易的計算器實例代碼

效果實現(xiàn):http://codepen.io/crystalYY/pen/jAkNVz

實現(xiàn)思路

1.使用table畫出整個界面。

借鑒了其他人已經(jīng)實現(xiàn)了的結(jié)構(gòu),發(fā)現(xiàn)他們有一些人沒有直接在td里寫1,2,3或者退格什么的,而是又嵌套了一個button,我其實到現(xiàn)在也沒有太理解為什么要這樣,只是在排版的時候感覺到有些作用:因為margin對td 不起作用,只能設(shè)置padding。

2.使用eval函數(shù)計算最終結(jié)果,并捕獲異常

function getResult(){
   function evalResult(){
    var result=eval(oinput.value);
    return result;   
   }
   //捕獲異常
   try{
    var x=evalResult();
    return x;
   }
   catch (e){
    oinput.className='showError';
    var errorHint=document.getElementById('errorHint');
    move(errorHint,'top',0);
    setTimeout(function(){
     oinput.className='';
     move(errorHint,'top',-282);
    },2000);
    return oinput.value;
   }
  }

eval函數(shù)第一次使用,w3c上對它的定義如下

eval() 函數(shù)可計算某個字符串,并執(zhí)行其中的的 JavaScript 代碼。

有了這個函數(shù)得到最終結(jié)果就很容易了。我的思路是在用戶輸入要計算的式子時不加干預(yù),最終的計算從input輸入框中獲取value值,然后把這個value值作為參數(shù)傳遞給eval,并使用try catch(exception)來捕獲并處理異常。

3.通過事件代理綁定事件

因為每個button都需要有一個點擊事件,如果一個一個去綁定,會導(dǎo)致代碼十分的不簡潔,而且效率也非常低。這時就可以考慮使用事件代理,由于事件冒泡的原理,我們可以把點擊事件綁定在table上,然后通過判斷事件發(fā)生的具體對象來做出不同的反應(yīng),調(diào)用不同的函數(shù)。

4.其他效果

可以根據(jù)自己的設(shè)計思路,添加其他的效果。我主要是添加了一個錯誤提示的動畫:如果eval函數(shù)拋出異常,則從上面緩慢滑下一個圖片,并且通過setTimeout來設(shè)置了停留的時間。

5.注意細節(jié)

在設(shè)置enter鍵按下獲得結(jié)果的時候,keydown事件對象應(yīng)該為整個table,并且應(yīng)該阻止enter鍵的默認(rèn)行為
獲取元素樣式時需要寫一個兼容函數(shù),因為obj.style.attr只能獲取行間樣式,要像獲取外部樣式需要用getComputedStyle(obj,false)[attr]或兼容IE的obj.currentStyle[attr]。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

分享文章:JavaScript實現(xiàn)一個簡易的計算器實例代碼
轉(zhuǎn)載注明:http://muchs.cn/article2/pipiic.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站策劃網(wǎng)站制作、標(biāo)簽優(yōu)化域名注冊、自適應(yīng)網(wǎng)站、響應(yīng)式網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

h5響應(yīng)式網(wǎng)站建設(shè)