2012-02-08 22:37:04 -05:00
|
|
|
//
|
|
|
|
// Applesoft BASIC in Javascript
|
|
|
|
// High Resolution Graphics (HiRes) Emulation
|
|
|
|
//
|
|
|
|
|
|
|
|
// Copyright (C) 2009-2011 Joshua Bell
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
|
|
|
|
// This variation does not rely on a canvas element with the same dimensions
|
|
|
|
// as the pixel buffer; instead, it paints pixels on the canvas using rectangles.
|
|
|
|
// This works even if the underlying canvas implementation does not support
|
|
|
|
// scaling (e.g. fallbacks for IE using VML or Flash)
|
|
|
|
|
|
|
|
// Usage:
|
|
|
|
//
|
|
|
|
// var hires = new LoRes( element, width, height )
|
2012-08-09 17:48:41 -04:00
|
|
|
// hires.clear( [color_index] )
|
2012-02-08 22:37:04 -05:00
|
|
|
// hires.setColor( color_index )
|
|
|
|
// hires.plot( x, y )
|
|
|
|
// hires.plot_to( x, x )
|
|
|
|
// hires.getPixel( x, x ) // for "HSCRN"
|
|
|
|
// hires.show( bool )
|
|
|
|
// { width: w, height: h } = hires.getScreenSize()
|
|
|
|
//
|
|
|
|
// Example:
|
|
|
|
//
|
|
|
|
// <script>
|
|
|
|
// hires = new HiRes( document.getElementById( 'hires' ), 140, 192 );
|
|
|
|
// interpreter = new BasicInterpreter( tty, lores, hires, paddle );
|
|
|
|
// </script>
|
|
|
|
// <canvas id="hires"></canvas>
|
|
|
|
|
|
|
|
function HiRes(element, width, height) {
|
|
|
|
var COLORS, // Apple II to HTML color table
|
|
|
|
last_x = 0,
|
|
|
|
last_y = 0,
|
|
|
|
pixels = [],
|
|
|
|
color = 0,
|
|
|
|
context = element.getContext("2d");
|
|
|
|
|
|
|
|
pixels.length = width * height;
|
|
|
|
|
|
|
|
// Colors c/o USENET:
|
|
|
|
// Date: Wed, 05 Sep 2007 01:04:20 +0200
|
|
|
|
// From: Linards Ticmanis <ticmanis@gmx.de>
|
|
|
|
// Newsgroups: comp.sys.apple2
|
|
|
|
// Subject: Re: Double hires mode color artifacts
|
|
|
|
// Message-ID: <46dde477$0$4527$9b4e6d93@newsspool3.arcor-online.net>
|
|
|
|
|
|
|
|
COLORS = [
|
|
|
|
'#000000', // Black 1
|
|
|
|
'#2fbc1a', // Green
|
|
|
|
'#d043e5', // Violet
|
|
|
|
'#ffffff', // White 1
|
|
|
|
'#000000', // Black 2
|
|
|
|
'#d06a1a', // Orange
|
|
|
|
'#2f95e5', // Medium Blue
|
|
|
|
'#ffffff' // White 2
|
|
|
|
];
|
|
|
|
|
2012-08-09 17:48:41 -04:00
|
|
|
this.clear = function(opt_color) {
|
2012-02-08 22:37:04 -05:00
|
|
|
var i;
|
2012-08-09 17:48:41 -04:00
|
|
|
context.clearRect(0, 0, element.width, element.height);
|
|
|
|
pixels = [];
|
|
|
|
pixels.length = width * height;
|
|
|
|
if (arguments.length >= 1) {
|
|
|
|
context.fillStyle = COLORS[opt_color];
|
2012-02-08 22:37:04 -05:00
|
|
|
context.fillRect(0, 0, element.width, element.height);
|
|
|
|
for (i = 0; i < pixels.length; i += 1) {
|
2012-08-09 17:48:41 -04:00
|
|
|
pixels[i] = opt_color;
|
2012-02-08 22:37:04 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
this.setColor = function(newColor) {
|
|
|
|
color = Math.floor(newColor) % COLORS.length;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
function drawPixel(x, y) {
|
|
|
|
var sx = element.width / width,
|
|
|
|
sy = element.height / height;
|
|
|
|
context.fillRect(x * sx, y * sy, sx, sy);
|
|
|
|
pixels[x + y * width] = color;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.plot = function(x, y) {
|
|
|
|
context.fillStyle = COLORS[color];
|
|
|
|
drawPixel(x, y);
|
|
|
|
|
|
|
|
last_x = x;
|
|
|
|
last_y = y;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
this.plot_to = function(x, y) {
|
|
|
|
var x0 = last_x, y0 = last_y, x1 = x, y1 = y,
|
|
|
|
dx = Math.abs(x1 - x0),
|
|
|
|
dy = Math.abs(y1 - y0),
|
|
|
|
sx = (x0 < x1) ? 1 : -1,
|
|
|
|
sy = (y0 < y1) ? 1 : -1,
|
|
|
|
err = dx - dy,
|
|
|
|
e2;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
this.plot(x0, y0);
|
|
|
|
|
|
|
|
if (x0 === x1 && y0 === y1) { return; }
|
|
|
|
e2 = 2 * err;
|
|
|
|
if (e2 > -dy) {
|
|
|
|
err -= dy;
|
|
|
|
x0 += sx;
|
|
|
|
}
|
|
|
|
if (e2 < dx) {
|
|
|
|
err += dx;
|
|
|
|
y0 += sy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
last_x = x;
|
|
|
|
last_y = y;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
this.getPixel = function(x, y) {
|
|
|
|
/*jslint bitwise:false*/
|
|
|
|
return pixels[y * width + x] >>> 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
this.getScreenSize = function() {
|
|
|
|
return { width: width, height: height };
|
|
|
|
};
|
|
|
|
|
|
|
|
this.show = function(state) {
|
|
|
|
element.style.visibility = state ? "visible" : "hidden";
|
|
|
|
};
|
|
|
|
}
|