From 13a38b3623aa5f0e5ba8e58b3a8dfb548e96e1ac Mon Sep 17 00:00:00 2001
From: Bram van Heuveln <56417002+jadeandtea@users.noreply.github.com>
Date: Fri, 11 Oct 2024 16:26:40 -0400
Subject: [PATCH] Fixing Merge Conflict
---
.../LastCellForNumberDirectRule/TestBoard | 19 ++
.../LastNumberForCellDirectRule/FullRegion | 19 ++
.../rules/RepeatedNumberContradictionRule/a | 0
.../10x10 Binary Hard/10x10 Binary Hard 1 | 111 +++++++++
.../10x10 Binary Hard/10x10 Binary Hard 2 | 111 +++++++++
.../10x10 Binary Hard/10x10 Binary Hard 3 | 111 +++++++++
.../10x10 Binary Medium/10x10 Binary Medium 1 | 111 +++++++++
.../10x10 Binary Medium/10x10 Binary Medium 2 | 111 +++++++++
.../10x10 Binary Medium/10x10 Binary Medium 3 | 111 +++++++++
.../10x10 Binary Very Hard 1 | 111 +++++++++
.../binary/6x6 Binary Easy/6x6 Binary Easy 1 | 22 ++
.../binary/6x6 Binary Easy/6x6 Binary Easy 2 | 21 ++
.../binary/6x6 Binary Easy/6x6 Binary Easy 3 | 25 ++
.../binary/6x6 Binary Easy/6x6 Binary Easy 4 | 28 +++
.../binary/6x6 Binary Easy/6x6 Binary Easy 5 | 47 ++++
.../binary/6x6 Binary Hard/6x6 Binary Hard 1 | 47 ++++
.../binary/6x6 Binary Hard/6x6 Binary Hard 2 | 47 ++++
.../binary/6x6 Binary Hard/6x6 Binary Hard 3 | 47 ++++
.../6x6 Binary Medium/6x6 Binary Medium 1 | 47 ++++
.../6x6 Binary Medium/6x6 Binary Medium 2 | 47 ++++
.../6x6 Binary Medium/6x6 Binary Medium 3 | 47 ++++
.../6x6 Binary Very Hard 1 | 47 ++++
.../6x6 Binary Very Hard 2 | 47 ++++
.../6x6 Binary Very Hard 3 | 47 ++++
.../binary/8x8 Binary Easy/8x8 Binary Easy 1 | 75 ++++++
.../binary/8x8 Binary Easy/8x8 Binary Easy 2 | 75 ++++++
.../binary/8x8 Binary Easy/8x8 Binary Easy 3 | 75 ++++++
.../binary/8x8 Binary Hard/8x8 Binary Hard 1 | 75 ++++++
.../binary/8x8 Binary Hard/8x8 Binary Hard 2 | 75 ++++++
.../binary/8x8 Binary Hard/8x8 Binary Hard 3 | 75 ++++++
.../8x8 Binary Medium/8x8 Binary Medium 1 | 75 ++++++
.../8x8 Binary Medium/8x8 Binary Medium 2 | 75 ++++++
.../8x8 Binary Medium/8x8 Binary Medium 3 | 75 ++++++
.../8x8 Binary Very Hard 1 | 75 ++++++
.../8x8 Binary Very Hard 2 | 75 ++++++
.../8x8 Binary Very Hard 3 | 75 ++++++
puzzles files/light-color-theme.txt | 99 ++++++++
.../5x5 Star Battle 1 star Normal 1.xml | 53 ----
.../6x6 Star Battle 1star Normal1.xml | 68 -----
.../6x6 Star Battle 1star Normal2.xml | 68 -----
.../6x6 StarBattle 1star Normal3.xml | 68 -----
.../7x7 Star Battle 1star Hard1.xml | 85 -------
.../7x7 Star Battle 1star Normal.xml | 85 -------
.../7x7 Star Battle 1star Normal1.xml | 85 -------
.../7x7 Star Battle 1star Normal2.xml | 85 -------
.../8x8 Star Battle 1star Normal1.xml | 105 --------
.../8x8 Star Battle 1star Normal2.xml | 104 --------
.../8x8 Star Battle 1star Normal3.xml | 104 --------
src/main/java/edu/rpi/legup/app/Config.java | 5 +
.../edu/rpi/legup/app/GameBoardFacade.java | 33 ++-
.../rpi/legup/app/InvalidConfigException.java | 5 +
.../edu/rpi/legup/app/LegupPreferences.java | 27 +-
.../edu/rpi/legup/controller/Controller.java | 5 +
.../controller/EditorElementController.java | 16 +-
.../legup/controller/ElementController.java | 18 +-
.../edu/rpi/legup/history/CommandError.java | 10 +
.../edu/rpi/legup/history/CommandState.java | 10 +
.../history/DeleteTreeElementCommand.java | 4 +
.../rpi/legup/history/EditDataCommand.java | 22 +-
.../java/edu/rpi/legup/history/History.java | 10 +-
.../rpi/legup/history/IHistoryListener.java | 9 +-
.../rpi/legup/history/IHistorySubject.java | 9 +-
.../InvalidCommandStateTransition.java | 7 +
.../history/ValidateDirectRuleCommand.java | 31 ++-
src/main/java/edu/rpi/legup/model/Puzzle.java | 55 ++--
.../edu/rpi/legup/model/PuzzleImporter.java | 22 +-
.../edu/rpi/legup/model/elements/Element.java | 43 ++++
.../edu/rpi/legup/model/gameboard/Board.java | 15 +-
.../rpi/legup/model/gameboard/CaseBoard.java | 48 ++++
.../legup/model/gameboard/ElementFactory.java | 3 +
.../rpi/legup/model/gameboard/GridBoard.java | 5 +
.../rpi/legup/model/gameboard/GridRegion.java | 7 +-
.../legup/model/gameboard/PuzzleElement.java | 20 ++
.../legup/model/observer/IBoardListener.java | 2 +-
.../legup/model/observer/IBoardSubject.java | 2 +-
.../legup/model/observer/ITreeSubject.java | 4 +-
.../edu/rpi/legup/model/rules/MergeRule.java | 2 +-
.../java/edu/rpi/legup/model/rules/Rule.java | 5 +
.../java/edu/rpi/legup/model/tree/Tree.java | 51 +++-
.../edu/rpi/legup/model/tree/TreeElement.java | 2 +-
.../edu/rpi/legup/model/tree/TreeNode.java | 4 +-
.../edu/rpi/legup/puzzle/binary/Binary.java | 23 +-
.../rpi/legup/puzzle/binary/BinaryBoard.java | 14 +-
.../rpi/legup/puzzle/binary/BinaryCell.java | 21 +-
.../puzzle/binary/BinaryCellFactory.java | 16 +-
.../legup/puzzle/binary/BinaryController.java | 6 +-
.../puzzle/binary/BinaryElementView.java | 104 ++------
.../legup/puzzle/binary/BinaryExporter.java | 2 +-
.../legup/puzzle/binary/BinaryImporter.java | 23 +-
.../puzzle/binary/elements/BlankTile.java | 1 -
.../binary_elements_reference_sheet.txt | 2 +
.../rules/CompleteRowColumnDirectRule.java | 22 +-
.../binary/rules/OneTileGapDirectRule.java | 64 -----
...> RepeatedRowColumnContradictionRule.java} | 26 +-
.../binary/rules/SurroundPairDirectRule.java | 48 ----
.../rules/ThreeAdjacentContradictionRule.java | 127 ----------
...UnbalancedRowColumnContradictionRule.java} | 27 +-
...roCaseRule.java => ZeroOrOneCaseRule.java} | 48 +++-
.../binary/rules/binary_reference_sheet.txt | 16 +-
.../legup/puzzle/fillapix/FillapixCell.java | 6 +-
.../puzzle/fillapix/elements/BlackTile.java | 2 +-
.../puzzle/fillapix/elements/NumberTile.java | 6 +-
.../puzzle/fillapix/elements/UnknownTile.java | 6 +-
.../puzzle/fillapix/elements/WhiteTile.java | 2 +-
.../fillapix_elements_reference_sheet.txt | 9 +-
.../fillapix/rules/BlackOrWhiteCaseRule.java | 3 +
.../fillapix/rules/SatisfyClueCaseRule.java | 3 +
.../edu/rpi/legup/puzzle/lightup/LightUp.java | 2 +-
.../legup/puzzle/lightup/LightUpBoard.java | 6 +-
.../rpi/legup/puzzle/lightup/LightUpCell.java | 8 +-
.../puzzle/lightup/elements/BlackTile.java | 6 +-
.../puzzle/lightup/elements/BulbTile.java | 2 +-
.../puzzle/lightup/elements/NumberTile.java | 8 +-
.../puzzle/lightup/elements/UnknownTile.java | 6 +-
.../lightup_elements_reference_sheet.txt | 4 +
.../lightup/rules/LightOrEmptyCaseRule.java | 4 +
.../lightup/rules/SatisfyNumberCaseRule.java | 6 +-
.../rules/TooFewBulbsContradictionRule.java | 2 +-
.../puzzle/minesweeper/elements/BombTile.java | 4 +-
.../minesweeper/elements/UnsetTile.java | 4 +-
.../legup/puzzle/nurikabe/NurikabeCell.java | 6 +-
.../puzzle/nurikabe/NurikabeExporter.java | 7 +
.../puzzle/nurikabe/elements/BlackTile.java | 2 +-
.../puzzle/nurikabe/elements/NumberTile.java | 6 +-
.../puzzle/nurikabe/elements/UnknownTile.java | 6 +-
.../puzzle/nurikabe/elements/WhiteTile.java | 2 +-
.../nurikabe_elements_reference_sheet.txt | 4 +
.../nurikabe/rules/BlackOrWhiteCaseRule.java | 6 +-
.../nurikabe/rules/FinishRoomCaseRule.java | 59 ++---
.../shorttruthtable/ShortTruthTableCell.java | 10 +-
.../elements/ArgumentElement.java | 6 +-
.../elements/GreenElement.java | 2 +-
.../elements/LogicSymbolElement.java | 6 +-
.../shorttruthtable/elements/RedElement.java | 2 +-
.../elements/UnknownElement.java | 2 +-
.../shorttruthtable_elements_reference_sheet | 11 +-
.../rules/caserule/CaseRuleAtomic.java | 3 +
.../caserule/CaseRule_GenericStatement.java | 5 +
.../legup/puzzle/skyscrapers/Skyscrapers.java | 2 +-
.../puzzle/skyscrapers/SkyscrapersBoard.java | 6 +-
.../puzzle/skyscrapers/SkyscrapersCell.java | 4 +-
.../skyscrapers/SkyscrapersClueView.java | 6 +-
.../puzzle/skyscrapers/elements/ClueTile.java | 14 --
.../skyscrapers/elements/NumberTile.java | 6 +-
.../skyscrapers/elements/UnknownTile.java | 6 +-
.../skyscrapers_elements_reference_sheet.txt | 5 +-
.../rules/CellForNumberCaseRule.java | 3 +
.../rules/NumberForCellCaseRule.java | 3 +
.../rpi/legup/puzzle/starbattle/allfiles.txt | 235 ------------------
.../puzzle/starbattle/elements/BlackTile.java | 4 +-
.../puzzle/starbattle/elements/StarTile.java | 4 +-
.../starbattle/elements/UnknownTile.java | 4 +-
.../starbattle_elements_reference_sheet.txt | 4 +
.../starbattle/rules/StarOrEmptyCaseRule.java | 4 +
.../rules/starbattle_reference_sheet.txt | 19 --
.../legup/puzzle/sudoku/ModelSudokuBoard.java | 17 ++
.../sudoku/PossibleNumberCaseBoard.java | 4 +-
.../edu/rpi/legup/puzzle/sudoku/Sudoku.java | 2 +
.../rpi/legup/puzzle/sudoku/SudokuCell.java | 2 +
.../puzzle/sudoku/SudokuCellController.java | 2 +-
.../legup/puzzle/sudoku/SudokuImporter.java | 10 -
.../rpi/legup/puzzle/sudoku/SudokuView.java | 1 +
.../puzzle/sudoku/elements/NumberTile.java | 9 +-
.../sudoku_elements_reference_sheet.txt | 2 +
.../rules/AdvancedDeductionDirectRule.java | 99 --------
.../rules/LastCellForNumberDirectRule.java | 136 ++++++++--
.../rules/LastNumberForCellDirectRule.java | 25 +-
...oCellForNumberColumnContradictionRule.java | 90 +++++++
...oCellForNumberRegionContradictionRule.java | 90 +++++++
.../NoCellForNumberRowContradictionRule.java | 90 +++++++
... => NoNumberForCellContradictionRule.java} | 30 ++-
...PossibleCellsForNumberColumnCaseRule.java} | 82 +++---
.../PossibleCellsForNumberRegionCaseRule.java | 104 ++++++++
.../PossibleCellsForNumberRowCaseRule.java | 107 ++++++++
...va => PossibleNumbersForCellCaseRule.java} | 73 +++---
.../RepeatedNumberContradictionRule.java | 64 +++--
.../sudoku/rules/sudoku_reference_sheet.txt | 14 +-
.../rpi/legup/puzzle/treetent/TreeTent.java | 15 +-
.../puzzle/treetent/elements/GrassTile.java | 2 +-
.../puzzle/treetent/elements/TentTile.java | 2 +-
.../puzzle/treetent/elements/TreeTile.java | 6 +-
.../puzzle/treetent/elements/UnknownTile.java | 6 +-
.../treetent_elements_reference_sheet.txt | 4 +
.../treetent/rules/FillinRowCaseRule.java | 4 +
.../treetent/rules/LinkTentCaseRule.java | 4 +
.../treetent/rules/LinkTreeCaseRule.java | 4 +
.../edu/rpi/legup/ui/CreatePuzzleDialog.java | 14 +-
.../java/edu/rpi/legup/ui/DynamicView.java | 35 +++
src/main/java/edu/rpi/legup/ui/HomePanel.java | 113 ++++-----
src/main/java/edu/rpi/legup/ui/LegupUI.java | 67 +++--
.../java/edu/rpi/legup/ui/PickGameDialog.java | 10 +
.../edu/rpi/legup/ui/PreferencesDialog.java | 22 ++
.../edu/rpi/legup/ui/ProofEditorPanel.java | 174 ++++++-------
.../edu/rpi/legup/ui/PuzzleEditorPanel.java | 199 ++++++++-------
.../java/edu/rpi/legup/ui/ScrollView.java | 5 +
.../java/edu/rpi/legup/ui/ToolbarName.java | 5 +-
.../java/edu/rpi/legup/ui/ZoomWidget.java | 9 +
.../java/edu/rpi/legup/ui/ZoomablePane.java | 4 +
.../edu/rpi/legup/ui/boardview/BoardView.java | 15 ++
.../legup/ui/boardview/DataSelectionView.java | 5 +
.../legup/ui/boardview/ElementSelection.java | 33 +++
.../rpi/legup/ui/boardview/ElementView.java | 46 ++++
.../rpi/legup/ui/boardview/GridBoardView.java | 9 +-
.../legup/ui/boardview/SelectionItemView.java | 5 +
.../ui/proofeditorui/rulesview/RuleFrame.java | 27 +-
.../ui/proofeditorui/rulesview/RulePanel.java | 32 ++-
.../treeview/TreeElementView.java | 4 +-
.../proofeditorui/treeview/TreeNodeView.java | 10 +
.../ui/proofeditorui/treeview/TreePanel.java | 25 ++
.../treeview/TreeToolBarButton.java | 11 +
.../treeview/TreeToolBarName.java | 4 +
.../treeview/TreeTransitionView.java | 52 ++--
.../ui/proofeditorui/treeview/TreeView.java | 100 ++++++--
.../elementsview/ElementFrame.java | 42 ++--
.../elementsview/ElementPanel.java | 3 +-
.../edu/rpi/legup/images/Legup/Reset.png | Bin 0 -> 662 bytes
.../rules/CompleteRowColumnDirectRule.png | Bin 2835 -> 2174 bytes
.../DuplicateRowOrColumnContradictionRule.png | Bin 3455 -> 0 bytes
.../images/binary/rules/OneOrZeroCaseRule.png | Bin 5578 -> 0 bytes
.../binary/rules/OneTileGapDirectRule.png | Bin 2751 -> 0 bytes
.../binary/rules/PreventTrioDirectRule.png | Bin 0 -> 1944 bytes
.../RepeatedRowColumnContradictionRule.png | Bin 0 -> 2124 bytes
.../binary/rules/SaveBlockerDirectRule.png | Bin 0 -> 2204 bytes
.../binary/rules/SurroundPairDirectRule.png | Bin 2982 -> 0 bytes
.../rules/ThreeAdjacentContradictionRule.png | Bin 2508 -> 0 bytes
.../binary/rules/TrioContradictionRule.png | Bin 0 -> 2071 bytes
.../UnbalancedRowColumnContradictionRule.png | Bin 2793 -> 1783 bytes
.../rules/UniqueRowColumnDirectRule.png | Bin 0 -> 3304 bytes
.../rules/WastedBlockerContradictionRule.png | Bin 0 -> 2006 bytes
.../images/binary/rules/ZeroOrOneCaseRule.png | Bin 0 -> 2186 bytes
.../legup/images/binary/tiles/NumberTile.png | Bin 0 -> 370 bytes
.../legup/images/binary/tiles/UnknownTile.png | Bin 0 -> 7231 bytes
.../skyscrapers/cases/CellForNumber.png | Bin 1871 -> 1531 bytes
.../skyscrapers/cases/NumberForCell.png | Bin 1875 -> 1603 bytes
.../contradictions/DuplicateNumber.png | Bin 954 -> 1991 bytes
.../contradictions/ExceedingVisibility.png | Bin 1184 -> 1949 bytes
.../contradictions/InsufficientVisibility.png | Bin 1291 -> 2071 bytes
.../contradictions/PreemptiveVisibility.png | Bin 1101 -> 1784 bytes
.../contradictions/UnresolvedCell.png | Bin 1165 -> 2108 bytes
.../contradictions/UnresolvedNumber.png | Bin 1357 -> 2658 bytes
.../images/skyscrapers/rules/FixedMax.png | Bin 1225 -> 1783 bytes
.../images/skyscrapers/rules/LastCell.png | Bin 1352 -> 2174 bytes
.../images/skyscrapers/rules/LastNumber.png | Bin 1335 -> 2083 bytes
.../legup/images/skyscrapers/rules/NEdge.png | Bin 1523 -> 2027 bytes
.../images/skyscrapers/rules/OneEdge.png | Bin 934 -> 1455 bytes
.../edu/rpi/legup/images/starbattle/black.gif | Bin 856 -> 0 bytes
.../starbattle/cases/StarOrEmptyCaseRule.png | Bin 4437 -> 0 bytes
.../ClashingOrbitContradictionRule.png | Bin 5135 -> 0 bytes
.../TooFewStarsContradictionRule.png | Bin 3218 -> 0 bytes
.../TooManyStarsContradictionRule.png | Bin 6199 -> 0 bytes
.../edu/rpi/legup/images/starbattle/empty.gif | Bin 857 -> 0 bytes
.../starbattle/rules/BlackOutDirectRule.png | Bin 4717 -> 0 bytes
.../rules/ColumnsWithinRegionsDirectRule.png | Bin 6456 -> 0 bytes
.../rules/ColumnsWithinRowsDirectRule.png | Bin 5331 -> 0 bytes
.../rules/FinishWithStarDirectRule.png | Bin 4326 -> 0 bytes
.../rules/RegionsWithinColumnsDirectRule.png | Bin 6402 -> 0 bytes
.../rules/RegionsWithinRowsDirectRule.png | Bin 6269 -> 0 bytes
.../rules/RowsWithinColumnsDirectRule.png | Bin 5117 -> 0 bytes
.../rules/RowsWithinRegionsDirectRule.png | Bin 6477 -> 0 bytes
.../images/starbattle/rules/SurroundStar.png | Bin 5102 -> 0 bytes
.../edu/rpi/legup/images/starbattle/star.gif | Bin 545 -> 0 bytes
.../legup/images/sudoku/AdvancedDeduction.png | Bin 3351 -> 0 bytes
.../sudoku/rules/NoCellForNumberColumn.png | Bin 0 -> 958 bytes
.../sudoku/rules/NoCellForNumberRegion.png | Bin 0 -> 1077 bytes
.../sudoku/rules/NoCellForNumberRow.png | Bin 0 -> 916 bytes
.../images/sudoku/{ => rules}/NoSolution.png | Bin
.../sudoku/{ => rules}/PossibleValues.png | Bin
.../sudoku/{ => rules}/RepeatedNumber.png | Bin
.../sudoku/{ => rules}/forcedByDeduction.png | Bin
.../{ => rules}/forcedByElimination.png | Bin
.../{ => rules}/possible_cells_number.png | Bin
.../rules/possible_cells_number_column.png | Bin 0 -> 3336 bytes
.../rules/possible_cells_number_region.png | Bin 0 -> 3623 bytes
.../rules/possible_cells_number_row.png | Bin 0 -> 803 bytes
.../legup/images/sudoku/{ => rules}/tem.png | Bin
.../white.gif => sudoku/tiles/NumberTile.png} | Bin 9700 -> 10067 bytes
.../tiles}/UnknownTile.png | Bin
src/main/resources/edu/rpi/legup/legup/config | 30 +--
.../rules/FinishRoomCaseRuleTest.java | 30 +--
...LastNumberForCellDirectRuleRegionTest.java | 100 ++++++++
.../RepeatedNumberContradictionRuleTest.java | 88 +++++++
.../SurroundTwoZerosWithTwoOnes | 10 +
.../binary/rules/SurroundPairDirectRule/test | 0
.../SurroundTwoZerosWithTwoOnes | 10 +
.../BlackoutDirectRule/ColumnBlackout | 40 ---
.../BlackoutDirectRule/RegionBlackout | 36 ---
.../BlackoutDirectRule/RowBlackout | 36 ---
.../CorneredRegion | 14 ++
.../LastNumberForCellDirectRule/FullMixed | 18 ++
.../LastNumberForCellDirectRule/FullRegion | 18 ++
.../rules/LastNumberForCellDirectRule/FullRow | 18 ++
.../BlankBoard4 | 12 +
.../BlankBoard7 | 11 +
293 files changed, 4995 insertions(+), 2620 deletions(-)
create mode 100644 output_path/test/src/resources/puzzles/sudoku/rules/LastCellForNumberDirectRule/TestBoard
create mode 100644 output_path/test/src/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullRegion
create mode 100644 output_path/test/src/resources/puzzles/sudoku/rules/RepeatedNumberContradictionRule/a
create mode 100644 puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 1
create mode 100644 puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 2
create mode 100644 puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 3
create mode 100644 puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 1
create mode 100644 puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 2
create mode 100644 puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 3
create mode 100644 puzzles files/binary/10x10 Binary Very Hard/10x10 Binary Very Hard 1
create mode 100644 puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 1
create mode 100644 puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 2
create mode 100644 puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 3
create mode 100644 puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 4
create mode 100644 puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 5
create mode 100644 puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 1
create mode 100644 puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 2
create mode 100644 puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 3
create mode 100644 puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 1
create mode 100644 puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 2
create mode 100644 puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 3
create mode 100644 puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 1
create mode 100644 puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 2
create mode 100644 puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 3
create mode 100644 puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 1
create mode 100644 puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 2
create mode 100644 puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 3
create mode 100644 puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 1
create mode 100644 puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 2
create mode 100644 puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 3
create mode 100644 puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 1
create mode 100644 puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 2
create mode 100644 puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 3
create mode 100644 puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 1
create mode 100644 puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 2
create mode 100644 puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 3
create mode 100644 puzzles files/light-color-theme.txt
delete mode 100644 puzzles files/starbattle/5x5 Star Battle 1 star Normal/5x5 Star Battle 1 star Normal 1.xml
delete mode 100644 puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal1.xml
delete mode 100644 puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal2.xml
delete mode 100644 puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 StarBattle 1star Normal3.xml
delete mode 100644 puzzles files/starbattle/7x7 Star Battle 1 star Hard/7x7 Star Battle 1star Hard1.xml
delete mode 100644 puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal.xml
delete mode 100644 puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal1.xml
delete mode 100644 puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal2.xml
delete mode 100644 puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal1.xml
delete mode 100644 puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal2.xml
delete mode 100644 puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal3.xml
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/binary/elements/BlankTile.java
create mode 100644 src/main/java/edu/rpi/legup/puzzle/binary/elements/binary_elements_reference_sheet.txt
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/binary/rules/OneTileGapDirectRule.java
rename src/main/java/edu/rpi/legup/puzzle/binary/rules/{DuplicateRowsOrColumnsContradictionRule.java => RepeatedRowColumnContradictionRule.java} (69%)
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/binary/rules/SurroundPairDirectRule.java
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/binary/rules/ThreeAdjacentContradictionRule.java
rename src/main/java/edu/rpi/legup/puzzle/binary/rules/{UnbalancedRowOrColumnContradictionRule.java => UnbalancedRowColumnContradictionRule.java} (66%)
rename src/main/java/edu/rpi/legup/puzzle/binary/rules/{OneOrZeroCaseRule.java => ZeroOrOneCaseRule.java} (64%)
create mode 100644 src/main/java/edu/rpi/legup/puzzle/lightup/elements/lightup_elements_reference_sheet.txt
create mode 100644 src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/nurikabe_elements_reference_sheet.txt
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/ClueTile.java
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/starbattle/allfiles.txt
create mode 100644 src/main/java/edu/rpi/legup/puzzle/starbattle/elements/starbattle_elements_reference_sheet.txt
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/starbattle/rules/starbattle_reference_sheet.txt
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/ModelSudokuBoard.java
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/elements/sudoku_elements_reference_sheet.txt
delete mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/rules/AdvancedDeductionDirectRule.java
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberColumnContradictionRule.java
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRegionContradictionRule.java
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRowContradictionRule.java
rename src/main/java/edu/rpi/legup/puzzle/sudoku/rules/{NoSolutionContradictionRule.java => NoNumberForCellContradictionRule.java} (72%)
rename src/main/java/edu/rpi/legup/puzzle/sudoku/rules/{PossibleNumberCaseRule.java => PossibleCellsForNumberColumnCaseRule.java} (53%)
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRegionCaseRule.java
create mode 100644 src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRowCaseRule.java
rename src/main/java/edu/rpi/legup/puzzle/sudoku/rules/{PossibleCellCaseRule.java => PossibleNumbersForCellCaseRule.java} (59%)
create mode 100644 src/main/java/edu/rpi/legup/puzzle/treetent/elements/treetent_elements_reference_sheet.txt
create mode 100644 src/main/resources/edu/rpi/legup/images/Legup/Reset.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/DuplicateRowOrColumnContradictionRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/OneOrZeroCaseRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/OneTileGapDirectRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/PreventTrioDirectRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/RepeatedRowColumnContradictionRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/SaveBlockerDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/SurroundPairDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/ThreeAdjacentContradictionRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/TrioContradictionRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/UniqueRowColumnDirectRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/WastedBlockerContradictionRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/rules/ZeroOrOneCaseRule.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/tiles/NumberTile.png
create mode 100644 src/main/resources/edu/rpi/legup/images/binary/tiles/UnknownTile.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/black.gif
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/cases/StarOrEmptyCaseRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/contradictions/ClashingOrbitContradictionRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/contradictions/TooFewStarsContradictionRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/contradictions/TooManyStarsContradictionRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/empty.gif
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/BlackOutDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/ColumnsWithinRegionsDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/ColumnsWithinRowsDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/FinishWithStarDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/RegionsWithinColumnsDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/RegionsWithinRowsDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/RowsWithinColumnsDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/RowsWithinRegionsDirectRule.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/rules/SurroundStar.png
delete mode 100644 src/main/resources/edu/rpi/legup/images/starbattle/star.gif
delete mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/AdvancedDeduction.png
create mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/rules/NoCellForNumberColumn.png
create mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/rules/NoCellForNumberRegion.png
create mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/rules/NoCellForNumberRow.png
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/NoSolution.png (100%)
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/PossibleValues.png (100%)
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/RepeatedNumber.png (100%)
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/forcedByDeduction.png (100%)
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/forcedByElimination.png (100%)
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/possible_cells_number.png (100%)
create mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/rules/possible_cells_number_column.png
create mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/rules/possible_cells_number_region.png
create mode 100644 src/main/resources/edu/rpi/legup/images/sudoku/rules/possible_cells_number_row.png
rename src/main/resources/edu/rpi/legup/images/sudoku/{ => rules}/tem.png (100%)
rename src/main/resources/edu/rpi/legup/images/{starbattle/white.gif => sudoku/tiles/NumberTile.png} (82%)
rename src/main/resources/edu/rpi/legup/images/{starbattle => sudoku/tiles}/UnknownTile.png (100%)
create mode 100644 src/test/java/puzzles/sudoku/rules/LastNumberForCellDirectRuleRegionTest.java
create mode 100644 src/test/java/puzzles/sudoku/rules/RepeatedNumberContradictionRuleTest.java
create mode 100644 src/test/resources/puzzles/binary/rules/SurroundPairDirectRule/SurroundTwoZerosWithTwoOnes
create mode 100644 src/test/resources/puzzles/binary/rules/SurroundPairDirectRule/test
create mode 100644 src/test/resources/puzzles/nurikabe/rules/BlackBetweenRegionsDirectRule/SurroundTwoZerosWithTwoOnes
delete mode 100644 src/test/resources/puzzles/starbattle.rules/BlackoutDirectRule/ColumnBlackout
delete mode 100644 src/test/resources/puzzles/starbattle.rules/BlackoutDirectRule/RegionBlackout
delete mode 100644 src/test/resources/puzzles/starbattle.rules/BlackoutDirectRule/RowBlackout
create mode 100644 src/test/resources/puzzles/sudoku/rules/LastCellForNumberDirectRule/CorneredRegion
create mode 100644 src/test/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullMixed
create mode 100644 src/test/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullRegion
create mode 100644 src/test/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullRow
create mode 100644 src/test/resources/puzzles/sudoku/rules/RepeatedNumberContradictionRule/BlankBoard4
create mode 100644 src/test/resources/puzzles/sudoku/rules/RepeatedNumberContradictionRule/BlankBoard7
diff --git a/output_path/test/src/resources/puzzles/sudoku/rules/LastCellForNumberDirectRule/TestBoard b/output_path/test/src/resources/puzzles/sudoku/rules/LastCellForNumberDirectRule/TestBoard
new file mode 100644
index 000000000..a41ad749c
--- /dev/null
+++ b/output_path/test/src/resources/puzzles/sudoku/rules/LastCellForNumberDirectRule/TestBoard
@@ -0,0 +1,19 @@
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
+
diff --git a/output_path/test/src/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullRegion b/output_path/test/src/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullRegion
new file mode 100644
index 000000000..49dae4aa6
--- /dev/null
+++ b/output_path/test/src/resources/puzzles/sudoku/rules/LastNumberForCellDirectRule/FullRegion
@@ -0,0 +1,19 @@
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
+
diff --git a/output_path/test/src/resources/puzzles/sudoku/rules/RepeatedNumberContradictionRule/a b/output_path/test/src/resources/puzzles/sudoku/rules/RepeatedNumberContradictionRule/a
new file mode 100644
index 000000000..e69de29bb
diff --git a/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 1 b/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 1
new file mode 100644
index 000000000..42ccf371b
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 1
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 2 b/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 2
new file mode 100644
index 000000000..d73caa5d2
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 2
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 3 b/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 3
new file mode 100644
index 000000000..99ec9769b
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Hard/10x10 Binary Hard 3
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 1 b/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 1
new file mode 100644
index 000000000..d203617c8
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 1
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 2 b/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 2
new file mode 100644
index 000000000..db56f04f3
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 2
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 3 b/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 3
new file mode 100644
index 000000000..11940a6eb
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Medium/10x10 Binary Medium 3
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/10x10 Binary Very Hard/10x10 Binary Very Hard 1 b/puzzles files/binary/10x10 Binary Very Hard/10x10 Binary Very Hard 1
new file mode 100644
index 000000000..828a450cf
--- /dev/null
+++ b/puzzles files/binary/10x10 Binary Very Hard/10x10 Binary Very Hard 1
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 1 b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 1
new file mode 100644
index 000000000..7b22ffc10
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 1
@@ -0,0 +1,22 @@
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
\ No newline at end of file
diff --git a/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 2 b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 2
new file mode 100644
index 000000000..ea8ef93b0
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 2
@@ -0,0 +1,21 @@
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
+
\ No newline at end of file
diff --git a/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 3 b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 3
new file mode 100644
index 000000000..0f0ff745e
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 3
@@ -0,0 +1,25 @@
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
+
\ No newline at end of file
diff --git a/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 4 b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 4
new file mode 100644
index 000000000..da76d067b
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 4
@@ -0,0 +1,28 @@
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
+
\ No newline at end of file
diff --git a/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 5 b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 5
new file mode 100644
index 000000000..a1ea13988
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Easy/6x6 Binary Easy 5
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 1 b/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 1
new file mode 100644
index 000000000..5f7f72a8a
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 1
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 2 b/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 2
new file mode 100644
index 000000000..a4ed30c31
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 2
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 3 b/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 3
new file mode 100644
index 000000000..fc0e413c1
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Hard/6x6 Binary Hard 3
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 1 b/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 1
new file mode 100644
index 000000000..a5ab8a2dc
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 1
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 2 b/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 2
new file mode 100644
index 000000000..4be5fdaad
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 2
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 3 b/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 3
new file mode 100644
index 000000000..eba370cab
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Medium/6x6 Binary Medium 3
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 1 b/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 1
new file mode 100644
index 000000000..faa68fa5e
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 1
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 2 b/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 2
new file mode 100644
index 000000000..3c707bdaa
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 2
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 3 b/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 3
new file mode 100644
index 000000000..217a032d8
--- /dev/null
+++ b/puzzles files/binary/6x6 Binary Very Hard/6x6 Binary Very Hard 3
@@ -0,0 +1,47 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 1 b/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 1
new file mode 100644
index 000000000..befd674f9
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 1
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 2 b/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 2
new file mode 100644
index 000000000..724426c26
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 2
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 3 b/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 3
new file mode 100644
index 000000000..92a96c72b
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Easy/8x8 Binary Easy 3
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 1 b/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 1
new file mode 100644
index 000000000..34eaf8388
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 1
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 2 b/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 2
new file mode 100644
index 000000000..9ef23277e
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 2
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 3 b/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 3
new file mode 100644
index 000000000..287ff6f68
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Hard/8x8 Binary Hard 3
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 1 b/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 1
new file mode 100644
index 000000000..47dae23dc
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 1
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 2 b/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 2
new file mode 100644
index 000000000..ae4cb8bb0
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 2
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 3 b/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 3
new file mode 100644
index 000000000..2f951ecc4
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Medium/8x8 Binary Medium 3
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 1 b/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 1
new file mode 100644
index 000000000..9c875523b
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 1
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 2 b/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 2
new file mode 100644
index 000000000..14f2e4ad2
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 2
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 3 b/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 3
new file mode 100644
index 000000000..ad319a4b7
--- /dev/null
+++ b/puzzles files/binary/8x8 Binary Very Hard/8x8 Binary Very Hard 3
@@ -0,0 +1,75 @@
+
+
+
+
+
+
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+ |
+
+
+
+
+
diff --git a/puzzles files/light-color-theme.txt b/puzzles files/light-color-theme.txt
new file mode 100644
index 000000000..5c45bc71b
--- /dev/null
+++ b/puzzles files/light-color-theme.txt
@@ -0,0 +1,99 @@
+correct: BLUE
+incorrect: RED
+error: RED_700
+info: GRAY_900
+ui-movement: GRAY_300
+password-field-background: LIGHT_BLUE_400
+password-field-unfocused-background: GRAY_200
+progress-bar-background: GRAY_200
+progress-bar-foreground: LIGHT_BLUE_400
+text-field-background: LIGHT_BLUE_400
+text-field-unfocused-background: GRAY_200
+light-line-border: GRAY_200
+thick-line-border: GRAY_200
+data-selection-background: GRAY
+element-view: BLACK
+button-highlight: GRAY_300
+button-background: GRAY_200
+button-foreground: BLACK
+checkbox-background: WHITE
+checkbox-foreground: BLACK
+combobox-background: WHITE
+combobox-foreground: BLACK
+combobox-button-background: GRAY_300
+combobox-selection-background: WHITE
+combobox-selection-foreground: BLACK
+combobox-selected-in-drop-down-background: GRAY_200
+label-background: WHITE
+label-foreground: BLACK
+menu-background: LIGHT_BLUE_100
+menu-foreground: BLACK
+menu-selection-background: GRAY_200
+menu-selection-foreground: BLACK
+menu-disabled-foreground: #000
+menu-bar-background: WHITE
+menu-bar-foreground: BLACK
+menu-item-disabled-foreground: #000
+menu-item-selection-background: GRAY_200
+menu-item-selection-foreground: BLACK
+menu-item-background: WHITE
+menu-item-foreground: BLACK
+option-pane-background: WHITE
+panel-background-color: WHITE
+popup-menu-background: WHITE
+popup-menu-foreground: BLACK
+radio-button-background: WHITE
+radio-button-foreground: BLACK
+spinner-background: WHITE
+spinner-foreground: BLACK
+spinner-arrow-button-background: GRAY_200
+scroll-bar-track: GRAY_200
+scroll-bar-thumb: GRAY_300
+scroll-bar-thumb-dark-shadow: GRAY_300
+scroll-bar-thumb-highlight: GRAY_300
+scroll-bar-thumb-shadow: GRAY_300
+scroll-bar-arrow-button-background: GRAY_300
+scroll-pane-background: WHITE
+slider-background: WHITE
+slider-foreground: GRAY_700
+slider-track-color: BLACK
+split-pane-background: WHITE
+tabbed-pane-background: WHITE
+tabbed-pane-foreground: BLACK
+tabbed-pane-highlight: GRAY_200
+tabbed-pane-border-highlight: GRAY_300
+table-selection-background: GRAY_100
+table-selection-foreground: BLACK
+table-background: WHITE
+table-grid-color: GRAY_200
+table-header-background: GRAY_200
+text-area-background: GRAY_200
+text-area-foreground: BLACK
+toggle-button-background: WHITE
+toggle-button-foreground: BLACK
+tool-bar-background: WHITE
+tool-bar-foreground: BLACK
+tool-bar-docking-background: LIGHT_GREEN_A100
+tool-bar-floating-background: GRAY_200
+tree-selection-foreground: BLACK
+tree-foreground: BLACK
+tree-selection-background: GRAY_200
+tree-background: WHITE
+radio-button-menu-item-foreground: BLACK
+radio-button-menu-item-selection-foreground: BLACK
+radio-button-menu-item-selection-background: GRAY_200
+checkbox-menu-item-selection-background: GRAY_200
+checkbox-menu-item-foreground: BLACK
+checkbox-menu-item-selection-foreground: BLACK
+text-pane-background: GRAY_50
+text-pane-selection-background: LIGHT_BLUE_200
+text-pane-inactive-foreground: GRAY_500
+editor-pane-background: GRAY_50
+editor-pane-selection-background: LIGHT_BLUE_200
+editor-pane-inactive-foreground: GRAY_500
+separator-background: GRAY_300
+separator-foreground: GRAY_300
+tool-tip-background: GRAY_500
+tool-tip-foreground: GRAY_50
+color-chooser-background: WHITE
+color-chooser-foreground: BLACK
diff --git a/puzzles files/starbattle/5x5 Star Battle 1 star Normal/5x5 Star Battle 1 star Normal 1.xml b/puzzles files/starbattle/5x5 Star Battle 1 star Normal/5x5 Star Battle 1 star Normal 1.xml
deleted file mode 100644
index b86e16ff2..000000000
--- a/puzzles files/starbattle/5x5 Star Battle 1 star Normal/5x5 Star Battle 1 star Normal 1.xml
+++ /dev/null
@@ -1,53 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
-
-
-
-
- |
-
-
-
-
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal1.xml b/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal1.xml
deleted file mode 100644
index 110dd9abe..000000000
--- a/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal1.xml
+++ /dev/null
@@ -1,68 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal2.xml b/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal2.xml
deleted file mode 100644
index 49efeba59..000000000
--- a/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 Star Battle 1star Normal2.xml
+++ /dev/null
@@ -1,68 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 StarBattle 1star Normal3.xml b/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 StarBattle 1star Normal3.xml
deleted file mode 100644
index 855943612..000000000
--- a/puzzles files/starbattle/6x6 Star Battle 1 star Normal/6x6 StarBattle 1star Normal3.xml
+++ /dev/null
@@ -1,68 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/7x7 Star Battle 1 star Hard/7x7 Star Battle 1star Hard1.xml b/puzzles files/starbattle/7x7 Star Battle 1 star Hard/7x7 Star Battle 1star Hard1.xml
deleted file mode 100644
index c1d7770f6..000000000
--- a/puzzles files/starbattle/7x7 Star Battle 1 star Hard/7x7 Star Battle 1star Hard1.xml
+++ /dev/null
@@ -1,85 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal.xml b/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal.xml
deleted file mode 100644
index cab0a0a5e..000000000
--- a/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal.xml
+++ /dev/null
@@ -1,85 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
-
-
-
-
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal1.xml b/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal1.xml
deleted file mode 100644
index 70b81e376..000000000
--- a/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal1.xml
+++ /dev/null
@@ -1,85 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal2.xml b/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal2.xml
deleted file mode 100644
index c541ece06..000000000
--- a/puzzles files/starbattle/7x7 Star Battle 1 star Normal/7x7 Star Battle 1star Normal2.xml
+++ /dev/null
@@ -1,85 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal1.xml b/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal1.xml
deleted file mode 100644
index 02dd5d6c0..000000000
--- a/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal1.xml
+++ /dev/null
@@ -1,105 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal2.xml b/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal2.xml
deleted file mode 100644
index 0df84ef62..000000000
--- a/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal2.xml
+++ /dev/null
@@ -1,104 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal3.xml b/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal3.xml
deleted file mode 100644
index 725c91d7f..000000000
--- a/puzzles files/starbattle/8x8 Star Battle 1 star Normal/8x8 Star Battle 1star Normal3.xml
+++ /dev/null
@@ -1,104 +0,0 @@
-
-
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
- |
- |
- |
- |
- |
- |
- |
- |
- |
-
-
-
-
-
-
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/app/Config.java b/src/main/java/edu/rpi/legup/app/Config.java
index 3d212a597..b04ef9b88 100644
--- a/src/main/java/edu/rpi/legup/app/Config.java
+++ b/src/main/java/edu/rpi/legup/app/Config.java
@@ -96,6 +96,11 @@ public static String convertDisplayNameToClassName(String displayName) {
return className;
}
+ /**
+ * Gets a list of all available puzzle display names
+ *
+ * @return a List of puzzle display names as Strings
+ */
public List getPuzzleNames() {
List names = new LinkedList();
for (String puzzle : this.getPuzzleClassNames()) {
diff --git a/src/main/java/edu/rpi/legup/app/GameBoardFacade.java b/src/main/java/edu/rpi/legup/app/GameBoardFacade.java
index 12f6b73d0..c6f92e5b5 100644
--- a/src/main/java/edu/rpi/legup/app/GameBoardFacade.java
+++ b/src/main/java/edu/rpi/legup/app/GameBoardFacade.java
@@ -89,6 +89,11 @@ public void initializeUI() {
});
}
+ /**
+ * Sets the current puzzle in the game board
+ *
+ * @param puzzle the Puzzle to set
+ */
public void setPuzzle(Puzzle puzzle) {
this.puzzle = puzzle;
this.puzzleSolver.setPuzzleView(puzzle);
@@ -113,11 +118,21 @@ public static void setupConfig() {
GameBoardFacade.getInstance().setConfig(config);
}
+ /**
+ * Sets the current puzzle editor with the given puzzle
+ *
+ * @param puzzle the Puzzle to set in the editor
+ */
public void setPuzzleEditor(Puzzle puzzle) {
this.puzzle = puzzle;
this.puzzleEditor.setPuzzleView(puzzle);
}
+ /**
+ * Sets the configuration object for the GameBoardFacade
+ *
+ * @param config config the Config object to set
+ */
public void setConfig(Config config) {
this.config = config;
}
@@ -176,7 +191,7 @@ public boolean validateTextInput(String game, String[] statements) throws Runtim
}
/**
- * Loads an empty puzzle
+ * Loads an empty puzzle with the specified dimensions
*
* @param game name of the puzzle
* @param rows the number of rows on the board
@@ -187,10 +202,10 @@ public void loadPuzzle(String game, int rows, int columns) throws RuntimeExcepti
String qualifiedClassName = config.getPuzzleClassForName(game);
LOGGER.debug("Loading " + qualifiedClassName);
- try {
- Class> c = Class.forName(qualifiedClassName);
- Constructor> cons = c.getConstructor();
- Puzzle puzzle = (Puzzle) cons.newInstance();
+ try {
+ Class> c = Class.forName(qualifiedClassName);
+ Constructor> cons = c.getConstructor();
+ Puzzle puzzle = (Puzzle) cons.newInstance();
PuzzleImporter importer = puzzle.getImporter();
if (importer == null) {
@@ -225,6 +240,12 @@ public void loadPuzzle(String game, int rows, int columns) throws RuntimeExcepti
}
}
+ /**
+ * Loads an empty puzzle with the specified input
+ *
+ * @param game name of the puzzle
+ * @param statements an array of statements to load the puzzle with
+ */
public void loadPuzzle(String game, String[] statements) {
String qualifiedClassName = config.getPuzzleClassForName(game);
LOGGER.debug("Loading " + qualifiedClassName);
@@ -267,7 +288,7 @@ public void loadPuzzle(String game, String[] statements) {
}
/**
- * Loads a puzzle file
+ * Loads a puzzle file from the specified file
*
* @param fileName file name of the board file
* @throws InvalidFileFormatException if the file format is invalid or if the file cannot be
diff --git a/src/main/java/edu/rpi/legup/app/InvalidConfigException.java b/src/main/java/edu/rpi/legup/app/InvalidConfigException.java
index 2526b3776..cad80b5ce 100644
--- a/src/main/java/edu/rpi/legup/app/InvalidConfigException.java
+++ b/src/main/java/edu/rpi/legup/app/InvalidConfigException.java
@@ -5,6 +5,11 @@
* errors
*/
public class InvalidConfigException extends Exception {
+ /**
+ * Constructs a new InvalidConfigException with the specified detail message
+ *
+ * @param message the detail message
+ */
public InvalidConfigException(String message) {
super(message);
}
diff --git a/src/main/java/edu/rpi/legup/app/LegupPreferences.java b/src/main/java/edu/rpi/legup/app/LegupPreferences.java
index 0bca27b49..3d17233bd 100644
--- a/src/main/java/edu/rpi/legup/app/LegupPreferences.java
+++ b/src/main/java/edu/rpi/legup/app/LegupPreferences.java
@@ -81,7 +81,7 @@ public class LegupPreferences {
* Gets the legup preferences singleton instance This method ensures that only one instance of
* LegupPreferences exists
*
- * @return legup preferences
+ * @return the singleton instance of LegupPreferences
*/
public static LegupPreferences getInstance() {
if (instance == null) {
@@ -100,23 +100,30 @@ private LegupPreferences() {}
* Gets the user preference by the string key
*
* @param key key name of the preference
- * @return value of the preference
+ * @return value of the preference or {@code null} if the preference does not exist
*/
public String getUserPref(String key) {
return preferencesMap.get(key);
}
/**
- * Gets the user preference by the string key, value pair
+ * Sets the user preference for the specified key to the provided value
*
- * @param key key name of the preference
- * @param value value of the preference
+ * @param key key to set for the preference
+ * @param value value to set for the preference
*/
public void setUserPref(String key, String value) {
preferences.put(key, value);
preferencesMap.put(key, value);
}
+ /**
+ * Retrieves the user preference associated with the specified key as a boolean
+ *
+ * @param key the key for the preference to retrieve
+ * @return the boolean value of the preference
+ * @throws RuntimeException if the preference value cannot be interpreted as a boolean
+ */
public boolean getUserPrefAsBool(String key) {
if (preferencesMap.get(key).equalsIgnoreCase(Boolean.toString(true))) {
return true;
@@ -129,10 +136,20 @@ public boolean getUserPrefAsBool(String key) {
}
}
+ /**
+ * Gets the saved path
+ *
+ * @return the saved path as a String
+ */
public String getSavedPath() {
return SAVED_PATH;
}
+ /**
+ * Sets the saved path to the specified value
+ *
+ * @param path the new saved path
+ */
public void setSavedPath(String path) {
SAVED_PATH = path;
}
diff --git a/src/main/java/edu/rpi/legup/controller/Controller.java b/src/main/java/edu/rpi/legup/controller/Controller.java
index 4910e24d9..823bac716 100644
--- a/src/main/java/edu/rpi/legup/controller/Controller.java
+++ b/src/main/java/edu/rpi/legup/controller/Controller.java
@@ -24,6 +24,11 @@ public Controller() {
pan = false;
}
+ /**
+ * Sets the ScrollView instance that this controller manages
+ *
+ * @param viewer The ScrollView instance to be set
+ */
public void setViewer(ScrollView viewer) {
this.viewer = viewer;
}
diff --git a/src/main/java/edu/rpi/legup/controller/EditorElementController.java b/src/main/java/edu/rpi/legup/controller/EditorElementController.java
index 9922ccaed..050f5bd65 100644
--- a/src/main/java/edu/rpi/legup/controller/EditorElementController.java
+++ b/src/main/java/edu/rpi/legup/controller/EditorElementController.java
@@ -25,19 +25,33 @@ public EditorElementController() {
prevButton = null;
}
+ /**
+ * Sets the ElementController instance for this controller
+ *
+ * @param elementController the ElementController instance to be set
+ */
public void setElementController(ElementController elementController) {
this.elementController = elementController;
}
+ /**
+ * Handles the event when a button associated with an Element is pressed
+ *
+ * @param element the Element associated with the button that was pressed
+ */
public void buttonPressed(Element element) {
// TODO: implement what happens when element is pressed
- System.out.printf("%s button pressed!\n", element.getElementName());
if (elementController != null) {
elementController.setSelectedElement(element);
}
}
+ /**
+ * Handles action events triggered by buttons
+ *
+ * @param e the event to be processed
+ */
@Override
public void actionPerformed(ActionEvent e) {
lastSource = e.getSource();
diff --git a/src/main/java/edu/rpi/legup/controller/ElementController.java b/src/main/java/edu/rpi/legup/controller/ElementController.java
index 5840650e1..436b078b9 100644
--- a/src/main/java/edu/rpi/legup/controller/ElementController.java
+++ b/src/main/java/edu/rpi/legup/controller/ElementController.java
@@ -26,13 +26,17 @@
import java.awt.*;
import java.awt.event.*;
+/**
+ * The ElementController class manages UI interactions related to elements in a {@link BoardView}.
+ * It handles mouse events, key events, and actions related to element selection and manipulation
+ */
public class ElementController
implements MouseListener, MouseMotionListener, ActionListener, KeyListener {
protected BoardView boardView;
private Element selectedElement;
/**
- * ElementController Constructor controller to handles ui events associated interacting with a
+ * ElementController Constructor controller to handle ui events associated interacting with a
* {@link BoardView}
*/
public ElementController() {
@@ -86,6 +90,7 @@ public void mouseReleased(MouseEvent e) {
if (boardView == null) {
boardView = getInstance().getLegupUI().getEditorBoardView();
}
+
Board board = boardView.getBoard();
ElementView elementView = boardView.getElement(e.getPoint());
TreeViewSelection selection = null;
@@ -137,17 +142,8 @@ public void mouseReleased(MouseEvent e) {
if (this.boardView.getBoard() instanceof TreeTentBoard) {
scaledPoint.setLocation(scaledPoint.getX() - 1, scaledPoint.getY() - 1);
}
- System.out.printf(
- "selected Element is NOT null, attempting to change board at (%d, %d)\n",
- scaledPoint.x, scaledPoint.y);
- // System.out.println("Before: " + b.getCell(scaledPoint.x,
- // scaledPoint.y).getData());
+
b.setCell(scaledPoint.x, scaledPoint.y, this.selectedElement, e);
- // System.out.println("After: " + b.getCell(scaledPoint.x,
- // scaledPoint.y).getData());
- // } else {
- // System.out.println("selected Element is null!");
- // }
boardView.repaint();
}
diff --git a/src/main/java/edu/rpi/legup/history/CommandError.java b/src/main/java/edu/rpi/legup/history/CommandError.java
index f848485eb..78027f6ab 100644
--- a/src/main/java/edu/rpi/legup/history/CommandError.java
+++ b/src/main/java/edu/rpi/legup/history/CommandError.java
@@ -25,10 +25,20 @@ public enum CommandError {
private String value;
+ /**
+ * Constructs a CommandError with the specified error message
+ *
+ * @param value The error message associated with the command error
+ */
CommandError(String value) {
this.value = value;
}
+ /**
+ * Returns the error message associated with this CommandError
+ *
+ * @return The error message
+ */
@Override
public String toString() {
return value;
diff --git a/src/main/java/edu/rpi/legup/history/CommandState.java b/src/main/java/edu/rpi/legup/history/CommandState.java
index 90e44b70c..b72f15480 100644
--- a/src/main/java/edu/rpi/legup/history/CommandState.java
+++ b/src/main/java/edu/rpi/legup/history/CommandState.java
@@ -12,10 +12,20 @@ public enum CommandState {
private String value;
+ /**
+ * Constructs a CommandState with the specified state name
+ *
+ * @param value The name associated with the command state
+ */
CommandState(String value) {
this.value = value;
}
+ /**
+ * Returns the name associated with this CommandState
+ *
+ * @return The state name
+ */
@Override
public String toString() {
return value;
diff --git a/src/main/java/edu/rpi/legup/history/DeleteTreeElementCommand.java b/src/main/java/edu/rpi/legup/history/DeleteTreeElementCommand.java
index f56437400..bfd9cd7f5 100644
--- a/src/main/java/edu/rpi/legup/history/DeleteTreeElementCommand.java
+++ b/src/main/java/edu/rpi/legup/history/DeleteTreeElementCommand.java
@@ -32,6 +32,9 @@ public void executeCommand() {
Puzzle puzzle = GameBoardFacade.getInstance().getPuzzleModule();
List selectedViews = selection.getSelectedViews();
+ if (selectedViews.isEmpty()) {
+ return;
+ }
TreeElementView firstSelectedView = selectedViews.get(0);
TreeElementView newSelectedView;
@@ -48,6 +51,7 @@ public void executeCommand() {
}
for (TreeElementView selectedView : selectedViews) {
+ System.out.println("DELETED");
TreeElement element = selectedView.getTreeElement();
tree.removeTreeElement(element);
puzzle.notifyTreeListeners(listener -> listener.onTreeElementRemoved(element));
diff --git a/src/main/java/edu/rpi/legup/history/EditDataCommand.java b/src/main/java/edu/rpi/legup/history/EditDataCommand.java
index a06a26854..0d51cc717 100644
--- a/src/main/java/edu/rpi/legup/history/EditDataCommand.java
+++ b/src/main/java/edu/rpi/legup/history/EditDataCommand.java
@@ -9,6 +9,7 @@
import edu.rpi.legup.model.tree.*;
import edu.rpi.legup.ui.boardview.BoardView;
import edu.rpi.legup.ui.boardview.ElementView;
+import edu.rpi.legup.ui.lookandfeel.materialdesign.MaterialColors;
import edu.rpi.legup.ui.proofeditorui.treeview.*;
import java.awt.*;
import java.awt.event.MouseEvent;
@@ -33,7 +34,7 @@ public class EditDataCommand extends PuzzleCommand {
* EditDataCommand Constructor create a puzzle command for editing a board
*
* @param elementView currently selected puzzle puzzleElement view that is being edited
- * @param selection currently selected tree puzzleElement views that is being edited
+ * @param selection currently selected tree puzzleElement views that are being edited
* @param event mouse event
*/
public EditDataCommand(ElementView elementView, TreeViewSelection selection, MouseEvent event) {
@@ -61,16 +62,13 @@ public void executeCommand() {
if (treeElement.getType() == TreeElementType.NODE) {
TreeNode treeNode = (TreeNode) treeElement;
-
if (treeNode.getChildren().isEmpty()) {
if (transition == null) {
transition = tree.addNewTransition(treeNode);
}
puzzle.notifyTreeListeners(listener -> listener.onTreeElementAdded(transition));
}
-
board = transition.getBoard();
-
puzzleElement = board.getPuzzleElement(selectedPuzzleElement);
savePuzzleElement = puzzleElement.copy();
} else {
@@ -80,7 +78,6 @@ public void executeCommand() {
}
Board prevBoard = transition.getParents().get(0).getBoard();
-
boardView.getElementController().changeCell(event, puzzleElement);
if (prevBoard.getPuzzleElement(selectedPuzzleElement).equalsData(puzzleElement)) {
@@ -109,28 +106,33 @@ public void executeCommand() {
public String getErrorString() {
List selectedViews = selection.getSelectedViews();
if (selectedViews.size() != 1) {
+ flashTreeViewRed();
return CommandError.ONE_SELECTED_VIEW.toString();
}
TreeElementView selectedView = selection.getFirstSelection();
Board board = selectedView.getTreeElement().getBoard();
PuzzleElement selectedPuzzleElement = elementView.getPuzzleElement();
if (selectedView.getType() == TreeElementType.NODE) {
-
TreeNodeView nodeView = (TreeNodeView) selectedView;
if (!nodeView.getChildrenViews().isEmpty()) {
+ flashTreeViewRed();
return CommandError.UNMODIFIABLE_BOARD.toString();
- } else {
- if (!board.getPuzzleElement(selectedPuzzleElement).isModifiable()) {
- return CommandError.UNMODIFIABLE_DATA.toString();
- }
+ } else if (!board.getPuzzleElement(selectedPuzzleElement).isModifiable()) {
+ flashTreeViewRed();
+ return CommandError.UNMODIFIABLE_DATA.toString();
}
} else {
TreeTransitionView transitionView = (TreeTransitionView) selectedView;
if (!transitionView.getTreeElement().getBoard().isModifiable()) {
+ flashTreeViewRed();
return CommandError.UNMODIFIABLE_BOARD.toString();
} else {
if (!board.getPuzzleElement(selectedPuzzleElement).isModifiable()) {
+ flashTreeViewRed();
return CommandError.UNMODIFIABLE_DATA.toString();
+ } else if (!board.getPuzzleElement(selectedPuzzleElement).isModifiableCaseRule()) {
+ flashTreeViewRed();
+ return CommandError.UNMODIFIABLE_DATA_CASE_RULE.toString();
}
}
}
diff --git a/src/main/java/edu/rpi/legup/history/History.java b/src/main/java/edu/rpi/legup/history/History.java
index 091d381ac..9eaeb3e75 100644
--- a/src/main/java/edu/rpi/legup/history/History.java
+++ b/src/main/java/edu/rpi/legup/history/History.java
@@ -48,7 +48,10 @@ public void pushChange(ICommand command) {
}
}
- /** Undoes an action */
+ /**
+ * Undoes the last action by calling the undo method of the command at the current index.
+ * Updates the current index and notifies listeners.
+ */
public void undo() {
synchronized (lock) {
if (curIndex > -1) {
@@ -63,7 +66,10 @@ public void undo() {
}
}
- /** Redoes an action */
+ /**
+ * Redoes the next action by calling the redo method of the command at the current index.
+ * Updates the current index and notifies listeners.
+ */
public void redo() {
synchronized (lock) {
if (curIndex < history.size() - 1) {
diff --git a/src/main/java/edu/rpi/legup/history/IHistoryListener.java b/src/main/java/edu/rpi/legup/history/IHistoryListener.java
index 344d8ede7..a752cd53a 100644
--- a/src/main/java/edu/rpi/legup/history/IHistoryListener.java
+++ b/src/main/java/edu/rpi/legup/history/IHistoryListener.java
@@ -6,15 +6,16 @@
* as pushing, undoing, redoing commands, and clearing the history.
*/
public interface IHistoryListener {
+
/**
- * Called when a action is pushed onto the edu.rpi.legup.history stack
+ * Called when a command is pushed onto the history stack.
*
- * @param command action to push onto the stack
+ * @param command the command that was pushed onto the stack
*/
void onPushChange(ICommand command);
/**
- * Called when an action is undone
+ * Called when a command is undone.
*
* @param isBottom true if there are no more actions to undo, false otherwise
* @param isTop true if there are no more changes to redo, false otherwise
@@ -22,7 +23,7 @@ public interface IHistoryListener {
void onUndo(boolean isBottom, boolean isTop);
/**
- * Called when an action is redone
+ * Called when a command is redone.
*
* @param isBottom true if there are no more actions to undo, false otherwise
* @param isTop true if there are no more changes to redo, false otherwise
diff --git a/src/main/java/edu/rpi/legup/history/IHistorySubject.java b/src/main/java/edu/rpi/legup/history/IHistorySubject.java
index 5c10e9b25..b526c7cf3 100644
--- a/src/main/java/edu/rpi/legup/history/IHistorySubject.java
+++ b/src/main/java/edu/rpi/legup/history/IHistorySubject.java
@@ -8,10 +8,11 @@
* listeners.
*/
public interface IHistorySubject {
+
/**
- * Adds a history listener
+ * Adds a history listener to receive updates about changes in the command history.
*
- * @param listener listener to add
+ * @param listener the listener to add
*/
void addHistoryListener(IHistoryListener listener);
@@ -19,12 +20,12 @@ public interface IHistorySubject {
* Removes a history listener, so it no longer receives updates about changes in the command
* history.
*
- * @param listener listener to remove
+ * @param listener the listener to remove
*/
void removeHistoryListener(IHistoryListener listener);
/**
- * Notifies listeners
+ * Notifies all registered listeners about a change in the command history.
*
* @param algorithm a Consumer function that takes an IHistoryListener and performs some action
* with it
diff --git a/src/main/java/edu/rpi/legup/history/InvalidCommandStateTransition.java b/src/main/java/edu/rpi/legup/history/InvalidCommandStateTransition.java
index 692c572d7..0e1ba1b0d 100644
--- a/src/main/java/edu/rpi/legup/history/InvalidCommandStateTransition.java
+++ b/src/main/java/edu/rpi/legup/history/InvalidCommandStateTransition.java
@@ -6,6 +6,13 @@
*/
public class InvalidCommandStateTransition extends RuntimeException {
+ /**
+ * Constructs a new InvalidCommandStateTransition exception with a detailed message
+ *
+ * @param puzzleCommand the PuzzleCommand involved in the invalid transition
+ * @param from the state from which the transition was attempted
+ * @param to the state to which the transition was attempted
+ */
public InvalidCommandStateTransition(
PuzzleCommand puzzleCommand, CommandState from, CommandState to) {
super(
diff --git a/src/main/java/edu/rpi/legup/history/ValidateDirectRuleCommand.java b/src/main/java/edu/rpi/legup/history/ValidateDirectRuleCommand.java
index b0daee986..49bf378ee 100644
--- a/src/main/java/edu/rpi/legup/history/ValidateDirectRuleCommand.java
+++ b/src/main/java/edu/rpi/legup/history/ValidateDirectRuleCommand.java
@@ -18,6 +18,7 @@
* ICommand interface.
*/
public class ValidateDirectRuleCommand extends PuzzleCommand {
+ private static final Logger LOGGER = LogManager.getLogger(History.class.getName());
private TreeViewSelection selection;
private Map oldRules;
@@ -49,14 +50,15 @@ public void executeCommand() {
for (TreeElementView selectedView : selectedViews) {
TreeElement element = selectedView.getTreeElement();
TreeTransitionView transitionView;
+
if (element.getType() == TreeElementType.NODE) {
TreeNodeView nodeView = (TreeNodeView) selectedView;
transitionView = nodeView.getChildrenViews().get(0);
} else {
transitionView = (TreeTransitionView) selectedView;
}
- TreeTransition transition = transitionView.getTreeElement();
+ TreeTransition transition = transitionView.getTreeElement();
oldRules.put(transition, transition.getRule());
transition.setRule(newRule);
@@ -73,17 +75,41 @@ public void executeCommand() {
final TreeNode finalNode = childNode;
puzzle.notifyTreeListeners(listener -> listener.onTreeElementAdded(finalNode));
}
- newSelection.addToSelection(treeView.getElementView(childNode));
+
+ TreeElementView childView = treeView.getElementView(childNode);
+ if (childView == null) {
+ LOGGER.error("Child view is null for child node: " + childNode);
+ continue;
+ }
+ newSelection.addToSelection(childView);
}
+
TreeElementView firstSelectedView = selection.getFirstSelection();
+
final TreeElement finalTreeElement;
if (firstSelectedView.getType() == TreeElementType.NODE) {
TreeNodeView nodeView = (TreeNodeView) firstSelectedView;
+ if (nodeView.getChildrenViews().isEmpty()) {
+ LOGGER.error("NodeView has no children views");
+ return;
+ }
finalTreeElement = nodeView.getChildrenViews().get(0).getTreeElement();
} else {
TreeTransitionView transitionView = (TreeTransitionView) firstSelectedView;
+ TreeNodeView childView = transitionView.getChildView();
+ if (childView == null) {
+ LOGGER.error("Child view is null for transition view: " + transitionView);
+ TreeNode childNode = transitionView.getTreeElement().getChildNode();
+ childView = (TreeNodeView) treeView.getElementView(childNode);
+ transitionView.setChildView(childView);
+ }
+ TreeTransition transition = transitionView.getTreeElement();
+ if (transition.getParents().get(0).getChildren().isEmpty()) {
+ transition.getParents().get(0).addChild(transition);
+ }
finalTreeElement = transitionView.getChildView().getTreeElement();
}
+
puzzle.notifyBoardListeners(listener -> listener.onTreeElementChanged(finalTreeElement));
puzzle.notifyTreeListeners(listener -> listener.onTreeSelectionChanged(newSelection));
}
@@ -133,6 +159,7 @@ public void undoCommand() {
transitionView = (TreeTransitionView) selectedView;
}
TreeTransition transition = transitionView.getTreeElement();
+
transition.setRule(oldRules.get(transition));
if (addNode.get(transition) != null) {
diff --git a/src/main/java/edu/rpi/legup/model/Puzzle.java b/src/main/java/edu/rpi/legup/model/Puzzle.java
index 1e3d0b3fe..da8f76ddd 100644
--- a/src/main/java/edu/rpi/legup/model/Puzzle.java
+++ b/src/main/java/edu/rpi/legup/model/Puzzle.java
@@ -58,7 +58,6 @@ public abstract class Puzzle implements IBoardSubject, ITreeSubject {
protected List contradictionRules;
protected List caseRules;
protected List placeableElements;
- protected List nonPlaceableElements;
/** Puzzle Constructor - creates a new Puzzle */
public Puzzle() {
@@ -70,7 +69,6 @@ public Puzzle() {
this.caseRules = new ArrayList<>();
this.placeableElements = new ArrayList<>();
- this.nonPlaceableElements = new ArrayList<>();
registerRules();
registerPuzzleElements();
@@ -109,9 +107,6 @@ private void registerPuzzleElements() {
case PLACEABLE:
this.addPlaceableElement((PlaceableElement) element);
break;
- case NONPLACEABLE:
- this.addNonPlaceableElement((NonPlaceableElement) element);
- break;
default:
break;
}
@@ -122,14 +117,6 @@ private void registerPuzzleElements() {
}
}
}
-
- // } catch (IOException | ClassNotFoundException | NoSuchMethodException |
- // InstantiationException | IllegalAccessException |
- // InvocationTargetException
- // e) {
- // LOGGER.error("Unable to find rules for " +
- // this.getClass().getSimpleName(), e);
- // }
} catch (Exception e) {
LOGGER.error("Unable to find elements for " + this.getClass().getSimpleName(), e);
}
@@ -185,14 +172,6 @@ private void registerRules() {
}
}
}
-
- // } catch (IOException | ClassNotFoundException | NoSuchMethodException |
- // InstantiationException | IllegalAccessException |
- // InvocationTargetException
- // e) {
- // LOGGER.error("Unable to find rules for " +
- // this.getClass().getSimpleName(), e);
- // }
} catch (Exception e) {
LOGGER.error("Unable to find rules for " + this.getClass().getSimpleName(), e);
}
@@ -221,10 +200,10 @@ public boolean isValidDimensions(int rows, int columns) {
}
/**
- * Checks if the given array of statements is valid text input for the given puzzle
+ * Checks if the provided text input is valid for the puzzle.
*
- * @param statements
- * @return
+ * @param statements array of statements to check
+ * @return true if input is valid, false otherwise
*/
public boolean isValidTextInput(String[] statements) {
return statements.length > 0;
@@ -356,14 +335,15 @@ public List getDirectRules() {
return directRules;
}
+ /**
+ * Gets the list of placeable elements.
+ *
+ * @return list of PlaceableElement instances
+ */
public List getPlaceableElements() {
return placeableElements;
}
- public List getNonPlaceableElements() {
- return nonPlaceableElements;
- }
-
/**
* Sets the list of direct rules
*
@@ -382,14 +362,15 @@ public void addDirectRule(DirectRule rule) {
directRules.add(rule);
}
+ /**
+ * Adds a placeable element to this puzzle.
+ *
+ * @param element PlaceableElement to add
+ */
public void addPlaceableElement(PlaceableElement element) {
placeableElements.add(element);
}
- public void addNonPlaceableElement(NonPlaceableElement element) {
- nonPlaceableElements.add(element);
- }
-
/**
* Remove a basic rule from this Puzzle
*
@@ -605,7 +586,7 @@ public void setFactory(ElementFactory factory) {
* Adds a board listener to the list of listeners. This allows the puzzle to notify the listener
* about changes to the board.
*
- * @param listener listener to add
+ * @param listener The IBoardListener to be added to the list of listeners.
*/
@Override
public void addBoardListener(IBoardListener listener) {
@@ -616,7 +597,7 @@ public void addBoardListener(IBoardListener listener) {
* Removes a board listener from the list of listeners. This prevents the puzzle from notifying
* the listener about future changes to the board.
*
- * @param listener listener to remove
+ * @param listener The IBoardListener to be removed from the list of listeners.
*/
@Override
public void removeBoardListener(IBoardListener listener) {
@@ -639,7 +620,7 @@ public void notifyBoardListeners(Consumer super IBoardListener> algorithm) {
* Adds a tree listener to the list of listeners. This allows the puzzle to notify the listener
* about changes to the tree.
*
- * @param listener listener to add
+ * @param listener The ITreeListener to be added to the list of listeners.
*/
@Override
public void addTreeListener(ITreeListener listener) {
@@ -650,7 +631,7 @@ public void addTreeListener(ITreeListener listener) {
* Removes a tree listener from the list of listeners. This prevents the puzzle from notifying
* the listener about future changes to the tree.
*
- * @param listener listener to remove
+ * @param listener The ITreeListener to be removed from the list of listeners.
*/
@Override
public void removeTreeListener(ITreeListener listener) {
@@ -673,7 +654,7 @@ public void notifyTreeListeners(Consumer super ITreeListener> algorithm) {
* Checks if the puzzle is valid. The implementation of this method can vary based on the
* specific criteria for puzzle validity.
*
- * @return if the puzzle is valid
+ * @return true if the puzzle is valid, false otherwise.
*/
public boolean checkValidity() {
return true;
diff --git a/src/main/java/edu/rpi/legup/model/PuzzleImporter.java b/src/main/java/edu/rpi/legup/model/PuzzleImporter.java
index 848e3169a..9d16d94e9 100644
--- a/src/main/java/edu/rpi/legup/model/PuzzleImporter.java
+++ b/src/main/java/edu/rpi/legup/model/PuzzleImporter.java
@@ -69,10 +69,10 @@ public void initializePuzzle(String[] statements)
}
/**
- * Initializes the puzzle attributes
+ * Initializes the puzzle attributes from the XML document node
*
- * @param node xml document node
- * @throws InvalidFileFormatException if file is invalid
+ * @param node the XML document node representing the puzzle
+ * @throws InvalidFileFormatException if the file format is invalid
*/
public void initializePuzzle(Node node) throws InvalidFileFormatException {
if (node.getNodeName().equalsIgnoreCase("puzzle")) {
@@ -125,19 +125,19 @@ public void initializePuzzle(Node node) throws InvalidFileFormatException {
}
/**
- * Creates the board for building
+ * Initializes the board with the specified number of rows and columns.
*
- * @param rows number of rows on the puzzle
- * @param columns number of columns on the puzzle
- * @throws RuntimeException if board can not be created
+ * @param rows the number of rows on the puzzle
+ * @param columns the number of columns on the puzzle
+ * @throws RuntimeException if the board cannot be created with the provided dimensions
*/
public abstract void initializeBoard(int rows, int columns);
/**
- * Creates an empty board for building
+ * Initializes the board from the XML document node.
*
- * @param node xml document node
- * @throws InvalidFileFormatException if file is invalid
+ * @param node the XML document node representing the board
+ * @throws InvalidFileFormatException if the file format is invalid
*/
public abstract void initializeBoard(Node node) throws InvalidFileFormatException;
@@ -454,7 +454,7 @@ protected void createDefaultTree() {
}
/**
- * Gets the result of building the Puzzle
+ * Gets the result of building the Puzzle object.
*
* @return puzzle
*/
diff --git a/src/main/java/edu/rpi/legup/model/elements/Element.java b/src/main/java/edu/rpi/legup/model/elements/Element.java
index 6f100c27d..ed47f6523 100644
--- a/src/main/java/edu/rpi/legup/model/elements/Element.java
+++ b/src/main/java/edu/rpi/legup/model/elements/Element.java
@@ -21,6 +21,14 @@ public abstract class Element {
private final String INVALID_USE_MESSAGE;
+ /**
+ * Constructs an Element with the specified ID, name, description, and image name
+ *
+ * @param elementID Unique identifier for the element
+ * @param elementName Name of the element
+ * @param description Description of the element
+ * @param imageName File name of the image associated with the element
+ */
public Element(String elementID, String elementName, String description, String imageName) {
this.elementID = elementID;
this.elementName = elementName;
@@ -55,30 +63,65 @@ private void loadImage() {
}
}
+ /**
+ * Gets the name of the element
+ *
+ * @return The name of the element
+ */
public String getElementName() {
return elementName;
}
+ /**
+ * Sets the name of the element
+ *
+ * @param elementName The new name for the element
+ */
public void setElementName(String elementName) {
this.elementName = elementName;
}
+ /**
+ * Gets the unique identifier of the element
+ *
+ * @return The ID of the element
+ */
public String getElementID() {
return elementID;
}
+ /**
+ * Gets the description of the element
+ *
+ * @return The description of the element
+ */
public String getDescription() {
return description;
}
+ /**
+ * Gets the image icon associated with the element
+ *
+ * @return The ImageIcon for the element
+ */
public ImageIcon getImageIcon() {
return image;
}
+ /**
+ * Gets the type of the element
+ *
+ * @return The ElementType of the element
+ */
public ElementType getElementType() {
return elementType;
}
+ /**
+ * Gets the message for invalid use of the rule
+ *
+ * @return The invalid use message
+ */
public String getInvalidUseOfRuleMessage() {
return this.INVALID_USE_MESSAGE;
}
diff --git a/src/main/java/edu/rpi/legup/model/gameboard/Board.java b/src/main/java/edu/rpi/legup/model/gameboard/Board.java
index f1fd27ad2..9e7d93441 100644
--- a/src/main/java/edu/rpi/legup/model/gameboard/Board.java
+++ b/src/main/java/edu/rpi/legup/model/gameboard/Board.java
@@ -35,21 +35,24 @@ public Board(int size) {
}
/**
- * Gets a specific {@link PuzzleElement} on this board.
+ * Gets a specific {@link PuzzleElement} from the board.
*
- * @param puzzleElement equivalent puzzleElement
- * @return equivalent puzzleElement on this board
+ * @param puzzleElement the puzzle element to retrieve
+ * @return the puzzle element at the corresponding index, or null if not found
*/
public PuzzleElement getPuzzleElement(PuzzleElement puzzleElement) {
+ if (puzzleElement == null) {
+ return null;
+ }
int index = puzzleElement.getIndex();
return index < puzzleElements.size() ? puzzleElements.get(index) : null;
}
/**
- * Sets a specific {@link PuzzleElement} on the board.
+ * Sets a specific {@link PuzzleElement} on the board
*
* @param index index of the puzzleElement
- * @param puzzleElement new puzzleElement at the index
+ * @param puzzleElement the puzzleElement to set at the index
*/
public void setPuzzleElement(int index, PuzzleElement puzzleElement) {
if (index < puzzleElements.size()) {
@@ -58,7 +61,7 @@ public void setPuzzleElement(int index, PuzzleElement puzzleElement) {
}
/**
- * Gets the number of elements on the board.
+ * Gets the number of elements on the board
*
* @return number of elements on the board
*/
diff --git a/src/main/java/edu/rpi/legup/model/gameboard/CaseBoard.java b/src/main/java/edu/rpi/legup/model/gameboard/CaseBoard.java
index 55484a2fc..07cc2947d 100644
--- a/src/main/java/edu/rpi/legup/model/gameboard/CaseBoard.java
+++ b/src/main/java/edu/rpi/legup/model/gameboard/CaseBoard.java
@@ -14,40 +14,88 @@ public class CaseBoard extends Board {
protected CaseRule caseRule;
protected Set pickablePuzzleElements;
+ /**
+ * Constructs a CaseBoard with a base board and a case rule.
+ *
+ * @param baseBoard the base board to use for this CaseBoard
+ * @param caseRule the case rule applied to this CaseBoard
+ */
public CaseBoard(Board baseBoard, CaseRule caseRule) {
this.baseBoard = baseBoard;
this.caseRule = caseRule;
this.pickablePuzzleElements = new HashSet<>();
}
+ /**
+ * Adds a puzzle element to the set of pickable elements.
+ *
+ * @param puzzleElement the puzzle element to add
+ */
public void addPickableElement(PuzzleElement puzzleElement) {
pickablePuzzleElements.add(puzzleElement);
}
+ /**
+ * Removes a puzzle element from the set of pickable elements.
+ *
+ * @param puzzleElement the puzzle element to remove
+ */
public void removePickableElement(PuzzleElement puzzleElement) {
pickablePuzzleElements.remove(puzzleElement);
}
+ /**
+ * Checks if a puzzle element is pickable based on the mouse event.
+ *
+ * @param puzzleElement the puzzle element to check
+ * @param e the mouse event
+ * @return true if the puzzle element is pickable, false otherwise
+ */
public boolean isPickable(PuzzleElement puzzleElement, MouseEvent e) {
return pickablePuzzleElements.contains(baseBoard.getPuzzleElement(puzzleElement));
}
+ /**
+ * Retrieves the base board for this CaseBoard.
+ *
+ * @return the base board
+ */
public Board getBaseBoard() {
return baseBoard;
}
+ /**
+ * Sets the base board for this CaseBoard.
+ *
+ * @param baseBoard the new base board
+ */
public void setBaseBoard(Board baseBoard) {
this.baseBoard = baseBoard;
}
+ /**
+ * Retrieves the case rule for this CaseBoard.
+ *
+ * @return the case rule
+ */
public CaseRule getCaseRule() {
return caseRule;
}
+ /**
+ * Sets the case rule for this CaseBoard.
+ *
+ * @param caseRule the new case rule
+ */
public void setCaseRule(CaseRule caseRule) {
this.caseRule = caseRule;
}
+ /**
+ * Gets the count of pickable puzzle elements.
+ *
+ * @return the number of pickable elements
+ */
public int getCount() {
return pickablePuzzleElements.size();
}
diff --git a/src/main/java/edu/rpi/legup/model/gameboard/ElementFactory.java b/src/main/java/edu/rpi/legup/model/gameboard/ElementFactory.java
index bfc785bdd..131feb122 100644
--- a/src/main/java/edu/rpi/legup/model/gameboard/ElementFactory.java
+++ b/src/main/java/edu/rpi/legup/model/gameboard/ElementFactory.java
@@ -5,6 +5,9 @@
import org.w3c.dom.Element;
import org.w3c.dom.Node;
+/**
+ * ElementFactory is an abstract class for importing and exporting {@link PuzzleElement} instances.
+ */
public abstract class ElementFactory {
/**
diff --git a/src/main/java/edu/rpi/legup/model/gameboard/GridBoard.java b/src/main/java/edu/rpi/legup/model/gameboard/GridBoard.java
index 9593690ce..7338132f8 100644
--- a/src/main/java/edu/rpi/legup/model/gameboard/GridBoard.java
+++ b/src/main/java/edu/rpi/legup/model/gameboard/GridBoard.java
@@ -6,6 +6,11 @@
import java.awt.*;
import java.awt.event.MouseEvent;
+/**
+ * GridBoard represents a grid-based board where each cell can be manipulated based on its
+ * coordinates. The board supports operations such as getting and setting cells, and provides
+ * dimensions of the grid. It also supports deep copying of the board.
+ */
public class GridBoard extends Board {
protected Dimension dimension;
diff --git a/src/main/java/edu/rpi/legup/model/gameboard/GridRegion.java b/src/main/java/edu/rpi/legup/model/gameboard/GridRegion.java
index a4e2343b2..79027d9d1 100644
--- a/src/main/java/edu/rpi/legup/model/gameboard/GridRegion.java
+++ b/src/main/java/edu/rpi/legup/model/gameboard/GridRegion.java
@@ -30,7 +30,7 @@ public void addCell(T cell) {
/**
* Removes the cell from the region
*
- * @param cell cell to be remove from the region
+ * @param cell cell to be removed from the region
*/
public void removeCell(T cell) {
regionCells.remove(cell);
@@ -53,9 +53,4 @@ public List getCells() {
public int getSize() {
return regionCells.size();
}
-
- /*
- public void colorRegion(){}
- */
-
}
diff --git a/src/main/java/edu/rpi/legup/model/gameboard/PuzzleElement.java b/src/main/java/edu/rpi/legup/model/gameboard/PuzzleElement.java
index 7664c7aa0..eca6f3be4 100644
--- a/src/main/java/edu/rpi/legup/model/gameboard/PuzzleElement.java
+++ b/src/main/java/edu/rpi/legup/model/gameboard/PuzzleElement.java
@@ -15,6 +15,7 @@ public abstract class PuzzleElement {
protected T data;
protected boolean isModifiable;
protected boolean isModified;
+ protected boolean isModifiableCaseRule;
protected boolean isGiven;
protected boolean isValid;
protected int casesDepended;
@@ -24,6 +25,7 @@ public PuzzleElement() {
this.index = -1;
this.data = null;
this.isModifiable = true;
+ this.isModifiableCaseRule = true;
this.isModified = false;
this.isGiven = false;
this.isValid = true;
@@ -80,6 +82,24 @@ public void setModifiable(boolean isModifiable) {
this.isModifiable = isModifiable;
}
+ /**
+ * Gets whether this puzzle element is modifiable as a result of a case rule.
+ *
+ * @return true if this puzzle element is modifiable, false otherwise
+ */
+ public boolean isModifiableCaseRule() {
+ return isModifiableCaseRule;
+ }
+
+ /**
+ * Sets whether this puzzle element is modifiable as a result of a case rule.
+ *
+ * @param isModifiableCaseRule true if this puzzle element is modifiable, false otherwise
+ */
+ public void setModifiableCaseRule(boolean isModifiableCaseRule) {
+ this.isModifiableCaseRule = isModifiableCaseRule;
+ }
+
/**
* Gets whether the puzzle element has been modified.
*
diff --git a/src/main/java/edu/rpi/legup/model/observer/IBoardListener.java b/src/main/java/edu/rpi/legup/model/observer/IBoardListener.java
index 7f5122a3d..7ba886fd4 100644
--- a/src/main/java/edu/rpi/legup/model/observer/IBoardListener.java
+++ b/src/main/java/edu/rpi/legup/model/observer/IBoardListener.java
@@ -17,7 +17,7 @@ public interface IBoardListener {
void onTreeElementChanged(TreeElement treeElement);
/**
- * Called when the a case board has been added to the view.
+ * Called when a case board has been added to the view.
*
* @param caseBoard case board to be added
*/
diff --git a/src/main/java/edu/rpi/legup/model/observer/IBoardSubject.java b/src/main/java/edu/rpi/legup/model/observer/IBoardSubject.java
index 2f9a3ff9b..6cdcd7dc2 100644
--- a/src/main/java/edu/rpi/legup/model/observer/IBoardSubject.java
+++ b/src/main/java/edu/rpi/legup/model/observer/IBoardSubject.java
@@ -22,7 +22,7 @@ public interface IBoardSubject {
void removeBoardListener(IBoardListener listener);
/**
- * Notifies all of the listeners using the specified algorithm.
+ * Notifies all the listeners using the specified algorithm.
*
* @param algorithm algorithm used to notify the listeners
*/
diff --git a/src/main/java/edu/rpi/legup/model/observer/ITreeSubject.java b/src/main/java/edu/rpi/legup/model/observer/ITreeSubject.java
index 6bb44296f..f7c033529 100644
--- a/src/main/java/edu/rpi/legup/model/observer/ITreeSubject.java
+++ b/src/main/java/edu/rpi/legup/model/observer/ITreeSubject.java
@@ -8,7 +8,7 @@
*/
public interface ITreeSubject {
/**
- * Adds a board listener.
+ * Adds a tree listener.
*
* @param listener listener to add
*/
@@ -22,7 +22,7 @@ public interface ITreeSubject {
void removeTreeListener(ITreeListener listener);
/**
- * Notifies all of the listeners using the specified algorithm.
+ * Notifies all the tree listeners using the specified algorithm.
*
* @param algorithm algorithm used to notify the listeners
*/
diff --git a/src/main/java/edu/rpi/legup/model/rules/MergeRule.java b/src/main/java/edu/rpi/legup/model/rules/MergeRule.java
index e7e3fd277..86c63fb7c 100644
--- a/src/main/java/edu/rpi/legup/model/rules/MergeRule.java
+++ b/src/main/java/edu/rpi/legup/model/rules/MergeRule.java
@@ -27,7 +27,7 @@ public MergeRule() {
/**
* Checks whether the transition logically follows from the parent node using this rule. This
- * method is the one that should overridden in child classes
+ * method is the one that should have overridden in child classes
*
* @param transition transition to check
* @return null if the child node logically follow from the parent node, otherwise error message
diff --git a/src/main/java/edu/rpi/legup/model/rules/Rule.java b/src/main/java/edu/rpi/legup/model/rules/Rule.java
index 6f5b1fd5c..edcab550b 100644
--- a/src/main/java/edu/rpi/legup/model/rules/Rule.java
+++ b/src/main/java/edu/rpi/legup/model/rules/Rule.java
@@ -164,6 +164,11 @@ public RuleType getRuleType() {
return ruleType;
}
+ /**
+ * Gets the message indicating an invalid use of the rule.
+ *
+ * @return the invalid use message
+ */
public String getInvalidUseOfRuleMessage() {
return this.INVALID_USE_MESSAGE;
}
diff --git a/src/main/java/edu/rpi/legup/model/tree/Tree.java b/src/main/java/edu/rpi/legup/model/tree/Tree.java
index 2f74ca3af..545976fd1 100644
--- a/src/main/java/edu/rpi/legup/model/tree/Tree.java
+++ b/src/main/java/edu/rpi/legup/model/tree/Tree.java
@@ -1,5 +1,6 @@
package edu.rpi.legup.model.tree;
+import edu.rpi.legup.controller.TreeController;
import edu.rpi.legup.model.gameboard.Board;
import edu.rpi.legup.ui.proofeditorui.treeview.TreeView;
import java.util.ArrayList;
@@ -29,6 +30,12 @@ public Tree() {
this.rootNode = null;
}
+ /**
+ * Adds a new transition to the specified node.
+ *
+ * @param treeNode the node to add a transition to
+ * @return the created transition
+ */
public TreeTransition addNewTransition(TreeNode treeNode) {
TreeTransition transition = new TreeTransition(treeNode, treeNode.getBoard().copy());
treeNode.addChild(transition);
@@ -36,13 +43,12 @@ public TreeTransition addNewTransition(TreeNode treeNode) {
return transition;
}
- public TreeNode addNode(TreeTransition transition) {
- TreeNode treeNode = new TreeNode(transition.getBoard().copy());
- transition.setChildNode(treeNode);
- treeNode.setParent(transition);
- return treeNode;
- }
-
+ /**
+ * Adds a tree element (node or transition) to the tree.
+ *
+ * @param element the tree element to add
+ * @return the added tree element
+ */
public TreeElement addTreeElement(TreeElement element) {
if (element.getType() == TreeElementType.NODE) {
TreeNode treeNode = (TreeNode) element;
@@ -51,30 +57,55 @@ public TreeElement addTreeElement(TreeElement element) {
} else {
TreeTransition transition = (TreeTransition) element;
Board copyBoard = transition.board.copy();
- copyBoard.setModifiable(false);
+ copyBoard.setModifiable(true);
return addTreeElement(transition, new TreeNode(copyBoard));
}
}
+ /**
+ * Adds a tree node and its associated transition to the tree.
+ *
+ * @param treeNode the tree node to add
+ * @param transition the transition to associate with the node
+ * @return the added transition
+ */
public TreeElement addTreeElement(TreeNode treeNode, TreeTransition transition) {
treeNode.addChild(transition);
treeNode.getChildren().forEach(TreeTransition::reverify);
return transition;
}
+ /**
+ * Adds a transition and its associated tree node to the tree.
+ *
+ * @param transition the transition to add
+ * @param treeNode the tree node to associate with the transition
+ * @return the added tree node
+ */
public TreeElement addTreeElement(TreeTransition transition, TreeNode treeNode) {
transition.setChildNode(treeNode);
treeNode.setParent(transition);
return treeNode;
}
+ /**
+ * Removes a tree element (node or transition) from the tree.
+ *
+ * @param element the tree element to remove
+ */
public void removeTreeElement(TreeElement element) {
if (element.getType() == TreeElementType.NODE) {
TreeNode node = (TreeNode) element;
+
+ node.getParent().removeChild(node);
node.getParent().setChildNode(null);
} else {
TreeTransition transition = (TreeTransition) element;
+
transition.getParents().forEach(n -> n.removeChild(transition));
+ TreeController treeController = new TreeController();
+ TreeView treeView = new TreeView(treeController);
+ treeView.removeTreeTransition(transition);
transition.getParents().get(0).getChildren().forEach(TreeTransition::reverify);
}
}
@@ -101,10 +132,10 @@ public Set getLeafTreeElements() {
}
/**
- * Gets a Set of TreeNodes that are leaf nodes from the sub tree rooted at the specified node
+ * Gets a Set of TreeNodes that are leaf nodes from the subtree rooted at the specified node
*
* @param node node that is input
- * @return Set of TreeNodes that are leaf nodes from the sub tree
+ * @return Set of TreeNodes that are leaf nodes from the subtree
*/
public Set getLeafTreeElements(TreeNode node) {
Set leafs = new HashSet<>();
diff --git a/src/main/java/edu/rpi/legup/model/tree/TreeElement.java b/src/main/java/edu/rpi/legup/model/tree/TreeElement.java
index e668a5c8f..8c865b89b 100644
--- a/src/main/java/edu/rpi/legup/model/tree/TreeElement.java
+++ b/src/main/java/edu/rpi/legup/model/tree/TreeElement.java
@@ -28,7 +28,7 @@ public TreeElement(TreeElementType type) {
public abstract boolean isContradictoryBranch();
/**
- * Recursively determines if the sub-tree rooted at this tree puzzleElement is valid by checking
+ * Recursively determines if the subtree rooted at this tree puzzleElement is valid by checking
* whether this tree puzzleElement and all descendants of this tree puzzleElement is justified
* and justified correctly
*
diff --git a/src/main/java/edu/rpi/legup/model/tree/TreeNode.java b/src/main/java/edu/rpi/legup/model/tree/TreeNode.java
index 2fe5ffcd2..d4095afa5 100644
--- a/src/main/java/edu/rpi/legup/model/tree/TreeNode.java
+++ b/src/main/java/edu/rpi/legup/model/tree/TreeNode.java
@@ -61,9 +61,9 @@ public boolean isValidBranch() {
}
/**
- * Gets all of the ancestors of this node
+ * Gets a list of the ancestors of this node
*
- * @return list of all of the ancestors for this node
+ * @return list of all the ancestors for this node
*/
public List getAncestors() {
List ancestors = new ArrayList<>();
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/Binary.java b/src/main/java/edu/rpi/legup/puzzle/binary/Binary.java
index d304dbb24..7b70a43a1 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/Binary.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/Binary.java
@@ -36,18 +36,12 @@ public Board generatePuzzle(int difficulty) {
return null;
}
- // /**
- // * Determines if the given dimensions are valid for Binary
- // *
- // * @param rows the number of rows
- // * @param columns the number of columns
- // * @return true if the given dimensions are valid for Binary, false otherwise
- // */
- // @Override
- // public boolean isValidDimensions(int rows, int columns){
- // return rows >= 2 && rows % 2 == 0 && columns >= 2 && columns % 2 == 0;
- // }
-
+ /**
+ * Determines if the current board is a valid state
+ *
+ * @param board board to check for validity
+ * @return true if board is valid, false otherwise
+ */
@Override
public boolean isBoardComplete(Board board) {
BinaryBoard binaryBoard = (BinaryBoard) board;
@@ -66,6 +60,11 @@ public boolean isBoardComplete(Board board) {
return true;
}
+ /**
+ * Callback for when the board puzzleElement changes
+ *
+ * @param board the board that has changed
+ */
@Override
public void onBoardChange(Board board) {}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryBoard.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryBoard.java
index 9a3d0f083..d5d65ed26 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryBoard.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryBoard.java
@@ -19,6 +19,13 @@ public BinaryBoard(int size) {
this.size = size;
}
+ /**
+ * Gets the cell at the (x,y) position
+ *
+ * @param x x-coordinate
+ * @param y y-coordinate
+ * @return BinaryCell cell at (x,y)
+ */
@Override
public BinaryCell getCell(int x, int y) {
if (y * dimension.width + x >= puzzleElements.size()
@@ -31,6 +38,12 @@ public BinaryCell getCell(int x, int y) {
return (BinaryCell) super.getCell(x, y);
}
+ /**
+ * Get all the binary cells in a row
+ *
+ * @param rowNum row number
+ * @return set of all binary cells in specified rowNum
+ */
public Set getRowCells(int rowNum) {
Set row = new HashSet<>();
for (int i = 0; i < size; i++) {
@@ -91,7 +104,6 @@ public ArrayList getColTypes(int colNum) {
*/
@Override
public BinaryBoard copy() {
- System.out.println("BinaryBoard copy()");
BinaryBoard copy = new BinaryBoard(dimension.width, dimension.height);
for (int x = 0; x < this.dimension.width; x++) {
for (int y = 0; y < this.dimension.height; y++) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCell.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCell.java
index e7be09b8d..0edb8444e 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCell.java
@@ -1,14 +1,26 @@
package edu.rpi.legup.puzzle.binary;
+import edu.rpi.legup.model.elements.Element;
import edu.rpi.legup.model.gameboard.GridCell;
import java.awt.Point;
import java.awt.event.MouseEvent;
public class BinaryCell extends GridCell {
- public BinaryCell(int valueInt, Point location) {
- super(valueInt, location);
+ /**
+ * BinaryCell Constructor - creates a BinaryCell from the specified value and location
+ *
+ * @param value value of the BinaryCell
+ * @param location position of the BinaryCell
+ */
+ public BinaryCell(int value, Point location) {
+ super(value, location);
}
+ /**
+ * Gets the type of this BinaryCell
+ *
+ * @return type of BinaryCell
+ */
public BinaryType getType() {
switch (data) {
case 0:
@@ -25,6 +37,11 @@ public BinaryType getType() {
return null;
}
+ /**
+ * Performs a deep copy on the BinaryCell
+ *
+ * @return a new copy of the BinaryCell that is independent of this one
+ */
@Override
public BinaryCell copy() {
BinaryCell copy = new BinaryCell(data, (Point) location.clone());
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCellFactory.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCellFactory.java
index 890c26656..a819177d6 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCellFactory.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryCellFactory.java
@@ -10,7 +10,14 @@
import org.w3c.dom.Node;
public class BinaryCellFactory extends ElementFactory {
-
+ /**
+ * Creates a puzzleElement based on the xml document Node and adds it to the board
+ *
+ * @param node node that represents the puzzleElement
+ * @param board board to add the newly created cell
+ * @return newly created cell from the xml document Node
+ * @throws InvalidFileFormatException if file is invalid
+ */
public BinaryCell importCell(Node node, Board board) throws InvalidFileFormatException {
try {
if (!node.getNodeName().equalsIgnoreCase("cell")) {
@@ -45,6 +52,13 @@ public BinaryCell importCell(Node node, Board board) throws InvalidFileFormatExc
}
}
+ /**
+ * Creates a xml document puzzleElement from a cell for exporting
+ *
+ * @param document xml document
+ * @param puzzleElement PuzzleElement cell
+ * @return xml PuzzleElement
+ */
public org.w3c.dom.Element exportCell(Document document, PuzzleElement puzzleElement) {
org.w3c.dom.Element cellElement = document.createElement("cell");
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryController.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryController.java
index f2a735c4d..5ec6669a0 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryController.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryController.java
@@ -39,13 +39,13 @@ public void changeCell(MouseEvent e, PuzzleElement data) {
}
} else {
if (e.getButton() == MouseEvent.BUTTON3) {
- if (cell.getData() == 0) {
+ if (cell.getData() == 2) {
data.setData(1);
} else {
if (cell.getData() == 1) {
- data.setData(2);
- } else {
data.setData(0);
+ } else {
+ data.setData(2);
}
}
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryElementView.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryElementView.java
index b523a74a9..74a761e89 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryElementView.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryElementView.java
@@ -7,6 +7,8 @@ public class BinaryElementView extends GridElementView {
private static final Font FONT = new Font("TimesRoman", Font.BOLD, 17);
private static final Color FONT_COLOR = Color.BLACK;
+ private static final Color GIVEN_COLOR = Color.LIGHT_GRAY;
+ private static final Color ELEMENT_COLOR = Color.WHITE;
public BinaryElementView(BinaryCell cell) {
super(cell);
@@ -51,95 +53,35 @@ public void drawElement(Graphics2D graphics2D) {
private void drawCell(Graphics2D graphics2D, Color bgColor) {
BinaryCell cell = (BinaryCell) puzzleElement;
BinaryType type = cell.getType();
- if (type == BinaryType.ZERO) {
+
+ if (type == BinaryType.ZERO || type == BinaryType.ONE) {
graphics2D.setStroke(new BasicStroke(1));
- graphics2D.setColor(Color.LIGHT_GRAY);
+ graphics2D.setColor(bgColor);
graphics2D.fillRect(location.x, location.y, size.width, size.height);
graphics2D.setColor(Color.BLACK);
graphics2D.drawRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(FONT_COLOR);
- graphics2D.setFont(FONT);
- FontMetrics metrics = graphics2D.getFontMetrics(FONT);
- String value = String.valueOf(puzzleElement.getData());
- int xText = location.x + (size.width - metrics.stringWidth(value)) / 2;
- int yText =
- location.y + ((size.height - metrics.getHeight()) / 2) + metrics.getAscent();
- graphics2D.drawString(String.valueOf(puzzleElement.getData()), xText, yText);
- } else {
- if (type == BinaryType.ONE) {
- graphics2D.setStroke(new BasicStroke(1));
- graphics2D.setColor(Color.LIGHT_GRAY);
- graphics2D.fillRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(Color.BLACK);
- graphics2D.drawRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(FONT_COLOR);
- graphics2D.setFont(FONT);
- FontMetrics metrics = graphics2D.getFontMetrics(FONT);
- String value = String.valueOf(puzzleElement.getData());
- int xText = location.x + (size.width - metrics.stringWidth(value)) / 2;
- int yText =
- location.y
- + ((size.height - metrics.getHeight()) / 2)
- + metrics.getAscent();
- graphics2D.drawString(String.valueOf(puzzleElement.getData()), xText, yText);
-
- } else {
- if (type == BinaryType.UNKNOWN) {
- graphics2D.setStroke(new BasicStroke(0));
- graphics2D.setColor(Color.WHITE);
- graphics2D.fillRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(Color.BLACK);
- graphics2D.drawRect(location.x, location.y, size.width, size.height);
- }
- }
- }
- }
-
- @Override
- public void drawElement(Graphics2D graphics2D) {
- BinaryCell cell = (BinaryCell) puzzleElement;
- BinaryType type = cell.getType();
- if (type == BinaryType.ZERO) {
- graphics2D.setStroke(new BasicStroke(1));
+ drawCenteredText(graphics2D);
+ } else if (type == BinaryType.UNKNOWN) {
+ graphics2D.setStroke(new BasicStroke(0));
graphics2D.setColor(Color.WHITE);
graphics2D.fillRect(location.x, location.y, size.width, size.height);
graphics2D.setColor(Color.BLACK);
graphics2D.drawRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(FONT_COLOR);
- graphics2D.setFont(FONT);
- FontMetrics metrics = graphics2D.getFontMetrics(FONT);
- String value = String.valueOf(puzzleElement.getData());
- int xText = location.x + (size.width - metrics.stringWidth(value)) / 2;
- int yText =
- location.y + ((size.height - metrics.getHeight()) / 2) + metrics.getAscent();
- graphics2D.drawString(String.valueOf(puzzleElement.getData()), xText, yText);
- } else {
- if (type == BinaryType.ONE) {
- graphics2D.setStroke(new BasicStroke(1));
- graphics2D.setColor(Color.WHITE);
- graphics2D.fillRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(Color.BLACK);
- graphics2D.drawRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(FONT_COLOR);
- graphics2D.setFont(FONT);
- FontMetrics metrics = graphics2D.getFontMetrics(FONT);
- String value = String.valueOf(puzzleElement.getData());
- int xText = location.x + (size.width - metrics.stringWidth(value)) / 2;
- int yText =
- location.y
- + ((size.height - metrics.getHeight()) / 2)
- + metrics.getAscent();
- graphics2D.drawString(String.valueOf(puzzleElement.getData()), xText, yText);
-
- } else {
- if (type == BinaryType.UNKNOWN) {
- graphics2D.setStroke(new BasicStroke(0));
- graphics2D.setColor(Color.WHITE);
- graphics2D.fillRect(location.x, location.y, size.width, size.height);
- graphics2D.setColor(Color.BLACK);
- graphics2D.drawRect(location.x, location.y, size.width, size.height);
- }
- }
}
}
+
+ /**
+ * Helper method to draw the centered text within the cell
+ *
+ * @param graphics2D The graphics object to draw on
+ */
+ private void drawCenteredText(Graphics2D graphics2D) {
+ graphics2D.setColor(FONT_COLOR);
+ graphics2D.setFont(FONT);
+ FontMetrics metrics = graphics2D.getFontMetrics(FONT);
+ String value = String.valueOf(puzzleElement.getData());
+ int xText = location.x + (size.width - metrics.stringWidth(value)) / 2;
+ int yText = location.y + ((size.height - metrics.getHeight()) / 2) + metrics.getAscent();
+ graphics2D.drawString(value, xText, yText);
+ }
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryExporter.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryExporter.java
index 43ea8caf4..c032b63a9 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryExporter.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryExporter.java
@@ -33,7 +33,7 @@ protected org.w3c.dom.Element createBoardElement(Document newDocument) {
org.w3c.dom.Element cellsElement = newDocument.createElement("cells");
for (PuzzleElement puzzleElement : board.getPuzzleElements()) {
BinaryCell cell = (BinaryCell) puzzleElement;
- if (cell.getData() != -2) {
+ if (cell.getData() != BinaryType.UNKNOWN.toValue()) {
org.w3c.dom.Element cellElement =
puzzle.getFactory().exportCell(newDocument, puzzleElement);
cellsElement.appendChild(cellElement);
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryImporter.java b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryImporter.java
index 746d3b4b6..419789060 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/BinaryImporter.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/BinaryImporter.java
@@ -12,16 +12,33 @@ public BinaryImporter(Binary binary) {
super(binary);
}
+ /**
+ * Determines if puzzle uses row and column input
+ *
+ * @return true if row and column input is used, false otherwise
+ */
@Override
public boolean acceptsRowsAndColumnsInput() {
return true;
}
+ /**
+ * Determines if puzzle uses text input
+ *
+ * @return true if text input is used, false otherwise
+ */
@Override
public boolean acceptsTextInput() {
return false;
}
+ /**
+ * Creates an empty board for building
+ *
+ * @param rows the number of rows on the board
+ * @param columns the number of columns on the board
+ * @throws RuntimeException if board can not be created
+ */
@Override
public void initializeBoard(int rows, int columns) {
BinaryBoard binaryBoard = new BinaryBoard(columns, rows);
@@ -48,12 +65,12 @@ public void initializeBoard(Node node) throws InvalidFileFormatException {
try {
if (!node.getNodeName().equalsIgnoreCase("board")) {
throw new InvalidFileFormatException(
- "binary Importer: cannot find board puzzleElement");
+ "Binary Importer: cannot find board puzzleElement");
}
Element boardElement = (Element) node;
if (boardElement.getElementsByTagName("cells").getLength() == 0) {
throw new InvalidFileFormatException(
- "binary Importer: no puzzleElement found for board");
+ "Binary Importer: no puzzleElement found for board");
}
Element dataElement = (Element) boardElement.getElementsByTagName("cells").item(0);
@@ -76,7 +93,7 @@ public void initializeBoard(Node node) throws InvalidFileFormatException {
int height = binaryBoard.getHeight();
if (binaryBoard == null || width % 2 != 0 || height % 2 != 0) {
- throw new InvalidFileFormatException("binary Importer: invalid board dimensions");
+ throw new InvalidFileFormatException("Binary Importer: invalid board dimensions");
}
for (int i = 0; i < elementDataList.getLength(); i++) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/elements/BlankTile.java b/src/main/java/edu/rpi/legup/puzzle/binary/elements/BlankTile.java
deleted file mode 100644
index 8b1378917..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/binary/elements/BlankTile.java
+++ /dev/null
@@ -1 +0,0 @@
-
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/elements/binary_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/binary/elements/binary_elements_reference_sheet.txt
new file mode 100644
index 000000000..54db0ee0b
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/elements/binary_elements_reference_sheet.txt
@@ -0,0 +1,2 @@
+BINA-ELEM-0001 : NumberTile
+BINA-ELEM-0002 : UnknownTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/CompleteRowColumnDirectRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/CompleteRowColumnDirectRule.java
index 09cca6dc5..6ced37a9c 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/CompleteRowColumnDirectRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/rules/CompleteRowColumnDirectRule.java
@@ -32,23 +32,27 @@ public CompleteRowColumnDirectRule() {
@Override
public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
BinaryBoard origBoard = (BinaryBoard) transition.getParents().get(0).getBoard();
- ContradictionRule contraRule = new UnbalancedRowOrColumnContradictionRule();
+ ContradictionRule contraRule = new UnbalancedRowColumnContradictionRule();
BinaryCell binaryCell = (BinaryCell) puzzleElement;
BinaryBoard modified = origBoard.copy();
- BinaryCell c = (BinaryCell) modified.getPuzzleElement(puzzleElement);
- // System.out.println("ORIG" + binaryCell.getData());
- // System.out.println("AFTER" + Math.abs(binaryCell.getData() - 1));
- modified.getPuzzleElement(puzzleElement).setData(binaryCell.getData());
- System.out.println(contraRule.checkContradictionAt(modified, puzzleElement));
-
- if (contraRule.checkContradictionAt(modified, puzzleElement) != null) {
+ // Flip the cell and check to see if there will be an unbalanced row/column contradiction,
+ // if so the rule is applied correctly
+ modified.getPuzzleElement(puzzleElement).setData(Math.abs(binaryCell.getData() - 1));
+ if (contraRule.checkContradictionAt(modified, puzzleElement) == null) {
return null;
}
- return "Grouping of Three Ones or Zeros not found";
+ return "Unbalanced row/column found";
}
+ /**
+ * Creates a transition {@link Board} that has this rule applied to it using the {@link
+ * TreeNode}.
+ *
+ * @param node tree node used to create default transition board
+ * @return default board or null if this rule cannot be applied to this tree node
+ */
@Override
public Board getDefaultBoard(TreeNode node) {
return null;
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/OneTileGapDirectRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/OneTileGapDirectRule.java
deleted file mode 100644
index 2e1e96fa5..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/OneTileGapDirectRule.java
+++ /dev/null
@@ -1,64 +0,0 @@
-package edu.rpi.legup.puzzle.binary.rules;
-
-import edu.rpi.legup.model.gameboard.Board;
-import edu.rpi.legup.model.gameboard.PuzzleElement;
-import edu.rpi.legup.model.rules.ContradictionRule;
-import edu.rpi.legup.model.rules.DirectRule;
-import edu.rpi.legup.model.tree.TreeNode;
-import edu.rpi.legup.model.tree.TreeTransition;
-import edu.rpi.legup.puzzle.binary.BinaryBoard;
-import edu.rpi.legup.puzzle.binary.BinaryCell;
-
-public class OneTileGapDirectRule extends DirectRule {
- private final String INVALID_USE_MESSAGE = "Number at cell is incorrect";
-
- public OneTileGapDirectRule() {
- super(
- "BINA-BASC-0002",
- "One Tile Gap",
- "If an empty tile is in between the same value, fill the gap with the other value.",
- "edu/rpi/legup/images/binary/rules/OneTileGapDirectRule.png");
- }
-
- boolean checkLeftRight(BinaryCell c, BinaryBoard board) {
- int x = c.getLocation().x;
- int y = c.getLocation().y;
- return board.getCell(x - 1, y).getType() != c.getType()
- || board.getCell(x + 1, y).getType() != c.getType();
- }
-
- boolean checkUpDown(BinaryCell c, BinaryBoard board) {
- int x = c.getLocation().x;
- int y = c.getLocation().y;
- return board.getCell(x, y - 1).getType() != c.getType()
- || board.getCell(x, y + 1).getType() != c.getType();
- }
-
- @Override
- public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
- BinaryBoard origBoard = (BinaryBoard) transition.getParents().get(0).getBoard();
- ContradictionRule contraRule = new ThreeAdjacentContradictionRule();
- BinaryCell binaryCell = (BinaryCell) puzzleElement;
- BinaryBoard modified = origBoard.copy();
-
- // System.out.println("ORIG" + binaryCell.getData());
- // System.out.println("AFTER" + Math.abs(binaryCell.getData() - 1));
- modified.getPuzzleElement(puzzleElement).setData(Math.abs(binaryCell.getData() - 1));
-
- PuzzleElement newP = binaryCell;
-
- System.out.println(contraRule.checkContradictionAt(modified, newP));
-
- if (contraRule.checkContradictionAt(modified, newP) == null) {
- return null;
- }
- modified.getPuzzleElement(puzzleElement).setData(Math.abs(binaryCell.getData() - 1));
-
- return "Grouping of Three Ones or Zeros not found";
- }
-
- @Override
- public Board getDefaultBoard(TreeNode node) {
- return null;
- }
-}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/DuplicateRowsOrColumnsContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/RepeatedRowColumnContradictionRule.java
similarity index 69%
rename from src/main/java/edu/rpi/legup/puzzle/binary/rules/DuplicateRowsOrColumnsContradictionRule.java
rename to src/main/java/edu/rpi/legup/puzzle/binary/rules/RepeatedRowColumnContradictionRule.java
index 6e009f123..a7a4ced12 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/DuplicateRowsOrColumnsContradictionRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/rules/RepeatedRowColumnContradictionRule.java
@@ -8,19 +8,28 @@
import edu.rpi.legup.puzzle.binary.BinaryType;
import java.util.ArrayList;
-public class DuplicateRowsOrColumnsContradictionRule extends ContradictionRule {
+public class RepeatedRowColumnContradictionRule extends ContradictionRule {
private final String NO_CONTRADICTION_MESSAGE =
"Does not contain a contradiction at this index";
private final String INVALID_USE_MESSAGE = "Row or column must have a value in each cell";
- public DuplicateRowsOrColumnsContradictionRule() {
+ public RepeatedRowColumnContradictionRule() {
super(
"BINA-CONT-0003",
- "Duplicate Rows Or Columns",
- "There must not be two rows or two columns that are duplicates",
- "edu/rpi/legup/images/binary/rules/DuplicateRowOrColumnContradictionRule.png");
+ "Repeated Row/Column",
+ "There must not be two of the same row or two of the same column in the puzzle",
+ "edu/rpi/legup/images/binary/rules/RepeatedRowColumnContradictionRule.png");
}
+ /**
+ * Checks whether the transition has a contradiction at the specific puzzleElement index using
+ * this rule
+ *
+ * @param board board to check contradiction
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the transition contains a contradiction at the specified puzzleElement,
+ * otherwise error message
+ */
@Override
public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
BinaryBoard binaryBoard = (BinaryBoard) board;
@@ -29,11 +38,9 @@ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
// Compare each row with row of current cell to see if they are equal, if so the rule is
// applied correctly
ArrayList row = binaryBoard.getRowTypes(cell.getLocation().y);
-
int size = row.size();
-
for (int i = 0; i < size; i++) {
- if (i > cell.getLocation().y) {
+ if (i != cell.getLocation().y) {
ArrayList currRow = binaryBoard.getRowTypes(i);
if (currRow.equals(row)) {
return null;
@@ -44,9 +51,8 @@ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
// Compare each column with column of current cell to see if they are equal, if so the rule
// is applied correctly
ArrayList col = binaryBoard.getColTypes(cell.getLocation().x);
-
for (int i = 0; i < size; i++) {
- if (i > cell.getLocation().x) {
+ if (i != cell.getLocation().x) {
ArrayList currCol = binaryBoard.getColTypes(i);
if (currCol.equals(col)) {
return null;
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/SurroundPairDirectRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/SurroundPairDirectRule.java
deleted file mode 100644
index dc2f07c8b..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/SurroundPairDirectRule.java
+++ /dev/null
@@ -1,48 +0,0 @@
-package edu.rpi.legup.puzzle.binary.rules;
-
-import edu.rpi.legup.model.gameboard.Board;
-import edu.rpi.legup.model.gameboard.PuzzleElement;
-import edu.rpi.legup.model.rules.ContradictionRule;
-import edu.rpi.legup.model.rules.DirectRule;
-import edu.rpi.legup.model.tree.TreeNode;
-import edu.rpi.legup.model.tree.TreeTransition;
-import edu.rpi.legup.puzzle.binary.BinaryBoard;
-import edu.rpi.legup.puzzle.binary.BinaryCell;
-
-public class SurroundPairDirectRule extends DirectRule {
-
- public SurroundPairDirectRule() {
- super(
- "BINA-BASC-0001",
- "Surround Pair",
- "If two adjacent tiles have the same value, surround the tiles with the other value.",
- "edu/rpi/legup/images/binary/rules/SurroundPairDirectRule.png");
- }
-
- public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
- BinaryBoard origBoard = (BinaryBoard) transition.getParents().get(0).getBoard();
- ContradictionRule contraRule = new ThreeAdjacentContradictionRule();
- BinaryCell binaryCell = (BinaryCell) puzzleElement;
- BinaryBoard modified = origBoard.copy();
-
- // System.out.println("ORIG" + binaryCell.getData());
- // System.out.println("AFTER" + Math.abs(binaryCell.getData() - 1));
- modified.getPuzzleElement(puzzleElement).setData(Math.abs(binaryCell.getData() - 1));
-
- PuzzleElement newP = binaryCell;
-
- System.out.println(contraRule.checkContradictionAt(modified, newP));
-
- if (contraRule.checkContradictionAt(modified, newP) == null) {
- return null;
- }
- modified.getPuzzleElement(puzzleElement).setData(Math.abs(binaryCell.getData() - 1));
-
- return "Grouping of Three Ones or Zeros not found";
- }
-
- @Override
- public Board getDefaultBoard(TreeNode node) {
- return null;
- }
-}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/ThreeAdjacentContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/ThreeAdjacentContradictionRule.java
deleted file mode 100644
index 075642246..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/ThreeAdjacentContradictionRule.java
+++ /dev/null
@@ -1,127 +0,0 @@
-package edu.rpi.legup.puzzle.binary.rules;
-
-import edu.rpi.legup.model.gameboard.Board;
-import edu.rpi.legup.model.gameboard.PuzzleElement;
-import edu.rpi.legup.model.rules.ContradictionRule;
-import edu.rpi.legup.puzzle.binary.BinaryBoard;
-import edu.rpi.legup.puzzle.binary.BinaryCell;
-import edu.rpi.legup.puzzle.binary.BinaryType;
-
-public class ThreeAdjacentContradictionRule extends ContradictionRule {
- private final String NO_CONTRADICTION_MESSAGE =
- "Does not contain a contradiction at this index";
- private final String INVALID_USE_MESSAGE = "Contradiction must be a zero or one";
-
- public ThreeAdjacentContradictionRule() {
- super(
- "BINA-CONT-0001",
- "Three Adjacent",
- "There must not be three adjacent zeros or three adjacent ones in a row or column",
- "edu/rpi/legup/images/binary/rules/ThreeAdjacentContradictionRule.png");
- }
-
- @Override
- public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
- BinaryBoard binaryBoard = (BinaryBoard) board;
- int height = binaryBoard.getHeight();
- int width = binaryBoard.getWidth();
-
- BinaryCell cell = (BinaryCell) binaryBoard.getPuzzleElement(puzzleElement);
- System.out.println("THE CELL IS : " + cell.getType());
- int cellX = cell.getLocation().x;
- int cellY = cell.getLocation().y;
- BinaryCell oneUp = null;
- BinaryCell oneDown = null;
- BinaryCell oneForward = null;
- BinaryCell oneBackward = null;
- BinaryCell twoUp = null;
- BinaryCell twoDown = null;
- BinaryCell twoForward = null;
- BinaryCell twoBackward = null;
- if (binaryBoard.getCell(cellX, cellY + 1) != null) {
- oneUp = binaryBoard.getCell(cellX, cellY + 1);
- }
- if (binaryBoard.getCell(cellX, cellY - 1) != null) {
- oneDown = binaryBoard.getCell(cellX, cellY - 1);
- }
- if (binaryBoard.getCell(cellX + 1, cellY) != null) {
- oneForward = binaryBoard.getCell(cellX + 1, cellY);
- }
- if (binaryBoard.getCell(cellX - 1, cellY) != null) {
- oneBackward = binaryBoard.getCell(cellX - 1, cellY);
- }
- if (binaryBoard.getCell(cellX, cellY + 2) != null) {
- twoUp = binaryBoard.getCell(cellX, cellY + 2);
- }
- if (binaryBoard.getCell(cellX, cellY - 2) != null) {
- twoDown = binaryBoard.getCell(cellX, cellY - 2);
- }
- if (binaryBoard.getCell(cellX + 2, cellY) != null) {
- twoForward = binaryBoard.getCell(cellX + 2, cellY);
- }
- if (binaryBoard.getCell(cellX - 2, cellY) != null) {
- twoBackward = binaryBoard.getCell(cellX - 2, cellY);
- }
-
- if (cell.getType() == BinaryType.ONE || cell.getType() == BinaryType.ZERO) {
- if (twoBackward != null
- && oneBackward != null
- && twoBackward.getType() != BinaryType.UNKNOWN
- && oneBackward.getType() != BinaryType.UNKNOWN) {
- if (twoBackward.getType() == cell.getType()
- && oneBackward.getType() == cell.getType()) {
- System.out.println("1");
- return null;
- }
- }
- if (twoForward != null
- && oneForward != null
- && twoForward.getType() != BinaryType.UNKNOWN
- && oneForward.getType() != BinaryType.UNKNOWN) {
- if (twoForward.getType() == cell.getType()
- && oneForward.getType() == cell.getType()) {
- System.out.println("2");
- return null;
- }
- }
- if (twoDown != null
- && oneDown != null
- && twoDown.getType() != BinaryType.UNKNOWN
- && oneDown.getType() != BinaryType.UNKNOWN) {
- if (twoDown.getType() == cell.getType() && oneDown.getType() == cell.getType()) {
- System.out.println("3");
- return null;
- }
- }
- if (twoUp != null
- && oneUp != null
- && twoUp.getType() != BinaryType.UNKNOWN
- && oneUp.getType() != BinaryType.UNKNOWN) {
- if (twoUp.getType() == cell.getType() && oneUp.getType() == cell.getType()) {
- System.out.println("4");
- return null;
- }
- }
- if (oneBackward != null
- && oneForward != null
- && oneBackward.getType() != BinaryType.UNKNOWN
- && oneForward.getType() != BinaryType.UNKNOWN) {
- if (oneBackward.getType() == cell.getType()
- && oneForward.getType() == cell.getType()) {
- System.out.println("5");
- return null;
- }
- }
- if (oneUp != null
- && oneDown != null
- && oneUp.getType() != BinaryType.UNKNOWN
- && oneDown.getType() != BinaryType.UNKNOWN) {
- if (oneUp.getType() == cell.getType() && oneDown.getType() == cell.getType()) {
- System.out.println("6");
- return null;
- }
- }
- }
- return super.getNoContradictionMessage() + ": " + this.NO_CONTRADICTION_MESSAGE;
- }
-}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/UnbalancedRowOrColumnContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/UnbalancedRowColumnContradictionRule.java
similarity index 66%
rename from src/main/java/edu/rpi/legup/puzzle/binary/rules/UnbalancedRowOrColumnContradictionRule.java
rename to src/main/java/edu/rpi/legup/puzzle/binary/rules/UnbalancedRowColumnContradictionRule.java
index 301cc6ec1..1d00f4da9 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/UnbalancedRowOrColumnContradictionRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/rules/UnbalancedRowColumnContradictionRule.java
@@ -8,20 +8,29 @@
import edu.rpi.legup.puzzle.binary.BinaryType;
import java.util.Set;
-public class UnbalancedRowOrColumnContradictionRule extends ContradictionRule {
+public class UnbalancedRowColumnContradictionRule extends ContradictionRule {
private final String NO_CONTRADICTION_MESSAGE =
"Does not contain a contradiction at this index";
private final String INVALID_USE_MESSAGE = "Row or column must have a value in each cell";
- public UnbalancedRowOrColumnContradictionRule() {
+ public UnbalancedRowColumnContradictionRule() {
super(
"BINA-CONT-0002",
- "Unbalanced Row Or Column",
+ "Unbalanced Row/Column",
"Each row or column must contain an equal number of zeros and ones",
"edu/rpi/legup/images/binary/rules/UnbalancedRowColumnContradictionRule.png");
}
+ /**
+ * Checks whether the transition has a contradiction at the specific puzzleElement index using
+ * this rule
+ *
+ * @param board board to check contradiction
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the transition contains a contradiction at the specified puzzleElement,
+ * otherwise error message
+ */
@Override
public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
BinaryBoard binaryBoard = (BinaryBoard) board;
@@ -41,8 +50,9 @@ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
}
}
- if (rowNumZeros == size / 2 && rowNumOnes == size / 2) {
- return super.getNoContradictionMessage() + ": " + this.NO_CONTRADICTION_MESSAGE;
+ // if there are too many zeros or ones in this row
+ if (rowNumZeros > size / 2 || rowNumOnes > size / 2) {
+ return null;
}
Set col = binaryBoard.getColCells(cell.getLocation().x);
@@ -59,10 +69,11 @@ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
}
}
- if (colNumZeros == size / 2 && colNumOnes == size / 2) {
- return super.getNoContradictionMessage() + ": " + this.NO_CONTRADICTION_MESSAGE;
+ // if there are too many zeros or ones in this column
+ if (colNumZeros > size / 2 || colNumOnes > size / 2) {
+ return null;
}
- return null;
+ return super.getNoContradictionMessage() + ": " + this.NO_CONTRADICTION_MESSAGE;
}
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/OneOrZeroCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/binary/rules/ZeroOrOneCaseRule.java
similarity index 64%
rename from src/main/java/edu/rpi/legup/puzzle/binary/rules/OneOrZeroCaseRule.java
rename to src/main/java/edu/rpi/legup/puzzle/binary/rules/ZeroOrOneCaseRule.java
index 21ee0dd5b..f47fe55b9 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/OneOrZeroCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/rules/ZeroOrOneCaseRule.java
@@ -11,16 +11,23 @@
import java.util.ArrayList;
import java.util.List;
-public class OneOrZeroCaseRule extends CaseRule {
+public class ZeroOrOneCaseRule extends CaseRule {
- public OneOrZeroCaseRule() {
+ public ZeroOrOneCaseRule() {
super(
"BINA-CASE-0001",
- "One or Zero",
- "Each blank cell is either a one or a zero.",
- "edu/rpi/legup/images/binary/rules/OneOrZeroCaseRule.png");
+ "Zero Or One",
+ "Each blank cell is either a zero or a one",
+ "edu/rpi/legup/images/binary/rules/ZeroOrOneCaseRule.png");
}
+ /**
+ * Checks whether the {@link TreeTransition} logically follows from the parent node using this
+ * rule. This method is the one that should be overridden in child classes.
+ *
+ * @param transition transition to check
+ * @return null if the child node logically follow from the parent node, otherwise error message
+ */
@Override
public String checkRuleRaw(TreeTransition transition) {
List childTransitions = transition.getParents().get(0).getChildren();
@@ -52,6 +59,13 @@ public String checkRuleRaw(TreeTransition transition) {
return null;
}
+ /**
+ * Generates a {@link CaseBoard} that includes all blank cells from the given board that this
+ * case rule can be applied to
+ *
+ * @param board The board to find locations where this case rule can be applied
+ * @return A CaseBoard containing pickable elements where the case rule can be applied
+ */
@Override
public CaseBoard getCaseBoard(Board board) {
BinaryBoard binaryBoard = (BinaryBoard) board.copy();
@@ -65,24 +79,44 @@ public CaseBoard getCaseBoard(Board board) {
return caseBoard;
}
+ /**
+ * Gets the possible cases at a specific location based on this case rule
+ *
+ * @param board the current board state
+ * @param puzzleElement equivalent puzzleElement
+ * @return a list of elements the specified could be
+ */
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
Board case1 = board.copy();
PuzzleElement data1 = case1.getPuzzleElement(puzzleElement);
- data1.setData(BinaryType.ZERO.toValue());
+ data1.setData(BinaryType.ONE.toValue());
case1.addModifiedData(data1);
cases.add(case1);
Board case2 = board.copy();
PuzzleElement data2 = case2.getPuzzleElement(puzzleElement);
- data2.setData(BinaryType.ONE.toValue());
+ data2.setData(BinaryType.ZERO.toValue());
case2.addModifiedData(data2);
cases.add(case2);
return cases;
}
+ /**
+ * Checks whether the child node logically follows from the parent node at the specific
+ * puzzleElement index using this rule
+ *
+ * @param transition transition to check
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the child node logically follow from the parent node at the specified
+ * puzzleElement, otherwise error message
+ */
@Override
public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
return null;
diff --git a/src/main/java/edu/rpi/legup/puzzle/binary/rules/binary_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/binary/rules/binary_reference_sheet.txt
index c8cb0d1b9..619d183a5 100644
--- a/src/main/java/edu/rpi/legup/puzzle/binary/rules/binary_reference_sheet.txt
+++ b/src/main/java/edu/rpi/legup/puzzle/binary/rules/binary_reference_sheet.txt
@@ -1,9 +1,11 @@
-BINA-BASC-0001 : SurroundPairDirectRule
-BINA-BASC-0002 : OneTileGapDirectRule
-BINA-BASC-0003 : CompleteRowColumnDirectRule
+BINA-BASC-0001 : PreventTrioContradictionRule
+BINA-BASC-0002 : CompleteRowColumnDirectRule
+BINA-BASC-0003 : SaveBlockerDirectRule
+BINA-BASC-0004 : UniqueRowColumnDirectRule
-BINA-CONT-0001 : ThreeAdjacentContradictionRule
-BINA-CONT-0002 : UnbalancedRowOrColumnContradictionRule
-BINA-CONT-0003 : DuplicateRowsOrColumnsContradictionRule
+BINA-CONT-0001 : TrioContradictionRule
+BINA-CONT-0002 : UnbalancedRowColumnContradictionRule
+BINA-CONT-0003 : RepeatedRowColumnContradictionRule
+BINA-CONT-0004 : WastedBlockerContradictionRule
-BINA-CASE-0001 : OneOrZeroCaseRule
\ No newline at end of file
+BINA-CASE-0001 : ZeroOrOneCaseRule
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/FillapixCell.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/FillapixCell.java
index a9e5aa2df..40c5e4a54 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/FillapixCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/FillapixCell.java
@@ -42,13 +42,13 @@ public void setCellType(FillapixCellType type) {
@Override
public void setType(Element e, MouseEvent m) {
switch (e.getElementID()) {
- case "FPIX-PLAC-0001":
+ case "FPIX-ELEM-0001":
this.setCellType(FillapixCellType.BLACK);
break;
- case "FPIX-PLAC-0002":
+ case "FPIX-ELEM-0004":
this.setCellType(FillapixCellType.WHITE);
break;
- case "FPIX-UNPL-0001":
+ case "FPIX-ELEM-0002":
int n = this.getNumber();
switch (m.getButton()) {
case MouseEvent.BUTTON1:
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/BlackTile.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/BlackTile.java
index 1d7c038a3..a6993778d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/BlackTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/BlackTile.java
@@ -5,7 +5,7 @@
public class BlackTile extends PlaceableElement {
public BlackTile() {
super(
- "FPIX-PLAC-0001",
+ "FPIX-ELEM-0001",
"Black Tile",
"The black tile",
"edu/rpi/legup/images/fillapix/tiles/BlackTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/NumberTile.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/NumberTile.java
index e869aeaf9..5852c1ad7 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/NumberTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/NumberTile.java
@@ -1,13 +1,13 @@
package edu.rpi.legup.puzzle.fillapix.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class NumberTile extends NonPlaceableElement {
+public class NumberTile extends PlaceableElement {
private int object_num;
public NumberTile() {
super(
- "FPIX-UNPL-0001",
+ "FPIX-ELEM-0002",
"Number Tile",
"A numbered tile",
"edu/rpi/legup/images/fillapix/tiles/NumberTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/UnknownTile.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/UnknownTile.java
index 6778c1758..82d0dffb9 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/UnknownTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/UnknownTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.fillapix.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnknownTile extends NonPlaceableElement {
+public class UnknownTile extends PlaceableElement {
public UnknownTile() {
super(
- "FPIX-UNPL-0002",
+ "FPIX-ELEM-0003",
"Unknown Tile",
"A blank tile",
"edu/rpi/legup/images/fillapix/tiles/UnknownTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/WhiteTile.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/WhiteTile.java
index 67065a7e9..b2eedfc09 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/WhiteTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/WhiteTile.java
@@ -5,7 +5,7 @@
public class WhiteTile extends PlaceableElement {
public WhiteTile() {
super(
- "FPIX-PLAC-0002",
+ "FPIX-ELEM-0004",
"White Tile",
"The white tile",
"edu/rpi/legup/images/fillapix/tiles/WhiteTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/fillapix_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/fillapix_elements_reference_sheet.txt
index 0409fa800..1aece4b97 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/fillapix_elements_reference_sheet.txt
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/elements/fillapix_elements_reference_sheet.txt
@@ -1,5 +1,4 @@
-FPIX-PLAC-0001 : BlackTile
-FPIX-PLAC-0002 : WhiteTile
-
-FPIX-UNPL-0001 : NumberTile
-FPIX-UNPL-0002 : UnknownTile
\ No newline at end of file
+FPIX-ELEM-0001 : BlackTile
+FPIX-ELEM-0002 : NumberTile
+FPIX-ELEM-0003 : UnknownTile
+FPIX-ELEM-0004 : WhiteTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/BlackOrWhiteCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/BlackOrWhiteCaseRule.java
index 860a6c011..f0194bd39 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/BlackOrWhiteCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/BlackOrWhiteCaseRule.java
@@ -37,6 +37,9 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
Board case1 = board.copy();
FillapixCell cell1 = (FillapixCell) case1.getPuzzleElement(puzzleElement);
diff --git a/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/SatisfyClueCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/SatisfyClueCaseRule.java
index 7db833f76..f8bb2d4f5 100644
--- a/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/SatisfyClueCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/fillapix/rules/SatisfyClueCaseRule.java
@@ -45,6 +45,9 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList();
+ if (puzzleElement == null) {
+ return cases;
+ }
// get value of cell
FillapixBoard fillapixBoard = (FillapixBoard) board.copy();
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/LightUp.java b/src/main/java/edu/rpi/legup/puzzle/lightup/LightUp.java
index ab95c4658..a73806cd7 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/LightUp.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/LightUp.java
@@ -47,7 +47,7 @@ public Board generatePuzzle(int difficulty) {
* @return true if the given dimensions are valid for Light Up, false otherwise
*/
public boolean isValidDimensions(int rows, int columns) {
- return rows > 0 && columns > 0;
+ return rows >= 0 && columns >= 0;
}
/**
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpBoard.java b/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpBoard.java
index 217ef79a8..21084b8c7 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpBoard.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpBoard.java
@@ -134,12 +134,12 @@ public int getNumAdjLite(LightUpCell cell) {
}
/**
- * Gets the number of adjacent cells that are placable
+ * Gets the number of adjacent cells that are placeable
*
* @param cell specified cell
- * @return number of adjacent cells that are placable
+ * @return number of adjacent cells that are placeable
*/
- public int getNumPlacble(LightUpCell cell) {
+ public int getNumPlaceable(LightUpCell cell) {
int num = 0;
Set adjCells = getAdj(cell);
for (LightUpCell c : adjCells) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpCell.java b/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpCell.java
index 8adf84cb4..6d890e67b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/LightUpCell.java
@@ -16,16 +16,16 @@ public LightUpCell(int valueInt, Point location) {
@Override
public void setType(Element e, MouseEvent m) {
switch (e.getElementID()) {
- case "LTUP-PLAC-0001":
+ case "LTUP-ELEM-0002":
this.data = -4;
break;
- case "LTUP-UNPL-0002":
+ case "LTUP-ELEM-0001":
this.data = -1;
break;
- case "LTUP-UNPL-0003":
+ case "LTUP-ELEM-0004":
this.data = -2;
break;
- case "LTUP-UNPL-0001":
+ case "LTUP-ELEM-0003":
switch (m.getButton()) {
case MouseEvent.BUTTON1:
if (this.data < 0 || this.data > 3) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BlackTile.java b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BlackTile.java
index 2ddb4f754..eed3795d7 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BlackTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BlackTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.lightup.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class BlackTile extends NonPlaceableElement {
+public class BlackTile extends PlaceableElement {
public BlackTile() {
super(
- "LTUP-UNPL-0002",
+ "LTUP-ELEM-0001",
"Black Tile",
"The black tile",
"edu/rpi/legup/images/lightup/black.gif");
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BulbTile.java b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BulbTile.java
index d238baa56..61ebac3d0 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BulbTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/BulbTile.java
@@ -5,7 +5,7 @@
public class BulbTile extends PlaceableElement {
public BulbTile() {
super(
- "LTUP-PLAC-0001",
+ "LTUP-ELEM-0002",
"Bulb Tile",
"The bulb tile",
"edu/rpi/legup/images/lightup/light.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/NumberTile.java b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/NumberTile.java
index ae314a4cf..26f9be46c 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/NumberTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/NumberTile.java
@@ -1,15 +1,15 @@
package edu.rpi.legup.puzzle.lightup.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class NumberTile extends NonPlaceableElement {
+public class NumberTile extends PlaceableElement {
int object_number;
// Follow the default format and resolves the NoSuchMethod error
public NumberTile() {
super(
- "LTUP-UNPL-0001",
+ "LTUP-ELEM-0003",
"Number Tile",
"The number tile",
"edu/rpi/legup/images/lightup/1.gif");
@@ -17,7 +17,7 @@ public NumberTile() {
public NumberTile(int num) {
super(
- "LTUP-UNPL-0001",
+ "LTUP-ELEM-0003",
"Number Tile",
"The number tile",
"edu/rpi/legup/images/lightup/" + num + ".gif");
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/UnknownTile.java b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/UnknownTile.java
index 24d420fe8..a724be600 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/UnknownTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/UnknownTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.lightup.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnknownTile extends NonPlaceableElement {
+public class UnknownTile extends PlaceableElement {
public UnknownTile() {
super(
- "LTUP-UNPL-0003",
+ "LTUP-ELEM-0004",
"Unknown Tile",
"A blank tile",
"edu/rpi/legup/images/lightup/UnknownTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/elements/lightup_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/lightup_elements_reference_sheet.txt
new file mode 100644
index 000000000..93c97de1c
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/elements/lightup_elements_reference_sheet.txt
@@ -0,0 +1,4 @@
+LTUP-ELEM-0001 : BlackTile
+LTUP-ELEM-0002 : BulbTile
+LTUP-ELEM-0003 : NumberTile
+LTUP-ELEM-0004 : UnknownTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/rules/LightOrEmptyCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/lightup/rules/LightOrEmptyCaseRule.java
index 4ba754731..53efb6587 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/rules/LightOrEmptyCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/rules/LightOrEmptyCaseRule.java
@@ -44,6 +44,10 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
Board case1 = board.copy();
PuzzleElement data1 = case1.getPuzzleElement(puzzleElement);
data1.setData(-4);
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/rules/SatisfyNumberCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/lightup/rules/SatisfyNumberCaseRule.java
index 490122874..f73a34b2d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/rules/SatisfyNumberCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/rules/SatisfyNumberCaseRule.java
@@ -47,6 +47,11 @@ public CaseBoard getCaseBoard(Board board) {
*/
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
+ ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
LightUpBoard lightUpBoard = (LightUpBoard) board;
LightUpCell cell = (LightUpCell) puzzleElement;
Point loc = cell.getLocation();
@@ -96,7 +101,6 @@ public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
}
}
- ArrayList cases = new ArrayList<>();
if (numNeeded == 0) {
return cases;
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/lightup/rules/TooFewBulbsContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/lightup/rules/TooFewBulbsContradictionRule.java
index 8cf68e570..de1f85edc 100644
--- a/src/main/java/edu/rpi/legup/puzzle/lightup/rules/TooFewBulbsContradictionRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/lightup/rules/TooFewBulbsContradictionRule.java
@@ -36,7 +36,7 @@ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
}
int bulbs = lightUpBoard.getNumAdj(cell, LightUpCellType.BULB);
- int placeable = lightUpBoard.getNumPlacble(cell);
+ int placeable = lightUpBoard.getNumPlaceable(cell);
if (bulbs + placeable < cell.getData()) {
return null;
diff --git a/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/BombTile.java b/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/BombTile.java
index 78a5d320c..cd5577eeb 100644
--- a/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/BombTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/BombTile.java
@@ -1,8 +1,8 @@
package edu.rpi.legup.puzzle.minesweeper.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class BombTile extends NonPlaceableElement {
+public class BombTile extends PlaceableElement {
public BombTile() {
super(
"MINE-UNPL-0001",
diff --git a/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/UnsetTile.java b/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/UnsetTile.java
index 447e2840c..6cfc34c8d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/UnsetTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/minesweeper/elements/UnsetTile.java
@@ -1,8 +1,8 @@
package edu.rpi.legup.puzzle.minesweeper.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnsetTile extends NonPlaceableElement {
+public class UnsetTile extends PlaceableElement {
public UnsetTile() {
super(
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeCell.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeCell.java
index c6cd2c64e..1e0e85ed8 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeCell.java
@@ -46,13 +46,13 @@ public NurikabeType getType() {
@Override
public void setType(Element e, MouseEvent m) {
switch (e.getElementID()) {
- case "NURI-PLAC-0001":
+ case "NURI-ELEM-0001":
this.data = -1;
break;
- case "NURI-PLAC-0002":
+ case "NURI-ELEM-0004":
this.data = 0;
break;
- case "NURI-UNPL-0001":
+ case "NURI-ELEM-0002":
switch (m.getButton()) {
case MouseEvent.BUTTON1:
if (this.data <= 0 || this.data > 8) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeExporter.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeExporter.java
index 23efd4724..e01821639 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeExporter.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/NurikabeExporter.java
@@ -10,6 +10,13 @@ public NurikabeExporter(Nurikabe nurikabe) {
super(nurikabe);
}
+ /**
+ * Generates an XML element for the nurikabe puzzle board, including its dimensions and the
+ * state of each cell. Nurikabe cells that are not empty are included in the XML.
+ *
+ * @param newDocument The XML document to which the board element belongs.
+ * @return The XML element representing the board.
+ */
@Override
protected org.w3c.dom.Element createBoardElement(Document newDocument) {
NurikabeBoard board;
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/BlackTile.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/BlackTile.java
index 459a809e0..a7972b9b2 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/BlackTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/BlackTile.java
@@ -5,7 +5,7 @@
public class BlackTile extends PlaceableElement {
public BlackTile() {
super(
- "NURI-PLAC-0001",
+ "NURI-ELEM-0001",
"Black Tile",
"The black tile",
"edu/rpi/legup/images/nurikabe/tiles/BlackTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/NumberTile.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/NumberTile.java
index 475b278da..2015d990b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/NumberTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/NumberTile.java
@@ -1,13 +1,13 @@
package edu.rpi.legup.puzzle.nurikabe.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class NumberTile extends NonPlaceableElement {
+public class NumberTile extends PlaceableElement {
private int object_num;
public NumberTile() {
super(
- "NURI-UNPL-0001",
+ "NURI-ELEM-0002",
"Number Tile",
"A numbered tile",
"edu/rpi/legup/images/nurikabe/tiles/NumberTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/UnknownTile.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/UnknownTile.java
index 85d47e208..8a18c80cc 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/UnknownTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/UnknownTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.nurikabe.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnknownTile extends NonPlaceableElement {
+public class UnknownTile extends PlaceableElement {
public UnknownTile() {
super(
- "NURI-UNPL-0002",
+ "NURI-ELEM-0003",
"Unknown Tile",
"A blank tile",
"edu/rpi/legup/images/nurikabe/tiles/UnknownTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/WhiteTile.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/WhiteTile.java
index 35eb63b81..ae07c6d76 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/WhiteTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/WhiteTile.java
@@ -5,7 +5,7 @@
public class WhiteTile extends PlaceableElement {
public WhiteTile() {
super(
- "NURI-PLAC-0002",
+ "NURI-ELEM-0004",
"White Tile",
"The white tile",
"edu/rpi/legup/images/nurikabe/tiles/WhiteTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/nurikabe_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/nurikabe_elements_reference_sheet.txt
new file mode 100644
index 000000000..667972fd6
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/elements/nurikabe_elements_reference_sheet.txt
@@ -0,0 +1,4 @@
+NURI-ELEM-0001 : BlackTile
+NURI-ELEM-0002 : NumberTile
+NURI-ELEM-0003 : UnknownTile
+NURI-ELEM-0004 : WhiteTile
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/BlackOrWhiteCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/BlackOrWhiteCaseRule.java
index ac0ab6df6..1c87d5cfa 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/BlackOrWhiteCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/BlackOrWhiteCaseRule.java
@@ -24,7 +24,7 @@ public BlackOrWhiteCaseRule() {
/**
* Checks whether the {@link TreeTransition} logically follows from the parent node using this
- * rule. This method is the one that should overridden in child classes.
+ * rule. This method is the one that should be overridden in child classes.
*
* @param transition transition to check
* @return null if the child node logically follow from the parent node, otherwise error message
@@ -84,6 +84,10 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
Board case1 = board.copy();
PuzzleElement data1 = case1.getPuzzleElement(puzzleElement);
data1.setData(NurikabeType.WHITE.toValue());
diff --git a/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/FinishRoomCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/FinishRoomCaseRule.java
index 24a261ec9..025212d96 100644
--- a/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/FinishRoomCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/nurikabe/rules/FinishRoomCaseRule.java
@@ -27,15 +27,16 @@ public FinishRoomCaseRule() {
super(
"NURI-CASE-0002",
"Finish Room",
- "Room can be finished in up to five ways",
+ "Room can be finished in up to nine ways",
"edu/rpi/legup/images/nurikabe/cases/FinishRoom.png");
- this.MAX_CASES = 5;
- this.MIN_CASES = 2;
+ this.MAX_CASES = 9;
+ this.MIN_CASES = 1;
+ this.uniqueCases = new HashSet<>();
}
/**
* Checks whether the {@link TreeTransition} logically follows from the parent node using this
- * rule. This method is the one that should overridden in child classes.
+ * rule. This method is the one that should have overridden in child classes.
*
* @param transition transition to check
* @return null if the child node logically follow from the parent node, otherwise error message
@@ -44,18 +45,18 @@ public FinishRoomCaseRule() {
public String checkRuleRaw(TreeTransition transition) {
NurikabeBoard destBoardState = (NurikabeBoard) transition.getBoard();
List childTransitions = transition.getParents().get(0).getChildren();
- if (childTransitions.size() > 5) {
+ if (childTransitions.size() > MAX_CASES) {
return super.getInvalidUseOfRuleMessage()
- + ": This case rule must have 5 or less children.";
+ + ": This case rule must have 9 or less children.";
}
- if (childTransitions.size() < 2) {
+ if (childTransitions.size() < MIN_CASES) {
return super.getInvalidUseOfRuleMessage()
- + ": This case rule must have 2 or more children.";
+ + ": This case rule must have 1 or more children.";
}
if (childTransitions.size() != legitCases) {
return super.getInvalidUseOfRuleMessage()
+ ": Cases can not be removed from the branch.";
- } // stops user from deleting 1 or mose generated cases and still having path show as green
+ } // stops user from deleting 1 or more generated cases and still having path show as green
Set locations = new HashSet<>();
for (TreeTransition t1 : childTransitions) {
locations.add(
@@ -100,14 +101,8 @@ public CaseBoard getCaseBoard(Board board) {
// if found another number tile and it's data is different from the element
// we're working with
if ((d.getType() == NurikabeType.NUMBER)
- && !(d.getData()
- .equals(
- ((NurikabeCell) element)
- .getData()))) { // if found another number tile
- // and it's data is different
- // than the element we're
- // working with
- only = false; // set only to false
+ && !(d.getData().equals(((NurikabeCell) element).getData()))) {
+ only = false;
}
}
// if size of region is 1 less than the number block and the number block is only
@@ -130,6 +125,10 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>(); // makes array list of cases
+ if (puzzleElement == null) {
+ return cases;
+ }
+
NurikabeBoard nuriBoard = (NurikabeBoard) board.copy(); // nurikabe board to edit
NurikabeCell numberCell =
nuriBoard.getCell(
@@ -231,23 +230,9 @@ private void generateCases(
break;
}
}
- if (!alreadyIn) { // if point wasn't already in
- Board casey =
- nuriBoard.copy(); // copy the current board with white tile
- // changed
- PuzzleElement datacasey =
- curr; // gets changed white tile as a puzzle element
- datacasey.setData(
- NurikabeType.WHITE
- .toValue()); // ensure set to white, probably redundant
- casey.addModifiedData(datacasey); // ensure confirmed white change
- regions =
- NurikabeUtilities.getNurikabeRegions(
- nuriBoard); // update regions
- cases.add(casey); // add this case to list of cases
- locations.add(
- here); // add location of new white tile to list of locations so
- // that we don't accidentally add it again later
+ if (unique) {
+ caseBoard.addModifiedData(newCell);
+ cases.add(caseBoard);
}
} else if (newRoomSet.size() < filledRoomSize) {
generateCases(
@@ -301,14 +286,10 @@ private boolean touchesDifferentRoom(
|| (adjacentPoint.x != origPoint.x || adjacentPoint.y != origPoint.y)) {
return true;
}
- curr.setData(NurikabeType.UNKNOWN.toValue()); // set cell type back to unknown
- nuriBoard.addModifiedData(curr); // confirms change back to unknown
- regions = NurikabeUtilities.getNurikabeRegions(nuriBoard); // updates regions
}
}
- legitCases = cases.size();
}
- return cases;
+ return false;
}
/**
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/ShortTruthTableCell.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/ShortTruthTableCell.java
index 75bba369f..ffd7c491d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/ShortTruthTableCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/ShortTruthTableCell.java
@@ -135,22 +135,22 @@ public void setType(Element e, MouseEvent m) {
}
// Red Element
- if (e.getElementID().equals("STTT-PLAC-0002")) {
+ if (e.getElementID().equals("STTT-ELEM-0004")) {
this.data = ShortTruthTableCellType.FALSE;
}
// Green Element
else {
- if (e.getElementID().equals("STTT-PLAC-0001")) {
+ if (e.getElementID().equals("STTT-ELEM-0002")) {
this.data = ShortTruthTableCellType.TRUE;
}
// Unknown Element
else {
- if (e.getElementID().equals("STTT-PLAC-0003")) {
+ if (e.getElementID().equals("STTT-ELEM-0005")) {
this.data = ShortTruthTableCellType.UNKNOWN;
}
// Argument Element
else {
- if (e.getElementID().equals("STTT-UNPL-0001")) {
+ if (e.getElementID().equals("STTT-ELEM-0001")) {
// Prevents non-argument symbols from being changed
if (!(this.symbol >= 'A' && this.symbol <= 'Z')) {
return;
@@ -172,7 +172,7 @@ public void setType(Element e, MouseEvent m) {
}
// And/Or Element
else {
- if (e.getElementID().equals("STTT-UNPL-0002")) {
+ if (e.getElementID().equals("STTT-ELEM-0003")) {
if (m.getButton() == MouseEvent.BUTTON1) {
if (this.symbol == '^') {
this.symbol = '|';
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/ArgumentElement.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/ArgumentElement.java
index 9294fba4e..912fd2672 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/ArgumentElement.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/ArgumentElement.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.shorttruthtable.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class ArgumentElement extends NonPlaceableElement {
+public class ArgumentElement extends PlaceableElement {
public ArgumentElement() {
super(
- "STTT-UNPL-0001",
+ "STTT-ELEM-0001",
"Argument Element",
"Argument of logic statement element",
"edu/rpi/legup/images/shorttruthtable/tiles/LetterTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/GreenElement.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/GreenElement.java
index 783186baa..56221fef3 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/GreenElement.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/GreenElement.java
@@ -5,7 +5,7 @@
public class GreenElement extends PlaceableElement {
public GreenElement() {
super(
- "STTT-PLAC-0001",
+ "STTT-ELEM-0002",
"Green Element",
"A green tile to set certain tiles to true",
"edu/rpi/legup/images/shorttruthtable/tiles/GreenTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/LogicSymbolElement.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/LogicSymbolElement.java
index 5fed4b1df..b82ebc2cb 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/LogicSymbolElement.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/LogicSymbolElement.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.shorttruthtable.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class LogicSymbolElement extends NonPlaceableElement {
+public class LogicSymbolElement extends PlaceableElement {
public LogicSymbolElement() {
super(
- "STTT-UNPL-0002",
+ "STTT-ELEM-0003",
"Logic Symbol Element",
"Logic symbol element",
"edu/rpi/legup/images/shorttruthtable/tiles/ConditionalBiconditionalTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/RedElement.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/RedElement.java
index e2a589b65..2114e62ec 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/RedElement.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/RedElement.java
@@ -5,7 +5,7 @@
public class RedElement extends PlaceableElement {
public RedElement() {
super(
- "STTT-PLAC-0002",
+ "STTT-ELEM-0004",
"Red Element",
"A red tile to set certain tiles to false",
"edu/rpi/legup/images/shorttruthtable/tiles/RedTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/UnknownElement.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/UnknownElement.java
index d475bc05d..52b54f202 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/UnknownElement.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/UnknownElement.java
@@ -5,7 +5,7 @@
public class UnknownElement extends PlaceableElement {
public UnknownElement() {
super(
- "STTT-PLAC-0003",
+ "STTT-ELEM-0005",
"Unknown Element",
"A blank tile",
"edu/rpi/legup/images/shorttruthtable/tiles/UnknownTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/shorttruthtable_elements_reference_sheet b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/shorttruthtable_elements_reference_sheet
index 471631553..c5421169f 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/shorttruthtable_elements_reference_sheet
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/elements/shorttruthtable_elements_reference_sheet
@@ -1,6 +1,5 @@
-STTT-UNPL-0001 : ArgumentElement
-STTT-UNPL-0002 : ConditionalBiconditionalElement
-
-STTT-PLAC-0001 : GreenElement
-STTT-PLAC-0002 : RedElement
-STTT-PLAC-0003 : UnknownElement
\ No newline at end of file
+STTT-ELEM-0001 : ArgumentElement
+STTT-ELEM-0002 : GreenElement
+STTT-ELEM-0003 : LogicSymbolElement
+STTT-ELEM-0004 : RedElement
+STTT-ELEM-0005 : UnknownElement
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRuleAtomic.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRuleAtomic.java
index 58d2068b2..22b49fd77 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRuleAtomic.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRuleAtomic.java
@@ -44,6 +44,9 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
Board case1 = board.copy();
PuzzleElement data1 = case1.getPuzzleElement(puzzleElement);
diff --git a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRule_GenericStatement.java b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRule_GenericStatement.java
index 99f771246..8aeb51a46 100644
--- a/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRule_GenericStatement.java
+++ b/src/main/java/edu/rpi/legup/puzzle/shorttruthtable/rules/caserule/CaseRule_GenericStatement.java
@@ -85,6 +85,11 @@ public CaseBoard getCaseBoard(Board board) {
@SuppressWarnings("unchecked")
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
+
+ if (puzzleElement == null) {
+ return new ArrayList();
+ }
+
ShortTruthTableBoard sttBoard = ((ShortTruthTableBoard) board);
ShortTruthTableCell cell = sttBoard.getCellFromElement(puzzleElement);
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/Skyscrapers.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/Skyscrapers.java
index df5ba78a3..44f416cef 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/Skyscrapers.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/Skyscrapers.java
@@ -46,7 +46,7 @@ public Board generatePuzzle(int difficulty) {
* @return true if the given dimensions are valid for Skyscrapers, false otherwise
*/
public boolean isValidDimensions(int rows, int columns) {
- return rows >= 4 && rows == columns;
+ return rows >= 3 && rows == columns;
}
/**
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersBoard.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersBoard.java
index 4cd09b254..0fc133786 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersBoard.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersBoard.java
@@ -209,7 +209,7 @@ public void setCell(int x, int y, Element e, MouseEvent m) {
SkyscrapersClue clue = this.getClue(x, y);
if (e == null) return;
if (clue != null) {
- if (!e.getElementID().equals("SKYS-UNPL-0003")) {
+ if (!e.getElementID().equals("SKYS-ELEM-0001")) {
return;
}
@@ -217,10 +217,10 @@ public void setCell(int x, int y, Element e, MouseEvent m) {
if (clue.getData() < dimension.height) {
clue.setData(clue.getData() + 1);
} else {
- clue.setData(0);
+ clue.setData(1);
}
} else {
- if (clue.getData() > 0) {
+ if (clue.getData() > 1) {
clue.setData(clue.getData() - 1);
} else {
clue.setData(dimension.height);
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersCell.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersCell.java
index 1cf9a357b..9e7283b20 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersCell.java
@@ -27,10 +27,10 @@ public SkyscrapersType getType() {
@Override
public void setType(Element e, MouseEvent m) {
switch (e.getElementID()) {
- case "SKYS-UNPL-0001":
+ case "SKYS-ELEM-0002":
this.data = 0;
break;
- case "SKYS-UNPL-0002":
+ case "SKYS-ELEM-0001":
switch (m.getButton()) {
case MouseEvent.BUTTON1:
if (this.data <= 0 || this.data >= this.max) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersClueView.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersClueView.java
index 4c75f0695..cfb3388ff 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersClueView.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/SkyscrapersClueView.java
@@ -25,11 +25,11 @@ public SkyscrapersClue getPuzzleElement() {
@Override
public void draw(Graphics2D graphics2D) {
drawElement(graphics2D);
+ if (this.isHover()) {
+ drawHover(graphics2D);
+ }
if (this.isShowCasePicker() && this.isCaseRulePickable()) {
drawCase(graphics2D);
- if (this.isHover()) {
- drawHover(graphics2D);
- }
}
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/ClueTile.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/ClueTile.java
deleted file mode 100644
index 64c9033e6..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/ClueTile.java
+++ /dev/null
@@ -1,14 +0,0 @@
-package edu.rpi.legup.puzzle.skyscrapers.elements;
-
-import edu.rpi.legup.model.elements.NonPlaceableElement;
-
-public class ClueTile extends NonPlaceableElement {
-
- public ClueTile() {
- super(
- "SKYS-UNPL-0003",
- "Clue Tile",
- "Clue Updater",
- "edu/rpi/legup/images/skyscrapers/tiles/ClueTile.png");
- }
-}
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/NumberTile.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/NumberTile.java
index 4d6b37c9a..f60e5fe8b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/NumberTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/NumberTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.skyscrapers.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class NumberTile extends NonPlaceableElement {
+public class NumberTile extends PlaceableElement {
public NumberTile() {
super(
- "SKYS-UNPL-0002",
+ "SKYS-ELEM-0001",
"Number Tile",
"A numbered tile",
"edu/rpi/legup/images/skyscrapers/tiles/ClueTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/UnknownTile.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/UnknownTile.java
index 2fb21193a..07f6a1238 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/UnknownTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/UnknownTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.skyscrapers.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnknownTile extends NonPlaceableElement {
+public class UnknownTile extends PlaceableElement {
public UnknownTile() {
super(
- "SKYS-UNPL-0001",
+ "SKYS-ELEM-0002",
"Unknown",
"A blank tile",
"edu/rpi/legup/images/skyscrapers/tiles/UnknownTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/skyscrapers_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/skyscrapers_elements_reference_sheet.txt
index 604e1824e..14e76a29d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/skyscrapers_elements_reference_sheet.txt
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/elements/skyscrapers_elements_reference_sheet.txt
@@ -1,3 +1,2 @@
-SKYS-UNPL-0001: Unknown Tile
-SKYS-UNPL-0002: Number Tile
-SKYS-UNPL-0003: Clue "Tile"
\ No newline at end of file
+SKYS-ELEM-0001: NumberTile
+SKYS-ELEM-0002: UnknownTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/CellForNumberCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/CellForNumberCaseRule.java
index 45bdadea3..b48962c41 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/CellForNumberCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/CellForNumberCaseRule.java
@@ -61,6 +61,9 @@ public CaseBoard getCaseBoard(Board board) {
public ArrayList getCasesFor(Board board, PuzzleElement puzzleElement, Integer number) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
SkyscrapersClue clue = (SkyscrapersClue) puzzleElement;
SkyscrapersBoard skyscrapersboard = (SkyscrapersBoard) board;
diff --git a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/NumberForCellCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/NumberForCellCaseRule.java
index 145dd6ee2..4f8e1df6b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/NumberForCellCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/skyscrapers/rules/NumberForCellCaseRule.java
@@ -47,6 +47,9 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
SkyscrapersCell cell = (SkyscrapersCell) puzzleElement;
SkyscrapersBoard skyscrapersboard = (SkyscrapersBoard) board;
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/allfiles.txt b/src/main/java/edu/rpi/legup/puzzle/starbattle/allfiles.txt
deleted file mode 100644
index 5a9ec0f0a..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/starbattle/allfiles.txt
+++ /dev/null
@@ -1,235 +0,0 @@
-//StarBattle.java
-
-package edu.rpi.legup.puzzle.starbattle;
-
-import edu.rpi.legup.model.Puzzle;
-import edu.rpi.legup.model.gameboard.Board;
-
-public class StarBattle extends Puzzle {
- public StarBattle() {
- super();
- this.name = "StarBattle";
-
- this.importer = new StarBattleImporter(this);
- this.exporter = new StarBattleExporter(this);
-
- this.factory = new StarBattleCellFactory();
- }
-
- @Override
- public void initializeView() {
- }
-
- @Override
- public Board generatePuzzle(int difficulty) {
- return null;
- }
-
- @Override
- public boolean isBoardComplete(Board board) {
- return true;
- }
-
- @Override
- public void onBoardChange(Board board) {
- }
-}
-
-//StarBattleBoard.java
-
-package edu.rpi.legup.puzzle.lightup;
-
-import edu.rpi.legup.model.gameboard.GridBoard;
-import edu.rpi.legup.model.gameboard.PuzzleElement;
-
-import java.awt.*;
-import java.util.HashSet;
-import java.util.Set;
-
-public class StarBattleBoard extends GridBoard {
-
- private int size;
- private vector group_sizes;
-
- /**
- * StarBattleBoard Constructor - create a new Star Battle board
- *
- * @param size size of one side of the star battle board
- */
-
- public StarBattleBoard(int size) {
- super(size, size);
- group_sizes = vector(size);
- }
-
- @Override
- public StarBattleCell getCell(int x, int y) {
- return (StarBattleCell) super.getCell(x, y);
- }
-
-
-}
-
-//StarBattleCell.java
-
-package edu.rpi.legup.puzzle.starbattle;
-
-import edu.rpi.legup.model.gameboard.GridCell;
-
-import java.awt.*;
-import java.util.HashSet;
-import java.util.Set;
-
-public class StarBattleCell extends GridCell {
- private int groupIndex;
- private int max;
-
- /**
- * StarBattleCell Constructor - creates a new StarBattle cell to hold the puzzleElement
- *
- * @param valueInt value of the star battle cell denoting its state
- * @param location location of the cell on the board
- * @param size size of the star battle cell
- */
- public StarBattleCell(int value, Point location, int groupIndex, int size) {
- super(value, location);
- this.groupIndex = groupIndex;
- this.max = size;
- }
-
- @Override
- public void setType(Element e, MouseEvent m) {
- switch (e.getElementID()) {
- case "SBUP-PLAC-0001":
- this.data = -3;
- break;
- case "SBUP-PLAC-0002":
- this.data = -2;
- break;
- case "SBUP-PLAC-0003":
- this.data = -1;
- break;
- case "SBUP-UNPL-0001"://Not sure how button events work
- switch (m.getButton()){
- case MouseEvent.BUTTON1:
- if (this.data < 0 || this.data > 3) {
- this.data = 0;
- }
- else {
- this.data = this.data + 1;
- }
- break;
- case MouseEvent.BUTTON3:
- if (this.data > 0) {
- this.data = this.data - 1;
- }
- else {
- this.data = 3;//Unsure
- }
- break;
- }
- break;
- }
- }
-
- public LightUpCellType getType() {
- switch (data) {
- case -3:
- return LightUpCellType.UNKNOWN;
- case -2:
- return LightUpCellType.STAR;
- case -1:
- return LightUpCellType.BLACK;
- default:
- if (data >= 0) {
- return StarBattleCellType.WHITE;
- }
- }
- return null;
- }
-
- /**
- * Gets the region index of the cell
- *
- * @return group index of the cell
- */
- public int getGroupIndex() {
- return groupIndex;
- }
-
- /**
- * Gets the size of the cell
- *
- * @return size of the cell
- */
-
- public int getMax() {
- return max;
- }
-
-}
-
-//StarBattleCellController.java
-
-package edu.rpi.legup.puzzle.starbattle;
-
-import edu.rpi.legup.controller.ElementController;
-import edu.rpi.legup.model.gameboard.PuzzleElement;
-
-import java.awt.event.MouseEvent;
-
-public class StarBattleCellController extends ElementController {
- @Override
- public void changeCell(MouseEvent e, PuzzleElement data) {
- StarBattleCell cell = (StarBattleCell) data;
- if (e.getButton() == MouseEvent.BUTTON1) {
- if (e.isControlDown()) {
- this.boardView.getSelectionPopupMenu().show(boardView, this.boardView.getCanvas().getX() + e.getX(), this.boardView.getCanvas().getY() + e.getY());
- }
- else {
- if (cell.getData() == 0) {
- data.setData(-3);
- }
- else {
- data.setData(cell.getData() + 1);
- }
- }
- }
- else {
- if (e.getButton() == MouseEvent.BUTTON3) {
- if (cell.getData() == -3) {
- data.setData(0);
- }
- else {
- data.setData(cell.getData() - 1);
- }
- }
- }
- }
-}
-
-//StarBattleCellFactory.java
-
-
-
-//StarBattleCellType.java
-package edu.rpi.legup.puzzle.starbattle;
-
-public enum StarBattleType {
- UNKNOWN(-3), STAR(-2), BLACK(-1), WHITE(0);
-
- public int value;
-
- StarBattleCell(int value) {
- this.value = value;
- }
-}
-
-//StarBattleExporter.java
-//StarBattleImporter.java
-//StarBattleView.java
-
-How to run Legup:
-
-./gradlew build
-Java -jar build/libs/Legup.jar
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/BlackTile.java b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/BlackTile.java
index 99f42886e..c4bbf7297 100644
--- a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/BlackTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/BlackTile.java
@@ -1,8 +1,8 @@
package edu.rpi.legup.puzzle.starbattle.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class BlackTile extends NonPlaceableElement {
+public class BlackTile extends PlaceableElement {
public BlackTile() {
super(
"STBL-PLAC-0002",
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/StarTile.java b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/StarTile.java
index 13ada3f4d..793d4dbeb 100644
--- a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/StarTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/StarTile.java
@@ -1,8 +1,8 @@
package edu.rpi.legup.puzzle.starbattle.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class StarTile extends NonPlaceableElement {
+public class StarTile extends PlaceableElement {
public StarTile() {
super(
"STBL-PLAC-0001",
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/UnknownTile.java b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/UnknownTile.java
index 425fb5d5e..30921de8d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/UnknownTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/UnknownTile.java
@@ -1,8 +1,8 @@
package edu.rpi.legup.puzzle.starbattle.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnknownTile extends NonPlaceableElement {
+public class UnknownTile extends PlaceableElement {
public UnknownTile() {
super(
"STBL-UNPL-0001",
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/starbattle_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/starbattle_elements_reference_sheet.txt
new file mode 100644
index 000000000..82352bd04
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/starbattle/elements/starbattle_elements_reference_sheet.txt
@@ -0,0 +1,4 @@
+STBL-ELEM-0001 : BlackTile
+STBL-ELEM-0002 : StarTile
+STBL-ELEM-0003 : UnknownTile
+STBL-ELEM-0004 : WhiteTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/StarOrEmptyCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/StarOrEmptyCaseRule.java
index df900dcd5..efd86bd7b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/StarOrEmptyCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/StarOrEmptyCaseRule.java
@@ -84,6 +84,10 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
Board case1 = board.copy();
PuzzleElement data1 = case1.getPuzzleElement(puzzleElement);
data1.setData(StarBattleCellType.STAR.value);
diff --git a/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/starbattle_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/starbattle_reference_sheet.txt
deleted file mode 100644
index f18965fd6..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/starbattle/rules/starbattle_reference_sheet.txt
+++ /dev/null
@@ -1,19 +0,0 @@
-Case Rules:
-Add Star: STBL-CASE-0001
-Star or Empty: STBL-CASE-0002
-
-Basic Rules:
-Blackout: STBL-BASC-0001
-Columns Within Regions: STBL-BASC-0002
-Columns Within Rows: STBL-BASC-0003
-Finish With Stars: STBL-BASC-0004
-Regions Within Columns: STBL-BASC-0005
-Regions Within Rows: STBL-BASC-0006
-Rows Within Columns: STBL-BASC-0007
-Rows Within Regions: STBL-BASC-0008
-Surround Star: STBL-BASC-0009
-
-Contradiction Rules:
-Too Many Stars: STBL-CONT-0001
-Too Few Stars: STBL-CONT-0002
-Clashing Orbit: STBL-CONT-0003
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/ModelSudokuBoard.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/ModelSudokuBoard.java
new file mode 100644
index 000000000..f7893ca32
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/ModelSudokuBoard.java
@@ -0,0 +1,17 @@
+package edu.rpi.legup.puzzle.sudoku;
+
+public class ModelSudokuBoard {
+ public int getModelRegionNumbers(int index) {
+ int columnMod = index % 3 + 1;
+ int rowMod = ((index / 9) % 3) * 3;
+ return columnMod + rowMod;
+ }
+
+ public int getModelRowNumbers(int index) {
+ return index % 9 + 1;
+ }
+
+ public int getModelColumnNumbers(int index) {
+ return index / 9 + 1;
+ }
+}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/PossibleNumberCaseBoard.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/PossibleNumberCaseBoard.java
index 0b6971235..c5f9eec2a 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/PossibleNumberCaseBoard.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/PossibleNumberCaseBoard.java
@@ -2,7 +2,7 @@
import edu.rpi.legup.model.gameboard.CaseBoard;
import edu.rpi.legup.model.gameboard.PuzzleElement;
-import edu.rpi.legup.puzzle.sudoku.rules.PossibleNumberCaseRule;
+import edu.rpi.legup.puzzle.sudoku.rules.PossibleCellsForNumberRegionCaseRule;
import java.awt.event.MouseEvent;
import java.util.HashSet;
import java.util.Set;
@@ -15,7 +15,7 @@ public class PossibleNumberCaseBoard extends CaseBoard {
private Set pickableCols;
public PossibleNumberCaseBoard(
- SudokuBoard baseBoard, PossibleNumberCaseRule caseRule, SudokuCell cell) {
+ SudokuBoard baseBoard, PossibleCellsForNumberRegionCaseRule caseRule, SudokuCell cell) {
super(baseBoard, caseRule);
this.cell = cell;
this.pickableRegions = new HashSet<>();
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/Sudoku.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/Sudoku.java
index 877c92665..c27269536 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/Sudoku.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/Sudoku.java
@@ -29,6 +29,8 @@ public BoardView getBoardView() {
@Override
public void initializeView() {
boardView = new SudokuView((SudokuBoard) currentBoard);
+ boardView.setBoard(currentBoard);
+ addBoardListener(boardView);
}
/**
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCell.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCell.java
index da009a255..75e5820a0 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCell.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCell.java
@@ -1,7 +1,9 @@
package edu.rpi.legup.puzzle.sudoku;
+import edu.rpi.legup.model.elements.Element;
import edu.rpi.legup.model.gameboard.GridCell;
import java.awt.*;
+import java.awt.event.MouseEvent;
import java.util.HashSet;
import java.util.Set;
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCellController.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCellController.java
index 9b24f13da..bcad1a0ce 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCellController.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuCellController.java
@@ -8,7 +8,7 @@ public class SudokuCellController extends ElementController {
@Override
public void changeCell(MouseEvent e, PuzzleElement data) {
SudokuCell cell = (SudokuCell) data;
- System.out.print(111);
+
if (e.getButton() == MouseEvent.BUTTON1) {
if (e.isControlDown()) {
this.boardView
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuImporter.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuImporter.java
index 68bf1e795..5084279c3 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuImporter.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuImporter.java
@@ -110,16 +110,6 @@ public void initializeBoard(Node node) throws InvalidFileFormatException {
}
}
}
- //
- // for(int y = 0; y < size; y++)
- // {
- // for(int x = 0; x < size; x++)
- // {
- // SudokuCell cell = sudokuBoard.getCell(x, y);
- // System.err.println("(" + x + ", " + y + ") - " +
- // cell.getGroupIndex());
- // }
- // }
puzzle.setCurrentBoard(sudokuBoard);
} catch (NumberFormatException e) {
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuView.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuView.java
index 9e8dc82ff..474feb342 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuView.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/SudokuView.java
@@ -30,6 +30,7 @@ public SudokuView(SudokuBoard board) {
// i * elementSize.height);
SudokuElementView element = new SudokuElementView(board.getCell(k, i));
element.setIndex(i * gridSize.width + k);
+ element.setIndex(i * gridSize.width);
element.setSize(elementSize);
element.setLocation(location);
elementViews.add(element);
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/NumberTile.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/NumberTile.java
index a75f32c5a..a94d10e64 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/NumberTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/NumberTile.java
@@ -3,10 +3,11 @@
import edu.rpi.legup.model.elements.PlaceableElement;
public class NumberTile extends PlaceableElement {
- private int object_num;
-
public NumberTile() {
- super("SUDO-PLAC-0001", "Number Tile", "A numbered tile", null);
- object_num = 0;
+ super(
+ "SUDO-ELEM-0001",
+ "Number Tile",
+ "A number tile",
+ "edu/rpi/legup/images/sudoku/tiles/NumberTile.png");
}
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/sudoku_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/sudoku_elements_reference_sheet.txt
new file mode 100644
index 000000000..b8df27eb6
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/elements/sudoku_elements_reference_sheet.txt
@@ -0,0 +1,2 @@
+SUDO-ELEM-0001 : NumberTile
+SUDO-ELEM-0002 : UnknownTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/AdvancedDeductionDirectRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/AdvancedDeductionDirectRule.java
deleted file mode 100644
index 190679b41..000000000
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/AdvancedDeductionDirectRule.java
+++ /dev/null
@@ -1,99 +0,0 @@
-package edu.rpi.legup.puzzle.sudoku.rules;
-
-import edu.rpi.legup.model.gameboard.Board;
-import edu.rpi.legup.model.gameboard.PuzzleElement;
-import edu.rpi.legup.model.rules.DirectRule;
-import edu.rpi.legup.model.tree.TreeNode;
-import edu.rpi.legup.model.tree.TreeTransition;
-import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
-import edu.rpi.legup.puzzle.sudoku.SudokuCell;
-
-public class AdvancedDeductionDirectRule extends DirectRule {
-
- public AdvancedDeductionDirectRule() {
- super(
- "SUDO-BASC-0001",
- "Advanced Deduction",
- "Use of group logic deduces more answers by means of forced by Location and forced"
- + " by Deduction",
- "edu/rpi/legup/images/sudoku/AdvancedDeduction.png");
- }
-
- /**
- * Checks whether the child node logically follows from the parent node at the specific
- * puzzleElement index using this rule
- *
- * @param transition transition to check
- * @param puzzleElement equivalent puzzleElement
- * @return null if the child node logically follow from the parent node at the specified
- * puzzleElement, otherwise error message
- */
- public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
- SudokuBoard initialBoard = (SudokuBoard) transition.getParents().get(0).getBoard();
- SudokuBoard finalBoard = (SudokuBoard) transition.getBoard();
-
- SudokuCell cell = (SudokuCell) finalBoard.getPuzzleElement(puzzleElement);
- int index = cell.getIndex();
- int groupSize = initialBoard.getWidth();
- int groupDim = (int) Math.sqrt(groupSize);
- int rowIndex = index / groupSize;
- int colIndex = index % groupSize;
- int relX = rowIndex / groupDim;
- int relY = colIndex % groupDim;
- int groupNum = rowIndex / groupDim * groupDim + colIndex / groupDim;
- boolean[][] possible = new boolean[groupDim][groupDim];
- for (int y = 0; y < groupDim; y++) {
- for (int x = 0; x < groupDim; x++) {
- SudokuCell c = initialBoard.getCell(groupNum, x, y);
- if (c.getData() == cell.getData() && x != relX && y != relY) {
- return super.getRuleName() + ": Duplicate value in sub-region";
- }
- possible[y][x] = c.getData() == 0;
- }
- }
- for (int y = 0; y < groupDim; y++) {
- for (int x = 0; x < groupSize; x++) {
- SudokuCell r = initialBoard.getCell(x, (groupNum / groupDim) * groupDim + y);
- SudokuCell c = initialBoard.getCell((groupNum % groupDim) * groupDim + y, x);
- if (r.getData() == cell.getData()) {
- for (int i = 0; i < groupDim; i++) {
- possible[y][i] = false;
- }
- }
- if (c.getData() == cell.getData()) {
- for (int i = 0; i < groupDim; i++) {
- possible[i][y] = false;
- }
- }
- }
- }
- boolean isForced = false;
- for (int y = 0; y < groupDim; y++) {
- for (int x = 0; x < groupDim; x++) {
- if (possible[y][x] && !isForced) {
- isForced = true;
- } else {
- if (possible[y][x]) {
- return super.getInvalidUseOfRuleMessage() + ": Not forced";
- }
- }
- }
- }
- if (!isForced) {
- return super.getInvalidUseOfRuleMessage() + ": Not forced";
- }
- return null;
- }
-
- /**
- * Creates a transition {@link Board} that has this rule applied to it using the {@link
- * TreeNode}.
- *
- * @param node tree node used to create default transition board
- * @return default board or null if this rule cannot be applied to this tree node
- */
- @Override
- public Board getDefaultBoard(TreeNode node) {
- return null;
- }
-}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastCellForNumberDirectRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastCellForNumberDirectRule.java
index fd03ef36c..6544bf7c3 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastCellForNumberDirectRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastCellForNumberDirectRule.java
@@ -15,7 +15,7 @@ public LastCellForNumberDirectRule() {
"SUDO-BASC-0002",
"Last Cell for Number",
"This is the only cell open in its group for some number.",
- "edu/rpi/legup/images/sudoku/forcedByElimination.png");
+ "edu/rpi/legup/images/sudoku/rules/forcedByElimination.png");
}
/**
@@ -32,52 +32,146 @@ public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElem
SudokuBoard finalBoard = (SudokuBoard) transition.getBoard();
SudokuCell cell = (SudokuCell) finalBoard.getPuzzleElement(puzzleElement);
+
+ // Check if empty cell placed
if (cell.getData() == 0) {
return super.getInvalidUseOfRuleMessage() + ": Cell is not forced at this index";
}
- int size = initialBoard.getSize();
-
+ // Get defaults
Set region = initialBoard.getRegion(cell.getGroupIndex());
Set row = initialBoard.getRow(cell.getLocation().y);
Set col = initialBoard.getCol(cell.getLocation().x);
- boolean contains = false;
- if (region.size() == size - 1) {
- for (SudokuCell c : region) {
- if (cell.getData() == c.getData()) {
+ // Check if new cell conflicts group
+ for (SudokuCell c : region) {
+ if (c.getData() == cell.getData()) {
+ return super.getInvalidUseOfRuleMessage() + ": Cell is not forced at this index";
+ }
+ }
+ for (SudokuCell c : row) {
+ if (c.getData() == cell.getData()) {
+ return super.getInvalidUseOfRuleMessage() + ": Cell is not forced at this index";
+ }
+ }
+ for (SudokuCell c : col) {
+ if (c.getData() == cell.getData()) {
+ return super.getInvalidUseOfRuleMessage() + ": Cell is not forced at this index";
+ }
+ }
+
+ // //
+ // Loop to see if the number is constrained to the cell
+ boolean restrained = true;
+ for (SudokuCell c : region) {
+ // Test if its not a valid testing cell
+ if (c.getData() != 0) {
+ continue;
+ }
+ if (c.getLocation().y == cell.getLocation().y
+ && c.getLocation().x == cell.getLocation().x) {
+ continue;
+ }
+ // Check if cell is eligible to hold number
+ Set crow = initialBoard.getRow(c.getLocation().y);
+ Set ccol = initialBoard.getCol(c.getLocation().x);
+ boolean contains = false;
+ for (SudokuCell rc : crow) {
+ if (rc.getData() == cell.getData()) {
contains = true;
- break;
}
}
+ for (SudokuCell cc : ccol) {
+ if (cc.getData() == cell.getData()) {
+ contains = true;
+ }
+ }
+ // Stop if another cell can hold number
if (!contains) {
- return null;
+ restrained = false;
+ break;
}
}
- if (row.size() == size - 1) {
- contains = false;
- for (SudokuCell c : row) {
- if (cell.getData() == c.getData()) {
+ // Output if success
+ if (restrained) {
+ return null;
+ }
+
+ // //
+ // Loop to see if the number is constrained to the cell
+ restrained = true;
+ for (SudokuCell c : row) {
+ // Test if its not a valid testing cell
+ if (c.getData() != 0) {
+ continue;
+ }
+ if (c.getLocation().y == cell.getLocation().y
+ && c.getLocation().x == cell.getLocation().x) {
+ continue;
+ }
+ // Check if cell is eligible to hold number
+ Set cregion = initialBoard.getRegion(c.getGroupIndex());
+ Set ccol = initialBoard.getCol(c.getLocation().x);
+ boolean contains = false;
+ for (SudokuCell rc : cregion) {
+ if (rc.getData() == cell.getData()) {
+ contains = true;
+ }
+ }
+ for (SudokuCell cc : ccol) {
+ if (cc.getData() == cell.getData()) {
contains = true;
- break;
}
}
+ // Stop if another cell can hold number
if (!contains) {
- return null;
+ restrained = false;
+ break;
}
}
- if (col.size() == size - 1) {
- contains = false;
- for (SudokuCell c : col) {
- if (cell.getData() == c.getData()) {
+ // Output if success
+ if (restrained) {
+ return null;
+ }
+
+ // //
+ // Loop to see if the number is constrained to the cell
+ restrained = true;
+ for (SudokuCell c : col) {
+ // Test if its not a valid testing cell
+ if (c.getData() != 0) {
+ continue;
+ }
+ if (c.getLocation().y == cell.getLocation().y
+ && c.getLocation().x == cell.getLocation().x) {
+ continue;
+ }
+ // Check if cell is eligible to hold number
+ Set cregion = initialBoard.getRegion(c.getGroupIndex());
+ Set crow = initialBoard.getRow(c.getLocation().y);
+ boolean contains = false;
+ for (SudokuCell rc : cregion) {
+ if (rc.getData() == cell.getData()) {
contains = true;
- break;
}
}
+ for (SudokuCell cc : crow) {
+ if (cc.getData() == cell.getData()) {
+ contains = true;
+ }
+ }
+ // Stop if another cell can hold number
if (!contains) {
- return null;
+ restrained = false;
+ break;
}
}
+ // Output if success
+ if (restrained) {
+ return null;
+ }
+
+ // Output fail
return super.getInvalidUseOfRuleMessage() + ": Cell is not forced at this index";
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastNumberForCellDirectRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastNumberForCellDirectRule.java
index ca0ac3023..333d91749 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastNumberForCellDirectRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/LastNumberForCellDirectRule.java
@@ -16,7 +16,7 @@ public LastNumberForCellDirectRule() {
"SUDO-BASC-0003",
"Last Number for Cell",
"This is the only number left that can fit in the cell of a group.",
- "edu/rpi/legup/images/sudoku/forcedByDeduction.png");
+ "edu/rpi/legup/images/sudoku/rules/forcedByDeduction.png");
}
/**
@@ -32,28 +32,37 @@ public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElem
SudokuBoard initialBoard = (SudokuBoard) transition.getParents().get(0).getBoard();
SudokuBoard finalBoard = (SudokuBoard) transition.getBoard();
- int index = puzzleElement.getIndex();
+ // Assign basics
int groupSize = initialBoard.getWidth();
int groupDim = (int) Math.sqrt(groupSize);
+
+ // Get position info
+ int index = puzzleElement.getIndex();
int rowIndex = index / groupSize;
int colIndex = index % groupSize;
- int groupNum = rowIndex / groupDim * groupDim + colIndex % groupDim;
+ int groupNum = (rowIndex / groupDim) * groupDim + (colIndex / groupDim);
+
+ // Create hashset of all numbers
HashSet numbers = new HashSet<>();
for (int i = 1; i <= groupSize; i++) {
numbers.add(i);
}
+
+ // Run through region, row, col to see contradicitng numbers
for (int i = 0; i < groupSize; i++) {
SudokuCell cell = initialBoard.getCell(groupNum, i % groupDim, i / groupDim);
numbers.remove(cell.getData());
}
for (int i = 0; i < groupSize; i++) {
- SudokuCell cell = initialBoard.getCell(i, colIndex);
+ SudokuCell cell = initialBoard.getCell(i, rowIndex);
numbers.remove(cell.getData());
}
for (int i = 0; i < groupSize; i++) {
- SudokuCell cell = initialBoard.getCell(rowIndex, i);
+ SudokuCell cell = initialBoard.getCell(colIndex, i);
numbers.remove(cell.getData());
}
+
+ // Check if plausible
if (numbers.size() > 1) {
return super.getInvalidUseOfRuleMessage() + ": The number at the index is not forced";
} else {
@@ -64,7 +73,11 @@ public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElem
+ ": The number at the index is forced but not correct";
}
}
- return null;
+ if (numbers.toArray(new Integer[1])[0] == puzzleElement.getData()) {
+ return null;
+ }
+ return super.getInvalidUseOfRuleMessage()
+ + ": The number at the index is forced but not correct";
}
/**
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberColumnContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberColumnContradictionRule.java
new file mode 100644
index 000000000..c8d627634
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberColumnContradictionRule.java
@@ -0,0 +1,90 @@
+package edu.rpi.legup.puzzle.sudoku.rules;
+
+import edu.rpi.legup.model.gameboard.Board;
+import edu.rpi.legup.model.gameboard.PuzzleElement;
+import edu.rpi.legup.model.rules.ContradictionRule;
+import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
+import edu.rpi.legup.puzzle.sudoku.SudokuCell;
+import java.util.HashSet;
+import java.util.Set;
+
+public class NoCellForNumberColumnContradictionRule extends ContradictionRule {
+
+ public NoCellForNumberColumnContradictionRule() {
+ super(
+ "SUDO-CONT-0003",
+ "No Cell for Number (Column)",
+ "Process of elimination yields no valid numbers for an empty cell in a column.",
+ "edu/rpi/legup/images/sudoku/rules/NoCellForNumberColumn.png");
+ }
+
+ /**
+ * Checks whether the transition has a contradiction at the specific puzzleElement index using
+ * this rule
+ *
+ * @param board board to check contradiction
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the transition contains a contradiction at the specified puzzleElement,
+ * otherwise error message
+ */
+ @Override
+ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
+ SudokuBoard sudokuBoard = (SudokuBoard) board;
+ SudokuCell cell = (SudokuCell) sudokuBoard.getPuzzleElement(puzzleElement);
+ if (cell.getData() != 0) {
+ return super.getNoContradictionMessage();
+ }
+
+ int groupSize = sudokuBoard.getSize();
+
+ Set col = sudokuBoard.getCol(cell.getGroupIndex());
+ Set numbersNotInColumn = new HashSet<>();
+
+ for (int i = 1; i <= groupSize; i++) {
+ numbersNotInColumn.add(i);
+ }
+ for (SudokuCell c : col) {
+ if (c.getData() != 0) {
+ numbersNotInColumn.remove(c.getData());
+ }
+ }
+
+ for (Integer i : numbersNotInColumn) {
+ // Check if number can be in cell
+ boolean canFit = false;
+ for (SudokuCell c : col) {
+ if (c.getData() != 0) {
+ continue;
+ }
+
+ // Get row and col groups
+ Set region = sudokuBoard.getRow(c.getLocation().y);
+ Set row = sudokuBoard.getCol(c.getLocation().x);
+
+ // Check if it alr exists in row or col
+ boolean duplicate = false;
+ for (SudokuCell rc : region) {
+ if (rc.getData() == i) {
+ duplicate = true;
+ }
+ }
+ for (SudokuCell cc : row) {
+ if (cc.getData() == i) {
+ duplicate = true;
+ }
+ }
+
+ // If there is no duplicate it can exist in the region
+ if (!duplicate) {
+ canFit = true;
+ break;
+ }
+ }
+ // If the number can't fit anywhere in region then contradiction
+ if (!canFit) {
+ return null;
+ }
+ }
+ return super.getNoContradictionMessage();
+ }
+}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRegionContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRegionContradictionRule.java
new file mode 100644
index 000000000..f5106b858
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRegionContradictionRule.java
@@ -0,0 +1,90 @@
+package edu.rpi.legup.puzzle.sudoku.rules;
+
+import edu.rpi.legup.model.gameboard.Board;
+import edu.rpi.legup.model.gameboard.PuzzleElement;
+import edu.rpi.legup.model.rules.ContradictionRule;
+import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
+import edu.rpi.legup.puzzle.sudoku.SudokuCell;
+import java.util.HashSet;
+import java.util.Set;
+
+public class NoCellForNumberRegionContradictionRule extends ContradictionRule {
+
+ public NoCellForNumberRegionContradictionRule() {
+ super(
+ "SUDO-CONT-0001",
+ "No Cell for Number (Region)",
+ "Process of elimination yields no valid numbers for an empty cell in a region.",
+ "edu/rpi/legup/images/sudoku/rules/NoCellForNumberRegion.png");
+ }
+
+ /**
+ * Checks whether the transition has a contradiction at the specific puzzleElement index using
+ * this rule
+ *
+ * @param board board to check contradiction
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the transition contains a contradiction at the specified puzzleElement,
+ * otherwise error message
+ */
+ @Override
+ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
+ SudokuBoard sudokuBoard = (SudokuBoard) board;
+ SudokuCell cell = (SudokuCell) sudokuBoard.getPuzzleElement(puzzleElement);
+ if (cell.getData() != 0) {
+ return super.getNoContradictionMessage();
+ }
+
+ int groupSize = sudokuBoard.getSize();
+
+ Set region = sudokuBoard.getRegion(cell.getGroupIndex());
+ Set numbersNotInRegion = new HashSet<>();
+
+ for (int i = 1; i <= groupSize; i++) {
+ numbersNotInRegion.add(i);
+ }
+ for (SudokuCell c : region) {
+ if (c.getData() != 0) {
+ numbersNotInRegion.remove(c.getData());
+ }
+ }
+
+ for (Integer i : numbersNotInRegion) {
+ // Check if number can be in cell
+ boolean canFit = false;
+ for (SudokuCell c : region) {
+ if (c.getData() != 0) {
+ continue;
+ }
+
+ // Get row and col groups
+ Set row = sudokuBoard.getRow(c.getLocation().y);
+ Set col = sudokuBoard.getCol(c.getLocation().x);
+
+ // Check if it alr exists in row or col
+ boolean duplicate = false;
+ for (SudokuCell rc : row) {
+ if (rc.getData() == i) {
+ duplicate = true;
+ }
+ }
+ for (SudokuCell cc : col) {
+ if (cc.getData() == i) {
+ duplicate = true;
+ }
+ }
+
+ // If there is no duplicate it can exist in the region
+ if (!duplicate) {
+ canFit = true;
+ break;
+ }
+ }
+ // If the number can't fit anywhere in region then contradiction
+ if (!canFit) {
+ return null;
+ }
+ }
+ return super.getNoContradictionMessage();
+ }
+}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRowContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRowContradictionRule.java
new file mode 100644
index 000000000..e3f9f764a
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoCellForNumberRowContradictionRule.java
@@ -0,0 +1,90 @@
+package edu.rpi.legup.puzzle.sudoku.rules;
+
+import edu.rpi.legup.model.gameboard.Board;
+import edu.rpi.legup.model.gameboard.PuzzleElement;
+import edu.rpi.legup.model.rules.ContradictionRule;
+import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
+import edu.rpi.legup.puzzle.sudoku.SudokuCell;
+import java.util.HashSet;
+import java.util.Set;
+
+public class NoCellForNumberRowContradictionRule extends ContradictionRule {
+
+ public NoCellForNumberRowContradictionRule() {
+ super(
+ "SUDO-CONT-0002",
+ "No Cell for Number (Row)",
+ "Process of elimination yields no valid numbers for an empty cell in a row.",
+ "edu/rpi/legup/images/sudoku/rules/NoCellForNumberRow.png");
+ }
+
+ /**
+ * Checks whether the transition has a contradiction at the specific puzzleElement index using
+ * this rule
+ *
+ * @param board board to check contradiction
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the transition contains a contradiction at the specified puzzleElement,
+ * otherwise error message
+ */
+ @Override
+ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
+ SudokuBoard sudokuBoard = (SudokuBoard) board;
+ SudokuCell cell = (SudokuCell) sudokuBoard.getPuzzleElement(puzzleElement);
+ if (cell.getData() != 0) {
+ return super.getNoContradictionMessage();
+ }
+
+ int groupSize = sudokuBoard.getSize();
+
+ Set row = sudokuBoard.getRow(cell.getGroupIndex());
+ Set numbersNotInRow = new HashSet<>();
+
+ for (int i = 1; i <= groupSize; i++) {
+ numbersNotInRow.add(i);
+ }
+ for (SudokuCell c : row) {
+ if (c.getData() != 0) {
+ numbersNotInRow.remove(c.getData());
+ }
+ }
+
+ for (Integer i : numbersNotInRow) {
+ // Check if number can be in cell
+ boolean canFit = false;
+ for (SudokuCell c : row) {
+ if (c.getData() != 0) {
+ continue;
+ }
+
+ // Get row and col groups
+ Set region = sudokuBoard.getRow(c.getLocation().y);
+ Set col = sudokuBoard.getCol(c.getLocation().x);
+
+ // Check if it alr exists in row or col
+ boolean duplicate = false;
+ for (SudokuCell rc : region) {
+ if (rc.getData() == i) {
+ duplicate = true;
+ }
+ }
+ for (SudokuCell cc : col) {
+ if (cc.getData() == i) {
+ duplicate = true;
+ }
+ }
+
+ // If there is no duplicate it can exist in the region
+ if (!duplicate) {
+ canFit = true;
+ break;
+ }
+ }
+ // If the number can't fit anywhere in region then contradiction
+ if (!canFit) {
+ return null;
+ }
+ }
+ return super.getNoContradictionMessage();
+ }
+}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoSolutionContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoNumberForCellContradictionRule.java
similarity index 72%
rename from src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoSolutionContradictionRule.java
rename to src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoNumberForCellContradictionRule.java
index e44728d3e..6ea8f0a2a 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoSolutionContradictionRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/NoNumberForCellContradictionRule.java
@@ -8,14 +8,14 @@
import java.util.HashSet;
import java.util.Set;
-public class NoSolutionContradictionRule extends ContradictionRule {
+public class NoNumberForCellContradictionRule extends ContradictionRule {
- public NoSolutionContradictionRule() {
+ public NoNumberForCellContradictionRule() {
super(
- "SUDO-CONT-0001",
- "No Solution for Cell",
+ "SUDO-CONT-0004",
+ "No Number for Cell",
"Process of elimination yields no valid numbers for an empty cell.",
- "edu/rpi/legup/images/sudoku/NoSolution.png");
+ "edu/rpi/legup/images/sudoku/rules/NoSolution.png");
}
/**
@@ -41,21 +41,19 @@ public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
Set row = sudokuBoard.getRow(cell.getLocation().y);
Set col = sudokuBoard.getCol(cell.getLocation().x);
Set solution = new HashSet<>();
- for (int i = 1; i <= groupSize; i++) {
- solution.add(i);
+ for (SudokuCell s : region) {
+ solution.add(s.getData());
}
-
- for (SudokuCell c : region) {
- solution.remove(c.getData());
- }
- for (SudokuCell c : row) {
- solution.remove(c.getData());
+ for (SudokuCell s : row) {
+ solution.add(s.getData());
}
- for (SudokuCell c : col) {
- solution.remove(c.getData());
+
+ for (SudokuCell s : col) {
+ solution.add(s.getData());
}
+ solution.remove(0);
- if (solution.isEmpty()) {
+ if (solution.size() == 9) {
return null;
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleNumberCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberColumnCaseRule.java
similarity index 53%
rename from src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleNumberCaseRule.java
rename to src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberColumnCaseRule.java
index e6ab0e64c..bab0bc79b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleNumberCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberColumnCaseRule.java
@@ -6,21 +6,26 @@
import edu.rpi.legup.model.rules.CaseRule;
import edu.rpi.legup.model.tree.TreeTransition;
import edu.rpi.legup.puzzle.sudoku.GroupType;
-import edu.rpi.legup.puzzle.sudoku.PossibleNumberCaseBoard;
+import edu.rpi.legup.puzzle.sudoku.ModelSudokuBoard;
import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
import edu.rpi.legup.puzzle.sudoku.SudokuCell;
import java.util.ArrayList;
-import java.util.List;
import java.util.Set;
-public class PossibleNumberCaseRule extends CaseRule {
+public class PossibleCellsForNumberColumnCaseRule extends CaseRule {
- public PossibleNumberCaseRule() {
+ // Board math for translating indexes to numbers
+ private ModelSudokuBoard model = new ModelSudokuBoard();
+
+ // Old board for caseBoard reference
+ private SudokuBoard lagBoard;
+
+ public PossibleCellsForNumberColumnCaseRule() {
super(
- "SUDO-CASE-0002",
- "Possible Numbers for Cell",
+ "SUDO-CASE-0004",
+ "Possible Cells for Number - Column",
"An empty cell has a limited set of possible numbers that can fill it.",
- "edu/rpi/legup/images/sudoku/PossibleValues.png");
+ "edu/rpi/legup/images/sudoku/rules/possible_cells_number_column.png");
}
/**
@@ -50,12 +55,12 @@ public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElem
@Override
public CaseBoard getCaseBoard(Board board) {
- SudokuBoard sudokuBoard = (SudokuBoard) board;
- PossibleNumberCaseBoard caseBoard = new PossibleNumberCaseBoard(sudokuBoard, this, null);
- for (int i = 0; i < sudokuBoard.getSize(); i++) {
- caseBoard.addPickableRegion(i);
- caseBoard.addPickableRow(i);
- caseBoard.addPickableCol(i);
+ SudokuBoard sudokuBoard = (SudokuBoard) board.copy();
+ lagBoard = (SudokuBoard) sudokuBoard.copy();
+ CaseBoard caseBoard = new CaseBoard(sudokuBoard, this);
+ for (PuzzleElement puzzleElement : sudokuBoard.getPuzzleElements()) {
+ puzzleElement.setData(model.getModelColumnNumbers(puzzleElement.getIndex()));
+ caseBoard.addPickableElement(puzzleElement);
}
return caseBoard;
}
@@ -69,7 +74,7 @@ public CaseBoard getCaseBoard(Board board) {
*/
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
- return getCases(board, puzzleElement, 1, GroupType.REGION);
+ return getCases(board, puzzleElement, 1, GroupType.COLUMN);
}
/**
@@ -84,48 +89,19 @@ public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
public ArrayList getCases(
Board board, PuzzleElement puzzleElement, int value, GroupType groupType) {
ArrayList cases = new ArrayList<>();
- SudokuBoard sudokuBoard = (SudokuBoard) board;
- List caseCells = new ArrayList<>();
- SudokuCell cell = (SudokuCell) puzzleElement;
+ SudokuBoard sudokuBoard = lagBoard;
+ SudokuCell sourceCell = (SudokuCell) puzzleElement;
- Set group;
- if (groupType == GroupType.REGION) {
- group = sudokuBoard.getRegion(cell.getGroupIndex());
- } else {
- if (groupType == GroupType.ROW) {
- group = sudokuBoard.getRow(cell.getLocation().y);
- } else {
- group = sudokuBoard.getCol(cell.getLocation().x);
+ Set group = sudokuBoard.getCol(sourceCell.getLocation().x);
+ for (SudokuCell cell : group) {
+ if (cell.getData() == 0) {
+ Board newCase = sudokuBoard.copy();
+ PuzzleElement element = newCase.getPuzzleElement(cell);
+ element.setData(model.getModelColumnNumbers(sourceCell.getIndex()));
+ newCase.addModifiedData(element);
+ cases.add(newCase);
}
}
-
- for (SudokuCell c : group) {
- if (c.getData() == 0) {
- Set blockableCells = sudokuBoard.getRegion(c.getGroupIndex());
- blockableCells.addAll(sudokuBoard.getRow(c.getLocation().y));
- blockableCells.addAll(sudokuBoard.getCol(c.getLocation().x));
-
- boolean repeat = false;
- for (SudokuCell bc : blockableCells) {
- if (bc.getData() == value) {
- repeat = true;
- break;
- }
- }
- if (!repeat) {
- caseCells.add(c);
- }
- }
- }
-
- for (SudokuCell c : caseCells) {
- Board newCase = sudokuBoard.copy();
- PuzzleElement element = newCase.getPuzzleElement(c);
- element.setData(value);
- newCase.addModifiedData(element);
- cases.add(newCase);
- }
-
return cases;
}
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRegionCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRegionCaseRule.java
new file mode 100644
index 000000000..47e408369
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRegionCaseRule.java
@@ -0,0 +1,104 @@
+package edu.rpi.legup.puzzle.sudoku.rules;
+
+import edu.rpi.legup.model.gameboard.Board;
+import edu.rpi.legup.model.gameboard.CaseBoard;
+import edu.rpi.legup.model.gameboard.PuzzleElement;
+import edu.rpi.legup.model.rules.CaseRule;
+import edu.rpi.legup.model.tree.TreeTransition;
+import edu.rpi.legup.puzzle.sudoku.*;
+import java.util.ArrayList;
+import java.util.Set;
+
+public class PossibleCellsForNumberRegionCaseRule extends CaseRule {
+
+ // Board math for translating indexes to numbers
+ private ModelSudokuBoard model = new ModelSudokuBoard();
+
+ // Old board for caseBoard reference
+ private SudokuBoard lagBoard;
+
+ public PossibleCellsForNumberRegionCaseRule() {
+ super(
+ "SUDO-CASE-0002",
+ "Possible Cells for Number - Region",
+ "An empty cell has a limited set of possible numbers that can fill it.",
+ "edu/rpi/legup/images/sudoku/rules/possible_cells_number_region.png");
+ }
+
+ /**
+ * Checks whether the transition logically follows from the parent node using this rule
+ *
+ * @param transition transition to check
+ * @return null if the child node logically follow from the parent node, otherwise error message
+ */
+ @Override
+ public String checkRuleRaw(TreeTransition transition) {
+ return null;
+ }
+
+ /**
+ * Checks whether the child node logically follows from the parent node at the specific
+ * puzzleElement index using this rule
+ *
+ * @param transition transition to check
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the child node logically follow from the parent node at the specified
+ * puzzleElement, otherwise error message
+ */
+ @Override
+ public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
+ return null;
+ }
+
+ @Override
+ public CaseBoard getCaseBoard(Board board) {
+ SudokuBoard sudokuBoard = (SudokuBoard) board.copy();
+ lagBoard = (SudokuBoard) sudokuBoard.copy();
+ CaseBoard caseBoard = new CaseBoard(sudokuBoard, this);
+ for (PuzzleElement puzzleElement : sudokuBoard.getPuzzleElements()) {
+ puzzleElement.setData(model.getModelRegionNumbers(puzzleElement.getIndex()));
+ caseBoard.addPickableElement(puzzleElement);
+ }
+ return caseBoard;
+ }
+
+ /**
+ * Gets the possible cases at a specific location based on this case rule
+ *
+ * @param board the current board state
+ * @param puzzleElement equivalent puzzleElement
+ * @return a list of elements the specified could be
+ */
+ @Override
+ public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
+ return getCases(board, puzzleElement, 1, GroupType.REGION);
+ }
+
+ /**
+ * Gets the possible cases at a specific location based on this case rule
+ *
+ * @param board the current board state
+ * @param puzzleElement equivalent puzzleElement
+ * @param value value that the rule will be applied from
+ * @param groupType group type
+ * @return a list of elements the specified could be
+ */
+ public ArrayList getCases(
+ Board board, PuzzleElement puzzleElement, int value, GroupType groupType) {
+ ArrayList cases = new ArrayList<>();
+ SudokuBoard sudokuBoard = lagBoard;
+ SudokuCell sourceCell = (SudokuCell) puzzleElement;
+
+ Set group = sudokuBoard.getRegion(sourceCell.getGroupIndex());
+ for (SudokuCell cell : group) {
+ if (cell.getData() == 0) {
+ Board newCase = sudokuBoard.copy();
+ PuzzleElement element = newCase.getPuzzleElement(cell);
+ element.setData(model.getModelRegionNumbers(sourceCell.getIndex()));
+ newCase.addModifiedData(element);
+ cases.add(newCase);
+ }
+ }
+ return cases;
+ }
+}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRowCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRowCaseRule.java
new file mode 100644
index 000000000..868541377
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellsForNumberRowCaseRule.java
@@ -0,0 +1,107 @@
+package edu.rpi.legup.puzzle.sudoku.rules;
+
+import edu.rpi.legup.model.gameboard.Board;
+import edu.rpi.legup.model.gameboard.CaseBoard;
+import edu.rpi.legup.model.gameboard.PuzzleElement;
+import edu.rpi.legup.model.rules.CaseRule;
+import edu.rpi.legup.model.tree.TreeTransition;
+import edu.rpi.legup.puzzle.sudoku.GroupType;
+import edu.rpi.legup.puzzle.sudoku.ModelSudokuBoard;
+import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
+import edu.rpi.legup.puzzle.sudoku.SudokuCell;
+import java.util.ArrayList;
+import java.util.Set;
+
+public class PossibleCellsForNumberRowCaseRule extends CaseRule {
+
+ // Board math for translating indexes to numbers
+ private ModelSudokuBoard model = new ModelSudokuBoard();
+
+ // Old board for caseBoard reference
+ private SudokuBoard lagBoard;
+
+ public PossibleCellsForNumberRowCaseRule() {
+ super(
+ "SUDO-CASE-0003",
+ "Possible Cells for Number - Row",
+ "An empty cell has a limited set of possible numbers that can fill it.",
+ "edu/rpi/legup/images/sudoku/rules/possible_cells_number_row.png");
+ }
+
+ /**
+ * Checks whether the transition logically follows from the parent node using this rule
+ *
+ * @param transition transition to check
+ * @return null if the child node logically follow from the parent node, otherwise error message
+ */
+ @Override
+ public String checkRuleRaw(TreeTransition transition) {
+ return null;
+ }
+
+ /**
+ * Checks whether the child node logically follows from the parent node at the specific
+ * puzzleElement index using this rule
+ *
+ * @param transition transition to check
+ * @param puzzleElement equivalent puzzleElement
+ * @return null if the child node logically follow from the parent node at the specified
+ * puzzleElement, otherwise error message
+ */
+ @Override
+ public String checkRuleRawAt(TreeTransition transition, PuzzleElement puzzleElement) {
+ return null;
+ }
+
+ @Override
+ public CaseBoard getCaseBoard(Board board) {
+ SudokuBoard sudokuBoard = (SudokuBoard) board.copy();
+ lagBoard = (SudokuBoard) sudokuBoard.copy();
+ CaseBoard caseBoard = new CaseBoard(sudokuBoard, this);
+ for (PuzzleElement puzzleElement : sudokuBoard.getPuzzleElements()) {
+ puzzleElement.setData(model.getModelRowNumbers(puzzleElement.getIndex()));
+ caseBoard.addPickableElement(puzzleElement);
+ }
+ return caseBoard;
+ }
+
+ /**
+ * Gets the possible cases at a specific location based on this case rule
+ *
+ * @param board the current board state
+ * @param puzzleElement equivalent puzzleElement
+ * @return a list of elements the specified could be
+ */
+ @Override
+ public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
+ return getCases(board, puzzleElement, 1, GroupType.ROW);
+ }
+
+ /**
+ * Gets the possible cases at a specific location based on this case rule
+ *
+ * @param board the current board state
+ * @param puzzleElement equivalent puzzleElement
+ * @param value value that the rule will be applied from
+ * @param groupType group type
+ * @return a list of elements the specified could be
+ */
+ public ArrayList getCases(
+ Board board, PuzzleElement puzzleElement, int value, GroupType groupType) {
+ ArrayList cases = new ArrayList<>();
+ SudokuBoard sudokuBoard = lagBoard;
+ SudokuCell sourceCell = (SudokuCell) puzzleElement;
+
+ Set group = sudokuBoard.getRow(sourceCell.getLocation().y);
+ for (SudokuCell cell : group) {
+ if (cell.getData() == 0) {
+ Board newCase = sudokuBoard.copy();
+ PuzzleElement element = newCase.getPuzzleElement(cell);
+ element.setData(model.getModelRowNumbers(sourceCell.getIndex()));
+ newCase.addModifiedData(element);
+ cases.add(newCase);
+ }
+ }
+ return cases;
+ }
+}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleNumbersForCellCaseRule.java
similarity index 59%
rename from src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellCaseRule.java
rename to src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleNumbersForCellCaseRule.java
index fb6da62d4..e17acc26b 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleCellCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/PossibleNumbersForCellCaseRule.java
@@ -5,19 +5,18 @@
import edu.rpi.legup.model.gameboard.PuzzleElement;
import edu.rpi.legup.model.rules.CaseRule;
import edu.rpi.legup.model.tree.TreeTransition;
-import edu.rpi.legup.puzzle.sudoku.SudokuBoard;
-import edu.rpi.legup.puzzle.sudoku.SudokuCell;
+import edu.rpi.legup.puzzle.sudoku.*;
import java.util.ArrayList;
-import java.util.HashSet;
-import java.util.Set;
+import java.util.List;
-public class PossibleCellCaseRule extends CaseRule {
- public PossibleCellCaseRule() {
+public class PossibleNumbersForCellCaseRule extends CaseRule {
+
+ public PossibleNumbersForCellCaseRule() {
super(
"SUDO-CASE-0001",
- "Possible Cells for Number",
- "A number has a limited set of cells in which it can be placed.",
- "edu/rpi/legup/images/sudoku/possible_cells_number.png");
+ "Possible Numbers for Cell",
+ "An empty cell has a limited set of possible numbers that can fill it.",
+ "edu/rpi/legup/images/sudoku/rules/PossibleValues.png");
}
/**
@@ -66,42 +65,34 @@ public CaseBoard getCaseBoard(Board board) {
*/
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
- ArrayList cases = new ArrayList<>();
- SudokuBoard sudokuBoard = (SudokuBoard) board;
- SudokuCell cell = (SudokuCell) puzzleElement;
-
- Set possibleValue = new HashSet<>();
- for (int i = 1; i <= sudokuBoard.getSize(); i++) {
- possibleValue.add(i);
- }
-
- int groupNum = cell.getGroupIndex();
- for (SudokuCell c : sudokuBoard.getRegion(groupNum)) {
- if (c.getData().equals(c.getData())) {
- possibleValue.remove(c.getData());
- }
- }
-
- int rowNum = cell.getLocation().y;
- for (SudokuCell c : sudokuBoard.getRegion(rowNum)) {
- if (c.getData().equals(c.getData())) {
- possibleValue.remove(c.getData());
- }
- }
+ return getCases(board, puzzleElement, 1, GroupType.REGION);
+ }
- int colNum = cell.getLocation().x;
- for (SudokuCell c : sudokuBoard.getRegion(colNum)) {
- if (c.getData().equals(c.getData())) {
- possibleValue.remove(c.getData());
- }
+ /**
+ * Gets the possible cases at a specific location based on this case rule
+ *
+ * @param board the current board state
+ * @param puzzleElement equivalent puzzleElement
+ * @param value value that the rule will be applied from
+ * @param groupType group type
+ * @return a list of elements the specified could be
+ */
+ public ArrayList getCases(
+ Board board, PuzzleElement puzzleElement, int value, GroupType groupType) {
+ ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
}
- for (Integer i : possibleValue) {
- SudokuBoard newCase = sudokuBoard.copy();
+ SudokuBoard sudokuBoard = (SudokuBoard) board;
+ List caseCells = new ArrayList<>();
+ SudokuCell cell = (SudokuCell) puzzleElement;
- PuzzleElement newCasePuzzleElement = newCase.getPuzzleElement(puzzleElement);
- newCasePuzzleElement.setData(i);
- newCase.addModifiedData(newCasePuzzleElement);
+ for (int i = 1; i <= 9; i++) {
+ Board newCase = sudokuBoard.copy();
+ PuzzleElement element = newCase.getPuzzleElement(puzzleElement);
+ element.setData(i);
+ newCase.addModifiedData(element);
cases.add(newCase);
}
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/RepeatedNumberContradictionRule.java b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/RepeatedNumberContradictionRule.java
index 955414e8e..f8172d071 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/RepeatedNumberContradictionRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/RepeatedNumberContradictionRule.java
@@ -12,10 +12,10 @@ public class RepeatedNumberContradictionRule extends ContradictionRule {
public RepeatedNumberContradictionRule() {
super(
- "SUDO-CONT-0002",
+ "SUDO-CONT-0005",
"Repeated Numbers",
"Two identical numbers are placed in the same group.",
- "edu/rpi/legup/images/sudoku/RepeatedNumber.png");
+ "edu/rpi/legup/images/sudoku/rules/RepeatedNumber.png");
}
/**
@@ -29,39 +29,51 @@ public RepeatedNumberContradictionRule() {
*/
@Override
public String checkContradictionAt(Board board, PuzzleElement puzzleElement) {
+ // Get board to check
SudokuBoard sudokuBoard = (SudokuBoard) board;
- SudokuCell cell = (SudokuCell) sudokuBoard.getPuzzleElement(puzzleElement);
- if (cell.getData() == 0) {
- return super.getNoContradictionMessage();
- }
- Set region = sudokuBoard.getRegion(cell.getGroupIndex());
- Set row = sudokuBoard.getRow(cell.getLocation().y);
- Set col = sudokuBoard.getCol(cell.getLocation().x);
+ // Loop all group indexes
+ for (int i = 0; i < 9; i++) {
+ // Get regions and sets to check duplicates
+ Set region = sudokuBoard.getRegion(i);
+ Set regionDup = new HashSet<>();
+
+ Set row = sudokuBoard.getRow(i);
+ Set rowDup = new HashSet<>();
- Set regionDup = new HashSet<>();
- Set rowDup = new HashSet<>();
- Set colDup = new HashSet<>();
+ Set col = sudokuBoard.getCol(i);
+ Set colDup = new HashSet<>();
- for (SudokuCell c : region) {
- if (regionDup.contains(c.getData())) {
- return null;
+ // Check for non zero duplicates to trigger contradiction
+ for (SudokuCell c : region) {
+ if (c.getData() == 0) {
+ continue;
+ }
+ if (regionDup.contains(c.getData())) {
+ return null;
+ }
+ regionDup.add(c.getData());
}
- regionDup.add(c.getData());
- }
- for (SudokuCell c : row) {
- if (rowDup.contains(c.getData())) {
- return null;
+ for (SudokuCell c : row) {
+ if (c.getData() == 0) {
+ continue;
+ }
+ if (rowDup.contains(c.getData())) {
+ return null;
+ }
+ rowDup.add(c.getData());
}
- rowDup.add(c.getData());
- }
- for (SudokuCell c : col) {
- if (colDup.contains(c.getData())) {
- return null;
+ for (SudokuCell c : col) {
+ if (c.getData() == 0) {
+ continue;
+ }
+ if (colDup.contains(c.getData())) {
+ return null;
+ }
+ colDup.add(c.getData());
}
- colDup.add(c.getData());
}
return super.getNoContradictionMessage();
diff --git a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/sudoku_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/sudoku_reference_sheet.txt
index a8635330d..ceffa168c 100644
--- a/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/sudoku_reference_sheet.txt
+++ b/src/main/java/edu/rpi/legup/puzzle/sudoku/rules/sudoku_reference_sheet.txt
@@ -1,9 +1,13 @@
-SUDO-BASC-0001 : AdvancedDeductionDirectRule
SUDO-BASC-0002 : LastCellForNumberDirectRule
SUDO-BASC-0003 : LastNumberForCellDirectRule
-SUDO-CONT-0001 : NoSolutionContradictionRule
-SUDO-CONT-0002 : RepeatedNumberContradictionRule
+SUDO-CONT-0001 : NoCellForNumberRegionContradictionRule
+SUDO-CONT-0002 : NoCellForNumberRowContradictionRule
+SUDO-CONT-0003 : NoCellForNumberColumnContradictionRule
+SUDO-CONT-0004 : NoNumberForCellContradictionRule
+SUDO-CONT-0005 : RepeatedNumberContradictionRule
-SUDO-CASE-0001 : PossibleCellCaseRule
-SUDO-CASE-0002 : PossibleNumberCaseRule
\ No newline at end of file
+SUDO-CASE-0001 : PossibleNumbersForCellCaseRule
+SUDO-CASE-0002 : PossibleCellsForNumberRegionCaseRule
+SUDO-CASE-0003 : PossibleCellsForNumberRowCaseRule
+SUDO-CASE-0004 : PossibleCellsForNumberColumnCaseRule
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/TreeTent.java b/src/main/java/edu/rpi/legup/puzzle/treetent/TreeTent.java
index 5ec135a22..35993303a 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/TreeTent.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/TreeTent.java
@@ -59,7 +59,20 @@ public boolean isValidDimensions(int rows, int columns) {
*/
@Override
public boolean isBoardComplete(Board board) {
- return false;
+ TreeTentBoard treeTentBoard = (TreeTentBoard) board;
+
+ for (ContradictionRule rule : contradictionRules) {
+ if (rule.checkContradiction(treeTentBoard) == null) {
+ return false;
+ }
+ }
+ for (PuzzleElement data : treeTentBoard.getPuzzleElements()) {
+ TreeTentCell cell = (TreeTentCell) data;
+ if (cell.getType() == TreeTentType.UNKNOWN) {
+ return false;
+ }
+ }
+ return true;
}
/**
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/GrassTile.java b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/GrassTile.java
index 5356120a8..1d33b9035 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/GrassTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/GrassTile.java
@@ -6,7 +6,7 @@ public class GrassTile extends PlaceableElement {
public GrassTile() {
super(
- "TREE-PlAC-0002",
+ "TREE-ELEM-0001",
"Grass Tile",
"The grass crest tile",
"edu/rpi/legup/images/treetent/grass.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TentTile.java b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TentTile.java
index 950aebfa7..96124a98d 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TentTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TentTile.java
@@ -6,7 +6,7 @@ public class TentTile extends PlaceableElement {
public TentTile() {
super(
- "TREE-PLAC-0001",
+ "TREE-ELEM-0002",
"Tent Tile",
"The tent tile",
"edu/rpi/legup/images/treetent/tent.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TreeTile.java b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TreeTile.java
index d04886ed5..3d94cbfba 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TreeTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/TreeTile.java
@@ -1,12 +1,12 @@
package edu.rpi.legup.puzzle.treetent.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class TreeTile extends NonPlaceableElement {
+public class TreeTile extends PlaceableElement {
public TreeTile() {
super(
- "TREE-UNPL-0001",
+ "TREE-ELEM-0003",
"Tree Tile",
"The tree tile",
"edu/rpi/legup/images/treetent/tree.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/UnknownTile.java b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/UnknownTile.java
index a54240efd..99b75b60c 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/UnknownTile.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/UnknownTile.java
@@ -1,11 +1,11 @@
package edu.rpi.legup.puzzle.treetent.elements;
-import edu.rpi.legup.model.elements.NonPlaceableElement;
+import edu.rpi.legup.model.elements.PlaceableElement;
-public class UnknownTile extends NonPlaceableElement {
+public class UnknownTile extends PlaceableElement {
public UnknownTile() {
super(
- "TREE-UNPL-0002",
+ "TREE-ELEM-0004",
"Unknown Tile",
"The blank tile",
"edu/rpi/legup/images/treetent/UnknownTile.png");
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/elements/treetent_elements_reference_sheet.txt b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/treetent_elements_reference_sheet.txt
new file mode 100644
index 000000000..e0cfc1dfa
--- /dev/null
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/elements/treetent_elements_reference_sheet.txt
@@ -0,0 +1,4 @@
+TREE-ELEM-0001 : GrassTile
+TREE-ELEM-0002 : TentTile
+TREE-ELEM-0003 : TreeTile
+TREE-ELEM-0004 : UnknownTile
\ No newline at end of file
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/rules/FillinRowCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/treetent/rules/FillinRowCaseRule.java
index aaa1a8fbc..8fe9b6873 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/rules/FillinRowCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/rules/FillinRowCaseRule.java
@@ -61,7 +61,11 @@ public CaseBoard getCaseBoard(Board board) {
*/
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
+ if (puzzleElement == null) {
+ return new ArrayList();
+ }
ArrayList cases;
+
List group;
int tentsLeft;
TreeTentClue clue = ((TreeTentClue) puzzleElement);
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTentCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTentCaseRule.java
index bd303174a..cbe91c3a7 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTentCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTentCaseRule.java
@@ -60,6 +60,10 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
TreeTentCell cell = (TreeTentCell) puzzleElement;
List adj = ((TreeTentBoard) board).getAdjacent(cell, TreeTentType.TREE);
List lines = ((TreeTentBoard) board).getLines();
diff --git a/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTreeCaseRule.java b/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTreeCaseRule.java
index 03d039898..153692ad0 100644
--- a/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTreeCaseRule.java
+++ b/src/main/java/edu/rpi/legup/puzzle/treetent/rules/LinkTreeCaseRule.java
@@ -62,6 +62,10 @@ public CaseBoard getCaseBoard(Board board) {
@Override
public ArrayList getCases(Board board, PuzzleElement puzzleElement) {
ArrayList cases = new ArrayList<>();
+ if (puzzleElement == null) {
+ return cases;
+ }
+
TreeTentBoard treeTentBoard = (TreeTentBoard) board;
TreeTentCell cell = (TreeTentCell) puzzleElement;
List adjCells = treeTentBoard.getAdjacent(cell, TreeTentType.TENT);
diff --git a/src/main/java/edu/rpi/legup/ui/CreatePuzzleDialog.java b/src/main/java/edu/rpi/legup/ui/CreatePuzzleDialog.java
index 32ddca34f..bde2df3d5 100644
--- a/src/main/java/edu/rpi/legup/ui/CreatePuzzleDialog.java
+++ b/src/main/java/edu/rpi/legup/ui/CreatePuzzleDialog.java
@@ -69,9 +69,7 @@ public void actionPerformed(ActionEvent e) {
*/
@Override
public void actionPerformed(ActionEvent ae) {
- String game =
- Config.convertDisplayNameToClassName(
- (String) gameBox.getSelectedItem());
+ String game = getGame();
// Check if all 3 TextFields are filled
if (game.equals("ShortTruthTable") && textArea.getText().isEmpty()) {
@@ -91,8 +89,8 @@ public void actionPerformed(ActionEvent ae) {
} else {
homePanel.openEditorWithNewPuzzle(
game,
- Integer.valueOf(rows.getText()),
- Integer.valueOf(columns.getText()));
+ Integer.valueOf(getRows()),
+ Integer.valueOf(getColumns()));
}
setVisible(false);
} catch (IllegalArgumentException e) {
@@ -116,6 +114,12 @@ public void actionPerformed(ActionEvent e) {
}
};
+ /**
+ * Constructs a new CreatePuzzleDialog
+ *
+ * @param parent the parent frame of the dialog
+ * @param homePanel the home panel where the created puzzle will be added
+ */
public CreatePuzzleDialog(JFrame parent, HomePanel homePanel) {
super(parent, true);
diff --git a/src/main/java/edu/rpi/legup/ui/DynamicView.java b/src/main/java/edu/rpi/legup/ui/DynamicView.java
index d6dbfb4c3..fa0004d0b 100644
--- a/src/main/java/edu/rpi/legup/ui/DynamicView.java
+++ b/src/main/java/edu/rpi/legup/ui/DynamicView.java
@@ -16,6 +16,10 @@
import javax.swing.*;
import javax.swing.event.ChangeEvent;
+/**
+ * A JPanel that provides a dynamic view with zooming capabilities for different types of content.
+ * This class supports views such as game boards or proof trees, allowing users to zoom in and out.
+ */
public class DynamicView extends JPanel {
private ScrollView scrollView;
@@ -29,6 +33,12 @@ public class DynamicView extends JPanel {
private static final Font INFO_FONT = MaterialFonts.REGULAR;
private static final Color INFO_COLOR = MaterialColors.GRAY_900;
+ /**
+ * Constructs a new DynamicView with the specified ScrollView and view type
+ *
+ * @param scrollView the ScrollView that provides the content to be displayed and zoomed
+ * @param type the type of dynamic view to set up (e.g., BOARD or PROOF_TREE)
+ */
public DynamicView(ScrollView scrollView, DynamicViewType type) {
this.scrollView = scrollView;
@@ -185,24 +195,49 @@ public void componentResized(ComponentEvent e) {
return zoomWrapper;
}
+ /**
+ * Gets the ScrollView component associated with this DynamicView
+ *
+ * @return the ScrollView component
+ */
public ScrollView getScrollView() {
return this.scrollView;
}
+ /**
+ * Gets the zoom wrapper that contains the zooming controls
+ *
+ * @return the zoom wrapper with zooming controls
+ */
public JPanel getZoomWrapper() {
return this.zoomWrapper;
}
+ /**
+ * Gets the zoomer that contains the zoomer component
+ *
+ * @return the zoomer with the zoomer component
+ */
public JPanel getZoomer() {
return this.zoomer;
}
+ /**
+ * Updates the status label with an informational message
+ *
+ * @param message the informational message to display
+ */
public void updateInfo(String message) {
status.setFont(INFO_FONT);
status.setForeground(INFO_COLOR);
status.setText(message);
}
+ /**
+ * Updates the status label with an error message
+ *
+ * @param message the error message to display
+ */
public void updateError(String message) {
status.setFont(ERROR_FONT);
status.setForeground(ERROR_COLOR);
diff --git a/src/main/java/edu/rpi/legup/ui/HomePanel.java b/src/main/java/edu/rpi/legup/ui/HomePanel.java
index 9e83dc9de..75761c475 100644
--- a/src/main/java/edu/rpi/legup/ui/HomePanel.java
+++ b/src/main/java/edu/rpi/legup/ui/HomePanel.java
@@ -46,40 +46,30 @@ public class HomePanel extends LegupPanel {
new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
- Object[] items = legupUI.getProofEditor().promptPuzzle();
- if (items == null) {
- // The attempt to prompt a puzzle ended gracefully (cancel)
- return;
- }
- String fileName = (String) items[0];
- File puzzleFile = (File) items[1];
- legupUI.getProofEditor().loadPuzzle(fileName, puzzleFile);
+ legupUI.getProofEditor().loadPuzzle("", null);
}
};
- private ActionListener openPuzzleListener =
- new ActionListener() {
- @Override
- public void actionPerformed(ActionEvent e) {
- Object[] items = legupUI.getPuzzleEditor().promptPuzzle();
- if (items == null) {
- // The attempt to prompt a puzzle ended gracefully (cancel)
- return;
- }
- String fileName = (String) items[0];
- File puzzleFile = (File) items[1];
- legupUI.getPuzzleEditor().loadPuzzle(fileName, puzzleFile);
- }
- };
-
- public HomePanel(FileDialog fileDialog, JFrame frame, LegupUI legupUI) {
+ /**
+ * Constructs a {@code HomePanel} with the specified {@code JFrame} and {@code LegupUI}.
+ *
+ * @param frame the main application frame
+ * @param legupUI the LEGUP user interface
+ */
+ public HomePanel(JFrame frame, LegupUI legupUI) {
this.legupUI = legupUI;
this.frame = frame;
setLayout(new GridLayout(1, 2));
+ setPreferredSize(new Dimension(440, 250));
initText();
initButtons();
}
+ /**
+ * Creates and returns the menu bar for this panel
+ *
+ * @return the menu bar
+ */
public JMenuBar getMenuBar() {
this.menuBar = new JMenuBar();
JMenu settings = new JMenu("Settings");
@@ -115,6 +105,14 @@ public void makeVisible() {
frame.setJMenuBar(this.getMenuBar());
}
+ /**
+ * Resizes the provided icon to the specified width and height
+ *
+ * @param icon the icon to resize
+ * @param width the target width
+ * @param height the target height
+ * @return the resized icon
+ */
private static ImageIcon resizeButtonIcon(ImageIcon icon, int width, int height) {
Image image = icon.getImage();
Image resizedImage = image.getScaledInstance(width, height, Image.SCALE_SMOOTH);
@@ -123,10 +121,10 @@ private static ImageIcon resizeButtonIcon(ImageIcon icon, int width, int height)
/** Initializes the buttons for this panel */
private void initButtons() {
- this.buttons = new JButton[4];
+ this.buttons = new JButton[3];
this.buttons[0] =
- new JButton("Solve Puzzle") {
+ new JButton("Puzzle Solver") {
{
setSize(buttonSize, buttonSize);
setMaximumSize(getSize());
@@ -144,7 +142,7 @@ private void initButtons() {
this.buttons[0].addActionListener(CursorController.createListener(this, openProofListener));
this.buttons[1] =
- new JButton("Create Puzzle") {
+ new JButton("Puzzle Editor") {
{
setSize(buttonSize, buttonSize);
setMaximumSize(getSize());
@@ -158,35 +156,17 @@ private void initButtons() {
this.buttons[1].setIcon(resizeButtonIcon(button1Icon, this.buttonSize, this.buttonSize));
this.buttons[1].setHorizontalTextPosition(AbstractButton.CENTER);
this.buttons[1].setVerticalTextPosition(AbstractButton.BOTTOM);
- this.buttons[1].addActionListener(l -> this.openNewPuzzleDialog());
-
- this.buttons[2] =
- new JButton("Edit Puzzle") {
- {
- setSize(buttonSize, buttonSize);
- setMaximumSize(getSize());
- }
- };
- URL button2IconLocation =
- ClassLoader.getSystemClassLoader()
- .getResource("edu/rpi/legup/images/Legup/homepanel/puzzle_file.png");
- ImageIcon button2Icon = new ImageIcon(button2IconLocation);
- this.buttons[2].setFocusPainted(false);
- this.buttons[2].setIcon(resizeButtonIcon(button2Icon, this.buttonSize, this.buttonSize));
- this.buttons[2].setHorizontalTextPosition(AbstractButton.CENTER);
- this.buttons[2].setVerticalTextPosition(AbstractButton.BOTTOM);
- this.buttons[2].addActionListener(
- CursorController.createListener(this, openPuzzleListener)); // PLACEHOLDER
+ this.buttons[1].addActionListener(l -> this.openPuzzleEditorDialog());
for (int i = 0; i < this.buttons.length - 1; i++) { // -1 to avoid the batch grader button
this.buttons[i].setBounds(200, 200, 700, 700);
}
- this.buttons[3] = new JButton("Batch Grader");
- this.buttons[3].setFocusPainted(false);
- this.buttons[3].setHorizontalTextPosition(AbstractButton.CENTER);
- this.buttons[3].setVerticalTextPosition(AbstractButton.BOTTOM);
+ this.buttons[2] = new JButton("Batch Grader");
+ this.buttons[2].setFocusPainted(false);
+ this.buttons[2].setHorizontalTextPosition(AbstractButton.CENTER);
+ this.buttons[2].setVerticalTextPosition(AbstractButton.BOTTOM);
- this.buttons[3].addActionListener(
+ this.buttons[2].addActionListener(
new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
@@ -284,7 +264,7 @@ public void checkFolder() {
}
} catch (IOException ex) {
LOGGER.error(ex.getMessage());
- this.buttons[3].addActionListener((ActionEvent e) -> use_xml_to_check());
+ this.buttons[2].addActionListener((ActionEvent e) -> use_xml_to_check());
}
}
@@ -523,7 +503,7 @@ private void render() {
this.removeAll();
this.setLayout(new BoxLayout(this, BoxLayout.PAGE_AXIS));
- this.legupUI.setTitle("LEGUP: A Better Way to Learn Formal Logic");
+ this.legupUI.setTitle("LEGUP: A Better Way To Learn Formal Logic");
JPanel buttons = new JPanel();
buttons.add(Box.createRigidArea(new Dimension(5, 0)));
@@ -531,11 +511,8 @@ private void render() {
buttons.add(Box.createRigidArea(new Dimension(5, 0)));
buttons.add(this.buttons[1]);
buttons.add(Box.createRigidArea(new Dimension(5, 0)));
- buttons.add(this.buttons[2]);
- buttons.add(Box.createRigidArea(new Dimension(5, 0)));
-
JPanel batchGraderButton = new JPanel();
- batchGraderButton.add(this.buttons[3]);
+ batchGraderButton.add(this.buttons[2]);
batchGraderButton.setAlignmentX(Component.CENTER_ALIGNMENT);
this.add(Box.createRigidArea(new Dimension(0, 5)));
@@ -566,6 +543,10 @@ private void openPuzzleEditorDialog() {
}
}
+ /**
+ * Opens a dialog to select a directory, recursively processes the directory to grade puzzles,
+ * and generates a CSV report of the grading results.
+ */
private void checkProofAll() {
/*
* Select dir to grade; recursively grade sub-dirs using traverseDir()
@@ -609,6 +590,14 @@ private void checkProofAll() {
JOptionPane.showMessageDialog(null, "Batch grading complete.");
}
+ /**
+ * Recursively traverses directories to grade puzzles and writes results to a CSV file
+ *
+ * @param folder the folder to traverse
+ * @param writer the BufferedWriter to write results to the CSV file
+ * @param path the current path within the directory structure
+ * @throws IOException if an I/O error occurs while writing to the CSV file
+ */
private void traverseDir(File folder, BufferedWriter writer, String path) throws IOException {
// Recursively traverse directory
GameBoardFacade facade = GameBoardFacade.getInstance();
@@ -661,6 +650,14 @@ private void traverseDir(File folder, BufferedWriter writer, String path) throws
}
}
+ /**
+ * Opens the puzzle editor for the specified puzzle with the specified dimensions
+ *
+ * @param game the name of the game
+ * @param rows the number of rows in the puzzle
+ * @param columns the number of columns in the puzzle
+ * @throws IllegalArgumentException if the dimensions are invalid
+ */
public void openEditorWithNewPuzzle(String game, int rows, int columns)
throws IllegalArgumentException {
if (game.isEmpty()) {
@@ -692,7 +689,7 @@ public void openEditorWithNewPuzzle(String game, int rows, int columns)
}
/**
- * Opens the puzzle editor for the specified game with the given statements
+ * Opens the puzzle editor for the specified puzzle with the given statements
*
* @param game a String containing the name of the game
* @param statements an array of statements
diff --git a/src/main/java/edu/rpi/legup/ui/LegupUI.java b/src/main/java/edu/rpi/legup/ui/LegupUI.java
index 201a2d4c7..75f822a6c 100644
--- a/src/main/java/edu/rpi/legup/ui/LegupUI.java
+++ b/src/main/java/edu/rpi/legup/ui/LegupUI.java
@@ -41,7 +41,7 @@ public static String getOS() {
return os;
}
- /** LegupUI Constructor - creates a new LegupUI to setup the menu and toolbar */
+ /** LegupUI Constructor - creates a new LegupUI to set up the menu and toolbar */
public LegupUI() {
setTitle("LEGUP");
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
@@ -103,11 +103,17 @@ private void initPanels() {
add(window);
panels = new LegupPanel[3];
- panels[0] = new HomePanel(this.fileDialog, this, this);
+ panels[0] = new HomePanel(this, this);
panels[1] = new ProofEditorPanel(this.fileDialog, this, this);
panels[2] = new PuzzleEditorPanel(this.fileDialog, this, this);
}
+ /**
+ * Displays the specified panel
+ *
+ * @param option the index of the panel to display
+ * @throws InvalidParameterException if the option is out of range
+ */
protected void displayPanel(int option) {
if (option > panels.length || option < 0) {
throw new InvalidParameterException("Invalid option");
@@ -121,10 +127,20 @@ protected void displayPanel(int option) {
repaint();
}
+ /**
+ * Gets the ProofEditorPanel instance
+ *
+ * @return the ProofEditorPanel
+ */
public ProofEditorPanel getProofEditor() {
return (ProofEditorPanel) panels[1];
}
+ /**
+ * Gets the PuzzleEditorPanel instance
+ *
+ * @return the PuzzleEditorPanel
+ */
public PuzzleEditorPanel getPuzzleEditor() {
return (PuzzleEditorPanel) panels[2];
}
@@ -134,17 +150,6 @@ public void repaintTree() {
getProofEditor().repaintTree();
}
- private void directions() {
- JOptionPane.showMessageDialog(
- null,
- "For every move you make, you must provide a rules for it (located in the Rules"
- + " panel).\n"
- + "While working on the edu.rpi.legup.puzzle, you may click on the \"Check\""
- + " button to test your proof for correctness.",
- "Directions",
- JOptionPane.PLAIN_MESSAGE);
- }
-
public void showStatus(String status, boolean error) {
showStatus(status, error, 1);
}
@@ -157,8 +162,13 @@ public void showStatus(String status, boolean error, int timer) {
// TODO: implement
}
- // ask to edu.rpi.legup.save current proof
- public boolean noquit(String instr) {
+ /**
+ * Prompts the user to confirm if they want to exit LEGUP
+ *
+ * @param instr the prompt message
+ * @return true if the user chooses not to quit, false otherwise
+ */
+ public boolean exit(String instr) {
int n = JOptionPane.showConfirmDialog(null, instr, "Confirm", JOptionPane.YES_NO_OPTION);
return n != JOptionPane.YES_OPTION;
}
@@ -168,7 +178,7 @@ public void windowOpened(WindowEvent e) {}
public void windowClosing(WindowEvent e) {
if (GameBoardFacade.getInstance().getHistory().getIndex() > -1) {
- if (noquit("Exiting LEGUP?")) {
+ if (exit("Exiting LEGUP?")) {
this.setDefaultCloseOperation(DO_NOTHING_ON_CLOSE);
} else {
this.setDefaultCloseOperation(EXIT_ON_CLOSE);
@@ -190,22 +200,47 @@ public void windowActivated(WindowEvent e) {}
public void windowDeactivated(WindowEvent e) {}
+ /**
+ * Gets the BoardView instance from the proof editor
+ *
+ * @return the BoardView
+ */
public BoardView getBoardView() {
return getProofEditor().getBoardView();
}
+ /**
+ * Gets the BoardView instance from the puzzle editor
+ *
+ * @return the BoardView
+ */
public BoardView getEditorBoardView() {
return getPuzzleEditor().getBoardView();
}
+ /**
+ * Gets the DynamicView instance from the proof editor
+ *
+ * @return the DynamicView
+ */
public DynamicView getDynamicBoardView() {
return getProofEditor().getDynamicBoardView();
}
+ /**
+ * Gets the DynamicView instance from the puzzle editor.
+ *
+ * @return the DynamicView
+ */
public DynamicView getEditorDynamicBoardView() {
return getPuzzleEditor().getDynamicBoardView();
}
+ /**
+ * Gets the TreePanel instance from the proof editor
+ *
+ * @return the TreePanel
+ */
public TreePanel getTreePanel() {
return getProofEditor().getTreePanel();
}
diff --git a/src/main/java/edu/rpi/legup/ui/PickGameDialog.java b/src/main/java/edu/rpi/legup/ui/PickGameDialog.java
index ed510c6b6..3b66931a7 100644
--- a/src/main/java/edu/rpi/legup/ui/PickGameDialog.java
+++ b/src/main/java/edu/rpi/legup/ui/PickGameDialog.java
@@ -136,10 +136,20 @@ public void initPuzzles() {
gameBox = new JComboBox(games);
}
+ /**
+ * Gets the selected puzzle file path
+ *
+ * @return the puzzle file path as a String
+ */
public String getPuzzle() {
return puzzleBox.getText();
}
+ /**
+ * Returns the selected puzzle
+ *
+ * @return the selected puzzle as a String
+ */
public String getGame() {
return (String) gameBox.getSelectedItem();
}
diff --git a/src/main/java/edu/rpi/legup/ui/PreferencesDialog.java b/src/main/java/edu/rpi/legup/ui/PreferencesDialog.java
index e9a0df4a2..e90d06640 100644
--- a/src/main/java/edu/rpi/legup/ui/PreferencesDialog.java
+++ b/src/main/java/edu/rpi/legup/ui/PreferencesDialog.java
@@ -53,12 +53,24 @@ public class PreferencesDialog extends JDialog {
}
}
+ /**
+ * Creates a new instance of PreferencesDialog for the proof editor
+ *
+ * @param frame the parent frame
+ * @param rules the RuleFrame associated with the proof editor
+ * @return a new instance of PreferencesDialog
+ */
public static PreferencesDialog CreateDialogForProofEditor(Frame frame, RuleFrame rules) {
PreferencesDialog p = new PreferencesDialog(frame);
p.rulesFrame = rules;
return p;
}
+ /**
+ * Constructs a PreferencesDialog
+ *
+ * @param frame the parent frame
+ */
public PreferencesDialog(Frame frame) {
super(frame);
@@ -108,6 +120,11 @@ public PreferencesDialog(Frame frame) {
setVisible(true);
}
+ /**
+ * Toggles between dark mode and light mode based on the given preferences
+ *
+ * @param prefs the LegupPreferences instance holding user preferences
+ */
private void toggleDarkMode(LegupPreferences prefs) {
try {
if (Boolean.valueOf(prefs.getUserPref(LegupPreferences.DARK_MODE))) {
@@ -121,6 +138,11 @@ private void toggleDarkMode(LegupPreferences prefs) {
}
}
+ /**
+ * Creates the general preferences tab
+ *
+ * @return a JScrollPane containing the general preferences panel
+ */
private JScrollPane createGeneralTab() {
LegupPreferences prefs = LegupPreferences.getInstance();
JScrollPane scrollPane = new JScrollPane();
diff --git a/src/main/java/edu/rpi/legup/ui/ProofEditorPanel.java b/src/main/java/edu/rpi/legup/ui/ProofEditorPanel.java
index ddabb5705..5ecbd5564 100644
--- a/src/main/java/edu/rpi/legup/ui/ProofEditorPanel.java
+++ b/src/main/java/edu/rpi/legup/ui/ProofEditorPanel.java
@@ -53,8 +53,8 @@ public class ProofEditorPanel extends LegupPanel implements IHistoryListener {
private DynamicView dynamicBoardView;
private JSplitPane topHalfPanel, mainPanel;
private TitledBorder boardBorder;
-
- private JButton[] toolBarButtons;
+ private JButton[] toolBar1Buttons;
+ private JButton[] toolBar2Buttons;
private JMenu file;
private JMenuItem newPuzzle,
resetPuzzle,
@@ -74,7 +74,8 @@ public class ProofEditorPanel extends LegupPanel implements IHistoryListener {
private JMenu about, help;
private JMenuItem helpLegup, aboutLegup;
- private JToolBar toolBar;
+ private JToolBar toolBar1;
+ private JToolBar toolBar2;
private BoardView boardView;
private JFileChooser folderBrowser;
@@ -88,7 +89,6 @@ public class ProofEditorPanel extends LegupPanel implements IHistoryListener {
public static final int IMD_FEEDBACK = 32;
public static final int INTERN_RO = 64;
public static final int AUTO_JUST = 128;
- static final int[] TOOLBAR_SEPARATOR_BEFORE = {2, 4, 8};
private static final String[] PROFILES = {
"No Assistance",
"Rigorous Proof",
@@ -118,6 +118,13 @@ public class ProofEditorPanel extends LegupPanel implements IHistoryListener {
protected JMenuItem testAI = new JMenuItem("Test AI!");
protected JMenuItem hintAI = new JMenuItem("Hint");
+ /**
+ * Constructs a new {@code ProofEditorPanel} with the specified parameters
+ *
+ * @param fileDialog the {@code FileDialog} used for file operations
+ * @param frame the {@code JFrame} that contains this panel
+ * @param legupUI the {@code LegupUI} instance managing the user interface
+ */
public ProofEditorPanel(FileDialog fileDialog, JFrame frame, LegupUI legupUI) {
this.fileDialog = fileDialog;
this.frame = frame;
@@ -134,7 +141,7 @@ public ProofEditorPanel(FileDialog fileDialog, JFrame frame, LegupUI legupUI) {
public void makeVisible() {
this.removeAll();
- setupToolBar();
+ setupToolBar1();
setupContent();
frame.setJMenuBar(getMenuBar());
}
@@ -165,7 +172,7 @@ public JMenuBar getMenuBar() {
file = new JMenu("File");
newPuzzle = new JMenuItem("Open");
- resetPuzzle = new JMenuItem("Reset Puzzle");
+ resetPuzzle = new JMenuItem("Reset");
// genPuzzle = new JMenuItem("Puzzle Generators"); // TODO: implement puzzle
// generator
saveProofAs = new JMenuItem("Save As"); // create a new file to save
@@ -341,6 +348,7 @@ public JMenuBar getMenuBar() {
} else {
resetPuzzle.setAccelerator(KeyStroke.getKeyStroke('R', InputEvent.CTRL_DOWN_MASK));
}
+
file.addSeparator();
file.add(saveProofAs);
@@ -547,6 +555,7 @@ public Object[] promptPuzzle() {
return null;
}
+ System.out.println(preferences.getSavedPath());
return new Object[] {fileName, puzzleFile};
}
@@ -712,16 +721,14 @@ private void helpTutorial() {
default:
url = "https://github.com/Bram-Hub/Legup/wiki/LEGUP-Tutorial";
}
- Runtime rt = Runtime.getRuntime();
try {
- // rt.exec("rundll32 url.dll,FileProtocolHandler "+url);
java.awt.Desktop.getDesktop().browse(java.net.URI.create(url));
} catch (IOException e) {
e.printStackTrace();
}
}
- // add the new function need to implement
+ // unfinished
public void add_drop() {
// add the mouse event then we can use the new listener to implement and
// we should create a need jbuttom for it to ship the rule we select.
@@ -765,13 +772,22 @@ private void saveProofChange() {
}
}
- // ask to edu.rpi.legup.save current proof
+ /**
+ * Displays a confirmation dialog with a specified message. Returns {@code true} if the user
+ * selects "No" or cancels the action, and {@code false} if the user selects "Yes".
+ *
+ * @param instr the message to display in the confirmation dialog
+ * @return {@code true} if the user chooses not to quit, {@code false} otherwise
+ */
public boolean noquit(String instr) {
int n = JOptionPane.showConfirmDialog(null, instr, "Confirm", JOptionPane.YES_NO_OPTION);
return n != JOptionPane.YES_OPTION;
}
- /** Sets the main content for the edu.rpi.legup.user interface */
+ /**
+ * Configures the layout and components for the main user interface. This includes setting up
+ * panels, split panes, and borders, and adding them to the main content pane.
+ */
protected void setupContent() {
// JPanel consoleBox = new JPanel(new BorderLayout());
JPanel treeBox = new JPanel(new BorderLayout());
@@ -804,16 +820,9 @@ protected void setupContent() {
ruleBox.add(boardPanel);
treeBox.add(ruleBox);
this.add(treeBox);
- // consoleBox.add(treeBox);
- //
- // getContentPane().add(consoleBox);
-
- // JPopupPanel popupPanel = new JPopupPanel();
- // setGlassPane(popupPanel);
- // popupPanel.setVisible(true);
mainPanel.setDividerLocation(mainPanel.getMaximumDividerLocation() + 100);
- // frame.pack();
+
revalidate();
}
@@ -828,68 +837,29 @@ private void setupToolBar1() {
toolBar1.setRollover(true);
setToolBar2Buttons(new JButton[1]);
- // Scale the image icons down to make the buttons smaller
- ImageIcon imageIcon = new ImageIcon(resourceLocation);
- Image image = imageIcon.getImage();
- imageIcon =
- new ImageIcon(
- image.getScaledInstance(
- this.TOOLBAR_ICON_SCALE,
- this.TOOLBAR_ICON_SCALE,
- Image.SCALE_SMOOTH));
-
- JButton button = new JButton(toolBarName, imageIcon);
- button.setFocusPainted(false);
- getToolBarButtons()[i] = button;
- }
+ URL open_url =
+ ClassLoader.getSystemClassLoader()
+ .getResource("edu/rpi/legup/images/Legup/Open.png");
- toolBar = new JToolBar();
- toolBar.setFloatable(false);
- toolBar.setRollover(true);
+ // Scale the image icons down to make the buttons smaller
+ ImageIcon OpenImageIcon = new ImageIcon(open_url);
+ Image OpenImage = OpenImageIcon.getImage();
+ OpenImageIcon =
+ new ImageIcon(
+ OpenImage.getScaledInstance(
+ this.TOOLBAR_ICON_SCALE,
+ this.TOOLBAR_ICON_SCALE,
+ Image.SCALE_SMOOTH));
- for (int i = 0; i < getToolBarButtons().length; i++) {
- for (int s = 0; s < TOOLBAR_SEPARATOR_BEFORE.length; s++) {
- if (i == TOOLBAR_SEPARATOR_BEFORE[s]) {
- toolBar.addSeparator();
- }
- }
- String toolBarName = ToolbarName.values()[i].toString();
+ JButton open = new JButton("Open", OpenImageIcon);
+ open.setFocusPainted(false);
- toolBar.add(getToolBarButtons()[i]);
- getToolBarButtons()[i].setToolTipText(toolBarName);
+ open.addActionListener((ActionEvent) -> loadPuzzle());
- getToolBarButtons()[i].setVerticalTextPosition(SwingConstants.BOTTOM);
- getToolBarButtons()[i].setHorizontalTextPosition(SwingConstants.CENTER);
- }
+ getToolBar2Buttons()[0] = open;
+ toolBar1.add(getToolBar2Buttons()[0]);
- // toolBarButtons[ToolbarName.OPEN_PUZZLE.ordinal()].addActionListener((ActionEvent
- // e) ->
- // promptPuzzle());
- // toolBarButtons[ToolbarName.SAVE.ordinal()].addActionListener((ActionEvent e) ->
- // saveProof());
- // toolBarButtons[ToolbarName.UNDO.ordinal()].addActionListener((ActionEvent e) ->
- // GameBoardFacade.getInstance().getHistory().undo());
- // toolBarButtons[ToolbarName.REDO.ordinal()].addActionListener((ActionEvent e) ->
- // GameBoardFacade.getInstance().getHistory().redo());
- toolBarButtons[ToolbarName.HINT.ordinal()].addActionListener((ActionEvent e) -> {});
- toolBarButtons[ToolbarName.CHECK.ordinal()].addActionListener(
- (ActionEvent e) -> checkProof());
- toolBarButtons[ToolbarName.SUBMIT.ordinal()].addActionListener((ActionEvent e) -> {});
- toolBarButtons[ToolbarName.DIRECTIONS.ordinal()].addActionListener((ActionEvent e) -> {});
-
- toolBarButtons[ToolbarName.CHECK_ALL.ordinal()].addActionListener(
- (ActionEvent e) -> checkProofAll());
-
- // toolBarButtons[ToolbarName.SAVE.ordinal()].setEnabled(false);
- // toolBarButtons[ToolbarName.UNDO.ordinal()].setEnabled(false);
- // toolBarButtons[ToolbarName.REDO.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.HINT.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.CHECK.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.SUBMIT.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.DIRECTIONS.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.CHECK_ALL.ordinal()].setEnabled(true);
-
- this.add(toolBar, BorderLayout.NORTH);
+ this.add(toolBar1, BorderLayout.NORTH);
}
/**
@@ -999,19 +969,19 @@ private void setupToolBar2() {
/**
* Sets the toolbar1 buttons
*
- * @param toolBarButtons toolbar buttons
+ * @param toolBar1Buttons toolbar buttons
*/
- public void setToolBarButtons(JButton[] toolBarButtons) {
- this.toolBarButtons = toolBarButtons;
+ public void setToolBar1Buttons(JButton[] toolBar1Buttons) {
+ this.toolBar1Buttons = toolBar1Buttons;
}
/**
- * Gets the toolbar buttons
+ * Sets the toolbar2 buttons
*
- * @return toolbar buttons
+ * @param toolBar2Buttons toolbar buttons
*/
- public JButton[] getToolBarButtons() {
- return toolBarButtons;
+ public void setToolBar2Buttons(JButton[] toolBar2Buttons) {
+ this.toolBar2Buttons = toolBar2Buttons;
}
/**
@@ -1112,6 +1082,12 @@ private void repaintAll() {
treePanel.repaint();
}
+ /**
+ * Initializes the dynamic board view, updates the tree panel, and sets rules and search panels
+ * based on the provided puzzle. It also updates toolbars and reloads the GUI.
+ *
+ * @param puzzle the puzzle to be displayed
+ */
public void setPuzzleView(Puzzle puzzle) {
this.boardView = puzzle.getBoardView();
@@ -1135,9 +1111,8 @@ public void setPuzzleView(Puzzle puzzle) {
ruleFrame.getContradictionPanel().setRules(puzzle.getContradictionRules());
ruleFrame.getSearchPanel().setSearchBar(puzzle);
- toolBarButtons[ToolbarName.CHECK.ordinal()].setEnabled(true);
- // toolBarButtons[ToolbarName.SAVE.ordinal()].setEnabled(true);
-
+ toolBar1.setVisible(false);
+ setupToolBar2();
reloadGui();
}
@@ -1203,6 +1178,15 @@ private boolean basicCheckProof(int[][] origCells) {
return false;
}
+ /**
+ * Traverses a given directory, grades the proofs found in the directory, and writes the results
+ * to the specified CSV writer.
+ *
+ * @param folder the folder to traverse
+ * @param writer the CSV writer
+ * @param path the current path in the directory traversal
+ * @throws IOException if an error occurs while writing to the CSV file
+ */
private void traverseDir(File folder, BufferedWriter writer, String path) throws IOException {
// Recursively traverse directory
GameBoardFacade facade = GameBoardFacade.getInstance();
@@ -1257,6 +1241,11 @@ private void traverseDir(File folder, BufferedWriter writer, String path) throws
}
}
+ /**
+ * Returns the current board view.
+ *
+ * @return the current {@link BoardView}
+ */
public BoardView getBoardView() {
return boardView;
}
@@ -1270,12 +1259,17 @@ public DynamicView getDynamicBoardView() {
return dynamicBoardView;
}
+ /**
+ * Returns the current tree panel.
+ *
+ * @return the current {@link TreePanel}
+ */
public TreePanel getTreePanel() {
return treePanel;
}
/**
- * Called when a action is pushed onto the edu.rpi.legup.history stack
+ * Called when an action is pushed onto the edu.rpi.legup.history stack
*
* @param command action to push onto the stack
*/
@@ -1283,9 +1277,7 @@ public TreePanel getTreePanel() {
public void onPushChange(ICommand command) {
LOGGER.info("Pushing " + command.getClass().getSimpleName() + " to stack.");
undo.setEnabled(true);
- // toolBarButtons[ToolbarName.UNDO.ordinal()].setEnabled(true);
redo.setEnabled(false);
- // toolBarButtons[ToolbarName.REDO.ordinal()].setEnabled(false);
String puzzleName = GameBoardFacade.getInstance().getPuzzleModule().getName();
File puzzleFile = new File(GameBoardFacade.getInstance().getCurFileName());
@@ -1308,9 +1300,7 @@ public void onClearHistory() {}
@Override
public void onRedo(boolean isBottom, boolean isTop) {
undo.setEnabled(!isBottom);
- // toolBarButtons[ToolbarName.UNDO.ordinal()].setEnabled(!isBottom);
redo.setEnabled(!isTop);
- // toolBarButtons[ToolbarName.REDO.ordinal()].setEnabled(!isTop);
if (isBottom) {
String puzzleName = GameBoardFacade.getInstance().getPuzzleModule().getName();
File puzzleFile = new File(GameBoardFacade.getInstance().getCurFileName());
@@ -1331,9 +1321,7 @@ public void onRedo(boolean isBottom, boolean isTop) {
@Override
public void onUndo(boolean isBottom, boolean isTop) {
undo.setEnabled(!isBottom);
- // toolBarButtons[ToolbarName.UNDO.ordinal()].setEnabled(!isBottom);
redo.setEnabled(!isTop);
- // toolBarButtons[ToolbarName.REDO.ordinal()].setEnabled(!isTop);
String puzzleName = GameBoardFacade.getInstance().getPuzzleModule().getName();
File puzzleFile = new File(GameBoardFacade.getInstance().getCurFileName());
if (isBottom) {
diff --git a/src/main/java/edu/rpi/legup/ui/PuzzleEditorPanel.java b/src/main/java/edu/rpi/legup/ui/PuzzleEditorPanel.java
index 6b65e72f7..7c2ba06ff 100644
--- a/src/main/java/edu/rpi/legup/ui/PuzzleEditorPanel.java
+++ b/src/main/java/edu/rpi/legup/ui/PuzzleEditorPanel.java
@@ -39,12 +39,14 @@ public class PuzzleEditorPanel extends LegupPanel implements IHistoryListener {
private JMenu[] menus;
private JMenuItem helpLegup, aboutLegup;
private JMenuBar menuBar;
- private JToolBar toolBar;
+ private JToolBar toolBar1;
+ private JToolBar toolBar2;
private JFileChooser folderBrowser;
private JFrame frame;
private JButton[] buttons;
JSplitPane splitPanel;
- private JButton[] toolBarButtons;
+ private JButton[] toolBar1Buttons;
+ private JButton[] toolBar2Buttons;
private JPanel elementPanel;
private DynamicView dynamicBoardView;
private BoardView boardView;
@@ -56,7 +58,11 @@ public class PuzzleEditorPanel extends LegupPanel implements IHistoryListener {
private JPanel treePanel;
private LegupUI legupUI;
private EditorElementController editorElementController;
- static final int[] TOOLBAR_SEPARATOR_BEFORE = {2, 4, 8};
+ private CreatePuzzleDialog cpd;
+ private HomePanel hp;
+ private boolean existingPuzzle;
+ private String fileName;
+ private File puzzleFile;
/**
* Constructs a {@code PuzzleEditorPanel} with the specified file dialog, frame, and Legup UI
@@ -124,14 +130,14 @@ public void setMenuBar() {
menus[0] = new JMenu("File");
// file>new
- JMenuItem newPuzzle = new JMenuItem("New");
- newPuzzle.addActionListener((ActionEvent) -> loadPuzzle());
+ JMenuItem openPuzzle = new JMenuItem("Open");
+ openPuzzle.addActionListener((ActionEvent) -> loadPuzzle());
if (os.equals("mac")) {
- newPuzzle.setAccelerator(
+ openPuzzle.setAccelerator(
KeyStroke.getKeyStroke(
- 'N', Toolkit.getDefaultToolkit().getMenuShortcutKeyMask()));
+ 'O', Toolkit.getDefaultToolkit().getMenuShortcutKeyMask()));
} else {
- newPuzzle.setAccelerator(KeyStroke.getKeyStroke('N', InputEvent.CTRL_DOWN_MASK));
+ openPuzzle.setAccelerator(KeyStroke.getKeyStroke('O', InputEvent.CTRL_DOWN_MASK));
}
// file>create
JMenuItem createPuzzle = new JMenuItem("Create");
@@ -144,11 +150,11 @@ public void setMenuBar() {
existingPuzzle = false;
});
if (os.equals("mac")) {
- newPuzzle.setAccelerator(
+ createPuzzle.setAccelerator(
KeyStroke.getKeyStroke(
- 'D', Toolkit.getDefaultToolkit().getMenuShortcutKeyMask()));
+ 'C', Toolkit.getDefaultToolkit().getMenuShortcutKeyMask()));
} else {
- newPuzzle.setAccelerator(KeyStroke.getKeyStroke('D', InputEvent.CTRL_DOWN_MASK));
+ createPuzzle.setAccelerator(KeyStroke.getKeyStroke('C', InputEvent.CTRL_DOWN_MASK));
}
JMenuItem exit = new JMenuItem("Exit");
@@ -279,8 +285,7 @@ public void exitEditor() {
@Override
public void makeVisible() {
this.removeAll();
-
- setupToolBar();
+ setupToolBar1();
setupContent();
setMenuBar();
}
@@ -292,21 +297,21 @@ public void makeVisible() {
private void setupToolBar1() {
setToolBar1Buttons(new JButton[2]);
- // Scale the image icons down to make the buttons smaller
- ImageIcon imageIcon = new ImageIcon(resourceLocation);
- Image image = imageIcon.getImage();
- imageIcon =
- new ImageIcon(
- image.getScaledInstance(
- this.TOOLBAR_ICON_SCALE,
- this.TOOLBAR_ICON_SCALE,
- Image.SCALE_SMOOTH));
-
- JButton button = new JButton(toolBarName, imageIcon);
- button.setFocusPainted(false);
- getToolBarButtons()[i] = button;
- lastone = i;
- }
+ URL open_url =
+ ClassLoader.getSystemClassLoader()
+ .getResource("edu/rpi/legup/images/Legup/Open.png");
+ ImageIcon OpenImageIcon = new ImageIcon(open_url);
+ Image OpenImage = OpenImageIcon.getImage();
+ OpenImageIcon =
+ new ImageIcon(
+ OpenImage.getScaledInstance(
+ this.TOOLBAR_ICON_SCALE,
+ this.TOOLBAR_ICON_SCALE,
+ Image.SCALE_SMOOTH));
+
+ JButton open = new JButton("Open", OpenImageIcon);
+ open.setFocusPainted(false);
+ open.addActionListener((ActionEvent) -> loadPuzzle());
getToolBar1Buttons()[0] = open;
@@ -414,81 +419,51 @@ private void setupToolBar2() {
URL save_and_solve =
ClassLoader.getSystemClassLoader()
.getResource("edu/rpi/legup/images/Legup/Check.png");
- ImageIcon imageIcon = new ImageIcon(check_and_save);
- Image image = imageIcon.getImage();
- imageIcon =
+ ImageIcon SaveSolveImageIcon = new ImageIcon(save_and_solve);
+ Image SaveSolveImage = SaveSolveImageIcon.getImage();
+ SaveSolveImageIcon =
new ImageIcon(
- image.getScaledInstance(
+ SaveSolveImage.getScaledInstance(
this.TOOLBAR_ICON_SCALE,
this.TOOLBAR_ICON_SCALE,
Image.SCALE_SMOOTH));
- JButton checkandsave = new JButton("check and Save", imageIcon);
- checkandsave.setFocusPainted(false);
- checkandsave.addActionListener(
+ JButton saveandsolve = new JButton("Save & Solve", SaveSolveImageIcon);
+ saveandsolve.setFocusPainted(false);
+ saveandsolve.addActionListener(
new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
- // savePuzzle();
- String filename = savePuzzle();
- File puzzlename = new File(filename);
- System.out.println(filename);
-
- GameBoardFacade.getInstance().getLegupUI().displayPanel(1);
- GameBoardFacade.getInstance()
- .getLegupUI()
- .getProofEditor()
- .loadPuzzle(filename, new File(filename));
- String puzzleName =
- GameBoardFacade.getInstance().getPuzzleModule().getName();
- frame.setTitle(puzzleName + " - " + puzzlename.getName());
+ if (GameBoardFacade.getInstance().getPuzzleModule() != null) {
+ String filename = savePuzzle();
+ File puzzlename = new File(filename);
+ System.out.println(filename);
+
+ GameBoardFacade.getInstance().getLegupUI().displayPanel(1);
+ GameBoardFacade.getInstance()
+ .getLegupUI()
+ .getProofEditor()
+ .loadPuzzle(filename, new File(filename));
+ String puzzleName =
+ GameBoardFacade.getInstance().getPuzzleModule().getName();
+ frame.setTitle(puzzleName + " - " + puzzlename.getName());
+ }
}
});
- getToolBarButtons()[lastone + 1] = checkandsave;
- System.out.println("it is create new file");
-
- toolBar = new JToolBar();
- toolBar.setFloatable(false);
- toolBar.setRollover(true);
-
- for (int i = 0; i < getToolBarButtons().length - 1; i++) {
- for (int s = 0; s < TOOLBAR_SEPARATOR_BEFORE.length; s++) {
- if (i == TOOLBAR_SEPARATOR_BEFORE[s]) {
- toolBar.addSeparator();
- }
- }
- String toolBarName = ToolbarName.values()[i].toString();
-
- toolBar.add(getToolBarButtons()[i]);
- getToolBarButtons()[i].setToolTipText(toolBarName);
-
- getToolBarButtons()[i].setVerticalTextPosition(SwingConstants.BOTTOM);
- getToolBarButtons()[i].setHorizontalTextPosition(SwingConstants.CENTER);
- }
+ getToolBar2Buttons()[2] = saveandsolve;
+ toolBar2.add(getToolBar2Buttons()[2]);
- // toolBarButtons[ToolbarName.OPEN_PUZZLE.ordinal()].addActionListener((ActionEvent
- // e) ->
- // promptPuzzle());
- // toolBarButtons[ToolbarName.SAVE.ordinal()].addActionListener((ActionEvent e) ->
- // saveProof());
- // toolBarButtons[ToolbarName.UNDO.ordinal()].addActionListener((ActionEvent e) ->
- // GameBoardFacade.getInstance().getHistory().undo());
- // toolBarButtons[ToolbarName.REDO.ordinal()].addActionListener((ActionEvent e) ->
- // GameBoardFacade.getInstance().getHistory().redo());
- toolBarButtons[ToolbarName.HINT.ordinal()].addActionListener((ActionEvent e) -> {});
- toolBarButtons[ToolbarName.SUBMIT.ordinal()].addActionListener((ActionEvent e) -> {});
- toolBarButtons[ToolbarName.DIRECTIONS.ordinal()].addActionListener((ActionEvent e) -> {});
-
- // toolBarButtons[ToolbarName.SAVE.ordinal()].setEnabled(false);
- // toolBarButtons[ToolbarName.UNDO.ordinal()].setEnabled(false);
- // toolBarButtons[ToolbarName.REDO.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.HINT.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.SUBMIT.ordinal()].setEnabled(false);
- toolBarButtons[ToolbarName.DIRECTIONS.ordinal()].setEnabled(false);
-
- this.add(toolBar, BorderLayout.NORTH);
+ this.add(toolBar2, BorderLayout.NORTH);
}
+ /**
+ * Initializes a puzzle based on the provided game name, rows, and columns.
+ *
+ * @param game the name of the game or puzzle to load
+ * @param rows the number of rows in the puzzle
+ * @param columns the number of columns in the puzzle
+ * @throws IllegalArgumentException if the provided arguments are invalid
+ */
public void loadPuzzleFromHome(String game, int rows, int columns)
throws IllegalArgumentException {
GameBoardFacade facade = GameBoardFacade.getInstance();
@@ -502,6 +477,13 @@ public void loadPuzzleFromHome(String game, int rows, int columns)
}
}
+ /**
+ * Initializes a puzzle based on the provided game name and an array of statements.
+ *
+ * @param game the name of the game or puzzle to load
+ * @param statements an array of statements to initialize the puzzle
+ * @throws IllegalArgumentException if the provided arguments are invalid
+ */
public void loadPuzzleFromHome(String game, String[] statements) {
GameBoardFacade facade = GameBoardFacade.getInstance();
try {
@@ -525,7 +507,7 @@ public void loadPuzzleFromHome(String game, String[] statements) {
public Object[] promptPuzzle() {
GameBoardFacade facade = GameBoardFacade.getInstance();
if (facade.getBoard() != null) {
- if (noQuit("Opening a new puzzle?")) {
+ if (noQuit("Open an existing puzzle?")) {
return new Object[0];
}
}
@@ -549,7 +531,6 @@ public Object[] promptPuzzle() {
fileBrowser.setAcceptAllFileFilterUsed(false);
File puzzlePath = fileBrowser.getSelectedFile();
- System.out.println(puzzlePath.getAbsolutePath());
if (puzzlePath != null) {
fileName = puzzlePath.getAbsolutePath();
@@ -594,6 +575,9 @@ public void loadPuzzle(String fileName, File puzzleFile) {
GameBoardFacade.getInstance().loadPuzzleEditor(fileName);
String puzzleName = GameBoardFacade.getInstance().getPuzzleModule().getName();
frame.setTitle(puzzleName + " - " + puzzleFile.getName());
+ existingPuzzle = true;
+ this.fileName = fileName;
+ this.puzzleFile = puzzleFile;
} catch (InvalidFileFormatException e) {
legupUI.displayPanel(0);
LOGGER.error(e.getMessage());
@@ -644,12 +628,22 @@ public BoardView getBoardView() {
return boardView;
}
- public JButton[] getToolBarButtons() {
- return toolBarButtons;
+ /**
+ * Returns the array of buttons for the first toolbar
+ *
+ * @return the array of toolbar1 buttons
+ */
+ public JButton[] getToolBar1Buttons() {
+ return toolBar1Buttons;
}
- public void setToolBarButtons(JButton[] toolBarButtons) {
- this.toolBarButtons = toolBarButtons;
+ /**
+ * Sets the array of buttons for the first toolbar
+ *
+ * @param toolBar1Buttons the array of toolbar1 buttons
+ */
+ public void setToolBar1Buttons(JButton[] toolBar1Buttons) {
+ this.toolBar1Buttons = toolBar1Buttons;
}
/**
@@ -696,13 +690,11 @@ public void setPuzzleView(Puzzle puzzle) {
dynamicBoardView.setBorder(titleBoard);
puzzle.addBoardListener(puzzle.getBoardView());
- System.out.println("Setting elements");
if (this.elementFrame != null) {
elementFrame.setElements(puzzle);
}
-
- toolBarButtons[ToolbarName.CHECK.ordinal()].setEnabled(true);
- // toolBarButtons[ToolbarName.SAVE.ordinal()].setEnabled(true);
+ toolBar1.setVisible(false);
+ setupToolBar2();
}
/** Saves a puzzle */
@@ -768,7 +760,7 @@ private String savePuzzle() {
folderBrowser.setAcceptAllFileFilterUsed(false);
String path = folderBrowser.getSelectedFile().getAbsolutePath();
-
+ preferences.setSavedPath(path);
if (path != null) {
try {
PuzzleExporter exporter = puzzle.getExporter();
@@ -783,6 +775,11 @@ private String savePuzzle() {
return path;
}
+ /**
+ * Returns the current dynamic board view
+ *
+ * @return the dynamic board view
+ */
public DynamicView getDynamicBoardView() {
return dynamicBoardView;
}
diff --git a/src/main/java/edu/rpi/legup/ui/ScrollView.java b/src/main/java/edu/rpi/legup/ui/ScrollView.java
index 4d53cd747..589573154 100644
--- a/src/main/java/edu/rpi/legup/ui/ScrollView.java
+++ b/src/main/java/edu/rpi/legup/ui/ScrollView.java
@@ -293,6 +293,11 @@ public void setSize(Dimension size) {
updateSize();
}
+ /**
+ * Gets the canvas for this {@code ScrollView}
+ *
+ * @return the ZoomablePane instance used as the canvas
+ */
public ZoomablePane getCanvas() {
return canvas;
}
diff --git a/src/main/java/edu/rpi/legup/ui/ToolbarName.java b/src/main/java/edu/rpi/legup/ui/ToolbarName.java
index 5cf9b5923..622d16d8d 100644
--- a/src/main/java/edu/rpi/legup/ui/ToolbarName.java
+++ b/src/main/java/edu/rpi/legup/ui/ToolbarName.java
@@ -5,11 +5,8 @@
* specific toolbar action.
*/
public enum ToolbarName {
- HINT,
- CHECK,
- SUBMIT,
DIRECTIONS,
- CHECK_ALL;
+ CHECK;
/**
* Gets the String representation of the ToolbarName enum
diff --git a/src/main/java/edu/rpi/legup/ui/ZoomWidget.java b/src/main/java/edu/rpi/legup/ui/ZoomWidget.java
index 3207462ed..40f36113f 100644
--- a/src/main/java/edu/rpi/legup/ui/ZoomWidget.java
+++ b/src/main/java/edu/rpi/legup/ui/ZoomWidget.java
@@ -10,6 +10,10 @@
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
+/**
+ * The {@code ZoomWidget} displays a zoom icon that, when clicked, shows a popup slider to adjust
+ * the zoom level of the associated {@code ScrollView}.
+ */
public class ZoomWidget extends JLabel {
private ScrollView parent;
private PopupSlider palette = new PopupSlider();
@@ -48,6 +52,11 @@ public PopupSlider() {
slider.addChangeListener(this);
}
+ /**
+ * Handles state changes in the slider by adjusting the zoom level of the {@code ScrollView}
+ *
+ * @param e the {@code ChangeEvent} indicating that the slider's state has changed
+ */
public void stateChanged(ChangeEvent e) {
if (slider.getValueIsAdjusting()) {
parent.zoomTo((double) slider.getValue() / 100.0);
diff --git a/src/main/java/edu/rpi/legup/ui/ZoomablePane.java b/src/main/java/edu/rpi/legup/ui/ZoomablePane.java
index 934d31c53..66af90abd 100644
--- a/src/main/java/edu/rpi/legup/ui/ZoomablePane.java
+++ b/src/main/java/edu/rpi/legup/ui/ZoomablePane.java
@@ -5,6 +5,10 @@
import java.awt.Graphics2D;
import javax.swing.*;
+/**
+ * The {@code ZoomablePane} class is used to display components in a zoomable and scalable manner.
+ * It uses {@code ScrollView} to handle scaling and drawing of the content.
+ */
public class ZoomablePane extends JLayeredPane {
private ScrollView viewer;
diff --git a/src/main/java/edu/rpi/legup/ui/boardview/BoardView.java b/src/main/java/edu/rpi/legup/ui/boardview/BoardView.java
index 0b395c774..18b47d98b 100644
--- a/src/main/java/edu/rpi/legup/ui/boardview/BoardView.java
+++ b/src/main/java/edu/rpi/legup/ui/boardview/BoardView.java
@@ -188,6 +188,11 @@ public ArrayList getElementViews() {
return elementViews;
}
+ /**
+ * Gets the ElementController associated with this board view.
+ *
+ * @return the ElementController
+ */
public ElementController getElementController() {
return elementController;
}
@@ -197,6 +202,11 @@ public void draw(Graphics2D graphics2D) {
drawBoard(graphics2D);
}
+ /**
+ * Draws the board and its elements.
+ *
+ * @param graphics2D the Graphics2D context used for drawing
+ */
public void drawBoard(Graphics2D graphics2D) {
for (ElementView element : elementViews) {
element.draw(graphics2D);
@@ -213,5 +223,10 @@ public void onBoardDataChanged(PuzzleElement puzzleElement) {
repaint();
}
+ /**
+ * Gets the selection popup menu for this board view.
+ *
+ * @return the DataSelectionView associated with this view
+ */
public abstract DataSelectionView getSelectionPopupMenu();
}
diff --git a/src/main/java/edu/rpi/legup/ui/boardview/DataSelectionView.java b/src/main/java/edu/rpi/legup/ui/boardview/DataSelectionView.java
index 44cee38fb..bc40e3d58 100644
--- a/src/main/java/edu/rpi/legup/ui/boardview/DataSelectionView.java
+++ b/src/main/java/edu/rpi/legup/ui/boardview/DataSelectionView.java
@@ -11,6 +11,11 @@
*/
public class DataSelectionView extends JPopupMenu {
+ /**
+ * Constructs a DataSelectionView with the given controller.
+ *
+ * @param controller The ElementController to handle UI events.
+ */
public DataSelectionView(ElementController controller) {
setBackground(Color.GRAY);
setBorder(new BevelBorder(BevelBorder.RAISED));
diff --git a/src/main/java/edu/rpi/legup/ui/boardview/ElementSelection.java b/src/main/java/edu/rpi/legup/ui/boardview/ElementSelection.java
index 5ddbab909..eaa85082d 100644
--- a/src/main/java/edu/rpi/legup/ui/boardview/ElementSelection.java
+++ b/src/main/java/edu/rpi/legup/ui/boardview/ElementSelection.java
@@ -12,16 +12,29 @@ public class ElementSelection {
private ElementView hover;
private Point mousePoint;
+ /**
+ * Constructs an ElementSelection instance with an empty selection and no hover or mouse point
+ */
public ElementSelection() {
this.selection = new ArrayList<>();
this.hover = null;
this.mousePoint = null;
}
+ /**
+ * Gets the list of currently selected ElementViews.
+ *
+ * @return the list of selected ElementViews
+ */
public ArrayList getSelection() {
return selection;
}
+ /**
+ * Gets the first ElementView in the selection, or null if the selection is empty.
+ *
+ * @return the first selected ElementView, or null if there are no selections
+ */
public ElementView getFirstSelection() {
return selection.size() == 0 ? null : selection.get(0);
}
@@ -62,10 +75,20 @@ public void clearSelection() {
selection.clear();
}
+ /**
+ * Gets the currently hovered ElementView.
+ *
+ * @return the currently hovered ElementView, or null if no element is hovered
+ */
public ElementView getHover() {
return hover;
}
+ /**
+ * Sets a new hovered ElementView, updating the hover state of the previous and new elements.
+ *
+ * @param newHovered the new ElementView to be hovered
+ */
public void newHover(ElementView newHovered) {
newHovered.setHover(true);
if (hover != null) {
@@ -82,10 +105,20 @@ public void clearHover() {
}
}
+ /**
+ * Gets the current mouse point location.
+ *
+ * @return the current mouse point location
+ */
public Point getMousePoint() {
return mousePoint;
}
+ /**
+ * Sets the mouse point location.
+ *
+ * @param point the new mouse point location
+ */
public void setMousePoint(Point point) {
this.mousePoint = point;
}
diff --git a/src/main/java/edu/rpi/legup/ui/boardview/ElementView.java b/src/main/java/edu/rpi/legup/ui/boardview/ElementView.java
index 48a33493c..eab13cc1d 100644
--- a/src/main/java/edu/rpi/legup/ui/boardview/ElementView.java
+++ b/src/main/java/edu/rpi/legup/ui/boardview/ElementView.java
@@ -97,8 +97,19 @@ public void drawElement(Graphics2D graphics2D) {
graphics2D.drawString(String.valueOf(puzzleElement.getData()), xText, yText);
}
+ /**
+ * Draws additional elements for given PuzzleElements (default implementation does nothing).
+ * Overriden in some puzzle element views.
+ *
+ * @param graphics2D the Graphics2D context to use for drawing
+ */
public void drawGiven(Graphics2D graphics2D) {}
+ /**
+ * Draws a hover effect on the ElementView.
+ *
+ * @param graphics2D the Graphics2D context to use for drawing
+ */
public void drawHover(Graphics2D graphics2D) {
graphics2D.setColor(hoverColor);
graphics2D.setStroke(new BasicStroke(2));
@@ -107,6 +118,11 @@ public void drawHover(Graphics2D graphics2D) {
location.x + 1.5f, location.y + 1.5f, size.width - 3, size.height - 3));
}
+ /**
+ * Draws a modified effect on the ElementView.
+ *
+ * @param graphics2D the Graphics2D context to use for drawing
+ */
public void drawModified(Graphics2D graphics2D) {
graphics2D.setColor(puzzleElement.isValid() ? modifiedColor : invalidColor);
graphics2D.setStroke(new BasicStroke(2));
@@ -115,6 +131,11 @@ public void drawModified(Graphics2D graphics2D) {
location.x + 1.5f, location.y + 1.5f, size.width - 3, size.height - 3));
}
+ /**
+ * Draws a case rule picker on the ElementView.
+ *
+ * @param graphics2D the Graphics2D context to use for drawing
+ */
public void drawCase(Graphics2D graphics2D) {
graphics2D.setColor(caseColor);
graphics2D.fill(
@@ -122,6 +143,11 @@ public void drawCase(Graphics2D graphics2D) {
location.x + 1.5f, location.y + 1.5f, size.width - 3, size.height - 3));
}
+ /**
+ * Creates an image representation of the ElementView.
+ *
+ * @return a BufferedImage of the ElementView
+ */
public BufferedImage getImage() {
BufferedImage image =
new BufferedImage(size.width, size.height, BufferedImage.TYPE_INT_RGB);
@@ -203,10 +229,20 @@ public void setPuzzleElement(PuzzleElement data) {
this.puzzleElement = data;
}
+ /**
+ * Checks if the case picker should be shown for this ElementView
+ *
+ * @return true if the case picker should be shown, false otherwise
+ */
public boolean isShowCasePicker() {
return showCasePicker;
}
+ /**
+ * Sets whether the case picker should be shown for this ElementView
+ *
+ * @param showCasePicker true if the case picker should be shown, false otherwise
+ */
public void setShowCasePicker(boolean showCasePicker) {
this.showCasePicker = showCasePicker;
}
@@ -414,11 +450,21 @@ public PathIterator getPathIterator(AffineTransform at, double flatness) {
.getPathIterator(at, flatness);
}
+ /**
+ * Returns the bounding rectangle of this ElementView
+ *
+ * @return a Rectangle representing the bounding box of this ElementView
+ */
@Override
public Rectangle getBounds() {
return new Rectangle(location.x, location.y, size.width, size.height);
}
+ /**
+ * Returns the bounding rectangle of this ElementView as a Rectangle2D
+ *
+ * @return a Rectangle2D representing the bounding box of this ElementView
+ */
@Override
public Rectangle2D getBounds2D() {
return new Rectangle(location.x, location.y, size.width, size.height);
diff --git a/src/main/java/edu/rpi/legup/ui/boardview/GridBoardView.java b/src/main/java/edu/rpi/legup/ui/boardview/GridBoardView.java
index db5ccbc82..e2b8bbc12 100644
--- a/src/main/java/edu/rpi/legup/ui/boardview/GridBoardView.java
+++ b/src/main/java/edu/rpi/legup/ui/boardview/GridBoardView.java
@@ -81,9 +81,9 @@ public void initSize() {
}
/**
- * Helper method to determine the proper dimension of the grid view
+ * Determines the proper dimension of the grid view based on grid size and element size.
*
- * @return proper dimension of the grid view
+ * @return the dimension of the grid view
*/
protected Dimension getProperSize() {
Dimension boardViewSize = new Dimension();
@@ -102,6 +102,11 @@ public DataSelectionView getSelectionPopupMenu() {
return null;
}
+ /**
+ * Gets the size of each element in the grid
+ *
+ * @return the dimension of each element in the grid
+ */
public Dimension getElementSize() {
return this.elementSize;
}
diff --git a/src/main/java/edu/rpi/legup/ui/boardview/SelectionItemView.java b/src/main/java/edu/rpi/legup/ui/boardview/SelectionItemView.java
index e3bb3592f..6a1429a61 100644
--- a/src/main/java/edu/rpi/legup/ui/boardview/SelectionItemView.java
+++ b/src/main/java/edu/rpi/legup/ui/boardview/SelectionItemView.java
@@ -58,6 +58,11 @@ public SelectionItemView(PuzzleElement data) {
this(data, (Integer) data.getData());
}
+ /**
+ * Gets the PuzzleElement associated with this menu item
+ *
+ * @return the PuzzleElement associated with this menu item
+ */
public PuzzleElement getData() {
return data;
}
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RuleFrame.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RuleFrame.java
index cf01f3ede..b14091351 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RuleFrame.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RuleFrame.java
@@ -131,7 +131,7 @@ public void resetSize() {
/**
* Set the status label to a value. Use resetStatus to clear it.
*
- * @param check true iff we want a check box, if false we'll have a red x box
+ * @param check true if we want a checkbox, if false we'll have a red x box
* @param text the text we're setting the label to display
*/
public void setStatus(boolean check, String text) {
@@ -169,22 +169,47 @@ public RuleController getController() {
return controller;
}
+ /**
+ * Gets the JTabbedPane used in this frame
+ *
+ * @return the JTabbedPane instance
+ */
public JTabbedPane getTabbedPane() {
return tabbedPane;
}
+ /**
+ * Gets the {@code DirectRulePanel} contained in this frame
+ *
+ * @return the {@link DirectRulePanel} instance
+ */
public DirectRulePanel getDirectRulePanel() {
return DirectRulePanel;
}
+ /**
+ * Gets the {@code CaseRulePanel} contained in this frame
+ *
+ * @return the {@link CaseRulePanel} instance
+ */
public CaseRulePanel getCasePanel() {
return casePanel;
}
+ /**
+ * Gets the {@code ContradictionRulePanel} contained in this frame
+ *
+ * @return the {@link ContradictionRulePanel} instance
+ */
public ContradictionRulePanel getContradictionPanel() {
return contradictionPanel;
}
+ /**
+ * Gets the {@code SearchBarPanel} contained in this frame
+ *
+ * @return the {@link SearchBarPanel} instance
+ */
public SearchBarPanel getSearchPanel() {
return searchPanel;
}
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RulePanel.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RulePanel.java
index c5e7327fd..85ddf82b5 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RulePanel.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/rulesview/RulePanel.java
@@ -36,7 +36,7 @@ public RulePanel(RuleFrame ruleFrame) {
}
/**
- * Gets the rule rule buttons
+ * Gets the array of rule buttons
*
* @return rule ruleButtons
*/
@@ -338,28 +338,58 @@ public List extends Rule> getRules() {
return rules;
}
+ /**
+ * Gets the icon associated with this panel
+ *
+ * @return The ImageIcon associated with this panel
+ */
public ImageIcon getIcon() {
return icon;
}
+ /**
+ * Sets the icon for this panel
+ *
+ * @return the ImageIcon associated with this panel
+ */
public void setIcon(ImageIcon icon) {
this.icon = icon;
}
+ /**
+ * Gets the name of this panel
+ *
+ * @return the name of this panel in a String
+ */
@Override
public String getName() {
return name;
}
+ /**
+ * Sets the name of this panel
+ *
+ * @param name the name to set for this panel
+ */
@Override
public void setName(String name) {
this.name = name;
}
+ /**
+ * Gets the tooltip text associated with this panel
+ *
+ * @return the tooltip text of this panel
+ */
public String getToolTip() {
return toolTip;
}
+ /**
+ * Sets the tooltip text for this panel
+ *
+ * @param toolTip the tooltip text to set for this panel
+ */
public void setToolTip(String toolTip) {
this.toolTip = toolTip;
}
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeElementView.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeElementView.java
index 0233ec6d4..e5e523e11 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeElementView.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeElementView.java
@@ -41,7 +41,7 @@ protected TreeElementView(TreeElementType type, TreeElement treeElement) {
public abstract void draw(Graphics2D graphics2D);
/**
- * Gets the span for the sub tree rooted at this view
+ * Gets the span for the subtree rooted at this view
*
* @return span bounded y span
*/
@@ -50,7 +50,7 @@ public double getSpan() {
}
/**
- * Sets the span for the sub tree rooted at this view.
+ * Sets the span for the subtree rooted at this view.
*
* @param span bounded y span
*/
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeNodeView.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeNodeView.java
index 8e9562d10..c9aedaef3 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeNodeView.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeNodeView.java
@@ -266,11 +266,21 @@ public int getRadius() {
return RADIUS;
}
+ /**
+ * Returns the bounding rectangle of this TreeNodeView
+ *
+ * @return a Rectangle representing the bounding box of this TreeNodeView
+ */
@Override
public Rectangle getBounds() {
return new Rectangle(location.x, location.y, DIAMETER, DIAMETER);
}
+ /**
+ * Returns the bounding rectangle of this TreeNodeView as a Rectangle2D
+ *
+ * @return a Rectangle2D representing the bounding box of this TreeNodeView
+ */
@Override
public Rectangle2D getBounds2D() {
return new Rectangle(location.x, location.y, DIAMETER, DIAMETER);
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreePanel.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreePanel.java
index 9efeb8dfb..936b0de95 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreePanel.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreePanel.java
@@ -66,10 +66,20 @@ public TreePanel(/*LegupUI legupUI*/ ) {
updateStatusTimer = 0;
}
+ /**
+ * Repaints the tree view with the provided {@link Tree} object
+ *
+ * @param tree the {@link Tree} object to update the view with
+ */
public void repaintTreeView(Tree tree) {
treeView.updateTreeView(tree);
}
+ /**
+ * Updates the status of the panel based on changes to the {@link Board}
+ *
+ * @param board the {@link Board} object representing the current board state
+ */
public void boardDataChanged(Board board) {
modifiedSinceSave = true;
modifiedSinceUndoPush = true;
@@ -89,18 +99,33 @@ public void updateStatus() {
this.status.setText("");
}
+ /**
+ * Updates the status display with the given status string
+ *
+ * @param statusString the status string to display
+ */
public void updateStatus(String statusString) {
status.setForeground(Color.BLACK);
status.setFont(MaterialFonts.REGULAR);
status.setText(statusString);
}
+ /**
+ * Updates the status display as an error with an error message
+ *
+ * @param error the error message to display
+ */
public void updateError(String error) {
status.setForeground(Color.RED);
status.setFont(MaterialFonts.ITALIC);
status.setText(error);
}
+ /**
+ * Gets the {@link TreeView} instance associated with this panel
+ *
+ * @return the {@link TreeView} instance
+ */
public TreeView getTreeView() {
return treeView;
}
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarButton.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarButton.java
index 2718f733d..232498723 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarButton.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarButton.java
@@ -9,6 +9,12 @@ public class TreeToolBarButton extends JButton {
private TreeToolBarName name;
private final Dimension MINIMUM_DIMENSION = new Dimension(60, 60);
+ /**
+ * Constructs a {@code TreeToolBarButton} with the specified icon and name.
+ *
+ * @param imageIcon the {@link ImageIcon} to be displayed on the button
+ * @param name the {@link TreeToolBarName} associated with this button
+ */
public TreeToolBarButton(ImageIcon imageIcon, TreeToolBarName name) {
super(imageIcon);
this.name = name;
@@ -17,6 +23,11 @@ public TreeToolBarButton(ImageIcon imageIcon, TreeToolBarName name) {
this.setFocusPainted(false);
}
+ /**
+ * Gets the {@link TreeToolBarName} associated with this button
+ *
+ * @return the {@link TreeToolBarName} associated with this button
+ */
public TreeToolBarName getToolBarName() {
return name;
}
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarName.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarName.java
index c805021be..3aec664be 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarName.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeToolBarName.java
@@ -1,5 +1,9 @@
package edu.rpi.legup.ui.proofeditorui.treeview;
+/**
+ * {@code TreeToolBarName} defines the names of actions represented by buttons in the tree toolbar.
+ * These actions are used for managing tree elements within the UI.
+ */
public enum TreeToolBarName {
ADD_CHILD,
DEL_CHILD,
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeTransitionView.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeTransitionView.java
index 70546616d..5a3aa7d5c 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeTransitionView.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeTransitionView.java
@@ -271,47 +271,67 @@ public void removeParentView(TreeNodeView nodeView) {
}
}
- public Point getEndPoint() {
- return endPoint;
- }
-
- public void setEndPoint(Point endPoint) {
- this.endPoint = endPoint;
- }
-
+ /**
+ * Gets the x-coordinate of the end point of the transition arrow
+ *
+ * @return the x-coordinate of the end point
+ */
public int getEndX() {
return endPoint.x;
}
+ /**
+ * Sets the x-coordinate of the end point of the transition arrow
+ *
+ * @param x the new x-coordinate of the end point
+ */
public void setEndX(int x) {
this.endPoint.x = x;
}
+ /**
+ * Gets the y-coordinate of the end point of the transition arrow
+ *
+ * @return the y-coordinate of the end point
+ */
public int getEndY() {
return endPoint.y;
}
+ /**
+ * Sets the y-coordinate of the end point of the transition arrow
+ *
+ * @param y the new y-coordinate of the end point
+ */
public void setEndY(int y) {
this.endPoint.y = y;
}
- public List getLineStartPoints() {
- return lineStartPoints;
- }
-
- public void setLineStartPoints(List lineStartPoints) {
- this.lineStartPoints = lineStartPoints;
- }
-
+ /**
+ * Gets the start point at the specified index from the list of start points
+ *
+ * @param index the index of the start point to retrieve
+ * @return the start point at the specified index, or null if the index is out of range
+ */
public Point getLineStartPoint(int index) {
return index < lineStartPoints.size() ? lineStartPoints.get(index) : null;
}
+ /**
+ * Returns the bounding rectangle of this TreeTransitionView
+ *
+ * @return a Rectangle representing the bounding box of this TreeTransitionView
+ */
@Override
public Rectangle getBounds() {
return arrowhead.getBounds();
}
+ /**
+ * Returns the bounding rectangle of this TreeTransitionView as a Rectangle2D
+ *
+ * @return a Rectangle2D representing the bounding box of this TreeTransitionView
+ */
@Override
public Rectangle2D getBounds2D() {
return arrowhead.getBounds2D();
diff --git a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeView.java b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeView.java
index 3d74d1e08..65f49ff0a 100644
--- a/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeView.java
+++ b/src/main/java/edu/rpi/legup/ui/proofeditorui/treeview/TreeView.java
@@ -57,6 +57,11 @@ public class TreeView extends ScrollView implements ITreeListener {
private TreeViewSelection selection;
+ /**
+ * Constructs a {@code TreeView} with the specified {@code TreeController}.
+ *
+ * @param treeController the {@code TreeController} used to manage tree operations
+ */
public TreeView(TreeController treeController) {
super(treeController);
currentStateBoxes = new ArrayList<>();
@@ -68,6 +73,11 @@ public TreeView(TreeController treeController) {
selection = new TreeViewSelection();
}
+ /**
+ * Gets the current tree view selection
+ *
+ * @return the {@code TreeViewSelection} object representing the current selection
+ */
public TreeViewSelection getSelection() {
return selection;
}
@@ -137,6 +147,11 @@ private TreeElementView getTreeElementView(Point point, TreeElementView elementV
return null;
}
+ /**
+ * Updates the tree view with the specified {@code Tree}
+ *
+ * @param tree the {@code Tree} to display in the view
+ */
public void updateTreeView(Tree tree) {
this.tree = tree;
if (selection.getSelectedViews().size() == 0) {
@@ -174,8 +189,9 @@ public void reset() {
* button is selected
*/
public void zoomFit() {
- double fitWidth = (viewport.getWidth() - 8.0) / (getSize().width - 200);
- double fitHeight = (viewport.getHeight() - 8.0) / (getSize().height - 120);
+ final int MIN_HEIGHT = 200;
+ double fitWidth = (viewport.getWidth() - 7.0) / (getSize().width - 75);
+ double fitHeight = (viewport.getHeight()) / Math.max((getSize().height - 115), MIN_HEIGHT);
zoomTo(Math.min(fitWidth, fitHeight));
viewport.setViewPosition(new Point(0, viewport.getHeight() / 2));
}
@@ -224,6 +240,11 @@ public void layoutContainer(Container parent) {
};
}
+ /**
+ * Draws the tree view on the provided {@code Graphics2D} context
+ *
+ * @param graphics2D the {@code Graphics2D} context to draw on
+ */
public void draw(Graphics2D graphics2D) {
currentStateBoxes.clear();
Tree tree = GameBoardFacade.getInstance().getTree();
@@ -255,6 +276,12 @@ public void zoomReset() {
viewport.setViewPosition(new Point(0, 0));
}
+ /**
+ * Recursively redraws the tree starting from the specified node view
+ *
+ * @param graphics2D the {@code Graphics2D} context to draw on
+ * @param nodeView the {@code TreeNodeView} to start drawing from
+ */
private void redrawTree(Graphics2D graphics2D, TreeNodeView nodeView) {
if (nodeView != null) {
nodeView.draw(graphics2D);
@@ -265,6 +292,11 @@ private void redrawTree(Graphics2D graphics2D, TreeNodeView nodeView) {
}
}
+ /**
+ * Removes the specified {@code TreeElementView} from the tree view
+ *
+ * @param view the {@code TreeElementView} to remove
+ */
public void removeTreeElement(TreeElementView view) {
if (view.getType() == NODE) {
TreeNodeView nodeView = (TreeNodeView) view;
@@ -424,8 +456,17 @@ public TreeElementView getElementView(TreeElement element) {
return viewMap.get(element);
}
- private void removeTreeNode(TreeNode node) {
+ /**
+ * Removes the specified {@link TreeNode} and its associated views
+ *
+ * @param node the {@link TreeNode} to be removed
+ */
+ public void removeTreeNode(TreeNode node) {
viewMap.remove(node);
+ if (node.getChildren() != null) {
+ node.getChildren().forEach(t -> removeTreeTransition(t));
+ }
+
List children = node.getChildren();
// if child is a case rule, unlock ancestor elements
@@ -452,11 +493,13 @@ private void removeTreeNode(TreeNode node) {
}
// set modifiable if started modifiable
- boolean modifiable =
- tree.getRootNode()
- .getBoard()
- .getPuzzleElement(oldElement)
- .isModifiable();
+ boolean modifiable = false;
+ if (tree != null) {
+ tree.getRootNode()
+ .getBoard()
+ .getPuzzleElement(oldElement)
+ .isModifiable();
+ }
// unmodifiable if already modified
TreeNode modNode = ancestor.getParent().getParents().get(0);
@@ -474,10 +517,14 @@ private void removeTreeNode(TreeNode node) {
}
}
}
- node.getChildren().forEach(t -> removeTreeTransition(t));
}
- private void removeTreeTransition(TreeTransition trans) {
+ /**
+ * Removes the specified {@link TreeTransition} and its associated views
+ *
+ * @param trans the {@link TreeTransition} to be removed
+ */
+ public void removeTreeTransition(TreeTransition trans) {
viewMap.remove(trans);
if (trans.getChildNode() != null) {
removeTreeNode(trans.getChildNode());
@@ -514,12 +561,16 @@ private void removeTreeTransition(TreeTransition trans) {
}
}
+ /**
+ * Adds the specified {@link TreeNode} and its associated views
+ *
+ * @param node the {@link TreeNode} to be added
+ */
private void addTreeNode(TreeNode node) {
TreeTransition parent = node.getParent();
-
TreeNodeView nodeView = new TreeNodeView(node);
- TreeTransitionView parentView = (TreeTransitionView) viewMap.get(parent);
+ TreeTransitionView parentView = (TreeTransitionView) viewMap.get(parent);
nodeView.setParentView(parentView);
parentView.setChildView(nodeView);
@@ -554,20 +605,25 @@ private void addTreeNode(TreeNode node) {
}
}
+ /**
+ * Adds the specified {@link TreeTransition} and its associated views
+ *
+ * @param trans The {@link TreeTransition} to be added
+ */
private void addTreeTransition(TreeTransition trans) {
List parents = trans.getParents();
-
TreeTransitionView transView = new TreeTransitionView(trans);
+
for (TreeNode parent : parents) {
TreeNodeView parentNodeView = (TreeNodeView) viewMap.get(parent);
transView.addParentView(parentNodeView);
parentNodeView.addChildrenView(transView);
+ viewMap.put(trans, transView);
+
// if transition is a new case rule, lock dependent ancestor elements
Rule rule = trans.getRule();
if (rule instanceof CaseRule && parent.getChildren().size() == 1) {
- CaseRule caseRule = (CaseRule) rule;
-
List ancestors = parent.getAncestors();
for (TreeNode ancestor : ancestors) {
// for all ancestors but root
@@ -584,15 +640,16 @@ private void addTreeTransition(TreeTransition trans) {
}
}
- viewMap.put(trans, transView);
-
if (trans.getChildNode() != null) {
addTreeNode(trans.getChildNode());
}
}
- /// New Draw Methods
-
+ /**
+ * Draws the tree using the provided {@link Graphics2D} object
+ *
+ * @param graphics2D the {@link Graphics2D} object used for drawing the tree
+ */
public void drawTree(Graphics2D graphics2D) {
if (tree == null) {
LOGGER.error("Unable to draw tree.");
@@ -618,6 +675,11 @@ public void drawTree(Graphics2D graphics2D) {
}
}
+ /**
+ * Creates views for the given {@link TreeNodeView} and its children
+ *
+ * @param nodeView the {@link TreeNodeView} for which to create views
+ */
public void createViews(TreeNodeView nodeView) {
if (nodeView != null) {
viewMap.put(nodeView.getTreeElement(), nodeView);
diff --git a/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementFrame.java b/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementFrame.java
index bcb69f6c0..1619ecbaa 100644
--- a/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementFrame.java
+++ b/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementFrame.java
@@ -19,45 +19,30 @@ public class ElementFrame extends JPanel {
private EditorElementController controller;
public ElementFrame(EditorElementController controller) {
+
this.controller = controller;
- MaterialTabbedPaneUI tabOverride =
- new MaterialTabbedPaneUI() {
- // this prevents the tabs from moving around when you select them
- @Override
- protected boolean shouldRotateTabRuns(int i) {
- return false;
- }
- };
-
- this.tabbedPane = new JTabbedPane();
- tabbedPane.setUI(tabOverride);
+
JLabel status = new JLabel("", SwingConstants.CENTER);
this.buttonGroup = new ButtonGroup();
- nonPlaceableElementPanel = new NonPlaceableElementPanel(this);
- // nonPlaceableElementPanel.setMinimumSize(new Dimension(100,200));
- tabbedPane.addTab(
- nonPlaceableElementPanel.getName(),
- nonPlaceableElementPanel.getIcon(),
- new JScrollPane(nonPlaceableElementPanel),
- nonPlaceableElementPanel.getToolTip());
+ // Parent panel to hold all elements
+ JPanel elementPanel = new JPanel();
+ elementPanel.setLayout(new BoxLayout(elementPanel, BoxLayout.Y_AXIS));
placeableElementPanel = new PlaceableElementPanel(this);
- // placeableElementPanel.setMinimuSize(new Dimension(100,200));
- tabbedPane.addTab(
- placeableElementPanel.getName(),
- placeableElementPanel.getIcon(),
- new JScrollPane(placeableElementPanel),
- placeableElementPanel.getToolTip());
- tabbedPane.setTabPlacement(JTabbedPane.TOP);
+ placeableElementPanel.setMinimumSize(new Dimension(100, 200));
+ elementPanel.add(new JScrollPane(placeableElementPanel));
+ // Set layout and dimensions for the main panel
setLayout(new BorderLayout());
setMinimumSize(new Dimension(250, 256));
setPreferredSize(new Dimension(330, 256));
- add(tabbedPane);
+ // Add components to the main panel
+ add(elementPanel, BorderLayout.CENTER);
add(status, BorderLayout.SOUTH);
+ // Center-align the titled border
TitledBorder title = BorderFactory.createTitledBorder("Elements");
title.setTitleJustification(TitledBorder.CENTER);
setBorder(title);
@@ -75,8 +60,9 @@ public ButtonGroup getButtonGroup() {
// }
public void setElements(Puzzle puzzle) {
- nonPlaceableElementPanel.setElements(puzzle.getNonPlaceableElements());
- placeableElementPanel.setElements(puzzle.getPlaceableElements());
+ if (puzzle != null) {
+ placeableElementPanel.setElements(puzzle.getPlaceableElements());
+ }
}
public EditorElementController getController() {
diff --git a/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementPanel.java b/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementPanel.java
index 46198e226..70826d25c 100644
--- a/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementPanel.java
+++ b/src/main/java/edu/rpi/legup/ui/puzzleeditorui/elementsview/ElementPanel.java
@@ -20,7 +20,7 @@ public ElementPanel(ElementFrame eFrame) {
setLayout(new WrapLayout());
}
- public void setElements(List extends Element> elements) {
+ public int setElements(List extends Element> elements) {
this.elements = elements;
clearButtons();
@@ -38,6 +38,7 @@ public void setElements(List extends Element> elements) {
add(elementButtons[i]);
}
revalidate();
+ return elements.size();
}
protected void clearButtons() {
diff --git a/src/main/resources/edu/rpi/legup/images/Legup/Reset.png b/src/main/resources/edu/rpi/legup/images/Legup/Reset.png
new file mode 100644
index 0000000000000000000000000000000000000000..fd6ffefa0fd001bd771519e6094c57919b51f2ba
GIT binary patch
literal 662
zcmV;H0%`q;P)(^b8FWQhbW?9;ba!ELWdL_~cP?peYja~^aAhuUa%Y?FJQ@H10ryEnK~z{r?UqSO
z#6TEE+hL&~;Jf$e3Vblb4i~dmL!9j@J(*YIms31lg-lTkxUpb^2)X4Dqu}o
z(3TvO)t~@=^_R498?s&U9g}QguSKwN%OB=r+gBXy{GGua~04_B}z_KKR
zEHb^>l?>b|fAYr)K>iIt23cf!!HVY|QguHK0Z_oZ)BYV~kVPihzlg73V_vdQ_T&W(
z0q{#%^~i)k23cf!t%UB)ckXuV<3Z!!#7ckR+3A=7>n%f(}Xb
zukwzDf*^}bvY8_yZ4LYfnVV0t0RBEy+a{^DD99p{Z03ka7VsfYDyO9hcPs+SBAYoP
z(r&?_JI=Uc5zM-yWmlz^jVO((8&MW{x8gK{w_?cqFiOH$GRO*g7(t%LaU7c*$;!p5
zv=E;c5}*%#Z5$aj@z*OEa!woH(TBc_VQe8}y_CKWWh72gaC{&})GxDgXcg07*qoM6N<$f~1Tez5oCK
literal 0
HcmV?d00001
diff --git a/src/main/resources/edu/rpi/legup/images/binary/rules/CompleteRowColumnDirectRule.png b/src/main/resources/edu/rpi/legup/images/binary/rules/CompleteRowColumnDirectRule.png
index a74654d4301f75defc639727604f3d59f05612a3..27ccec97290109fe20d727d468465a57af330906 100644
GIT binary patch
literal 2174
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
W-$D%jr3*N)f}HK$?8r8Oq<;bxYC}%|
diff --git a/src/main/resources/edu/rpi/legup/images/binary/rules/DuplicateRowOrColumnContradictionRule.png b/src/main/resources/edu/rpi/legup/images/binary/rules/DuplicateRowOrColumnContradictionRule.png
deleted file mode 100644
index 214aa53483ccc0f6749bffda004ccfc2f70ded29..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 3455
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
diff --git a/src/main/resources/edu/rpi/legup/images/binary/rules/OneOrZeroCaseRule.png b/src/main/resources/edu/rpi/legup/images/binary/rules/OneOrZeroCaseRule.png
deleted file mode 100644
index 73072f2ce5ea9d189fe55058aa6a4fa09ecc3979..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 5578
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
diff --git a/src/main/resources/edu/rpi/legup/images/binary/rules/OneTileGapDirectRule.png b/src/main/resources/edu/rpi/legup/images/binary/rules/OneTileGapDirectRule.png
deleted file mode 100644
index b68f67e441d3519f17daffd48da37f35e9b3bcfb..0000000000000000000000000000000000000000
GIT binary patch
literal 0
HcmV?d00001
literal 2751
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
diff --git a/src/main/resources/edu/rpi/legup/images/binary/rules/PreventTrioDirectRule.png b/src/main/resources/edu/rpi/legup/images/binary/rules/PreventTrioDirectRule.png
new file mode 100644
index 0000000000000000000000000000000000000000..ed0701f26fc1880098ae5f07e8c309e113a63662
GIT binary patch
literal 1944
zcmbVN`8OK~7mZpHwfAFCu|!)MM6D$)v46IvRHJr+qKa0fG|ea_H4!9bx~SG(2{N&E
zYFA5Yr$eJgYKyHXB2%?hFqwa$=iK|=x#zs|&bjaY@QALL?In(%JPrT=Bpe-V-S|E1
zKY~Q~RoTjSkKY6^ZuZuI$`9Zbz7PttLRtX;HIK!3fx>((8tveP;g6>LM**&q!5RPn
zggM$;p|Jkzc|JE}Jro`uA!{37>~|r9APZ-ZkYR^jJj~5u1ZU-{BraT&_#+h20igi#ojLq7dHy0oD~tYL!BN$_r_vo-cOn49brRL&vy{vVf)Po
z7N&!_In{5D5)3hihM3T~V*rMZxK-!EQ%EYTM^C(Bk+-+0XJS&%zTi$NFBhejmdg8<
z(u_<@7~|vP$*U+iF`p01%VoJt#=^pBaVe>Rmns?>8tCiSZO6V#XGD2&!%2ENgr7uR
zZb`{$cqu_n`4HX4aNPVoLo?U()DPRHHRap=W)fKJf2Kzs#H*Xf{pLcx!K`elc>gK_=h)
z*X_;*sNAEJln(j^lsCc>$(NFdPMKGLH@e9LhRPn60!~HER@YU7I;R5r=PmVM!=0y4##7ZE9U^Wu-0znC_&muHHUjAzAQXOJCM_AEMqZeE2Y+(%t6i#S`(<
zfIqX}TMg~BS^BzVD72TInCzocsf8^q;7Y${k(!#C7!r=oq{2#DTm4?;j&z9bew%PR
zTc(`!*sOUL-{7+7ZOl6#d(0cRS{P9eOfd5nt09hfFR*WXq|vk*a9gwoUq_P8XqK_O
z+d&U|@+2BZHnd^kGG23UY)FOu9S0Ef3lctvYgn)mQ6#*B0d0E6aCs4V@-x@#d*$~(tDbArirdEf-FmN9+EiHL^+TS#q
z4OnBnu`iiQXjAQE45?78^Udxbo2b7{=Rh&Unb4hp&njTK%7E(d5Kael;O{x
zk0T0P?t^L3r-A#3@ega`HG(l1OqT)u4^B$JyLhil9+kdODvQO6-u!~Z@^)*=&dKxM
zdDcZui2UJE^~->3##d9+tb-({5eo~SYKqOw&DF`KEp?`d1>KB&RAGM7C0485w0g54
z;Q+_Qn#DlXeTG9_oGeV+sgVr^|$1%y0?IVpU7pwa0PK{
z9v+V`M(F9OQMFjLAz}+lOExw(__E*tp$)@wvlu-x-;DfRA@1{CzGdLl)KoNQ5}s8d
zCo9{2u)kCE`n3)Hs&JghKOrZ^S>_2m)5z%-#hVck&y*>TNwV9(p(
z;a5%)w7A%~yB5IgU?sGlTzh0#Xy}(`7L&Yv_@1#~^=8Fu*Id+@zjY<4t-FF3iW0u7
z8Qt|@%oJ4N77xJMl9K}TwIN4)*~g}}I~eZ3tQ5(*x)7|qPENV1@7$Fp=n|y>xROE~V^HjmF+LfbkfTMeYVVacM??MRn4*+C>B**8XCpF==@faf
zMR1s0BEm&X;|D+3;W&H-q?+xNjEfxVeW@}iNuS-qg&?JnNMtu2C?&qxJ98`T|$CPGbez2o-NW+6xHbT#B6n+jq
zJ=L^k&%&)j3dS@uU&iYTb#c)
zcZIBvK-^kh7%jir&E;0pSBj|A?&a0hC&uQZpFSy?o152Cfdj;;FN08(-2($tO9ZN}
z2v>a+6kYfKpESomi4U1`ge0B2$ApiGb?D!LLZP)(0y>h>sZ!rPjCYB+F0c#i(WWcP
TsFd*d-xT0zciFbm8lC(vAL*4l
literal 0
HcmV?d00001
diff --git a/src/main/resources/edu/rpi/legup/images/binary/rules/RepeatedRowColumnContradictionRule.png b/src/main/resources/edu/rpi/legup/images/binary/rules/RepeatedRowColumnContradictionRule.png
new file mode 100644
index 0000000000000000000000000000000000000000..5cc030a7671f6e567d56c9d71ddd640bd5a33336
GIT binary patch
literal 2124
zcmai0dpHy7AK#k$t&OG`5l(UtO|E4!*H&_gL#7RLmI>}({(;9>v(KoWh<
z@)AFL^MfEF#9!U;*d%@ek}lbp1Ij)s%=3YO59%Tc04Prv-}4gWW6>Mu+(-bxk;K0O
zn9g1!006?1XiJnc8Ml<@cP`si;eAxdTlw4PoMme-AL_3WIsbEx`aO5Z9S5;G;v)n&
zyu8*yZ*W+w$t10wU^;F$H_Lsq5Vh_Z>5vOty!
zH$*BKk?HSeL$8mISDVc46dI{4NC`KBgr3#M3R||!EiZ>i`FzZ$hL7v%85mgC21<|B
zQfzF}6ejDtxPj8K)9K*XT==}4qB5QwpELTc`V6`Zybm({wV&ATmrn8^7$^pPWT+-}W<4rav_N-04uiopVBb`D!IC6EO@!pk9(TKYRcH{co<$T`mUOF)%prE2wwz&_NyoPge()RX^(*3lPC+#F*
zt3^9_@<}7eCI!~G?ccBga~iLUV0mwRdSFr712gc4rOKU3V0n5c`y@)pHL+kQ50SV6yzwyg2mOQ6=NX!w7J+d_NHhp=#zB85<
z+ZGfQ^r^qpJ*+i$(>8`o%h+BS?-UmhFp8J_!0X5}?CtHf9H|WO_OhkO`4^f-v~7*O
z3)MM8@UVnh_+Z@J&{Qfl|I
z(nBPL^C|PV{fqTPnVRz?|2dn}J|&s^_K=Y-D?_gn4cju=9ZeBd%7|vDHjv4iAMw^I=~I
z1j5|}1Zq1ac(H%&GBR+?1XCNXZ!I{hwmWnDvCQ!{z_b9FNTapY=-Jr1XM!tGUj
z6yh=l!|x2exzMM%WjS6HT7aZvsoB!Z