minesweeper/src/main/java/org/apidesign/demo/minesweeper/MinesModel.java
author Jaroslav Tulach <jtulach@netbeans.org>
Thu, 17 Jul 2014 08:23:25 +0200
changeset 174 a57b2414b855
parent 166 e6667c8206fc
child 176 1e482b09b814
permissions -rw-r--r--
One should not be able to win the game by marking all the squares
jtulach@63
     1
/**
jtulach@63
     2
 * The MIT License (MIT)
jtulach@63
     3
 *
jtulach@63
     4
 * Copyright (C) 2013 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jtulach@63
     5
 *
jtulach@63
     6
 * Permission is hereby granted, free of charge, to any person obtaining a copy
jtulach@63
     7
 * of this software and associated documentation files (the "Software"), to deal
jtulach@63
     8
 * in the Software without restriction, including without limitation the rights
jtulach@63
     9
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
jtulach@63
    10
 * copies of the Software, and to permit persons to whom the Software is
jtulach@63
    11
 * furnished to do so, subject to the following conditions:
jtulach@63
    12
 *
jtulach@63
    13
 * The above copyright notice and this permission notice shall be included in
jtulach@63
    14
 * all copies or substantial portions of the Software.
jtulach@63
    15
 *
jtulach@63
    16
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
jtulach@63
    17
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
jtulach@63
    18
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
jtulach@63
    19
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
jtulach@63
    20
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
jtulach@63
    21
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
jtulach@63
    22
 * THE SOFTWARE.
jtulach@63
    23
 */
jtulach@63
    24
package org.apidesign.demo.minesweeper;
jtulach@63
    25
jtulach@63
    26
import java.util.ArrayList;
jtulach@63
    27
import java.util.List;
jtulach@63
    28
import java.util.Random;
jtulach@63
    29
import net.java.html.json.ComputedProperty;
jtulach@63
    30
import net.java.html.json.Function;
jtulach@63
    31
import net.java.html.json.Model;
jtulach@63
    32
import net.java.html.json.ModelOperation;
jtulach@63
    33
import net.java.html.json.Property;
jtulach@79
    34
import net.java.html.sound.AudioClip;
jtulach@63
    35
jtulach@63
    36
/** Model of the mine field.
jtulach@63
    37
 */
jtulach@63
    38
@Model(className = "Mines", properties = {
jtulach@63
    39
    @Property(name = "state", type = MinesModel.GameState.class),
jtulach@63
    40
    @Property(name = "rows", type = Row.class, array = true),
jtulach@63
    41
})
jtulach@90
    42
