当前位置: 首页 > 工具软件 > scala-steward > 使用案例 >

Scala - scala examples : Scala spreasheet

靳高明
2023-12-01

In this post, we will check an example that create a SCells Spreadsheet. 

So we first check on the visual framework. 

The Visual framework

extends the ScrollPane, which gives it a Scroll-bars at the bottom and right. It contains two sub-component named Table and rowHeader.
Table display the data, while the rowHeader component contains the row-number headers at the left of the spreadsheet. 
the GUI code that forms the skeleton is as follow. 

// the code as below 
package org.stairwaybook.scelles

import swing._

class Spreadsheet(val height : Int, val width : Int) 
	extends ScrollPane { 
  val table = new Table(height, width) {
    rowHeight = 25
    autoResizeMode = Table.AutoResizeMode.Off
    showGrid = true
    gridColor = new java.awt.Color(150, 150, 150)
  }
  
  val rowHeader = 
    new ListView((0 until height) map (_.toString)) {
      fixedCellWidth = 30
      fixedCellHeight = table.rowHeight
  }
  
  viewportView = table
  rowHeaderView = rowHeader
}
To try this rudimentary spreadsheet out, here is the main class. 

// the code as below 
package org.stairwaybook.scelles

import swing._

object Main extends SimpleGUIApplication { 
  def top = new MainFrame { 
    title  = "ScalaSheet"
    contents = new Spreadsheet(100, 20)
  }
}

Disconnectng data entry and display 

A good UI code, may have demands that the code separation between UI and model. This is the same in Scala as well in elsewhere. 

First, we need to refactor the UI code, we want to show the contents with meaning value when it is beeing edited or when it is just display the result. so we will use the renderedComponent method, it is an override method, and it works as follow. 

// the code as below 
package org.stairwaybook.scelles

import swing._

class Spreadsheet(val height : Int, val width : Int) 
	extends ScrollPane { 
  // the data
  val cellMode = new Model(height, width)
  import cellMode._
  
  
  val table = new Table(height, width) {
    rowHeight = 25
    autoResizeMode = Table.AutoResizeMode.Off
    showGrid = true
    gridColor = new java.awt.Color(150, 150, 150)
    
    override def rendererComponent(isSelected : Boolean, hasFocus : Boolean, row : Int, column : Int) : Component = {
	    if (hasFocus) new TextField(userData(row, column))
	    else 
	      new Label(cells(row)(column).toString) {
	      xAlignment = Alignment.Right
	    }
	}
	  
	  def userData(row : Int, column : Int) : String = { 
	    val v = this(row, column)
	    if (v == null) "" else v.toString
	}
  
  }
  
  val rowHeader = 
    new ListView((0 until height) map (_.toString)) {
      fixedCellWidth = 30
      fixedCellHeight = table.rowHeight
  }
  
  viewportView = table
  rowHeaderView = rowHeader
}
as we see that we uses userData, and that uses cells from the Model class. here is the code of the Model class. 

package org.stairwaybook.scelles

import swing._

class Model(val height : Int, val width: Int) {
  case class Cell(row :Int, column : Int)
  
  val cells = new Array[Array[Cell]](height, width)
  
  for (i <- 0 until height; j <- 0 until width) 
    cells(i)(j) = new Cell(i, j)
}
now we have basic Spreadsheet, but it does not allow us to do useful things such as computation, it just display string as you hasve inputed it. 

Formulas

In reality , a spread sheet cell holds two things : An actual value and a formula to compute this value. there are three types of formulas in a spreadsheets

1. Numeric  value such as 1.22, -3, or 0 
2. Textual value such as Annual sales, Deprecation, or total
3. Formula that compute a new value from the content of cells, such as "=add(A1, B2)" or "=sum((mul(2, A2), C1: D16)"

Coord       for cell coordinates such as A3
Range       for cell ranges such as A3:B17
Number      for floating-point numbers such as 3.1415
Textual     for textual labels such as Deprecation
Application for function applications such as sum(A1, A2).


so we have defined the following Formula classes. 

package org.stairwaybook.scelles

trait Formula

case class Coord(row :Int, column : Int) extends Formula {
  override def toString = ('A' + column).toChar .toString + row
}

case class Range(c1 : Coord, c2: Coord) extends Formula { 
  override def toString = c1.toString + ":" + c2.toString
}

case class Number(value : Double) extends Formula { 
  override def toString = value.toString
}

case class Textual(value : String) extends Formula { 
  override def toString = value
}

// stands for a formula which should have a function name and a list of arguments. 
case class Application(function: String, arguments : List[Formula]) extends Formula {
  override def toString = function + arguments.mkString("(", ",", ")")
}

object Empty extends Textual("")

Parsing formulas 

