├── .cache ├── .classpath ├── .gitignore ├── .project ├── .settings ├── org.eclipse.core.resources.prefs └── org.scala-ide.sdt.core.prefs ├── Procfile ├── bin └── application.conf ├── build.sbt ├── policy ├── project ├── build.properties ├── cdn.scala ├── plugins.sbt └── translate-zh.scala ├── readme.md ├── sbt ├── sbt ├── sbt-launch-0.11.3-2.jar └── sbt.cmd ├── scala-tour-zh.md ├── scala-tour.md ├── src ├── main │ ├── resources │ │ └── application.conf │ └── scala │ │ └── com │ │ └── yankay │ │ └── scalaTour │ │ ├── ScalaScriptRunner.scala │ │ ├── Web.scala │ │ └── script │ │ └── TaskMem.scala └── test │ └── scala │ ├── com │ └── yankay │ │ └── scalaTour │ │ ├── ScalaScriptCompilerTest.scala │ │ └── script │ │ └── TaskMem.scala │ └── test.scala └── webapp ├── apple-touch-icon.png ├── css ├── bootstrap-responsive.css ├── bootstrap-responsive.min.css ├── bootstrap.css ├── bootstrap.min.css ├── codemirror.css ├── elegant.css ├── impress-demo.css ├── scala-tour.css └── solarized.css ├── favicon.png ├── img ├── Lambda_lc.svg.png ├── glyphicons-halflings-white.png ├── glyphicons-halflings.png ├── glyphicons_065_tag.png ├── glyphicons_221_unshare.png └── logo.png ├── index-zh.html ├── index.html └── js ├── bootstrap.js ├── bootstrap.min.js ├── clike.js ├── codemirror.js ├── codemirror.min.js ├── go.js ├── impress.js ├── jquery.js ├── jquery.min.js ├── matchbrackets.js └── scala-tour.js /.cache: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/.cache -------------------------------------------------------------------------------- /.classpath: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .idea 2 | .idea_modules 3 | target 4 | project/boot 5 | project/target 6 | project/plugins/target 7 | sftp-config.json 8 | -------------------------------------------------------------------------------- /.project: -------------------------------------------------------------------------------- 1 | 2 | scala-tour 3 | 4 | 5 | org.scala-ide.sdt.core.scalabuilder 6 | 7 | 8 | 9 | org.scala-ide.sdt.core.scalanature 10 | org.eclipse.jdt.core.javanature 11 | 12 | -------------------------------------------------------------------------------- /.settings/org.eclipse.core.resources.prefs: -------------------------------------------------------------------------------- 1 | eclipse.preferences.version=1 2 | encoding/=UTF-8 3 | -------------------------------------------------------------------------------- /.settings/org.scala-ide.sdt.core.prefs: -------------------------------------------------------------------------------- 1 | Xcheck-null=false 2 | Xcheckinit=false 3 | Xdisable-assertions=false 4 | Xelide-below=-2147483648 5 | Xexperimental=false 6 | Xfatal-warnings=false 7 | Xfuture=false 8 | Xlog-implicits=false 9 | Xno-uescape=false 10 | Xpluginsdir=C\:\\Program Files\\eclipse\\configuration\\org.eclipse.osgi\\bundles\\245\\1\\.cp\\lib 11 | Ybuild-manager-debug=false 12 | Yno-generic-signatures=false 13 | Yno-imports=false 14 | Ypresentation-debug=false 15 | Ypresentation-delay=0 16 | Ypresentation-verbose=false 17 | Yrecursion=0 18 | Yself-in-annots=false 19 | Ystruct-dispatch=poly-cache 20 | Ywarn-dead-code=false 21 | buildmanager=sbt 22 | compileorder=Mixed 23 | debugIncremental=true 24 | deprecation=false 25 | eclipse.preferences.version=1 26 | explaintypes=false 27 | formatter.alignParameters=false 28 | formatter.alignSingleLineCaseStatements=false 29 | formatter.alignSingleLineCaseStatements.maxArrowIndent=40 30 | formatter.compactControlReadability=false 31 | formatter.compactStringConcatenation=false 32 | formatter.doubleIndentClassDeclaration=false 33 | formatter.formatXml=true 34 | formatter.indentLocalDefs=false 35 | formatter.indentPackageBlocks=true 36 | formatter.indentSpaces=2 37 | formatter.indentWithTabs=false 38 | formatter.multilineScaladocCommentsStartOnFirstLine=false 39 | formatter.placeScaladocAsterisksBeneathSecondAsterisk=false 40 | formatter.preserveDanglingCloseParenthesis=false 41 | formatter.preserveSpaceBeforeArguments=false 42 | formatter.rewriteArrowSymbols=false 43 | formatter.spaceBeforeColon=false 44 | formatter.spaceInsideBrackets=false 45 | formatter.spaceInsideParentheses=false 46 | formatter.spacesWithinPatternBinders=true 47 | g=vars 48 | no-specialization=false 49 | nowarn=false 50 | optimise=false 51 | organizeimports.expandcollapse=expand 52 | organizeimports.groups=java$scala$org$com 53 | organizeimports.scalapackage=false 54 | organizeimports.wildcards=scalaz$scalaz.Scalaz 55 | scala.compiler.useProjectSettings=true 56 | stopBuildOnError=true 57 | syntaxColouring.bracket.backgroundColourEnabled=false 58 | syntaxColouring.bracket.bold=false 59 | syntaxColouring.bracket.colour=0,0,0 60 | syntaxColouring.bracket.enabled=true 61 | syntaxColouring.bracket.italic=false 62 | syntaxColouring.bracket.underline=false 63 | syntaxColouring.default.backgroundColourEnabled=false 64 | syntaxColouring.default.bold=false 65 | syntaxColouring.default.colour=0,0,0 66 | syntaxColouring.default.enabled=true 67 | syntaxColouring.default.italic=false 68 | syntaxColouring.default.underline=false 69 | syntaxColouring.keyword.backgroundColourEnabled=false 70 | syntaxColouring.keyword.bold=true 71 | syntaxColouring.keyword.colour=127,0,85 72 | syntaxColouring.keyword.enabled=true 73 | syntaxColouring.keyword.italic=false 74 | syntaxColouring.keyword.underline=false 75 | syntaxColouring.operator.backgroundColourEnabled=false 76 | syntaxColouring.operator.bold=false 77 | syntaxColouring.operator.colour=0,0,0 78 | syntaxColouring.operator.enabled=true 79 | syntaxColouring.operator.italic=false 80 | syntaxColouring.operator.underline=false 81 | syntaxColouring.scaladoc.backgroundColourEnabled=false 82 | syntaxColouring.scaladoc.bold=false 83 | syntaxColouring.scaladoc.colour=63,95,191 84 | syntaxColouring.scaladoc.enabled=true 85 | syntaxColouring.scaladoc.italic=false 86 | syntaxColouring.scaladoc.underline=false 87 | syntaxColouring.singleLineComment.backgroundColourEnabled=false 88 | syntaxColouring.singleLineComment.bold=false 89 | syntaxColouring.singleLineComment.colour=63,127,95 90 | syntaxColouring.singleLineComment.enabled=true 91 | syntaxColouring.singleLineComment.italic=false 92 | syntaxColouring.singleLineComment.underline=false 93 | syntaxColouring.string.backgroundColourEnabled=false 94 | syntaxColouring.string.bold=false 95 | syntaxColouring.string.colour=42,0,255 96 | syntaxColouring.string.enabled=true 97 | syntaxColouring.string.italic=false 98 | syntaxColouring.string.underline=false 99 | syntaxColouring.xml.attributeName.backgroundColourEnabled=false 100 | syntaxColouring.xml.attributeName.bold=false 101 | syntaxColouring.xml.attributeName.colour=127,0,127 102 | syntaxColouring.xml.attributeName.enabled=true 103 | syntaxColouring.xml.attributeName.italic=false 104 | syntaxColouring.xml.attributeName.underline=false 105 | syntaxColouring.xml.attributeValue.backgroundColourEnabled=false 106 | syntaxColouring.xml.attributeValue.bold=false 107 | syntaxColouring.xml.attributeValue.colour=42,0,255 108 | syntaxColouring.xml.attributeValue.enabled=true 109 | syntaxColouring.xml.attributeValue.italic=true 110 | syntaxColouring.xml.attributeValue.underline=false 111 | syntaxColouring.xml.comment.backgroundColourEnabled=false 112 | syntaxColouring.xml.comment.bold=false 113 | syntaxColouring.xml.comment.colour=63,85,191 114 | syntaxColouring.xml.comment.enabled=true 115 | syntaxColouring.xml.comment.italic=false 116 | syntaxColouring.xml.comment.underline=false 117 | syntaxColouring.xml.equals.backgroundColourEnabled=false 118 | syntaxColouring.xml.equals.bold=false 119 | syntaxColouring.xml.equals.colour=0,0,0 120 | syntaxColouring.xml.equals.enabled=true 121 | syntaxColouring.xml.equals.italic=false 122 | syntaxColouring.xml.equals.underline=false 123 | syntaxColouring.xml.processingInstruction.backgroundColourEnabled=false 124 | syntaxColouring.xml.processingInstruction.bold=false 125 | syntaxColouring.xml.processingInstruction.colour=0,128,128 126 | syntaxColouring.xml.processingInstruction.enabled=true 127 | syntaxColouring.xml.processingInstruction.italic=false 128 | syntaxColouring.xml.processingInstruction.underline=false 129 | syntaxColouring.xml.tagDelimiter.backgroundColourEnabled=false 130 | syntaxColouring.xml.tagDelimiter.bold=false 131 | syntaxColouring.xml.tagDelimiter.colour=0,128,128 132 | syntaxColouring.xml.tagDelimiter.enabled=true 133 | syntaxColouring.xml.tagDelimiter.italic=false 134 | syntaxColouring.xml.tagDelimiter.underline=false 135 | syntaxColouring.xml.tagName.backgroundColourEnabled=false 136 | syntaxColouring.xml.tagName.bold=false 137 | syntaxColouring.xml.tagName.colour=63,127,127 138 | syntaxColouring.xml.tagName.enabled=true 139 | syntaxColouring.xml.tagName.italic=false 140 | syntaxColouring.xml.tagName.underline=false 141 | target=jvm-1.6 142 | unchecked=false 143 | verbose=false 144 | -------------------------------------------------------------------------------- /Procfile: -------------------------------------------------------------------------------- 1 | web: target/start com.yankay.scalaTour.Web 2 | -------------------------------------------------------------------------------- /bin/application.conf: -------------------------------------------------------------------------------- 1 | remote { 2 | akka { 3 | 4 | actor { 5 | provider = "akka.remote.RemoteActorRefProvider" 6 | } 7 | remote { 8 | transport = "akka.remote.netty.NettyRemoteTransport" 9 | netty { 10 | hostname = "127.0.0.1" 11 | port = 2552 12 | } 13 | } 14 | 15 | } 16 | } -------------------------------------------------------------------------------- /build.sbt: -------------------------------------------------------------------------------- 1 | import com.typesafe.sbt.SbtStartScript 2 | 3 | seq(SbtStartScript.startScriptForClassesSettings: _*) 4 | 5 | net.virtualvoid.sbt.graph.Plugin.graphSettings 6 | 7 | name := "scala-tour" 8 | 9 | version := "1.0" 10 | 11 | scalaVersion := "2.10.1" 12 | 13 | libraryDependencies ++= Seq( 14 | "org.scala-lang" % "scala-actors" % "2.10.1", 15 | "org.scala-lang" % "scala-compiler" % "2.10.1", 16 | "com.typesafe.akka" %% "akka-testkit" % "2.1.2", 17 | "com.typesafe.akka" %% "akka-actor" % "2.1.2", 18 | "com.typesafe.akka" %% "akka-remote" % "2.1.2", 19 | "org.eclipse.jetty" % "jetty-server" % "8.1.10.v20130312", 20 | "org.eclipse.jetty" % "jetty-servlet" % "8.1.10.v20130312", 21 | "org.eclipse.jetty" % "jetty-servlets" % "8.1.10.v20130312", 22 | "org.json4s" %% "json4s-jackson" % "3.2.2", 23 | "commons-beanutils" % "commons-beanutils" % "1.8.3", 24 | "org.specs2" % "specs2_2.10" % "1.14" 25 | ) -------------------------------------------------------------------------------- /policy: -------------------------------------------------------------------------------- 1 | /* AUTOMATICALLY GENERATED ON Sat Aug 10 20:31:30 PDT 2013*/ 2 | /* DO NOT EDIT */ 3 | 4 | grant { 5 | permission java.io.FilePermission "${user.home}/.ivy2/cache/com.typesafe", "read"; 6 | permission java.io.FilePermission "${user.home}/.ivy2/cache/com.typesafe/-", "read"; 7 | permission java.io.FilePermission "${user.home}/.ivy2/cache/com.typesafe.akka", "read"; 8 | permission java.io.FilePermission "${user.home}/.ivy2/cache/com.typesafe.akka/-", "read"; 9 | permission java.io.FilePermission "./target/specs2-reports", "read, write"; 10 | permission java.io.FilePermission "./target/specs2-reports/-", "read, write"; 11 | 12 | permission java.io.FilePermission "/tmp", "read"; 13 | 14 | permission java.io.FilePermission "./bin/application.conf", "read"; 15 | permission java.io.FilePermission "/usr/lib/jvm/java-*/-", "read"; 16 | permission java.io.FilePermission "/usr/lib/jvm/java-*/jre/-", "read"; 17 | permission java.io.FilePermission "/usr/java/packages/lib/ext/-", "read"; 18 | permission java.io.FilePermission "/usr/java/packages/lib/ext", "read"; 19 | permission java.io.FilePermission "/usr/share/java/-", "read"; 20 | 21 | permission java.util.PropertyPermission "java.version", "read"; 22 | permission java.util.PropertyPermission "java.vendor", "read"; 23 | permission java.util.PropertyPermission "os.name", "read"; 24 | permission java.util.PropertyPermission "os.version", "read"; 25 | permission java.util.PropertyPermission "os.arch", "read"; 26 | 27 | permission java.io.FilePermission "/proc/self/stat", "read"; 28 | 29 | permission java.net.SocketPermission "github.com:443", "connect, resolve"; 30 | permission java.net.SocketPermission "raw.github.com:443", "connect, resolve"; 31 | permission java.net.SocketPermission "scala-lang.org:80", "connect, resolve"; 32 | permission java.net.SocketPermission "127.0.0.1:2552", "connect, resolve, listen, accept"; 33 | 34 | permission java.lang.RuntimePermission "getenv.AKKA_HOME", "read"; 35 | permission java.util.PropertyPermission "akka.home", "read"; 36 | permission java.util.PropertyPermission "*", "read, write"; 37 | permission java.lang.RuntimePermission "setContextClassLoader"; 38 | permission java.lang.reflect.ReflectPermission "suppressAccessChecks"; 39 | permission java.lang.RuntimePermission "accessClassInPackage.sun.misc"; 40 | permission java.lang.RuntimePermission "accessDeclaredMembers"; 41 | permission java.lang.RuntimePermission "modifyThread"; 42 | }; 43 | 44 | -------------------------------------------------------------------------------- /project/build.properties: -------------------------------------------------------------------------------- 1 | sbt.version=0.12.2 2 | -------------------------------------------------------------------------------- /project/cdn.scala: -------------------------------------------------------------------------------- 1 | object CDN { 2 | 3 | def main(args: Array[String]) { 4 | val url = args(0) 5 | val cdn = scala.collection.mutable.Map[String, String]() 6 | cdn += ("""href="css/""" -> ("href=\"" + url + "css/").toString()) 7 | cdn += ("""src="js/""" -> ("src=\"" + url + "js/").toString()) 8 | cdn += ("""src="img/""" -> ("src=\"" + url + "img/").toString()) 9 | 10 | scala.io.Source.stdin.getLines.foreach( 11 | x => { 12 | var line = x 13 | cdn.foreach(p => line = line.replace(p._1, p._2)) 14 | println(line) 15 | }) 16 | } 17 | } -------------------------------------------------------------------------------- /project/plugins.sbt: -------------------------------------------------------------------------------- 1 | addSbtPlugin("com.typesafe.sbt" % "sbt-start-script" % "0.7.0") 2 | 3 | addSbtPlugin("com.typesafe.sbteclipse" % "sbteclipse-plugin" % "2.1.1") 4 | 5 | addSbtPlugin("net.virtual-void" % "sbt-dependency-graph" % "0.7.1") 6 | -------------------------------------------------------------------------------- /readme.md: -------------------------------------------------------------------------------- 1 | # Scala Tour 2 | 3 | A cooler tour the same as scala language 4 | 5 | [http://www.scala-tour.com/](http://www.scala-tour.com/) 6 | 7 | 8 | ## Compile And Run 9 | ``` 10 | Linux/Mac: 11 | ./sbt/sbt stage 12 | ./target/start 13 | 14 | Windows can only compile 15 | sbt\sbt stage 16 | ``` 17 | view 18 | http://localhost:8080/#/welcome 19 | 20 | ## Pure Text 21 | [Scala tour](https://github.com/yankay/scala-tour/blob/master/scala-tour.md) 22 | 23 | ## Translate 24 | The tour is can be translate to Chinese by these scripts: 25 | ``` 26 | cat webapp/index.html | scala project/translate-zh.scala > webapp/index-cn.html 27 | cat scala-tour.md | scala project/translate-zh.scala > scala-tour-cn.md 28 | 29 | ``` 30 | -------------------------------------------------------------------------------- /sbt/sbt: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | EXTRA_ARGS="" 3 | if [ "$MESOS_HOME" != "" ]; then 4 | EXTRA_ARGS="-Djava.library.path=$MESOS_HOME/lib/java" 5 | fi 6 | export SPARK_HOME=$(cd "$(dirname $0)/.."; pwd) 7 | export SPARK_TESTING=1 # To put test classes on classpath 8 | java -Xmx1200M -XX:MaxPermSize=200m $EXTRA_ARGS -jar $SPARK_HOME/sbt/sbt-launch-*.jar "$@" 9 | -------------------------------------------------------------------------------- /sbt/sbt-launch-0.11.3-2.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/sbt/sbt-launch-0.11.3-2.jar -------------------------------------------------------------------------------- /sbt/sbt.cmd: -------------------------------------------------------------------------------- 1 | @echo off 2 | set EXTRA_ARGS= 3 | if not "%MESOS_HOME%x"=="x" set EXTRA_ARGS=-Djava.library.path=%MESOS_HOME%\lib\java 4 | set SPARK_HOME=%~dp0.. 5 | java -Xmx1200M -XX:MaxPermSize=200m %EXTRA_ARGS% -jar %SPARK_HOME%\sbt\sbt-launch-0.11.3-2.jar "%*" 6 | -------------------------------------------------------------------------------- /scala-tour-zh.md: -------------------------------------------------------------------------------- 1 | # Scala 指南 2 | 3 | [http://zh.scala-修改为ur.com/](http://zh.scala-修改为ur.com/) 4 | 5 | ## 基本 6 | 7 | ### 表达式和值 8 | 9 | 在Scala中,几乎所有的语言元素都是表达式。println("hello wolrd")是一个表达式, "hello"+" world" 10 | 也是一个表达式。 11 | 12 | 可以通过val定义一个常量,亦可以通过var定义一个变量。推荐多使用常量。 13 | 14 | ``` 15 | var helloWorld = "hello" + " world" 16 | println(helloWorld) 17 | 18 | val again = " again" 19 | helloWorld = helloWorld + again 20 | println(helloWorld) 21 | ``` 22 | 23 | ### 函数是一等公民 24 | 25 | 可以使用def来定义一个函数。函数体是一个表达式。 26 | 27 | 使用Block表达式的时候,默认最后一行的返回是返回值,无需显式指定。 28 | 29 | 函数还可以像值一样,赋值给var或val。因此他也可以作为参数传给另一个函数。 30 | 31 | ``` 32 | def square(a: Int) = a * a 33 | 34 | def squareWithBlock(a: Int) = { 35 | a * a 36 | } 37 | 38 | val squareVal = (a: Int) => a * a 39 | 40 | def addOne(f: Int => Int, arg: Int) = f(arg) + 1 41 | 42 | println("square(2):" + square(2)) 43 | println("squareWithBlock(2):" + squareWithBlock(2)) 44 | println("squareVal(2):" + squareVal(2)) 45 | println("addOne(squareVal,2):" + addOne(squareVal, 2)) 46 | ``` 47 | 48 | 49 | 50 | ### 借贷模式 51 | 52 | 由于函数可以像值一样作为参数传递,所以可以方便的实现借贷模式。 53 | 54 | 这个例子是从/proc/self/stat文件中读取当前进程的pid。 55 | 56 | withScanner封装了try-finally块,所以调用者不用再close。 57 | 58 | 注:当表达式没有返回值时,默认返回Unit。 59 | 60 | ``` 61 | import scala.reflect.io.File 62 | import java.util.Scanner 63 | 64 | def withScanner(f: File, op: Scanner => Unit) = { 65 | val scanner = new Scanner(f.bufferedReader) 66 | try { 67 | op(scanner) 68 | } finally { 69 | scanner.close() 70 | } 71 | } 72 | 73 | withScanner(File("/proc/self/stat"), 74 | scanner => println("pid is " + scanner.next())) 75 | ``` 76 | 77 | 78 | ### 按名称传递参数 79 | 80 | 这个例子演示了按名称传递参数,由于有除以0,所以运行该程序会产生异常。 81 | 82 | 试着将def log(msg: String)修改为def log(msg: => String)。由按值传递修改为按名称传递后将不会产生异常。 83 | 84 | 85 | 因为log函数的参数是按名称传递,参数会等到真正访问的时候才会计算,由于logEnable = false,所以被跳过。 86 | 87 | 88 | 按名称传递参数可以减少不必要的计算和异常。 89 | 90 | 91 | ``` 92 | val logEnable = false 93 | 94 | def log(msg: String) = 95 | if (logEnable) println(msg) 96 | 97 | val MSG = "programing is running" 98 | 99 | log(MSG + 1 / 0) 100 | ``` 101 | 102 | ### 类定义 103 | 104 | 可以用class关键字来定义类。并通过new来创建类。 105 | 在定义类时可以定义字段,如firstName,lastName。这样做还可以自动生成构造函数。 106 | 可以在类中通过def定义函数。var和val定义字段。 107 | 函数名是任何字符如+,-,*,/。 108 | 例子中obama.age_=(51)的函数调用,可以简化为obama.age = 51 。 109 | obama.age()的函数调用,可以省略小括号,简化为obama.age。 110 | 111 | ``` 112 | class Person(val firstName: String, val lastName: String) { 113 | 114 | private var _age = 0 115 | def age = _age 116 | def age_=(newAge: Int) = _age = newAge 117 | 118 | def fullName() = firstName + " " + lastName 119 | 120 | override def 修改为String() = fullName() 121 | } 122 | 123 | val obama: Person = new Person("Barack", "Obama") 124 | 125 | println("Person: " + obama) 126 | println("firstName: " + obama.firstName) 127 | println("lastName: " + obama.lastName) 128 | obama.age_=(51) 129 | println("age: " + obama.age()) 130 | ``` 131 | 132 | 133 | ### 鸭子类型 134 | 135 | 走起来像鸭子,叫起来像鸭子,就是鸭子。 136 | 这个例子中使用{ def close(): Unit }作为参数类型。因此任何含有close()的函数的类都可以作为参数。 137 | 不必使用继承这种不够灵活的特性。 138 | 139 | ``` 140 | def withClose(closeAble: { def close(): Unit }, op: { def close(): Unit } => Unit) { 141 | try { 142 | op(closeAble) 143 | } finally { 144 | closeAble.close() 145 | } 146 | } 147 | 148 | class Connection { 149 | def close() = println("close Connection") 150 | } 151 | 152 | val conn: Connection = new Connection() 153 | withClose(conn, conn => 154 | println("do something with Connection")) 155 | ``` 156 | 157 | 158 | ### 柯里化 159 | 160 | 这个例子和上面的功能相同。不同的是使用了柯里化(Currying)的技术。 161 | def add(x:Int, y:Int) = x + y 是普通的函数 162 | def add(x:Int) = (y:Int) => x + y 是柯里化后的函数,相当于返回一个匿名函数表达式。 163 | def add(x:Int)(y:Int) = x + y 是上面的简化写法 164 | 柯里化可以让我们构造出更像原生语言提供的功能的代码 165 | 例子中的withclose(...)(...)换成withclose(...){...} 166 | 是否和java中的synchronized关键字用法很像? 167 | 168 | ``` 169 | def withClose(closeAble: { def close(): Unit })(op: { def close(): Unit } => Unit) { 170 | try { 171 | op(closeAble) 172 | } finally { 173 | closeAble.close() 174 | } 175 | } 176 | 177 | class Connection { 178 | def close() = println("close Connection") 179 | } 180 | 181 | val conn: Connection = new Connection() 182 | withClose(conn)(conn => 183 | println("do something with Connection")) 184 | ``` 185 | 186 | 187 | ### 范型 188 | 189 | 之前的例子可以使用泛型变得更简洁更灵活。 190 | 试着将val msg = "123456"修改为val msg = 123456。 191 | 虽然msg由String类型变为Int类型,但是由于使用了泛型,代码依旧可以正常运行。 192 | 193 | ``` 194 | def withClose[A <: { def close(): Unit }, B](closeAble: A)(op: A => B) { 195 | try { 196 | op(closeAble) 197 | } finally { 198 | closeAble.close() 199 | } 200 | } 201 | 202 | class Connection { 203 | val msg = "123456" 204 | def close() = println("close Connection") 205 | } 206 | 207 | val conn: Connection = new Connection() 208 | val msg = withClose(conn) { conn => 209 | { 210 | println("do something with Connection") 211 | conn.msg 212 | } 213 | } 214 | 215 | println(msg) 216 | ``` 217 | 218 | ### Traits 219 | 220 | Traits就像是有函数体的Interface。使用with关键字来混入。 221 | 这个例子是给java.util.ArrayList添加了foreach的功能。 222 | 试着再在后面加上with JsonAble,给list添加修改为Json的能力 223 | 224 | ``` 225 | trait ForEachAble[A] { 226 | def itera修改为r: java.util.Itera修改为r[A] 227 | def foreach(f: A => Unit) = { 228 | val iter = itera修改为r 229 | while (iter.hasNext) 230 | f(iter.next) 231 | } 232 | } 233 | 234 | trait JsonAble { 235 | override def 修改为String() = 236 | scala.util.parsing.json.JSONFormat.defaultFormatter(this) 237 | } 238 | 239 | val list = new java.util.ArrayList[Int]() with ForEachAble[Int] 240 | list.add(1); list.add(2) 241 | 242 | list.foreach(x => println(x)) 243 | println("Json: " + list.修改为String()) 244 | ``` 245 | 246 | 247 | ## 函数式编程 248 | 249 | ### 模式匹配 250 | 251 | 模式匹配是类似switch-case特性,但更加灵活;也类似if-else,但更加简约。 252 | 这个例子展示的使用用模式匹配实现斐波那契。 253 | 使用case来匹配参数,如果case _,则可以匹配任何参数。 254 | 这个例子有所不足,当输入负数时,会无限循环。 255 | 可以在case后添加if语句判断,将case n: Int 修改为 case n: Int if (n > 1)即可。 256 | Try 修改为 add case n: String => fibonacci(n.修改为Int) before case _ 这样就可以匹配String类型 257 | 在最后添加 println(fibonacci(-3));println(fibonacci("3"));来检查刚刚修改的效果。 258 | 259 | 260 | ``` 261 | def fibonacci(in: Any): Int = in match { 262 | case 0 => 0 263 | case 1 => 1 264 | case n: Int => fibonacci(n - 1) + fibonacci(n - 2) 265 | case _ => 0 266 | } 267 | 268 | println(fibonacci(3)) 269 | ``` 270 | 271 | ### Case Class 272 | 273 | case class 顾名思义就是为case语句专门设计的类。 274 | 在普通类的基础上添加了和类名一致的工厂方法, 275 | 还添加了hashcode,equals和修改为String等方法。 276 | 试试最后添加 println(Sum(1,2)) 。 277 | 由于使用了require(n >= 0)来检验参数,尝试使用负数,会抛出异常。 278 | 279 | 280 | ``` 281 | abstract class Expr 282 | 283 | case class FibonacciExpr(n: Int) extends Expr { 284 | require(n >= 0) 285 | } 286 | 287 | case class SumExpr(a: Expr, b: Expr) extends Expr 288 | 289 | def value(in: Expr): Int = in match { 290 | case FibonacciExpr(0) => 0 291 | case FibonacciExpr(1) => 1 292 | case FibonacciExpr(n) => value(SumExpr(FibonacciExpr(n - 1), FibonacciExpr(n - 2))) 293 | case SumExpr(a, b) => value(a) + value(b) 294 | case _ => 0 295 | } 296 | println(value(FibonacciExpr(3))) 297 | ``` 298 | 299 | ### 函数式的威力 300 | 301 | 这个例子是判断一个List中是否含有奇数。 302 | 第一行到倒数第二行是使用for循环的指令式编程解决。最后一行是通过函数式编程解决。 303 | 通过将函数作为参数,可以使程序极为简洁。其中 _ % 2 == 1 是 (x: Int) => x % 2 == 1 的简化写法。 304 | 相比于Ruby等动态语言,这威力来自于科学而不是魔法 305 | 306 | ``` 307 | val list = List(1, 2, 3, 4) 308 | 309 | def containsOdd(list: List[Int]): Boolean = { 310 | for (i <- list) { 311 | if (i % 2 == 1) 312 | return true; 313 | } 314 | return false; 315 | } 316 | 317 | println("list containsOdd by for loop:" + containsOdd(list)) 318 | 319 | println("list containsOdd by funtional:" + list.exists(_ % 2 == 1)) 320 | ``` 321 | 322 | ### 函数式真正的威力 323 | 324 | 函数式除了能简化代码外,更重要的是他关注的是Input和Output,函数本身没有副作用。 325 | 就是Unix pipeline一样,简单的命令组合在一起威力无穷。 326 | List的filter方法接受一个过滤函数,返回一个新的List。 327 | 如果你喜欢Unix pipeline的方式,你一定也会喜欢函数式编程。 328 | 这个例子是用函数式的代码模拟“cat file | grep 'warn' | grep '2013' | wc”的行为。 329 | 330 | ``` 331 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 332 | 333 | println("cat file | grep 'warn' | grep '2013' | wc : " 334 | + file.filter(_.contains("warn")).filter(_.contains("2013")).size) 335 | ``` 336 | ### Word Count 337 | Word Count是一个MapReduce的一个经典示例。巧合的是,使用函数式的编程法,用类似MapReduce的方法实现word count也是最直观的。 338 | 这个例子介绍了List的两个重要的高阶方法map和reduceLeft。 339 | List的map方法接受一个转换函数,返回一个经过转换的List。 340 | List的reduceLeft方法接受一个合并函数,依次遍历合并。第一个参数是合并后的值,第二个参数是下一个需要合并的值。 341 | 将reduceLeft(_ + _)修改为foldLeft(0)(_ + _)。foldLeft比将reduceLeft更常用,因为他可以提供一个初始参数。 342 | Map和foldLeft可以代替大部分需要for循环的操作,并且使代码更清晰 343 | 344 | ``` 345 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 346 | 347 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 348 | 349 | val num = file.map(wordcount).reduceLeft(_ + _) 350 | 351 | println("wordcount:" + num) 352 | ``` 353 | ### 尾递归 354 | 355 | 这个例子是用尾递归实现foldLeft。尾递归是递归的一种,特点在于会在函数的最末调用自身。 356 | 当用List做match case时,可以使用 :: 来解构。返回第一个元素head和剩余元素tail。 357 | 尾递归会在编译期优化,因此不用担心一般递归造成的栈溢出问题。 358 | 359 | ``` 360 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 361 | 362 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 363 | 364 | def foldLeft(list: List[Int])(init: Int)(f: (Int, Int) => Int): Int = { 365 | list match { 366 | case List() => init 367 | case head :: tail => foldLeft(tail)(f(init, head))(f) 368 | } 369 | } 370 | 371 | val num = foldLeft(file.map(wordcount))(0)(_ + _) 372 | 373 | println("wordcount:" + num) 374 | ``` 375 | 376 | ### 更强大的For 377 | 378 | 循环语句是指令式编程的特产,Scala对其加以改进,成为适应函数式风格的利器。 379 | For循环也是有返回值的,其返回是一个List。在每一轮迭代中加入yield,yield后的值可以加入到List中。 380 | 这个例子是使用for循环代替map函数。 381 | 382 | ``` 383 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 384 | 385 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 386 | 387 | val counts = 388 | for (line <- file) 389 | yield wordcount(line) 390 | 391 | val num = counts.reduceLeft(_ + _) 392 | 393 | println("wordcount:" + num) 394 | ``` 395 | ### Option 396 | 397 | NullException是Java中最常见的异常,要想避免他只有不断检查null。Scala提供了Option机制来解决。 398 | 这个例子包装了可能返回null的getProperty方法,使其返回一个Option。 399 | 这样就可以不再漫无目的地null检查。只要Option类型的值即可。 400 | 使用pattern match来检查是常见做法。也可以使用getOrElse来提供当为None时的默认值。 401 | 给力的是Option还可以看作是最大长度为1的List,其的强大功能都可以使用。 402 | 尝试在最后添加 osName.foreach(print _) 。 403 | 404 | 405 | ``` 406 | 407 | def getProperty(name: String): Option[String] = { 408 | val value = System.getProperty(name) 409 | if (value != null) Some(value) else None 410 | } 411 | 412 | val osName = getProperty("os.name") 413 | 414 | osName match { 415 | case Some(value) => println(value) 416 | case _ => println("none") 417 | } 418 | 419 | println(osName.getOrElse("none")) 420 | 421 | 422 | 423 | ``` 424 | 425 | 426 | ### Lazy 427 | 428 | Lazy可以延迟初始化。加上lazy的字段会在第一次访问的时候初始化。 429 | 这个例子是从github获得Scala的版本号,由于访问网络需要较多时间。 430 | 如果费尽力气获取到,而调用它的代码却不去访问就会很浪费。 431 | 可以使用lazy来延迟获取。 432 | 433 | ``` 434 | class ScalaCurrentVersion(val url: String) { 435 | lazy val source= { 436 | println("fetching from url...") 437 | scala.io.Source.fromURL(url).getLines().修改为List 438 | } 439 | lazy val majorVersion = source.find(_.contains("version.major")) 440 | lazy val minorVersion = source.find(_.contains("version.minor")) 441 | } 442 | val version = new ScalaCurrentVersion("https://raw.github.com/scala/scala/master/build.number") 443 | println("get scala version from " + version.url) 444 | version.majorVersion.foreach(println _) 445 | version.minorVersion.foreach(println _) 446 | ``` 447 | 448 | 449 | ## 并发 450 | 451 | ### 使用Ac修改为r 452 | 453 | Ac修改为r是Scala的并发模型。在2.10之后的版本中,使用[http://akka.io/](Akka)作为其推荐Ac修改为r实现。 454 | Ac修改为r是类似线程的实体,有一个邮箱。 455 | 可以通过system.ac修改为rOf来创建,receive获取邮箱消息,!向邮箱发送消息。 456 | 这个例子是一个EchoServer,接受信息并打印。 457 | 458 | ``` 459 | import akka.ac修改为r.{ Ac修改为r, Ac修改为rSystem, Props } 460 | 461 | val system = Ac修改为rSystem() 462 | 463 | class EchoServer extends Ac修改为r { 464 | def receive = { 465 | case msg: String => println("echo " + msg) 466 | } 467 | } 468 | 469 | val echoServer = system.ac修改为rOf(Props[EchoServer]) 470 | echoServer ! "hi" 471 | 472 | system.shutdown 473 | ``` 474 | 475 | ### Ac修改为r更简化的用法 476 | 477 | 可以通过更简化的办法声明Ac修改为r。 478 | 导入akka.ac修改为r.Ac修改为rDSL中的ac修改为r函数。 479 | 这个函数可以接受一个Ac修改为r的构造器Act,启动并返回Ac修改为r。 480 | 481 | 482 | ``` 483 | import akka.ac修改为r.Ac修改为rDSL._ 484 | import akka.ac修改为r.Ac修改为rSystem 485 | 486 | implicit val system = Ac修改为rSystem() 487 | 488 | val echoServer = ac修改为r(new Act { 489 | become { 490 | case msg => println("echo " + msg) 491 | } 492 | }) 493 | echoServer ! "hi" 494 | system.shutdown 495 | ``` 496 | ### Ac修改为r原理 497 | Ac修改为r比线程轻量。在Scala中可以创建数以百万级的Ac修改为r。奥秘在于Ac修改为r可以复用线程。 498 | Ac修改为r和线程是不同的抽象,他们的对应关系是由Dispatcher决定的。 499 | 这个例子创建4个Ac修改为r,每次调用的时候打印自身线程。 500 | 可以发现Ac修改为r和线程之间没有一对一的对应关系。 501 | 一个Ac修改为r可以使用多个线程,一个线程也会被多个Ac修改为r复用。 502 | 503 | ``` 504 | import akka.ac修改为r.{ Ac修改为r, Props, Ac修改为rSystem } 505 | import akka.testkit.CallingThreadDispatcher 506 | 507 | implicit val system = Ac修改为rSystem() 508 | 509 | class EchoServer(name: String) extends Ac修改为r { 510 | def receive = { 511 | case msg => println("server" + name + " echo " + msg + 512 | " by " + Thread.currentThread().getName()) 513 | } 514 | } 515 | 516 | val echoServers = (1 修改为 10).map(x => 517 | system.ac修改为rOf(Props(new EchoServer(x.修改为String)) 518 | .withDispatcher(CallingThreadDispatcher.Id))) 519 | (1 修改为 10).foreach(msg => 520 | echoServers(scala.util.Random.nextInt(10)) ! msg.修改为String) 521 | 522 | system.shutdown 523 | ``` 524 | 525 | 526 | 527 | ### 同步返回 528 | 529 | Ac修改为r非常适合于较耗时的操作。比如获取网络资源。 530 | 这个例子通过调用ask函数来获取一个Future。 531 | 在Ac修改为r内部通过 sender ! 传递结果。 532 | Future像Option一样有很多高阶方法,可以使用foreach查看结果。 533 | 534 | 535 | ``` 536 | import akka.ac修改为r.Ac修改为rDSL._ 537 | import akka.pattern.ask 538 | 539 | implicit val ec = scala.concurrent.ExecutionContext.Implicits.global 540 | implicit val system = akka.ac修改为r.Ac修改为rSystem() 541 | 542 | val versionUrl = "https://raw.github.com/scala/scala/master/starr.number" 543 | 544 | val fromURL = ac修改为r(new Act { 545 | become { 546 | case url: String => sender ! scala.io.Source.fromURL(url) 547 | .getLines().mkString("\n") 548 | } 549 | }) 550 | 551 | val version = fromURL.ask(versionUrl)(akka.util.Timeout(5 * 1000)) 552 | version.foreach(println _) 553 | 554 | system.shutdown 555 | ``` 556 | 557 | ### 异步返回 558 | 559 | 异步操作可以最大发挥效能。Scala的Futrue很强大,可以异步返回。 560 | 可以实现Futrue的onComplete方法。当Futrue结束的时候就会回调。 561 | 在调用ask的时候,可以设定超时。 562 | 563 | ``` 564 | import akka.ac修改为r.Ac修改为rDSL._ 565 | import akka.pattern.ask 566 | 567 | implicit val ec = scala.concurrent.ExecutionContext.Implicits.global 568 | implicit val system = akka.ac修改为r.Ac修改为rSystem() 569 | 570 | val versionUrl = "https://raw.github.com/scala/scala/master/starr.number" 571 | 572 | val fromURL = ac修改为r(new Act { 573 | become { 574 | case url: String => sender ! scala.io.Source.fromURL(url) 575 | .getLines().mkString("\n") 576 | } 577 | }) 578 | 579 | val version = fromURL.ask(versionUrl)(akka.util.Timeout(5 * 1000)) 580 | version onComplete { 581 | case msg => println(msg); system.shutdown 582 | } 583 | ``` 584 | ### 并发集合 585 | 586 | 这个例子是访问若干URL,并记录时间。 587 | 如果能并发访问,就可以大幅提高性能。 588 | 尝试将urls.map修改为urls.par.map。这样每个map中的函数都可以并发执行。 589 | 当函数式和并发结合,就会这样让人兴奋。 590 | 591 | ``` 592 | import scala.io.Codec 593 | import java.nio.charset.CodingErrorAction 594 | 595 | implicit val codec = Codec("UTF-8") 596 | codec.onMalformedInput(CodingErrorAction.REPLACE) 597 | codec.onUnmappableCharacter(CodingErrorAction.REPLACE) 598 | 599 | val urls = "http://scala-lang.org" :: "https://github.com/yankay/scala-修改为ur" :: Nil 600 | 601 | def fromURL(url: String) = scala.io.Source.fromURL(url).getLines().mkString("\n") 602 | 603 | val s = System.currentTimeMillis() 604 | time(urls.map(fromURL(_))) 605 | println("time: " + (System.currentTimeMillis - s) + "ms") 606 | ``` 607 | 608 | ### 并发wordcount 609 | 并发集合支持大部分集合的功能。 610 | 在前面有一个word count例子,也可以用并行集合加以实现。 611 | 不增加程序复杂性,却能大幅提高程序利用多核的能力。 612 | 613 | ``` 614 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 615 | 616 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 617 | 618 | val num = file.par.map(wordcount).par.reduceLeft(_ + _) 619 | 620 | println("wordcount:" + num) 621 | ``` 622 | 623 | ### 远程Ac修改为r 624 | Ac修改为r是并发模型,也使用于分布式。 625 | 这个例子创建一个Echo服务器,通过ac修改为rOf来注册自己。 626 | 然后再创建一个client,通过Akka url来寻址。 627 | 除了是通过url创建的,其他使用的方法和普通Ac修改为r一样。 628 | 629 | ``` 630 | import akka.ac修改为r.{ Ac修改为r, Ac修改为rSystem, Props } 631 | import com.typesafe.config.ConfigFac修改为ry 632 | 633 | implicit val system = akka.ac修改为r.Ac修改为rSystem("RemoteSystem", 634 | ConfigFac修改为ry.load.getConfig("remote")) 635 | class EchoServer extends Ac修改为r { 636 | def receive = { 637 | case msg: String => println("echo " + msg) 638 | } 639 | } 640 | 641 | val server = system.ac修改为rOf(Props[EchoServer], name = "echoServer") 642 | 643 | val echoClient = system 644 | .ac修改为rFor("akka://RemoteSystem@127.0.0.1:2552/user/echoServer") 645 | echoClient ! "Hi Remote" 646 | 647 | system.shutdown 648 | 649 | ``` 650 | 651 | ## 实践 652 | 653 | ### 使用Java 654 | 655 | Scala和Java可以非常方便的互操作,前面已经有大量Scala直接使用Java的例子。 656 | 同样Java也可以使用Scala。这个例子演示使用@BeanProperty注解来生成Java Style的Bean。 657 | 尝试将在var name前加上@BeanProperty。这样就给bean添加了getter/setter 658 | Apache BeanUtils就可以正常工作。 659 | 660 | ``` 661 | import org.apache.commons.beanutils.BeanUtils 662 | import scala.beans.BeanProperty 663 | 664 | class SimpleBean( var name: String) { 665 | } 666 | 667 | val bean = new SimpleBean("foo") 668 | 669 | println(BeanUtils.describe(bean)) 670 | ``` 671 | 672 | ### 相等性 673 | 在Scala中==操作等效于equals,这一点和Java不同。更自然一些。 674 | 这个例子定义了一个equals函数,并验证。 675 | 写一个完全正确的equal函数并不容易,这个例子也有子类会不对称的Bug。 676 | 尝试将class修改为case class并删除equals函数。 677 | case类会自动生成正确的equals函数。 678 | 679 | ``` 680 | class Person(val name: String) { 681 | override def equals(other: Any) = other match { 682 | case that: Person => name.equals(that.name) 683 | case _ => false 684 | } 685 | } 686 | 687 | println(new Person("Black") == new Person("Black")) 688 | ``` 689 | 690 | 691 | ### 抽取器 692 | 抽取器可以帮助pattern match进行解构。 693 | 这个例子是构建一个Email抽取器,只要实现unapply函数就可以了。 694 | Scala的正则表达式会自带抽取器,可以抽取出一个List。List的元素是匹配()里的表达式。 695 | 抽取器很有用,短短的例子里就有两处使用抽取器: 696 | case user :: domain :: Nil解构List;case Email(user, domain) 解构Email。 697 | 698 | ``` 699 | object Email { 700 | def apply(user: String, domain: String) = user + "@" + domain 701 | 702 | def unapply(str: String) = new Regex("""(.*)@(.*)""").unapplySeq(str).get match { 703 | case user :: domain :: Nil => Some(user, domain) 704 | case _ => None 705 | } 706 | } 707 | 708 | "user@domain.com" match { 709 | case Email(user, domain) => println(user + "@" + domain) 710 | } 711 | ``` 712 | 713 | ### 记忆模式 714 | 记忆模式可以解决手动编写存取cache代码的麻烦。 715 | 这个例子中,memo可以将一个不含cache函数,包装成一个含有cache功能的。 716 | 还是斐波那契的例子,通过cache可以使性能提高。 717 | 尝试将fibonacci_(n - 1) + fibonacci_(n - 2)修改memo(fibonacci_)(n - 1) + memo(fibonacci_)(n - 2),可以提高更多。 718 | 719 | ``` 720 | import scala.collection.mutable.WeakHashMap 721 | 722 | val cache = new WeakHashMap[Int, Int] 723 | def memo(f: Int => Int) = (x: Int) => cache.getOrElseUpdate(x, f(x)) 724 | 725 | def fibonacci_(in: Int): Int = in match { 726 | case 0 => 0; 727 | case 1 => 1; 728 | case n: Int => fibonacci_(n - 1) + fibonacci_(n - 2) 729 | } 730 | 731 | val fibonacci: Int => Int = memo(fibonacci_) 732 | 733 | val t1 = System.currentTimeMillis() 734 | println(fibonacci(40)) 735 | println("it takes " + (System.currentTimeMillis() - t1) + "ms") 736 | 737 | val t2 = System.currentTimeMillis() 738 | println(fibonacci(40)) 739 | println("it takes " + (System.currentTimeMillis() - t2) + "ms") 740 | ``` 741 | ### 隐式转换 742 | implicit可以定义一个转换函数,可以在下面的使用到的时候自动转换。 743 | 这个例子可以将String自动转换为Date类型。隐式转换时实现DSL的重要工具。 744 | 745 | ``` 746 | implicit def strToDate(str: String) = new SimpleDateFormat("yyyy-MM-dd").parse(str) 747 | 748 | println("2013-01-01 unix time: " + "2013-01-01".getTime()/1000l) 749 | ``` 750 | 751 | ### DSL 752 | DSL是Scala最强大武器,Scala可以使一些描述性代码变得极为简单。 753 | 这个例子是使用DSL生成JSON。Scala很多看似是语言级的特性也是用DSL做到的。 754 | 自己编写DSL有点复杂,但使用方便灵活的。 755 | 756 | ``` 757 | import org.json4s._ 758 | import org.json4s.JsonDSL._ 759 | 760 | import org.json4s.jackson.JsonMethods._ 761 | 762 | case class Twitter(id: Long, text: String, publishedAt: Option[java.util.Date]) 763 | 764 | var twitters = Twitter(1, "hello scala", Some(new Date())) :: Twitter(2, "I like scala 修改为ur", None) :: Nil 765 | 766 | var json = ("twitters" 767 | -> twitters.map( 768 | t => ("id" -> t.id) 769 | ~ ("text" -> t.text) 770 | ~ ("published_at" -> t.publishedAt.修改为String()))) 771 | 772 | println(pretty(render(json))) 773 | ``` 774 | 775 | ### 测试 776 | Scala DSL可以使测试更方便。 777 | 这个例子是测试一个阶乘函数。使用should/in来建立测试用例。 778 | 测试是默认并发执行的。 779 | 780 | ``` 781 | import org.specs2.mutable._ 782 | 783 | class Fac修改为rialSpec extends Specification { 784 | args.report(color = false) 785 | 786 | def fac修改为rial(n: Int) = (1 修改为 n).reduce(_ * _) 787 | 788 | "The 'Hello world' string" should { 789 | "fac修改为rial 3 must be 6" in { 790 | fac修改为rial(3) mustEqual 6 791 | } 792 | "fac修改为rial 4 must be 6" in { 793 | fac修改为rial(4) must greaterThan(6) 794 | } 795 | } 796 | } 797 | specs2.run(new Fac修改为rialSpec) 798 | ``` 799 | 800 | 801 | ### Simple Build Tool 802 | SBT是Scala的最佳编译工具。 803 | 在他的帮助下,you can develop Scala even without installing anything except JRE. 804 | This example is 修改为 run this Scala 指南 in your computer. 805 | 806 | ``` 807 | #Linux/Mac(compile & run): 808 | git clone https://github.com/yankay/scala-修改为ur-zh.git 809 | cd scala-修改为ur-zh 810 | ./sbt/sbt stage 811 | ./target/start 812 | 813 | #Windows(can only compile): 814 | git clone https://github.com/yankay/scala-修改为ur-zh.git 815 | cd scala-修改为ur-zh 816 | sbt\sbt stage 817 | ``` 818 | -------------------------------------------------------------------------------- /scala-tour.md: -------------------------------------------------------------------------------- 1 | # Scala Tour 2 | 3 | [http://www.scala-tour.com/](http://www.scala-tour.com/) 4 | 5 | ## Basic 6 | 7 | ### Expressions and Values 8 | 9 | In Scala, almost everything is an expression.println("hello wolrd")is an expression, "hello"+" world" 10 | is also an expression. 11 | 12 | Constants can be created with val, and variables can be created with var. More constants are better. 13 | 14 | ``` 15 | var helloWorld = "hello" + " world" 16 | println(helloWorld) 17 | 18 | val again = " again" 19 | helloWorld = helloWorld + again 20 | println(helloWorld) 21 | ``` 22 | 23 | ### First class Functions 24 | 25 | You can create functions with def. And the function body is an expression. 26 | 27 | When the body is a block expression, it returns the value of the last line. So there's no need to use the return keyword 28 | 29 | And like values, functions can also be assigned using var or val So it can be passed as an argument to another function. 30 | 31 | ``` 32 | def square(a: Int) = a * a 33 | 34 | def squareWithBlock(a: Int) = { 35 | a * a 36 | } 37 | 38 | val squareVal = (a: Int) => a * a 39 | 40 | def addOne(f: Int => Int, arg: Int) = f(arg) + 1 41 | 42 | println("square(2):" + square(2)) 43 | println("squareWithBlock(2):" + squareWithBlock(2)) 44 | println("squareVal(2):" + squareVal(2)) 45 | println("addOne(squareVal,2):" + addOne(squareVal, 2)) 46 | ``` 47 | 48 | 49 | 50 | ### Loan Pattern 51 | 52 | For functions which can be passed as arguments, the 'Loan' pattern is easy to implement. 53 | 54 | This example reads the self pid from /proc/self/stat. 55 | 56 | Because the 'withScanner' function encapsulates the 'try-finally' block,there's no need to call 'close()' any more. 57 | 58 | PS: the expression's return type is 'Unit' when it doesn't return a value. 59 | 60 | ``` 61 | import scala.reflect.io.File 62 | import java.util.Scanner 63 | 64 | def withScanner(f: File, op: Scanner => Unit) = { 65 | val scanner = new Scanner(f.bufferedReader) 66 | try { 67 | op(scanner) 68 | } finally { 69 | scanner.close() 70 | } 71 | } 72 | 73 | withScanner(File("/proc/self/stat"), 74 | scanner => println("pid is " + scanner.next())) 75 | ``` 76 | 77 | 78 | ### Call-by-Name 79 | 80 | This example shows the call by name, When the last line tries to calculate '1 / 0', the program will throw an exception. 81 | 82 | Try to change 'def log(msg: String)' to 'def log(msg: => String)'.The program will not throw an exception because it has been changed to call-by-name 83 | 84 | 85 | Call-by-name means that the argument will be calculated when it be actually called. Because 'logEnable = false', the '1 / 0' would be skipped. 86 | 87 | 88 | Call-by-name can reduce the useless calculation and exception. 89 | 90 | 91 | ``` 92 | val logEnable = false 93 | 94 | def log(msg: String) = 95 | if (logEnable) println(msg) 96 | 97 | val MSG = "programing is running" 98 | 99 | log(MSG + 1 / 0) 100 | ``` 101 | 102 | ### Define Class 103 | 104 | The 'class' keyword defines a class, and the 'new' keyword creates an instance. 105 | The fields can be also defined in class, like the 'firstName' and 'lastName'.These are automatically generated from the constructor's arguments. 106 | Methods can be defined with def, and fields can be defined with val or var 107 | The function name can be any characters like +,-,*,/. 108 | The 'obama.age_=(51)' can be simplified as 'obama.age = 51'. 109 | And the 'obama.age()' can be simplified as 'obama.age'. 110 | 111 | ``` 112 | class Person(val firstName: String, val lastName: String) { 113 | 114 | private var _age = 0 115 | def age = _age 116 | def age_=(newAge: Int) = _age = newAge 117 | 118 | def fullName() = firstName + " " + lastName 119 | 120 | override def toString() = fullName() 121 | } 122 | 123 | val obama: Person = new Person("Barack", "Obama") 124 | 125 | println("Person: " + obama) 126 | println("firstName: " + obama.firstName) 127 | println("lastName: " + obama.lastName) 128 | obama.age_=(51) 129 | println("age: " + obama.age()) 130 | ``` 131 | 132 | 133 | ### Duck Typing 134 | 135 | When I see a bird that walks like a duck and swims like a duck and quacks like a duck, I call that bird a duck. 136 | This example uses '{ def close(): Unit }' as the type of argument. So any class contains methods 'close()' can be passed. 137 | And there's no need to use 'inherit'。 138 | 139 | ``` 140 | def withClose(closeAble: { def close(): Unit }, op: { def close(): Unit } => Unit) { 141 | try { 142 | op(closeAble) 143 | } finally { 144 | closeAble.close() 145 | } 146 | } 147 | 148 | class Connection { 149 | def close() = println("close Connection") 150 | } 151 | 152 | val conn: Connection = new Connection() 153 | withClose(conn, conn => 154 | println("do something with Connection")) 155 | ``` 156 | 157 | 158 | ### Currying 159 | 160 | This example is similar to the previous one.The difference between them is this one leverage currying technology。 161 | def add(x:Int, y:Int) = x + y is a normal function 162 | def add(x:Int) = (y:Int) => x + y is a curried function.The return value is a function expression. 163 | def add(x:Int)(y:Int) = x + y is syntactic sugar 164 | Currying can let our code look like it is part of the language. 165 | Change the withclose(...)(...) to withclose(...){...} 166 | Is it similar with Java's synchronized block? 167 | 168 | ``` 169 | def withClose(closeAble: { def close(): Unit })(op: { def close(): Unit } => Unit) { 170 | try { 171 | op(closeAble) 172 | } finally { 173 | closeAble.close() 174 | } 175 | } 176 | 177 | class Connection { 178 | def close() = println("close Connection") 179 | } 180 | 181 | val conn: Connection = new Connection() 182 | withClose(conn)(conn => 183 | println("do something with Connection")) 184 | ``` 185 | 186 | 187 | ### Generic 188 | 189 | The sample before can be more simplified with generics. 190 | Try to change val msg = "123456" to val msg = 123456. 191 | Although the type of msg changed from String to Int, the program still compiles. 192 | 193 | ``` 194 | def withClose[A <: { def close(): Unit }, B](closeAble: A)(op: A => B) { 195 | try { 196 | op(closeAble) 197 | } finally { 198 | closeAble.close() 199 | } 200 | } 201 | 202 | class Connection { 203 | val msg = "123456" 204 | def close() = println("close Connection") 205 | } 206 | 207 | val conn: Connection = new Connection() 208 | val msg = withClose(conn) { conn => 209 | { 210 | println("do something with Connection") 211 | conn.msg 212 | } 213 | } 214 | 215 | println(msg) 216 | ``` 217 | 218 | ### Traits 219 | 220 | Traits look like Java's interfaces, but with function blocks. One class can extend several traits using the with keyword. 221 | This example extends java.util.ArrayList with a foreach loop. 222 | Try to append with JsonAble to extend the ability of toJson. 223 | 224 | ``` 225 | trait ForEachAble[A] { 226 | def iterator: java.util.Iterator[A] 227 | def foreach(f: A => Unit) = { 228 | val iter = iterator 229 | while (iter.hasNext) 230 | f(iter.next) 231 | } 232 | } 233 | 234 | trait JsonAble { 235 | override def toString() = 236 | scala.util.parsing.json.JSONFormat.defaultFormatter(this) 237 | } 238 | 239 | val list = new java.util.ArrayList[Int]() with ForEachAble[Int] 240 | list.add(1); list.add(2) 241 | 242 | list.foreach(x => println(x)) 243 | println("Json: " + list.toString()) 244 | ``` 245 | 246 | 247 | ## Functional Programing 248 | 249 | ### Pattern Matching 250 | 251 | Pattern Matching is more flexible than switch-case and simpler than if-else. 252 | This example shows a Fibonacci function implemented with pattern matching. 253 | The case keyword matches on a value. The case _ means it can match anything. 254 | But the example has a bug. If the input is a negative number, it will loop endlessly. 255 | Try to add if after case. Change case n: Int to case n: Int if (n > 1). 256 | Try to add case n: String => fibonacci(n.toInt) before case _ to match String type. 257 | Try to add println(fibonacci(-3));println(fibonacci("3")) to test the program. 258 | 259 | 260 | ``` 261 | def fibonacci(in: Any): Int = in match { 262 | case 0 => 0 263 | case 1 => 1 264 | case n: Int => fibonacci(n - 1) + fibonacci(n - 2) 265 | case _ => 0 266 | } 267 | 268 | println(fibonacci(3)) 269 | ``` 270 | 271 | ### Case Class 272 | 273 | Case classes are used to conveniently store and match on the contents of a class. 274 | You can construct them without using new. 275 | It also has hashcode, equality and nice toString methods. 276 | Try to append println(Sum(1,2)) last. 277 | Because of the require(n >= 0), it will throw an exception when the input is negative. 278 | 279 | 280 | ``` 281 | abstract class Expr 282 | 283 | case class FibonacciExpr(n: Int) extends Expr { 284 | require(n >= 0) 285 | } 286 | 287 | case class SumExpr(a: Expr, b: Expr) extends Expr 288 | 289 | def value(in: Expr): Int = in match { 290 | case FibonacciExpr(0) => 0 291 | case FibonacciExpr(1) => 1 292 | case FibonacciExpr(n) => value(SumExpr(FibonacciExpr(n - 1), FibonacciExpr(n - 2))) 293 | case SumExpr(a, b) => value(a) + value(b) 294 | case _ => 0 295 | } 296 | println(value(FibonacciExpr(3))) 297 | ``` 298 | 299 | ### The power of functional programing 300 | 301 | This example determines whether there is an odd number in the list. 302 | Every line of the code excluding the last line is created using imperative programming.The last line is created using functional programming. 303 | Treating function expression as function arguments can simplify code effectively.The _ % 2 == 1 is the syntactic sugar of (x: Int) => x % 2 == 1. 304 | Ruby's power is from magic, but Scala's power is from science. 305 | 306 | ``` 307 | val list = List(1, 2, 3, 4) 308 | 309 | def containsOdd(list: List[Int]): Boolean = { 310 | for (i <- list) { 311 | if (i % 2 == 1) 312 | return true; 313 | } 314 | return false; 315 | } 316 | 317 | println("list containsOdd by for loop:" + containsOdd(list)) 318 | 319 | println("list containsOdd by funtional:" + list.exists(_ % 2 == 1)) 320 | ``` 321 | 322 | ### The true power of functional programming 323 | 324 | Besides simplifying code, functional programming is more concerned with Input & Output without side-effects. 325 | Like the Unix pipeline, simple commands can be combined together. 326 | The filter method in List can accept a filter function to return a new List. 327 | If you like the way Unix pipelines commands, you may also like functional programming 328 | This example uses Scala code to simulate the Unix command line "cat file | grep 'warn' | grep '2013' | wc." 329 | 330 | ``` 331 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 332 | 333 | println("cat file | grep 'warn' | grep '2013' | wc : " 334 | + file.filter(_.contains("warn")).filter(_.contains("2013")).size) 335 | ``` 336 | ### Word Count 337 | Word Count is a classic use case for Map Reduce. Map Reduce with functional programming is also a wonderful way to implement word count. 338 | The example shows two important functions 'map' and 'reduceLeft' in List. 339 | The map function accepts a translation expression and returns the translated list. 340 | The reduceLeft function accepts a combining expression and returns the combined result.The first argument is the reduced value, and the second argument is the value next. 341 | Try to change reduceLeft(_ + _) into foldLeft(0)(_ + _).foldLeft is more popular than reduceLeft for it can provide a initial value. 342 | Map and ReduceLeft can replace a for-loop expression, making code cleaner. 343 | 344 | ``` 345 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 346 | 347 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 348 | 349 | val num = file.map(wordcount).reduceLeft(_ + _) 350 | 351 | println("wordcount:" + num) 352 | ``` 353 | ### Tail Recursion 354 | 355 | This example shows how to implement foldLeft with Tail RecursionTail Recursion is one type of Recursion, in which a function calls itself as its last expression. 356 | Lists can be pattern matched by '::', the first element returned is head, and the others are the tail. 357 | Tail Recursion can be optimized at compile time. So there's no need to worry about stack overflow. 358 | 359 | ``` 360 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 361 | 362 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 363 | 364 | def foldLeft(list: List[Int])(init: Int)(f: (Int, Int) => Int): Int = { 365 | list match { 366 | case List() => init 367 | case head :: tail => foldLeft(tail)(f(init, head))(f) 368 | } 369 | } 370 | 371 | val num = foldLeft(file.map(wordcount))(0)(_ + _) 372 | 373 | println("wordcount:" + num) 374 | ``` 375 | 376 | ### Powerful For Expression 377 | 378 | Loop expression is feature of imperative programming.So Scala improved it to suit functional programming. 379 | For expressions in Scala can also return a List. With 'yield' in the loop, the value after yield will be appended to the List. 380 | This example replaces the map function with for loop. 381 | 382 | ``` 383 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 384 | 385 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 386 | 387 | val counts = 388 | for (line <- file) 389 | yield wordcount(line) 390 | 391 | val num = counts.reduceLeft(_ + _) 392 | 393 | println("wordcount:" + num) 394 | ``` 395 | ### Option 396 | 397 | NullException is the most common exception in Java. The only way to avoid it is to check null everywhere.Scala provides an Option feature to solve it. 398 | This example creates a getProperty function, which returns Option instead of null. 399 | We don't need to check null, getting a value from Option is enough. 400 | Using pattern matching is a common way to get the value of Option. Use getOrElse() to set a default value when Option is None. 401 | Another important think is that Option has lots of functions in List, so it can work like a list in most of time. 402 | Try to add 'osName.foreach(print _)' at last. 403 | 404 | 405 | ``` 406 | 407 | def getProperty(name: String): Option[String] = { 408 | val value = System.getProperty(name) 409 | if (value != null) Some(value) else None 410 | } 411 | 412 | val osName = getProperty("os.name") 413 | 414 | osName match { 415 | case Some(value) => println(value) 416 | case _ => println("none") 417 | } 418 | 419 | println(osName.getOrElse("none")) 420 | 421 | 422 | 423 | ``` 424 | 425 | 426 | ### Lazy 427 | 428 | Lazy is lazy initial value. A field with the lazy keyword is only initialized when it is first accessed. 429 | This example is to get the Scala Version Code from Github. It takes a little time because of network latency. 430 | It waste of time that if we get it with latency but we do not use it later. 431 | So we can get it with lazy. 432 | 433 | ``` 434 | class ScalaCurrentVersion(val url: String) { 435 | lazy val source= { 436 | println("fetching from url...") 437 | scala.io.Source.fromURL(url).getLines().toList 438 | } 439 | lazy val majorVersion = source.find(_.contains("version.major")) 440 | lazy val minorVersion = source.find(_.contains("version.minor")) 441 | } 442 | val version = new ScalaCurrentVersion("https://raw.github.com/scala/scala/master/build.number") 443 | println("get scala version from " + version.url) 444 | version.majorVersion.foreach(println _) 445 | version.minorVersion.foreach(println _) 446 | ``` 447 | 448 | 449 | ## Concurrent 450 | 451 | ### Using Actor 452 | 453 | Actors are one of Scala's concurrent models.Users of Scala earlier than version 2.10 must install [http://akka.io/](Akka). 454 | An Actor is a like a thread instance with a mailbox. 455 | It can be created with system.actorOf: use receive to get a message and ! to send a message. 456 | This example is an EchoServer which can receive messages then print them. 457 | 458 | ``` 459 | import akka.actor.{ Actor, ActorSystem, Props } 460 | 461 | val system = ActorSystem() 462 | 463 | class EchoServer extends Actor { 464 | def receive = { 465 | case msg: String => println("echo " + msg) 466 | } 467 | } 468 | 469 | val echoServer = system.actorOf(Props[EchoServer]) 470 | echoServer ! "hi" 471 | 472 | system.shutdown 473 | ``` 474 | 475 | ### Simplify Actor 476 | 477 | There is a simpler way to define an Actor. 478 | Import the actor function from akka.actor.ActorDSL. 479 | This function accepts an Actor instance, and returns a started Actor. 480 | 481 | 482 | ``` 483 | import akka.actor.ActorDSL._ 484 | import akka.actor.ActorSystem 485 | 486 | implicit val system = ActorSystem() 487 | 488 | val echoServer = actor(new Act { 489 | become { 490 | case msg => println("echo " + msg) 491 | } 492 | }) 493 | echoServer ! "hi" 494 | system.shutdown 495 | ``` 496 | ### Actor Implementation 497 | An Actor is more lightweight than a thread. Millions of actors can be generated in Scala. The secret is that an Actor can reuse a thread. 498 | The mapping relationship between an Actor and a Thread is decided by a Dispatcher. 499 | This example creates 4 Actors, and prints its thread name when invoked. 500 | You will find there is no fixed mapping relationship between Actors and Threads. 501 | An Actor can use many threads. And a thread can be used by many Actors. 502 | 503 | ``` 504 | import akka.actor.{ Actor, Props, ActorSystem } 505 | import akka.testkit.CallingThreadDispatcher 506 | 507 | implicit val system = ActorSystem() 508 | 509 | class EchoServer(name: String) extends Actor { 510 | def receive = { 511 | case msg => println("server" + name + " echo " + msg + 512 | " by " + Thread.currentThread().getName()) 513 | } 514 | } 515 | 516 | val echoServers = (1 to 10).map(x => 517 | system.actorOf(Props(new EchoServer(x.toString)) 518 | .withDispatcher(CallingThreadDispatcher.Id))) 519 | (1 to 10).foreach(msg => 520 | echoServers(scala.util.Random.nextInt(10)) ! msg.toString) 521 | 522 | system.shutdown 523 | ``` 524 | 525 | 526 | 527 | ### Synchronized Return 528 | 529 | Actors are very suitable for long-running operations, like getting resources over a network. 530 | This example creates a Future with the ask function. 531 | In the actor we use 'sender !' to return the value. 532 | Like Option, Future has lots of functions. The result can be printed with a foreach. 533 | 534 | 535 | ``` 536 | import akka.actor.ActorDSL._ 537 | import akka.pattern.ask 538 | 539 | implicit val ec = scala.concurrent.ExecutionContext.Implicits.global 540 | implicit val system = akka.actor.ActorSystem() 541 | 542 | val versionUrl = "https://raw.github.com/scala/scala/master/starr.number" 543 | 544 | val fromURL = actor(new Act { 545 | become { 546 | case url: String => sender ! scala.io.Source.fromURL(url) 547 | .getLines().mkString("\n") 548 | } 549 | }) 550 | 551 | val version = fromURL.ask(versionUrl)(akka.util.Timeout(5 * 1000)) 552 | version.foreach(println _) 553 | 554 | system.shutdown 555 | ``` 556 | 557 | ### Asynchronous Return 558 | 559 | Asynchronous operations can provide better performance. A Future in Scala is very powerful, it can execute asynchronously. 560 | The Future will call the 'onComplete' function when it is finished. 561 | It can also set a TIMEOUT when specified. 562 | 563 | ``` 564 | import akka.actor.ActorDSL._ 565 | import akka.pattern.ask 566 | 567 | implicit val ec = scala.concurrent.ExecutionContext.Implicits.global 568 | implicit val system = akka.actor.ActorSystem() 569 | 570 | val versionUrl = "https://raw.github.com/scala/scala/master/starr.number" 571 | 572 | val fromURL = actor(new Act { 573 | become { 574 | case url: String => sender ! scala.io.Source.fromURL(url) 575 | .getLines().mkString("\n") 576 | } 577 | }) 578 | 579 | val version = fromURL.ask(versionUrl)(akka.util.Timeout(5 * 1000)) 580 | version onComplete { 581 | case msg => println(msg); system.shutdown 582 | } 583 | ``` 584 | ### Concurrent Collection 585 | 586 | This example prints the time needed to access several URLs. 587 | If we access them concurrently, the performance can be better. 588 | Try to change the 'urls.map' to 'urls.par.map'.Now, the functions in map will run concurrently. 589 | It's exciting to combine functional and concurrent programming! 590 | 591 | ``` 592 | import scala.io.Codec 593 | import java.nio.charset.CodingErrorAction 594 | 595 | implicit val codec = Codec("UTF-8") 596 | codec.onMalformedInput(CodingErrorAction.REPLACE) 597 | codec.onUnmappableCharacter(CodingErrorAction.REPLACE) 598 | 599 | val urls = "http://scala-lang.org" :: "https://github.com/yankay/scala-tour" :: Nil 600 | 601 | def fromURL(url: String) = scala.io.Source.fromURL(url).getLines().mkString("\n") 602 | 603 | val s = System.currentTimeMillis() 604 | time(urls.map(fromURL(_))) 605 | println("time: " + (System.currentTimeMillis - s) + "ms") 606 | ``` 607 | 608 | ### Concurrentwordcount 609 | Concurrent Collection support most functions in normal ones. 610 | Here is the word count example from earlier, improved using a parallel collection. 611 | It can use the power of multiple cores without increasing the complexity. 612 | 613 | ``` 614 | val file = List("warn 2013 msg", "warn 2012 msg", "error 2013 msg", "warn 2013 msg") 615 | 616 | def wordcount(str: String): Int = str.split(" ").count("msg" == _) 617 | 618 | val num = file.par.map(wordcount).par.reduceLeft(_ + _) 619 | 620 | println("wordcount:" + num) 621 | ``` 622 | 623 | ### Remote Actor 624 | Actor is not only a concurrency model, it can also be used for distributed computing. 625 | This example builds an EchoServer using an Actor. 626 | Then it creates a client to access the Akka URL. 627 | The usage is the same as with a normal Actor. 628 | 629 | ``` 630 | import akka.actor.{ Actor, ActorSystem, Props } 631 | import com.typesafe.config.ConfigFactory 632 | 633 | implicit val system = akka.actor.ActorSystem("RemoteSystem", 634 | ConfigFactory.load.getConfig("remote")) 635 | class EchoServer extends Actor { 636 | def receive = { 637 | case msg: String => println("echo " + msg) 638 | } 639 | } 640 | 641 | val server = system.actorOf(Props[EchoServer], name = "echoServer") 642 | 643 | val echoClient = system 644 | .actorFor("akka://RemoteSystem@127.0.0.1:2552/user/echoServer") 645 | echoClient ! "Hi Remote" 646 | 647 | system.shutdown 648 | 649 | ``` 650 | 651 | ## Practice 652 | 653 | ### Using Java 654 | 655 | Scala can execute Java code very easily. There have already been many examples of this. 656 | Java can also use Scala. This example shows how to use the @BeanProperty Annotation to create a Java Style Bean. 657 | Try to add @BeanProperty before var name. Now the bean contains getter/setter functions. 658 | And the Apache BeanUtils can work correctly. 659 | 660 | ``` 661 | import org.apache.commons.beanutils.BeanUtils 662 | import scala.beans.BeanProperty 663 | 664 | class SimpleBean( var name: String) { 665 | } 666 | 667 | val bean = new SimpleBean("foo") 668 | 669 | println(BeanUtils.describe(bean)) 670 | ``` 671 | 672 | ### Equality 673 | In Scala == is the same as equals function. It's not the same as Java, but it's more reasonable. 674 | This example defines a equals function, and prints the result. 675 | Correctly writing an equals function is difficult. This example has an issue with subclasses. 676 | Try to change 'class' to 'case class', and delete the equals function. 677 | Case Class correctly generates the equals function for us. 678 | 679 | ``` 680 | class Person(val name: String) { 681 | override def equals(other: Any) = other match { 682 | case that: Person => name.equals(that.name) 683 | case _ => false 684 | } 685 | } 686 | 687 | println(new Person("Black") == new Person("Black")) 688 | ``` 689 | 690 | 691 | ### Extractor 692 | Extractor objects can deconstruct pattern matches. 693 | This example builds an Email Extractor, only the 'unapply function' is needed. 694 | Scala's Regex contains an extractor, which extracts a List. The List elements sequentially match expressions captured in (). 695 | Extractor is very useful. There are 2 cases in this example. 696 | case user :: domain :: Nil extracts a List. case Email(user, domain) extracts an Email. 697 | 698 | ``` 699 | object Email { 700 | def apply(user: String, domain: String) = user + "@" + domain 701 | 702 | def unapply(str: String) = new Regex("""(.*)@(.*)""").unapplySeq(str).get match { 703 | case user :: domain :: Nil => Some(user, domain) 704 | case _ => None 705 | } 706 | } 707 | 708 | "user@domain.com" match { 709 | case Email(user, domain) => println(user + "@" + domain) 710 | } 711 | ``` 712 | 713 | ### Memory Pattern 714 | Memory Pattern can be used to simplify caching. 715 | In this example, the 'memo function' wraps a function without caching to add the simple cache capability. 716 | In this Fibonacci example, a cache improves performance after the first call. 717 | Try to change fibonacci_(n - 1) + fibonacci_(n - 2) to memo(fibonacci_)(n - 1) + memo(fibonacci_)(n - 2), it can improve more. 718 | 719 | ``` 720 | import scala.collection.mutable.WeakHashMap 721 | 722 | val cache = new WeakHashMap[Int, Int] 723 | def memo(f: Int => Int) = (x: Int) => cache.getOrElseUpdate(x, f(x)) 724 | 725 | def fibonacci_(in: Int): Int = in match { 726 | case 0 => 0; 727 | case 1 => 1; 728 | case n: Int => fibonacci_(n - 1) + fibonacci_(n - 2) 729 | } 730 | 731 | val fibonacci: Int => Int = memo(fibonacci_) 732 | 733 | val t1 = System.currentTimeMillis() 734 | println(fibonacci(40)) 735 | println("it takes " + (System.currentTimeMillis() - t1) + "ms") 736 | 737 | val t2 = System.currentTimeMillis() 738 | println(fibonacci(40)) 739 | println("it takes " + (System.currentTimeMillis() - t2) + "ms") 740 | ``` 741 | Implicit Conversion 742 | Implicit can be used to define a Conversion function. Types are automatically implicitly converted when needed. 743 | This example converts String to Date automatically. Implicit is the most important feature when implementing a DSL. 744 | 745 | ``` 746 | implicit def strToDate(str: String) = new SimpleDateFormat("yyyy-MM-dd").parse(str) 747 | 748 | println("2013-01-01 unix time: " + "2013-01-01".getTime()/1000l) 749 | ``` 750 | 751 | ### DSL 752 | DSL is the most powerful tool in Scala. With it Scala code can become more descriptive. 753 | This example generates Json with a DSL. Some of the features look like native features but are created by a DSL. 754 | It's complex to write your own DSL. But it's very easy to use. 755 | 756 | ``` 757 | import org.json4s._ 758 | import org.json4s.JsonDSL._ 759 | 760 | import org.json4s.jackson.JsonMethods._ 761 | 762 | case class Twitter(id: Long, text: String, publishedAt: Option[java.util.Date]) 763 | 764 | var twitters = Twitter(1, "hello scala", Some(new Date())) :: Twitter(2, "I like scala tour", None) :: Nil 765 | 766 | var json = ("twitters" 767 | -> twitters.map( 768 | t => ("id" -> t.id) 769 | ~ ("text" -> t.text) 770 | ~ ("published_at" -> t.publishedAt.toString()))) 771 | 772 | println(pretty(render(json))) 773 | ``` 774 | 775 | ### Testing 776 | Scala DSL can make testing even easier. 777 | This example tests a Factorial function. It creates a test case with should/in. 778 | Test cases run concurrently by default. 779 | 780 | ``` 781 | import org.specs2.mutable._ 782 | 783 | class FactorialSpec extends Specification { 784 | args.report(color = false) 785 | 786 | def factorial(n: Int) = (1 to n).reduce(_ * _) 787 | 788 | "The 'Hello world' string" should { 789 | "factorial 3 must be 6" in { 790 | factorial(3) mustEqual 6 791 | } 792 | "factorial 4 must be 6" in { 793 | factorial(4) must greaterThan(6) 794 | } 795 | } 796 | } 797 | specs2.run(new FactorialSpec) 798 | ``` 799 | 800 | 801 | ### Simple Build Tool 802 | SBT is a very popular build tool for Scala. 803 | With its help, you can develop Scala even without installing anything except JRE. 804 | This example is to run this Scala Tour in your computer. 805 | 806 | ``` 807 | #Linux/Mac(compile & run): 808 | git clone https://github.com/yankay/scala-tour-zh.git 809 | cd scala-tour-zh 810 | ./sbt/sbt stage 811 | ./target/start 812 | 813 | #Windows(can only compile): 814 | git clone https://github.com/yankay/scala-tour-zh.git 815 | cd scala-tour-zh 816 | sbt\sbt stage 817 | ``` 818 | -------------------------------------------------------------------------------- /src/main/resources/application.conf: -------------------------------------------------------------------------------- 1 | remote { 2 | akka { 3 | 4 | actor { 5 | provider = "akka.remote.RemoteActorRefProvider" 6 | } 7 | remote { 8 | transport = "akka.remote.netty.NettyRemoteTransport" 9 | netty { 10 | hostname = "127.0.0.1" 11 | port = 2552 12 | } 13 | } 14 | 15 | } 16 | } -------------------------------------------------------------------------------- /src/main/scala/com/yankay/scalaTour/ScalaScriptRunner.scala: -------------------------------------------------------------------------------- 1 | package com.yankay.scalaTour 2 | 3 | import java.io.OutputStream 4 | import java.io.PrintStream 5 | import java.io.PrintWriter 6 | import scala.collection.mutable.StringBuilder 7 | import scala.reflect.io.Directory 8 | import scala.reflect.io.File 9 | import scala.reflect.io.Path 10 | import scala.reflect.io.Path.string2path 11 | import scala.sys.process.Process 12 | import scala.sys.process.ProcessBuilder 13 | import scala.sys.process.ProcessCreation 14 | import scala.sys.process.ProcessIO 15 | import scala.sys.process.ProcessLogger 16 | import scala.tools.nsc.GenericRunnerCommand 17 | import scala.tools.nsc.GenericRunnerSettings 18 | import scala.tools.nsc.Global 19 | import scala.tools.nsc.Settings 20 | import scala.tools.nsc.reporters.ConsoleReporter 21 | import scala.tools.nsc.reporters.Reporter 22 | import scala.tools.nsc.util.HasClassPath 23 | import scala.util.Properties 24 | import scala.tools.nsc.io.Jar 25 | 26 | object ScalaScriptCompiler { 27 | // /** Default name to use for the wrapped script */ 28 | val defaultScriptMain = "Main" 29 | // 30 | // /** Pick a main object name from the specified settings */ 31 | // def scriptMain(settings: Settings) = settings.script.value match { 32 | // case "" => defaultScriptMain 33 | // case x => x 34 | // } 35 | 36 | val settings: GenericRunnerSettings = { 37 | val command = new GenericRunnerCommand(List(), Console.err println _) 38 | val CP = Properties.propOrEmpty("java.class.path") 39 | val PS = Properties.propOrEmpty("path.separator") 40 | CP.split(PS).foreach((x: String) => { 41 | command.settings.classpath.append(x) 42 | command.settings.bootclasspath.append(x) 43 | }) 44 | command.settings.nc.value = true 45 | command.settings.feature.value = true 46 | command.settings.language.appendToValue("reflectiveCalls") 47 | command.settings.language.appendToValue("implicitConversions") 48 | command.settings.nospecialization.value = true 49 | command.settings.nouescape.value = true 50 | command.settings; 51 | } 52 | 53 | def compile(script: String, err: OutputStream): Option[File] = { 54 | val scriptFile = File.makeTemp("scala-script", ".scala") 55 | // save the command to the file 56 | 57 | val illegalCodePattern = """.*(\.sys\.).*""".r 58 | 59 | val strippedScript = { 60 | if (illegalCodePattern.findAllIn(script).toList.isEmpty) script 61 | else """println("No need to import anything")""" 62 | } 63 | 64 | val scriptAll = """ 65 | import scala.io.Codec 66 | import java.nio.charset.CodingErrorAction 67 | object Main { 68 | def main(args: Array[String]) { 69 | implicit val codec = Codec("UTF-8") 70 | codec.onMalformedInput(CodingErrorAction.REPLACE) 71 | codec.onUnmappableCharacter(CodingErrorAction.REPLACE) 72 | """ + strippedScript + """ 73 | } 74 | } 75 | """ 76 | scriptFile.writeAll(scriptAll) 77 | try compile(scriptFile, err) 78 | finally scriptFile.delete() // in case there was a compilation error 79 | } 80 | 81 | def compile( 82 | scriptFile: File, err: OutputStream): Option[File] = { 83 | // def mainClass = scriptMain(settings) 84 | val compiledPath = Directory makeTemp "scalascript" 85 | 86 | // delete the directory after the user code has finished 87 | // sys.addShutdownHook(compiledPath.deleteRecursively()) 88 | 89 | settings.outdir.value = compiledPath.path 90 | 91 | /** 92 | * Setting settings.script.value informs the compiler this is not a 93 | * self contained compilation unit. 94 | */ 95 | // settings.script.value = mainClass 96 | val reporter = new ConsoleReporter(settings, Console.in, new PrintWriter(err, true)) 97 | val compiler = Global(settings, reporter) 98 | 99 | new compiler.Run compile List(scriptFile.path) 100 | if (reporter.hasErrors) None else 101 | try createjar(compiledPath) 102 | finally compiledPath.deleteRecursively; 103 | } 104 | 105 | def createjar(compiledPath: Directory): Option[File] = { 106 | val jarpath = File.makeTemp("scalascript", ".jar") 107 | jarpath.deleteIfExists() 108 | try { 109 | Jar.create(jarpath, compiledPath, defaultScriptMain) 110 | Some(jarpath) 111 | } catch { case _: Exception => jarpath.delete(); None } 112 | 113 | } 114 | } 115 | 116 | object ScalaScriptProcess { 117 | def javaf(): String = { 118 | val head = Properties.propOrEmpty("java.home") + File.separator + "bin" + File.separator + "java" 119 | if (File.separator == "/") 120 | head + "" 121 | else 122 | head + ".exe" 123 | } 124 | 125 | def create(compiledLocation: File, out: OutputStream, err: OutputStream): Option[ScalaScriptProcess] = { 126 | val policy = new java.io.File("policy") 127 | val sep = Properties.propOrEmpty("path.separator") 128 | val CP = Properties.propOrEmpty("java.class.path") + sep + compiledLocation + sep + "./bin" + sep + "." 129 | var args = List("-Djava.security.manager", "-Djava.security.policy=" + policy.getAbsolutePath ,"-cp", CP, "Main") 130 | if (Path(javaf()).exists) { 131 | val outp = new PrintStream(out, true); 132 | val errp = new PrintStream(err, true); 133 | val pl = ProcessLogger( 134 | outp println _, 135 | errp println _) 136 | Some(new ScalaScriptProcess(Process(javaf(), args), pl)) 137 | } else None 138 | 139 | } 140 | } 141 | 142 | class ScalaScriptProcess(val builder: ProcessBuilder, val logger: ProcessLogger) { 143 | def run(): Process = { 144 | // println(builder.toString) 145 | builder.run(logger); 146 | } 147 | } 148 | -------------------------------------------------------------------------------- /src/main/scala/com/yankay/scalaTour/Web.scala: -------------------------------------------------------------------------------- 1 | package com.yankay.scalaTour 2 | 3 | import java.io.ByteArrayOutputStream 4 | import scala.actors.Actor 5 | import scala.actors.TIMEOUT 6 | import scala.reflect.io.File 7 | import scala.util.Properties 8 | import org.eclipse.jetty.server.Handler 9 | import org.eclipse.jetty.server.Server 10 | import org.eclipse.jetty.servlet.ServletContextHandler 11 | import org.eclipse.jetty.servlet.ServletHolder 12 | import org.json4s.jackson.JsonMethods 13 | import javax.servlet.http.HttpServlet 14 | import javax.servlet.http.HttpServletRequest 15 | import javax.servlet.http.HttpServletResponse 16 | import org.eclipse.jetty.server.handler.ResourceHandler 17 | import org.eclipse.jetty.servlet.DefaultServlet 18 | import org.eclipse.jetty.servlet.FilterHolder 19 | import org.eclipse.jetty.server.handler.GzipHandler 20 | import org.eclipse.jetty.servlets.GzipFilter 21 | import java.util.EnumSet 22 | import javax.servlet.DispatcherType 23 | import scala.collection.mutable.WeakHashMap 24 | import scala.util.matching.Regex 25 | 26 | object Web { 27 | val cache = new WeakHashMap[String, RunResponse] 28 | 29 | def handler(): Handler = { 30 | val context = new ServletContextHandler 31 | context.setContextPath("/"); 32 | context.setResourceBase("webapp") 33 | context.addServlet(new ServletHolder(new RunServlet()), "/run"); 34 | context.addServlet(new ServletHolder(new DefaultServlet()), "/"); 35 | val fh = new FilterHolder(new GzipFilter()) 36 | fh.setInitParameter("mimeTypes", "text/html,text/css,application/x-javascript,imgage/png") 37 | context.addFilter(fh, "/*", EnumSet.of(DispatcherType.FORWARD, 38 | DispatcherType.INCLUDE, 39 | DispatcherType.REQUEST, 40 | DispatcherType.ASYNC, 41 | DispatcherType.ERROR)) 42 | context 43 | } 44 | 45 | def main(args: Array[String]) { 46 | 47 | val server = new Server(Properties.envOrElse("PORT", "8080").toInt); 48 | server.setHandler(handler()) 49 | println("Start") 50 | server.start() 51 | } 52 | } 53 | 54 | class RunServlet extends HttpServlet { 55 | 56 | def compileAndRun(code: String): RunResponse = { 57 | // println("code:" + code) 58 | val buffer = new ByteArrayOutputStream(); 59 | val file = ScalaScriptCompiler.compile(code, buffer); 60 | val error = new String(buffer.toByteArray()).lines.toList 61 | // println(new String(buffer.toByteArray())) 62 | // println(error); 63 | def replaceErrorCodeNum(src: String): String = { 64 | val reg = new Regex("""main\.scala:([0-9]*): error:.*""") 65 | reg.unapplySeq(src).getOrElse(src) match { 66 | case l :: Nil => "main.scala:" + (l.toString.toInt - 8).toString + src.substring("main.scala:".length() + l.toString.length()) 67 | case _ => src 68 | } 69 | } 70 | val errorMsg = error.map(x => x.replaceAll("/tmp/scala-script.*scala", "main.scala")).map(replaceErrorCodeNum) 71 | 72 | file match { 73 | case Some(f) => { 74 | try { 75 | val (outEvents, errEvents, exitValue) = run(f); 76 | new RunResponse(errorMsg.toList, outEvents, errEvents, exitValue) 77 | } finally { 78 | f.deleteIfExists(); 79 | } 80 | } 81 | case _ => new RunResponse(errorMsg, List(), List(), -1) 82 | } 83 | } 84 | 85 | def run(file: File): (List[String], List[String], Int) = { 86 | val out = new ByteArrayOutputStream(); 87 | val err = new ByteArrayOutputStream(); 88 | val proc = ScalaScriptProcess.create(file, out, err); 89 | proc match { 90 | case Some(p) => { 91 | val pid = p.run(); 92 | val timeout = new TimeoutActor(pid, 10 * 1000) 93 | timeout.start 94 | val existValue = pid.exitValue() 95 | timeout ! existValue 96 | val outEvents = new String(out.toByteArray()).lines.toList 97 | val errEvents = new String(err.toByteArray()).lines.toList.filterNot(_.startsWith("Picked up JAVA_TOOL_OPTIONS")) 98 | existValue match { 99 | case 0 => (outEvents, errEvents, 0) 100 | case x => (outEvents, errEvents ::: List("exit value is " + x), x) 101 | } 102 | 103 | } 104 | case _ => (List(), List(), -1) 105 | } 106 | } 107 | 108 | class TimeoutActor(proc: scala.sys.process.Process, timeout: Long) extends Actor { 109 | def act() { 110 | receiveWithin(timeout) { 111 | case Int => 112 | case TIMEOUT => proc.destroy() 113 | } 114 | } 115 | } 116 | 117 | def json(mode: RunResponse): String = { 118 | import org.json4s._ 119 | import org.json4s.JsonDSL._ 120 | val json = ("Errors" -> mode.errors) ~ ("Events" -> mode.events) ~ ("ErrEvents" -> mode.errEvents) 121 | JsonMethods.pretty(JsonMethods.render(json)) 122 | } 123 | 124 | def memo(f: String => RunResponse) = { 125 | (x: String) => 126 | { 127 | val response = Web.cache.get(x) 128 | response match { 129 | case Some(resp) => resp 130 | case None => { 131 | val resp = f(x) 132 | resp.exitValue match { 133 | case 0 => Web.cache.getOrElseUpdate(x, resp) 134 | case _ => resp 135 | } 136 | } 137 | 138 | } 139 | } 140 | } 141 | override def doGet(req: HttpServletRequest, resp: HttpServletResponse) = { 142 | val code = req.getParameter("code") 143 | if (code == null) 144 | resp.setStatus(404) 145 | else { 146 | var f = memo(compileAndRun) 147 | val model = f(code) 148 | resp.getWriter().print(json(model)) 149 | resp.getWriter().flush() 150 | resp.setStatus(200) 151 | } 152 | } 153 | 154 | override def doPost(req: HttpServletRequest, resp: HttpServletResponse) = { 155 | doGet(req, resp) 156 | } 157 | } 158 | 159 | case class RunResponse(errors: List[String], events: List[String], errEvents: List[String], exitValue: Int) { 160 | 161 | } -------------------------------------------------------------------------------- /src/main/scala/com/yankay/scalaTour/script/TaskMem.scala: -------------------------------------------------------------------------------- 1 | package com.yankay.scalaTour.script 2 | 3 | import java.util.Date 4 | import scala.actors.Actor._ 5 | import scala.actors.TIMEOUT 6 | import org.apache.commons.logging.LogFactory 7 | 8 | object CompileStatus extends Enumeration { 9 | type T = Value 10 | val Perpared, Compileing, Compiled = Value 11 | } 12 | 13 | object RunStatus extends Enumeration { 14 | type T = Value 15 | val Perpared, Running, Runned = Value 16 | } 17 | 18 | case class Task(val taskId: Int, val script: String, 19 | val compiledPath: Option[String], val compileLog: List[String], val compileStatus: CompileStatus.T, 20 | val stdOut: List[String], val stdErr: List[String], 21 | val runStart: Option[Date], val runEnd: Option[Date], 22 | val runStatus: RunStatus.T, val runCode: Option[Int], 23 | val expireAt: Date) 24 | 25 | object TaskMem { 26 | 27 | val logger = LogFactory.getLog("TaskMem") 28 | 29 | var tasks = List[Task]() 30 | 31 | def getById(id: Int): Option[Task] = { 32 | TaskMem.synchronized( 33 | tasks.find(_.taskId == id)) 34 | } 35 | 36 | def getByScript(script: String): Option[Task] = { 37 | TaskMem.synchronized( 38 | tasks.find(_.script == script)) 39 | } 40 | 41 | def putAndReplace(task: Task) = { 42 | logger.debug("putAndReplace " + task) 43 | TaskMem.synchronized( 44 | tasks = task :: tasks.filterNot(_.taskId == task.taskId)) 45 | } 46 | 47 | actor { 48 | loop { 49 | reactWithin(60 * 1000) { 50 | case TIMEOUT => removeExpire 51 | } 52 | } 53 | } 54 | 55 | def removeExpire() = { 56 | logger.debug("removeExpire") 57 | TaskMem.synchronized( 58 | tasks = tasks.filter(_.expireAt.after(new Date()))) 59 | } 60 | 61 | var id = 0 62 | 63 | def generateId(): Int = { 64 | TaskMem.synchronized { 65 | id = id + 1; 66 | return id 67 | } 68 | } 69 | 70 | } -------------------------------------------------------------------------------- /src/test/scala/com/yankay/scalaTour/ScalaScriptCompilerTest.scala: -------------------------------------------------------------------------------- 1 | package com.yankay.scalaTour 2 | 3 | import java.util.Date 4 | import java.text.SimpleDateFormat 5 | import scala.util.matching.Regex 6 | 7 | object ScalaScriptCompilerTest { 8 | // this only handles functions with a single argument. 9 | 10 | } 11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/test/scala/com/yankay/scalaTour/script/TaskMem.scala: -------------------------------------------------------------------------------- 1 | package com.yankay.scalaTour.script 2 | 3 | import java.util.Date 4 | 5 | object TaskMemTest { 6 | def main(args: Array[String]) { 7 | 8 | } 9 | 10 | 11 | 12 | 13 | } -------------------------------------------------------------------------------- /src/test/scala/test.scala: -------------------------------------------------------------------------------- 1 | 2 | 3 | import scala.util.matching.Regex 4 | 5 | object test { 6 | def main(args: Array[String]) { 7 | import scala.collection.mutable.WeakHashMap 8 | 9 | val cache = new WeakHashMap[Int, Int] 10 | def memo(f: Int => Int) = (x: Int) => cache.getOrElseUpdate(x, f(x)) 11 | 12 | def fibonacci_(in: Int): Int = in match { 13 | case 0 => 0; 14 | case 1 => 1; 15 | case n: Int => memo(fibonacci_)(n - 1) + memo(fibonacci_)(n - 2) 16 | } 17 | 18 | val fibonacci: Int => Int = memo(fibonacci_) 19 | 20 | val t1 = System.currentTimeMillis() 21 | println(fibonacci(40)) 22 | println("it takes " + (System.currentTimeMillis() - t1) + "ms") 23 | 24 | val t2 = System.currentTimeMillis() 25 | println(fibonacci(40)) 26 | println("it takes " + (System.currentTimeMillis() - t2) + "ms") 27 | 28 | } 29 | 30 | } 31 | 32 | 33 | 34 | -------------------------------------------------------------------------------- /webapp/apple-touch-icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/apple-touch-icon.png -------------------------------------------------------------------------------- /webapp/css/bootstrap-responsive.css: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap Responsive v2.3.1 3 | * 4 | * Copyright 2012 Twitter, Inc 5 | * Licensed under the Apache License v2.0 6 | * http://www.apache.org/licenses/LICENSE-2.0 7 | * 8 | * Designed and built with all the love in the world @twitter by @mdo and @fat. 9 | */ 10 | 11 | .clearfix { 12 | *zoom: 1; 13 | } 14 | 15 | .clearfix:before, 16 | .clearfix:after { 17 | display: table; 18 | line-height: 0; 19 | content: ""; 20 | } 21 | 22 | .clearfix:after { 23 | clear: both; 24 | } 25 | 26 | .hide-text { 27 | font: 0/0 a; 28 | color: transparent; 29 | text-shadow: none; 30 | background-color: transparent; 31 | border: 0; 32 | } 33 | 34 | .input-block-level { 35 | display: block; 36 | width: 100%; 37 | min-height: 30px; 38 | -webkit-box-sizing: border-box; 39 | -moz-box-sizing: border-box; 40 | box-sizing: border-box; 41 | } 42 | 43 | @-ms-viewport { 44 | width: device-width; 45 | } 46 | 47 | .hidden { 48 | display: none; 49 | visibility: hidden; 50 | } 51 | 52 | .visible-phone { 53 | display: none !important; 54 | } 55 | 56 | .visible-tablet { 57 | display: none !important; 58 | } 59 | 60 | .hidden-desktop { 61 | display: none !important; 62 | } 63 | 64 | .visible-desktop { 65 | display: inherit !important; 66 | } 67 | 68 | @media (min-width: 768px) and (max-width: 979px) { 69 | .hidden-desktop { 70 | display: inherit !important; 71 | } 72 | .visible-desktop { 73 | display: none !important ; 74 | } 75 | .visible-tablet { 76 | display: inherit !important; 77 | } 78 | .hidden-tablet { 79 | display: none !important; 80 | } 81 | } 82 | 83 | @media (max-width: 767px) { 84 | .hidden-desktop { 85 | display: inherit !important; 86 | } 87 | .visible-desktop { 88 | display: none !important; 89 | } 90 | .visible-phone { 91 | display: inherit !important; 92 | } 93 | .hidden-phone { 94 | display: none !important; 95 | } 96 | } 97 | 98 | .visible-print { 99 | display: none !important; 100 | } 101 | 102 | @media print { 103 | .visible-print { 104 | display: inherit !important; 105 | } 106 | .hidden-print { 107 | display: none !important; 108 | } 109 | } 110 | 111 | @media (min-width: 1200px) { 112 | .row { 113 | margin-left: -30px; 114 | *zoom: 1; 115 | } 116 | .row:before, 117 | .row:after { 118 | display: table; 119 | line-height: 0; 120 | content: ""; 121 | } 122 | .row:after { 123 | clear: both; 124 | } 125 | [class*="span"] { 126 | float: left; 127 | min-height: 1px; 128 | margin-left: 30px; 129 | } 130 | .container, 131 | .navbar-static-top .container, 132 | .navbar-fixed-top .container, 133 | .navbar-fixed-bottom .container { 134 | width: 1170px; 135 | } 136 | .span12 { 137 | width: 1170px; 138 | } 139 | .span11 { 140 | width: 1070px; 141 | } 142 | .span10 { 143 | width: 970px; 144 | } 145 | .span9 { 146 | width: 870px; 147 | } 148 | .span8 { 149 | width: 770px; 150 | } 151 | .span7 { 152 | width: 670px; 153 | } 154 | .span6 { 155 | width: 570px; 156 | } 157 | .span5 { 158 | width: 470px; 159 | } 160 | .span4 { 161 | width: 370px; 162 | } 163 | .span3 { 164 | width: 270px; 165 | } 166 | .span2 { 167 | width: 170px; 168 | } 169 | .span1 { 170 | width: 70px; 171 | } 172 | .offset12 { 173 | margin-left: 1230px; 174 | } 175 | .offset11 { 176 | margin-left: 1130px; 177 | } 178 | .offset10 { 179 | margin-left: 1030px; 180 | } 181 | .offset9 { 182 | margin-left: 930px; 183 | } 184 | .offset8 { 185 | margin-left: 830px; 186 | } 187 | .offset7 { 188 | margin-left: 730px; 189 | } 190 | .offset6 { 191 | margin-left: 630px; 192 | } 193 | .offset5 { 194 | margin-left: 530px; 195 | } 196 | .offset4 { 197 | margin-left: 430px; 198 | } 199 | .offset3 { 200 | margin-left: 330px; 201 | } 202 | .offset2 { 203 | margin-left: 230px; 204 | } 205 | .offset1 { 206 | margin-left: 130px; 207 | } 208 | .row-fluid { 209 | width: 100%; 210 | *zoom: 1; 211 | } 212 | .row-fluid:before, 213 | .row-fluid:after { 214 | display: table; 215 | line-height: 0; 216 | content: ""; 217 | } 218 | .row-fluid:after { 219 | clear: both; 220 | } 221 | .row-fluid [class*="span"] { 222 | display: block; 223 | float: left; 224 | width: 100%; 225 | min-height: 30px; 226 | margin-left: 2.564102564102564%; 227 | *margin-left: 2.5109110747408616%; 228 | -webkit-box-sizing: border-box; 229 | -moz-box-sizing: border-box; 230 | box-sizing: border-box; 231 | } 232 | .row-fluid [class*="span"]:first-child { 233 | margin-left: 0; 234 | } 235 | .row-fluid .controls-row [class*="span"] + [class*="span"] { 236 | margin-left: 2.564102564102564%; 237 | } 238 | .row-fluid .span12 { 239 | width: 100%; 240 | *width: 99.94680851063829%; 241 | } 242 | .row-fluid .span11 { 243 | width: 91.45299145299145%; 244 | *width: 91.39979996362975%; 245 | } 246 | .row-fluid .span10 { 247 | width: 82.90598290598291%; 248 | *width: 82.8527914166212%; 249 | } 250 | .row-fluid .span9 { 251 | width: 74.35897435897436%; 252 | *width: 74.30578286961266%; 253 | } 254 | .row-fluid .span8 { 255 | width: 65.81196581196582%; 256 | *width: 65.75877432260411%; 257 | } 258 | .row-fluid .span7 { 259 | width: 57.26495726495726%; 260 | *width: 57.21176577559556%; 261 | } 262 | .row-fluid .span6 { 263 | width: 48.717948717948715%; 264 | *width: 48.664757228587014%; 265 | } 266 | .row-fluid .span5 { 267 | width: 40.17094017094017%; 268 | *width: 40.11774868157847%; 269 | } 270 | .row-fluid .span4 { 271 | width: 31.623931623931625%; 272 | *width: 31.570740134569924%; 273 | } 274 | .row-fluid .span3 { 275 | width: 23.076923076923077%; 276 | *width: 23.023731587561375%; 277 | } 278 | .row-fluid .span2 { 279 | width: 14.52991452991453%; 280 | *width: 14.476723040552828%; 281 | } 282 | .row-fluid .span1 { 283 | width: 5.982905982905983%; 284 | *width: 5.929714493544281%; 285 | } 286 | .row-fluid .offset12 { 287 | margin-left: 105.12820512820512%; 288 | *margin-left: 105.02182214948171%; 289 | } 290 | .row-fluid .offset12:first-child { 291 | margin-left: 102.56410256410257%; 292 | *margin-left: 102.45771958537915%; 293 | } 294 | .row-fluid .offset11 { 295 | margin-left: 96.58119658119658%; 296 | *margin-left: 96.47481360247316%; 297 | } 298 | .row-fluid .offset11:first-child { 299 | margin-left: 94.01709401709402%; 300 | *margin-left: 93.91071103837061%; 301 | } 302 | .row-fluid .offset10 { 303 | margin-left: 88.03418803418803%; 304 | *margin-left: 87.92780505546462%; 305 | } 306 | .row-fluid .offset10:first-child { 307 | margin-left: 85.47008547008548%; 308 | *margin-left: 85.36370249136206%; 309 | } 310 | .row-fluid .offset9 { 311 | margin-left: 79.48717948717949%; 312 | *margin-left: 79.38079650845607%; 313 | } 314 | .row-fluid .offset9:first-child { 315 | margin-left: 76.92307692307693%; 316 | *margin-left: 76.81669394435352%; 317 | } 318 | .row-fluid .offset8 { 319 | margin-left: 70.94017094017094%; 320 | *margin-left: 70.83378796144753%; 321 | } 322 | .row-fluid .offset8:first-child { 323 | margin-left: 68.37606837606839%; 324 | *margin-left: 68.26968539734497%; 325 | } 326 | .row-fluid .offset7 { 327 | margin-left: 62.393162393162385%; 328 | *margin-left: 62.28677941443899%; 329 | } 330 | .row-fluid .offset7:first-child { 331 | margin-left: 59.82905982905982%; 332 | *margin-left: 59.72267685033642%; 333 | } 334 | .row-fluid .offset6 { 335 | margin-left: 53.84615384615384%; 336 | *margin-left: 53.739770867430444%; 337 | } 338 | .row-fluid .offset6:first-child { 339 | margin-left: 51.28205128205128%; 340 | *margin-left: 51.175668303327875%; 341 | } 342 | .row-fluid .offset5 { 343 | margin-left: 45.299145299145295%; 344 | *margin-left: 45.1927623204219%; 345 | } 346 | .row-fluid .offset5:first-child { 347 | margin-left: 42.73504273504273%; 348 | *margin-left: 42.62865975631933%; 349 | } 350 | .row-fluid .offset4 { 351 | margin-left: 36.75213675213675%; 352 | *margin-left: 36.645753773413354%; 353 | } 354 | .row-fluid .offset4:first-child { 355 | margin-left: 34.18803418803419%; 356 | *margin-left: 34.081651209310785%; 357 | } 358 | .row-fluid .offset3 { 359 | margin-left: 28.205128205128204%; 360 | *margin-left: 28.0987452264048%; 361 | } 362 | .row-fluid .offset3:first-child { 363 | margin-left: 25.641025641025642%; 364 | *margin-left: 25.53464266230224%; 365 | } 366 | .row-fluid .offset2 { 367 | margin-left: 19.65811965811966%; 368 | *margin-left: 19.551736679396257%; 369 | } 370 | .row-fluid .offset2:first-child { 371 | margin-left: 17.094017094017094%; 372 | *margin-left: 16.98763411529369%; 373 | } 374 | .row-fluid .offset1 { 375 | margin-left: 11.11111111111111%; 376 | *margin-left: 11.004728132387708%; 377 | } 378 | .row-fluid .offset1:first-child { 379 | margin-left: 8.547008547008547%; 380 | *margin-left: 8.440625568285142%; 381 | } 382 | input, 383 | textarea, 384 | .uneditable-input { 385 | margin-left: 0; 386 | } 387 | .controls-row [class*="span"] + [class*="span"] { 388 | margin-left: 30px; 389 | } 390 | input.span12, 391 | textarea.span12, 392 | .uneditable-input.span12 { 393 | width: 1156px; 394 | } 395 | input.span11, 396 | textarea.span11, 397 | .uneditable-input.span11 { 398 | width: 1056px; 399 | } 400 | input.span10, 401 | textarea.span10, 402 | .uneditable-input.span10 { 403 | width: 956px; 404 | } 405 | input.span9, 406 | textarea.span9, 407 | .uneditable-input.span9 { 408 | width: 856px; 409 | } 410 | input.span8, 411 | textarea.span8, 412 | .uneditable-input.span8 { 413 | width: 756px; 414 | } 415 | input.span7, 416 | textarea.span7, 417 | .uneditable-input.span7 { 418 | width: 656px; 419 | } 420 | input.span6, 421 | textarea.span6, 422 | .uneditable-input.span6 { 423 | width: 556px; 424 | } 425 | input.span5, 426 | textarea.span5, 427 | .uneditable-input.span5 { 428 | width: 456px; 429 | } 430 | input.span4, 431 | textarea.span4, 432 | .uneditable-input.span4 { 433 | width: 356px; 434 | } 435 | input.span3, 436 | textarea.span3, 437 | .uneditable-input.span3 { 438 | width: 256px; 439 | } 440 | input.span2, 441 | textarea.span2, 442 | .uneditable-input.span2 { 443 | width: 156px; 444 | } 445 | input.span1, 446 | textarea.span1, 447 | .uneditable-input.span1 { 448 | width: 56px; 449 | } 450 | .thumbnails { 451 | margin-left: -30px; 452 | } 453 | .thumbnails > li { 454 | margin-left: 30px; 455 | } 456 | .row-fluid .thumbnails { 457 | margin-left: 0; 458 | } 459 | } 460 | 461 | @media (min-width: 768px) and (max-width: 979px) { 462 | .row { 463 | margin-left: -20px; 464 | *zoom: 1; 465 | } 466 | .row:before, 467 | .row:after { 468 | display: table; 469 | line-height: 0; 470 | content: ""; 471 | } 472 | .row:after { 473 | clear: both; 474 | } 475 | [class*="span"] { 476 | float: left; 477 | min-height: 1px; 478 | margin-left: 20px; 479 | } 480 | .container, 481 | .navbar-static-top .container, 482 | .navbar-fixed-top .container, 483 | .navbar-fixed-bottom .container { 484 | width: 724px; 485 | } 486 | .span12 { 487 | width: 724px; 488 | } 489 | .span11 { 490 | width: 662px; 491 | } 492 | .span10 { 493 | width: 600px; 494 | } 495 | .span9 { 496 | width: 538px; 497 | } 498 | .span8 { 499 | width: 476px; 500 | } 501 | .span7 { 502 | width: 414px; 503 | } 504 | .span6 { 505 | width: 352px; 506 | } 507 | .span5 { 508 | width: 290px; 509 | } 510 | .span4 { 511 | width: 228px; 512 | } 513 | .span3 { 514 | width: 166px; 515 | } 516 | .span2 { 517 | width: 104px; 518 | } 519 | .span1 { 520 | width: 42px; 521 | } 522 | .offset12 { 523 | margin-left: 764px; 524 | } 525 | .offset11 { 526 | margin-left: 702px; 527 | } 528 | .offset10 { 529 | margin-left: 640px; 530 | } 531 | .offset9 { 532 | margin-left: 578px; 533 | } 534 | .offset8 { 535 | margin-left: 516px; 536 | } 537 | .offset7 { 538 | margin-left: 454px; 539 | } 540 | .offset6 { 541 | margin-left: 392px; 542 | } 543 | .offset5 { 544 | margin-left: 330px; 545 | } 546 | .offset4 { 547 | margin-left: 268px; 548 | } 549 | .offset3 { 550 | margin-left: 206px; 551 | } 552 | .offset2 { 553 | margin-left: 144px; 554 | } 555 | .offset1 { 556 | margin-left: 82px; 557 | } 558 | .row-fluid { 559 | width: 100%; 560 | *zoom: 1; 561 | } 562 | .row-fluid:before, 563 | .row-fluid:after { 564 | display: table; 565 | line-height: 0; 566 | content: ""; 567 | } 568 | .row-fluid:after { 569 | clear: both; 570 | } 571 | .row-fluid [class*="span"] { 572 | display: block; 573 | float: left; 574 | width: 100%; 575 | min-height: 30px; 576 | margin-left: 2.7624309392265194%; 577 | *margin-left: 2.709239449864817%; 578 | -webkit-box-sizing: border-box; 579 | -moz-box-sizing: border-box; 580 | box-sizing: border-box; 581 | } 582 | .row-fluid [class*="span"]:first-child { 583 | margin-left: 0; 584 | } 585 | .row-fluid .controls-row [class*="span"] + [class*="span"] { 586 | margin-left: 2.7624309392265194%; 587 | } 588 | .row-fluid .span12 { 589 | width: 100%; 590 | *width: 99.94680851063829%; 591 | } 592 | .row-fluid .span11 { 593 | width: 91.43646408839778%; 594 | *width: 91.38327259903608%; 595 | } 596 | .row-fluid .span10 { 597 | width: 82.87292817679558%; 598 | *width: 82.81973668743387%; 599 | } 600 | .row-fluid .span9 { 601 | width: 74.30939226519337%; 602 | *width: 74.25620077583166%; 603 | } 604 | .row-fluid .span8 { 605 | width: 65.74585635359117%; 606 | *width: 65.69266486422946%; 607 | } 608 | .row-fluid .span7 { 609 | width: 57.18232044198895%; 610 | *width: 57.12912895262725%; 611 | } 612 | .row-fluid .span6 { 613 | width: 48.61878453038674%; 614 | *width: 48.56559304102504%; 615 | } 616 | .row-fluid .span5 { 617 | width: 40.05524861878453%; 618 | *width: 40.00205712942283%; 619 | } 620 | .row-fluid .span4 { 621 | width: 31.491712707182323%; 622 | *width: 31.43852121782062%; 623 | } 624 | .row-fluid .span3 { 625 | width: 22.92817679558011%; 626 | *width: 22.87498530621841%; 627 | } 628 | .row-fluid .span2 { 629 | width: 14.3646408839779%; 630 | *width: 14.311449394616199%; 631 | } 632 | .row-fluid .span1 { 633 | width: 5.801104972375691%; 634 | *width: 5.747913483013988%; 635 | } 636 | .row-fluid .offset12 { 637 | margin-left: 105.52486187845304%; 638 | *margin-left: 105.41847889972962%; 639 | } 640 | .row-fluid .offset12:first-child { 641 | margin-left: 102.76243093922652%; 642 | *margin-left: 102.6560479605031%; 643 | } 644 | .row-fluid .offset11 { 645 | margin-left: 96.96132596685082%; 646 | *margin-left: 96.8549429881274%; 647 | } 648 | .row-fluid .offset11:first-child { 649 | margin-left: 94.1988950276243%; 650 | *margin-left: 94.09251204890089%; 651 | } 652 | .row-fluid .offset10 { 653 | margin-left: 88.39779005524862%; 654 | *margin-left: 88.2914070765252%; 655 | } 656 | .row-fluid .offset10:first-child { 657 | margin-left: 85.6353591160221%; 658 | *margin-left: 85.52897613729868%; 659 | } 660 | .row-fluid .offset9 { 661 | margin-left: 79.8342541436464%; 662 | *margin-left: 79.72787116492299%; 663 | } 664 | .row-fluid .offset9:first-child { 665 | margin-left: 77.07182320441989%; 666 | *margin-left: 76.96544022569647%; 667 | } 668 | .row-fluid .offset8 { 669 | margin-left: 71.2707182320442%; 670 | *margin-left: 71.16433525332079%; 671 | } 672 | .row-fluid .offset8:first-child { 673 | margin-left: 68.50828729281768%; 674 | *margin-left: 68.40190431409427%; 675 | } 676 | .row-fluid .offset7 { 677 | margin-left: 62.70718232044199%; 678 | *margin-left: 62.600799341718584%; 679 | } 680 | .row-fluid .offset7:first-child { 681 | margin-left: 59.94475138121547%; 682 | *margin-left: 59.838368402492065%; 683 | } 684 | .row-fluid .offset6 { 685 | margin-left: 54.14364640883978%; 686 | *margin-left: 54.037263430116376%; 687 | } 688 | .row-fluid .offset6:first-child { 689 | margin-left: 51.38121546961326%; 690 | *margin-left: 51.27483249088986%; 691 | } 692 | .row-fluid .offset5 { 693 | margin-left: 45.58011049723757%; 694 | *margin-left: 45.47372751851417%; 695 | } 696 | .row-fluid .offset5:first-child { 697 | margin-left: 42.81767955801105%; 698 | *margin-left: 42.71129657928765%; 699 | } 700 | .row-fluid .offset4 { 701 | margin-left: 37.01657458563536%; 702 | *margin-left: 36.91019160691196%; 703 | } 704 | .row-fluid .offset4:first-child { 705 | margin-left: 34.25414364640884%; 706 | *margin-left: 34.14776066768544%; 707 | } 708 | .row-fluid .offset3 { 709 | margin-left: 28.45303867403315%; 710 | *margin-left: 28.346655695309746%; 711 | } 712 | .row-fluid .offset3:first-child { 713 | margin-left: 25.69060773480663%; 714 | *margin-left: 25.584224756083227%; 715 | } 716 | .row-fluid .offset2 { 717 | margin-left: 19.88950276243094%; 718 | *margin-left: 19.783119783707537%; 719 | } 720 | .row-fluid .offset2:first-child { 721 | margin-left: 17.12707182320442%; 722 | *margin-left: 17.02068884448102%; 723 | } 724 | .row-fluid .offset1 { 725 | margin-left: 11.32596685082873%; 726 | *margin-left: 11.219583872105325%; 727 | } 728 | .row-fluid .offset1:first-child { 729 | margin-left: 8.56353591160221%; 730 | *margin-left: 8.457152932878806%; 731 | } 732 | input, 733 | textarea, 734 | .uneditable-input { 735 | margin-left: 0; 736 | } 737 | .controls-row [class*="span"] + [class*="span"] { 738 | margin-left: 20px; 739 | } 740 | input.span12, 741 | textarea.span12, 742 | .uneditable-input.span12 { 743 | width: 710px; 744 | } 745 | input.span11, 746 | textarea.span11, 747 | .uneditable-input.span11 { 748 | width: 648px; 749 | } 750 | input.span10, 751 | textarea.span10, 752 | .uneditable-input.span10 { 753 | width: 586px; 754 | } 755 | input.span9, 756 | textarea.span9, 757 | .uneditable-input.span9 { 758 | width: 524px; 759 | } 760 | input.span8, 761 | textarea.span8, 762 | .uneditable-input.span8 { 763 | width: 462px; 764 | } 765 | input.span7, 766 | textarea.span7, 767 | .uneditable-input.span7 { 768 | width: 400px; 769 | } 770 | input.span6, 771 | textarea.span6, 772 | .uneditable-input.span6 { 773 | width: 338px; 774 | } 775 | input.span5, 776 | textarea.span5, 777 | .uneditable-input.span5 { 778 | width: 276px; 779 | } 780 | input.span4, 781 | textarea.span4, 782 | .uneditable-input.span4 { 783 | width: 214px; 784 | } 785 | input.span3, 786 | textarea.span3, 787 | .uneditable-input.span3 { 788 | width: 152px; 789 | } 790 | input.span2, 791 | textarea.span2, 792 | .uneditable-input.span2 { 793 | width: 90px; 794 | } 795 | input.span1, 796 | textarea.span1, 797 | .uneditable-input.span1 { 798 | width: 28px; 799 | } 800 | } 801 | 802 | @media (max-width: 767px) { 803 | body { 804 | padding-right: 20px; 805 | padding-left: 20px; 806 | } 807 | .navbar-fixed-top, 808 | .navbar-fixed-bottom, 809 | .navbar-static-top { 810 | margin-right: -20px; 811 | margin-left: -20px; 812 | } 813 | .container-fluid { 814 | padding: 0; 815 | } 816 | .dl-horizontal dt { 817 | float: none; 818 | width: auto; 819 | clear: none; 820 | text-align: left; 821 | } 822 | .dl-horizontal dd { 823 | margin-left: 0; 824 | } 825 | .container { 826 | width: auto; 827 | } 828 | .row-fluid { 829 | width: 100%; 830 | } 831 | .row, 832 | .thumbnails { 833 | margin-left: 0; 834 | } 835 | .thumbnails > li { 836 | float: none; 837 | margin-left: 0; 838 | } 839 | [class*="span"], 840 | .uneditable-input[class*="span"], 841 | .row-fluid [class*="span"] { 842 | display: block; 843 | float: none; 844 | width: 100%; 845 | margin-left: 0; 846 | -webkit-box-sizing: border-box; 847 | -moz-box-sizing: border-box; 848 | box-sizing: border-box; 849 | } 850 | .span12, 851 | .row-fluid .span12 { 852 | width: 100%; 853 | -webkit-box-sizing: border-box; 854 | -moz-box-sizing: border-box; 855 | box-sizing: border-box; 856 | } 857 | .row-fluid [class*="offset"]:first-child { 858 | margin-left: 0; 859 | } 860 | .input-large, 861 | .input-xlarge, 862 | .input-xxlarge, 863 | input[class*="span"], 864 | select[class*="span"], 865 | textarea[class*="span"], 866 | .uneditable-input { 867 | display: block; 868 | width: 100%; 869 | min-height: 30px; 870 | -webkit-box-sizing: border-box; 871 | -moz-box-sizing: border-box; 872 | box-sizing: border-box; 873 | } 874 | .input-prepend input, 875 | .input-append input, 876 | .input-prepend input[class*="span"], 877 | .input-append input[class*="span"] { 878 | display: inline-block; 879 | width: auto; 880 | } 881 | .controls-row [class*="span"] + [class*="span"] { 882 | margin-left: 0; 883 | } 884 | .modal { 885 | position: fixed; 886 | top: 20px; 887 | right: 20px; 888 | left: 20px; 889 | width: auto; 890 | margin: 0; 891 | } 892 | .modal.fade { 893 | top: -100px; 894 | } 895 | .modal.fade.in { 896 | top: 20px; 897 | } 898 | } 899 | 900 | @media (max-width: 480px) { 901 | .nav-collapse { 902 | -webkit-transform: translate3d(0, 0, 0); 903 | } 904 | .page-header h1 small { 905 | display: block; 906 | line-height: 20px; 907 | } 908 | input[type="checkbox"], 909 | input[type="radio"] { 910 | border: 1px solid #ccc; 911 | } 912 | .form-horizontal .control-label { 913 | float: none; 914 | width: auto; 915 | padding-top: 0; 916 | text-align: left; 917 | } 918 | .form-horizontal .controls { 919 | margin-left: 0; 920 | } 921 | .form-horizontal .control-list { 922 | padding-top: 0; 923 | } 924 | .form-horizontal .form-actions { 925 | padding-right: 10px; 926 | padding-left: 10px; 927 | } 928 | .media .pull-left, 929 | .media .pull-right { 930 | display: block; 931 | float: none; 932 | margin-bottom: 10px; 933 | } 934 | .media-object { 935 | margin-right: 0; 936 | margin-left: 0; 937 | } 938 | .modal { 939 | top: 10px; 940 | right: 10px; 941 | left: 10px; 942 | } 943 | .modal-header .close { 944 | padding: 10px; 945 | margin: -10px; 946 | } 947 | .carousel-caption { 948 | position: static; 949 | } 950 | } 951 | 952 | @media (max-width: 979px) { 953 | body { 954 | padding-top: 0; 955 | } 956 | .navbar-fixed-top, 957 | .navbar-fixed-bottom { 958 | position: static; 959 | } 960 | .navbar-fixed-top { 961 | margin-bottom: 20px; 962 | } 963 | .navbar-fixed-bottom { 964 | margin-top: 20px; 965 | } 966 | .navbar-fixed-top .navbar-inner, 967 | .navbar-fixed-bottom .navbar-inner { 968 | padding: 5px; 969 | } 970 | .navbar .container { 971 | width: auto; 972 | padding: 0; 973 | } 974 | .navbar .brand { 975 | padding-right: 10px; 976 | padding-left: 10px; 977 | margin: 0 0 0 -5px; 978 | } 979 | .nav-collapse { 980 | clear: both; 981 | } 982 | .nav-collapse .nav { 983 | float: none; 984 | margin: 0 0 10px; 985 | } 986 | .nav-collapse .nav > li { 987 | float: none; 988 | } 989 | .nav-collapse .nav > li > a { 990 | margin-bottom: 2px; 991 | } 992 | .nav-collapse .nav > .divider-vertical { 993 | display: none; 994 | } 995 | .nav-collapse .nav .nav-header { 996 | color: #777777; 997 | text-shadow: none; 998 | } 999 | .nav-collapse .nav > li > a, 1000 | .nav-collapse .dropdown-menu a { 1001 | padding: 9px 15px; 1002 | font-weight: bold; 1003 | color: #777777; 1004 | -webkit-border-radius: 3px; 1005 | -moz-border-radius: 3px; 1006 | border-radius: 3px; 1007 | } 1008 | .nav-collapse .btn { 1009 | padding: 4px 10px 4px; 1010 | font-weight: normal; 1011 | -webkit-border-radius: 4px; 1012 | -moz-border-radius: 4px; 1013 | border-radius: 4px; 1014 | } 1015 | .nav-collapse .dropdown-menu li + li a { 1016 | margin-bottom: 2px; 1017 | } 1018 | .nav-collapse .nav > li > a:hover, 1019 | .nav-collapse .nav > li > a:focus, 1020 | .nav-collapse .dropdown-menu a:hover, 1021 | .nav-collapse .dropdown-menu a:focus { 1022 | background-color: #f2f2f2; 1023 | } 1024 | .navbar-inverse .nav-collapse .nav > li > a, 1025 | .navbar-inverse .nav-collapse .dropdown-menu a { 1026 | color: #999999; 1027 | } 1028 | .navbar-inverse .nav-collapse .nav > li > a:hover, 1029 | .navbar-inverse .nav-collapse .nav > li > a:focus, 1030 | .navbar-inverse .nav-collapse .dropdown-menu a:hover, 1031 | .navbar-inverse .nav-collapse .dropdown-menu a:focus { 1032 | background-color: #111111; 1033 | } 1034 | .nav-collapse.in .btn-group { 1035 | padding: 0; 1036 | margin-top: 5px; 1037 | } 1038 | .nav-collapse .dropdown-menu { 1039 | position: static; 1040 | top: auto; 1041 | left: auto; 1042 | display: none; 1043 | float: none; 1044 | max-width: none; 1045 | padding: 0; 1046 | margin: 0 15px; 1047 | background-color: transparent; 1048 | border: none; 1049 | -webkit-border-radius: 0; 1050 | -moz-border-radius: 0; 1051 | border-radius: 0; 1052 | -webkit-box-shadow: none; 1053 | -moz-box-shadow: none; 1054 | box-shadow: none; 1055 | } 1056 | .nav-collapse .open > .dropdown-menu { 1057 | display: block; 1058 | } 1059 | .nav-collapse .dropdown-menu:before, 1060 | .nav-collapse .dropdown-menu:after { 1061 | display: none; 1062 | } 1063 | .nav-collapse .dropdown-menu .divider { 1064 | display: none; 1065 | } 1066 | .nav-collapse .nav > li > .dropdown-menu:before, 1067 | .nav-collapse .nav > li > .dropdown-menu:after { 1068 | display: none; 1069 | } 1070 | .nav-collapse .navbar-form, 1071 | .nav-collapse .navbar-search { 1072 | float: none; 1073 | padding: 10px 15px; 1074 | margin: 10px 0; 1075 | border-top: 1px solid #f2f2f2; 1076 | border-bottom: 1px solid #f2f2f2; 1077 | -webkit-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1); 1078 | -moz-box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1); 1079 | box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.1), 0 1px 0 rgba(255, 255, 255, 0.1); 1080 | } 1081 | .navbar-inverse .nav-collapse .navbar-form, 1082 | .navbar-inverse .nav-collapse .navbar-search { 1083 | border-top-color: #111111; 1084 | border-bottom-color: #111111; 1085 | } 1086 | .navbar .nav-collapse .nav.pull-right { 1087 | float: none; 1088 | margin-left: 0; 1089 | } 1090 | .nav-collapse, 1091 | .nav-collapse.collapse { 1092 | height: 0; 1093 | overflow: hidden; 1094 | } 1095 | .navbar .btn-navbar { 1096 | display: block; 1097 | } 1098 | .navbar-static .navbar-inner { 1099 | padding-right: 10px; 1100 | padding-left: 10px; 1101 | } 1102 | } 1103 | 1104 | @media (min-width: 980px) { 1105 | .nav-collapse.collapse { 1106 | height: auto !important; 1107 | overflow: visible !important; 1108 | } 1109 | } 1110 | -------------------------------------------------------------------------------- /webapp/css/bootstrap-responsive.min.css: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap Responsive v2.3.1 3 | * 4 | * Copyright 2012 Twitter, Inc 5 | * Licensed under the Apache License v2.0 6 | * http://www.apache.org/licenses/LICENSE-2.0 7 | * 8 | * Designed and built with all the love in the world @twitter by @mdo and @fat. 9 | */.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;line-height:0;content:""}.clearfix:after{clear:both}.hide-text{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.input-block-level{display:block;width:100%;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}@-ms-viewport{width:device-width}.hidden{display:none;visibility:hidden}.visible-phone{display:none!important}.visible-tablet{display:none!important}.hidden-desktop{display:none!important}.visible-desktop{display:inherit!important}@media(min-width:768px) and (max-width:979px){.hidden-desktop{display:inherit!important}.visible-desktop{display:none!important}.visible-tablet{display:inherit!important}.hidden-tablet{display:none!important}}@media(max-width:767px){.hidden-desktop{display:inherit!important}.visible-desktop{display:none!important}.visible-phone{display:inherit!important}.hidden-phone{display:none!important}}.visible-print{display:none!important}@media print{.visible-print{display:inherit!important}.hidden-print{display:none!important}}@media(min-width:1200px){.row{margin-left:-30px;*zoom:1}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}[class*="span"]{float:left;min-height:1px;margin-left:30px}.container,.navbar-static-top .container,.navbar-fixed-top .container,.navbar-fixed-bottom .container{width:1170px}.span12{width:1170px}.span11{width:1070px}.span10{width:970px}.span9{width:870px}.span8{width:770px}.span7{width:670px}.span6{width:570px}.span5{width:470px}.span4{width:370px}.span3{width:270px}.span2{width:170px}.span1{width:70px}.offset12{margin-left:1230px}.offset11{margin-left:1130px}.offset10{margin-left:1030px}.offset9{margin-left:930px}.offset8{margin-left:830px}.offset7{margin-left:730px}.offset6{margin-left:630px}.offset5{margin-left:530px}.offset4{margin-left:430px}.offset3{margin-left:330px}.offset2{margin-left:230px}.offset1{margin-left:130px}.row-fluid{width:100%;*zoom:1}.row-fluid:before,.row-fluid:after{display:table;line-height:0;content:""}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;float:left;width:100%;min-height:30px;margin-left:2.564102564102564%;*margin-left:2.5109110747408616%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .controls-row [class*="span"]+[class*="span"]{margin-left:2.564102564102564%}.row-fluid .span12{width:100%;*width:99.94680851063829%}.row-fluid .span11{width:91.45299145299145%;*width:91.39979996362975%}.row-fluid .span10{width:82.90598290598291%;*width:82.8527914166212%}.row-fluid .span9{width:74.35897435897436%;*width:74.30578286961266%}.row-fluid .span8{width:65.81196581196582%;*width:65.75877432260411%}.row-fluid .span7{width:57.26495726495726%;*width:57.21176577559556%}.row-fluid .span6{width:48.717948717948715%;*width:48.664757228587014%}.row-fluid .span5{width:40.17094017094017%;*width:40.11774868157847%}.row-fluid .span4{width:31.623931623931625%;*width:31.570740134569924%}.row-fluid .span3{width:23.076923076923077%;*width:23.023731587561375%}.row-fluid .span2{width:14.52991452991453%;*width:14.476723040552828%}.row-fluid .span1{width:5.982905982905983%;*width:5.929714493544281%}.row-fluid .offset12{margin-left:105.12820512820512%;*margin-left:105.02182214948171%}.row-fluid .offset12:first-child{margin-left:102.56410256410257%;*margin-left:102.45771958537915%}.row-fluid .offset11{margin-left:96.58119658119658%;*margin-left:96.47481360247316%}.row-fluid .offset11:first-child{margin-left:94.01709401709402%;*margin-left:93.91071103837061%}.row-fluid .offset10{margin-left:88.03418803418803%;*margin-left:87.92780505546462%}.row-fluid .offset10:first-child{margin-left:85.47008547008548%;*margin-left:85.36370249136206%}.row-fluid .offset9{margin-left:79.48717948717949%;*margin-left:79.38079650845607%}.row-fluid .offset9:first-child{margin-left:76.92307692307693%;*margin-left:76.81669394435352%}.row-fluid .offset8{margin-left:70.94017094017094%;*margin-left:70.83378796144753%}.row-fluid .offset8:first-child{margin-left:68.37606837606839%;*margin-left:68.26968539734497%}.row-fluid .offset7{margin-left:62.393162393162385%;*margin-left:62.28677941443899%}.row-fluid .offset7:first-child{margin-left:59.82905982905982%;*margin-left:59.72267685033642%}.row-fluid .offset6{margin-left:53.84615384615384%;*margin-left:53.739770867430444%}.row-fluid .offset6:first-child{margin-left:51.28205128205128%;*margin-left:51.175668303327875%}.row-fluid .offset5{margin-left:45.299145299145295%;*margin-left:45.1927623204219%}.row-fluid .offset5:first-child{margin-left:42.73504273504273%;*margin-left:42.62865975631933%}.row-fluid .offset4{margin-left:36.75213675213675%;*margin-left:36.645753773413354%}.row-fluid .offset4:first-child{margin-left:34.18803418803419%;*margin-left:34.081651209310785%}.row-fluid .offset3{margin-left:28.205128205128204%;*margin-left:28.0987452264048%}.row-fluid .offset3:first-child{margin-left:25.641025641025642%;*margin-left:25.53464266230224%}.row-fluid .offset2{margin-left:19.65811965811966%;*margin-left:19.551736679396257%}.row-fluid .offset2:first-child{margin-left:17.094017094017094%;*margin-left:16.98763411529369%}.row-fluid .offset1{margin-left:11.11111111111111%;*margin-left:11.004728132387708%}.row-fluid .offset1:first-child{margin-left:8.547008547008547%;*margin-left:8.440625568285142%}input,textarea,.uneditable-input{margin-left:0}.controls-row [class*="span"]+[class*="span"]{margin-left:30px}input.span12,textarea.span12,.uneditable-input.span12{width:1156px}input.span11,textarea.span11,.uneditable-input.span11{width:1056px}input.span10,textarea.span10,.uneditable-input.span10{width:956px}input.span9,textarea.span9,.uneditable-input.span9{width:856px}input.span8,textarea.span8,.uneditable-input.span8{width:756px}input.span7,textarea.span7,.uneditable-input.span7{width:656px}input.span6,textarea.span6,.uneditable-input.span6{width:556px}input.span5,textarea.span5,.uneditable-input.span5{width:456px}input.span4,textarea.span4,.uneditable-input.span4{width:356px}input.span3,textarea.span3,.uneditable-input.span3{width:256px}input.span2,textarea.span2,.uneditable-input.span2{width:156px}input.span1,textarea.span1,.uneditable-input.span1{width:56px}.thumbnails{margin-left:-30px}.thumbnails>li{margin-left:30px}.row-fluid .thumbnails{margin-left:0}}@media(min-width:768px) and (max-width:979px){.row{margin-left:-20px;*zoom:1}.row:before,.row:after{display:table;line-height:0;content:""}.row:after{clear:both}[class*="span"]{float:left;min-height:1px;margin-left:20px}.container,.navbar-static-top .container,.navbar-fixed-top .container,.navbar-fixed-bottom .container{width:724px}.span12{width:724px}.span11{width:662px}.span10{width:600px}.span9{width:538px}.span8{width:476px}.span7{width:414px}.span6{width:352px}.span5{width:290px}.span4{width:228px}.span3{width:166px}.span2{width:104px}.span1{width:42px}.offset12{margin-left:764px}.offset11{margin-left:702px}.offset10{margin-left:640px}.offset9{margin-left:578px}.offset8{margin-left:516px}.offset7{margin-left:454px}.offset6{margin-left:392px}.offset5{margin-left:330px}.offset4{margin-left:268px}.offset3{margin-left:206px}.offset2{margin-left:144px}.offset1{margin-left:82px}.row-fluid{width:100%;*zoom:1}.row-fluid:before,.row-fluid:after{display:table;line-height:0;content:""}.row-fluid:after{clear:both}.row-fluid [class*="span"]{display:block;float:left;width:100%;min-height:30px;margin-left:2.7624309392265194%;*margin-left:2.709239449864817%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.row-fluid [class*="span"]:first-child{margin-left:0}.row-fluid .controls-row [class*="span"]+[class*="span"]{margin-left:2.7624309392265194%}.row-fluid .span12{width:100%;*width:99.94680851063829%}.row-fluid .span11{width:91.43646408839778%;*width:91.38327259903608%}.row-fluid .span10{width:82.87292817679558%;*width:82.81973668743387%}.row-fluid .span9{width:74.30939226519337%;*width:74.25620077583166%}.row-fluid .span8{width:65.74585635359117%;*width:65.69266486422946%}.row-fluid .span7{width:57.18232044198895%;*width:57.12912895262725%}.row-fluid .span6{width:48.61878453038674%;*width:48.56559304102504%}.row-fluid .span5{width:40.05524861878453%;*width:40.00205712942283%}.row-fluid .span4{width:31.491712707182323%;*width:31.43852121782062%}.row-fluid .span3{width:22.92817679558011%;*width:22.87498530621841%}.row-fluid .span2{width:14.3646408839779%;*width:14.311449394616199%}.row-fluid .span1{width:5.801104972375691%;*width:5.747913483013988%}.row-fluid .offset12{margin-left:105.52486187845304%;*margin-left:105.41847889972962%}.row-fluid .offset12:first-child{margin-left:102.76243093922652%;*margin-left:102.6560479605031%}.row-fluid .offset11{margin-left:96.96132596685082%;*margin-left:96.8549429881274%}.row-fluid .offset11:first-child{margin-left:94.1988950276243%;*margin-left:94.09251204890089%}.row-fluid .offset10{margin-left:88.39779005524862%;*margin-left:88.2914070765252%}.row-fluid .offset10:first-child{margin-left:85.6353591160221%;*margin-left:85.52897613729868%}.row-fluid .offset9{margin-left:79.8342541436464%;*margin-left:79.72787116492299%}.row-fluid .offset9:first-child{margin-left:77.07182320441989%;*margin-left:76.96544022569647%}.row-fluid .offset8{margin-left:71.2707182320442%;*margin-left:71.16433525332079%}.row-fluid .offset8:first-child{margin-left:68.50828729281768%;*margin-left:68.40190431409427%}.row-fluid .offset7{margin-left:62.70718232044199%;*margin-left:62.600799341718584%}.row-fluid .offset7:first-child{margin-left:59.94475138121547%;*margin-left:59.838368402492065%}.row-fluid .offset6{margin-left:54.14364640883978%;*margin-left:54.037263430116376%}.row-fluid .offset6:first-child{margin-left:51.38121546961326%;*margin-left:51.27483249088986%}.row-fluid .offset5{margin-left:45.58011049723757%;*margin-left:45.47372751851417%}.row-fluid .offset5:first-child{margin-left:42.81767955801105%;*margin-left:42.71129657928765%}.row-fluid .offset4{margin-left:37.01657458563536%;*margin-left:36.91019160691196%}.row-fluid .offset4:first-child{margin-left:34.25414364640884%;*margin-left:34.14776066768544%}.row-fluid .offset3{margin-left:28.45303867403315%;*margin-left:28.346655695309746%}.row-fluid .offset3:first-child{margin-left:25.69060773480663%;*margin-left:25.584224756083227%}.row-fluid .offset2{margin-left:19.88950276243094%;*margin-left:19.783119783707537%}.row-fluid .offset2:first-child{margin-left:17.12707182320442%;*margin-left:17.02068884448102%}.row-fluid .offset1{margin-left:11.32596685082873%;*margin-left:11.219583872105325%}.row-fluid .offset1:first-child{margin-left:8.56353591160221%;*margin-left:8.457152932878806%}input,textarea,.uneditable-input{margin-left:0}.controls-row [class*="span"]+[class*="span"]{margin-left:20px}input.span12,textarea.span12,.uneditable-input.span12{width:710px}input.span11,textarea.span11,.uneditable-input.span11{width:648px}input.span10,textarea.span10,.uneditable-input.span10{width:586px}input.span9,textarea.span9,.uneditable-input.span9{width:524px}input.span8,textarea.span8,.uneditable-input.span8{width:462px}input.span7,textarea.span7,.uneditable-input.span7{width:400px}input.span6,textarea.span6,.uneditable-input.span6{width:338px}input.span5,textarea.span5,.uneditable-input.span5{width:276px}input.span4,textarea.span4,.uneditable-input.span4{width:214px}input.span3,textarea.span3,.uneditable-input.span3{width:152px}input.span2,textarea.span2,.uneditable-input.span2{width:90px}input.span1,textarea.span1,.uneditable-input.span1{width:28px}}@media(max-width:767px){body{padding-right:20px;padding-left:20px}.navbar-fixed-top,.navbar-fixed-bottom,.navbar-static-top{margin-right:-20px;margin-left:-20px}.container-fluid{padding:0}.dl-horizontal dt{float:none;width:auto;clear:none;text-align:left}.dl-horizontal dd{margin-left:0}.container{width:auto}.row-fluid{width:100%}.row,.thumbnails{margin-left:0}.thumbnails>li{float:none;margin-left:0}[class*="span"],.uneditable-input[class*="span"],.row-fluid [class*="span"]{display:block;float:none;width:100%;margin-left:0;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.span12,.row-fluid .span12{width:100%;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.row-fluid [class*="offset"]:first-child{margin-left:0}.input-large,.input-xlarge,.input-xxlarge,input[class*="span"],select[class*="span"],textarea[class*="span"],.uneditable-input{display:block;width:100%;min-height:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.input-prepend input,.input-append input,.input-prepend input[class*="span"],.input-append input[class*="span"]{display:inline-block;width:auto}.controls-row [class*="span"]+[class*="span"]{margin-left:0}.modal{position:fixed;top:20px;right:20px;left:20px;width:auto;margin:0}.modal.fade{top:-100px}.modal.fade.in{top:20px}}@media(max-width:480px){.nav-collapse{-webkit-transform:translate3d(0,0,0)}.page-header h1 small{display:block;line-height:20px}input[type="checkbox"],input[type="radio"]{border:1px solid #ccc}.form-horizontal .control-label{float:none;width:auto;padding-top:0;text-align:left}.form-horizontal .controls{margin-left:0}.form-horizontal .control-list{padding-top:0}.form-horizontal .form-actions{padding-right:10px;padding-left:10px}.media .pull-left,.media .pull-right{display:block;float:none;margin-bottom:10px}.media-object{margin-right:0;margin-left:0}.modal{top:10px;right:10px;left:10px}.modal-header .close{padding:10px;margin:-10px}.carousel-caption{position:static}}@media(max-width:979px){body{padding-top:0}.navbar-fixed-top,.navbar-fixed-bottom{position:static}.navbar-fixed-top{margin-bottom:20px}.navbar-fixed-bottom{margin-top:20px}.navbar-fixed-top .navbar-inner,.navbar-fixed-bottom .navbar-inner{padding:5px}.navbar .container{width:auto;padding:0}.navbar .brand{padding-right:10px;padding-left:10px;margin:0 0 0 -5px}.nav-collapse{clear:both}.nav-collapse .nav{float:none;margin:0 0 10px}.nav-collapse .nav>li{float:none}.nav-collapse .nav>li>a{margin-bottom:2px}.nav-collapse .nav>.divider-vertical{display:none}.nav-collapse .nav .nav-header{color:#777;text-shadow:none}.nav-collapse .nav>li>a,.nav-collapse .dropdown-menu a{padding:9px 15px;font-weight:bold;color:#777;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px}.nav-collapse .btn{padding:4px 10px 4px;font-weight:normal;-webkit-border-radius:4px;-moz-border-radius:4px;border-radius:4px}.nav-collapse .dropdown-menu li+li a{margin-bottom:2px}.nav-collapse .nav>li>a:hover,.nav-collapse .nav>li>a:focus,.nav-collapse .dropdown-menu a:hover,.nav-collapse .dropdown-menu a:focus{background-color:#f2f2f2}.navbar-inverse .nav-collapse .nav>li>a,.navbar-inverse .nav-collapse .dropdown-menu a{color:#999}.navbar-inverse .nav-collapse .nav>li>a:hover,.navbar-inverse .nav-collapse .nav>li>a:focus,.navbar-inverse .nav-collapse .dropdown-menu a:hover,.navbar-inverse .nav-collapse .dropdown-menu a:focus{background-color:#111}.nav-collapse.in .btn-group{padding:0;margin-top:5px}.nav-collapse .dropdown-menu{position:static;top:auto;left:auto;display:none;float:none;max-width:none;padding:0;margin:0 15px;background-color:transparent;border:0;-webkit-border-radius:0;-moz-border-radius:0;border-radius:0;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none}.nav-collapse .open>.dropdown-menu{display:block}.nav-collapse .dropdown-menu:before,.nav-collapse .dropdown-menu:after{display:none}.nav-collapse .dropdown-menu .divider{display:none}.nav-collapse .nav>li>.dropdown-menu:before,.nav-collapse .nav>li>.dropdown-menu:after{display:none}.nav-collapse .navbar-form,.nav-collapse .navbar-search{float:none;padding:10px 15px;margin:10px 0;border-top:1px solid #f2f2f2;border-bottom:1px solid #f2f2f2;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);-moz-box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1);box-shadow:inset 0 1px 0 rgba(255,255,255,0.1),0 1px 0 rgba(255,255,255,0.1)}.navbar-inverse .nav-collapse .navbar-form,.navbar-inverse .nav-collapse .navbar-search{border-top-color:#111;border-bottom-color:#111}.navbar .nav-collapse .nav.pull-right{float:none;margin-left:0}.nav-collapse,.nav-collapse.collapse{height:0;overflow:hidden}.navbar .btn-navbar{display:block}.navbar-static .navbar-inner{padding-right:10px;padding-left:10px}}@media(min-width:980px){.nav-collapse.collapse{height:auto!important;overflow:visible!important}} 10 | -------------------------------------------------------------------------------- /webapp/css/codemirror.css: -------------------------------------------------------------------------------- 1 | /* BASICS */ 2 | 3 | .CodeMirror { 4 | /* Set height, width, borders, and global font properties here */ 5 | font-family: monospace; 6 | height: 300px; 7 | } 8 | .CodeMirror-scroll { 9 | /* Set scrolling behaviour here */ 10 | overflow: auto; 11 | } 12 | 13 | /* PADDING */ 14 | 15 | .CodeMirror-lines { 16 | padding: 4px 0; /* Vertical padding around content */ 17 | } 18 | .CodeMirror pre { 19 | padding: 0 4px; /* Horizontal padding of content */ 20 | } 21 | 22 | .CodeMirror-scrollbar-filler { 23 | background-color: white; /* The little square between H and V scrollbars */ 24 | } 25 | 26 | /* GUTTER */ 27 | 28 | .CodeMirror-gutters { 29 | border-right: 1px solid #ddd; 30 | background-color: #f7f7f7; 31 | } 32 | .CodeMirror-linenumbers {} 33 | .CodeMirror-linenumber { 34 | padding: 0 3px 0 0px; 35 | min-width: 20px; 36 | text-align: right; 37 | color: #999; 38 | } 39 | 40 | /* CURSOR */ 41 | 42 | .CodeMirror div.CodeMirror-cursor { 43 | border-left: 1px solid black; 44 | z-index: 3; 45 | } 46 | /* Shown when moving in bi-directional text */ 47 | .CodeMirror div.CodeMirror-secondarycursor { 48 | border-left: 1px solid silver; 49 | } 50 | .CodeMirror.cm-keymap-fat-cursor div.CodeMirror-cursor { 51 | width: auto; 52 | border: 0; 53 | background: #7e7; 54 | z-index: 1; 55 | } 56 | /* Can style cursor different in overwrite (non-insert) mode */ 57 | .CodeMirror div.CodeMirror-cursor.CodeMirror-overwrite {} 58 | 59 | .cm-tab { display: inline-block; } 60 | 61 | /* DEFAULT THEME */ 62 | 63 | .cm-s-default .cm-keyword {color: #708;} 64 | .cm-s-default .cm-atom {color: #219;} 65 | .cm-s-default .cm-number {color: #164;} 66 | .cm-s-default .cm-def {color: #00f;} 67 | .cm-s-default .cm-variable {color: black;} 68 | .cm-s-default .cm-variable-2 {color: #05a;} 69 | .cm-s-default .cm-variable-3 {color: #085;} 70 | .cm-s-default .cm-property {color: black;} 71 | .cm-s-default .cm-operator {color: black;} 72 | .cm-s-default .cm-comment {color: #a50;} 73 | .cm-s-default .cm-string {color: #a11;} 74 | .cm-s-default .cm-string-2 {color: #f50;} 75 | .cm-s-default .cm-meta {color: #555;} 76 | .cm-s-default .cm-error {color: #f00;} 77 | .cm-s-default .cm-qualifier {color: #555;} 78 | .cm-s-default .cm-builtin {color: #30a;} 79 | .cm-s-default .cm-bracket {color: #997;} 80 | .cm-s-default .cm-tag {color: #170;} 81 | .cm-s-default .cm-attribute {color: #00c;} 82 | .cm-s-default .cm-header {color: blue;} 83 | .cm-s-default .cm-quote {color: #090;} 84 | .cm-s-default .cm-hr {color: #999;} 85 | .cm-s-default .cm-link {color: #00c;} 86 | 87 | .cm-negative {color: #d44;} 88 | .cm-positive {color: #292;} 89 | .cm-header, .cm-strong {font-weight: bold;} 90 | .cm-em {font-style: italic;} 91 | .cm-link {text-decoration: underline;} 92 | 93 | .cm-invalidchar {color: #f00;} 94 | 95 | div.CodeMirror span.CodeMirror-matchingbracket {color: #0f0;} 96 | div.CodeMirror span.CodeMirror-nonmatchingbracket {color: #f22;} 97 | 98 | /* STOP */ 99 | 100 | /* The rest of this file contains styles related to the mechanics of 101 | the editor. You probably shouldn't touch them. */ 102 | 103 | .CodeMirror { 104 | line-height: 1; 105 | position: relative; 106 | overflow: hidden; 107 | background: white; 108 | color: black; 109 | } 110 | 111 | .CodeMirror-scroll { 112 | /* 30px is the magic margin used to hide the element's real scrollbars */ 113 | /* See overflow: hidden in .CodeMirror, and the paddings in .CodeMirror-sizer */ 114 | margin-bottom: -30px; margin-right: -30px; 115 | padding-bottom: 30px; padding-right: 30px; 116 | height: 100%; 117 | outline: none; /* Prevent dragging from highlighting the element */ 118 | position: relative; 119 | } 120 | .CodeMirror-sizer { 121 | position: relative; 122 | } 123 | 124 | /* The fake, visible scrollbars. Used to force redraw during scrolling 125 | before actuall scrolling happens, thus preventing shaking and 126 | flickering artifacts. */ 127 | .CodeMirror-vscrollbar, .CodeMirror-hscrollbar, .CodeMirror-scrollbar-filler { 128 | position: absolute; 129 | z-index: 6; 130 | display: none; 131 | } 132 | .CodeMirror-vscrollbar { 133 | right: 0; top: 0; 134 | overflow-x: hidden; 135 | overflow-y: scroll; 136 | } 137 | .CodeMirror-hscrollbar { 138 | bottom: 0; left: 0; 139 | overflow-y: hidden; 140 | overflow-x: scroll; 141 | } 142 | .CodeMirror-scrollbar-filler { 143 | right: 0; bottom: 0; 144 | z-index: 6; 145 | } 146 | 147 | .CodeMirror-gutters { 148 | position: absolute; left: 0; top: 0; 149 | height: 100%; 150 | padding-bottom: 30px; 151 | z-index: 3; 152 | } 153 | .CodeMirror-gutter { 154 | height: 100%; 155 | padding-bottom: 30px; 156 | margin-bottom: -32px; 157 | display: inline-block; 158 | /* Hack to make IE7 behave */ 159 | *zoom:1; 160 | *display:inline; 161 | } 162 | .CodeMirror-gutter-elt { 163 | position: absolute; 164 | cursor: default; 165 | z-index: 4; 166 | } 167 | 168 | .CodeMirror-lines { 169 | cursor: text; 170 | } 171 | .CodeMirror pre { 172 | /* Reset some styles that the rest of the page might have set */ 173 | -moz-border-radius: 0; -webkit-border-radius: 0; border-radius: 0; 174 | border-width: 0; 175 | background: transparent; 176 | font-family: inherit; 177 | font-size: inherit; 178 | margin: 0; 179 | white-space: pre; 180 | word-wrap: normal; 181 | line-height: inherit; 182 | color: inherit; 183 | z-index: 2; 184 | position: relative; 185 | overflow: visible; 186 | } 187 | .CodeMirror-wrap pre { 188 | word-wrap: break-word; 189 | white-space: pre-wrap; 190 | word-break: normal; 191 | } 192 | .CodeMirror-linebackground { 193 | position: absolute; 194 | left: 0; right: 0; top: 0; bottom: 0; 195 | z-index: 0; 196 | } 197 | 198 | .CodeMirror-linewidget { 199 | position: relative; 200 | z-index: 2; 201 | overflow: auto; 202 | } 203 | 204 | .CodeMirror-widget { 205 | display: inline-block; 206 | } 207 | 208 | .CodeMirror-wrap .CodeMirror-scroll { 209 | overflow-x: hidden; 210 | } 211 | 212 | .CodeMirror-measure { 213 | position: absolute; 214 | width: 100%; height: 0px; 215 | overflow: hidden; 216 | visibility: hidden; 217 | } 218 | .CodeMirror-measure pre { position: static; } 219 | 220 | .CodeMirror div.CodeMirror-cursor { 221 | position: absolute; 222 | visibility: hidden; 223 | border-right: none; 224 | width: 0; 225 | } 226 | .CodeMirror-focused div.CodeMirror-cursor { 227 | visibility: visible; 228 | } 229 | 230 | .CodeMirror-selected { background: #d9d9d9; } 231 | .CodeMirror-focused .CodeMirror-selected { background: #d7d4f0; } 232 | 233 | .cm-searching { 234 | background: #ffa; 235 | background: rgba(255, 255, 0, .4); 236 | } 237 | 238 | /* IE7 hack to prevent it from returning funny offsetTops on the spans */ 239 | .CodeMirror span { *vertical-align: text-bottom; } 240 | 241 | @media print { 242 | /* Hide the cursor when printing */ 243 | .CodeMirror div.CodeMirror-cursor { 244 | visibility: hidden; 245 | } 246 | } 247 | -------------------------------------------------------------------------------- /webapp/css/elegant.css: -------------------------------------------------------------------------------- 1 | .cm-s-elegant span.cm-number, .cm-s-elegant span.cm-string, .cm-s-elegant span.cm-atom {color: #762;} 2 | .cm-s-elegant span.cm-comment {color: #262; font-style: italic; line-height: 1em;} 3 | .cm-s-elegant span.cm-meta {color: #555; font-style: italic; line-height: 1em;} 4 | .cm-s-elegant span.cm-variable {color: black;} 5 | .cm-s-elegant span.cm-variable-2 {color: #b11;} 6 | .cm-s-elegant span.cm-qualifier {color: #555;} 7 | .cm-s-elegant span.cm-keyword {color: #730;} 8 | .cm-s-elegant span.cm-builtin {color: #30a;} 9 | .cm-s-elegant span.cm-error {background-color: #fdd;} 10 | .cm-s-elegant span.cm-link {color: #762;} 11 | -------------------------------------------------------------------------------- /webapp/css/impress-demo.css: -------------------------------------------------------------------------------- 1 | /* 2 | So you like the style of impress.js demo? 3 | Or maybe you are just curious how it was done? 4 | 5 | You couldn't find a better place to find out! 6 | 7 | Welcome to the stylesheet impress.js demo presentation. 8 | 9 | Please remember that it is not meant to be a part of impress.js and is 10 | not required by impress.js. 11 | I expect that anyone creating a presentation for impress.js would create 12 | their own set of styles. 13 | 14 | But feel free to read through it and learn how to get the most of what 15 | impress.js provides. 16 | 17 | And let me be your guide. 18 | 19 | Shall we begin? 20 | */ 21 | 22 | 23 | /* 24 | We start with a good ol' reset. 25 | That's the one by Eric Meyer http://meyerweb.com/eric/tools/css/reset/ 26 | 27 | You can probably argue if it is needed here, or not, but for sure it 28 | doesn't do any harm and gives us a fresh start. 29 | */ 30 | 31 | html, body, div, span, applet, object, iframe, 32 | h1, h2, h3, h4, h5, h6, p, blockquote, pre, 33 | a, abbr, acronym, address, big, cite, code, 34 | del, dfn, em, img, ins, kbd, q, s, samp, 35 | small, strike, strong, sub, sup, tt, var, 36 | b, u, i, center, 37 | dl, dt, dd, ol, ul, li, 38 | fieldset, form, label, legend, 39 | table, caption, tbody, tfoot, thead, tr, th, td, 40 | article, aside, canvas, details, embed, 41 | figure, figcaption, footer, header, hgroup, 42 | menu, nav, output, ruby, section, summary, 43 | time, mark, audio, video { 44 | margin: 0; 45 | padding: 0; 46 | border: 0; 47 | font-size: 100%; 48 | font: inherit; 49 | vertical-align: baseline; 50 | } 51 | 52 | /* HTML5 display-role reset for older browsers */ 53 | article, aside, details, figcaption, figure, 54 | footer, header, hgroup, menu, nav, section { 55 | display: block; 56 | } 57 | body { 58 | line-height: 1; 59 | } 60 | ol, ul { 61 | list-style: none; 62 | } 63 | blockquote, q { 64 | quotes: none; 65 | } 66 | blockquote:before, blockquote:after, 67 | q:before, q:after { 68 | content: ''; 69 | content: none; 70 | } 71 | 72 | table { 73 | border-collapse: collapse; 74 | border-spacing: 0; 75 | } 76 | 77 | /* 78 | Now here is when interesting things start to appear. 79 | 80 | We set up styles with default font and nice gradient in the background. 81 | And yes, there is a lot of repetition there because of -prefixes but we don't 82 | want to leave anybody behind. 83 | */ 84 | body { 85 | font-family: 'PT Sans', sans-serif; 86 | min-height: 640px; 87 | 88 | background: rgb(215, 215, 215); 89 | background: -webkit-gradient(radial, 50% 50%, 0, 50% 50%, 500, from(rgb(240, 240, 240)), to(rgb(190, 190, 190))); 90 | background: -webkit-radial-gradient(rgb(240, 240, 240), rgb(190, 190, 190)); 91 | background: -moz-radial-gradient(rgb(240, 240, 240), rgb(190, 190, 190)); 92 | background: -ms-radial-gradient(rgb(240, 240, 240), rgb(190, 190, 190)); 93 | background: -o-radial-gradient(rgb(240, 240, 240), rgb(190, 190, 190)); 94 | background: radial-gradient(rgb(240, 240, 240), rgb(190, 190, 190)); 95 | } 96 | 97 | /* 98 | Now let's bring some text styles back ... 99 | */ 100 | b, strong { font-weight: bold } 101 | i, em { font-style: italic } 102 | 103 | /* 104 | ... and give links a nice look. 105 | */ 106 | a { 107 | color: inherit; 108 | text-decoration: none; 109 | padding: 0 0.1em; 110 | background: rgba(255,255,255,0.5); 111 | text-shadow: -1px -1px 2px rgba(100,100,100,0.9); 112 | border-radius: 0.2em; 113 | 114 | -webkit-transition: 0.5s; 115 | -moz-transition: 0.5s; 116 | -ms-transition: 0.5s; 117 | -o-transition: 0.5s; 118 | transition: 0.5s; 119 | } 120 | 121 | a:hover, 122 | a:focus { 123 | background: rgba(255,255,255,1); 124 | text-shadow: -1px -1px 2px rgba(100,100,100,0.5); 125 | } 126 | 127 | /* 128 | Because the main point behind the impress.js demo is to demo impress.js 129 | we display a fallback message for users with browsers that don't support 130 | all the features required by it. 131 | 132 | All of the content will be still fully accessible for them, but I want 133 | them to know that they are missing something - that's what the demo is 134 | about, isn't it? 135 | 136 | And then we hide the message, when support is detected in the browser. 137 | */ 138 | 139 | .fallback-message { 140 | font-family: sans-serif; 141 | line-height: 1.3; 142 | 143 | width: 780px; 144 | padding: 10px 10px 0; 145 | margin: 20px auto; 146 | 147 | border: 1px solid #E4C652; 148 | border-radius: 10px; 149 | background: #EEDC94; 150 | } 151 | 152 | .fallback-message p { 153 | margin-bottom: 10px; 154 | } 155 | 156 | .impress-supported .fallback-message { 157 | display: none; 158 | } 159 | 160 | .impress-not-supported #impress { 161 | display: none; 162 | } 163 | .impress-not-supported .hint { 164 | display: none; 165 | } 166 | 167 | 168 | /* 169 | Now let's style the presentation steps. 170 | 171 | We start with basics to make sure it displays correctly in everywhere ... 172 | */ 173 | 174 | .step { 175 | position: relative; 176 | width: 1024px; 177 | margin: 20px auto; 178 | 179 | -webkit-box-sizing: border-box; 180 | -moz-box-sizing: border-box; 181 | -ms-box-sizing: border-box; 182 | -o-box-sizing: border-box; 183 | box-sizing: border-box; 184 | 185 | font-family: 'PT Serif', georgia, serif; 186 | /* 187 | padding: 40px; 188 | font-size: 48px; 189 | line-height: 1.5; 190 | */ 191 | } 192 | 193 | /* 194 | ... and we enhance the styles for impress.js. 195 | 196 | Basically we remove the margin and make inactive steps a little bit transparent. 197 | */ 198 | .impress-enabled .step { 199 | margin: 20px 0 0 0; 200 | opacity: 0; 201 | 202 | -webkit-transition: opacity 1s; 203 | -moz-transition: opacity 1s; 204 | -ms-transition: opacity 1s; 205 | -o-transition: opacity 1s; 206 | transition: opacity 1s; 207 | } 208 | 209 | .impress-enabled .step.active { opacity: 1 } 210 | 211 | /* 212 | These 'slide' step styles were heavily inspired by HTML5 Slides: 213 | http://html5slides.googlecode.com/svn/trunk/styles.css 214 | 215 | ;) 216 | 217 | They cover everything what you see on first three steps of the demo. 218 | */ 219 | .slide { 220 | display: block; 221 | 222 | width: 1024px; 223 | height: 600px; 224 | color: rgb(102, 102, 102); 225 | text-shadow: 0 2px 2px rgba(0, 0, 0, .1); 226 | 227 | font-family: 'Open Sans', Arial, sans-serif; 228 | 229 | /* 230 | padding: 40px 60px; 231 | 232 | font-size: 30px; 233 | line-height: 36px; 234 | letter-spacing: -1px;*/ 235 | } 236 | 237 | .slide_background{ 238 | background-color: white; 239 | border: 1px solid rgba(0, 0, 0, .3); 240 | border-radius: 10px; 241 | box-shadow: 0 2px 6px rgba(0, 0, 0, .1); 242 | } 243 | 244 | .slide q { 245 | display: block; 246 | line-height: 72px; 247 | 248 | margin-top: 100px; 249 | /* 250 | font-size: 50px; 251 | */ 252 | } 253 | 254 | .slide q strong { 255 | white-space: nowrap; 256 | } 257 | 258 | /* 259 | And now we start to style each step separately. 260 | 261 | I agree that this may be not the most efficient, object-oriented and 262 | scalable way of styling, but most of steps have quite a custom look 263 | and typography tricks here and there, so they had to be styled separately. 264 | 265 | First is the title step with a big