public final class MinesModel {
jtulach@63
    43
    enum GameState {
jaroslav@164
    44
        IN_PROGRESS, MARKING_MINE, WON, LOST;
jtulach@63
    45
    }
jtulach@63
    46
    
jaroslav@166
    47
    @ComputedProperty static String gameStyle(GameState state) {
jaroslav@166
    48
        return state == GameState.MARKING_MINE ? "MARKING" : null;
jaroslav@166
    49
    }
jaroslav@166
    50
    
jtulach@63
    51
    @Model(className = "Row", properties = {
jtulach@63
    52
        @Property(name = "columns", type = Square.class, array = true)
jtulach@63
    53
    })
jtulach@63
    54
    static class RowModel {
jtulach@63
    55
    }
jtulach@63
    56
jtulach@63
    57
    @Model(className = "Square", properties = {
jtulach@63
    58
        @Property(name = "state", type = SquareType.class),
jtulach@63
    59
        @Property(name = "mine", type = boolean.class)
jtulach@63
    60
    })
jtulach@63
    61
    static class SquareModel {
jtulach@65
    62
        @ComputedProperty static String style(SquareType state) {
jtulach@65
    63
            return state == null ? null : state.toString();
jtulach@65
    64
        }
jtulach@63
    65
    }
jtulach@63
    66
    
jtulach@63
    67
    enum SquareType {
jtulach@63
    68
        N_0, N_1, N_2, N_3, N_4, N_5, N_6, N_7, N_8,
jaroslav@164
    69
        UNKNOWN, EXPLOSION, DISCOVERED, MARKED;
jtulach@64
    70
        
jtulach@64
    71
        final boolean isVisible() {
jtulach@64
    72
            return name().startsWith("N_");
jtulach@64
    73
        }
jtulach@64
    74
jtulach@69
    75
        final SquareType moreBombsAround() {
jtulach@68
    76
            switch (this) {
jtulach@68
    77
                case EXPLOSION:
jtulach@68
    78
                case UNKNOWN:
jtulach@68
    79
                case DISCOVERED:
jtulach@68
    80
                case N_8:
jtulach@68
    81
                    return this;
jtulach@64
    82
            }
jtulach@64
    83
            return values()[ordinal() + 1];
jtulach@64
    84
        }
jtulach@63
    85
    }
jtulach@63
    86
    
jtulach@76
    87
    @ComputedProperty static boolean fieldShowing(GameState state) {
jtulach@76
    88
        return state != null;
jtulach@76
    89
    }
jtulach@76
    90
    
jtulach@76
    91
    @Function static void showHelp(Mines model) {
jtulach@76
    92
        model.setState(null);
jtulach@76
    93
    }
jtulach@76
    94
    
jtulach@70
    95
    @Function static void smallGame(Mines model) {
jtulach@70
    96
        model.init(5, 5, 5);
jtulach@70
    97
    }
jtulach@70
    98
    @Function static void normalGame(Mines model) {
jtulach@70
    99
        model.init(10, 10, 10);
jtulach@70
   100
    }
jtulach@70
   101
    
jtulach@72
   102
    @Function static void giveUp(Mines model) {
jtulach@72
   103
        showAllBombs(model, SquareType.EXPLOSION);
jaroslav@152
   104
        model.setState(GameState.LOST);
jtulach@72
   105
    }
jtulach@72
   106
    
jaroslav@164
   107
    @Function static void markMine(Mines model) {
jaroslav@164
   108
        if (model.getState() == GameState.IN_PROGRESS) {
jaroslav@164
   109
            model.setState(GameState.MARKING_MINE);
jaroslav@164
   110
        }
jaroslav@164
   111
    }
jaroslav@164
   112
    
jtulach@63
   113
    @ModelOperation static void init(Mines model, int width, int height, int mines) {
jtulach@107
   114
        List<Row> rows = model.getRows();
jtulach@107
   115
        if (rows.size() != height || rows.get(0).getColumns().size() != width) {
jtulach@107
   116
            rows = new ArrayList<Row>(height);
jtulach@107
   117
            for (int y = 0; y < height; y++) {
jtulach@107
   118
                Square[] columns = new Square[width];
jtulach@107
   119
                for (int x = 0; x < width; x++) {
jtulach@107
   120
                    columns[x] = new Square(SquareType.UNKNOWN, false);
jtulach@107
   121
                }
jtulach@107
   122
                rows.add(new Row(columns));
jtulach@63
   123
            }
jtulach@107
   124
        } else {
jtulach@107
   125
            for (Row row : rows) {
jtulach@107
   126
                for (Square sq : row.getColumns()) {
jtulach@107
   127
                    sq.setState(SquareType.UNKNOWN);
jtulach@107
   128
                    sq.setMine(false);
jtulach@107
   129
                }
jtulach@107
   130
            }
jtulach@63
   131
        }
jtulach@63
   132
        
jtulach@63
   133
        Random r = new Random();
jtulach@63
   134
        while (mines > 0) {
jtulach@63
   135
            int x = r.nextInt(width);
jtulach@63
   136
            int y = r.nextInt(height);
jtulach@63
   137
            final Square s = rows.get(y).getColumns().get(x);
jtulach@63
   138
            if (s.isMine()) {
jtulach@63
   139
                continue;
jtulach@63
   140
            }
jtulach@63
   141
            s.setMine(true);
jtulach@63
   142
            mines--;
jtulach@63
   143
        }
jtulach@63
   144
jtulach@63
   145
        model.setState(GameState.IN_PROGRESS);
jtulach@107
   146
        if (rows != model.getRows()) {
jtulach@107
   147
            model.getRows().clear();
jtulach@107
   148
            model.getRows().addAll(rows);
jtulach@107
   149
        }
jtulach@63
   150
    }
jtulach@63
   151
    
jtulach@64
   152
    @ModelOperation static void computeMines(Mines model) {
jtulach@64
   153
        List<Integer> xBombs = new ArrayList<Integer>();
jtulach@64
   154
        List<Integer> yBombs = new ArrayList<Integer>();
jtulach@64
   155
        final List<Row> rows = model.getRows();
jtulach@66
   156
        boolean emptyHidden = false;
jtulach@69
   157
        SquareType[][] arr = new SquareType[rows.size()][];
jtulach@64
   158
        for (int y = 0; y < rows.size(); y++) {
jtulach@64
   159
            final List<Square> columns = rows.get(y).getColumns();
jtulach@69
   160
            arr[y] = new SquareType[columns.size()];
jtulach@64
   161
            for (int x = 0; x < columns.size(); x++) {
jtulach@64
   162
                Square sq = columns.get(x);
jtulach@64
   163
                if (sq.isMine()) {
jtulach@64
   164
                    xBombs.add(x);
jtulach@64
   165
                    yBombs.add(y);
jtulach@64
   166
                }
jtulach@64
   167
                if (sq.getState().isVisible()) {
jtulach@69
   168
                    arr[y][x] = SquareType.N_0;
jtulach@66
   169
                } else {
jtulach@66
   170
                    if (!sq.isMine()) {
jtulach@66
   171
                        emptyHidden = true;
jtulach@66
   172
                    }
jtulach@64
   173
                }
jtulach@64
   174
            }
jtulach@64
   175
        }
jtulach@64
   176
        for (int i = 0; i < xBombs.size(); i++) {
jtulach@64
   177
            int x = xBombs.get(i);
jtulach@64
   178
            int y = yBombs.get(i);
jtulach@64
   179
            
jtulach@69
   180
            incrementAround(arr, x, y);
jtulach@69
   181
        }
jtulach@69
   182
        for (int y = 0; y < rows.size(); y++) {
jtulach@69
   183
            final List<Square> columns = rows.get(y).getColumns();
jtulach@69
   184
            for (int x = 0; x < columns.size(); x++) {
jtulach@69
   185
                Square sq = columns.get(x);
jtulach@69
   186
                final SquareType newState = arr[y][x];
jtulach@69
   187
                if (newState != null && newState != sq.getState()) {
jtulach@69
   188
                    sq.setState(newState);
jtulach@69
   189
                }
jtulach@69
   190
            }
jtulach@64
   191
        }
jtulach@66
   192
        
jtulach@66
   193
        if (!emptyHidden) {
jtulach@66
   194
            model.setState(GameState.WON);
jtulach@68
   195
            showAllBombs(model, SquareType.DISCOVERED);
jtulach@153
   196
            AudioClip applause = AudioClip.create("applause.mp3");
jaroslav@136
   197
            applause.play();
jtulach@66
   198
        }
jtulach@64
   199
    }
jtulach@64
   200
    
jtulach@69
   201
    private static void incrementAround(SquareType[][] arr, int x, int y) {
jtulach@69
   202
        incrementAt(arr, x - 1, y - 1);
jtulach@69
   203
        incrementAt(arr, x - 1, y);
jtulach@69
   204
        incrementAt(arr, x - 1, y + 1);
jtulach@64
   205
jtulach@69
   206
        incrementAt(arr, x + 1, y - 1);
jtulach@69
   207
        incrementAt(arr, x + 1, y);
jtulach@69
   208
        incrementAt(arr, x + 1, y + 1);
jtulach@64
   209
        
jtulach@69
   210
        incrementAt(arr, x, y - 1);
jtulach@69
   211
        incrementAt(arr, x, y + 1);
jtulach@64
   212
    }
jtulach@64
   213
    
jtulach@69
   214
    private static void incrementAt(SquareType[][] arr, int x, int y) {
jtulach@69
   215
        if (y >= 0 && y < arr.length) {
jtulach@69
   216
            SquareType[] r = arr[y];
jtulach@69
   217
            if (x >= 0 && x < r.length) {
jtulach@69
   218
                SquareType sq = r[x];
jtulach@69
   219
                if (sq != null) {
jtulach@69
   220
                    r[x] = sq.moreBombsAround();
jtulach@69
   221
                }
jtulach@64
   222
            }
jtulach@64
   223
        }
jtulach@64
   224
    }
jtulach@64
   225
    
jtulach@68
   226
    static void showAllBombs(Mines model, SquareType state) {
jtulach@63
   227
        for (Row row : model.getRows()) {
jtulach@63
   228
            for (Square square : row.getColumns()) {
jtulach@63
   229
                if (square.isMine()) {
jtulach@68
   230
                    square.setState(state);
jtulach@63
   231
                }
jtulach@63
   232
            }
jtulach@63
   233
        }
jtulach@63
   234
    }
jtulach@63
   235
    
jtulach@63
   236
    @Function static void click(Mines model, Square data) {
jaroslav@164
   237
        if (model.getState() == GameState.MARKING_MINE) {
jaroslav@164
   238
            if (data.getState() == SquareType.UNKNOWN) {
jaroslav@164
   239
                data.setState(SquareType.MARKED);
jaroslav@165
   240
                if (allMarked(model)) {
jaroslav@165
   241
                    model.setState(GameState.WON);
jaroslav@165
   242
                    return;
jaroslav@165
   243
                }
jaroslav@164
   244
            }
jaroslav@164
   245
            model.setState(GameState.IN_PROGRESS);
jaroslav@164
   246
            return;
jaroslav@164
   247
        }
jtulach@63
   248
        if (model.getState() != GameState.IN_PROGRESS) {
jtulach@63
   249
            return;
jtulach@63
   250
        }
jaroslav@164
   251
        if (data.getState() == SquareType.MARKED) {
jaroslav@164
   252
            data.setState(SquareType.UNKNOWN);
jtulach@174
   253
            if (allMarked(model)) {
jtulach@174
   254
                model.setState(GameState.WON);
jtulach@174
   255
            }
jaroslav@164
   256
            return;
jaroslav@164
   257
        }
jaroslav@152
   258
        if (data.getState() != SquareType.UNKNOWN) {
jaroslav@152
   259
            return;
jaroslav@152
   260
        }
jaroslav@152
   261
        if (data.isMine()) {
jaroslav@152
   262
            Square fair = atLeastOnePlaceWhereBombCantBe(model);
jaroslav@152
   263
            if (fair == null) {
jaroslav@152
   264
                if (placeBombElseWhere(model, data)) {
jaroslav@152
   265
                    cleanedUp(model, data);
jaroslav@152
   266
                    return;
jtulach@63
   267
                }
jaroslav@152
   268
            }
jaroslav@152
   269
            explosion(model);
jaroslav@152
   270
        } else {
jaroslav@152
   271
            Square takeFrom = tryStealBomb(model, data);
jaroslav@152
   272
            if (takeFrom != null) {
jaroslav@152
   273
                final Square fair = atLeastOnePlaceWhereBombCantBe(model);
jaroslav@152
   274
                if (fair != null) {
jaroslav@152
   275
                    takeFrom.setMine(false);
jaroslav@152
   276
                    data.setMine(true);
jaroslav@152
   277
                    explosion(model);
jaroslav@152
   278
                    return;
jaroslav@152
   279
                }
jaroslav@152
   280
            }
jaroslav@152
   281
            cleanedUp(model, data);
jtulach@63
   282
        }
jtulach@63
   283
    }
jaroslav@152
   284
jaroslav@152
   285
    private static void cleanedUp(Mines model, Square data) {
jaroslav@152
   286
        AudioClip touch = AudioClip.create("move.mp3");
jaroslav@152
   287
        touch.play();
jaroslav@152
   288
        expandKnown(model, data);
jaroslav@152
   289
        model.computeMines();
jaroslav@152
   290
    }
jaroslav@152
   291
jaroslav@152
   292
    private static void explosion(Mines model) {
jaroslav@152
   293
        showAllBombs(model, SquareType.EXPLOSION);
jaroslav@152
   294
        model.setState(GameState.LOST);
jtulach@153
   295
        AudioClip oops = AudioClip.create("oops.mp3");
jaroslav@152
   296
        oops.play();
jaroslav@152
   297
    }
jaroslav@152
   298
    
jaroslav@152
   299
    private static Square tryStealBomb(Mines model, Square data) {
jaroslav@152
   300
        data.setMine(true);
jaroslav@152
   301
        final List<Row> rows = model.getRows();
jaroslav@152
   302
        for (int y = 0; y < rows.size(); y++) {
jaroslav@152
   303
            final List<Square> columns = rows.get(y).getColumns();
jaroslav@152
   304
            for (int x = 0; x < columns.size(); x++) {
jaroslav@152
   305
                Square sq = columns.get(x);
jaroslav@152
   306
                if (sq == data) {
jaroslav@152
   307
                    continue;
jaroslav@152
   308
                }
jaroslav@152
   309
                if (sq.isMine()) {
jaroslav@152
   310
                    sq.setMine(false);
jaroslav@152
   311
                    final boolean ok = isConsistent(model);
jaroslav@152
   312
                    sq.setMine(true);
jaroslav@152
   313
                    if (ok) {
jaroslav@152
   314
                        data.setMine(false);
jaroslav@152
   315
                        return sq;
jaroslav@152
   316
                    }
jaroslav@152
   317
                }
jaroslav@152
   318
            }
jaroslav@152
   319
        }
jaroslav@152
   320
        data.setMine(false);        
jaroslav@152
   321
        return null;
jaroslav@152
   322
    }
jaroslav@152
   323
    
jaroslav@152
   324
    private static Square atLeastOnePlaceWhereBombCantBe(Mines model) {
jaroslav@152
   325
        final List<Row> rows = model.getRows();
jaroslav@152
   326
        Square cantBe = null;
jaroslav@152
   327
        int discovered = 0;
jaroslav@152
   328
        for (int y = 0; y < rows.size(); y++) {
jaroslav@152
   329
            final List<Square> columns = rows.get(y).getColumns();
jaroslav@152
   330
            for (int x = 0; x < columns.size(); x++) {
jaroslav@152
   331
                Square sq = columns.get(x);
jaroslav@152
   332
                if (sq.getState() == SquareType.UNKNOWN) {
jaroslav@152
   333
                    if (!sq.isMine()) {
jaroslav@152
   334
                        if (tryStealBomb(model, sq) == null) {
jaroslav@152
   335
                            cantBe = sq;
jaroslav@152
   336
                        }
jaroslav@152
   337
                    }
jaroslav@152
   338
                } else {
jaroslav@152
   339
                    discovered++;
jaroslav@152
   340
                }
jaroslav@152
   341
            }
jaroslav@152
   342
        }
jaroslav@152
   343
        
jaroslav@152
   344
        if (discovered > 5) {
jaroslav@152
   345
            return cantBe;
jaroslav@152
   346
        }
jaroslav@152
   347
        
jaroslav@152
   348
        return null;
jaroslav@152
   349
    }
jaroslav@152
   350
    
jaroslav@152
   351
    private static boolean placeBombElseWhere(Mines model, Square moveBomb) {
jaroslav@152
   352
        List<Square> ok = new ArrayList<Square>();
jaroslav@152
   353
        moveBomb.setMine(false);
jaroslav@152
   354
        final List<Row> rows = model.getRows();
jaroslav@152
   355
        for (int y = 0; y < rows.size(); y++) {
jaroslav@152
   356
            final List<Square> columns = rows.get(y).getColumns();
jaroslav@152
   357
            for (int x = 0; x < columns.size(); x++) {
jaroslav@152
   358
                Square sq = columns.get(x);
jaroslav@152
   359
                if (sq == moveBomb || sq.isMine() || sq.getState().isVisible()) {
jaroslav@152
   360
                    continue;
jaroslav@152
   361
                }
jaroslav@152
   362
                sq.setMine(true);
jaroslav@152
   363
                if (isConsistent(model)) {
jaroslav@152
   364
                    ok.add(sq);
jaroslav@152
   365
                }
jaroslav@152
   366
                sq.setMine(false);
jaroslav@152
   367
            }
jaroslav@152
   368
        }
jaroslav@152
   369
        if (ok.isEmpty()) {
jaroslav@152
   370
            moveBomb.setMine(true);
jaroslav@152
   371
            return false;
jaroslav@152
   372
        } else {
jaroslav@152
   373
            int r = new Random().nextInt(ok.size());
jaroslav@152
   374
            ok.get(r).setMine(true);
jaroslav@152
   375
            return true;
jaroslav@152
   376
        }
jaroslav@152
   377
    }
jaroslav@152
   378
    
jtulach@68
   379
    private static void expandKnown(Mines model, Square data) {
jtulach@68
   380
        final List<Row> rows = model.getRows();
jtulach@68
   381
        for (int y = 0; y < rows.size(); y++) {
jtulach@68
   382
            final List<Square> columns = rows.get(y).getColumns();
jtulach@68
   383
            for (int x = 0; x < columns.size(); x++) {
jtulach@68
   384
                Square sq = columns.get(x);
jtulach@68
   385
                if (sq == data) {
jtulach@68
   386
                    expandKnown(model, x, y);
jtulach@68
   387
                    return;
jtulach@68
   388
                }
jtulach@68
   389
            }
jtulach@68
   390
        }
jtulach@68
   391
    }
jtulach@68
   392
    private static void expandKnown(Mines model, int x , int y) {
jtulach@68
   393
        if (y < 0 || y >= model.getRows().size()) {
jtulach@68
   394
            return;
jtulach@68
   395
        }
jtulach@68
   396
        final List<Square> columns = model.getRows().get(y).getColumns();
jtulach@68
   397
        if (x < 0 || x >= columns.size()) {
jtulach@68
   398
            return;
jtulach@68
   399
        }
jtulach@68
   400
        final Square sq = columns.get(x);
jtulach@68
   401
        if (sq.getState() == SquareType.UNKNOWN) {
jaroslav@152
   402
            int around = around(model, x, y);
jtulach@107
   403
            final SquareType t = SquareType.valueOf("N_" + around);
jtulach@107
   404
            sq.setState(t);
jtulach@107
   405
            if (t == SquareType.N_0) {
jtulach@68
   406
                expandKnown(model, x - 1, y - 1);
jtulach@68
   407
                expandKnown(model, x - 1, y);
jtulach@68
   408
                expandKnown(model, x - 1, y + 1);
jtulach@68
   409
                expandKnown(model, x , y - 1);
jtulach@68
   410
                expandKnown(model, x, y + 1);
jtulach@68
   411
                expandKnown(model, x + 1, y - 1);
jtulach@68
   412
                expandKnown(model, x + 1, y);
jtulach@68
   413
                expandKnown(model, x + 1, y + 1);
jtulach@68
   414
            }
jtulach@68
   415
        }
jtulach@68
   416
    }
jaroslav@152
   417
jaroslav@152
   418
    private static int around(Mines model, int x, int y) {
jaroslav@152
   419
        return 
jaroslav@152
   420
            minesAt(model, x - 1, y - 1) +
jaroslav@152
   421
            minesAt(model, x - 1, y) +
jaroslav@152
   422
            minesAt(model, x - 1, y + 1) +
jaroslav@152
   423
            minesAt(model, x , y - 1) +
jaroslav@152
   424
            minesAt(model, x, y + 1) +
jaroslav@152
   425
            minesAt(model, x + 1, y - 1) +
jaroslav@152
   426
            minesAt(model, x + 1, y) +
jaroslav@152
   427
            minesAt(model, x + 1, y + 1);
jaroslav@152
   428
    }
jtulach@90
   429
    
jtulach@107
   430
    private static int minesAt(Mines model, int x, int y) {
jtulach@107
   431
        if (y < 0 || y >= model.getRows().size()) {
jtulach@107
   432
            return 0;
jtulach@107
   433
        }
jtulach@107
   434
        final List<Square> columns = model.getRows().get(y).getColumns();
jtulach@107
   435
        if (x < 0 || x >= columns.size()) {
jtulach@107
   436
            return 0;
jtulach@107
   437
        }
jtulach@107
   438
        Square sq = columns.get(x);
jtulach@107
   439
        return sq.isMine() ? 1 : 0;
jtulach@107
   440
    }
jaroslav@152
   441
    
jaroslav@152
   442
    private static boolean isConsistent(Mines m) {
jaroslav@152
   443
        for (int row = 0; row < m.getRows().size(); row++) {
jaroslav@152
   444
            Row r = m.getRows().get(row);
jaroslav@152
   445
            for (int col = 0; col < r.getColumns().size(); col++) {
jaroslav@152
   446
                Square sq = r.getColumns().get(col);
jaroslav@152
   447
                if (sq.getState().isVisible()) {
jaroslav@152
   448
                    int around = around(m, col, row);
jaroslav@152
   449
                    if (around != sq.getState().ordinal()) {
jaroslav@152
   450
                        return false;
jaroslav@152
   451
                    }
jaroslav@152
   452
                }
jaroslav@152
   453
            }
jaroslav@152
   454
        }
jaroslav@152
   455
        return true;
jaroslav@152
   456
    }
jaroslav@165
   457
    
jaroslav@165
   458
    private static boolean allMarked(Mines m) {
jaroslav@165
   459
        for (Row r : m.getRows()) {
jaroslav@165
   460
            for (Square sq : r.getColumns()) {
jtulach@174
   461
                if (sq.isMine() == (sq.getState() != SquareType.MARKED)) {
jaroslav@165
   462
                    return false;
jaroslav@165
   463
                }
jaroslav@165
   464
            }
jaroslav@165
   465
        }
jaroslav@165
   466
        for (Row r : m.getRows()) {
jaroslav@165
   467
            for (Square sq : r.getColumns()) {
jaroslav@165
   468
                if (sq.isMine()) {
jaroslav@165
   469
                    sq.setState(SquareType.DISCOVERED);
jaroslav@165
   470
                } else {
jaroslav@165
   471
                    sq.setState(SquareType.N_0);
jaroslav@165
   472
                }
jaroslav@165
   473
            }
jaroslav@165
   474
        }
jaroslav@165
   475
        computeMines(m);
jaroslav@165
   476
        return true;
jaroslav@165
   477
    }
jtulach@116
   478
jtulach@90
   479
    /**
jtulach@90
   480
     * Called when page is ready
jtulach@90
   481
     */
jtulach@90
   482
    public static void main(String... args) throws Exception {
jtulach@90
   483
        Mines m = new Mines();
jtulach@90
   484
        m.applyBindings();
jtulach@90
   485
    }
jtulach@63
   486
}