297 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			297 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
<!doctype html>
 | 
						|
<!-- prototyped this in ~24 hours to help so i don't have to manually enter lists of coordinates for gameboy sprite animations -->
 | 
						|
<html>
 | 
						|
<head>
 | 
						|
	<title>curve editor for gameboy animation</title>
 | 
						|
	<style>
 | 
						|
		body {
 | 
						|
			display: flex;
 | 
						|
			flex-direction: row;
 | 
						|
			gap: 1em;
 | 
						|
		}
 | 
						|
		#controls {
 | 
						|
			display: flex;
 | 
						|
			flex-direction: column;
 | 
						|
			gap: 1em;
 | 
						|
		}
 | 
						|
	</style>
 | 
						|
</head>
 | 
						|
<body>
 | 
						|
	<svg 
 | 
						|
		viewBox="0 0 176 176" 
 | 
						|
		style="border: 1px solid black; display: block; max-height: 700px"
 | 
						|
		xmlns:xlink="http://www.w3.org/1999/xlink">
 | 
						|
		<defs>
 | 
						|
			<g id="point">
 | 
						|
				<rect id="sprite" x="0" y="0" width="8" height="8" stroke="none" fill="gray" fill-opacity="25%"/>
 | 
						|
				<circle cx="0.5" cy="0.5" r="2" fill="black" fill-opacity="25%"/>
 | 
						|
				<circle cx="0.5" cy="0.5" r="0.5" fill="black" fill-opacity="100%"/>
 | 
						|
			</g>
 | 
						|
	    <pattern id="smallGrid" width="1" height="1" patternUnits="userSpaceOnUse">
 | 
						|
	      <path d="M 1 0 L 0 0 0 1" fill="none" stroke="darkseagreen" stroke-width="0.25"/>
 | 
						|
	    </pattern>
 | 
						|
	    <pattern id="grid" width="8" height="8" patternUnits="userSpaceOnUse">
 | 
						|
	      <rect width="8" height="8" fill="url(#smallGrid)"/>
 | 
						|
	      <path d="M 8 0 L 0 0 0 8" fill="none" stroke="darkseagreen" stroke-width="0.5"/>
 | 
						|
	    </pattern>
 | 
						|
	  </defs>
 | 
						|
	      
 | 
						|
	  <image 
 | 
						|
	  	id="background-container"
 | 
						|
	  	width="160"
 | 
						|
	  	height="144"
 | 
						|
	  	x="8" y="16"
 | 
						|
	  	opacity="50%"
 | 
						|
	  	xlink:href="" />
 | 
						|
		<rect width="100%" height="100%" fill="url(#grid)" />
 | 
						|
		<rect x="8" y="16" width="160" height="144" style="fill: none; stroke: darkseagreen; stroke-width: 0.5"/>
 | 
						|
		<g id="lines">
 | 
						|
		</g>
 | 
						|
		<g id="sequence">
 | 
						|
		</g>
 | 
						|
	</svg>
 | 
						|
	<div id="controls">
 | 
						|
		<label for="background-selector">file for background</label>
 | 
						|
		<input type="file" id="background-selector" name="background-selector"/>
 | 
						|
		<fieldset>
 | 
						|
			<legend>control mode</legend>
 | 
						|
			<div>
 | 
						|
				<input type="radio" name="mode" id="add" value="add" checked/>
 | 
						|
				<label for="add"><strong>a</strong>dd points</label>
 | 
						|
			</div>
 | 
						|
			<div>
 | 
						|
				<input type="radio" name="mode" id="delete" value="delete"/>
 | 
						|
				<label for="delete"><strong>d</strong>elete points</label>
 | 
						|
			</div>
 | 
						|
			<div>
 | 
						|
				<input type="radio" name="mode" id="move" value="move"/>
 | 
						|
				<label for="move"><strong>g</strong>rab and move points</label>
 | 
						|
			</div>
 | 
						|
		</fieldset>
 | 
						|
		<div>
 | 
						|
			<label for="sprite_width">width in 8x8 sprites:</label>
 | 
						|
			<input type="number" 
 | 
						|
				id="sprite_width" 
 | 
						|
				name="sprite_width" 
 | 
						|
				min="1" max="16" step="1" value="1" 
 | 
						|
				onchange="sprite_dims_change()"/>
 | 
						|
		</div>
 | 
						|
		<div>
 | 
						|
			<label for="sprite_height">height in 8x8 sprites:</label>
 | 
						|
			<input type="number" 
 | 
						|
				id="sprite_height" 
 | 
						|
				name="sprite_height" 
 | 
						|
				min="1" max="16" step="1" value="1"
 | 
						|
				onchange="sprite_dims_change()"/>
 | 
						|
		</div>
 | 
						|
		<p>output is a length-prefixed list of y,x pairs.</p>
 | 
						|
		<textarea id="output">db 0,</textarea>
 | 
						|
	</div>
 | 
						|