(no room for padding) and some 266 | 3D positioning along Z axis. 267 | */ 268 | 269 | #title { 270 | padding: 0; 271 | } 272 | 273 | #title .try { 274 | font-size: 64px; 275 | position: absolute; 276 | top: -0.5em; 277 | left: 1.5em; 278 | 279 | -webkit-transform: translateZ(20px); 280 | -moz-transform: translateZ(20px); 281 | -ms-transform: translateZ(20px); 282 | -o-transform: translateZ(20px); 283 | transform: translateZ(20px); 284 | } 285 | 286 | #title h1 { 287 | font-size: 190px; 288 | 289 | -webkit-transform: translateZ(50px); 290 | -moz-transform: translateZ(50px); 291 | -ms-transform: translateZ(50px); 292 | -o-transform: translateZ(50px); 293 | transform: translateZ(50px); 294 | } 295 | 296 | #title .footnote { 297 | font-size: 32px; 298 | } 299 | 300 | /* 301 | Second step is nothing special, just a text with a link, so it doesn't need 302 | any special styling. 303 | 304 | Let's move to 'big thoughts' with centered text and custom font sizes. 305 | */ 306 | #big { 307 | width: 600px; 308 | text-align: center; 309 | font-size: 60px; 310 | line-height: 1; 311 | } 312 | 313 | #big b { 314 | display: block; 315 | font-size: 250px; 316 | line-height: 250px; 317 | } 318 | 319 | #big .thoughts { 320 | font-size: 90px; 321 | line-height: 150px; 322 | } 323 | 324 | /* 325 | 'Tiny ideas' just need some tiny styling. 326 | */ 327 | #tiny { 328 | width: 500px; 329 | text-align: center; 330 | } 331 | 332 | /* 333 | This step has some animated text ... 334 | */ 335 | #ing { width: 500px } 336 | 337 | /* 338 | ... so we define display to `inline-block` to enable transforms and 339 | transition duration to 0.5s ... 340 | */ 341 | #ing b { 342 | display: inline-block; 343 | -webkit-transition: 0.5s; 344 | -moz-transition: 0.5s; 345 | -ms-transition: 0.5s; 346 | -o-transition: 0.5s; 347 | transition: 0.5s; 348 | } 349 | 350 | /* 351 | ... and we want 'positioning` word to move up a bit when the step gets 352 | `present` class ... 353 | */ 354 | #ing.present .positioning { 355 | -webkit-transform: translateY(-10px); 356 | -moz-transform: translateY(-10px); 357 | -ms-transform: translateY(-10px); 358 | -o-transform: translateY(-10px); 359 | transform: translateY(-10px); 360 | } 361 | 362 | /* 363 | ... 'rotating' to rotate a quarter of a second later ... 364 | */ 365 | #ing.present .rotating { 366 | -webkit-transform: rotate(-10deg); 367 | -moz-transform: rotate(-10deg); 368 | -ms-transform: rotate(-10deg); 369 | -o-transform: rotate(-10deg); 370 | transform: rotate(-10deg); 371 | 372 | -webkit-transition-delay: 0.25s; 373 | -moz-transition-delay: 0.25s; 374 | -ms-transition-delay: 0.25s; 375 | -o-transition-delay: 0.25s; 376 | transition-delay: 0.25s; 377 | } 378 | 379 | /* 380 | ... and 'scaling' to scale down after another quarter of a second. 381 | */ 382 | #ing.present .scaling { 383 | -webkit-transform: scale(0.7); 384 | -moz-transform: scale(0.7); 385 | -ms-transform: scale(0.7); 386 | -o-transform: scale(0.7); 387 | transform: scale(0.7); 388 | 389 | -webkit-transition-delay: 0.5s; 390 | -moz-transition-delay: 0.5s; 391 | -ms-transition-delay: 0.5s; 392 | -o-transition-delay: 0.5s; 393 | transition-delay: 0.5s; 394 | } 395 | 396 | /* 397 | The 'imagination' step is again some boring font-sizing. 398 | */ 399 | 400 | #imagination { 401 | width: 600px; 402 | } 403 | 404 | #imagination .imagination { 405 | font-size: 78px; 406 | } 407 | 408 | /* 409 | There is nothing really special about 'use the source, Luke' step, too, 410 | except maybe of the Yoda background. 411 | 412 | As you can see below I've 'hard-coded' it in data URL. 413 | That's not the best way to serve images, but because that's just this one 414 | I decided it will be OK to have it this way. 415 | 416 | Just make sure you don't blindly copy this approach. 417 | */ 418 | #source { 419 | width: 700px; 420 | padding-bottom: 300px; 421 | 422 | /* Yoda Icon :: Pixel Art from Star Wars http://www.pixeljoint.com/pixelart/1423.htm */ 423 | background-image: url(); 424 | background-position: bottom right; 425 | background-repeat: no-repeat; 426 | } 427 | 428 | #source q { 429 | font-size: 60px; 430 | } 431 | 432 | /* 433 | And the "it's in 3D" step again brings some 3D typography - just for fun. 434 | 435 | Because we want to position elements in 3D we set transform-style to 436 | `preserve-3d` on the paragraph. 437 | It is not needed by webkit browsers, but it is in Firefox. It's hard to say 438 | which behaviour is correct as 3D transforms spec is not very clear about it. 439 | */ 440 | #its-in-3d p { 441 | -webkit-transform-style: preserve-3d; 442 | -moz-transform-style: preserve-3d; /* Y U need this Firefox?! */ 443 | -ms-transform-style: preserve-3d; 444 | -o-transform-style: preserve-3d; 445 | transform-style: preserve-3d; 446 | } 447 | 448 | /* 449 | Below we position each word separately along Z axis and we want it to transition 450 | to default position in 0.5s when the step gets `present` class. 451 | 452 | Quite a simple idea, but lot's of styles and prefixes. 453 | */ 454 | #its-in-3d span, 455 | #its-in-3d b { 456 | display: inline-block; 457 | -webkit-transform: translateZ(40px); 458 | -moz-transform: translateZ(40px); 459 | -ms-transform: translateZ(40px); 460 | -o-transform: translateZ(40px); 461 | transform: translateZ(40px); 462 | 463 | -webkit-transition: 0.5s; 464 | -moz-transition: 0.5s; 465 | -ms-transition: 0.5s; 466 | -o-transition: 0.5s; 467 | transition: 0.5s; 468 | } 469 | 470 | #its-in-3d .have { 471 | -webkit-transform: translateZ(-40px); 472 | -moz-transform: translateZ(-40px); 473 | -ms-transform: translateZ(-40px); 474 | -o-transform: translateZ(-40px); 475 | transform: translateZ(-40px); 476 | } 477 | 478 | #its-in-3d .you { 479 | -webkit-transform: translateZ(20px); 480 | -moz-transform: translateZ(20px); 481 | -ms-transform: translateZ(20px); 482 | -o-transform: translateZ(20px); 483 | transform: translateZ(20px); 484 | } 485 | 486 | #its-in-3d .noticed { 487 | -webkit-transform: translateZ(-40px); 488 | -moz-transform: translateZ(-40px); 489 | -ms-transform: translateZ(-40px); 490 | -o-transform: translateZ(-40px); 491 | transform: translateZ(-40px); 492 | } 493 | 494 | #its-in-3d .its { 495 | -webkit-transform: translateZ(60px); 496 | -moz-transform: translateZ(60px); 497 | -ms-transform: translateZ(60px); 498 | -o-transform: translateZ(60px); 499 | transform: translateZ(60px); 500 | } 501 | 502 | #its-in-3d .in { 503 | -webkit-transform: translateZ(-10px); 504 | -moz-transform: translateZ(-10px); 505 | -ms-transform: translateZ(-10px); 506 | -o-transform: translateZ(-10px); 507 | transform: translateZ(-10px); 508 | } 509 | 510 | #its-in-3d .footnote { 511 | font-size: 32px; 512 | 513 | -webkit-transform: translateZ(-10px); 514 | -moz-transform: translateZ(-10px); 515 | -ms-transform: translateZ(-10px); 516 | -o-transform: translateZ(-10px); 517 | transform: translateZ(-10px); 518 | } 519 | 520 | #its-in-3d.present span, 521 | #its-in-3d.present b { 522 | -webkit-transform: translateZ(0px); 523 | -moz-transform: translateZ(0px); 524 | -ms-transform: translateZ(0px); 525 | -o-transform: translateZ(0px); 526 | transform: translateZ(0px); 527 | } 528 | 529 | /* 530 | The last step is an overview. 531 | There is no content in it, so we make sure it's not visible because we want 532 | to be able to click on other steps. 533 | 534 | */ 535 | #overview { display: none } 536 | 537 | /* 538 | We also make other steps visible and give them a pointer cursor using the 539 | `impress-on-` class. 540 | */ 541 | .impress-on-overview .step { 542 | opacity: 1; 543 | cursor: pointer; 544 | } 545 | 546 | 547 | /* 548 | Now, when we have all the steps styled let's give users a hint how to navigate 549 | around the presentation. 550 | 551 | The best way to do this would be to use JavaScript, show a delayed hint for a 552 | first time users, then hide it and store a status in cookie or localStorage... 553 | 554 | But I wanted to have some CSS fun and avoid additional scripting... 555 | 556 | Let me explain it first, so maybe the transition magic will be more readable 557 | when you read the code. 558 | 559 | First of all I wanted the hint to appear only when user is idle for a while. 560 | You can't detect the 'idle' state in CSS, but I delayed a appearing of the 561 | hint by 5s using transition-delay. 562 | 563 | You also can't detect in CSS if the user is a first-time visitor, so I had to 564 | make an assumption that I'll only show the hint on the first step. And when 565 | the step is changed hide the hint, because I can assume that user already 566 | knows how to navigate. 567 | 568 | To summarize it - hint is shown when the user is on the first step for longer 569 | than 5 seconds. 570 | 571 | The other problem I had was caused by the fact that I wanted the hint to fade 572 | in and out. It can be easily achieved by transitioning the opacity property. 573 | But that also meant that the hint was always on the screen, even if totally 574 | transparent. It covered part of the screen and you couldn't correctly clicked 575 | through it. 576 | Unfortunately you cannot transition between display `block` and `none` in pure 577 | CSS, so I needed a way to not only fade out the hint but also move it out of 578 | the screen. 579 | 580 | I solved this problem by positioning the hint below the bottom of the screen 581 | with CSS transform and moving it up to show it. But I also didn't want this move 582 | to be visible. I wanted the hint only to fade in and out visually, so I delayed 583 | the fade in transition, so it starts when the hint is already in its correct 584 | position on the screen. 585 | 586 | I know, it sounds complicated ... maybe it would be easier with the code? 587 | */ 588 | 589 | .hint { 590 | /* 591 | We hide the hint until presentation is started and from browsers not supporting 592 | impress.js, as they will have a linear scrollable view ... 593 | */ 594 | display: none; 595 | 596 | /* 597 | ... and give it some fixed position and nice styles. 598 | */ 599 | position: fixed; 600 | left: 0; 601 | right: 0; 602 | bottom: 200px; 603 | 604 | background: rgba(0,0,0,0.5); 605 | color: #EEE; 606 | text-align: center; 607 | 608 | font-size: 50px; 609 | padding: 20px; 610 | 611 | z-index: 100; 612 | 613 | /* 614 | By default we don't want the hint to be visible, so we make it transparent ... 615 | */ 616 | opacity: 0; 617 | 618 | /* 619 | ... and position it below the bottom of the screen (relative to it's fixed position) 620 | */ 621 | -webkit-transform: translateY(400px); 622 | -moz-transform: translateY(400px); 623 | -ms-transform: translateY(400px); 624 | -o-transform: translateY(400px); 625 | transform: translateY(400px); 626 | 627 | /* 628 | Now let's imagine that the hint is visible and we want to fade it out and move out 629 | of the screen. 630 | 631 | So we define the transition on the opacity property with 1s duration and another 632 | transition on transform property delayed by 1s so it will happen after the fade out 633 | on opacity finished. 634 | 635 | This way user will not see the hint moving down. 636 | */ 637 | -webkit-transition: opacity 1s, -webkit-transform 0.5s 1s; 638 | -moz-transition: opacity 1s, -moz-transform 0.5s 1s; 639 | -ms-transition: opacity 1s, -ms-transform 0.5s 1s; 640 | -o-transition: opacity 1s, -o-transform 0.5s 1s; 641 | transition: opacity 1s, transform 0.5s 1s; 642 | } 643 | 644 | /* 645 | Now we 'enable' the hint when presentation is initialized ... 646 | */ 647 | .impress-enabled .hint { display: block } 648 | 649 | /* 650 | ... and we will show it when the first step (with id 'bored') is active. 651 | */ 652 | .impress-on-welcome .hint { 653 | /* 654 | We remove the transparency and position the hint in its default fixed 655 | position. 656 | */ 657 | opacity: 1; 658 | 659 | -webkit-transform: translateY(0px); 660 | -moz-transform: translateY(0px); 661 | -ms-transform: translateY(0px); 662 | -o-transform: translateY(0px); 663 | transform: translateY(0px); 664 | 665 | /* 666 | Now for fade in transition we have the oposite situation from the one 667 | above. 668 | 669 | First after 4.5s delay we animate the transform property to move the hint 670 | into its correct position and after that we fade it in with opacity 671 | transition. 672 | */ 673 | -webkit-transition: opacity 1s 5s, -webkit-transform 0.5s 4.5s; 674 | -moz-transition: opacity 1s 5s, -moz-transform 0.5s 4.5s; 675 | -ms-transition: opacity 1s 5s, -ms-transform 0.5s 4.5s; 676 | -o-transition: opacity 1s 5s, -o-transform 0.5s 4.5s; 677 | transition: opacity 1s 5s, transform 0.5s 4.5s; 678 | 679 | } 680 | 681 | /* 682 | And as the last thing there is a workaround for quite strange bug. 683 | It happens a lot in Chrome. I don't remember if I've seen it in Firefox. 684 | 685 | Sometimes the element positioned in 3D (especially when it's moved back 686 | along Z axis) is not clickable, because it falls 'behind' the 687 | element. 688 | 689 | To prevent this, I decided to make non clickable by setting 690 | pointer-events property to `none` value. 691 | Value if this property is inherited, so to make everything else clickable 692 | I bring it back on the #impress element. 693 | 694 | If you want to know more about `pointer-events` here are some docs: 695 | https://developer.mozilla.org/en/CSS/pointer-events 696 | 697 | There is one very important thing to notice about this workaround - it makes 698 | everything 'unclickable' except what's in #impress element. 699 | 700 | So use it wisely ... or don't use at all. 701 | */ 702 | .impress-enabled { pointer-events: none } 703 | .impress-enabled #impress { pointer-events: auto } 704 | 705 | /* 706 | There is one funny thing I just realized. 707 | 708 | Thanks to this workaround above everything except #impress element is invisible 709 | for click events. That means that the hint element is also not clickable. 710 | So basically all of this transforms and delayed transitions trickery was probably 711 | not needed at all... 712 | 713 | But it was fun to learn about it, wasn't it? 714 | */ 715 | 716 | /* 717 | That's all I have for you in this file. 718 | Thanks for reading. I hope you enjoyed it at least as much as I enjoyed writing it 719 | for you. 720 | */ 721 | 722 | -------------------------------------------------------------------------------- /webapp/css/scala-tour.css: -------------------------------------------------------------------------------- 1 | 2 | .workspace { 3 | position: fixed; 4 | top: 0; 5 | bottom: 0; 6 | left: 0; 7 | right: 350px; 8 | min-height: 200px; 9 | text-shadow: 0 0 0 black 10 | 11 | } 12 | 13 | .workspace-top { 14 | position: absolute; 15 | top: 0; 16 | bottom: 30%; 17 | right: 0; 18 | left: 0; 19 | } 20 | 21 | .workspace-bottom { 22 | position: absolute; 23 | top: 70%; 24 | bottom: 0; 25 | right: 0; 26 | left: 0; 27 | } 28 | 29 | 30 | .workspace-editor { 31 | position: absolute; 32 | top: .6em; 33 | bottom: .3em; 34 | right: .6em; 35 | left: .6em; 36 | border: 1px solid #375EAB; 37 | background: #FFFFD8; 38 | -webkit-border-top-left-radius: 2px; 39 | -webkit-border-top-right-radius: 2px; 40 | -moz-border-radius-topleft: 2px; 41 | -moz-border-radius-topright: 2px; 42 | border-top-left-radius: 2px; 43 | border-top-right-radius: 2px; 44 | } 45 | 46 | .editor { 47 | position: absolute; 48 | top: 5px; 49 | bottom: 5px; 50 | left: 5px; 51 | right: 5px; 52 | border: none; 53 | outline: none; 54 | resize: none; 55 | background: #FFFFD8; 56 | font-size: 16px; 57 | } 58 | 59 | .output { 60 | position: absolute; 61 | top: .3em; 62 | bottom: .6em; 63 | right: .6em; 64 | left: .6em; 65 | border: 1px solid #375EAB; 66 | background: #FFFFFF; 67 | -webkit-border-bottom-left-radius: 5px; 68 | -webkit-border-bottom-right-radius: 5px; 69 | -moz-border-radius-bottomleft: 5px; 70 | -moz-border-radius-tottomright: 5px; 71 | border-bottom-left-radius: 5px; 72 | border-bottom-right-radius: 5px; 73 | overflow: auto; 74 | } 75 | .output pre { 76 | margin: 5px; 77 | padding: 0; 78 | background: none; 79 | font-size: 16px; 80 | } 81 | .output .loading { 82 | color: #999; 83 | } 84 | .output .stdout, .output pre { 85 | color: black; 86 | } 87 | .output .stderr, .output .error { 88 | color: #a11; 89 | } 90 | .output .system, .output .exit { 91 | color: #375EAB; 92 | } 93 | .output .stderr, .output .err { 94 | color: #a11; 95 | } 96 | .output img { 97 | margin: 5px; 98 | padding: 0; 99 | border: none; 100 | } 101 | 102 | pre, code { 103 | font-family: Menlo, monospace; 104 | font-size: 14px; 105 | border: 0px solid black; 106 | } 107 | 108 | div.source { 109 | display: none; 110 | } 111 | 112 | .CodeMirror { 113 | position: absolute; 114 | top: 0; 115 | bottom: 0; 116 | left: 0; 117 | right: 0; 118 | background: none; 119 | font-family: Menlo, monospace; 120 | } 121 | .CodeMirror { 122 | height: auto; 123 | } 124 | 125 | .CodeMirror-scroll { 126 | height: 100%; 127 | } 128 | 129 | .CodeMirror-gutter { 130 | background: inherit; 131 | border-right: 1px solid #999; 132 | } 133 | 134 | .CodeMirror-gutter-text { 135 | color: #999; 136 | } 137 | 138 | .CodeMirror-matchingbracket { 139 | color: #000 !important; 140 | text-decoration: underline; 141 | } 142 | 143 | .CodeMirror-focused div.CodeMirror-selected { 144 | background: #d0d0d0; 145 | } 146 | 147 | .CodeMirror .errLine { 148 | background: #FDD !important; 149 | } 150 | 151 | 152 | 153 | pre, code { 154 | font-family: Menlo, monospace; 155 | font-size: 14px; 156 | } 157 | 158 | 159 | .controls { 160 | position: absolute; 161 | z-index: 10; 162 | top: 1.2em; 163 | right: 2em; 164 | text-align: right; 165 | } 166 | 167 | .output , .error { 168 | color: #a11; 169 | } 170 | 171 | #brower-not-support{ 172 | margin-top: 100px; 173 | font-size: 200%; 174 | } 175 | 176 | #welcome h1 { 177 | font-size: 96px; 178 | } 179 | 180 | 181 | #welcome p { 182 | font-size: 48px; 183 | line-height:3; 184 | } 185 | 186 | .scala-red { 187 | color:rgb(192, 32, 0); 188 | } 189 | 190 | #logo img{ 191 | margin-right: 4px; 192 | } 193 | 194 | .navbar a { 195 | background: rgba(255,255,255,0); 196 | } 197 | 198 | .navbar .lang { 199 | display: block; 200 | float: right; 201 | padding: 10px 10px 10px; 202 | color: #777777; 203 | text-shadow: 0 1px 0 #ffffff; 204 | } 205 | 206 | .navbar .doc { 207 | display: block; 208 | float: right; 209 | padding: 10px 10px 10px; 210 | 211 | } 212 | 213 | .navbar .doc a{ 214 | color: #777777; 215 | text-shadow: 0 1px 0 #ffffff; 216 | } 217 | 218 | 219 | .slide p{ 220 | font-family: Helvetica, Arial, sans-serif; 221 | font-size: 16px; 222 | line-height: 1.6em; 223 | color: #222; 224 | } 225 | 226 | 227 | .impress-enabled .deeply{ 228 | -webkit-transition: opacity 0s; 229 | -moz-transition: opacity 0s; 230 | -ms-transition: opacity 0s; 231 | -o-transition: opacity 0s; 232 | transition: opacity 0s; 233 | } 234 | .onload{ 235 | opacity:0; 236 | } 237 | 238 | .chapter h2{ 239 | font-size: 120px; 240 | } 241 | 242 | .chapter .btn{ 243 | margin-left: 20px; 244 | margin-top: 150px; 245 | } 246 | 247 | .active .chapter .btn{ 248 | } 249 | 250 | 251 | .step-nav{ 252 | position: absolute; 253 | top: 90%; 254 | bottom: 0; 255 | right: 0; 256 | } 257 | 258 | #designed-by { 259 | position: absolute; 260 | top: 0; 261 | } 262 | 263 | #designed-by a { 264 | background-color: rgba(0, 0, 0, 0); 265 | } 266 | 267 | #compass { 268 | position: absolute; 269 | width: 300px; 270 | top: 0; 271 | bottom: 0; 272 | left: 0; 273 | right: 0; 274 | } 275 | 276 | #about { 277 | font-size: 24px; 278 | line-height: 3; 279 | } 280 | #about li{ 281 | line-height: 2; 282 | } 283 | #about a{ 284 | background: rgba(255,255,255,0); 285 | } 286 | 287 | #using-actor p a{ 288 | background: rgba(255,255,255,0); 289 | } 290 | 291 | .contents h2{ 292 | font-size: 36px; 293 | line-height: 2; 294 | } 295 | 296 | .contents li{ 297 | list-style-image:url('../img/glyphicons_065_tag.png'); 298 | } 299 | .contents .overview-li{ 300 | list-style-image:url('../img/glyphicons_221_unshare.png'); 301 | } 302 | 303 | .contents li{ 304 | line-height: 2; 305 | } 306 | 307 | .contents li a { 308 | text-shadow: 0 2px 2px rgba(0, 0, 0, .1); 309 | font-size: 24px; 310 | color: inherit; 311 | background: rgba(255,255,255,0); 312 | } 313 | 314 | .important{ 315 | font-weight:bold; 316 | } 317 | 318 | #disqus_thread { 319 | height:600px; 320 | overflow: auto; 321 | } 322 | 323 | .deeply h2{ 324 | font-size: 30px; 325 | } -------------------------------------------------------------------------------- /webapp/css/solarized.css: -------------------------------------------------------------------------------- 1 | /* 2 | Solarized theme for code-mirror 3 | http://ethanschoonover.com/solarized 4 | */ 5 | 6 | /* 7 | Solarized color pallet 8 | http://ethanschoonover.com/solarized/img/solarized-palette.png 9 | */ 10 | 11 | .solarized.base03 { color: #002b36; } 12 | .solarized.base02 { color: #073642; } 13 | .solarized.base01 { color: #586e75; } 14 | .solarized.base00 { color: #657b83; } 15 | .solarized.base0 { color: #839496; } 16 | .solarized.base1 { color: #93a1a1; } 17 | .solarized.base2 { color: #eee8d5; } 18 | .solarized.base3 { color: #fdf6e3; } 19 | .solarized.solar-yellow { color: #b58900; } 20 | .solarized.solar-orange { color: #cb4b16; } 21 | .solarized.solar-red { color: #dc322f; } 22 | .solarized.solar-magenta { color: #d33682; } 23 | .solarized.solar-violet { color: #6c71c4; } 24 | .solarized.solar-blue { color: #268bd2; } 25 | .solarized.solar-cyan { color: #2aa198; } 26 | .solarized.solar-green { color: #859900; } 27 | 28 | /* Color scheme for code-mirror */ 29 | 30 | .cm-s-solarized { 31 | line-height: 1.45em; 32 | font-family: Menlo,Monaco,"Andale Mono","lucida console","Courier New",monospace !important; 33 | color-profile: sRGB; 34 | rendering-intent: auto; 35 | } 36 | .cm-s-solarized.cm-s-dark { 37 | color: #839496; 38 | background-color: #002b36; 39 | text-shadow: #002b36 0 1px; 40 | } 41 | .cm-s-solarized.cm-s-light { 42 | background-color: #FFFFD8; 43 | color: #657b83; 44 | text-shadow: #eee8d5 0 1px; 45 | } 46 | 47 | .cm-s-solarized .CodeMirror-widget { 48 | text-shadow: none; 49 | } 50 | 51 | 52 | .cm-s-solarized .cm-keyword { color: #cb4b16 } 53 | .cm-s-solarized .cm-atom { color: #d33682; } 54 | .cm-s-solarized .cm-number { color: #d33682; } 55 | .cm-s-solarized .cm-def { color: #2aa198; } 56 | 57 | .cm-s-solarized .cm-variable { color: #268bd2; } 58 | .cm-s-solarized .cm-variable-2 { color: #b58900; } 59 | .cm-s-solarized .cm-variable-3 { color: #6c71c4; } 60 | 61 | .cm-s-solarized .cm-property { color: #2aa198; } 62 | .cm-s-solarized .cm-operator {color: #6c71c4;} 63 | 64 | .cm-s-solarized .cm-comment { color: #586e75; font-style:italic; } 65 | 66 | .cm-s-solarized .cm-string { color: #859900; } 67 | .cm-s-solarized .cm-string-2 { color: #b58900; } 68 | 69 | .cm-s-solarized .cm-meta { color: #859900; } 70 | .cm-s-solarized .cm-error, 71 | .cm-s-solarized .cm-invalidchar { 72 | color: #586e75; 73 | border-bottom: 1px dotted #dc322f; 74 | } 75 | .cm-s-solarized .cm-qualifier { color: #b58900; } 76 | .cm-s-solarized .cm-builtin { color: #d33682; } 77 | .cm-s-solarized .cm-bracket { color: #cb4b16; } 78 | .cm-s-solarized .CodeMirror-matchingbracket { color: #859900; } 79 | .cm-s-solarized .CodeMirror-nonmatchingbracket { color: #dc322f; } 80 | .cm-s-solarized .cm-tag { color: #93a1a1 } 81 | .cm-s-solarized .cm-attribute { color: #2aa198; } 82 | .cm-s-solarized .cm-header { color: #586e75; } 83 | .cm-s-solarized .cm-quote { color: #93a1a1; } 84 | .cm-s-solarized .cm-hr { 85 | color: transparent; 86 | border-top: 1px solid #586e75; 87 | display: block; 88 | } 89 | .cm-s-solarized .cm-link { color: #93a1a1; cursor: pointer; } 90 | .cm-s-solarized .cm-special { color: #6c71c4; } 91 | .cm-s-solarized .cm-em { 92 | color: #999; 93 | text-decoration: underline; 94 | text-decoration-style: dotted; 95 | } 96 | .cm-s-solarized .cm-strong { color: #eee; } 97 | .cm-s-solarized .cm-tab:before { 98 | content: "➤"; /*visualize tab character*/ 99 | color: #586e75; 100 | } 101 | 102 | .cm-s-solarized.cm-s-dark .CodeMirror-focused .CodeMirror-selected { 103 | background: #386774; 104 | color: inherit; 105 | } 106 | 107 | .cm-s-solarized.cm-s-dark ::selection { 108 | background: #386774; 109 | color: inherit; 110 | } 111 | 112 | .cm-s-solarized.cm-s-dark .CodeMirror-selected { 113 | background: #586e75; 114 | } 115 | 116 | .cm-s-solarized.cm-s-light .CodeMirror-focused .CodeMirror-selected { 117 | background: #eee8d5; 118 | color: inherit; 119 | } 120 | 121 | .cm-s-solarized.cm-s-light ::selection { 122 | background: #eee8d5; 123 | color: inherit; 124 | } 125 | 126 | .cm-s-solarized.cm-s-light .CodeMirror-selected { 127 | background: #93a1a1; 128 | } 129 | 130 | 131 | 132 | /* Editor styling */ 133 | 134 | 135 | 136 | /* Little shadow on the view-port of the buffer view */ 137 | .cm-s-solarized.CodeMirror { 138 | -moz-box-shadow: inset 7px 0 12px -6px #000; 139 | -webkit-box-shadow: inset 7px 0 12px -6px #000; 140 | box-shadow: inset 7px 0 12px -6px #000; 141 | } 142 | 143 | /* Gutter border and some shadow from it */ 144 | /*.cm-s-solarized .CodeMirror-gutters { 145 | padding: 0 15px 0 10px; 146 | box-shadow: 0 10px 20px black; 147 | border-right: 1px solid;*/ 148 | } 149 | */ 150 | /* Gutter colors and line number styling based of color scheme (dark / light) */ 151 | 152 | /* Dark */ 153 | .cm-s-solarized.cm-s-dark .CodeMirror-gutters { 154 | background-color: #073642; 155 | border-color: #00232c; 156 | } 157 | 158 | .cm-s-solarized.cm-s-dark .CodeMirror-linenumber { 159 | text-shadow: #021014 0 -1px; 160 | } 161 | 162 | /* Light */ 163 | /*.cm-s-solarized.cm-s-light .CodeMirror-gutters { 164 | background-color: #eee8d5; 165 | border-color: #eee8d5; 166 | }*/ 167 | 168 | /* Common */ 169 | .cm-s-solarized .CodeMirror-linenumber { 170 | color: #586e75; 171 | } 172 | 173 | .cm-s-solarized .CodeMirror-gutter .CodeMirror-gutter-text { 174 | color: #586e75; 175 | } 176 | 177 | .cm-s-solarized .CodeMirror-lines { 178 | padding-left: 5px; 179 | } 180 | 181 | .cm-s-solarized .CodeMirror-lines .CodeMirror-cursor { 182 | border-left: 1px solid #819090; 183 | } 184 | 185 | /* 186 | Active line. Negative margin compensates left padding of the text in the 187 | view-port 188 | */ 189 | .cm-s-solarized .activeline { 190 | margin-left: -20px; 191 | } 192 | 193 | .cm-s-solarized.cm-s-dark .activeline { 194 | background: rgba(255, 255, 255, 0.05); 195 | 196 | } 197 | .cm-s-solarized.cm-s-light .activeline { 198 | background: rgba(0, 0, 0, 0.05); 199 | } 200 | 201 | -------------------------------------------------------------------------------- /webapp/favicon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/favicon.png -------------------------------------------------------------------------------- /webapp/img/Lambda_lc.svg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/img/Lambda_lc.svg.png -------------------------------------------------------------------------------- /webapp/img/glyphicons-halflings-white.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/img/glyphicons-halflings-white.png -------------------------------------------------------------------------------- /webapp/img/glyphicons-halflings.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/img/glyphicons-halflings.png -------------------------------------------------------------------------------- /webapp/img/glyphicons_065_tag.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/img/glyphicons_065_tag.png -------------------------------------------------------------------------------- /webapp/img/glyphicons_221_unshare.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/img/glyphicons_221_unshare.png -------------------------------------------------------------------------------- /webapp/img/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/yankay/scala-tour/7a482eea52e1ee429cb5961cb7ee5272227b44b1/webapp/img/logo.png -------------------------------------------------------------------------------- /webapp/js/bootstrap.min.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * Bootstrap.js by @fat & @mdo 3 | * Copyright 2012 Twitter, Inc. 4 | * http://www.apache.org/licenses/LICENSE-2.0.txt 5 | */ 6 | !function(e){"use strict";e(function(){e.support.transition=function(){var e=function(){var e=document.createElement("bootstrap"),t={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"},n;for(n in t)if(e.style[n]!==undefined)return t[n]}();return e&&{end:e}}()})}(window.jQuery),!function(e){"use strict";var t='[data-dismiss="alert"]',n=function(n){e(n).on("click",t,this.close)};n.prototype.close=function(t){function s(){i.trigger("closed").remove()}var n=e(this),r=n.attr("data-target"),i;r||(r=n.attr("href"),r=r&&r.replace(/.*(?=#[^\s]*$)/,"")),i=e(r),t&&t.preventDefault(),i.length||(i=n.hasClass("alert")?n:n.parent()),i.trigger(t=e.Event("close"));if(t.isDefaultPrevented())return;i.removeClass("in"),e.support.transition&&i.hasClass("fade")?i.on(e.support.transition.end,s):s()};var r=e.fn.alert;e.fn.alert=function(t){return this.each(function(){var r=e(this),i=r.data("alert");i||r.data("alert",i=new n(this)),typeof t=="string"&&i[t].call(r)})},e.fn.alert.Constructor=n,e.fn.alert.noConflict=function(){return e.fn.alert=r,this},e(document).on("click.alert.data-api",t,n.prototype.close)}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.options=e.extend({},e.fn.button.defaults,n)};t.prototype.setState=function(e){var t="disabled",n=this.$element,r=n.data(),i=n.is("input")?"val":"html";e+="Text",r.resetText||n.data("resetText",n[i]()),n[i](r[e]||this.options[e]),setTimeout(function(){e=="loadingText"?n.addClass(t).attr(t,t):n.removeClass(t).removeAttr(t)},0)},t.prototype.toggle=function(){var e=this.$element.closest('[data-toggle="buttons-radio"]');e&&e.find(".active").removeClass("active"),this.$element.toggleClass("active")};var n=e.fn.button;e.fn.button=function(n){return this.each(function(){var r=e(this),i=r.data("button"),s=typeof n=="object"&&n;i||r.data("button",i=new t(this,s)),n=="toggle"?i.toggle():n&&i.setState(n)})},e.fn.button.defaults={loadingText:"loading..."},e.fn.button.Constructor=t,e.fn.button.noConflict=function(){return e.fn.button=n,this},e(document).on("click.button.data-api","[data-toggle^=button]",function(t){var n=e(t.target);n.hasClass("btn")||(n=n.closest(".btn")),n.button("toggle")})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.$indicators=this.$element.find(".carousel-indicators"),this.options=n,this.options.pause=="hover"&&this.$element.on("mouseenter",e.proxy(this.pause,this)).on("mouseleave",e.proxy(this.cycle,this))};t.prototype={cycle:function(t){return t||(this.paused=!1),this.interval&&clearInterval(this.interval),this.options.interval&&!this.paused&&(this.interval=setInterval(e.proxy(this.next,this),this.options.interval)),this},getActiveIndex:function(){return this.$active=this.$element.find(".item.active"),this.$items=this.$active.parent().children(),this.$items.index(this.$active)},to:function(t){var n=this.getActiveIndex(),r=this;if(t>this.$items.length-1||t<0)return;return this.sliding?this.$element.one("slid",function(){r.to(t)}):n==t?this.pause().cycle():this.slide(t>n?"next":"prev",e(this.$items[t]))},pause:function(t){return t||(this.paused=!0),this.$element.find(".next, .prev").length&&e.support.transition.end&&(this.$element.trigger(e.support.transition.end),this.cycle(!0)),clearInterval(this.interval),this.interval=null,this},next:function(){if(this.sliding)return;return this.slide("next")},prev:function(){if(this.sliding)return;return this.slide("prev")},slide:function(t,n){var r=this.$element.find(".item.active"),i=n||r[t](),s=this.interval,o=t=="next"?"left":"right",u=t=="next"?"first":"last",a=this,f;this.sliding=!0,s&&this.pause(),i=i.length?i:this.$element.find(".item")[u](),f=e.Event("slide",{relatedTarget:i[0],direction:o});if(i.hasClass("active"))return;this.$indicators.length&&(this.$indicators.find(".active").removeClass("active"),this.$element.one("slid",function(){var t=e(a.$indicators.children()[a.getActiveIndex()]);t&&t.addClass("active")}));if(e.support.transition&&this.$element.hasClass("slide")){this.$element.trigger(f);if(f.isDefaultPrevented())return;i.addClass(t),i[0].offsetWidth,r.addClass(o),i.addClass(o),this.$element.one(e.support.transition.end,function(){i.removeClass([t,o].join(" ")).addClass("active"),r.removeClass(["active",o].join(" ")),a.sliding=!1,setTimeout(function(){a.$element.trigger("slid")},0)})}else{this.$element.trigger(f);if(f.isDefaultPrevented())return;r.removeClass("active"),i.addClass("active"),this.sliding=!1,this.$element.trigger("slid")}return s&&this.cycle(),this}};var n=e.fn.carousel;e.fn.carousel=function(n){return this.each(function(){var r=e(this),i=r.data("carousel"),s=e.extend({},e.fn.carousel.defaults,typeof n=="object"&&n),o=typeof n=="string"?n:s.slide;i||r.data("carousel",i=new t(this,s)),typeof n=="number"?i.to(n):o?i[o]():s.interval&&i.pause().cycle()})},e.fn.carousel.defaults={interval:5e3,pause:"hover"},e.fn.carousel.Constructor=t,e.fn.carousel.noConflict=function(){return e.fn.carousel=n,this},e(document).on("click.carousel.data-api","[data-slide], [data-slide-to]",function(t){var n=e(this),r,i=e(n.attr("data-target")||(r=n.attr("href"))&&r.replace(/.*(?=#[^\s]+$)/,"")),s=e.extend({},i.data(),n.data()),o;i.carousel(s),(o=n.attr("data-slide-to"))&&i.data("carousel").pause().to(o).cycle(),t.preventDefault()})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.options=e.extend({},e.fn.collapse.defaults,n),this.options.parent&&(this.$parent=e(this.options.parent)),this.options.toggle&&this.toggle()};t.prototype={constructor:t,dimension:function(){var e=this.$element.hasClass("width");return e?"width":"height"},show:function(){var t,n,r,i;if(this.transitioning||this.$element.hasClass("in"))return;t=this.dimension(),n=e.camelCase(["scroll",t].join("-")),r=this.$parent&&this.$parent.find("> .accordion-group > .in");if(r&&r.length){i=r.data("collapse");if(i&&i.transitioning)return;r.collapse("hide"),i||r.data("collapse",null)}this.$element[t](0),this.transition("addClass",e.Event("show"),"shown"),e.support.transition&&this.$element[t](this.$element[0][n])},hide:function(){var t;if(this.transitioning||!this.$element.hasClass("in"))return;t=this.dimension(),this.reset(this.$element[t]()),this.transition("removeClass",e.Event("hide"),"hidden"),this.$element[t](0)},reset:function(e){var t=this.dimension();return this.$element.removeClass("collapse")[t](e||"auto")[0].offsetWidth,this.$element[e!==null?"addClass":"removeClass"]("collapse"),this},transition:function(t,n,r){var i=this,s=function(){n.type=="show"&&i.reset(),i.transitioning=0,i.$element.trigger(r)};this.$element.trigger(n);if(n.isDefaultPrevented())return;this.transitioning=1,this.$element[t]("in"),e.support.transition&&this.$element.hasClass("collapse")?this.$element.one(e.support.transition.end,s):s()},toggle:function(){this[this.$element.hasClass("in")?"hide":"show"]()}};var n=e.fn.collapse;e.fn.collapse=function(n){return this.each(function(){var r=e(this),i=r.data("collapse"),s=e.extend({},e.fn.collapse.defaults,r.data(),typeof n=="object"&&n);i||r.data("collapse",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.collapse.defaults={toggle:!0},e.fn.collapse.Constructor=t,e.fn.collapse.noConflict=function(){return e.fn.collapse=n,this},e(document).on("click.collapse.data-api","[data-toggle=collapse]",function(t){var n=e(this),r,i=n.attr("data-target")||t.preventDefault()||(r=n.attr("href"))&&r.replace(/.*(?=#[^\s]+$)/,""),s=e(i).data("collapse")?"toggle":n.data();n[e(i).hasClass("in")?"addClass":"removeClass"]("collapsed"),e(i).collapse(s)})}(window.jQuery),!function(e){"use strict";function r(){e(t).each(function(){i(e(this)).removeClass("open")})}function i(t){var n=t.attr("data-target"),r;n||(n=t.attr("href"),n=n&&/#/.test(n)&&n.replace(/.*(?=#[^\s]*$)/,"")),r=n&&e(n);if(!r||!r.length)r=t.parent();return r}var t="[data-toggle=dropdown]",n=function(t){var n=e(t).on("click.dropdown.data-api",this.toggle);e("html").on("click.dropdown.data-api",function(){n.parent().removeClass("open")})};n.prototype={constructor:n,toggle:function(t){var n=e(this),s,o;if(n.is(".disabled, :disabled"))return;return s=i(n),o=s.hasClass("open"),r(),o||s.toggleClass("open"),n.focus(),!1},keydown:function(n){var r,s,o,u,a,f;if(!/(38|40|27)/.test(n.keyCode))return;r=e(this),n.preventDefault(),n.stopPropagation();if(r.is(".disabled, :disabled"))return;u=i(r),a=u.hasClass("open");if(!a||a&&n.keyCode==27)return n.which==27&&u.find(t).focus(),r.click();s=e("[role=menu] li:not(.divider):visible a",u);if(!s.length)return;f=s.index(s.filter(":focus")),n.keyCode==38&&f>0&&f--,n.keyCode==40&&f').appendTo(document.body),this.$backdrop.click(this.options.backdrop=="static"?e.proxy(this.$element[0].focus,this.$element[0]):e.proxy(this.hide,this)),i&&this.$backdrop[0].offsetWidth,this.$backdrop.addClass("in");if(!t)return;i?this.$backdrop.one(e.support.transition.end,t):t()}else!this.isShown&&this.$backdrop?(this.$backdrop.removeClass("in"),e.support.transition&&this.$element.hasClass("fade")?this.$backdrop.one(e.support.transition.end,t):t()):t&&t()}};var n=e.fn.modal;e.fn.modal=function(n){return this.each(function(){var r=e(this),i=r.data("modal"),s=e.extend({},e.fn.modal.defaults,r.data(),typeof n=="object"&&n);i||r.data("modal",i=new t(this,s)),typeof n=="string"?i[n]():s.show&&i.show()})},e.fn.modal.defaults={backdrop:!0,keyboard:!0,show:!0},e.fn.modal.Constructor=t,e.fn.modal.noConflict=function(){return e.fn.modal=n,this},e(document).on("click.modal.data-api",'[data-toggle="modal"]',function(t){var n=e(this),r=n.attr("href"),i=e(n.attr("data-target")||r&&r.replace(/.*(?=#[^\s]+$)/,"")),s=i.data("modal")?"toggle":e.extend({remote:!/#/.test(r)&&r},i.data(),n.data());t.preventDefault(),i.modal(s).one("hide",function(){n.focus()})})}(window.jQuery),!function(e){"use strict";var t=function(e,t){this.init("tooltip",e,t)};t.prototype={constructor:t,init:function(t,n,r){var i,s,o,u,a;this.type=t,this.$element=e(n),this.options=this.getOptions(r),this.enabled=!0,o=this.options.trigger.split(" ");for(a=o.length;a--;)u=o[a],u=="click"?this.$element.on("click."+this.type,this.options.selector,e.proxy(this.toggle,this)):u!="manual"&&(i=u=="hover"?"mouseenter":"focus",s=u=="hover"?"mouseleave":"blur",this.$element.on(i+"."+this.type,this.options.selector,e.proxy(this.enter,this)),this.$element.on(s+"."+this.type,this.options.selector,e.proxy(this.leave,this)));this.options.selector?this._options=e.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()},getOptions:function(t){return t=e.extend({},e.fn[this.type].defaults,this.$element.data(),t),t.delay&&typeof t.delay=="number"&&(t.delay={show:t.delay,hide:t.delay}),t},enter:function(t){var n=e.fn[this.type].defaults,r={},i;this._options&&e.each(this._options,function(e,t){n[e]!=t&&(r[e]=t)},this),i=e(t.currentTarget)[this.type](r).data(this.type);if(!i.options.delay||!i.options.delay.show)return i.show();clearTimeout(this.timeout),i.hoverState="in",this.timeout=setTimeout(function(){i.hoverState=="in"&&i.show()},i.options.delay.show)},leave:function(t){var n=e(t.currentTarget)[this.type](this._options).data(this.type);this.timeout&&clearTimeout(this.timeout);if(!n.options.delay||!n.options.delay.hide)return n.hide();n.hoverState="out",this.timeout=setTimeout(function(){n.hoverState=="out"&&n.hide()},n.options.delay.hide)},show:function(){var t,n,r,i,s,o,u=e.Event("show");if(this.hasContent()&&this.enabled){this.$element.trigger(u);if(u.isDefaultPrevented())return;t=this.tip(),this.setContent(),this.options.animation&&t.addClass("fade"),s=typeof this.options.placement=="function"?this.options.placement.call(this,t[0],this.$element[0]):this.options.placement,t.detach().css({top:0,left:0,display:"block"}),this.options.container?t.appendTo(this.options.container):t.insertAfter(this.$element),n=this.getPosition(),r=t[0].offsetWidth,i=t[0].offsetHeight;switch(s){case"bottom":o={top:n.top+n.height,left:n.left+n.width/2-r/2};break;case"top":o={top:n.top-i,left:n.left+n.width/2-r/2};break;case"left":o={top:n.top+n.height/2-i/2,left:n.left-r};break;case"right":o={top:n.top+n.height/2-i/2,left:n.left+n.width}}this.applyPlacement(o,s),this.$element.trigger("shown")}},applyPlacement:function(e,t){var n=this.tip(),r=n[0].offsetWidth,i=n[0].offsetHeight,s,o,u,a;n.offset(e).addClass(t).addClass("in"),s=n[0].offsetWidth,o=n[0].offsetHeight,t=="top"&&o!=i&&(e.top=e.top+i-o,a=!0),t=="bottom"||t=="top"?(u=0,e.left<0&&(u=e.left*-2,e.left=0,n.offset(e),s=n[0].offsetWidth,o=n[0].offsetHeight),this.replaceArrow(u-r+s,s,"left")):this.replaceArrow(o-i,o,"top"),a&&n.offset(e)},replaceArrow:function(e,t,n){this.arrow().css(n,e?50*(1-e/t)+"%":"")},setContent:function(){var e=this.tip(),t=this.getTitle();e.find(".tooltip-inner")[this.options.html?"html":"text"](t),e.removeClass("fade in top bottom left right")},hide:function(){function i(){var t=setTimeout(function(){n.off(e.support.transition.end).detach()},500);n.one(e.support.transition.end,function(){clearTimeout(t),n.detach()})}var t=this,n=this.tip(),r=e.Event("hide");this.$element.trigger(r);if(r.isDefaultPrevented())return;return n.removeClass("in"),e.support.transition&&this.$tip.hasClass("fade")?i():n.detach(),this.$element.trigger("hidden"),this},fixTitle:function(){var e=this.$element;(e.attr("title")||typeof e.attr("data-original-title")!="string")&&e.attr("data-original-title",e.attr("title")||"").attr("title","")},hasContent:function(){return this.getTitle()},getPosition:function(){var t=this.$element[0];return e.extend({},typeof t.getBoundingClientRect=="function"?t.getBoundingClientRect():{width:t.offsetWidth,height:t.offsetHeight},this.$element.offset())},getTitle:function(){var e,t=this.$element,n=this.options;return e=t.attr("data-original-title")||(typeof n.title=="function"?n.title.call(t[0]):n.title),e},tip:function(){return this.$tip=this.$tip||e(this.options.template)},arrow:function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")},validate:function(){this.$element[0].parentNode||(this.hide(),this.$element=null,this.options=null)},enable:function(){this.enabled=!0},disable:function(){this.enabled=!1},toggleEnabled:function(){this.enabled=!this.enabled},toggle:function(t){var n=t?e(t.currentTarget)[this.type](this._options).data(this.type):this;n.tip().hasClass("in")?n.hide():n.show()},destroy:function(){this.hide().$element.off("."+this.type).removeData(this.type)}};var n=e.fn.tooltip;e.fn.tooltip=function(n){return this.each(function(){var r=e(this),i=r.data("tooltip"),s=typeof n=="object"&&n;i||r.data("tooltip",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.tooltip.Constructor=t,e.fn.tooltip.defaults={animation:!0,placement:"top",selector:!1,template:'
',trigger:"hover focus",title:"",delay:0,html:!1,container:!1},e.fn.tooltip.noConflict=function(){return e.fn.tooltip=n,this}}(window.jQuery),!function(e){"use strict";var t=function(e,t){this.init("popover",e,t)};t.prototype=e.extend({},e.fn.tooltip.Constructor.prototype,{constructor:t,setContent:function(){var e=this.tip(),t=this.getTitle(),n=this.getContent();e.find(".popover-title")[this.options.html?"html":"text"](t),e.find(".popover-content")[this.options.html?"html":"text"](n),e.removeClass("fade top bottom left right in")},hasContent:function(){return this.getTitle()||this.getContent()},getContent:function(){var e,t=this.$element,n=this.options;return e=(typeof n.content=="function"?n.content.call(t[0]):n.content)||t.attr("data-content"),e},tip:function(){return this.$tip||(this.$tip=e(this.options.template)),this.$tip},destroy:function(){this.hide().$element.off("."+this.type).removeData(this.type)}});var n=e.fn.popover;e.fn.popover=function(n){return this.each(function(){var r=e(this),i=r.data("popover"),s=typeof n=="object"&&n;i||r.data("popover",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.popover.Constructor=t,e.fn.popover.defaults=e.extend({},e.fn.tooltip.defaults,{placement:"right",trigger:"click",content:"",template:'