we need to have a way to parse the formula into a formula tree. 

import scala.util.parsing.combinator._

object FormulaParsers extends RegexParsers { 
  def ident : Parser[String] = """[a-zA-Z_]\w*""".r
  def decimal : Parser[String] = """-?\d+(\.\d*)?""".r
  // next, we will need to capture Cell
  def cell : Parser[Coord] = 
    """[a-zA-Z_]\d+""".r ^^ { s =>  
    val column = s.charAt(0).toUpper - 'A'
    val row = s.substring(1).toInt
    Coord(row, column)
  }
  
  def range: Parser[Range] = 
    cell ~":"~cell ^^ { 
    case c1~":"~c2 => Range(c1, c2)
  }
  
  def number : Parser[Number] = 
    decimal ^^ (d => Number(d.toDouble))
    
  def application : Parser[Application] = 
    ident ~ "(" ~ repsep(expr, ",") ~ ")" ^^ { 
      case f ~ "(" ~ ps ~ ")" => Application(f, ps)
    }
 
  def expr : Parser[Formula] = 
    range | cell | number | application 
    
  def textual : Parser[Textual] = 
    """[^=].*""".r ^^ Textual // this is very neat... , you can directly apply the Textual (as Function1) on the ParserReult[T] on the Textual case class
  
  def formula : Parser[Formula] = 
    number | textual | "=" ~> expr
    
  def parse(input  : String ) : Formula = 
    parseAll(formula, input) match { 
      case Success (e, _) => e
      case f : NoSuccess => Textual("[" + f.msg + "]")
    }
} // end FormulaParsers
this is using the Combinator framework . 

To support the formula on Cells, here is the new version of Model , which has the formula members. 

import swing._

class Model(val height : Int, val width: Int) {
  case class Cell(row :Int, column : Int) {
    var formula : Formula = Empty 
    override def toString = formula.toString
  }
  
  val cells = new Array[Array[Cell]](height, width)
  
  for (i <- 0 until height; j <- 0 until width) 
    cells(i)(j) = new Cell(i, j)
}
now with the Formula, we still display the string just as the formula's string representation. 

Evaluation 

we create a new Evaluation trait, which has the evaluate method. 

trait Evaluator { this : Model => 
  def evaluate(e: Formula) : Double = try { 
    e match { 
      case Coord(row, column) => 
        cells(row)(column).value 
      case Number(v) => 
        v
      case Textual(_) => 
        0
      case Application(function, arguments) => 
        val argvals = arguments flatMap evalList 
        operations(function)(argvals)
    }
  } catch {
    case ex : Exception => Double.NaN
  }

  type Op = List[Double] => Double
  val operations = new collection.mutable.HashMap[String, Op]
  
  private def evalList(e: Formula) : List[Double] = e match { 
    case Range(_, _) => references(e) map (_.value)
    case _ => List(evaluate(e)) // the recursion goes here
  }
  
  def references(e : Formula) : List[Cell] = e match { 
    case Coord (row, column) => 
      List(cells(row)(column))
    case Range(Coord(r1, c1), Coord(r2, c2)) => 
      for (row <- (r1 to r2).toList; column <- c1 to c2)
        yield cells(row)(column)
    case Application(function, arguments) =>
      arguments flatMap references
    case _ => 
      List()
  }
} // end Evaluator
the key is the Application ,which calles operations, which is a map from function name to the function's application. 

type Op = List[Double] => Double
  val operations = new collection.mutable.HashMap[String, Op]
and to support evaluation on nested arguments, we have evalList. 

private def evalList(e: Formula) : List[Double] = e match { 
    case Range(_, _) => references(e) map (_.value)
    case _ => List(evaluate(e)) // the recursion goes here
  }
and in order to support argument such as [C1:C5] or C1 and others types, we define a method references , which can turn a formula classes into a cell references. as follow. 

def references(e : Formula) : List[Cell] = e match { 
    case Coord (row, column) => 
      List(cells(row)(column))
    case Range(Coord(r1, c1), Coord(r2, c2)) => 
      for (row <- (r1 to r2).toList; column <- c1 to c2)
        yield cells(row)(column)
    case Application(function, arguments) =>
      arguments flatMap references
    case _ => 
      List()
  }

equally, when we 

Operations libraries

To actually convert the formula into the operations, so we can really convert to real calculation. 

// the trait populate the operations table during its initialization. 
trait Arithmetic { this : Evaluator => 
    operations += (
      "add" -> { case List(x, y) => x + y },
      "sub" -> { case List(x, y) => x - y },
      "div" -> { case List(x, y) => x / y },
      "mul" -> { case List(x, y) => x * y },
      "mod" -> { case List(x, y) => x % y },
      "sum" -> { xs => (0.0 /: xs) (_ + _) },
      "prod" -> { xs => (1.0 /: xs) (_ * _) }
    )
}
this trait does not export any exported members, we need to revise the Model class, here ist he model clases that introduced the Formula. 

