All Downloads are FREE. Search and download functionalities are using the official Maven repository.

examples.Life.src.main.gosu.life.Board.gs Maven / Gradle / Ivy

package life

uses javax.swing.JPanel
uses javax.swing.border.LineBorder
uses java.awt.Graphics2D
uses java.awt.Graphics
uses java.awt.Dimension
uses java.awt.Color
uses java.awt.Point
uses java.awt.image.BufferedImage
uses java.util.concurrent.Executors
uses java.util.concurrent.ExecutorService

class Board extends JPanel {
  final static var SELCTION_COLOR = new Color( 0, 0, 255, 128 )
  var _model: LifeModel as Model
  var _density: int as Density
  var _frequency: int as Frequency
  var _origin: Point as Origin
  var _modelProcessor: ModelProcessor
  var _scroller: ScrollHandler
  var _zoomer: ZoomHandler
  var _selector: SelectionHandler
  var _drawHandler: DrawHandler
  var _listeners: List
  var _freqListeners: Set
  var _colorScheme: ColorScheme as BoardColorScheme
  var _buffer: BufferedImage

  construct( model: LifeModel, density: int = 16, frequency: int = 100 ) {
    _origin = new Point( 0, 0 )
    _density = density
    _frequency = frequency
    _colorScheme = Manila
    _model = model
    Border = new LineBorder( _colorScheme.Grid )
    _scroller = new ScrollHandler( this )
    _zoomer = new ZoomHandler( this )
    _selector = new SelectionHandler( this )
    _drawHandler = new DrawHandler( this )
    _listeners = {}
    _freqListeners = {}
  }

  function start() {
    if( _modelProcessor == null ) {
      _modelProcessor = new()
      _modelProcessor.start()
    }
  }
  function stop() {
    _modelProcessor?.shutdown()
    _modelProcessor = null
  }

  function resetTimer( frequency: int ) {
    stop()
    Frequency = frequency
    start()
  }

  function reset( size: Dimension, infinite = false, multithreaded = true, populator: Populator = Random, cellSize = 100, frequency = 40, density = 16, start = true ) {
    stop()
    _model = new LifeModel( infinite, multithreaded, cellSize, size, getPopulatorCells( size, populator ) )
    _density = density
    _frequency = frequency
    _origin = new( 0, 0 )
    fireReset()
    if( start ) {
      start()
    }
    invalidate()
    revalidate()
  }

  property get Running(): boolean {
    return _modelProcessor != null
  }

  property get CellSize() : int {
    return _model.CellSize
  }

  property get XCells() : int {
    return Width / CellSize
  }
  property get YCells() : int {
    return Height/CellSize
  }

  property set Origin( value: Point ) {
    _origin = value
    repaintBuffer()
  }

  property get OriginPoint() : Point {
    return new( Origin.x * CellSize, Origin.y * CellSize )
  }

  override function paintComponent( g: Graphics ) {
    if( _buffer != null ) {
      g.drawImage( _buffer, 0, 0, null )
    }
  }

  function repaintBuffer() {
    _buffer = paintComponentViaBuffer()
    repaint()
  }

  function paintComponentViaBuffer() : BufferedImage {
    var bi = new BufferedImage( Width, Height, BufferedImage.TYPE_INT_ARGB )
    using( var g = bi.createGraphics() ) {
      g.Color = _colorScheme.Cell
      g.fillRect( 0, 0, Width, Height )
      drawGrid( g )
      drawCells( g, _model.LiveCells, _colorScheme.Live )
      drawCells( g, _selector.Selection, _selector.Dragging ? _colorScheme.Cell : SELCTION_COLOR )
      drawCells( g, _drawHandler.Line, _drawHandler.Dragging ? _colorScheme.Cell : SELCTION_COLOR )
    }
    return bi
  }

  private function drawGrid( g: Graphics2D ) {
    var step = CellSize
    if( step < 2 ) {
      return
    }
    g.Color = _colorScheme.Grid
    for( i in 1..XCells ) {
      g.drawLine( step * i, 0, step * i, Height )
    }
    for( i in 1..YCells ) {
      g.drawLine( 0, step * i, Width, step * i )
    }
  }