'}),e.fn.popover.noConflict=function(){return e.fn.popover=n,this}}(window.jQuery),!function(e){"use strict";function t(t,n){var r=e.proxy(this.process,this),i=e(t).is("body")?e(window):e(t),s;this.options=e.extend({},e.fn.scrollspy.defaults,n),this.$scrollElement=i.on("scroll.scroll-spy.data-api",r),this.selector=(this.options.target||(s=e(t).attr("href"))&&s.replace(/.*(?=#[^\s]+$)/,"")||"")+" .nav li > a",this.$body=e("body"),this.refresh(),this.process()}t.prototype={constructor:t,refresh:function(){var t=this,n;this.offsets=e([]),this.targets=e([]),n=this.$body.find(this.selector).map(function(){var n=e(this),r=n.data("target")||n.attr("href"),i=/^#\w/.test(r)&&e(r);return i&&i.length&&[[i.position().top+(!e.isWindow(t.$scrollElement.get(0))&&t.$scrollElement.scrollTop()),r]]||null}).sort(function(e,t){return e[0]-t[0]}).each(function(){t.offsets.push(this[0]),t.targets.push(this[1])})},process:function(){var e=this.$scrollElement.scrollTop()+this.options.offset,t=this.$scrollElement[0].scrollHeight||this.$body[0].scrollHeight,n=t-this.$scrollElement.height(),r=this.offsets,i=this.targets,s=this.activeTarget,o;if(e>=n)return s!=(o=i.last()[0])&&this.activate(o);for(o=r.length;o--;)s!=i[o]&&e>=r[o]&&(!r[o+1]||e<=r[o+1])&&this.activate(i[o])},activate:function(t){var n,r;this.activeTarget=t,e(this.selector).parent(".active").removeClass("active"),r=this.selector+'[data-target="'+t+'"],'+this.selector+'[href="'+t+'"]',n=e(r).parent("li").addClass("active"),n.parent(".dropdown-menu").length&&(n=n.closest("li.dropdown").addClass("active")),n.trigger("activate")}};var n=e.fn.scrollspy;e.fn.scrollspy=function(n){return this.each(function(){var r=e(this),i=r.data("scrollspy"),s=typeof n=="object"&&n;i||r.data("scrollspy",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.scrollspy.Constructor=t,e.fn.scrollspy.defaults={offset:10},e.fn.scrollspy.noConflict=function(){return e.fn.scrollspy=n,this},e(window).on("load",function(){e('[data-spy="scroll"]').each(function(){var t=e(this);t.scrollspy(t.data())})})}(window.jQuery),!function(e){"use strict";var t=function(t){this.element=e(t)};t.prototype={constructor:t,show:function(){var t=this.element,n=t.closest("ul:not(.dropdown-menu)"),r=t.attr("data-target"),i,s,o;r||(r=t.attr("href"),r=r&&r.replace(/.*(?=#[^\s]*$)/,""));if(t.parent("li").hasClass("active"))return;i=n.find(".active:last a")[0],o=e.Event("show",{relatedTarget:i}),t.trigger(o);if(o.isDefaultPrevented())return;s=e(r),this.activate(t.parent("li"),n),this.activate(s,s.parent(),function(){t.trigger({type:"shown",relatedTarget:i})})},activate:function(t,n,r){function o(){i.removeClass("active").find("> .dropdown-menu > .active").removeClass("active"),t.addClass("active"),s?(t[0].offsetWidth,t.addClass("in")):t.removeClass("fade"),t.parent(".dropdown-menu")&&t.closest("li.dropdown").addClass("active"),r&&r()}var i=n.find("> .active"),s=r&&e.support.transition&&i.hasClass("fade");s?i.one(e.support.transition.end,o):o(),i.removeClass("in")}};var n=e.fn.tab;e.fn.tab=function(n){return this.each(function(){var r=e(this),i=r.data("tab");i||r.data("tab",i=new t(this)),typeof n=="string"&&i[n]()})},e.fn.tab.Constructor=t,e.fn.tab.noConflict=function(){return e.fn.tab=n,this},e(document).on("click.tab.data-api",'[data-toggle="tab"], [data-toggle="pill"]',function(t){t.preventDefault(),e(this).tab("show")})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.$element=e(t),this.options=e.extend({},e.fn.typeahead.defaults,n),this.matcher=this.options.matcher||this.matcher,this.sorter=this.options.sorter||this.sorter,this.highlighter=this.options.highlighter||this.highlighter,this.updater=this.options.updater||this.updater,this.source=this.options.source,this.$menu=e(this.options.menu),this.shown=!1,this.listen()};t.prototype={constructor:t,select:function(){var e=this.$menu.find(".active").attr("data-value");return this.$element.val(this.updater(e)).change(),this.hide()},updater:function(e){return e},show:function(){var t=e.extend({},this.$element.position(),{height:this.$element[0].offsetHeight});return this.$menu.insertAfter(this.$element).css({top:t.top+t.height,left:t.left}).show(),this.shown=!0,this},hide:function(){return this.$menu.hide(),this.shown=!1,this},lookup:function(t){var n;return this.query=this.$element.val(),!this.query||this.query.length"+t+""})},render:function(t){var n=this;return t=e(t).map(function(t,r){return t=e(n.options.item).attr("data-value",r),t.find("a").html(n.highlighter(r)),t[0]}),t.first().addClass("active"),this.$menu.html(t),this},next:function(t){var n=this.$menu.find(".active").removeClass("active"),r=n.next();r.length||(r=e(this.$menu.find("li")[0])),r.addClass("active")},prev:function(e){var t=this.$menu.find(".active").removeClass("active"),n=t.prev();n.length||(n=this.$menu.find("li").last()),n.addClass("active")},listen:function(){this.$element.on("focus",e.proxy(this.focus,this)).on("blur",e.proxy(this.blur,this)).on("keypress",e.proxy(this.keypress,this)).on("keyup",e.proxy(this.keyup,this)),this.eventSupported("keydown")&&this.$element.on("keydown",e.proxy(this.keydown,this)),this.$menu.on("click",e.proxy(this.click,this)).on("mouseenter","li",e.proxy(this.mouseenter,this)).on("mouseleave","li",e.proxy(this.mouseleave,this))},eventSupported:function(e){var t=e in this.$element;return t||(this.$element.setAttribute(e,"return;"),t=typeof this.$element[e]=="function"),t},move:function(e){if(!this.shown)return;switch(e.keyCode){case 9:case 13:case 27:e.preventDefault();break;case 38:e.preventDefault(),this.prev();break;case 40:e.preventDefault(),this.next()}e.stopPropagation()},keydown:function(t){this.suppressKeyPressRepeat=~e.inArray(t.keyCode,[40,38,9,13,27]),this.move(t)},keypress:function(e){if(this.suppressKeyPressRepeat)return;this.move(e)},keyup:function(e){switch(e.keyCode){case 40:case 38:case 16:case 17:case 18:break;case 9:case 13:if(!this.shown)return;this.select();break;case 27:if(!this.shown)return;this.hide();break;default:this.lookup()}e.stopPropagation(),e.preventDefault()},focus:function(e){this.focused=!0},blur:function(e){this.focused=!1,!this.mousedover&&this.shown&&this.hide()},click:function(e){e.stopPropagation(),e.preventDefault(),this.select(),this.$element.focus()},mouseenter:function(t){this.mousedover=!0,this.$menu.find(".active").removeClass("active"),e(t.currentTarget).addClass("active")},mouseleave:function(e){this.mousedover=!1,!this.focused&&this.shown&&this.hide()}};var n=e.fn.typeahead;e.fn.typeahead=function(n){return this.each(function(){var r=e(this),i=r.data("typeahead"),s=typeof n=="object"&&n;i||r.data("typeahead",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.typeahead.defaults={source:[],items:8,menu:'',item:'
  • ',minLength:1},e.fn.typeahead.Constructor=t,e.fn.typeahead.noConflict=function(){return e.fn.typeahead=n,this},e(document).on("focus.typeahead.data-api",'[data-provide="typeahead"]',function(t){var n=e(this);if(n.data("typeahead"))return;n.typeahead(n.data())})}(window.jQuery),!function(e){"use strict";var t=function(t,n){this.options=e.extend({},e.fn.affix.defaults,n),this.$window=e(window).on("scroll.affix.data-api",e.proxy(this.checkPosition,this)).on("click.affix.data-api",e.proxy(function(){setTimeout(e.proxy(this.checkPosition,this),1)},this)),this.$element=e(t),this.checkPosition()};t.prototype.checkPosition=function(){if(!this.$element.is(":visible"))return;var t=e(document).height(),n=this.$window.scrollTop(),r=this.$element.offset(),i=this.options.offset,s=i.bottom,o=i.top,u="affix affix-top affix-bottom",a;typeof i!="object"&&(s=o=i),typeof o=="function"&&(o=i.top()),typeof s=="function"&&(s=i.bottom()),a=this.unpin!=null&&n+this.unpin<=r.top?!1:s!=null&&r.top+this.$element.height()>=t-s?"bottom":o!=null&&n<=o?"top":!1;if(this.affixed===a)return;this.affixed=a,this.unpin=a=="bottom"?r.top-n:null,this.$element.removeClass(u).addClass("affix"+(a?"-"+a:""))};var n=e.fn.affix;e.fn.affix=function(n){return this.each(function(){var r=e(this),i=r.data("affix"),s=typeof n=="object"&&n;i||r.data("affix",i=new t(this,s)),typeof n=="string"&&i[n]()})},e.fn.affix.Constructor=t,e.fn.affix.defaults={offset:0},e.fn.affix.noConflict=function(){return e.fn.affix=n,this},e(window).on("load",function(){e('[data-spy="affix"]').each(function(){var t=e(this),n=t.data();n.offset=n.offset||{},n.offsetBottom&&(n.offset.bottom=n.offsetBottom),n.offsetTop&&(n.offset.top=n.offsetTop),t.affix(n)})})}(window.jQuery); -------------------------------------------------------------------------------- /webapp/js/clike.js: -------------------------------------------------------------------------------- 1 | CodeMirror.defineMode("clike", function(config, parserConfig) { 2 | var indentUnit = config.indentUnit, 3 | statementIndentUnit = parserConfig.statementIndentUnit || indentUnit, 4 | dontAlignCalls = parserConfig.dontAlignCalls, 5 | keywords = parserConfig.keywords || {}, 6 | builtin = parserConfig.builtin || {}, 7 | blockKeywords = parserConfig.blockKeywords || {}, 8 | atoms = parserConfig.atoms || {}, 9 | hooks = parserConfig.hooks || {}, 10 | multiLineStrings = parserConfig.multiLineStrings; 11 | var isOperatorChar = /[+\-*&%=<>!?|\/]/; 12 | 13 | var curPunc; 14 | 15 | function tokenBase(stream, state) { 16 | var ch = stream.next(); 17 | if (hooks[ch]) { 18 | var result = hooks[ch](stream, state); 19 | if (result !== false) return result; 20 | } 21 | if (ch == '"' || ch == "'") { 22 | state.tokenize = tokenString(ch); 23 | return state.tokenize(stream, state); 24 | } 25 | if (/[\[\]{}\(\),;\:\.]/.test(ch)) { 26 | curPunc = ch; 27 | return null; 28 | } 29 | if (/\d/.test(ch)) { 30 | stream.eatWhile(/[\w\.]/); 31 | return "number"; 32 | } 33 | if (ch == "/") { 34 | if (stream.eat("*")) { 35 | state.tokenize = tokenComment; 36 | return tokenComment(stream, state); 37 | } 38 | if (stream.eat("/")) { 39 | stream.skipToEnd(); 40 | return "comment"; 41 | } 42 | } 43 | if (isOperatorChar.test(ch)) { 44 | stream.eatWhile(isOperatorChar); 45 | return "operator"; 46 | } 47 | stream.eatWhile(/[\w\$_]/); 48 | var cur = stream.current(); 49 | if (keywords.propertyIsEnumerable(cur)) { 50 | if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; 51 | return "keyword"; 52 | } 53 | if (builtin.propertyIsEnumerable(cur)) { 54 | if (blockKeywords.propertyIsEnumerable(cur)) curPunc = "newstatement"; 55 | return "builtin"; 56 | } 57 | if (atoms.propertyIsEnumerable(cur)) return "atom"; 58 | return "variable"; 59 | } 60 | 61 | function tokenString(quote) { 62 | return function(stream, state) { 63 | var escaped = false, next, end = false; 64 | while ((next = stream.next()) != null) { 65 | if (next == quote && !escaped) {end = true; break;} 66 | escaped = !escaped && next == "\\"; 67 | } 68 | if (end || !(escaped || multiLineStrings)) 69 | state.tokenize = null; 70 | return "string"; 71 | }; 72 | } 73 | 74 | function tokenComment(stream, state) { 75 | var maybeEnd = false, ch; 76 | while (ch = stream.next()) { 77 | if (ch == "/" && maybeEnd) { 78 | state.tokenize = null; 79 | break; 80 | } 81 | maybeEnd = (ch == "*"); 82 | } 83 | return "comment"; 84 | } 85 | 86 | function Context(indented, column, type, align, prev) { 87 | this.indented = indented; 88 | this.column = column; 89 | this.type = type; 90 | this.align = align; 91 | this.prev = prev; 92 | } 93 | function pushContext(state, col, type) { 94 | var indent = state.indented; 95 | if (state.context && state.context.type == "statement") 96 | indent = state.context.indented; 97 | return state.context = new Context(indent, col, type, null, state.context); 98 | } 99 | function popContext(state) { 100 | var t = state.context.type; 101 | if (t == ")" || t == "]" || t == "}") 102 | state.indented = state.context.indented; 103 | return state.context = state.context.prev; 104 | } 105 | 106 | // Interface 107 | 108 | return { 109 | startState: function(basecolumn) { 110 | return { 111 | tokenize: null, 112 | context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), 113 | indented: 0, 114 | startOfLine: true 115 | }; 116 | }, 117 | 118 | token: function(stream, state) { 119 | var ctx = state.context; 120 | if (stream.sol()) { 121 | if (ctx.align == null) ctx.align = false; 122 | state.indented = stream.indentation(); 123 | state.startOfLine = true; 124 | } 125 | if (stream.eatSpace()) return null; 126 | curPunc = null; 127 | var style = (state.tokenize || tokenBase)(stream, state); 128 | if (style == "comment" || style == "meta") return style; 129 | if (ctx.align == null) ctx.align = true; 130 | 131 | if ((curPunc == ";" || curPunc == ":" || curPunc == ",") && ctx.type == "statement") popContext(state); 132 | else if (curPunc == "{") pushContext(state, stream.column(), "}"); 133 | else if (curPunc == "[") pushContext(state, stream.column(), "]"); 134 | else if (curPunc == "(") pushContext(state, stream.column(), ")"); 135 | else if (curPunc == "}") { 136 | while (ctx.type == "statement") ctx = popContext(state); 137 | if (ctx.type == "}") ctx = popContext(state); 138 | while (ctx.type == "statement") ctx = popContext(state); 139 | } 140 | else if (curPunc == ctx.type) popContext(state); 141 | else if (((ctx.type == "}" || ctx.type == "top") && curPunc != ';') || (ctx.type == "statement" && curPunc == "newstatement")) 142 | pushContext(state, stream.column(), "statement"); 143 | state.startOfLine = false; 144 | return style; 145 | }, 146 | 147 | indent: function(state, textAfter) { 148 | if (state.tokenize != tokenBase && state.tokenize != null) return CodeMirror.Pass; 149 | var ctx = state.context, firstChar = textAfter && textAfter.charAt(0); 150 | if (ctx.type == "statement" && firstChar == "}") ctx = ctx.prev; 151 | var closing = firstChar == ctx.type; 152 | if (ctx.type == "statement") return ctx.indented + (firstChar == "{" ? 0 : statementIndentUnit); 153 | else if (dontAlignCalls && ctx.type == ")" && !closing) return ctx.indented + statementIndentUnit; 154 | else if (ctx.align) return ctx.column + (closing ? 0 : 1); 155 | else return ctx.indented + (closing ? 0 : indentUnit); 156 | }, 157 | 158 | electricChars: "{}" 159 | }; 160 | }); 161 | 162 | (function() { 163 | function words(str) { 164 | var obj = {}, words = str.split(" "); 165 | for (var i = 0; i < words.length; ++i) obj[words[i]] = true; 166 | return obj; 167 | } 168 | var cKeywords = "auto if break int case long char register continue return default short do sizeof " + 169 | "double static else struct entry switch extern typedef float union for unsigned " + 170 | "goto while enum void const signed volatile"; 171 | 172 | function cppHook(stream, state) { 173 | if (!state.startOfLine) return false; 174 | for (;;) { 175 | if (stream.skipTo("\\")) { 176 | stream.next(); 177 | if (stream.eol()) { 178 | state.tokenize = cppHook; 179 | break; 180 | } 181 | } else { 182 | stream.skipToEnd(); 183 | state.tokenize = null; 184 | break; 185 | } 186 | } 187 | return "meta"; 188 | } 189 | 190 | // C#-style strings where "" escapes a quote. 191 | function tokenAtString(stream, state) { 192 | var next; 193 | while ((next = stream.next()) != null) { 194 | if (next == '"' && !stream.eat('"')) { 195 | state.tokenize = null; 196 | break; 197 | } 198 | } 199 | return "string"; 200 | } 201 | 202 | function mimes(ms, mode) { 203 | for (var i = 0; i < ms.length; ++i) CodeMirror.defineMIME(ms[i], mode); 204 | } 205 | 206 | mimes(["text/x-csrc", "text/x-c", "text/x-chdr"], { 207 | name: "clike", 208 | keywords: words(cKeywords), 209 | blockKeywords: words("case do else for if switch while struct"), 210 | atoms: words("null"), 211 | hooks: {"#": cppHook} 212 | }); 213 | mimes(["text/x-c++src", "text/x-c++hdr"], { 214 | name: "clike", 215 | keywords: words(cKeywords + " asm dynamic_cast namespace reinterpret_cast try bool explicit new " + 216 | "static_cast typeid catch operator template typename class friend private " + 217 | "this using const_cast inline public throw virtual delete mutable protected " + 218 | "wchar_t"), 219 | blockKeywords: words("catch class do else finally for if struct switch try while"), 220 | atoms: words("true false null"), 221 | hooks: {"#": cppHook} 222 | }); 223 | CodeMirror.defineMIME("text/x-java", { 224 | name: "clike", 225 | keywords: words("abstract assert boolean break byte case catch char class const continue default " + 226 | "do double else enum extends final finally float for goto if implements import " + 227 | "instanceof int interface long native new package private protected public " + 228 | "return short static strictfp super switch synchronized this throw throws transient " + 229 | "try void volatile while"), 230 | blockKeywords: words("catch class do else finally for if switch try while"), 231 | atoms: words("true false null"), 232 | hooks: { 233 | "@": function(stream) { 234 | stream.eatWhile(/[\w\$_]/); 235 | return "meta"; 236 | } 237 | } 238 | }); 239 | CodeMirror.defineMIME("text/x-csharp", { 240 | name: "clike", 241 | keywords: words("abstract as base break case catch checked class const continue" + 242 | " default delegate do else enum event explicit extern finally fixed for" + 243 | " foreach goto if implicit in interface internal is lock namespace new" + 244 | " operator out override params private protected public readonly ref return sealed" + 245 | " sizeof stackalloc static struct switch this throw try typeof unchecked" + 246 | " unsafe using virtual void volatile while add alias ascending descending dynamic from get" + 247 | " global group into join let orderby partial remove select set value var yield"), 248 | blockKeywords: words("catch class do else finally for foreach if struct switch try while"), 249 | builtin: words("Boolean Byte Char DateTime DateTimeOffset Decimal Double" + 250 | " Guid Int16 Int32 Int64 Object SByte Single String TimeSpan UInt16 UInt32" + 251 | " UInt64 bool byte char decimal double short int long object" + 252 | " sbyte float string ushort uint ulong"), 253 | atoms: words("true false null"), 254 | hooks: { 255 | "@": function(stream, state) { 256 | if (stream.eat('"')) { 257 | state.tokenize = tokenAtString; 258 | return tokenAtString(stream, state); 259 | } 260 | stream.eatWhile(/[\w\$_]/); 261 | return "meta"; 262 | } 263 | } 264 | }); 265 | CodeMirror.defineMIME("text/x-scala", { 266 | name: "clike", 267 | keywords: words( 268 | 269 | /* scala */ 270 | "abstract case catch class def do else extends false final finally for forSome if " + 271 | "implicit import lazy match new null object override package private protected return " + 272 | "sealed super this throw trait try trye type val var while with yield _ : = => <- <: " + 273 | "<% >: # @ " + 274 | 275 | /* package scala */ 276 | "assert assume require print println printf readLine readBoolean readByte readShort " + 277 | "readChar readInt readLong readFloat readDouble " + 278 | 279 | "AnyVal App Application Array BufferedIterator BigDecimal BigInt Char Console Either " + 280 | "Enumeration Equiv Error Exception Fractional Function IndexedSeq Integral Iterable " + 281 | "Iterator List Map Numeric Nil NotNull Option Ordered Ordering PartialFunction PartialOrdering " + 282 | "Product Proxy Range Responder Seq Serializable Set Specializable Stream StringBuilder " + 283 | "StringContext Symbol Throwable Traversable TraversableOnce Tuple Unit Vector :: #:: " + 284 | 285 | /* package java.lang */ 286 | "Boolean Byte Character CharSequence Class ClassLoader Cloneable Comparable " + 287 | "Compiler Double Exception Float Integer Long Math Number Object Package Pair Process " + 288 | "Runtime Runnable SecurityManager Short StackTraceElement StrictMath String " + 289 | "StringBuffer System Thread ThreadGroup ThreadLocal Throwable Triple Void" 290 | 291 | 292 | ), 293 | blockKeywords: words("catch class do else finally for forSome if match switch try while"), 294 | atoms: words("true false null"), 295 | hooks: { 296 | "@": function(stream) { 297 | stream.eatWhile(/[\w\$_]/); 298 | return "meta"; 299 | } 300 | } 301 | }); 302 | }()); 303 | -------------------------------------------------------------------------------- /webapp/js/go.js: -------------------------------------------------------------------------------- 1 | CodeMirror.defineMode("go", function(config) { 2 | var indentUnit = config.indentUnit; 3 | 4 | var keywords = { 5 | "break":true, "case":true, "chan":true, "const":true, "continue":true, 6 | "default":true, "defer":true, "else":true, "fallthrough":true, "for":true, 7 | "func":true, "go":true, "goto":true, "if":true, "import":true, 8 | "interface":true, "map":true, "package":true, "range":true, "return":true, 9 | "select":true, "struct":true, "switch":true, "type":true, "var":true, 10 | "bool":true, "byte":true, "complex64":true, "complex128":true, 11 | "float32":true, "float64":true, "int8":true, "int16":true, "int32":true, 12 | "int64":true, "string":true, "uint8":true, "uint16":true, "uint32":true, 13 | "uint64":true, "int":true, "uint":true, "uintptr":true 14 | }; 15 | 16 | var atoms = { 17 | "true":true, "false":true, "iota":true, "nil":true, "append":true, 18 | "cap":true, "close":true, "complex":true, "copy":true, "imag":true, 19 | "len":true, "make":true, "new":true, "panic":true, "print":true, 20 | "println":true, "real":true, "recover":true 21 | }; 22 | 23 | var isOperatorChar = /[+\-*&^%:=<>!|\/]/; 24 | 25 | var curPunc; 26 | 27 | function tokenBase(stream, state) { 28 | var ch = stream.next(); 29 | if (ch == '"' || ch == "'" || ch == "`") { 30 | state.tokenize = tokenString(ch); 31 | return state.tokenize(stream, state); 32 | } 33 | if (/[\d\.]/.test(ch)) { 34 | if (ch == ".") { 35 | stream.match(/^[0-9]+([eE][\-+]?[0-9]+)?/); 36 | } else if (ch == "0") { 37 | stream.match(/^[xX][0-9a-fA-F]+/) || stream.match(/^0[0-7]+/); 38 | } else { 39 | stream.match(/^[0-9]*\.?[0-9]*([eE][\-+]?[0-9]+)?/); 40 | } 41 | return "number"; 42 | } 43 | if (/[\[\]{}\(\),;\:\.]/.test(ch)) { 44 | curPunc = ch; 45 | return null; 46 | } 47 | if (ch == "/") { 48 | if (stream.eat("*")) { 49 | state.tokenize = tokenComment; 50 | return tokenComment(stream, state); 51 | } 52 | if (stream.eat("/")) { 53 | stream.skipToEnd(); 54 | return "comment"; 55 | } 56 | } 57 | if (isOperatorChar.test(ch)) { 58 | stream.eatWhile(isOperatorChar); 59 | return "operator"; 60 | } 61 | stream.eatWhile(/[\w\$_]/); 62 | var cur = stream.current(); 63 | if (keywords.propertyIsEnumerable(cur)) { 64 | if (cur == "case" || cur == "default") curPunc = "case"; 65 | return "keyword"; 66 | } 67 | if (atoms.propertyIsEnumerable(cur)) return "atom"; 68 | return "variable"; 69 | } 70 | 71 | function tokenString(quote) { 72 | return function(stream, state) { 73 | var escaped = false, next, end = false; 74 | while ((next = stream.next()) != null) { 75 | if (next == quote && !escaped) {end = true; break;} 76 | escaped = !escaped && next == "\\"; 77 | } 78 | if (end || !(escaped || quote == "`")) 79 | state.tokenize = tokenBase; 80 | return "string"; 81 | }; 82 | } 83 | 84 | function tokenComment(stream, state) { 85 | var maybeEnd = false, ch; 86 | while (ch = stream.next()) { 87 | if (ch == "/" && maybeEnd) { 88 | state.tokenize = tokenBase; 89 | break; 90 | } 91 | maybeEnd = (ch == "*"); 92 | } 93 | return "comment"; 94 | } 95 | 96 | function Context(indented, column, type, align, prev) { 97 | this.indented = indented; 98 | this.column = column; 99 | this.type = type; 100 | this.align = align; 101 | this.prev = prev; 102 | } 103 | function pushContext(state, col, type) { 104 | return state.context = new Context(state.indented, col, type, null, state.context); 105 | } 106 | function popContext(state) { 107 | var t = state.context.type; 108 | if (t == ")" || t == "]" || t == "}") 109 | state.indented = state.context.indented; 110 | return state.context = state.context.prev; 111 | } 112 | 113 | // Interface 114 | 115 | return { 116 | startState: function(basecolumn) { 117 | return { 118 | tokenize: null, 119 | context: new Context((basecolumn || 0) - indentUnit, 0, "top", false), 120 | indented: 0, 121 | startOfLine: true 122 | }; 123 | }, 124 | 125 | token: function(stream, state) { 126 | var ctx = state.context; 127 | if (stream.sol()) { 128 | if (ctx.align == null) ctx.align = false; 129 | state.indented = stream.indentation(); 130 | state.startOfLine = true; 131 | if (ctx.type == "case") ctx.type = "}"; 132 | } 133 | if (stream.eatSpace()) return null; 134 | curPunc = null; 135 | var style = (state.tokenize || tokenBase)(stream, state); 136 | if (style == "comment") return style; 137 | if (ctx.align == null) ctx.align = true; 138 | 139 | if (curPunc == "{") pushContext(state, stream.column(), "}"); 140 | else if (curPunc == "[") pushContext(state, stream.column(), "]"); 141 | else if (curPunc == "(") pushContext(state, stream.column(), ")"); 142 | else if (curPunc == "case") ctx.type = "case"; 143 | else if (curPunc == "}" && ctx.type == "}") ctx = popContext(state); 144 | else if (curPunc == ctx.type) popContext(state); 145 | state.startOfLine = false; 146 | return style; 147 | }, 148 | 149 | indent: function(state, textAfter) { 150 | if (state.tokenize != tokenBase && state.tokenize != null) return 0; 151 | var ctx = state.context, firstChar = textAfter && textAfter.charAt(0); 152 | if (ctx.type == "case" && /^(?:case|default)\b/.test(textAfter)) { 153 | state.context.type = "}"; 154 | return ctx.indented; 155 | } 156 | var closing = firstChar == ctx.type; 157 | if (ctx.align) return ctx.column + (closing ? 0 : 1); 158 | else return ctx.indented + (closing ? 0 : indentUnit); 159 | }, 160 | 161 | electricChars: "{}:" 162 | }; 163 | }); 164 | 165 | CodeMirror.defineMIME("text/x-go", "go"); 166 | -------------------------------------------------------------------------------- /webapp/js/matchbrackets.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | var ie_lt8 = /MSIE \d/.test(navigator.userAgent) && 3 | (document.documentMode == null || document.documentMode < 8); 4 | 5 | var Pos = CodeMirror.Pos; 6 | // Disable brace matching in long lines, since it'll cause hugely slow updates 7 | var maxLineLen = 1000; 8 | 9 | var matching = {"(": ")>", ")": "(<", "[": "]>", "]": "[<", "{": "}>", "}": "{<"}; 10 | function findMatchingBracket(cm) { 11 | var cur = cm.getCursor(), line = cm.getLineHandle(cur.line), pos = cur.ch - 1; 12 | var match = (pos >= 0 && matching[line.text.charAt(pos)]) || matching[line.text.charAt(++pos)]; 13 | if (!match) return null; 14 | var forward = match.charAt(1) == ">", d = forward ? 1 : -1; 15 | var style = cm.getTokenAt(Pos(cur.line, pos + 1)).type; 16 | 17 | var stack = [line.text.charAt(pos)], re = /[(){}[\]]/; 18 | function scan(line, lineNo, start) { 19 | if (!line.text) return; 20 | var pos = forward ? 0 : line.text.length - 1, end = forward ? line.text.length : -1; 21 | if (start != null) pos = start + d; 22 | for (; pos != end; pos += d) { 23 | var ch = line.text.charAt(pos); 24 | if (re.test(ch) && cm.getTokenAt(Pos(lineNo, pos + 1)).type == style) { 25 | var match = matching[ch]; 26 | if (match.charAt(1) == ">" == forward) stack.push(ch); 27 | else if (stack.pop() != match.charAt(0)) return {pos: pos, match: false}; 28 | else if (!stack.length) return {pos: pos, match: true}; 29 | } 30 | } 31 | } 32 | for (var i = cur.line, found, e = forward ? Math.min(i + 100, cm.lineCount()) : Math.max(-1, i - 100); i != e; i+=d) { 33 | if (i == cur.line) found = scan(line, i, pos); 34 | else found = scan(cm.getLineHandle(i), i); 35 | if (found) break; 36 | } 37 | return {from: Pos(cur.line, pos), to: found && Pos(i, found.pos), match: found && found.match}; 38 | } 39 | 40 | function matchBrackets(cm, autoclear) { 41 | var found = findMatchingBracket(cm); 42 | if (!found || cm.getLine(found.from.line).length > maxLineLen || 43 | found.to && cm.getLine(found.to.line).length > maxLineLen) 44 | return; 45 | 46 | var style = found.match ? "CodeMirror-matchingbracket" : "CodeMirror-nonmatchingbracket"; 47 | var one = cm.markText(found.from, Pos(found.from.line, found.from.ch + 1), {className: style}); 48 | var two = found.to && cm.markText(found.to, Pos(found.to.line, found.to.ch + 1), {className: style}); 49 | // Kludge to work around the IE bug from issue #1193, where text 50 | // input stops going to the textare whever this fires. 51 | if (ie_lt8 && cm.state.focused) cm.display.input.focus(); 52 | var clear = function() { 53 | cm.operation(function() { one.clear(); two && two.clear(); }); 54 | }; 55 | if (autoclear) setTimeout(clear, 800); 56 | else return clear; 57 | } 58 | 59 | var currentlyHighlighted = null; 60 | function doMatchBrackets(cm) { 61 | cm.operation(function() { 62 | if (currentlyHighlighted) {currentlyHighlighted(); currentlyHighlighted = null;} 63 | if (!cm.somethingSelected()) currentlyHighlighted = matchBrackets(cm, false); 64 | }); 65 | } 66 | 67 | CodeMirror.defineOption("matchBrackets", false, function(cm, val) { 68 | if (val) cm.on("cursorActivity", doMatchBrackets); 69 | else cm.off("cursorActivity", doMatchBrackets); 70 | }); 71 | 72 | CodeMirror.defineExtension("matchBrackets", function() {matchBrackets(this, true);}); 73 | CodeMirror.defineExtension("findMatchingBracket", function(){return findMatchingBracket(this);}); 74 | })(); 75 | -------------------------------------------------------------------------------- /webapp/js/scala-tour.js: -------------------------------------------------------------------------------- 1 | function loading(outputDiv) { 2 | // $(".output").html('
    '+('waiting')+'
    '); 3 | outputDiv.html('
    ' + ('waiting') + '
    '); 4 | } 5 | 6 | function runFunc(codeStr, outputDiv) { 7 | $.ajax("/run?code=" + encodeURIComponent(codeStr), { 8 | type: "GET", 9 | dataType: "json", 10 | success: function(data) { 11 | if (!data) { 12 | return; 13 | } 14 | if (data.Errors && data.Errors.length > 0) { 15 | setOutput(outputDiv, null, null, data.Errors); 16 | return; 17 | } 18 | setOutput(outputDiv, data.Events, data.ErrEvents, false); 19 | }, 20 | error: function() { 21 | outputDiv.addClass("error").text( 22 | "Error communicating with remote server."); 23 | } 24 | }); 25 | } 26 | 27 | function setOutput(output, events, errevents, error) { 28 | output.empty(); 29 | if (events) { 30 | 31 | var msg = "" 32 | for (var i = 0; i < events.length; i++) { 33 | msg += events[i] + "\n" 34 | } 35 | output.text(msg); 36 | 37 | msg = "" 38 | for (var i = 0; i < errevents.length; i++) { 39 | msg += errevents[i] + "\n" 40 | } 41 | 42 | if (msg != "") { 43 | var err = $(''); 44 | err.text(msg); 45 | err.appendTo(output); 46 | } 47 | 48 | var exit = $(''); 49 | exit.text("\nProgram exited."); 50 | exit.appendTo(output); 51 | } 52 | // Display errors. 53 | if (error) { 54 | var errorText = "" 55 | for (var i = 0; i < error.length; i++) { 56 | errorText += error[i] + "\n" 57 | } 58 | output.addClass("error").text(errorText); 59 | } 60 | } 61 | 62 | var editors = [] 63 | var editorsMap = {} 64 | 65 | $(".run").click(function() { 66 | var editorText = $(this).parent().parent().parent().find(".editor"); 67 | var outputDiv = $(this).parent().parent().parent().find(".output"); 68 | loading(outputDiv); 69 | var editor = editorsMap[getElementPath(editorText)] 70 | runFunc(editor.getValue(), outputDiv.find("pre")); 71 | }); 72 | 73 | function getElementPath(element) { 74 | return "//" + $(element).parents().andSelf().map(function() { 75 | var $this = $(this); 76 | var tagName = this.nodeName; 77 | if ($this.siblings(tagName).length > 0) { 78 | tagName += "[" + $this.prevAll(tagName).length + "]"; 79 | } 80 | return tagName; 81 | }).get().join("/").toUpperCase(); 82 | } 83 | 84 | 85 | 86 | document.addEventListener("impress:add-active", function(event) { 87 | //restore 88 | var editorsRemove=[] 89 | 90 | for(var i = 0; i < editors.length; i++){ 91 | var editor=editors[i] 92 | editorsRemove.push(editor) 93 | } 94 | editors = [] 95 | editorsMap = {} 96 | //add 97 | var textareas = $(".active .editor").get() 98 | for (var i = 0; i < textareas.length; i++) { 99 | var textarea = textareas[i] 100 | var editor = CodeMirror.fromTextArea(textarea, { 101 | theme: "solarized light", 102 | matchBrackets: true, 103 | indentUnit: 2, 104 | tabSize: 2, 105 | indentWithTabs: false, 106 | mode: "text/x-scala", 107 | smartIndent :false, 108 | lineNumbers: false 109 | }); 110 | editors.push(editor) 111 | editorsMap[getElementPath(textarea)]=editor 112 | } 113 | 114 | for(var i = 0; i < editorsRemove.length; i++){ 115 | var editor=editorsRemove[i] 116 | window.setTimeout(function() { 117 | editor.save() 118 | editor.toTextArea() 119 | }, 1000 /* but after 2000 ms */); 120 | } 121 | 122 | 123 | }, false); 124 | 125 | 126 | document.getElementById("body").className = document.getElementById("body").className.replace('onload', ' '); 127 | impress().init(); 128 | 129 | 130 | 131 | // var textareas = document.getElementsByClassName("editor") 132 | 133 | 134 | // $(window).load(function() { 135 | 136 | 137 | 138 | 139 | 140 | // }); --------------------------------------------------------------------------------