NonogramSolver.jl

Formulates and solves nonogram puzzles (a.k.a. Picross and paint-by-number) in Julia. Uses a new integer linear programming formulation, and solves it with JuMP.
Author kamilkhanlab
Popularity
0 Stars
Updated Last
7 Months Ago
Started In
June 2023

NonogramSolver.jl

Stable Dev Build Status Coverage

This package provides a Julia module for formulating nonogram puzzles (a.k.a. Picross, paint-by-number, and crucipixel), and for solving these puzzles using integer linear programming (ILP) via JuMP. A new effective ILP formulation by Khan is employed.

Monochrome and multicolored puzzles are both supported. If a puzzle has multiple solutions, then only one will be identified. If a puzzle has no solutions, then this will be identified, and a fake solution will be returned with no cells colored.

A simple example is included below; for further information and API details, please see the package's documentation linked above.

If you make use of this implementation in your own work, please cite the accompanying article:

Kamil A. Khan, Solving nonograms using integer programming without coloring, IEEE Transactions on Games, 14(1): 56-63, 2022. doi:10.1109/TG.2020.3036687

See also:

Puzzle overview

The goal of this puzzle is to color certain cells in a rectangular grid, in a manner that is consistent with clues pertaining to each row and each column. Each row's clues include a number for each block of contiguous colored cells in that row. The numerical value in this clue indicates how many cells are in the corresponding contiguous block. Columns' clues are analogous.

For example, the following puzzle instance:

Puzzle prompt

has the unique solution:

Puzzle solution

While "real-world" instances are often deliberately designed to be solvable by hand, this puzzle in general is NP-complete.

Installation

To install this package, enter the following command in Julia's REPL:

import Pkg; Pkg.add("NonogramSolver")

Example

With the package installed, we may solve the above example problem as follows in Julia. For more details, refer to the API documentation.

  1. Load the package's module:

    using NonogramSolver
  2. Enter the row clues as a Vector{Vector{Int}}, and the column clues similarly. Note that an uncolored row (or column), with no numerical clues at all, is entered as Int[]:

    rowClues = [[1,1], [1,1], Int[], [1,2], [3]]
    colClues = [[1], [2,1], [1], [2,2], [1]]
  3. Construct a Puzzle object to hold the puzzle's formulation:

    p = Puzzle(rowClues, colClues)
  4. Use JuMP to solve this puzzle as an integer linear program, using the freely available solver GLPK by default.

    solution = solve_puzzle(p; verbosity=0)

    You can customize the choice of solver and its options. Without verbosity=0, JuMP's solution summary would be printed as well.

  5. If the previous step was carried out in Julia's REPL, then the solution has already been displayed using colored-square emojis. Otherwise, retrieve the solution in one of three ways:

    • displayed to the REPL/terminal, via:

      @show solution
    • as a String containing the printed solution, with:

      stringOutput = repr(solution)
    • as a Matrix of 0s and 1s, with:

      matrixOutput = solution.z

Used By Packages

No packages found.