  private function drawCells( g: Graphics2D, cells: Collection, color: Color )  {
    g.Color = color
    var cellSize = Math.max( CellSize, 1 )
    var borderSize = cellSize > 1 ? 1 : 0
    g.translate( Origin.x * cellSize, Origin.y * cellSize )
    for( cell in cells ) {
      g.fillRect( cell.X * cellSize + borderSize, cell.Y * cellSize + borderSize, cellSize - borderSize, cellSize - borderSize )
    }
    g.translate( -(Origin.x * cellSize), -(Origin.y * cellSize) )
  }

  function cellAt( loc: Point ) : Cell {
    var cellLoc = cellLocationAt( loc )
    if( !Model.Infinite ) {
      if( cellLoc.x < 0 || cellLoc.y < 0 || cellLoc.x >= XCells || cellLoc.y >= YCells ) {
        return null
      }
    }
    return Model.getCell( cellLoc.x, cellLoc.y )
  }

  function cellLocationAt( loc: Point ) : Point {
    loc = translate( loc )
    var x = loc.x / CellSize
    var y = loc.y / CellSize
    return new( x, y )
  }

  function translate( pt: Point ) : Point {
    return new( pt.x - OriginPoint.x, pt.y - OriginPoint.y )
  }

  function enableScrolling( enable: boolean ) {
    if( enable && !_scroller.Enabled ) {
      _scroller.Enabled = true
    }
    else {
      _scroller.Enabled = false
    }
  }

  function enableSelection( enable: boolean ) {
    if( enable && !_selector.Enabled ) {
      _selector.Enabled = true
    }
    else {
      _selector.Enabled = false
    }
  }

  function enableDrawing( enable: boolean ) {
    if( enable && !_drawHandler.Enabled ) {
      _drawHandler.Enabled = true
    }
    else {
      _drawHandler.Enabled = false
    }
  }

  function addResetListener( listener( board: Board ) ) {
    _listeners.add( listener )
  }
  function fireReset() {
    _listeners.each( \l -> l( this ))
  }


  property set Frequency( value: int ) {
    _frequency = value
    _freqListeners.each( \l -> l() )
  }
  function addFrequencyListener( listener() ) {
    _freqListeners.add( listener )
  }

  function postModelChange( change() ) {
    if( _modelProcessor == null ) {
      change()
    }
    else {
      _modelProcessor.submit( change )
    }
  }

  private function getPopulatorCells( size: Dimension, populator: Populator ) : Collection {
    if( populator.LiveCells != null ) {
      return populator.LiveCells
    }
    return randomize( size, populator.LiveCells )
  }

  private function randomize( size: Dimension, cells: Collection ) : List {
    var live = new ArrayList()
    for( x in 0..|size.width ) {
      for( y in 0..|size.height ) {
        if( Math.random() * 100 < Density ) {
          live.add( new( x, y ) )
        }
      }
    }
    return live
  }

  private class ModelProcessor {
    var _executor: ExecutorService
    var _shutdown: boolean as Shutdown

    construct() {
      _executor = Executors.newSingleThreadExecutor()
    }

    function start() {
      _executor.submit( \-> process( System.currentTimeMillis() ) )
    }

    function submit( change() ) {
      _executor.submit( new Change( change ) )
    }

    function shutdown() {
      _shutdown = true
      _executor.shutdown()
    }

    function process( last: long ) {
      if( _shutdown ) {
        return
      }

      var current = System.currentTimeMillis()
      if( current - last >= Frequency ) {
        last = current
        _model.process()
        _buffer = paintComponentViaBuffer()
        repaint()
      }
      else {
        using( this as IMonitorLock ) {
          wait( Math.max( Frequency/10, 1 ) )
        }
      }
      _executor.submit( \-> process( last ) )
    }

    class Change implements Runnable {
      var _change ()

      construct( change () ) {
        _change = change
      }

      override function run() {
        if( !Shutdown ) {
          _change()
        }
      }
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy