![]() |
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> <HTML LANG="ja"> <HEAD> <META HTTP-EQUIV="Content-Type" CONTENT="text/html;charset=Shift_JIS"> <TITLE>スパイログラフ</TITLE> </HEAD> <BODY BGCOLOR="#CCFFFF"> <CENTER> <B>スパイログラフ (Netscape 7)</B> <BR><BR> <FORM> <TABLE><TR> <TD VALIGN=TOP> <IMG ID="img" SRC="" WIDTH=480 HEIGHT=480> </TD> <TD WIDTH=10></TD> <TD VALIGN=TOP NOWRAP> <BR> <IMG SRC="images/spiro.gif" WIDTH=149 HEIGHT=165><BR><BR> r1: <INPUT TYPE=TEXT ID="r1" SIZE=4> (整数)<BR> r2: <INPUT TYPE=TEXT ID="r2" SIZE=4> (整数)<BR> r3: <INPUT TYPE=TEXT ID="r3" SIZE=8><BR> スケール: <INPUT TYPE=TEXT ID="scale" SIZE=8><BR><BR> 色: <SELECT ID="color"> <OPTION SELECTED>黒 <OPTION>赤 <OPTION>ライム <OPTION>青 <OPTION>黄 <OPTION>赤紫 <OPTION>水色 <OPTION>栗色 <OPTION>緑 <OPTION>ネイビー <OPTION>オリーブ <OPTION>紫 <OPTION>青緑 <OPTION>橙 <OPTION>灰 </SELECT><BR><BR> <INPUT TYPE=CHECKBOX ID="quiet">途中経過を表示しない<BR><BR> <INPUT TYPE=BUTTON ID="start" VALUE="描画開始" onClick="start_drawing()"> <INPUT TYPE=BUTTON ID="can" VALUE="描画中止" DISABLED onClick="cancel = true"> <INPUT TYPE=BUTTON ID="undo" VALUE="やり直し" DISABLED onClick="restore_image()"><BR><BR> <INPUT TYPE=BUTTON ID="clear" VALUE="クリア" onClick="clear_image()"><BR><BR> <DIV ID="prog" ALIGN=CENTER STYLE="visibility:hidden"><FONT SIZE="+1">◆ 処理中 ◆</FONT></DIV> </TD> </TR></TABLE> </FORM> </CENTER> <SCRIPT LANGUAGE="JavaScript1.2" TYPE="text/javascript"> <!-- // 整数 キー入力チェック function check_int(e) { return (e.which < 0x20 || e.which >= 0x30 && e.which <= 0x39 || e.ctrlKey || e.metaKey); } // 小数 キー入力チェック function check_float(e) { return (e.which < 0x20 || e.which >= 0x30 && e.which <= 0x39 || e.which == 0x2e || e.ctrlKey || e.metaKey); } // 描画開始 function start_drawing() { // 入力チェック var r1 = elem_r1.value; if(r1.search(/^\s*\d+\s*$/) == -1) r1 = 0; else r1 = parseInt(r1, 10); if((isNaN(r1)) ? true : (r1 == 0 || r1 >= 10000)) { alert("r1 の入力に誤りがあります。"); return; } var r2 = elem_r2.value; if(r2.search(/^\s*\d+\s*$/) == -1) r2 = 0; else r2 = parseInt(r2, 10); if((isNaN(r2)) ? true : (r2 == 0 || r2 >= 10000)) { alert("r2 の入力に誤りがあります。"); return; } if(r2 >= r1) { alert("r2 には r1 より小さい値を入力してください。"); return; } var r3 = elem_r3.value; if(r3.search(/^\s*\d*\.?\d*\s*$/) == -1) r3 = 0; else r3 = parseFloat(r3); if((isNaN(r3)) ? true : (r3 == 0 || r3 >= 10000)) { alert("r3 の入力に誤りがあります。"); return; } var scale = elem_scale.value; if(scale.search(/\S/) == -1) { // 未入力 scale = 0; } else { if(scale.search(/^\s*\d*\.?\d*\s*$/) == -1) scale = 0; else scale = parseFloat(scale); if((isNaN(scale)) ? true : (scale == 0 || scale >= 10000)) { alert("スケールの入力に誤りがあります。"); return; } if(scale < r1 - r2 + r3) { alert("スケールには r1 - r2 + r3 以上の値を入力してください。"); return; } } color = document.getElementById("color").selectedIndex; // 色 quiet = document.getElementById("quiet").checked // 途中経過を表示しない cancel = false; save_data = data; elem_start.disabled = true; elem_undo.disabled = true; elem_clear.disabled = true; if(quiet) elem_prog.style.visibility = "visible"; else elem_can.disabled = false; var rd = r1 - r2; rr = rd / r2; if(!scale) scale = r1 - r2 + r3; rdp = rd * 239.99 / scale; r3p = r3 * 239.99 / scale; // GCD を求める var a, b; for(a = r1, b = r2; ; ) { var w; if(a < b) { w = a; a = b; b = w; } if(!b) break; w = a % b; a = b; b = w; } div_end = NDIV * (r2 / a); prev_x = Math.floor(rdp + r3p) + 240; prev_y = 240; setTimeout(draw, 0, 0); } // 画像復旧 function restore_image() { elem_undo.disabled = true; data = save_data; elem_img.src = data; } // クリア function clear_image() { elem_undo.disabled = true; var d3 = b64.charAt(0x3f); d3 += d3; data = header; for(var i = 76800; i; i--) data += d3; elem_img.src = data; } // 描画 function draw(div) { if(cancel) { elem_can.disabled = true; elem_start.disabled = false; elem_clear.disabled = false; data = save_data; elem_img.src = data; return; } for(var cnt = 10; ; ) { var x, y; if(div == div_end) { x = Math.floor(rdp + r3p) + 240; y = 240; } else { var th1 = PI2DIV * div; var th2 = rr * (th1); x = Math.floor(rdp * Math.cos(th1) + r3p * Math.cos(th2)) + 240; y = Math.floor(rdp * Math.sin(th1) - r3p * Math.sin(th2)) + 240; } if(x != prev_x || y != prev_y) { // 直線補間 var dx = Math.abs(x - prev_x); var dy = Math.abs(y - prev_y); var rem = 0; var xi = (prev_x < x) ? 1 : -1; var yi = (prev_y < y) ? 1 : -1; if(dx >= dy) { for(var d = dx; d; d--) { set_pixel(prev_x, prev_y); if((rem += dy) >= dx) { prev_y += yi; rem -= dx; } prev_x += xi; } } else { for(var d = dy; d; d--) { set_pixel(prev_x, prev_y); if((rem += dx) >= dy) { prev_x += xi; rem -= dy; } prev_y += yi; } } if(!quiet) cnt--; } if(div++ == div_end) { // 終了 elem_can.disabled = true; elem_prog.style.visibility = "hidden"; elem_start.disabled = false; elem_undo.disabled = false; elem_clear.disabled = false; elem_img.src = data; return; } if(!cnt) { elem_img.src = data; setTimeout(draw, 0, div); return; } } } // 1 ピクセル描画 function set_pixel(x, y) { var i = y * 320 + (Math.floor(x / 3) << 1) + header_len; switch(x % 3) { case 0: data = data.substr(0, i) + b64.charAt((color << 2) | b64.indexOf(data.charAt(i)) & 0x3) + data.substr(i + 1); break; case 1: data = data.substr(0, i) + b64.charAt(b64.indexOf(data.charAt(i)) & 0x3c | (color >> 2)) + b64.charAt(((color & 0x3) << 4) | b64.indexOf(data.charAt(i + 1)) & 0xf) + data.substr(i + 2); break; default: data = data.substr(0, i + 1) + b64.charAt(b64.indexOf(data.charAt(i + 1)) & 0x30 | color) + data.substr(i + 2); } } //---------------------------------------------------------- // BMP ヘッダ bmp_header = [ // BITMAPFILEHEADER 0x42, 0x4d, // BM 0x78, 0xc2, 0x01, 0x00, // Size 115320 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, // OffBits 120 // BITMAPINFOHEADER 0x28, 0x00, 0x00, 0x00, // Size 40 0xe0, 0x01, 0x00, 0x00, // Width 480 0xe0, 0x01, 0x00, 0x00, // Height 480 0x01, 0x00, // Planes 1 0x04, 0x00, // BitCount 4 0x00, 0x00, 0x00, 0x00, // Compression 0x00, 0xc2, 0x01, 0x00, // SizeImage 115200 0xc4, 0x0e, 0x00, 0x00, // XPelsPerMeter 3780 0xc4, 0x0e, 0x00, 0x00, // YPelsPerMeter 3780 0x00, 0x00, 0x00, 0x00, // ClrUsed 0x00, 0x00, 0x00, 0x00, // ClrImportant // カラー テーブル(RGBQUAD) 0x00, 0x00, 0x00, 0x00, // 黒 0x00, 0x00, 0xff, 0x00, // 赤 0x00, 0xff, 0x00, 0x00, // ライム 0xff, 0x00, 0x00, 0x00, // 青 0x00, 0xff, 0xff, 0x00, // 黄 0xff, 0x00, 0xff, 0x00, // 赤紫 0xff, 0xff, 0x00, 0x00, // 水色 0x00, 0x00, 0x80, 0x00, // 栗色 0x00, 0x80, 0x00, 0x00, // 緑 0x80, 0x00, 0x00, 0x00, // ネイビー 0x00, 0x80, 0x80, 0x00, // オリーブ 0x80, 0x00, 0x80, 0x00, // 紫 0x80, 0x80, 0x00, 0x00, // 青緑 0x00, 0x80, 0xff, 0x00, // 橙 0x80, 0x80, 0x80, 0x00, // 灰 0xff, 0xff, 0xff, 0x00, // 白 // BASE64 エンコードするのに都合がよくなるよう,ピクセル データのオフセットが // 3 の倍数になるようにパディングを入れておく. 0x00, 0x00 ]; // BASE64 エンコード キャラクタ b64 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; NDIV = 500; // 2π の分割数 PI2DIV = Math.PI * 2 / NDIV; header = "data:image/bmp;base64,"; for(i = 0; i < bmp_header.length; i += 3) { b3 = (bmp_header[i] << 16) | (bmp_header[i + 1] << 8) | bmp_header[i + 2]; header += b64.charAt(b3 >> 18); header += b64.charAt((b3 >> 12) & 0x3f); header += b64.charAt((b3 >> 6) & 0x3f); header += b64.charAt(b3 & 0x3f); } header_len = header.length; elem_img = document.getElementById("img"); elem_r1 = document.getElementById("r1"); elem_r2 = document.getElementById("r2"); elem_r3 = document.getElementById("r3"); elem_scale = document.getElementById("scale"); elem_start = document.getElementById("start"); elem_can = document.getElementById("can"); elem_undo = document.getElementById("undo"); elem_clear = document.getElementById("clear"); elem_prog = document.getElementById("prog"); elem_r1.onkeypress = check_int; elem_r2.onkeypress = check_int; elem_r3.onkeypress = check_float; elem_scale.onkeypress = check_float; clear_image(); document.forms[0].reset(); // ページを再ロードしたときのため //--> </SCRIPT> </BODY> </HTML> |