about summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--027debug.cc17
-rw-r--r--030container.cc4
-rw-r--r--032array.cc4
-rw-r--r--edit.mu391
4 files changed, 171 insertions, 245 deletions
diff --git a/027debug.cc b/027debug.cc
index a855aad8..301cd01a 100644
--- a/027debug.cc
+++ b/027debug.cc
@@ -75,7 +75,22 @@ _DUMP_TRACE,
 Recipe_ordinal["$dump-trace"] = _DUMP_TRACE;
 :(before "End Primitive Recipe Implementations")
 case _DUMP_TRACE: {
-  DUMP("");
+  if (ingredients.empty()) {
+    DUMP("");
+  }
+  else {
+    DUMP(current_instruction().ingredients.at(0).name);
+  }
+  break;
+}
+
+:(before "End Primitive Recipe Declarations")
+_CLEAR_TRACE,
+:(before "End Primitive Recipe Numbers")
+Recipe_ordinal["$clear-trace"] = _CLEAR_TRACE;
+:(before "End Primitive Recipe Implementations")
+case _CLEAR_TRACE: {
+  CLEAR_TRACE;
   break;
 }
 
diff --git a/030container.cc b/030container.cc
index a679626e..4b013450 100644
--- a/030container.cc
+++ b/030container.cc
@@ -112,6 +112,8 @@ case GET: {
   assert(is_literal(current_instruction().ingredients.at(1)));
   assert(scalar(ingredients.at(1)));
   long long int offset = ingredients.at(1).at(0);
+  assert(offset >= 0);
+  assert(offset < size_of(base));
   long long int src = base_address;
   for (long long int i = 0; i < offset; ++i) {
     src += size_of(Type[base_type].elements.at(i));
@@ -161,6 +163,8 @@ case GET_ADDRESS: {
   assert(is_literal(current_instruction().ingredients.at(1)));
   assert(scalar(ingredients.at(1)));
   long long int offset = ingredients.at(1).at(0);
+  assert(offset >= 0);
+  assert(offset < size_of(base));
   long long int result = base_address;
   for (long long int i = 0; i < offset; ++i) {
     result += size_of(Type[base_type].elements.at(i));
diff --git a/032array.cc b/032array.cc
index 3afb11a4..20cba6fe 100644
--- a/032array.cc
+++ b/032array.cc
@@ -87,6 +87,8 @@ case INDEX: {
   vector<type_ordinal> element_type = array_element(base.types);
 //?   trace(Primitive_recipe_depth, "run") << "offset: " << offset_val.at(0); //? 1
 //?   trace(Primitive_recipe_depth, "run") << "size of elements: " << size_of(element_type); //? 1
+  assert(offset_val.at(0) >= 0);
+  assert(offset_val.at(0) < Memory[base_address]*size_of(element_type) + 1);
   long long int src = base_address + 1 + offset_val.at(0)*size_of(element_type);
   trace(Primitive_recipe_depth, "run") << "address to copy is " << src;
   trace(Primitive_recipe_depth, "run") << "its type is " << Type[element_type.at(0)].name;
@@ -137,6 +139,8 @@ case INDEX_ADDRESS: {
   reagent offset = canonize(current_instruction().ingredients.at(1));
   vector<double> offset_val(read_memory(offset));
   vector<type_ordinal> element_type = array_element(base.types);
+  assert(offset_val.at(0) >= 0);
+  assert(offset_val.at(0) < Memory[base_address]*size_of(element_type) + 1);
   long long int result = base_address + 1 + offset_val.at(0)*size_of(element_type);
   products.resize(1);
   products.at(0).push_back(result);
diff --git a/edit.mu b/edit.mu
index 8239fb0b..f21f5ea8 100644
--- a/edit.mu
+++ b/edit.mu
@@ -3,48 +3,45 @@
 recipe main [
   default-space:address:array:location <- new location:type, 30:literal
   open-console
-  programming-environment 0:literal/screen, 0:literal/console
-  close-console
+  initial-recipe:address:array:character <- new [recipe new-add [
+  x:number <- next-ingredient
+  y:number <- next-ingredient
+  z:number <- add x:number, y:number
+  reply z:number
+]]
+  initial-sandbox:address:array:character <- new [new-add 2:literal, 3:literal]
+  programming-environment 0:literal/screen, 0:literal/console, initial-recipe:address:array:character, initial-sandbox:address:array:character
 ]
 
 recipe programming-environment [
   default-space:address:array:location <- new location:type, 30:literal
   screen:address <- next-ingredient
   console:address <- next-ingredient
+  initial-recipe-contents:address:array:character <- next-ingredient
+  initial-sandbox-contents:address:array:character <- next-ingredient
   width:number <- screen-width screen:address
   height:number <- screen-height screen:address
   # top menu
   draw-horizontal screen:address, 0:literal, 0:literal/left, width:number, 32:literal/space, 0:literal/black, 238:literal/grey
   button-start:number <- subtract width:number, 20:literal
+  button-on-screen?:boolean <- greater-or-equal button-start:number, 0:literal
+  assert button-on-screen?:boolean, [screen too narrow for menu]
   move-cursor screen:address, 0:literal/row, button-start:number/column
   run-button:address:array:character <- new [ run (F10)  ]
   print-string screen:address, run-button:address:array:character, 255:literal/white, 161:literal/reddish
   # dotted line down the middle
   divider:number, _ <- divide-with-remainder width:number, 2:literal
   draw-vertical screen:address, divider:number, 1:literal/top, height:number, 9482:literal/vertical-dotted
-  # editor on the left
-  left:address:array:character <- new [recipe new-add [
-  x:number <- next-ingredient
-  y:number <- next-ingredient
-  z:number <- add x:number, y:number
-  reply z:number
-]]
-  left-editor:address:editor-data <- new-editor left:address:array:character, screen:address, 1:literal/top, 0:literal/left, divider:number/right
-  # editor on the right
-  right:address:array:character <- new [new-add 2:literal, 3:literal]
+  # recipe editor on the left
+  left-editor:address:editor-data <- new-editor initial-recipe-contents:address:array:character, screen:address, 0:literal/left, divider:number/right
+  # sandbox editor on the right
   new-left:number <- add divider:number, 1:literal
   new-right:number <- add new-left:number, 5:literal
-  right-editor:address:editor-data <- new-editor right:address:array:character, screen:address, 1:literal/top, new-left:number, width:number
-  # chain
-  x:address:address:editor-data <- get-address left-editor:address:editor-data/deref, next-editor:offset
-  x:address:address:editor-data/deref <- copy right-editor:address:editor-data
-  # initialize focus
-  reset-focus left-editor:address:editor-data
-  cursor-row:number <- get left-editor:address:editor-data/deref, cursor-row:offset
-  cursor-column:number <- get left-editor:address:editor-data/deref, cursor-column:offset
-  move-cursor screen:address, cursor-row:number, cursor-column:number
+  right-editor:address:editor-data <- new-editor initial-sandbox-contents:address:array:character, screen:address, new-left:number, width:number
+  # initialize cursor
+  update-cursor screen:address, left-editor:address:editor-data, right-editor:address:editor-data, 0:literal/focus-in-recipe
   # and we're off!
-  event-loop screen:address, console:address, left-editor:address:editor-data
+  event-loop screen:address, console:address, left-editor:address:editor-data, right-editor:address:editor-data
 ]
 
 scenario editor-initially-prints-string-to-screen [
@@ -63,6 +60,12 @@ scenario editor-initially-prints-string-to-screen [
 ## In which we introduce the editor data structure, and show how it displays
 ## text to the screen.
 
+container programming-environment-data [
+  recipes:address:editor-data
+  current-sandbox:address:editor-data
+  focus:boolean  # false => focus in recipes; true => focus in current-sandbox
+]
+
 container editor-data [
   # editable text: doubly linked list of characters (head contains a special sentinel)
   data:address:duplex-list
@@ -81,11 +84,6 @@ container editor-data [
   # raw screen coordinates of cursor
   cursor-row:number
   cursor-column:number
-
-  # pointer to another editor, responsible for a different area of screen.
-  # helps organize editors in a 'chain'.
-  next-editor:address:editor-data
-  in-focus?:boolean  # set for the one editor in this chain currently being edited
 ]
 
 # editor:address, screen:address <- new-editor s:address:array:character, screen:address, left:number, right:number
@@ -120,10 +118,6 @@ recipe new-editor [
   init:address:address:duplex-list/deref <- push-duplex 167:literal/§, 0:literal/tail
   y:address:address:duplex-list <- get-address result:address:editor-data/deref, before-cursor:offset
   y:address:address:duplex-list/deref <- copy init:address:address:duplex-list/deref
-  # set focus
-  # if using multiple editors, must call reset-focus after chaining them all
-  b:address:boolean <- get-address result:address:editor-data/deref, in-focus?:offset
-  b:address:boolean/deref <- copy 1:literal/true
   # early exit if s is empty
   reply-unless s:address:array:character, result:address:editor-data
   len:number <- length s:address:array:character/deref
@@ -132,15 +126,9 @@ recipe new-editor [
   # now we can start appending the rest, character by character
   curr:address:duplex-list <- copy init:address:address:duplex-list/deref
   {
-#?     $print idx:number, [ vs ], len:number, [ 
-#? ] #? 1
-#?     $print [append to ], curr:address:duplex-list, [ 
-#? ] #? 1
     done?:boolean <- greater-or-equal idx:number, len:number
     break-if done?:boolean
     c:character <- index s:address:array:character/deref, idx:number
-#?     $print [aa: ], c:character, [ 
-#? ] #? 1
     insert-duplex c:character, curr:address:duplex-list
     # next iter
     curr:address:duplex-list <- next-duplex curr:address:duplex-list
@@ -151,8 +139,8 @@ recipe new-editor [
   y:address:address:duplex-list <- get-address result:address:editor-data/deref, before-cursor:offset
   y:address:address:duplex-list/deref <- copy init:address:address:duplex-list/deref
   # perform initial rendering to screen
-  result:address:editor-data <- render screen:address, result:address:editor-data
-  reply result:address:editor-data
+  result:address:editor-data, screen:address <- render screen:address, result:address:editor-data
+  reply result:address:editor-data, screen:address/same-as-ingredient:0
 ]
 
 scenario editor-initializes-without-data [
@@ -182,8 +170,7 @@ recipe render [
   default-space:address:array:location <- new location:type, 40:literal
   screen:address <- next-ingredient
   editor:address:editor-data <- next-ingredient
-#?   $print [=== render
-#? ] #? 2
+  reply-unless editor:address:editor-data, editor:address:editor-data/same-as-ingredient:1, screen:address/same-as-ingredient:0
   left:number <- get editor:address:editor-data/deref, left:offset
   screen-height:number <- screen-height screen:address
   right:number <- get editor:address:editor-data/deref, right:offset
@@ -329,8 +316,6 @@ recipe render [
   }
   # update cursor
   {
-    in-focus?:boolean <- get editor:address:editor-data/deref, in-focus?:offset
-    break-unless in-focus?:boolean
     cursor-inside-right-margin?:boolean <- lesser-or-equal cursor-column:address:number/deref, right:number
     assert cursor-inside-right-margin?:boolean, [cursor outside right margin]
     cursor-inside-left-margin?:boolean <- greater-or-equal cursor-column:address:number/deref, left:number
@@ -338,7 +323,7 @@ recipe render [
     move-cursor screen:address, cursor-row:address:number/deref, cursor-column:address:number/deref
   }
   show-screen screen:address
-  reply editor:address:editor-data/same-as-ingredient:1
+  reply editor:address:editor-data/same-as-ingredient:1, screen:address/same-as-ingredient:0
 ]
 
 # row:number <- render-string s:address:array:character, editor:address:editor-data, color:number, row:number
@@ -531,13 +516,16 @@ scenario editor-initializes-empty-text [
 
 ## handling events from the keyboard, mouse, touch screen, ...
 
-# Takes a chain of editors (chained using editor-data.next-editor), sends each
-# event from the console to each editor.
+# takes two editors, sends each event from the console to each editor
 recipe event-loop [
   default-space:address:array:location <- new location:type, 30:literal
   screen:address <- next-ingredient
   console:address <- next-ingredient
-  editor:address:editor-data <- next-ingredient
+  recipes:address:editor-data <- next-ingredient
+  current-sandbox:address:editor-data <- next-ingredient
+  sandbox-in-focus?:boolean <- next-ingredient
+#?   $print [sandbox in focus? ], sandbox-in-focus?:boolean, [ 
+#? ] #? 1
   {
     # looping over each (keyboard or touch) event as it occurs
     +next-event
@@ -555,32 +543,36 @@ recipe event-loop [
       {
         do-run?:boolean <- equal k:address:number/deref, 65526:literal/F10
         break-unless do-run?:boolean
-        run-sandboxes editor:address:editor-data
-        render-all screen:address, editor:address:editor-data
+#?         trace [app], [run] #? 1
+        run-sandboxes recipes:address:editor-data, current-sandbox:address:editor-data
+        render-all screen:address, recipes:address:editor-data, current-sandbox:address:editor-data, sandbox-in-focus?:boolean
         loop +next-event:label  # done with this event; no need to send to editors
       }
     }
-    # if it's not global, locate the editor the event should go to
-    curr:address:editor-data <- copy editor:address:editor-data
+    # 'touch' event - send to both editors
     {
-      break-unless curr:address:editor-data
-      handle-event screen:address, console:address, curr:address:editor-data, e:event
-      curr:address:editor-data <- get curr:address:editor-data/deref, next-editor:offset
-      loop
+      t:address:touch-event <- maybe-convert e:event, touch:variant
+      break-unless t:address:touch-event
+#?       trace [app], [touch] #? 1
+      _ <- move-cursor-in-editor recipes:address:editor-data, t:address:touch-event/deref
+      sandbox-in-focus?:boolean <- move-cursor-in-editor current-sandbox:address:editor-data, t:address:touch-event/deref
+      render-all screen:address, recipes:address:editor-data, current-sandbox:address:editor-data, sandbox-in-focus?:boolean
+      loop +next-event:label
     }
-    # ..and position the cursor
-    curr:address:editor-data <- copy editor:address:editor-data
+    # if it's not global, send to appropriate editor
     {
-      break-unless curr:address:editor-data
       {
-        in-focus?:boolean <- get curr:address:editor-data/deref, in-focus?:offset
-        break-unless in-focus?:boolean
-        cursor-row:number <- get curr:address:editor-data/deref, cursor-row:offset
-        cursor-column:number <- get curr:address:editor-data/deref, cursor-column:offset
-        move-cursor screen:address, cursor-row:number, cursor-column:number
+        break-if sandbox-in-focus?:boolean
+#?         $print [event in recipes
+#? ] #? 1
+        handle-event screen:address, console:address, recipes:address:editor-data, e:event
+      }
+      {
+        break-unless sandbox-in-focus?:boolean
+#?         $print [event in current-sandbox: ], sandbox-in-focus?:boolean, [ 
+#? ] #? 1
+        handle-event screen:address, console:address, current-sandbox:address:editor-data, e:event
       }
-      curr:address:editor-data <- get curr:address:editor-data/deref, next-editor:offset
-      loop
     }
     loop
   }
@@ -592,25 +584,7 @@ recipe handle-event [
   console:address <- next-ingredient
   editor:address:editor-data <- next-ingredient
   e:event <- next-ingredient
-  # 'touch' event
-  {
-    t:address:touch-event <- maybe-convert e:event, touch:variant
-    break-unless t:address:touch-event
-    move-cursor-in-editor editor:address:editor-data, t:address:touch-event/deref
-    jump +render:label
-  }
-  # other events trigger only if this editor is in focus
-#?   $print [checking ], editor:address:editor-data, [ 
-#? ] #? 1
-#?   x:address:boolean <- get-address editor:address:editor-data/deref, in-focus?:offset #? 1
-#?   $print [address of focus: ], x:address:boolean, [ 
-#? ] #? 1
-  in-focus?:address:boolean <- get-address editor:address:editor-data/deref, in-focus?:offset
-#?   $print [ at ], in-focus?:address:boolean, [ 
-#? ] #? 1
-  reply-unless in-focus?:address:boolean/deref  # no need to render
-#?   $print [in focus: ], editor:address:editor-data, [ 
-#? ] #? 1
+  reply-unless editor:address:editor-data
   # typing a character
   {
     c:address:character <- maybe-convert e:event, text:variant
@@ -684,6 +658,7 @@ recipe handle-event [
   {
     move-to-previous-character?:boolean <- equal k:address:number/deref, 65515:literal/left-arrow
     break-unless move-to-previous-character?:boolean
+#?     trace [app], [left arrow] #? 1
     # if not at start of text (before-cursor at § sentinel)
     prev:address:duplex-list <- prev-duplex before-cursor:address:address:duplex-list/deref
     break-unless prev:address:duplex-list
@@ -691,6 +666,7 @@ recipe handle-event [
     {
       at-left-margin?:boolean <- equal cursor-column:address:number/deref, 0:literal
       break-if at-left-margin?:boolean
+#?       trace [app], [decrementing] #? 1
       cursor-column:address:number/deref <- subtract cursor-column:address:number/deref, 1:literal
       jump +render:label
     }
@@ -701,6 +677,7 @@ recipe handle-event [
       prevc:character <- get before-cursor:address:address:duplex-list/deref/deref, value:offset
       previous-character-is-newline?:boolean <- equal prevc:character, 10:literal/newline
       break-unless previous-character-is-newline?:boolean
+#?       trace [app], [previous line] #? 1
       # compute length of previous line
       end-of-line:number <- previous-line-length before-cursor:address:address:duplex-list/deref, d:address:duplex-list
       cursor-row:address:number/deref <- subtract cursor-row:address:number/deref, 1:literal
@@ -740,39 +717,33 @@ recipe handle-event [
   {
     more-events?:boolean <- has-more-events? console:address
     break-if more-events?:boolean
+#?     $print [trying to call render
+#? ] #? 1
     render screen:address, editor:address:editor-data
   }
 ]
 
+# process click, return if it was on current editor
+# todo: ignores menu bar (for now just displays shortcuts)
 recipe move-cursor-in-editor [
   default-space:address:array:location <- new location:type, 30:literal
   editor:address:editor-data <- next-ingredient
   t:touch-event <- next-ingredient
-  # clicks on the menu bar shouldn't affect focus
-  click-row:number <- get t:touch-event, row:offset
-  too-far-up?:boolean <- lesser-than click-row:number, 1:literal/top
-  reply-if too-far-up?:boolean
-  # not on menu? reset focus then set it if necessary
-  in-focus?:address:boolean <- get-address editor:address:editor-data/deref, in-focus?:offset
-  in-focus?:address:boolean/deref <- copy 0:literal/true
+  reply-unless editor:address:editor-data, 0:literal/false
   click-column:number <- get t:touch-event, column:offset
   left:number <- get editor:address:editor-data/deref, left:offset
   too-far-left?:boolean <- lesser-than click-column:number, left:number
-  reply-if too-far-left?:boolean
+  reply-if too-far-left?:boolean, 0:literal/false
   right:number <- get editor:address:editor-data/deref, right:offset
   too-far-right?:boolean <- greater-than click-column:number, right:number
-  reply-if too-far-right?:boolean
-#?   $print [focus now at ], editor:address:editor-data, [ 
-#? ] #? 2
-  # click on this window; gain focus
-  in-focus?:address:boolean/deref <- copy 1:literal/true
+  reply-if too-far-right?:boolean, 0:literal/false
   # update cursor
   cursor-row:address:number <- get-address editor:address:editor-data/deref, cursor-row:offset
   cursor-row:address:number/deref <- get t:touch-event, row:offset
   cursor-column:address:number <- get-address editor:address:editor-data/deref, cursor-column:offset
   cursor-column:address:number/deref <- get t:touch-event, column:offset
-#?   $print [column is at: ], cursor-column:address:number, [ 
-#? ] #? 1
+  # gain focus
+  reply 1:literal/true
 ]
 
 recipe insert-at-cursor [
@@ -869,47 +840,43 @@ recipe previous-line-length [
 recipe render-all [
   default-space:address:array:location <- new location:type, 40:literal
   screen:address <- next-ingredient
-  editor:address:editor-data <- next-ingredient
-  curr:address:editor-data <- copy editor:address:editor-data
+  recipes:address:editor-data <- next-ingredient
+  current-sandbox:address:editor-data <- next-ingredient
+  sandbox-in-focus?:boolean <- next-ingredient
+  render screen:address, recipes:address:editor-data
+  render screen:address, current-sandbox:address:editor-data
+  update-cursor screen:address, recipes:address:editor-data, current-sandbox:address:editor-data, sandbox-in-focus?:boolean
+]
+
+recipe update-cursor [
+  default-space:address:array:location <- new location:type, 40:literal
+  screen:address <- next-ingredient
+  recipes:address:editor-data <- next-ingredient
+  current-sandbox:address:editor-data <- next-ingredient
+  sandbox-in-focus?:boolean <- next-ingredient
   {
-    break-unless curr:address:editor-data
-    render screen:address, curr:address:editor-data
-    curr:address:editor-data <- get curr:address:editor-data/deref, next-editor:offset
-    loop
+    break-if sandbox-in-focus?:boolean
+#?     trace [app], [recipes in focus] #? 1
+    cursor-row:number <- get recipes:address:editor-data/deref, cursor-row:offset
+    cursor-column:number <- get recipes:address:editor-data/deref, cursor-column:offset
   }
-  # update cursor
-  curr:address:editor-data <- copy editor:address:editor-data
   {
-    +next-editor
-    in-focus?:boolean <- get curr:address:editor-data/deref, in-focus?:offset
-    {
-      break-if in-focus?:boolean
-      curr:address:editor-data <- get curr:address:editor-data/deref, next-editor:offset
-      loop +next-editor:label
-    }
-    cursor-row:number <- get curr:address:editor-data/deref, cursor-row:offset
-    cursor-column:number <- get curr:address:editor-data/deref, cursor-column:offset
-    move-cursor screen:address, cursor-row:number, cursor-column:number
+    break-unless sandbox-in-focus?:boolean
+#?     trace [app], [sandbox in focus] #? 1
+    cursor-row:number <- get current-sandbox:address:editor-data/deref, cursor-row:offset
+    cursor-column:number <- get current-sandbox:address:editor-data/deref, cursor-column:offset
   }
+  move-cursor screen:address, cursor-row:number, cursor-column:number
 ]
 
 scenario editor-handles-empty-event-queue [
   assume-screen 10:literal/width, 5:literal/height
-#?   3:number <- get screen:address/deref, num-rows:offset #? 1
-#?   $print [0: ], screen:address, [: ], 3:number, [ 
-#? ] #? 1
   1:address:array:character <- new [abc]
-#?   $print [1: ], screen:address, [ 
-#? ] #? 1
   2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 10:literal/right
   assume-console []
-#?   $print [8: ], screen:address, [ 
-#? ] #? 1
   run [
     event-loop screen:address, console:address, 2:address:editor-data
   ]
-#?   $print [9: ], screen:address, [ 
-#? ] #? 1
   screen-should-contain [
     .          .
     .abc       .
@@ -1547,17 +1514,18 @@ d]
   assume-console [
     left-click 2, 0
     press 65515  # left arrow
-    type [0]
   ]
   run [
     event-loop screen:address, console:address, 2:address:editor-data
+    3:number <- get 2:address:editor-data/deref, cursor-row:offset
+    4:number <- get 2:address:editor-data/deref, cursor-column:offset
   ]
-  screen-should-contain [
-    .          .
-    .abc0      .
-    .d         .
-    .          .
+  memory-should-contain [
+    3 <- 1
+    4 <- 3
   ]
+#?   $dump-trace [app] #? 1
+#?   $exit #? 1
 ]
 
 scenario editor-moves-cursor-to-previous-line-with-left-arrow-at-start-of-line-2 [
@@ -1745,39 +1713,37 @@ de]
 
 scenario point-at-multiple-editors [
   assume-screen 10:literal/width, 5:literal/height
-  # initialize an editor covering left half of screen
+  # initialize recipe editor covering left half of screen
   1:address:array:character <- new [abc]
   2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 5:literal/right
+  # initialize sandbox editor covering right half of screen
   3:address:array:character <- new [def]
-  # chain new editor to it, covering the right half of the screen
-  4:address:address:editor-data <- get-address 2:address:editor-data/deref, next-editor:offset
-  4:address:address:editor-data/deref <- new-editor 3:address:array:character, screen:address, 5:literal/left, 10:literal/right
-  # type one letter in each of them
+  4:address:editor-data <- new-editor 3:address:array:character, screen:address, 5:literal/left, 10:literal/right
+  # focus on both sides
   assume-console [
     left-click 1, 1
     left-click 1, 8
   ]
+  # check cursor column in each
   run [
-    event-loop screen:address, console:address, 2:address:editor-data
-    5:number <- get 2:address:editor-data/deref, cursor-column:offset
-    6:number <- get 4:address:address:editor-data/deref/deref, cursor-column:offset
+    event-loop screen:address, console:address, 2:address:editor-data, 4:address:editor-data
+    6:number <- get 2:address:editor-data/deref, cursor-column:offset
+    7:number <- get 4:address:editor-data/deref, cursor-column:offset
   ]
   memory-should-contain [
-    5 <- 1
-    6 <- 8
+    6 <- 1
+    7 <- 8
   ]
 ]
 
 scenario editors-chain-to-cover-multiple-columns [
   assume-screen 10:literal/width, 5:literal/height
-  # initialize an editor covering left half of screen
+  # initialize recipe editor covering left half of screen
   1:address:array:character <- new [abc]
   2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 5:literal/right
+  # initialize sandbox editor covering right half of screen
   3:address:array:character <- new [def]
-  # chain new editor to it, covering the right half of the screen
-  4:address:address:editor-data <- get-address 2:address:editor-data/deref, next-editor:offset
-  4:address:address:editor-data/deref <- new-editor 3:address:array:character, screen:address, 5:literal/left, 10:literal/right
-  reset-focus 2:address:editor-data
+  4:address:editor-data <- new-editor 3:address:array:character, screen:address, 5:literal/left, 10:literal/right
   # type one letter in each of them
   assume-console [
     left-click 1, 1
@@ -1786,9 +1752,9 @@ scenario editors-chain-to-cover-multiple-columns [
     type [1]
   ]
   run [
-    event-loop screen:address, console:address, 2:address:editor-data
+    event-loop screen:address, console:address, 2:address:editor-data, 4:address:editor-data
     5:number <- get 2:address:editor-data/deref, cursor-column:offset
-    6:number <- get 4:address:address:editor-data/deref/deref, cursor-column:offset
+    6:number <- get 4:address:editor-data/deref, cursor-column:offset
   ]
   screen-should-contain [
     .          .
@@ -1796,8 +1762,8 @@ scenario editors-chain-to-cover-multiple-columns [
     .          .
   ]
   memory-should-contain [
-    5 <- 2
-    6 <- 7
+    5 <- 2  # cursor column of recipe editor
+    6 <- 7  # cursor column of sandbox editor
   ]
   # show the cursor at the right window
   run [
@@ -1815,12 +1781,11 @@ scenario multiple-editors-cover-only-their-own-areas [
   run [
     # draw a divider
     draw-vertical screen:address, 5:literal/divider, 1:literal/top, 5:literal/height
-    # initialize editors on both sides of it and chain the two
+    # initialize editors on both sides of it
     1:address:array:character <- new [abc]
     2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 5:literal/right
     3:address:array:character <- new [def]
-    4:address:address:editor-data <- get-address 2:address:editor-data/deref, next-editor:offset
-    4:address:address:editor-data/deref <- new-editor 3:address:array:character, screen:address, 6:literal/left, 10:literal/right
+    4:address:editor-data <- new-editor 3:address:array:character, screen:address, 6:literal/left, 10:literal/right
   ]
   # divider isn't messed up
   screen-should-contain [
@@ -1833,72 +1798,43 @@ scenario multiple-editors-cover-only-their-own-areas [
 ]
 
 scenario editor-in-focus-keeps-cursor [
-  assume-screen 10:literal/width, 5:literal/height
-  # initialize an editor covering left half of screen
+  assume-screen 30:literal/width, 5:literal/height
   1:address:array:character <- new [abc]
-  2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 5:literal/right
-  3:address:array:character <- new [def]
-  # chain new editor to it, covering the right half of the screen
-  4:address:address:editor-data <- get-address 2:address:editor-data/deref, next-editor:offset
-  4:address:address:editor-data/deref <- new-editor 3:address:array:character, screen:address, 5:literal/left, 10:literal/right
-  # initialize cursor
+  2:address:array:character <- new [def]
+  # initialize programming environment and highlight cursor
+  assume-console []
   run [
-    reset-focus 2:address:editor-data
-    5:number <- get 2:address:editor-data/deref, cursor-row:offset
-    6:number <- get 2:address:editor-data/deref, cursor-column:offset
-    move-cursor screen:address, 5:number, 6:number
+    programming-environment screen:address, console:address, 1:address:array:character, 2:address:array:character
     screen:address <- print-character screen:address, 9251:literal/␣
   ]
-  # is it at the right place?
+  # is cursor at the right place?
   screen-should-contain [
-    .          .
-    .␣bc  def  .
-    .          .
+    .           run (F10)          .
+    .␣bc            ┊def           .
+    .               ┊              .
   ]
   # now try typing a letter
   assume-console [
     type [z]
   ]
   run [
-    event-loop screen:address, console:address, 2:address:editor-data
+    programming-environment screen:address, console:address, 1:address:array:character, 2:address:array:character
     screen:address <- print-character screen:address, 9251:literal/␣
   ]
   # cursor should still be right
   screen-should-contain [
-    .          .
-    .z␣bc def  .
-    .          .
+    .           run (F10)          .
+    .z␣bc           ┊def           .
+    .               ┊              .
   ]
 ]
 
-# set focus to first editor, reset it in later ones
-recipe reset-focus [
-  default-space:address:array:location <- new location:type, 30:literal
-  editor:address:editor-data <- next-ingredient
-  in-focus:address:boolean <- get-address editor:address:editor-data/deref, in-focus?:offset
-  in-focus:address:boolean/deref <- copy 1:literal/true
-  e:address:editor-data <- get editor:address:editor-data/deref, next-editor:offset
-  {
-    break-unless e:address:editor-data
-#?     $print [resetting focus in ], e:address:editor-data, [ 
-#? ] #? 1
-    x:address:boolean <- get-address e:address:editor-data/deref, in-focus?:offset
-#?     $print [ at ], x:address:boolean, [ 
-#? ] #? 1
-    x:address:boolean/deref <- copy 0:literal/false
-    e:address:editor-data <- get e:address:editor-data/deref, next-editor:offset
-    loop
-  }
-]
-
 ## Running code from the editors
 
 recipe editor-contents [
   default-space:address:array:location <- new location:type, 30:literal
   editor:address:editor-data <- next-ingredient
   buf:address:buffer <- new-buffer 80:literal
-#?   $print [buffer: ], buf:address:buffer, [ 
-#? ] #? 1
   curr:address:duplex-list <- get editor:address:editor-data/deref, data:offset
   # skip § sentinel
   assert curr:address:duplex-list, [editor without data is illegal; must have at least a sentinel]
@@ -1906,8 +1842,6 @@ recipe editor-contents [
   {
     break-unless curr:address:duplex-list
     c:character <- get curr:address:duplex-list/deref, value:offset
-#?     $print [appending ], c:character, [ 
-#? ] #? 1
     buffer-append buf:address:buffer, c:character
     curr:address:duplex-list <- next-duplex curr:address:duplex-list
     loop
@@ -1928,32 +1862,26 @@ scenario editor-provides-edited-contents [
     event-loop screen:address, console:address, 2:address:editor-data
     3:address:array:character <- editor-contents 2:address:editor-data
     4:array:character <- copy 3:address:array:character/deref
-#?     $dump-memory #? 1
   ]
   memory-should-contain [
     4:string <- [abdefc]
   ]
 ]
 
-## running code in editors
-
 scenario run-and-show-results [
   assume-screen 60:literal/width, 5:literal/height
-  # left editor is empty
+  # recipe editor is empty
   1:address:array:character <- new []
   2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 5:literal/right
-  # right editor contains an instruction without storing outputs
+  # sandbox editor contains an instruction without storing outputs
   3:address:array:character <- new [divide-with-remainder 11:literal, 3:literal]
-  4:address:address:editor-data <- get-address 2:address:editor-data/deref, next-editor:offset
-  4:address:address:editor-data/deref <- new-editor 3:address:array:character, screen:address, 5:literal/left, 60:literal/right
-  reset-focus 2:address:editor-data
+  4:address:editor-data <- new-editor 3:address:array:character, screen:address, 5:literal/left, 60:literal/right
   # run the code in the editors
   assume-console [
     press 65526  # F10
   ]
   run [
-    # now run query for it
-    event-loop screen:address, console:address, 2:address:editor-data
+    event-loop screen:address, console:address, 2:address:editor-data, 4:address:editor-data
   ]
   # check that screen prints the results
   screen-should-contain [
@@ -1963,7 +1891,6 @@ scenario run-and-show-results [
     .     2                                                      .
     .                                                            .
   ]
-#?   $exit #? 1
   screen-should-contain-in-color 7:literal/white, [
     .                                                            .
     .     divide-with-remainder 11:literal, 3:literal            .
@@ -1982,77 +1909,53 @@ scenario run-and-show-results [
 
 recipe run-sandboxes [
   default-space:address:array:location <- new location:type, 30:literal
-  editor:address:editor-data <- next-ingredient
-  # load code from left editor (recipes), save any warnings. recipes never generate output.
-  in:address:array:character <- editor-contents editor:address:editor-data
-  warnings:address:address:array:character <- get-address editor:address:editor-data/deref, warnings:offset
+  recipes:address:editor-data <- next-ingredient
+  current-sandbox:address:editor-data <- next-ingredient
+  # load code from recipe editor, save any warnings
+  in:address:array:character <- editor-contents recipes:address:editor-data
+  warnings:address:address:array:character <- get-address recipes:address:editor-data/deref, warnings:offset
   warnings:address:address:array:character/deref <- reload in:address:array:character
   # run contents of right editor (sandbox), save any warnings or output
-  sandbox:address:editor-data <- get editor:address:editor-data/deref, next-editor:offset
-  in:address:array:character <- editor-contents sandbox:address:editor-data
-  response:address:address:array:character <- get-address sandbox:address:editor-data/deref, response:offset
-  warnings:address:address:array:character <- get-address sandbox:address:editor-data/deref, warnings:offset
+  in:address:array:character <- editor-contents current-sandbox:address:editor-data
+  response:address:address:array:character <- get-address current-sandbox:address:editor-data/deref, response:offset
+  warnings:address:address:array:character <- get-address current-sandbox:address:editor-data/deref, warnings:offset
   response:address:address:array:character/deref, warnings:address:address:array:character/deref <- run-interactive in:address:array:character
-#?   print-string 0:literal/screen, response:address:address:array:character/deref #? 1
-#?   wait-for-some-interaction #? 1
-#?   close-console #? 1
-#?   $print response:address:address:array:character/deref, [, ], warnings:address:address:array:character/deref #? 1
-#?   $exit #? 1
 ]
 
 scenario run-instruction-and-print-warnings [
-#?   $print [=====
-#? ] #? 2
   assume-screen 60:literal/width, 5:literal/height
   # left editor is empty
   1:address:array:character <- new []
   2:address:editor-data <- new-editor 1:address:array:character, screen:address, 0:literal/left, 5:literal/right
   # right editor contains an illegal instruction
   3:address:array:character <- new [get 1234:number, foo:offset]
-  4:address:address:editor-data <- get-address 2:address:editor-data/deref, next-editor:offset
-  4:address:address:editor-data/deref <- new-editor 3:address:array:character, screen:address, 5:literal/left, 60:literal/right
-  reset-focus 2:address:editor-data
+  4:address:editor-data <- new-editor 3:address:array:character, screen:address, 5:literal/left, 60:literal/right
   # run the code in the editors
   assume-console [
     press 65526  # F10
   ]
   run [
-    # now run query for it
-#?     $print [about to start event loop
-#? ] #? 1
-    event-loop screen:address, console:address, 2:address:editor-data
+    event-loop screen:address, console:address, 2:address:editor-data, 4:address:editor-data
   ]
-  # check that screen prints the value in location 12
-#?   $print [a0
-#? ] #? 1
-#?   $dump-screen #? 1
-#?   get 1234:number, foo:offset #? 1
+  # check that screen prints error message in red
   screen-should-contain [
     .                                                            .
     .     get 1234:number, foo:offset                            .
     .     unknown element foo in container number                .
     .                                                            .
   ]
-#?   $print [a1
-#? ] #? 1
-#?   $dump-trace #? 1
-#?   $exit #? 2
   screen-should-contain-in-color 7:literal/white, [
     .                                                            .
     .     get 1234:number, foo:offset                            .
     .                                                            .
     .                                                            .
   ]
-#?   $print [a2
-#? ] #? 1
   screen-should-contain-in-color, 1:literal/red, [
     .                                                            .
     .                                                            .
     .     unknown element foo in container number                .
     .                                                            .
   ]
-#?   $print [a3
-#? ] #? 1
 ]
 
 ## helpers for drawing editor borders