shithub: puzzles

Download patch

ref: fcf573c672b0aa862ccc087a74256ee214c277b5
parent: d0500732f7bdab97ecaece2c5c74dd18f8122ea4
author: Jonas Kölker <jonaskoelker@yahoo.com>
date: Thu Oct 1 12:44:43 EDT 2015

Update Sixteen keyboard system for faster entry.

Pressing Ctrl-arrow or Shift-arrow on a tile now moves the row or
column under the tile. With Ctrl, the cursor moves as well so you can
keep making moves that affect the same tile; with Shift, the cursor
stays put so you can keep making moves that affect the same location.

--- a/puzzles.but
+++ b/puzzles.but
@@ -662,6 +662,13 @@
 around the edge of the grid, and use the return key to move the
 row/column in the direction indicated. 
 
+You can also move the tiles directly.  Move the cursor onto a tile,
+hold Control and press an arrow key to move the tile under the
+cursor and move the cursor along with the tile.  Or, hold Shift to
+move only the tile.  Pressing Enter simulates holding down Control
+(press Enter again to release), while pressing Space simulates
+holding down shift.
+
 (All the actions described in \k{common-actions} are also available.)
 
 \H{sixteen-params} \I{parameters, for Sixteen}Sixteen parameters
--- a/sixteen.c
+++ b/sixteen.c
@@ -27,6 +27,9 @@
 #define Y(state, i) ( (i) / (state)->w )
 #define C(state, x, y) ( (y) * (state)->w + (x) )
 
+#define TILE_CURSOR(i, state, x, y) ((i) == C((state), (x), (y)) &&     \
+                                     0 <= (x) && (x) < (state)->w &&    \
+                                     0 <= (y) && (y) < (state)->h)
 enum {
     COL_BACKGROUND,
     COL_TEXT,
@@ -555,9 +558,12 @@
     return ret;
 }
 
+enum cursor_mode { unlocked, lock_tile, lock_position };
+
 struct game_ui {
     int cur_x, cur_y;
     int cur_visible;
+    enum cursor_mode cur_mode;
 };
 
 static game_ui *new_ui(const game_state *state)
@@ -564,8 +570,9 @@
 {
     game_ui *ui = snew(game_ui);
     ui->cur_x = 0;
-    ui->cur_y = -1;
+    ui->cur_y = 0;
     ui->cur_visible = FALSE;
+    ui->cur_mode = unlocked;
 
     return ui;
 }
@@ -572,7 +579,7 @@
 
 static void free_ui(game_ui *ui)
 {
-  sfree(ui);
+    sfree(ui);
 }
 
 static char *encode_ui(const game_ui *ui)
@@ -603,21 +610,71 @@
 {
     int cx = -1, cy = -1, dx, dy;
     char buf[80];
+    int shift = button & MOD_SHFT, control = button & MOD_CTRL,
+        pad = button & MOD_NUM_KEYPAD;
 
     button &= ~MOD_MASK;
 
-    if (IS_CURSOR_MOVE(button)) {
-        /* right/down rotates cursor clockwise,
-         * left/up rotates anticlockwise. */
-        int cpos, diff;
-        cpos = c2pos(state->w, state->h, ui->cur_x, ui->cur_y);
-        diff = c2diff(state->w, state->h, ui->cur_x, ui->cur_y, button);
+    if (IS_CURSOR_MOVE(button) || pad) {
+        if (!ui->cur_visible) {
+            ui->cur_visible = 1;
+            return "";
+        }
 
-        cpos += diff;
-        pos2c(state->w, state->h, cpos, &ui->cur_x, &ui->cur_y);
+        if (control || shift || ui->cur_mode) {
+            int x = ui->cur_x, y = ui->cur_y, xwrap = x, ywrap = y;
+            if (x < 0 || x >= state->w || y < 0 || y >= state->h)
+                return NULL;
+            move_cursor(button | pad, &x, &y,
+                        state->w, state->h, FALSE);
+            move_cursor(button | pad, &xwrap, &ywrap,
+                        state->w, state->h, TRUE);
 
-        ui->cur_visible = 1;
-        return "";
+            if (x != xwrap) {
+                sprintf(buf, "R%d,%c1", y, x ? '+' : '-');
+            } else if (y != ywrap) {
+                sprintf(buf, "C%d,%c1", x, y ? '+' : '-');
+            } else if (x == ui->cur_x)
+                sprintf(buf, "C%d,%d", x, y - ui->cur_y);
+            else
+                sprintf(buf, "R%d,%d", y, x - ui->cur_x);
+
+            if (control || (!shift && ui->cur_mode == lock_tile)) {
+                ui->cur_x = xwrap;
+                ui->cur_y = ywrap;
+            }
+
+            return dupstr(buf);
+        } else {
+            int x = ui->cur_x + 1, y = ui->cur_y + 1;
+
+            move_cursor(button | pad, &x, &y,
+                        state->w + 2, state->h + 2, FALSE);
+
+            if (x == 0 && y == 0) {
+                int t = ui->cur_x;
+                ui->cur_x = ui->cur_y;
+                ui->cur_y = t;
+            } else if (x == 0 && y == state->h + 1) {
+                int t = ui->cur_x;
+                ui->cur_x = (state->h - 1) - ui->cur_y;
+                ui->cur_y = (state->h - 1) - t;
+            } else if (x == state->w + 1 && y == 0) {
+                int t = ui->cur_x;
+                ui->cur_x = (state->w - 1) - ui->cur_y;
+                ui->cur_y = (state->w - 1) - t;
+            } else if (x == state->w + 1 && y == state->h + 1) {
+                int t = ui->cur_x;
+                ui->cur_x = state->w - state->h + ui->cur_y;
+                ui->cur_y = state->h - state->w + t;
+            } else {
+                ui->cur_x = x - 1;
+                ui->cur_y = y - 1;
+            }
+
+            ui->cur_visible = 1;
+            return "";
+        }
     }
 
     if (button == LEFT_BUTTON || button == RIGHT_BUTTON) {
@@ -626,8 +683,16 @@
         ui->cur_visible = 0;
     } else if (IS_CURSOR_SELECT(button)) {
         if (ui->cur_visible) {
-            cx = ui->cur_x;
-            cy = ui->cur_y;
+            if (ui->cur_x == -1 || ui->cur_x == state->w ||
+                ui->cur_y == -1 || ui->cur_y == state->h) {
+                cx = ui->cur_x;
+                cy = ui->cur_y;
+            } else {
+                const enum cursor_mode m = (button == CURSOR_SELECT2 ?
+                                            lock_position : lock_tile);
+                ui->cur_mode = (ui->cur_mode == m ? unlocked : m);
+                return "";
+            }
         } else {
             ui->cur_visible = 1;
             return "";
@@ -853,7 +918,7 @@
     else if (cur_y == ds->h) /* Bottom row */
         draw_arrow(dr, ds, COORD(cur_x+1), COORD(ds->h), -1, 0, cur);
     else
-        assert(!"Invalid cursor position");
+        return;
 
     draw_update(dr, COORD(cur_x), COORD(cur_y),
                 TILE_SIZE, TILE_SIZE);
@@ -922,11 +987,11 @@
     if (ui->cur_visible) {
         cur_x = ui->cur_x; cur_y = ui->cur_y;
     }
+
     if (cur_x != ds->cur_x || cur_y != ds->cur_y) {
         /* Cursor has changed; redraw two (prev and curr) arrows. */
         draw_arrow_for_cursor(dr, ds, cur_x, cur_y, 1);
         draw_arrow_for_cursor(dr, ds, ds->cur_x, ds->cur_y, 0);
-        ds->cur_x = cur_x; ds->cur_y = cur_y;
     }
 
     /*
@@ -952,8 +1017,11 @@
 	t0 = t;
 
 	if (ds->bgcolour != bgcolour ||   /* always redraw when flashing */
-	    ds->tiles[i] != t || ds->tiles[i] == -1 || t == -1) {
-	    int x, y, x2, y2;
+            ds->tiles[i] != t || ds->tiles[i] == -1 || t == -1 ||
+            ((ds->cur_x != cur_x || ds->cur_y != cur_y) && /* cursor moved */
+             (TILE_CURSOR(i, state, ds->cur_x, ds->cur_y) ||
+              TILE_CURSOR(i, state, cur_x, cur_y)))) {
+            int x, y, x2, y2;
 
 	    /*
 	     * Figure out what to _actually_ draw, and where to
@@ -1021,12 +1089,18 @@
 		x2 = y2 = -1;
 	    }
 
-	    draw_tile(dr, ds, state, x, y, t, bgcolour);
+	    draw_tile(dr, ds, state, x, y, t,
+		      (x2 == -1 && TILE_CURSOR(i, state, cur_x, cur_y)) ?
+                      COL_LOWLIGHT : bgcolour);
+
 	    if (x2 != -1 || y2 != -1)
 		draw_tile(dr, ds, state, x2, y2, t, bgcolour);
 	}
 	ds->tiles[i] = t0;
     }
+
+    ds->cur_x = cur_x;
+    ds->cur_y = cur_y;
 
     unclip(dr);