【初始编辑】2013/4/21 14:55
【更新 2013/5/1 14:36 】
更新日志:加上坐标轴的箭头,加了两个函数(其实这个不算更新吧。。)。
优化了一下绘制方法:修改了清除画布的方法,极大提高效率。
自定义函数的功能还是没有修好。
代码中已经包含了一部分下次更新的功能:差不多是坐标轴的单位线
昨天我翻出来了以前一时兴起做的一个渣渣函数图绘制的代码,然后又一时兴起想把它改好一点,于是就出现了今天的这篇文章。
【我可是冒着期中考试前不复习的风险来做这个的】
【用chrome浏览效果更佳,其他浏览器可能会有卡顿现象】
现在已经做成了这个样子
源地址在这里,佳佳实验室的地址在这里,代码在文章最下面,自带注释
介绍,这是一个基于x坐标进行绘图[至少现在还是这样]的东东。
已经完成的功能:
画图
用鼠标拖动
滚轮放大缩小
回到中心
然后有一个功能”自定义函数”虽然显示在那里了,但还是不可以用的,有些错误还没时间解决。
现在的用法是打开源代码修改里面的函数数组,demo里的函数是这些
/*格式:函数,颜色,线宽,是否显示*/ functionlist[0]=["3*x+2","#66ccff",1,true]; functionlist[1]=["x*x+2*x-7","#123456",1,true]; functionlist[2]=["Math.tan(x)","#654321",1,true]; functionlist[3]=["Math.sin(x)","rgba(221,123,159,0.5)",20,true]; functionlist[4]=["1","#66ccff",1,true];
从这个代码里可以看出其实还有很多功能还没有做出来[上学的孩子苦逼啊]。
在以后的某个时间里我可能又会突然兴起在升级一下,现在就做到这样了,毕竟作为一个苦逼又たま的大天朝学生没有那么多时间来做这个,况且还有iti要做。
版权所有,转载和修改请注明出处。
<html> <meta charset="utf-8"> <head> <script src="jquery.js"> </script> <script> /*这里摆了一些图片的datauri,为了防止太多零碎文件就都放在这里了*/ icons = new Object; icons.home = ""; /*回到中心 25*25 */ icons.upjiantou = ""; /*上箭头 7*10 */ icons.rightjiantou = ""; /*右箭头 10*7 */ icons.heng5px = ""; /*横着5px*/ icons.shu5px = ""; /*竖着5px*/ upjiantouimg = new Image(); rightjiantouimg = new Image(); heng5px = new Image(); shu5px = new Image(); heng5px.src = icons.heng5px; shu5px.src = icons.shu5px; upjiantouimg.src = icons.upjiantou; rightjiantouimg.src = icons.rightjiantou; /*把图标载入image对象*/ </script> <style> body { padding: 0; margin: 0; cursor: default; } #zuobiaozhou { position: absolute; top: 0; left: 0; cursor: default; z-index: 0; } #hanshutu { position: absolute; top: 0; left: 0; cursor: default; z-index: 1; } div#control { position: fixed; right: -266px; top: 0px; height: 100%; width: 250px; background-color: rgba(129, 150, 141, 0.6); padding: 1em; overflow-x: hidden; z-index: 999; } div#home { height: 25px; width: 25px; cursor: pointer; } div.sidediv { width: 100%; color: rgb(67, 49, 162); font-size: 13px; } textarea#zidingyifunction { width: 100%; max-width: 100%; max-height: 160px; height: 160px; } </style> <script> var functionlist = new Array(); /*格式:函数,颜色,线宽,是否显示*/ functionlist[0] = ["3*x+2", "#66ccff", 1, true]; functionlist[1] = ["x*x+2*x-7", "#123456", 1, true]; functionlist[3] = ["Math.sin(x)", "rgba(221,123,159,0.5)", 10, true]; functionlist[4] = ["1", "#66ccff", 1, true]; functionlist[5] = ["Math.pow(-1,x)", "#CC33FF", 2, true]; functionlist[2] = ["Math.pow(2,x)", "#CC6633", 1, true]; functionlist[6] = ["Math.sqrt(64-x*x)", "rgb(255,204,0)", 6, true]; functionlist[7] = ["-Math.sqrt(64-x*x)", "red", 2, true]; functionlist[8] = ["Math.acos(Math.sqrt(Math.sin(x)))", "blue", 1, true]; functionlist[9] = ["x/(x*x+3*x+1)", "blue", 1, true]; </script> </head> <body> <canvas id="zuobiaozhou"> </canvas> <canvas id="hanshutu"> 如果你什么都看不到,请到 <a href="http://browsehappy.com/" title="browsehappy" target="_blank" style="color:rgb(153,153,51);text-decoration:none;"> 这里 </a> 寻找最新的浏览器~ </canvas> <div id="control"> <abbr title="回到中心"> <div id="home"> </div> </abbr> <div class="sidediv"> 自定义函数 <span style="float:right;cursor:pointer;" id="refreshzidingfun"> 刷新自定义函数 </span> <br> <textarea id="zidingyifunction"> </textarea> </div> <br> <div class="sidediv"> <span style="cursor:pointer;" id="refreshfuns"> 刷新函数 </span> <br> </div> </div> <script> /*变量名中的V觉绝大部分意思是虚拟*/ /*定义变量*/ var zuobiao$obj, pianyix = 0, pianyiy = 0, /*偏移量*/ bei = 1, /*缩放倍数*/ clickpointx = 0, clickpointy = 0, /*点击坐标*/ vzoom = new Object, /*虚拟坐标系空间函数对象*/ zuobiaocanvasobj, /*坐标轴canvas dom对象*/ zuobiaozhoudrawobj, /*坐标轴绘图上下文*/ hanshutucanvasobj, /*函数图canvas dom对象*/ hanshutudrawobj, /*函数图绘图上下文*/ windowwidth = window.innerWidth, windowheight = window.innerHeight, /*用于快速存取的整体长宽变量*/ xcenter = windowwidth / 2, /*中心坐标*/ ycenter = windowheight / 2, hanshutuurltmp, /*没有用*/ realxtmp, realytmp, /*真实坐标的临时变量*/ vxtmp, vytmp, /*虚拟坐标的临时变量*/ v1danwei, /*虚拟坐标里的一个单位的真实像素数*/ basetimestemp = 0, /*单位基数乘次计数*/ basenumber = 1, /*单位基数*/ zuobiaozhoux, zuobiaozhouy, /*坐标轴的两条轴位置*/ zuobiaoxianx, zuobiaoxiany, zuobiao$obj = $("#zuobiaozhou"), hanshu$obj = $("#hanshutu"), zuobiaocanvasobj = zuobiao$obj[0], hanshutucanvasobj = hanshu$obj[0], zuobiaozhoudrawobj = zuobiaocanvasobj.getContext("2d"), hanshutudrawobj = hanshutucanvasobj.getContext("2d"); hanshutudrawobj.fillStyle = 'rgba(255,255,255,0)'; zuobiaozhoudrawobj.fillStyle = 'rgba(255,255,255,0)'; /*虚拟坐标算法和反算法*/ //参数:虚拟位置坐标。返回值:在canvas元素里的实际坐标 vzoom.realx = function(vx) { return xcenter + pianyix + vx * 40 * bei; } vzoom.realy = function(vy) { return ycenter - pianyiy - vy * 40 * bei; } //参数:实际位置坐标。返回值:虚拟坐标 vzoom.vx = function(realx) { return (realx - xcenter - pianyix) / bei / 40; } vzoom.vy = function(realy) { return (realy - ycenter + pianyiy) / -bei / 40; } /*函数解析方法*/ var funs = new Array(); funs.refreshfuns = function() { var tmpfun for (i = 0; functionlist[i]; i++) { tmpfun = functionlist[i][0]; try { eval('funs[' + i + ']=function(x){return ' + tmpfun + ';}'); } catch(e) { console.log("第" + (i + 1) + "个函数解析失败"); continue; }; } } /*设置宽高*/ function setkuangao() { zuobiaocanvasobj.width = windowwidth; zuobiaocanvasobj.height = windowheight; hanshutucanvasobj.width = windowwidth; hanshutucanvasobj.height = windowheight; } /*画线*/ function paintline(x, y) { hanshutudrawobj.lineTo(x, y); } /*画函数图*/ function drawhanshutu() { hanshutudrawobj.clearRect(0, 0, windowwidth, windowheight); for (funnunm = 0; functionlist[funnunm]; funnunm++) { if (functionlist[funnunm][3]) { hanshutudrawobj.strokeStyle = functionlist[funnunm][1]; hanshutudrawobj.lineWidth = functionlist[funnunm][2]; hanshutudrawobj.beginPath(); for (i = 0; i < windowwidth; i++) { vxtmp = vzoom.vx(i); if (funs[funnunm](vxtmp) != Infinity) { paintline(i, vzoom.realy(funs[funnunm](vxtmp))); } else { nextpoint = vzoom.vx(i + 1); hanshutudrawobj.moveTo(i + 1, funs[funnunm](nextpoint)) } } } hanshutudrawobj.stroke(); } } /*画坐标轴*/ function drawzuobiaozhou() { zuobiaozhoudrawobj.beginPath(); zuobiaozhoudrawobj.clearRect(0, 0, windowwidth, windowheight); zuobiaozhoudrawobj.strokeStyle = "#000"; zuobiaozhoudrawobj.lineWidth = 1; realxtmp = vzoom.realx(0); realytmp = vzoom.realy(0); //console.log(realxtmp+" "+realytmp);//调试用 信息输出 /*以下都是让xy轴保持显示在显示区内*/ if (realxtmp >= windowwidth) { zuobiaozhoux = windowwidth - 0.5; } else if (realxtmp <= 0) { zuobiaozhoux = 0.5; } else { zuobiaozhoux = xcenter + pianyix; } if (realytmp <= 0) { zuobiaozhouy = 0.5; } else if (realytmp >= windowheight) { zuobiaozhouy = windowheight - 0.5; } else { zuobiaozhouy = ycenter - pianyiy; } zuobiaozhoudrawobj.moveTo(zuobiaozhoux, 0); zuobiaozhoudrawobj.lineTo(zuobiaozhoux, windowheight); zuobiaozhoudrawobj.moveTo(0, zuobiaozhouy); zuobiaozhoudrawobj.lineTo(windowwidth, zuobiaozhouy); zuobiaozhoudrawobj.drawImage(upjiantouimg, zuobiaozhoux - 3.5, 0); zuobiaozhoudrawobj.drawImage(rightjiantouimg, windowwidth - 10, zuobiaozhouy - 3.5); zuobiaoxianx = vzoom.vx(0); zuobiaoxiany = vzoom.vy(windowheight); /*for*/ zuobiaozhoudrawobj.stroke(); } /*取距离1、2、5距离最近的数*/ function getfrom125(number) { if (number <= 1) return 1; else if (number > 1 && number <= 2.5) return 2; else if (number > 2.5) return 5; } /*设置单位基数*/ function setbasenumber() { v1danwei = 50 / (vzoom.realx(1) - vzoom.realx(0)); if (v1danwei < 1) { for (; v1danwei < 1; basetimestemp++) { v1danwei *= Math.pow(10, basetimestemp); } basenumber = getfrom125(v1danwei); basenumber *= Math.pow(10, -basetimestemp); } else if (v1danwei > 10) { for (; v1danwei > 10; basetimestemp--) { v1danwei *= Math.pow(10, basetimestemp); } basenumber = getfrom125(v1danwei); basenumber *= Math.pow(10, -basetimestemp); } else { basenumber = 1; } } function refreshzidinghanshu() { localStorage.zidingyifunction = document.getElementById("zidingyifunction").value; eval(document.getElementById("zidingyifunction").value); drawhanshutu(); } function loadzidinghanshu() { if (localStorage.zidingyifunction) { eval(localStorage.zidingyifunction); document.getElementById("zidingyifunction").value = localStorage.zidingyifunction; } } /*由鼠标引发的移动事件*/ hanshu$obj.mousedown(function(event) { event.preventDefault(); clickpointx = event.clientX; clickpointy = event.clientY; /*hanshu$obj.unbind("mouseup"); hanshu$obj.unbind("mousemove");*/ hanshu$obj.mouseup(function(event) { hanshu$obj.unbind("mousemove"); hanshu$obj.unbind("mouseup"); hanshu$obj.unbind("mouseout"); }); hanshu$obj.mousemove(function(event) { event.preventDefault(); pianyix = pianyix + event.clientX - clickpointx; pianyiy = pianyiy - event.clientY + clickpointy; drawzuobiaozhou(); drawhanshutu(); clickpointx = event.clientX; clickpointy = event.clientY; }); hanshu$obj.mouseout(function() { hanshu$obj.unbind("mouseup"); hanshu$obj.unbind("mousemove"); }); }); $(window).resize(function() { /*尺寸改变的时候*/ windowwidth = window.innerWidth; /*重新计算虚拟坐标所需的宽高*/ windowheight = window.innerHeight; xcenter = windowwidth / 2; ycenter = windowheight / 2; setkuangao(); /*重新设置宽高*/ drawzuobiaozhou(); /*重画*/ drawhanshutu(); }); setkuangao(); funs.refreshfuns(); /*开始画之前先刷新一下函数列表*/ drawzuobiaozhou(); drawhanshutu(); $("#home").click(function(e) { /*回到中心*/ pianyix = pianyiy = 0; bei = 1; drawzuobiaozhou(); drawhanshutu(); }); var controlpannel = $("#control"); controlpannel.mouseenter(function(e) { controlpannel.stop().animate({ right: 0 }, "fast"); }); controlpannel.mouseleave(function(e) { controlpannel.stop().animate({ right: -266 }, "fast"); }); /*滚轮控制部分*/ var scrollFunc = function(e) { var direct = 0; e = e || window.event; var t1 = document.getElementById("wheelDelta"); var t2 = document.getElementById("detail"); var data = e.wheelDelta ? e.wheelDelta: e.detail; if (data == -3 || data == 120) { /*不知为什么firefox的数值和其他浏览器是反的。。*/ bei *= 1.1; /*按1.1倍放大*/ pianyix *= 1.1, pianyiy *= 1.1; /*同时倍增偏移量使放大中心保持在显示区中心*/ } else if (data == 3 || data == -120) { bei *= 0.9; pianyix *= 0.9, pianyiy *= 0.9; } drawzuobiaozhou(); drawhanshutu(); } if (document.addEventListener) { document.addEventListener('DOMMouseScroll', scrollFunc, false); } hanshutucanvasobj.onmousewheel = scrollFunc; /*刷新自定义函数*/ $("#refreshzidingfun").click(function(e) { refreshzidinghanshu(); }); /*刷新函数列表*/ $("#refreshfuns").click(function(e) { funs.refreshfuns(); }); /*下面都杂项函数*/ function fillicons() { document.getElementById("home").style.backgroundImage = "url(" + icons.home + ")"; } </script> <script> fillicons(); upjiantouimg.onload = function() { drawzuobiaozhou(); } rightjiantouimg.onload = function() { drawzuobiaozhou(); } </script> </body> </html>
本文发布于 https://luojia.me
本站文章未经文下加注授权不得拷贝发布。