To support evaluation, Model now mix in Evaluator trait to support evaluate 

import swing._

class Model(val height : Int, val width: Int) 
      extends Evaluator with Arithmetic {
  case class Cell(row :Int, column : Int) {
    var formula : Formula = Empty 
    
    def value = evaluate(formula)
    
    override def toString = formula match { 
      case Textual(s) => s
      case _ => value.toString
    }
  }
  
  val cells = new Array[Array[Cell]](height, width)
  
  for (i <- 0 until height; j <- 0 until width) 
    cells(i)(j) = new Cell(i, j)
}

Change Propagation 

so we have all the meat and potatos, but we don't have change propagations, if one of hte dependent cell changes, we are not being informed and the cell is not updated. 

we need to extend the Model class. This we extend our Cell case class to let it extend from the Swing.event.Publisher class and we redefine the formula class to let it have the subscription and unsubcription wired up . as follow. 

class Model(val height : Int, val width: Int) 
      extends Evaluator with Arithmetic {
 
  case class Cell(row :Int, column : Int) extends Publisher {
    private var f : Formula = Empty
    def formula_=(f : Formula) { 
      
      for (c <- references(formula)) deafTo(c)
      this.f = f
      for (c <- references(formula)) listenTo(c)
      value = evaluate(f)
    }
    def formula : Formula = f 
    
    private var v : Double = 0
    def value = evaluate(formula)
    def value_= (w: Double) {
      if (!(v == w || v.isNaN && w.isNaN)) {
        v = w
        publish(ValueChanged(this))
      }
    }
    
    override def toString = formula match { 
      case Textual(s) => s
      case _ => value.toString
    }
    
    // reactions is from the Publisher classes
    reactions += {
      case ValueChanged(_) => value = evaluate(formula)
    }
  } // end class Cell
  
  case class ValueChanged(cell : Cell) extends event.Event
  
  val cells = new Array[Array[Cell]](height, width)
  
  for (i <- 0 until height; j <- 0 until width) 
    cells(i)(j) = new Cell(i, j)
}
in this code, we define a new event called ValueChanged, when a cell is given a different value, the event will be fired. and we need as well to handle the ValueChanged event (this is because some upstream cells when it call publich(ValueChanged(this)), the downstream should react and reevaluate the formula. 

Now, the cell has changed, we need as well to extend the Spreadsheet class to 1. listen to all cells 2. create a handler for the ValueChanged event to redraw content for the cell. the key is to call cellUpdate(row, cell). here is the new Spreadsheet class. 

import swing._
import swing.event._

class Spreadsheet(val height : Int, val width : Int) 
	extends ScrollPane { 
  // the data
  val cellMode = new Model(height, width)
  import cellMode._
  
  
  val table = new Table(height, width) {
    rowHeight = 25
    autoResizeMode = Table.AutoResizeMode.Off
    showGrid = true
    gridColor = new java.awt.Color(150, 150, 150)
    
    override def rendererComponent(isSelected : Boolean, hasFocus : Boolean, row : Int, column : Int) : Component = {
	    if (hasFocus) new TextField(userData(row, column))
	    else 
	      new Label(cells(row)(column).toString) {
	      xAlignment = Alignment.Right
	    }
	}
	  
	def userData(row : Int, column : Int) : String = { 
	    val v = this(row, column)
	    if (v == null) "" else v.toString
	}
	  
	reactions += {
	  case TableUpdated(table, rows, column) => 
	    for (row <- rows) 
	      cells(row)(column).formula = FormulaParsers.parse(userData(row, column))
	  case ValueChanged(cell) => 
	    updateCell(cell.row, cell.column) // redraw the cell with the updateCell call 
	}
	
	for (row <- cells; cell <- row) listenTo(cell) // wire up and listen to all the cells 
  
  }
  
  val rowHeader = 
    new ListView((0 until height) map (_.toString)) {
      fixedCellWidth = 30
      fixedCellHeight = table.rowHeight
  }
  
  viewportView = table
  rowHeaderView = rowHeader
}

The main class

To run the code, we need a Main class, it is as follow. 

import swing._

object Main extends SimpleGUIApplication { 
  def top = new MainFrame { 
    title  = "ScalaSheet"
    contents = new Spreadsheet(100, 20)
  }
}
run the following code 
scala Main

and if you have the package statement above, you should run it like this:
scala org.stairwaybook.scelles.Main
or just 

Main.main(Array[String]())
if you are running from the Eclipse interactive shel. 

转载于:https://my.oschina.net/u/854138/blog/142974

 类似资料: