ref: 865e8ad6ca3d83ad2a585ceeb1809e9f68c18a20
parent: a1be37343cfab49053396a3a70c7cca14d6f376f
author: Jacob Nevins <[email protected]>
date: Thu May 26 09:40:38 EDT 2005
Add origin-shifting (Shift+cursors) and source-shifting (Ctrl+cursors) to Net. (Adding modifier+cursors handling has had minor knock-on effects on the other puzzles, so that they can continue to ignore modifiers.) (An unfortunate side effect of this is some artifacts in exterior barrier drawing; notably, a disconnected corner can now appear at the corner of the grid under some circumstances. I haven't found a satisfactory way round this yet.) [originally from svn r5844]
--- a/cube.c
+++ b/cube.c
@@ -1013,6 +1013,8 @@
int i, j, dest, mask;
struct solid *poly;
+ button = button & (~MOD_MASK | MOD_NUM_KEYPAD);
+
/*
* All moves are made with the cursor keys or numeric keypad.
*/
--- a/fifteen.c
+++ b/fifteen.c
@@ -456,6 +456,8 @@
int gx, gy, dx, dy, ux, uy, up, p;
game_state *ret;
+ button &= ~MOD_MASK;
+
gx = X(from, from->gap_pos);
gy = Y(from, from->gap_pos);
--- a/gtk.c
+++ b/gtk.c
@@ -317,24 +317,26 @@
{
frontend *fe = (frontend *)data;
int keyval;
+ int shift = (event->state & GDK_SHIFT_MASK) ? MOD_SHFT : 0;
+ int ctrl = (event->state & GDK_CONTROL_MASK) ? MOD_CTRL : 0;
if (!fe->pixmap)
return TRUE;
if (event->keyval == GDK_Up)
- keyval = CURSOR_UP;
+ keyval = shift | ctrl | CURSOR_UP;
else if (event->keyval == GDK_KP_Up || event->keyval == GDK_KP_8)
keyval = MOD_NUM_KEYPAD | '8';
else if (event->keyval == GDK_Down)
- keyval = CURSOR_DOWN;
+ keyval = shift | ctrl | CURSOR_DOWN;
else if (event->keyval == GDK_KP_Down || event->keyval == GDK_KP_2)
keyval = MOD_NUM_KEYPAD | '2';
else if (event->keyval == GDK_Left)
- keyval = CURSOR_LEFT;
+ keyval = shift | ctrl | CURSOR_LEFT;
else if (event->keyval == GDK_KP_Left || event->keyval == GDK_KP_4)
keyval = MOD_NUM_KEYPAD | '4';
else if (event->keyval == GDK_Right)
- keyval = CURSOR_RIGHT;
+ keyval = shift | ctrl | CURSOR_RIGHT;
else if (event->keyval == GDK_KP_Right || event->keyval == GDK_KP_6)
keyval = MOD_NUM_KEYPAD | '6';
else if (event->keyval == GDK_KP_Home || event->keyval == GDK_KP_7)
--- a/net.c
+++ b/net.c
@@ -57,6 +57,13 @@
#define ROTATE_TIME 0.13F
#define FLASH_FRAME 0.07F
+/* Transform physical coords to game coords using game_drawstate ds */
+#define GX(x) (((x) + ds->org_x) % ds->width)
+#define GY(y) (((y) + ds->org_y) % ds->height)
+/* ...and game coords to physical coords */
+#define RX(x) (((x) + ds->width - ds->org_x) % ds->width)
+#define RY(y) (((y) + ds->height - ds->org_y) % ds->height)
+
enum {
COL_BACKGROUND,
COL_LOCKED,
@@ -82,7 +89,7 @@
};
struct game_state {
- int width, height, cx, cy, wrapping, completed;
+ int width, height, wrapping, completed;
int last_rotate_x, last_rotate_y, last_rotate_dir;
int used_solve, just_used_solve;
unsigned char *tiles;
@@ -1570,8 +1577,6 @@
state = snew(game_state);
w = state->width = params->width;
h = state->height = params->height;
- state->cx = state->width / 2;
- state->cy = state->height / 2;
state->wrapping = params->wrapping;
state->last_rotate_dir = state->last_rotate_x = state->last_rotate_y = 0;
state->completed = state->used_solve = state->just_used_solve = FALSE;
@@ -1623,6 +1628,22 @@
barrier(state, 0, y) |= L;
barrier(state, state->width-1, y) |= R;
}
+ } else {
+ /*
+ * We check whether this is de-facto a non-wrapping game
+ * despite the parameters, in case we were passed the
+ * description of a non-wrapping game. This is so that we
+ * can change some aspects of the UI behaviour.
+ */
+ state->wrapping = FALSE;
+ for (x = 0; x < state->width; x++)
+ if (!(barrier(state, x, 0) & U) ||
+ !(barrier(state, x, state->height-1) & D))
+ state->wrapping = TRUE;
+ for (y = 0; y < state->width; y++)
+ if (!(barrier(state, 0, y) & L) ||
+ !(barrier(state, state->width-1, y) & R))
+ state->wrapping = TRUE;
}
/*
@@ -1644,30 +1665,20 @@
if (barrier(state, x, y) & dir2)
corner = TRUE;
- x1 = x + X(dir), y1 = y + Y(dir);
- if (x1 >= 0 && x1 < state->width &&
- y1 >= 0 && y1 < state->height &&
- (barrier(state, x1, y1) & dir2))
+ OFFSET(x1, y1, x, y, dir, state);
+ if (barrier(state, x1, y1) & dir2)
corner = TRUE;
- x2 = x + X(dir2), y2 = y + Y(dir2);
- if (x2 >= 0 && x2 < state->width &&
- y2 >= 0 && y2 < state->height &&
- (barrier(state, x2, y2) & dir))
+ OFFSET(x2, y2, x, y, dir2, state);
+ if (barrier(state, x2, y2) & dir)
corner = TRUE;
if (corner) {
barrier(state, x, y) |= (dir << 4);
- if (x1 >= 0 && x1 < state->width &&
- y1 >= 0 && y1 < state->height)
- barrier(state, x1, y1) |= (A(dir) << 4);
- if (x2 >= 0 && x2 < state->width &&
- y2 >= 0 && y2 < state->height)
- barrier(state, x2, y2) |= (C(dir) << 4);
- x3 = x + X(dir) + X(dir2), y3 = y + Y(dir) + Y(dir2);
- if (x3 >= 0 && x3 < state->width &&
- y3 >= 0 && y3 < state->height)
- barrier(state, x3, y3) |= (F(dir) << 4);
+ barrier(state, x1, y1) |= (A(dir) << 4);
+ barrier(state, x2, y2) |= (C(dir) << 4);
+ OFFSET(x3, y3, x1, y1, dir2, state);
+ barrier(state, x3, y3) |= (F(dir) << 4);
}
}
}
@@ -1683,8 +1694,6 @@
ret = snew(game_state);
ret->width = state->width;
ret->height = state->height;
- ret->cx = state->cx;
- ret->cy = state->cy;
ret->wrapping = state->wrapping;
ret->completed = state->completed;
ret->used_solve = state->used_solve;
@@ -1748,7 +1757,7 @@
* completed - just call this function and see whether every square
* is marked active.
*/
-static unsigned char *compute_active(game_state *state)
+static unsigned char *compute_active(game_state *state, int cx, int cy)
{
unsigned char *active;
tree234 *todo;
@@ -1762,8 +1771,8 @@
* xyd_cmp and just store direction 0 every time.
*/
todo = newtree234(xyd_cmp_nc);
- index(state, active, state->cx, state->cy) = ACTIVE;
- add234(todo, new_xyd(state->cx, state->cy, 0));
+ index(state, active, cx, cy) = ACTIVE;
+ add234(todo, new_xyd(cx, cy, 0));
while ( (xyd = delpos234(todo, 0)) != NULL) {
int x1, y1, d1, x2, y2, d2;
@@ -1799,6 +1808,8 @@
}
struct game_ui {
+ int org_x, org_y; /* origin */
+ int cx, cy; /* source tile (game coordinates) */
int cur_x, cur_y;
int cur_visible;
random_state *rs; /* used for jumbling */
@@ -1809,8 +1820,9 @@
void *seed;
int seedsize;
game_ui *ui = snew(game_ui);
- ui->cur_x = state->width / 2;
- ui->cur_y = state->height / 2;
+ ui->org_x = ui->org_y = 0;
+ ui->cur_x = ui->cx = state->width / 2;
+ ui->cur_y = ui->cy = state->height / 2;
ui->cur_visible = FALSE;
get_random_seed(&seed, &seedsize);
ui->rs = random_init(seed, seedsize);
@@ -1833,7 +1845,9 @@
{
game_state *ret, *nullret;
int tx, ty, orig;
+ int shift = button & MOD_SHFT, ctrl = button & MOD_CTRL;
+ button &= ~MOD_MASK;
nullret = NULL;
if (button == LEFT_BUTTON ||
@@ -1856,23 +1870,44 @@
ty = y / TILE_SIZE;
if (tx >= state->width || ty >= state->height)
return nullret;
+ /* Transform from physical to game coords */
+ tx = (tx + ui->org_x) % state->width;
+ ty = (ty + ui->org_y) % state->height;
if (x % TILE_SIZE >= TILE_SIZE - TILE_BORDER ||
y % TILE_SIZE >= TILE_SIZE - TILE_BORDER)
return nullret;
} else if (button == CURSOR_UP || button == CURSOR_DOWN ||
button == CURSOR_RIGHT || button == CURSOR_LEFT) {
- if (button == CURSOR_UP && ui->cur_y > 0)
- ui->cur_y--;
- else if (button == CURSOR_DOWN && ui->cur_y < state->height-1)
- ui->cur_y++;
- else if (button == CURSOR_LEFT && ui->cur_x > 0)
- ui->cur_x--;
- else if (button == CURSOR_RIGHT && ui->cur_x < state->width-1)
- ui->cur_x++;
- else
- return nullret; /* no cursor movement */
- ui->cur_visible = TRUE;
- return state; /* UI activity has occurred */
+ int dir;
+ switch (button) {
+ case CURSOR_UP: dir = U; break;
+ case CURSOR_DOWN: dir = D; break;
+ case CURSOR_LEFT: dir = L; break;
+ case CURSOR_RIGHT: dir = R; break;
+ default: return nullret;
+ }
+ if (shift) {
+ /*
+ * Move origin.
+ */
+ if (state->wrapping) {
+ OFFSET(ui->org_x, ui->org_y, ui->org_x, ui->org_y, dir, state);
+ } else return nullret; /* disallowed for non-wrapping grids */
+ }
+ if (ctrl) {
+ /*
+ * Change source tile.
+ */
+ OFFSET(ui->cx, ui->cy, ui->cx, ui->cy, dir, state);
+ }
+ if (!shift && !ctrl) {
+ /*
+ * Move keyboard cursor.
+ */
+ OFFSET(ui->cur_x, ui->cur_y, ui->cur_x, ui->cur_y, dir, state);
+ ui->cur_visible = TRUE;
+ }
+ return state; /* UI activity has occurred */
} else if (button == 'a' || button == 's' || button == 'd' ||
button == 'A' || button == 'S' || button == 'D') {
tx = ui->cur_x;
@@ -1961,7 +1996,7 @@
* Check whether the game has been completed.
*/
{
- unsigned char *active = compute_active(ret);
+ unsigned char *active = compute_active(ret, ui->cx, ui->cy);
int x1, y1;
int complete = TRUE;
@@ -1989,6 +2024,7 @@
struct game_drawstate {
int started;
int width, height;
+ int org_x, org_y;
unsigned char *visible;
};
@@ -1999,6 +2035,7 @@
ds->started = FALSE;
ds->width = state->width;
ds->height = state->height;
+ ds->org_x = ds->org_y = -1;
ds->visible = snewn(state->width * state->height, unsigned char);
memset(ds->visible, 0xFF, state->width * state->height);
@@ -2096,7 +2133,11 @@
draw_rect(fe, mx, my, dx, dy, colour);
}
-static void draw_barrier_corner(frontend *fe, int x, int y, int dir, int phase)
+/*
+ * draw_barrier_corner() and draw_barrier() are passed physical coords
+ */
+static void draw_barrier_corner(frontend *fe, int x, int y, int dir, int phase,
+ int barrier)
{
int bx = WINDOW_OFFSET + TILE_SIZE * x;
int by = WINDOW_OFFSET + TILE_SIZE * y;
@@ -2113,18 +2154,19 @@
if (phase == 0) {
draw_rect_coords(fe, bx+x1, by+y1,
bx+x1-TILE_BORDER*dx, by+y1-(TILE_BORDER-1)*dy,
- COL_WIRE);
+ barrier ? COL_WIRE : COL_BACKGROUND);
draw_rect_coords(fe, bx+x1, by+y1,
bx+x1-(TILE_BORDER-1)*dx, by+y1-TILE_BORDER*dy,
- COL_WIRE);
+ barrier ? COL_WIRE : COL_BACKGROUND);
} else {
draw_rect_coords(fe, bx+x1, by+y1,
bx+x1-(TILE_BORDER-1)*dx, by+y1-(TILE_BORDER-1)*dy,
- COL_BARRIER);
+ barrier ? COL_BARRIER : COL_BORDER);
}
}
-static void draw_barrier(frontend *fe, int x, int y, int dir, int phase)
+static void draw_barrier(frontend *fe, int x, int y, int dir, int phase,
+ int barrier)
{
int bx = WINDOW_OFFSET + TILE_SIZE * x;
int by = WINDOW_OFFSET + TILE_SIZE * y;
@@ -2136,14 +2178,19 @@
h = (Y(dir) ? TILE_BORDER : TILE_SIZE - TILE_BORDER);
if (phase == 0) {
- draw_rect(fe, bx+x1-X(dir), by+y1-Y(dir), w, h, COL_WIRE);
+ draw_rect(fe, bx+x1-X(dir), by+y1-Y(dir), w, h,
+ barrier ? COL_WIRE : COL_BACKGROUND);
} else {
- draw_rect(fe, bx+x1, by+y1, w, h, COL_BARRIER);
+ draw_rect(fe, bx+x1, by+y1, w, h,
+ barrier ? COL_BARRIER : COL_BORDER);
}
}
-static void draw_tile(frontend *fe, game_state *state, int x, int y, int tile,
- float angle, int cursor)
+/*
+ * draw_tile() is passed physical coordinates
+ */
+static void draw_tile(frontend *fe, game_state *state, game_drawstate *ds,
+ int x, int y, int tile, int src, float angle, int cursor)
{
int bx = WINDOW_OFFSET + TILE_SIZE * x;
int by = WINDOW_OFFSET + TILE_SIZE * y;
@@ -2238,7 +2285,7 @@
* otherwise not at all.
*/
col = -1;
- if (x == state->cx && y == state->cy)
+ if (src)
col = COL_WIRE;
else if (COUNT(tile) == 1) {
col = (tile & ACTIVE ? COL_POWERED : COL_ENDPOINT);
@@ -2279,7 +2326,7 @@
if (ox < 0 || ox >= state->width || oy < 0 || oy >= state->height)
continue;
- if (!(tile(state, ox, oy) & F(dir)))
+ if (!(tile(state, GX(ox), GY(oy)) & F(dir)))
continue;
px = bx + (int)(dx>0 ? TILE_SIZE + TILE_BORDER - 1 : dx<0 ? 0 : cx);
@@ -2315,11 +2362,11 @@
*/
for (phase = 0; phase < 2; phase++) {
for (dir = 1; dir < 0x10; dir <<= 1)
- if (barrier(state, x, y) & (dir << 4))
- draw_barrier_corner(fe, x, y, dir << 4, phase);
+ if (barrier(state, GX(x), GY(y)) & (dir << 4))
+ draw_barrier_corner(fe, x, y, dir << 4, phase, TRUE);
for (dir = 1; dir < 0x10; dir <<= 1)
- if (barrier(state, x, y) & dir)
- draw_barrier(fe, x, y, dir, phase);
+ if (barrier(state, GX(x), GY(y)) & dir)
+ draw_barrier(fe, x, y, dir, phase, TRUE);
}
draw_update(fe, bx, by, TILE_SIZE+TILE_BORDER, TILE_SIZE+TILE_BORDER);
@@ -2328,17 +2375,14 @@
static void game_redraw(frontend *fe, game_drawstate *ds, game_state *oldstate,
game_state *state, int dir, game_ui *ui, float t, float ft)
{
- int x, y, tx, ty, frame, last_rotate_dir;
+ int x, y, tx, ty, frame, last_rotate_dir, moved_origin = FALSE;
unsigned char *active;
float angle = 0.0;
/*
- * Clear the screen and draw the exterior barrier lines if this
- * is our first call.
+ * Clear the screen if this is our first call.
*/
if (!ds->started) {
- int phase;
-
ds->started = TRUE;
draw_rect(fe, 0, 0,
@@ -2345,40 +2389,46 @@
WINDOW_OFFSET * 2 + TILE_SIZE * state->width + TILE_BORDER,
WINDOW_OFFSET * 2 + TILE_SIZE * state->height + TILE_BORDER,
COL_BACKGROUND);
+
+ }
+
+ /*
+ * If the origin has changed, we need to redraw the exterior
+ * barrier lines.
+ */
+ if (ui->org_x != ds->org_x || ui->org_y != ds->org_y) {
+ int phase;
+
+ ds->org_x = ui->org_x;
+ ds->org_y = ui->org_y;
+ moved_origin = TRUE;
+
draw_update(fe, 0, 0,
WINDOW_OFFSET*2 + TILE_SIZE*state->width + TILE_BORDER,
WINDOW_OFFSET*2 + TILE_SIZE*state->height + TILE_BORDER);
-
+
for (phase = 0; phase < 2; phase++) {
for (x = 0; x < ds->width; x++) {
- if (barrier(state, x, 0) & UL)
- draw_barrier_corner(fe, x, -1, LD, phase);
- if (barrier(state, x, 0) & RU)
- draw_barrier_corner(fe, x, -1, DR, phase);
- if (barrier(state, x, 0) & U)
- draw_barrier(fe, x, -1, D, phase);
- if (barrier(state, x, ds->height-1) & DR)
- draw_barrier_corner(fe, x, ds->height, RU, phase);
- if (barrier(state, x, ds->height-1) & LD)
- draw_barrier_corner(fe, x, ds->height, UL, phase);
- if (barrier(state, x, ds->height-1) & D)
- draw_barrier(fe, x, ds->height, U, phase);
+ int ub = barrier(state, GX(x), GY(0));
+ int db = barrier(state, GX(x), GY(ds->height-1));
+ draw_barrier_corner(fe, x, -1, LD, phase, ub & UL);
+ draw_barrier_corner(fe, x, -1, DR, phase, ub & RU);
+ draw_barrier(fe, x, -1, D, phase, ub & U);
+ draw_barrier_corner(fe, x, ds->height, RU, phase, db & DR);
+ draw_barrier_corner(fe, x, ds->height, UL, phase, db & LD);
+ draw_barrier(fe, x, ds->height, U, phase, db & D);
}
for (y = 0; y < ds->height; y++) {
- if (barrier(state, 0, y) & UL)
- draw_barrier_corner(fe, -1, y, RU, phase);
- if (barrier(state, 0, y) & LD)
- draw_barrier_corner(fe, -1, y, DR, phase);
- if (barrier(state, 0, y) & L)
- draw_barrier(fe, -1, y, R, phase);
- if (barrier(state, ds->width-1, y) & RU)
- draw_barrier_corner(fe, ds->width, y, UL, phase);
- if (barrier(state, ds->width-1, y) & DR)
- draw_barrier_corner(fe, ds->width, y, LD, phase);
- if (barrier(state, ds->width-1, y) & R)
- draw_barrier(fe, ds->width, y, L, phase);
+ int lb = barrier(state, GX(0), GY(y));
+ int rb = barrier(state, GX(ds->width-1), GY(y));
+ draw_barrier_corner(fe, -1, y, RU, phase, lb & UL);
+ draw_barrier_corner(fe, -1, y, DR, phase, lb & LD);
+ draw_barrier(fe, -1, y, R, phase, lb & L);
+ draw_barrier_corner(fe, ds->width, y, UL, phase, rb & RU);
+ draw_barrier_corner(fe, ds->width, y, LD, phase, rb & DR);
+ draw_barrier(fe, ds->width, y, L, phase, rb & R);
}
}
}
@@ -2409,11 +2459,16 @@
/*
* Draw any tile which differs from the way it was last drawn.
*/
- active = compute_active(state);
+ active = compute_active(state, ui->cx, ui->cy);
for (x = 0; x < ds->width; x++)
for (y = 0; y < ds->height; y++) {
- unsigned char c = tile(state, x, y) | index(state, active, x, y);
+ unsigned char c = tile(state, GX(x), GY(y)) |
+ index(state, active, GX(x), GY(y));
+ int is_src = GX(x) == ui->cx && GY(y) == ui->cy;
+ int is_anim = GX(x) == tx && GY(y) == ty;
+ int is_cursor = ui->cur_visible &&
+ GX(x) == ui->cur_x && GY(y) == ui->cur_y;
/*
* In a completion flash, we adjust the LOCKED bit
@@ -2421,9 +2476,10 @@
* the frame number.
*/
if (frame >= 0) {
+ int rcx = RX(ui->cx), rcy = RY(ui->cy);
int xdist, ydist, dist;
- xdist = (x < state->cx ? state->cx - x : x - state->cx);
- ydist = (y < state->cy ? state->cy - y : y - state->cy);
+ xdist = (x < rcx ? rcx - x : x - rcx);
+ ydist = (y < rcy ? rcy - y : y - rcy);
dist = (xdist > ydist ? xdist : ydist);
if (frame >= dist && frame < dist+4) {
@@ -2433,15 +2489,13 @@
}
}
- if (index(state, ds->visible, x, y) != c ||
+ if (moved_origin ||
+ index(state, ds->visible, x, y) != c ||
index(state, ds->visible, x, y) == 0xFF ||
- (x == tx && y == ty) ||
- (ui->cur_visible && x == ui->cur_x && y == ui->cur_y)) {
- draw_tile(fe, state, x, y, c,
- (x == tx && y == ty ? angle : 0.0F),
- (ui->cur_visible && x == ui->cur_x && y == ui->cur_y));
- if ((x == tx && y == ty) ||
- (ui->cur_visible && x == ui->cur_x && y == ui->cur_y))
+ is_src || is_anim || is_cursor) {
+ draw_tile(fe, state, ds, x, y, c,
+ is_src, (is_anim ? angle : 0.0F), is_cursor);
+ if (is_src || is_anim || is_cursor)
index(state, ds->visible, x, y) = 0xFF;
else
index(state, ds->visible, x, y) = c;
@@ -2505,16 +2559,11 @@
*/
if (!oldstate->completed && newstate->completed &&
!oldstate->used_solve && !newstate->used_solve) {
- int size;
- size = 0;
- if (size < newstate->cx+1)
- size = newstate->cx+1;
- if (size < newstate->cy+1)
- size = newstate->cy+1;
- if (size < newstate->width - newstate->cx)
- size = newstate->width - newstate->cx;
- if (size < newstate->height - newstate->cy)
- size = newstate->height - newstate->cy;
+ int size = 0;
+ if (size < newstate->width)
+ size = newstate->width;
+ if (size < newstate->height)
+ size = newstate->height;
return FLASH_FRAME * (size+4);
}
--- a/netslide.c
+++ b/netslide.c
@@ -1006,6 +1006,8 @@
int n, dx, dy;
game_state *ret;
+ button &= ~MOD_MASK;
+
if (button != LEFT_BUTTON && button != RIGHT_BUTTON)
return NULL;
--- a/pattern.c
+++ b/pattern.c
@@ -769,6 +769,8 @@
{
game_state *ret;
+ button &= ~MOD_MASK;
+
x = FROMCOORD(from->w, x);
y = FROMCOORD(from->h, y);
--- a/puzzles.but
+++ b/puzzles.but
@@ -320,6 +320,21 @@
also unlock it again, but while it's locked you can't accidentally
turn it.
+The following controls are not necessary to complete the game, but may
+be useful:
+
+\dt \e{Shift grid}: Shift + arrow keys
+
+\dd On grids that wrap, you can move the origin of the grid, so that
+tiles that were on opposite sides of the grid can be seen together.
+
+\dt \e{Move centre}: Ctrl + arrow keys
+
+\dd You can change which tile is used as the source of highlighting.
+(It doesn't ultimately matter which tile this is, as every tile will
+be connected to every other tile in a correct solution, but it may be
+helpful in the intermediate stages of solving the puzzle.)
+
\dt \e{Jumble tiles}: \q{J} key
\dd This key turns all tiles that are not locked to random
--- a/puzzles.h
+++ b/puzzles.h
@@ -32,7 +32,10 @@
CURSOR_LEFT,
CURSOR_RIGHT,
- MOD_NUM_KEYPAD = 0x40000000
+ MOD_CTRL = 0x10000000,
+ MOD_SHFT = 0x20000000,
+ MOD_NUM_KEYPAD = 0x40000000,
+ MOD_MASK = 0x70000000 /* mask for all modifiers */
};
#define IS_MOUSE_DOWN(m) ( (unsigned)((m) - LEFT_BUTTON) <= \
--- a/rect.c
+++ b/rect.c
@@ -2127,6 +2127,8 @@
int startdrag = FALSE, enddrag = FALSE, active = FALSE;
game_state *ret;
+ button &= ~MOD_MASK;
+
if (button == LEFT_BUTTON) {
startdrag = TRUE;
} else if (button == LEFT_RELEASE) {
--- a/sixteen.c
+++ b/sixteen.c
@@ -572,6 +572,7 @@
int dx, dy, tx, ty, n;
game_state *ret;
+ button &= ~MOD_MASK;
if (button != LEFT_BUTTON && button != RIGHT_BUTTON)
return NULL;
--- a/solo.c
+++ b/solo.c
@@ -1828,7 +1828,7 @@
int tx, ty;
game_state *ret;
- button &= ~MOD_NUM_KEYPAD; /* we treat this the same as normal */
+ button &= ~MOD_MASK;
tx = (x + TILE_SIZE - BORDER) / TILE_SIZE - 1;
ty = (y + TILE_SIZE - BORDER) / TILE_SIZE - 1;
--- a/twiddle.c
+++ b/twiddle.c
@@ -594,6 +594,8 @@
game_state *ret;
int dir;
+ button = button & (~MOD_MASK | MOD_NUM_KEYPAD);
+
if (button == LEFT_BUTTON || button == RIGHT_BUTTON) {
/*
* Determine the coordinates of the click. We offset by n-1
--- a/windows.c
+++ b/windows.c
@@ -1221,31 +1221,35 @@
case WM_KEYDOWN:
{
int key = -1;
+ BYTE keystate[256];
+ int r = GetKeyboardState(keystate);
+ int shift = (r && (keystate[VK_SHIFT] & 0x80)) ? MOD_SHFT : 0;
+ int ctrl = (r && (keystate[VK_CONTROL] & 0x80)) ? MOD_CTRL : 0;
switch (wParam) {
case VK_LEFT:
if (!(lParam & 0x01000000))
key = MOD_NUM_KEYPAD | '4';
- else
- key = CURSOR_LEFT;
+ else
+ key = shift | ctrl | CURSOR_LEFT;
break;
case VK_RIGHT:
if (!(lParam & 0x01000000))
key = MOD_NUM_KEYPAD | '6';
- else
- key = CURSOR_RIGHT;
+ else
+ key = shift | ctrl | CURSOR_RIGHT;
break;
case VK_UP:
if (!(lParam & 0x01000000))
key = MOD_NUM_KEYPAD | '8';
- else
- key = CURSOR_UP;
+ else
+ key = shift | ctrl | CURSOR_UP;
break;
case VK_DOWN:
if (!(lParam & 0x01000000))
key = MOD_NUM_KEYPAD | '2';
- else
- key = CURSOR_DOWN;
+ else
+ key = shift | ctrl | CURSOR_DOWN;
break;
/*
* Diagonal keys on the numeric keypad.