diff --git a/app/views/coordinate/home.scala.html b/app/views/coordinate/home.scala.html index 7f0842b6b3..86eb034ad6 100644 --- a/app/views/coordinate/home.scala.html +++ b/app/views/coordinate/home.scala.html @@ -30,7 +30,7 @@ active = siteMenu.puzzle.some) {

@trans.training()

Puzzle - Coordinate + Coord
@if(ctx.isAuth) { @scoreOption.map { score => diff --git a/app/views/opening/JsData.scala b/app/views/opening/JsData.scala index 0e373ad77d..46efe01f82 100644 --- a/app/views/opening/JsData.scala +++ b/app/views/opening/JsData.scala @@ -19,13 +19,19 @@ object JsData extends lila.Steroids { "attempts" -> opening.attempts, "fen" -> opening.fen, "color" -> opening.color.name, - "moves" -> JsArray(opening.scoredMoves.map { - case ScoredMove(move, score) => Json.obj( + "moves" -> JsArray(opening.qualityMoves.map { + case QualityMove(move, quality) => Json.obj( "first" -> move.first, "cp" -> move.cp, "line" -> move.line.mkString(" "), - "score" -> score.name) + "quality" -> quality.name) }), "url" -> s"$netBaseUrl${routes.Opening.show(opening.id)}" - )))) + ), + "user" -> userInfos.map { i => + Json.obj( + "score" -> i.score, + "history" -> i.history.nonEmpty.option(Json.toJson(i.chart)) + ) + }))) } diff --git a/app/views/opening/show.scala.html b/app/views/opening/show.scala.html index 9ced7ca44a..3b15552d98 100644 --- a/app/views/opening/show.scala.html +++ b/app/views/opening/show.scala.html @@ -9,7 +9,10 @@ document.querySelector('#lichess .round'), @JsData(opening, userInfos), openingRoutes.controllers, @Html(J.stringify(i18nJsObject( -trans.training +trans.training, +trans.toTrackYourProgress, +trans.signUp, +trans.trainingSignupExplanation ))) ); } diff --git a/modules/opening/src/main/Opening.scala b/modules/opening/src/main/Opening.scala index a1fe63ee10..52bab34740 100644 --- a/modules/opening/src/main/Opening.scala +++ b/modules/opening/src/main/Opening.scala @@ -17,29 +17,40 @@ case class Opening( attempts: Int, score: Double) { - def scoredMoves = moves.map { move => - ScoredMove(move, Score.Good) + def qualityMoves: List[QualityMove] = { + val bestCp = moves.foldLeft(Int.MaxValue) { + case (cp, move) => if (move.cp < cp) move.cp else cp + } + moves.map { move => + QualityMove(move, Quality(move.cp - bestCp)) + } } } -sealed trait Score { - def name = toString.toLowerCase +sealed abstract class Quality(val threshold: Int) { + val name = toString.toLowerCase } -object Score { - case object Great extends Score - case object Good extends Score - case object Dubious extends Score - case object Bad extends Score +object Quality { + case object Good extends Quality(30) + case object Dubious extends Quality(80) + case object Bad extends Quality(Int.MaxValue) + + def apply(cp: Int) = + if (cp < Good.threshold) Good + else if (cp < Dubious.threshold) Dubious + else Bad } -case class ScoredMove( +case class QualityMove( move: Move, - score: Score) + quality: Quality) object Opening { type ID = Int + val defaultScore = 50 + def make( fen: String, color: Color, diff --git a/modules/opening/src/main/UserInfos.scala b/modules/opening/src/main/UserInfos.scala index f9960233f4..16aef0c0f4 100644 --- a/modules/opening/src/main/UserInfos.scala +++ b/modules/opening/src/main/UserInfos.scala @@ -2,27 +2,33 @@ package lila.opening import reactivemongo.bson._ import reactivemongo.bson.Macros +import play.api.libs.json._ import lila.db.Types.Coll import lila.rating.Glicko import lila.user.User -case class UserInfos(user: User, history: List[Attempt]) +case class UserInfos(user: User, history: List[Attempt], chart: JsArray) { + + def score = if (history.isEmpty) 50f + else history.foldLeft(0)(_ + _.score) / history.size +} object UserInfos { private def historySize = 20 + private def chartSize = 12 import Attempt.attemptBSONHandler def apply(attemptColl: Coll) = new { def apply(user: User): Fu[UserInfos] = fetchAttempts(user.id) map { attempts => - new UserInfos(user, makeHistory(attempts)) + new UserInfos(user, makeHistory(attempts), makeChart(attempts)) } recover { case e: Exception => - play.api.Logger("Puzzle UserInfos").error(e.getMessage) - new UserInfos(user, Nil) + play.api.Logger("Opening UserInfos").error(e.getMessage) + new UserInfos(user, Nil, JsArray()) } def apply(user: Option[User]): Fu[Option[UserInfos]] = @@ -33,8 +39,14 @@ object UserInfos { Attempt.BSONFields.userId -> userId )).sort(BSONDocument( Attempt.BSONFields.date -> -1 - )).cursor[Attempt].collect[List](historySize) + )).cursor[Attempt].collect[List](math.max(historySize, chartSize)) } private def makeHistory(attempts: List[Attempt]) = attempts.take(historySize) + + private def makeChart(attempts: List[Attempt]) = JsArray { + val scores = attempts.take(chartSize).reverse map (_.score) + val filled = List.fill(chartSize - scores.size)(Glicko.default.intRating) ::: scores + filled map { JsNumber(_) } + } } diff --git a/public/stylesheets/opening.css b/public/stylesheets/opening.css index e69de29bb2..b9427e0155 100644 --- a/public/stylesheets/opening.css +++ b/public/stylesheets/opening.css @@ -0,0 +1,211 @@ +#opening .lulzbar { + position: relative; + display: block; + width: 472px; + height: 20px; + padding: 10px 20px; + border-bottom: 1px solid rgba(255, 255, 255, 0.25); + border-radius: 16px; + margin: 20px 0 0 0; + box-shadow: 0px 4px 4px -4px rgba(255, 255, 255, 0.4), 0px -3px 3px -3px rgba(255, 255, 255, 0.25), inset 0px 0px 12px 0px rgba(0, 0, 0, 0.5); +} +#opening .lulzbar:before { + position: absolute; + display: block; + content: ""; + width: 470px; + height: 18px; + top: 10px; + left: 20px; + border-radius: 20px; + background: #222; + box-shadow: inset 0px 0px 6px 0px rgba(0, 0, 0, 0.85); + border: 1px solid rgba(0, 0, 0, 0.8); +} +#opening .lulzbar .bar { + position: absolute; + display: block; + width: 0px; + height: 16px; + top: 12px; + left: 22px; + background: rgb(126, 234, 25); + background: -moz-linear-gradient(top, rgba(126, 234, 25, 1) 0%, rgba(83, 173, 0, 1) 100%); + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(126, 234, 25, 1)), color-stop(100%, rgba(83, 173, 0, 1))); + background: -webkit-linear-gradient(top, rgba(126, 234, 25, 1) 0%, rgba(83, 173, 0, 1) 100%); + background: -o-linear-gradient(top, rgba(126, 234, 25, 1) 0%, rgba(83, 173, 0, 1) 100%); + background: -ms-linear-gradient(top, rgba(126, 234, 25, 1) 0%, rgba(83, 173, 0, 1) 100%); + background: linear-gradient(to bottom, rgba(126, 234, 25, 1) 0%, rgba(83, 173, 0, 1) 100%); + border-radius: 16px; + box-shadow: 0px 0px 12px 0px rgba(126, 234, 25, 1), inset 0px 1px 0px 0px rgba(255, 255, 255, 0.45), inset 1px 0px 0px 0px rgba(255, 255, 255, 0.25), inset -1px 0px 0px 0px rgba(255, 255, 255, 0.25); + overflow: hidden; + transition: width 1s; +} +#opening .lulzbar .bar.yellow { + background: rgb(229, 195, 25); + background: -moz-linear-gradient(top, rgba(229, 195, 25, 1) 0%, rgba(168, 140, 0, 1) 100%); + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(229, 195, 25, 1)), color-stop(100%, rgba(168, 140, 0, 1))); + background: -webkit-linear-gradient(top, rgba(229, 195, 25, 1) 0%, rgba(168, 140, 0, 1) 100%); + background: -o-linear-gradient(top, rgba(229, 195, 25, 1) 0%, rgba(168, 140, 0, 1) 100%); + background: -ms-linear-gradient(top, rgba(229, 195, 25, 1) 0%, rgba(168, 140, 0, 1) 100%); + background: linear-gradient(to bottom, rgba(229, 195, 25, 1) 0%, rgba(168, 140, 0, 1) 100%); + box-shadow: 0px 0px 12px 0px rgba(229, 195, 25, 1), inset 0px 1px 0px 0px rgba(255, 255, 255, 0.45), inset 1px 0px 0px 0px rgba(255, 255, 255, 0.25), inset -1px 0px 0px 0px rgba(255, 255, 255, 0.25); +} +#opening .lulzbar .bar.red { + background: rgb(232, 25, 87); + background: -moz-linear-gradient(top, rgba(232, 25, 87, 1) 0%, rgba(170, 0, 51, 1) 100%); + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(232, 25, 87, 1)), color-stop(100%, rgba(170, 0, 51, 1))); + background: -webkit-linear-gradient(top, rgba(232, 25, 87, 1) 0%, rgba(170, 0, 51, 1) 100%); + background: -o-linear-gradient(top, rgba(232, 25, 87, 1) 0%, rgba(170, 0, 51, 1) 100%); + background: -ms-linear-gradient(top, rgba(232, 25, 87, 1) 0%, rgba(170, 0, 51, 1) 100%); + background: linear-gradient(to bottom, rgba(232, 25, 87, 1) 0%, rgba(170, 0, 51, 1) 100%); + box-shadow: 0px 0px 12px 0px rgba(232, 25, 87, 1), inset 0px 1px 0px 0px rgba(255, 255, 255, 0.45), inset 1px 0px 0px 0px rgba(255, 255, 255, 0.25), inset -1px 0px 0px 0px rgba(255, 255, 255, 0.25); +} +#opening .lulzbar .bar.blue { + background: rgb(24, 109, 226); + background: -moz-linear-gradient(top, rgba(24, 109, 226, 1) 0%, rgba(0, 69, 165, 1) 100%); + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(24, 109, 226, 1)), color-stop(100%, rgba(0, 69, 165, 1))); + background: -webkit-linear-gradient(top, rgba(24, 109, 226, 1) 0%, rgba(0, 69, 165, 1) 100%); + background: -o-linear-gradient(top, rgba(24, 109, 226, 1) 0%, rgba(0, 69, 165, 1) 100%); + background: -ms-linear-gradient(top, rgba(24, 109, 226, 1) 0%, rgba(0, 69, 165, 1) 100%); + background: linear-gradient(to bottom, rgba(24, 109, 226, 1) 0%, rgba(0, 69, 165, 1) 100%); + box-shadow: 0px 0px 12px 0px rgba(24, 109, 226, 1), inset 0px 1px 0px 0px rgba(255, 255, 255, 0.45), inset 1px 0px 0px 0px rgba(255, 255, 255, 0.25), inset -1px 0px 0px 0px rgba(255, 255, 255, 0.25); +} +#opening .lulzbar .bar:before { + position: absolute; + display: block; + content: ""; + width: 606px; + height: 150%; + top: -25%; + left: -25px; + background: -moz-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.01) 50%, rgba(255, 255, 255, 0) 51%, rgba(255, 255, 255, 0) 100%); + background: -webkit-gradient(radial, center center, 0px, center center, 100%, color-stop(0%, rgba(255, 255, 255, 0.35)), color-stop(50%, rgba(255, 255, 255, 0.01)), color-stop(51%, rgba(255, 255, 255, 0)), color-stop(100%, rgba(255, 255, 255, 0))); + background: -webkit-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.01) 50%, rgba(255, 255, 255, 0) 51%, rgba(255, 255, 255, 0) 100%); + background: -o-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.01) 50%, rgba(255, 255, 255, 0) 51%, rgba(255, 255, 255, 0) 100%); + background: -ms-radial-gradient(center, ellipse cover, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.01) 50%, rgba(255, 255, 255, 0) 51%, rgba(255, 255, 255, 0) 100%); + background: radial-gradient(ellipse at center, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.01) 50%, rgba(255, 255, 255, 0) 51%, rgba(255, 255, 255, 0) 100%); +} +#opening .lulzbar .bar:after { + position: absolute; + display: block; + content: ""; + width: 64px; + height: 16px; + right: 0; + top: 0; + border-radius: 0px 16px 16px 0px; + background: -moz-linear-gradient(left, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.6) 98%, rgba(255, 255, 255, 0) 100%); + background: -webkit-gradient(linear, left top, right top, color-stop(0%, rgba(255, 255, 255, 0)), color-stop(98%, rgba(255, 255, 255, 0.6)), color-stop(100%, rgba(255, 255, 255, 0))); + background: -webkit-linear-gradient(left, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.6) 98%, rgba(255, 255, 255, 0) 100%); + background: -o-linear-gradient(left, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.6) 98%, rgba(255, 255, 255, 0) 100%); + background: -ms-linear-gradient(left, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.6) 98%, rgba(255, 255, 255, 0) 100%); + background: linear-gradient(to right, rgba(255, 255, 255, 0) 0%, rgba(255, 255, 255, 0.6) 98%, rgba(255, 255, 255, 0) 100%); +} +#opening .lulzbar .bar span { + position: absolute; + display: block; + width: 100%; + height: 64px; + border-radius: 16px; + top: 0; + left: 0; + background: url("") 0 0; + -webkit-animation: sparkle 1500ms linear infinite; + -moz-animation: sparkle 1500ms linear infinite; + -o-animation: sparkle 1500ms linear infinite; + animation: sparkle 1500ms linear infinite; + opacity: 0.2; +} +#opening .lulzbar .label { + font-family: monospace; + position: absolute; + display: block; + width: 40px; + height: 30px; + line-height: 30px; + top: 38px; + left: 0px; + background: rgb(76, 76, 76); + background: -moz-linear-gradient(top, rgba(76, 76, 76, 1) 0%, rgba(38, 38, 38, 1) 100%); + background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(76, 76, 76, 1)), color-stop(100%, rgba(38, 38, 38, 1))); + background: -webkit-linear-gradient(top, rgba(76, 76, 76, 1) 0%, rgba(38, 38, 38, 1) 100%); + background: -o-linear-gradient(top, rgba(76, 76, 76, 1) 0%, rgba(38, 38, 38, 1) 100%); + background: -ms-linear-gradient(top, rgba(76, 76, 76, 1) 0%, rgba(38, 38, 38, 1) 100%); + background: linear-gradient(to bottom, rgba(76, 76, 76, 1) 0%, rgba(38, 38, 38, 1) 100%); + font-weight: bold; + font-size: 14px; + color: #fff; + text-align: center; + border-radius: 6px; + border: 1px solid rgba(0, 0, 0, 0.2); + box-shadow: 0px 1px 4px 0px rgba(0, 0, 0, 0.3); + text-shadow: 0px -1px 0px #000000, 0px 1px 1px #000000; + filter: dropshadow(color=#000000, offx=0, offy=-1); + transition: left 1s; +} +#opening .lulzbar .label span { + position: absolute; + display: block; + width: 12px; + height: 9px; + top: -9px; + left: 14px; + background: transparent; + overflow: hidden; +} +#opening .lulzbar .label span:before { + position: absolute; + display: block; + content: ""; + width: 8px; + height: 8px; + top: 4px; + left: 2px; + border: 1px solid rgba(0, 0, 0, 0.5); + background: rgb(86, 86, 86); + background: -moz-linear-gradient(-45deg, rgba(86, 86, 86, 1) 0%, rgba(76, 76, 76, 1) 50%); + background: -webkit-gradient(linear, left top, right bottom, color-stop(0%, rgba(86, 86, 86, 1)), color-stop(50%, rgba(76, 76, 76, 1))); + background: -webkit-linear-gradient(-45deg, rgba(86, 86, 86, 1) 0%, rgba(76, 76, 76, 1) 50%); + background: -o-linear-gradient(-45deg, rgba(86, 86, 86, 1) 0%, rgba(76, 76, 76, 1) 50%); + background: -ms-linear-gradient(-45deg, rgba(86, 86, 86, 1) 0%, rgba(76, 76, 76, 1) 50%); + background: linear-gradient(135deg, rgba(86, 86, 86, 1) 0%, rgba(76, 76, 76, 1) 50%); + box-shadow: 0px -1px 2px 0px rgba(0, 0, 0, 0.15); + -moz-transform: rotate(45deg); + -webkit-transform: rotate(45deg); + -o-transform: rotate(45deg); + -ms-transform: rotate(45deg); + transform: rotate(45deg); +} +@-webkit-keyframes sparkle { + from { + background-position: 0 0; + } + to { + background-position: 0 -64px; + } +} +@-moz-keyframes sparkle { + from { + background-position: 0 0; + } + to { + background-position: 0 -64px; + } +} +@-o-keyframes sparkle { + from { + background-position: 0 0; + } + to { + background-position: 0 -64px; + } +} +@keyframes sparkle { + from { + background-position: 0 0; + } + to { + background-position: 0 -64px; + } +} diff --git a/ui/opening/package.json b/ui/opening/package.json index 4d7feb93fd..496f7d1a8b 100644 --- a/ui/opening/package.json +++ b/ui/opening/package.json @@ -30,9 +30,10 @@ "watchify": "^1.0.2" }, "dependencies": { - "chessground": "1.8.0", + "chessground": "1.8.7", + "chessli.js": "file:../chessli", "lodash-node": "^2.4.1", "merge": "^1.2.0", - "mithril": "0.1.24" + "mithril": "0.1.28" } } diff --git a/ui/opening/src/ctrl.js b/ui/opening/src/ctrl.js index 0e71be001b..f7b7a0d6d0 100644 --- a/ui/opening/src/ctrl.js +++ b/ui/opening/src/ctrl.js @@ -1,23 +1,95 @@ var m = require('mithril'); var chessground = require('chessground'); +var Chess = require('chessli.js').Chess; module.exports = function(cfg, router, i18n) { this.data = cfg; + console.log(this.data); + + this.vm = { + nbGood: this.data.opening.moves.filter(function(m) { + return m.quality === 'good'; + }).length, + figuredOut: [], + messedUp: [], + flash: null + }; + + var chess = new Chess(this.data.opening.fen); + var init = { + dests: chess.dests(), + check: chess.in_check() + }; + + var onMove = function(orig, dest, meta) { + submitMove(orig + dest); + setTimeout(function() { + this.chessground.set({ + fen: this.data.opening.fen, + lastMove: null, + turnColor: this.data.opening.color, + check: init.check, + premovable: { + enabled: false + }, + movable: { + dests: init.dests + } + }); + }.bind(this), 1000); + m.redraw(); + }.bind(this); this.chessground = new chessground.controller({ fen: this.data.opening.fen, orientation: this.data.opening.color, - viewOnly: true, + turnColor: this.data.opening.color, + check: init.check, + movable: { + color: this.data.opening.color, + free: false, + dests: init.dests, + events: { + after: onMove + } + }, }); + var submitMove = function(move) { + var found = this.data.opening.moves.filter(function(m) { + return m.first === move; + })[0]; + if (found && found.quality === 'good') { + if (this.vm.figuredOut.indexOf(move) === -1) this.vm.figuredOut.push(move); + } else if (found && found.quality === 'dubious') { + flash('dubious'); + } else if (!found || found.quality === 'bad') { + if (this.vm.messedUp.indexOf(move) === -1) this.vm.messedUp.push(move); + flash('bad'); + } + }.bind(this); + + var flash = function(f) { + this.vm.flash = f; + setTimeout(function() { + this.vm.flash = null; + m.redraw(); + }.bind(this), 1000); + }.bind(this); + this.router = router; this.trans = function() { - var str = i18n[arguments[0]] + var str = i18n[arguments[0]] || untranslated[arguments[0]] || arguments[0]; Array.prototype.slice.call(arguments, 1).forEach(function(arg) { str = str.replace('%s', arg); }); return str; }; + + var untranslated = { + yourOpeningScoreX: 'Your opening score: %s', + findNbGoodMoves: 'Find %s good moves', + }; }; diff --git a/ui/opening/src/main.js b/ui/opening/src/main.js index 5df733cefe..7c46982d3c 100644 --- a/ui/opening/src/main.js +++ b/ui/opening/src/main.js @@ -1,3 +1,4 @@ +var m = require('mithril'); var ctrl = require('./ctrl'); var view = require('./view'); @@ -7,5 +8,8 @@ module.exports = function(element, config, router, i18n) { controller: function () { return controller; }, view: view }); - controller.initiate(); }; + +// lol, that's for the rest of lichess to access mithril +// without having to include it a second time +window.Chessground = require('chessground'); diff --git a/ui/opening/src/view.js b/ui/opening/src/view.js index 3be4a6f78f..8d296b63a1 100644 --- a/ui/opening/src/view.js +++ b/ui/opening/src/view.js @@ -1,5 +1,9 @@ -var chessground = require('chessground'); var m = require('mithril'); +var chessground = require('chessground'); + +function strong(txt) { + return '' + txt + ''; +} function renderTable(ctrl) { return m('div.table', @@ -7,13 +11,101 @@ function renderTable(ctrl) { ); } +function renderUserInfos(ctrl) { + return m('div.chart_container', [ + m('p', m.trust(ctrl.trans('yourOpeningScoreX', strong(ctrl.data.user.score)))), + ctrl.data.user.history ? m('div.user_chart', { + config: function(el, isUpdate, context) { + var hash = ctrl.data.user.history.join(''); + if (hash == context.hash) return; + context.hash = hash; + var dark = document.body.classList.contains('dark'); + jQuery(el).sparkline(ctrl.data.user.history, { + type: 'line', + width: '213px', + height: '80px', + lineColor: dark ? '#4444ff' : '#0000ff', + fillColor: dark ? '#222255' : '#ccccff' + }); + } + }) : null + ]); +} + +function renderTrainingBox(ctrl) { + return m('div.box', [ + m('h1', ctrl.trans('training')), + m('div.tabs.buttonset', [ + m('a.button', { + href: '/training' + }, 'Puzzle'), + m('a.button', { + href: '/training/coordinate' + }, 'Coord'), + m('a.button.active', { + href: '/training/opening' + }, 'Opening'), + ]), + ctrl.data.user ? renderUserInfos(ctrl) : m('div.register', [ + m('p', ctrl.trans('toTrackYourProgress')), + m('p.signup', + m('a.button', { + href: '/signup', + }, ctrl.trans('signUp')) + ), + m('p', ctrl.trans('trainingSignupExplanation')) + ]) + ]); +} + +function renderResult(ctrl) { + return [ + m('div.goal', m.trust(ctrl.trans('findNbGoodMoves', strong(ctrl.vm.nbGood)))), + ]; +} + +function renderSide(ctrl) { + return m('div.side', [ + renderTrainingBox(ctrl), + renderResult(ctrl) + ]); +} + module.exports = function(ctrl) { + var percent = Math.ceil(ctrl.vm.figuredOut.length * 100 / ctrl.vm.nbGood) + '%'; + var color; + switch (ctrl.vm.flash) { + case 'dubious': + color = 'yellow'; + break; + case 'bad': + color = 'red'; + break; + default: + color = 'green'; + } return m('div#opening.training', [ + renderSide(ctrl), m('div.board_and_ground', [ m('div', chessground.view(ctrl.chessground)), m('div.right', renderTable(ctrl)) ]), m('div.center', [ + m('div.lulzbar', [ + m('div.bar.' + color, { + style: { + width: percent + } + }, m('span')), + m('div.label', { + style: { + left: percent + } + }, [ + m('span'), + m('div.perc', percent) + ]) + ]) ]) ]); }; diff --git a/ui/puzzle/src/view.js b/ui/puzzle/src/view.js index ad59a32f2f..9417591aed 100644 --- a/ui/puzzle/src/view.js +++ b/ui/puzzle/src/view.js @@ -40,7 +40,7 @@ function renderTrainingBox(ctrl) { }, 'Puzzle'), m('a.button', { href: '/training/coordinate' - }, 'Coordinate') + }, 'Coord') ]), ctrl.data.user ? renderUserInfos(ctrl) : m('div.register', [ m('p', ctrl.trans('toTrackYourProgress')),