index — krinitsin.com @ a214ca4f86169ca8cf4f3eba12862c6688cfd644

personal website

games/wick-dodge/index.html (view raw)

   1<html><head>
   2<title>PICO-8 Cartridge</title>
   3<meta name="viewport" content="width=device-width, user-scalable=no">
   4<script type="text/javascript">
   5
   6	// Default shell for PICO-8 0.2.6 (includes @weeble's gamepad mod 1.0)
   7	
   8	// options
   9
  10	// fullscreen, sound, close button at top when playing on touchscreen
  11	var p8_allow_mobile_menu = true;
  12
  13	// p8_autoplay true to boot the cartridge automatically after page load when possible
  14	// if the browser can not create an audio context outside of a user gesture (e.g. on iOS), p8_autoplay has no effect
  15	var p8_autoplay = false;
  16
  17	// When pico8_state is defined, PICO-8 will set .is_paused, .sound_volume and .frame_number each frame 
  18	// (used for determining button icons)
  19	var pico8_state = [];
  20
  21	// When pico8_buttons is defined, PICO-8 reads each int as a bitfield holding that player's button states
  22	// 0x1 left, 0x2 right, 0x4 up, 0x8 right, 0x10 O, 0x20 X, 0x40 menu
  23	// (used by p8_update_gamepads)
  24	var pico8_buttons = [0, 0, 0, 0, 0, 0, 0, 0]; // max 8 players
  25
  26	// When pico8_mouse is defined, PICO-8 reads the 3 integers as X, Y and a bitfield for buttons: 0x1 LMB, 0x2 RMB
  27	var pico8_mouse = [];
  28
  29	// used to display number of detected joysticks
  30	var pico8_gamepads = {};
  31	pico8_gamepads.count = 0;
  32
  33	// When pico8_gpio is defined, reading and writing to gpio pins will read and write to these values
  34	var pico8_gpio = new Array(128);
  35
  36	// When pico8_audio_context context is defined, the html shell (this file) is responsible for creating and managing it.
  37	// This makes satisfying browser requirements easier -- e.g. initialising audio from a short script in response to a user action.
  38	// Otherwise PICO-8 will try to create and use its own context.
  39
  40	var pico8_audio_context;
  41
  42
  43	// menu button and controller graphics
  44	p8_gfx_dat={
  45			"p8b_pause1": "",
  46"p8b_controls":"",
  47"p8b_full":"",
  48"p8b_pause0":"",
  49"p8b_sound0":"",
  50"p8b_sound1":"",
  51"p8b_close":"",
  52
  53"controls_left_panel":"",
  54
  55
  56"controls_right_panel":"",
  57
  58	};
  59
  60
  61	// added 0.2.1: work-around for iOS/Safari running from an iFrame (e.g. from itch.io page):
  62	// touch events only register after adding dummy listeners on document.
  63
  64	document.addEventListener('touchstart', {});
  65	document.addEventListener('touchmove', {});
  66	document.addEventListener('touchend', {});
  67
  68
  69	// --------------------------------------------------------------------------------------------------------------------------------
  70	// pico-8 0.2.2: allow dropping files
  71	var p8_dropped_cart = null;
  72	var p8_dropped_cart_name = "";
  73	function p8_drop_file(e)
  74	{
  75		// console.log("@@ dropping file...");
  76	
  77		e.stopPropagation();
  78		e.preventDefault();
  79
  80		if (e.dataTransfer && e.dataTransfer.files && e.dataTransfer.files[0])
  81		{
  82			// read from file
  83			reader = new FileReader();
  84			reader.onload = function (event)
  85			{
  86				p8_dropped_cart_name = 'untitled.p8';
  87				if (typeof e.dataTransfer.files[0].name !== 'undefined') p8_dropped_cart_name = e.dataTransfer.files[0].name;
  88				if (typeof e.dataTransfer.files[0].fileName !== 'undefined') p8_dropped_cart_name = e.dataTransfer.files[0].fileName;
  89				p8_dropped_cart = reader.result;
  90				// data:image/png;base64
  91				e.stopPropagation();
  92				e.preventDefault();
  93				codo_command = 9; // read directly from p8_dropped_cart with libb64 decoder
  94			};
  95			reader.readAsDataURL(e.dataTransfer.files[0]);
  96			
  97		}
  98		else
  99		{
 100			// read from url (or data url)
 101			txt = e.dataTransfer.getData('Text');
 102			if (txt){
 103				p8_dropped_cart_name = "untitled.p8.png";
 104				p8_dropped_cart = txt;
 105				codo_command = 9;
 106			}
 107		}
 108	}
 109	function nop(evt) {
 110		evt.stopPropagation();
 111		evt.preventDefault();
 112	}
 113	function dragover(evt) {
 114		evt.stopPropagation();
 115		evt.preventDefault();
 116		Module.pico8DragOver();
 117	}
 118	function dragstop(evt) {
 119		evt.stopPropagation();
 120		evt.preventDefault();
 121		Module.pico8DragStop();
 122	}
 123
 124	// download (pico-8 0.2.4d web exports can save a .wav file)
 125	function download_browser_file(filename, contents)
 126	{
 127	  var element = document.createElement('a');
 128	  if (filename.substr(filename.length - 7) == ".p8.png")
 129	  	element.setAttribute('href', 'data:image/png;base64,' + encodeURIComponent(contents));
 130	  else if (filename.substr(filename.length - 4) == ".wav")
 131	  	element.setAttribute('href', 'data:audio/x-wav;base64,' + encodeURIComponent(contents));
 132	  else
 133	  	element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(contents));
 134	  element.setAttribute('download', filename);
 135	  element.style.display = 'none';
 136	  document.body.appendChild(element);
 137	  element.click();
 138	  document.body.removeChild(element);
 139	}
 140	// --------------------------------------------------------------------------------------------------------------------------------
 141
 142
 143	var p8_buttons_hash = -1;
 144	function p8_update_button_icons()
 145	{
 146		// buttons only appear when running
 147		if (!p8_is_running)
 148		{
 149			requestAnimationFrame(p8_update_button_icons);
 150			return;
 151		}
 152		var is_fullscreen=(document.fullscreenElement || document.mozFullScreenElement || document.webkitIsFullScreen || document.msFullscreenElement);
 153		
 154		// hash based on: pico8_state.sound_volume  pico8_state.is_paused bottom_margin left is_fullscreen p8_touch_detected
 155		var hash = 0;
 156		hash = pico8_state.sound_volume;
 157		if (pico8_state.is_paused) hash += 0x100;
 158		if (p8_touch_detected)     hash += 0x200;
 159		if (is_fullscreen)         hash += 0x400;
 160
 161		if (p8_buttons_hash == hash)
 162		{
 163			requestAnimationFrame(p8_update_button_icons);
 164			return;
 165		}
 166		
 167		p8_buttons_hash = hash;
 168		// console.log("@@ updating button icons");
 169
 170		els = document.getElementsByClassName('p8_menu_button');
 171		for (i = 0; i < els.length; i++)
 172		{
 173			el = els[i];
 174			index = el.id;			
 175			if (index == 'p8b_sound') index += (pico8_state.sound_volume == 0 ? "0" : "1"); // 1 if undefined
 176			if (index == 'p8b_pause') index += (pico8_state.is_paused > 0 ? "1" : "0");     // 0 if undefined
 177
 178			new_str = '<img width=24 height=24 style="pointer-events:none" src="'+p8_gfx_dat[index]+'">';
 179			if (el.innerHTML != new_str)
 180				el.innerHTML = new_str;
 181
 182
 183
 184
 185			// hide all buttons for touch mode (can pause with menu buttons)
 186			
 187			var is_visible = p8_is_running;
 188
 189			if ((!p8_touch_detected || !p8_allow_mobile_menu) && el.parentElement.id == "p8_menu_buttons_touch")
 190				is_visible = false;
 191
 192			if (p8_touch_detected && el.parentElement.id == "p8_menu_buttons")
 193				is_visible = false;
 194
 195			if (is_fullscreen) 
 196				is_visible = false;
 197
 198			if (is_visible)
 199				el.style.display="";
 200			else
 201				el.style.display="none";
 202		}
 203		requestAnimationFrame(p8_update_button_icons);
 204	}
 205
 206
 207
 208	function abs(x)
 209	{
 210		return x < 0 ? -x : x;
 211	}
 212	
 213	// step 0 down 1 drag 2 up (not used)
 214	function pico8_buttons_event(e, step)
 215	{
 216		if (!p8_is_running) return;
 217	
 218		pico8_buttons[0] = 0;
 219
 220		if (step == 2 && typeof(pico8_mouse) !== 'undefined')
 221		{
 222			pico8_mouse[2] = 0;
 223		}
 224		
 225		var num = 0;
 226		if (e.touches) num = e.touches.length;
 227
 228		if (num == 0 && typeof(pico8_mouse) !== 'undefined')
 229		{
 230			//  no active touches: release mouse button from anywhere on page. (maybe redundant? but just in case)
 231			pico8_mouse[2] = 0;
 232		}
 233
 234		
 235		for (var i = 0; i < num; i++)
 236		{
 237			var touch = e.touches[i];
 238			var x = touch.clientX;
 239			var y = touch.clientY;
 240			var w = window.innerWidth;
 241			var h = window.innerHeight;
 242
 243			var r = Math.min(w,h) / 12;
 244			if (r > 40) r = 40;
 245
 246			// mouse (0.1.12d)
 247
 248			let canvas = document.getElementById("canvas");
 249			if (p8_touch_detected)
 250			if (typeof(pico8_mouse) !== 'undefined')
 251			if (canvas)
 252			{
 253				var rect = canvas.getBoundingClientRect();
 254				//console.log(rect.top, rect.right, rect.bottom, rect.left, x, y);
 255
 256				if (x >= rect.left && x < rect.right && y >= rect.top && y < rect.bottom)
 257				{
 258					pico8_mouse = [
 259						Math.floor((x - rect.left) * 128 / (rect.right - rect.left)),
 260						Math.floor((y - rect.top) * 128 / (rect.bottom - rect.top)),
 261						step < 2 ? 1 : 0
 262					];
 263					// return; // commented -- blocks overlapping buttons
 264				}else
 265				{
 266					pico8_mouse[2] = 0;
 267				}
 268			}
 269
 270
 271			// buttons
 272			
 273			b = 0;
 274
 275			if (y < h - r*8)
 276			{
 277				// no controller buttons up here; includes canvas and menu buttons at top in touch mode
 278			}
 279			else
 280			{
 281				e.preventDefault();
 282
 283				if ((y < h - r*6) && y > (h - r*8))
 284				{
 285					// menu button: half as high as X O button
 286					// stretch across right-hand half above X O buttons
 287					if (x > w - r*3) 
 288						b |= 0x40;
 289				}
 290				else if (x < w/2 && x < r*6)
 291				{
 292					// stick
 293
 294					mask = 0xf; // dpad
 295					var cx = 0 + r*3;
 296					var cy = h - r*3;
 297
 298					deadzone = r/3;
 299					var dx = x - cx;
 300					var dy = y - cy;
 301
 302					if (abs(dx) > abs(dy) * 0.6) // horizontal 
 303					{
 304						if (dx < -deadzone) b |= 0x1;
 305						if (dx > deadzone) b |= 0x2;
 306					}
 307					if (abs(dy) > abs(dx) * 0.6) // vertical
 308					{
 309						if (dy < -deadzone) b |= 0x4;
 310						if (dy > deadzone) b |= 0x8;
 311					}
 312				}
 313				else if (x > w - r*6)
 314				{
 315					// button; diagonal split from bottom right corner
 316				
 317					mask = 0x30;
 318					
 319					// one or both of [X], [O]
 320					if ( (h-y) > (w-x) * 0.8) b |= 0x10;
 321					if ( (w-x) > (h-y) * 0.8) b |= 0x20;
 322				}
 323			}
 324
 325			pico8_buttons[0] |= b;
 326		
 327		}
 328	}
 329
 330	// p8_update_layout_hash is used to decide when to update layout (expensive especially when part of a heavy page)
 331	var p8_update_layout_hash = -1;
 332	var last_windowed_container_height = 512;
 333	var p8_layout_frames = 0;
 334
 335	function p8_update_layout()
 336	{
 337		var canvas = document.getElementById("canvas");
 338		var p8_playarea = document.getElementById("p8_playarea");
 339		var p8_container = document.getElementById("p8_container");
 340		var p8_frame = document.getElementById("p8_frame");
 341		var csize = 512;
 342		var margin_top = 0;
 343		var margin_left = 0;
 344
 345		// page didn't load yet? first call should be after p8_frame is created so that layout doesn't jump around.
 346		if (!canvas || !p8_playarea || !p8_container || !p8_frame)
 347		{
 348			p8_update_layout_hash = -1;
 349			requestAnimationFrame(p8_update_layout);
 350			return;
 351		}
 352
 353		p8_layout_frames ++;
 354
 355		// assumes frame doesn't have padding
 356		
 357		var is_fullscreen=(document.fullscreenElement || document.mozFullScreenElement || document.webkitIsFullScreen || document.msFullscreenElement);
 358		var frame_width = p8_frame.offsetWidth;
 359		var frame_height = p8_frame.offsetHeight;
 360
 361		if (is_fullscreen)
 362		{
 363			// same as window
 364			frame_width = window.innerWidth;
 365			frame_height = window.innerHeight;
 366		}
 367		else{
 368			// never larger than window  // (happens when address bar is down in portraight mode on phone)
 369			frame_width  = Math.min(frame_width, window.innerWidth);
 370			frame_height = Math.min(frame_height, window.innerHeight);
 371		}
 372
 373		// as big as will fit in a frame..
 374		csize =  Math.min(frame_width,frame_height);
 375
 376		// .. but never more than 2/3 of longest side for touch (e.g. leave space for controls on iPad)
 377		if (p8_touch_detected && p8_is_running)
 378		{
 379			var longest_side = Math.max(window.innerWidth,window.innerHeight);
 380			csize = Math.min(csize, longest_side * 2/3);
 381		}
 382
 383		// pixel perfect: quantize to closest multiple of 128
 384		// only when large display (desktop)
 385		if (frame_width >= 512 && frame_height >= 512)
 386		{
 387			csize = (csize+1) & ~0x7f;
 388		}
 389
 390		// csize should never be higher than parent frame
 391		// (otherwise stretched large when fullscreen and then return)
 392		if (!is_fullscreen && p8_frame) 
 393			csize = Math.min(csize, last_windowed_container_height); // p8_frame_0 parent
 394		
 395
 396		if (is_fullscreen)
 397		{
 398			// always center horizontally
 399			margin_left = (frame_width - csize)/2;
 400
 401			if (p8_touch_detected)
 402			{
 403				if (window.innerWidth < window.innerHeight)
 404				{
 405					// portrait: keep at y=40 (avoid rounded top corners / camera nub thing etc.)
 406					margin_top = Math.min(40, frame_height - csize);
 407				}
 408				else
 409				{
 410					// landscape: put a little above vertical center
 411					margin_top = (frame_height - csize)/4;
 412				}
 413			}
 414			else{
 415				// non-touch: center vertically
 416				margin_top = (frame_height - csize)/2;
 417			}
 418		}
 419
 420		// skip if relevant state has not changed
 421
 422		var update_hash = csize + margin_top * 1000.3 + margin_left * 0.001 + frame_width * 333.33 + frame_height * 772.15134;
 423		if (is_fullscreen) update_hash += 0.1237;
 424
 425		// unexpected things can happen in the first few seconds, so just keep re-calculating layout. wasm version breaks layout otherwise.
 426		// also: bonus refresh at 5, 8 seconds just in case ._.
 427		if (p8_layout_frames < 180 || p8_layout_frames == 60*5 || p8_layout_frames == 60*8 )
 428			update_hash = p8_layout_frames;
 429
 430		if (!is_fullscreen) // fullscreen: update every frame for safety. should be cheap!
 431		if (!p8_touch_detected) // mobile: update every frame because nothing can be trusted
 432		if (p8_update_layout_hash == update_hash)
 433		{
 434			//console.log("p8_update_layout(): skipping");
 435			requestAnimationFrame(p8_update_layout);
 436			return;
 437		}
 438		p8_update_layout_hash = update_hash;
 439
 440		// record this for returning to original size after fullscreen pushes out container height (argh)
 441		if (!is_fullscreen && p8_frame)
 442			last_windowed_container_height = p8_frame.parentNode.parentNode.offsetHeight;
 443		
 444
 445		// mobile in portrait mode: put screen at top (w / a little extra space for fullscreen button if needed)
 446		// (don't cart too about buttons overlapping screen)
 447		if (p8_touch_detected && p8_is_running && document.body.clientWidth < document.body.clientHeight)
 448			p8_playarea.style.marginTop = p8_allow_mobile_menu ? 32 : 8;
 449		else if (p8_touch_detected && p8_is_running) // landscape: slightly above vertical center (only relevant for iPad / highres devices)
 450			p8_playarea.style.marginTop = (document.body.clientHeight - csize) / 4;
 451		else
 452			p8_playarea.style.marginTop = "";
 453
 454		canvas.style.width = csize;
 455		canvas.style.height = csize;
 456
 457		// to do: this should just happen from css layout
 458		canvas.style.marginLeft = margin_left;
 459		canvas.style.marginTop = margin_top;
 460
 461		p8_container.style.width = csize;
 462		p8_container.style.height = csize;
 463
 464		// set menu buttons position to bottom right
 465		el = document.getElementById("p8_menu_buttons");
 466		el.style.marginTop = csize - el.offsetHeight;
 467
 468		if (p8_touch_detected && p8_is_running)
 469		{
 470			// turn off pointer events to prevent double-tap zoom etc (works on Android)
 471			// don't want this for desktop because breaks mouse input & click-to-focus when using codo_textarea
 472			canvas.style.pointerEvents = "none";
 473
 474			p8_container.style.marginTop = "0px";
 475
 476			// buttons
 477			
 478			// same as touch event handling
 479			var w = window.innerWidth;
 480			var h = window.innerHeight;
 481			var r = Math.min(w,h) / 12;
 482
 483			if (r > 40) r = 40;
 484
 485			el = document.getElementById("controls_right_panel");
 486			el.style.left = w-r*6;
 487			el.style.top = h-r*7;
 488			el.style.width = r*6;
 489			el.style.height = r*7;
 490			if (el.getAttribute("src") != p8_gfx_dat["controls_right_panel"]) // optimisation: avoid reload? (browser should handle though)
 491				el.setAttribute("src", p8_gfx_dat["controls_right_panel"]);
 492
 493			el = document.getElementById("controls_left_panel");
 494			el.style.left = 0;
 495			el.style.top = h-r*6;
 496			el.style.width = r*6;
 497			el.style.height = r*6;
 498			if (el.getAttribute("src") != p8_gfx_dat["controls_left_panel"]) // optimisation: avoid reload? (browser should handle though)
 499				el.setAttribute("src", p8_gfx_dat["controls_left_panel"]);
 500			
 501			// scroll to cart (commented; was a failed attempt to prevent scroll-on-drag on some browsers)
 502			// p8_frame.scrollIntoView(true);
 503
 504			document.getElementById("touch_controls_gfx").style.display="table";
 505			document.getElementById("touch_controls_background").style.display="table";
 506
 507		}
 508		else{
 509			document.getElementById("touch_controls_gfx").style.display="none";
 510			document.getElementById("touch_controls_background").style.display="none";
 511		}
 512
 513		if (!p8_is_running)
 514		{
 515			p8_playarea.style.display="none";
 516			p8_container.style.display="flex";
 517			p8_container.style.marginTop="auto";
 518
 519			el = document.getElementById("p8_start_button");
 520			if (el) el.style.display="flex";
 521		}
 522		requestAnimationFrame(p8_update_layout);
 523	}
 524
 525
 526	var p8_touch_detected = false;
 527	addEventListener("touchstart", function(event)
 528	{
 529		p8_touch_detected = true;
 530
 531		// hide codo_textarea -- clipboard support on mobile is not feasible
 532		el = document.getElementById("codo_textarea");
 533		if (el && el.style.display != "none"){
 534			el.style.display="none";
 535		}
 536
 537	},  {passive: true});
 538
 539	function p8_create_audio_context()
 540	{
 541		if (pico8_audio_context) 
 542		{
 543			try {
 544				pico8_audio_context.resume();
 545			}
 546			catch(err) {
 547				console.log("** pico8_audio_context.resume() failed");
 548			}	
 549			return;
 550		}
 551
 552		var webAudioAPI = window.AudioContext || window.webkitAudioContext || window.mozAudioContext || window.oAudioContext || window.msAudioContext;			
 553		if (webAudioAPI)
 554		{
 555			pico8_audio_context = new webAudioAPI;
 556
 557			// wake up iOS
 558			if (pico8_audio_context)
 559			{
 560				try {
 561					var dummy_source_sfx = pico8_audio_context.createBufferSource();
 562					dummy_source_sfx.buffer = pico8_audio_context.createBuffer(1, 1, 22050); // dummy
 563					dummy_source_sfx.connect(pico8_audio_context.destination);
 564					dummy_source_sfx.start(1, 0.25); // gives InvalidStateError -- why? hasn't been played before 
 565					//dummy_source_sfx.noteOn(0); // deleteme
 566				}
 567				catch(err) {
 568					console.log("** dummy_source_sfx.start(1, 0.25) failed");
 569				}
 570			}
 571		}
 572	}
 573
 574	function p8_close_cart()
 575	{
 576		// just reload page! used for touch buttons -- hard to roll back state 
 577		window.location.hash = ""; // triggers reload
 578	}
 579
 580	var p8_is_running = false;
 581	var p8_script = null;
 582	var Module = null;
 583	function p8_run_cart()
 584	{
 585		if (p8_is_running) return;
 586		p8_is_running = true;
 587
 588		// touch: hide everything except p8_frame_0
 589		if (p8_touch_detected)
 590		{
 591			el = document.getElementById("body_0");
 592			el2 = document.getElementById("p8_frame_0");
 593			if (el && el2)
 594			{
 595				el.style.display="none";
 596				el.parentNode.appendChild(el2);
 597			}
 598		}
 599
 600		// create audio context and wake it up (for iOS -- needs happen inside touch event)		
 601		p8_create_audio_context();
 602
 603		// show touch elements
 604		els = document.getElementsByClassName('p8_controller_area');
 605		for (i = 0; i < els.length; i++)
 606			els[i].style.display="";
 607
 608
 609		// install touch events. These also serve to block scrolling / pinching / zooming on phones when p8_is_running
 610			// moved event.preventDefault(); calls into pico8_buttons_event() (want to let top buttons pass through)
 611		addEventListener("touchstart", function(event){ pico8_buttons_event(event, 0); }, {passive: false});
 612		addEventListener("touchmove",  function(event){ pico8_buttons_event(event, 1); }, {passive: false});
 613		addEventListener("touchend",   function(event){ pico8_buttons_event(event, 2); }, {passive: false});
 614
 615
 616		// load and run script
 617		e = document.createElement("script");
 618		p8_script = e;
 619		e.onload = function(){
 620			
 621			// show canvas / menu buttons only after loading
 622			el = document.getElementById("p8_playarea");
 623			if (el) el.style.display="table";
 624
 625			if (typeof(p8_update_layout_hash) !== 'undefined')
 626				p8_update_layout_hash = -77;
 627			if (typeof(p8_buttons_hash) !== 'undefined')
 628				p8_buttons_hash = -33;
 629
 630
 631		}
 632		e.type = "application/javascript";
 633		e.src = "wick-dodge.js";
 634		e.id = "e_script";
 635		
 636		document.body.appendChild(e); // load and run
 637
 638		// hide start button and show canvas / menu buttons. hide start button
 639		el = document.getElementById("p8_start_button");
 640		if (el) el.style.display="none";
 641
 642		// add #playing for touchscreen devices (allows back button to close)
 643		// X button can also be used to trigger this 
 644		if (p8_touch_detected)
 645		{
 646			window.location.hash = "#playing";
 647			window.onhashchange = function()
 648			{
 649				if (window.location.hash.search("playing") < 0)
 650					window.location.reload();
 651			}
 652		}
 653
 654		// install drag&drop listeners
 655		{
 656			let canvas = document.getElementById("canvas");
 657			if (canvas)
 658			{
 659				canvas.addEventListener('dragenter', dragover, false);
 660				canvas.addEventListener('dragover', dragover, false);
 661				canvas.addEventListener('dragleave', dragstop, false);
 662				canvas.addEventListener('drop', nop, false);
 663				canvas.addEventListener('drop', p8_drop_file, false);
 664			}
 665		}
 666	}
 667
 668	
 669		// Gamepad code
 670	
 671	var P8_BUTTON_O = {action:'button', code: 0x10};
 672	var P8_BUTTON_X = {action:'button', code: 0x20};
 673	var P8_DPAD_LEFT = {action:'button', code: 0x1};
 674	var P8_DPAD_RIGHT = {action:'button', code: 0x2};
 675	var P8_DPAD_UP = {action:'button', code: 0x4};
 676	var P8_DPAD_DOWN = {action:'button', code: 0x8};
 677	var P8_MENU = {action:'menu'};
 678	var P8_NO_ACTION = {action:'none'};
 679
 680	var P8_BUTTON_MAPPING = [
 681		// ref: https://w3c.github.io/gamepad/#remapping
 682		P8_BUTTON_O,          // Bottom face button
 683		P8_BUTTON_X,          // Right face button
 684		P8_BUTTON_X,          // Left face button
 685		P8_BUTTON_O,          // Top face button
 686		P8_NO_ACTION,         // Near left shoulder button (L1)
 687		P8_NO_ACTION,         // Near right shoulder button (R1)
 688		P8_NO_ACTION,         // Far left shoulder button (L2)
 689		P8_NO_ACTION,         // Far right shoulder button (R2)
 690		P8_MENU,              // Left auxiliary button (select)
 691		P8_MENU,              // Right auxiliary button (start)
 692		P8_NO_ACTION,         // Left stick button
 693		P8_NO_ACTION,         // Right stick button
 694		P8_DPAD_UP,           // Dpad up
 695		P8_DPAD_DOWN,         // Dpad down
 696		P8_DPAD_LEFT,         // Dpad left
 697		P8_DPAD_RIGHT,        // Dpad right
 698	];
 699
 700	// Track which player is controller by each gamepad. Gamepad index i controls the
 701	// player with index pico8_gamepads_mapping[i]. Gamepads with null player are
 702	// currently unassigned - they get assigned to a player when a button is pressed.
 703	var pico8_gamepads_mapping = [];
 704
 705	function p8_unassign_gamepad(gamepad_index) {
 706		if (pico8_gamepads_mapping[gamepad_index] == null) {
 707			return;
 708		}
 709		pico8_buttons[pico8_gamepads_mapping[gamepad_index]] = 0;
 710		pico8_gamepads_mapping[gamepad_index] = null;
 711	}
 712
 713
 714	function p8_first_player_without_gamepad(max_players) {
 715		var allocated_players = pico8_gamepads_mapping.filter(function(x) { return x != null; });
 716		var sorted_players = Array.from(allocated_players).sort();
 717		for (var desired = 0; desired < sorted_players.length && desired < max_players; ++desired) {
 718			if (desired != sorted_players[desired]) {
 719				return desired;
 720			}
 721		}
 722		if (sorted_players.length < max_players) {
 723			return sorted_players.length;
 724		}
 725		return null;
 726	}
 727
 728	function p8_assign_gamepad_to_player(gamepad_index, player_index) {
 729		p8_unassign_gamepad(gamepad_index);
 730		pico8_gamepads_mapping[gamepad_index] = player_index;
 731	}
 732
 733
 734
 735	function p8_convert_standard_gamepad_to_button_state(gamepad, axis_threshold, button_threshold) {
 736		// Given a gamepad object, return:
 737		// {
 738		//     button_state: the binary encoded Pico 8 button state
 739		//     menu_button: true if any menu-mapped button was pressed
 740		//     any_button: true if any button was pressed, including d-pad
 741		//         buttons and unmapped buttons
 742		// }
 743		if (!gamepad || !gamepad.axes || !gamepad.buttons) {
 744			return {
 745				button_state: 0,
 746				menu_button: false,
 747				any_button: false
 748			};
 749		}
 750		function button_state_from_axis(axis, low_state, high_state, default_state) {
 751			if (axis && axis < -axis_threshold) return low_state;
 752			if (axis && axis > axis_threshold) return high_state;
 753			return default_state;
 754		}
 755		var axes_actions = [
 756			button_state_from_axis(gamepad.axes[0], P8_DPAD_LEFT, P8_DPAD_RIGHT, P8_NO_ACTION),
 757			button_state_from_axis(gamepad.axes[1], P8_DPAD_UP, P8_DPAD_DOWN, P8_NO_ACTION),
 758		];
 759
 760		var button_actions = gamepad.buttons.map(function (button, index) {
 761			var pressed = button.value > button_threshold || button.pressed;
 762			if (!pressed) return P8_NO_ACTION;
 763			return P8_BUTTON_MAPPING[index] || P8_NO_ACTION;
 764		});
 765
 766		var all_actions = axes_actions.concat(button_actions);
 767		
 768		var menu_button = button_actions.some(function (action) { return action.action == 'menu'; });
 769		var button_state = (all_actions
 770			.filter(function (a) { return a.action == 'button'; })
 771			.map(function (a) { return a.code; })
 772			.reduce(function (result, code) { return result | code; }, 0)
 773		);
 774		var any_button = gamepad.buttons.some(function (button) {
 775			return button.value > button_threshold || button.pressed;
 776		});
 777
 778		any_button |= button_state; //jww: include axes 0,1 as might be first intended action
 779
 780		return {
 781			button_state,
 782			menu_button,
 783			any_button
 784		};
 785	}
 786
 787	// jww: pico-8 0.2.1 version for unmapped gamepads, following p8_convert_standard_gamepad_to_button_state
 788	// axes 0,1 & buttons 0,1,2,3 are reasonably safe. don't try to read dpad.
 789	// menu buttons are unpredictable, but use 6..8 anyway (better to have a weird menu button than none)
 790
 791	function p8_convert_unmapped_gamepad_to_button_state(gamepad, axis_threshold, button_threshold) {
 792
 793		if (!gamepad || !gamepad.axes || !gamepad.buttons) {
 794			return {
 795				button_state: 0,
 796				menu_button: false,
 797				any_button: false
 798			};
 799		}
 800
 801		var button_state = 0;
 802
 803		if (gamepad.axes[0] && gamepad.axes[0] < -axis_threshold) button_state |= 0x1;
 804		if (gamepad.axes[0] && gamepad.axes[0] > axis_threshold)  button_state |= 0x2;
 805		if (gamepad.axes[1] && gamepad.axes[1] < -axis_threshold) button_state |= 0x4;
 806		if (gamepad.axes[1] && gamepad.axes[1] > axis_threshold)  button_state |= 0x8;
 807
 808		// buttons: first 4 taken to be O/X, 6..8 taken to be menu button
 809
 810		for (j = 0; j < gamepad.buttons.length; j++)
 811		if (gamepad.buttons[j].value > 0 || gamepad.buttons[j].pressed)
 812		{
 813			if (j < 4)
 814				button_state |= (0x10 << (((j+1)/2)&1)); // 0 1 1 0 -- A,X -> O,X on xbox360
 815			else if (j >= 6 && j <= 8)
 816				button_state |= 0x40;
 817		}
 818
 819		var menu_button = button_state & 0x40;
 820
 821		var any_button = gamepad.buttons.some(function (button) {
 822			return button.value > button_threshold || button.pressed;
 823		});
 824
 825		any_button |= button_state; //jww: include axes 0,1 as might be first intended action
 826		
 827		return {
 828			button_state,
 829			menu_button,
 830			any_button
 831		};
 832	}
 833
 834	
 835	// gamepad  https://developer.mozilla.org/en-US/docs/Web/API/Gamepad_API/Using_the_Gamepad_API
 836	// (sets bits in pico8_buttons[])
 837	function p8_update_gamepads() {
 838		var axis_threshold = 0.3;
 839		var button_threshold = 0.5; // Should be unnecessary, we should be able to trust .pressed
 840		var max_players = 8;
 841		var gps = navigator.getGamepads() || navigator.webkitGetGamepads();
 842
 843		if (!gps) return;
 844
 845		// In Chrome, gps is iterable but it's not an array.
 846		gps = Array.from(gps);
 847
 848		pico8_gamepads.count = gps.length;
 849		while (gps.length > pico8_gamepads_mapping.length) {
 850		    pico8_gamepads_mapping.push(null);
 851		}
 852
 853		var menu_button = false;
 854		var gamepad_states = gps.map(function (gp) {
 855			return (gp && gp.mapping == "standard") ?
 856				p8_convert_standard_gamepad_to_button_state(gp, axis_threshold, button_threshold) :
 857				p8_convert_unmapped_gamepad_to_button_state(gp, axis_threshold, button_threshold);
 858		});
 859
 860		// Unassign disconnected gamepads.
 861		// gps.forEach(function (gp, i) { if (gp && !gp.connected) { p8_unassign_gamepad(i); }});
 862		gps.forEach(function (gp, i) { if (!gp || !gp.connected) { p8_unassign_gamepad(i); }}); // https://www.lexaloffle.com/bbs/?pid=87132#p
 863
 864
 865		// Assign unassigned gamepads when any button is pressed.
 866		gamepad_states.forEach(function (state, i) {
 867			if (state.any_button && pico8_gamepads_mapping[i] == null) {
 868				var first_free_player = p8_first_player_without_gamepad(max_players);
 869				p8_assign_gamepad_to_player(i, first_free_player);
 870			}
 871		});
 872
 873		// Update pico8_buttons array.
 874		gamepad_states.forEach(function (gamepad_state, i) {
 875			if (pico8_gamepads_mapping[i] != null) {
 876				pico8_buttons[pico8_gamepads_mapping[i]] = gamepad_state.button_state;
 877			}
 878		});
 879
 880		// Update menu button.
 881		// Pico 8 only recognises the menu button on the first player, so we
 882		// press it when any gamepad has pressed a button mapped to menu.
 883		if (gamepad_states.some(function (state) { return state.menu_button; })) {
 884			pico8_buttons[0] |= 0x40;
 885		}
 886
 887		requestAnimationFrame(p8_update_gamepads);
 888	}
 889	requestAnimationFrame(p8_update_gamepads);
 890
 891	// End of gamepad code
 892
 893
 894	// key blocker. prevent browser operations while playing cart so that PICO-8 can use those keys e.g. cursors to scroll, ctrl-r to reload
 895	document.addEventListener('keydown',
 896	function (event) {
 897		event = event || window.event;
 898		if (!p8_is_running) return;
 899
 900		if (pico8_state.has_focus == 1)
 901			if ([32, 37, 38, 39, 40, 77, 82, 80, 9].indexOf(event.keyCode) > -1)       // block only cursors, M R P, tab
 902				if (event.preventDefault) event.preventDefault();
 903
 904	},{passive: false});
 905
 906	// when using codo_textarea to determine focus, need to explicitly hand focus back when clicking a p8_menu_button
 907	function p8_give_focus()
 908	{
 909		el = (typeof codo_textarea === 'undefined') ? document.getElementById("codo_textarea") : codo_textarea;
 910		if (el)
 911		{
 912			el.focus();
 913			el.select();
 914		}
 915	}
 916
 917	function p8_request_fullscreen() {
 918
 919		var is_fullscreen=(document.fullscreenElement || document.mozFullScreenElement || document.webkitIsFullScreen || document.msFullscreenElement);
 920
 921		if (is_fullscreen)
 922		{
 923			 if (document.exitFullscreen) {
 924		        document.exitFullscreen();
 925		    } else if (document.webkitExitFullscreen) {
 926		        document.webkitExitFullscreen();
 927		    } else if (document.mozCancelFullScreen) {
 928		        document.mozCancelFullScreen();
 929		    } else if (document.msExitFullscreen) {
 930		        document.msExitFullscreen();
 931		    }
 932			return;
 933		}
 934		
 935		var el = document.getElementById("p8_playarea");
 936
 937		if ( el.requestFullscreen ) {
 938			el.requestFullscreen();
 939		} else if ( el.mozRequestFullScreen ) {
 940			el.mozRequestFullScreen();
 941		} else if ( el.webkitRequestFullScreen ) {
 942			el.webkitRequestFullScreen( Element.ALLOW_KEYBOARD_INPUT );
 943		}
 944	}
 945
 946</script>
 947
 948<STYLE TYPE="text/css">
 949<!--
 950.p8_menu_button{
 951	opacity:0.3;
 952	padding:4px;
 953	display:table;
 954	width:24px;
 955	height:24px;
 956	float:right;
 957}
 958
 959@media screen and (min-width:512px) {
 960	.p8_menu_button{
 961		width:24px; margin-left:12px; margin-bottom:8px;
 962	}
 963}
 964.p8_menu_button:hover{
 965	opacity:1.0;
 966	cursor:pointer;
 967}
 968
 969canvas{
 970    image-rendering: optimizeSpeed;
 971    image-rendering: -moz-crisp-edges;
 972    image-rendering: -webkit-optimize-contrast;
 973    image-rendering: optimize-contrast;
 974    image-rendering: pixelated;
 975    -ms-interpolation-mode: nearest-neighbor;
 976	border: 0px;
 977	cursor: none;
 978}
 979
 980
 981.p8_start_button{
 982	cursor:pointer;
 983	background:url("");
 984	-repeat center;
 985	-webkit-background-size:cover; -moz-background-size:cover; -o-background-size:cover; background-size:cover;
 986}
 987
 988.button_gfx{
 989	stroke-width:2;
 990	stroke: #ffffff;
 991	stroke-opacity:0.4;
 992	fill-opacity:0.2;
 993	fill:black;
 994}
 995
 996.button_gfx_icon{
 997	stroke-width:3;
 998	stroke: #909090;
 999	stroke-opacity:0.7;
1000	fill:none;
1001}
1002
1003-->
1004</STYLE>
1005
1006</head>
1007
1008<body style="padding:0px; margin:0px; background-color:#222; color:#ccc">
1009<div id="body_0"> <!-- hide this when playing in mobile (p8_touch_detected) so that elements don't affect layout -->
1010
1011
1012<!-- Add any content above the cart here -->
1013
1014
1015<div id="p8_frame_0" style="max-width:800px; max-height:800px; margin:auto;"> <!-- double function: limit size, and display only this div for touch devices -->
1016<div id="p8_frame" style="display:flex; width:100%; max-width:95vw; height:100vw; max-height:95vh; margin:auto;">
1017
1018	<div id="p8_menu_buttons_touch" style="position:absolute; width:100%; z-index:10; left:0px;">
1019		<div class="p8_menu_button" id="p8b_full"  style="float:left;margin-left:10px" onClick="p8_give_focus(); p8_request_fullscreen();"></div>
1020		<div class="p8_menu_button" id="p8b_sound" style="float:left;margin-left:10px" onClick="p8_give_focus(); p8_create_audio_context(); Module.pico8ToggleSound();"></div>
1021		<div class="p8_menu_button" id="p8b_close" style="float:right; margin-right:10px" onClick="p8_close_cart();"></div>
1022	</div>
1023
1024	<div id="p8_container"
1025		style="margin:auto; display:table;"
1026		onclick="p8_create_audio_context(); p8_run_cart();">
1027
1028		<div id="p8_start_button" class="p8_start_button" style="width:100%; height:100%; display:flex;">
1029			<img width=80 height=80 style="margin:auto;"
1030		src=""/>
1031		</div>
1032
1033		<div id="p8_playarea" style="display:none; margin:auto;
1034				-webkit-user-select:none; -moz-user-select: none; user-select: none; -webkit-touch-callout:none;
1035		">
1036
1037			<div  id="touch_controls_background"
1038				  style=" pointer-events:none; display:none; background-color:#000;
1039						 position:fixed; top:0px; left:0px; border:0; width:100vw; height:100vh">
1040				&nbsp
1041			</div>
1042
1043			<div style="display:flex; position:relative">
1044				<!-- pointer-events turned off for mobile in p8_update_layout because need for desktop mouse -->
1045				<canvas class="emscripten" id="canvas" oncontextmenu="event.preventDefault();" >
1046				</canvas>
1047				<div class=p8_menu_buttons id="p8_menu_buttons" style="margin-left:10px;">
1048					<div class="p8_menu_button" style="position:absolute; bottom:125px" id="p8b_controls" onClick="p8_give_focus(); Module.pico8ToggleControlMenu();"></div>					
1049					<div class="p8_menu_button" style="position:absolute; bottom:90px" id="p8b_pause" onClick="p8_give_focus(); Module.pico8TogglePaused(); p8_update_layout_hash = -22;"></div>
1050					<div class="p8_menu_button" style="position:absolute; bottom:55px" id="p8b_sound" onClick="p8_give_focus(); p8_create_audio_context(); Module.pico8ToggleSound();"></div>
1051					<div class="p8_menu_button" style="position:absolute; bottom:20px" id="p8b_full" onClick="p8_give_focus(); p8_request_fullscreen();"></div>
1052				</div>
1053			</div>
1054
1055
1056			<!-- display after first layout update -->
1057			<div  id="touch_controls_gfx"
1058				  style=" pointer-events:none; display:table; 
1059						 position:fixed; top:0px; left:0px; border:0; width:100vw; height:100vh">
1060
1061					<img src="" id="controls_right_panel" style="position:absolute; opacity:0.5;">
1062					<img src="" id="controls_left_panel" style="position:absolute;  opacity:0.5;">
1063						
1064			
1065			</div> <!-- touch_controls_gfx -->
1066
1067			<!-- used for clipboard access & keyboard input; displayed and used by PICO-8 only once needed. can be safely removed if clipboard / key presses not needed. -->
1068			<!-- (needs to be inside p8_playarea so that it still works under Chrome when fullscreened) -->
1069			<!-- 0.2.5: added "display:none"; pico8.js shows on demand to avoid mac osx accent character selector // https://www.lexaloffle.com/bbs/?tid=47743 -->
1070
1071			<textarea id="codo_textarea" class="emscripten" style="display:none; position:absolute; left:-9999px; height:0px; overflow:hidden"></textarea>
1072
1073		</div> <!--p8_playarea -->
1074
1075	</div> <!-- p8_container -->
1076
1077</div> <!-- p8_frame -->
1078</div> <!-- p8_frame_0 size limit -->
1079
1080<script type="text/javascript">
1081
1082	p8_update_layout();
1083	p8_update_button_icons();
1084
1085	var canvas = document.getElementById("canvas");
1086	Module = {};
1087	Module.canvas = canvas;
1088
1089	// from @ultrabrite's shell: test if an AudioContext can be created outside of an event callback.
1090	// If it can't be created, then require pressing the start button to run the cartridge
1091
1092	if (p8_autoplay)
1093	{
1094		var temp_context = new AudioContext();
1095		temp_context.onstatechange = function ()
1096		{
1097			if (temp_context.state=='running')
1098			{
1099				p8_run_cart();
1100				temp_context.close();
1101			}
1102		};
1103	}
1104
1105	// pointer lock request needs to be inside a canvas interaction event
1106	// pico8_state.request_pointer_lock is true when 0x5f2d bit 0 and bit 2 are set -- poke(0x5f2d,0x5) 
1107	// note on mouse acceleration for future: // https://github.com/w3c/pointerlock/pull/49
1108	canvas.addEventListener("click", function()
1109	{
1110		if (!p8_touch_detected)
1111			if (pico8_state.request_pointer_lock)
1112				canvas.requestPointerLock();
1113	});
1114	
1115</script>
1116
1117
1118
1119<!-- Add content below the cart here -->
1120
1121
1122
1123
1124</div> <!-- body_0 -->
1125</body></html>
1126