</body>
 | 
						|
<script>
 | 
						|
	var svg = null;
 | 
						|
	var svg_group = null;
 | 
						|
	var text_output = null;
 | 
						|
	var moving_point_index = -1;
 | 
						|
	var the_list = [];
 | 
						|
	var current_viewport = {center: {x: 0, y: 0}, radius: {x: 176, y: 176}};
 | 
						|
	var max_viewport = {center: {x: 0, y: 0}, radius: {x: 176, y:176}};
 | 
						|
 | 
						|
	function length(dx, dy) {return Math.sqrt(dx*dx + dy*dy);}
 | 
						|
 | 
						|
	const clicksvg = (event) => {
 | 
						|
		const mode = document.querySelector('input[name=mode]:checked').value; // either 'add', 'delete', or 'move'
 | 
						|
		const {x, y} = dom_to_svg_coords(svg, event.clientX, event.clientY);
 | 
						|
 | 
						|
		if (mode == "add") { 
 | 
						|
			the_list.push({x: Number((x-0.5).toFixed()),y: Number((y-0.5).toFixed())});
 | 
						|
			console.log(Number(x.toFixed()))
 | 
						|
			regenerate_svg_list();
 | 
						|
			regenerate_text_list();
 | 
						|
		}
 | 
						|
 | 
						|
		if (mode == "delete") {
 | 
						|
			let closest = -1;
 | 
						|
			let distance = 3;
 | 
						|
			the_list.forEach((point, index) => {
 | 
						|
				let new_distance = length(point.x-x, point.y-y);
 | 
						|
				if (new_distance < distance) {
 | 
						|
					distance = new_distance;
 | 
						|
					closest = index;
 | 
						|
				}
 | 
						|
			});
 | 
						|
 | 
						|
			if (closest != -1) {
 | 
						|
				the_list.splice(closest, 1);
 | 
						|
			}
 | 
						|
 | 
						|
			regenerate_svg_list();
 | 
						|
			regenerate_text_list();
 | 
						|
		}
 | 
						|
 | 
						|
		if (mode == "move" && moving_point_index != -1) {
 | 
						|
			const elem = document.querySelector(`g#sequence use:nth-child(${moving_point_index+1})`);
 | 
						|
 | 
						|
			let newX = Number(Number(elem.getAttributeNS(null, "x")).toFixed());
 | 
						|
			let newY = Number(Number(elem.getAttributeNS(null, "y")).toFixed());
 | 
						|
			the_list[moving_point_index] = {x: newX, y: newY};
 | 
						|
			moving_point_index = -1;
 | 
						|
			
 | 
						|
			regenerate_svg_list();
 | 
						|
			regenerate_text_list();
 | 
						|
		}
 | 
						|
	}
 | 
						|
	const mousemovesvg = (event) => {
 | 
						|
		const mode = document.querySelector('input[name=mode]:checked').value; // either 'add', 'delete', or 'move'
 | 
						|
		const {x, y} = dom_to_svg_coords(svg, event.clientX, event.clientY);
 | 
						|
 | 
						|
		if (mode == 'move' && moving_point_index != -1) {
 | 
						|
			const previous_mouse_location = {x: event.clientX - event.movementX, y: event.clientY - event.movementY};
 | 
						|
			const prev_location_in_gbspace = dom_to_svg_coords(svg, previous_mouse_location.x, previous_mouse_location.y);
 | 
						|
			const gb_movement = {x: x - prev_location_in_gbspace.x, y: y - prev_location_in_gbspace.y};
 | 
						|
 | 
						|
			const point = the_list[moving_point_index];
 | 
						|
			const elem = document.querySelector(`g#sequence use:nth-child(${moving_point_index+1})`);
 | 
						|
 | 
						|
			let newX = Number(elem.getAttributeNS(null, "x")) + gb_movement.x;
 | 
						|
			let newY = Number(elem.getAttributeNS(null, 'y')) + gb_movement.y;
 | 
						|
			elem.setAttributeNS(null, "x", newX);
 | 
						|
			elem.setAttributeNS(null, "y", newY);
 | 
						|
 | 
						|
			const lineTo = document.querySelector(`g#lines line:nth-child(${moving_point_index})`);
 | 
						|
			if (lineTo) {
 | 
						|
				lineTo.setAttributeNS(null, "x2", newX+0.5);
 | 
						|
				lineTo.setAttributeNS(null, "y2", newY+0.5);
 | 
						|
			}
 | 
						|
 | 
						|
			const lineFrom = document.querySelector(`g#lines line:nth-child(${moving_point_index+1})`);
 | 
						|
			if (lineFrom) {
 | 
						|
				lineFrom.setAttributeNS(null, "x1", newX+0.5);
 | 
						|
				lineFrom.setAttributeNS(null, "y1", newY+0.5);
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	const mousedownsvg = (event) => { 
 | 
						|
		const mode = document.querySelector('input[name=mode]:checked').value; // either 'add', 'delete', or 'move'
 | 
						|
		const {x, y} = dom_to_svg_coords(svg, event.clientX, event.clientY);
 | 
						|
		let closest = -1;
 | 
						|
		let distance = 3;
 | 
						|
		the_list.forEach((point, index) => {
 | 
						|
			let new_distance = length(point.x-x, point.y-y);
 | 
						|
			if (new_distance < distance) {
 | 
						|
				distance = new_distance;
 | 
						|
				closest = index;
 | 
						|
			}
 | 
						|
		});
 | 
						|
 | 
						|
		if (mode == 'move') {
 | 
						|
			moving_point_index = closest;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	const changetextarea = (event) => {
 | 
						|
		let raw = event.target.value;
 | 
						|
		let stripped = raw.replace(/db [0-9]+,/, "");
 | 
						|
		let pairs = stripped.matchAll(/ ?[0-9]+, ?[0-9]+/g);
 | 
						|
		let splits = pairs.map((x) => x[0].split(/,/));
 | 
						|
		let points = splits.map(([a, b]) => { return {y: Number(a), x: Number(b)}; } );
 | 
						|
 | 
						|
		the_list = [... points];
 | 
						|
 | 
						|
		regenerate_svg_list();
 | 
						|
		regenerate_text_list();
 | 
						|
	}
 | 
						|
 | 
						|
	const sprite_dims_change = (event) => {
 | 
						|
		const width = document.querySelector('input#sprite_width').value;
 | 
						|
		const height = document.querySelector('input#sprite_height').value;
 | 
						|
 | 
						|
		const rect = document.querySelector('g#point rect#sprite')
 | 
						|
		rect.setAttributeNS(null, "width", width*8);
 | 
						|
		rect.setAttributeNS(null, "height", height*8);
 | 
						|
	}
 | 
						|
 | 
						|
	
 | 
						|
 | 
						|
	function regenerate_svg_list() {
 | 
						|
		svg_group.textContent = '';
 | 
						|
		for (const {x, y} of the_list) {
 | 
						|
			const newcircle = document.createElementNS("http://www.w3.org/2000/svg", "use");
 | 
						|
			newcircle.setAttributeNS(null, "href", "#point");
 | 
						|
			newcircle.setAttributeNS(null, "x", x);
 | 
						|
			newcircle.setAttributeNS(null, "y", y);
 | 
						|
			svg_group.appendChild(newcircle);
 | 
						|
		}
 | 
						|
		document.querySelector('g#lines').textContent = '';
 | 
						|
		for (const [second, first] of the_list.slice(1, the_list.length).map((e,i) => [e, the_list[i]])) {
 | 
						|
			const newline = document.createElementNS("http://www.w3.org/2000/svg", "line");
 | 
						|
			newline.setAttributeNS(null, "x1", first.x+0.5);
 | 
						|
			newline.setAttributeNS(null, "y1", first.y+0.5);
 | 
						|
			newline.setAttributeNS(null, "x2", second.x+0.5);
 | 
						|
			newline.setAttributeNS(null, "y2", second.y+0.5);
 | 
						|
			newline.setAttributeNS(null, "stroke", "black");
 | 
						|
			newline.setAttributeNS(null, "stroke-width", "0.25");
 | 
						|
			newline.setAttributeNS(null, "stroke-opacity", "50%");
 | 
						|
			document.querySelector('g#lines').appendChild(newline);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	function regenerate_text_list() {
 | 
						|
		s = `db ${the_list.length}, `;
 | 
						|
		for (const {x, y} of the_list) {
 | 
						|
			s += `${y}, ${x}, `;
 | 
						|
		}
 | 
						|
		text_output.value = s;
 | 
						|
	}
 | 
						|
 | 
						|
	function dom_to_svg_coords(s, clientX, clientY) {
 | 
						|
		// thanks to https://www.sitepoint.com/how-to-translate-from-dom-to-svg-coordinates-and-back-again/
 | 
						|
		const pt = s.createSVGPoint();
 | 
						|
		pt.x = clientX;
 | 
						|
		pt.y = clientY;
 | 
						|
		const svgP = pt.matrixTransform( s.getScreenCTM().inverse() );
 | 
						|
		return svgP;
 | 
						|
	}
 | 
						|
 | 
						|
	window.onload = () => {
 | 
						|
		svg = document.querySelector('svg');
 | 
						|
		svg.onclick = clicksvg;
 | 
						|
		svg.onmousemove = mousemovesvg;
 | 
						|
		svg.onmousedown = mousedownsvg;
 | 
						|
		svg_group = document.querySelector('g#sequence');
 | 
						|
		text_output = document.querySelector('textarea#output');
 | 
						|
		text_output.onchange = changetextarea;
 | 
						|
		sprite_dims_change();
 | 
						|
 | 
						|
		let filePicker = document.querySelector('input#background-selector');
 | 
						|
		filePicker.onchange = (event) => {
 | 
						|
			var file = event.target.files[0];
 | 
						|
 | 
						|
			// setting up the reader
 | 
						|
		  var reader = new FileReader();
 | 
						|
		  reader.readAsDataURL(file,'UTF-8');
 | 
						|
 | 
						|
		  // here we tell the reader what to do when it's done reading...
 | 
						|
		  reader.onload = readerEvent => {
 | 
						|
		    var content = readerEvent.target.result; // this is the content!
 | 
						|
		    document
 | 
						|
		    	.querySelector('image#background-container')
 | 
						|
		    	.setAttributeNS("http://www.w3.org/1999/xlink", "href", content);
 | 
						|
		  }
 | 
						|
		}
 | 
						|
		if (filePicker.files[0]) {
 | 
						|
			filePicker.onchange( {target: filePicker} );
 | 
						|
		}
 | 
						|
 | 
						|
		document.onkeyup = (event) => {
 | 
						|
			const add_radio = document.querySelector('input[name=mode]#add'); // either 'add', 'delete', or 'move'
 | 
						|
			const delete_radio = document.querySelector('input[name=mode]#delete');
 | 
						|
			const move_radio = document.querySelector('input[name=mode]#move');
 | 
						|
 | 
						|
			if (event.key == 'a') { add_radio.checked = true; }
 | 
						|
			if (event.key == 'd') { delete_radio.checked = true; }
 | 
						|
			if (event.key == 'g') { move_radio.checked = true; }
 | 
						|
		}
 | 
						|
 | 
						|
 | 
						|
	}
 | 
						|
</script>
 | 
						|
</html> |