Wednesday, December 23, 2015

OpenCV 3.1 with Java Support on OsX El Capitan V10.11.1

In this blog post I'll describe what you have to do in order to build OpenCV 3.1 with Java Support on a Mac with El Capitan.

Painting the tanker 'Borgsten'


(If you don't want to do source builds and you don't have all necessary tools installed, there are also some alternatives like brew or macports available. In fact, I blogged about using this approach some years ago.)

Compiling OpenCV 3.1


First, you will have to download the sources directly from the official opencv.org site. It is a rather large source distribution, so the download may take a while.

Unpack the source ball into a directory, I did it like this:

~/opencv/ (1)
~/opencv/opencv-3.1.0/ (2)

I created a directory named "build"

~/opencv/build/ (3)

(1) ... a directory which includes all opencv versions, so you have everything in place if you want to follow more than one version and you don't use git for doing this
(2) ... the sources like you've downloaded them
(3) ... the build directory which will be cluttered up by cmake with all kinds of stuff

Change to the directory (3) and issue following command:

cmake ../opencv-3.1.0/

This command will investigate what is already installed on your system. It will do its best to download stuff it will need to compile what is possible. Certain tools or libraries you still need to download manually, though. This command will finish in a minute and end with a thorough report of how the build is configured. Check this list and see if your feature is configured aswell. To my surprise and as opposed to the last try with version 2.4.6 the java bindings are activated per default which is great. Huge thanks for the build guys of this library.

The cmake command is setup in a way that in theory you just have to invoke another command, named

make

which will invoke by itself all programs which contribute to the final build artifacts. This means it will invoke c++ compilers, c compilers, helper programs and whatnot. This command will also take some time to complete.

After waiting for some minutes the command exits successfully and in the bin folder there is a jar file which contains all bindings to the opencv library. This API is then the entrance for using the excellent library from your Java applications.

Using OpenCV 3.1.0 with Maven


After having compiled opencv like described above, several directories and files had been created. Amongst those, in the 'bin' folder, there is a file called

opencv-310.jar

This jar file can be installed in the local maven repository like this:

mvn install:install-file 
       -Dfile=opencv-310.jar 
    -DgroupId=org.opencv 
 -DartifactId=opencv-java 
    -Dversion=3.1.0 
  -Dpackaging=jar

From now on you should be able to reference the library with maven in the pom.xml like this:

<dependency>
  <groupId>org.opencv</groupId>
  <artifactId>opencv-java</artifactId>
  <version>3.1.0</version>
</dependency>

Still, in order to get OpenCV to work, you'll need the native part of the library which contains "the real thing" - the Java code is just a thin wrapper which makes it more convenient to call the native code. Those files are generated in the "lib" folder:

libopencv_objdetect.3.1.0.dylib
libopencv_calib3d.3.1.0.dylib
libopencv_objdetect.3.1.dylib
libopencv_calib3d.3.1.dylib
...

One of them is called "libopencv_java310.so" - this library contains the native code for the api which is accessible through the jar file. Make sure this library is loaded before you call the first time into the jar file via 

System.load(new File("/path/to/libopencv_java310.so"))

This should help you start developing against the OpenCV API with Java and MacOsX.

It may happen however, like mentioned, that you will need to install additional software packages such that the compilation step of OpenCV is successful for you.

Friday, December 11, 2015

JavaFX Tableview

In this very short blogpost I'll present you an example for the usage of the JavaFX tableview component.

A JavaFX TableView Screenshot

Yes, the app does not much - it displays some random data in a javafx TableView control. 

The interesting thing maybe is how the table columns are initialized:


I think it is quite obvious what is going on - the columns are somehow connected to the properties of ... erm ... the row entries.

More often than not this is what you want from a tableview, just display some data. 

Consulting javadocs for TableView or TableColumn never hurts, but sometimes it can be a little bit hard to get the big picture, especially when you don't have much experience with JavaFX.

The abstractions provided here help you to display some data quickly (in terms of development time) and can be reused in various scenarios. A self contained project is hosted on my github site, but for the impatient i'll repost the essential parts here as a gist:




package net.ladstatt.fx.tableview
import java.net.URL
import java.util.ResourceBundle
import javafx.application.Application
import javafx.beans.property.{SimpleDoubleProperty, SimpleIntegerProperty, SimpleStringProperty}
import javafx.beans.value.ObservableValue
import javafx.collections.{FXCollections, ObservableList}
import javafx.fxml._
import javafx.scene.control.{TableColumn, TableView}
import javafx.scene.{Parent, Scene}
import javafx.stage.Stage
import javafx.util.Callback
import scala.collection.JavaConversions
import scala.util.Random
import scala.util.control.NonFatal
/**
* Shows a way to use a JavaFX TableView with Scala
*/
object TableViewExample {
def main(args: Array[String]) {
Application.launch(classOf[TableViewApp], args: _*)
}
}
/**
* Setup for the javafx app
*/
class TableViewApp extends javafx.application.Application {
val loader = new FXMLLoader(getClass.getResource("TableView.fxml"))
override def start(stage: Stage): Unit =
try {
stage.setTitle("TableView Example App")
loader.load[Parent]()
stage.setScene(new Scene(loader.getRoot[Parent]))
stage.show()
} catch {
case NonFatal(e) => e.printStackTrace()
}
}
/**
* domain object
*/
case class Article(id: Int, name: String, price: Double)
/**
* domain object, but usable with javafx
*/
class MutableArticle {
val idProperty: SimpleIntegerProperty = new SimpleIntegerProperty()
val nameProperty: SimpleStringProperty = new SimpleStringProperty()
val priceProperty: SimpleDoubleProperty = new SimpleDoubleProperty()
def setId(id: Int) = idProperty.set(id)
def setName(name: String) = nameProperty.set(name)
def setPrice(price: Double) = priceProperty.set(price)
}
/**
* companion object to get a better initialisation story
*/
object MutableArticle {
def apply(a: Article): MutableArticle = {
val ma = new MutableArticle
ma.setId(a.id)
ma.setName(a.name)
ma.setPrice(a.price)
ma
}
}
/**
* util functions to bridge the javafx / scala gap
*/
object JfxUtils {
type TCDF[S, T] = TableColumn.CellDataFeatures[S, T]
import JavaConversions._
def mkObservableList[T](collection: Iterable[T]): ObservableList[T] = {
FXCollections.observableList(new java.util.ArrayList[T](collection))
}
private def mkCellValueFactory[S, T](fn: TCDF[S, T] => ObservableValue[T]): Callback[TCDF[S, T], ObservableValue[T]] = {
new Callback[TCDF[S, T], ObservableValue[T]] {
def call(cdf: TCDF[S, T]): ObservableValue[T] = fn(cdf)
}
}
def initTableViewColumnCellValueFactory[S, T](tc: TableColumn[S, T], f: S => Any): Unit = {
tc.setCellValueFactory(mkCellValueFactory(cdf => f(cdf.getValue).asInstanceOf[ObservableValue[T]]))
}
}
/**
* simulates a database for example
*/
object DataSource {
val data =
(1 to 1000) map {
case i => Article(i, s"name $i", Random.nextDouble() * i)
}
}
/**
* populates data in the table view controller
*/
class TableViewAppController extends Initializable {
import JfxUtils._
type ArticleTC[T] = TableColumn[MutableArticle, T]
@FXML var tableView: TableView[MutableArticle] = _
@FXML var columnId: ArticleTC[Int] = _
@FXML var columnName: ArticleTC[String] = _
@FXML var columnPrice: ArticleTC[Double] = _
val mutableArticles = mkObservableList(DataSource.data.map(MutableArticle(_)))
/**
* provide a table column and a generator function for the value to put into
* the column.
*
* @tparam T the type which is contained in the property
* @return
*/
def initTableViewColumn[T]: (ArticleTC[T], (MutableArticle) => Any) => Unit =
initTableViewColumnCellValueFactory[MutableArticle, T]
override def initialize(location: URL, resources: ResourceBundle): Unit = {
tableView.setItems(mutableArticles)
initTableViewColumn[Int](columnId, _.idProperty)
initTableViewColumn[String](columnName, _.nameProperty)
initTableViewColumn[Double](columnPrice, _.priceProperty)
}
}
Thanks for reading.