From 07019577d5af8d83c7a29ea83fda51847fb045d5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Paul-Etienne=20Fran=C3=A7ois?= Date: Wed, 8 May 2024 14:02:11 +0000 Subject: [PATCH 001/168] Place the user actions in a dropdown window with labels --- app/views/user/show/header.scala | 129 +++++++++++++++---------------- ui/bits/css/user/_profile.scss | 5 ++ ui/bits/css/user/_show.scss | 52 +++++++++++++ 3 files changed, 121 insertions(+), 65 deletions(-) diff --git a/app/views/user/show/header.scala b/app/views/user/show/header.scala index 984b08a20861a..a798c8ff53e85 100644 --- a/app/views/user/show/header.scala +++ b/app/views/user/show/header.scala @@ -39,7 +39,70 @@ object header: )(patronIconChar) ) ), - u.enabled.no.option(span(cls := "closed")("CLOSED")) + u.enabled.no.option(span(cls := "closed")("CLOSED")), + div(cls := "user-actions dropdown")( + a( + cls := "text", + dataIcon := Icon.List + ), + div(cls := "dropdown-window")( + (ctx + .is(u)) + .option( + frag( + a( + cls := "text", + href := routes.Account.profile, + dataIcon := Icon.Gear + )(trans.site.editProfile.txt()), + a( + cls := "text", + href := routes.Relation.blocks(), + dataIcon := Icon.NotAllowed + )(trans.site.listBlockedPlayers.txt()) + ) + ), + isGranted(_.UserModView).option( + a( + cls := "text mod-zone-toggle", + href := routes.User.mod(u.username), + dataIcon := Icon.Agent + )("Mod zone (Hotkey: m)") + ), + a( + cls := "text", + href := routes.User.tv(u.username), + dataIcon := Icon.AnalogTv + )(trans.site.watchGames.txt()), + ctx + .isnt(u) + .option( + views.relation.actions( + u.light, + relation = social.relation, + followable = social.followable, + blocked = social.blocked + ) + ), + a( + cls := "text", + href := s"${routes.UserAnalysis.index}#explorer/${u.username}", + dataIcon := Icon.Book + )(trans.site.openingExplorer.txt()), + a( + cls := "text", + href := routes.User.download(u.username), + dataIcon := Icon.Download + )(trans.site.exportGames.txt()), + (ctx.isAuth && ctx.kid.no && ctx.isnt(u)).option( + a( + cls := "text", + href := s"${routes.Report.form}?username=${u.username}", + dataIcon := Icon.CautionTriangle + )(trans.site.reportXToModerators.txt(u.username)) + ) + ) + ) ), div(cls := "user-show__social")( div(cls := "number-menu")( @@ -89,70 +152,6 @@ object header: ), (ctx.isAuth && ctx.isnt(u)) .option(a(cls := "nm-item note-zone-toggle")(splitNumber(s"${social.notes.size} Notes"))) - ), - div(cls := "user-actions btn-rack")( - (ctx - .is(u)) - .option( - frag( - a( - cls := "btn-rack__btn", - href := routes.Account.profile, - titleOrText(trans.site.editProfile.txt()), - dataIcon := Icon.Gear - ), - a( - cls := "btn-rack__btn", - href := routes.Relation.blocks(), - titleOrText(trans.site.listBlockedPlayers.txt()), - dataIcon := Icon.NotAllowed - ) - ) - ), - isGranted(_.UserModView).option( - a( - cls := "btn-rack__btn mod-zone-toggle", - href := routes.User.mod(u.username), - titleOrText("Mod zone (Hotkey: m)"), - dataIcon := Icon.Agent - ) - ), - a( - cls := "btn-rack__btn", - href := routes.User.tv(u.username), - titleOrText(trans.site.watchGames.txt()), - dataIcon := Icon.AnalogTv - ), - ctx - .isnt(u) - .option( - views.relation.actions( - u.light, - relation = social.relation, - followable = social.followable, - blocked = social.blocked - ) - ), - a( - cls := "btn-rack__btn", - href := s"${routes.UserAnalysis.index}#explorer/${u.username}", - titleOrText(trans.site.openingExplorer.txt()), - dataIcon := Icon.Book - ), - a( - cls := "btn-rack__btn", - href := routes.User.download(u.username), - titleOrText(trans.site.exportGames.txt()), - dataIcon := Icon.Download - ), - (ctx.isAuth && ctx.kid.no && ctx.isnt(u)).option( - a( - titleOrText(trans.site.reportXToModerators.txt(u.username)), - cls := "btn-rack__btn", - href := s"${routes.Report.form}?username=${u.username}", - dataIcon := Icon.CautionTriangle - ) - ) ) ), ctx.isnt(u).option(noteUi.zone(u, social.notes)), diff --git a/ui/bits/css/user/_profile.scss b/ui/bits/css/user/_profile.scss index 8709a80865b21..f5367e1a75206 100644 --- a/ui/bits/css/user/_profile.scss +++ b/ui/bits/css/user/_profile.scss @@ -46,6 +46,11 @@ unicode-bidi: plaintext; text-align: start; + + time { + font-size: 100%; + opacity: 1; + } } .insight { diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 3bd273be9965c..3b15bba810ec6 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -45,6 +45,58 @@ } } + &__header { + .dropdown { + position: relative; + font-size: x-large; + > a { + color: $c-font-page; + float: right; + padding-left: 0.5em; + } + + .dropdown-window { + visibility: hidden; + background: $c-bg-header-dropdown; + border-radius: 3px 0 3px 3px; + box-shadow: 2px 5px 6px rgba(0, 0, 0, 0.3); + position: absolute; + top: 1.45em; + right: 0; + a { + width: 16em; + font-size: small; + display: block; + padding: 0.6rem 1rem; + color: $c-header-dropdown; + &:first-child { + border-radius: 3px 0 0 0; + } + &:last-child { + border-radius: 0 0 3px 3px; + } + &:hover { + background: $c-primary; + &, + &::after { + color: $c-over; + } + } + } + } + } + + .dropdown:hover { + > a { + background: $c-bg-header-dropdown; + box-shadow: 2px 5px 6px rgba(0, 0, 0, 0.3); + } + .dropdown-window { + visibility: visible; + } + } + } + .claim-title { @extend %box-padding; From 665576b8c937a73c6283173486ef819152f483fa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Paul-Etienne=20Fran=C3=A7ois?= Date: Wed, 15 May 2024 19:48:37 +0000 Subject: [PATCH 002/168] fix: Display relation actions correctly --- app/views/user/show/header.scala | 100 +++++++++--------- modules/relation/src/main/ui/RelationUi.scala | 43 +++----- ui/bits/css/user/_show.scss | 8 -- 3 files changed, 67 insertions(+), 84 deletions(-) diff --git a/app/views/user/show/header.scala b/app/views/user/show/header.scala index a798c8ff53e85..7b3883b78eb40 100644 --- a/app/views/user/show/header.scala +++ b/app/views/user/show/header.scala @@ -40,6 +40,56 @@ object header: ) ), u.enabled.no.option(span(cls := "closed")("CLOSED")), + ), + div(cls := "user-show__social")( + div(cls := "number-menu")( + u.noBot.option( + a( + href := routes.UserTournament.path(u.username, "recent"), + cls := "nm-item", + dataToints := u.toints + )( + splitNumber(trans.site.nbTournamentPoints.pluralSame(u.toints)) + ) + ), + (info.nbSimuls > 0).option( + a( + href := routes.Simul.byUser(u.username), + cls := "nm-item" + )( + splitNumber(trans.site.nbSimuls.pluralSame(info.nbSimuls)) + ) + ), + (info.nbRelays > 0).option( + a( + href := routes.RelayTour.by(u.username), + cls := "nm-item" + )( + splitNumber(trans.broadcast.nbBroadcasts.pluralSame(info.nbRelays)) + ) + ), + a(href := routes.Study.byOwnerDefault(u.username), cls := "nm-item")( + splitNumber(trans.site.`nbStudies`.pluralSame(info.nbStudies)) + ), + ctx.kid.no.option( + a( + cls := "nm-item", + href := routes.ForumPost.search("user:" + u.username, 1).url + )( + splitNumber(trans.site.nbForumPosts.pluralSame(info.nbForumPosts)) + ) + ), + (ctx.kid.no && (info.ublog.exists(_.nbPosts > 0) || ctx.is(u))).option( + a( + cls := "nm-item", + href := routes.Ublog.index(u.username) + )( + splitNumber(s"${info.ublog.so(_.nbPosts)} blog posts") + ) + ), + (ctx.isAuth && ctx.isnt(u)) + .option(a(cls := "nm-item note-zone-toggle")(splitNumber(s"${social.notes.size} Notes"))) + ), div(cls := "user-actions dropdown")( a( cls := "text", @@ -104,56 +154,6 @@ object header: ) ) ), - div(cls := "user-show__social")( - div(cls := "number-menu")( - u.noBot.option( - a( - href := routes.UserTournament.path(u.username, "recent"), - cls := "nm-item", - dataToints := u.toints - )( - splitNumber(trans.site.nbTournamentPoints.pluralSame(u.toints)) - ) - ), - (info.nbSimuls > 0).option( - a( - href := routes.Simul.byUser(u.username), - cls := "nm-item" - )( - splitNumber(trans.site.nbSimuls.pluralSame(info.nbSimuls)) - ) - ), - (info.nbRelays > 0).option( - a( - href := routes.RelayTour.by(u.username), - cls := "nm-item" - )( - splitNumber(trans.broadcast.nbBroadcasts.pluralSame(info.nbRelays)) - ) - ), - a(href := routes.Study.byOwnerDefault(u.username), cls := "nm-item")( - splitNumber(trans.site.`nbStudies`.pluralSame(info.nbStudies)) - ), - ctx.kid.no.option( - a( - cls := "nm-item", - href := routes.ForumPost.search("user:" + u.username, 1).url - )( - splitNumber(trans.site.nbForumPosts.pluralSame(info.nbForumPosts)) - ) - ), - (ctx.kid.no && (info.ublog.exists(_.nbPosts > 0) || ctx.is(u))).option( - a( - cls := "nm-item", - href := routes.Ublog.index(u.username) - )( - splitNumber(s"${info.ublog.so(_.nbPosts)} blog posts") - ) - ), - (ctx.isAuth && ctx.isnt(u)) - .option(a(cls := "nm-item note-zone-toggle")(splitNumber(s"${social.notes.size} Notes"))) - ) - ), ctx.isnt(u).option(noteUi.zone(u, social.notes)), isGranted(_.UserModView).option(div(cls := "mod-zone mod-zone-full none")), standardFlash, diff --git a/modules/relation/src/main/ui/RelationUi.scala b/modules/relation/src/main/ui/RelationUi.scala index b2a46d8f04d07..315c2cb951a67 100644 --- a/modules/relation/src/main/ui/RelationUi.scala +++ b/modules/relation/src/main/ui/RelationUi.scala @@ -51,14 +51,13 @@ final class RelationUi(helpers: Helpers): blocked: Boolean, signup: Boolean = false )(using ctx: Context) = - div(cls := "relation-actions btn-rack")( + div(cls := "relation-actions")( (ctx.isnt(user) && !blocked).option( a( - titleOrText(trans.challenge.challengeToPlay.txt()), + cls := "text", href := s"${routes.Lobby.home}?user=${user.name}#friend", - cls := "btn-rack__btn", dataIcon := Icon.Swords - ) + )(trans.challenge.challengeToPlay.txt()) ), ctx.userId .map: myId => @@ -67,54 +66,46 @@ final class RelationUi(helpers: Helpers): frag( (!blocked && !user.isBot).option( a( - titleOrText(trans.site.composeMessage.txt()), + cls := "text", href := routes.Msg.convo(user.name), - cls := "btn-rack__btn", dataIcon := Icon.BubbleSpeech - ) + )(trans.site.composeMessage.txt()) ), (!blocked && !user.isPatron).option( a( - titleOrText(trans.patron.giftPatronWingsShort.txt()), + cls := "text", href := s"${routes.Plan.list}?dest=gift&giftUsername=${user.name}", - cls := "btn-rack__btn", dataIcon := Icon.Wings - ) + )(trans.patron.giftPatronWingsShort.txt()) ), relation match case None => frag( (followable && !blocked).option( a( - cls := "btn-rack__btn relation-button", + cls := "text relation-button", href := routes.Relation.follow(user.name), - titleOrText(trans.site.follow.txt()), dataIcon := Icon.ThumbsUp - ) + )(trans.site.follow.txt()) ), a( - cls := "btn-rack__btn relation-button", + cls := "text relation-button", href := routes.Relation.block(user.name), - titleOrText(trans.site.block.txt()), dataIcon := Icon.NotAllowed - ) + )(trans.site.block.txt()) ) case Some(Relation.Follow) => a( - dataIcon := Icon.ThumbsUp, - cls := "btn-rack__btn relation-button text hover-text", + cls := "text relation-button", href := routes.Relation.unfollow(user.name), - titleOrText(trans.site.following.txt()), - dataHoverText := trans.site.unfollow.txt() - ) + dataIcon := Icon.ThumbsUp + )(trans.site.unfollow.txt()) case Some(Relation.Block) => a( - dataIcon := Icon.NotAllowed, - cls := "btn-rack__btn relation-button text hover-text", + cls := "text relation-button", href := routes.Relation.unblock(user.name), - titleOrText(trans.site.blocked.txt()), - dataHoverText := trans.site.unblock.txt() - ) + dataIcon := Icon.NotAllowed + )(trans.site.unblock.txt()) ) ) .getOrElse: diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 3b15bba810ec6..fa3fc0f71425e 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -38,14 +38,6 @@ } } - .relation-actions { - @extend %flex-center-nowrap; - - border-inline-end: $border; - } - } - - &__header { .dropdown { position: relative; font-size: x-large; From d5e7788d046df91f3ee1fe3bea26f5d1aa20316f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Paul-Etienne=20Fran=C3=A7ois?= Date: Thu, 16 May 2024 00:42:53 +0200 Subject: [PATCH 003/168] Format code --- app/views/user/show/header.scala | 2 +- modules/relation/src/main/ui/RelationUi.scala | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/app/views/user/show/header.scala b/app/views/user/show/header.scala index 7b3883b78eb40..4cc92a766941d 100644 --- a/app/views/user/show/header.scala +++ b/app/views/user/show/header.scala @@ -39,7 +39,7 @@ object header: )(patronIconChar) ) ), - u.enabled.no.option(span(cls := "closed")("CLOSED")), + u.enabled.no.option(span(cls := "closed")("CLOSED")) ), div(cls := "user-show__social")( div(cls := "number-menu")( diff --git a/modules/relation/src/main/ui/RelationUi.scala b/modules/relation/src/main/ui/RelationUi.scala index 315c2cb951a67..5d6bdea39c2b0 100644 --- a/modules/relation/src/main/ui/RelationUi.scala +++ b/modules/relation/src/main/ui/RelationUi.scala @@ -83,14 +83,14 @@ final class RelationUi(helpers: Helpers): frag( (followable && !blocked).option( a( - cls := "text relation-button", - href := routes.Relation.follow(user.name), + cls := "text relation-button", + href := routes.Relation.follow(user.name), dataIcon := Icon.ThumbsUp )(trans.site.follow.txt()) ), a( - cls := "text relation-button", - href := routes.Relation.block(user.name), + cls := "text relation-button", + href := routes.Relation.block(user.name), dataIcon := Icon.NotAllowed )(trans.site.block.txt()) ) From 2313d1c6321bc578b565acc475e9f0c5701613df Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 17 May 2024 09:30:07 +0200 Subject: [PATCH 004/168] update UserName.historicalRegex - closes #15314 for https://lichess.org/@/__kamensky --- modules/core/src/main/userId.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/core/src/main/userId.scala b/modules/core/src/main/userId.scala index 9c0f8053af1d9..943d4c4f3c8a6 100644 --- a/modules/core/src/main/userId.scala +++ b/modules/core/src/main/userId.scala @@ -57,7 +57,7 @@ object userId: extension (e: UserName) def str = UserStr(e) given UserIdOf[UserName] = n => UserId(n.value.toLowerCase) // what existing usernames are like - val historicalRegex = "(?i)[a-z0-9][a-z0-9_-]{0,28}[a-z0-9]".r + val historicalRegex = "(?i)[a-z0-9_][a-z0-9_-]{0,28}[a-z0-9]".r val anonymous: UserName = "Anonymous" val lichess: UserName = "lichess" val anonMod: String = "A Lichess Moderator" From ad68b2388a137ac5a3b4c01d980e7210d2427ffa Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Tue, 11 Jun 2024 17:06:17 -0700 Subject: [PATCH 005/168] Tweak the branch styling for nested variations - Remove the "head" rendered as part of the element as it doesn't accurately represent where we're branching from - Remove the "tail" that extends past the bottom variation as it's visually unnecessary. This means the height is no longer the same as the parent height, so we need a new element with variable height. Tested on Firefox & Chrome with inline & column mode in various zoom levels. --- ui/analyse/src/treeView/columnView.ts | 8 +++--- ui/analyse/src/treeView/inlineView.ts | 8 +++--- ui/tree/css/_tree.scss | 40 ++++++++++++++++----------- 3 files changed, 32 insertions(+), 24 deletions(-) diff --git a/ui/analyse/src/treeView/columnView.ts b/ui/analyse/src/treeView/columnView.ts index f08103f5e4a08..ed340a416f036 100644 --- a/ui/analyse/src/treeView/columnView.ts +++ b/ui/analyse/src/treeView/columnView.ts @@ -100,16 +100,16 @@ function renderLines(ctx: Ctx, nodes: Tree.Node[], opts: Opts): VNode { nodes.map(n => { return ( retroLine(ctx, n) || - h( - 'line', - renderMoveAndChildrenOf(ctx, n, { + h('line', [ + h('branch'), + ...renderMoveAndChildrenOf(ctx, n, { parentPath: opts.parentPath, isMainline: false, withIndex: true, noConceal: opts.noConceal, truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, }), - ) + ]) ); }), ); diff --git a/ui/analyse/src/treeView/inlineView.ts b/ui/analyse/src/treeView/inlineView.ts index 8c2181700b585..6f8903793db75 100644 --- a/ui/analyse/src/treeView/inlineView.ts +++ b/ui/analyse/src/treeView/inlineView.ts @@ -68,15 +68,15 @@ function renderLines(ctx: Ctx, nodes: Tree.Node[], opts: Opts): VNode { nodes.map(n => { return ( retroLine(ctx, n) || - h( - 'line', - renderMoveAndChildrenOf(ctx, n, { + h('line', [ + h('branch'), + ...renderMoveAndChildrenOf(ctx, n, { parentPath: opts.parentPath, isMainline: false, withIndex: true, truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, }), - ) + ]) ); }), ); diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index 9c5bf91bd9998..d4e3dfa5ea539 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -301,7 +301,7 @@ margin-top: 2px; margin-inline-start: 6px; margin-bottom: 0.8em; - border-inline-start: 2px solid $c-border; + padding-inline-start: 2px; } > interrupt > lines { @@ -319,37 +319,45 @@ line { display: block; padding-inline-start: 7px; + position: relative; } &-column line { margin: 2px 0; } - lines lines move { - font-size: 13px; + line > branch { + border-inline-start: 2px solid $c-border; + margin-inline-start: -9px; + position: absolute; + width: 8px; + height: 100%; } - lines lines { - margin-inline-start: 1px; + line:last-child > branch { + height: calc(0.8em + 2px); } - lines lines::before { + line > branch::before { + margin-top: 0.8em; content: ' '; border-top: 2px solid $c-border; position: absolute; - margin-inline-start: -11px; - width: 9px; + width: 7px; height: 6px; } - lines line::before { - margin-top: 0.65em; - margin-inline-start: -8px; - content: ' '; - border-top: 2px solid $c-border; - position: absolute; - width: 8px; - height: 6px; + &-column line > branch { + margin-top: -2px; + height: calc(100% + 2px); + } + + lines lines move { + font-size: 13px; + } + + lines lines { + margin-inline-start: 1px; } lines lines:last-child { From 6f4171f5702604685793c28d4efb0df32dbe5428 Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Wed, 12 Jun 2024 00:33:31 -0700 Subject: [PATCH 006/168] Collapse/expand variations --- modules/analyse/src/main/ui/AnalyseI18n.scala | 2 ++ modules/coreI18n/src/main/key.scala | 2 ++ translation/source/site.xml | 2 ++ ui/@types/lichess/tree.d.ts | 1 + ui/analyse/src/ctrl.ts | 5 ++++ ui/analyse/src/treeView/columnView.ts | 24 ++++++++++++------- ui/analyse/src/treeView/contextMenu.ts | 4 ++++ ui/analyse/src/treeView/inlineView.ts | 23 ++++++++++++------ ui/analyse/src/view/util.ts | 4 ++-- ui/tree/css/_tree.scss | 15 +++++++++++- ui/tree/src/tree.ts | 6 +++++ 11 files changed, 70 insertions(+), 18 deletions(-) diff --git a/modules/analyse/src/main/ui/AnalyseI18n.scala b/modules/analyse/src/main/ui/AnalyseI18n.scala index 4b788558741d6..ce5ab5661c83c 100644 --- a/modules/analyse/src/main/ui/AnalyseI18n.scala +++ b/modules/analyse/src/main/ui/AnalyseI18n.scala @@ -91,6 +91,8 @@ final class AnalyseI18n(helpers: Helpers): site.promoteVariation, site.makeMainLine, site.deleteFromHere, + site.collapseVariations, + site.expandVariations, site.forceVariation, site.copyVariationPgn, // practice (also uses checkmate, draw) diff --git a/modules/coreI18n/src/main/key.scala b/modules/coreI18n/src/main/key.scala index 9ae5db0295c50..0bf92e2084696 100644 --- a/modules/coreI18n/src/main/key.scala +++ b/modules/coreI18n/src/main/key.scala @@ -86,6 +86,8 @@ object I18nKey: val `promoteVariation`: I18nKey = "promoteVariation" val `makeMainLine`: I18nKey = "makeMainLine" val `deleteFromHere`: I18nKey = "deleteFromHere" + val `collapseVariations`: I18nKey = "collapseVariations" + val `expandVariations`: I18nKey = "expandVariations" val `forceVariation`: I18nKey = "forceVariation" val `copyVariationPgn`: I18nKey = "copyVariationPgn" val `move`: I18nKey = "move" diff --git a/translation/source/site.xml b/translation/source/site.xml index 02920c3bb6df8..d3bfc03e4e9e6 100644 --- a/translation/source/site.xml +++ b/translation/source/site.xml @@ -70,6 +70,8 @@ Promote variation Make mainline Delete from here + Collapse variations + Expand variations Force variation Copy variation PGN Move diff --git a/ui/@types/lichess/tree.d.ts b/ui/@types/lichess/tree.d.ts index d578442b66434..f47e999fb169e 100644 --- a/ui/@types/lichess/tree.d.ts +++ b/ui/@types/lichess/tree.d.ts @@ -72,6 +72,7 @@ declare namespace Tree { } export interface Node extends NodeBase { children: Node[]; + collapsed?: boolean; } export interface NodeCrazy { diff --git a/ui/analyse/src/ctrl.ts b/ui/analyse/src/ctrl.ts index c4f87723efc2c..58fe5dea3c894 100644 --- a/ui/analyse/src/ctrl.ts +++ b/ui/analyse/src/ctrl.ts @@ -636,6 +636,11 @@ export default class AnalyseCtrl { this.treeVersion++; } + setCollapsed(path: Tree.Path, collapsed: boolean): void { + this.tree.setCollapsedAt(path, collapsed); + this.redraw(); + } + forceVariation(path: Tree.Path, force: boolean): void { this.tree.forceVariationAt(path, force); this.jump(path); diff --git a/ui/analyse/src/treeView/columnView.ts b/ui/analyse/src/treeView/columnView.ts index ed340a416f036..c12e6ade4cd11 100644 --- a/ui/analyse/src/treeView/columnView.ts +++ b/ui/analyse/src/treeView/columnView.ts @@ -1,5 +1,6 @@ import { VNode } from 'snabbdom'; import { isEmpty } from 'common'; +import * as licon from 'common/licon'; import { LooseVNodes, looseH as h } from 'common/snabbdom'; import { fixCrazySan } from 'chess'; import { path as treePath, ops as treeOps } from 'tree'; @@ -63,7 +64,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): LooseVNodes | h( 'interrupt', commentTags.concat( - renderLines(ctx, main.forceVariation ? cs : cs.slice(1), { + renderLines(ctx, node, main.forceVariation ? cs : cs.slice(1), { parentPath: opts.parentPath, isMainline: passOpts.isMainline, conceal, @@ -77,7 +78,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): LooseVNodes | ]; } if (!cs[1]) return renderMoveAndChildrenOf(ctx, main, opts); - return renderInlined(ctx, cs, opts) || [renderLines(ctx, cs, opts)]; + return renderInlined(ctx, cs, opts) || [renderLines(ctx, node, cs, opts)]; } function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): LooseVNodes | undefined { @@ -93,11 +94,18 @@ function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): LooseVNodes | }); } -function renderLines(ctx: Ctx, nodes: Tree.Node[], opts: Opts): VNode { - return h( - 'lines', - { class: { single: !nodes[1] } }, - nodes.map(n => { +function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: Opts): VNode { + return h('lines', { class: { single: !nodes[1], collapsed: parentNode.collapsed || false } }, [ + parentNode.collapsed + ? h('line', { class: { expand: true } }, [ + h('branch'), + h('a', { + attrs: { 'data-icon': licon.PlusButton, title: ctx.ctrl.trans.noarg('expandVariations') }, + on: { click: () => ctx.ctrl.setCollapsed(opts.parentPath, false) }, + }), + ]) + : null, + ...nodes.map(n => { return ( retroLine(ctx, n) || h('line', [ @@ -112,7 +120,7 @@ function renderLines(ctx: Ctx, nodes: Tree.Node[], opts: Opts): VNode { ]) ); }), - ); + ]); } function renderMoveOf(ctx: Ctx, node: Tree.Node, opts: Opts): VNode { diff --git a/ui/analyse/src/treeView/contextMenu.ts b/ui/analyse/src/treeView/contextMenu.ts index f7f4a42af9ffc..06dc1a1b53d9b 100644 --- a/ui/analyse/src/treeView/contextMenu.ts +++ b/ui/analyse/src/treeView/contextMenu.ts @@ -82,6 +82,10 @@ function view(opts: Opts, coords: Coords): VNode { action(licon.Trash, trans('deleteFromHere'), () => ctrl.deleteNode(opts.path)), + action(licon.PlusButton, trans('expandVariations'), () => ctrl.setCollapsed(opts.path, false)), + + action(licon.MinusButton, trans('collapseVariations'), () => ctrl.setCollapsed(opts.path, true)), + ...(ctrl.study ? studyView.contextMenu(ctrl.study, opts.path, node) : []), onMainline && diff --git a/ui/analyse/src/treeView/inlineView.ts b/ui/analyse/src/treeView/inlineView.ts index 6f8903793db75..ba4c480119570 100644 --- a/ui/analyse/src/treeView/inlineView.ts +++ b/ui/analyse/src/treeView/inlineView.ts @@ -3,6 +3,7 @@ import { fixCrazySan } from 'chess'; import { path as treePath, ops as treeOps } from 'tree'; import * as moveView from '../view/moveView'; import AnalyseCtrl from '../ctrl'; +import * as licon from 'common/licon'; import { MaybeVNodes } from 'common/snabbdom'; import { mainHook, nodeClasses, renderInlineCommentsOf, retroLine, Ctx, Opts, renderingCtx } from './common'; @@ -31,7 +32,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes | ]), h( 'interrupt', - renderLines(ctx, main.forceVariation ? cs : cs.slice(1), { + renderLines(ctx, node, main.forceVariation ? cs : cs.slice(1), { parentPath: opts.parentPath, isMainline: true, }), @@ -47,7 +48,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes | ); } if (!cs[1]) return renderMoveAndChildrenOf(ctx, main, opts); - return renderInlined(ctx, cs, opts) || [renderLines(ctx, cs, opts)]; + return renderInlined(ctx, cs, opts) || [renderLines(ctx, node, cs, opts)]; } function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): MaybeVNodes | undefined { @@ -62,10 +63,18 @@ function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): MaybeVNodes | }); } -function renderLines(ctx: Ctx, nodes: Tree.Node[], opts: Opts): VNode { - return h( - 'lines', - nodes.map(n => { +function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: Opts): VNode { + return h('lines', { class: { collapsed: parentNode.collapsed || false } }, [ + parentNode.collapsed + ? h('line', { class: { expand: true } }, [ + h('branch'), + h('a', { + attrs: { 'data-icon': licon.PlusButton, title: ctx.ctrl.trans.noarg('expandVariations') }, + on: { click: () => ctx.ctrl.setCollapsed(opts.parentPath, false) }, + }), + ]) + : null, + ...nodes.map(n => { return ( retroLine(ctx, n) || h('line', [ @@ -79,7 +88,7 @@ function renderLines(ctx: Ctx, nodes: Tree.Node[], opts: Opts): VNode { ]) ); }), - ); + ]); } function renderMoveAndChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes { diff --git a/ui/analyse/src/view/util.ts b/ui/analyse/src/view/util.ts index 030f3b104ae57..9f81a9df22f39 100644 --- a/ui/analyse/src/view/util.ts +++ b/ui/analyse/src/view/util.ts @@ -1,8 +1,8 @@ import { fixCrazySan } from 'chess'; -import { attributesModule, classModule, init, h } from 'snabbdom'; +import { attributesModule, classModule, eventListenersModule, init, h } from 'snabbdom'; import { plyToTurn } from '../util'; -export const patch = init([classModule, attributesModule]); +export const patch = init([classModule, attributesModule, eventListenersModule]); export const emptyRedButton = 'button.button.button-red.button-empty'; diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index d4e3dfa5ea539..ee30461fe032f 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -334,7 +334,8 @@ height: 100%; } - line:last-child > branch { + line:last-child > branch, + line.expand > branch { height: calc(0.8em + 2px); } @@ -364,6 +365,18 @@ margin-bottom: 0; } + lines.collapsed > * { + display: none; + } + + lines.collapsed > line.expand { + display: block; + } + + line.expand > a { + color: var(--c-font-dim); + } + inline { display: inline; font-style: italic; diff --git a/ui/tree/src/tree.ts b/ui/tree/src/tree.ts index 3ccac90adca99..47f881956e608 100644 --- a/ui/tree/src/tree.ts +++ b/ui/tree/src/tree.ts @@ -26,6 +26,7 @@ export interface TreeWrapper { lastMainlineNode(path: Tree.Path): Tree.Node; pathExists(path: Tree.Path): boolean; deleteNodeAt(path: Tree.Path): void; + setCollapsedAt(path: Tree.Path, collapsed: boolean): MaybeNode; promoteAt(path: Tree.Path, toMainline: boolean): void; forceVariationAt(path: Tree.Path, force: boolean): MaybeNode; getCurrentNodesAfterPly(nodeList: Tree.Node[], mainline: Tree.Node[], ply: number): Tree.Node[]; @@ -212,6 +213,11 @@ export function build(root: Tree.Node): TreeWrapper { node.clock = clock; }); }, + setCollapsedAt(path: Tree.Path, collapsed: boolean) { + return updateAt(path, function (node) { + node.collapsed = collapsed; + }); + }, pathIsMainline, pathIsForcedVariation, lastMainlineNode(path: Tree.Path): Tree.Node { From 00679f2fa401ec9cc1d0953d7ddfcd32b241e25c Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Wed, 12 Jun 2024 16:35:26 -0700 Subject: [PATCH 007/168] Recursively expand/collapse --- ui/analyse/src/ctrl.ts | 8 ++++++++ ui/analyse/src/treeView/contextMenu.ts | 4 ++-- ui/tree/src/tree.ts | 13 +++++++++++++ 3 files changed, 23 insertions(+), 2 deletions(-) diff --git a/ui/analyse/src/ctrl.ts b/ui/analyse/src/ctrl.ts index 58fe5dea3c894..dc989d737b5c0 100644 --- a/ui/analyse/src/ctrl.ts +++ b/ui/analyse/src/ctrl.ts @@ -641,6 +641,14 @@ export default class AnalyseCtrl { this.redraw(); } + setAllCollapsed(path: Tree.Path, collapsed: boolean): void { + // Also update parent + const parentPath = treePath.init(path); + this.tree.setCollapsedAt(parentPath, collapsed); + this.tree.setCollapsedRecursive(path, collapsed); + this.redraw(); + } + forceVariation(path: Tree.Path, force: boolean): void { this.tree.forceVariationAt(path, force); this.jump(path); diff --git a/ui/analyse/src/treeView/contextMenu.ts b/ui/analyse/src/treeView/contextMenu.ts index 06dc1a1b53d9b..edcadf8ef1c28 100644 --- a/ui/analyse/src/treeView/contextMenu.ts +++ b/ui/analyse/src/treeView/contextMenu.ts @@ -82,9 +82,9 @@ function view(opts: Opts, coords: Coords): VNode { action(licon.Trash, trans('deleteFromHere'), () => ctrl.deleteNode(opts.path)), - action(licon.PlusButton, trans('expandVariations'), () => ctrl.setCollapsed(opts.path, false)), + action(licon.PlusButton, trans('expandVariations'), () => ctrl.setAllCollapsed(opts.path, false)), - action(licon.MinusButton, trans('collapseVariations'), () => ctrl.setCollapsed(opts.path, true)), + action(licon.MinusButton, trans('collapseVariations'), () => ctrl.setAllCollapsed(opts.path, true)), ...(ctrl.study ? studyView.contextMenu(ctrl.study, opts.path, node) : []), diff --git a/ui/tree/src/tree.ts b/ui/tree/src/tree.ts index 47f881956e608..e29acbbc3b7c7 100644 --- a/ui/tree/src/tree.ts +++ b/ui/tree/src/tree.ts @@ -27,6 +27,7 @@ export interface TreeWrapper { pathExists(path: Tree.Path): boolean; deleteNodeAt(path: Tree.Path): void; setCollapsedAt(path: Tree.Path, collapsed: boolean): MaybeNode; + setCollapsedRecursive(path: Tree.Path, collapsed: boolean): void; promoteAt(path: Tree.Path, toMainline: boolean): void; forceVariationAt(path: Tree.Path, force: boolean): MaybeNode; getCurrentNodesAfterPly(nodeList: Tree.Node[], mainline: Tree.Node[], ply: number): Tree.Node[]; @@ -110,6 +111,13 @@ export function build(root: Tree.Node): TreeWrapper { return; } + function updateRecursive(path: Tree.Path, update: (node: Tree.Node) => void): void { + const node = nodeAtPathOrNull(path); + if (node) { + ops.updateAll(node, update); + } + } + // returns new path function addNode(node: Tree.Node, path: Tree.Path): Tree.Path | undefined { const newPath = path + node.id, @@ -218,6 +226,11 @@ export function build(root: Tree.Node): TreeWrapper { node.collapsed = collapsed; }); }, + setCollapsedRecursive(path: Tree.Path, collapsed: boolean) { + return updateRecursive(path, function (node) { + node.collapsed = collapsed; + }); + }, pathIsMainline, pathIsForcedVariation, lastMainlineNode(path: Tree.Path): Tree.Node { From bc78ab3475457d7a8fea48f961c15ce1846c06b8 Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Wed, 12 Jun 2024 16:35:47 -0700 Subject: [PATCH 008/168] Auto-expand when navigating through moves --- ui/analyse/src/ctrl.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/ui/analyse/src/ctrl.ts b/ui/analyse/src/ctrl.ts index dc989d737b5c0..07a7a20b17477 100644 --- a/ui/analyse/src/ctrl.ts +++ b/ui/analyse/src/ctrl.ts @@ -253,6 +253,9 @@ export default class AnalyseCtrl { this.path = path; this.nodeList = this.tree.getNodeList(path); this.node = treeOps.last(this.nodeList) as Tree.Node; + for (let i = 0; i < this.nodeList.length; i++) { + this.nodeList[i].collapsed = false; + } this.mainline = treeOps.mainlineNodeList(this.tree.root); this.onMainline = this.tree.pathIsMainline(path); this.fenInput = undefined; From 87864009188d954ea4d2dd5cfdd68c914e15a2c4 Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Wed, 12 Jun 2024 18:13:45 -0700 Subject: [PATCH 009/168] Auto-collapse deeply nested variations on page load The node's "collapse" flag we now treat as a tri-state where "undefined" means to use the default collapse state. --- ui/analyse/src/treeView/columnView.ts | 36 +++++++++++++++++++++------ ui/analyse/src/treeView/common.ts | 1 + ui/analyse/src/treeView/inlineView.ts | 21 +++++++++++++--- 3 files changed, 48 insertions(+), 10 deletions(-) diff --git a/ui/analyse/src/treeView/columnView.ts b/ui/analyse/src/treeView/columnView.ts index c12e6ade4cd11..2dee229521913 100644 --- a/ui/analyse/src/treeView/columnView.ts +++ b/ui/analyse/src/treeView/columnView.ts @@ -49,13 +49,28 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): LooseVNodes | if (!cs[1] && isEmpty(commentTags) && !main.forceVariation) return [ isWhite && moveView.renderIndex(main.ply, false), - ...renderMoveAndChildrenOf(ctx, main, { parentPath: opts.parentPath, isMainline: true, conceal }), + ...renderMoveAndChildrenOf(ctx, main, { + parentPath: opts.parentPath, + isMainline: true, + depth: opts.depth, + conceal, + }), ]; const mainChildren = !main.forceVariation && - renderChildrenOf(ctx, main, { parentPath: opts.parentPath + main.id, isMainline: true, conceal }); - - const passOpts = { parentPath: opts.parentPath, isMainline: !main.forceVariation, conceal }; + renderChildrenOf(ctx, main, { + parentPath: opts.parentPath + main.id, + isMainline: true, + depth: opts.depth, + conceal, + }); + + const passOpts = { + parentPath: opts.parentPath, + isMainline: !main.forceVariation, + depth: opts.depth, + conceal, + }; return [ isWhite && moveView.renderIndex(main.ply, false), @@ -67,6 +82,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): LooseVNodes | renderLines(ctx, node, main.forceVariation ? cs : cs.slice(1), { parentPath: opts.parentPath, isMainline: passOpts.isMainline, + depth: opts.depth, conceal, noConceal: !conceal, }), @@ -89,14 +105,17 @@ function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): LooseVNodes | return renderMoveAndChildrenOf(ctx, nodes[0], { parentPath: opts.parentPath, isMainline: false, + depth: opts.depth, noConceal: opts.noConceal, inline: nodes[1], }); } function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: Opts): VNode { - return h('lines', { class: { single: !nodes[1], collapsed: parentNode.collapsed || false } }, [ - parentNode.collapsed + const collapsed = + parentNode.collapsed === undefined ? opts.depth >= 2 && opts.depth % 2 === 0 : parentNode.collapsed; + return h('lines', { class: { single: !nodes[1], collapsed } }, [ + collapsed ? h('line', { class: { expand: true } }, [ h('branch'), h('a', { @@ -113,6 +132,7 @@ function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: ...renderMoveAndChildrenOf(ctx, n, { parentPath: opts.parentPath, isMainline: false, + depth: opts.depth + 1, withIndex: true, noConceal: opts.noConceal, truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, @@ -154,6 +174,7 @@ function renderMoveAndChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): LooseVN ...(renderChildrenOf(ctx, node, { parentPath: path, isMainline: opts.isMainline, + depth: opts.depth, noConceal: opts.noConceal, truncate: opts.truncate ? opts.truncate - 1 : undefined, }) || []), @@ -167,6 +188,7 @@ function renderInline(ctx: Ctx, node: Tree.Node, opts: Opts): VNode { withIndex: true, parentPath: opts.parentPath, isMainline: false, + depth: opts.depth, noConceal: opts.noConceal, truncate: opts.truncate, }), @@ -209,6 +231,6 @@ export default function (ctrl: AnalyseCtrl, concealOf?: ConcealOf): VNode { !isEmpty(commentTags) && h('interrupt', commentTags), blackStarts && moveView.renderIndex(root.ply, false), blackStarts && emptyMove(), - ...(renderChildrenOf(ctx, root, { parentPath: '', isMainline: true }) || []), + ...(renderChildrenOf(ctx, root, { parentPath: '', isMainline: true, depth: 0 }) || []), ]); } diff --git a/ui/analyse/src/treeView/common.ts b/ui/analyse/src/treeView/common.ts index cb6c03bc49ad9..782cd7afcdabd 100644 --- a/ui/analyse/src/treeView/common.ts +++ b/ui/analyse/src/treeView/common.ts @@ -164,6 +164,7 @@ export interface Ctx { export interface Opts { parentPath: Tree.Path; isMainline: boolean; + depth: number; inline?: Tree.Node; withIndex?: boolean; truncate?: number; diff --git a/ui/analyse/src/treeView/inlineView.ts b/ui/analyse/src/treeView/inlineView.ts index ba4c480119570..5b81597411a34 100644 --- a/ui/analyse/src/treeView/inlineView.ts +++ b/ui/analyse/src/treeView/inlineView.ts @@ -16,6 +16,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes | return renderMoveAndChildrenOf(ctx, main, { parentPath: opts.parentPath, isMainline: true, + depth: opts.depth, withIndex: opts.withIndex, }); return ( @@ -26,6 +27,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes | renderMoveOf(ctx, main, { parentPath: opts.parentPath, isMainline: true, + depth: opts.depth, withIndex: opts.withIndex, }), ...renderInlineCommentsOf(ctx, main, opts.parentPath), @@ -35,6 +37,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes | renderLines(ctx, node, main.forceVariation ? cs : cs.slice(1), { parentPath: opts.parentPath, isMainline: true, + depth: opts.depth, }), ), ...(main.forceVariation @@ -42,6 +45,7 @@ function renderChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes | : renderChildrenOf(ctx, main, { parentPath: opts.parentPath + main.id, isMainline: true, + depth: opts.depth, withIndex: true, }) || []), ] @@ -59,13 +63,16 @@ function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): MaybeVNodes | return renderMoveAndChildrenOf(ctx, nodes[0], { parentPath: opts.parentPath, isMainline: opts.isMainline, + depth: opts.depth, inline: nodes[1], }); } function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: Opts): VNode { - return h('lines', { class: { collapsed: parentNode.collapsed || false } }, [ - parentNode.collapsed + const collapsed = + parentNode.collapsed === undefined ? opts.depth >= 2 && opts.depth % 2 === 0 : parentNode.collapsed; + return h('lines', { class: { collapsed } }, [ + collapsed ? h('line', { class: { expand: true } }, [ h('branch'), h('a', { @@ -82,6 +89,7 @@ function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: ...renderMoveAndChildrenOf(ctx, n, { parentPath: opts.parentPath, isMainline: false, + depth: opts.depth + 1, withIndex: true, truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, }), @@ -102,6 +110,7 @@ function renderMoveAndChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVN renderChildrenOf(ctx, node, { parentPath: path, isMainline: opts.isMainline, + depth: opts.depth, truncate: opts.truncate ? opts.truncate - 1 : undefined, withIndex: !!comments[0], }) || [], @@ -113,7 +122,12 @@ function renderInline(ctx: Ctx, node: Tree.Node, opts: Opts): VNode { if (retro) return h('interrupt', h('lines', retro)); return h( 'inline', - renderMoveAndChildrenOf(ctx, node, { withIndex: true, parentPath: opts.parentPath, isMainline: false }), + renderMoveAndChildrenOf(ctx, node, { + withIndex: true, + parentPath: opts.parentPath, + isMainline: false, + depth: opts.depth, + }), ); } @@ -134,6 +148,7 @@ export default function (ctrl: AnalyseCtrl): VNode { ...(renderChildrenOf(ctx, ctrl.tree.root, { parentPath: '', isMainline: true, + depth: 0, }) || []), ]); } From 4e598bc47b81d563c9131d2284cc12436b894df0 Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Wed, 12 Jun 2024 19:03:48 -0700 Subject: [PATCH 010/168] Fix autoscroll offset We've added "position: relative" to the elements for styling purposes, but that affects the calculation of the "offsetTop" property. We can instead calculate the offset ourselves. --- ui/analyse/src/treeView/common.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ui/analyse/src/treeView/common.ts b/ui/analyse/src/treeView/common.ts index 782cd7afcdabd..70d9fe8480295 100644 --- a/ui/analyse/src/treeView/common.ts +++ b/ui/analyse/src/treeView/common.ts @@ -59,7 +59,8 @@ const autoScroll = throttle(200, (ctrl: AnalyseCtrl, el: HTMLElement) => { cont.scrollTop = ctrl.path ? 99999 : 0; return; } - cont.scrollTop = target.offsetTop - cont.offsetHeight / 2 + target.offsetHeight; + const targetOffset = target.getBoundingClientRect().y - el.getBoundingClientRect().y; + cont.scrollTop = targetOffset - cont.offsetHeight / 2 + target.offsetHeight; }); export interface NodeClasses { From aefe9be30f19255f5a9ffbae93d9b7f03c8a2df7 Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Wed, 12 Jun 2024 19:53:28 -0700 Subject: [PATCH 011/168] Add hover styling for expand variations button --- ui/tree/css/_tree.scss | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index ee30461fe032f..9819ef30d6b1c 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -377,6 +377,10 @@ color: var(--c-font-dim); } + line.expand > a:hover { + color: var(--c-font); + } + inline { display: inline; font-style: italic; From fdd82e33a349cb5ec1142121c4dc8e13417b6ac5 Mon Sep 17 00:00:00 2001 From: Trevor Fitzgerald Date: Thu, 13 Jun 2024 12:29:17 -0400 Subject: [PATCH 012/168] Vitest with jsdom for frontend tests --- .github/workflows/assets.yml | 8 +- .github/workflows/lint.yml | 2 +- pnpm-lock.yaml | 3010 ++++++----------- ui/@types/lichess/index.d.ts | 6 +- ui/README.md | 11 +- ui/chat/src/spam.ts | 2 +- ui/chat/tests/spam.test.ts | 25 + ui/common/src/richText.ts | 6 +- ui/common/tests/richText.test.ts | 31 + ui/game/src/status.ts | 2 +- ui/jest.config.js | 9 - ui/keyboardMove/package.json | 1 - .../{src => tests}/keyboardSubmit.test.ts | 152 +- ui/package.json | 7 +- ui/site/src/clockWidget.ts | 2 +- ui/site/src/log.ts | 4 +- ui/site/src/once.ts | 9 +- ui/site/src/powertip.ts | 2 +- ui/site/tests/once.test.ts | 14 + ui/site/tests/timeago.test.ts | 40 + ui/vitest.config.ts | 7 + 21 files changed, 1280 insertions(+), 2070 deletions(-) create mode 100644 ui/chat/tests/spam.test.ts create mode 100644 ui/common/tests/richText.test.ts delete mode 100644 ui/jest.config.js rename ui/keyboardMove/{src => tests}/keyboardSubmit.test.ts (83%) create mode 100644 ui/site/tests/once.test.ts create mode 100644 ui/site/tests/timeago.test.ts create mode 100644 ui/vitest.config.ts diff --git a/.github/workflows/assets.yml b/.github/workflows/assets.yml index 2288702812daf..571be6c445e7e 100644 --- a/.github/workflows/assets.yml +++ b/.github/workflows/assets.yml @@ -25,7 +25,10 @@ jobs: - uses: actions/checkout@v4 with: submodules: recursive - - uses: pnpm/action-setup@v3 + - uses: actions/setup-node@v4 + with: + node-version: 22 + - uses: pnpm/action-setup@v4 with: run_install: true - uses: actions/checkout@v4 @@ -39,7 +42,8 @@ jobs: - run: pnpm link "$GITHUB_WORKSPACE/ab" if: steps.ab.outcome == 'success' - run: ./ui/build --no-install -p - - run: cd ui && pnpm run test && cd - + - run: pnpm test + working-directory: ui - run: mkdir assets && mv public assets/ && cp -p bin/download-lifat LICENSE COPYING.md README.md assets/ && git log -n 1 --pretty=oneline > assets/commit.txt - run: cd assets && tar --zstd -cvpf ../assets.tar.zst . && cd - - uses: actions/upload-artifact@v4 diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 88bc7f0d357bf..501a94459dccb 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -17,7 +17,7 @@ jobs: security-events: write steps: - uses: actions/checkout@v4 - - uses: pnpm/action-setup@v3 + - uses: pnpm/action-setup@v4 with: run_install: true - uses: github/codeql-action/init@v3 diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 85d700620dbe0..f4e6dc0eceb28 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -53,12 +53,12 @@ importers: ui: dependencies: - '@types/jest': - specifier: ^29.5.12 - version: 29.5.12 - jest: - specifier: ^29.7.0 - version: 29.7.0(@types/node@20.12.2) + jsdom: + specifier: ^24.1.0 + version: 24.1.0 + vitest: + specifier: ^1.6.0 + version: 1.6.0(@types/node@20.12.2)(jsdom@24.1.0) ui/@types/lichess: {} @@ -394,9 +394,6 @@ importers: ui/keyboardMove: dependencies: - '@jest/globals': - specifier: ^29.7.0 - version: 29.7.0 chess: specifier: workspace:* version: link:../chess @@ -782,185 +779,152 @@ packages: resolution: {integrity: sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==} engines: {node: '>=0.10.0'} - '@ampproject/remapping@2.3.0': - resolution: {integrity: sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==} - engines: {node: '>=6.0.0'} - - '@babel/code-frame@7.24.2': - resolution: {integrity: sha512-y5+tLQyV8pg3fsiln67BVLD1P13Eg4lh5RW9mF0zUuvLrv9uIQ4MCL+CRT+FTsBlBjcIan6PGsLcBN0m3ClUyQ==} - engines: {node: '>=6.9.0'} - - '@babel/compat-data@7.24.1': - resolution: {integrity: sha512-Pc65opHDliVpRHuKfzI+gSA4zcgr65O4cl64fFJIWEEh8JoHIHh0Oez1Eo8Arz8zq/JhgKodQaxEwUPRtZylVA==} - engines: {node: '>=6.9.0'} - - '@babel/core@7.24.3': - resolution: {integrity: sha512-5FcvN1JHw2sHJChotgx8Ek0lyuh4kCKelgMTTqhYJJtloNvUfpAFMeNQUtdlIaktwrSV9LtCdqwk48wL2wBacQ==} - engines: {node: '>=6.9.0'} - - '@babel/generator@7.24.1': - resolution: {integrity: sha512-DfCRfZsBcrPEHUfuBMgbJ1Ut01Y/itOs+hY2nFLgqsqXd52/iSiVq5TITtUasIUgm+IIKdY2/1I7auiQOEeC9A==} - engines: {node: '>=6.9.0'} - - '@babel/helper-compilation-targets@7.23.6': - resolution: {integrity: sha512-9JB548GZoQVmzrFgp8o7KxdgkTGm6xs9DW0o/Pim72UDjzr5ObUQ6ZzYPqA+g9OTS2bBQoctLJrky0RDCAWRgQ==} - engines: {node: '>=6.9.0'} - - '@babel/helper-environment-visitor@7.22.20': - resolution: {integrity: sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==} - engines: {node: '>=6.9.0'} - - '@babel/helper-function-name@7.23.0': - resolution: {integrity: sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==} - engines: {node: '>=6.9.0'} - - '@babel/helper-hoist-variables@7.22.5': - resolution: {integrity: sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==} - engines: {node: '>=6.9.0'} - - '@babel/helper-module-imports@7.24.3': - resolution: {integrity: sha512-viKb0F9f2s0BCS22QSF308z/+1YWKV/76mwt61NBzS5izMzDPwdq1pTrzf+Li3npBWX9KdQbkeCt1jSAM7lZqg==} - engines: {node: '>=6.9.0'} - - '@babel/helper-module-transforms@7.23.3': - resolution: {integrity: sha512-7bBs4ED9OmswdfDzpz4MpWgSrV7FXlc3zIagvLFjS5H+Mk7Snr21vQ6QwrsoCGMfNC4e4LQPdoULEt4ykz0SRQ==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0 - - '@babel/helper-plugin-utils@7.24.0': - resolution: {integrity: sha512-9cUznXMG0+FxRuJfvL82QlTqIzhVW9sL0KjMPHhAOOvpQGL8QtdxnBKILjBqxlHyliz0yCa1G903ZXI/FuHy2w==} - engines: {node: '>=6.9.0'} - - '@babel/helper-simple-access@7.22.5': - resolution: {integrity: sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w==} - engines: {node: '>=6.9.0'} - - '@babel/helper-split-export-declaration@7.22.6': - resolution: {integrity: sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==} - engines: {node: '>=6.9.0'} - - '@babel/helper-string-parser@7.24.1': - resolution: {integrity: sha512-2ofRCjnnA9y+wk8b9IAREroeUP02KHp431N2mhKniy2yKIDKpbrHv9eXwm8cBeWQYcJmzv5qKCu65P47eCF7CQ==} - engines: {node: '>=6.9.0'} - - '@babel/helper-validator-identifier@7.22.20': - resolution: {integrity: sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==} - engines: {node: '>=6.9.0'} + '@badrap/result@0.2.13': + resolution: {integrity: sha512-Qvyzz0dmGY0h8pwvBFo1BznAKf5Y5NXIDiqhPALWtfU7oHbAToCtPu4FlYQ3uysskSWLx8GUiyhe0nv0nDd/7Q==} - '@babel/helper-validator-option@7.23.5': - resolution: {integrity: sha512-85ttAOMLsr53VgXkTbkx8oA6YTfT4q7/HzXSLEYmjcSTJPMPQtvq1BD79Byep5xMUYbGRzEpDsjUf3dyp54IKw==} - engines: {node: '>=6.9.0'} + '@blakeembrey/deque@1.0.5': + resolution: {integrity: sha512-6xnwtvp9DY1EINIKdTfvfeAtCYw4OqBZJhtiqkT3ivjnEfa25VQ3TsKvaFfKm8MyGIEfE95qLe+bNEt3nB0Ylg==} - '@babel/helpers@7.24.1': - resolution: {integrity: sha512-BpU09QqEe6ZCHuIHFphEFgvNSrubve1FtyMton26ekZ85gRGi6LrTF7zArARp2YvyFxloeiRmtSCq5sjh1WqIg==} - engines: {node: '>=6.9.0'} + '@blakeembrey/template@1.1.0': + resolution: {integrity: sha512-iZf+UWfL+DogJVpd/xMQyP6X6McYd6ArdYoPMiv/zlOTzeXXfQbYxBNJJBF6tThvsjLMbA8tLjkCdm9RWMFCCw==} - '@babel/highlight@7.24.2': - resolution: {integrity: sha512-Yac1ao4flkTxTteCDZLEvdxg2fZfz1v8M4QpaGypq/WPDqg3ijHYbDfs+LG5hvzSoqaSZ9/Z9lKSP3CjZjv+pA==} - engines: {node: '>=6.9.0'} + '@esbuild/aix-ppc64@0.21.5': + resolution: {integrity: sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [aix] - '@babel/parser@7.24.1': - resolution: {integrity: sha512-Zo9c7N3xdOIQrNip7Lc9wvRPzlRtovHVE4lkz8WEDr7uYh/GMQhSiIgFxGIArRHYdJE5kxtZjAf8rT0xhdLCzg==} - engines: {node: '>=6.0.0'} - hasBin: true + '@esbuild/android-arm64@0.21.5': + resolution: {integrity: sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] - '@babel/plugin-syntax-async-generators@7.8.4': - resolution: {integrity: sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/android-arm@0.21.5': + resolution: {integrity: sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] - '@babel/plugin-syntax-bigint@7.8.3': - resolution: {integrity: sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/android-x64@0.21.5': + resolution: {integrity: sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] - '@babel/plugin-syntax-class-properties@7.12.13': - resolution: {integrity: sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/darwin-arm64@0.21.5': + resolution: {integrity: sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] - '@babel/plugin-syntax-import-meta@7.10.4': - resolution: {integrity: sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/darwin-x64@0.21.5': + resolution: {integrity: sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] - '@babel/plugin-syntax-json-strings@7.8.3': - resolution: {integrity: sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/freebsd-arm64@0.21.5': + resolution: {integrity: sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] - '@babel/plugin-syntax-jsx@7.24.1': - resolution: {integrity: sha512-2eCtxZXf+kbkMIsXS4poTvT4Yu5rXiRa+9xGVT56raghjmBTKMpFNc9R4IDiB4emao9eO22Ox7CxuJG7BgExqA==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/freebsd-x64@0.21.5': + resolution: {integrity: sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] - '@babel/plugin-syntax-logical-assignment-operators@7.10.4': - resolution: {integrity: sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-arm64@0.21.5': + resolution: {integrity: sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] - '@babel/plugin-syntax-nullish-coalescing-operator@7.8.3': - resolution: {integrity: sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-arm@0.21.5': + resolution: {integrity: sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] - '@babel/plugin-syntax-numeric-separator@7.10.4': - resolution: {integrity: sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-ia32@0.21.5': + resolution: {integrity: sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] - '@babel/plugin-syntax-object-rest-spread@7.8.3': - resolution: {integrity: sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-loong64@0.21.5': + resolution: {integrity: sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] - '@babel/plugin-syntax-optional-catch-binding@7.8.3': - resolution: {integrity: sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-mips64el@0.21.5': + resolution: {integrity: sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] - '@babel/plugin-syntax-optional-chaining@7.8.3': - resolution: {integrity: sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-ppc64@0.21.5': + resolution: {integrity: sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] - '@babel/plugin-syntax-top-level-await@7.14.5': - resolution: {integrity: sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-riscv64@0.21.5': + resolution: {integrity: sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] - '@babel/plugin-syntax-typescript@7.24.1': - resolution: {integrity: sha512-Yhnmvy5HZEnHUty6i++gcfH1/l68AHnItFHnaCv6hn9dNh0hQvvQJsxpi4BMBFN5DLeHBuucT/0DgzXif/OyRw==} - engines: {node: '>=6.9.0'} - peerDependencies: - '@babel/core': ^7.0.0-0 + '@esbuild/linux-s390x@0.21.5': + resolution: {integrity: sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] - '@babel/template@7.24.0': - resolution: {integrity: sha512-Bkf2q8lMB0AFpX0NFEqSbx1OkTHf0f+0j82mkw+ZpzBnkk7e9Ql0891vlfgi+kHwOk8tQjiQHpqh4LaSa0fKEA==} - engines: {node: '>=6.9.0'} + '@esbuild/linux-x64@0.21.5': + resolution: {integrity: sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] - '@babel/traverse@7.24.1': - resolution: {integrity: sha512-xuU6o9m68KeqZbQuDt2TcKSxUw/mrsvavlEqQ1leZ/B+C9tk6E4sRWy97WaXgvq5E+nU3cXMxv3WKOCanVMCmQ==} - engines: {node: '>=6.9.0'} + '@esbuild/netbsd-x64@0.21.5': + resolution: {integrity: sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] - '@babel/types@7.24.0': - resolution: {integrity: sha512-+j7a5c253RfKh8iABBhywc8NSfP5LURe7Uh4qpsh6jc+aLJguvmIUBdjSdEMQv2bENrCR5MfRdjGo7vzS/ob7w==} - engines: {node: '>=6.9.0'} + '@esbuild/openbsd-x64@0.21.5': + resolution: {integrity: sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] - '@badrap/result@0.2.13': - resolution: {integrity: sha512-Qvyzz0dmGY0h8pwvBFo1BznAKf5Y5NXIDiqhPALWtfU7oHbAToCtPu4FlYQ3uysskSWLx8GUiyhe0nv0nDd/7Q==} + '@esbuild/sunos-x64@0.21.5': + resolution: {integrity: sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] - '@bcoe/v8-coverage@0.2.3': - resolution: {integrity: sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==} + '@esbuild/win32-arm64@0.21.5': + resolution: {integrity: sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] - '@blakeembrey/deque@1.0.5': - resolution: {integrity: sha512-6xnwtvp9DY1EINIKdTfvfeAtCYw4OqBZJhtiqkT3ivjnEfa25VQ3TsKvaFfKm8MyGIEfE95qLe+bNEt3nB0Ylg==} + '@esbuild/win32-ia32@0.21.5': + resolution: {integrity: sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] - '@blakeembrey/template@1.1.0': - resolution: {integrity: sha512-iZf+UWfL+DogJVpd/xMQyP6X6McYd6ArdYoPMiv/zlOTzeXXfQbYxBNJJBF6tThvsjLMbA8tLjkCdm9RWMFCCw==} + '@esbuild/win32-x64@0.21.5': + resolution: {integrity: sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] '@eslint-community/eslint-utils@4.4.0': resolution: {integrity: sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==} @@ -1003,98 +967,13 @@ packages: '@humanwhocodes/object-schema@2.0.3': resolution: {integrity: sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==} - '@istanbuljs/load-nyc-config@1.1.0': - resolution: {integrity: sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==} - engines: {node: '>=8'} - - '@istanbuljs/schema@0.1.3': - resolution: {integrity: sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==} - engines: {node: '>=8'} - - '@jest/console@29.7.0': - resolution: {integrity: sha512-5Ni4CU7XHQi32IJ398EEP4RrB8eV09sXP2ROqD4bksHrnTree52PsxvX8tpL8LvTZ3pFzXyPbNQReSN41CAhOg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/core@29.7.0': - resolution: {integrity: sha512-n7aeXWKMnGtDA48y8TLWJPJmLmmZ642Ceo78cYWEpiD7FzDgmNDV/GCVRorPABdXLJZ/9wzzgZAlHjXjxDHGsg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - peerDependencies: - node-notifier: ^8.0.1 || ^9.0.0 || ^10.0.0 - peerDependenciesMeta: - node-notifier: - optional: true - - '@jest/environment@29.7.0': - resolution: {integrity: sha512-aQIfHDq33ExsN4jP1NWGXhxgQ/wixs60gDiKO+XVMd8Mn0NWPWgc34ZQDTb2jKaUWQ7MuwoitXAsN2XVXNMpAw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/expect-utils@29.7.0': - resolution: {integrity: sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/expect@29.7.0': - resolution: {integrity: sha512-8uMeAMycttpva3P1lBHB8VciS9V0XAr3GymPpipdyQXbBcuhkLQOSe8E/p92RyAdToS6ZD1tFkX+CkhoECE0dQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/fake-timers@29.7.0': - resolution: {integrity: sha512-q4DH1Ha4TTFPdxLsqDXK1d3+ioSL7yL5oCMJZgDYm6i+6CygW5E5xVr/D1HdsGxjt1ZWSfUAs9OxSB/BNelWrQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/globals@29.7.0': - resolution: {integrity: sha512-mpiz3dutLbkW2MNFubUGUEVLkTGiqW6yLVTA+JbP6fI6J5iL9Y0Nlg8k95pcF8ctKwCS7WVxteBs29hhfAotzQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/reporters@29.7.0': - resolution: {integrity: sha512-DApq0KJbJOEzAFYjHADNNxAE3KbhxQB1y5Kplb5Waqw6zVbuWatSnMjE5gs8FUgEPmNsnZA3NCWl9NG0ia04Pg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - peerDependencies: - node-notifier: ^8.0.1 || ^9.0.0 || ^10.0.0 - peerDependenciesMeta: - node-notifier: - optional: true - '@jest/schemas@29.6.3': resolution: {integrity: sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==} engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - '@jest/source-map@29.6.3': - resolution: {integrity: sha512-MHjT95QuipcPrpLM+8JMSzFx6eHp5Bm+4XeFDJlwsvVBjmKNiIAvasGK2fxz2WbGRlnvqehFbh07MMa7n3YJnw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/test-result@29.7.0': - resolution: {integrity: sha512-Fdx+tv6x1zlkJPcWXmMDAG2HBnaR9XPSd5aDWQVsfrZmLVT3lU1cwyxLgRmXR9yrq4NBoEm9BMsfgFzTQAbJYA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/test-sequencer@29.7.0': - resolution: {integrity: sha512-GQwJ5WZVrKnOJuiYiAF52UNUJXgTZx1NHjFSEB0qEMmSZKAkdMoIzw/Cj6x6NF4AvV23AUqDpFzQkN/eYCYTxw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/transform@29.7.0': - resolution: {integrity: sha512-ok/BTPFzFKVMwO5eOHRrvnBVHdRy9IrsrW1GpMaQ9MCnilNLXQKmAX8s1YXDFaai9xJpac2ySzV0YeRRECr2Vw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jest/types@29.6.3': - resolution: {integrity: sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - '@jridgewell/gen-mapping@0.3.5': - resolution: {integrity: sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==} - engines: {node: '>=6.0.0'} - - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} - - '@jridgewell/set-array@1.2.1': - resolution: {integrity: sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==} - engines: {node: '>=6.0.0'} - '@jridgewell/sourcemap-codec@1.4.15': resolution: {integrity: sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==} - '@jridgewell/trace-mapping@0.3.25': - resolution: {integrity: sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==} - '@juggle/resize-observer@3.4.0': resolution: {integrity: sha512-dfLbk+PwWvFzSxwk3n5ySL0hfBog779o8h68wK/7/APo/7cgyWp5jcXockbxdk5kFRkbeXWm4Fbi9FrdN381sA==} @@ -1117,14 +996,88 @@ packages: resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} engines: {node: '>= 8'} - '@sinclair/typebox@0.27.8': - resolution: {integrity: sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==} + '@rollup/rollup-android-arm-eabi@4.18.0': + resolution: {integrity: sha512-Tya6xypR10giZV1XzxmH5wr25VcZSncG0pZIjfePT0OVBvqNEurzValetGNarVrGiq66EBVAFn15iYX4w6FKgQ==} + cpu: [arm] + os: [android] + + '@rollup/rollup-android-arm64@4.18.0': + resolution: {integrity: sha512-avCea0RAP03lTsDhEyfy+hpfr85KfyTctMADqHVhLAF3MlIkq83CP8UfAHUssgXTYd+6er6PaAhx/QGv4L1EiA==} + cpu: [arm64] + os: [android] + + '@rollup/rollup-darwin-arm64@4.18.0': + resolution: {integrity: sha512-IWfdwU7KDSm07Ty0PuA/W2JYoZ4iTj3TUQjkVsO/6U+4I1jN5lcR71ZEvRh52sDOERdnNhhHU57UITXz5jC1/w==} + cpu: [arm64] + os: [darwin] + + '@rollup/rollup-darwin-x64@4.18.0': + resolution: {integrity: sha512-n2LMsUz7Ynu7DoQrSQkBf8iNrjOGyPLrdSg802vk6XT3FtsgX6JbE8IHRvposskFm9SNxzkLYGSq9QdpLYpRNA==} + cpu: [x64] + os: [darwin] - '@sinonjs/commons@3.0.1': - resolution: {integrity: sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==} + '@rollup/rollup-linux-arm-gnueabihf@4.18.0': + resolution: {integrity: sha512-C/zbRYRXFjWvz9Z4haRxcTdnkPt1BtCkz+7RtBSuNmKzMzp3ZxdM28Mpccn6pt28/UWUCTXa+b0Mx1k3g6NOMA==} + cpu: [arm] + os: [linux] + + '@rollup/rollup-linux-arm-musleabihf@4.18.0': + resolution: {integrity: sha512-l3m9ewPgjQSXrUMHg93vt0hYCGnrMOcUpTz6FLtbwljo2HluS4zTXFy2571YQbisTnfTKPZ01u/ukJdQTLGh9A==} + cpu: [arm] + os: [linux] + + '@rollup/rollup-linux-arm64-gnu@4.18.0': + resolution: {integrity: sha512-rJ5D47d8WD7J+7STKdCUAgmQk49xuFrRi9pZkWoRD1UeSMakbcepWXPF8ycChBoAqs1pb2wzvbY6Q33WmN2ftw==} + cpu: [arm64] + os: [linux] + + '@rollup/rollup-linux-arm64-musl@4.18.0': + resolution: {integrity: sha512-be6Yx37b24ZwxQ+wOQXXLZqpq4jTckJhtGlWGZs68TgdKXJgw54lUUoFYrg6Zs/kjzAQwEwYbp8JxZVzZLRepQ==} + cpu: [arm64] + os: [linux] + + '@rollup/rollup-linux-powerpc64le-gnu@4.18.0': + resolution: {integrity: sha512-hNVMQK+qrA9Todu9+wqrXOHxFiD5YmdEi3paj6vP02Kx1hjd2LLYR2eaN7DsEshg09+9uzWi2W18MJDlG0cxJA==} + cpu: [ppc64] + os: [linux] + + '@rollup/rollup-linux-riscv64-gnu@4.18.0': + resolution: {integrity: sha512-ROCM7i+m1NfdrsmvwSzoxp9HFtmKGHEqu5NNDiZWQtXLA8S5HBCkVvKAxJ8U+CVctHwV2Gb5VUaK7UAkzhDjlg==} + cpu: [riscv64] + os: [linux] + + '@rollup/rollup-linux-s390x-gnu@4.18.0': + resolution: {integrity: sha512-0UyyRHyDN42QL+NbqevXIIUnKA47A+45WyasO+y2bGJ1mhQrfrtXUpTxCOrfxCR4esV3/RLYyucGVPiUsO8xjg==} + cpu: [s390x] + os: [linux] + + '@rollup/rollup-linux-x64-gnu@4.18.0': + resolution: {integrity: sha512-xuglR2rBVHA5UsI8h8UbX4VJ470PtGCf5Vpswh7p2ukaqBGFTnsfzxUBetoWBWymHMxbIG0Cmx7Y9qDZzr648w==} + cpu: [x64] + os: [linux] + + '@rollup/rollup-linux-x64-musl@4.18.0': + resolution: {integrity: sha512-LKaqQL9osY/ir2geuLVvRRs+utWUNilzdE90TpyoX0eNqPzWjRm14oMEE+YLve4k/NAqCdPkGYDaDF5Sw+xBfg==} + cpu: [x64] + os: [linux] + + '@rollup/rollup-win32-arm64-msvc@4.18.0': + resolution: {integrity: sha512-7J6TkZQFGo9qBKH0pk2cEVSRhJbL6MtfWxth7Y5YmZs57Pi+4x6c2dStAUvaQkHQLnEQv1jzBUW43GvZW8OFqA==} + cpu: [arm64] + os: [win32] + + '@rollup/rollup-win32-ia32-msvc@4.18.0': + resolution: {integrity: sha512-Txjh+IxBPbkUB9+SXZMpv+b/vnTEtFyfWZgJ6iyCmt2tdx0OF5WhFowLmnh8ENGNpfUlUZkdI//4IEmhwPieNg==} + cpu: [ia32] + os: [win32] + + '@rollup/rollup-win32-x64-msvc@4.18.0': + resolution: {integrity: sha512-UOo5FdvOL0+eIVTgS4tIdbW+TtnBLWg1YBCcU2KWM7nuNwRz9bksDX1bekJJCpu25N1DVWaCwnT39dVQxzqS8g==} + cpu: [x64] + os: [win32] - '@sinonjs/fake-timers@10.3.0': - resolution: {integrity: sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==} + '@sinclair/typebox@0.27.8': + resolution: {integrity: sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==} '@textcomplete/core@0.1.13': resolution: {integrity: sha512-C4S+ihQU5HsKQ/TbsmS0e7hfPZtLZbEXj5NDUgRnhu/1Nezpu892bjNZGeErZm+R8iyDIT6wDu6EgIhng4M8eQ==} @@ -1143,18 +1096,6 @@ packages: '@types/audioworklet@0.0.54': resolution: {integrity: sha512-WR1XcwT2LhCaUiKDDgHdTjrVjoBZnTz6FhszeIKgY9i2UYfIRKtnNvqToUDnbCPXBpVuu4Qo5+mMJt+wDphRew==} - '@types/babel__core@7.20.5': - resolution: {integrity: sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==} - - '@types/babel__generator@7.6.8': - resolution: {integrity: sha512-ASsj+tpEDsEiFr1arWrlN6V3mdfjRMZt6LtK/Vp/kreFLnr5QH5+DhvD5nINYZXzwJvXeGq+05iUXcAzVrqWtw==} - - '@types/babel__template@7.4.4': - resolution: {integrity: sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==} - - '@types/babel__traverse@7.20.5': - resolution: {integrity: sha512-WXCyOcRtH37HAUkpXhUduaxdm82b4GSlyTqajXviN4EfiuPgNYR109xMCKvpl6zPIpua0DGlMEDCq+g8EdoheQ==} - '@types/chess.js@0.10.1': resolution: {integrity: sha512-cuFMyrUYOWpKm5PRayvk22jLV3BOuWs7iJ7Q4PZJXMWoF7uhjWEojMS4GPk441gn4Ki9BHxbKsll7mrwJ3KT3g==} @@ -1164,24 +1105,12 @@ packages: '@types/dragscroll@0.0.0': resolution: {integrity: sha512-rWr/ryzOUi9r/zUA2GK2qLWGBIBmDeIojBQXuvR76pulHUoEGMJ2A7NWShUaA5AE90ha+l9tlsyGz2UioQE9cg==} + '@types/estree@1.0.5': + resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} + '@types/fnando__sparkline@0.3.7': resolution: {integrity: sha512-irYrS2clNGbnKBBIBAulPH6hDZ/HlBkNsDbYPfO8B2obcxtC9UzKFS5IBSzrUtfKbwf8U01xHj+5iEJ7TtY04Q==} - '@types/graceful-fs@4.1.9': - resolution: {integrity: sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==} - - '@types/istanbul-lib-coverage@2.0.6': - resolution: {integrity: sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==} - - '@types/istanbul-lib-report@3.0.3': - resolution: {integrity: sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA==} - - '@types/istanbul-reports@3.0.4': - resolution: {integrity: sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ==} - - '@types/jest@29.5.12': - resolution: {integrity: sha512-eDC8bTvT/QhYdxJAulQikueigY5AsdBRH2yDKW3yveW7svY3+DzN84/2NUgkw10RTiJbWqZrTtoGVdYlvFJdLw==} - '@types/json-schema@7.0.15': resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} @@ -1203,9 +1132,6 @@ packages: '@types/sortablejs@1.15.8': resolution: {integrity: sha512-b79830lW+RZfwaztgs1aVPgbasJ8e7AXtZYHTELNXZPsERt4ymJdjV4OccDbHQAvHrCcFpbF78jkm0R6h/pZVg==} - '@types/stack-utils@2.0.3': - resolution: {integrity: sha512-9aEbYZ3TbYMznPdcdr3SmIrLXwC/AKZXQeCf9Pgao5CKb8CyHuEX5jzWPTkvregvhRJHcpRO6BFoGW9ycaOkYw==} - '@types/web@0.0.142': resolution: {integrity: sha512-QWDvMW+P3sdq8rhiw3dNyBR64O5adSY80gBjRd2xI3BADGsAFpAglblWucsGUjKVKyPm4EbYZkvFs7BRxPsBOg==} @@ -1215,12 +1141,6 @@ packages: '@types/yaireo__tagify@4.17.5': resolution: {integrity: sha512-nYk4xqky1ZnbgTlP7dO24GA/Kz4Q7mvNGNOsRPygnaVV8kBXIhROFVLG141Y0EF+TZKbmIMUolaJ+Z69Pr9FwQ==} - '@types/yargs-parser@21.0.3': - resolution: {integrity: sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==} - - '@types/yargs@17.0.32': - resolution: {integrity: sha512-xQ67Yc/laOG5uMfX/093MRlGGCIBzZMarVa+gfNKJxWAIgykYpVGkBdbqEzGDDfCrVUj6Hiff4mTZ5BA6TmAog==} - '@types/zxcvbn@4.4.4': resolution: {integrity: sha512-Tuk4q7q0DnpzyJDI4aMeghGuFu2iS1QAdKpabn8JfbtfGmVDUgvZv1I7mEjP61Bvnp3ljKCC8BE6YYSTNxmvRQ==} @@ -1285,6 +1205,21 @@ packages: '@ungap/structured-clone@1.2.0': resolution: {integrity: sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==} + '@vitest/expect@1.6.0': + resolution: {integrity: sha512-ixEvFVQjycy/oNgHjqsL6AZCDduC+tflRluaHIzKIsdbzkLn2U/iBnVeJwB6HsIjQBdfMR8Z0tRxKUsvFJEeWQ==} + + '@vitest/runner@1.6.0': + resolution: {integrity: sha512-P4xgwPjwesuBiHisAVz/LSSZtDjOTPYZVmNAnpHHSR6ONrf8eCJOFRvUwdHn30F5M1fxhqtl7QZQUk2dprIXAg==} + + '@vitest/snapshot@1.6.0': + resolution: {integrity: sha512-+Hx43f8Chus+DCmygqqfetcAZrDJwvTj0ymqjQq4CvmpKFSTVteEOBzCusu1x2tt4OJcvBflyHUE0DZSLgEMtQ==} + + '@vitest/spy@1.6.0': + resolution: {integrity: sha512-leUTap6B/cqi/bQkXUu6bQV5TZPx7pmMBKBQiI0rJA8c3pB56ZsaTbREnF7CJfmvAS4V2cXIBAh/3rVwrrCYgw==} + + '@vitest/utils@1.6.0': + resolution: {integrity: sha512-21cPiuGMoMZwiOHa2i4LXkMkMkCGzA+MVFV70jRwHo95dL4x/ts5GZhML1QWuy7yfp3WzK3lRvZi3JnXTYqrBw==} + '@yaireo/tagify@4.17.9': resolution: {integrity: sha512-x9aZy22hzte7BNmMrFcYNrZH71ombgH5PnzcOVXqPevRV/m/ItSnWIvY5fOHYzpC9Uxy0+h/1P5v62fIvwq2MA==} engines: {node: '>=14.20.0', npm: '>=8.0.0'} @@ -1303,18 +1238,22 @@ packages: peerDependencies: acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 + acorn-walk@8.3.2: + resolution: {integrity: sha512-cjkyv4OtNCIeqhHrfS81QWXoCBPExR/J62oyEqepVw8WaQeSqpW2uhuLPh1m9eWhDuOo/jUXVTlifvesOWp/4A==} + engines: {node: '>=0.4.0'} + acorn@8.11.3: resolution: {integrity: sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==} engines: {node: '>=0.4.0'} hasBin: true + agent-base@7.1.1: + resolution: {integrity: sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==} + engines: {node: '>= 14'} + ajv@6.12.6: resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} - ansi-escapes@4.3.2: - resolution: {integrity: sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==} - engines: {node: '>=8'} - ansi-escapes@6.2.1: resolution: {integrity: sha512-4nJ3yixlEthEJ9Rk4vPcdBRkZvQZlYyu8j4/Mqz5sgIkddmEnH2Yj2ZrnP9S3tQOvSNRUIgVNF/1yPpRAGNRig==} engines: {node: '>=14.16'} @@ -1327,10 +1266,6 @@ packages: resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} engines: {node: '>=12'} - ansi-styles@3.2.1: - resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} - engines: {node: '>=4'} - ansi-styles@4.3.0: resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} engines: {node: '>=8'} @@ -1353,9 +1288,6 @@ packages: arg@4.1.3: resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} - argparse@1.0.10: - resolution: {integrity: sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==} - argparse@2.0.1: resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} @@ -1363,30 +1295,11 @@ packages: resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} engines: {node: '>=8'} - babel-jest@29.7.0: - resolution: {integrity: sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - peerDependencies: - '@babel/core': ^7.8.0 - - babel-plugin-istanbul@6.1.1: - resolution: {integrity: sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==} - engines: {node: '>=8'} - - babel-plugin-jest-hoist@29.6.3: - resolution: {integrity: sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - babel-preset-current-node-syntax@1.0.1: - resolution: {integrity: sha512-M7LQ0bxarkxQoN+vz5aJPsLBn77n8QgTFmo8WK0/44auK2xlCXrYcUxHFxgU7qW5Yzw/CjmLRK2uJzaCd7LvqQ==} - peerDependencies: - '@babel/core': ^7.0.0 + assertion-error@1.1.0: + resolution: {integrity: sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==} - babel-preset-jest@29.6.3: - resolution: {integrity: sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - peerDependencies: - '@babel/core': ^7.0.0 + asynckit@0.4.0: + resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} balanced-match@1.0.2: resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} @@ -1405,16 +1318,9 @@ packages: resolution: {integrity: sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==} engines: {node: '>=8'} - browserslist@4.23.0: - resolution: {integrity: sha512-QW8HiM1shhT2GuzkvklfjcKDiWFXHOeFCIA/huJPwHsslwcydgk7X+z2zXpEijP98UCY7HbubZt5J2Zgvf0CaQ==} - engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7} - hasBin: true - - bser@2.1.1: - resolution: {integrity: sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==} - - buffer-from@1.1.2: - resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} + cac@6.7.14: + resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} + engines: {node: '>=8'} callsites@3.1.0: resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} @@ -1424,15 +1330,8 @@ packages: resolution: {integrity: sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==} engines: {node: '>=6'} - camelcase@6.3.0: - resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} - engines: {node: '>=10'} - - caniuse-lite@1.0.30001605: - resolution: {integrity: sha512-nXwGlFWo34uliI9z3n6Qc0wZaf7zaZWA1CPZ169La5mV3I/gem7bst0vr5XQH5TJXZIMfDeZyOrZnSlVzKxxHQ==} - - chalk@2.4.2: - resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} + chai@4.4.1: + resolution: {integrity: sha512-13sOfMv2+DWduEU+/xbun3LScLoqN17nBeTLUsmDfKdoiC1fr0n9PU4guu4AhRcOVFk/sW8LyZWHuhWtQZiF+g==} engines: {node: '>=4'} chalk@4.1.2: @@ -1443,10 +1342,6 @@ packages: resolution: {integrity: sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==} engines: {node: ^12.17.0 || ^14.13 || >=16.0.0} - char-regex@1.0.2: - resolution: {integrity: sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==} - engines: {node: '>=10'} - chart.js@4.4.0: resolution: {integrity: sha512-vQEj6d+z0dcsKLlQvbKIMYFHd3t8W/7L2vfJIbYcfyPcRx92CsHqECpueN8qVGNlKyDcr5wBrYAYKnfu/9Q1hQ==} engines: {pnpm: '>=7'} @@ -1468,6 +1363,9 @@ packages: peerDependencies: chart.js: '>=3.2.0' + check-error@1.0.3: + resolution: {integrity: sha512-iKEoDYaRmd1mxM90a2OEfWhjsjPpYPuQ+lMYsoxB126+t8fw7ySEO48nmDg5COTjxDI65/Y2OWpeEHk3ZOe8zg==} + chess.js@https://codeload.github.com/ornicar/chess.js/tar.gz/ad0709c0b07773d9d0da8e4605a9a2a28f00d249: resolution: {tarball: https://codeload.github.com/ornicar/chess.js/tar.gz/ad0709c0b07773d9d0da8e4605a9a2a28f00d249} version: 0.10.2 @@ -1485,13 +1383,6 @@ packages: resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} engines: {node: '>= 8.10.0'} - ci-info@3.9.0: - resolution: {integrity: sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==} - engines: {node: '>=8'} - - cjs-module-lexer@1.2.3: - resolution: {integrity: sha512-0TNiGstbQmCFwt4akjjBg5pLRTSyj/PkWQ1ZoO2zntmg9yLqSRxwEa4iCfQLGjqhiqBfOJa7W/E8wfGrTDmlZQ==} - cli-cursor@4.0.0: resolution: {integrity: sha512-VGtlMu3x/4DOtIUwEkRezxUZ2lBacNJCHash0N0WeZDBS+7Ux1dm3XWAgWYxLJFMMdOeXMHXorshEFhbMSGelg==} engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} @@ -1503,33 +1394,20 @@ packages: cliui@6.0.0: resolution: {integrity: sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==} - cliui@8.0.1: - resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} - engines: {node: '>=12'} - - co@4.6.0: - resolution: {integrity: sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==} - engines: {iojs: '>= 1.0.0', node: '>= 0.12.0'} - - collect-v8-coverage@1.0.2: - resolution: {integrity: sha512-lHl4d5/ONEbLlJvaJNtsF/Lz+WvB07u2ycqTYbdrq7UypDXailES4valYb2eWiJFxZlVmpGekfqoxQhzyFdT4Q==} - - color-convert@1.9.3: - resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} - color-convert@2.0.1: resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} engines: {node: '>=7.0.0'} - color-name@1.1.3: - resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} - color-name@1.1.4: resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} colorette@2.0.20: resolution: {integrity: sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==} + combined-stream@1.0.8: + resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} + engines: {node: '>= 0.8'} + commander@11.1.0: resolution: {integrity: sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ==} engines: {node: '>=16'} @@ -1537,13 +1415,8 @@ packages: concat-map@0.0.1: resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} - convert-source-map@2.0.0: - resolution: {integrity: sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==} - - create-jest@29.7.0: - resolution: {integrity: sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - hasBin: true + confbox@0.1.7: + resolution: {integrity: sha512-uJcB/FKZtBMCJpK8MQji6bJHgu1tixKPxRLeGkNzBoOZzpnZUJm0jm2/sBDWcuBx1dYgxV4JU+g5hmNxCyAmdA==} cropperjs@1.6.1: resolution: {integrity: sha512-F4wsi+XkDHCOMrHMYjrTEE4QBOrsHHN5/2VsVAaRq8P7E5z7xQpT75S+f/9WikmBEailas3+yo+6zPIomW+NOA==} @@ -1552,9 +1425,17 @@ packages: resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} engines: {node: '>= 8'} + cssstyle@4.0.1: + resolution: {integrity: sha512-8ZYiJ3A/3OkDd093CBT/0UKDWry7ak4BdPTFP2+QEP7cmhouyq/Up709ASSj2cK02BbZiMgk7kYjZNS4QP5qrQ==} + engines: {node: '>=18'} + csstype@3.1.3: resolution: {integrity: sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==} + data-urls@5.0.0: + resolution: {integrity: sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg==} + engines: {node: '>=18'} + date-fns@2.29.3: resolution: {integrity: sha512-dDCnyH2WnnKusqvZZ6+jA1O51Ibt8ZMRNkDZdyAyK4YfbDwa/cEmuztzG5pk6hqlp9aSBPYcjOlktquahGwGeA==} engines: {node: '>=0.11'} @@ -1578,13 +1459,12 @@ packages: resolution: {integrity: sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==} engines: {node: '>=0.10.0'} - dedent@1.5.1: - resolution: {integrity: sha512-+LxW+KLWxu3HW3M2w2ympwtqPrqYRzU8fqi6Fhd18fBALe15blJPI/I4+UHveMVG6lJqB4JNd4UG0S5cnVHwIg==} - peerDependencies: - babel-plugin-macros: ^3.1.0 - peerDependenciesMeta: - babel-plugin-macros: - optional: true + decimal.js@10.4.3: + resolution: {integrity: sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==} + + deep-eql@4.1.4: + resolution: {integrity: sha512-SUwdGfqdKOwxCPeVYjwSyRpJ7Z+fhpwIAtmCUdZIWZ/YP5R9WAsyuSgpLVDi9bjWoN2LXHNss/dk3urXtdQxGg==} + engines: {node: '>=6'} deep-is@0.1.4: resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} @@ -1593,9 +1473,9 @@ packages: resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} engines: {node: '>=0.10.0'} - detect-newline@3.1.0: - resolution: {integrity: sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==} - engines: {node: '>=8'} + delayed-stream@1.0.0: + resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} + engines: {node: '>=0.4.0'} dialog-polyfill@0.5.6: resolution: {integrity: sha512-ZbVDJI9uvxPAKze6z146rmfUZjBqNEwcnFTVamQzXH+svluiV7swmVIGr7miwADgfgt1G2JQIytypM9fbyhX4w==} @@ -1621,13 +1501,6 @@ packages: dragscroll@0.0.8: resolution: {integrity: sha512-nMrx/KErHpEkiKZlrghcT/nLWCj8vEJgv6s6TF84gmgn6uROPx2wRvClkcnjSEyvppYY9okOI1DIv573Toql1w==} - electron-to-chromium@1.4.723: - resolution: {integrity: sha512-rxFVtrMGMFROr4qqU6n95rUi9IlfIm+lIAt+hOToy/9r6CDv0XiEcQdC3VP71y1pE5CFTzKV0RvxOGYCPWWHPw==} - - emittery@0.13.1: - resolution: {integrity: sha512-DeWwawk6r5yR9jFgnDKYt4sLS0LmHJJi3ZOnb5/JdbYwj3nW+FxQnHIjhBKz8YLC7oRNPVM9NQ47I3CVx34eqQ==} - engines: {node: '>=12'} - emoji-mart@5.5.2: resolution: {integrity: sha512-Sqc/nso4cjxhOwWJsp9xkVm8OF5c+mJLZJFoFfzRuKO+yWiN7K8c96xmtughYb0d/fZ8UC6cLIQ/p4BR6Pv3/A==} @@ -1640,20 +1513,14 @@ packages: encode-utf8@1.0.3: resolution: {integrity: sha512-ucAnuBEhUK4boH2HjVYG5Q2mQyPorvv0u/ocS+zhdw0S8AlHYY+GOFhP1Gio5z4icpP2ivFSvhtFjQi8+T9ppw==} - error-ex@1.3.2: - resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} - - escalade@3.1.2: - resolution: {integrity: sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==} - engines: {node: '>=6'} - - escape-string-regexp@1.0.5: - resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} - engines: {node: '>=0.8.0'} + entities@4.5.0: + resolution: {integrity: sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==} + engines: {node: '>=0.12'} - escape-string-regexp@2.0.0: - resolution: {integrity: sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==} - engines: {node: '>=8'} + esbuild@0.21.5: + resolution: {integrity: sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==} + engines: {node: '>=12'} + hasBin: true escape-string-regexp@4.0.0: resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} @@ -1676,11 +1543,6 @@ packages: resolution: {integrity: sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} - esprima@4.0.1: - resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==} - engines: {node: '>=4'} - hasBin: true - esquery@1.5.0: resolution: {integrity: sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==} engines: {node: '>=0.10'} @@ -1693,6 +1555,9 @@ packages: resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} engines: {node: '>=4.0'} + estree-walker@3.0.3: + resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} + esutils@2.0.3: resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} engines: {node: '>=0.10.0'} @@ -1703,22 +1568,10 @@ packages: eventemitter3@5.0.1: resolution: {integrity: sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==} - execa@5.1.1: - resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==} - engines: {node: '>=10'} - execa@8.0.1: resolution: {integrity: sha512-VyhnebXciFV2DESc+p6B+y0LjSm0krU4OgJN44qFAhBY0TJ+1V61tYD2+wHusZ6F9n5K+vl8k0sTy7PEfV4qpg==} engines: {node: '>=16.17'} - exit@0.1.2: - resolution: {integrity: sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==} - engines: {node: '>= 0.8.0'} - - expect@29.7.0: - resolution: {integrity: sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - fast-deep-equal@3.1.3: resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} @@ -1735,9 +1588,6 @@ packages: fastq@1.17.1: resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} - fb-watchman@2.0.2: - resolution: {integrity: sha512-p5161BqbuCaSnB8jIbzQHOlpgsPmK5rJVDfDKO91Axs5NC1uu3HRQm6wt9cd9/+GtQQIO53JdGXXoyDpTAsgYA==} - file-entry-cache@6.0.1: resolution: {integrity: sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==} engines: {node: ^10.12.0 || >=12.0.0} @@ -1764,6 +1614,10 @@ packages: flatted@3.3.1: resolution: {integrity: sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==} + form-data@4.0.0: + resolution: {integrity: sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==} + engines: {node: '>= 6'} + fs.realpath@1.0.0: resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} @@ -1772,13 +1626,6 @@ packages: engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} os: [darwin] - function-bind@1.1.2: - resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - - gensync@1.0.0-beta.2: - resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==} - engines: {node: '>=6.9.0'} - get-caller-file@2.0.5: resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} engines: {node: 6.* || 8.* || >= 10.*} @@ -1787,13 +1634,8 @@ packages: resolution: {integrity: sha512-2nk+7SIVb14QrgXFHcm84tD4bKQz0RxPuMT8Ag5KPOq7J5fEmAg0UbXdTOSHqNuHSU28k55qnceesxXRZGzKWA==} engines: {node: '>=18'} - get-package-type@0.1.0: - resolution: {integrity: sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==} - engines: {node: '>=8.0.0'} - - get-stream@6.0.1: - resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} - engines: {node: '>=10'} + get-func-name@2.0.2: + resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} get-stream@8.0.1: resolution: {integrity: sha512-VaUJspBffn/LMCJVoMvSAdmscJyS1auj5Zulnn5UoYcY531UWmdwhRWkcGKnGU93m5HSXP9LP2usOryrBtQowA==} @@ -1810,10 +1652,6 @@ packages: glob@7.2.3: resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} - globals@11.12.0: - resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==} - engines: {node: '>=4'} - globals@13.24.0: resolution: {integrity: sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==} engines: {node: '>=8'} @@ -1822,9 +1660,6 @@ packages: resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} engines: {node: '>=10'} - graceful-fs@4.2.11: - resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} - graphemer@1.4.0: resolution: {integrity: sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==} @@ -1832,29 +1667,30 @@ packages: resolution: {integrity: sha512-tSQXBXS/MWQOn/RKckawJ61vvsDpCom87JgxiYdGwHdOa0ht0vzUWDlfioofFCRU0L+6NGDt6XzbgoJvZkMeRQ==} engines: {node: '>=0.8.0'} - has-flag@3.0.0: - resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} - engines: {node: '>=4'} - has-flag@4.0.0: resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} engines: {node: '>=8'} - hasown@2.0.2: - resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} - engines: {node: '>= 0.4'} + html-encoding-sniffer@4.0.0: + resolution: {integrity: sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==} + engines: {node: '>=18'} - html-escaper@2.0.2: - resolution: {integrity: sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==} + http-proxy-agent@7.0.2: + resolution: {integrity: sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==} + engines: {node: '>= 14'} - human-signals@2.1.0: - resolution: {integrity: sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==} - engines: {node: '>=10.17.0'} + https-proxy-agent@7.0.4: + resolution: {integrity: sha512-wlwpilI7YdjSkWaQ/7omYBMTliDcmCN8OLihO6I9B86g06lMyAoqgoDpV0XqoaPOKj+0DIdAvnsWfyAAhmimcg==} + engines: {node: '>= 14'} human-signals@5.0.0: resolution: {integrity: sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ==} engines: {node: '>=16.17.0'} + iconv-lite@0.6.3: + resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==} + engines: {node: '>=0.10.0'} + idb-keyval@6.2.1: resolution: {integrity: sha512-8Sb3veuYCyrZL+VBt9LJfZjLUPWVvqn8tG28VqYNFCo43KHcKuq+b4EiXGeuaLAQWL2YmyDgMp2aSpH9JHsEQg==} @@ -1866,11 +1702,6 @@ packages: resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} engines: {node: '>=6'} - import-local@3.1.0: - resolution: {integrity: sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==} - engines: {node: '>=8'} - hasBin: true - imurmurhash@0.1.4: resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} engines: {node: '>=0.8.19'} @@ -1881,16 +1712,10 @@ packages: inherits@2.0.4: resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - is-arrayish@0.2.1: - resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} - is-binary-path@2.1.0: resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} engines: {node: '>=8'} - is-core-module@2.13.1: - resolution: {integrity: sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==} - is-extglob@2.1.1: resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} engines: {node: '>=0.10.0'} @@ -1907,10 +1732,6 @@ packages: resolution: {integrity: sha512-OVa3u9kkBbw7b8Xw5F9P+D/T9X+Z4+JruYVNapTjPYZYUznQ5YfWeFkOj606XYYW8yugTfC8Pj0hYqvi4ryAhA==} engines: {node: '>=18'} - is-generator-fn@2.1.0: - resolution: {integrity: sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==} - engines: {node: '>=6'} - is-glob@4.0.3: resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} engines: {node: '>=0.10.0'} @@ -1923,9 +1744,8 @@ packages: resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==} engines: {node: '>=8'} - is-stream@2.0.1: - resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} - engines: {node: '>=8'} + is-potential-custom-element-name@1.0.1: + resolution: {integrity: sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==} is-stream@3.0.0: resolution: {integrity: sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA==} @@ -1934,180 +1754,27 @@ packages: isexe@2.0.0: resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - istanbul-lib-coverage@3.2.2: - resolution: {integrity: sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==} - engines: {node: '>=8'} - - istanbul-lib-instrument@5.2.1: - resolution: {integrity: sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==} - engines: {node: '>=8'} + js-tokens@4.0.0: + resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - istanbul-lib-instrument@6.0.2: - resolution: {integrity: sha512-1WUsZ9R1lA0HtBSohTkm39WTPlNKSJ5iFk7UwqXkBLoHQT+hfqPsfsTDVuZdKGaBwn7din9bS7SsnoAr943hvw==} - engines: {node: '>=10'} + js-tokens@9.0.0: + resolution: {integrity: sha512-WriZw1luRMlmV3LGJaR6QOJjWwgLUTf89OwT2lUOyjX2dJGBwgmIkbcz+7WFZjrZM635JOIR517++e/67CP9dQ==} - istanbul-lib-report@3.0.1: - resolution: {integrity: sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==} - engines: {node: '>=10'} + js-yaml@4.1.0: + resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} + hasBin: true - istanbul-lib-source-maps@4.0.1: - resolution: {integrity: sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==} - engines: {node: '>=10'} + jsdom@24.1.0: + resolution: {integrity: sha512-6gpM7pRXCwIOKxX47cgOyvyQDN/Eh0f1MeKySBV2xGdKtqJBLj8P25eY3EVCWo2mglDDzozR2r2MW4T+JiNUZA==} + engines: {node: '>=18'} + peerDependencies: + canvas: ^2.11.2 + peerDependenciesMeta: + canvas: + optional: true - istanbul-reports@3.1.7: - resolution: {integrity: sha512-BewmUXImeuRk2YY0PVbxgKAysvhRPUQE0h5QRM++nVWyubKGV0l8qQ5op8+B2DOmwSe63Jivj0BjkPQVf8fP5g==} - engines: {node: '>=8'} - - jest-changed-files@29.7.0: - resolution: {integrity: sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-circus@29.7.0: - resolution: {integrity: sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-cli@29.7.0: - resolution: {integrity: sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - hasBin: true - peerDependencies: - node-notifier: ^8.0.1 || ^9.0.0 || ^10.0.0 - peerDependenciesMeta: - node-notifier: - optional: true - - jest-config@29.7.0: - resolution: {integrity: sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - peerDependencies: - '@types/node': '*' - ts-node: '>=9.0.0' - peerDependenciesMeta: - '@types/node': - optional: true - ts-node: - optional: true - - jest-diff@29.7.0: - resolution: {integrity: sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-docblock@29.7.0: - resolution: {integrity: sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-each@29.7.0: - resolution: {integrity: sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-environment-node@29.7.0: - resolution: {integrity: sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-get-type@29.6.3: - resolution: {integrity: sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-haste-map@29.7.0: - resolution: {integrity: sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-leak-detector@29.7.0: - resolution: {integrity: sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-matcher-utils@29.7.0: - resolution: {integrity: sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-message-util@29.7.0: - resolution: {integrity: sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-mock@29.7.0: - resolution: {integrity: sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-pnp-resolver@1.2.3: - resolution: {integrity: sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==} - engines: {node: '>=6'} - peerDependencies: - jest-resolve: '*' - peerDependenciesMeta: - jest-resolve: - optional: true - - jest-regex-util@29.6.3: - resolution: {integrity: sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-resolve-dependencies@29.7.0: - resolution: {integrity: sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-resolve@29.7.0: - resolution: {integrity: sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-runner@29.7.0: - resolution: {integrity: sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-runtime@29.7.0: - resolution: {integrity: sha512-gUnLjgwdGqW7B4LvOIkbKs9WGbn+QLqRQQ9juC6HndeDiezIwhDP+mhMwHWCEcfQ5RUXa6OPnFF8BJh5xegwwQ==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-snapshot@29.7.0: - resolution: {integrity: sha512-Rm0BMWtxBcioHr1/OX5YCP8Uov4riHvKPknOGs804Zg9JGZgmIBkbtlxJC/7Z4msKYVbIJtfU+tKb8xlYNfdkw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-util@29.7.0: - resolution: {integrity: sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-validate@29.7.0: - resolution: {integrity: sha512-ZB7wHqaRGVw/9hST/OuFUReG7M8vKeq0/J2egIGLdvjHCmYqGARhzXmtgi+gVeZ5uXFF219aOc3Ls2yLg27tkw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-watcher@29.7.0: - resolution: {integrity: sha512-49Fg7WXkU3Vl2h6LbLtMQ/HyB6rXSIX7SqvBLQmssRBGN9I0PNvPmAmCWSOY6SOvrjhI/F7/bGAv9RtnsPA03g==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest-worker@29.7.0: - resolution: {integrity: sha512-eIz2msL/EzL9UFTFFx7jBTkeZfku0yUAyZZZmJ93H2TYEiroIx2PQjEXcwYtYl8zXCxb+PAmA2hLIt/6ZEkPHw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - - jest@29.7.0: - resolution: {integrity: sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==} - engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - hasBin: true - peerDependencies: - node-notifier: ^8.0.1 || ^9.0.0 || ^10.0.0 - peerDependenciesMeta: - node-notifier: - optional: true - - js-tokens@4.0.0: - resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} - - js-yaml@3.14.1: - resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} - hasBin: true - - js-yaml@4.1.0: - resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} - hasBin: true - - jsesc@2.5.2: - resolution: {integrity: sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==} - engines: {node: '>=4'} - hasBin: true - - json-buffer@3.0.1: - resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} - - json-parse-even-better-errors@2.3.1: - resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} + json-buffer@3.0.1: + resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} json-schema-traverse@0.4.1: resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} @@ -2115,22 +1782,9 @@ packages: json-stable-stringify-without-jsonify@1.0.1: resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} - json5@2.2.3: - resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} - engines: {node: '>=6'} - hasBin: true - keyv@4.5.4: resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} - kleur@3.0.3: - resolution: {integrity: sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==} - engines: {node: '>=6'} - - leven@3.1.0: - resolution: {integrity: sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==} - engines: {node: '>=6'} - levn@0.4.1: resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} engines: {node: '>= 0.8.0'} @@ -2145,9 +1799,6 @@ packages: resolution: {integrity: sha512-K2U4W2Ff5ibV7j7ydLr+zLAkIg5JJ4lPn1Ltsdt+Tz/IjQ8buJ55pZAxoP34lqIiwtF9iAvtLv3JGv7CAyAg+g==} engines: {node: '>=14'} - lines-and-columns@1.2.4: - resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} - lint-staged@15.2.2: resolution: {integrity: sha512-TiTt93OPh1OZOsb5B7k96A/ATl2AjIZo+vnzFZ6oHK5FuTk63ByDtxGQpHm+kFETjEWqgkF95M8FRXKR/LEBcw==} engines: {node: '>=18.12.0'} @@ -2157,6 +1808,10 @@ packages: resolution: {integrity: sha512-ovJXBXkKGfq+CwmKTjluEqFi3p4h8xvkxGQQAQan22YCgef4KZ1mKGjzfGh6PL6AW5Csw0QiQPNuQyH+6Xk3hA==} engines: {node: '>=18.0.0'} + local-pkg@0.5.0: + resolution: {integrity: sha512-ok6z3qlYyCDS4ZEU27HaU6x/xZa9Whf8jD4ptH5UZTQYZVYeb9bnZ3ojVhiJNLiXK1Hfc0GNbLXcmZ5plLDDBg==} + engines: {node: '>=14'} + locate-path@5.0.0: resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==} engines: {node: '>=8'} @@ -2176,19 +1831,15 @@ packages: resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} hasBin: true - lru-cache@5.1.1: - resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} + loupe@2.3.7: + resolution: {integrity: sha512-zSMINGVYkdpYSOBmLi0D1Uo7JU9nVdQKrHxC8eYlV+9YKK9WePqAlL7lSlorG/U2Fw1w0hTBmaa/jrQ3UbPHtA==} lru-cache@6.0.0: resolution: {integrity: sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==} engines: {node: '>=10'} - make-dir@4.0.0: - resolution: {integrity: sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==} - engines: {node: '>=10'} - - makeerror@1.0.12: - resolution: {integrity: sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==} + magic-string@0.30.10: + resolution: {integrity: sha512-iIRwTIf0QKV3UAnYK4PU8uiEc4SRh5jX0mwpIwETPpHdhVM4f53RSwS/vXvN1JhGX+Cs7B8qIq3d6AH49O5fAQ==} merge-stream@2.0.0: resolution: {integrity: sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==} @@ -2201,6 +1852,14 @@ packages: resolution: {integrity: sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==} engines: {node: '>=8.6'} + mime-db@1.52.0: + resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} + engines: {node: '>= 0.6'} + + mime-types@2.1.35: + resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} + engines: {node: '>= 0.6'} + mimic-fn@2.1.0: resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} engines: {node: '>=6'} @@ -2216,18 +1875,20 @@ packages: resolution: {integrity: sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==} engines: {node: '>=16 || 14 >=14.17'} + mlly@1.7.1: + resolution: {integrity: sha512-rrVRZRELyQzrIUAVMHxP97kv+G786pHmOKzuFII8zDYahFBS7qnHh2AlYSl1GAHhaMPCz6/oHjVMcfFYgFYHgA==} + ms@2.1.2: resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} + nanoid@3.3.7: + resolution: {integrity: sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==} + engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} + hasBin: true + natural-compare@1.4.0: resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} - node-int64@0.4.0: - resolution: {integrity: sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==} - - node-releases@2.0.14: - resolution: {integrity: sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==} - normalize-path@3.0.0: resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} engines: {node: '>=0.10.0'} @@ -2235,14 +1896,13 @@ packages: nouislider@15.7.1: resolution: {integrity: sha512-5N7C1ru/i8y3dg9+Z6ilj6+m1EfabvOoaRa7ztpxBSKKRZso4vA52DGSbBJjw5XLtFr/LZ9SgGAXqyVtlVHO5w==} - npm-run-path@4.0.1: - resolution: {integrity: sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==} - engines: {node: '>=8'} - npm-run-path@5.3.0: resolution: {integrity: sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ==} engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + nwsapi@2.2.10: + resolution: {integrity: sha512-QK0sRs7MKv0tKe1+5uZIQk/C8XGza4DAnztJG8iD+TpJIORARrCxczA738awHrZoHeTjSSoHqao2teO0dC/gFQ==} + object-assign@4.1.1: resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} engines: {node: '>=0.10.0'} @@ -2277,6 +1937,10 @@ packages: resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} engines: {node: '>=10'} + p-limit@5.0.0: + resolution: {integrity: sha512-/Eaoq+QyLSiXQ4lyYV23f14mZRQcXnxfHrN0vCai+ak9G0pp9iEQukIIZq5NccEvwRB8PUnZT0KsOoDCINS1qQ==} + engines: {node: '>=18'} + p-locate@4.1.0: resolution: {integrity: sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==} engines: {node: '>=8'} @@ -2293,9 +1957,8 @@ packages: resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} engines: {node: '>=6'} - parse-json@5.2.0: - resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} - engines: {node: '>=8'} + parse5@7.1.2: + resolution: {integrity: sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==} path-exists@4.0.0: resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} @@ -2313,13 +1976,16 @@ packages: resolution: {integrity: sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==} engines: {node: '>=12'} - path-parse@1.0.7: - resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} - path-type@4.0.0: resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} engines: {node: '>=8'} + pathe@1.1.2: + resolution: {integrity: sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ==} + + pathval@1.1.1: + resolution: {integrity: sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==} + peerjs-js-binarypack@2.1.0: resolution: {integrity: sha512-YIwCC+pTzp3Bi8jPI9UFKO0t0SLo6xALnHkiNt/iUFmUUZG0fEEmEyFKvjsDKweiFitzHRyhuh6NvyJZ4nNxMg==} engines: {node: '>= 14.0.0'} @@ -2340,18 +2006,17 @@ packages: engines: {node: '>=0.10'} hasBin: true - pirates@4.0.6: - resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} - engines: {node: '>= 6'} - - pkg-dir@4.2.0: - resolution: {integrity: sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==} - engines: {node: '>=8'} + pkg-types@1.1.1: + resolution: {integrity: sha512-ko14TjmDuQJ14zsotODv7dBlwxKhUKQEhuhmbqo1uCi9BB0Z2alo/wAXg6q1dTR5TyuqYyWhjtfe/Tsh+X28jQ==} pngjs@5.0.0: resolution: {integrity: sha512-40QW5YalBNfQo5yRYmiw7Yz6TKKVr3h6970B2YE+3fQpsWcrbj1PzJgxeJ19DRQjhMbKPIuMY8rFaXc8moolVw==} engines: {node: '>=10.13.0'} + postcss@8.4.38: + resolution: {integrity: sha512-Wglpdk03BSfXkHoQa3b/oulrotAkwrlLDRSOb9D0bN86FdRyE9lppSp33aHNPgBa0JKCoB+drFLZkQoRRYae5A==} + engines: {node: ^10 || ^12 || >=14} + prelude-ls@1.2.1: resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==} engines: {node: '>= 0.8.0'} @@ -2365,10 +2030,6 @@ packages: resolution: {integrity: sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==} engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} - prompts@2.4.2: - resolution: {integrity: sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==} - engines: {node: '>= 6'} - prop-types@15.8.1: resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} @@ -2396,18 +2057,21 @@ packages: prosemirror-view@1.33.3: resolution: {integrity: sha512-P4Ao/bc4OrU/2yLIf8dL4lJaEtjLR3QjIvQHgJYp2jUS7kYM4bSR6okbBjkqzOs/FwUon6UGjTLdKMnPL1MZqw==} + psl@1.9.0: + resolution: {integrity: sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==} + punycode@2.3.1: resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} engines: {node: '>=6'} - pure-rand@6.1.0: - resolution: {integrity: sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==} - qrcode@1.5.3: resolution: {integrity: sha512-puyri6ApkEHYiVl4CFzo1tDkAZ+ATcnbJrJ6RiBM1Fhctdn/ix9MTE3hRph33omisEbC/2fcfemsseiKgBPKZg==} engines: {node: '>=10.13.0'} hasBin: true + querystringify@2.2.0: + resolution: {integrity: sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==} + queue-microtask@1.2.3: resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} @@ -2428,26 +2092,13 @@ packages: require-main-filename@2.0.0: resolution: {integrity: sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==} - resolve-cwd@3.0.0: - resolution: {integrity: sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==} - engines: {node: '>=8'} + requires-port@1.0.0: + resolution: {integrity: sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==} resolve-from@4.0.0: resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} engines: {node: '>=4'} - resolve-from@5.0.0: - resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} - engines: {node: '>=8'} - - resolve.exports@2.0.2: - resolution: {integrity: sha512-X2UW6Nw3n/aMgDVy+0rSqgHlv39WZAlZrXCdnbyEiKm17DSqHX4MmQMaST3FbeWR5FTuRcUwYAziZajji0Y7mg==} - engines: {node: '>=10'} - - resolve@1.22.8: - resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} - hasBin: true - restore-cursor@4.0.0: resolution: {integrity: sha512-I9fPXU9geO9bHOt9pHHOhOkYerIMsmVaWB0rA2AI9ERh/+x/i7MV5HKBNrg+ljO5eoPVgCcnFuRjJ9uH6I/3eg==} engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} @@ -2463,22 +2114,36 @@ packages: resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==} hasBin: true + rollup@4.18.0: + resolution: {integrity: sha512-QmJz14PX3rzbJCN1SG4Xe/bAAX2a6NpCP8ab2vfu2GiUr8AQcr2nCV/oEO3yneFarB67zk8ShlIyWb2LGTb3Sg==} + engines: {node: '>=18.0.0', npm: '>=8.0.0'} + hasBin: true + rope-sequence@1.3.4: resolution: {integrity: sha512-UT5EDe2cu2E/6O4igUr5PSFs23nvvukicWHx6GnOPlHAiiYbzNuCRQCuiUdHJQcqKalLKlrYJnjY0ySGsXNQXQ==} + rrweb-cssom@0.6.0: + resolution: {integrity: sha512-APM0Gt1KoXBz0iIkkdB/kfvGOwC4UuJFeG/c+yV7wSc7q96cG/kJ0HiYCnzivD9SB53cLV1MlHFNfOuPaadYSw==} + + rrweb-cssom@0.7.1: + resolution: {integrity: sha512-TrEMa7JGdVm0UThDJSx7ddw5nVm3UJS9o9CCIZ72B1vSyEZoziDqBYP3XIoi/12lKrJR8rE3jeFHMok2F/Mnsg==} + run-parallel@1.2.0: resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} + safer-buffer@2.1.2: + resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} + sax@1.3.0: resolution: {integrity: sha512-0s+oAmw9zLl1V1cS9BtZN7JAd0cW5e0QH4W3LWEK6a4LaLEA2OTpGYWDY+6XasBLtz6wkm3u1xRw95mRuJ59WA==} + saxes@6.0.0: + resolution: {integrity: sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==} + engines: {node: '>=v12.22.7'} + sdp@3.2.0: resolution: {integrity: sha512-d7wDPgDV3DDiqulJjKiV2865wKsJ34YI+NDREbm+FySq6WuKOikwyNQcm+doLAZ1O6ltdO0SeKle2xMpN3Brgw==} - semver@6.3.1: - resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} - hasBin: true - semver@7.6.0: resolution: {integrity: sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==} engines: {node: '>=10'} @@ -2499,6 +2164,9 @@ packages: resolution: {integrity: sha512-2hbz3N7GuuTjI7y3sfnoqKnH0cNhExx67IJtCTGQI2KhBEyvegsDYW5qjj5BlvvVtQjmL/O/J1GQEciwfoZWpw==} engines: {node: 16.* || >= 18} + siginfo@2.0.0: + resolution: {integrity: sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==} + signal-exit@3.0.7: resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} @@ -2506,9 +2174,6 @@ packages: resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} engines: {node: '>=14'} - sisteransi@1.0.5: - resolution: {integrity: sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==} - slash@3.0.0: resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} engines: {node: '>=8'} @@ -2528,19 +2193,15 @@ packages: sortablejs@1.15.2: resolution: {integrity: sha512-FJF5jgdfvoKn1MAKSdGs33bIqLi3LmsgVTliuX6iITj834F+JRQZN90Z93yql8h0K2t0RwDPBmxwlbZfDcxNZA==} - source-map-support@0.5.13: - resolution: {integrity: sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w==} - - source-map@0.6.1: - resolution: {integrity: sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==} + source-map-js@1.2.0: + resolution: {integrity: sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==} engines: {node: '>=0.10.0'} - sprintf-js@1.0.3: - resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==} + stackback@0.0.2: + resolution: {integrity: sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==} - stack-utils@2.0.6: - resolution: {integrity: sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==} - engines: {node: '>=10'} + std-env@3.7.0: + resolution: {integrity: sha512-JPbdCEQLj1w5GilpiHAx3qJvFndqybBysA3qUOnznweH4QbNYUsW/ea8QzSrnh0vNsezMMw5bcVool8lM0gwzg==} stockfish-mv.wasm@0.6.1: resolution: {integrity: sha512-6QE1LeWbv9NNMABHCjDGyjgob+1CfMpIszxUKCUcCH0znreWPXz48G09rhKTUpRzC8YsOTHPQL/s74gDcqXJGw==} @@ -2558,10 +2219,6 @@ packages: resolution: {integrity: sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q==} engines: {node: '>=0.6.19'} - string-length@4.0.2: - resolution: {integrity: sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==} - engines: {node: '>=10'} - string-width@4.2.3: resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} engines: {node: '>=8'} @@ -2578,14 +2235,6 @@ packages: resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} engines: {node: '>=12'} - strip-bom@4.0.0: - resolution: {integrity: sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==} - engines: {node: '>=8'} - - strip-final-newline@2.0.0: - resolution: {integrity: sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==} - engines: {node: '>=6'} - strip-final-newline@3.0.0: resolution: {integrity: sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw==} engines: {node: '>=12'} @@ -2594,22 +2243,13 @@ packages: resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} engines: {node: '>=8'} - supports-color@5.5.0: - resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} - engines: {node: '>=4'} + strip-literal@2.1.0: + resolution: {integrity: sha512-Op+UycaUt/8FbN/Z2TWPBLge3jWrP3xj10f3fnYxf052bKuS3EKs1ZQcVGjnEMdsNVAM+plXRdmjrZ/KgG3Skw==} supports-color@7.2.0: resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} engines: {node: '>=8'} - supports-color@8.1.1: - resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} - engines: {node: '>=10'} - - supports-preserve-symlinks-flag@1.0.0: - resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} - engines: {node: '>= 0.4'} - svg.draggable.js@2.2.2: resolution: {integrity: sha512-JzNHBc2fLQMzYCZ90KZHN2ohXL0BQJGQimK1kGk6AvSeibuKcIdDX9Kr0dT9+UJ5O8nYA0RB839Lhvk4CY4MZw==} engines: {node: '>= 0.8.0'} @@ -2641,30 +2281,41 @@ packages: resolution: {integrity: sha512-h5IS/hKkuVCbKSieR9uQCj9w+zLHoPh+ce19bBYyqF53g6mnPB8sAtIbe1s9dh2S2fCmYX2xel1Ln3PJBbK4kw==} engines: {node: '>= 0.8.0'} + symbol-tree@3.2.4: + resolution: {integrity: sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==} + tablesort@5.3.0: resolution: {integrity: sha512-WkfcZBHsp47gVH9CBHG0ZXopriG01IA87arGrchvIe868d4RiXVvoYPS1zMq9IdW05kBs5iGsqxTABqLyWonbg==} - test-exclude@6.0.0: - resolution: {integrity: sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==} - engines: {node: '>=8'} - text-table@0.2.0: resolution: {integrity: sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==} textarea-caret@3.1.0: resolution: {integrity: sha512-cXAvzO9pP5CGa6NKx0WYHl+8CHKZs8byMkt3PCJBCmq2a34YA9pO1NrQET5pzeqnBjBdToF5No4rrmkDUgQC2Q==} - tmpl@1.0.5: - resolution: {integrity: sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==} + tinybench@2.8.0: + resolution: {integrity: sha512-1/eK7zUnIklz4JUUlL+658n58XO2hHLQfSk1Zf2LKieUjxidN16eKFEoDEfjHc3ohofSSqK3X5yO6VGb6iW8Lw==} - to-fast-properties@2.0.0: - resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==} - engines: {node: '>=4'} + tinypool@0.8.4: + resolution: {integrity: sha512-i11VH5gS6IFeLY3gMBQ00/MmLncVP7JLXOw1vlgkytLmJK7QnEr7NXf0LBdxfmNPAeyetukOk0bOYrJrFGjYJQ==} + engines: {node: '>=14.0.0'} + + tinyspy@2.2.1: + resolution: {integrity: sha512-KYad6Vy5VDWV4GH3fjpseMQ/XU2BhIYP7Vzd0LG44qRWm/Yt2WCOTicFdvmgo6gWaqooMQCawTtILVQJupKu7A==} + engines: {node: '>=14.0.0'} to-regex-range@5.0.1: resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} engines: {node: '>=8.0'} + tough-cookie@4.1.4: + resolution: {integrity: sha512-Loo5UUvLD9ScZ6jh8beX1T6sO1w2/MpCRpEP7V280GKMVUQ0Jzar2U3UJPsrdbziLEMMhu3Ujnq//rhiFuIeag==} + engines: {node: '>=6'} + + tr46@5.0.0: + resolution: {integrity: sha512-tk2G5R2KRwBd+ZN0zaEXpmzdKyOYksXwywulIX95MBODjSzMIuQnQ3m8JxgbhnL1LeVo7lqQKsYa1O3Htl7K5g==} + engines: {node: '>=18'} + tree-kill@1.2.2: resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} hasBin: true @@ -2687,10 +2338,6 @@ packages: resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==} engines: {node: '>=10'} - type-fest@0.21.3: - resolution: {integrity: sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==} - engines: {node: '>=10'} - types-serviceworker@0.0.1: resolution: {integrity: sha512-EKO/SZ3AsHEZsqv+bsdlTCz5k955riOksnYGlG6JhVwNTVsPWj/TScTbiNVZ5+mmX8TcEXF0C8aSxUw0jTDpIw==} @@ -2699,28 +2346,86 @@ packages: engines: {node: '>=14.17'} hasBin: true + ufo@1.5.3: + resolution: {integrity: sha512-Y7HYmWaFwPUmkoQCUIAYpKqkOf+SbVj/2fJJZ4RJMCfZp0rTGwRbzQD+HghfnhKOjL9E01okqz+ncJskGYfBNw==} + undate@0.3.0: resolution: {integrity: sha512-ssH8QTNBY6B+2fRr3stSQ+9m2NT8qTaun3ExTx5ibzYQvP7yX4+BnX0McNxFCvh6S5ia/DYu6bsCKQx/U4nb/Q==} undici-types@5.26.5: resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} - update-browserslist-db@1.0.13: - resolution: {integrity: sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg==} - hasBin: true - peerDependencies: - browserslist: '>= 4.21.0' + universalify@0.2.0: + resolution: {integrity: sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==} + engines: {node: '>= 4.0.0'} uri-js@4.4.1: resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} + url-parse@1.5.10: + resolution: {integrity: sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==} + uuid@9.0.0: resolution: {integrity: sha512-MXcSTerfPa4uqyzStbRoTgt5XIe3x5+42+q1sDuy3R5MDk66URdLMOZe5aPX/SQd+kuYAh0FdP/pO28IkQyTeg==} hasBin: true - v8-to-istanbul@9.2.0: - resolution: {integrity: sha512-/EH/sDgxU2eGxajKdwLCDmQ4FWq+kpi3uCmBGpw1xJtnAxEjlD8j8PEiGWpCIMIs3ciNAgH0d3TTJiUkYzyZjA==} - engines: {node: '>=10.12.0'} + vite-node@1.6.0: + resolution: {integrity: sha512-de6HJgzC+TFzOu0NTC4RAIsyf/DY/ibWDYQUcuEA84EMHhcefTUGkjFHKKEJhQN4A+6I0u++kr3l36ZF2d7XRw==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + + vite@5.3.0: + resolution: {integrity: sha512-hA6vAVK977NyW1Qw+fLvqSo7xDPej7von7C3DwwqPRmnnnK36XEBC/J3j1V5lP8fbt7y0TgTKJbpNGSwM+Bdeg==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + peerDependencies: + '@types/node': ^18.0.0 || >=20.0.0 + less: '*' + lightningcss: ^1.21.0 + sass: '*' + stylus: '*' + sugarss: '*' + terser: ^5.4.0 + peerDependenciesMeta: + '@types/node': + optional: true + less: + optional: true + lightningcss: + optional: true + sass: + optional: true + stylus: + optional: true + sugarss: + optional: true + terser: + optional: true + + vitest@1.6.0: + resolution: {integrity: sha512-H5r/dN06swuFnzNFhq/dnz37bPXnq8xB2xB5JOVk8K09rUtoeNN+LHWkoQ0A/i3hvbUKKcCei9KpbxqHMLhLLA==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + peerDependencies: + '@edge-runtime/vm': '*' + '@types/node': ^18.0.0 || >=20.0.0 + '@vitest/browser': 1.6.0 + '@vitest/ui': 1.6.0 + happy-dom: '*' + jsdom: '*' + peerDependenciesMeta: + '@edge-runtime/vm': + optional: true + '@types/node': + optional: true + '@vitest/browser': + optional: true + '@vitest/ui': + optional: true + happy-dom: + optional: true + jsdom: + optional: true vosk-browser@0.0.8: resolution: {integrity: sha512-df9MuChirUGKxP/+4k8tBV6hJOic8IRdJXv64+ET0Ue5CddOzLZINiHOWCka765gBzHJWNtQKgaYakS0FXZIJA==} @@ -2728,13 +2433,30 @@ packages: w3c-keyname@2.2.8: resolution: {integrity: sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==} - walker@1.0.8: - resolution: {integrity: sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==} + w3c-xmlserializer@5.0.0: + resolution: {integrity: sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==} + engines: {node: '>=18'} + + webidl-conversions@7.0.0: + resolution: {integrity: sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==} + engines: {node: '>=12'} webrtc-adapter@9.0.1: resolution: {integrity: sha512-1AQO+d4ElfVSXyzNVTOewgGT/tAomwwztX/6e3totvyyzXPvXIIuUUjAmyZGbKBKbZOXauuJooZm3g6IuFuiNQ==} engines: {node: '>=6.0.0', npm: '>=3.10.0'} + whatwg-encoding@3.1.1: + resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==} + engines: {node: '>=18'} + + whatwg-mimetype@4.0.0: + resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==} + engines: {node: '>=18'} + + whatwg-url@14.0.0: + resolution: {integrity: sha512-1lfMEm2IEr7RIV+f4lUNPOqfFL+pO+Xw3fJSqmjX9AbXcXcYOkCe1P6+9VBZB6n94af16NfZf+sSk0JCBZC9aw==} + engines: {node: '>=18'} + which-module@2.0.1: resolution: {integrity: sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ==} @@ -2743,14 +2465,15 @@ packages: engines: {node: '>= 8'} hasBin: true + why-is-node-running@2.2.2: + resolution: {integrity: sha512-6tSwToZxTOcotxHeA+qGCq1mVzKR3CwcJGmVcY+QE8SHy6TnpFnh8PAvPNHYr7EcuVeG0QSMxtYCuO1ta/G/oA==} + engines: {node: '>=8'} + hasBin: true + wrap-ansi@6.2.0: resolution: {integrity: sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==} engines: {node: '>=8'} - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} - wrap-ansi@9.0.0: resolution: {integrity: sha512-G8ura3S+3Z2G+mkgNRq8dqaFZAuxfsxpBB8OCTGRTCtp+l/v9nbFNmCUP1BZMts3G1142MsZfn6eeUKrr4PD1Q==} engines: {node: '>=18'} @@ -2758,9 +2481,21 @@ packages: wrappy@1.0.2: resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} - write-file-atomic@4.0.2: - resolution: {integrity: sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==} - engines: {node: ^12.13.0 || ^14.15.0 || >=16.0.0} + ws@8.17.0: + resolution: {integrity: sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==} + engines: {node: '>=10.0.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: '>=5.0.2' + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + + xml-name-validator@5.0.0: + resolution: {integrity: sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==} + engines: {node: '>=18'} xml2js@0.5.0: resolution: {integrity: sha512-drPFnkQJik/O+uPKpqSgr22mpuFHqKdbS835iAQrUC73L2F5WkboIRd63ai/2Yg6I1jzifPFKH2NTK+cfglkIA==} @@ -2770,16 +2505,12 @@ packages: resolution: {integrity: sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==} engines: {node: '>=4.0'} + xmlchars@2.2.0: + resolution: {integrity: sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==} + y18n@4.0.3: resolution: {integrity: sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==} - y18n@5.0.8: - resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} - engines: {node: '>=10'} - - yallist@3.1.1: - resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==} - yallist@4.0.0: resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==} @@ -2791,22 +2522,18 @@ packages: resolution: {integrity: sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==} engines: {node: '>=6'} - yargs-parser@21.1.1: - resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} - engines: {node: '>=12'} - yargs@15.4.1: resolution: {integrity: sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==} engines: {node: '>=8'} - yargs@17.7.2: - resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} - engines: {node: '>=12'} - yocto-queue@0.1.0: resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} engines: {node: '>=10'} + yocto-queue@1.0.0: + resolution: {integrity: sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==} + engines: {node: '>=12.20'} + zxcvbn@4.4.2: resolution: {integrity: sha512-Bq0B+ixT/DMyG8kgX2xWcI5jUvCwqrMxSFam7m0lAf78nf04hv6lNCsyLYdyYTrCVMqNDY/206K7eExYCeSyUQ==} @@ -2814,216 +2541,80 @@ snapshots: '@aashutoshrathi/word-wrap@1.2.6': {} - '@ampproject/remapping@2.3.0': - dependencies: - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 + '@badrap/result@0.2.13': {} - '@babel/code-frame@7.24.2': - dependencies: - '@babel/highlight': 7.24.2 - picocolors: 1.0.0 + '@blakeembrey/deque@1.0.5': {} - '@babel/compat-data@7.24.1': {} - - '@babel/core@7.24.3': - dependencies: - '@ampproject/remapping': 2.3.0 - '@babel/code-frame': 7.24.2 - '@babel/generator': 7.24.1 - '@babel/helper-compilation-targets': 7.23.6 - '@babel/helper-module-transforms': 7.23.3(@babel/core@7.24.3) - '@babel/helpers': 7.24.1 - '@babel/parser': 7.24.1 - '@babel/template': 7.24.0 - '@babel/traverse': 7.24.1 - '@babel/types': 7.24.0 - convert-source-map: 2.0.0 - debug: 4.3.4 - gensync: 1.0.0-beta.2 - json5: 2.2.3 - semver: 6.3.1 - transitivePeerDependencies: - - supports-color + '@blakeembrey/template@1.1.0': {} - '@babel/generator@7.24.1': - dependencies: - '@babel/types': 7.24.0 - '@jridgewell/gen-mapping': 0.3.5 - '@jridgewell/trace-mapping': 0.3.25 - jsesc: 2.5.2 + '@esbuild/aix-ppc64@0.21.5': + optional: true - '@babel/helper-compilation-targets@7.23.6': - dependencies: - '@babel/compat-data': 7.24.1 - '@babel/helper-validator-option': 7.23.5 - browserslist: 4.23.0 - lru-cache: 5.1.1 - semver: 6.3.1 + '@esbuild/android-arm64@0.21.5': + optional: true - '@babel/helper-environment-visitor@7.22.20': {} + '@esbuild/android-arm@0.21.5': + optional: true - '@babel/helper-function-name@7.23.0': - dependencies: - '@babel/template': 7.24.0 - '@babel/types': 7.24.0 + '@esbuild/android-x64@0.21.5': + optional: true - '@babel/helper-hoist-variables@7.22.5': - dependencies: - '@babel/types': 7.24.0 + '@esbuild/darwin-arm64@0.21.5': + optional: true - '@babel/helper-module-imports@7.24.3': - dependencies: - '@babel/types': 7.24.0 + '@esbuild/darwin-x64@0.21.5': + optional: true - '@babel/helper-module-transforms@7.23.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-environment-visitor': 7.22.20 - '@babel/helper-module-imports': 7.24.3 - '@babel/helper-simple-access': 7.22.5 - '@babel/helper-split-export-declaration': 7.22.6 - '@babel/helper-validator-identifier': 7.22.20 + '@esbuild/freebsd-arm64@0.21.5': + optional: true - '@babel/helper-plugin-utils@7.24.0': {} + '@esbuild/freebsd-x64@0.21.5': + optional: true - '@babel/helper-simple-access@7.22.5': - dependencies: - '@babel/types': 7.24.0 + '@esbuild/linux-arm64@0.21.5': + optional: true - '@babel/helper-split-export-declaration@7.22.6': - dependencies: - '@babel/types': 7.24.0 + '@esbuild/linux-arm@0.21.5': + optional: true - '@babel/helper-string-parser@7.24.1': {} + '@esbuild/linux-ia32@0.21.5': + optional: true - '@babel/helper-validator-identifier@7.22.20': {} + '@esbuild/linux-loong64@0.21.5': + optional: true - '@babel/helper-validator-option@7.23.5': {} + '@esbuild/linux-mips64el@0.21.5': + optional: true - '@babel/helpers@7.24.1': - dependencies: - '@babel/template': 7.24.0 - '@babel/traverse': 7.24.1 - '@babel/types': 7.24.0 - transitivePeerDependencies: - - supports-color + '@esbuild/linux-ppc64@0.21.5': + optional: true - '@babel/highlight@7.24.2': - dependencies: - '@babel/helper-validator-identifier': 7.22.20 - chalk: 2.4.2 - js-tokens: 4.0.0 - picocolors: 1.0.0 + '@esbuild/linux-riscv64@0.21.5': + optional: true - '@babel/parser@7.24.1': - dependencies: - '@babel/types': 7.24.0 - - '@babel/plugin-syntax-async-generators@7.8.4(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-bigint@7.8.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-class-properties@7.12.13(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-import-meta@7.10.4(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-json-strings@7.8.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-jsx@7.24.1(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-logical-assignment-operators@7.10.4(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-nullish-coalescing-operator@7.8.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-numeric-separator@7.10.4(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-object-rest-spread@7.8.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-optional-catch-binding@7.8.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-optional-chaining@7.8.3(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-top-level-await@7.14.5(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 - - '@babel/plugin-syntax-typescript@7.24.1(@babel/core@7.24.3)': - dependencies: - '@babel/core': 7.24.3 - '@babel/helper-plugin-utils': 7.24.0 + '@esbuild/linux-s390x@0.21.5': + optional: true - '@babel/template@7.24.0': - dependencies: - '@babel/code-frame': 7.24.2 - '@babel/parser': 7.24.1 - '@babel/types': 7.24.0 + '@esbuild/linux-x64@0.21.5': + optional: true - '@babel/traverse@7.24.1': - dependencies: - '@babel/code-frame': 7.24.2 - '@babel/generator': 7.24.1 - '@babel/helper-environment-visitor': 7.22.20 - '@babel/helper-function-name': 7.23.0 - '@babel/helper-hoist-variables': 7.22.5 - '@babel/helper-split-export-declaration': 7.22.6 - '@babel/parser': 7.24.1 - '@babel/types': 7.24.0 - debug: 4.3.4 - globals: 11.12.0 - transitivePeerDependencies: - - supports-color + '@esbuild/netbsd-x64@0.21.5': + optional: true - '@babel/types@7.24.0': - dependencies: - '@babel/helper-string-parser': 7.24.1 - '@babel/helper-validator-identifier': 7.22.20 - to-fast-properties: 2.0.0 + '@esbuild/openbsd-x64@0.21.5': + optional: true - '@badrap/result@0.2.13': {} + '@esbuild/sunos-x64@0.21.5': + optional: true - '@bcoe/v8-coverage@0.2.3': {} + '@esbuild/win32-arm64@0.21.5': + optional: true - '@blakeembrey/deque@1.0.5': {} + '@esbuild/win32-ia32@0.21.5': + optional: true - '@blakeembrey/template@1.1.0': {} + '@esbuild/win32-x64@0.21.5': + optional: true '@eslint-community/eslint-utils@4.4.0(eslint@8.57.0)': dependencies: @@ -3073,223 +2664,80 @@ snapshots: '@humanwhocodes/object-schema@2.0.3': {} - '@istanbuljs/load-nyc-config@1.1.0': - dependencies: - camelcase: 5.3.1 - find-up: 4.1.0 - get-package-type: 0.1.0 - js-yaml: 3.14.1 - resolve-from: 5.0.0 - - '@istanbuljs/schema@0.1.3': {} - - '@jest/console@29.7.0': - dependencies: - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - chalk: 4.1.2 - jest-message-util: 29.7.0 - jest-util: 29.7.0 - slash: 3.0.0 - - '@jest/core@29.7.0': + '@jest/schemas@29.6.3': dependencies: - '@jest/console': 29.7.0 - '@jest/reporters': 29.7.0 - '@jest/test-result': 29.7.0 - '@jest/transform': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - ansi-escapes: 4.3.2 - chalk: 4.1.2 - ci-info: 3.9.0 - exit: 0.1.2 - graceful-fs: 4.2.11 - jest-changed-files: 29.7.0 - jest-config: 29.7.0(@types/node@20.12.2) - jest-haste-map: 29.7.0 - jest-message-util: 29.7.0 - jest-regex-util: 29.6.3 - jest-resolve: 29.7.0 - jest-resolve-dependencies: 29.7.0 - jest-runner: 29.7.0 - jest-runtime: 29.7.0 - jest-snapshot: 29.7.0 - jest-util: 29.7.0 - jest-validate: 29.7.0 - jest-watcher: 29.7.0 - micromatch: 4.0.5 - pretty-format: 29.7.0 - slash: 3.0.0 - strip-ansi: 6.0.1 - transitivePeerDependencies: - - babel-plugin-macros - - supports-color - - ts-node + '@sinclair/typebox': 0.27.8 - '@jest/environment@29.7.0': - dependencies: - '@jest/fake-timers': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - jest-mock: 29.7.0 + '@jridgewell/sourcemap-codec@1.4.15': {} - '@jest/expect-utils@29.7.0': - dependencies: - jest-get-type: 29.6.3 + '@juggle/resize-observer@3.4.0': {} - '@jest/expect@29.7.0': - dependencies: - expect: 29.7.0 - jest-snapshot: 29.7.0 - transitivePeerDependencies: - - supports-color + '@kurkle/color@0.3.2': {} - '@jest/fake-timers@29.7.0': - dependencies: - '@jest/types': 29.6.3 - '@sinonjs/fake-timers': 10.3.0 - '@types/node': 20.12.2 - jest-message-util: 29.7.0 - jest-mock: 29.7.0 - jest-util: 29.7.0 + '@msgpack/msgpack@2.8.0': {} - '@jest/globals@29.7.0': + '@nodelib/fs.scandir@2.1.5': dependencies: - '@jest/environment': 29.7.0 - '@jest/expect': 29.7.0 - '@jest/types': 29.6.3 - jest-mock: 29.7.0 - transitivePeerDependencies: - - supports-color + '@nodelib/fs.stat': 2.0.5 + run-parallel: 1.2.0 - '@jest/reporters@29.7.0': - dependencies: - '@bcoe/v8-coverage': 0.2.3 - '@jest/console': 29.7.0 - '@jest/test-result': 29.7.0 - '@jest/transform': 29.7.0 - '@jest/types': 29.6.3 - '@jridgewell/trace-mapping': 0.3.25 - '@types/node': 20.12.2 - chalk: 4.1.2 - collect-v8-coverage: 1.0.2 - exit: 0.1.2 - glob: 7.2.3 - graceful-fs: 4.2.11 - istanbul-lib-coverage: 3.2.2 - istanbul-lib-instrument: 6.0.2 - istanbul-lib-report: 3.0.1 - istanbul-lib-source-maps: 4.0.1 - istanbul-reports: 3.1.7 - jest-message-util: 29.7.0 - jest-util: 29.7.0 - jest-worker: 29.7.0 - slash: 3.0.0 - string-length: 4.0.2 - strip-ansi: 6.0.1 - v8-to-istanbul: 9.2.0 - transitivePeerDependencies: - - supports-color + '@nodelib/fs.stat@2.0.5': {} - '@jest/schemas@29.6.3': + '@nodelib/fs.walk@1.2.8': dependencies: - '@sinclair/typebox': 0.27.8 + '@nodelib/fs.scandir': 2.1.5 + fastq: 1.17.1 - '@jest/source-map@29.6.3': - dependencies: - '@jridgewell/trace-mapping': 0.3.25 - callsites: 3.1.0 - graceful-fs: 4.2.11 + '@rollup/rollup-android-arm-eabi@4.18.0': + optional: true - '@jest/test-result@29.7.0': - dependencies: - '@jest/console': 29.7.0 - '@jest/types': 29.6.3 - '@types/istanbul-lib-coverage': 2.0.6 - collect-v8-coverage: 1.0.2 + '@rollup/rollup-android-arm64@4.18.0': + optional: true - '@jest/test-sequencer@29.7.0': - dependencies: - '@jest/test-result': 29.7.0 - graceful-fs: 4.2.11 - jest-haste-map: 29.7.0 - slash: 3.0.0 + '@rollup/rollup-darwin-arm64@4.18.0': + optional: true - '@jest/transform@29.7.0': - dependencies: - '@babel/core': 7.24.3 - '@jest/types': 29.6.3 - '@jridgewell/trace-mapping': 0.3.25 - babel-plugin-istanbul: 6.1.1 - chalk: 4.1.2 - convert-source-map: 2.0.0 - fast-json-stable-stringify: 2.1.0 - graceful-fs: 4.2.11 - jest-haste-map: 29.7.0 - jest-regex-util: 29.6.3 - jest-util: 29.7.0 - micromatch: 4.0.5 - pirates: 4.0.6 - slash: 3.0.0 - write-file-atomic: 4.0.2 - transitivePeerDependencies: - - supports-color + '@rollup/rollup-darwin-x64@4.18.0': + optional: true - '@jest/types@29.6.3': - dependencies: - '@jest/schemas': 29.6.3 - '@types/istanbul-lib-coverage': 2.0.6 - '@types/istanbul-reports': 3.0.4 - '@types/node': 20.12.2 - '@types/yargs': 17.0.32 - chalk: 4.1.2 + '@rollup/rollup-linux-arm-gnueabihf@4.18.0': + optional: true - '@jridgewell/gen-mapping@0.3.5': - dependencies: - '@jridgewell/set-array': 1.2.1 - '@jridgewell/sourcemap-codec': 1.4.15 - '@jridgewell/trace-mapping': 0.3.25 + '@rollup/rollup-linux-arm-musleabihf@4.18.0': + optional: true - '@jridgewell/resolve-uri@3.1.2': {} + '@rollup/rollup-linux-arm64-gnu@4.18.0': + optional: true - '@jridgewell/set-array@1.2.1': {} + '@rollup/rollup-linux-arm64-musl@4.18.0': + optional: true - '@jridgewell/sourcemap-codec@1.4.15': {} + '@rollup/rollup-linux-powerpc64le-gnu@4.18.0': + optional: true - '@jridgewell/trace-mapping@0.3.25': - dependencies: - '@jridgewell/resolve-uri': 3.1.2 - '@jridgewell/sourcemap-codec': 1.4.15 + '@rollup/rollup-linux-riscv64-gnu@4.18.0': + optional: true - '@juggle/resize-observer@3.4.0': {} + '@rollup/rollup-linux-s390x-gnu@4.18.0': + optional: true - '@kurkle/color@0.3.2': {} + '@rollup/rollup-linux-x64-gnu@4.18.0': + optional: true - '@msgpack/msgpack@2.8.0': {} + '@rollup/rollup-linux-x64-musl@4.18.0': + optional: true - '@nodelib/fs.scandir@2.1.5': - dependencies: - '@nodelib/fs.stat': 2.0.5 - run-parallel: 1.2.0 + '@rollup/rollup-win32-arm64-msvc@4.18.0': + optional: true - '@nodelib/fs.stat@2.0.5': {} + '@rollup/rollup-win32-ia32-msvc@4.18.0': + optional: true - '@nodelib/fs.walk@1.2.8': - dependencies: - '@nodelib/fs.scandir': 2.1.5 - fastq: 1.17.1 + '@rollup/rollup-win32-x64-msvc@4.18.0': + optional: true '@sinclair/typebox@0.27.8': {} - '@sinonjs/commons@3.0.1': - dependencies: - type-detect: 4.0.8 - - '@sinonjs/fake-timers@10.3.0': - dependencies: - '@sinonjs/commons': 3.0.1 - '@textcomplete/core@0.1.13': dependencies: eventemitter3: 5.0.1 @@ -3316,53 +2764,15 @@ snapshots: '@types/audioworklet@0.0.54': {} - '@types/babel__core@7.20.5': - dependencies: - '@babel/parser': 7.24.1 - '@babel/types': 7.24.0 - '@types/babel__generator': 7.6.8 - '@types/babel__template': 7.4.4 - '@types/babel__traverse': 7.20.5 - - '@types/babel__generator@7.6.8': - dependencies: - '@babel/types': 7.24.0 - - '@types/babel__template@7.4.4': - dependencies: - '@babel/parser': 7.24.1 - '@babel/types': 7.24.0 - - '@types/babel__traverse@7.20.5': - dependencies: - '@babel/types': 7.24.0 - '@types/chess.js@0.10.1': {} '@types/debounce-promise@3.1.9': {} '@types/dragscroll@0.0.0': {} - '@types/fnando__sparkline@0.3.7': {} - - '@types/graceful-fs@4.1.9': - dependencies: - '@types/node': 20.12.2 - - '@types/istanbul-lib-coverage@2.0.6': {} - - '@types/istanbul-lib-report@3.0.3': - dependencies: - '@types/istanbul-lib-coverage': 2.0.6 - - '@types/istanbul-reports@3.0.4': - dependencies: - '@types/istanbul-lib-report': 3.0.3 + '@types/estree@1.0.5': {} - '@types/jest@29.5.12': - dependencies: - expect: 29.7.0 - pretty-format: 29.7.0 + '@types/fnando__sparkline@0.3.7': {} '@types/json-schema@7.0.15': {} @@ -3385,8 +2795,6 @@ snapshots: '@types/sortablejs@1.15.8': {} - '@types/stack-utils@2.0.3': {} - '@types/web@0.0.142': {} '@types/webrtc@0.0.33': {} @@ -3395,12 +2803,6 @@ snapshots: dependencies: '@types/react': 18.2.74 - '@types/yargs-parser@21.0.3': {} - - '@types/yargs@17.0.32': - dependencies: - '@types/yargs-parser': 21.0.3 - '@types/zxcvbn@4.4.4': {} '@typescript-eslint/eslint-plugin@7.5.0(@typescript-eslint/parser@7.5.0(eslint@8.57.0)(typescript@5.4.3))(eslint@8.57.0)(typescript@5.4.3)': @@ -3491,6 +2893,35 @@ snapshots: '@ungap/structured-clone@1.2.0': {} + '@vitest/expect@1.6.0': + dependencies: + '@vitest/spy': 1.6.0 + '@vitest/utils': 1.6.0 + chai: 4.4.1 + + '@vitest/runner@1.6.0': + dependencies: + '@vitest/utils': 1.6.0 + p-limit: 5.0.0 + pathe: 1.1.2 + + '@vitest/snapshot@1.6.0': + dependencies: + magic-string: 0.30.10 + pathe: 1.1.2 + pretty-format: 29.7.0 + + '@vitest/spy@1.6.0': + dependencies: + tinyspy: 2.2.1 + + '@vitest/utils@1.6.0': + dependencies: + diff-sequences: 29.6.3 + estree-walker: 3.0.3 + loupe: 2.3.7 + pretty-format: 29.7.0 + '@yaireo/tagify@4.17.9(prop-types@15.8.1)': dependencies: prop-types: 15.8.1 @@ -3503,8 +2934,16 @@ snapshots: dependencies: acorn: 8.11.3 + acorn-walk@8.3.2: {} + acorn@8.11.3: {} + agent-base@7.1.1: + dependencies: + debug: 4.3.4 + transitivePeerDependencies: + - supports-color + ajv@6.12.6: dependencies: fast-deep-equal: 3.1.3 @@ -3512,20 +2951,12 @@ snapshots: json-schema-traverse: 0.4.1 uri-js: 4.4.1 - ansi-escapes@4.3.2: - dependencies: - type-fest: 0.21.3 - ansi-escapes@6.2.1: {} ansi-regex@5.0.1: {} ansi-regex@6.0.1: {} - ansi-styles@3.2.1: - dependencies: - color-convert: 1.9.3 - ansi-styles@4.3.0: dependencies: color-convert: 2.0.1 @@ -3551,65 +2982,13 @@ snapshots: arg@4.1.3: {} - argparse@1.0.10: - dependencies: - sprintf-js: 1.0.3 - argparse@2.0.1: {} array-union@2.1.0: {} - babel-jest@29.7.0(@babel/core@7.24.3): - dependencies: - '@babel/core': 7.24.3 - '@jest/transform': 29.7.0 - '@types/babel__core': 7.20.5 - babel-plugin-istanbul: 6.1.1 - babel-preset-jest: 29.6.3(@babel/core@7.24.3) - chalk: 4.1.2 - graceful-fs: 4.2.11 - slash: 3.0.0 - transitivePeerDependencies: - - supports-color + assertion-error@1.1.0: {} - babel-plugin-istanbul@6.1.1: - dependencies: - '@babel/helper-plugin-utils': 7.24.0 - '@istanbuljs/load-nyc-config': 1.1.0 - '@istanbuljs/schema': 0.1.3 - istanbul-lib-instrument: 5.2.1 - test-exclude: 6.0.0 - transitivePeerDependencies: - - supports-color - - babel-plugin-jest-hoist@29.6.3: - dependencies: - '@babel/template': 7.24.0 - '@babel/types': 7.24.0 - '@types/babel__core': 7.20.5 - '@types/babel__traverse': 7.20.5 - - babel-preset-current-node-syntax@1.0.1(@babel/core@7.24.3): - dependencies: - '@babel/core': 7.24.3 - '@babel/plugin-syntax-async-generators': 7.8.4(@babel/core@7.24.3) - '@babel/plugin-syntax-bigint': 7.8.3(@babel/core@7.24.3) - '@babel/plugin-syntax-class-properties': 7.12.13(@babel/core@7.24.3) - '@babel/plugin-syntax-import-meta': 7.10.4(@babel/core@7.24.3) - '@babel/plugin-syntax-json-strings': 7.8.3(@babel/core@7.24.3) - '@babel/plugin-syntax-logical-assignment-operators': 7.10.4(@babel/core@7.24.3) - '@babel/plugin-syntax-nullish-coalescing-operator': 7.8.3(@babel/core@7.24.3) - '@babel/plugin-syntax-numeric-separator': 7.10.4(@babel/core@7.24.3) - '@babel/plugin-syntax-object-rest-spread': 7.8.3(@babel/core@7.24.3) - '@babel/plugin-syntax-optional-catch-binding': 7.8.3(@babel/core@7.24.3) - '@babel/plugin-syntax-optional-chaining': 7.8.3(@babel/core@7.24.3) - '@babel/plugin-syntax-top-level-await': 7.14.5(@babel/core@7.24.3) - - babel-preset-jest@29.6.3(@babel/core@7.24.3): - dependencies: - '@babel/core': 7.24.3 - babel-plugin-jest-hoist: 29.6.3 - babel-preset-current-node-syntax: 1.0.1(@babel/core@7.24.3) + asynckit@0.4.0: {} balanced-match@1.0.2: {} @@ -3628,32 +3007,21 @@ snapshots: dependencies: fill-range: 7.0.1 - browserslist@4.23.0: - dependencies: - caniuse-lite: 1.0.30001605 - electron-to-chromium: 1.4.723 - node-releases: 2.0.14 - update-browserslist-db: 1.0.13(browserslist@4.23.0) - - bser@2.1.1: - dependencies: - node-int64: 0.4.0 - - buffer-from@1.1.2: {} + cac@6.7.14: {} callsites@3.1.0: {} camelcase@5.3.1: {} - camelcase@6.3.0: {} - - caniuse-lite@1.0.30001605: {} - - chalk@2.4.2: + chai@4.4.1: dependencies: - ansi-styles: 3.2.1 - escape-string-regexp: 1.0.5 - supports-color: 5.5.0 + assertion-error: 1.1.0 + check-error: 1.0.3 + deep-eql: 4.1.4 + get-func-name: 2.0.2 + loupe: 2.3.7 + pathval: 1.1.1 + type-detect: 4.0.8 chalk@4.1.2: dependencies: @@ -3662,8 +3030,6 @@ snapshots: chalk@5.3.0: {} - char-regex@1.0.2: {} - chart.js@4.4.0: dependencies: '@kurkle/color': 0.3.2 @@ -3682,6 +3048,10 @@ snapshots: chart.js: 4.4.0 hammerjs: 2.0.8 + check-error@1.0.3: + dependencies: + get-func-name: 2.0.2 + chess.js@https://codeload.github.com/ornicar/chess.js/tar.gz/ad0709c0b07773d9d0da8e4605a9a2a28f00d249: {} chessground@9.1.1: {} @@ -3706,10 +3076,6 @@ snapshots: optionalDependencies: fsevents: 2.3.3 - ci-info@3.9.0: {} - - cjs-module-lexer@1.2.3: {} - cli-cursor@4.0.0: dependencies: restore-cursor: 4.0.0 @@ -3725,50 +3091,23 @@ snapshots: strip-ansi: 6.0.1 wrap-ansi: 6.2.0 - cliui@8.0.1: - dependencies: - string-width: 4.2.3 - strip-ansi: 6.0.1 - wrap-ansi: 7.0.0 - - co@4.6.0: {} - - collect-v8-coverage@1.0.2: {} - - color-convert@1.9.3: - dependencies: - color-name: 1.1.3 - color-convert@2.0.1: dependencies: color-name: 1.1.4 - color-name@1.1.3: {} - color-name@1.1.4: {} colorette@2.0.20: {} + combined-stream@1.0.8: + dependencies: + delayed-stream: 1.0.0 + commander@11.1.0: {} concat-map@0.0.1: {} - convert-source-map@2.0.0: {} - - create-jest@29.7.0(@types/node@20.12.2): - dependencies: - '@jest/types': 29.6.3 - chalk: 4.1.2 - exit: 0.1.2 - graceful-fs: 4.2.11 - jest-config: 29.7.0(@types/node@20.12.2) - jest-util: 29.7.0 - prompts: 2.4.2 - transitivePeerDependencies: - - '@types/node' - - babel-plugin-macros - - supports-color - - ts-node + confbox@0.1.7: {} cropperjs@1.6.1: {} @@ -3778,8 +3117,17 @@ snapshots: shebang-command: 2.0.0 which: 2.0.2 + cssstyle@4.0.1: + dependencies: + rrweb-cssom: 0.6.0 + csstype@3.1.3: {} + data-urls@5.0.0: + dependencies: + whatwg-mimetype: 4.0.0 + whatwg-url: 14.0.0 + date-fns@2.29.3: {} dayjs@1.11.10: {} @@ -3792,13 +3140,17 @@ snapshots: decamelize@1.2.0: {} - dedent@1.5.1: {} + decimal.js@10.4.3: {} + + deep-eql@4.1.4: + dependencies: + type-detect: 4.0.8 deep-is@0.1.4: {} deepmerge@4.3.1: {} - detect-newline@3.1.0: {} + delayed-stream@1.0.0: {} dialog-polyfill@0.5.6: {} @@ -3818,10 +3170,6 @@ snapshots: dragscroll@0.0.8: {} - electron-to-chromium@1.4.723: {} - - emittery@0.13.1: {} - emoji-mart@5.5.2: {} emoji-regex@10.3.0: {} @@ -3830,15 +3178,33 @@ snapshots: encode-utf8@1.0.3: {} - error-ex@1.3.2: - dependencies: - is-arrayish: 0.2.1 - - escalade@3.1.2: {} - - escape-string-regexp@1.0.5: {} + entities@4.5.0: {} - escape-string-regexp@2.0.0: {} + esbuild@0.21.5: + optionalDependencies: + '@esbuild/aix-ppc64': 0.21.5 + '@esbuild/android-arm': 0.21.5 + '@esbuild/android-arm64': 0.21.5 + '@esbuild/android-x64': 0.21.5 + '@esbuild/darwin-arm64': 0.21.5 + '@esbuild/darwin-x64': 0.21.5 + '@esbuild/freebsd-arm64': 0.21.5 + '@esbuild/freebsd-x64': 0.21.5 + '@esbuild/linux-arm': 0.21.5 + '@esbuild/linux-arm64': 0.21.5 + '@esbuild/linux-ia32': 0.21.5 + '@esbuild/linux-loong64': 0.21.5 + '@esbuild/linux-mips64el': 0.21.5 + '@esbuild/linux-ppc64': 0.21.5 + '@esbuild/linux-riscv64': 0.21.5 + '@esbuild/linux-s390x': 0.21.5 + '@esbuild/linux-x64': 0.21.5 + '@esbuild/netbsd-x64': 0.21.5 + '@esbuild/openbsd-x64': 0.21.5 + '@esbuild/sunos-x64': 0.21.5 + '@esbuild/win32-arm64': 0.21.5 + '@esbuild/win32-ia32': 0.21.5 + '@esbuild/win32-x64': 0.21.5 escape-string-regexp@4.0.0: {} @@ -3898,8 +3264,6 @@ snapshots: acorn-jsx: 5.3.2(acorn@8.11.3) eslint-visitor-keys: 3.4.3 - esprima@4.0.1: {} - esquery@1.5.0: dependencies: estraverse: 5.3.0 @@ -3910,24 +3274,16 @@ snapshots: estraverse@5.3.0: {} + estree-walker@3.0.3: + dependencies: + '@types/estree': 1.0.5 + esutils@2.0.3: {} eventemitter3@4.0.7: {} eventemitter3@5.0.1: {} - execa@5.1.1: - dependencies: - cross-spawn: 7.0.3 - get-stream: 6.0.1 - human-signals: 2.1.0 - is-stream: 2.0.1 - merge-stream: 2.0.0 - npm-run-path: 4.0.1 - onetime: 5.1.2 - signal-exit: 3.0.7 - strip-final-newline: 2.0.0 - execa@8.0.1: dependencies: cross-spawn: 7.0.3 @@ -3940,16 +3296,6 @@ snapshots: signal-exit: 4.1.0 strip-final-newline: 3.0.0 - exit@0.1.2: {} - - expect@29.7.0: - dependencies: - '@jest/expect-utils': 29.7.0 - jest-get-type: 29.6.3 - jest-matcher-utils: 29.7.0 - jest-message-util: 29.7.0 - jest-util: 29.7.0 - fast-deep-equal@3.1.3: {} fast-glob@3.3.2: @@ -3968,10 +3314,6 @@ snapshots: dependencies: reusify: 1.0.4 - fb-watchman@2.0.2: - dependencies: - bser: 2.1.1 - file-entry-cache@6.0.1: dependencies: flat-cache: 3.2.0 @@ -3998,518 +3340,179 @@ snapshots: flatpickr@4.6.13: {} - flatted@3.3.1: {} - - fs.realpath@1.0.0: {} - - fsevents@2.3.3: - optional: true - - function-bind@1.1.2: {} - - gensync@1.0.0-beta.2: {} - - get-caller-file@2.0.5: {} - - get-east-asian-width@1.2.0: {} - - get-package-type@0.1.0: {} - - get-stream@6.0.1: {} - - get-stream@8.0.1: {} - - glob-parent@5.1.2: - dependencies: - is-glob: 4.0.3 - - glob-parent@6.0.2: - dependencies: - is-glob: 4.0.3 - - glob@7.2.3: - dependencies: - fs.realpath: 1.0.0 - inflight: 1.0.6 - inherits: 2.0.4 - minimatch: 3.1.2 - once: 1.4.0 - path-is-absolute: 1.0.1 - - globals@11.12.0: {} - - globals@13.24.0: - dependencies: - type-fest: 0.20.2 - - globby@11.1.0: - dependencies: - array-union: 2.1.0 - dir-glob: 3.0.1 - fast-glob: 3.3.2 - ignore: 5.3.1 - merge2: 1.4.1 - slash: 3.0.0 - - graceful-fs@4.2.11: {} - - graphemer@1.4.0: {} - - hammerjs@2.0.8: {} - - has-flag@3.0.0: {} - - has-flag@4.0.0: {} - - hasown@2.0.2: - dependencies: - function-bind: 1.1.2 - - html-escaper@2.0.2: {} - - human-signals@2.1.0: {} - - human-signals@5.0.0: {} - - idb-keyval@6.2.1: {} - - ignore@5.3.1: {} - - import-fresh@3.3.0: - dependencies: - parent-module: 1.0.1 - resolve-from: 4.0.0 - - import-local@3.1.0: - dependencies: - pkg-dir: 4.2.0 - resolve-cwd: 3.0.0 - - imurmurhash@0.1.4: {} - - inflight@1.0.6: - dependencies: - once: 1.4.0 - wrappy: 1.0.2 - - inherits@2.0.4: {} - - is-arrayish@0.2.1: {} - - is-binary-path@2.1.0: - dependencies: - binary-extensions: 2.3.0 - - is-core-module@2.13.1: - dependencies: - hasown: 2.0.2 - - is-extglob@2.1.1: {} - - is-fullwidth-code-point@3.0.0: {} - - is-fullwidth-code-point@4.0.0: {} - - is-fullwidth-code-point@5.0.0: - dependencies: - get-east-asian-width: 1.2.0 - - is-generator-fn@2.1.0: {} - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-number@7.0.0: {} - - is-path-inside@3.0.3: {} - - is-stream@2.0.1: {} - - is-stream@3.0.0: {} - - isexe@2.0.0: {} - - istanbul-lib-coverage@3.2.2: {} - - istanbul-lib-instrument@5.2.1: - dependencies: - '@babel/core': 7.24.3 - '@babel/parser': 7.24.1 - '@istanbuljs/schema': 0.1.3 - istanbul-lib-coverage: 3.2.2 - semver: 6.3.1 - transitivePeerDependencies: - - supports-color - - istanbul-lib-instrument@6.0.2: - dependencies: - '@babel/core': 7.24.3 - '@babel/parser': 7.24.1 - '@istanbuljs/schema': 0.1.3 - istanbul-lib-coverage: 3.2.2 - semver: 7.6.0 - transitivePeerDependencies: - - supports-color - - istanbul-lib-report@3.0.1: - dependencies: - istanbul-lib-coverage: 3.2.2 - make-dir: 4.0.0 - supports-color: 7.2.0 - - istanbul-lib-source-maps@4.0.1: - dependencies: - debug: 4.3.4 - istanbul-lib-coverage: 3.2.2 - source-map: 0.6.1 - transitivePeerDependencies: - - supports-color - - istanbul-reports@3.1.7: - dependencies: - html-escaper: 2.0.2 - istanbul-lib-report: 3.0.1 - - jest-changed-files@29.7.0: - dependencies: - execa: 5.1.1 - jest-util: 29.7.0 - p-limit: 3.1.0 - - jest-circus@29.7.0: - dependencies: - '@jest/environment': 29.7.0 - '@jest/expect': 29.7.0 - '@jest/test-result': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - chalk: 4.1.2 - co: 4.6.0 - dedent: 1.5.1 - is-generator-fn: 2.1.0 - jest-each: 29.7.0 - jest-matcher-utils: 29.7.0 - jest-message-util: 29.7.0 - jest-runtime: 29.7.0 - jest-snapshot: 29.7.0 - jest-util: 29.7.0 - p-limit: 3.1.0 - pretty-format: 29.7.0 - pure-rand: 6.1.0 - slash: 3.0.0 - stack-utils: 2.0.6 - transitivePeerDependencies: - - babel-plugin-macros - - supports-color - - jest-cli@29.7.0(@types/node@20.12.2): - dependencies: - '@jest/core': 29.7.0 - '@jest/test-result': 29.7.0 - '@jest/types': 29.6.3 - chalk: 4.1.2 - create-jest: 29.7.0(@types/node@20.12.2) - exit: 0.1.2 - import-local: 3.1.0 - jest-config: 29.7.0(@types/node@20.12.2) - jest-util: 29.7.0 - jest-validate: 29.7.0 - yargs: 17.7.2 - transitivePeerDependencies: - - '@types/node' - - babel-plugin-macros - - supports-color - - ts-node - - jest-config@29.7.0(@types/node@20.12.2): - dependencies: - '@babel/core': 7.24.3 - '@jest/test-sequencer': 29.7.0 - '@jest/types': 29.6.3 - babel-jest: 29.7.0(@babel/core@7.24.3) - chalk: 4.1.2 - ci-info: 3.9.0 - deepmerge: 4.3.1 - glob: 7.2.3 - graceful-fs: 4.2.11 - jest-circus: 29.7.0 - jest-environment-node: 29.7.0 - jest-get-type: 29.6.3 - jest-regex-util: 29.6.3 - jest-resolve: 29.7.0 - jest-runner: 29.7.0 - jest-util: 29.7.0 - jest-validate: 29.7.0 - micromatch: 4.0.5 - parse-json: 5.2.0 - pretty-format: 29.7.0 - slash: 3.0.0 - strip-json-comments: 3.1.1 - optionalDependencies: - '@types/node': 20.12.2 - transitivePeerDependencies: - - babel-plugin-macros - - supports-color - - jest-diff@29.7.0: - dependencies: - chalk: 4.1.2 - diff-sequences: 29.6.3 - jest-get-type: 29.6.3 - pretty-format: 29.7.0 - - jest-docblock@29.7.0: - dependencies: - detect-newline: 3.1.0 + flatted@3.3.1: {} - jest-each@29.7.0: + form-data@4.0.0: dependencies: - '@jest/types': 29.6.3 - chalk: 4.1.2 - jest-get-type: 29.6.3 - jest-util: 29.7.0 - pretty-format: 29.7.0 + asynckit: 0.4.0 + combined-stream: 1.0.8 + mime-types: 2.1.35 - jest-environment-node@29.7.0: - dependencies: - '@jest/environment': 29.7.0 - '@jest/fake-timers': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - jest-mock: 29.7.0 - jest-util: 29.7.0 + fs.realpath@1.0.0: {} + + fsevents@2.3.3: + optional: true + + get-caller-file@2.0.5: {} - jest-get-type@29.6.3: {} + get-east-asian-width@1.2.0: {} + + get-func-name@2.0.2: {} + + get-stream@8.0.1: {} - jest-haste-map@29.7.0: + glob-parent@5.1.2: dependencies: - '@jest/types': 29.6.3 - '@types/graceful-fs': 4.1.9 - '@types/node': 20.12.2 - anymatch: 3.1.3 - fb-watchman: 2.0.2 - graceful-fs: 4.2.11 - jest-regex-util: 29.6.3 - jest-util: 29.7.0 - jest-worker: 29.7.0 - micromatch: 4.0.5 - walker: 1.0.8 - optionalDependencies: - fsevents: 2.3.3 + is-glob: 4.0.3 - jest-leak-detector@29.7.0: + glob-parent@6.0.2: dependencies: - jest-get-type: 29.6.3 - pretty-format: 29.7.0 + is-glob: 4.0.3 - jest-matcher-utils@29.7.0: + glob@7.2.3: dependencies: - chalk: 4.1.2 - jest-diff: 29.7.0 - jest-get-type: 29.6.3 - pretty-format: 29.7.0 + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 3.1.2 + once: 1.4.0 + path-is-absolute: 1.0.1 - jest-message-util@29.7.0: + globals@13.24.0: dependencies: - '@babel/code-frame': 7.24.2 - '@jest/types': 29.6.3 - '@types/stack-utils': 2.0.3 - chalk: 4.1.2 - graceful-fs: 4.2.11 - micromatch: 4.0.5 - pretty-format: 29.7.0 - slash: 3.0.0 - stack-utils: 2.0.6 + type-fest: 0.20.2 - jest-mock@29.7.0: + globby@11.1.0: dependencies: - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - jest-util: 29.7.0 + array-union: 2.1.0 + dir-glob: 3.0.1 + fast-glob: 3.3.2 + ignore: 5.3.1 + merge2: 1.4.1 + slash: 3.0.0 - jest-pnp-resolver@1.2.3(jest-resolve@29.7.0): - optionalDependencies: - jest-resolve: 29.7.0 + graphemer@1.4.0: {} - jest-regex-util@29.6.3: {} + hammerjs@2.0.8: {} - jest-resolve-dependencies@29.7.0: - dependencies: - jest-regex-util: 29.6.3 - jest-snapshot: 29.7.0 - transitivePeerDependencies: - - supports-color + has-flag@4.0.0: {} - jest-resolve@29.7.0: + html-encoding-sniffer@4.0.0: dependencies: - chalk: 4.1.2 - graceful-fs: 4.2.11 - jest-haste-map: 29.7.0 - jest-pnp-resolver: 1.2.3(jest-resolve@29.7.0) - jest-util: 29.7.0 - jest-validate: 29.7.0 - resolve: 1.22.8 - resolve.exports: 2.0.2 - slash: 3.0.0 + whatwg-encoding: 3.1.1 - jest-runner@29.7.0: + http-proxy-agent@7.0.2: dependencies: - '@jest/console': 29.7.0 - '@jest/environment': 29.7.0 - '@jest/test-result': 29.7.0 - '@jest/transform': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - chalk: 4.1.2 - emittery: 0.13.1 - graceful-fs: 4.2.11 - jest-docblock: 29.7.0 - jest-environment-node: 29.7.0 - jest-haste-map: 29.7.0 - jest-leak-detector: 29.7.0 - jest-message-util: 29.7.0 - jest-resolve: 29.7.0 - jest-runtime: 29.7.0 - jest-util: 29.7.0 - jest-watcher: 29.7.0 - jest-worker: 29.7.0 - p-limit: 3.1.0 - source-map-support: 0.5.13 + agent-base: 7.1.1 + debug: 4.3.4 transitivePeerDependencies: - supports-color - jest-runtime@29.7.0: + https-proxy-agent@7.0.4: dependencies: - '@jest/environment': 29.7.0 - '@jest/fake-timers': 29.7.0 - '@jest/globals': 29.7.0 - '@jest/source-map': 29.6.3 - '@jest/test-result': 29.7.0 - '@jest/transform': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - chalk: 4.1.2 - cjs-module-lexer: 1.2.3 - collect-v8-coverage: 1.0.2 - glob: 7.2.3 - graceful-fs: 4.2.11 - jest-haste-map: 29.7.0 - jest-message-util: 29.7.0 - jest-mock: 29.7.0 - jest-regex-util: 29.6.3 - jest-resolve: 29.7.0 - jest-snapshot: 29.7.0 - jest-util: 29.7.0 - slash: 3.0.0 - strip-bom: 4.0.0 + agent-base: 7.1.1 + debug: 4.3.4 transitivePeerDependencies: - supports-color - jest-snapshot@29.7.0: - dependencies: - '@babel/core': 7.24.3 - '@babel/generator': 7.24.1 - '@babel/plugin-syntax-jsx': 7.24.1(@babel/core@7.24.3) - '@babel/plugin-syntax-typescript': 7.24.1(@babel/core@7.24.3) - '@babel/types': 7.24.0 - '@jest/expect-utils': 29.7.0 - '@jest/transform': 29.7.0 - '@jest/types': 29.6.3 - babel-preset-current-node-syntax: 1.0.1(@babel/core@7.24.3) - chalk: 4.1.2 - expect: 29.7.0 - graceful-fs: 4.2.11 - jest-diff: 29.7.0 - jest-get-type: 29.6.3 - jest-matcher-utils: 29.7.0 - jest-message-util: 29.7.0 - jest-util: 29.7.0 - natural-compare: 1.4.0 - pretty-format: 29.7.0 - semver: 7.6.0 - transitivePeerDependencies: - - supports-color + human-signals@5.0.0: {} - jest-util@29.7.0: + iconv-lite@0.6.3: dependencies: - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - chalk: 4.1.2 - ci-info: 3.9.0 - graceful-fs: 4.2.11 - picomatch: 2.3.1 + safer-buffer: 2.1.2 + + idb-keyval@6.2.1: {} + + ignore@5.3.1: {} - jest-validate@29.7.0: + import-fresh@3.3.0: dependencies: - '@jest/types': 29.6.3 - camelcase: 6.3.0 - chalk: 4.1.2 - jest-get-type: 29.6.3 - leven: 3.1.0 - pretty-format: 29.7.0 + parent-module: 1.0.1 + resolve-from: 4.0.0 + + imurmurhash@0.1.4: {} - jest-watcher@29.7.0: + inflight@1.0.6: dependencies: - '@jest/test-result': 29.7.0 - '@jest/types': 29.6.3 - '@types/node': 20.12.2 - ansi-escapes: 4.3.2 - chalk: 4.1.2 - emittery: 0.13.1 - jest-util: 29.7.0 - string-length: 4.0.2 + once: 1.4.0 + wrappy: 1.0.2 + + inherits@2.0.4: {} + + is-binary-path@2.1.0: + dependencies: + binary-extensions: 2.3.0 + + is-extglob@2.1.1: {} + + is-fullwidth-code-point@3.0.0: {} + + is-fullwidth-code-point@4.0.0: {} - jest-worker@29.7.0: + is-fullwidth-code-point@5.0.0: dependencies: - '@types/node': 20.12.2 - jest-util: 29.7.0 - merge-stream: 2.0.0 - supports-color: 8.1.1 + get-east-asian-width: 1.2.0 - jest@29.7.0(@types/node@20.12.2): + is-glob@4.0.3: dependencies: - '@jest/core': 29.7.0 - '@jest/types': 29.6.3 - import-local: 3.1.0 - jest-cli: 29.7.0(@types/node@20.12.2) - transitivePeerDependencies: - - '@types/node' - - babel-plugin-macros - - supports-color - - ts-node + is-extglob: 2.1.1 + + is-number@7.0.0: {} + + is-path-inside@3.0.3: {} + + is-potential-custom-element-name@1.0.1: {} + + is-stream@3.0.0: {} + + isexe@2.0.0: {} js-tokens@4.0.0: {} - js-yaml@3.14.1: - dependencies: - argparse: 1.0.10 - esprima: 4.0.1 + js-tokens@9.0.0: {} js-yaml@4.1.0: dependencies: argparse: 2.0.1 - jsesc@2.5.2: {} + jsdom@24.1.0: + dependencies: + cssstyle: 4.0.1 + data-urls: 5.0.0 + decimal.js: 10.4.3 + form-data: 4.0.0 + html-encoding-sniffer: 4.0.0 + http-proxy-agent: 7.0.2 + https-proxy-agent: 7.0.4 + is-potential-custom-element-name: 1.0.1 + nwsapi: 2.2.10 + parse5: 7.1.2 + rrweb-cssom: 0.7.1 + saxes: 6.0.0 + symbol-tree: 3.2.4 + tough-cookie: 4.1.4 + w3c-xmlserializer: 5.0.0 + webidl-conversions: 7.0.0 + whatwg-encoding: 3.1.1 + whatwg-mimetype: 4.0.0 + whatwg-url: 14.0.0 + ws: 8.17.0 + xml-name-validator: 5.0.0 + transitivePeerDependencies: + - bufferutil + - supports-color + - utf-8-validate json-buffer@3.0.1: {} - json-parse-even-better-errors@2.3.1: {} - json-schema-traverse@0.4.1: {} json-stable-stringify-without-jsonify@1.0.1: {} - json5@2.2.3: {} - keyv@4.5.4: dependencies: json-buffer: 3.0.1 - kleur@3.0.3: {} - - leven@3.1.0: {} - levn@0.4.1: dependencies: prelude-ls: 1.2.1 @@ -4526,8 +3529,6 @@ snapshots: lilconfig@3.0.0: {} - lines-and-columns@1.2.4: {} - lint-staged@15.2.2: dependencies: chalk: 5.3.0 @@ -4552,6 +3553,11 @@ snapshots: rfdc: 1.3.1 wrap-ansi: 9.0.0 + local-pkg@0.5.0: + dependencies: + mlly: 1.7.1 + pkg-types: 1.1.1 + locate-path@5.0.0: dependencies: p-locate: 4.1.0 @@ -4574,21 +3580,17 @@ snapshots: dependencies: js-tokens: 4.0.0 - lru-cache@5.1.1: + loupe@2.3.7: dependencies: - yallist: 3.1.1 + get-func-name: 2.0.2 lru-cache@6.0.0: dependencies: yallist: 4.0.0 - make-dir@4.0.0: - dependencies: - semver: 7.6.0 - - makeerror@1.0.12: + magic-string@0.30.10: dependencies: - tmpl: 1.0.5 + '@jridgewell/sourcemap-codec': 1.4.15 merge-stream@2.0.0: {} @@ -4599,6 +3601,12 @@ snapshots: braces: 3.0.2 picomatch: 2.3.1 + mime-db@1.52.0: {} + + mime-types@2.1.35: + dependencies: + mime-db: 1.52.0 + mimic-fn@2.1.0: {} mimic-fn@4.0.0: {} @@ -4611,26 +3619,29 @@ snapshots: dependencies: brace-expansion: 2.0.1 - ms@2.1.2: {} + mlly@1.7.1: + dependencies: + acorn: 8.11.3 + pathe: 1.1.2 + pkg-types: 1.1.1 + ufo: 1.5.3 - natural-compare@1.4.0: {} + ms@2.1.2: {} - node-int64@0.4.0: {} + nanoid@3.3.7: {} - node-releases@2.0.14: {} + natural-compare@1.4.0: {} normalize-path@3.0.0: {} nouislider@15.7.1: {} - npm-run-path@4.0.1: - dependencies: - path-key: 3.1.1 - npm-run-path@5.3.0: dependencies: path-key: 4.0.0 + nwsapi@2.2.10: {} + object-assign@4.1.1: {} once@1.4.0: @@ -4674,6 +3685,10 @@ snapshots: dependencies: yocto-queue: 0.1.0 + p-limit@5.0.0: + dependencies: + yocto-queue: 1.0.0 + p-locate@4.1.0: dependencies: p-limit: 2.3.0 @@ -4688,12 +3703,9 @@ snapshots: dependencies: callsites: 3.1.0 - parse-json@5.2.0: + parse5@7.1.2: dependencies: - '@babel/code-frame': 7.24.2 - error-ex: 1.3.2 - json-parse-even-better-errors: 2.3.1 - lines-and-columns: 1.2.4 + entities: 4.5.0 path-exists@4.0.0: {} @@ -4703,10 +3715,12 @@ snapshots: path-key@4.0.0: {} - path-parse@1.0.7: {} - path-type@4.0.0: {} + pathe@1.1.2: {} + + pathval@1.1.1: {} + peerjs-js-binarypack@2.1.0: {} peerjs@1.5.4: @@ -4722,14 +3736,20 @@ snapshots: pidtree@0.6.0: {} - pirates@4.0.6: {} - - pkg-dir@4.2.0: + pkg-types@1.1.1: dependencies: - find-up: 4.1.0 + confbox: 0.1.7 + mlly: 1.7.1 + pathe: 1.1.2 pngjs@5.0.0: {} + postcss@8.4.38: + dependencies: + nanoid: 3.3.7 + picocolors: 1.0.0 + source-map-js: 1.2.0 + prelude-ls@1.2.1: {} prettier@3.0.2: {} @@ -4740,11 +3760,6 @@ snapshots: ansi-styles: 5.2.0 react-is: 18.2.0 - prompts@2.4.2: - dependencies: - kleur: 3.0.3 - sisteransi: 1.0.5 - prop-types@15.8.1: dependencies: loose-envify: 1.4.0 @@ -4794,9 +3809,9 @@ snapshots: prosemirror-state: 1.4.3 prosemirror-transform: 1.8.0 - punycode@2.3.1: {} + psl@1.9.0: {} - pure-rand@6.1.0: {} + punycode@2.3.1: {} qrcode@1.5.3: dependencies: @@ -4805,6 +3820,8 @@ snapshots: pngjs: 5.0.0 yargs: 15.4.1 + querystringify@2.2.0: {} + queue-microtask@1.2.3: {} react-is@16.13.1: {} @@ -4819,22 +3836,10 @@ snapshots: require-main-filename@2.0.0: {} - resolve-cwd@3.0.0: - dependencies: - resolve-from: 5.0.0 + requires-port@1.0.0: {} resolve-from@4.0.0: {} - resolve-from@5.0.0: {} - - resolve.exports@2.0.2: {} - - resolve@1.22.8: - dependencies: - is-core-module: 2.13.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 - restore-cursor@4.0.0: dependencies: onetime: 5.1.2 @@ -4848,17 +3853,47 @@ snapshots: dependencies: glob: 7.2.3 + rollup@4.18.0: + dependencies: + '@types/estree': 1.0.5 + optionalDependencies: + '@rollup/rollup-android-arm-eabi': 4.18.0 + '@rollup/rollup-android-arm64': 4.18.0 + '@rollup/rollup-darwin-arm64': 4.18.0 + '@rollup/rollup-darwin-x64': 4.18.0 + '@rollup/rollup-linux-arm-gnueabihf': 4.18.0 + '@rollup/rollup-linux-arm-musleabihf': 4.18.0 + '@rollup/rollup-linux-arm64-gnu': 4.18.0 + '@rollup/rollup-linux-arm64-musl': 4.18.0 + '@rollup/rollup-linux-powerpc64le-gnu': 4.18.0 + '@rollup/rollup-linux-riscv64-gnu': 4.18.0 + '@rollup/rollup-linux-s390x-gnu': 4.18.0 + '@rollup/rollup-linux-x64-gnu': 4.18.0 + '@rollup/rollup-linux-x64-musl': 4.18.0 + '@rollup/rollup-win32-arm64-msvc': 4.18.0 + '@rollup/rollup-win32-ia32-msvc': 4.18.0 + '@rollup/rollup-win32-x64-msvc': 4.18.0 + fsevents: 2.3.3 + rope-sequence@1.3.4: {} + rrweb-cssom@0.6.0: {} + + rrweb-cssom@0.7.1: {} + run-parallel@1.2.0: dependencies: queue-microtask: 1.2.3 + safer-buffer@2.1.2: {} + sax@1.3.0: {} - sdp@3.2.0: {} + saxes@6.0.0: + dependencies: + xmlchars: 2.2.0 - semver@6.3.1: {} + sdp@3.2.0: {} semver@7.6.0: dependencies: @@ -4877,12 +3912,12 @@ snapshots: '@floating-ui/dom': 1.6.3 deepmerge: 4.3.1 + siginfo@2.0.0: {} + signal-exit@3.0.7: {} signal-exit@4.1.0: {} - sisteransi@1.0.5: {} - slash@3.0.0: {} slice-ansi@5.0.0: @@ -4899,18 +3934,11 @@ snapshots: sortablejs@1.15.2: {} - source-map-support@0.5.13: - dependencies: - buffer-from: 1.1.2 - source-map: 0.6.1 - - source-map@0.6.1: {} + source-map-js@1.2.0: {} - sprintf-js@1.0.3: {} + stackback@0.0.2: {} - stack-utils@2.0.6: - dependencies: - escape-string-regexp: 2.0.0 + std-env@3.7.0: {} stockfish-mv.wasm@0.6.1: {} @@ -4922,11 +3950,6 @@ snapshots: string-argv@0.3.2: {} - string-length@4.0.2: - dependencies: - char-regex: 1.0.2 - strip-ansi: 6.0.1 - string-width@4.2.3: dependencies: emoji-regex: 8.0.0 @@ -4947,28 +3970,18 @@ snapshots: dependencies: ansi-regex: 6.0.1 - strip-bom@4.0.0: {} - - strip-final-newline@2.0.0: {} - strip-final-newline@3.0.0: {} strip-json-comments@3.1.1: {} - supports-color@5.5.0: + strip-literal@2.1.0: dependencies: - has-flag: 3.0.0 + js-tokens: 9.0.0 supports-color@7.2.0: dependencies: has-flag: 4.0.0 - supports-color@8.1.1: - dependencies: - has-flag: 4.0.0 - - supports-preserve-symlinks-flag@1.0.0: {} - svg.draggable.js@2.2.2: dependencies: svg.js: 2.7.1 @@ -5000,26 +4013,35 @@ snapshots: dependencies: svg.js: 2.7.1 - tablesort@5.3.0: {} + symbol-tree@3.2.4: {} - test-exclude@6.0.0: - dependencies: - '@istanbuljs/schema': 0.1.3 - glob: 7.2.3 - minimatch: 3.1.2 + tablesort@5.3.0: {} text-table@0.2.0: {} textarea-caret@3.1.0: {} - tmpl@1.0.5: {} + tinybench@2.8.0: {} + + tinypool@0.8.4: {} - to-fast-properties@2.0.0: {} + tinyspy@2.2.1: {} to-regex-range@5.0.1: dependencies: is-number: 7.0.0 + tough-cookie@4.1.4: + dependencies: + psl: 1.9.0 + punycode: 2.3.1 + universalify: 0.2.0 + url-parse: 1.5.10 + + tr46@5.0.0: + dependencies: + punycode: 2.3.1 + tree-kill@1.2.2: {} ts-api-utils@1.3.0(typescript@5.4.3): @@ -5034,33 +4056,88 @@ snapshots: type-fest@0.20.2: {} - type-fest@0.21.3: {} - types-serviceworker@0.0.1: {} typescript@5.4.3: {} + ufo@1.5.3: {} + undate@0.3.0: {} undici-types@5.26.5: {} - update-browserslist-db@1.0.13(browserslist@4.23.0): - dependencies: - browserslist: 4.23.0 - escalade: 3.1.2 - picocolors: 1.0.0 + universalify@0.2.0: {} uri-js@4.4.1: dependencies: punycode: 2.3.1 + url-parse@1.5.10: + dependencies: + querystringify: 2.2.0 + requires-port: 1.0.0 + uuid@9.0.0: {} - v8-to-istanbul@9.2.0: + vite-node@1.6.0(@types/node@20.12.2): + dependencies: + cac: 6.7.14 + debug: 4.3.4 + pathe: 1.1.2 + picocolors: 1.0.0 + vite: 5.3.0(@types/node@20.12.2) + transitivePeerDependencies: + - '@types/node' + - less + - lightningcss + - sass + - stylus + - sugarss + - supports-color + - terser + + vite@5.3.0(@types/node@20.12.2): + dependencies: + esbuild: 0.21.5 + postcss: 8.4.38 + rollup: 4.18.0 + optionalDependencies: + '@types/node': 20.12.2 + fsevents: 2.3.3 + + vitest@1.6.0(@types/node@20.12.2)(jsdom@24.1.0): dependencies: - '@jridgewell/trace-mapping': 0.3.25 - '@types/istanbul-lib-coverage': 2.0.6 - convert-source-map: 2.0.0 + '@vitest/expect': 1.6.0 + '@vitest/runner': 1.6.0 + '@vitest/snapshot': 1.6.0 + '@vitest/spy': 1.6.0 + '@vitest/utils': 1.6.0 + acorn-walk: 8.3.2 + chai: 4.4.1 + debug: 4.3.4 + execa: 8.0.1 + local-pkg: 0.5.0 + magic-string: 0.30.10 + pathe: 1.1.2 + picocolors: 1.0.0 + std-env: 3.7.0 + strip-literal: 2.1.0 + tinybench: 2.8.0 + tinypool: 0.8.4 + vite: 5.3.0(@types/node@20.12.2) + vite-node: 1.6.0(@types/node@20.12.2) + why-is-node-running: 2.2.2 + optionalDependencies: + '@types/node': 20.12.2 + jsdom: 24.1.0 + transitivePeerDependencies: + - less + - lightningcss + - sass + - stylus + - sugarss + - supports-color + - terser vosk-browser@0.0.8: dependencies: @@ -5068,27 +4145,39 @@ snapshots: w3c-keyname@2.2.8: {} - walker@1.0.8: + w3c-xmlserializer@5.0.0: dependencies: - makeerror: 1.0.12 + xml-name-validator: 5.0.0 + + webidl-conversions@7.0.0: {} webrtc-adapter@9.0.1: dependencies: sdp: 3.2.0 + whatwg-encoding@3.1.1: + dependencies: + iconv-lite: 0.6.3 + + whatwg-mimetype@4.0.0: {} + + whatwg-url@14.0.0: + dependencies: + tr46: 5.0.0 + webidl-conversions: 7.0.0 + which-module@2.0.1: {} which@2.0.2: dependencies: isexe: 2.0.0 - wrap-ansi@6.2.0: + why-is-node-running@2.2.2: dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 + siginfo: 2.0.0 + stackback: 0.0.2 - wrap-ansi@7.0.0: + wrap-ansi@6.2.0: dependencies: ansi-styles: 4.3.0 string-width: 4.2.3 @@ -5102,10 +4191,9 @@ snapshots: wrappy@1.0.2: {} - write-file-atomic@4.0.2: - dependencies: - imurmurhash: 0.1.4 - signal-exit: 3.0.7 + ws@8.17.0: {} + + xml-name-validator@5.0.0: {} xml2js@0.5.0: dependencies: @@ -5114,11 +4202,9 @@ snapshots: xmlbuilder@11.0.1: {} - y18n@4.0.3: {} - - y18n@5.0.8: {} + xmlchars@2.2.0: {} - yallist@3.1.1: {} + y18n@4.0.3: {} yallist@4.0.0: {} @@ -5129,8 +4215,6 @@ snapshots: camelcase: 5.3.1 decamelize: 1.2.0 - yargs-parser@21.1.1: {} - yargs@15.4.1: dependencies: cliui: 6.0.0 @@ -5145,16 +4229,8 @@ snapshots: y18n: 4.0.3 yargs-parser: 18.1.3 - yargs@17.7.2: - dependencies: - cliui: 8.0.1 - escalade: 3.1.2 - get-caller-file: 2.0.5 - require-directory: 2.1.1 - string-width: 4.2.3 - y18n: 5.0.8 - yargs-parser: 21.1.1 - yocto-queue@0.1.0: {} + yocto-queue@1.0.0: {} + zxcvbn@4.4.2: {} diff --git a/ui/@types/lichess/index.d.ts b/ui/@types/lichess/index.d.ts index 8b3963431e128..d9c566a98c71b 100644 --- a/ui/@types/lichess/index.d.ts +++ b/ui/@types/lichess/index.d.ts @@ -13,7 +13,11 @@ // file://./../../site/src/site.ts interface Site { debug: boolean; - info: any; + info: { + commit: string; + message: string; + date: string; + }; StrongSocket: { // file://./../../site/src/socket.ts new (url: string, version: number | false, cfg?: any): any; diff --git a/ui/README.md b/ui/README.md index 0722d069f0022..7da754f10cdae 100644 --- a/ui/README.md +++ b/ui/README.md @@ -6,7 +6,7 @@ Client builds are performed by the ui/build script. Stick to `ui/build -w` and l Usage examples: -``` +```bash ui/build # builds all client assets in dev mode ui/build -w # builds all client assets and watches for changes ui/build -p # builds minified client assets (prod builds) @@ -22,11 +22,14 @@ Usage examples: ## Testing -The frontend uses the Jest testing framework. +The frontend uses the [Vitest](https://vitest.dev/) testing framework. + +```bash +cd ui -``` -cd ui/ pnpm test +## or +pnpm test:watch ``` ## CSS diff --git a/ui/chat/src/spam.ts b/ui/chat/src/spam.ts index cf66bf23856b0..459bbe5baccc9 100644 --- a/ui/chat/src/spam.ts +++ b/ui/chat/src/spam.ts @@ -5,7 +5,7 @@ export const skip = (txt: string) => (suspLink(txt) || followMe(txt)) && !isKnow export const selfReport = (txt: string) => { if (isKnownSpammer()) return; const hasSuspLink = suspLink(txt); - if (hasSuspLink) xhr.text(`/jslog/${window.location.href.substr(-12)}?n=spam`, { method: 'post' }); + if (hasSuspLink) xhr.text(`/jslog/${window.location.href.slice(-12)}?n=spam`, { method: 'post' }); if (hasSuspLink || followMe(txt)) site.storage.set('chat-spam', '1'); }; diff --git a/ui/chat/tests/spam.test.ts b/ui/chat/tests/spam.test.ts new file mode 100644 index 0000000000000..1d43e1d0b5aa7 --- /dev/null +++ b/ui/chat/tests/spam.test.ts @@ -0,0 +1,25 @@ +import { expect, test, vi } from 'vitest'; +import { selfReport } from '../src/spam'; +import * as xhr from 'common/xhr'; + +test('self report', () => { + vi.stubGlobal('window', { location: { href: 'https://lichess.org/abcdef123456' } }); + + vi.stubGlobal('site', { + storage: { + set: (key: string, value: string) => { + expect(key).toBe('chat-spam'); + expect(value).toBe('1'); + }, + get: () => '0', + }, + }); + + const spy = vi.spyOn(xhr, 'text').mockImplementation((url: string, init?: RequestInit) => { + expect(url).toBe('/jslog/abcdef123456?n=spam'); + return Promise.resolve('ok'); + }); + + selfReport('check out bit.ly/spam'); + expect(spy).toHaveBeenCalled(); +}); diff --git a/ui/common/src/richText.ts b/ui/common/src/richText.ts index 0a9d44385904b..d037c7a11141d 100644 --- a/ui/common/src/richText.ts +++ b/ui/common/src/richText.ts @@ -6,7 +6,7 @@ import { VNode, Hooks } from 'snabbdom'; export const linkRegex = /(^|[\s\n]|<[A-Za-z]*\/?>)((?:(?:https?|ftp):\/\/|lichess\.org)[\-A-Z0-9+\u0026\u2019@#\/%?=()~_|!:,.;]*[\-A-Z0-9+\u0026@#\/%=~()_|])/gi; export const newLineRegex = /\n/g; -export const userPattern = /(^|[^\w@#/])@([a-z0-9][a-z0-9_-]{0,28}[a-z0-9])/gi; +export const userPattern = /(^|[^\w@#/])@([a-zA-Z0-9_-]{2,30})/gi; // looks like it has a @mention or #gameid or a url.tld export const isMoreThanText = (str: string) => /(\n|(@|#|\.)\w{2,}|(board|game) \d)/i.test(str); @@ -59,8 +59,8 @@ export function richHTML(text: string, newLines = true): Hooks { const linkPattern = /\b\b(?:https?:\/\/)?(lichess\.org\/[-–—\w+&'@#\/%?=()~|!:,.;]+[\w+&@#\/%=~|])/gi; const pawnDropPattern = /^[a-h][2-7]$/; -const movePattern = - /\b(\d+)\s*(\.+)\s*(?:[o0-]+[o0]|[NBRQKP\u2654\u2655\u2656\u2657\u2658\u2659]?[a-h]?[1-8]?[x@]?[a-z][1-8](?:=[NBRQK\u2654\u2655\u2656\u2657\u2658\u2659])?)\+?#?[!\?=]{0,5}/gi; +export const movePattern = + /\b(\d+)\s*(\.+)\s*(?:[o0-]+[o0]|[NBRQKP\u2654\u2655\u2656\u2657\u2658\u2659]?[a-h]?[1-8]?[x@]?[a-h][1-8](?:=[NBRQK\u2654\u2655\u2656\u2657\u2658\u2659])?)\+?#?[!\?=]{0,5}/gi; const boardPattern = /\b(?:board|game)\s(\d{1,2})/gi; function moveReplacer(match: string, turn: number, dots: string) { diff --git a/ui/common/tests/richText.test.ts b/ui/common/tests/richText.test.ts new file mode 100644 index 0000000000000..1abc3a98c49a8 --- /dev/null +++ b/ui/common/tests/richText.test.ts @@ -0,0 +1,31 @@ +import { describe, expect, test, vi } from 'vitest'; +import { movePattern, userPattern } from '../src/richText'; + +describe('test regex patterns', () => { + test('username mentions', () => { + expect('@foo'.match(userPattern)).toStrictEqual(['@foo']); + expect('@foo-'.match(userPattern)).toStrictEqual(['@foo-']); + expect('@__foo'.match(userPattern)).toStrictEqual(['@__foo']); + }); + + test.each([ + ['1.e4'], + ['1. e4'], + ['5...Nf6'], + ['5... Nf6'], + ['12.♔d1'], + ['10.O-O-O'], + ['10.o-o-o'], + ['10.0-0-0'], + ])('moves', move => { + expect(move.match(movePattern)).toStrictEqual([move]); + }); + + test('move with comment', () => { + expect('I considered 34. f7+ instead'.match(movePattern)).toStrictEqual(['34. f7+']); + }); + + test('not a move', () => { + expect('4.m3'.match(movePattern)).toBeNull(); + }); +}); diff --git a/ui/game/src/status.ts b/ui/game/src/status.ts index 44332623ee6cb..0e82af555f360 100644 --- a/ui/game/src/status.ts +++ b/ui/game/src/status.ts @@ -1,6 +1,6 @@ import { GameData } from './interfaces'; -// https://github.com/lichess-org/scalachess/blob/master/src/main/scala/Status.scala +// https://github.com/lichess-org/scalachess/blob/master/core/src/main/scala/Status.scala export const ids = { created: 10, diff --git a/ui/jest.config.js b/ui/jest.config.js deleted file mode 100644 index f83c256ad492d..0000000000000 --- a/ui/jest.config.js +++ /dev/null @@ -1,9 +0,0 @@ -/* eslint-env node */ -module.exports = { - testMatch: ['**/dist/**/*.test.js'], - testEnvironment: 'node', - transform: {}, - globals: { - site: {}, - }, -}; diff --git a/ui/keyboardMove/package.json b/ui/keyboardMove/package.json index 70b661488e72d..5af514527cb00 100644 --- a/ui/keyboardMove/package.json +++ b/ui/keyboardMove/package.json @@ -14,7 +14,6 @@ "author": "Thibault Duplessis", "license": "AGPL-3.0-or-later", "dependencies": { - "@jest/globals": "^29.7.0", "chess": "workspace:*", "common": "workspace:*", "game": "workspace:*", diff --git a/ui/keyboardMove/src/keyboardSubmit.test.ts b/ui/keyboardMove/tests/keyboardSubmit.test.ts similarity index 83% rename from ui/keyboardMove/src/keyboardSubmit.test.ts rename to ui/keyboardMove/tests/keyboardSubmit.test.ts index ddebe92f0ba31..16de93bf5a79d 100644 --- a/ui/keyboardMove/src/keyboardSubmit.test.ts +++ b/ui/keyboardMove/tests/keyboardSubmit.test.ts @@ -1,6 +1,6 @@ -import { jest, beforeEach, describe, expect, test } from '@jest/globals'; +import { beforeEach, describe, expect, test, vi } from 'vitest'; import { Prop, propWithEffect } from 'common'; -import { makeSubmit } from './keyboardSubmit'; +import { makeSubmit } from '../src/keyboardSubmit'; import { Dests, destsToUcis, sanWriter } from 'chess'; // Tips for working with this file: @@ -42,24 +42,18 @@ const defaultCtrl = { }; const defaultClear = unexpectedErrorThrower('clear'); -// we don't have access to DOM elements in jest (testEnvironment: 'node'), so we need to mock this -const input = { - value: '', - classList: { contains: () => false, toggle: () => {} }, -} as unknown as HTMLInputElement; - describe('keyboardSubmit', () => { - let mockClear = jest.fn(); + let mockClear = vi.fn(); beforeEach(() => { - mockClear = jest.fn(); + mockClear = vi.fn(); }); test('resigns game', () => { - const mockResign = jest.fn(); + const mockResign = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, resign: mockResign, @@ -75,10 +69,10 @@ describe('keyboardSubmit', () => { }); test('draws game', () => { - const mockDraw = jest.fn(); + const mockDraw = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, draw: mockDraw, @@ -94,10 +88,10 @@ describe('keyboardSubmit', () => { }); test('goes to next puzzle', () => { - const mockNext = jest.fn(); + const mockNext = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, next: mockNext, @@ -113,10 +107,10 @@ describe('keyboardSubmit', () => { }); test('up votes puzzle', () => { - const mockVote = jest.fn(); + const mockVote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, vote: mockVote, @@ -133,10 +127,10 @@ describe('keyboardSubmit', () => { }); test('down votes puzzle', () => { - const mockVote = jest.fn(); + const mockVote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, vote: mockVote, @@ -153,10 +147,10 @@ describe('keyboardSubmit', () => { }); test('reads out clock', () => { - const mockSpeakClock = jest.fn(); + const mockSpeakClock = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, speakClock: mockSpeakClock, @@ -172,10 +166,10 @@ describe('keyboardSubmit', () => { }); test('berserks a game', () => { - const mockGoBerserk = jest.fn(); + const mockGoBerserk = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, goBerserk: mockGoBerserk, @@ -190,11 +184,30 @@ describe('keyboardSubmit', () => { expect(mockClear).toHaveBeenCalledTimes(1); }); + test('speaks opponent name', () => { + vi.stubGlobal('site', { sound: { say: vi.fn() } }); + + const submit = makeSubmit( + { + input: document.createElement('input'), + ctrl: { + ...defaultCtrl, + opponent: 'opponent-name', + }, + }, + mockClear, + ); + + submit('who', { isTrusted: true }); + expect(site.sound.say).toHaveBeenCalledTimes(1); + expect(site.sound.say).toBeCalledWith('opponent-name', false, true); + }); + test('opens help modal with ?', () => { - const mockSetHelpModalOpen = jest.fn(); + const mockSetHelpModalOpen = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, helpModalOpen: mockSetHelpModalOpen as Prop, @@ -212,10 +225,10 @@ describe('keyboardSubmit', () => { describe('from starting position', () => { test('plays e4 via SAN', () => { - const mockSan = jest.fn(); + const mockSan = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -233,10 +246,10 @@ describe('keyboardSubmit', () => { }); test('selects e2 via UCI', () => { - const mockSelect = jest.fn(); + const mockSelect = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -254,11 +267,11 @@ describe('keyboardSubmit', () => { }); test('with e2 selected, plays e4 via UCI', () => { - const mockSan = jest.fn(); - const mockSelect = jest.fn(); + const mockSan = vi.fn(); + const mockSelect = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -281,10 +294,10 @@ describe('keyboardSubmit', () => { }); test('selects e2 via ICCF', () => { - const mockSelect = jest.fn(); + const mockSelect = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -302,11 +315,11 @@ describe('keyboardSubmit', () => { }); test('with e2 selected, plays e4 via ICCF', () => { - const mockSan = jest.fn(); - const mockSelect = jest.fn(); + const mockSan = vi.fn(); + const mockSelect = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -333,10 +346,10 @@ describe('keyboardSubmit', () => { const ambiguousPawnBishopCapture = '4k3/8/8/8/8/2r5/1P1B4/4K3 w - - 0 1'; test('does pawn capture', () => { - const mockSan = jest.fn(); + const mockSan = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(ambiguousPawnBishopCapture, { b2: ['c3'], d2: ['c3'] }), @@ -355,10 +368,10 @@ describe('keyboardSubmit', () => { }); test('does bishop capture', () => { - const mockSan = jest.fn(); + const mockSan = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(ambiguousPawnBishopCapture, { b2: ['c3'], d2: ['c3'] }), @@ -382,7 +395,7 @@ describe('keyboardSubmit', () => { test('does not castle short', () => { const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(ambiguousCastlingFen, { e1: ['c1', 'g1'] }), @@ -397,10 +410,10 @@ describe('keyboardSubmit', () => { }); test('does castle long', () => { - const mockSan = jest.fn(); + const mockSan = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(ambiguousCastlingFen, { e1: ['c1', 'g1'] }), @@ -422,10 +435,10 @@ describe('keyboardSubmit', () => { const promotablePawnFen = 'r3k3/1P6/8/8/8/8/8/4K3 w - - 0 1'; test('with no piece specified does not promote by advancing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -442,10 +455,10 @@ describe('keyboardSubmit', () => { }); test('with piece specified does promote by advancing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -462,10 +475,10 @@ describe('keyboardSubmit', () => { }); test('with no piece specified does not promote by capturing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -482,10 +495,10 @@ describe('keyboardSubmit', () => { }); test('with piece specified does promote by capturing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -503,10 +516,10 @@ describe('keyboardSubmit', () => { describe('with pawn selected', () => { test('with no piece specified does not promote by advancing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -524,10 +537,10 @@ describe('keyboardSubmit', () => { }); test('with piece specified does promote by advancing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -545,10 +558,10 @@ describe('keyboardSubmit', () => { }); test('with no piece specified does not promote by capturing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -566,10 +579,10 @@ describe('keyboardSubmit', () => { }); test('with piece specified does promote by capturing', () => { - const mockPromote = jest.fn(); + const mockPromote = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(promotablePawnFen, { b7: ['a8', 'b8'] }), @@ -590,10 +603,10 @@ describe('keyboardSubmit', () => { describe('in crazyhouse variant', () => { test('with incomplete crazyhouse entry does nothing', () => { - const mockDrop = jest.fn(); + const mockDrop = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -610,10 +623,10 @@ describe('keyboardSubmit', () => { }); test('with complete crazyhouse entry does a drop', () => { - const mockDrop = jest.fn(); + const mockDrop = vi.fn(); const submit = makeSubmit( { - input, + input: document.createElement('input'), ctrl: { ...defaultCtrl, legalSans: fenDestsToSans(startingFen, { e2: ['e4'] }), @@ -631,8 +644,9 @@ describe('keyboardSubmit', () => { }); test('with incorrect entry marks it wrong', () => { - input.classList.toggle = jest.fn() as any; - site.sound = { play: jest.fn() } as any; + vi.stubGlobal('site', { sound: { play: vi.fn() } }); + + const input = document.createElement('input'); const submit = makeSubmit( { input, @@ -646,8 +660,8 @@ describe('keyboardSubmit', () => { submit('j4', { isTrusted: true }); - expect(input.classList.toggle).toHaveBeenCalledTimes(1); - expect(input.classList.toggle).toBeCalledWith('wrong', true); + expect(input.classList.contains('wrong')).toBe(true); expect(site.sound.play).toHaveBeenCalledTimes(1); + expect(site.sound.play).toBeCalledWith('error'); }); }); diff --git a/ui/package.json b/ui/package.json index 19906028729d8..80d6ca633c58d 100644 --- a/ui/package.json +++ b/ui/package.json @@ -9,10 +9,11 @@ "author": "Thibault Duplessis", "license": "AGPL-3.0-or-later", "dependencies": { - "@types/jest": "^29.5.12", - "jest": "^29.7.0" + "jsdom": "^24.1.0", + "vitest": "^1.6.0" }, "scripts": { - "test": "node --experimental-vm-modules node_modules/jest/bin/jest.js" + "test": "vitest run", + "test:watch": "vitest" } } diff --git a/ui/site/src/clockWidget.ts b/ui/site/src/clockWidget.ts index 0ac793b5e148b..fa721560fd68c 100644 --- a/ui/site/src/clockWidget.ts +++ b/ui/site/src/clockWidget.ts @@ -14,7 +14,7 @@ export default function (el: HTMLElement, opts: Opts) { class ClockWidget { target: number; - interval: number; + interval: Timeout; constructor( readonly el: HTMLElement, private opts: Opts, diff --git a/ui/site/src/log.ts b/ui/site/src/log.ts index c4802ee89c8c0..933060ff3ec83 100644 --- a/ui/site/src/log.ts +++ b/ui/site/src/log.ts @@ -44,7 +44,9 @@ export default function makeLog(): LichessLog { } const log: LichessLog = async (...args: any[]) => { - const msg = `#${site.info ? `${site.info.commit.substr(0, 7)} - ` : ''}${args.map(stringify).join(' ')}`; + const msg = `#${site.info ? `${site.info.commit.substring(0, 7)} - ` : ''}${args + .map(stringify) + .join(' ')}`; let nextKey = Date.now(); console.log(...args); if (nextKey === lastKey) { diff --git a/ui/site/src/once.ts b/ui/site/src/once.ts index 633208e0ed09b..757bc47eb8466 100644 --- a/ui/site/src/once.ts +++ b/ui/site/src/once.ts @@ -2,9 +2,8 @@ import { storage } from './storage'; export default function once(key: string, mod?: 'always' | undefined) { if (mod === 'always') return true; - if (!storage.get(key)) { - storage.set(key, '1'); - return true; - } - return false; + if (storage.get(key)) return false; + + storage.set(key, '1'); + return true; } diff --git a/ui/site/src/powertip.ts b/ui/site/src/powertip.ts index 8bf17d8d2353a..0abe0bec83ed5 100644 --- a/ui/site/src/powertip.ts +++ b/ui/site/src/powertip.ts @@ -245,7 +245,7 @@ function cssCoordinates(): Coords { class DisplayController { scoped: { [key: string]: any } = {}; - hoverTimer?: number; + hoverTimer?: Timeout; el: WithTooltip; constructor( diff --git a/ui/site/tests/once.test.ts b/ui/site/tests/once.test.ts new file mode 100644 index 0000000000000..b11a6d4cb7f3a --- /dev/null +++ b/ui/site/tests/once.test.ts @@ -0,0 +1,14 @@ +import { describe, expect, test } from 'vitest'; +import once from '../src/once'; + +describe('test once', () => { + test('once', () => { + expect(once('foo')).toBe(true); + + // subsequent calls should return false + expect(once('foo')).toBe(false); + expect(once('foo')).toBe(false); + + expect(once('foo', 'always')).toBe(true); + }); +}); diff --git a/ui/site/tests/timeago.test.ts b/ui/site/tests/timeago.test.ts new file mode 100644 index 0000000000000..5b81377c10646 --- /dev/null +++ b/ui/site/tests/timeago.test.ts @@ -0,0 +1,40 @@ +import { beforeEach, describe, expect, test, vi } from 'vitest'; + +describe('test formatter', () => { + beforeEach(() => { + vi.resetModules(); + }); + + test.each([ + ['en-US', 'Jan 1, 2024, 5:00 PM'], + ['en-UK', '1 Jan 2024, 17:00'], + ['fr', '1 janv. 2024, 17:00'], + ])('lang code formatting', async (lang, expected) => { + document.documentElement.lang = lang; + const formatter = await import('../src/timeago').then(m => m.formatter); + expect(formatter()(new Date(2024, 0, 1, 17, 0, 0))).toBe(expected); + }); + + test.each([ + ['ar-ae'], + ['ar-bh'], + ['ar-dz'], + ['ar-eg'], + ['ar-iq'], + ['ar-jo'], + ['ar-kw'], + ['ar-lb'], + ['ar-ly'], + ['ar-ma'], + ['ar-om'], + ['ar-qa'], + ['ar-sa'], + ['ar-sy'], + ['ar-tn'], + ['ar-ye'], + ])('arabic lang code uses the gregorian calendar', async lang => { + document.documentElement.lang = lang; + const formatter = await import('../src/timeago').then(m => m.formatter); + expect(formatter()(new Date(2024, 0, 1, 17, 0, 0))).toBe('1 يناير 2024، 5:00 م'); + }); +}); diff --git a/ui/vitest.config.ts b/ui/vitest.config.ts new file mode 100644 index 0000000000000..647a9e54c8c99 --- /dev/null +++ b/ui/vitest.config.ts @@ -0,0 +1,7 @@ +import { defineConfig } from 'vitest/config'; + +export default defineConfig({ + test: { + environment: 'jsdom', + }, +}); From 7571264bdc65c0c1e2885129055e22e6d995c6ae Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 00:16:34 +0200 Subject: [PATCH 013/168] rename relay Plan --- modules/relay/src/main/RelaySync.scala | 2 +- modules/relay/src/main/RelayUpdatePlan.scala | 7 ++-- .../relay/src/test/RelayUpdatePlanTest.scala | 34 +++++++++---------- 3 files changed, 20 insertions(+), 23 deletions(-) diff --git a/modules/relay/src/main/RelaySync.scala b/modules/relay/src/main/RelaySync.scala index 0a89675098637..50b8ee4b7cc9f 100644 --- a/modules/relay/src/main/RelaySync.scala +++ b/modules/relay/src/main/RelaySync.scala @@ -20,7 +20,7 @@ final private class RelaySync( study <- studyApi.byId(rt.round.studyId).orFail("Missing relay study!") chapters <- chapterRepo.orderedByStudyLoadingAllInMemory(study.id) games = RelayInputSanity.fixGames(rawGames) - plan = RelayUpdatePlan(chapters, games).output + plan = RelayUpdatePlan(chapters, games) _ <- plan.reorder.so(studyApi.sortChapters(study.id, _)(who(study.ownerId))) updates <- plan.update.sequentially: (chapter, game) => updateChapter(rt.tour, study, game, chapter) diff --git a/modules/relay/src/main/RelayUpdatePlan.scala b/modules/relay/src/main/RelayUpdatePlan.scala index b62fca9e87c87..5e9b1df8de182 100644 --- a/modules/relay/src/main/RelayUpdatePlan.scala +++ b/modules/relay/src/main/RelayUpdatePlan.scala @@ -9,7 +9,7 @@ object RelayUpdatePlan: override def toString: String = s"Input(chapters = ${chapters.map(_.name)}, games = ${games.map(_.tags.names)})" - case class Output( + case class Plan( reorder: Option[List[StudyChapterId]], update: List[(Chapter, RelayGame)], append: RelayGames @@ -17,8 +17,6 @@ object RelayUpdatePlan: override def toString: String = s"Output(reorder = $reorder, update = ${update.map(_._1.name)}, append = ${append.map(_.tags.names)})" - case class Plan(input: Input, output: Output) - def apply(chapters: List[Chapter], games: RelayGames): Plan = apply(Input(chapters, games)) @@ -52,9 +50,8 @@ object RelayUpdatePlan: val ids = updates.map(_._1.id) Option.when(ids.size == chapters.size && ids != chapters.map(_.id))(ids) - val output = Output( + Plan( reorder = reorder, update = updates, append = appends ) - Plan(input, output) diff --git a/modules/relay/src/test/RelayUpdatePlanTest.scala b/modules/relay/src/test/RelayUpdatePlanTest.scala index 2e96615c59954..12d742f863d7a 100644 --- a/modules/relay/src/test/RelayUpdatePlanTest.scala +++ b/modules/relay/src/test/RelayUpdatePlanTest.scala @@ -7,84 +7,84 @@ class RelayUpdatePlanTest extends munit.FunSuite: import RelayPlanUpdateFixtures.* - def output(input: Input)(check: PartialFunction[Output, Unit]): Unit = - val out = RelayUpdatePlan(input).output + def output(input: Input)(check: PartialFunction[Plan, Unit]): Unit = + val out = RelayUpdatePlan(input) check.applyOrElse(out, _ => fail(s"Unexpected output: $out")) test("add no game to empty relay"): - assertEquals(RelayUpdatePlan(Input(Nil, games.take(0))).output, Output(None, Nil, Vector.empty)) + assertEquals(RelayUpdatePlan(Input(Nil, games.take(0))), Plan(None, Nil, Vector.empty)) test("add one game to empty relay"): output(Input(Nil, games.take(1))): - case Output(None, Nil, append) => assertEquals(append, games.take(1)) + case Plan(None, Nil, append) => assertEquals(append, games.take(1)) test("add all games to empty relay"): output(Input(Nil, games)): - case Output(None, Nil, append) => assertEquals(append, games) + case Plan(None, Nil, append) => assertEquals(append, games) test("add no game to relay with initial chapter"): assertEquals( - RelayUpdatePlan(Input(List(initialChapter), games.take(0))).output, - Output(None, Nil, Vector.empty) + RelayUpdatePlan(Input(List(initialChapter), games.take(0))), + Plan(None, Nil, Vector.empty) ) test("add one game to relay with initial chapter"): output(Input(List(initialChapter), games.take(1))): - case Output(None, update, Vector()) => + case Plan(None, update, Vector()) => assertEquals(update, List(initialChapter -> games(0))) test("add all games to relay with initial chapter"): output(Input(List(initialChapter), games)): - case Output(None, update, append) => + case Plan(None, update, append) => assertEquals(update, List(initialChapter -> games(0))) assertEquals(append, games.drop(1)) test("1 chapter, 1 game, matching tags"): output(Input(chapters.take(1), games.take(1))): - case Output(None, update, append) => + case Plan(None, update, append) => assertEquals(update, List(chapters(0) -> games(0))) assert(append.isEmpty) test("1 chapter, 1 game, not matching tags"): output(Input(chapters.take(1), games.drop(1).take(1))): - case Output(None, Nil, append) => assertEquals(append, games.drop(1).take(1)) + case Plan(None, Nil, append) => assertEquals(append, games.drop(1).take(1)) test("1 chapter, 5 games, first matching tags"): output(Input(chapters.take(1), games)): - case Output(None, update, append) => + case Plan(None, update, append) => assertEquals(update, List(chapters(0) -> games(0))) assertEquals(append, games.drop(1)) test("5 chapters, 5 games, matching tags"): output(Input(chapters, games)): - case Output(None, update, append) => + case Plan(None, update, append) => assertEquals(update, chapters.zip(games)) assert(append.isEmpty) test("2 chapters, 2 games, reverse order"): val in = Input(chapters.take(2), games.take(2).reverse) output(in): - case Output(reorder, update, Vector()) => + case Plan(reorder, update, Vector()) => assertEquals(update, in.chapters.reverse.zip(in.games)) assertEquals(reorder, in.chapters.reverse.map(_.id).some) test("5 chapters, 5 games, reverse order"): val in = Input(chapters, games.reverse) output(in): - case Output(Some(reorder), update, Vector()) => + case Plan(Some(reorder), update, Vector()) => assertEquals(update, in.chapters.reverse.zip(in.games)) assertEquals(reorder, in.chapters.reverse.map(_.id)) test("5 chapters, 2 games, reverse order"): output(Input(chapters, games.take(2).reverse)): - case Output(None, update, Vector()) => + case Plan(None, update, Vector()) => assertEquals(update, List(chapters(1) -> games(1), chapters(0) -> games(0))) test("2 chapters, 3 games, reverse order"): // A B, C B A val in = Input(chapters.take(2), games.take(3).reverse) output(in): - case Output(Some(reorder), update, append) => + case Plan(Some(reorder), update, append) => assertEquals(update, List(chapters(1) -> games(1), chapters(0) -> games(0))) assertEquals(reorder, in.chapters.reverse.map(_.id)) assertEquals(append, Vector(games(2))) From 8a53bf8660938849ecbba40c002073823338cc51 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 08:56:47 +0200 Subject: [PATCH 014/168] use scss variables --- ui/tree/css/_tree.scss | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index 9819ef30d6b1c..aac5fed98067a 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -374,11 +374,11 @@ } line.expand > a { - color: var(--c-font-dim); + color: $c-font-dim; } line.expand > a:hover { - color: var(--c-font); + color: $c-font; } inline { From 5ef62f45a3d6322152b42b91000e02a8c0c1eb8d Mon Sep 17 00:00:00 2001 From: Ben Olden-Cooligan Date: Fri, 14 Jun 2024 00:30:55 -0700 Subject: [PATCH 015/168] Don't render collapsed nodes (instead of just hiding) --- ui/analyse/src/treeView/columnView.ts | 39 ++++++++++++++------------- ui/analyse/src/treeView/inlineView.ts | 37 ++++++++++++------------- ui/tree/css/_tree.scss | 8 ------ 3 files changed, 39 insertions(+), 45 deletions(-) diff --git a/ui/analyse/src/treeView/columnView.ts b/ui/analyse/src/treeView/columnView.ts index 2dee229521913..472b62c93dd46 100644 --- a/ui/analyse/src/treeView/columnView.ts +++ b/ui/analyse/src/treeView/columnView.ts @@ -114,7 +114,9 @@ function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): LooseVNodes | function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: Opts): VNode { const collapsed = parentNode.collapsed === undefined ? opts.depth >= 2 && opts.depth % 2 === 0 : parentNode.collapsed; - return h('lines', { class: { single: !nodes[1], collapsed } }, [ + return h( + 'lines', + { class: { single: !nodes[1], collapsed } }, collapsed ? h('line', { class: { expand: true } }, [ h('branch'), @@ -123,24 +125,23 @@ function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: on: { click: () => ctx.ctrl.setCollapsed(opts.parentPath, false) }, }), ]) - : null, - ...nodes.map(n => { - return ( - retroLine(ctx, n) || - h('line', [ - h('branch'), - ...renderMoveAndChildrenOf(ctx, n, { - parentPath: opts.parentPath, - isMainline: false, - depth: opts.depth + 1, - withIndex: true, - noConceal: opts.noConceal, - truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, - }), - ]) - ); - }), - ]); + : nodes.map(n => { + return ( + retroLine(ctx, n) || + h('line', [ + h('branch'), + ...renderMoveAndChildrenOf(ctx, n, { + parentPath: opts.parentPath, + isMainline: false, + depth: opts.depth + 1, + withIndex: true, + noConceal: opts.noConceal, + truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, + }), + ]) + ); + }), + ); } function renderMoveOf(ctx: Ctx, node: Tree.Node, opts: Opts): VNode { diff --git a/ui/analyse/src/treeView/inlineView.ts b/ui/analyse/src/treeView/inlineView.ts index 5b81597411a34..13cdf38f62fe9 100644 --- a/ui/analyse/src/treeView/inlineView.ts +++ b/ui/analyse/src/treeView/inlineView.ts @@ -71,7 +71,9 @@ function renderInlined(ctx: Ctx, nodes: Tree.Node[], opts: Opts): MaybeVNodes | function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: Opts): VNode { const collapsed = parentNode.collapsed === undefined ? opts.depth >= 2 && opts.depth % 2 === 0 : parentNode.collapsed; - return h('lines', { class: { collapsed } }, [ + return h( + 'lines', + { class: { collapsed } }, collapsed ? h('line', { class: { expand: true } }, [ h('branch'), @@ -80,23 +82,22 @@ function renderLines(ctx: Ctx, parentNode: Tree.Node, nodes: Tree.Node[], opts: on: { click: () => ctx.ctrl.setCollapsed(opts.parentPath, false) }, }), ]) - : null, - ...nodes.map(n => { - return ( - retroLine(ctx, n) || - h('line', [ - h('branch'), - ...renderMoveAndChildrenOf(ctx, n, { - parentPath: opts.parentPath, - isMainline: false, - depth: opts.depth + 1, - withIndex: true, - truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, - }), - ]) - ); - }), - ]); + : nodes.map(n => { + return ( + retroLine(ctx, n) || + h('line', [ + h('branch'), + ...renderMoveAndChildrenOf(ctx, n, { + parentPath: opts.parentPath, + isMainline: false, + depth: opts.depth + 1, + withIndex: true, + truncate: n.comp && !treePath.contains(ctx.ctrl.path, opts.parentPath + n.id) ? 3 : undefined, + }), + ]) + ); + }), + ); } function renderMoveAndChildrenOf(ctx: Ctx, node: Tree.Node, opts: Opts): MaybeVNodes { diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index aac5fed98067a..406581eeb43db 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -365,14 +365,6 @@ margin-bottom: 0; } - lines.collapsed > * { - display: none; - } - - lines.collapsed > line.expand { - display: block; - } - line.expand > a { color: $c-font-dim; } From b94531f5674ae930d87c6d15a806a0428596e404 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 11:07:28 +0200 Subject: [PATCH 016/168] fide player not found page - closes #15457 --- app/controllers/Fide.scala | 18 +++++++++++------- app/views/fide.scala | 9 ++++----- modules/fide/src/main/ui/FideUi.scala | 24 ++++++++++++++++++++++++ 3 files changed, 39 insertions(+), 12 deletions(-) diff --git a/app/controllers/Fide.scala b/app/controllers/Fide.scala index 04f652a4d5e1a..e2704d8d9b410 100644 --- a/app/controllers/Fide.scala +++ b/app/controllers/Fide.scala @@ -22,13 +22,17 @@ final class Fide(env: Env) extends LilaController(env): yield Ok(renderedPage) def show(id: chess.FideId, slug: String, page: Int) = Open: - Found(env.fide.repo.player.fetch(id)): player => - if player.slug != slug then Redirect(routes.Fide.show(id, player.slug)) - else - for - tours <- env.relay.playerTour.playerTours(player, page) - rendered <- renderPage(views.fide.player.show(player, tours)) - yield Ok(rendered) + env.fide.repo.player + .fetch(id) + .flatMap: + case None => NotFound.page(views.fide.player.notFound(id)) + case Some(player) => + if player.slug != slug then Redirect(routes.Fide.show(id, player.slug)) + else + for + tours <- env.relay.playerTour.playerTours(player, page) + rendered <- renderPage(views.fide.player.show(player, tours)) + yield Ok(rendered) def federations(page: Int) = Open: for diff --git a/app/views/fide.scala b/app/views/fide.scala index dbfd78573a19e..18d32b8bffe3c 100644 --- a/app/views/fide.scala +++ b/app/views/fide.scala @@ -10,13 +10,12 @@ lazy val ui = lila.fide.ui.FideUi(helpers)(active => Context ?=> views.relay.tou export ui.federation object player: - export ui.player.index + export ui.player.{ index, notFound } def show(player: FidePlayer, tours: Paginator[RelayTour.WithLastRound])(using Context) = ui.player.show( player, - (tours.nbResults > 0).option( - views.relay.tour.renderPager(views.relay.tour.asRelayPager(tours)): page => - routes.Fide.show(player.id, player.slug, page) - ) + (tours.nbResults > 0).option: + views.relay.tour.renderPager(views.relay.tour.asRelayPager(tours)): + routes.Fide.show(player.id, player.slug, _) ) diff --git a/modules/fide/src/main/ui/FideUi.scala b/modules/fide/src/main/ui/FideUi.scala index 67147eb4c1460..83b0707dd95b0 100644 --- a/modules/fide/src/main/ui/FideUi.scala +++ b/modules/fide/src/main/ui/FideUi.scala @@ -111,6 +111,30 @@ final class FideUi(helpers: Helpers)(menu: String => Context ?=> Frag): playerList(players, np => routes.Fide.index(np, query.some.filter(_.nonEmpty))) ) + def notFound(id: chess.FideId)(using Context) = + page("FIDE player not found", "players")( + cls := "fide-players", + boxTop( + h1("FIDE player not found"), + div(cls := "box__top__actions"): + searchForm("") + ), + div(cls := "box__pad")( + p( + "We could not find anyone with the FIDE ID \"", + strong(id), + "\", please make sure the number is correct." + ), + p( + "If the player appears on the ", + a(href := "https://ratings.fide.com/", targetBlank)("official FIDE website"), + ", then the player was not included in the latest rating export from FIDE.", + br, + "FIDE exports are provided once a month and includes players who have at least one official rating." + ) + ) + ) + def searchForm(q: String) = st.form(cls := "fide-players__search-form", action := routes.Fide.index(1), method := "get")( input( From 553c67cb50bd04c5f8186078f77d86ed344fb7ed Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 11:11:01 +0200 Subject: [PATCH 017/168] New Crowdin updates (#15490) * New translations: site.xml (Persian) * New translations: learn.xml (Bengali) * New translations: coordinates.xml (Russian) * New translations: coordinates.xml (Chinese Simplified) * New translations: arena.xml (Swiss German) * New translations: site.xml (English, United States) * New translations: lag.xml (English, United States) --- translation/dest/arena/gsw-CH.xml | 6 +++--- translation/dest/coordinates/ru-RU.xml | 1 + translation/dest/coordinates/zh-CN.xml | 1 + translation/dest/lag/en-US.xml | 2 +- translation/dest/learn/bn-BD.xml | 4 ++-- translation/dest/site/en-US.xml | 8 ++++---- translation/dest/site/fa-IR.xml | 4 ++-- 7 files changed, 14 insertions(+), 12 deletions(-) diff --git a/translation/dest/arena/gsw-CH.xml b/translation/dest/arena/gsw-CH.xml index df7e485c61591..0e803538cbd62 100644 --- a/translation/dest/arena/gsw-CH.xml +++ b/translation/dest/arena/gsw-CH.xml @@ -57,7 +57,7 @@ Wer schnäll schpillt und sofort zrugg id Turnierübersicht gaht, hät meh Schpi Lass d\'Schpiller ime Chatruum diskutiere Arena Sieges Serie Nach 2 Sieg in Serie gits 4 Pünkt anstatt 2. - Berserk nöd möglich + Berserk nöd erlaubt Kei Arena-Serie Durchschnittlichi Leischtig Durchschnittlichi Punktzahl @@ -80,8 +80,8 @@ Bischpil: Rang 3 ime Turnier mit 100 Schpiler = 3% oder Rang 10 bi 1000 Schpiler All Durchschnittswert uf dere Site sind %s. Total Punktedurchschnitt - Azahl Pünkt - Rang - Durchschnitt + Azahl Pünkt + Rang - Durchschnitt Turnier - Sieger Turnier - Schilder Nur Schpiler mit Titel diff --git a/translation/dest/coordinates/ru-RU.xml b/translation/dest/coordinates/ru-RU.xml index d21b68ad390e2..25db548b72e3d 100644 --- a/translation/dest/coordinates/ru-RU.xml +++ b/translation/dest/coordinates/ru-RU.xml @@ -13,6 +13,7 @@ У вас есть 30 секунд на то, чтобы правильно отметить как можно больше полей! Продвиньтесь так далеко, как сможете. Время не ограничено! Показывать координаты + Координаты на каждом поле Показывать фигуры Начать тренировку Найти поле diff --git a/translation/dest/coordinates/zh-CN.xml b/translation/dest/coordinates/zh-CN.xml index 5e78dbcde088e..215243eb3e570 100644 --- a/translation/dest/coordinates/zh-CN.xml +++ b/translation/dest/coordinates/zh-CN.xml @@ -13,6 +13,7 @@ 您有30秒时间正确配对尽可能多的棋格。 没有时间限制,尽情练习吧! 显示坐标 + 在每个格子上显示坐标 显示棋子 开始训练 找格子 diff --git a/translation/dest/lag/en-US.xml b/translation/dest/lag/en-US.xml index 828a6ba2f9d20..ab6af8b78b00d 100644 --- a/translation/dest/lag/en-US.xml +++ b/translation/dest/lag/en-US.xml @@ -7,7 +7,7 @@ Yes. It will be fixed soon! And now, the long answer! Game lag is composed of two unrelated values (lower is better): Lichess server latency - The time it takes to process a move on the server. It is the same for everybody, and only depends on the server load. The more players and the higher it gets, but Lichess developers do their best to keep it low. It rarely exceeds 10ms. + The time it takes to process a move on the server. It is the same for everybody, and only depends on the server load. The more players, the higher it gets, but Lichess developers do their best to keep it low. It rarely exceeds 10ms. Network between Lichess and you The time it takes to send a move from your computer to Lichess server, and get the response back. It\'s specific to your distance to Lichess (France) as well as the quality of your Internet connection. Lichess developers cannot fix your wifi or make light go faster. You can find both these values at any time, by clicking your username in the top bar. diff --git a/translation/dest/learn/bn-BD.xml b/translation/dest/learn/bn-BD.xml index 099628fb2c943..df7563ce67dd3 100644 --- a/translation/dest/learn/bn-BD.xml +++ b/translation/dest/learn/bn-BD.xml @@ -7,9 +7,9 @@ আমার অগ্রগতি রিসেট হোক আপনি আপনার সকল অগ্রগতি হারাবেন! খেলুন! - দাবার টুকরা + দাবার ঘুটি নৌকা - এটা সোজা রেখা গুলোতে সরান + এটা সোজাসুজি (সোজা রেখার মতো) যায় দাবার নৌকা হয় একটি শক্তিশালী টুকরা. এটাকে নির্দেশ দিতে আপনি কি প্রস্তুত? তারায় নিয়ে আসতে নৌকায় ক্লিক করুন! সকল তারা দখল করুন! diff --git a/translation/dest/site/en-US.xml b/translation/dest/site/en-US.xml index 806e22f8e97a8..bbcb425b17843 100644 --- a/translation/dest/site/en-US.xml +++ b/translation/dest/site/en-US.xml @@ -223,8 +223,8 @@ You are leaving Lichess Never type your Lichess password on another site! Proceed to %s - Do not set a password suggested by someone else. They could use it to steal your account. - Do not set an email address suggested by someone else. They could use it to steal your account. + Do not set a password suggested by someone else. They will use it to steal your account. + Do not set an email address suggested by someone else. They will use it to steal your account. Help with email confirmation Didn\'t receive your confirmation email after signing up? What username did you use to sign up? @@ -233,10 +233,10 @@ We have sent an email to %s. It can take some time to arrive. Wait 5 minutes and refresh your email inbox. - If you do not receive a confirmation email, check your spam folder. Be sure to indicate messages from Xxxxx as \'safe\' so you can stay informed of important communication. + Also check your spam folder, it might end up there. If so, mark it as not spam. If you still have questions, please send us an email: Copy and paste the above text and send it to %s - We will be back with you shortly to help complete your signup process. + We will come back to you shortly to help you complete your signup. The user %s is successfully confirmed. You can login right now as %s. You do not need a confirmation email. diff --git a/translation/dest/site/fa-IR.xml b/translation/dest/site/fa-IR.xml index cae1141a4a3ad..e16aacef370da 100644 --- a/translation/dest/site/fa-IR.xml +++ b/translation/dest/site/fa-IR.xml @@ -147,8 +147,8 @@ پیشنهاد مساوی مساوی - %s نفر بازیکن آنلاین - %s نفر بازیکن آنلاین + %s بازیکن + %s بازیکن تساوی با توافق طرفین قانون ۵۰ حرکت From 5321ff3f7a30cbe0afc8c1d60582e100a747a494 Mon Sep 17 00:00:00 2001 From: Vincent Velociter Date: Fri, 14 Jun 2024 11:33:24 +0200 Subject: [PATCH 018/168] Add mobile scope to more challenge api endpoints --- app/controllers/Challenge.scala | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/app/controllers/Challenge.scala b/app/controllers/Challenge.scala index 6ef5fea5ec0bd..78aa8761188a3 100644 --- a/app/controllers/Challenge.scala +++ b/app/controllers/Challenge.scala @@ -26,7 +26,7 @@ final class Challenge( api.allFor(me).map(env.challenge.jsonView.apply).map(JsonOk) } - def apiList = ScopedBody(_.Challenge.Read) { ctx ?=> me ?=> + def apiList = ScopedBody(_.Challenge.Read, _.Web.Mobile) { ctx ?=> me ?=> api.allFor(me, 300).map { all => JsonOk: Json.obj( @@ -39,7 +39,7 @@ final class Challenge( def show(id: ChallengeId, _color: Option[Color]) = Open: showId(id) - def apiShow(id: ChallengeId) = Scoped(_.Challenge.Read) { ctx ?=> _ ?=> + def apiShow(id: ChallengeId) = Scoped(_.Challenge.Read, _.Web.Mobile) { ctx ?=> _ ?=> Found(api.byId(id)): c => val direction: Option[Direction] = if isMine(c) then Direction.Out.some @@ -125,7 +125,7 @@ final class Challenge( ) def apiAccept(id: ChallengeId) = - Scoped(_.Challenge.Write, _.Bot.Play, _.Board.Play) { _ ?=> me ?=> + Scoped(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { _ ?=> me ?=> def tryRematch = env.bot.player.rematchAccept(id.into(GameId)).flatMap { if _ then jsonOkResult @@ -173,7 +173,7 @@ final class Challenge( ) .inject(NoContent) } - def apiDecline(id: ChallengeId) = ScopedBody(_.Challenge.Write, _.Bot.Play, _.Board.Play) { ctx ?=> me ?=> + def apiDecline(id: ChallengeId) = ScopedBody(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { ctx ?=> me ?=> api.activeByIdFor(id, me).flatMap { case None => env.bot.player.rematchDecline(id.into(GameId)).flatMap { @@ -195,7 +195,7 @@ final class Challenge( then api.cancel(c).inject(NoContent) else notFound - def apiCancel(id: ChallengeId) = Scoped(_.Challenge.Write, _.Bot.Play, _.Board.Play) { ctx ?=> me ?=> + def apiCancel(id: ChallengeId) = Scoped(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { ctx ?=> me ?=> api.activeByIdBy(id, me).flatMap { case Some(c) => api.cancel(c).inject(jsonOkResult) case None => From 6dd7dffb711726dadc614873fdb8a0f7a53bd4fa Mon Sep 17 00:00:00 2001 From: Vincent Velociter Date: Fri, 14 Jun 2024 11:42:59 +0200 Subject: [PATCH 019/168] Fix format --- app/controllers/Challenge.scala | 110 ++++++++++++++++---------------- 1 file changed, 56 insertions(+), 54 deletions(-) diff --git a/app/controllers/Challenge.scala b/app/controllers/Challenge.scala index 78aa8761188a3..04adf129889d5 100644 --- a/app/controllers/Challenge.scala +++ b/app/controllers/Challenge.scala @@ -173,19 +173,20 @@ final class Challenge( ) .inject(NoContent) } - def apiDecline(id: ChallengeId) = ScopedBody(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { ctx ?=> me ?=> - api.activeByIdFor(id, me).flatMap { - case None => - env.bot.player.rematchDecline(id.into(GameId)).flatMap { - if _ then jsonOkResult - else notFoundJson() - } - case Some(c) => - bindForm(env.challenge.forms.decline)( - jsonFormError, - data => api.decline(c, data.realReason).inject(jsonOkResult) - ) - } + def apiDecline(id: ChallengeId) = ScopedBody(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { + ctx ?=> me ?=> + api.activeByIdFor(id, me).flatMap { + case None => + env.bot.player.rematchDecline(id.into(GameId)).flatMap { + if _ then jsonOkResult + else notFoundJson() + } + case Some(c) => + bindForm(env.challenge.forms.decline)( + jsonFormError, + data => api.decline(c, data.realReason).inject(jsonOkResult) + ) + } } def cancel(id: ChallengeId) = @@ -195,49 +196,50 @@ final class Challenge( then api.cancel(c).inject(NoContent) else notFound - def apiCancel(id: ChallengeId) = Scoped(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { ctx ?=> me ?=> - api.activeByIdBy(id, me).flatMap { - case Some(c) => api.cancel(c).inject(jsonOkResult) - case None => - api.activeByIdFor(id, me).flatMap { - case Some(c) => api.decline(c, ChallengeModel.DeclineReason.default).inject(jsonOkResult) - case None => - import lila.core.misc.map.Tell - import lila.core.round.Abort - import lila.core.round.AbortForce - env.game.gameRepo - .game(id.into(GameId)) - .dmap { - _.flatMap { Pov(_, me) } - } - .flatMapz { p => - env.round.proxyRepo.upgradeIfPresent(p).dmap(some) - } - .flatMap { - case Some(pov) if pov.game.abortableByUser => - lila.common.Bus.publish(Tell(id.value, Abort(pov.playerId)), "roundSocket") - jsonOkResult - case Some(pov) if pov.game.playable => - Bearer.from(get("opponentToken")) match - case Some(bearer) => - val required = OAuthScope.select(_.Challenge.Write).into(EndpointScopes) - env.oAuth.server.auth(bearer, required, ctx.req.some).map { - case Right(access) if pov.opponent.isUser(access.me) => + def apiCancel(id: ChallengeId) = Scoped(_.Challenge.Write, _.Bot.Play, _.Board.Play, _.Web.Mobile) { + ctx ?=> me ?=> + api.activeByIdBy(id, me).flatMap { + case Some(c) => api.cancel(c).inject(jsonOkResult) + case None => + api.activeByIdFor(id, me).flatMap { + case Some(c) => api.decline(c, ChallengeModel.DeclineReason.default).inject(jsonOkResult) + case None => + import lila.core.misc.map.Tell + import lila.core.round.Abort + import lila.core.round.AbortForce + env.game.gameRepo + .game(id.into(GameId)) + .dmap { + _.flatMap { Pov(_, me) } + } + .flatMapz { p => + env.round.proxyRepo.upgradeIfPresent(p).dmap(some) + } + .flatMap { + case Some(pov) if pov.game.abortableByUser => + lila.common.Bus.publish(Tell(id.value, Abort(pov.playerId)), "roundSocket") + jsonOkResult + case Some(pov) if pov.game.playable => + Bearer.from(get("opponentToken")) match + case Some(bearer) => + val required = OAuthScope.select(_.Challenge.Write).into(EndpointScopes) + env.oAuth.server.auth(bearer, required, ctx.req.some).map { + case Right(access) if pov.opponent.isUser(access.me) => + lila.common.Bus.publish(Tell(id.value, AbortForce), "roundSocket") + jsonOkResult + case Right(_) => BadRequest(jsonError("Not the opponent token")) + case Left(err) => BadRequest(jsonError(err.message)) + } + case None if api.isOpenBy(id, me) => + if pov.game.abortable then lila.common.Bus.publish(Tell(id.value, AbortForce), "roundSocket") jsonOkResult - case Right(_) => BadRequest(jsonError("Not the opponent token")) - case Left(err) => BadRequest(jsonError(err.message)) - } - case None if api.isOpenBy(id, me) => - if pov.game.abortable then - lila.common.Bus.publish(Tell(id.value, AbortForce), "roundSocket") - jsonOkResult - else BadRequest(jsonError("The game can no longer be aborted")) - case None => BadRequest(jsonError("Missing opponentToken")) - case _ => notFoundJson() - } - } - } + else BadRequest(jsonError("The game can no longer be aborted")) + case None => BadRequest(jsonError("Missing opponentToken")) + case _ => notFoundJson() + } + } + } } def apiStartClocks(id: GameId) = Anon: From 232eed23bcf429571f2f8f35407d4c2508d25875 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 14:21:16 +0200 Subject: [PATCH 020/168] smoother transitions during layout switch --- ui/bits/css/user/_sub-rating.scss | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/ui/bits/css/user/_sub-rating.scss b/ui/bits/css/user/_sub-rating.scss index 2264ecdaab299..dacb734627a5b 100644 --- a/ui/bits/css/user/_sub-rating.scss +++ b/ui/bits/css/user/_sub-rating.scss @@ -6,7 +6,7 @@ white-space: nowrap; - @include transition; + @include transition(background); &.empty { opacity: 0.5; @@ -17,7 +17,9 @@ opacity: 0.5; margin-inline-end: 0.2em; - @include transition; + transition: + opacity $transition-duration, + color $transition-duration; } &[href=''] { From 58398cf4be77824cd1256264ef6efbb4016de80d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 14:23:41 +0200 Subject: [PATCH 021/168] fix transitions --- ui/bits/css/coach/_editor.scss | 2 +- ui/bits/css/team/_list.scss | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/ui/bits/css/coach/_editor.scss b/ui/bits/css/coach/_editor.scss index b924e061d30bc..31af0981bd6ba 100644 --- a/ui/bits/css/coach/_editor.scss +++ b/ui/bits/css/coach/_editor.scss @@ -107,7 +107,7 @@ .status { opacity: 0; - @include transition(); + @include transition; text-align: center; color: $c-good; } diff --git a/ui/bits/css/team/_list.scss b/ui/bits/css/team/_list.scss index fa75daa12a062..a33cc23d17b3b 100644 --- a/ui/bits/css/team/_list.scss +++ b/ui/bits/css/team/_list.scss @@ -35,12 +35,12 @@ .team__quit { opacity: 0; - @include transition(0.1s); + transition: opacity 0.1s; } tr:hover .team__quit { opacity: 1; - @include transition(0.6s); + transition: opacity 0.6s; } .info { From 7f97977243d3c920b2489ed16e5ec207084bd15e Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 14:24:26 +0200 Subject: [PATCH 022/168] user dropdown icon --- app/views/user/show/header.scala | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/app/views/user/show/header.scala b/app/views/user/show/header.scala index e4c0ed963a539..5143d91f10480 100644 --- a/app/views/user/show/header.scala +++ b/app/views/user/show/header.scala @@ -90,10 +90,7 @@ object header: .option(a(cls := "nm-item note-zone-toggle")(splitNumber(s"${social.notes.size} Notes"))) ), div(cls := "user-actions dropdown")( - a( - cls := "text", - dataIcon := Icon.List - ), + a(cls := "text", dataIcon := Icon.Hamburger), div(cls := "dropdown-window")( (ctx .is(u)) From 976e6e02b1ebd8a2ab84740942666f4f6371bd4e Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 15:39:53 +0200 Subject: [PATCH 023/168] scala codegolf --- app/views/user/show/header.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app/views/user/show/header.scala b/app/views/user/show/header.scala index 5143d91f10480..fdf75ef45443d 100644 --- a/app/views/user/show/header.scala +++ b/app/views/user/show/header.scala @@ -92,8 +92,8 @@ object header: div(cls := "user-actions dropdown")( a(cls := "text", dataIcon := Icon.Hamburger), div(cls := "dropdown-window")( - (ctx - .is(u)) + ctx + .is(u) .option( frag( a( From e22abf72f03a436730a71bafa327403d0771da48 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 15:40:26 +0200 Subject: [PATCH 024/168] tweak user menu css, fix mobile view it can't wrap back to the left side --- ui/bits/css/user/_show.scss | 37 +++++++++++++++++++++---------------- 1 file changed, 21 insertions(+), 16 deletions(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index fa3fc0f71425e..4520832709804 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -22,16 +22,19 @@ } &__social { - @extend %flex-between; + @extend %flex-between-nowrap; background: $c-bg-zebra; .number-menu { + flex: 0 1 auto; + overflow: hidden; margin: 0 0 0.2em 1em; } .user-actions { - margin: 1em; + flex: 0 0 auto; + margin: 1em 1em 1em 0.3em; form { display: inline; @@ -40,24 +43,26 @@ .dropdown { position: relative; - font-size: x-large; + font-size: 2em; > a { + display: block; + height: 1.5em; color: $c-font-page; - float: right; padding-left: 0.5em; } .dropdown-window { + @extend %dropdown-shadow; + z-index: z('link-overlay'); visibility: hidden; background: $c-bg-header-dropdown; border-radius: 3px 0 3px 3px; - box-shadow: 2px 5px 6px rgba(0, 0, 0, 0.3); position: absolute; - top: 1.45em; + top: 1.5em; right: 0; a { width: 16em; - font-size: small; + font-size: 1rem; display: block; padding: 0.6rem 1rem; color: $c-header-dropdown; @@ -76,15 +81,15 @@ } } } - } - - .dropdown:hover { - > a { - background: $c-bg-header-dropdown; - box-shadow: 2px 5px 6px rgba(0, 0, 0, 0.3); - } - .dropdown-window { - visibility: visible; + &:hover { + > a { + border-radius: 3px 3px 0 0; + background: $c-bg-header-dropdown; + box-shadow: 2px 5px 6px rgba(0, 0, 0, 0.3); + } + .dropdown-window { + visibility: visible; + } } } } From 65d8bc8fa960e77bdd037e9fe6df827aefd34756 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 15:48:51 +0200 Subject: [PATCH 025/168] fix fide player page layout - closes #15496 --- modules/fide/src/main/ui/FideUi.scala | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/modules/fide/src/main/ui/FideUi.scala b/modules/fide/src/main/ui/FideUi.scala index 83b0707dd95b0..f56dd7d6c271f 100644 --- a/modules/fide/src/main/ui/FideUi.scala +++ b/modules/fide/src/main/ui/FideUi.scala @@ -212,7 +212,7 @@ final class FideUi(helpers: Helpers)(menu: String => Context ?=> Frag): ), tcTrans.map: (tc, name) => card(name(), player.ratingOf(tc).fold("Unrated")(_.toString)), - tours.map: tours => - div(cls := "fide-player__tours")(h2("Recent tournaments"), tours) - ) + ), + tours.map: tours => + div(cls := "fide-player__tours")(h2("Recent tournaments"), tours) ) From ea77f084138bb2684abe08737345dcdafdf2ec51 Mon Sep 17 00:00:00 2001 From: Niklas Fiekas Date: Fri, 14 Jun 2024 16:30:36 +0200 Subject: [PATCH 026/168] svgo cooke pieces --- public/piece-css/cooke.css | 24 +++++----- public/piece/cooke/bB.svg | 77 +------------------------------- public/piece/cooke/bK.svg | 66 +-------------------------- public/piece/cooke/bN.svg | 88 +----------------------------------- public/piece/cooke/bP.svg | 91 +------------------------------------- public/piece/cooke/bQ.svg | 69 +---------------------------- public/piece/cooke/bR.svg | 86 +---------------------------------- public/piece/cooke/wB.svg | 72 +----------------------------- public/piece/cooke/wK.svg | 69 +---------------------------- public/piece/cooke/wN.svg | 84 +---------------------------------- public/piece/cooke/wP.svg | 84 +---------------------------------- public/piece/cooke/wQ.svg | 68 +--------------------------- public/piece/cooke/wR.svg | 89 +------------------------------------ 13 files changed, 24 insertions(+), 943 deletions(-) diff --git a/public/piece-css/cooke.css b/public/piece-css/cooke.css index 2a9f3bca1aa7e..6ea20bfbd7e29 100644 --- a/public/piece-css/cooke.css +++ b/public/piece-css/cooke.css @@ -1,12 +1,12 @@ -.is2d .pawn.white {background-image:url('')} -.is2d .knight.white {background-image:url('')} -.is2d .bishop.white {background-image:url('')} -.is2d .rook.white {background-image:url('')} -.is2d .queen.white {background-image:url('')} -.is2d .king.white {background-image:url('')} -.is2d .pawn.black {background-image:url('')} -.is2d .knight.black {background-image:url('')} -.is2d .bishop.black {background-image:url('')} -.is2d .rook.black {background-image:url('')} -.is2d .queen.black {background-image:url('')} -.is2d .king.black {background-image:url('')} +.is2d .pawn.white {background-image:url('')} +.is2d .knight.white {background-image:url('')} +.is2d .bishop.white {background-image:url('')} +.is2d .rook.white {background-image:url('')} +.is2d .queen.white {background-image:url('')} +.is2d .king.white {background-image:url('')} +.is2d .pawn.black {background-image:url('')} +.is2d .knight.black {background-image:url('')} +.is2d .bishop.black {background-image:url('')} +.is2d .rook.black {background-image:url('')} +.is2d .queen.black {background-image:url('')} +.is2d .king.black {background-image:url('')} diff --git a/public/piece/cooke/bB.svg b/public/piece/cooke/bB.svg index acd017f6615c6..8b3edfeb8a8a2 100644 --- a/public/piece/cooke/bB.svg +++ b/public/piece/cooke/bB.svg @@ -1,76 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/bK.svg b/public/piece/cooke/bK.svg index 373b8827f5324..48b91bc6ad8b0 100644 --- a/public/piece/cooke/bK.svg +++ b/public/piece/cooke/bK.svg @@ -1,65 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/bN.svg b/public/piece/cooke/bN.svg index 6875e4d199153..b120f611acec9 100644 --- a/public/piece/cooke/bN.svg +++ b/public/piece/cooke/bN.svg @@ -1,87 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/bP.svg b/public/piece/cooke/bP.svg index b9515ac790919..cafb3ffb8d146 100644 --- a/public/piece/cooke/bP.svg +++ b/public/piece/cooke/bP.svg @@ -1,90 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/bQ.svg b/public/piece/cooke/bQ.svg index be666f974178f..72819d6440ae0 100644 --- a/public/piece/cooke/bQ.svg +++ b/public/piece/cooke/bQ.svg @@ -1,68 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/bR.svg b/public/piece/cooke/bR.svg index 43304a074fa9f..be06839d5ff68 100644 --- a/public/piece/cooke/bR.svg +++ b/public/piece/cooke/bR.svg @@ -1,85 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/wB.svg b/public/piece/cooke/wB.svg index bc46ff2fe3444..428f6be5d764d 100644 --- a/public/piece/cooke/wB.svg +++ b/public/piece/cooke/wB.svg @@ -1,71 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/wK.svg b/public/piece/cooke/wK.svg index 19111ab772185..42ad237ae5219 100644 --- a/public/piece/cooke/wK.svg +++ b/public/piece/cooke/wK.svg @@ -1,68 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/wN.svg b/public/piece/cooke/wN.svg index 170212b9aba5b..6de7b018d5f22 100644 --- a/public/piece/cooke/wN.svg +++ b/public/piece/cooke/wN.svg @@ -1,83 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/wP.svg b/public/piece/cooke/wP.svg index ee712fc0bd25f..ad46cddad2ddf 100644 --- a/public/piece/cooke/wP.svg +++ b/public/piece/cooke/wP.svg @@ -1,83 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/wQ.svg b/public/piece/cooke/wQ.svg index 4b98dcd5c9850..414078b8aa768 100644 --- a/public/piece/cooke/wQ.svg +++ b/public/piece/cooke/wQ.svg @@ -1,67 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/cooke/wR.svg b/public/piece/cooke/wR.svg index 2b9ef3c97b4a7..79373b65b5f99 100644 --- a/public/piece/cooke/wR.svg +++ b/public/piece/cooke/wR.svg @@ -1,88 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file + \ No newline at end of file From 7fd2fb0d6d50df9a47bbd1d743874b44ba10a126 Mon Sep 17 00:00:00 2001 From: Scala Steward Date: Fri, 14 Jun 2024 17:00:49 +0000 Subject: [PATCH 027/168] Update scalafmt-core to 3.8.2 --- .scalafmt.conf | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.scalafmt.conf b/.scalafmt.conf index 03d951870dc2a..f1245a49e7c76 100644 --- a/.scalafmt.conf +++ b/.scalafmt.conf @@ -1,4 +1,4 @@ -version = "3.8.1" +version = "3.8.2" runner.dialect = scala3 align.preset = more From 2b774ffaf8926dc9986954f3c427cbe479e05f6a Mon Sep 17 00:00:00 2001 From: Scala Steward Date: Fri, 14 Jun 2024 17:02:40 +0000 Subject: [PATCH 028/168] Reformat with scalafmt 3.8.2 Executed command: scalafmt --non-interactive --- modules/activity/src/main/activities.scala | 4 +- modules/coach/src/main/ui/CoachUi.scala | 3 +- modules/db/src/main/dsl.scala | 20 ++++---- modules/game/src/main/GameDiff.scala | 4 +- modules/game/src/main/JsonView.scala | 50 +++++++++---------- modules/msg/src/main/BsonHandlers.scala | 6 +-- modules/rating/src/main/RatingRegulator.scala | 4 +- .../src/main/TournamentStandingApi.scala | 4 +- 8 files changed, 48 insertions(+), 47 deletions(-) diff --git a/modules/activity/src/main/activities.scala b/modules/activity/src/main/activities.scala index 07d5c1752a24a..19856416d59a1 100644 --- a/modules/activity/src/main/activities.scala +++ b/modules/activity/src/main/activities.scala @@ -15,7 +15,7 @@ object activities: extension (a: Games) def add(pt: PerfKey, score: Score): Games = a.value + (pt -> a.value.get(pt).fold(score)(_.plus(score))) - def hasNonCorres = a.value.exists(_._1 != PerfKey.correspondence) + def hasNonCorres = a.value.exists(_._1 != PerfKey.correspondence) given Zero[Games] = Zero(Map.empty) opaque type ForumPosts = List[ForumPostId] @@ -62,7 +62,7 @@ object activities: extension (a: Practice) def +(studyId: StudyId): Practice = a.value + (studyId -> a.value.get(studyId).fold(1)(1 +)) - given Zero[Practice] = Zero(Map.empty) + given Zero[Practice] = Zero(Map.empty) opaque type Simuls = List[SimulId] object Simuls extends TotalWrapper[Simuls, List[SimulId]]: diff --git a/modules/coach/src/main/ui/CoachUi.scala b/modules/coach/src/main/ui/CoachUi.scala index 30114635ec9f2..97dd2087c1235 100644 --- a/modules/coach/src/main/ui/CoachUi.scala +++ b/modules/coach/src/main/ui/CoachUi.scala @@ -148,7 +148,8 @@ final class CoachUi(helpers: Helpers)( cls := "text button button-empty", dataIcon := Icon.BubbleSpeech, href := s"${routes.Msg.convo(c.user.username)}" - )(trc.sendPM()), + )(trc.sendPM()) + , ), div(cls := "coach-show__main coach-main box")( div(cls := "coach-widget")(widget(c, link = false)), diff --git a/modules/db/src/main/dsl.scala b/modules/db/src/main/dsl.scala index a2eff998db74e..8370c9fc9b665 100644 --- a/modules/db/src/main/dsl.scala +++ b/modules/db/src/main/dsl.scala @@ -339,21 +339,21 @@ object dsl extends dsl with Handlers: // like headOption, but with stopOnError defaulting to false def uno: Fu[Option[A]] = - c.collect[Iterable](1, Cursor.ContOnError[Iterable[A]]()).map(_.headOption) + c.collect[Iterable](1, Cursor.ContOnError[Iterable[A]]()).map(_.headOption) - // extension [A](cursor: Cursor.WithOps[A])(using Executor) + // extension [A](cursor: Cursor.WithOps[A])(using Executor) - // def gather[M[_]](upTo: Int)(using Factory[A, M[A]]): Fu[M[A]] = - // cursor.collect[M](upTo, Cursor.ContOnError[M[A]]()) + // def gather[M[_]](upTo: Int)(using Factory[A, M[A]]): Fu[M[A]] = + // cursor.collect[M](upTo, Cursor.ContOnError[M[A]]()) - // def list(): Fu[List[A]] = - // gather[List](Int.MaxValue) + // def list(): Fu[List[A]] = + // gather[List](Int.MaxValue) - // def list(limit: Int): Fu[List[A]] = - // gather[List](limit) + // def list(limit: Int): Fu[List[A]] = + // gather[List](limit) - // def list(limit: Option[Int]): Fu[List[A]] = - // gather[List](limit | Int.MaxValue) + // def list(limit: Option[Int]): Fu[List[A]] = + // gather[List](limit | Int.MaxValue) def vector(limit: Int): Fu[Vector[A]] = gather[Vector](limit) diff --git a/modules/game/src/main/GameDiff.scala b/modules/game/src/main/GameDiff.scala index ab73a10237385..e52890c4ce2dc 100644 --- a/modules/game/src/main/GameDiff.scala +++ b/modules/game/src/main/GameDiff.scala @@ -33,7 +33,7 @@ object GameDiff: val vb = getter(b) if getter(a) != vb then if vb == None || vb == null || vb == "" then unsetBuilder += (name -> bTrue) - else setBuilder += name -> toBson(vb) + else setBuilder += name -> toBson(vb) def dOpt[A](name: String, getter: Game => A, toBson: A => Option[BSONValue]): Unit = val vb = getter(b) @@ -42,7 +42,7 @@ object GameDiff: else toBson(vb) match case None => unsetBuilder += (name -> bTrue) - case Some(x) => setBuilder += name -> x + case Some(x) => setBuilder += name -> x def dTry[A](name: String, getter: Game => A, toBson: A => Try[BSONValue]): Unit = d[A](name, getter, a => toBson(a).get) diff --git a/modules/game/src/main/JsonView.scala b/modules/game/src/main/JsonView.scala index ff6a26fa29b8b..8d4743f51d5a5 100644 --- a/modules/game/src/main/JsonView.scala +++ b/modules/game/src/main/JsonView.scala @@ -13,31 +13,31 @@ final class JsonView(rematches: Rematches): import JsonView.given def base(game: Game, initialFen: Option[Fen.Full]) = - Json - .obj( - "id" -> game.id, - "variant" -> game.variant, - "speed" -> game.speed.key, - "perf" -> game.perfKey, - "rated" -> game.rated, - "fen" -> Fen.write(game.chess), - "turns" -> game.ply, - "source" -> game.source, - "status" -> game.status, - "createdAt" -> game.createdAt - ) - .add("startedAtTurn" -> game.chess.startedAtPly.some.filter(_ > 0)) - .add("initialFen" -> initialFen) - .add("threefold" -> game.history.threefoldRepetition) - .add("boosted" -> game.boosted) - .add("tournamentId" -> game.tournamentId) - .add("swissId" -> game.swissId) - .add("winner" -> game.winnerColor) - .add("rematch" -> rematches.getAcceptedId(game.id)) - .add("rules" -> game.metadata.nonEmptyRules) - .add("drawOffers" -> (!game.drawOffers.isEmpty).option(game.drawOffers.normalizedPlies)) - - // adds fields that could be computed by the client instead + Json + .obj( + "id" -> game.id, + "variant" -> game.variant, + "speed" -> game.speed.key, + "perf" -> game.perfKey, + "rated" -> game.rated, + "fen" -> Fen.write(game.chess), + "turns" -> game.ply, + "source" -> game.source, + "status" -> game.status, + "createdAt" -> game.createdAt + ) + .add("startedAtTurn" -> game.chess.startedAtPly.some.filter(_ > 0)) + .add("initialFen" -> initialFen) + .add("threefold" -> game.history.threefoldRepetition) + .add("boosted" -> game.boosted) + .add("tournamentId" -> game.tournamentId) + .add("swissId" -> game.swissId) + .add("winner" -> game.winnerColor) + .add("rematch" -> rematches.getAcceptedId(game.id)) + .add("rules" -> game.metadata.nonEmptyRules) + .add("drawOffers" -> (!game.drawOffers.isEmpty).option(game.drawOffers.normalizedPlies)) + + // adds fields that could be computed by the client instead def baseWithChessDenorm(game: Game, initialFen: Option[Fen.Full]) = base(game, initialFen) ++ Json .obj( diff --git a/modules/msg/src/main/BsonHandlers.scala b/modules/msg/src/main/BsonHandlers.scala index 42791a3d27b33..3ec345eff99a1 100644 --- a/modules/msg/src/main/BsonHandlers.scala +++ b/modules/msg/src/main/BsonHandlers.scala @@ -42,9 +42,9 @@ private object BsonHandlers: ) def writeThread(thread: MsgThread, delBy: List[UserId]): Bdoc = - threadHandler.writeTry(thread).get ++ $doc("del" -> delBy) - ++ $doc("maskWith" -> $doc("date" -> thread.lastMsg.date)) - // looks weird, but maybe.. it is the way + threadHandler.writeTry(thread).get ++ $doc("del" -> delBy) + ++ $doc("maskWith" -> $doc("date" -> thread.lastMsg.date)) + // looks weird, but maybe.. it is the way def selectNotDeleted(using me: Me) = if UserId.lichess.is(me) then $empty // using "del" is too expensive diff --git a/modules/rating/src/main/RatingRegulator.scala b/modules/rating/src/main/RatingRegulator.scala index 3821942d81441..cbdded7184650 100644 --- a/modules/rating/src/main/RatingRegulator.scala +++ b/modules/rating/src/main/RatingRegulator.scala @@ -11,8 +11,8 @@ object RatingRegulator: } private def apply(factor: RatingFactor, perfType: PerfType, before: Perf, after: Perf): Perf = - if (after.nb == before.nb + 1) && // after playing one game - (after.glicko.rating > before.glicko.rating) // and gaining rating + if (after.nb == before.nb + 1) && // after playing one game + (after.glicko.rating > before.glicko.rating) // and gaining rating then val diff = after.glicko.rating - before.glicko.rating val extra = diff * (factor.value - 1) diff --git a/modules/tournament/src/main/TournamentStandingApi.scala b/modules/tournament/src/main/TournamentStandingApi.scala index 30b91e2c1ff00..b7acb3ca7c11d 100644 --- a/modules/tournament/src/main/TournamentStandingApi.scala +++ b/modules/tournament/src/main/TournamentStandingApi.scala @@ -56,8 +56,8 @@ final class TournamentStandingApi( compute(tourId, page, withScores = true) def clearCache(tour: Tournament): Unit = - first.invalidate(tour.id) - // no need to invalidate createdCache, these are only cached when tour.isCreated + first.invalidate(tour.id) + // no need to invalidate createdCache, these are only cached when tour.isCreated private def compute(id: TourId, page: Int, withScores: Boolean): Fu[JsObject] = cached.tourCache.byId(id).orFail(s"No such tournament: $id").flatMap { compute(_, page, withScores) } From c661d9014cc1d4d43a36c5f0ee5a98a1c9ceb7ee Mon Sep 17 00:00:00 2001 From: Scala Steward Date: Fri, 14 Jun 2024 17:02:40 +0000 Subject: [PATCH 029/168] Add 'Reformat with scalafmt 3.8.2' to .git-blame-ignore-revs --- .git-blame-ignore-revs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.git-blame-ignore-revs b/.git-blame-ignore-revs index 9d1ff9eb089b9..be304c6e33c6e 100644 --- a/.git-blame-ignore-revs +++ b/.git-blame-ignore-revs @@ -57,3 +57,6 @@ cd9708a0d772fe445635ae05306f1cb9365902de # Scala Steward: Reformat with scalafmt 3.8.1 1aae0c7b022f9e505e1ca7fec3b32efeef9f9981 + +# Scala Steward: Reformat with scalafmt 3.8.2 +2b774ffaf8926dc9986954f3c427cbe479e05f6a From cdc25555b67c35be6f72f85f061b2f645a92a119 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 20:55:32 +0200 Subject: [PATCH 030/168] scalafmtAll --- modules/coach/src/main/ui/CoachUi.scala | 1 - modules/db/src/main/dsl.scala | 2 +- modules/game/src/main/JsonView.scala | 46 +++++++++---------- modules/msg/src/main/BsonHandlers.scala | 4 +- .../src/main/TournamentStandingApi.scala | 2 +- 5 files changed, 27 insertions(+), 28 deletions(-) diff --git a/modules/coach/src/main/ui/CoachUi.scala b/modules/coach/src/main/ui/CoachUi.scala index 97dd2087c1235..d3efb32c1e4ce 100644 --- a/modules/coach/src/main/ui/CoachUi.scala +++ b/modules/coach/src/main/ui/CoachUi.scala @@ -149,7 +149,6 @@ final class CoachUi(helpers: Helpers)( dataIcon := Icon.BubbleSpeech, href := s"${routes.Msg.convo(c.user.username)}" )(trc.sendPM()) - , ), div(cls := "coach-show__main coach-main box")( div(cls := "coach-widget")(widget(c, link = false)), diff --git a/modules/db/src/main/dsl.scala b/modules/db/src/main/dsl.scala index 8370c9fc9b665..10c81749777af 100644 --- a/modules/db/src/main/dsl.scala +++ b/modules/db/src/main/dsl.scala @@ -339,7 +339,7 @@ object dsl extends dsl with Handlers: // like headOption, but with stopOnError defaulting to false def uno: Fu[Option[A]] = - c.collect[Iterable](1, Cursor.ContOnError[Iterable[A]]()).map(_.headOption) + c.collect[Iterable](1, Cursor.ContOnError[Iterable[A]]()).map(_.headOption) // extension [A](cursor: Cursor.WithOps[A])(using Executor) diff --git a/modules/game/src/main/JsonView.scala b/modules/game/src/main/JsonView.scala index 8d4743f51d5a5..05c2912856ca3 100644 --- a/modules/game/src/main/JsonView.scala +++ b/modules/game/src/main/JsonView.scala @@ -13,29 +13,29 @@ final class JsonView(rematches: Rematches): import JsonView.given def base(game: Game, initialFen: Option[Fen.Full]) = - Json - .obj( - "id" -> game.id, - "variant" -> game.variant, - "speed" -> game.speed.key, - "perf" -> game.perfKey, - "rated" -> game.rated, - "fen" -> Fen.write(game.chess), - "turns" -> game.ply, - "source" -> game.source, - "status" -> game.status, - "createdAt" -> game.createdAt - ) - .add("startedAtTurn" -> game.chess.startedAtPly.some.filter(_ > 0)) - .add("initialFen" -> initialFen) - .add("threefold" -> game.history.threefoldRepetition) - .add("boosted" -> game.boosted) - .add("tournamentId" -> game.tournamentId) - .add("swissId" -> game.swissId) - .add("winner" -> game.winnerColor) - .add("rematch" -> rematches.getAcceptedId(game.id)) - .add("rules" -> game.metadata.nonEmptyRules) - .add("drawOffers" -> (!game.drawOffers.isEmpty).option(game.drawOffers.normalizedPlies)) + Json + .obj( + "id" -> game.id, + "variant" -> game.variant, + "speed" -> game.speed.key, + "perf" -> game.perfKey, + "rated" -> game.rated, + "fen" -> Fen.write(game.chess), + "turns" -> game.ply, + "source" -> game.source, + "status" -> game.status, + "createdAt" -> game.createdAt + ) + .add("startedAtTurn" -> game.chess.startedAtPly.some.filter(_ > 0)) + .add("initialFen" -> initialFen) + .add("threefold" -> game.history.threefoldRepetition) + .add("boosted" -> game.boosted) + .add("tournamentId" -> game.tournamentId) + .add("swissId" -> game.swissId) + .add("winner" -> game.winnerColor) + .add("rematch" -> rematches.getAcceptedId(game.id)) + .add("rules" -> game.metadata.nonEmptyRules) + .add("drawOffers" -> (!game.drawOffers.isEmpty).option(game.drawOffers.normalizedPlies)) // adds fields that could be computed by the client instead def baseWithChessDenorm(game: Game, initialFen: Option[Fen.Full]) = diff --git a/modules/msg/src/main/BsonHandlers.scala b/modules/msg/src/main/BsonHandlers.scala index 3ec345eff99a1..22261f37b1e53 100644 --- a/modules/msg/src/main/BsonHandlers.scala +++ b/modules/msg/src/main/BsonHandlers.scala @@ -42,8 +42,8 @@ private object BsonHandlers: ) def writeThread(thread: MsgThread, delBy: List[UserId]): Bdoc = - threadHandler.writeTry(thread).get ++ $doc("del" -> delBy) - ++ $doc("maskWith" -> $doc("date" -> thread.lastMsg.date)) + threadHandler.writeTry(thread).get ++ $doc("del" -> delBy) + ++ $doc("maskWith" -> $doc("date" -> thread.lastMsg.date)) // looks weird, but maybe.. it is the way def selectNotDeleted(using me: Me) = diff --git a/modules/tournament/src/main/TournamentStandingApi.scala b/modules/tournament/src/main/TournamentStandingApi.scala index b7acb3ca7c11d..67ec89c763426 100644 --- a/modules/tournament/src/main/TournamentStandingApi.scala +++ b/modules/tournament/src/main/TournamentStandingApi.scala @@ -56,7 +56,7 @@ final class TournamentStandingApi( compute(tourId, page, withScores = true) def clearCache(tour: Tournament): Unit = - first.invalidate(tour.id) + first.invalidate(tour.id) // no need to invalidate createdCache, these are only cached when tour.isCreated private def compute(id: TourId, page: Int, withScores: Boolean): Fu[JsObject] = From 1af9c1114105ea1b3aefda5f8b7dae2ed141ba50 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Gl=C3=B3rias?= <9739913+SergioGlorias@users.noreply.github.com> Date: Fri, 14 Jun 2024 21:00:32 +0100 Subject: [PATCH 031/168] Reduce anarcandy --- public/piece-css/anarcandy.css | 24 ++++++++++++------------ public/piece/anarcandy/bB.svg | 2 +- public/piece/anarcandy/bK.svg | 2 +- public/piece/anarcandy/bN.svg | 2 +- public/piece/anarcandy/bP.svg | 2 +- public/piece/anarcandy/bQ.svg | 2 +- public/piece/anarcandy/bR.svg | 2 +- public/piece/anarcandy/wB.svg | 2 +- public/piece/anarcandy/wK.svg | 2 +- public/piece/anarcandy/wN.svg | 2 +- public/piece/anarcandy/wP.svg | 2 +- public/piece/anarcandy/wQ.svg | 2 +- public/piece/anarcandy/wR.svg | 2 +- 13 files changed, 24 insertions(+), 24 deletions(-) diff --git a/public/piece-css/anarcandy.css b/public/piece-css/anarcandy.css index 13a6ed98633a9..334042c3bcd4a 100644 --- a/public/piece-css/anarcandy.css +++ b/public/piece-css/anarcandy.css @@ -1,12 +1,12 @@ -.is2d .pawn.white {background-image:url('')} -.is2d .knight.white {background-image:url('')} -.is2d .bishop.white {background-image:url('')} -.is2d .rook.white {background-image:url('')} -.is2d .queen.white {background-image:url('')} -.is2d .king.white {background-image:url('')} -.is2d .pawn.black {background-image:url('')} -.is2d .knight.black {background-image:url('')} -.is2d .bishop.black {background-image:url('')} -.is2d .rook.black {background-image:url('')} -.is2d .queen.black {background-image:url('')} -.is2d .king.black {background-image:url('')} +.is2d .pawn.white {background-image:url('')} +.is2d .knight.white {background-image:url('')} +.is2d .bishop.white {background-image:url('')} +.is2d .rook.white {background-image:url('')} +.is2d .queen.white {background-image:url('')} +.is2d .king.white {background-image:url('')} +.is2d .pawn.black {background-image:url('')} +.is2d .knight.black {background-image:url('')} +.is2d .bishop.black {background-image:url('')} +.is2d .rook.black {background-image:url('')} +.is2d .queen.black {background-image:url('')} +.is2d .king.black {background-image:url('')} diff --git a/public/piece/anarcandy/bB.svg b/public/piece/anarcandy/bB.svg index 706f3c14b3a9f..9e9fdeac1a6c1 100644 --- a/public/piece/anarcandy/bB.svg +++ b/public/piece/anarcandy/bB.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/bK.svg b/public/piece/anarcandy/bK.svg index 8530018b17831..289ce8ae396e4 100644 --- a/public/piece/anarcandy/bK.svg +++ b/public/piece/anarcandy/bK.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/bN.svg b/public/piece/anarcandy/bN.svg index 1c65e1a227e3f..1324689d9b72a 100644 --- a/public/piece/anarcandy/bN.svg +++ b/public/piece/anarcandy/bN.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/bP.svg b/public/piece/anarcandy/bP.svg index faacdcc128b82..8885435dca787 100644 --- a/public/piece/anarcandy/bP.svg +++ b/public/piece/anarcandy/bP.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/bQ.svg b/public/piece/anarcandy/bQ.svg index 99a342f6a0826..43f962bddcc47 100644 --- a/public/piece/anarcandy/bQ.svg +++ b/public/piece/anarcandy/bQ.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/bR.svg b/public/piece/anarcandy/bR.svg index e072cc156e664..c1bdae13a548c 100644 --- a/public/piece/anarcandy/bR.svg +++ b/public/piece/anarcandy/bR.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/wB.svg b/public/piece/anarcandy/wB.svg index 96db00c3fe55e..712e646e0898b 100644 --- a/public/piece/anarcandy/wB.svg +++ b/public/piece/anarcandy/wB.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/wK.svg b/public/piece/anarcandy/wK.svg index e433f29263a22..2b58c65dbcf5f 100644 --- a/public/piece/anarcandy/wK.svg +++ b/public/piece/anarcandy/wK.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/wN.svg b/public/piece/anarcandy/wN.svg index 7665324eae442..00332cf2f0fc1 100644 --- a/public/piece/anarcandy/wN.svg +++ b/public/piece/anarcandy/wN.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/wP.svg b/public/piece/anarcandy/wP.svg index 479740bef219d..662e07f8f1de2 100644 --- a/public/piece/anarcandy/wP.svg +++ b/public/piece/anarcandy/wP.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/wQ.svg b/public/piece/anarcandy/wQ.svg index 263064878d939..9104dbb2adcba 100644 --- a/public/piece/anarcandy/wQ.svg +++ b/public/piece/anarcandy/wQ.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/anarcandy/wR.svg b/public/piece/anarcandy/wR.svg index 7690eff6a4ce5..d5f608c9ac6fb 100644 --- a/public/piece/anarcandy/wR.svg +++ b/public/piece/anarcandy/wR.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file From a93bad40a16e7057fe2d7b0d96809123be141868 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 22:06:57 +0200 Subject: [PATCH 032/168] sync only a list of relay tour ids useful in dev env --- conf/base.conf | 3 +++ modules/relay/src/main/Env.scala | 4 ++++ modules/relay/src/main/RelayApi.scala | 16 ++++++++++------ modules/relay/src/main/RelayFetch.scala | 7 +++++-- 4 files changed, 22 insertions(+), 8 deletions(-) diff --git a/conf/base.conf b/conf/base.conf index cb4084b954715..bd47354c11d86 100644 --- a/conf/base.conf +++ b/conf/base.conf @@ -125,6 +125,9 @@ puzzle { path = puzzle2_path } } +relay { + syncOnlyIds = null +} storm.secret = "somethingElseInProd" coordinate { collection { diff --git a/modules/relay/src/main/Env.scala b/modules/relay/src/main/Env.scala index 8ba1817eaa08d..bf3c2415fcdc9 100644 --- a/modules/relay/src/main/Env.scala +++ b/modules/relay/src/main/Env.scala @@ -16,6 +16,7 @@ import lila.relay.RelayTour.WithLastRound @Module final class Env( + config: play.api.Configuration, ws: StandaloneWSClient, db: lila.db.Db, yoloDb: lila.db.AsyncDb @@ lila.db.YoloDb, @@ -114,6 +115,9 @@ final class Env( private val relayFidePlayerApi = wire[RelayFidePlayerApi] + import lila.common.config.given + private val syncOnlyIds = config.getOptional[List[String]]("relay.syncOnlyIds").map(RelayTourId.from) + // start the sync scheduler wire[RelayFetch] diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index ad63e3ad2b6cf..a2a36920842ea 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -110,16 +110,16 @@ final class RelayApi( get(tour.id).map(RelayTour.WithGroupTours(tour, _)) def invalidate(id: RelayTourId) = cache.underlying.synchronous.invalidate(id) - private def toSyncSelect = $doc( + private def toSyncSelect(onlyIds: Option[List[RelayTourId]]) = $doc( "sync.until".$exists(true), "sync.nextAt".$lt(nowInstant) - ) + ) ++ onlyIds.so(ids => $doc("tourId".$in(ids))) - private[relay] def toSyncOfficial(max: Max): Fu[List[WithTour]] = + private[relay] def toSyncOfficial(max: Max, onlyIds: Option[List[RelayTourId]]): Fu[List[WithTour]] = roundRepo.coll .aggregateList(max.value, _.pri): framework => import framework.* - Match(toSyncSelect) -> List( + Match(toSyncSelect(onlyIds)) -> List( PipelineOperator(tourRepo.lookup("tourId")), UnwindField("tour"), Match($doc("tour.tier".$exists(true))), @@ -128,11 +128,15 @@ final class RelayApi( ) .map(_.flatMap(readRoundWithTour)) - private[relay] def toSyncUser(max: Max, maxPerUser: Max = Max(5)): Fu[List[WithTour]] = + private[relay] def toSyncUser( + max: Max, + onlyIds: Option[List[RelayTourId]], + maxPerUser: Max = Max(5) + ): Fu[List[WithTour]] = roundRepo.coll .aggregateList(max.value, _.pri): framework => import framework.* - Match(toSyncSelect) -> List( + Match(toSyncSelect(onlyIds)) -> List( PipelineOperator(tourRepo.lookup("tourId")), UnwindField("tour"), Match($doc("tour.tier".$exists(false))), diff --git a/modules/relay/src/main/RelayFetch.scala b/modules/relay/src/main/RelayFetch.scala index c15ba31bb3792..eb8f1344de023 100644 --- a/modules/relay/src/main/RelayFetch.scala +++ b/modules/relay/src/main/RelayFetch.scala @@ -27,7 +27,8 @@ final private class RelayFetch( fidePlayers: RelayFidePlayerApi, gameRepo: GameRepo, pgnDump: PgnDump, - gameProxy: lila.core.game.GameProxy + gameProxy: lila.core.game.GameProxy, + onlyIds: Option[List[RelayTourId]] = None )(using Executor, Scheduler, lila.core.i18n.Translator)(using mode: play.api.Mode): import RelayFetch.* @@ -51,7 +52,9 @@ final private class RelayFetch( private val maxRelaysToSync = Max(50) private def syncRelays(official: Boolean): Funit = - val relays = if official then api.toSyncOfficial(maxRelaysToSync) else api.toSyncUser(maxRelaysToSync) + val relays = + if official then api.toSyncOfficial(maxRelaysToSync, onlyIds) + else api.toSyncUser(maxRelaysToSync, onlyIds) relays .flatMap: relays => lila.mon.relay.ongoing(official).update(relays.size) From 8d00782625cb7bc3ff2f091f17898aef3488f504 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 22:09:46 +0200 Subject: [PATCH 033/168] list all participated broadcasts on fide profiles - closes #14950 even if they're part of a group and not the first one of the group --- modules/relay/src/main/RelayPager.scala | 48 +++++++++++++++---------- 1 file changed, 30 insertions(+), 18 deletions(-) diff --git a/modules/relay/src/main/RelayPager.scala b/modules/relay/src/main/RelayPager.scala index daff274c6e7c6..88016822b2547 100644 --- a/modules/relay/src/main/RelayPager.scala +++ b/modules/relay/src/main/RelayPager.scala @@ -115,12 +115,14 @@ final class RelayPager( forSelector( $inIds(ids) ++ selectors.officialPublic, page, + onlyKeepGroupFirst = false, List("syncedAt") ) private def forSelector( selector: Bdoc, page: Int, + onlyKeepGroupFirst: Boolean = true, sortFields: List[String] = List("tier", "syncedAt", "createdAt") ): Fu[Paginator[WithLastRound]] = Paginator( @@ -132,7 +134,7 @@ final class RelayPager( import framework.* Match(selector) -> { List(Sort(sortFields.map(Descending(_))*)) ::: - aggregateRoundAndUnwind(framework) ::: + aggregateRoundAndUnwind(framework, onlyKeepGroupFirst) ::: List(Skip(offset), Limit(length)) } .map(readToursWithRound) @@ -141,26 +143,36 @@ final class RelayPager( maxPerPage = maxPerPage ) - private def aggregateRoundAndUnwind(framework: tourRepo.coll.AggregationFramework.type) = - aggregateRound(framework) ::: List(framework.UnwindField("round")) - - private def aggregateRound(framework: tourRepo.coll.AggregationFramework.type) = List( - framework.PipelineOperator(RelayListing.group.lookup(colls.group)), - framework.Match(RelayListing.group.filter), - framework.PipelineOperator( - $lookup.pipeline( - from = roundRepo.coll, - as = "round", - local = "_id", - foreign = "tourId", - pipe = List( - $doc("$sort" -> RelayRoundRepo.sort.start), - $doc("$limit" -> 1), - $doc("$addFields" -> $doc("sync.log" -> $arr())) + private def aggregateRoundAndUnwind( + framework: tourRepo.coll.AggregationFramework.type, + onlyKeepGroupFirst: Boolean = true + ) = + aggregateRound(framework, onlyKeepGroupFirst) ::: List(framework.UnwindField("round")) + + private def aggregateRound( + framework: tourRepo.coll.AggregationFramework.type, + onlyKeepGroupFirst: Boolean = true + ) = + onlyKeepGroupFirst.so( + List( + framework.PipelineOperator(RelayListing.group.lookup(colls.group)), + framework.Match(RelayListing.group.filter) + ) + ) ::: List( + framework.PipelineOperator( + $lookup.pipeline( + from = roundRepo.coll, + as = "round", + local = "_id", + foreign = "tourId", + pipe = List( + $doc("$sort" -> RelayRoundRepo.sort.start), + $doc("$limit" -> 1), + $doc("$addFields" -> $doc("sync.log" -> $arr())) + ) ) ) ) - ) private def readToursWithRound(docs: List[Bdoc]): List[WithLastRound] = for doc <- docs From 38ea2cefa6053fc11f6b4f0e9f0ab3b3cab6209b Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Fri, 14 Jun 2024 23:40:13 +0200 Subject: [PATCH 034/168] better broadcast search ranking - closes #15504 mixing date, tier, and search score --- modules/relay/src/main/RelayPager.scala | 25 +++++++++++++++++++++---- 1 file changed, 21 insertions(+), 4 deletions(-) diff --git a/modules/relay/src/main/RelayPager.scala b/modules/relay/src/main/RelayPager.scala index 88016822b2547..ba23a71bb593e 100644 --- a/modules/relay/src/main/RelayPager.scala +++ b/modules/relay/src/main/RelayPager.scala @@ -109,20 +109,36 @@ final class RelayPager( ) def search(query: String, page: Int): Fu[Paginator[WithLastRound]] = - forSelector($text(query) ++ selectors.officialPublic, page) + val day = 1000L * 3600 * 24 + forSelector( + selector = $text(query) ++ selectors.officialPublic, + page = page, + onlyKeepGroupFirst = false, + addFields = $doc( + "searchDate" -> $doc( + "$add" -> $arr( + $doc("$ifNull" -> $arr("$syncedAt", "$createdAt")), + $doc("$multiply" -> $arr($doc("$add" -> $arr("$tier", -RelayTour.Tier.NORMAL)), 60 * day)), + $doc("$multiply" -> $arr($doc("$meta" -> "textScore"), 30 * day)) + ) + ) + ).some, + sortFields = List("searchDate") + ) def byIds(ids: List[RelayTourId], page: Int): Fu[Paginator[WithLastRound]] = forSelector( $inIds(ids) ++ selectors.officialPublic, - page, + page = page, onlyKeepGroupFirst = false, - List("syncedAt") + sortFields = List("syncedAt") ) private def forSelector( selector: Bdoc, page: Int, onlyKeepGroupFirst: Boolean = true, + addFields: Option[Bdoc] = None, sortFields: List[String] = List("tier", "syncedAt", "createdAt") ): Fu[Paginator[WithLastRound]] = Paginator( @@ -133,7 +149,8 @@ final class RelayPager( .aggregateList(length, _.sec): framework => import framework.* Match(selector) -> { - List(Sort(sortFields.map(Descending(_))*)) ::: + addFields.map(AddFields(_)).toList ::: + List(Sort(sortFields.map(Descending(_))*)) ::: aggregateRoundAndUnwind(framework, onlyKeepGroupFirst) ::: List(Skip(offset), Limit(length)) } From f6225d8fb5685f448a84018b3100fbfce0a166e5 Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Sat, 15 Jun 2024 09:40:31 +0700 Subject: [PATCH 035/168] Bump lila-search RC8 And use count output as Long --- modules/gameSearch/src/main/GameSearchForm.scala | 2 +- modules/search/src/main/PaginatorBuilder.scala | 2 +- modules/search/src/main/SearchReadApi.scala | 2 +- modules/studySearch/src/main/Env.scala | 2 +- project/Dependencies.scala | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/modules/gameSearch/src/main/GameSearchForm.scala b/modules/gameSearch/src/main/GameSearchForm.scala index c18a05900f664..fba10520ccd4f 100644 --- a/modules/gameSearch/src/main/GameSearchForm.scala +++ b/modules/gameSearch/src/main/GameSearchForm.scala @@ -7,7 +7,7 @@ import java.time.LocalDate import lila.common.Form.* import lila.core.i18n.Translate -import lila.search.spec.{ Sorting as SpecSorting, Clocking, IntRange, DateRange, Query } +import lila.search.spec.{ Sorting as SpecSorting, IntRange, DateRange, Query } import smithy4s.Timestamp final private[gameSearch] class GameSearchForm: diff --git a/modules/search/src/main/PaginatorBuilder.scala b/modules/search/src/main/PaginatorBuilder.scala index f59515efb12a3..2302d3e1da784 100644 --- a/modules/search/src/main/PaginatorBuilder.scala +++ b/modules/search/src/main/PaginatorBuilder.scala @@ -10,7 +10,7 @@ final class PaginatorBuilder[A, Q]( def apply(query: Q, page: Int): Fu[Paginator[A]] = Paginator( adapter = new AdapterLike[A]: - def nbResults = searchApi.count(query) + def nbResults = searchApi.count(query).dmap(_.toInt) def slice(offset: Int, length: Int) = searchApi.search(query, From(offset), Size(length)) , diff --git a/modules/search/src/main/SearchReadApi.scala b/modules/search/src/main/SearchReadApi.scala index b6b03488009a2..acb96d65434fe 100644 --- a/modules/search/src/main/SearchReadApi.scala +++ b/modules/search/src/main/SearchReadApi.scala @@ -4,4 +4,4 @@ trait SearchReadApi[A, Q]: def search(query: Q, from: From, size: Size): Fu[List[A]] - def count(query: Q): Fu[Int] + def count(query: Q): Fu[Long] diff --git a/modules/studySearch/src/main/Env.scala b/modules/studySearch/src/main/Env.scala index 8f679c25973f4..5348917f1fc9f 100644 --- a/modules/studySearch/src/main/Env.scala +++ b/modules/studySearch/src/main/Env.scala @@ -27,7 +27,7 @@ final class Env( Paginator[Study.WithChaptersAndLiked]( adapter = new AdapterLike[Study]: def query = Query.study(text.take(100), me.map(_.id.value)) - def nbResults = api.count(query) + def nbResults = api.count(query).dmap(_.toInt) def slice(offset: Int, length: Int) = api.search(query, From(offset), Size(length)) .mapFutureList(pager.withChaptersAndLiking(me)), currentPage = page, diff --git a/project/Dependencies.scala b/project/Dependencies.scala index 955da39388b30..273cb5afdb76f 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -26,7 +26,7 @@ object Dependencies { val lettuce = "io.lettuce" % "lettuce-core" % "6.3.2.RELEASE" val nettyTransport = ("io.netty" % s"netty-transport-native-$notifier" % "4.1.111.Final").classifier(s"$os-$arch") - val lilaSearch = "org.lichess.search" %% "client" % "3.0.0-RC7" + val lilaSearch = "org.lichess.search" %% "client" % "3.0.0-RC8" val munit = "org.scalameta" %% "munit" % "1.0.0" % Test val uaparser = "org.uaparser" %% "uap-scala" % "0.17.0" val apacheText = "org.apache.commons" % "commons-text" % "1.12.0" From 334f70b7ada8bc2a250f83f150c981798a928f91 Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Sat, 15 Jun 2024 10:40:35 +0700 Subject: [PATCH 036/168] Bump lila-search RC9 and fix forum's post date As We used to store date as milliseconds --- modules/forumSearch/src/main/ForumSearchApi.scala | 3 +-- project/Dependencies.scala | 2 +- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/modules/forumSearch/src/main/ForumSearchApi.scala b/modules/forumSearch/src/main/ForumSearchApi.scala index 9ab9a94b5d2e0..3a223b357ca45 100644 --- a/modules/forumSearch/src/main/ForumSearchApi.scala +++ b/modules/forumSearch/src/main/ForumSearchApi.scala @@ -7,7 +7,6 @@ import lila.core.forum.{ ForumPostApi, ForumPostMini, ForumPostMiniView } import lila.core.id.ForumPostId import lila.search.client.SearchClient import lila.search.spec.{ ForumSource, Query } -import smithy4s.Timestamp final class ForumSearchApi( client: SearchClient, @@ -37,7 +36,7 @@ final class ForumSearchApi( author = view.post.userId.map(_.value), topicId = view.topic.id.value, troll = view.post.troll, - date = Timestamp.fromInstant(view.post.createdAt) + date = view.post.createdAt.toEpochMilli() ) def reset = diff --git a/project/Dependencies.scala b/project/Dependencies.scala index 273cb5afdb76f..52412dd976ab2 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -26,7 +26,7 @@ object Dependencies { val lettuce = "io.lettuce" % "lettuce-core" % "6.3.2.RELEASE" val nettyTransport = ("io.netty" % s"netty-transport-native-$notifier" % "4.1.111.Final").classifier(s"$os-$arch") - val lilaSearch = "org.lichess.search" %% "client" % "3.0.0-RC8" + val lilaSearch = "org.lichess.search" %% "client" % "3.0.0-RC9" val munit = "org.scalameta" %% "munit" % "1.0.0" % Test val uaparser = "org.uaparser" %% "uap-scala" % "0.17.0" val apacheText = "org.apache.commons" % "commons-text" % "1.12.0" From f1bbf837f7086078884873980c202628cc9f80e4 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 07:21:34 +0200 Subject: [PATCH 037/168] New Crowdin updates (#15495) * New translations: site.xml (Danish) * New translations: site.xml (Spanish) * New translations: site.xml (Catalan) * New translations: site.xml (German) * New translations: site.xml (Polish) * New translations: site.xml (Slovak) * New translations: site.xml (Albanian) * New translations: site.xml (Portuguese) * New translations: site.xml (Portuguese, Brazilian) * New translations: site.xml (Galician) * New translations: site.xml (Sorani (Kurdish)) * New translations: site.xml (English, United States) * New translations: site.xml (French) * New translations: site.xml (Norwegian Nynorsk) * New translations: faq.xml (English, United States) * New translations: site.xml (Persian) * New translations: site.xml (Swiss German) * New translations: site.xml (Afrikaans) * New translations: site.xml (Hebrew) * New translations: site.xml (Slovenian) * New translations: site.xml (Russian) * New translations: site.xml (Danish) * New translations: challenge.xml (Danish) * New translations: site.xml (Vietnamese) --- translation/dest/challenge/da-DK.xml | 2 +- translation/dest/faq/en-US.xml | 2 +- translation/dest/site/af-ZA.xml | 2 ++ translation/dest/site/ca-ES.xml | 2 ++ translation/dest/site/ckb-IR.xml | 2 ++ translation/dest/site/da-DK.xml | 4 +++- translation/dest/site/de-DE.xml | 2 ++ translation/dest/site/en-US.xml | 2 ++ translation/dest/site/es-ES.xml | 2 ++ translation/dest/site/fa-IR.xml | 2 ++ translation/dest/site/fr-FR.xml | 1 + translation/dest/site/gl-ES.xml | 2 ++ translation/dest/site/gsw-CH.xml | 2 ++ translation/dest/site/he-IL.xml | 2 ++ translation/dest/site/nn-NO.xml | 2 ++ translation/dest/site/pl-PL.xml | 2 ++ translation/dest/site/pt-BR.xml | 2 ++ translation/dest/site/pt-PT.xml | 2 ++ translation/dest/site/ru-RU.xml | 2 ++ translation/dest/site/sk-SK.xml | 2 ++ translation/dest/site/sl-SI.xml | 2 ++ translation/dest/site/sq-AL.xml | 2 ++ translation/dest/site/vi-VN.xml | 2 ++ 23 files changed, 44 insertions(+), 3 deletions(-) diff --git a/translation/dest/challenge/da-DK.xml b/translation/dest/challenge/da-DK.xml index 1bb763b89c775..d93acfe401ffd 100644 --- a/translation/dest/challenge/da-DK.xml +++ b/translation/dest/challenge/da-DK.xml @@ -12,7 +12,7 @@ Kan ikke udfordre på grund af provisorisk %s rating. %s accepterer kun udfordringer fra venner. Jeg accepterer ikke udfordringer i øjeblikket. - Det er ikke et godt tidspunkt for mig, spørg venligst igen senere. + Det er ikke et godt tidspunkt for mig, spørg gerne igen senere. Denne tidskontrol er for hurtig for mig. Udfordr mig gerne igen med et langsommere spil. Denne tidskontrol er for langsom for mig. Udfordr mig gerne igen med et hurtigere spil. Jeg accepterer ikke udfordringer med denne tidskontrol. diff --git a/translation/dest/faq/en-US.xml b/translation/dest/faq/en-US.xml index 3ecb689732a68..df2a03671a1ae 100644 --- a/translation/dest/faq/en-US.xml +++ b/translation/dest/faq/en-US.xml @@ -167,7 +167,7 @@ We show the red icon to alert you when this happens. Often you can explicitly al 3. Click Cookies and Site Permissions 4. Scroll down and click Media autoplay 5. Add lichess.org to Allow - Stop me from playing. + Stop myself from playing stand-alone mental health condition Lichess userstyles fewer lobby pools diff --git a/translation/dest/site/af-ZA.xml b/translation/dest/site/af-ZA.xml index e6f5a6d2e36c1..7d5e2587694c4 100644 --- a/translation/dest/site/af-ZA.xml +++ b/translation/dest/site/af-ZA.xml @@ -70,6 +70,8 @@ Bevorder variasie Maak hooflyn Verwyder hiervandaan + Verberg variasies + Vertoon variasies Forseer variasie Kopieer variasie-PGN Skuif diff --git a/translation/dest/site/ca-ES.xml b/translation/dest/site/ca-ES.xml index 4e5da1414ddd1..88b4f62d71dd2 100644 --- a/translation/dest/site/ca-ES.xml +++ b/translation/dest/site/ca-ES.xml @@ -70,6 +70,8 @@ Promoure variant Convertir en línia principal Esborrar des d\'aquí + Amagar variacions + Expandir variacions Forçar variant Copia el PGN de la variació Moviment diff --git a/translation/dest/site/ckb-IR.xml b/translation/dest/site/ckb-IR.xml index 68697cad916f3..31055d425193a 100644 --- a/translation/dest/site/ckb-IR.xml +++ b/translation/dest/site/ckb-IR.xml @@ -70,6 +70,8 @@ برەوپێدانی جۆراوجۆر کردن بە رستەی سەرەکی لێرە بیسڕەوە + تێکدانی لایەنەکانی دیکە + درێژکردنەوەی لایەنەکانی دیکە هێزی جۆراوجۆر کۆپی گۆڕانکاری PGN جوڵە diff --git a/translation/dest/site/da-DK.xml b/translation/dest/site/da-DK.xml index fe3de8d53c803..404fa19acad46 100644 --- a/translation/dest/site/da-DK.xml +++ b/translation/dest/site/da-DK.xml @@ -70,6 +70,8 @@ Forfrem variant Gør til hovedlinjen Slet herfra + Fold variationer sammen + Udvid variationer Gennemtving variation Kopiér variant-PGN Træk @@ -799,7 +801,7 @@ Nulstil farver til standard Brikker Indlejr på din hjemmeside - Dette brugernavn er allerede i brug. Vælg venligst et andet og prøv igen. + Dette brugernavn er allerede i brug. Vælg et andet og prøv igen. Brugernavnet skal begynde med et bogstav. Brugernavnet skal afsluttes med et bogstav eller et tal. Brugernavnet må kun indeholde bogstaver, tal, underscore og bindestreger. diff --git a/translation/dest/site/de-DE.xml b/translation/dest/site/de-DE.xml index fb7b076f3b623..ae4a17e4c7e0c 100644 --- a/translation/dest/site/de-DE.xml +++ b/translation/dest/site/de-DE.xml @@ -70,6 +70,8 @@ Variante aufwerten Zur Hauptvariante machen Ab hier löschen + Varianten einklappen + Varianten ausklappen Variante erzwingen PGN-Variante kopieren Zug diff --git a/translation/dest/site/en-US.xml b/translation/dest/site/en-US.xml index bbcb425b17843..332903fbaaaaa 100644 --- a/translation/dest/site/en-US.xml +++ b/translation/dest/site/en-US.xml @@ -70,6 +70,8 @@ Promote variation Make main line Delete from here + Collapse variations + Expand variations Force variation Copy variation PGN Move diff --git a/translation/dest/site/es-ES.xml b/translation/dest/site/es-ES.xml index 411a758e6657d..c1e3946adc929 100644 --- a/translation/dest/site/es-ES.xml +++ b/translation/dest/site/es-ES.xml @@ -70,6 +70,8 @@ Promocionar variante Convertir en línea principal Borrar a partir de aquí + Cerrar variantes + Abrir variantes Convertir en variante Copiar PGN de la variante Movimiento diff --git a/translation/dest/site/fa-IR.xml b/translation/dest/site/fa-IR.xml index e16aacef370da..f4d20c9beedd6 100644 --- a/translation/dest/site/fa-IR.xml +++ b/translation/dest/site/fa-IR.xml @@ -70,6 +70,8 @@ افزایش عمق شاخه اصلی خط کنونی را به خط اصلی تبدیل کنید از اینجا به بعد را پاک کنید + بستن شاخه‌ها + باز کردن شاخه‌ها نتیجه تحلیل را به عنوان یکی از تنوعهای بازی انتخاب نمایید کپی PGN این شاخه انتقال بدهید diff --git a/translation/dest/site/fr-FR.xml b/translation/dest/site/fr-FR.xml index c07b89a488af3..117f82cc56078 100644 --- a/translation/dest/site/fr-FR.xml +++ b/translation/dest/site/fr-FR.xml @@ -70,6 +70,7 @@ Promouvoir la variante En faire la variante principale Supprimer à partir d\'ici + Afficher les variantes Forcer la variante Copier le PGN de la variante Coup diff --git a/translation/dest/site/gl-ES.xml b/translation/dest/site/gl-ES.xml index 9089d78ff9c18..2a36432053d19 100644 --- a/translation/dest/site/gl-ES.xml +++ b/translation/dest/site/gl-ES.xml @@ -70,6 +70,8 @@ Promover variante Converter en liña principal Borrar desde aquí + Contraer as variantes + Despregar as variantes Forzar variante Copiar o PGN da variante Movemento diff --git a/translation/dest/site/gsw-CH.xml b/translation/dest/site/gsw-CH.xml index b482e0232a6e8..7c442c77447cf 100644 --- a/translation/dest/site/gsw-CH.xml +++ b/translation/dest/site/gsw-CH.xml @@ -70,6 +70,8 @@ Variante ufwerte Zur Hauptvariante mache Ab da lösche + Variante kollabiere + Variante erwitere Variante erzwinge PGN Variante kopiere Zug diff --git a/translation/dest/site/he-IL.xml b/translation/dest/site/he-IL.xml index ac2748d70ecdd..5db647b9d37b3 100644 --- a/translation/dest/site/he-IL.xml +++ b/translation/dest/site/he-IL.xml @@ -72,6 +72,8 @@ העדפת וריאנט קביעה כוריאנט הראשי מחיקה מכאן והלאה + הסתרת מהלכים חלופיים + הצגת מהלכים חלופיים וריאנט יחיד העתקת ה-PGN של הוריאנט מסע diff --git a/translation/dest/site/nn-NO.xml b/translation/dest/site/nn-NO.xml index bc31504d6a541..64a80714a7d24 100644 --- a/translation/dest/site/nn-NO.xml +++ b/translation/dest/site/nn-NO.xml @@ -70,6 +70,8 @@ Forfremjingsvariant Gjer til hovudline Slett herifrå + Skjul variantar + Vis variantar Tving fram varianten Kopier variant-PGN Trekk diff --git a/translation/dest/site/pl-PL.xml b/translation/dest/site/pl-PL.xml index 950fdaf959d33..bb9d250caf713 100644 --- a/translation/dest/site/pl-PL.xml +++ b/translation/dest/site/pl-PL.xml @@ -72,6 +72,8 @@ Promuj ten wariant Promuj na główny wariant Usuń od tego miejsca + Zwiń warianty + Rozwiń warianty Przedstaw jako wariant Skopiuj wariant PGN Ruch diff --git a/translation/dest/site/pt-BR.xml b/translation/dest/site/pt-BR.xml index 77ea363fdf7c8..705434a0b1373 100644 --- a/translation/dest/site/pt-BR.xml +++ b/translation/dest/site/pt-BR.xml @@ -70,6 +70,8 @@ Promover variante Transformar em linha principal Excluir a partir daqui + Esconder variantes + Mostrar variantes Variante forçada Copiar PGN da variante Movimentos diff --git a/translation/dest/site/pt-PT.xml b/translation/dest/site/pt-PT.xml index 1442735f6adf9..b6e72e924705f 100644 --- a/translation/dest/site/pt-PT.xml +++ b/translation/dest/site/pt-PT.xml @@ -70,6 +70,8 @@ Promover variante Tornar variante principal Eliminar a partir de aqui + Recolher variações + Expandir variações Forçar variante Copiar variação PGN Jogada diff --git a/translation/dest/site/ru-RU.xml b/translation/dest/site/ru-RU.xml index a39c228b6eaac..97db45e465e21 100644 --- a/translation/dest/site/ru-RU.xml +++ b/translation/dest/site/ru-RU.xml @@ -72,6 +72,8 @@ Повысить приоритет варианта Сделать этот вариант главным Удалить с этого места + Свернуть варианты + Развернуть варианты Сделать вариантом Скопировать вариант в формате PGN Ход diff --git a/translation/dest/site/sk-SK.xml b/translation/dest/site/sk-SK.xml index 71f953fda6f79..f7eb9ef00fe9f 100644 --- a/translation/dest/site/sk-SK.xml +++ b/translation/dest/site/sk-SK.xml @@ -72,6 +72,8 @@ Povýšiť variant Povýšiť na hlavnú variantu Vymazať odtiaľto + Skryť varianty + Ukázať varianty Povýšiť variant Kopírovať PGN variantu Ťah diff --git a/translation/dest/site/sl-SI.xml b/translation/dest/site/sl-SI.xml index 5f7b9ff27ed38..45b0341f50a14 100644 --- a/translation/dest/site/sl-SI.xml +++ b/translation/dest/site/sl-SI.xml @@ -72,6 +72,8 @@ Promoviraj variacijo Nastavite kot glavno varianto Izbriši od tukaj + Strni različice + Razširite različice Vsili varianto Kopiraj različico PGN Poteza diff --git a/translation/dest/site/sq-AL.xml b/translation/dest/site/sq-AL.xml index ad98c27935570..81ebc24c7f7be 100644 --- a/translation/dest/site/sq-AL.xml +++ b/translation/dest/site/sq-AL.xml @@ -70,6 +70,8 @@ Promovoni variacion Bëje variantin kryesor Fshije nga këtu + Tkurri variantet + Shfaqi variantet Detyro variant Kopjo PGN varianti Lëvizje diff --git a/translation/dest/site/vi-VN.xml b/translation/dest/site/vi-VN.xml index 95c6b1edb3eba..d74461afe3a23 100644 --- a/translation/dest/site/vi-VN.xml +++ b/translation/dest/site/vi-VN.xml @@ -69,6 +69,8 @@ Thay đổi biến Biến chính Xoá từ đây + Thu gọn các biến + Mở rộng các biến Đổi biến Sao chép biến PGN Nước cờ From ca49c021164b98c04854ac0ac8f8c33a58f66b26 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 08:03:08 +0200 Subject: [PATCH 038/168] read forum posts from primary --- modules/forum/src/main/ForumPostRepo.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/forum/src/main/ForumPostRepo.scala b/modules/forum/src/main/ForumPostRepo.scala index daf971b555f8b..33029a19f59fc 100644 --- a/modules/forum/src/main/ForumPostRepo.scala +++ b/modules/forum/src/main/ForumPostRepo.scala @@ -121,4 +121,4 @@ final class ForumPostRepo(val coll: Coll, filter: Filter = Safe)(using val filter = since.fold(noGhost)(instant => $and(noGhost, $doc("createdAt".$gt(instant)))) coll .find(filter, miniProjection.some) - .cursor[ForumPostMini](ReadPref.sec) + .cursor[ForumPostMini](ReadPref.priTemp) From 02eff3a3b8a3d35aa8b7f0761da729fd99a18b9b Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 08:27:51 +0200 Subject: [PATCH 039/168] scala rewrite with for --- app/controllers/Study.scala | 115 ++++++++++++++++++------------------ 1 file changed, 58 insertions(+), 57 deletions(-) diff --git a/app/controllers/Study.scala b/app/controllers/Study.scala index 231bc2b23e552..891c315883d64 100644 --- a/app/controllers/Study.scala +++ b/app/controllers/Study.scala @@ -33,13 +33,14 @@ final class Study( def search(text: String, page: Int) = OpenBody: Reasonable(page): if text.trim.isEmpty then - env.study.pager - .all(Orders.default, page) - .flatMap: pag => - preloadMembers(pag) >> negotiate( - Ok.page(views.study.list.all(pag, Orders.default)), - apiStudies(pag) - ) + for + pag <- env.study.pager.all(Orders.default, page) + _ <- preloadMembers(pag) + res <- negotiate( + Ok.page(views.study.list.all(pag, Orders.default)), + apiStudies(pag) + ) + yield res else env .studySearch(ctx.me)(text, page) @@ -62,13 +63,14 @@ final class Study( case order if !Orders.withoutSelector.contains(order) => Redirect(routes.Study.allDefault(page)) case order => - env.study.pager - .all(order, page) - .flatMap: pag => - preloadMembers(pag) >> negotiate( - Ok.page(views.study.list.all(pag, order)), - apiStudies(pag) - ) + for + pag <- env.study.pager.all(order, page) + _ <- preloadMembers(pag) + res <- negotiate( + Ok.page(views.study.list.all(pag, order)), + apiStudies(pag) + ) + yield res def byOwnerDefault(username: UserStr, page: Int) = byOwner(username, Orders.default, page) @@ -83,50 +85,55 @@ final class Study( ) def mine(order: Order, page: Int) = Auth { ctx ?=> me ?=> - env.study.pager - .mine(order, page) - .flatMap: pag => - preloadMembers(pag) >> negotiate( - env.study.topicApi.userTopics(me).flatMap { topics => - Ok.page(views.study.list.mine(pag, order, topics)) - }, - apiStudies(pag) - ) + for + pag <- env.study.pager.mine(order, page) + _ <- preloadMembers(pag) + res <- negotiate( + env.study.topicApi.userTopics(me).flatMap { topics => + Ok.page(views.study.list.mine(pag, order, topics)) + }, + apiStudies(pag) + ) + yield res } def minePublic(order: Order, page: Int) = Auth { ctx ?=> me ?=> - env.study.pager - .minePublic(order, page) - .flatMap: pag => - preloadMembers(pag) >> negotiate( - Ok.page(views.study.list.minePublic(pag, order)), - apiStudies(pag) - ) + for + pag <- env.study.pager.minePublic(order, page) + _ <- preloadMembers(pag) + res <- negotiate( + Ok.page(views.study.list.minePublic(pag, order)), + apiStudies(pag) + ) + yield res } def minePrivate(order: Order, page: Int) = Auth { ctx ?=> me ?=> - env.study.pager - .minePrivate(order, page) - .flatMap: pag => - preloadMembers(pag) >> negotiate( - Ok.page(views.study.list.minePrivate(pag, order)), - apiStudies(pag) - ) + for + pag <- env.study.pager.minePrivate(order, page) + _ <- preloadMembers(pag) + res <- negotiate( + Ok.page(views.study.list.minePrivate(pag, order)), + apiStudies(pag) + ) + yield res + } def mineMember(order: Order, page: Int) = Auth { ctx ?=> me ?=> - env.study.pager - .mineMember(order, page) - .flatMap: pag => - preloadMembers(pag) >> negotiate( - Ok.async: - env.study.topicApi - .userTopics(me) - .map: - views.study.list.mineMember(pag, order, _) - , - apiStudies(pag) - ) + for + pag <- env.study.pager.mineMember(order, page) + _ <- preloadMembers(pag) + res <- negotiate( + Ok.async: + env.study.topicApi + .userTopics(me) + .map: + views.study.list.mineMember(pag, order, _) + , + apiStudies(pag) + ) + yield res } def mineLikes(order: Order, page: Int) = Auth { ctx ?=> me ?=> @@ -327,9 +334,8 @@ final class Study( def clearChat(id: StudyId) = Auth { _ ?=> me ?=> env.study.api .isOwnerOrAdmin(id, me) - .flatMapz { + .flatMapz: env.chat.api.userChat.clear(id.into(ChatId)) - } .inject(Redirect(routes.Study.show(id))) } @@ -339,12 +345,7 @@ final class Study( val chapterDatas = data.toChapterDatas limit.studyPgnImport(me, rateLimited, cost = chapterDatas.size): env.study.api - .importPgns( - id, - chapterDatas, - sticky = data.sticky, - ctx.pref.showRatings - )(Who(me, sri)) + .importPgns(id, chapterDatas, sticky = data.sticky, ctx.pref.showRatings)(Who(me, sri)) .map(f) def importPgn(id: StudyId) = AuthBody { ctx ?=> me ?=> From 5e38f28c02fead339f5b6650f63462b298da18c9 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 09:05:45 +0200 Subject: [PATCH 040/168] fix relay form help indentation --- modules/relay/src/main/ui/FormUi.scala | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 26d1da70df36d..e9108a463e6cc 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -268,7 +268,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): pre("player name / rating / title / new name"), "All values are optional. Example:", pre("""Magnus Carlsen / 2863 / GM - YouGotLittUp / 1890 / / Louis Litt""") +YouGotLittUp / 1890 / / Louis Litt""") ).some, half = true )(form3.textarea(_)(rows := 3)), @@ -280,7 +280,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): pre("Team name; Fide Id or Player name"), "Example:", pre("""Team Cats ; 3408230 - Team Dogs ; Scooby Doo"""), +Team Dogs ; Scooby Doo"""), "By default the PGN tags WhiteTeam and BlackTeam are used." ).some, half = true @@ -389,8 +389,8 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): br, "Example:", pre("""Youth Championship 2024 - tour1-id Youth Championship 2024 | G20 - tour2-id Youth Championship 2024 | G16 - """) +tour1-id Youth Championship 2024 | G20 +tour2-id Youth Championship 2024 | G16 +""") ) ) From b4d06bd54e9a9199a0920768f7d37a5d4bda1ab2 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 09:08:54 +0200 Subject: [PATCH 041/168] rename broadcast private tier --- modules/relay/src/main/RelayTour.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayTour.scala b/modules/relay/src/main/RelayTour.scala index 749caecec038d..1f3b1ccd1cc25 100644 --- a/modules/relay/src/main/RelayTour.scala +++ b/modules/relay/src/main/RelayTour.scala @@ -60,7 +60,7 @@ object RelayTour: NORMAL.toString -> "Official: normal tier", HIGH.toString -> "Official: high tier", BEST.toString -> "Official: best tier", - PRIVATE.toString -> "Official: Private" + PRIVATE.toString -> "Private" ) def name(tier: Tier) = options.collectFirst { case (t, n) if t == tier.toString => n From 7fed8a65a4bb07721e903c0ffd2df3167c20586a Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 09:18:52 +0200 Subject: [PATCH 042/168] hide broadcast pinned streamer form --- modules/relay/src/main/ui/FormUi.scala | 63 ++++++++++++++------------ 1 file changed, 33 insertions(+), 30 deletions(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index e9108a463e6cc..d8bda1ef7337c 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -320,38 +320,41 @@ Team Dogs ; Scooby Doo"""), form3.select(_, langList.popularLanguagesForm.choices) ), tg.map: t => - div( - cls := "relay-pinned-streamer-edit", - data("post-url") := routes.RelayTour.image(t.tour.id, "pinnedStreamerImage".some) - )( + details( + summary("Pinned streamer"), div( - form3.group( - form("pinnedStreamer"), - "Pinned streamer", - help = frag( - p("The pinned streamer is featured even when they're not watching the broadcast."), - p("An optional placeholder image will embed their stream when clicked."), - p( - "To upload one, you must first submit this form with a pinned streamer. " - + "Then return to this page and choose an image." - ) - ).some - )(form3.input(_)), - span( - button(tpe := "button", cls := "button streamer-select-image")("select image"), - button( - tpe := "button", - cls := "button button-empty button-red streamer-delete-image", - data("post-url") := routes.RelayTour.image(t.tour.id, "pinnedStreamerImage".some) - )("delete image") - ) - ), - ui.thumbnail(t.tour.pinnedStreamerImage, _.Size.Small16x9)( - cls := List( - "streamer-drop-target" -> true, - "user-image" -> t.tour.pinnedStreamerImage.isDefined + cls := "relay-pinned-streamer-edit", + data("post-url") := routes.RelayTour.image(t.tour.id, "pinnedStreamerImage".some) + )( + div( + form3.group( + form("pinnedStreamer"), + "Pinned streamer", + help = frag( + p("The pinned streamer is featured even when they're not watching the broadcast."), + p("An optional placeholder image will embed their stream when clicked."), + p( + "To upload one, you must first submit this form with a pinned streamer. " + + "Then return to this page and choose an image." + ) + ).some + )(form3.input(_)), + span( + button(tpe := "button", cls := "button streamer-select-image")("select image"), + button( + tpe := "button", + cls := "button button-empty button-red streamer-delete-image", + data("post-url") := routes.RelayTour.image(t.tour.id, "pinnedStreamerImage".some) + )("delete image") + ) ), - attr("draggable") := "true" + ui.thumbnail(t.tour.pinnedStreamerImage, _.Size.Small16x9)( + cls := List( + "streamer-drop-target" -> true, + "user-image" -> t.tour.pinnedStreamerImage.isDefined + ), + attr("draggable") := "true" + ) ) ) ) From 3fbb922212761ec8096a6484d8277ae173da1362 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 09:59:26 +0200 Subject: [PATCH 043/168] fix lobby game load background in light theme - closes #15502 --- ui/lobby/css/app/_app.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/lobby/css/app/_app.scss b/ui/lobby/css/app/_app.scss index 112ad2ba37c43..2fe0bb360b589 100644 --- a/ui/lobby/css/app/_app.scss +++ b/ui/lobby/css/app/_app.scss @@ -35,7 +35,7 @@ } .lredir { - background: $c-bg-box; + background: $c-bg-box !important; display: flex; .spinner { From f48a2f13dd61a9ec5c8e6ca967fb6e4c6eb2f65e Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 09:59:42 +0200 Subject: [PATCH 044/168] monospace in technical broadcast form textareas --- ui/bits/css/relay/_form.scss | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/ui/bits/css/relay/_form.scss b/ui/bits/css/relay/_form.scss index 4518e6ad7041c..5e2d055418f52 100644 --- a/ui/bits/css/relay/_form.scss +++ b/ui/bits/css/relay/_form.scss @@ -28,6 +28,13 @@ margin-bottom: 2em; } } + +#form3-grouping, +#form3-players, +#form3-teams { + font-family: monospace; +} + .relay-image { max-width: 100%; height: auto; From 72e520f645c9d13c402d332de83ff973983580f8 Mon Sep 17 00:00:00 2001 From: Jonathan Gamble Date: Sat, 15 Jun 2024 04:06:25 -0500 Subject: [PATCH 045/168] use scss variables for important z-index values --- ui/analyse/css/_context-menu.scss | 2 +- ui/analyse/css/_player-clock.scss | 2 +- ui/analyse/css/study/relay/_video-player.scss | 4 +- ui/bits/css/build/bits.lpv.scss | 2 +- ui/bits/css/streamer/_header.scss | 4 +- ui/bits/css/user/_show.scss | 2 +- ui/chess/css/_promotion.scss | 2 +- ui/common/css/abstract/_extends.scss | 4 +- ui/common/css/abstract/_z-index.scss | 80 ++++++++----------- ui/common/css/component/_announce.scss | 2 +- ui/common/css/component/_complete.scss | 2 +- ui/common/css/component/_dialog.scss | 4 +- ui/common/css/component/_friend-box.scss | 2 +- ui/common/css/component/_mselect.scss | 2 +- ui/common/css/component/_power-tip.scss | 2 +- ui/common/css/component/_reconnecting.scss | 2 +- ui/common/css/component/_subnav.scss | 2 +- ui/common/css/component/_zen-toggle.scss | 2 +- ui/common/css/header/_buttons.scss | 2 +- ui/common/css/header/_header.scss | 2 +- ui/common/css/header/_topnav-hidden.scss | 6 +- ui/common/css/theme/board/_chessground.scss | 14 ++-- ui/game/css/_row.scss | 2 +- ui/lobby/css/app/_hook-chart.scss | 2 +- ui/mod/css/_inquiry.scss | 2 +- ui/mod/css/_user.scss | 2 +- ui/round/css/_meta.scss | 2 +- ui/tutor/css/_util.scss | 2 +- 28 files changed, 72 insertions(+), 86 deletions(-) diff --git a/ui/analyse/css/_context-menu.scss b/ui/analyse/css/_context-menu.scss index 09dba62142b6b..93ba9a97d95ab 100644 --- a/ui/analyse/css/_context-menu.scss +++ b/ui/analyse/css/_context-menu.scss @@ -4,7 +4,7 @@ background: $c-bg-box; position: absolute; display: none; - z-index: z('context-menu'); + z-index: $z-context-menu-108; cursor: default; user-select: none; -webkit-user-select: none; diff --git a/ui/analyse/css/_player-clock.scss b/ui/analyse/css/_player-clock.scss index d4f0fdff870ce..94a777e2700ab 100644 --- a/ui/analyse/css/_player-clock.scss +++ b/ui/analyse/css/_player-clock.scss @@ -9,7 +9,7 @@ $clock-height: 20px; &.top { top: #{-$clock-height}; - z-index: z('above-site-header'); + z-index: $z-above-site-header-107; .is3d & { top: #{-$clock-height - 35px}; diff --git a/ui/analyse/css/study/relay/_video-player.scss b/ui/analyse/css/study/relay/_video-player.scss index 4b8333a02eda0..44a90f13872ad 100644 --- a/ui/analyse/css/study/relay/_video-player.scss +++ b/ui/analyse/css/study/relay/_video-player.scss @@ -1,5 +1,5 @@ #video-player { - z-index: z('video-player'); + z-index: $z-video-player-100; display: none; position: absolute; border: 0; @@ -11,7 +11,7 @@ } img.video-player-close { - z-index: z('video-player-controls'); + z-index: $z-video-player-controls-101; position: absolute; height: 20px; width: 20px; diff --git a/ui/bits/css/build/bits.lpv.scss b/ui/bits/css/build/bits.lpv.scss index 1df05a5fa6ac0..15552bd0997bd 100644 --- a/ui/bits/css/build/bits.lpv.scss +++ b/ui/bits/css/build/bits.lpv.scss @@ -8,7 +8,7 @@ } .lpv__gamebook { @extend %popup-shadow; - z-index: z('cg__board.overlay'); + z-index: $z-cg__board_overlay-100; position: absolute; top: 50%; @include inline-start(50%); diff --git a/ui/bits/css/streamer/_header.scss b/ui/bits/css/streamer/_header.scss index 9600d0805c774..087e134ee4aa9 100644 --- a/ui/bits/css/streamer/_header.scss +++ b/ui/bits/css/streamer/_header.scss @@ -133,11 +133,11 @@ gap: 1em; align-self: flex-start; font-weight: bold; - z-index: z('link-overlay'); + z-index: $z-link-overlay-2; } .streamer-profile { - z-index: z('link-overlay'); + z-index: $z-link-overlay-2; } .live .streamer-lang { diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 4520832709804..b6ed004f1b19e 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -53,7 +53,7 @@ .dropdown-window { @extend %dropdown-shadow; - z-index: z('link-overlay'); + z-index: $z-link-overlay-2; visibility: hidden; background: $c-bg-header-dropdown; border-radius: 3px 0 3px 3px; diff --git a/ui/chess/css/_promotion.scss b/ui/chess/css/_promotion.scss index 7c6ab1922e90a..b87cb4dbc313f 100644 --- a/ui/chess/css/_promotion.scss +++ b/ui/chess/css/_promotion.scss @@ -1,6 +1,6 @@ #promotion-choice { background: $m-bg--fade-30; - z-index: z('cg__promotion'); + z-index: $z-cg__promotion-205; position: absolute; width: var(---cg-width, 100%); diff --git a/ui/common/css/abstract/_extends.scss b/ui/common/css/abstract/_extends.scss index 83afbac2698d2..7c4aa78db700c 100644 --- a/ui/common/css/abstract/_extends.scss +++ b/ui/common/css/abstract/_extends.scss @@ -255,7 +255,7 @@ width: 100vw; // escape from bounding box of any fixed position parent height: 100vh; background: $c-page-mask; - z-index: z('fullscreen-mask'); + z-index: $z-fullscreen-mask-110; } %link-overlay { @@ -264,7 +264,7 @@ left: 0; width: 100%; height: 100%; - z-index: z('link-overlay'); + z-index: $z-link-overlay-2; } %abs-100 { diff --git a/ui/common/css/abstract/_z-index.scss b/ui/common/css/abstract/_z-index.scss index 21b23109deab7..46462a3eb330c 100644 --- a/ui/common/css/abstract/_z-index.scss +++ b/ui/common/css/abstract/_z-index.scss @@ -1,48 +1,34 @@ -/// Z-indexes map, gathering all Z layers of the application -/// @access private -/// @type Map -/// @prop {String} key - Layer’s name -/// @prop {Number} value - Z value mapped to the key -$z-indexes: ( - 'cg__promotion': 205, - 'cg__piece.dragging': 204, - 'cg__board.overlay': 100, - 'cg__piece.anim': 3, - 'cg__svg.cg-shapes': 2, - 'cg__svg.cg-custom-svgs': 4, - 'cg__cg-auto-pieces': 2, - 'cg__piece': 2, - 'cg__piece.fading': 1, - 'powertip': 120, - 'complete': 113, - 'inquiry': 112, - 'zen-toggle': 111, - 'modal': 111, - 'mselect': 111, - 'topnav': 111, - 'fullscreen-mask': 110, - 'dropdown': 109, - 'context-menu': 108, - 'above-site-header': 107, - 'site-header': 106, - 'network-status': 105, - 'tour-reminder': 104, - 'video-player-controls': 101, - 'video-player': 100, - 'mz-menu': 4, - 'above-link-overlay': 3, - 'friend-box': 2, - 'link-overlay': 2, - 'game-bookmark': 2, - 'subnav-side': 2, - 'default': 0, -); +/// Z-indexes, gathering all Z layers of the application -/// Get a z-index value from a layer name -/// @access public -/// @param {String} $layer - Layer’s name -/// @return {Number} -/// @require $z-indexes -@function z($layer) { - @return map-get($z-indexes, $layer); -} +$z-cg__promotion-205: 205; +$z-cg__piece_dragging-204: 204; +$z-cg__board_overlay-100: 100; +$z-cg__piece_anim-3: 3; +$z-cg__svg_cg-shapes-2: 2; +$z-cg__svg_cg-custom-svgs-4: 4; +$z-cg__cg-auto-pieces-2: 2; +$z-cg__piece-2: 2; +$z-cg__piece_fading-1: 1; +$z-powertip-120: 120; +$z-complete-113: 113; +$z-inquiry-112: 112; +$z-zen-toggle-111: 111; +$z-modal-111: 111; +$z-mselect-111: 111; +$z-topnav-111: 111; +$z-fullscreen-mask-110: 110; +$z-dropdown-109: 109; +$z-context-menu-108: 108; +$z-above-site-header-107: 107; +$z-site-header-106: 106; +$z-network-status-105: 105; +$z-tour-reminder-104: 104; +$z-video-player-controls-101: 101; +$z-video-player-100: 100; +$z-mz-menu-4: 4; +$z-above-link-overlay-3: 3; +$z-friend-box-2: 2; +$z-link-overlay-2: 2; +$z-game-bookmark-2: 2; +$z-subnav-side-2: 2; +$z-default-0: 0; diff --git a/ui/common/css/component/_announce.scss b/ui/common/css/component/_announce.scss index cc9b4f6c7c056..2592adc209745 100644 --- a/ui/common/css/component/_announce.scss +++ b/ui/common/css/component/_announce.scss @@ -8,7 +8,7 @@ background: $c-primary; color: $c-over; padding: 0.7rem 1rem; - z-index: z('tour-reminder'); + z-index: $z-tour-reminder-104; width: 100%; @media (min-width: at-least($xx-small)) { diff --git a/ui/common/css/component/_complete.scss b/ui/common/css/component/_complete.scss index 3bf7122d06a9f..7f63b3834c135 100644 --- a/ui/common/css/component/_complete.scss +++ b/ui/common/css/component/_complete.scss @@ -8,7 +8,7 @@ position: absolute; top: 100%; - z-index: z('complete'); + z-index: $z-complete-113; width: 14em; min-height: 2em; background-color: $c-bg-box; diff --git a/ui/common/css/component/_dialog.scss b/ui/common/css/component/_dialog.scss index 9f544157d558a..a78efd82223ec 100644 --- a/ui/common/css/component/_dialog.scss +++ b/ui/common/css/component/_dialog.scss @@ -9,7 +9,7 @@ dialog { @include if-rtl { transform: translate(50%, -50%); } - z-index: z('modal'); + z-index: $z-modal-111; padding: 0; border: none; background: $c-bg-high; @@ -38,7 +38,7 @@ dialog { @include inline-end(4px); width: 40px; // bigger for phones height: 40px; - z-index: z('modal') + 1; + z-index: $z-modal-111 + 1; background: $c-bg-high; color: $c-font; border-radius: 6px; diff --git a/ui/common/css/component/_friend-box.scss b/ui/common/css/component/_friend-box.scss index 303efdeba76b7..fe993d6df1a68 100644 --- a/ui/common/css/component/_friend-box.scss +++ b/ui/common/css/component/_friend-box.scss @@ -8,7 +8,7 @@ position: fixed; bottom: 0; @include inline-end(0); - z-index: z('friend-box'); + z-index: $z-friend-box-2; background: $c-bg-popup; border: $border; border-inline-end: 0; diff --git a/ui/common/css/component/_mselect.scss b/ui/common/css/component/_mselect.scss index e9fc66f17f8b5..1a41d794b5ccc 100644 --- a/ui/common/css/component/_mselect.scss +++ b/ui/common/css/component/_mselect.scss @@ -44,7 +44,7 @@ $c-mselect: $c-primary; min-width: 100%; max-height: 60vh; overflow-y: auto; - z-index: z('mselect'); + z-index: $z-mselect-111; background: $c-bg-popup; transform: scale(1, 0); transform-origin: top; diff --git a/ui/common/css/component/_power-tip.scss b/ui/common/css/component/_power-tip.scss index bc14800365541..29066ac33ddc1 100644 --- a/ui/common/css/component/_power-tip.scss +++ b/ui/common/css/component/_power-tip.scss @@ -8,7 +8,7 @@ background: $c-bg-popup; display: none; position: absolute; - z-index: z('powertip'); + z-index: $z-powertip-120; .mini-game__player { @include padding-direction(3px, 0.5em, 0.3em, 0.7em); diff --git a/ui/common/css/component/_reconnecting.scss b/ui/common/css/component/_reconnecting.scss index dee881bf20cbf..15a930ad39300 100644 --- a/ui/common/css/component/_reconnecting.scss +++ b/ui/common/css/component/_reconnecting.scss @@ -25,7 +25,7 @@ $recon-height: 2.5rem; height: $recon-height; padding: 0 1rem; border-top-right-radius: 3px; - z-index: z('network-status'); + z-index: $z-network-status-105; opacity: 0; transform: translateY($recon-height); diff --git a/ui/common/css/component/_subnav.scss b/ui/common/css/component/_subnav.scss index 4d0b792dc52c8..5b05b9267a07c 100644 --- a/ui/common/css/component/_subnav.scss +++ b/ui/common/css/component/_subnav.scss @@ -56,7 +56,7 @@ @include mq-subnav-side { margin-top: 5px; - z-index: z('subnav-side'); + z-index: $z-subnav-side-2; /* active border must go over the page content */ a { diff --git a/ui/common/css/component/_zen-toggle.scss b/ui/common/css/component/_zen-toggle.scss index 9171b9dc94d20..6645e30213d98 100644 --- a/ui/common/css/component/_zen-toggle.scss +++ b/ui/common/css/component/_zen-toggle.scss @@ -3,7 +3,7 @@ display: none; flex-flow: row nowrap; align-items: center; - z-index: z('zen-toggle'); + z-index: $z-zen-toggle-111; body.zenable.zen & { display: flex; diff --git a/ui/common/css/header/_buttons.scss b/ui/common/css/header/_buttons.scss index 0df6d88607981..b3660e62dab63 100644 --- a/ui/common/css/header/_buttons.scss +++ b/ui/common/css/header/_buttons.scss @@ -55,7 +55,7 @@ @include inline-end(0); top: $site-header-height; background: $c-bg-header-dropdown; - z-index: z('dropdown'); + z-index: $z-dropdown-109; a, button { diff --git a/ui/common/css/header/_header.scss b/ui/common/css/header/_header.scss index 5594f5ed03a82..a246b28ed6a96 100644 --- a/ui/common/css/header/_header.scss +++ b/ui/common/css/header/_header.scss @@ -8,7 +8,7 @@ body > header { display: flex; justify-content: space-between; position: relative; - z-index: z('site-header'); + z-index: $z-site-header-106; max-width: 1800px; margin: 0 auto; user-select: none; diff --git a/ui/common/css/header/_topnav-hidden.scss b/ui/common/css/header/_topnav-hidden.scss index a227284c3b0d8..d4c694efc368c 100644 --- a/ui/common/css/header/_topnav-hidden.scss +++ b/ui/common/css/header/_topnav-hidden.scss @@ -13,7 +13,7 @@ width: $site-header-height; height: $site-header-height; cursor: pointer; - z-index: z('topnav'); + z-index: $z-topnav-111; &__in { &, @@ -145,7 +145,7 @@ section { flex: 1 0 50%; margin-top: 1rem; - z-index: z('topnav'); + z-index: $z-topnav-111; > a { font-size: 1.2em; @@ -174,7 +174,7 @@ } .topnav-toggle:checked ~ & { - z-index: z('topnav'); + z-index: $z-topnav-111; transform: translateX(0); a { opacity: 1; diff --git a/ui/common/css/theme/board/_chessground.scss b/ui/common/css/theme/board/_chessground.scss index 2e1ed8c562e28..2691386ed9872 100644 --- a/ui/common/css/theme/board/_chessground.scss +++ b/ui/common/css/theme/board/_chessground.scss @@ -127,21 +127,21 @@ piece { width: 12.5%; height: 12.5%; background-size: cover; - z-index: z('cg__piece'); + z-index: $z-cg__piece-2; will-change: transform; pointer-events: none; &.dragging { cursor: move; - z-index: z('cg__piece.dragging') !important; + z-index: $z-cg__piece_dragging-204 !important; } &.anim { - z-index: z('cg__piece.anim'); + z-index: $z-cg__piece_anim-3; } &.fading { - z-index: z('cg__piece.fading'); + z-index: $z-cg__piece_fading-1; opacity: 0.5; } @@ -173,11 +173,11 @@ cg-auto-pieces { cg-container .cg-shapes { opacity: 0.6; - z-index: z('cg__svg.cg-shapes'); + z-index: $z-cg__svg_cg-shapes-2; } cg-container .cg-custom-svgs { - z-index: z('cg__svg.cg-custom-svgs'); + z-index: $z-cg__svg_cg-custom-svgs-4; } cg-container .cg-shapes { @@ -189,7 +189,7 @@ cg-container .cg-custom-svgs svg { } cg-auto-pieces { - z-index: z('cg__cg-auto-pieces'); + z-index: $z-cg__cg-auto-pieces-2; piece { opacity: 0.3; diff --git a/ui/game/css/_row.scss b/ui/game/css/_row.scss index 81d3b733e41d7..c7fa1ae8c60c7 100644 --- a/ui/game/css/_row.scss +++ b/ui/game/css/_row.scss @@ -89,7 +89,7 @@ a { font-weight: bold; position: relative; - z-index: z('above-link-overlay'); + z-index: $z-above-link-overlay-3; } .anon { diff --git a/ui/lobby/css/app/_hook-chart.scss b/ui/lobby/css/app/_hook-chart.scss index 9f1eafda45a49..78e831b970374 100644 --- a/ui/lobby/css/app/_hook-chart.scss +++ b/ui/lobby/css/app/_hook-chart.scss @@ -77,7 +77,7 @@ display: none; background: $c-bg-box; position: absolute; - z-index: z('powertip'); + z-index: $z-powertip-120; .inner { @extend %flex-column; diff --git a/ui/mod/css/_inquiry.scss b/ui/mod/css/_inquiry.scss index 8d3a1352743f4..894dcf6db0e95 100644 --- a/ui/mod/css/_inquiry.scss +++ b/ui/mod/css/_inquiry.scss @@ -34,7 +34,7 @@ body.no-inquiry { top: 0; left: 0; right: 0; - z-index: z('inquiry'); + z-index: $z-inquiry-112; .costello { flex: 0 0 160px; diff --git a/ui/mod/css/_user.scss b/ui/mod/css/_user.scss index 78757cfc9dffa..a312d662e47d1 100644 --- a/ui/mod/css/_user.scss +++ b/ui/mod/css/_user.scss @@ -136,7 +136,7 @@ position: fixed; bottom: 0; background: $c-brag; - z-index: z('mz-menu'); + z-index: $z-mz-menu-4; // border-top: 1px solid $c-brag; // border-width: 3px 2px 0 2px; diff --git a/ui/round/css/_meta.scss b/ui/round/css/_meta.scss index cb441e3ad9eb9..ae32ae923e614 100644 --- a/ui/round/css/_meta.scss +++ b/ui/round/css/_meta.scss @@ -52,7 +52,7 @@ .bookmark { position: absolute; @include inline-end(0); - z-index: z('game-bookmark'); + z-index: $z-game-bookmark-2; color: $c-font-dim; ::before { diff --git a/ui/tutor/css/_util.scss b/ui/tutor/css/_util.scss index 2b7836022c967..74c33c42f7da2 100644 --- a/ui/tutor/css/_util.scss +++ b/ui/tutor/css/_util.scss @@ -36,7 +36,7 @@ padding: 0.8em 1.5em; width: 300px; position: absolute; - z-index: z('powertip'); + z-index: $z-powertip-120; } } } From 5c40632abfa87669652b982d4fb2b07cba17c5cd Mon Sep 17 00:00:00 2001 From: Ben Rollin Date: Sat, 15 Jun 2024 02:08:09 -0700 Subject: [PATCH 046/168] clear scenario timeouts when switching levels --- ui/learn/src/levelCtrl.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/ui/learn/src/levelCtrl.ts b/ui/learn/src/levelCtrl.ts index 3c224242cad48..002c0dea5a800 100644 --- a/ui/learn/src/levelCtrl.ts +++ b/ui/learn/src/levelCtrl.ts @@ -57,6 +57,8 @@ export class LevelCtrl { readonly opts: LevelOpts, readonly redraw: () => void, ) { + timeouts.clearTimeouts(); + this.isAppleLevel = prop(blueprint.apples?.length > 0); this.items = makeItems({ apples: blueprint.apples }); this.chess = makeChess(blueprint.fen, blueprint.emptyApples ? [] : this.items.appleKeys()); From 9cecd70978c0b00191583d1b5aa3cb7bfafb284d Mon Sep 17 00:00:00 2001 From: kraktus Date: Sat, 15 Jun 2024 12:23:15 +0200 Subject: [PATCH 047/168] thinner profile header (back to before profile menu) close https://github.com/lichess-org/lila/issues/15512 --- ui/bits/css/user/_show.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 4520832709804..7970466e5bd08 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -34,7 +34,7 @@ .user-actions { flex: 0 0 auto; - margin: 1em 1em 1em 0.3em; + margin: 0 1em 0 0.3em; form { display: inline; From 89710b0d377c12a7766b415673a4c8316d9bf6ff Mon Sep 17 00:00:00 2001 From: kraktus Date: Sat, 15 Jun 2024 12:56:20 +0200 Subject: [PATCH 048/168] Profile, fix dropdown button covering others when screen too narrow --- ui/bits/css/user/_show.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 4520832709804..475e276e3a994 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -28,7 +28,7 @@ .number-menu { flex: 0 1 auto; - overflow: hidden; + overflow-x: auto; margin: 0 0 0.2em 1em; } From 84415481c611b8e7a206f3a2f0983f930e785ebf Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 13:04:37 +0200 Subject: [PATCH 049/168] study search text.take(100) --- app/controllers/Study.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/controllers/Study.scala b/app/controllers/Study.scala index 891c315883d64..13d7f269a664c 100644 --- a/app/controllers/Study.scala +++ b/app/controllers/Study.scala @@ -43,7 +43,7 @@ final class Study( yield res else env - .studySearch(ctx.me)(text, page) + .studySearch(ctx.me)(text.take(100), page) .flatMap: pag => negotiate( Ok.page(views.study.list.search(pag, text)), From d842b21e69c489583d85c5a2716ba2cfb27b69ba Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 13:22:33 +0200 Subject: [PATCH 050/168] monitor broadcast crowds --- modules/common/src/main/mon.scala | 1 + modules/relay/src/main/Env.scala | 2 +- modules/relay/src/main/RelayApi.scala | 5 +++++ modules/relay/src/main/RelayRoundRepo.scala | 13 +++++++++++++ 4 files changed, 20 insertions(+), 1 deletion(-) diff --git a/modules/common/src/main/mon.scala b/modules/common/src/main/mon.scala index ee84be6d53dff..6fcfe9420ec23 100644 --- a/modules/common/src/main/mon.scala +++ b/modules/common/src/main/mon.scala @@ -284,6 +284,7 @@ object mon: def moves(official: Boolean, slug: String) = counter("relay.moves").withTags(relay(official, slug)) def fetchTime(official: Boolean, slug: String) = timer("relay.fetch.time").withTags(relay(official, slug)) def syncTime(official: Boolean, slug: String) = timer("relay.sync.time").withTags(relay(official, slug)) + def tourCrowd(tourId: RelayTourId) = gauge("relay.tour.crowd").withTag("tour", tourId.value) def httpGet(host: String, proxy: Option[String]) = future("relay.http.get", tags("host" -> host, "proxy" -> proxy.getOrElse("none"))) diff --git a/modules/relay/src/main/Env.scala b/modules/relay/src/main/Env.scala index bf3c2415fcdc9..98ddc40f3070f 100644 --- a/modules/relay/src/main/Env.scala +++ b/modules/relay/src/main/Env.scala @@ -122,7 +122,7 @@ final class Env( wire[RelayFetch] scheduler.scheduleWithFixedDelay(1 minute, 1 minute): () => - api.autoStart >> api.autoFinishNotSyncing + api.autoStart >> api.autoFinishNotSyncing >> api.monitorCrowd lila.common.Bus.subscribeFuns( "study" -> { case lila.core.study.RemoveStudy(studyId) => diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index a2a36920842ea..b29b071a4de2b 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -408,6 +408,11 @@ final class RelayApi( update(relay)(_.finish) } + private[relay] def monitorCrowd: Funit = + roundRepo.tourCrowds.map: crowds => + crowds.pp.foreach: (tourId, crowd) => + lila.mon.relay.tourCrowd(tourId).update(crowd) + private[relay] def WithRelay[A: Zero](id: RelayRoundId)(f: RelayRound => Fu[A]): Fu[A] = byId(id).flatMapz(f) diff --git a/modules/relay/src/main/RelayRoundRepo.scala b/modules/relay/src/main/RelayRoundRepo.scala index a288e03cbf128..1b8fab34c82f3 100644 --- a/modules/relay/src/main/RelayRoundRepo.scala +++ b/modules/relay/src/main/RelayRoundRepo.scala @@ -49,6 +49,19 @@ final private class RelayRoundRepo(val coll: Coll)(using Executor): def studyIdsOf(tourId: RelayTourId): Fu[List[StudyId]] = coll.distinctEasy[StudyId, List]("_id", selectors.tour(tourId)) + def tourCrowds: Fu[List[(RelayTourId, Int)]] = + coll + .aggregateList(maxDocs = 500, _.sec): framework => + import framework.* + Match($doc("sync.until" -> $exists(true))) -> + List(GroupField("_id")("crowd" -> SumField("crowd"))) + .map: docs => + for + doc <- docs + id <- doc.getAsOpt[RelayTourId]("_id") + crowd <- doc.getAsOpt[Int]("crowd") + yield (id, crowd) + private object RelayRoundRepo: object sort: From 0cc944c9c3c7d5ab6f3c3b3d70b0d80072a97156 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 14:20:57 +0200 Subject: [PATCH 051/168] upgrade chart.js --- pnpm-lock.yaml | 31 ++++++++++++++++++++++++------- ui/chart/package.json | 2 +- 2 files changed, 25 insertions(+), 8 deletions(-) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index f4e6dc0eceb28..b12793ea45865 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -257,17 +257,17 @@ importers: specifier: workspace:* version: link:../ceval chart.js: - specifier: 4.4.0 - version: 4.4.0 + specifier: 4.4.3 + version: 4.4.3 chartjs-adapter-dayjs-4: specifier: ^1.0.4 - version: 1.0.4(chart.js@4.4.0)(dayjs@1.11.10) + version: 1.0.4(chart.js@4.4.3)(dayjs@1.11.10) chartjs-plugin-datalabels: specifier: ^2.2.0 - version: 2.2.0(chart.js@4.4.0) + version: 2.2.0(chart.js@4.4.3) chartjs-plugin-zoom: specifier: ^2.0.1 - version: 2.0.1(chart.js@4.4.0) + version: 2.0.1(chart.js@4.4.3) common: specifier: workspace:* version: link:../common @@ -1346,6 +1346,10 @@ packages: resolution: {integrity: sha512-vQEj6d+z0dcsKLlQvbKIMYFHd3t8W/7L2vfJIbYcfyPcRx92CsHqECpueN8qVGNlKyDcr5wBrYAYKnfu/9Q1hQ==} engines: {pnpm: '>=7'} + chart.js@4.4.3: + resolution: {integrity: sha512-qK1gkGSRYcJzqrrzdR6a+I0vQ4/R+SoODXyAjscQ/4mzuNzySaMCd+hyVxitSY1+L2fjPD1Gbn+ibNqRmwQeLw==} + engines: {pnpm: '>=8'} + chartjs-adapter-dayjs-4@1.0.4: resolution: {integrity: sha512-yy9BAYW4aNzPVrCWZetbILegTRb7HokhgospPoC3b5iZ5qdlqNmXts2KdSp6AqnjkPAp/YWyHDxLvIvwt5x81w==} engines: {node: '>=10'} @@ -3034,18 +3038,31 @@ snapshots: dependencies: '@kurkle/color': 0.3.2 + chart.js@4.4.3: + dependencies: + '@kurkle/color': 0.3.2 + chartjs-adapter-dayjs-4@1.0.4(chart.js@4.4.0)(dayjs@1.11.10): dependencies: chart.js: 4.4.0 dayjs: 1.11.10 + chartjs-adapter-dayjs-4@1.0.4(chart.js@4.4.3)(dayjs@1.11.10): + dependencies: + chart.js: 4.4.3 + dayjs: 1.11.10 + chartjs-plugin-datalabels@2.2.0(chart.js@4.4.0): dependencies: chart.js: 4.4.0 - chartjs-plugin-zoom@2.0.1(chart.js@4.4.0): + chartjs-plugin-datalabels@2.2.0(chart.js@4.4.3): dependencies: - chart.js: 4.4.0 + chart.js: 4.4.3 + + chartjs-plugin-zoom@2.0.1(chart.js@4.4.3): + dependencies: + chart.js: 4.4.3 hammerjs: 2.0.8 check-error@1.0.3: diff --git a/ui/chart/package.json b/ui/chart/package.json index 4308cd4a6a40b..47c82da99ac7f 100644 --- a/ui/chart/package.json +++ b/ui/chart/package.json @@ -11,7 +11,7 @@ "dependencies": { "@juggle/resize-observer": "^3.4.0", "ceval": "workspace:*", - "chart.js": "4.4.0", + "chart.js": "4.4.3", "chartjs-adapter-dayjs-4": "^1.0.4", "chartjs-plugin-datalabels": "^2.2.0", "chartjs-plugin-zoom": "^2.0.1", From cffbbca3d5a028616d17ef68702e52d9b3657c4b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Gl=C3=B3rias?= <9739913+SergioGlorias@users.noreply.github.com> Date: Sat, 15 Jun 2024 13:55:35 +0100 Subject: [PATCH 052/168] Reduce caliente --- public/piece-css/caliente.css | 24 +-- public/piece/caliente/bB.svg | 206 +------------------------- public/piece/caliente/bK.svg | 232 +---------------------------- public/piece/caliente/bN.svg | 167 +-------------------- public/piece/caliente/bP.svg | 144 +----------------- public/piece/caliente/bQ.svg | 267 +--------------------------------- public/piece/caliente/bR.svg | 227 +---------------------------- public/piece/caliente/wB.svg | 199 +------------------------ public/piece/caliente/wK.svg | 226 +--------------------------- public/piece/caliente/wN.svg | 157 +------------------- public/piece/caliente/wP.svg | 139 +----------------- public/piece/caliente/wQ.svg | 261 +-------------------------------- public/piece/caliente/wR.svg | 218 +-------------------------- 13 files changed, 24 insertions(+), 2443 deletions(-) diff --git a/public/piece-css/caliente.css b/public/piece-css/caliente.css index f684f5f5340b9..a6850e5b72754 100644 --- a/public/piece-css/caliente.css +++ b/public/piece-css/caliente.css @@ -1,12 +1,12 @@ -.is2d .pawn.white {background-image:url('')} -.is2d .knight.white {background-image:url('')} -.is2d .bishop.white {background-image:url('')} -.is2d .rook.white {background-image:url('')} -.is2d .queen.white {background-image:url('')} -.is2d .king.white {background-image:url('')} -.is2d .pawn.black {background-image:url('')} -.is2d .knight.black {background-image:url('')} -.is2d .bishop.black {background-image:url('')} -.is2d .rook.black {background-image:url('')} -.is2d .queen.black {background-image:url('')} -.is2d .king.black {background-image:url('')} +.is2d .pawn.white {background-image:url('')} +.is2d .knight.white {background-image:url('')} +.is2d .bishop.white {background-image:url('')} +.is2d .rook.white {background-image:url('')} +.is2d .queen.white {background-image:url('')} +.is2d .king.white {background-image:url('')} +.is2d .pawn.black {background-image:url('')} +.is2d .knight.black {background-image:url('')} +.is2d .bishop.black {background-image:url('')} +.is2d .rook.black {background-image:url('')} +.is2d .queen.black {background-image:url('')} +.is2d .king.black {background-image:url('')} diff --git a/public/piece/caliente/bB.svg b/public/piece/caliente/bB.svg index 4935501e0ef2e..97efccbf874a8 100644 --- a/public/piece/caliente/bB.svg +++ b/public/piece/caliente/bB.svg @@ -1,205 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/bK.svg b/public/piece/caliente/bK.svg index 435def07cb935..28a36438750bc 100644 --- a/public/piece/caliente/bK.svg +++ b/public/piece/caliente/bK.svg @@ -1,231 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/bN.svg b/public/piece/caliente/bN.svg index c416437d88e6d..aa1590b3bce43 100644 --- a/public/piece/caliente/bN.svg +++ b/public/piece/caliente/bN.svg @@ -1,166 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/bP.svg b/public/piece/caliente/bP.svg index 08eeb11d80f09..73ceb65b70144 100644 --- a/public/piece/caliente/bP.svg +++ b/public/piece/caliente/bP.svg @@ -1,143 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/bQ.svg b/public/piece/caliente/bQ.svg index 343805525b364..71b790fde257e 100644 --- a/public/piece/caliente/bQ.svg +++ b/public/piece/caliente/bQ.svg @@ -1,266 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/bR.svg b/public/piece/caliente/bR.svg index e7a78ae2d0bf4..313d8fc10cb49 100644 --- a/public/piece/caliente/bR.svg +++ b/public/piece/caliente/bR.svg @@ -1,226 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/wB.svg b/public/piece/caliente/wB.svg index 9c1d32b054bb5..2fb7d754e8962 100644 --- a/public/piece/caliente/wB.svg +++ b/public/piece/caliente/wB.svg @@ -1,198 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/wK.svg b/public/piece/caliente/wK.svg index bcae54ae99066..32291bab79f78 100644 --- a/public/piece/caliente/wK.svg +++ b/public/piece/caliente/wK.svg @@ -1,225 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/wN.svg b/public/piece/caliente/wN.svg index 695bcbb8a868c..f1bbcd14a0acf 100644 --- a/public/piece/caliente/wN.svg +++ b/public/piece/caliente/wN.svg @@ -1,156 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/wP.svg b/public/piece/caliente/wP.svg index 103fc9b2156f8..09e31d9b1fb05 100644 --- a/public/piece/caliente/wP.svg +++ b/public/piece/caliente/wP.svg @@ -1,138 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/wQ.svg b/public/piece/caliente/wQ.svg index fef54e18f6a54..fe1af43da9bad 100644 --- a/public/piece/caliente/wQ.svg +++ b/public/piece/caliente/wQ.svg @@ -1,260 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file diff --git a/public/piece/caliente/wR.svg b/public/piece/caliente/wR.svg index 746ed7b0bc71e..9bbf690c4602e 100644 --- a/public/piece/caliente/wR.svg +++ b/public/piece/caliente/wR.svg @@ -1,217 +1 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + \ No newline at end of file From ed086ca8281858de3b79e2c438cf0d5759170dd7 Mon Sep 17 00:00:00 2001 From: kraktus Date: Sat, 15 Jun 2024 15:05:48 +0200 Subject: [PATCH 053/168] Appeal: display "my mark" after isolating an account --- modules/mod/src/main/ModlogApi.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/mod/src/main/ModlogApi.scala b/modules/mod/src/main/ModlogApi.scala index 8cdf6c3462b66..7e939d7371aef 100644 --- a/modules/mod/src/main/ModlogApi.scala +++ b/modules/mod/src/main/ModlogApi.scala @@ -22,7 +22,7 @@ final class ModlogApi(repo: ModlogRepo, userRepo: UserRepo, ircApi: IrcApi, pres private given Conversion[Me, ModId] = _.modId private val markActions = - List(Modlog.alt, Modlog.booster, Modlog.closeAccount, Modlog.engine, Modlog.troll, Modlog.rankban) + List(Modlog.alt, Modlog.booster, Modlog.closeAccount, Modlog.engine, Modlog.troll, Modlog.rankban, Modlog.isolate) def streamerDecline(streamerId: UserId)(using MyId) = add: Modlog(streamerId.some, Modlog.streamerDecline) From 1dd4e95fe009348585cdd49135a5ebea02ca8482 Mon Sep 17 00:00:00 2001 From: kraktus Date: Sat, 15 Jun 2024 15:06:41 +0200 Subject: [PATCH 054/168] scalafmt --- modules/mod/src/main/ModlogApi.scala | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/modules/mod/src/main/ModlogApi.scala b/modules/mod/src/main/ModlogApi.scala index 7e939d7371aef..4722e03d9f8b6 100644 --- a/modules/mod/src/main/ModlogApi.scala +++ b/modules/mod/src/main/ModlogApi.scala @@ -22,7 +22,15 @@ final class ModlogApi(repo: ModlogRepo, userRepo: UserRepo, ircApi: IrcApi, pres private given Conversion[Me, ModId] = _.modId private val markActions = - List(Modlog.alt, Modlog.booster, Modlog.closeAccount, Modlog.engine, Modlog.troll, Modlog.rankban, Modlog.isolate) + List( + Modlog.alt, + Modlog.booster, + Modlog.closeAccount, + Modlog.engine, + Modlog.troll, + Modlog.rankban, + Modlog.isolate + ) def streamerDecline(streamerId: UserId)(using MyId) = add: Modlog(streamerId.some, Modlog.streamerDecline) From 15c7ad02ddf5f5073cfee146b4ecdc2d051449ec Mon Sep 17 00:00:00 2001 From: kraktus Date: Sat, 15 Jun 2024 15:07:20 +0200 Subject: [PATCH 055/168] remove debug from broadcast crowd monitoring api --- modules/relay/src/main/RelayApi.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index b29b071a4de2b..f115665a50419 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -410,7 +410,7 @@ final class RelayApi( private[relay] def monitorCrowd: Funit = roundRepo.tourCrowds.map: crowds => - crowds.pp.foreach: (tourId, crowd) => + crowds.foreach: (tourId, crowd) => lila.mon.relay.tourCrowd(tourId).update(crowd) private[relay] def WithRelay[A: Zero](id: RelayRoundId)(f: RelayRound => Fu[A]): Fu[A] = From f872ad973b5a97c1ca7cb03e35aef665d8d2c290 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 15:16:21 +0200 Subject: [PATCH 056/168] delete bin/relay --- bin/relay/package.json | 15 --------------- bin/relay/server.js | 43 ------------------------------------------ bin/relay/yarn.lock | 25 ------------------------ 3 files changed, 83 deletions(-) delete mode 100644 bin/relay/package.json delete mode 100644 bin/relay/server.js delete mode 100644 bin/relay/yarn.lock diff --git a/bin/relay/package.json b/bin/relay/package.json deleted file mode 100644 index 8f288991d1811..0000000000000 --- a/bin/relay/package.json +++ /dev/null @@ -1,15 +0,0 @@ -{ - "name": "pgnstudyrelay-data", - "version": "1.0.0", - "description": "As an example, gct-09 has a complete set of files from before the games started until after they end", - "main": "server.js", - "scripts": { - "test": "echo \"Error: no test specified\" && exit 1", - "start": "node server.js" - }, - "author": "", - "license": "ISC", - "dependencies": { - "fs-extra": "^4.0.2" - } -} diff --git a/bin/relay/server.js b/bin/relay/server.js deleted file mode 100644 index 6ecb67b58e70d..0000000000000 --- a/bin/relay/server.js +++ /dev/null @@ -1,43 +0,0 @@ -const http = require('http'); -const fs = require('fs-extra'); - -const port = parseInt(process.argv[2]); -const dir = process.argv[3]; -const delay = parseInt(process.argv[4] || 1000); - -let files, - file, - completion = -1; - -fs.readdir(dir).then(list => { - files = list.filter(n => n.endsWith('.pgn')); - serveIndex(0); -}); - -function serveIndex(index) { - if (!files[index]) index = 0; - const percent = Math.floor((index * 100) / files.length); - if (percent > completion) { - completion = percent; - console.log(`${percent}%`); - } - file = files[index]; - setTimeout(() => serveIndex(index + 1, delay), delay); -} - -http - .createServer((request, response) => { - const path = `${dir}/${file}`; - return fs - .readFile(path, { - encoding: 'utf8', - }) - .then(content => { - console.log(`${path} ${content.length}`); - response.end(content); - }); - }) - .listen(port, err => { - if (err) return console.log(err); - console.log(`server is listening on ${port}`); - }); diff --git a/bin/relay/yarn.lock b/bin/relay/yarn.lock deleted file mode 100644 index f36b9aa6d04cf..0000000000000 --- a/bin/relay/yarn.lock +++ /dev/null @@ -1,25 +0,0 @@ -# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. -# yarn lockfile v1 - - -fs-extra@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-4.0.2.tgz#f91704c53d1b461f893452b0c307d9997647ab6b" - dependencies: - graceful-fs "^4.1.2" - jsonfile "^4.0.0" - universalify "^0.1.0" - -graceful-fs@^4.1.2, graceful-fs@^4.1.6: - version "4.1.11" - resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.1.11.tgz#0e8bdfe4d1ddb8854d64e04ea7c00e2a026e5658" - -jsonfile@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb" - optionalDependencies: - graceful-fs "^4.1.6" - -universalify@^0.1.0: - version "0.1.1" - resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.1.tgz#fa71badd4437af4c148841e3b3b165f9e9e590b7" From a3b524381e698e92d3cdd6038c7aa7ba1ed55ba4 Mon Sep 17 00:00:00 2001 From: Trevor Fitzgerald Date: Sat, 15 Jun 2024 09:33:31 -0400 Subject: [PATCH 057/168] Add study flair to json page view --- modules/study/src/main/JsonView.scala | 24 +++++++++++++----------- 1 file changed, 13 insertions(+), 11 deletions(-) diff --git a/modules/study/src/main/JsonView.scala b/modules/study/src/main/JsonView.scala index 200d4e35725fd..cb85ddc58e5ca 100644 --- a/modules/study/src/main/JsonView.scala +++ b/modules/study/src/main/JsonView.scala @@ -80,17 +80,19 @@ final class JsonView( .pipe(addChapterMode(c)) def pagerData(s: Study.WithChaptersAndLiked) = - Json.obj( - "id" -> s.study.id, - "name" -> s.study.name, - "liked" -> s.liked, - "likes" -> s.study.likes, - "updatedAt" -> s.study.updatedAt, - "owner" -> lightUserApi.sync(s.study.ownerId), - "chapters" -> s.chapters.take(Study.previewNbChapters), - "topics" -> s.study.topicsOrEmpty, - "members" -> s.study.members.members.values.take(Study.previewNbMembers) - ) + Json + .obj( + "id" -> s.study.id, + "name" -> s.study.name, + "liked" -> s.liked, + "likes" -> s.study.likes, + "updatedAt" -> s.study.updatedAt, + "owner" -> lightUserApi.sync(s.study.ownerId), + "chapters" -> s.chapters.take(Study.previewNbChapters), + "topics" -> s.study.topicsOrEmpty, + "members" -> s.study.members.members.values.take(Study.previewNbMembers) + ) + .add("flair", s.study.flair) private def addChapterMode(c: Chapter)(js: JsObject): JsObject = js.add("practice", c.isPractice) From 58a0ed56ba66b9c2f50c303904056f3109421b4b Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Sat, 15 Jun 2024 19:20:19 +0700 Subject: [PATCH 058/168] Stop using http request to index forum posts We use lila-search's ingestor instead --- modules/core/src/main/forum.scala | 2 -- modules/forum/src/main/ForumDelete.scala | 9 ++++----- modules/forum/src/main/ForumPostApi.scala | 5 +---- modules/forumSearch/src/main/Env.scala | 6 +----- 4 files changed, 6 insertions(+), 16 deletions(-) diff --git a/modules/core/src/main/forum.scala b/modules/core/src/main/forum.scala index 69840a3a78704..3493b6e9a0d33 100644 --- a/modules/core/src/main/forum.scala +++ b/modules/core/src/main/forum.scala @@ -10,9 +10,7 @@ import reactivemongo.api.bson.Macros.Annotations.Key enum BusForum: case CreatePost(post: ForumPostMini) case RemovePost(id: ForumPostId, by: Option[UserId], text: String, asAdmin: Boolean)(using val me: MyId) - case RemovePosts(ids: List[ForumPostId]) // erasing = blanking, still in db but with empty text - case ErasePost(id: ForumPostId) case ErasePosts(ids: List[ForumPostId]) object BusForum: diff --git a/modules/forum/src/main/ForumDelete.scala b/modules/forum/src/main/ForumDelete.scala index ace794c95053d..0d508311e8dbc 100644 --- a/modules/forum/src/main/ForumDelete.scala +++ b/modules/forum/src/main/ForumDelete.scala @@ -19,16 +19,15 @@ final class ForumDelete( .allByUserCursor(user) .documentSource() .mapAsyncUnordered(4): post => - postApi.viewOf(post).flatMap { _.so(deletePost) } + postApi.viewOf(post).flatMap(_.so(deletePost)) .runWith(Sink.ignore) .void def deleteTopic(view: PostView)(using Me): Funit = for - postIds <- postRepo.idsByTopicId(view.topic.id) - _ <- postRepo.removeByTopic(view.topic.id) - _ <- topicRepo.remove(view.topic) - _ <- categApi.denormalize(view.categ) + _ <- postRepo.removeByTopic(view.topic.id) + _ <- topicRepo.remove(view.topic) + _ <- categApi.denormalize(view.categ) yield publishDelete(view.post) def deletePost(view: PostView)(using Me): Funit = diff --git a/modules/forum/src/main/ForumPostApi.scala b/modules/forum/src/main/ForumPostApi.scala index 8411bde90e844..77dae8e497674 100644 --- a/modules/forum/src/main/ForumPostApi.scala +++ b/modules/forum/src/main/ForumPostApi.scala @@ -223,14 +223,11 @@ final class ForumPostApi( postRepo.coll.update .one($id(post.id), post.erase) .void - .andDo: - Bus.pub(BusForum.ErasePost(post.id)) def eraseFromSearchIndex(user: User): Funit = postRepo.coll .distinctEasy[ForumPostId, List]("_id", $doc("userId" -> user.id), _.sec) - .map: ids => - Bus.pub(BusForum.ErasePosts(ids)) + .map(ids => Bus.pub(BusForum.ErasePosts(ids))) def teamIdOfPostId(postId: ForumPostId): Fu[Option[TeamId]] = postRepo.coll.byId[ForumPost](postId).flatMapz { post => diff --git a/modules/forumSearch/src/main/Env.scala b/modules/forumSearch/src/main/Env.scala index 9ec7cbf024c8c..89dc672311df7 100644 --- a/modules/forumSearch/src/main/Env.scala +++ b/modules/forumSearch/src/main/Env.scala @@ -45,8 +45,4 @@ final class Env( private lazy val paginatorBuilder = lila.search.PaginatorBuilder(api, config.maxPerPage) lila.common.Bus.sub[BusForum]: - case CreatePost(post) => api.store(post) - case RemovePost(id, _, _, _) => client.deleteById(index, id.value) - case RemovePosts(ids) => client.deleteByIds(index, ids.map(_.value)) - case ErasePost(id) => client.deleteById(index, id.value) - case ErasePosts(ids) => client.deleteByIds(index, ids.map(_.value)) + case ErasePosts(ids) => client.deleteByIds(index, ids.map(_.value)) From 81e6907bdf38f137e398cb9919c1c615f056ec41 Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Sat, 15 Jun 2024 19:21:26 +0700 Subject: [PATCH 059/168] Remove unneded forum search write apis --- modules/api/src/main/Cli.scala | 1 - modules/forumSearch/src/main/Env.scala | 12 ------- .../forumSearch/src/main/ForumSearchApi.scala | 35 ------------------- 3 files changed, 48 deletions(-) diff --git a/modules/api/src/main/Cli.scala b/modules/api/src/main/Cli.scala index 8078f3c419278..da551708e9062 100644 --- a/modules/api/src/main/Cli.scala +++ b/modules/api/src/main/Cli.scala @@ -56,7 +56,6 @@ final private[api] class Cli( private def processors = security.cli.process .orElse(teamSearch.cli.process) - .orElse(forumSearch.cli.process) .orElse(tournament.cli.process) .orElse(fishnet.cli.process) .orElse(study.cli.process) diff --git a/modules/forumSearch/src/main/Env.scala b/modules/forumSearch/src/main/Env.scala index 89dc672311df7..badff6d3f567f 100644 --- a/modules/forumSearch/src/main/Env.scala +++ b/modules/forumSearch/src/main/Env.scala @@ -30,18 +30,6 @@ final class Env( def apply(text: String, page: Int, troll: Boolean) = paginatorBuilder(Query.forum(text.take(100), troll), page) - def cli: lila.common.Cli = new: - def process = { - case "forum" :: "search" :: "reset" :: Nil => api.reset.inject("done") - case "forum" :: "search" :: "backfill" :: epochSeconds :: Nil => - Either - .catchNonFatal(java.lang.Long.parseLong(epochSeconds)) - .fold( - e => fufail(s"Invalid epochSeconds: $e"), - since => api.backfill(java.time.Instant.ofEpochSecond(since)).inject("done") - ) - } - private lazy val paginatorBuilder = lila.search.PaginatorBuilder(api, config.maxPerPage) lila.common.Bus.sub[BusForum]: diff --git a/modules/forumSearch/src/main/ForumSearchApi.scala b/modules/forumSearch/src/main/ForumSearchApi.scala index 3a223b357ca45..3fea6a027cf2f 100644 --- a/modules/forumSearch/src/main/ForumSearchApi.scala +++ b/modules/forumSearch/src/main/ForumSearchApi.scala @@ -22,38 +22,3 @@ final class ForumSearchApi( def count(query: Query.Forum) = client.count(query).dmap(_.count) - - def store(post: ForumPostMini) = - postApi - .toMiniView(post) - .flatMapz: view => - client.storeForum(view.post.id.value, toDoc(view)) - - private def toDoc(view: ForumPostMiniView) = - ForumSource( - body = view.post.text.take(10000), - topic = view.topic.name, - author = view.post.userId.map(_.value), - topicId = view.topic.id.value, - troll = view.post.troll, - date = view.post.createdAt.toEpochMilli() - ) - - def reset = - client.mapping(index) >> - readAndIndexPosts(none) >> - client.refresh(index) - - def backfill(since: Instant) = - readAndIndexPosts(since.some) - - private def readAndIndexPosts(since: Option[Instant]) = - postApi - .nonGhostCursor(since) - .documentSource() - .via(lila.common.LilaStream.logRate("forum index")(logger)) - .grouped(200) - .mapAsync(1)(posts => postApi.toMiniViews(posts.toList)) - .map(_.map(v => v.post.id.value -> toDoc(v))) - .mapAsyncUnordered(2)(client.storeBulkForum) - .runWith(Sink.ignore) From 7ac6ae9b11c9a8973eeb2668f0bdd22d2743b348 Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Sat, 15 Jun 2024 20:32:30 +0700 Subject: [PATCH 060/168] Code tweak & scalafmt --- modules/forumSearch/src/main/Env.scala | 6 ++---- modules/forumSearch/src/main/ForumSearchApi.scala | 8 ++------ 2 files changed, 4 insertions(+), 10 deletions(-) diff --git a/modules/forumSearch/src/main/Env.scala b/modules/forumSearch/src/main/Env.scala index badff6d3f567f..6092317e9eff1 100644 --- a/modules/forumSearch/src/main/Env.scala +++ b/modules/forumSearch/src/main/Env.scala @@ -13,15 +13,13 @@ import lila.search.client.SearchClient import lila.search.spec.Query @Module -private class ForumSearchConfig( - @ConfigName("paginator.max_per_page") val maxPerPage: MaxPerPage -) +private class ForumSearchConfig(@ConfigName("paginator.max_per_page") val maxPerPage: MaxPerPage) final class Env( appConfig: Configuration, postApi: lila.core.forum.ForumPostApi, client: SearchClient -)(using Executor, akka.stream.Materializer): +)(using Executor): private val config = appConfig.get[ForumSearchConfig]("forumSearch")(AutoConfig.loader) diff --git a/modules/forumSearch/src/main/ForumSearchApi.scala b/modules/forumSearch/src/main/ForumSearchApi.scala index 3fea6a027cf2f..3b85b2094005d 100644 --- a/modules/forumSearch/src/main/ForumSearchApi.scala +++ b/modules/forumSearch/src/main/ForumSearchApi.scala @@ -8,17 +8,13 @@ import lila.core.id.ForumPostId import lila.search.client.SearchClient import lila.search.spec.{ ForumSource, Query } -final class ForumSearchApi( - client: SearchClient, - postApi: ForumPostApi -)(using Executor, akka.stream.Materializer) +final class ForumSearchApi(client: SearchClient, postApi: ForumPostApi)(using Executor) extends SearchReadApi[ForumPostId, Query.Forum]: def search(query: Query.Forum, from: From, size: Size) = client .search(query, from.value, size.value) - .map: res => - res.hitIds.map(ForumPostId.apply) + .map(res => res.hitIds.map(ForumPostId.apply)) def count(query: Query.Forum) = client.count(query).dmap(_.count) From a4f9d204e2a1c37b504d62dfb4a500030e9e902a Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 15:39:13 +0200 Subject: [PATCH 061/168] broadcast stats WIP What is done: - DB storage, recording and fetching stats of a tournament - Empty stats page linked on the tournament page To do: - use chart.js to draw the viewers graph --- app/controllers/RelayTour.scala | 7 ++++ conf/routes | 1 + modules/relay/src/main/Env.scala | 6 +++- modules/relay/src/main/RelayApi.scala | 5 --- modules/relay/src/main/RelayRoundRepo.scala | 4 +-- modules/relay/src/main/RelayStatsApi.scala | 38 +++++++++++++++++++++ modules/relay/src/main/ui/RelayTourUi.scala | 17 +++++++++ ui/analyse/src/study/relay/relayTourView.ts | 7 ++++ ui/bits/package.json | 1 + ui/bits/src/bits.relayStats.ts | 3 ++ 10 files changed, 81 insertions(+), 8 deletions(-) create mode 100644 modules/relay/src/main/RelayStatsApi.scala create mode 100644 ui/bits/src/bits.relayStats.ts diff --git a/app/controllers/RelayTour.scala b/app/controllers/RelayTour.scala index c3008f2ad4854..465ba63651d20 100644 --- a/app/controllers/RelayTour.scala +++ b/app/controllers/RelayTour.scala @@ -188,6 +188,13 @@ final class RelayTour(env: Env, apiC: => Api) extends LilaController(env): asAttachmentStream(s"${env.relay.pgnStream.filename(tour)}.pgn"): Ok.chunked(source).as(pgnContentType) + def stats(id: RelayTourId) = Open: + Found(env.relay.api.tourById(id)): tour => + env.relay.stats + .get(tour.id) + .flatMap: stats => + Ok.page(views.relay.tour.stats(tour, stats)) + def apiIndex = Anon: apiC.jsonDownload: env.relay.tourStream diff --git a/conf/routes b/conf/routes index ba6d629e8aaa5..8cc00de17428e 100644 --- a/conf/routes +++ b/conf/routes @@ -262,6 +262,7 @@ GET /broadcast/all-private controllers.RelayTour.allPrivate(p GET /broadcast/:ts/$id<\w{8}> controllers.RelayTour.show(ts, id: RelayTourId) GET /api/broadcast/$id<\w{8}> controllers.RelayTour.apiShow(id: RelayTourId) GET /api/broadcast/$tourId<\w{8}>.pgn controllers.RelayTour.pgn(tourId: RelayTourId) +GET /broadcast/$tourId<\w{8}>/stats controllers.RelayTour.stats(tourId: RelayTourId) GET /broadcast/$tourId<\w{8}>/edit controllers.RelayTour.edit(tourId: RelayTourId) POST /broadcast/$tourId<\w{8}>/edit controllers.RelayTour.update(tourId: RelayTourId) POST /broadcast/$tourId<\w{8}>/delete controllers.RelayTour.delete(tourId: RelayTourId) diff --git a/modules/relay/src/main/Env.scala b/modules/relay/src/main/Env.scala index 98ddc40f3070f..c5dbf759d6e33 100644 --- a/modules/relay/src/main/Env.scala +++ b/modules/relay/src/main/Env.scala @@ -90,6 +90,9 @@ final class Env( private lazy val delay = wire[RelayDelay] + // must instanciate eagerly to start the scheduler + val stats = wire[RelayStatsApi] + import SettingStore.CredentialsOption.given val proxyCredentials = settingStore[Option[Credentials]]( "relayProxyCredentials", @@ -122,7 +125,7 @@ final class Env( wire[RelayFetch] scheduler.scheduleWithFixedDelay(1 minute, 1 minute): () => - api.autoStart >> api.autoFinishNotSyncing >> api.monitorCrowd + api.autoStart >> api.autoFinishNotSyncing lila.common.Bus.subscribeFuns( "study" -> { case lila.core.study.RemoveStudy(studyId) => @@ -150,6 +153,7 @@ private class RelayColls(mainDb: lila.db.Db, yoloDb: lila.db.AsyncDb @@ lila.db. val tour = mainDb(CollName("relay_tour")) val group = mainDb(CollName("relay_group")) val delay = yoloDb(CollName("relay_delay")) + val stats = mainDb(CollName("relay_stats")) private trait ProxyCredentials private trait ProxyHostPort diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index f115665a50419..a2a36920842ea 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -408,11 +408,6 @@ final class RelayApi( update(relay)(_.finish) } - private[relay] def monitorCrowd: Funit = - roundRepo.tourCrowds.map: crowds => - crowds.foreach: (tourId, crowd) => - lila.mon.relay.tourCrowd(tourId).update(crowd) - private[relay] def WithRelay[A: Zero](id: RelayRoundId)(f: RelayRound => Fu[A]): Fu[A] = byId(id).flatMapz(f) diff --git a/modules/relay/src/main/RelayRoundRepo.scala b/modules/relay/src/main/RelayRoundRepo.scala index 1b8fab34c82f3..675529bf87a58 100644 --- a/modules/relay/src/main/RelayRoundRepo.scala +++ b/modules/relay/src/main/RelayRoundRepo.scala @@ -53,8 +53,8 @@ final private class RelayRoundRepo(val coll: Coll)(using Executor): coll .aggregateList(maxDocs = 500, _.sec): framework => import framework.* - Match($doc("sync.until" -> $exists(true))) -> - List(GroupField("_id")("crowd" -> SumField("crowd"))) + Match($doc("sync.until" -> $exists(true), "crowd".$gt(0))) -> + List(GroupField("tourId")("crowd" -> SumField("crowd"))) .map: docs => for doc <- docs diff --git a/modules/relay/src/main/RelayStatsApi.scala b/modules/relay/src/main/RelayStatsApi.scala new file mode 100644 index 0000000000000..ffc2979976060 --- /dev/null +++ b/modules/relay/src/main/RelayStatsApi.scala @@ -0,0 +1,38 @@ +package lila.relay + +import lila.db.dsl.{ *, given } + +object RelayStats: + type Minute = Int + type Crowd = Int + type Graph = List[(Minute, Crowd)] + +final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using scheduler: Scheduler)(using + Executor +): + import RelayStats.* + + // on measurement by minute at most; the storage depends on it. + scheduler.scheduleWithFixedDelay(0 seconds, 1 minute)(() => record()) + + def get(id: RelayTourId): Fu[Graph] = + colls.stats + .primitiveOne[List[Int]]($id(id), "d") + .mapz: + _.grouped(2) + .collect: + case List(minute, crowd) => (minute, crowd) + .toList + + private def record(): Funit = for + crowds <- roundRepo.tourCrowds + nowMinutes = nowSeconds / 60 + update = colls.stats.update(ordered = false) + elements <- crowds.sequentially: (tourId, crowd) => + update.element( + q = $id(tourId), + u = $push("d" -> $doc("$each" -> $arr(nowMinutes, crowd))), + upsert = true + ) + _ <- elements.nonEmpty.so(update.many(elements).void) + yield () diff --git a/modules/relay/src/main/ui/RelayTourUi.scala b/modules/relay/src/main/ui/RelayTourUi.scala index 78df22aa49466..5d10ced899eb6 100644 --- a/modules/relay/src/main/ui/RelayTourUi.scala +++ b/modules/relay/src/main/ui/RelayTourUi.scala @@ -6,6 +6,7 @@ import ScalatagsTemplate.{ *, given } import scalalib.paginator.Paginator import lila.relay.RelayTour.WithLastRound import lila.core.LightUser +import play.api.libs.json.Json final class RelayTourUi(helpers: Helpers, ui: RelayUi): import helpers.{ *, given } @@ -101,6 +102,22 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): ) ) + def stats(t: RelayTour, graph: RelayStats.Graph)(using Context) = + Page(s"${t.name.value} - Stats") + .css("bits.relay.index") + .js( + PageModule( + "bits.relayStats", + Json + .obj("points" -> graph.map: (minute, crowd) => + Json.arr(minute * 60, crowd)) + ) + ): + main(cls := "relay-tour page box box-pad")( + boxTop(h1(a(href := routes.RelayTour.show(t.slug, t.id).url)(t.name), " - Stats")), + "Here, a graph shows the number of viewers over time." + ) + def page(title: String, pageBody: Frag, active: String)(using Context): Page = Page(title) .css("bits.page") diff --git a/ui/analyse/src/study/relay/relayTourView.ts b/ui/analyse/src/study/relay/relayTourView.ts index 64af89bbd0294..edac7327fd2af 100644 --- a/ui/analyse/src/study/relay/relayTourView.ts +++ b/ui/analyse/src/study/relay/relayTourView.ts @@ -285,6 +285,13 @@ const makeTabs = (ctrl: AnalyseCtrl) => { makeTab('boards', 'Boards'), relay.teams && makeTab('teams', 'Teams'), relay.data.tour.leaderboard ? makeTab('leaderboard', 'Leaderboard') : undefined, + study.members.myMember() + ? h( + 'a.text', + { attrs: { ...dataIcon(licon.LineGraph), href: `/broadcast/${relay.data.tour.id}/stats` } }, + 'Popularity stats', + ) + : undefined, ]); }; diff --git a/ui/bits/package.json b/ui/bits/package.json index 4dfd6503ed721..cc1cbc6100f36 100644 --- a/ui/bits/package.json +++ b/ui/bits/package.json @@ -72,6 +72,7 @@ "src/bits.publicChats.ts", "src/bits.qrcode.ts", "src/bits.relayForm.ts", + "src/bits.relayStats.ts", "src/bits.soundMove.ts", "src/bits.streamer.ts", "src/bits.team.ts", diff --git a/ui/bits/src/bits.relayStats.ts b/ui/bits/src/bits.relayStats.ts new file mode 100644 index 0000000000000..04fd2c9080fcb --- /dev/null +++ b/ui/bits/src/bits.relayStats.ts @@ -0,0 +1,3 @@ +export function initModule(data: any) { + console.log(data); +} From b906c073476b439b7c282aeed48ec3d5adf8159b Mon Sep 17 00:00:00 2001 From: slither77 <132110444+slither77@users.noreply.github.com> Date: Sat, 15 Jun 2024 13:11:53 -0400 Subject: [PATCH 062/168] added monarch piece set made by slither77 --- public/piece/monarchy/bB.svg | 1 + public/piece/monarchy/bK.svg | 1 + public/piece/monarchy/bN.svg | 1 + public/piece/monarchy/bP.svg | 1 + public/piece/monarchy/bQ.svg | 1 + public/piece/monarchy/bR.svg | 1 + public/piece/monarchy/wB.svg | 1 + public/piece/monarchy/wK.svg | 1 + public/piece/monarchy/wN.svg | 1 + public/piece/monarchy/wP.svg | 1 + public/piece/monarchy/wQ.svg | 1 + public/piece/monarchy/wR.svg | 1 + 12 files changed, 12 insertions(+) create mode 100644 public/piece/monarchy/bB.svg create mode 100644 public/piece/monarchy/bK.svg create mode 100644 public/piece/monarchy/bN.svg create mode 100644 public/piece/monarchy/bP.svg create mode 100644 public/piece/monarchy/bQ.svg create mode 100644 public/piece/monarchy/bR.svg create mode 100644 public/piece/monarchy/wB.svg create mode 100644 public/piece/monarchy/wK.svg create mode 100644 public/piece/monarchy/wN.svg create mode 100644 public/piece/monarchy/wP.svg create mode 100644 public/piece/monarchy/wQ.svg create mode 100644 public/piece/monarchy/wR.svg diff --git a/public/piece/monarchy/bB.svg b/public/piece/monarchy/bB.svg new file mode 100644 index 0000000000000..a5ac5db2167c0 --- /dev/null +++ b/public/piece/monarchy/bB.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/bK.svg b/public/piece/monarchy/bK.svg new file mode 100644 index 0000000000000..3f4e68aadf871 --- /dev/null +++ b/public/piece/monarchy/bK.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/bN.svg b/public/piece/monarchy/bN.svg new file mode 100644 index 0000000000000..0213ea8382250 --- /dev/null +++ b/public/piece/monarchy/bN.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/bP.svg b/public/piece/monarchy/bP.svg new file mode 100644 index 0000000000000..854569e647b71 --- /dev/null +++ b/public/piece/monarchy/bP.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/bQ.svg b/public/piece/monarchy/bQ.svg new file mode 100644 index 0000000000000..df3811688a7f5 --- /dev/null +++ b/public/piece/monarchy/bQ.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/bR.svg b/public/piece/monarchy/bR.svg new file mode 100644 index 0000000000000..312278e8cfc45 --- /dev/null +++ b/public/piece/monarchy/bR.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/wB.svg b/public/piece/monarchy/wB.svg new file mode 100644 index 0000000000000..671a6f5c8ddaa --- /dev/null +++ b/public/piece/monarchy/wB.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/wK.svg b/public/piece/monarchy/wK.svg new file mode 100644 index 0000000000000..6ddeb192c48ae --- /dev/null +++ b/public/piece/monarchy/wK.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/wN.svg b/public/piece/monarchy/wN.svg new file mode 100644 index 0000000000000..7a728d5b4973a --- /dev/null +++ b/public/piece/monarchy/wN.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/wP.svg b/public/piece/monarchy/wP.svg new file mode 100644 index 0000000000000..c0cfccfa8f1b3 --- /dev/null +++ b/public/piece/monarchy/wP.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/wQ.svg b/public/piece/monarchy/wQ.svg new file mode 100644 index 0000000000000..e73504e4b8323 --- /dev/null +++ b/public/piece/monarchy/wQ.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/public/piece/monarchy/wR.svg b/public/piece/monarchy/wR.svg new file mode 100644 index 0000000000000..5e7a364ff6f56 --- /dev/null +++ b/public/piece/monarchy/wR.svg @@ -0,0 +1 @@ + \ No newline at end of file From 931e61f44f546d4a20b422733a05c26cca6cb4f7 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sat, 15 Jun 2024 22:28:42 +0200 Subject: [PATCH 063/168] integrate monarchy piece set --- COPYING.md | 1 + bin/gen/piece-sprite | 1 + modules/pref/src/main/PieceSet.scala | 1 + public/piece-css/monarchy.css | 12 ++++++++++++ public/piece-css/monarchy.external.css | 12 ++++++++++++ 5 files changed, 27 insertions(+) create mode 100644 public/piece-css/monarchy.css create mode 100644 public/piece-css/monarchy.external.css diff --git a/COPYING.md b/COPYING.md index 0362eaf249abe..6531d7f304a28 100644 --- a/COPYING.md +++ b/COPYING.md @@ -59,6 +59,7 @@ public/piece/disguised | danegraphics | [CC BY-NC-SA 4.0](https://creativecommon public/piece/kiwen-suwi | [neverRare](https://github.com/neverRare) | [CC BY 4.0](https://creativecommons.org/licenses/by/4.0/) public/piece/mpchess | [Maxime Chupin](https://github.com/chupinmaxime) | [GPL3v3+](https://www.gnu.org/licenses/quick-guide-gplv3.en.html) public/piece/cooke | [fejfar](https://github.com/fejfar) | [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) +public/piece/monarchy | [slither77](https://github.com/slither77) | [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) public/sounds/futuristic | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ public/sounds/nes | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ public/sounds/piano | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ diff --git a/bin/gen/piece-sprite b/bin/gen/piece-sprite index afcb9537bb27c..cf2f374d4c976 100755 --- a/bin/gen/piece-sprite +++ b/bin/gen/piece-sprite @@ -44,6 +44,7 @@ themes = [ ['kiwen-suwi', 'svg'], ['mpchess','svg'], ['cooke','svg'], + ['monarchy', 'svg'], ] types = { 'svg' => 'svg+xml;base64,', diff --git a/modules/pref/src/main/PieceSet.scala b/modules/pref/src/main/PieceSet.scala index 55178a1e70e5f..38874ecf48c98 100644 --- a/modules/pref/src/main/PieceSet.scala +++ b/modules/pref/src/main/PieceSet.scala @@ -47,6 +47,7 @@ object PieceSet extends PieceSetObject: PieceSet("tatiana"), PieceSet("staunty"), PieceSet("cooke"), + PieceSet("monarchy"), PieceSet("governor"), PieceSet("dubrovny"), PieceSet("icpieces"), diff --git a/public/piece-css/monarchy.css b/public/piece-css/monarchy.css new file mode 100644 index 0000000000000..87558a658876f --- /dev/null +++ b/public/piece-css/monarchy.css @@ -0,0 +1,12 @@ +.is2d .pawn.white {background-image:url('')} +.is2d .knight.white {background-image:url('')} +.is2d .bishop.white {background-image:url('')} +.is2d .rook.white {background-image:url('')} +.is2d .queen.white {background-image:url('')} +.is2d .king.white {background-image:url('')} +.is2d .pawn.black {background-image:url('')} +.is2d .knight.black {background-image:url('')} +.is2d .bishop.black {background-image:url('')} +.is2d .rook.black {background-image:url('')} +.is2d .queen.black {background-image:url('')} +.is2d .king.black {background-image:url('')} diff --git a/public/piece-css/monarchy.external.css b/public/piece-css/monarchy.external.css new file mode 100644 index 0000000000000..4f8779b0e84fe --- /dev/null +++ b/public/piece-css/monarchy.external.css @@ -0,0 +1,12 @@ +.is2d .pawn.white {background-image:url('/assets/piece/monarchy/wP.svg')} +.is2d .knight.white {background-image:url('/assets/piece/monarchy/wN.svg')} +.is2d .bishop.white {background-image:url('/assets/piece/monarchy/wB.svg')} +.is2d .rook.white {background-image:url('/assets/piece/monarchy/wR.svg')} +.is2d .queen.white {background-image:url('/assets/piece/monarchy/wQ.svg')} +.is2d .king.white {background-image:url('/assets/piece/monarchy/wK.svg')} +.is2d .pawn.black {background-image:url('/assets/piece/monarchy/bP.svg')} +.is2d .knight.black {background-image:url('/assets/piece/monarchy/bN.svg')} +.is2d .bishop.black {background-image:url('/assets/piece/monarchy/bB.svg')} +.is2d .rook.black {background-image:url('/assets/piece/monarchy/bR.svg')} +.is2d .queen.black {background-image:url('/assets/piece/monarchy/bQ.svg')} +.is2d .king.black {background-image:url('/assets/piece/monarchy/bK.svg')} From 0b44d019240e08fd56b4773828cfc781207bbdc8 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 00:48:51 +0200 Subject: [PATCH 064/168] broadcast card css tweaks --- ui/bits/css/relay/_card.scss | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ui/bits/css/relay/_card.scss b/ui/bits/css/relay/_card.scss index a461e6cdcce13..ad1b0d43eca4b 100644 --- a/ui/bits/css/relay/_card.scss +++ b/ui/bits/css/relay/_card.scss @@ -57,7 +57,7 @@ &__body { @extend %flex-column; - padding: 0.3em 0.5em 1em 1em; + padding: 0.6em 0.5em 0.6em 1em; } &__info { @@ -78,10 +78,10 @@ .relay-cards--tier-best & { font-size: 1.4em; } + padding: 0.2em 0 0.3em 0; } &__desc { @extend %roboto; color: $c-font-dim; - font-size: 0.9em; } } From ed021f2194783741bf49f287b5e3a06fdaf0029f Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 01:10:56 +0200 Subject: [PATCH 065/168] remove confusing abstraction in relay model --- modules/relay/src/main/RelayTour.scala | 3 +-- modules/relay/src/main/ui/RelayTourUi.scala | 2 +- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/modules/relay/src/main/RelayTour.scala b/modules/relay/src/main/RelayTour.scala index 1f3b1ccd1cc25..21e72c6fba25f 100644 --- a/modules/relay/src/main/RelayTour.scala +++ b/modules/relay/src/main/RelayTour.scala @@ -84,8 +84,7 @@ object RelayTour: display: RelayRound, // which round to show on the tour link link: RelayRound, // which round to actually link to group: Option[RelayGroup.Name] - ) extends RelayRound.AndTourAndGroup: - export display.{ hasStarted as ongoing } + ) extends RelayRound.AndTourAndGroup case class WithLastRound(tour: RelayTour, round: RelayRound, group: Option[RelayGroup.Name]) extends RelayRound.AndTourAndGroup: diff --git a/modules/relay/src/main/ui/RelayTourUi.scala b/modules/relay/src/main/ui/RelayTourUi.scala index 78df22aa49466..ecde8f219cc3e 100644 --- a/modules/relay/src/main/ui/RelayTourUi.scala +++ b/modules/relay/src/main/ui/RelayTourUi.scala @@ -22,7 +22,7 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): val selected = active.filter(_.tour.tierIs(selector)) selected.nonEmpty.option(st.section(cls := s"relay-cards relay-cards--tier-$tier"): selected.map: - card.render(_, ongoing = _.ongoing) + card.render(_, ongoing = _.display.hasStarted) ) Page(trc.liveBroadcasts.txt()) .css("bits.relay.index") From c07aa1e39644ff1652ef7c974a7ebd8594608740 Mon Sep 17 00:00:00 2001 From: Aaron Leslie Date: Sat, 15 Jun 2024 21:49:23 +0000 Subject: [PATCH 066/168] Allow voice and keyboard input of 'next' to start new puzzle streak game --- ui/puzzle/src/ctrl.ts | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) mode change 100644 => 100755 ui/puzzle/src/ctrl.ts diff --git a/ui/puzzle/src/ctrl.ts b/ui/puzzle/src/ctrl.ts old mode 100644 new mode 100755 index 8e97d8d1ac15c..f80f878802edc --- a/ui/puzzle/src/ctrl.ts +++ b/ui/puzzle/src/ctrl.ts @@ -428,7 +428,10 @@ export default class PuzzleCtrl implements ParentCtrl { private isPuzzleData = (d: PuzzleData | ReplayEnd): d is PuzzleData => 'puzzle' in d; nextPuzzle = (): void => { - if (this.streak && this.lastFeedback != 'win') return; + if (this.streak && this.lastFeedback != 'win') { + if (this.lastFeedback == 'fail') window.location.replace('/streak'); + return; + } if (this.mode !== 'view') return; this.ceval.stop(); From f442db7347f25e4cfecb95b8783cef6bd75cb820 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 09:33:06 +0200 Subject: [PATCH 067/168] show ongoing broadcasts in /broadcast WIP --- modules/relay/src/main/RelayApi.scala | 11 ++++++---- modules/relay/src/main/RelayListing.scala | 25 ++++++++++++++++------ modules/relay/src/main/RelayRound.scala | 2 ++ modules/relay/src/main/RelayTour.scala | 1 + modules/relay/src/main/RelayTourForm.scala | 1 + modules/relay/src/main/RelayTourRepo.scala | 4 ++-- 6 files changed, 31 insertions(+), 13 deletions(-) diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index f115665a50419..d95cdc03c88de 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -76,9 +76,12 @@ final class RelayApi( def withRounds(tour: RelayTour) = roundRepo.byTourOrdered(tour).dmap(tour.withRounds) def denormalizeTourActive(tourId: RelayTourId): Funit = - roundRepo.coll.exists(RelayRoundRepo.selectors.tour(tourId) ++ $doc("finished" -> false)).flatMap { - tourRepo.setActive(tourId, _) - } + val unfinished = RelayRoundRepo.selectors.tour(tourId) ++ $doc("finished" -> false) + for + active <- roundRepo.coll.exists(unfinished) + ongoing <- active.so(roundRepo.coll.exists(unfinished ++ $doc("startedAt".$exists(true)))) + _ <- tourRepo.setActive(tourId, active, ongoing) + yield () object countOwnedByUser: private val cache = cacheApi[UserId, Int](16_384, "relay.nb.owned"): @@ -251,7 +254,7 @@ final class RelayApi( _ <- roundRepo.coll.update.one($id(round.id), round).void _ <- (round.sync.playing != from.sync.playing) .so(sendToContributors(round.id, "relaySync", jsonView.sync(round))) - _ <- (round.finished != from.finished).so(denormalizeTourActive(round.tourId)) + _ <- (round.stateHash != from.stateHash).so(denormalizeTourActive(round.tourId)) yield round.sync.log.events.lastOption .ifTrue(round.sync.log != from.sync.log) diff --git a/modules/relay/src/main/RelayListing.scala b/modules/relay/src/main/RelayListing.scala index a2642aece2164..165e5b05e9dd4 100644 --- a/modules/relay/src/main/RelayListing.scala +++ b/modules/relay/src/main/RelayListing.scala @@ -43,7 +43,7 @@ final class RelayListing( ) -> List( Sort(Descending("tier")), PipelineOperator(group.lookup(colls.group)), - Match(group.filter), + // Match(group.filter), PipelineOperator(roundLookup), UnwindField("round"), Limit(max) @@ -149,18 +149,29 @@ private object RelayListing: def lookup(groupColl: Coll) = $lookup.pipelineFull( from = groupColl.name, as = "group", - let = $doc("id" -> "$_id"), + let = $doc("tourId" -> "$_id"), pipe = List( - $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("$$id", "$tours")))), + $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("$$tourId", "$tours")))), $doc: "$project" -> $doc( - "_id" -> false, - "name" -> true, - "first" -> $doc("$eq" -> $arr("$$id", $doc("$first" -> "$tours"))) + "_id" -> false, + "name" -> true, + "isFirst" -> $doc("$eq" -> $arr("$$tourId", $doc("$first" -> "$tours"))) ) ) ) - val filter = $doc("group.0.first".$ne(false)) + val filter = $doc("group.0.isFirst".$ne(false)) + + def groupOngoingTourLookup(tourColl: Coll) = $lookup.pipelineFull( + from = tourColl.name, + as = "ongoingTour", + let = $doc("tourIds" -> "$tours"), + pipe = List( + $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("_id", "$$tourIds")))), + $doc("$match" -> $doc("ongoing" -> true)), + $doc("$project" -> $doc("_id" -> false, "id" -> true)) + ) + ) def readFrom(doc: Bdoc): Option[RelayGroup.Name] = for garr <- doc.getAsOpt[Barr]("group") diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index 7b3f59ac7f79e..75505b7cb08e6 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -51,6 +51,8 @@ case class RelayRound( case Some(at) => at.isBefore(nowInstant.minusHours(3)) case None => createdAt.isBefore(nowInstant.minusDays(1)) + def stateHash = (hasStarted, finished) + def withSync(f: RelayRound.Sync => RelayRound.Sync) = copy(sync = f(sync)) def withTour(tour: RelayTour) = RelayRound.WithTour(this, tour) diff --git a/modules/relay/src/main/RelayTour.scala b/modules/relay/src/main/RelayTour.scala index 21e72c6fba25f..249357726b7c8 100644 --- a/modules/relay/src/main/RelayTour.scala +++ b/modules/relay/src/main/RelayTour.scala @@ -15,6 +15,7 @@ case class RelayTour( createdAt: Instant, tier: Option[RelayTour.Tier], // if present, it's an official broadcast active: Boolean, // a round is scheduled or ongoing + ongoing: Boolean, // a round is ongoing syncedAt: Option[Instant], // last time a round was synced spotlight: Option[RelayTour.Spotlight] = None, autoLeaderboard: Boolean = true, diff --git a/modules/relay/src/main/RelayTourForm.scala b/modules/relay/src/main/RelayTourForm.scala index a7c2e04b9afab..63418a328e515 100644 --- a/modules/relay/src/main/RelayTourForm.scala +++ b/modules/relay/src/main/RelayTourForm.scala @@ -82,6 +82,7 @@ object RelayTourForm: ownerId = me, tier = tier.ifTrue(Granter(_.Relay)), active = false, + ongoing = false, createdAt = nowInstant, syncedAt = none, autoLeaderboard = autoLeaderboard, diff --git a/modules/relay/src/main/RelayTourRepo.scala b/modules/relay/src/main/RelayTourRepo.scala index d3a0fd8af7202..abd5be9668fea 100644 --- a/modules/relay/src/main/RelayTourRepo.scala +++ b/modules/relay/src/main/RelayTourRepo.scala @@ -11,8 +11,8 @@ final private class RelayTourRepo(val coll: Coll)(using Executor): def setSyncedNow(tour: RelayTour): Funit = coll.updateField($id(tour.id), "syncedAt", nowInstant).void - def setActive(tourId: RelayTourId, active: Boolean): Funit = - coll.updateField($id(tourId), "active", active).void + def setActive(tourId: RelayTourId, active: Boolean, ongoing: Boolean): Funit = + coll.update.one($id(tourId), $set("active" -> active, "ongoing" -> ongoing)).void def lookup(local: String) = $lookup.simple(coll, "tour", local, "_id") From 1beb12ecbd29f2b9e5e271d3d79988870132125e Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 09:54:34 +0200 Subject: [PATCH 068/168] remove transitive dependencies --- build.sbt | 34 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/build.sbt b/build.sbt index 0249ad11ed20a..bb26114807e4c 100644 --- a/build.sbt +++ b/build.sbt @@ -136,12 +136,12 @@ lazy val rating = module("rating", ).dependsOn(common % "test->test") lazy val cms = module("cms", - Seq(memo, coreI18n, ui), + Seq(memo, ui), Seq() ) lazy val puzzle = module("puzzle", - Seq(coreI18n, tree, memo, rating, ui), + Seq(tree, memo, rating), tests.bundle ) @@ -161,7 +161,7 @@ lazy val video = module("video", ) lazy val coach = module("coach", - Seq(memo, rating, ui), + Seq(memo, rating), Seq() ) @@ -181,7 +181,7 @@ lazy val feed = module("feed", ) lazy val ublog = module("ublog", - Seq(coreI18n, memo, ui), + Seq(memo, ui), Seq(bloomFilter) ) @@ -191,7 +191,7 @@ lazy val evaluation = module("evaluation", ) lazy val perfStat = module("perfStat", - Seq(ui, memo, rating), + Seq(memo, rating), Seq() ) @@ -221,7 +221,7 @@ lazy val timeline = module("timeline", ) lazy val event = module("event", - Seq(memo, coreI18n, ui), + Seq(memo, ui), Seq() ) @@ -241,7 +241,7 @@ lazy val game = module("game", ) lazy val gameSearch = module("gameSearch", - Seq(coreI18n, search, ui), + Seq(search, ui), tests.bundle ) @@ -257,7 +257,7 @@ lazy val bot = module("bot", ) lazy val analyse = module("analyse", - Seq(coreI18n, tree, memo, ui), + Seq(tree, memo, ui), tests.bundle ) @@ -267,7 +267,7 @@ lazy val round = module("round", ) lazy val pool = module("pool", - Seq(coreI18n, db, rating), + Seq(db, rating), Seq() ) @@ -282,7 +282,7 @@ lazy val lobby = module("lobby", ) lazy val setup = module("setup", - Seq(lobby, ui), + Seq(lobby), Seq() ) @@ -292,12 +292,12 @@ lazy val insight = module("insight", ) lazy val tutor = module("tutor", - Seq(insight, ui), + Seq(insight), tests.bundle ) lazy val opening = module("opening", - Seq(coreI18n, memo, ui), + Seq(memo, ui), tests.bundle ) @@ -307,7 +307,7 @@ lazy val gathering = module("gathering", ) lazy val tournament = module("tournament", - Seq(gathering, room, memo, ui), + Seq(gathering, room, memo), Seq(lettuce) ++ tests.bundle ) @@ -332,7 +332,7 @@ lazy val irwin = module("irwin", ) lazy val oauth = module("oauth", - Seq(memo, coreI18n, ui), + Seq(memo, ui), Seq() ) @@ -362,7 +362,7 @@ lazy val title = module("title", ) lazy val study = module("study", - Seq(coreI18n, tree, memo, room, ui), + Seq(tree, memo, room, ui), Seq(lettuce) ++ tests.bundle ++ Seq(scalacheck, munitCheck, chess.testKit) ).dependsOn(common % "test->test") @@ -412,7 +412,7 @@ lazy val mailer = module("mailer", ) lazy val plan = module("plan", - Seq(coreI18n, memo, ui), + Seq(memo, ui), tests.bundle ) @@ -422,7 +422,7 @@ lazy val relation = module("relation", ) lazy val pref = module("pref", - Seq(coreI18n, memo, ui), + Seq(memo, ui), Seq() ) From 23c88e3b104f0191de755cb3e7cf71eb3c35813e Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 10:00:37 +0200 Subject: [PATCH 069/168] more search .ignore --- .ignore | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.ignore b/.ignore index 09f3c819eec4d..bda35ecf32bd5 100644 --- a/.ignore +++ b/.ignore @@ -6,3 +6,6 @@ public/sound public/piece public/piece-css public/flair +public/images +public/cursors +*.png From 6c20d4977672b7c5fbc269af72c9563b7c8de600 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 10:06:31 +0200 Subject: [PATCH 070/168] better redirect to new streak --- ui/puzzle/src/ctrl.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/puzzle/src/ctrl.ts b/ui/puzzle/src/ctrl.ts index f80f878802edc..7b5ccfc82f2ed 100755 --- a/ui/puzzle/src/ctrl.ts +++ b/ui/puzzle/src/ctrl.ts @@ -429,7 +429,7 @@ export default class PuzzleCtrl implements ParentCtrl { nextPuzzle = (): void => { if (this.streak && this.lastFeedback != 'win') { - if (this.lastFeedback == 'fail') window.location.replace('/streak'); + if (this.lastFeedback == 'fail') site.redirect(router.withLang('/streak')); return; } if (this.mode !== 'view') return; From 4bb02c8215a363787406df1a9dfa236e389c65eb Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 10:50:59 +0200 Subject: [PATCH 071/168] user profile mod zone button and actions dropdown tweaks --- app/views/user/show/header.scala | 111 ++++++++++++++++--------------- ui/bits/css/user/_show.scss | 35 +++++----- 2 files changed, 77 insertions(+), 69 deletions(-) diff --git a/app/views/user/show/header.scala b/app/views/user/show/header.scala index fdf75ef45443d..f8ae017695850 100644 --- a/app/views/user/show/header.scala +++ b/app/views/user/show/header.scala @@ -89,63 +89,66 @@ object header: (ctx.isAuth && ctx.isnt(u)) .option(a(cls := "nm-item note-zone-toggle")(splitNumber(s"${social.notes.size} Notes"))) ), - div(cls := "user-actions dropdown")( - a(cls := "text", dataIcon := Icon.Hamburger), - div(cls := "dropdown-window")( - ctx - .is(u) - .option( - frag( - a( - cls := "text", - href := routes.Account.profile, - dataIcon := Icon.Gear - )(trans.site.editProfile.txt()), - a( - cls := "text", - href := routes.Relation.blocks(), - dataIcon := Icon.NotAllowed - )(trans.site.listBlockedPlayers.txt()) - ) - ), - isGranted(_.UserModView).option( - a( - cls := "text mod-zone-toggle", - href := routes.User.mod(u.username), - dataIcon := Icon.Agent - )("Mod zone (Hotkey: m)") - ), - a( - cls := "text", - href := routes.User.tv(u.username), - dataIcon := Icon.AnalogTv - )(trans.site.watchGames.txt()), - ctx - .isnt(u) - .option( - views.relation.actions( - u.light, - relation = social.relation, - followable = social.followable, - blocked = social.blocked - ) - ), - a( - cls := "text", - href := s"${routes.UserAnalysis.index}#explorer/${u.username}", - dataIcon := Icon.Book - )(trans.site.openingExplorer.txt()), + div(cls := "user-actions")( + isGranted(_.UserModView).option( a( - cls := "text", - href := routes.User.download(u.username), - dataIcon := Icon.Download - )(trans.site.exportGames.txt()), - (ctx.isAuth && ctx.kid.no && ctx.isnt(u)).option( + cls := "mod-zone-toggle", + href := routes.User.mod(u.username), + dataIcon := Icon.Agent, + title := "Mod zone (Hotkey: m)" + ) + ), + div(cls := "dropdown")( + a(dataIcon := Icon.Hamburger), + div(cls := "dropdown-window")( + ctx + .is(u) + .option( + frag( + a( + cls := "text", + href := routes.Account.profile, + dataIcon := Icon.Gear + )(trans.site.editProfile.txt()), + a( + cls := "text", + href := routes.Relation.blocks(), + dataIcon := Icon.NotAllowed + )(trans.site.listBlockedPlayers.txt()) + ) + ), a( cls := "text", - href := s"${routes.Report.form}?username=${u.username}", - dataIcon := Icon.CautionTriangle - )(trans.site.reportXToModerators.txt(u.username)) + href := routes.User.tv(u.username), + dataIcon := Icon.AnalogTv + )(trans.site.watchGames.txt()), + ctx + .isnt(u) + .option( + views.relation.actions( + u.light, + relation = social.relation, + followable = social.followable, + blocked = social.blocked + ) + ), + a( + cls := "text", + href := s"${routes.UserAnalysis.index}#explorer/${u.username}", + dataIcon := Icon.Book + )(trans.site.openingExplorer.txt()), + a( + cls := "text", + href := routes.User.download(u.username), + dataIcon := Icon.Download + )(trans.site.exportGames.txt()), + (ctx.isAuth && ctx.kid.no && ctx.isnt(u)).option( + a( + cls := "text", + href := s"${routes.Report.form}?username=${u.username}", + dataIcon := Icon.CautionTriangle + )(trans.site.reportXToModerators.txt(u.username)) + ) ) ) ) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index b6ed004f1b19e..6fe8b52163c85 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -33,23 +33,25 @@ } .user-actions { + @extend %flex-center-nowrap; flex: 0 0 auto; margin: 1em 1em 1em 0.3em; - form { - display: inline; + .mod-zone-toggle, + .dropdown > a { + display: block; + font-size: 2em; + height: 1.5em; + color: $c-font-dim; + padding: 0 0.7em; + } + .mod-zone-toggle:hover { + color: $c-font-clear; } } .dropdown { position: relative; - font-size: 2em; - > a { - display: block; - height: 1.5em; - color: $c-font-page; - padding-left: 0.5em; - } .dropdown-window { @extend %dropdown-shadow; @@ -58,14 +60,17 @@ background: $c-bg-header-dropdown; border-radius: 3px 0 3px 3px; position: absolute; - top: 1.5em; + top: 3rem; right: 0; a { - width: 16em; - font-size: 1rem; + width: 20em; display: block; - padding: 0.6rem 1rem; + padding: 0.7rem 1rem; color: $c-header-dropdown; + &::before { + margin-inline: 0 1rem; + font-size: 1.4em; + } &:first-child { border-radius: 3px 0 0 0; } @@ -83,9 +88,9 @@ } &:hover { > a { - border-radius: 3px 3px 0 0; + @extend %box-radius-top, %dropdown-shadow; background: $c-bg-header-dropdown; - box-shadow: 2px 5px 6px rgba(0, 0, 0, 0.3); + color: $c-font-clear; } .dropdown-window { visibility: visible; From f6f784824a60947ab8cc032af9ea1295ae22cf6d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 11:03:40 +0200 Subject: [PATCH 072/168] hide broken user actions in blocked list - closes #15529 --- modules/relation/src/main/ui/RelationUi.scala | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/modules/relation/src/main/ui/RelationUi.scala b/modules/relation/src/main/ui/RelationUi.scala index 08517bf5db7bf..8a3a39a07fbec 100644 --- a/modules/relation/src/main/ui/RelationUi.scala +++ b/modules/relation/src/main/ui/RelationUi.scala @@ -126,7 +126,7 @@ final class RelationUi(helpers: Helpers): trans.site.friends() ) ), - pagTable(pag, routes.Relation.following(u.username)) + pagTable(pag, routes.Relation.following(u.username), withActions = true) ) def blocks(u: User, pag: Paginator[Related[UserWithPerfs]])(using Context) = @@ -136,7 +136,7 @@ final class RelationUi(helpers: Helpers): h1(userLink(u, withOnline = false)), div(cls := "actions")(trans.site.blocks.pluralSame(pag.nbResults)) ), - pagTable(pag, routes.Relation.blocks()) + pagTable(pag, routes.Relation.blocks(), withActions = false) ) def opponents(u: User, sugs: List[Related[UserWithPerfs]])(using ctx: Context) = @@ -177,7 +177,9 @@ final class RelationUi(helpers: Helpers): .wrap: body => main(cls := "box page-small")(body) - private def pagTable(pager: Paginator[Related[UserWithPerfs]], call: Call)(using ctx: Context) = + private def pagTable(pager: Paginator[Related[UserWithPerfs]], call: Call, withActions: Boolean)(using + ctx: Context + ) = table(cls := "slist slist-pad")( if pager.nbResults > 0 then @@ -189,7 +191,8 @@ final class RelationUi(helpers: Helpers): td(trans.site.nbGames.plural(r.user.count.game, r.user.count.game.localize)), td(r.user.seenAt.map: seen => trans.site.lastSeenActive(momentFromNow(seen))), - td(actions(r.user.light, relation = r.relation, followable = r.followable, blocked = false)) + withActions.option: + td(actions(r.user.light, relation = r.relation, followable = r.followable, blocked = false)) ), pagerNextTable(pager, np => addQueryParam(call.url, "page", np.toString)) ) From e64b23b205e5a19b7c08aacf2a65ea196e76aaa5 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 11:41:47 +0200 Subject: [PATCH 073/168] set initial tour.ongoing --- modules/relay/src/main/RelayApi.scala | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index d95cdc03c88de..8b3735f5e4564 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -194,10 +194,9 @@ final class RelayApi( def create(data: RelayRoundForm.Data, tour: RelayTour)(using me: Me): Fu[RelayRound.WithTourAndStudy] = roundRepo .lastByTour(tour) - .flatMapz { last => + .flatMapz: last => studyRepo.byId(last.studyId) - } - .flatMap { lastStudy => + .flatMap: lastStudy => import lila.study.{ StudyMember, StudyMembers } val relay = data.make(me, tour) for @@ -226,10 +225,9 @@ final class RelayApi( ) .orFail(s"Can't create study for relay $relay") _ <- roundRepo.coll.insert.one(relay) - _ <- tourRepo.setActive(tour.id, true) + _ <- tourRepo.setActive(tour.id, true, relay.hasStarted) _ <- studyApi.addTopics(relay.studyId, List(StudyTopic.broadcast.value)) yield relay.withTour(tour).withStudy(study.study) - } def requestPlay(id: RelayRoundId, v: Boolean): Funit = WithRelay(id): relay => From cc93e452796bf76de63a99e617e65b9e60e338f0 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 11:52:23 +0200 Subject: [PATCH 074/168] fix relay_tour compat --- modules/relay/src/main/RelayTour.scala | 2 +- modules/relay/src/main/RelayTourForm.scala | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/relay/src/main/RelayTour.scala b/modules/relay/src/main/RelayTour.scala index 249357726b7c8..aa544c263cdb0 100644 --- a/modules/relay/src/main/RelayTour.scala +++ b/modules/relay/src/main/RelayTour.scala @@ -15,7 +15,7 @@ case class RelayTour( createdAt: Instant, tier: Option[RelayTour.Tier], // if present, it's an official broadcast active: Boolean, // a round is scheduled or ongoing - ongoing: Boolean, // a round is ongoing + ongoing: Option[Boolean], // a round is ongoing syncedAt: Option[Instant], // last time a round was synced spotlight: Option[RelayTour.Spotlight] = None, autoLeaderboard: Boolean = true, diff --git a/modules/relay/src/main/RelayTourForm.scala b/modules/relay/src/main/RelayTourForm.scala index 63418a328e515..2bd953d351c43 100644 --- a/modules/relay/src/main/RelayTourForm.scala +++ b/modules/relay/src/main/RelayTourForm.scala @@ -82,7 +82,7 @@ object RelayTourForm: ownerId = me, tier = tier.ifTrue(Granter(_.Relay)), active = false, - ongoing = false, + ongoing = none, createdAt = nowInstant, syncedAt = none, autoLeaderboard = autoLeaderboard, From 8d63e1cd70b8c15ff9741ea01a347d69504bad36 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 11:56:32 +0200 Subject: [PATCH 075/168] rename relay tour "live" --- modules/relay/src/main/RelayApi.scala | 6 +++--- modules/relay/src/main/RelayListing.scala | 8 ++++---- modules/relay/src/main/RelayTour.scala | 2 +- modules/relay/src/main/RelayTourForm.scala | 2 +- modules/relay/src/main/RelayTourRepo.scala | 4 ++-- modules/relay/src/main/ui/RelayTourUi.scala | 20 ++++++++++---------- ui/bits/css/relay/_card.scss | 4 ++-- 7 files changed, 23 insertions(+), 23 deletions(-) diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index 8b3735f5e4564..739a7ef4d14d5 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -78,9 +78,9 @@ final class RelayApi( def denormalizeTourActive(tourId: RelayTourId): Funit = val unfinished = RelayRoundRepo.selectors.tour(tourId) ++ $doc("finished" -> false) for - active <- roundRepo.coll.exists(unfinished) - ongoing <- active.so(roundRepo.coll.exists(unfinished ++ $doc("startedAt".$exists(true)))) - _ <- tourRepo.setActive(tourId, active, ongoing) + active <- roundRepo.coll.exists(unfinished) + live <- active.so(roundRepo.coll.exists(unfinished ++ $doc("startedAt".$exists(true)))) + _ <- tourRepo.setActive(tourId, active, live) yield () object countOwnedByUser: diff --git a/modules/relay/src/main/RelayListing.scala b/modules/relay/src/main/RelayListing.scala index 165e5b05e9dd4..cb9382c327953 100644 --- a/modules/relay/src/main/RelayListing.scala +++ b/modules/relay/src/main/RelayListing.scala @@ -56,7 +56,7 @@ final class RelayListing( yield (tour, round, group) sorted = tours.sortBy: (tour, round, _) => ( - !round.startedAt.isDefined, // ongoing tournaments first + !round.hasStarted, // ongoing tournaments first 0 - ~tour.tier, // then by tier 0 - ~round.crowd, // then by viewers round.startsAt.fold(Long.MaxValue)(_.toMillis) // then by next round date @@ -162,13 +162,13 @@ private object RelayListing: ) val filter = $doc("group.0.isFirst".$ne(false)) - def groupOngoingTourLookup(tourColl: Coll) = $lookup.pipelineFull( + def groupliveTourLookup(tourColl: Coll) = $lookup.pipelineFull( from = tourColl.name, - as = "ongoingTour", + as = "liveTour", let = $doc("tourIds" -> "$tours"), pipe = List( $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("_id", "$$tourIds")))), - $doc("$match" -> $doc("ongoing" -> true)), + $doc("$match" -> $doc("live" -> true)), $doc("$project" -> $doc("_id" -> false, "id" -> true)) ) ) diff --git a/modules/relay/src/main/RelayTour.scala b/modules/relay/src/main/RelayTour.scala index aa544c263cdb0..f0c106eed16e5 100644 --- a/modules/relay/src/main/RelayTour.scala +++ b/modules/relay/src/main/RelayTour.scala @@ -15,7 +15,7 @@ case class RelayTour( createdAt: Instant, tier: Option[RelayTour.Tier], // if present, it's an official broadcast active: Boolean, // a round is scheduled or ongoing - ongoing: Option[Boolean], // a round is ongoing + live: Option[Boolean], // a round is live, i.e. started and not finished syncedAt: Option[Instant], // last time a round was synced spotlight: Option[RelayTour.Spotlight] = None, autoLeaderboard: Boolean = true, diff --git a/modules/relay/src/main/RelayTourForm.scala b/modules/relay/src/main/RelayTourForm.scala index 2bd953d351c43..b5bbc133ed8b3 100644 --- a/modules/relay/src/main/RelayTourForm.scala +++ b/modules/relay/src/main/RelayTourForm.scala @@ -82,7 +82,7 @@ object RelayTourForm: ownerId = me, tier = tier.ifTrue(Granter(_.Relay)), active = false, - ongoing = none, + live = none, createdAt = nowInstant, syncedAt = none, autoLeaderboard = autoLeaderboard, diff --git a/modules/relay/src/main/RelayTourRepo.scala b/modules/relay/src/main/RelayTourRepo.scala index abd5be9668fea..fed24b208718d 100644 --- a/modules/relay/src/main/RelayTourRepo.scala +++ b/modules/relay/src/main/RelayTourRepo.scala @@ -11,8 +11,8 @@ final private class RelayTourRepo(val coll: Coll)(using Executor): def setSyncedNow(tour: RelayTour): Funit = coll.updateField($id(tour.id), "syncedAt", nowInstant).void - def setActive(tourId: RelayTourId, active: Boolean, ongoing: Boolean): Funit = - coll.update.one($id(tourId), $set("active" -> active, "ongoing" -> ongoing)).void + def setActive(tourId: RelayTourId, active: Boolean, live: Boolean): Funit = + coll.update.one($id(tourId), $set("active" -> active, "live" -> live)).void def lookup(local: String) = $lookup.simple(coll, "tour", local, "_id") diff --git a/modules/relay/src/main/ui/RelayTourUi.scala b/modules/relay/src/main/ui/RelayTourUi.scala index ecde8f219cc3e..d0dd6ab58a424 100644 --- a/modules/relay/src/main/ui/RelayTourUi.scala +++ b/modules/relay/src/main/ui/RelayTourUi.scala @@ -22,7 +22,7 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): val selected = active.filter(_.tour.tierIs(selector)) selected.nonEmpty.option(st.section(cls := s"relay-cards relay-cards--tier-$tier"): selected.map: - card.render(_, ongoing = _.display.hasStarted) + card.render(_, live = _.display.hasStarted) ) Page(trc.liveBroadcasts.txt()) .css("bits.relay.index") @@ -40,7 +40,7 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): h2(cls := "relay-index__section")("Upcoming broadcasts"), st.section(cls := "relay-cards relay-cards--upcoming"): upcoming.map: - card.render(_, ongoing = _ => false) + card.render(_, live = _ => false) ) ), h2(cls := "relay-index__section")("Past broadcasts"), @@ -146,24 +146,24 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): ) private object card: - private def link(t: RelayTour, url: String, ongoing: Boolean) = a( + private def link(t: RelayTour, url: String, live: Boolean) = a( href := url, cls := List( - "relay-card" -> true, - "relay-card--active" -> t.active, - "relay-card--ongoing" -> ongoing + "relay-card" -> true, + "relay-card--active" -> t.active, + "relay-card--live" -> live ) ) private def image(t: RelayTour) = t.image.fold(ui.thumbnail.fallback(cls := "relay-card__image")): id => img(cls := "relay-card__image", src := ui.thumbnail.url(id, _.Size.Small)) - def render[A <: RelayRound.AndTourAndGroup](tr: A, ongoing: A => Boolean)(using Context) = - link(tr.tour, tr.path, ongoing(tr))( + def render[A <: RelayRound.AndTourAndGroup](tr: A, live: A => Boolean)(using Context) = + link(tr.tour, tr.path, live(tr))( image(tr.tour), span(cls := "relay-card__body")( span(cls := "relay-card__info")( tr.tour.active.option(span(cls := "relay-card__round")(tr.display.name)), - if ongoing(tr) + if live(tr) then span(cls := "relay-card__live")( "LIVE", @@ -204,7 +204,7 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): def renderPager(pager: Paginator[RelayTour | WithLastRound])(next: Int => Call)(using Context): Tag = st.section(cls := "infinite-scroll relay-cards")( pager.currentPageResults.map: - case w: WithLastRound => card.render(w, ongoing = _ => false)(cls := "paginated") + case w: WithLastRound => card.render(w, live = _ => false)(cls := "paginated") case t: RelayTour => card.empty(t)(cls := "paginated") , pagerNext(pager, next(_).url) diff --git a/ui/bits/css/relay/_card.scss b/ui/bits/css/relay/_card.scss index ad1b0d43eca4b..7f046141c6538 100644 --- a/ui/bits/css/relay/_card.scss +++ b/ui/bits/css/relay/_card.scss @@ -29,7 +29,7 @@ 0 0 20px $c-link; } - &--ongoing { + &--live { outline: 3px solid $m-bad--alpha-50; &:hover { box-shadow: @@ -42,7 +42,7 @@ width: 100%; height: auto; opacity: 0.7; - .relay-card--ongoing & { + .relay-card--live & { opacity: 0.9; } @include transition(opacity); From 30d1b9ac080bd6bb5d111170787dc73f07c67db2 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 14:13:55 +0200 Subject: [PATCH 076/168] Revert "add monarch piece set" --- COPYING.md | 1 - bin/gen/piece-sprite | 1 - modules/pref/src/main/PieceSet.scala | 1 - public/piece-css/monarchy.css | 12 ------------ public/piece-css/monarchy.external.css | 12 ------------ public/piece/monarchy/bB.svg | 1 - public/piece/monarchy/bK.svg | 1 - public/piece/monarchy/bN.svg | 1 - public/piece/monarchy/bP.svg | 1 - public/piece/monarchy/bQ.svg | 1 - public/piece/monarchy/bR.svg | 1 - public/piece/monarchy/wB.svg | 1 - public/piece/monarchy/wK.svg | 1 - public/piece/monarchy/wN.svg | 1 - public/piece/monarchy/wP.svg | 1 - public/piece/monarchy/wQ.svg | 1 - public/piece/monarchy/wR.svg | 1 - 17 files changed, 39 deletions(-) delete mode 100644 public/piece-css/monarchy.css delete mode 100644 public/piece-css/monarchy.external.css delete mode 100644 public/piece/monarchy/bB.svg delete mode 100644 public/piece/monarchy/bK.svg delete mode 100644 public/piece/monarchy/bN.svg delete mode 100644 public/piece/monarchy/bP.svg delete mode 100644 public/piece/monarchy/bQ.svg delete mode 100644 public/piece/monarchy/bR.svg delete mode 100644 public/piece/monarchy/wB.svg delete mode 100644 public/piece/monarchy/wK.svg delete mode 100644 public/piece/monarchy/wN.svg delete mode 100644 public/piece/monarchy/wP.svg delete mode 100644 public/piece/monarchy/wQ.svg delete mode 100644 public/piece/monarchy/wR.svg diff --git a/COPYING.md b/COPYING.md index 6531d7f304a28..0362eaf249abe 100644 --- a/COPYING.md +++ b/COPYING.md @@ -59,7 +59,6 @@ public/piece/disguised | danegraphics | [CC BY-NC-SA 4.0](https://creativecommon public/piece/kiwen-suwi | [neverRare](https://github.com/neverRare) | [CC BY 4.0](https://creativecommons.org/licenses/by/4.0/) public/piece/mpchess | [Maxime Chupin](https://github.com/chupinmaxime) | [GPL3v3+](https://www.gnu.org/licenses/quick-guide-gplv3.en.html) public/piece/cooke | [fejfar](https://github.com/fejfar) | [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) -public/piece/monarchy | [slither77](https://github.com/slither77) | [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) public/sounds/futuristic | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ public/sounds/nes | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ public/sounds/piano | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ diff --git a/bin/gen/piece-sprite b/bin/gen/piece-sprite index cf2f374d4c976..afcb9537bb27c 100755 --- a/bin/gen/piece-sprite +++ b/bin/gen/piece-sprite @@ -44,7 +44,6 @@ themes = [ ['kiwen-suwi', 'svg'], ['mpchess','svg'], ['cooke','svg'], - ['monarchy', 'svg'], ] types = { 'svg' => 'svg+xml;base64,', diff --git a/modules/pref/src/main/PieceSet.scala b/modules/pref/src/main/PieceSet.scala index 38874ecf48c98..55178a1e70e5f 100644 --- a/modules/pref/src/main/PieceSet.scala +++ b/modules/pref/src/main/PieceSet.scala @@ -47,7 +47,6 @@ object PieceSet extends PieceSetObject: PieceSet("tatiana"), PieceSet("staunty"), PieceSet("cooke"), - PieceSet("monarchy"), PieceSet("governor"), PieceSet("dubrovny"), PieceSet("icpieces"), diff --git a/public/piece-css/monarchy.css b/public/piece-css/monarchy.css deleted file mode 100644 index 87558a658876f..0000000000000 --- a/public/piece-css/monarchy.css +++ /dev/null @@ -1,12 +0,0 @@ -.is2d .pawn.white {background-image:url('')} -.is2d .knight.white {background-image:url('')} -.is2d .bishop.white {background-image:url('')} -.is2d .rook.white {background-image:url('')} -.is2d .queen.white {background-image:url('')} -.is2d .king.white {background-image:url('')} -.is2d .pawn.black {background-image:url('')} -.is2d .knight.black {background-image:url('')} -.is2d .bishop.black {background-image:url('')} -.is2d .rook.black {background-image:url('')} -.is2d .queen.black {background-image:url('')} -.is2d .king.black {background-image:url('')} diff --git a/public/piece-css/monarchy.external.css b/public/piece-css/monarchy.external.css deleted file mode 100644 index 4f8779b0e84fe..0000000000000 --- a/public/piece-css/monarchy.external.css +++ /dev/null @@ -1,12 +0,0 @@ -.is2d .pawn.white {background-image:url('/assets/piece/monarchy/wP.svg')} -.is2d .knight.white {background-image:url('/assets/piece/monarchy/wN.svg')} -.is2d .bishop.white {background-image:url('/assets/piece/monarchy/wB.svg')} -.is2d .rook.white {background-image:url('/assets/piece/monarchy/wR.svg')} -.is2d .queen.white {background-image:url('/assets/piece/monarchy/wQ.svg')} -.is2d .king.white {background-image:url('/assets/piece/monarchy/wK.svg')} -.is2d .pawn.black {background-image:url('/assets/piece/monarchy/bP.svg')} -.is2d .knight.black {background-image:url('/assets/piece/monarchy/bN.svg')} -.is2d .bishop.black {background-image:url('/assets/piece/monarchy/bB.svg')} -.is2d .rook.black {background-image:url('/assets/piece/monarchy/bR.svg')} -.is2d .queen.black {background-image:url('/assets/piece/monarchy/bQ.svg')} -.is2d .king.black {background-image:url('/assets/piece/monarchy/bK.svg')} diff --git a/public/piece/monarchy/bB.svg b/public/piece/monarchy/bB.svg deleted file mode 100644 index a5ac5db2167c0..0000000000000 --- a/public/piece/monarchy/bB.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/bK.svg b/public/piece/monarchy/bK.svg deleted file mode 100644 index 3f4e68aadf871..0000000000000 --- a/public/piece/monarchy/bK.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/bN.svg b/public/piece/monarchy/bN.svg deleted file mode 100644 index 0213ea8382250..0000000000000 --- a/public/piece/monarchy/bN.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/bP.svg b/public/piece/monarchy/bP.svg deleted file mode 100644 index 854569e647b71..0000000000000 --- a/public/piece/monarchy/bP.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/bQ.svg b/public/piece/monarchy/bQ.svg deleted file mode 100644 index df3811688a7f5..0000000000000 --- a/public/piece/monarchy/bQ.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/bR.svg b/public/piece/monarchy/bR.svg deleted file mode 100644 index 312278e8cfc45..0000000000000 --- a/public/piece/monarchy/bR.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/wB.svg b/public/piece/monarchy/wB.svg deleted file mode 100644 index 671a6f5c8ddaa..0000000000000 --- a/public/piece/monarchy/wB.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/wK.svg b/public/piece/monarchy/wK.svg deleted file mode 100644 index 6ddeb192c48ae..0000000000000 --- a/public/piece/monarchy/wK.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/wN.svg b/public/piece/monarchy/wN.svg deleted file mode 100644 index 7a728d5b4973a..0000000000000 --- a/public/piece/monarchy/wN.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/wP.svg b/public/piece/monarchy/wP.svg deleted file mode 100644 index c0cfccfa8f1b3..0000000000000 --- a/public/piece/monarchy/wP.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/wQ.svg b/public/piece/monarchy/wQ.svg deleted file mode 100644 index e73504e4b8323..0000000000000 --- a/public/piece/monarchy/wQ.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/piece/monarchy/wR.svg b/public/piece/monarchy/wR.svg deleted file mode 100644 index 5e7a364ff6f56..0000000000000 --- a/public/piece/monarchy/wR.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file From 9e667fdeb60ed69d3d8e9768a4b0a014358e634a Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 15:53:51 +0200 Subject: [PATCH 077/168] ensure settings user ids are normalized --- modules/memo/src/main/SettingStore.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/memo/src/main/SettingStore.scala b/modules/memo/src/main/SettingStore.scala index ec7f80fabdb32..422bca2363fa2 100644 --- a/modules/memo/src/main/SettingStore.scala +++ b/modules/memo/src/main/SettingStore.scala @@ -79,7 +79,7 @@ object SettingStore: given StringReader[Strings] = StringReader.fromIso(using stringsIso) object UserIds: val userIdsIso = Strings.stringsIso.map[lila.core.data.UserIds]( - strs => lila.core.data.UserIds(UserId.from(strs.value)), + strs => lila.core.data.UserIds(UserStr.from(strs.value).map(_.id)), uids => lila.core.data.Strings(UserId.raw(uids.value)) ) given BSONHandler[UserIds] = lila.db.dsl.isoHandler(using userIdsIso) From 9cd7cff84b4a90453176b56b866f455a27819bd9 Mon Sep 17 00:00:00 2001 From: Trevor Fitzgerald Date: Sun, 16 Jun 2024 10:05:30 -0400 Subject: [PATCH 078/168] use const --- ui/bits/src/bits.importer.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/src/bits.importer.ts b/ui/bits/src/bits.importer.ts index e9ac93fbe8154..8261015017360 100644 --- a/ui/bits/src/bits.importer.ts +++ b/ui/bits/src/bits.importer.ts @@ -1,4 +1,4 @@ -var $form = $('main.importer form'); +const $form = $('main.importer form'); $form.on('submit', () => setTimeout(() => $form.html(site.spinnerHtml), 50)); From 7a75da2f0f7bd6382473369d5bac5966661031b9 Mon Sep 17 00:00:00 2001 From: Trevor Fitzgerald Date: Sun, 16 Jun 2024 10:06:14 -0400 Subject: [PATCH 079/168] concise username regex --- ui/common/src/richText.ts | 2 +- ui/common/tests/richText.test.ts | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/ui/common/src/richText.ts b/ui/common/src/richText.ts index d037c7a11141d..d0f63929f702c 100644 --- a/ui/common/src/richText.ts +++ b/ui/common/src/richText.ts @@ -6,7 +6,7 @@ import { VNode, Hooks } from 'snabbdom'; export const linkRegex = /(^|[\s\n]|<[A-Za-z]*\/?>)((?:(?:https?|ftp):\/\/|lichess\.org)[\-A-Z0-9+\u0026\u2019@#\/%?=()~_|!:,.;]*[\-A-Z0-9+\u0026@#\/%=~()_|])/gi; export const newLineRegex = /\n/g; -export const userPattern = /(^|[^\w@#/])@([a-zA-Z0-9_-]{2,30})/gi; +export const userPattern = /(^|[^\w@#/])@([a-z0-9_-]{2,30})/gi; // looks like it has a @mention or #gameid or a url.tld export const isMoreThanText = (str: string) => /(\n|(@|#|\.)\w{2,}|(board|game) \d)/i.test(str); diff --git a/ui/common/tests/richText.test.ts b/ui/common/tests/richText.test.ts index 1abc3a98c49a8..42ca3618d3706 100644 --- a/ui/common/tests/richText.test.ts +++ b/ui/common/tests/richText.test.ts @@ -1,4 +1,4 @@ -import { describe, expect, test, vi } from 'vitest'; +import { describe, expect, test } from 'vitest'; import { movePattern, userPattern } from '../src/richText'; describe('test regex patterns', () => { @@ -6,6 +6,7 @@ describe('test regex patterns', () => { expect('@foo'.match(userPattern)).toStrictEqual(['@foo']); expect('@foo-'.match(userPattern)).toStrictEqual(['@foo-']); expect('@__foo'.match(userPattern)).toStrictEqual(['@__foo']); + expect('@Foo'.match(userPattern)).toStrictEqual(['@Foo']); }); test.each([ From 846b2a2d1ebadd92d8d32971802760c7c3e9831b Mon Sep 17 00:00:00 2001 From: Trevor Fitzgerald Date: Sun, 16 Jun 2024 12:00:02 -0400 Subject: [PATCH 080/168] remove duplicate css property --- ui/bits/css/_titleRequest.scss | 1 - 1 file changed, 1 deletion(-) diff --git a/ui/bits/css/_titleRequest.scss b/ui/bits/css/_titleRequest.scss index a93532c8c0688..41acc120bf25f 100644 --- a/ui/bits/css/_titleRequest.scss +++ b/ui/bits/css/_titleRequest.scss @@ -3,7 +3,6 @@ gap: $block-gap; grid-template-columns: repeat(auto-fill, minmax(25em, 1fr)); margin-bottom: 2em; - gap: 2em; } .title-image-edit { From 2afdddcd847a227326498f2069c408efb95ffac7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Gl=C3=B3rias?= <9739913+SergioGlorias@users.noreply.github.com> Date: Sun, 16 Jun 2024 17:03:52 +0100 Subject: [PATCH 081/168] Reduce california pieces SVG (#15534) * Reduce california pieces SVG * FIX svg need width and height * fix base64 * better no touch in diffrent css --- public/piece-css/california.css | 24 ++++++++++++------------ public/piece/california/bB.svg | 2 +- public/piece/california/bK.svg | 2 +- public/piece/california/bN.svg | 2 +- public/piece/california/bP.svg | 2 +- public/piece/california/bQ.svg | 2 +- public/piece/california/bR.svg | 2 +- public/piece/california/wB.svg | 2 +- public/piece/california/wK.svg | 2 +- public/piece/california/wN.svg | 2 +- public/piece/california/wP.svg | 2 +- public/piece/california/wQ.svg | 2 +- public/piece/california/wR.svg | 2 +- 13 files changed, 24 insertions(+), 24 deletions(-) diff --git a/public/piece-css/california.css b/public/piece-css/california.css index 0cd63bd4ceb0f..598fb2297b1ea 100644 --- a/public/piece-css/california.css +++ b/public/piece-css/california.css @@ -1,12 +1,12 @@ -.is2d .pawn.white {background-image:url('')} -.is2d .knight.white {background-image:url('')} -.is2d .bishop.white {background-image:url('')} -.is2d .rook.white {background-image:url('')} -.is2d .queen.white {background-image:url('')} -.is2d .king.white {background-image:url('')} -.is2d .pawn.black {background-image:url('')} -.is2d .knight.black {background-image:url('')} -.is2d .bishop.black {background-image:url('')} -.is2d .rook.black {background-image:url('')} -.is2d .queen.black {background-image:url('')} -.is2d .king.black {background-image:url('')} +.is2d .pawn.white {background-image:url('')} +.is2d .knight.white {background-image:url('')} +.is2d .bishop.white {background-image:url('')} +.is2d .rook.white {background-image:url('')} +.is2d .queen.white {background-image:url('')} +.is2d .king.white {background-image:url('')} +.is2d .pawn.black {background-image:url('')} +.is2d .knight.black {background-image:url('')} +.is2d .bishop.black {background-image:url('')} +.is2d .rook.black {background-image:url('')} +.is2d .queen.black {background-image:url('')} +.is2d .king.black {background-image:url('')} diff --git a/public/piece/california/bB.svg b/public/piece/california/bB.svg index 3ccbe575cd9cf..e7289234afb04 100644 --- a/public/piece/california/bB.svg +++ b/public/piece/california/bB.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/bK.svg b/public/piece/california/bK.svg index 66f5719890fae..4db7f7b87ddc7 100644 --- a/public/piece/california/bK.svg +++ b/public/piece/california/bK.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/bN.svg b/public/piece/california/bN.svg index 585c006d45fb5..8bbe5005aeb2a 100644 --- a/public/piece/california/bN.svg +++ b/public/piece/california/bN.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/bP.svg b/public/piece/california/bP.svg index e688531ed613a..bb4ce3f9192b3 100644 --- a/public/piece/california/bP.svg +++ b/public/piece/california/bP.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/bQ.svg b/public/piece/california/bQ.svg index 23e395b8cbcd0..479e1f95f5e84 100644 --- a/public/piece/california/bQ.svg +++ b/public/piece/california/bQ.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/bR.svg b/public/piece/california/bR.svg index 2cd8f50dae8fa..aae79ba472194 100644 --- a/public/piece/california/bR.svg +++ b/public/piece/california/bR.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/wB.svg b/public/piece/california/wB.svg index f412208eaf232..03c71b7e39cc4 100644 --- a/public/piece/california/wB.svg +++ b/public/piece/california/wB.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/wK.svg b/public/piece/california/wK.svg index 842934894faa0..5478848b58eb1 100644 --- a/public/piece/california/wK.svg +++ b/public/piece/california/wK.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/wN.svg b/public/piece/california/wN.svg index dba04ffd1b9dd..45b2bda22934b 100644 --- a/public/piece/california/wN.svg +++ b/public/piece/california/wN.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/wP.svg b/public/piece/california/wP.svg index 0247d305b9fb8..bba8d0f49d55d 100644 --- a/public/piece/california/wP.svg +++ b/public/piece/california/wP.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/wQ.svg b/public/piece/california/wQ.svg index a6459435ee7a5..a857fc4106f53 100644 --- a/public/piece/california/wQ.svg +++ b/public/piece/california/wQ.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/public/piece/california/wR.svg b/public/piece/california/wR.svg index 2482b5088f0d6..4b2f598f08ba5 100644 --- a/public/piece/california/wR.svg +++ b/public/piece/california/wR.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file From c101feb7034c034dc64b807fd2abbfe40f28470a Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 18:21:14 +0200 Subject: [PATCH 082/168] insane mongodb aggregation magic to achieve the desired featuring where the first ongoing tournament of a group is selected --- modules/relay/src/main/RelayListing.scala | 114 ++++++++++++++++++---- modules/relay/src/main/RelayPager.scala | 4 +- 2 files changed, 95 insertions(+), 23 deletions(-) diff --git a/modules/relay/src/main/RelayListing.scala b/modules/relay/src/main/RelayListing.scala index cb9382c327953..5907619781f27 100644 --- a/modules/relay/src/main/RelayListing.scala +++ b/modules/relay/src/main/RelayListing.scala @@ -35,15 +35,92 @@ final class RelayListing( for upcoming <- upcoming.get({}) max = 100 + tourIds <- colls.tour.distinctEasy[RelayTourId, List]( + "_id", + RelayTourRepo.selectors.officialActive ++ $doc("_id".$nin(upcoming.map(_.tour.id))) + ) + groupToursDocs <- colls.group.aggregateList(Int.MaxValue): framework => + import framework.* + Match($doc("tours".$in(tourIds))) -> List( + PipelineOperator( + $lookup.pipelineFull( + from = colls.tour.name, + as = "tours", + let = $doc("tourIds" -> "$tours"), + pipe = List( + $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("$_id", "$$tourIds")))), + $doc("$addFields" -> $doc("__order" -> $doc("$indexOfArray" -> $arr("$$tourIds", "$_id")))), + $doc("$sort" -> $doc("tier" -> -1, "__order" -> 1)), + $doc("$project" -> $doc("live" -> true)) + ) + ) + ), + Project( + $doc( + "tours" -> $doc( + "$ifNull" -> $arr( + $doc( + "$first" -> $doc( + "$filter" -> $doc( + "input" -> "$tours", + "as" -> "tour", + "cond" -> "$$tour.live", + "limit" -> 1 + ) + ) + ), + $doc("$first" -> "$tours") + ) + ) + ) + ), + Project($doc("_id" -> true, "tour" -> "$tours._id")) + ) + groupTourPairs = for + doc <- groupToursDocs + groupId <- doc.getAsOpt[RelayGroup.Id]("_id") + tour <- doc.getAsOpt[RelayTourId]("tour") + yield s"$groupId$tour" docs <- colls.tour .aggregateList(max): framework => import framework.* - Match( - RelayTourRepo.selectors.officialActive ++ $doc("_id".$nin(upcoming.map(_.tour.id))) - ) -> List( + Match($inIds(tourIds)) -> List( Sort(Descending("tier")), - PipelineOperator(group.lookup(colls.group)), - // Match(group.filter), + Project( + $doc("subscribers" -> false, "notified" -> false, "teams" -> false, "players" -> false) + ), + PipelineOperator( + $lookup.pipelineFull( + from = colls.group.name, + as = "group", + let = $doc("tourId" -> "$_id"), + pipe = List( + $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("$$tourId", "$tours")))), + $doc( + "$project" -> $doc( + "_id" -> true, + "name" -> true + ) + ) + ) + ) + ), + AddFields($doc("group" -> $doc("$first" -> "$group"))), + AddFields( + $doc( + "isGroupTour" -> + $doc( + "$let" -> $doc( + "vars" -> $doc( + "allPairs" -> groupTourPairs, + "pair" -> $doc("$concat" -> $arr("$group._id", "$_id")) + ), + "in" -> $doc("$in" -> $arr("$$pair", "$$allPairs")) + ) + ) + ) + ), + Match($doc($or("group".$exists(false), "isGroupTour".$eq(true)))), PipelineOperator(roundLookup), UnwindField("round"), Limit(max) @@ -52,7 +129,7 @@ final class RelayListing( doc <- docs tour <- doc.asOpt[RelayTour] round <- doc.getAsOpt[RelayRound]("round") - group = RelayListing.group.readFrom(doc) + group = RelayListing.group.readFromOne(doc) yield (tour, round, group) sorted = tours.sortBy: (tour, round, _) => ( @@ -82,8 +159,8 @@ final class RelayListing( import framework.* Match(RelayTourRepo.selectors.officialActive) -> List( Sort(Descending("tier")), - PipelineOperator(group.lookup(colls.group)), - Match(group.filter), + PipelineOperator(group.firstLookup(colls.group)), + Match(group.firstFilter), PipelineOperator: $lookup.pipeline( from = colls.round, @@ -143,10 +220,11 @@ final class RelayListing( private object RelayListing: object group: + // look at the groups where the tour appears. // only keep the tour if there is no group, // or if the tour is the first in the group. - def lookup(groupColl: Coll) = $lookup.pipelineFull( + def firstLookup(groupColl: Coll) = $lookup.pipelineFull( from = groupColl.name, as = "group", let = $doc("tourId" -> "$_id"), @@ -160,21 +238,15 @@ private object RelayListing: ) ) ) - val filter = $doc("group.0.isFirst".$ne(false)) - - def groupliveTourLookup(tourColl: Coll) = $lookup.pipelineFull( - from = tourColl.name, - as = "liveTour", - let = $doc("tourIds" -> "$tours"), - pipe = List( - $doc("$match" -> $doc("$expr" -> $doc("$in" -> $arr("_id", "$$tourIds")))), - $doc("$match" -> $doc("live" -> true)), - $doc("$project" -> $doc("_id" -> false, "id" -> true)) - ) - ) + val firstFilter = $doc("group.0.isFirst".$ne(false)) def readFrom(doc: Bdoc): Option[RelayGroup.Name] = for garr <- doc.getAsOpt[Barr]("group") gdoc <- garr.getAsOpt[Bdoc](0) name <- gdoc.getAsOpt[RelayGroup.Name]("name") yield name + + def readFromOne(doc: Bdoc): Option[RelayGroup.Name] = for + gdoc <- doc.getAsOpt[Bdoc]("group") + name <- gdoc.getAsOpt[RelayGroup.Name]("name") + yield name diff --git a/modules/relay/src/main/RelayPager.scala b/modules/relay/src/main/RelayPager.scala index ba23a71bb593e..fc9b33dce4343 100644 --- a/modules/relay/src/main/RelayPager.scala +++ b/modules/relay/src/main/RelayPager.scala @@ -172,8 +172,8 @@ final class RelayPager( ) = onlyKeepGroupFirst.so( List( - framework.PipelineOperator(RelayListing.group.lookup(colls.group)), - framework.Match(RelayListing.group.filter) + framework.PipelineOperator(RelayListing.group.firstLookup(colls.group)), + framework.Match(RelayListing.group.firstFilter) ) ) ::: List( framework.PipelineOperator( From 7a76183305b829590e7056ae717a506aed1764da Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 18:41:30 +0200 Subject: [PATCH 083/168] broadcast: ignore source games with an unknown player --- modules/relay/src/main/RelayGame.scala | 6 ++++++ modules/relay/src/main/RelayInputSanity.scala | 6 +++++- 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayGame.scala b/modules/relay/src/main/RelayGame.scala index 081364dbe1206..180df1a9fd52f 100644 --- a/modules/relay/src/main/RelayGame.scala +++ b/modules/relay/src/main/RelayGame.scala @@ -38,6 +38,10 @@ case class RelayGame( def fideIdsPair: Option[PairOf[Option[chess.FideId]]] = tags.fideIds.some.filter(_.forall(_.isDefined)).map(_.toPair) + def hasUnknownPlayer: Boolean = + List(RelayGame.whiteTags, RelayGame.blackTags).exists: + _.forall(tag => tags(tag).isEmpty) + private object RelayGame: val lichessDomains = List("lichess.org", "lichess.dev") @@ -46,6 +50,8 @@ private object RelayGame: val eventTags: TagNames = List(_.Event, _.Site) val nameTags: TagNames = List(_.White, _.Black) val fideIdTags: TagNames = List(_.WhiteFideId, _.BlackFideId) + val whiteTags: TagNames = List(_.White, _.WhiteFideId) + val blackTags: TagNames = List(_.Black, _.BlackFideId) import scalalib.Iso import chess.format.pgn.{ InitialComments, Pgn } diff --git a/modules/relay/src/main/RelayInputSanity.scala b/modules/relay/src/main/RelayInputSanity.scala index df4a50f91333a..22f3f6c97c71e 100644 --- a/modules/relay/src/main/RelayInputSanity.scala +++ b/modules/relay/src/main/RelayInputSanity.scala @@ -8,7 +8,11 @@ import lila.study.* private object RelayInputSanity: def fixGames(games: RelayGames): RelayGames = - fixDgtKingsInTheCenter(games) + fixDgtKingsInTheCenter: + removeGamesWithUnknownPlayer(games) + + private def removeGamesWithUnknownPlayer(games: RelayGames): RelayGames = + games.filterNot(_.hasUnknownPlayer) // DGT puts the kings in the center on game end // and sends it as actual moves if the kings were close to the center From 0aa0c5a5dde827b6863bd5c7795e16a411073d24 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 20:29:04 +0200 Subject: [PATCH 084/168] broadcast viewers stats per round --- modules/relay/src/main/JsonView.scala | 31 +++++++++------ modules/relay/src/main/RelayRoundRepo.scala | 6 +-- modules/relay/src/main/RelayStatsApi.scala | 44 +++++++++++++++------ modules/relay/src/main/ui/RelayTourUi.scala | 12 ++---- 4 files changed, 57 insertions(+), 36 deletions(-) diff --git a/modules/relay/src/main/JsonView.scala b/modules/relay/src/main/JsonView.scala index 91ddf7521bfd7..0bcdab2f91e7d 100644 --- a/modules/relay/src/main/JsonView.scala +++ b/modules/relay/src/main/JsonView.scala @@ -45,18 +45,6 @@ final class JsonView( "tours" -> g.withShorterTourNames.tours ) - given OWrites[RelayRound] = OWrites: r => - Json - .obj( - "id" -> r.id, - "name" -> r.name, - "slug" -> r.slug, - "createdAt" -> r.createdAt - ) - .add("finished" -> r.finished) - .add("ongoing" -> (r.hasStarted && !r.finished)) - .add("startsAt" -> r.startsAt.orElse(r.startedAt)) - def fullTour(tour: RelayTour): JsObject = Json .toJsObject(tour) @@ -165,6 +153,25 @@ object JsonView: given OWrites[SyncLog.Event] = Json.writes + given OWrites[RelayRound] = OWrites: r => + Json + .obj( + "id" -> r.id, + "name" -> r.name, + "slug" -> r.slug, + "createdAt" -> r.createdAt + ) + .add("finished" -> r.finished) + .add("ongoing" -> (r.hasStarted && !r.finished)) + .add("startsAt" -> r.startsAt.orElse(r.startedAt)) + + given OWrites[RelayStats.RoundStats] = OWrites: r => + Json.obj( + "round" -> r.round, + "viewers" -> r.viewers.map: (minute, crowd) => + Json.arr(minute * 60, crowd) + ) + private given OWrites[RelayRound.Sync] = OWrites: s => Json .obj( diff --git a/modules/relay/src/main/RelayRoundRepo.scala b/modules/relay/src/main/RelayRoundRepo.scala index 675529bf87a58..f869994a2c5c0 100644 --- a/modules/relay/src/main/RelayRoundRepo.scala +++ b/modules/relay/src/main/RelayRoundRepo.scala @@ -49,16 +49,16 @@ final private class RelayRoundRepo(val coll: Coll)(using Executor): def studyIdsOf(tourId: RelayTourId): Fu[List[StudyId]] = coll.distinctEasy[StudyId, List]("_id", selectors.tour(tourId)) - def tourCrowds: Fu[List[(RelayTourId, Int)]] = + def roundCrowds: Fu[List[(RelayRoundId, Int)]] = coll .aggregateList(maxDocs = 500, _.sec): framework => import framework.* Match($doc("sync.until" -> $exists(true), "crowd".$gt(0))) -> - List(GroupField("tourId")("crowd" -> SumField("crowd"))) + List(Project($doc("_id" -> 1, "crowd" -> 1))) .map: docs => for doc <- docs - id <- doc.getAsOpt[RelayTourId]("_id") + id <- doc.getAsOpt[RelayRoundId]("_id") crowd <- doc.getAsOpt[Int]("crowd") yield (id, crowd) diff --git a/modules/relay/src/main/RelayStatsApi.scala b/modules/relay/src/main/RelayStatsApi.scala index ffc2979976060..56e59039b1238 100644 --- a/modules/relay/src/main/RelayStatsApi.scala +++ b/modules/relay/src/main/RelayStatsApi.scala @@ -6,31 +6,51 @@ object RelayStats: type Minute = Int type Crowd = Int type Graph = List[(Minute, Crowd)] + case class RoundStats(round: RelayRound, viewers: Graph) final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using scheduler: Scheduler)(using Executor ): import RelayStats.* + import BSONHandlers.given // on measurement by minute at most; the storage depends on it. - scheduler.scheduleWithFixedDelay(0 seconds, 1 minute)(() => record()) + scheduler.scheduleWithFixedDelay(1 minute, 1 minute)(() => record()) - def get(id: RelayTourId): Fu[Graph] = - colls.stats - .primitiveOne[List[Int]]($id(id), "d") - .mapz: - _.grouped(2) - .collect: - case List(minute, crowd) => (minute, crowd) - .toList + def get(id: RelayTourId): Fu[List[RoundStats]] = + colls.round + .aggregateList(128): framework => + import framework.* + Match($doc("tourId" -> id)) -> List( + Sort(Ascending("createdAt")), + AddFields($doc("sync.log" -> $arr())), + PipelineOperator( + $lookup.simple(colls.stats, "stats", "_id", "_id") + ), + AddFields($doc("stats" -> $doc("$first" -> "$stats"))) + ) + .map: docs => + for + doc <- docs + round <- doc.asOpt[RelayRound] + data = for + doc <- doc.getAsOpt[Bdoc]("stats") + data <- doc.getAsOpt[List[Int]]("d") + yield data + stats = data.so: + _.grouped(2) + .collect: + case List(minute, crowd) => (minute, crowd) + .toList + yield RoundStats(round, stats) private def record(): Funit = for - crowds <- roundRepo.tourCrowds + crowds <- roundRepo.roundCrowds nowMinutes = nowSeconds / 60 update = colls.stats.update(ordered = false) - elements <- crowds.sequentially: (tourId, crowd) => + elements <- crowds.sequentially: (roundId, crowd) => update.element( - q = $id(tourId), + q = $id(roundId), u = $push("d" -> $doc("$each" -> $arr(nowMinutes, crowd))), upsert = true ) diff --git a/modules/relay/src/main/ui/RelayTourUi.scala b/modules/relay/src/main/ui/RelayTourUi.scala index dffc3bea5351a..54defb524d4d6 100644 --- a/modules/relay/src/main/ui/RelayTourUi.scala +++ b/modules/relay/src/main/ui/RelayTourUi.scala @@ -102,17 +102,11 @@ final class RelayTourUi(helpers: Helpers, ui: RelayUi): ) ) - def stats(t: RelayTour, graph: RelayStats.Graph)(using Context) = + def stats(t: RelayTour, stats: List[RelayStats.RoundStats])(using Context) = + import JsonView.given Page(s"${t.name.value} - Stats") .css("bits.relay.index") - .js( - PageModule( - "bits.relayStats", - Json - .obj("points" -> graph.map: (minute, crowd) => - Json.arr(minute * 60, crowd)) - ) - ): + .js(PageModule("bits.relayStats", Json.obj("rounds" -> stats))): main(cls := "relay-tour page box box-pad")( boxTop(h1(a(href := routes.RelayTour.show(t.slug, t.id).url)(t.name), " - Stats")), "Here, a graph shows the number of viewers over time." From 41e3db1b4e2a8cbf375bf5184e00350b2547c8e2 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 20:37:02 +0200 Subject: [PATCH 085/168] refactor broadcast stats and add logging --- modules/relay/src/main/RelayRoundRepo.scala | 13 ------------- modules/relay/src/main/RelayStatsApi.scala | 18 +++++++++++++++++- 2 files changed, 17 insertions(+), 14 deletions(-) diff --git a/modules/relay/src/main/RelayRoundRepo.scala b/modules/relay/src/main/RelayRoundRepo.scala index f869994a2c5c0..a288e03cbf128 100644 --- a/modules/relay/src/main/RelayRoundRepo.scala +++ b/modules/relay/src/main/RelayRoundRepo.scala @@ -49,19 +49,6 @@ final private class RelayRoundRepo(val coll: Coll)(using Executor): def studyIdsOf(tourId: RelayTourId): Fu[List[StudyId]] = coll.distinctEasy[StudyId, List]("_id", selectors.tour(tourId)) - def roundCrowds: Fu[List[(RelayRoundId, Int)]] = - coll - .aggregateList(maxDocs = 500, _.sec): framework => - import framework.* - Match($doc("sync.until" -> $exists(true), "crowd".$gt(0))) -> - List(Project($doc("_id" -> 1, "crowd" -> 1))) - .map: docs => - for - doc <- docs - id <- doc.getAsOpt[RelayRoundId]("_id") - crowd <- doc.getAsOpt[Int]("crowd") - yield (id, crowd) - private object RelayRoundRepo: object sort: diff --git a/modules/relay/src/main/RelayStatsApi.scala b/modules/relay/src/main/RelayStatsApi.scala index 56e59039b1238..311a9ae6793cb 100644 --- a/modules/relay/src/main/RelayStatsApi.scala +++ b/modules/relay/src/main/RelayStatsApi.scala @@ -45,7 +45,7 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc yield RoundStats(round, stats) private def record(): Funit = for - crowds <- roundRepo.roundCrowds + crowds <- fetchRoundCrowds nowMinutes = nowSeconds / 60 update = colls.stats.update(ordered = false) elements <- crowds.sequentially: (roundId, crowd) => @@ -56,3 +56,19 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc ) _ <- elements.nonEmpty.so(update.many(elements).void) yield () + + private def fetchRoundCrowds: Fu[List[(RelayRoundId, Crowd)]] = + val max = 500 + colls.round + .aggregateList(maxDocs = max, _.sec): framework => + import framework.* + Match($doc("sync.until" -> $exists(true), "crowd".$gt(0))) -> + List(Project($doc("_id" -> 1, "crowd" -> 1))) + .map: docs => + if docs.size == max + then logger.warn(s"RelayStats.fetchRoundCrowds: $max docs fetched") + for + doc <- docs + id <- doc.getAsOpt[RelayRoundId]("_id") + crowd <- doc.getAsOpt[Crowd]("crowd") + yield (id, crowd) From caf15219903470ba48d31646736c8c7c93790303 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 20:45:26 +0200 Subject: [PATCH 086/168] hide the broadcast stats link for now but start recording data --- ui/analyse/src/study/relay/relayTourView.ts | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/ui/analyse/src/study/relay/relayTourView.ts b/ui/analyse/src/study/relay/relayTourView.ts index edac7327fd2af..e79ddb1317860 100644 --- a/ui/analyse/src/study/relay/relayTourView.ts +++ b/ui/analyse/src/study/relay/relayTourView.ts @@ -285,13 +285,13 @@ const makeTabs = (ctrl: AnalyseCtrl) => { makeTab('boards', 'Boards'), relay.teams && makeTab('teams', 'Teams'), relay.data.tour.leaderboard ? makeTab('leaderboard', 'Leaderboard') : undefined, - study.members.myMember() - ? h( - 'a.text', - { attrs: { ...dataIcon(licon.LineGraph), href: `/broadcast/${relay.data.tour.id}/stats` } }, - 'Popularity stats', - ) - : undefined, + // study.members.myMember() + // ? h( + // 'a.text', + // { attrs: { ...dataIcon(licon.LineGraph), href: `/broadcast/${relay.data.tour.id}/stats` } }, + // 'Popularity stats', + // ) + // : undefined, ]); }; From 222ad9dcb67dbda4e69b7e67a1d94175a1d57fb4 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Sun, 16 Jun 2024 20:52:40 +0200 Subject: [PATCH 087/168] disable blocked player's dropdown items - closes #15536 --- modules/relation/src/main/ui/RelationUi.scala | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/modules/relation/src/main/ui/RelationUi.scala b/modules/relation/src/main/ui/RelationUi.scala index 8a3a39a07fbec..5ba6e197671c1 100644 --- a/modules/relation/src/main/ui/RelationUi.scala +++ b/modules/relation/src/main/ui/RelationUi.scala @@ -51,8 +51,9 @@ final class RelationUi(helpers: Helpers): blocked: Boolean, signup: Boolean = false )(using ctx: Context) = + val blocks = relation.contains(Relation.Block) div(cls := "relation-actions")( - (ctx.isnt(user) && !blocked).option( + (ctx.isnt(user) && !blocked && !blocks).option( a( cls := "text", href := s"${routes.Lobby.home}?user=${user.name}#friend", @@ -64,14 +65,14 @@ final class RelationUi(helpers: Helpers): (!user.is(myId)) .so( frag( - (!blocked && !user.isBot).option( + (!blocked && !blocks && !user.isBot).option( a( cls := "text", href := routes.Msg.convo(user.name), dataIcon := Icon.BubbleSpeech )(trans.site.composeMessage.txt()) ), - (!blocked && !user.isPatron).option( + (!blocked && !blocks && !user.isPatron).option( a( cls := "text", href := s"${routes.Plan.list}?dest=gift&giftUsername=${user.name}", From 3e23d94cc384bf1c3096c62f6258b686f8bbf126 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 07:27:35 +0200 Subject: [PATCH 088/168] New Crowdin updates (#15509) * New translations: site.xml (Norwegian Bokmal) * New translations: faq.xml (Norwegian Bokmal) * New translations: site.xml (Danish) * New translations: site.xml (Norwegian Bokmal) * New translations: site.xml (Japanese) * New translations: team.xml (Arabic) * New translations: team.xml (Arabic) * New translations: coordinates.xml (Finnish) * New translations: site.xml (Finnish) * New translations: team.xml (Finnish) --- translation/dest/coordinates/fi-FI.xml | 1 + translation/dest/faq/nb-NO.xml | 4 ++-- translation/dest/site/da-DK.xml | 4 ++-- translation/dest/site/fi-FI.xml | 2 ++ translation/dest/site/ja-JP.xml | 2 ++ translation/dest/site/nb-NO.xml | 10 ++++++---- translation/dest/team/ar-SA.xml | 9 +++++++++ translation/dest/team/fi-FI.xml | 1 + 8 files changed, 25 insertions(+), 8 deletions(-) diff --git a/translation/dest/coordinates/fi-FI.xml b/translation/dest/coordinates/fi-FI.xml index 2132e97b438f5..22641c1cfa74f 100644 --- a/translation/dest/coordinates/fi-FI.xml +++ b/translation/dest/coordinates/fi-FI.xml @@ -13,6 +13,7 @@ Sinulla on 30 sekuntia aikaa paikantaa niin monta ruutua kuin ehdit! Tee sitä kaikessa rauhassa, aikarajaa ei ole! Näytä koordinaatit + Koordinaatit jokaisessa ruudussa Näytä nappulat Aloita harjoitus Etsi ruutu diff --git a/translation/dest/faq/nb-NO.xml b/translation/dest/faq/nb-NO.xml index 37b62c8ec90d2..60f24e37bc3c4 100644 --- a/translation/dest/faq/nb-NO.xml +++ b/translation/dest/faq/nb-NO.xml @@ -51,7 +51,7 @@ Lichess bruker Stockfish til maskinanalyse. Tap på tid, remis og utilstrekkelig materiell Hvis tiden løper ut for en spiller, vil vedkommende vanligvis tape partiet. Partiet er likevel remis hvis stillingen er slik at motstanderen ikke kan matte spillerens konge ved noen lovlig trekkrekkefølge (%1$s). -Dette kan i sjeldne tilfeller være vanskelig å avgjøre automatisk (tvungne linjer, festninger). Da holder vi med spilleren der tiden ikke løp ut. +Dette kan i sjeldne tilfeller være vanskelig å avgjøre automatisk (tvungne varianter, festninger). Da holder vi med spilleren der tiden ikke løp ut. Legg merke til at den kan være mulig å matte med en enkelt springer eller løper hvis motstanderen har en brikke som kan blokkere kongen. FIDEs håndbok @@ -167,7 +167,7 @@ Vi viser det røde ikonet for å varsle deg når dette skjer. Ofte kan du ekspli 3. Klikk på Informasjonskapsler og nettstedstillatelser 4. Rull ned og klikk på Automatisk avspilling av medier 5. Legg til lichess.org under Tillat - Stoppe meg selv fra å spille? + Stoppe meg selv fra å spille? egen psykisk helsetilstand Lichess-brukerstiler færre lobbypuljer diff --git a/translation/dest/site/da-DK.xml b/translation/dest/site/da-DK.xml index 404fa19acad46..3e0f9b7520432 100644 --- a/translation/dest/site/da-DK.xml +++ b/translation/dest/site/da-DK.xml @@ -38,8 +38,8 @@ Hvid i trækket Sort i trækket - Din modstander har forladt partiet. Du kan kræve sejr om %s sekund. - Din modstander har forladt partiet. Du kan kræve sejr om %s sekunder. + Din modstander har forladt partiet. Du kan kræve at få tildelt sejren om %s sekund. + Din modstander har forladt partiet. Du kan kræve at få tildelt sejren om %s sekunder. Din modstander har forladt partiet. Du kan fremtvinge modstanderens kapitulation eller vente. Kræv sejr diff --git a/translation/dest/site/fi-FI.xml b/translation/dest/site/fi-FI.xml index f3732010a2742..62eaa8ba665cc 100644 --- a/translation/dest/site/fi-FI.xml +++ b/translation/dest/site/fi-FI.xml @@ -70,6 +70,8 @@ Nosta muunnelmaa Päämuunnelmaksi Poista tästä alkaen + Taita kokoon muunnelmat + Laajenna muunnelmat Pakota muunnelmaksi Kopioi muunnelman PGN Siirto diff --git a/translation/dest/site/ja-JP.xml b/translation/dest/site/ja-JP.xml index 5c3959d19164c..7406145dafbf2 100644 --- a/translation/dest/site/ja-JP.xml +++ b/translation/dest/site/ja-JP.xml @@ -69,6 +69,8 @@ 変化を主手順にする 主手順にする これ以降を削除 + 変化手順をかくす + 変化手順を表示する 変化として表示 手順の PGN をコピー diff --git a/translation/dest/site/nb-NO.xml b/translation/dest/site/nb-NO.xml index 1a9d2737b47d6..c1a31af2191f6 100644 --- a/translation/dest/site/nb-NO.xml +++ b/translation/dest/site/nb-NO.xml @@ -70,6 +70,8 @@ Forfrem variant Gjør til hovedvariant Slett herfra + Skjul varianter + Vis varianter Vis som variant Kopier variant-PGN Trekk @@ -119,7 +121,7 @@ Pil for beste trekk Vis variantpiler Evalueringsmåler - Flere linjer + Flere varianter Prosessorer Minne Uendelig analyse @@ -330,7 +332,7 @@ Gi %s sekunder Denne kontoen brøt vilkårene for å bruke Lichess - Åpningsutforsker & tabellbase + Åpningsutforsker og tabellbase Angre Foreslå å angre Angreforslag sendt @@ -865,8 +867,8 @@ Ingen betingede forhåndstrekk Spill %s - og lagre %s linje med forhåndstrekk - og lagre %s linjer med forhåndstrekk + og lagre %s forhåndstrekkvariant + og lagre %s forhåndstrekkvarianter Du har mottatt en privat melding fra Lichess. Klikk her for å lese den diff --git a/translation/dest/team/ar-SA.xml b/translation/dest/team/ar-SA.xml index 9fff9f41a1e68..be798a3abfa64 100644 --- a/translation/dest/team/ar-SA.xml +++ b/translation/dest/team/ar-SA.xml @@ -67,4 +67,13 @@ الطلبات المرفوضة انضم إلى فريق %s الرسمي للأخبار والأحداث صفحة الفريق + انتهت هذه البطولة، ولم يعد من الممكن تحديث الفريق. + قائمة الفرق التي ستتنافس في هذه المعركة. + فريق واحد لكل سطر. استخدم الإكمال التلقائي. + يمكنك نسخ ولصق هذه القائمة من بطولة إلى أخرى! + +لا يمكنك إزالة فريق إذا كان اللاعب قد انضم فعلًا إلى البطولة معه. + عدد القادة لكل فريق. مجموع درجاتهم هو نتيجة الفريق. + لا يجب عليك تغيير هذه القيمة بعد بدء البطولة! + الفريق الداخلي diff --git a/translation/dest/team/fi-FI.xml b/translation/dest/team/fi-FI.xml index 7f6c3f77ae6f2..741c07a11e079 100644 --- a/translation/dest/team/fi-FI.xml +++ b/translation/dest/team/fi-FI.xml @@ -63,4 +63,5 @@ Pelaajat, jotka eivät mielellään ota vastaan viestejäsi, saattavat lähteä Et voi poistaa joukkuetta, jos pelaaja on jo liittynyt turnaukseen sen kanssa. Johtajien määrä kussakin joukkueessa. Heidän pisteidensä summa muodostaa joukkueen pistemäärän. Tätä arvoa ei todellakaan tulisi muuttaa turnauksen jo käynnistyttyä! + Sisäinen joukkue From 9fc0cbaec62456d9f6bc45d57e87033bafaa8968 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 07:54:59 +0200 Subject: [PATCH 089/168] fix mongodb aggregation --- modules/relay/src/main/RelayListing.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/relay/src/main/RelayListing.scala b/modules/relay/src/main/RelayListing.scala index 5907619781f27..49191e9c3f330 100644 --- a/modules/relay/src/main/RelayListing.scala +++ b/modules/relay/src/main/RelayListing.scala @@ -64,8 +64,8 @@ final class RelayListing( "$filter" -> $doc( "input" -> "$tours", "as" -> "tour", - "cond" -> "$$tour.live", - "limit" -> 1 + "cond" -> "$$tour.live" + // "limit" -> 1 // TODO unsupported by mongodb 4.4 (but also not needed here) ) ) ), From bef8bb058c68d6f7b10a1a567e9100371db1ec4c Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 08:54:25 +0200 Subject: [PATCH 090/168] fix some z-index after #15510 --- ui/analyse/css/study/relay/_tour.scss | 2 +- ui/board/css/_menu.scss | 4 ++-- ui/ceval/css/_settings.scss | 2 +- ui/voice/css/_voice.scss | 2 +- 4 files changed, 5 insertions(+), 5 deletions(-) diff --git a/ui/analyse/css/study/relay/_tour.scss b/ui/analyse/css/study/relay/_tour.scss index a5fe56f09aeb5..2f15dc0c698db 100644 --- a/ui/analyse/css/study/relay/_tour.scss +++ b/ui/analyse/css/study/relay/_tour.scss @@ -456,7 +456,7 @@ $hover-bg: $m-primary_bg--mix-30; max-height: 12em; overflow-y: auto; background: $c-bg-zebra; - z-index: z(mz-menu); + z-index: $z-mz-menu-4; width: 100%; border: $c-primary 2px solid; } diff --git a/ui/board/css/_menu.scss b/ui/board/css/_menu.scss index 86d4ad0f3685d..ee76dbac89ef5 100644 --- a/ui/board/css/_menu.scss +++ b/ui/board/css/_menu.scss @@ -8,14 +8,14 @@ position: absolute; right: 0; min-width: 100%; - z-index: z(mz-menu); + z-index: $z-mz-menu-4; } @media (max-width: at-most($small)) { position: fixed; top: 50% !important; left: 50%; transform: translate(-50%, -50%); - z-index: z(context-menu); + z-index: $z-context-menu-108; min-width: 80%; border-radius: $box-radius-size; } diff --git a/ui/ceval/css/_settings.scss b/ui/ceval/css/_settings.scss index cc19d82570867..3597e77e54350 100644 --- a/ui/ceval/css/_settings.scss +++ b/ui/ceval/css/_settings.scss @@ -6,7 +6,7 @@ @extend %flex-column, %dropdown-shadow, %box-radius-bottom; position: absolute; border-top: 2px solid $c-primary; - z-index: z(mz-menu); + z-index: $z-mz-menu-4; width: 100%; background: $c-bg-high; gap: 1.5em; diff --git a/ui/voice/css/_voice.scss b/ui/voice/css/_voice.scss index 8c67c1d17dcee..be0f41a9db84e 100644 --- a/ui/voice/css/_voice.scss +++ b/ui/voice/css/_voice.scss @@ -114,7 +114,7 @@ button#microphone-button { @extend %flex-column, %dropdown-shadow, %box-radius-bottom; position: absolute; border-top: 2px solid $c-primary; - z-index: z(mz-menu); + z-index: $z-mz-menu-4; width: 100%; background: $c-bg-high; gap: 1.5em; From 6b44f3be73c1312afb63418b403adf43dd66a479 Mon Sep 17 00:00:00 2001 From: Allan Joseph Date: Mon, 17 Jun 2024 07:18:47 +0000 Subject: [PATCH 091/168] upgrade chartjs everywhere --- pnpm-lock.yaml | 21 ++++++--------------- ui/insight/package.json | 2 +- ui/opening/package.json | 2 +- 3 files changed, 8 insertions(+), 17 deletions(-) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index b12793ea45865..d3e1e761fb0ff 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -380,11 +380,11 @@ importers: specifier: workspace:^ version: link:../chart chart.js: - specifier: ^4.4.0 - version: 4.4.0 + specifier: 4.4.3 + version: 4.4.3 chartjs-plugin-datalabels: specifier: ^2.2.0 - version: 2.2.0(chart.js@4.4.0) + version: 2.2.0(chart.js@4.4.3) common: specifier: workspace:* version: link:../common @@ -503,11 +503,11 @@ importers: specifier: ^3.1.9 version: 3.1.9 chart.js: - specifier: 4.4.0 - version: 4.4.0 + specifier: 4.4.3 + version: 4.4.3 chartjs-adapter-dayjs-4: specifier: ^1.0.4 - version: 1.0.4(chart.js@4.4.0)(dayjs@1.11.10) + version: 1.0.4(chart.js@4.4.3)(dayjs@1.11.10) common: specifier: workspace:* version: link:../common @@ -3042,20 +3042,11 @@ snapshots: dependencies: '@kurkle/color': 0.3.2 - chartjs-adapter-dayjs-4@1.0.4(chart.js@4.4.0)(dayjs@1.11.10): - dependencies: - chart.js: 4.4.0 - dayjs: 1.11.10 - chartjs-adapter-dayjs-4@1.0.4(chart.js@4.4.3)(dayjs@1.11.10): dependencies: chart.js: 4.4.3 dayjs: 1.11.10 - chartjs-plugin-datalabels@2.2.0(chart.js@4.4.0): - dependencies: - chart.js: 4.4.0 - chartjs-plugin-datalabels@2.2.0(chart.js@4.4.3): dependencies: chart.js: 4.4.3 diff --git a/ui/insight/package.json b/ui/insight/package.json index 52e7f67a6b37d..2cc781fb9337c 100644 --- a/ui/insight/package.json +++ b/ui/insight/package.json @@ -13,7 +13,7 @@ "license": "AGPL-3.0-or-later", "dependencies": { "chart": "workspace:^", - "chart.js": "^4.4.0", + "chart.js": "4.4.3", "chartjs-plugin-datalabels": "^2.2.0", "common": "workspace:*", "snabbdom": "3.5.1" diff --git a/ui/opening/package.json b/ui/opening/package.json index 4bd09189b3789..a8db21be1c34e 100644 --- a/ui/opening/package.json +++ b/ui/opening/package.json @@ -7,7 +7,7 @@ "license": "AGPL-3.0-or-later", "dependencies": { "@types/debounce-promise": "^3.1.9", - "chart.js": "4.4.0", + "chart.js": "4.4.3", "chartjs-adapter-dayjs-4": "^1.0.4", "common": "workspace:*", "dayjs": "^1.11.10", From fca5d062c47ba0fce62d1dba37d969bb4d77c10b Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 11:30:59 +0200 Subject: [PATCH 092/168] sneak in another rounding pixel --- ui/common/css/abstract/_variables.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/common/css/abstract/_variables.scss b/ui/common/css/abstract/_variables.scss index 006f8171352c5..9dbb7930583a0 100644 --- a/ui/common/css/abstract/_variables.scss +++ b/ui/common/css/abstract/_variables.scss @@ -7,7 +7,7 @@ $viewport-min-width: 320px; $block-gap: var(---block-gap); -$box-radius-size: 5px; +$box-radius-size: 6px; $box-padding-vert: 5vh; $transition-duration: 150ms; From 62bbbee01df5d7843ff343703a39da297d315389 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 11:31:07 +0200 Subject: [PATCH 093/168] broadcast form navigation --- app/controllers/RelayRound.scala | 28 +-- app/controllers/RelayTour.scala | 29 +-- modules/relay/src/main/RelayApi.scala | 19 +- modules/relay/src/main/RelayRoundRepo.scala | 8 +- modules/relay/src/main/ui/FormUi.scala | 187 +++++++++++------- .../src/study/relay/relayManagerView.ts | 12 +- ui/bits/css/relay/_form.scss | 26 +++ ui/bits/css/user/_show.scss | 6 +- 8 files changed, 196 insertions(+), 119 deletions(-) diff --git a/app/controllers/RelayRound.scala b/app/controllers/RelayRound.scala index 3d3bd95b4fd10..c1c2dade5058d 100644 --- a/app/controllers/RelayRound.scala +++ b/app/controllers/RelayRound.scala @@ -9,6 +9,7 @@ import scala.annotation.nowarn import lila.app.{ *, given } import lila.common.HTTPRequest import lila.relay.{ RelayRound as RoundModel, RelayTour as TourModel } +import lila.relay.ui.FormNavigation import lila.core.id.{ RelayRoundId, RelayTourId } final class RelayRound( @@ -21,7 +22,7 @@ final class RelayRound( NoLameOrBot: WithTourAndRoundsCanUpdate(tourId): trs => Ok.page: - views.relay.form.round.create(env.relay.roundForm.create(trs), trs.tour) + views.relay.form.round.create(env.relay.roundForm.create(trs), FormNavigation(trs, none)) } def create(tourId: RelayTourId) = AuthOrScopedBody(_.Study.Write) { ctx ?=> me ?=> @@ -35,7 +36,7 @@ final class RelayRound( bindForm(env.relay.roundForm.create(trs))( err => negotiate( - BadRequest.page(views.relay.form.round.create(err, tour)), + BadRequest.page(views.relay.form.round.create(err, FormNavigation(trs, none))), jsonFormError(err) ), setup => @@ -51,30 +52,31 @@ final class RelayRound( } def edit(id: RelayRoundId) = Auth { ctx ?=> me ?=> - FoundPage(env.relay.api.byIdAndContributor(id)): rt => - views.relay.form.round.edit(rt, env.relay.roundForm.edit(rt.round)) + FoundPage(env.relay.api.formNavigation(id)): (round, nav) => + views.relay.form.round.edit(round, env.relay.roundForm.edit(round), nav) } def update(id: RelayRoundId) = AuthOrScopedBody(_.Study.Write) { ctx ?=> me ?=> given play.api.Mode = env.mode env.relay.api - .byIdAndContributor(id) - .flatMapz: rt => - bindForm(env.relay.roundForm.edit(rt.round))( - err => fuccess(Left(rt -> err)), + .formNavigation(id) + .flatMapz: (round, nav) => + bindForm(env.relay.roundForm.edit(round))( + err => fuccess(Left((round, nav) -> err)), data => env.relay.api - .update(rt.round)(data.update(rt.tour.official)) - .dmap(_.withTour(rt.tour)) + .update(round)(data.update(nav.tour.official)) + .dmap(_.withTour(nav.tour)) .dmap(Right(_)) ).dmap(some) .orNotFound: _.fold( - (old, err) => + { case ((round, nav), err) => negotiate( - BadRequest.page(views.relay.form.round.edit(old, err)), + BadRequest.page(views.relay.form.round.edit(round, err, nav)), jsonFormError(err) - ), + ) + }, rt => negotiate(Redirect(rt.path), JsonOk(env.relay.jsonView.withUrl(rt, withTour = true))) ) } diff --git a/app/controllers/RelayTour.scala b/app/controllers/RelayTour.scala index 465ba63651d20..ec77fba160af9 100644 --- a/app/controllers/RelayTour.scala +++ b/app/controllers/RelayTour.scala @@ -6,6 +6,7 @@ import scalalib.Json.given import lila.app.{ *, given } import lila.core.net.IpAddress import lila.relay.RelayTour as TourModel +import lila.relay.ui.FormNavigation import lila.core.id.RelayTourId final class RelayTour(env: Env, apiC: => Api) extends LilaController(env): @@ -98,23 +99,23 @@ final class RelayTour(env: Env, apiC: => Api) extends LilaController(env): } def edit(id: RelayTourId) = Auth { ctx ?=> _ ?=> - WithTourCanUpdate(id): tg => + WithTourCanUpdate(id): nav => Ok.page: - views.relay.form.tour.edit(tg, env.relay.tourForm.edit(tg)) + views.relay.form.tour.edit(env.relay.tourForm.edit(nav.tourWithGroup), nav) } def update(id: RelayTourId) = AuthOrScopedBody(_.Study.Write) { ctx ?=> me ?=> - WithTourCanUpdate(id): tg => - bindForm(env.relay.tourForm.edit(tg))( + WithTourCanUpdate(id): nav => + bindForm(env.relay.tourForm.edit(nav.tourWithGroup))( err => negotiate( - BadRequest.page(views.relay.form.tour.edit(tg, err)), + BadRequest.page(views.relay.form.tour.edit(err, nav)), jsonFormError(err) ), setup => - env.relay.api.tourUpdate(tg.tour, setup) >> + env.relay.api.tourUpdate(nav.tour, setup) >> negotiate( - Redirect(routes.RelayTour.show(tg.tour.slug, tg.tour.id)), + Redirect(routes.RelayTour.show(nav.tour.slug, nav.tour.id)), jsonOkResult ) ) @@ -126,16 +127,16 @@ final class RelayTour(env: Env, apiC: => Api) extends LilaController(env): } def image(id: RelayTourId, tag: Option[String]) = AuthBody(parse.multipartFormData) { ctx ?=> me ?=> - WithTourCanUpdate(id): tg => + WithTourCanUpdate(id): nav => ctx.body.body.file("image") match case Some(image) => limit.imageUpload(ctx.ip, rateLimited): - (env.relay.api.image.upload(me, tg.tour, image, tag) >> { + (env.relay.api.image.upload(me, nav.tour, image, tag) >> { Ok }).recover { case e: Exception => BadRequest(e.getMessage) } - case None => env.relay.api.image.delete(tg.tour, tag) >> Ok + case None => env.relay.api.image.delete(nav.tour, tag) >> Ok } def leaderboardView(id: RelayTourId) = Open: @@ -212,12 +213,12 @@ final class RelayTour(env: Env, apiC: => Api) extends LilaController(env): private def WithTourCanUpdate( id: RelayTourId - )(f: TourModel.WithGroupTours => Fu[Result])(using ctx: Context): Fu[Result] = + )(f: (FormNavigation) => Fu[Result])(using Context, Me): Fu[Result] = WithTour(id): tour => - ctx.me - .soUse { env.relay.api.canUpdate(tour) } + env.relay.api + .canUpdate(tour) .elseNotFound: - env.relay.api.withTours.addTo(tour).flatMap(f) + env.relay.api.formNavigation(tour).flatMap(f) private[controllers] def rateLimitCreation( fail: => Fu[Result] diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index a6c7868fd6405..a88a458f4069b 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -45,9 +45,20 @@ final class RelayApi( def byIdAndContributor(id: RelayRoundId)(using me: Me): Fu[Option[WithTour]] = byIdWithTourAndStudy(id).map: _.collect: - case RelayRound.WithTourAndStudy(relay, tour, study) if study.canContribute(me) => + case RelayRound.WithTourAndStudy(relay, tour, study) + if study.canContribute(me) || Granter(_.StudyAdmin) => relay.withTour(tour) + def formNavigation(id: RelayRoundId)(using me: Me): Fu[Option[(RelayRound, ui.FormNavigation)]] = + byIdAndContributor(id).flatMapz: rt => + formNavigation(rt.tour).map: nav => + (rt.round, nav.copy(round = rt.round.id.some)).some + + def formNavigation(tour: RelayTour)(using me: Me): Fu[ui.FormNavigation] = for + group <- withTours.get(tour.id) + rounds <- roundRepo.byTourOrdered(tour.id) + yield ui.FormNavigation(group, tour, rounds, none) + def byIdWithTourAndStudy(id: RelayRoundId): Fu[Option[RelayRound.WithTourAndStudy]] = byIdWithTour(id).flatMapz { case WithTour(relay, tour) => studyApi @@ -64,7 +75,7 @@ final class RelayApi( RelayRound.WithStudy(relay, _) def byTourOrdered(tour: RelayTour): Fu[List[WithTour]] = - roundRepo.byTourOrdered(tour).dmap(_.map(_.withTour(tour))) + roundRepo.byTourOrdered(tour.id).dmap(_.map(_.withTour(tour))) def roundIdsById(tourId: RelayTourId): Fu[List[StudyId]] = roundRepo.idsByTourId(tourId) @@ -73,7 +84,7 @@ final class RelayApi( roundIdsById(tourId).flatMap: _.sequentiallyVoid(studyApi.kick(_, userId, who)) - def withRounds(tour: RelayTour) = roundRepo.byTourOrdered(tour).dmap(tour.withRounds) + def withRounds(tour: RelayTour) = roundRepo.byTourOrdered(tour.id).dmap(tour.withRounds) def denormalizeTourActive(tourId: RelayTourId): Funit = val unfinished = RelayRoundRepo.selectors.tour(tourId) ++ $doc("finished" -> false) @@ -314,7 +325,7 @@ final class RelayApi( ) tourRepo.coll.insert.one(tour) >> roundRepo - .byTourOrderedCursor(from) + .byTourOrderedCursor(from.id) .documentSource() .mapAsync(1)(cloneWithStudy(_, tour)) .runWith(Sink.ignore) diff --git a/modules/relay/src/main/RelayRoundRepo.scala b/modules/relay/src/main/RelayRoundRepo.scala index a288e03cbf128..eedde0a66ac91 100644 --- a/modules/relay/src/main/RelayRoundRepo.scala +++ b/modules/relay/src/main/RelayRoundRepo.scala @@ -10,14 +10,14 @@ final private class RelayRoundRepo(val coll: Coll)(using Executor): import RelayRoundRepo.* import BSONHandlers.given - def byTourOrderedCursor(tour: RelayTour) = + def byTourOrderedCursor(tourId: RelayTourId) = coll - .find(selectors.tour(tour.id)) + .find(selectors.tour(tourId)) .sort(sort.chrono) .cursor[RelayRound]() - def byTourOrdered(tour: RelayTour): Fu[List[RelayRound]] = - byTourOrderedCursor(tour).list(RelayTour.maxRelays) + def byTourOrdered(tourId: RelayTourId): Fu[List[RelayRound]] = + byTourOrderedCursor(tourId).list(RelayTour.maxRelays) def idsByTourOrdered(tour: RelayTour): Fu[List[RelayRoundId]] = coll.primitive[RelayRoundId]( diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index d8bda1ef7337c..ae17b19facc46 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -8,40 +8,94 @@ import ScalatagsTemplate.{ *, given } import play.api.data.Form import lila.core.id.ImageId +case class FormNavigation( + group: Option[RelayGroup.WithTours], + tour: RelayTour, + rounds: List[RelayRound], + round: Option[RelayRoundId], + newRound: Boolean = false +): + def tourWithGroup = RelayTour.WithGroupTours(tour, group) + +object FormNavigation: + def apply(trs: RelayTour.WithRounds, roundId: Option[RelayRoundId]): FormNavigation = + FormNavigation(none, trs.tour, trs.rounds, roundId) + final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): import helpers.{ *, given } import trans.{ broadcast as trb } + private def navigationMenu(nav: FormNavigation)(using Context) = + def tourAndRounds(shortName: Option[RelayTour.Name]) = frag( + a( + href := routes.RelayTour.edit(nav.tour.id), + cls := List( + "relay-form__subnav__tour-parent" -> shortName.isDefined, + "active" -> (nav.round.isEmpty && !nav.newRound) + ) + )( + shortName.fold(frag(nav.tour.name))(strong(_)) + ), + frag( + nav.rounds.map: r => + a( + href := routes.RelayRound.edit(r.id), + cls := List("subnav__subitem" -> true, "active" -> nav.round.has(r.id)) + )(r.name), + a( + href := routes.RelayRound.create(nav.tour.id), + cls := List("subnav__subitem text" -> true, "active" -> nav.newRound), + dataIcon := Icon.PlusButton + )("New round") + ) + ) + lila.ui.bits.pageMenuSubnav( + cls := "relay-form__subnav", + nav.group match + case None => tourAndRounds(none) + case Some(g) => + frag( + span(cls := "relay-form__subnav__group")(g.group.name), + g.withShorterTourNames.tours.map: t => + if nav.tour.id == t.id then tourAndRounds(t.name.some) + else a(href := routes.RelayTour.edit(t.id), cls := List("subnav__item" -> true))(t.name) + ) + ) + object round: - private def page(title: String)(using Context) = + private def page(title: String, nav: FormNavigation)(using Context) = Page(title) .css("bits.relay.form") .js(EsmInit("bits.flatpickr")) .wrap: body => - main(cls := "page-small box box-pad")(body) - - def create(form: Form[RelayRoundForm.Data], tour: RelayTour)(using Context) = - page(trans.broadcast.newBroadcast.txt()): - frag( - boxTop(h1(a(href := routes.RelayTour.edit(tour.id))(tour.name), " • ", trans.broadcast.addRound())), - standardFlash, - inner(form, routes.RelayRound.create(tour.id), tour, create = true) - ) + main(cls := "page page-menu")( + navigationMenu(nav), + div(cls := "page-menu__content box box-pad")(body) + ) - def edit(rt: RelayRound.WithTour, form: Form[RelayRoundForm.Data])(using Context) = - page(rt.fullName): + def create(form: Form[RelayRoundForm.Data], nav: FormNavigation)(using Context) = + page(trans.broadcast.newBroadcast.txt(), nav.copy(newRound = true)): frag( boxTop( - h1(dataIcon := Icon.Pencil, cls := "text")( - a(href := routes.RelayTour.edit(rt.tour.id))(rt.tour.name), + h1( + a(href := routes.RelayTour.edit(nav.tour.id))(nav.tour.name), " • ", - a(href := rt.path)(rt.round.name) + trans.broadcast.addRound() ) ), - inner(form, routes.RelayRound.update(rt.round.id), rt.tour, create = false), + standardFlash, + inner(form, routes.RelayRound.create(nav.tour.id), nav.tour, create = true) + ) + + def edit(r: RelayRound, form: Form[RelayRoundForm.Data], nav: FormNavigation)(using Context) = + page(r.name.value, nav): + val rt = r.withTour(nav.tour) + frag( + boxTop(h1(a(href := rt.path)(rt.fullName))), + inner(form, routes.RelayRound.update(r.id), nav.tour, create = false), div(cls := "relay-form__actions")( - postForm(action := routes.RelayRound.reset(rt.round.id))( + postForm(action := routes.RelayRound.reset(r.id))( submitButton( cls := "button button-red button-empty confirm" )( @@ -49,7 +103,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): em(trb.deleteAllGamesOfThisRound()) ) ), - postForm(action := routes.Study.delete(rt.round.studyId))( + postForm(action := routes.Study.delete(r.studyId))( submitButton( cls := "button button-red button-empty confirm" )(strong(trb.deleteRound()), em(trb.definitivelyDeleteRound())) @@ -146,21 +200,18 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): object tour: - private def page(title: String, menu: Option[String])(using Context) = + private def page(title: String, menu: Either[String, FormNavigation])(using Context) = Page(title) .css("bits.relay.form") .js(EsmInit("bits.relayForm")) .wrap: body => - menu match - case Some(active) => - main(cls := "page page-menu")( - tourUi.pageMenu(active), - div(cls := "page-menu__content box box-pad")(body) - ) - case None => main(cls := "page-small box box-pad")(body) + main(cls := "page page-menu")( + menu.fold(tourUi.pageMenu(_), navigationMenu), + div(cls := "page-menu__content box box-pad")(body) + ) def create(form: Form[lila.relay.RelayTourForm.Data])(using Context) = - page(trans.broadcast.newBroadcast.txt(), menu = "new".some): + page(trans.broadcast.newBroadcast.txt(), menu = Left("new")): frag( boxTop(h1(dataIcon := Icon.RadioTower, cls := "text")(trans.broadcast.newBroadcast())), postForm(cls := "form3", action := routes.RelayTour.create)( @@ -172,23 +223,19 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): ) ) - def edit(tg: RelayTour.WithGroupTours, form: Form[RelayTourForm.Data])(using Context) = - page(tg.tour.name.value, menu = none): + def edit(form: Form[RelayTourForm.Data], nav: FormNavigation)(using Context) = + page(nav.tour.name.value, menu = Right(nav)): frag( - boxTop: - h1(dataIcon := Icon.Pencil, cls := "text"): - a(href := routes.RelayTour.show(tg.tour.slug, tg.tour.id))(tg.tour.name) - , - image(tg.tour), - postForm(cls := "form3", action := routes.RelayTour.update(tg.tour.id))( - inner(form, tg.some), + boxTop(h1(a(href := routes.RelayTour.show(nav.tour.slug, nav.tour.id))(nav.tour.name))), + postForm(cls := "form3", action := routes.RelayTour.update(nav.tour.id))( + inner(form, nav.tourWithGroup.some), form3.actions( - a(href := routes.RelayTour.show(tg.tour.slug, tg.tour.id))(trans.site.cancel()), + a(href := routes.RelayTour.show(nav.tour.slug, nav.tour.id))(trans.site.cancel()), form3.submit(trans.site.apply()) ) ), div(cls := "relay-form__actions")( - postForm(action := routes.RelayTour.delete(tg.tour.id))( + postForm(action := routes.RelayTour.delete(nav.tour.id))( submitButton( cls := "button button-red button-empty confirm" )(strong(trb.deleteTournament()), em(trb.definitivelyDeleteTournament())) @@ -196,7 +243,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): Granter .opt(_.Relay) .option( - postForm(action := routes.RelayTour.cloneTour(tg.tour.id))( + postForm(action := routes.RelayTour.cloneTour(nav.tour.id))( submitButton( cls := "button button-green button-empty confirm" )( @@ -268,7 +315,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): pre("player name / rating / title / new name"), "All values are optional. Example:", pre("""Magnus Carlsen / 2863 / GM -YouGotLittUp / 1890 / / Louis Litt""") + YouGotLittUp / 1890 / / Louis Litt""") ).some, half = true )(form3.textarea(_)(rows := 3)), @@ -280,7 +327,7 @@ YouGotLittUp / 1890 / / Louis Litt""") pre("Team name; Fide Id or Player name"), "Example:", pre("""Team Cats ; 3408230 -Team Dogs ; Scooby Doo"""), + Team Dogs ; Scooby Doo"""), "By default the PGN tags WhiteTeam and BlackTeam are used." ).some, half = true @@ -361,39 +408,39 @@ Team Dogs ; Scooby Doo"""), ) ) - private def image(t: RelayTour)(using ctx: Context) = - div(cls := "relay-image-edit", data("post-url") := routes.RelayTour.image(t.id))( - ui.thumbnail(t.image, _.Size.Small)( - cls := List("drop-target" -> true, "user-image" -> t.image.isDefined), - attr("draggable") := "true" + private def image(t: RelayTour)(using ctx: Context) = + div(cls := "relay-image-edit", data("post-url") := routes.RelayTour.image(t.id))( + ui.thumbnail(t.image, _.Size.Small)( + cls := List("drop-target" -> true, "user-image" -> t.image.isDefined), + attr("draggable") := "true" + ), + div( + p("Upload a beautiful image to represent your tournament."), + p("The image must be twice as wide as it is tall. Recommended resolution: 1000x500."), + p( + "A picture of the city where the tournament takes place is a good idea, but feel free to design something different." ), - div( - p("Upload a beautiful image to represent your tournament."), - p("The image must be twice as wide as it is tall. Recommended resolution: 1000x500."), - p( - "A picture of the city where the tournament takes place is a good idea, but feel free to design something different." - ), - p(trans.streamer.maxSize(s"${lila.memo.PicfitApi.uploadMaxMb}MB.")), - form3.file.selectImage() - ) + p(trans.streamer.maxSize(s"${lila.memo.PicfitApi.uploadMaxMb}MB.")), + form3.file.selectImage() ) + ) - def grouping(form: Form[RelayTourForm.Data])(using Context) = - form3.split(cls := "relay-form__grouping")( - form3.group( - form("grouping"), - "Optional: assign tournaments to a group", - half = true - )(form3.textarea(_)(rows := 5)), - div(cls := "form-group form-half form-help")( // do not translate - "First line is the group name. Subsequent lines are the tournament IDs and names in the group. Names are facultative and only used for display in this textarea.", - br, - "You can add, remove, and re-order tournaments; and you can rename the group.", - br, - "Example:", - pre("""Youth Championship 2024 + def grouping(form: Form[RelayTourForm.Data])(using Context) = + form3.split(cls := "relay-form__grouping")( + form3.group( + form("grouping"), + "Optional: assign tournaments to a group", + half = true + )(form3.textarea(_)(rows := 5)), + div(cls := "form-group form-half form-help")( // do not translate + "First line is the group name. Subsequent lines are the tournament IDs and names in the group. Names are facultative and only used for display in this textarea.", + br, + "You can add, remove, and re-order tournaments; and you can rename the group.", + br, + "Example:", + pre("""Youth Championship 2024 tour1-id Youth Championship 2024 | G20 tour2-id Youth Championship 2024 | G16 """) - ) ) + ) diff --git a/ui/analyse/src/study/relay/relayManagerView.ts b/ui/analyse/src/study/relay/relayManagerView.ts index e03f810adb5f8..21d7fbb86f598 100644 --- a/ui/analyse/src/study/relay/relayManagerView.ts +++ b/ui/analyse/src/study/relay/relayManagerView.ts @@ -14,17 +14,7 @@ export default function (ctrl: RelayCtrl, study: StudyCtrl): MaybeVNode { ? h('div.relay-admin', { hook: onInsert(_ => site.asset.loadCssPath('analyse.relay-admin')) }, [ h('h2', [ h('span.text', { attrs: dataIcon(licon.RadioTower) }, 'Broadcast manager'), - h('span', [ - h('a', { attrs: { href: `/broadcast/round/${ctrl.id}/edit`, 'data-icon': licon.Gear } }), - ' ', - h('a', { - attrs: { - href: `/broadcast/${ctrl.data.tour.id}/new`, - title: 'New round', - 'data-icon': licon.PlusButton, - }, - }), - ]), + h('a', { attrs: { href: `/broadcast/round/${ctrl.id}/edit`, 'data-icon': licon.Gear } }), ]), ctrl.data.sync?.url || ctrl.data.sync?.ids ? (ctrl.data.sync.ongoing ? stateOn : stateOff)(ctrl) diff --git a/ui/bits/css/relay/_form.scss b/ui/bits/css/relay/_form.scss index 5e2d055418f52..48e0c22d62b45 100644 --- a/ui/bits/css/relay/_form.scss +++ b/ui/bits/css/relay/_form.scss @@ -54,3 +54,29 @@ @extend %flex-between; } } + +@include mq-subnav-side { + .relay-form__subnav__group { + display: block; + margin-bottom: 1em; + font-weight: bold; + color: $c-font-dimmer; + } + .relay-form__subnav__tour-parent { + background: $c-bg-box; + border-radius: $box-radius-size 0 0 $box-radius-size; + } + .subnav { + width: 30ch; + a { + white-space: wrap; + } + .subnav__subitem { + margin-left: 1em; + } + .subnav__subitem2 { + margin-left: 2em; + letter-spacing: 0px; + } + } +} diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 0f0276e1f266d..f0cc93d2c88f6 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -58,7 +58,7 @@ z-index: $z-link-overlay-2; visibility: hidden; background: $c-bg-header-dropdown; - border-radius: 3px 0 3px 3px; + border-radius: $box-radius-size 0 $box-radius-size $box-radius-size; position: absolute; top: 3rem; right: 0; @@ -72,10 +72,10 @@ font-size: 1.4em; } &:first-child { - border-radius: 3px 0 0 0; + border-radius: $box-radius-size 0 0 0; } &:last-child { - border-radius: 0 0 3px 3px; + @extend %box-radius-bottom; } &:hover { background: $c-primary; From bd12c53962caabce5bc043d8c81a04dd2655cd37 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 11:58:41 +0200 Subject: [PATCH 094/168] better permission denied pages for broadcast forms --- app/controllers/RelayRound.scala | 13 +++++++++++-- app/controllers/RelayTour.scala | 9 +++++---- modules/relay/src/main/RelayApi.scala | 8 +++++--- modules/relay/src/main/ui/FormUi.scala | 12 ++++++++++++ 4 files changed, 33 insertions(+), 9 deletions(-) diff --git a/app/controllers/RelayRound.scala b/app/controllers/RelayRound.scala index c1c2dade5058d..3fd3f37d5e7b2 100644 --- a/app/controllers/RelayRound.scala +++ b/app/controllers/RelayRound.scala @@ -52,8 +52,17 @@ final class RelayRound( } def edit(id: RelayRoundId) = Auth { ctx ?=> me ?=> - FoundPage(env.relay.api.formNavigation(id)): (round, nav) => - views.relay.form.round.edit(round, env.relay.roundForm.edit(round), nav) + env.relay.api + .byIdAndContributor(id) + .flatMap: + case None => + Found(env.relay.api.formNavigation(id)): (_, nav) => + Forbidden.page(views.relay.form.noAccess(nav)) + case Some(rt) => + env.relay.api + .formNavigation(rt) + .flatMap: (round, nav) => + Ok.page(views.relay.form.round.edit(round, env.relay.roundForm.edit(round), nav)) } def update(id: RelayRoundId) = AuthOrScopedBody(_.Study.Write) { ctx ?=> me ?=> diff --git a/app/controllers/RelayTour.scala b/app/controllers/RelayTour.scala index ec77fba160af9..1ea846a2b2e83 100644 --- a/app/controllers/RelayTour.scala +++ b/app/controllers/RelayTour.scala @@ -215,10 +215,11 @@ final class RelayTour(env: Env, apiC: => Api) extends LilaController(env): id: RelayTourId )(f: (FormNavigation) => Fu[Result])(using Context, Me): Fu[Result] = WithTour(id): tour => - env.relay.api - .canUpdate(tour) - .elseNotFound: - env.relay.api.formNavigation(tour).flatMap(f) + for + canUpdate <- env.relay.api.canUpdate(tour) + nav <- env.relay.api.formNavigation(tour) + res <- if canUpdate then f(nav) else Forbidden.page(views.relay.form.noAccess(nav)) + yield res private[controllers] def rateLimitCreation( fail: => Fu[Result] diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index a88a458f4069b..c7ab5f3e6fbb4 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -50,9 +50,11 @@ final class RelayApi( relay.withTour(tour) def formNavigation(id: RelayRoundId)(using me: Me): Fu[Option[(RelayRound, ui.FormNavigation)]] = - byIdAndContributor(id).flatMapz: rt => - formNavigation(rt.tour).map: nav => - (rt.round, nav.copy(round = rt.round.id.some)).some + byIdWithTour(id).flatMapz(rt => formNavigation(rt).dmap(some)) + + def formNavigation(rt: RelayRound.WithTour)(using me: Me): Fu[(RelayRound, ui.FormNavigation)] = + formNavigation(rt.tour).map: nav => + (rt.round, nav.copy(round = rt.round.id.some)) def formNavigation(tour: RelayTour)(using me: Me): Fu[ui.FormNavigation] = for group <- withTours.get(tour.id) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index ae17b19facc46..8423f18f8c3d4 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -62,6 +62,18 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): ) ) + def noAccess(nav: FormNavigation)(using Context) = + Page("Insufficient permissions") + .css("bits.relay.form") + .wrap: body => + main(cls := "page page-menu")( + navigationMenu(nav), + div(cls := "page-menu__content box box-pad")( + boxTop(h1("Insufficient permissions")), + p("You are not allowed to edit this broadcast or round.") + ) + ) + object round: private def page(title: String, nav: FormNavigation)(using Context) = From 6ea27c619742d1e50f2cc27dcfaacc46a8a7b89a Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 14:32:35 +0200 Subject: [PATCH 095/168] guess new broadcast round settings based on previous 2 rounds --- modules/relay/src/main/RelayRoundForm.scala | 49 +++++++++++++++++---- 1 file changed, 41 insertions(+), 8 deletions(-) diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index ab4c1c246cc0d..9a952c8f55f90 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -38,24 +38,57 @@ final class RelayRoundForm(using mode: Mode): )(Data.apply)(unapply) .verifying("This source requires a round number. See the new form field below.", !_.roundMissing) - def create(trs: RelayTour.WithRounds) = Form { + def create(trs: RelayTour.WithRounds) = Form( roundMapping .verifying( s"Maximum rounds per tournament: ${RelayTour.maxRelays}", _ => trs.rounds.sizeIs < RelayTour.maxRelays ) - }.fill( - Data( - name = RelayRound.Name(s"Round ${trs.rounds.size + 1}"), - caption = none, - syncUrlRound = Some(trs.rounds.size + 1) - ) - ) + ).fill(fillFromPrevRounds(trs.rounds)) def edit(r: RelayRound) = Form(roundMapping).fill(Data.make(r)) object RelayRoundForm: + def fillFromPrevRounds(rounds: List[RelayRound]): Data = + val prevs: Option[(RelayRound, RelayRound)] = rounds.reverse match + case a :: b :: _ => (a, b).some + case _ => none + val prev: Option[RelayRound] = prevs.map(_._1) + val roundNumberRegex = """([^\d]*)(\d{1,2})([^\d]*)""".r + val roundNumberIn: String => Option[Int] = + case roundNumberRegex(_, n, _) => n.toIntOption + case _ => none + def replaceRoundNumber(s: String, n: Int): String = + roundNumberRegex.replaceAllIn(s, m => s"${m.group(1)}${n}${m.group(3)}") + val prevNumber: Option[Int] = prev.flatMap(p => roundNumberIn(p.name.value)) + val nextNumber = (prevNumber | rounds.size) + 1 + val guessName = for + n <- prevNumber + if prevs + .map(_._2) + .fold(true): old => + roundNumberIn(old.name.value).contains(n - 1) + p <- prev + yield replaceRoundNumber(p.name.value, nextNumber) + val nextUrl = + prev.flatMap(_.sync.upstream).flatMap(_.asUrl).map(_.withRound).filter(_.round.isDefined).map(_.url) + val guessDate = for + (prev, old) <- prevs + prevDate <- prev.startsAt + oldDate <- old.startsAt + delta = prevDate.toEpochMilli - oldDate.toEpochMilli + yield prevDate.plusMillis(delta) + Data( + name = RelayRound.Name(guessName | s"Round ${nextNumber}"), + caption = none, + syncUrl = nextUrl, + syncUrlRound = nextUrl.isDefined.option(nextNumber), + startsAt = guessDate, + period = prev.flatMap(_.sync.period), + delay = prev.flatMap(_.sync.delay) + ) + case class GameIds(ids: List[GameId]) private def toGameIds(ids: String): Option[GameIds] = From e757228a2b78acaa23716b4bc9191b3c9cbc821b Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 14:39:57 +0200 Subject: [PATCH 096/168] redirect to round edit --- app/controllers/RelayRound.scala | 8 ++++++-- modules/relay/src/main/ui/FormUi.scala | 1 + 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/app/controllers/RelayRound.scala b/app/controllers/RelayRound.scala index 3fd3f37d5e7b2..69d57d93d6929 100644 --- a/app/controllers/RelayRound.scala +++ b/app/controllers/RelayRound.scala @@ -45,7 +45,7 @@ final class RelayRound( .create(setup, tour) .flatMap: rt => negotiate( - Redirect(routes.RelayRound.show(tour.slug, rt.relay.slug, rt.relay.id)), + Redirect(routes.RelayRound.edit(rt.relay.id)).flashSuccess, JsonOk(env.relay.jsonView.myRound(rt)) ) ) @@ -86,7 +86,11 @@ final class RelayRound( jsonFormError(err) ) }, - rt => negotiate(Redirect(rt.path), JsonOk(env.relay.jsonView.withUrl(rt, withTour = true))) + rt => + negotiate( + Redirect(routes.RelayRound.edit(id)).flashSuccess, + JsonOk(env.relay.jsonView.withUrl(rt, withTour = true)) + ) ) } diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 8423f18f8c3d4..570253f0650c1 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -105,6 +105,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): val rt = r.withTour(nav.tour) frag( boxTop(h1(a(href := rt.path)(rt.fullName))), + standardFlash, inner(form, routes.RelayRound.update(r.id), nav.tour, create = false), div(cls := "relay-form__actions")( postForm(action := routes.RelayRound.reset(r.id))( From 40777e754b52433833aaa515ff72bac89173a0d1 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 15:03:25 +0200 Subject: [PATCH 097/168] remove unused argument --- modules/relay/src/main/ui/RelayUi.scala | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/modules/relay/src/main/ui/RelayUi.scala b/modules/relay/src/main/ui/RelayUi.scala index 5e00bb851d9d8..7d551ed7ae657 100644 --- a/modules/relay/src/main/ui/RelayUi.scala +++ b/modules/relay/src/main/ui/RelayUi.scala @@ -24,8 +24,7 @@ final class RelayUi(helpers: Helpers)( rt: lila.relay.RelayRound.WithTourAndStudy, data: lila.relay.JsonView.JsData, chatOption: Option[(JsObject, Frag)], - socketVersion: lila.core.socket.SocketVersion, - crossSiteIsolation: Boolean = true + socketVersion: lila.core.socket.SocketVersion )(using ctx: Context) = Page(rt.fullName) .css("analyse.relay") From e51abf0f31e30467ef47d8fc372e00c0708d4bbb Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 15:10:14 +0200 Subject: [PATCH 098/168] always guess url round number for first 2 rounds --- modules/relay/src/main/RelayRoundForm.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index 9a952c8f55f90..e036288a3af70 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -83,7 +83,7 @@ object RelayRoundForm: name = RelayRound.Name(guessName | s"Round ${nextNumber}"), caption = none, syncUrl = nextUrl, - syncUrlRound = nextUrl.isDefined.option(nextNumber), + syncUrlRound = (prevs.isEmpty || nextUrl.isDefined).option(nextNumber), startsAt = guessDate, period = prev.flatMap(_.sync.period), delay = prev.flatMap(_.sync.delay) From cc8dc78957e928c1da5844697656fee6e9de5c1d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 15:21:21 +0200 Subject: [PATCH 099/168] rewrite with for --- modules/security/src/main/EmailChange.scala | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/modules/security/src/main/EmailChange.scala b/modules/security/src/main/EmailChange.scala index a2cddab1b4631..8dfe31f52bb44 100644 --- a/modules/security/src/main/EmailChange.scala +++ b/modules/security/src/main/EmailChange.scala @@ -50,10 +50,11 @@ ${trans.common_orPaste.txt()} // also returns the previous email address def confirm(token: String): Fu[Option[(Me, Option[EmailAddress])]] = tokener.read(token).dmap(_.flatten).flatMapz { case TokenPayload(userId, email) => - userRepo.email(userId).flatMap { previous => - (userRepo.setEmail(userId, email).recoverDefault >> userRepo.me(userId)) - .map2(_ -> previous) - } + for + previous <- userRepo.email(userId) + _ <- userRepo.setEmail(userId, email).recoverDefault + me <- userRepo.me(userId) + yield me.map(_ -> previous) } case class TokenPayload(userId: UserId, email: EmailAddress) From 34ba5eff31305c9f4d91090d2767c4e968280d0f Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 15:23:20 +0200 Subject: [PATCH 100/168] log email change --- modules/security/src/main/EmailChange.scala | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/modules/security/src/main/EmailChange.scala b/modules/security/src/main/EmailChange.scala index 8dfe31f52bb44..910345216ad88 100644 --- a/modules/security/src/main/EmailChange.scala +++ b/modules/security/src/main/EmailChange.scala @@ -54,7 +54,9 @@ ${trans.common_orPaste.txt()} previous <- userRepo.email(userId) _ <- userRepo.setEmail(userId, email).recoverDefault me <- userRepo.me(userId) - yield me.map(_ -> previous) + yield + logger.info(s"Set email for $id: $email") + me.map(_ -> previous) } case class TokenPayload(userId: UserId, email: EmailAddress) From 0c11e70b041ffe78dbb07e975e9b30b42ca85698 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 15:46:57 +0200 Subject: [PATCH 101/168] fix lobby ublog bg --- ui/lobby/css/_lobby.scss | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/ui/lobby/css/_lobby.scss b/ui/lobby/css/_lobby.scss index 3c4fc0eed7691..de05fd279966d 100644 --- a/ui/lobby/css/_lobby.scss +++ b/ui/lobby/css/_lobby.scss @@ -105,8 +105,11 @@ body { @extend %page-text-shadow !optional; font-size: 1.2em; } - .ublog-post-card:hover { - box-shadow: none; + .ublog-post-card { + background: $c-bg-box; + &:hover { + box-shadow: none; + } } } From 777f9bdfcbbc351a6067f13bea62910f02214e94 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 16:14:22 +0200 Subject: [PATCH 102/168] lobby pool buttons without borders --- ui/common/css/theme/_default.scss | 1 + ui/lobby/css/app/_pool.scss | 10 ++++++---- 2 files changed, 7 insertions(+), 4 deletions(-) diff --git a/ui/common/css/theme/_default.scss b/ui/common/css/theme/_default.scss index 33743f19086f3..2f4dd98fad6b3 100644 --- a/ui/common/css/theme/_default.scss +++ b/ui/common/css/theme/_default.scss @@ -122,4 +122,5 @@ $c-clearer: #fff; --c-good-move: #{$c-good-move}; --c-brilliant: #{$c-brilliant}; --c-interesting: #{$c-interesting}; + --c-pool-button: #{hsla($site-hue, 7%, 19%, 66%)}; } diff --git a/ui/lobby/css/app/_pool.scss b/ui/lobby/css/app/_pool.scss index fdfb119465b8f..18bf734a9ae3e 100644 --- a/ui/lobby/css/app/_pool.scss +++ b/ui/lobby/css/app/_pool.scss @@ -5,7 +5,9 @@ grid-template-columns: repeat(3, 1fr); grid-template-rows: repeat(4, 1fr); grid-gap: 9px; - padding: 9px; + padding-top: 9px; + background-color: transparent !important; + box-shadow: none; @include fluid-size('font-size', 14px, 25px); @@ -15,18 +17,18 @@ justify-content: center; align-items: center; cursor: pointer; - border: $border; - background: $m-font--fade-95; + background: $c-pool-button; @include if-light { background: $m-bg--fade-50; } - color: $c-font; + color: $c-font-dim; @include transition; &:hover { + color: $c-font; background: $m-accent--fade-80 !important; opacity: 1; } From 1a9c3431d496412c69d1c120d0888ca43afe9d22 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 16:23:10 +0200 Subject: [PATCH 103/168] fill round caption --- modules/relay/src/main/RelayRoundForm.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index e036288a3af70..a4578ba10f03a 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -81,7 +81,7 @@ object RelayRoundForm: yield prevDate.plusMillis(delta) Data( name = RelayRound.Name(guessName | s"Round ${nextNumber}"), - caption = none, + caption = prev.flatMap(_.caption), syncUrl = nextUrl, syncUrlRound = (prevs.isEmpty || nextUrl.isDefined).option(nextNumber), startsAt = guessDate, From 346d934da45d1badae484acb65fc2cd832d4e7e6 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 16:36:34 +0200 Subject: [PATCH 104/168] stop teaching study admins how to use broadcasts --- modules/relay/src/main/ui/FormUi.scala | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 570253f0650c1..eda99f52892ed 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -129,12 +129,14 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): ) = val isLcc = form("syncUrl").value.exists(RelayRound.Sync.UpstreamUrl.LccRegex.matches) postForm(cls := "form3", action := url)( - div(cls := "form-group")( - ui.howToUse, - (create && t.createdAt.isBefore(nowInstant.minusMinutes(1))).option: - p(dataIcon := Icon.InfoCircle, cls := "text"): - trb.theNewRoundHelp() - ), + (!Granter.opt(_.StudyAdmin)).option: + div(cls := "form-group")( + div(cls := "form-group")(ui.howToUse), + (create && t.createdAt.isBefore(nowInstant.minusMinutes(1))).option: + p(dataIcon := Icon.InfoCircle, cls := "text"): + trb.theNewRoundHelp() + ) + , form3.globalError(form), form3.split( form3.group(form("name"), trb.roundName(), half = true)(form3.input(_)(autofocus)), @@ -270,7 +272,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): private def inner(form: Form[RelayTourForm.Data], tg: Option[RelayTour.WithGroupTours])(using Context) = frag( - div(cls := "form-group")(ui.howToUse), + (!Granter.opt(_.StudyAdmin)).option(div(cls := "form-group")(ui.howToUse)), form3.globalError(form), form3.split( form3.group(form("name"), trb.tournamentName(), half = true)(form3.input(_)(autofocus)), From f2e73393316108358d0d53dd22da4eacab2b5654 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 16:38:10 +0200 Subject: [PATCH 105/168] use translation --- modules/relay/src/main/ui/FormUi.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index eda99f52892ed..9dd2607ec0b72 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -46,7 +46,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): href := routes.RelayRound.create(nav.tour.id), cls := List("subnav__subitem text" -> true, "active" -> nav.newRound), dataIcon := Icon.PlusButton - )("New round") + )(trb.addRound()) ) ) lila.ui.bits.pageMenuSubnav( @@ -92,7 +92,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): boxTop( h1( a(href := routes.RelayTour.edit(nav.tour.id))(nav.tour.name), - " • ", + " / ", trans.broadcast.addRound() ) ), From d5186650afa75502ad8c3e1be07cc23ce191f8ef Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 16:43:20 +0200 Subject: [PATCH 106/168] broadcast status icons in form navigation --- modules/relay/src/main/ui/FormUi.scala | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 9dd2607ec0b72..5bc04ed385ad6 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -40,7 +40,11 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): nav.rounds.map: r => a( href := routes.RelayRound.edit(r.id), - cls := List("subnav__subitem" -> true, "active" -> nav.round.has(r.id)) + cls := List("subnav__subitem text" -> true, "active" -> nav.round.has(r.id)), + dataIcon := r.stateHash.match + case (_, true) => Icon.Checkmark + case (true, false) => Icon.DiscBig + case _ => Icon.DiscOutline )(r.name), a( href := routes.RelayRound.create(nav.tour.id), From 25e9fd2be9bd2eb59dc5d28b6f610a4a6ba04dd3 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 16:52:24 +0200 Subject: [PATCH 107/168] fix broadcast form navigation on new round page --- app/controllers/RelayRound.scala | 22 +++++++++++----------- modules/relay/src/main/RelayApi.scala | 6 +++--- modules/relay/src/main/ui/FormUi.scala | 7 ++----- 3 files changed, 16 insertions(+), 19 deletions(-) diff --git a/app/controllers/RelayRound.scala b/app/controllers/RelayRound.scala index 69d57d93d6929..57586cd408e68 100644 --- a/app/controllers/RelayRound.scala +++ b/app/controllers/RelayRound.scala @@ -20,29 +20,29 @@ final class RelayRound( def form(tourId: RelayTourId) = Auth { ctx ?=> _ ?=> NoLameOrBot: - WithTourAndRoundsCanUpdate(tourId): trs => + WithNavigationCanUpdate(tourId): nav => Ok.page: - views.relay.form.round.create(env.relay.roundForm.create(trs), FormNavigation(trs, none)) + views.relay.form.round + .create(env.relay.roundForm.create(nav.tourWithRounds), nav) } def create(tourId: RelayTourId) = AuthOrScopedBody(_.Study.Write) { ctx ?=> me ?=> NoLameOrBot: - WithTourAndRoundsCanUpdate(tourId): trs => - val tour = trs.tour + WithNavigationCanUpdate(tourId): nav => def whenRateLimited = negotiate( - Redirect(routes.RelayTour.show(tour.slug, tour.id)), + Redirect(routes.RelayTour.show(nav.tour.slug, nav.tour.id)), rateLimited ) - bindForm(env.relay.roundForm.create(trs))( + bindForm(env.relay.roundForm.create(nav.tourWithRounds))( err => negotiate( - BadRequest.page(views.relay.form.round.create(err, FormNavigation(trs, none))), + BadRequest.page(views.relay.form.round.create(err, nav)), jsonFormError(err) ), setup => rateLimitCreation(whenRateLimited): env.relay.api - .create(setup, tour) + .create(setup, nav.tour) .flatMap: rt => negotiate( Redirect(routes.RelayRound.edit(rt.relay.id)).flashSuccess, @@ -187,14 +187,14 @@ final class RelayRound( )(using Context): Fu[Result] = Found(env.relay.api.tourById(id))(f) - private def WithTourAndRoundsCanUpdate(id: RelayTourId)( - f: TourModel.WithRounds => Fu[Result] + private def WithNavigationCanUpdate(id: RelayTourId)( + f: FormNavigation => Fu[Result] )(using ctx: Context): Fu[Result] = WithTour(id): tour => ctx.me .soUse { env.relay.api.canUpdate(tour) } .elseNotFound: - env.relay.api.withRounds(tour).flatMap(f) + env.relay.api.formNavigation(tour).flatMap(f) private def doShow(rt: RoundModel.WithTour, oldSc: lila.study.Study.WithChapter, embed: Option[UserStr])( using ctx: Context diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index c7ab5f3e6fbb4..edeb6d4087f53 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -49,14 +49,14 @@ final class RelayApi( if study.canContribute(me) || Granter(_.StudyAdmin) => relay.withTour(tour) - def formNavigation(id: RelayRoundId)(using me: Me): Fu[Option[(RelayRound, ui.FormNavigation)]] = + def formNavigation(id: RelayRoundId): Fu[Option[(RelayRound, ui.FormNavigation)]] = byIdWithTour(id).flatMapz(rt => formNavigation(rt).dmap(some)) - def formNavigation(rt: RelayRound.WithTour)(using me: Me): Fu[(RelayRound, ui.FormNavigation)] = + def formNavigation(rt: RelayRound.WithTour): Fu[(RelayRound, ui.FormNavigation)] = formNavigation(rt.tour).map: nav => (rt.round, nav.copy(round = rt.round.id.some)) - def formNavigation(tour: RelayTour)(using me: Me): Fu[ui.FormNavigation] = for + def formNavigation(tour: RelayTour): Fu[ui.FormNavigation] = for group <- withTours.get(tour.id) rounds <- roundRepo.byTourOrdered(tour.id) yield ui.FormNavigation(group, tour, rounds, none) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 5bc04ed385ad6..a333161ef3460 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -15,11 +15,8 @@ case class FormNavigation( round: Option[RelayRoundId], newRound: Boolean = false ): - def tourWithGroup = RelayTour.WithGroupTours(tour, group) - -object FormNavigation: - def apply(trs: RelayTour.WithRounds, roundId: Option[RelayRoundId]): FormNavigation = - FormNavigation(none, trs.tour, trs.rounds, roundId) + def tourWithGroup = RelayTour.WithGroupTours(tour, group) + def tourWithRounds = RelayTour.WithRounds(tour, rounds) final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): import helpers.{ *, given } From 1beb0fac39e51cb20609ea967ece4c688c0b2bd8 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 17:28:06 +0200 Subject: [PATCH 108/168] scala tweak --- modules/relay/src/main/ui/FormUi.scala | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index a333161ef3460..5f92c7b225353 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -38,10 +38,11 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): a( href := routes.RelayRound.edit(r.id), cls := List("subnav__subitem text" -> true, "active" -> nav.round.has(r.id)), - dataIcon := r.stateHash.match - case (_, true) => Icon.Checkmark - case (true, false) => Icon.DiscBig - case _ => Icon.DiscOutline + dataIcon := ( + if r.finished then Icon.Checkmark + else if r.hasStarted then Icon.DiscBig + else Icon.DiscOutline + ) )(r.name), a( href := routes.RelayRound.create(nav.tour.id), From 08248f4b18b2afd38642d83100a88582cb09f86d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 18:49:55 +0200 Subject: [PATCH 109/168] fix user dropdown RTL - closes #15528 --- ui/bits/css/user/_show.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index f0cc93d2c88f6..d4e087579654a 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -61,7 +61,7 @@ border-radius: $box-radius-size 0 $box-radius-size $box-radius-size; position: absolute; top: 3rem; - right: 0; + @include inline-end(0); a { width: 20em; display: block; From 0a178ae88836a87916fdace97abc5d1c92ce5537 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 19:13:20 +0200 Subject: [PATCH 110/168] fix broadcast selector dropped shadow I couldn't figure out how to achieve it while keeping the text overflow ellipsis --- ui/analyse/css/study/relay/_tour.scss | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ui/analyse/css/study/relay/_tour.scss b/ui/analyse/css/study/relay/_tour.scss index 2f15dc0c698db..efaa4b2d0d335 100644 --- a/ui/analyse/css/study/relay/_tour.scss +++ b/ui/analyse/css/study/relay/_tour.scss @@ -165,10 +165,10 @@ $hover-bg: $m-primary_bg--mix-30; } &__mselect { - @extend %ellipsis; position: unset; flex: auto; @include fluid-size('font-size', 12px, 20px); + white-space: wrap; &.mselect__active { overflow: visible; } @@ -193,6 +193,7 @@ $hover-bg: $m-primary_bg--mix-30; &__status { font-size: 1rem; margin-inline-end: 1ch; + white-space: nowrap; } &__label { .round-state { From 27b5238e1614e668715f15a8d8cab6d5892abef7 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 20:44:06 +0200 Subject: [PATCH 111/168] optimize relay stats storage by only recording new values --- modules/relay/src/main/RelayStatsApi.scala | 29 +++++++++++++++++----- 1 file changed, 23 insertions(+), 6 deletions(-) diff --git a/modules/relay/src/main/RelayStatsApi.scala b/modules/relay/src/main/RelayStatsApi.scala index 311a9ae6793cb..e02458ab49fbf 100644 --- a/modules/relay/src/main/RelayStatsApi.scala +++ b/modules/relay/src/main/RelayStatsApi.scala @@ -1,6 +1,7 @@ package lila.relay import lila.db.dsl.{ *, given } +import reactivemongo.api.bson.BSONInteger object RelayStats: type Minute = Int @@ -47,13 +48,29 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc private def record(): Funit = for crowds <- fetchRoundCrowds nowMinutes = nowSeconds / 60 - update = colls.stats.update(ordered = false) - elements <- crowds.sequentially: (roundId, crowd) => - update.element( - q = $id(roundId), - u = $push("d" -> $doc("$each" -> $arr(nowMinutes, crowd))), - upsert = true + lastValuesDocs <- colls.stats.aggregateList(crowds.size): framework => + import framework.* + Match($inIds(crowds.map(_._1))) -> List( + Project($doc("last" -> $doc("$arrayElemAt" -> $arr("$d", -1)))) ) + lastValues = for + doc <- lastValuesDocs + last <- doc.getAsOpt[Crowd]("last") + id <- doc.getAsOpt[RelayRoundId]("_id") + yield (id, last) + lastValuesMap = lastValues.toMap + update = colls.stats.update(ordered = false) + elementOpts <- crowds.sequentially: (roundId, crowd) => + val lastValue = ~lastValuesMap.get(roundId) + (lastValue != crowd).so: + update + .element( + q = $id(roundId), + u = $push("d" -> $doc("$each" -> $arr(nowMinutes, crowd))), + upsert = true + ) + .dmap(some) + elements = elementOpts.flatten _ <- elements.nonEmpty.so(update.many(elements).void) yield () From fd01f5652dd0592f75704b4c475cb58445578045 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 20:44:44 +0200 Subject: [PATCH 112/168] extract ui/analyse glyphs to ui/chess --- pnpm-lock.yaml | 3 +++ ui/analyse/src/autoShape.ts | 4 +-- ui/chess/package.json | 3 ++- ui/{analyse => chess}/src/glyphs.ts | 38 ++++++++++++----------------- 4 files changed, 22 insertions(+), 26 deletions(-) rename ui/{analyse => chess}/src/glyphs.ts (90%) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index d3e1e761fb0ff..47e38a009a786 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -292,6 +292,9 @@ importers: ui/chess: dependencies: + chessops: + specifier: ^0.14.1 + version: 0.14.1 common: specifier: workspace:* version: link:../common diff --git a/ui/analyse/src/autoShape.ts b/ui/analyse/src/autoShape.ts index 38c0521d8d5cc..c47f9f8133137 100644 --- a/ui/analyse/src/autoShape.ts +++ b/ui/analyse/src/autoShape.ts @@ -4,7 +4,7 @@ import { winningChances } from 'ceval'; import * as cg from 'chessground/types'; import { opposite } from 'chessground/util'; import { DrawModifiers, DrawShape } from 'chessground/draw'; -import { annotationShapes } from './glyphs'; +import { annotationShapes } from 'chess/glyphs'; import AnalyseCtrl from './ctrl'; const pieceDrop = (key: cg.Key, role: cg.Role, color: Color): DrawShape => ({ @@ -122,7 +122,7 @@ export function compute(ctrl: AnalyseCtrl): DrawShape[] { } }); } - shapes = shapes.concat(annotationShapes(ctrl)); + if (ctrl.showMoveAnnotation()) shapes = shapes.concat(annotationShapes(ctrl.node)); if (ctrl.showVariationArrows()) hiliteVariations(ctrl, shapes); return shapes; } diff --git a/ui/chess/package.json b/ui/chess/package.json index d7fb184595928..cf863743a7091 100644 --- a/ui/chess/package.json +++ b/ui/chess/package.json @@ -25,6 +25,7 @@ "license": "AGPL-3.0-or-later", "dependencies": { "common": "workspace:*", - "snabbdom": "3.5.1" + "snabbdom": "3.5.1", + "chessops": "^0.14.1" } } diff --git a/ui/analyse/src/glyphs.ts b/ui/chess/src/glyphs.ts similarity index 90% rename from ui/analyse/src/glyphs.ts rename to ui/chess/src/glyphs.ts index e77172d6b2b62..47ae3409b6790 100644 --- a/ui/analyse/src/glyphs.ts +++ b/ui/chess/src/glyphs.ts @@ -1,11 +1,9 @@ import { parseUci, makeSquare, squareRank } from 'chessops/util'; -import AnalyseCtrl from './ctrl'; import { DrawShape } from 'chessground/draw'; -export function annotationShapes(ctrl: AnalyseCtrl): DrawShape[] { - const shapes: DrawShape[] = []; - const { uci, glyphs, san } = ctrl.node; - if (ctrl.showMoveAnnotation() && uci && san && glyphs && glyphs.length > 0) { +export function annotationShapes(node: Tree.Node): DrawShape[] { + const { uci, glyphs, san } = node; + if (uci && san && glyphs?.[0]) { const move = parseUci(uci)!; const destSquare = san.startsWith('O-O') // castle, short or long ? squareRank(move.to) === 0 // white castle @@ -17,29 +15,23 @@ export function annotationShapes(ctrl: AnalyseCtrl): DrawShape[] { : 'g8' : makeSquare(move.to); const prerendered = glyphToSvg[glyphs[0].symbol]; - shapes.push({ - orig: destSquare, - brush: prerendered ? '' : undefined, - customSvg: prerendered ? { html: prerendered } : undefined, - label: prerendered ? undefined : { text: glyphs[0].symbol, fill: 'purple' }, - // keep some purple just to keep feedback forum on their toes - }); - } - return shapes; + return [ + { + orig: destSquare, + brush: prerendered ? '' : undefined, + customSvg: prerendered ? { html: prerendered } : undefined, + label: prerendered ? undefined : { text: glyphs[0].symbol, fill: 'purple' }, + // keep some purple just to keep feedback forum on their toes + }, + ]; + } else return []; } // We can render glyphs as text, but people are used to these SVGs as the "Big 5" glyphs // and right now they look better const prependDropShadow = (svgBase: string) => - ` - - - - - -` + - svgBase + - ''; + ` +${svgBase}`; // NOTE: // Base svg was authored with Inkscape. // On Inkscape, by using "Object to Path", text is converted to path, which enables consistent layout on browser. From 4584d4fa89050f7eb57aa21ea90782b35db131e7 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 20:58:19 +0200 Subject: [PATCH 113/168] fix puzzle good move color --- ui/common/css/theme/_default.scss | 4 ++-- ui/common/css/theme/_light.scss | 2 +- ui/tree/css/_tree.scss | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/ui/common/css/theme/_default.scss b/ui/common/css/theme/_default.scss index 2f4dd98fad6b3..237e629dc7303 100644 --- a/ui/common/css/theme/_default.scss +++ b/ui/common/css/theme/_default.scss @@ -33,7 +33,7 @@ $c-shade: hsl(0, 0%, 30%); $c-inaccuracy: hsl(202, 78%, 62%); $c-mistake: hsl(41, 100%, 45%); $c-blunder: hsl(0, 69%, 60%); -$c-good-move: hsl(130, 67%, 62%); +$c-good: $c-secondary; $c-brilliant: hsl(129, 71%, 45%); $c-interesting: hsl(307, 80%, 70%); $c-dark: #333; @@ -119,7 +119,7 @@ $c-clearer: #fff; --c-inaccuracy: #{$c-inaccuracy}; --c-mistake: #{$c-mistake}; --c-blunder: #{$c-blunder}; - --c-good-move: #{$c-good-move}; + --c-good-move: #{$c-good}; --c-brilliant: #{$c-brilliant}; --c-interesting: #{$c-interesting}; --c-pool-button: #{hsla($site-hue, 7%, 19%, 66%)}; diff --git a/ui/common/css/theme/_light.scss b/ui/common/css/theme/_light.scss index 0b3a5a689bbc0..7eb701b3a1729 100644 --- a/ui/common/css/theme/_light.scss +++ b/ui/common/css/theme/_light.scss @@ -24,7 +24,7 @@ $c-shade: hsl(0, 0%, 84%); $c-inaccuracy: hsl(202, 78%, 40%); $c-mistake: hsl(41, 100%, 35%); $c-blunder: hsl(0, 68%, 50%); -$c-good-move: hsl(130, 67%, 40%); +$c-good: $c-secondary; $c-brilliant: hsl(129, 71%, 30%); $c-interesting: hsl(307, 80%, 59%); $c-dimmer: #fff; diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index 406581eeb43db..129196d651bbe 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -37,9 +37,9 @@ } } &.good { - color: $c-good-move; + color: $c-good; &:hover { - background: $m-good-move_bg--mix-30; + background: $m-good_bg--mix-30; } } &.brilliant { From 4caa0b9e6ff3c238f1366df9cb759ac9506de16f Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 21:00:07 +0200 Subject: [PATCH 114/168] ui/puzzle typing --- ui/puzzle/src/view/tree.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/ui/puzzle/src/view/tree.ts b/ui/puzzle/src/view/tree.ts index 9aaf8def95f54..73c98bfb3c179 100644 --- a/ui/puzzle/src/view/tree.ts +++ b/ui/puzzle/src/view/tree.ts @@ -21,9 +21,9 @@ interface Glyph { symbol: string; } -const autoScroll = throttle(150, (ctrl: PuzzleCtrl, el) => { - const cont = el.parentNode; - const target = el.querySelector('.active'); +const autoScroll = throttle(150, (ctrl: PuzzleCtrl, el: HTMLElement) => { + const cont = el.parentNode as HTMLElement; + const target = el.querySelector('.active') as HTMLElement | null; if (!target) { cont.scrollTop = ctrl.path === treePath.root ? 0 : 99999; return; From fdacb97428b0fddb06d912a6609a735bbd2e4e32 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 21:09:27 +0200 Subject: [PATCH 115/168] fix puzzle move tree autoScroll after #15488 --- ui/puzzle/src/view/tree.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ui/puzzle/src/view/tree.ts b/ui/puzzle/src/view/tree.ts index 73c98bfb3c179..7664e9c323d9c 100644 --- a/ui/puzzle/src/view/tree.ts +++ b/ui/puzzle/src/view/tree.ts @@ -28,7 +28,8 @@ const autoScroll = throttle(150, (ctrl: PuzzleCtrl, el: HTMLElement) => { cont.scrollTop = ctrl.path === treePath.root ? 0 : 99999; return; } - cont.scrollTop = target.offsetTop - cont.offsetHeight / 2 + target.offsetHeight; + const targetOffset = target.getBoundingClientRect().y - el.getBoundingClientRect().y; + cont.scrollTop = targetOffset - cont.offsetHeight / 2 + target.offsetHeight; }); function pathContains(ctx: Ctx, path: Tree.Path): boolean { From e6b9beabf09c5f8ebc74f8d142a4bcd11c4d7b78 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 21:21:24 +0200 Subject: [PATCH 116/168] fix lobby preload lpools is the tab that's loaded most of the time and it has no background --- app/views/lobby/bits.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/views/lobby/bits.scala b/app/views/lobby/bits.scala index 6906963989872..ba745279203d9 100644 --- a/app/views/lobby/bits.scala +++ b/app/views/lobby/bits.scala @@ -8,7 +8,7 @@ object bits: val lobbyApp = div(cls := "lobby__app")( div(cls := "tabs-horiz")(span(nbsp)), - div(cls := "lobby__app__content") + div(cls := "lobby__app__content lpools") ) def underboards( From 98c335bf0367c1868015721276b082b360519fd5 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 21:22:14 +0200 Subject: [PATCH 117/168] fix css var --- ui/common/css/component/_lichess-pgn-viewer.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/common/css/component/_lichess-pgn-viewer.scss b/ui/common/css/component/_lichess-pgn-viewer.scss index 6d0e86d9c7990..30f918240adda 100644 --- a/ui/common/css/component/_lichess-pgn-viewer.scss +++ b/ui/common/css/component/_lichess-pgn-viewer.scss @@ -27,7 +27,7 @@ --c-lpv-bg-inaccuracy-hover: #{$m-inaccuracy_bg--mix-30}; --c-lpv-bg-mistake-hover: #{$m-mistake_bg--mix-30}; --c-lpv-bg-blunder-hover: #{$m-blunder_bg--mix-30}; - --c-lpv-bg-good-hover: #{$m-good-move_bg--mix-30}; + --c-lpv-bg-good-hover: #{$m-good_bg--mix-30}; --c-lpv-bg-brilliant-hover: #{$m-brilliant_bg--mix-30}; --c-lpv-bg-interesting-hover: #{$m-interesting_bg--mix-30}; From 59094af7e011f1c486f85981377c743f1c7bfb67 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 22:06:27 +0200 Subject: [PATCH 118/168] show puzzle glyphs over the board - closes #15379 --- ui/chess/src/glyphs.ts | 10 ++++++++-- ui/puzzle/src/autoShape.ts | 25 +++++++++++++++---------- ui/puzzle/src/ctrl.ts | 5 ++++- ui/puzzle/src/view/tree.ts | 4 +--- ui/tree/css/_tree.scss | 3 +++ 5 files changed, 31 insertions(+), 16 deletions(-) diff --git a/ui/chess/src/glyphs.ts b/ui/chess/src/glyphs.ts index 47ae3409b6790..0c3bf5cb6c77a 100644 --- a/ui/chess/src/glyphs.ts +++ b/ui/chess/src/glyphs.ts @@ -14,19 +14,25 @@ export function annotationShapes(node: Tree.Node): DrawShape[] { ? 'c8' : 'g8' : makeSquare(move.to); - const prerendered = glyphToSvg[glyphs[0].symbol]; + const symbol = glyphs[0].symbol; + const prerendered = glyphToSvg[symbol]; return [ { orig: destSquare, brush: prerendered ? '' : undefined, customSvg: prerendered ? { html: prerendered } : undefined, - label: prerendered ? undefined : { text: glyphs[0].symbol, fill: 'purple' }, + label: prerendered ? undefined : { text: symbol, fill: fills[symbol] || 'purple' }, // keep some purple just to keep feedback forum on their toes }, ]; } else return []; } +const fills: { [key: string]: string } = { + '✓': '#168226', + '✗': '#df5353', +}; + // We can render glyphs as text, but people are used to these SVGs as the "Big 5" glyphs // and right now they look better const prependDropShadow = (svgBase: string) => diff --git a/ui/puzzle/src/autoShape.ts b/ui/puzzle/src/autoShape.ts index 5e98eca04c4c1..66f9fa129fcc1 100644 --- a/ui/puzzle/src/autoShape.ts +++ b/ui/puzzle/src/autoShape.ts @@ -1,4 +1,5 @@ import { winningChances, CevalCtrl } from 'ceval'; +import { annotationShapes } from 'chess/glyphs'; import { DrawModifiers, DrawShape } from 'chessground/draw'; import { Api as CgApi } from 'chessground/api'; import { opposite, parseUci, makeSquare } from 'chessops/util'; @@ -42,14 +43,8 @@ export default function (opts: Opts): DrawShape[] { let nextBest: Uci | undefined = opts.nextNodeBest(); if (!nextBest && opts.ceval.enabled() && n.ceval) nextBest = n.ceval.pvs[0].moves[0]; if (nextBest) shapes = shapes.concat(makeAutoShapesFromUci(color, nextBest, 'paleBlue')); - if ( - opts.ceval.enabled() && - n.ceval && - n.ceval.pvs && - n.ceval.pvs[1] && - !(opts.threatMode() && n.threat && n.threat.pvs[2]) - ) { - n.ceval.pvs.forEach(function (pv) { + if (opts.ceval.enabled() && n.ceval?.pvs?.[1] && !(opts.threatMode() && n.threat?.pvs[2])) { + n.ceval.pvs.forEach(pv => { if (pv.moves[0] === nextBest) return; const shift = winningChances.povDiff(color, n.ceval!.pvs[0], pv); if (shift > 0.2 || isNaN(shift) || shift < 0) return; @@ -65,7 +60,7 @@ export default function (opts: Opts): DrawShape[] { if (opts.ceval.enabled() && opts.threatMode() && n.threat) { if (n.threat.pvs[1]) { shapes = shapes.concat(makeAutoShapesFromUci(opposite(color), n.threat.pvs[0].moves[0], 'paleRed')); - n.threat.pvs.slice(1).forEach(function (pv) { + n.threat.pvs.slice(1).forEach(pv => { const shift = winningChances.povDiff(opposite(color), pv, n.threat!.pvs[0]); if (shift > 0.2 || isNaN(shift) || shift < 0) return; shapes = shapes.concat( @@ -76,5 +71,15 @@ export default function (opts: Opts): DrawShape[] { }); } else shapes = shapes.concat(makeAutoShapesFromUci(opposite(color), n.threat.pvs[0].moves[0], 'red')); } - return shapes; + let glyph: Tree.Glyph | undefined; + switch (n.puzzle) { + case 'good': + case 'win': + glyph = { id: 7, name: 'good', symbol: '✓' }; + break; + case 'fail': + glyph = { id: 4, name: 'fail', symbol: '✗' }; + } + const withPuzzleGlyphs = glyph ? { ...n, glyphs: [glyph] } : n; + return shapes.concat(annotationShapes(withPuzzleGlyphs)); } diff --git a/ui/puzzle/src/ctrl.ts b/ui/puzzle/src/ctrl.ts index 7b5ccfc82f2ed..01d7ecfef6d9b 100755 --- a/ui/puzzle/src/ctrl.ts +++ b/ui/puzzle/src/ctrl.ts @@ -255,7 +255,10 @@ export default class PuzzleCtrl implements ParentCtrl { return config; }; - showGround = (g: CgApi): void => g.set(this.makeCgOpts()); + showGround = (g: CgApi): void => { + g.set(this.makeCgOpts()); + this.setAutoShapes(); + }; pluginMove = (orig: Key, dest: Key, role?: Role) => { if (role) this.playUserMove(orig, dest, role); diff --git a/ui/puzzle/src/view/tree.ts b/ui/puzzle/src/view/tree.ts index 7664e9c323d9c..493cc2b8eb8c6 100644 --- a/ui/puzzle/src/view/tree.ts +++ b/ui/puzzle/src/view/tree.ts @@ -100,9 +100,7 @@ function renderMainlineMoveOf(ctx: Ctx, node: Tree.Node, opts: RenderOpts): VNod return h('move', { attrs: { p: path }, class: classes }, renderMove(ctx, node)); } -function renderGlyph(glyph: Glyph): VNode { - return h('glyph', { attrs: { title: glyph.name } }, glyph.symbol); -} +const renderGlyph = (glyph: Glyph): VNode => h('glyph', { attrs: { title: glyph.name } }, glyph.symbol); function puzzleGlyph(ctx: Ctx, node: Tree.Node): MaybeVNode { switch (node.puzzle) { diff --git a/ui/tree/css/_tree.scss b/ui/tree/css/_tree.scss index 129196d651bbe..feb399aab18d4 100644 --- a/ui/tree/css/_tree.scss +++ b/ui/tree/css/_tree.scss @@ -38,18 +38,21 @@ } &.good { color: $c-good; + font-weight: bold; &:hover { background: $m-good_bg--mix-30; } } &.brilliant { color: $c-brilliant; + font-weight: bold; &:hover { background: $m-brilliant_bg--mix-30; } } &.interesting { color: $c-interesting; + font-weight: bold; &:hover { background: $m-interesting_bg--mix-30; } From 1fe946bdee80db6ae98ae156427340f210054834 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Gl=C3=B3rias?= Date: Mon, 17 Jun 2024 20:17:49 +0000 Subject: [PATCH 119/168] Use translete --- modules/web/src/main/ui/TopNav.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/web/src/main/ui/TopNav.scala b/modules/web/src/main/ui/TopNav.scala index 17e0aa3fb2de2..537b2a5366dc2 100644 --- a/modules/web/src/main/ui/TopNav.scala +++ b/modules/web/src/main/ui/TopNav.scala @@ -39,7 +39,7 @@ final class TopNav(helpers: Helpers): linkTitle(puzzleUrl, trans.site.puzzles()), div(role := "group")( a(href := puzzleUrl)(trans.site.puzzles()), - a(href := langHref(routes.Puzzle.themes))("Puzzle Themes"), + a(href := langHref(routes.Puzzle.themes))(trans.puzzle.puzzleThemes()), a(href := routes.Puzzle.dashboard(30, "home", none))(trans.puzzle.puzzleDashboard()), a(href := langHref(routes.Puzzle.streak))("Puzzle Streak"), a(href := langHref(routes.Storm.home))("Puzzle Storm"), From 21a65f0f308b185ce6aeb7e52dd67455bdfe7103 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Mon, 17 Jun 2024 23:55:20 +0200 Subject: [PATCH 120/168] bump broadcast max delay to 1h --- modules/relay/src/main/RelayDelay.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayDelay.scala b/modules/relay/src/main/RelayDelay.scala index bdda1e47590e8..ee90565ae3e9f 100644 --- a/modules/relay/src/main/RelayDelay.scala +++ b/modules/relay/src/main/RelayDelay.scala @@ -84,6 +84,6 @@ final private class RelayDelay(colls: RelayColls)(using Executor): _.flatMap(_.getAsOpt[PgnStr]("pgn")) private object RelayDelay: - val maxSeconds = Seconds(1800) + val maxSeconds = Seconds(60 * 60) private case class GamesSeenBy(games: Fu[RelayGames], seenBy: Set[RelayRoundId]) From 13e68ab3e557103143aa760ca1bc9f7a51b7d47e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=C4=90inh=20Ho=C3=A0ng=20Vi=E1=BB=87t?= <134517889+M-DinhHoangViet@users.noreply.github.com> Date: Tue, 18 Jun 2024 10:33:19 +0700 Subject: [PATCH 121/168] Align multi-line text next to icon --- ui/bits/css/user/_show.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index d4e087579654a..1e47ada555f60 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -64,7 +64,7 @@ @include inline-end(0); a { width: 20em; - display: block; + display: flex; padding: 0.7rem 1rem; color: $c-header-dropdown; &::before { From 8cbc395581a2c5e088bd01940c601c3841590501 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 07:46:21 +0200 Subject: [PATCH 122/168] New Crowdin updates (#15540) * New translations: site.xml (French) * New translations: site.xml (Dutch) --- translation/dest/site/fr-FR.xml | 3 ++- translation/dest/site/nl-NL.xml | 2 ++ 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/translation/dest/site/fr-FR.xml b/translation/dest/site/fr-FR.xml index 117f82cc56078..a53b18564860f 100644 --- a/translation/dest/site/fr-FR.xml +++ b/translation/dest/site/fr-FR.xml @@ -70,7 +70,8 @@ Promouvoir la variante En faire la variante principale Supprimer à partir d\'ici - Afficher les variantes + Fermer les variantes + Afficher les variantes Forcer la variante Copier le PGN de la variante Coup diff --git a/translation/dest/site/nl-NL.xml b/translation/dest/site/nl-NL.xml index 1182ab32dca32..6de9fd1f1ec8c 100644 --- a/translation/dest/site/nl-NL.xml +++ b/translation/dest/site/nl-NL.xml @@ -70,6 +70,8 @@ Promoveer variant Maak hoofdvariant Verwijder vanaf hier + Varianten verbergen + Varianten weergeven Forceer variatie Kopieer variatie PGN Zet From 156e1e148765ad9001587f97da690ddedee6ac41 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 09:04:36 +0200 Subject: [PATCH 123/168] earlier start for delayed broadcast rounds --- modules/relay/src/main/RelayApi.scala | 19 +++++++++++-------- 1 file changed, 11 insertions(+), 8 deletions(-) diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index edeb6d4087f53..5c9cc8eced7c3 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -392,17 +392,21 @@ final class RelayApi( .list[RelayRound]( $doc( "startsAt" - .$lt(nowInstant.plusMinutes(30)) // start 30 minutes early to fetch boards - .$gt(nowInstant.minusDays(1)), // bit late now + // start early to fetch boards + .$lt(nowInstant.plusSeconds(RelayDelay.maxSeconds.value)) + .$gt(nowInstant.minusDays(1)), // bit late now "startedAt".$exists(false), "sync.until".$exists(false) ) ) .flatMap: - _.sequentiallyVoid { relay => - logger.info(s"Automatically start $relay") - requestPlay(relay.id, v = true) - } + _.sequentiallyVoid: relay => + val earlyMinutes = Math.min(60, 30 + relay.sync.delay.so(_.value / 60)) + relay.startsAt + .exists(_.isBefore(nowInstant.plusMinutes(earlyMinutes))) + .so: + logger.info(s"Automatically start $relay") + requestPlay(relay.id, v = true) private[relay] def autoFinishNotSyncing: Funit = roundRepo.coll @@ -417,10 +421,9 @@ final class RelayApi( ) ) .flatMap: - _.sequentiallyVoid { relay => + _.sequentiallyVoid: relay => logger.info(s"Automatically finish $relay") update(relay)(_.finish) - } private[relay] def WithRelay[A: Zero](id: RelayRoundId)(f: RelayRound => Fu[A]): Fu[A] = byId(id).flatMapz(f) From 8ac885be32b0c4733d4a0ca2042a211bcc8431d4 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 09:06:53 +0200 Subject: [PATCH 124/168] only auto-start broadcast rounds with sync.upstream --- modules/relay/src/main/RelayApi.scala | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index 5c9cc8eced7c3..5f4485d92583d 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -396,7 +396,8 @@ final class RelayApi( .$lt(nowInstant.plusSeconds(RelayDelay.maxSeconds.value)) .$gt(nowInstant.minusDays(1)), // bit late now "startedAt".$exists(false), - "sync.until".$exists(false) + "sync.until".$exists(false), + "sync.upstream".$exists(true) ) ) .flatMap: From 59a56f06ecc8df381022a1bdf738f9da941c1f7b Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 10:26:03 +0200 Subject: [PATCH 125/168] fix broadcast form lacks image field --- modules/relay/src/main/ui/FormUi.scala | 1 + 1 file changed, 1 insertion(+) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 5f92c7b225353..0fc8b62c27e42 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -244,6 +244,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): page(nav.tour.name.value, menu = Right(nav)): frag( boxTop(h1(a(href := routes.RelayTour.show(nav.tour.slug, nav.tour.id))(nav.tour.name))), + image(nav.tour), postForm(cls := "form3", action := routes.RelayTour.update(nav.tour.id))( inner(form, nav.tourWithGroup.some), form3.actions( From 4fbafb6f2ab74f6f365089e16c246951d4ae288c Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 10:32:49 +0200 Subject: [PATCH 126/168] align user dropdown icons --- ui/bits/css/user/_show.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 1e47ada555f60..66f4531ebcfa7 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -63,8 +63,8 @@ top: 3rem; @include inline-end(0); a { + @extend %flex-center-nowrap; width: 20em; - display: flex; padding: 0.7rem 1rem; color: $c-header-dropdown; &::before { From 9eb36d0c338fcb31ebde39d5739a5a080a9f6aa0 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 10:58:01 +0200 Subject: [PATCH 127/168] more subtle lobby pool start loading screen --- ui/lobby/src/view/main.ts | 5 +++-- ui/lobby/src/view/pools.ts | 6 +++--- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/ui/lobby/src/view/main.ts b/ui/lobby/src/view/main.ts index 8fc5228b8f213..6991babedccbe 100644 --- a/ui/lobby/src/view/main.ts +++ b/ui/lobby/src/view/main.ts @@ -10,7 +10,8 @@ import LobbyController from '../ctrl'; export default function (ctrl: LobbyController) { let body, data: VNodeData = {}; - if (ctrl.redirecting) body = spinner(); + const redirBlock = ctrl.redirecting && ctrl.tab != 'pools'; + if (redirBlock) body = spinner(); else switch (ctrl.tab) { case 'pools': @@ -29,6 +30,6 @@ export default function (ctrl: LobbyController) { } return h('div.lobby__app.lobby__app-' + ctrl.tab, [ h('div.tabs-horiz', { attrs: { role: 'tablist' } }, renderTabs(ctrl)), - h('div.lobby__app__content.l' + (ctrl.redirecting ? 'redir' : ctrl.tab), data, body), + h(`div.lobby__app__content.l${redirBlock ? 'redir' : ctrl.tab}`, data, body), ]); } diff --git a/ui/lobby/src/view/pools.ts b/ui/lobby/src/view/pools.ts index db41d3293e32c..e0767ab8911ad 100644 --- a/ui/lobby/src/view/pools.ts +++ b/ui/lobby/src/view/pools.ts @@ -3,10 +3,11 @@ import { spinnerVdom as spinner } from 'common/spinner'; import { bind } from 'common/snabbdom'; import LobbyController from '../ctrl'; -export function hooks(ctrl: LobbyController): Hooks { - return bind( +export const hooks = (ctrl: LobbyController): Hooks => + bind( 'click', e => { + if (ctrl.redirecting) return; const id = (e.target as HTMLElement).dataset['id'] || ((e.target as HTMLElement).parentNode as HTMLElement).dataset['id']; @@ -15,7 +16,6 @@ export function hooks(ctrl: LobbyController): Hooks { }, ctrl.redraw, ); -} export function render(ctrl: LobbyController) { const member = ctrl.poolMember; From ede45dd70f8e0d8728f96a3d95c8d2ec5dd9aa1d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 11:05:53 +0200 Subject: [PATCH 128/168] lobby pool css tweaks --- ui/lobby/css/app/_pool.scss | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ui/lobby/css/app/_pool.scss b/ui/lobby/css/app/_pool.scss index 18bf734a9ae3e..8165ebdd83423 100644 --- a/ui/lobby/css/app/_pool.scss +++ b/ui/lobby/css/app/_pool.scss @@ -8,6 +8,7 @@ padding-top: 9px; background-color: transparent !important; box-shadow: none; + overflow: visible !important; // for button shadows @include fluid-size('font-size', 14px, 25px); @@ -36,7 +37,6 @@ .active { @extend %popup-shadow; - @include back-blur(); // for all themes, not just transp .perf { display: none; @@ -44,7 +44,7 @@ } .transp { - opacity: 0.4; + opacity: 0.35; } .spinner { From 4ea5af0297e5ed3a6c61b86294fe5a9a071f5ff0 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 11:30:34 +0200 Subject: [PATCH 129/168] TS code golf --- ui/lobby/src/view/pools.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ui/lobby/src/view/pools.ts b/ui/lobby/src/view/pools.ts index e0767ab8911ad..67cd9de24b33d 100644 --- a/ui/lobby/src/view/pools.ts +++ b/ui/lobby/src/view/pools.ts @@ -21,12 +21,12 @@ export function render(ctrl: LobbyController) { const member = ctrl.poolMember; return ctrl.pools .map(pool => { - const active = !!member && member.id === pool.id, + const active = member?.id === pool.id, transp = !!member && !active; return h( 'div', { - class: { active, transp: !active && transp }, + class: { active, transp }, attrs: { role: 'button', 'data-id': pool.id }, }, [ From 49a91e034db59167480665f3d32c5f9ddec3949d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 11:30:53 +0200 Subject: [PATCH 130/168] don't show that pool is left while redirecting to new game --- ui/lobby/src/lobby.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/lobby/src/lobby.ts b/ui/lobby/src/lobby.ts index d6728ecd7de95..50b37ae5cf247 100644 --- a/ui/lobby/src/lobby.ts +++ b/ui/lobby/src/lobby.ts @@ -42,8 +42,8 @@ export function initModule(opts: LobbyOpts) { site.contentLoaded(); }, redirect(e: RedirectTo) { - lobbyCtrl.leavePool(); lobbyCtrl.setRedirecting(); + lobbyCtrl.leavePool(); site.redirect(e, true); return true; }, From 0126c9bef4250efc1b435a2b561b5b0ee1c481c6 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 12:34:19 +0200 Subject: [PATCH 131/168] fix user actions in blind mode - after #15260 - closes #15545 --- ui/bits/css/user/_show.scss | 3 +++ 1 file changed, 3 insertions(+) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 66f4531ebcfa7..44eb3a1a8827a 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -62,6 +62,9 @@ position: absolute; top: 3rem; @include inline-end(0); + body.blind-mode & { + visibility: visible; + } a { @extend %flex-center-nowrap; width: 20em; From 176832738a8127edd7c09956bf8a4fd253e63735 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 12:34:36 +0200 Subject: [PATCH 132/168] fix user actions style --- ui/bits/css/user/_show.scss | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/ui/bits/css/user/_show.scss b/ui/bits/css/user/_show.scss index 44eb3a1a8827a..39864cfd5d664 100644 --- a/ui/bits/css/user/_show.scss +++ b/ui/bits/css/user/_show.scss @@ -74,12 +74,6 @@ margin-inline: 0 1rem; font-size: 1.4em; } - &:first-child { - border-radius: $box-radius-size 0 0 0; - } - &:last-child { - @extend %box-radius-bottom; - } &:hover { background: $c-primary; &, @@ -88,6 +82,14 @@ } } } + > a { + &:first-child { + border-radius: $box-radius-size 0 0 0; + } + &:last-child { + @extend %box-radius-bottom; + } + } } &:hover { > a { From 8dee6e213969b7c2f910082b595598d17de7afd4 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 14:44:58 +0200 Subject: [PATCH 133/168] fix /opening/tree js data - closes #15547 --- modules/opening/src/main/ui/OpeningBits.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/modules/opening/src/main/ui/OpeningBits.scala b/modules/opening/src/main/ui/OpeningBits.scala index 79ea81800c68e..4457a12a7b3b8 100644 --- a/modules/opening/src/main/ui/OpeningBits.scala +++ b/modules/opening/src/main/ui/OpeningBits.scala @@ -1,7 +1,7 @@ package lila.opening package ui -import play.api.libs.json.Json +import play.api.libs.json.* import chess.opening.{ Opening, OpeningKey } import lila.ui.* @@ -15,7 +15,7 @@ final class OpeningBits(helpers: Helpers): def pageModule(page: Option[OpeningPage])(using Context) = PageModule( "opening", - page.so: p => + page.fold(JsNull): p => import lila.common.Json.given Json.obj("history" -> p.explored.so[List[Float]](_.history), "sans" -> p.query.sans) ) From cac60a4b59bc38b636959344d03f9e736255ab1d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 14:45:51 +0200 Subject: [PATCH 134/168] remove debug --- ui/opening/src/wiki.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/ui/opening/src/wiki.ts b/ui/opening/src/wiki.ts index b7d89b6d00ddc..e7303e39214c9 100644 --- a/ui/opening/src/wiki.ts +++ b/ui/opening/src/wiki.ts @@ -48,8 +48,6 @@ async function fetchAndRender(data: OpeningPage, render: (html: string) => void) console.warn('error: unexpected API response:
' + JSON.stringify(page) + '
'); return; } else { - console.log(page.extract, title); - console.log(transform(page.extract, title)); return render(transform(page.extract, title)); } } else return; From 7cc6a9bc4e7e9ebc649d4749be6109cbc51fcefc Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 15:34:25 +0200 Subject: [PATCH 135/168] filter broadcast source games by round --- modules/relay/src/main/RelayFetch.scala | 1 + modules/relay/src/main/RelayGame.scala | 4 ++++ modules/relay/src/main/RelayRound.scala | 1 + modules/relay/src/main/RelayRoundForm.scala | 15 +++++++++------ modules/relay/src/main/ui/FormUi.scala | 10 ++++++++++ 5 files changed, 25 insertions(+), 6 deletions(-) diff --git a/modules/relay/src/main/RelayFetch.scala b/modules/relay/src/main/RelayFetch.scala index eb8f1344de023..620b671c5f44d 100644 --- a/modules/relay/src/main/RelayFetch.scala +++ b/modules/relay/src/main/RelayFetch.scala @@ -80,6 +80,7 @@ final private class RelayFetch( if !rt.round.sync.playing then fuccess(updating(_.withSync(_.play(rt.tour.official)))) else fetchGames(rt) + .map(RelayGame.filter(rt.round.sync.onlyRound)) .map(games => rt.tour.players.fold(games)(_.update(games))) .flatMap(fidePlayers.enrichGames(rt.tour)) .map(games => rt.tour.teams.fold(games)(_.update(games))) diff --git a/modules/relay/src/main/RelayGame.scala b/modules/relay/src/main/RelayGame.scala index 180df1a9fd52f..208bc56878de5 100644 --- a/modules/relay/src/main/RelayGame.scala +++ b/modules/relay/src/main/RelayGame.scala @@ -73,3 +73,7 @@ private object RelayGame: , mul => RelayFetch.multiPgnToGames(mul).fold(e => throw e, identity) ) + + def filter(onlyRound: Option[Int])(games: RelayGames): RelayGames = + onlyRound.fold(games): round => + games.filter(_.tags.roundNumber.has(round)) diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index 75505b7cb08e6..90c1fb2fe7c12 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -75,6 +75,7 @@ object RelayRound: nextAt: Option[Instant], // when to run next sync period: Option[Seconds], // override time between two sync (rare) delay: Option[Seconds], // add delay between the source and the study + onlyRound: Option[Int], // only keep games with [Round "x"] log: SyncLog ): def hasUpstream = upstream.isDefined diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index a4578ba10f03a..4b9ca1e688fcf 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -32,9 +32,8 @@ final class RelayRoundForm(using mode: Mode): "startsAt" -> optional(ISOInstantOrTimestamp.mapping), "finished" -> optional(boolean), "period" -> optional(number(min = 2, max = 60).into[Seconds]), - "delay" -> optional( - number(min = 0, max = RelayDelay.maxSeconds.value).into[Seconds] - ) // don't increase the max + "delay" -> optional(number(min = 0, max = RelayDelay.maxSeconds.value).into[Seconds]), + "onlyRound" -> optional(number(min = 1, max = 999)) )(Data.apply)(unapply) .verifying("This source requires a round number. See the new form field below.", !_.roundMissing) @@ -54,7 +53,7 @@ object RelayRoundForm: val prevs: Option[(RelayRound, RelayRound)] = rounds.reverse match case a :: b :: _ => (a, b).some case _ => none - val prev: Option[RelayRound] = prevs.map(_._1) + val prev: Option[RelayRound] = rounds.lastOption val roundNumberRegex = """([^\d]*)(\d{1,2})([^\d]*)""".r val roundNumberIn: String => Option[Int] = case roundNumberRegex(_, n, _) => n.toIntOption @@ -86,7 +85,8 @@ object RelayRoundForm: syncUrlRound = (prevs.isEmpty || nextUrl.isDefined).option(nextNumber), startsAt = guessDate, period = prev.flatMap(_.sync.period), - delay = prev.flatMap(_.sync.delay) + delay = prev.flatMap(_.sync.delay), + onlyRound = prev.flatMap(_.sync.onlyRound).map(_ + 1) ) case class GameIds(ids: List[GameId]) @@ -143,7 +143,8 @@ object RelayRoundForm: startsAt: Option[Instant] = None, finished: Option[Boolean] = None, period: Option[Seconds] = None, - delay: Option[Seconds] = None + delay: Option[Seconds] = None, + onlyRound: Option[Int] = None ): def requiresRound = syncUrl.exists(RelayRound.Sync.UpstreamUrl.LccRegex.matches) @@ -176,6 +177,7 @@ object RelayRoundForm: nextAt = none, period = period.ifTrue(Granter.ofUser(_.StudyAdmin)(user)), delay = delay, + onlyRound = onlyRound, log = SyncLog.empty ) @@ -207,5 +209,6 @@ object RelayRoundForm: startsAt = relay.startsAt, finished = relay.finished.option(true), period = relay.sync.period, + onlyRound = relay.sync.onlyRound, delay = relay.sync.delay ) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 0fc8b62c27e42..79bb1feb6ce13 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -209,6 +209,16 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): )(form3.input(_, typ = "number")) ) ), + form3.split( + form3.group( + form("onlyRound"), + raw("Filter games by round number"), + help = frag( + "Optional, only keep games from the source that match a round number." + ).some, + half = true + )(form3.input(_, typ = "number")) + ), form3.actions( a(href := routes.RelayTour.show(t.slug, t.id))(trans.site.cancel()), form3.submit(trans.site.apply()) From c7a904a4d497c3b4e5e117852f4a0bfdf438d9c5 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 16:15:41 +0200 Subject: [PATCH 136/168] broadcast game slicing --- modules/relay/src/main/BSONHandlers.scala | 3 ++ modules/relay/src/main/RelayFetch.scala | 1 + modules/relay/src/main/RelayGame.scala | 38 +++++++++++++++++++++ modules/relay/src/main/RelayRound.scala | 1 + modules/relay/src/main/RelayRoundForm.scala | 13 +++++-- modules/relay/src/main/ui/FormUi.scala | 19 ++++++++++- 6 files changed, 71 insertions(+), 4 deletions(-) diff --git a/modules/relay/src/main/BSONHandlers.scala b/modules/relay/src/main/BSONHandlers.scala index 8c8967dc4cfc9..3d85ca7675240 100644 --- a/modules/relay/src/main/BSONHandlers.scala +++ b/modules/relay/src/main/BSONHandlers.scala @@ -30,6 +30,9 @@ object BSONHandlers: given BSONHandler[SyncLog] = isoHandler[SyncLog, Vector[Event]](_.events, SyncLog.apply) + given BSONHandler[List[RelayGame.Slice]] = + stringIsoHandler[List[RelayGame.Slice]](using RelayGame.Slices.iso) + given BSONDocumentHandler[Sync] = Macros.handler given BSONDocumentHandler[RelayRound] = Macros.handler diff --git a/modules/relay/src/main/RelayFetch.scala b/modules/relay/src/main/RelayFetch.scala index 620b671c5f44d..3eb0640e970b2 100644 --- a/modules/relay/src/main/RelayFetch.scala +++ b/modules/relay/src/main/RelayFetch.scala @@ -81,6 +81,7 @@ final private class RelayFetch( else fetchGames(rt) .map(RelayGame.filter(rt.round.sync.onlyRound)) + .map(RelayGame.Slices.filter(~rt.round.sync.slices)) .map(games => rt.tour.players.fold(games)(_.update(games))) .flatMap(fidePlayers.enrichGames(rt.tour)) .map(games => rt.tour.teams.fold(games)(_.update(games))) diff --git a/modules/relay/src/main/RelayGame.scala b/modules/relay/src/main/RelayGame.scala index 208bc56878de5..b997c4e689aaf 100644 --- a/modules/relay/src/main/RelayGame.scala +++ b/modules/relay/src/main/RelayGame.scala @@ -77,3 +77,41 @@ private object RelayGame: def filter(onlyRound: Option[Int])(games: RelayGames): RelayGames = onlyRound.fold(games): round => games.filter(_.tags.roundNumber.has(round)) + + // 1-indexed, both inclusive + case class Slice(from: Int, to: Int) + + object Slices: + def filter(slices: List[Slice])(games: RelayGames): RelayGames = + if slices.isEmpty then games + else + games.view.zipWithIndex + .filter: (g, i) => + val n = i + 1 + slices.exists: s => + n >= s.from && n <= s.to + .map(_._1) + .toVector + + // 1-5,12-15,20 + def parse(str: String): List[Slice] = str.trim + .split(',') + .toList + .map(_.trim) + .flatMap: s => + s.split('-').toList.map(_.trim) match + case Nil => none + case from :: Nil => from.toIntOption.map(f => Slice(f, f)) + case from :: to :: _ => + for + f <- from.toIntOption + t <- to.toIntOption + yield Slice(f, t) + + def show(slices: List[Slice]): String = slices.pp + .map: + case Slice(f, t) if f == t => f.toString + case Slice(f, t) => s"$f-$t" + .mkString(",") + + val iso: Iso.StringIso[List[Slice]] = Iso(parse, show) diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index 90c1fb2fe7c12..4641c65112263 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -76,6 +76,7 @@ object RelayRound: period: Option[Seconds], // override time between two sync (rare) delay: Option[Seconds], // add delay between the source and the study onlyRound: Option[Int], // only keep games with [Round "x"] + slices: Option[List[RelayGame.Slice]] = none, log: SyncLog ): def hasUpstream = upstream.isDefined diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index 4b9ca1e688fcf..3ad8b5db581ea 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -33,7 +33,10 @@ final class RelayRoundForm(using mode: Mode): "finished" -> optional(boolean), "period" -> optional(number(min = 2, max = 60).into[Seconds]), "delay" -> optional(number(min = 0, max = RelayDelay.maxSeconds.value).into[Seconds]), - "onlyRound" -> optional(number(min = 1, max = 999)) + "onlyRound" -> optional(number(min = 1, max = 999)), + "slices" -> optional: + nonEmptyText + .transform[List[RelayGame.Slice]](RelayGame.Slices.parse, RelayGame.Slices.show) )(Data.apply)(unapply) .verifying("This source requires a round number. See the new form field below.", !_.roundMissing) @@ -86,7 +89,8 @@ object RelayRoundForm: startsAt = guessDate, period = prev.flatMap(_.sync.period), delay = prev.flatMap(_.sync.delay), - onlyRound = prev.flatMap(_.sync.onlyRound).map(_ + 1) + onlyRound = prev.flatMap(_.sync.onlyRound).map(_ + 1), + slices = prev.flatMap(_.sync.slices) ) case class GameIds(ids: List[GameId]) @@ -144,7 +148,8 @@ object RelayRoundForm: finished: Option[Boolean] = None, period: Option[Seconds] = None, delay: Option[Seconds] = None, - onlyRound: Option[Int] = None + onlyRound: Option[Int] = None, + slices: Option[List[RelayGame.Slice]] = None ): def requiresRound = syncUrl.exists(RelayRound.Sync.UpstreamUrl.LccRegex.matches) @@ -178,6 +183,7 @@ object RelayRoundForm: period = period.ifTrue(Granter.ofUser(_.StudyAdmin)(user)), delay = delay, onlyRound = onlyRound, + slices = slices, log = SyncLog.empty ) @@ -210,5 +216,6 @@ object RelayRoundForm: finished = relay.finished.option(true), period = relay.sync.period, onlyRound = relay.sync.onlyRound, + slices = relay.sync.slices, delay = relay.sync.delay ) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 79bb1feb6ce13..c9dd9ee6758c6 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -217,7 +217,24 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): "Optional, only keep games from the source that match a round number." ).some, half = true - )(form3.input(_, typ = "number")) + )(form3.input(_, typ = "number")), + form3.group( + form("slices"), + raw("Select slices of the games"), + help = frag( + "Optional. Select games based on their position in the source.", + br, + pre(""" +1 only select the first board +1-4 only select the first 4 boards +1,2,3,4 same as above, first 4 boards +11-20,21-30 boards 11 to 20, and boards 21 to 30 +2,3,7-9 boards 2, 3, 7, 8, and 9 +"""), + "Slicing is done after filtering by round number." + ).some, + half = true + )(form3.input(_)) ), form3.actions( a(href := routes.RelayTour.show(t.slug, t.id))(trans.site.cancel()), From a4f55eb5f9baa044d0604bbf477db80007fd8cb5 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 16:29:45 +0200 Subject: [PATCH 137/168] remove debug --- modules/relay/src/main/RelayGame.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayGame.scala b/modules/relay/src/main/RelayGame.scala index b997c4e689aaf..8b59b6f530cd0 100644 --- a/modules/relay/src/main/RelayGame.scala +++ b/modules/relay/src/main/RelayGame.scala @@ -108,7 +108,7 @@ private object RelayGame: t <- to.toIntOption yield Slice(f, t) - def show(slices: List[Slice]): String = slices.pp + def show(slices: List[Slice]): String = slices .map: case Slice(f, t) if f == t => f.toString case Slice(f, t) => s"$f-$t" From fbca504acaf9c305458c69e113b3b97adf1ab6ff Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Tue, 18 Jun 2024 16:40:43 +0200 Subject: [PATCH 138/168] better board slicing example --- modules/relay/src/main/ui/FormUi.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index c9dd9ee6758c6..c389d3145a39e 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -228,7 +228,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): 1 only select the first board 1-4 only select the first 4 boards 1,2,3,4 same as above, first 4 boards -11-20,21-30 boards 11 to 20, and boards 21 to 30 +11-15,21-25 boards 11 to 15, and boards 21 to 25 2,3,7-9 boards 2, 3, 7, 8, and 9 """), "Slicing is done after filtering by round number." From 73202fdc905fe7ec59cd6f1899a8cd4944ef2f43 Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Tue, 18 Jun 2024 07:54:53 +0700 Subject: [PATCH 139/168] Clean up leftover of forumSearch ingestor --- modules/forum/src/main/ForumPostRepo.scala | 4 +--- modules/forumSearch/src/main/Env.scala | 2 -- modules/forumSearch/src/main/ForumSearchApi.scala | 5 +---- 3 files changed, 2 insertions(+), 9 deletions(-) diff --git a/modules/forum/src/main/ForumPostRepo.scala b/modules/forum/src/main/ForumPostRepo.scala index 33029a19f59fc..e2e253e0d4a28 100644 --- a/modules/forum/src/main/ForumPostRepo.scala +++ b/modules/forum/src/main/ForumPostRepo.scala @@ -8,9 +8,7 @@ import lila.forum.Filter.* import lila.core.forum.ForumPostMini import reactivemongo.api.CursorOps -final class ForumPostRepo(val coll: Coll, filter: Filter = Safe)(using - Executor -): +final class ForumPostRepo(val coll: Coll, filter: Filter = Safe)(using Executor): def forUser(user: Option[User]) = withFilter(user.filter(_.marks.troll).fold[Filter](Safe) { u => diff --git a/modules/forumSearch/src/main/Env.scala b/modules/forumSearch/src/main/Env.scala index 6092317e9eff1..d9b08fa2af131 100644 --- a/modules/forumSearch/src/main/Env.scala +++ b/modules/forumSearch/src/main/Env.scala @@ -4,11 +4,9 @@ import com.softwaremill.macwire.* import play.api.Configuration import lila.common.autoconfig.{ *, given } -import lila.search.* import lila.core.forum.BusForum import BusForum.* import lila.core.config.ConfigName -import lila.core.id.ForumPostId import lila.search.client.SearchClient import lila.search.spec.Query diff --git a/modules/forumSearch/src/main/ForumSearchApi.scala b/modules/forumSearch/src/main/ForumSearchApi.scala index 3b85b2094005d..23bad2e6ae357 100644 --- a/modules/forumSearch/src/main/ForumSearchApi.scala +++ b/modules/forumSearch/src/main/ForumSearchApi.scala @@ -1,14 +1,11 @@ package lila.forumSearch -import akka.stream.scaladsl.* - import lila.search.* -import lila.core.forum.{ ForumPostApi, ForumPostMini, ForumPostMiniView } import lila.core.id.ForumPostId import lila.search.client.SearchClient import lila.search.spec.{ ForumSource, Query } -final class ForumSearchApi(client: SearchClient, postApi: ForumPostApi)(using Executor) +final class ForumSearchApi(client: SearchClient)(using Executor) extends SearchReadApi[ForumPostId, Query.Forum]: def search(query: Query.Forum, from: From, size: Size) = From 811fb46c66d248deb26461b282ed3719b9f7c371 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=A9rgio=20Gl=C3=B3rias?= <9739913+SergioGlorias@users.noreply.github.com> Date: Tue, 18 Jun 2024 20:51:27 +0100 Subject: [PATCH 140/168] Fix texts and add missing info --- modules/relay/src/main/ui/FormUi.scala | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index c389d3145a39e..d624489f02aba 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -356,11 +356,13 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): br, """"Jorge Rick Vito" will match "Jorge Rick", "jorge vito", "Rick, Vito", etc.""", br, + "If the player is NM or WNM, you can:", + pre("""Player Name = FIDE ID / Title"""), "Alternatively, you may set tags manually, like so:", pre("player name / rating / title / new name"), "All values are optional. Example:", pre("""Magnus Carlsen / 2863 / GM - YouGotLittUp / 1890 / / Louis Litt""") +YouGotLittUp / 1890 / / Louis Litt""") ).some, half = true )(form3.textarea(_)(rows := 3)), @@ -372,7 +374,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): pre("Team name; Fide Id or Player name"), "Example:", pre("""Team Cats ; 3408230 - Team Dogs ; Scooby Doo"""), +Team Dogs ; Scooby Doo"""), "By default the PGN tags WhiteTeam and BlackTeam are used." ).some, half = true From 26747a4ecea8b33f3207f39abe0abec3b9d659e6 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 12:48:04 +0200 Subject: [PATCH 141/168] relay multi-url wip --- bin/mongodb/relay-lcc-migrate.js | 17 +++ modules/coreI18n/src/main/key.scala | 4 +- modules/relay/src/main/BSONHandlers.scala | 20 ++- modules/relay/src/main/JsonView.scala | 13 +- modules/relay/src/main/RelayApi.scala | 2 +- modules/relay/src/main/RelayFetch.scala | 51 ++++--- modules/relay/src/main/RelayFormat.scala | 25 ++-- modules/relay/src/main/RelayRound.scala | 25 ++-- modules/relay/src/main/RelayRoundForm.scala | 134 ++++++++++++------ modules/relay/src/main/ui/FormUi.scala | 119 ++++++++++------ modules/ui/src/main/helper/Form3.scala | 1 + translation/source/broadcast.xml | 4 +- ui/analyse/src/study/relay/interfaces.ts | 1 + .../src/study/relay/relayManagerView.ts | 10 +- ui/bits/css/relay/_form.scss | 10 ++ ui/bits/src/bits.relayForm.ts | 16 ++- ui/common/css/form/_form3.scss | 4 +- ui/simul/css/_form.scss | 7 - 18 files changed, 292 insertions(+), 171 deletions(-) create mode 100644 bin/mongodb/relay-lcc-migrate.js diff --git a/bin/mongodb/relay-lcc-migrate.js b/bin/mongodb/relay-lcc-migrate.js new file mode 100644 index 0000000000000..e2c53262ca08c --- /dev/null +++ b/bin/mongodb/relay-lcc-migrate.js @@ -0,0 +1,17 @@ +const regex = /.*view\.livechesscloud\.com\/?#?([0-9a-f\-]+)/; +let done = 0; +let failed = 0; +db.relay.find({ 'sync.upstream.url': /view\.livechesscloud\.com/ }).forEach(relay => { + const url = relay.sync.upstream.url; + try { + const id = url.match(regex)[1]; + const round = parseInt(url.split(' ')[1]) || 1; + if (!id) throw new Error('No id in ' + url); + db.relay.updateOne({ _id: relay._id }, { $set: { 'sync.upstream': { lcc: { id, round } } } }); + done++; + } catch (e) { + failed++; + } +}); +console.log(done + ' done'); +console.log(failed + ' failed'); diff --git a/modules/coreI18n/src/main/key.scala b/modules/coreI18n/src/main/key.scala index f94c0339668e6..08b853fee5c82 100644 --- a/modules/coreI18n/src/main/key.scala +++ b/modules/coreI18n/src/main/key.scala @@ -1665,9 +1665,9 @@ object I18nKey: val `tournamentDescription`: I18nKey = "broadcast:tournamentDescription" val `fullDescription`: I18nKey = "broadcast:fullDescription" val `fullDescriptionHelp`: I18nKey = "broadcast:fullDescriptionHelp" - val `sourceUrlOrGameIds`: I18nKey = "broadcast:sourceUrlOrGameIds" + val `sourceUrl`: I18nKey = "broadcast:sourceUrl" val `sourceUrlHelp`: I18nKey = "broadcast:sourceUrlHelp" - val `gameIdsHelp`: I18nKey = "broadcast:gameIdsHelp" + val `sourceGameIds`: I18nKey = "broadcast:sourceGameIds" val `startDate`: I18nKey = "broadcast:startDate" val `startDateHelp`: I18nKey = "broadcast:startDateHelp" val `credits`: I18nKey = "broadcast:credits" diff --git a/modules/relay/src/main/BSONHandlers.scala b/modules/relay/src/main/BSONHandlers.scala index 3d85ca7675240..03b4f2377624c 100644 --- a/modules/relay/src/main/BSONHandlers.scala +++ b/modules/relay/src/main/BSONHandlers.scala @@ -10,18 +10,24 @@ object BSONHandlers: given BSONHandler[RelayTeamsTextarea] = stringAnyValHandler(_.text, RelayTeamsTextarea(_)) import RelayRound.Sync - import Sync.{ Upstream, UpstreamIds, UpstreamUrl } - given upstreamUrlHandler: BSONDocumentHandler[UpstreamUrl] = Macros.handler - given upstreamIdsHandler: BSONDocumentHandler[UpstreamIds] = Macros.handler + import Sync.{ Upstream, UpstreamIds, UpstreamUrl, UpstreamLcc, UpstreamUrls } + given upstreamUrlHandler: BSONDocumentHandler[UpstreamUrl] = Macros.handler + given upstreamLccHandler: BSONDocumentHandler[UpstreamLcc] = Macros.handler + given upstreamUrlsHandler: BSONDocumentHandler[UpstreamUrls] = Macros.handler + given upstreamIdsHandler: BSONDocumentHandler[UpstreamIds] = Macros.handler given BSONHandler[Upstream] = tryHandler( { - case d: BSONDocument if d.contains("url") => upstreamUrlHandler.readTry(d) - case d: BSONDocument if d.contains("ids") => upstreamIdsHandler.readTry(d) + case d: BSONDocument if d.contains("url") => upstreamUrlHandler.readTry(d) + case d: BSONDocument if d.contains("lcc") => upstreamLccHandler.readTry(d) + case d: BSONDocument if d.contains("urls") => upstreamUrlsHandler.readTry(d) + case d: BSONDocument if d.contains("ids") => upstreamIdsHandler.readTry(d) }, { - case url: UpstreamUrl => upstreamUrlHandler.writeTry(url).get - case ids: UpstreamIds => upstreamIdsHandler.writeTry(ids).get + case url: UpstreamUrl => upstreamUrlHandler.writeTry(url).get + case lcc: UpstreamLcc => upstreamLccHandler.writeTry(lcc).get + case urls: UpstreamUrls => upstreamUrlsHandler.writeTry(urls).get + case ids: UpstreamIds => upstreamIdsHandler.writeTry(ids).get } ) diff --git a/modules/relay/src/main/JsonView.scala b/modules/relay/src/main/JsonView.scala index 0bcdab2f91e7d..8625ba6c272c4 100644 --- a/modules/relay/src/main/JsonView.scala +++ b/modules/relay/src/main/JsonView.scala @@ -93,9 +93,7 @@ final class JsonView( rt: RelayRound.WithTourAndStudy, previews: ChapterPreview.AsJsons, group: Option[RelayGroup.WithTours] - )(using - Option[Me] - ): JsObject = + )(using Option[Me]): JsObject = myRound(rt) ++ Json.obj("games" -> previews).add("group" -> group) def sync(round: RelayRound) = Json.toJsObject(round.sync) @@ -172,7 +170,8 @@ object JsonView: Json.arr(minute * 60, crowd) ) - private given OWrites[RelayRound.Sync] = OWrites: s => + import RelayRound.Sync + private given OWrites[Sync] = OWrites: s => Json .obj( "ongoing" -> s.ongoing, @@ -180,6 +179,8 @@ object JsonView: ) .add("delay" -> s.delay) ++ s.upstream.so { - case url: RelayRound.Sync.UpstreamUrl => Json.obj("url" -> url.withRound.url) - case RelayRound.Sync.UpstreamIds(ids) => Json.obj("ids" -> ids) + case Sync.UpstreamUrl(url) => Json.obj("url" -> url) + case Sync.UpstreamLcc(url, round) => Json.obj("url" -> url, "round" -> round) + case Sync.UpstreamUrls(urls) => Json.obj("urls" -> urls.map(_.url)) + case Sync.UpstreamIds(ids) => Json.obj("ids" -> ids) } diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index 5f4485d92583d..760270c8a686a 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -244,7 +244,7 @@ final class RelayApi( def requestPlay(id: RelayRoundId, v: Boolean): Funit = WithRelay(id): relay => - relay.sync.upstream.flatMap(_.asUrl).map(_.withRound).foreach(formatApi.refresh) + relay.sync.upstream.foreach(formatApi.refresh) isOfficial(relay.id).flatMap: official => update(relay): r => if v diff --git a/modules/relay/src/main/RelayFetch.scala b/modules/relay/src/main/RelayFetch.scala index 3eb0640e970b2..98de665690b4d 100644 --- a/modules/relay/src/main/RelayFetch.scala +++ b/modules/relay/src/main/RelayFetch.scala @@ -14,7 +14,7 @@ import lila.memo.CacheApi import lila.study.{ MultiPgn, StudyPgnImport } import lila.tree.Node.Comments -import RelayRound.Sync.{ UpstreamIds, UpstreamUrl } +import RelayRound.Sync.{ UpstreamIds, UpstreamUrl, UpstreamUrls } import RelayFormat.CanProxy import scalalib.model.Seconds @@ -142,8 +142,7 @@ final private class RelayFetch( Seconds(60) else round.sync.period | Seconds: - if upstream.local then 3 - else if upstream.asUrl.exists(_.isLcc) && !tour.official then 10 + if upstream.asUrl.exists(_.isLcc) && !tour.official then 12 else 5 updating: _.withSync: @@ -168,26 +167,32 @@ final private class RelayFetch( private def fetchGames(rt: RelayRound.WithTour): Fu[RelayGames] = rt.round.sync.upstream.so: - case UpstreamIds(ids) => - gameRepo - .gamesFromSecondary(ids) - .flatMap(gameProxy.upgradeIfPresent) - .flatMap(gameRepo.withInitialFens) - .flatMap { games => - if games.size == ids.size then - val pgnFlags = gameIdsUpstreamPgnFlags.copy(delayMoves = !rt.tour.official) - given play.api.i18n.Lang = lila.core.i18n.defaultLang - games - .sequentially: (game, fen) => - pgnDump(game, fen, pgnFlags).dmap(_.render) - .dmap(MultiPgn.apply) - else - throw LilaInvalid: - s"Invalid game IDs: ${ids.filter(id => !games.exists(_._1.id == id)).mkString(", ")}" - } - .flatMap(multiPgnToGames(_).toFuture) - case url: UpstreamUrl => - delayer(url, rt.round, fetchFromUpstream(using CanProxy(rt.tour.official))) + case UpstreamIds(ids) => fetchFromGameIds(rt.tour, ids) + case url: UpstreamUrl => delayer(url, rt.round, fetchFromUpstream(using CanProxy(rt.tour.official))) + case UpstreamUrls(urls) => + urls + .traverse: url => + delayer(url, rt.round, fetchFromUpstream(using CanProxy(rt.tour.official))) + .map(_.flatten.toVector) + + private def fetchFromGameIds(tour: RelayTour, ids: List[GameId]): Fu[RelayGames] = + gameRepo + .gamesFromSecondary(ids) + .flatMap(gameProxy.upgradeIfPresent) + .flatMap(gameRepo.withInitialFens) + .flatMap { games => + if games.size == ids.size then + val pgnFlags = gameIdsUpstreamPgnFlags.copy(delayMoves = !tour.official) + given play.api.i18n.Lang = lila.core.i18n.defaultLang + games + .sequentially: (game, fen) => + pgnDump(game, fen, pgnFlags).dmap(_.render) + .dmap(MultiPgn.apply) + else + throw LilaInvalid: + s"Invalid game IDs: ${ids.filter(id => !games.exists(_._1.id == id)).mkString(", ")}" + } + .flatMap(multiPgnToGames(_).toFuture) private def fetchFromUpstream(using canProxy: CanProxy)(upstream: UpstreamUrl, max: Max): Fu[RelayGames] = import DgtJson.* diff --git a/modules/relay/src/main/RelayFormat.scala b/modules/relay/src/main/RelayFormat.scala index 38bdd5b465dfe..21d55e19cbd8c 100644 --- a/modules/relay/src/main/RelayFormat.scala +++ b/modules/relay/src/main/RelayFormat.scala @@ -29,34 +29,33 @@ final private class RelayFormatApi( )(using Executor): import RelayFormat.* - import RelayRound.Sync.UpstreamUrl + import RelayRound.Sync.{ Upstream, UpstreamUrl, UpstreamLcc } - private val cache = cacheApi[(UpstreamUrl.WithRound, CanProxy), RelayFormat](64, "relay.format"): + private val cache = cacheApi[(Upstream, CanProxy), RelayFormat](64, "relay.format"): _.expireAfterWrite(5 minutes) .buildAsyncFuture: (url, proxy) => guessFormat(url)(using proxy) - def get(upstream: UpstreamUrl.WithRound)(using proxy: CanProxy): Fu[RelayFormat] = + def get(upstream: Upstream)(using proxy: CanProxy): Fu[RelayFormat] = cache.get(upstream -> proxy) - def refresh(upstream: UpstreamUrl.WithRound): Unit = + def refresh(upstream: Upstream): Unit = CanProxy .from(List(false, true)) .foreach: proxy => cache.invalidate(upstream -> proxy) - private def guessFormat(upstream: UpstreamUrl.WithRound)(using CanProxy): Fu[RelayFormat] = { + private def guessFormat(upstream: Upstream)(using CanProxy): Fu[RelayFormat] = { - val originalUrl = URL.parse(upstream.url) + // val originalUrl = URL.parse(upstream.url) // http://view.livechesscloud.com/ed5fb586-f549-4029-a470-d590f8e30c76 - def guessLcc(url: URL)(using CanProxy): Fu[Option[RelayFormat]] = - url.toString match - case UpstreamUrl.LccRegex(id) => - guessManyFiles: - URL.parse: - s"http://1.pool.livechesscloud.com/get/$id/round-${upstream.round | 1}/index.json" - case _ => fuccess(none) + def guessLcc(using CanProxy): Fu[Option[RelayFormat]] = upstream match + case UpstreamLcc(id, round) => + guessManyFiles: + URL.parse: + s"http://1.pool.livechesscloud.com/get/$id/round-$round/index.json" + case _ => fuccess(none) def guessSingleFile(url: URL)(using CanProxy): Fu[Option[RelayFormat]] = List( diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index 4641c65112263..84c31915d0699 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -114,21 +114,16 @@ object RelayRound: override def toString = upstream.toString object Sync: - sealed trait Upstream: - def asUrl: Option[UpstreamUrl] = this match - case url: UpstreamUrl => url.some - case _ => none - def local = asUrl.fold(true)(_.isLocal) - case class UpstreamUrl(url: String) extends Upstream: - def isLocal = url.contains("://127.0.0.1") || url.contains("://[::1]") || url.contains("://localhost") - def withRound = url.split(" ", 2) match - case Array(u, round) => UpstreamUrl.WithRound(u, round.toIntOption) - case _ => UpstreamUrl.WithRound(url, none) - def isLcc: Boolean = UpstreamUrl.LccRegex.matches(url) - object UpstreamUrl: - case class WithRound(url: String, round: Option[Int]) - val LccRegex = """.*view\.livechesscloud\.com/?#?([0-9a-f\-]+)""".r - case class UpstreamIds(ids: List[GameId]) extends Upstream + sealed trait Upstream + case class UpstreamUrl(url: String) extends Upstream + case class UpstreamLcc(id: String, round: Int) extends Upstream: + def url = s"https://view.livechesscloud.com/#$id" + case class UpstreamUrls(urls: List[UpstreamUrl]) extends Upstream + case class UpstreamIds(ids: List[GameId]) extends Upstream + val LccRegex = """.*view\.livechesscloud\.com/?#?([0-9a-f\-]+)""".r + def tryLcc(url: String): Option[UpstreamLcc] = url match + case LccRegex(id) => UpstreamLcc(id, 1).some + case _ => none trait AndTour: val tour: RelayTour diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index 3ad8b5db581ea..8c144eb8788b7 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -1,33 +1,55 @@ package lila.relay +import scala.util.Try import io.mola.galimatias.URL import play.api.data.* import play.api.data.Forms.* +import play.api.data.format.Formatter import play.api.Mode - -import scala.util.Try -import scala.util.chaining.* - -import lila.common.Form.{ cleanText, into } import scalalib.model.Seconds + +import lila.common.Form.{ cleanText, into, stringIn, formatter } import lila.core.perm.Granter import lila.relay.RelayRound.Sync +import lila.relay.RelayRound.Sync.UpstreamUrl final class RelayRoundForm(using mode: Mode): import RelayRoundForm.* import lila.common.Form.ISOInstantOrTimestamp + private given Formatter[Sync.UpstreamUrl] = formatter.stringTryFormatter(validateUpstreamUrl, _.url) + private given Formatter[Sync.UpstreamUrls] = formatter.stringTryFormatter( + _.linesIterator.toList + .map(_.trim) + .filter(_.nonEmpty) + .traverse(validateUpstreamUrl) + .map(_.distinct) + .map(Sync.UpstreamUrls.apply), + _.urls.map(_.url).mkString("\n") + ) + private given Formatter[Sync.UpstreamIds] = formatter.stringTryFormatter( + _.split(' ').toList + .map(_.trim) + .traverse: i => + GameId.from(i.trim).toRight(s"Invalid game ID: $i") + .left + .map(_.mkString(", ")) + .filterOrElse(_.sizeIs <= RelayFetch.maxChapters.value, s"Max games: ${RelayFetch.maxChapters}") + .map(_.distinct) + .map(Sync.UpstreamIds.apply), + _.ids.mkString(" ") + ) + val roundMapping = mapping( - "name" -> cleanText(minLength = 3, maxLength = 80).into[RelayRound.Name], - "caption" -> optional(cleanText(minLength = 3, maxLength = 80).into[RelayRound.Caption]), - "syncUrl" -> optional { - cleanText(minLength = 8, maxLength = 600) - .verifying("Invalid source", validSource) - .verifying("The source URL cannot specify a port", url => mode.notProd || validSourcePort(url)) - }, + "name" -> cleanText(minLength = 3, maxLength = 80).into[RelayRound.Name], + "caption" -> optional(cleanText(minLength = 3, maxLength = 80).into[RelayRound.Caption]), + "syncSource" -> optional(stringIn(sourceTypes.map(_._1).toSet)), + "syncUrl" -> optional(of[Sync.UpstreamUrl]), + "syncUrls" -> optional(of[Sync.UpstreamUrls]), + "syncIds" -> optional(of[Sync.UpstreamIds]), "syncUrlRound" -> optional(number(min = 1, max = 999)), "startsAt" -> optional(ISOInstantOrTimestamp.mapping), "finished" -> optional(boolean), @@ -52,6 +74,13 @@ final class RelayRoundForm(using mode: Mode): object RelayRoundForm: + val sourceTypes = List( + "url" -> "Single PGN URL", + "urls" -> "Combine several PGN URLs", + "ids" -> "Lichess game IDs", + "push" -> "Push local games" + ) + def fillFromPrevRounds(rounds: List[RelayRound]): Data = val prevs: Option[(RelayRound, RelayRound)] = rounds.reverse match case a :: b :: _ => (a, b).some @@ -73,8 +102,11 @@ object RelayRoundForm: roundNumberIn(old.name.value).contains(n - 1) p <- prev yield replaceRoundNumber(p.name.value, nextNumber) - val nextUrl = - prev.flatMap(_.sync.upstream).flatMap(_.asUrl).map(_.withRound).filter(_.round.isDefined).map(_.url) + val nextLcc: Option[Sync.UpstreamLcc] = prev + .flatMap(_.sync.upstream) + .flatMap: + case lcc: Sync.UpstreamLcc => lcc.copy(round = nextNumber).some + case _ => none val guessDate = for (prev, old) <- prevs prevDate <- prev.startsAt @@ -84,8 +116,9 @@ object RelayRoundForm: Data( name = RelayRound.Name(guessName | s"Round ${nextNumber}"), caption = prev.flatMap(_.caption), - syncUrl = nextUrl, - syncUrlRound = (prevs.isEmpty || nextUrl.isDefined).option(nextNumber), + syncSource = prev.map(Data.make).flatMap(_.syncSource), + syncUrl = nextLcc.map(l => Sync.UpstreamUrl(l.url)), + syncUrlRound = nextLcc.map(_.round), startsAt = guessDate, period = prev.flatMap(_.sync.period), delay = prev.flatMap(_.sync.delay), @@ -99,9 +132,6 @@ object RelayRoundForm: val list = ids.split(' ').view.flatMap(i => GameId.from(i.trim)).toList (list.sizeIs > 0 && list.sizeIs <= RelayFetch.maxChapters.value).option(GameIds(list)) - private def validSource(source: String)(using mode: Mode): Boolean = - cleanUrl(source).isDefined || toGameIds(source).isDefined - private def cleanUrl(source: String)(using mode: Mode): Option[String] = for url <- Try(URL.parse(source)).toOption @@ -112,10 +142,19 @@ object RelayRoundForm: if !subdomain(host, "chess.com") || url.toString.startsWith("https://api.chess.com/pub") yield url.toString.stripSuffix("/") + private def validateUpstreamUrl(s: String)(using Mode): Either[String, Sync.UpstreamUrl] = for + url <- cleanUrl(s).toRight("Invalid source URL") + url <- if !validSourcePort(url) then Left("The source URL cannot specify a port") else Right(url) + yield Sync.UpstreamUrl(url) + + private def cleanUrls(source: String)(using mode: Mode): Option[List[String]] = + source.linesIterator.toList.flatMap(cleanUrl).some.filter(_.nonEmpty) + private val validPorts = Set(-1, 80, 443, 8080, 8491) - private def validSourcePort(source: String): Boolean = - Try(URL.parse(source)).toOption.forall: url => - validPorts(url.port) + private def validSourcePort(source: String)(using mode: Mode): Boolean = + mode.notProd || + Try(URL.parse(source)).toOption.forall: url => + validPorts(url.port) private def subdomain(host: String, domain: String) = s".$host".endsWith(s".$domain") @@ -142,7 +181,10 @@ object RelayRoundForm: case class Data( name: RelayRound.Name, caption: Option[RelayRound.Caption], - syncUrl: Option[String] = None, + syncSource: Option[String], + syncUrl: Option[Sync.UpstreamUrl] = None, + syncUrls: Option[Sync.UpstreamUrls] = None, + syncIds: Option[Sync.UpstreamIds] = None, syncUrlRound: Option[Int] = None, startsAt: Option[Instant] = None, finished: Option[Boolean] = None, @@ -151,33 +193,32 @@ object RelayRoundForm: onlyRound: Option[Int] = None, slices: Option[List[RelayGame.Slice]] = None ): + def upstream: Option[Sync.Upstream] = syncSource match + case None => syncUrl.orElse(syncUrls).orElse(syncIds) + case Some("url") => + syncUrl.map: u => + (Sync.tryLcc(u.url), syncUrlRound).mapN((l, r) => l.copy(round = r)) | u + case Some("urls") => syncUrls + case Some("ids") => syncIds + case _ => None - def requiresRound = syncUrl.exists(RelayRound.Sync.UpstreamUrl.LccRegex.matches) - - def roundMissing = requiresRound && syncUrlRound.isEmpty - - def gameIds = syncUrl.flatMap(toGameIds) + def roundMissing = upstream.exists: + case u: Sync.UpstreamUrl => Sync.tryLcc(u.url).isDefined + case _ => false def update(official: Boolean)(relay: RelayRound)(using me: Me)(using mode: Mode) = + val sync = makeSync(me) relay.copy( name = name, caption = caption, - sync = makeSync(me).pipe: sync => - if relay.sync.playing then sync.play(official) else sync, + sync = if relay.sync.playing then sync.play(official) else sync, startsAt = startsAt, finished = ~finished ) private def makeSync(user: User)(using mode: Mode): Sync = RelayRound.Sync( - upstream = syncUrl - .flatMap(cleanUrl) - .map { u => - RelayRound.Sync.UpstreamUrl(s"$u${syncUrlRound.so(" " +)}") - } - .orElse(gameIds.map { ids => - RelayRound.Sync.UpstreamIds(ids.ids) - }), + upstream = upstream, until = none, nextAt = none, period = period.ifTrue(Granter.ofUser(_.StudyAdmin)(user)), @@ -207,10 +248,19 @@ object RelayRoundForm: Data( name = relay.name, caption = relay.caption, - syncUrl = relay.sync.upstream.map { - case url: RelayRound.Sync.UpstreamUrl => url.withRound.url - case RelayRound.Sync.UpstreamIds(ids) => ids.mkString(" ") - }, + syncSource = relay.sync.upstream + .fold("push"): + case _: Sync.UpstreamUrl => "url" + case _: Sync.UpstreamUrls => "urls" + case _: Sync.UpstreamIds => "ids" + .some, + syncUrl = relay.sync.upstream.collect: + case url: Sync.UpstreamUrl => url.withoutRound, + syncUrls = relay.sync.upstream.collect: + case url: Sync.UpstreamUrl => Sync.UpstreamUrls(List(url.withoutRound)) + case urls: Sync.UpstreamUrls => urls, + syncIds = relay.sync.upstream.collect: + case ids: Sync.UpstreamIds => ids, syncUrlRound = relay.sync.upstream.flatMap(_.asUrl).flatMap(_.withRound.round), startsAt = relay.startsAt, finished = relay.finished.option(true), diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index c389d3145a39e..39c038579e1bb 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -81,7 +81,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): private def page(title: String, nav: FormNavigation)(using Context) = Page(title) .css("bits.relay.form") - .js(EsmInit("bits.flatpickr")) + .js(List(EsmInit("bits.flatpickr"), EsmInit("bits.relayForm")).map(some)) .wrap: body => main(cls := "page page-menu")( navigationMenu(nav), @@ -154,25 +154,77 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): form3.input(_) ) ), - form3.group( - form("syncUrl"), - trb.sourceUrlOrGameIds(), - help = frag( - trb.sourceUrlHelp(), - br, - trb.gameIdsHelp(), - br, - "Or leave empty to push games from another program." - ).some - )(form3.input(_)), - form3 - .group( - form("syncUrlRound"), - trb.roundNumber(), - help = frag("Only for livechesscloud source URLs").some + form3.fieldset("Source")(cls := "box-pad")( + form3.group( + form("syncSource"), + "Where do the games come from?" + )(form3.select(_, RelayRoundForm.sourceTypes)), + div( + cls := List( + "relay-form__sync relay-form__sync-url" -> true, + "form-split relay-form__sync-lcc" -> isLcc + ) )( - form3.input(_, typ = "number") - )(cls := (!isLcc).option("none")), + form3.group( + form("syncUrl"), + trb.sourceUrl(), + help = trb.sourceUrlHelp().some, + half = isLcc + )(form3.input(_)), + isLcc.option: + form3.group( + form("syncUrlRound"), + trb.roundNumber(), + help = frag("Only for livechesscloud source URLs").some, + half = true + )(form3.input(_, typ = "number")) + ), + form3.group( + form("syncUrls"), + "Multiple source URLs, one per line.", + help = frag("The games will be combined in the order of the URLs.").some, + half = false + )(form3.textarea(_)(rows := 5))(cls := "relay-form__sync relay-form__sync-urls none"), + form3.group( + form("syncIds"), + trb.sourceGameIds(), + half = false + )(form3.input(_))(cls := "relay-form__sync relay-form__sync-ids none"), + div(cls := "form-group relay-form__sync relay-form__sync-push none")( + p( + "Send your local games to Lichess using ", + a(href := "https://github.com/lichess-org/broadcaster")("the Lichess Broadcaster App"), + "." + ) + ), + form3.split(cls := "relay-form__sync relay-form__sync-url relay-form__sync-urls none")( + form3.group( + form("onlyRound"), + raw("Filter games by round number"), + help = frag( + "Optional, only keep games from the source that match a round number." + ).some, + half = true + )(form3.input(_, typ = "number"))(cls := List("none" -> isLcc)), + form3.group( + form("slices"), + raw("Select slices of the games"), + help = frag( + "Optional. Select games based on their position in the source.", + br, + pre(""" +1 only select the first board +1-4 only select the first 4 boards +1,2,3,4 same as above, first 4 boards +11-15,21-25 boards 11 to 15, and boards 21 to 25 +2,3,7-9 boards 2, 3, 7, 8, and 9 +"""), + "Slicing is done after filtering by round number." + ).some, + half = true + )(form3.input(_)) + ) + ), form3.split( form3.group( form("startsAt"), @@ -209,33 +261,6 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): )(form3.input(_, typ = "number")) ) ), - form3.split( - form3.group( - form("onlyRound"), - raw("Filter games by round number"), - help = frag( - "Optional, only keep games from the source that match a round number." - ).some, - half = true - )(form3.input(_, typ = "number")), - form3.group( - form("slices"), - raw("Select slices of the games"), - help = frag( - "Optional. Select games based on their position in the source.", - br, - pre(""" -1 only select the first board -1-4 only select the first 4 boards -1,2,3,4 same as above, first 4 boards -11-15,21-25 boards 11 to 15, and boards 21 to 25 -2,3,7-9 boards 2, 3, 7, 8, and 9 -"""), - "Slicing is done after filtering by round number." - ).some, - half = true - )(form3.input(_)) - ), form3.actions( a(href := routes.RelayTour.show(t.slug, t.id))(trans.site.cancel()), form3.submit(trans.site.apply()) @@ -470,7 +495,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): ) ) - def grouping(form: Form[RelayTourForm.Data])(using Context) = + private def grouping(form: Form[RelayTourForm.Data])(using Context) = form3.split(cls := "relay-form__grouping")( form3.group( form("grouping"), diff --git a/modules/ui/src/main/helper/Form3.scala b/modules/ui/src/main/helper/Form3.scala index 27d75919f35d2..e7a014b499010 100644 --- a/modules/ui/src/main/helper/Form3.scala +++ b/modules/ui/src/main/helper/Form3.scala @@ -6,6 +6,7 @@ import scalalib.Render import lila.ui.ScalatagsTemplate.{ *, given } import lila.core.user.FlairApi import lila.core.i18n.{ I18nKey as trans, Translate } +import scalatags.Text.TypedTag final class Form3(formHelper: FormHelper & I18nHelper, flairApi: FlairApi): diff --git a/translation/source/broadcast.xml b/translation/source/broadcast.xml index 011b9d493b95c..d7ddd12a58b7c 100644 --- a/translation/source/broadcast.xml +++ b/translation/source/broadcast.xml @@ -23,9 +23,9 @@ Short tournament description Full tournament description Optional long description of the tournament. %1$s is available. Length must be less than %2$s characters. - Source URL, or game IDs + Source URL, or game IDs URL that Lichess will check to get PGN updates. It must be publicly accessible from the Internet. - Alternatively, you can enter up to 64 Lichess game IDs, separated by spaces. + Up to 64 Lichess game IDs, separated by spaces. Start date in your own timezone Optional, if you know when the event starts Credit the source diff --git a/ui/analyse/src/study/relay/interfaces.ts b/ui/analyse/src/study/relay/interfaces.ts index d70bd288bbf25..4a0d89a0a04e4 100644 --- a/ui/analyse/src/study/relay/interfaces.ts +++ b/ui/analyse/src/study/relay/interfaces.ts @@ -44,6 +44,7 @@ export interface RelayTour { export interface RelaySync { ongoing: boolean; url?: string; + urls?: [{ url: string }]; ids?: string; log: LogEvent[]; delay?: number; diff --git a/ui/analyse/src/study/relay/relayManagerView.ts b/ui/analyse/src/study/relay/relayManagerView.ts index 21d7fbb86f598..7cc2bd15db661 100644 --- a/ui/analyse/src/study/relay/relayManagerView.ts +++ b/ui/analyse/src/study/relay/relayManagerView.ts @@ -7,7 +7,8 @@ import { side as studyViewSide } from '../studyView'; import StudyCtrl from '../studyCtrl'; export default function (ctrl: RelayCtrl, study: StudyCtrl): MaybeVNode { - const contributor = ctrl.members.canContribute(); + const contributor = ctrl.members.canContribute(), + sync = ctrl.data.sync; return contributor || study.data.admin ? h('div.relay-admin__container', [ contributor @@ -16,9 +17,7 @@ export default function (ctrl: RelayCtrl, study: StudyCtrl): MaybeVNode { h('span.text', { attrs: dataIcon(licon.RadioTower) }, 'Broadcast manager'), h('a', { attrs: { href: `/broadcast/round/${ctrl.id}/edit`, 'data-icon': licon.Gear } }), ]), - ctrl.data.sync?.url || ctrl.data.sync?.ids - ? (ctrl.data.sync.ongoing ? stateOn : stateOff)(ctrl) - : null, + sync?.url || sync?.ids || sync?.urls ? (sync.ongoing ? stateOn : stateOff)(ctrl) : null, renderLog(ctrl), ]) : undefined, @@ -52,6 +51,7 @@ function renderLog(ctrl: RelayCtrl) { function stateOn(ctrl: RelayCtrl) { const sync = ctrl.data.sync, url = sync?.url, + urls = sync?.urls, ids = sync?.ids; return h( 'div.state.on.clickable', @@ -67,6 +67,8 @@ function stateOn(ctrl: RelayCtrl) { ] : ids ? ['Connected to', h('br'), ids.length, ' game(s)'] + : urls + ? ['Connected to', h('br'), urls.length, ' urls'] : [], ), ], diff --git a/ui/bits/css/relay/_form.scss b/ui/bits/css/relay/_form.scss index 48e0c22d62b45..62030d3ddcda8 100644 --- a/ui/bits/css/relay/_form.scss +++ b/ui/bits/css/relay/_form.scss @@ -55,6 +55,16 @@ } } +.relay-form__sync-lcc { + .form-half:first-child { + flex: 0 1 76%; + } + .form-half:last-child { + flex: 0 1 20%; + min-width: 3ch; + } +} + @include mq-subnav-side { .relay-form__subnav__group { display: block; diff --git a/ui/bits/src/bits.relayForm.ts b/ui/bits/src/bits.relayForm.ts index f1f177ebd95b5..3c1b2b7a63d1f 100644 --- a/ui/bits/src/bits.relayForm.ts +++ b/ui/bits/src/bits.relayForm.ts @@ -7,6 +7,20 @@ site.load.then(() => { selectClicks: $('.select-image, .drop-target'), selectDrags: $('.drop-target'), }); + + pinnedStreamerForm(); + + const $source = $('#form3-syncSource'), + showSource = () => + $('.relay-form__sync').each(function (this: HTMLElement) { + this.classList.toggle('none', !this.classList.contains(`relay-form__sync-${$source.val()}`)); + }); + + $source.on('change', showSource); + showSource(); +}); + +function pinnedStreamerForm() { const pinned = document.querySelector('.relay-pinned-streamer-edit') as HTMLElement; if (!pinned) return; @@ -27,4 +41,4 @@ site.load.then(() => { else deleteImageBtn.onclick = () => fetch(deleteImageBtn.dataset.postUrl!, { method: 'POST', body: new FormData() }).then(site.reload); -}); +} diff --git a/ui/common/css/form/_form3.scss b/ui/common/css/form/_form3.scss index 9d000a5dc0d99..6b49d14c12ca6 100644 --- a/ui/common/css/form/_form3.scss +++ b/ui/common/css/form/_form3.scss @@ -133,10 +133,12 @@ textarea.form-control { .form-fieldset { @extend %box-radius; - margin: 1.5em 0; + margin: 1rem 0 3rem 0; border-color: rgba(128, 128, 128, 0.3); + padding: 2rem 2rem 0 2rem; legend { + padding: 0 1em; text-align: center; font-style: italic; font-size: 1.2em; diff --git a/ui/simul/css/_form.scss b/ui/simul/css/_form.scss index 664ffad554216..e150195fd23a9 100644 --- a/ui/simul/css/_form.scss +++ b/ui/simul/css/_form.scss @@ -23,11 +23,4 @@ } } } - - .form-fieldset { - legend { - padding: 0 1em; - } - padding: 1em; - } } From c375705198429ff65c324f8a45e509148006dd27 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 13:19:42 +0200 Subject: [PATCH 142/168] mapN --- modules/relay/src/main/RelayGame.scala | 10 +++------- 1 file changed, 3 insertions(+), 7 deletions(-) diff --git a/modules/relay/src/main/RelayGame.scala b/modules/relay/src/main/RelayGame.scala index 8b59b6f530cd0..dee3cc97ad08f 100644 --- a/modules/relay/src/main/RelayGame.scala +++ b/modules/relay/src/main/RelayGame.scala @@ -100,13 +100,9 @@ private object RelayGame: .map(_.trim) .flatMap: s => s.split('-').toList.map(_.trim) match - case Nil => none - case from :: Nil => from.toIntOption.map(f => Slice(f, f)) - case from :: to :: _ => - for - f <- from.toIntOption - t <- to.toIntOption - yield Slice(f, t) + case Nil => none + case from :: Nil => from.toIntOption.map(f => Slice(f, f)) + case from :: to :: _ => (from.toIntOption, to.toIntOption).mapN(Slice.apply) def show(slices: List[Slice]): String = slices .map: From 89c334cb8b3c37364db415dbfd58b7c8113b151d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 14:46:12 +0200 Subject: [PATCH 143/168] refactor broadcast sync source --- bin/mongodb/relay-lcc-migrate.js | 2 +- modules/relay/src/main/RelayApi.scala | 5 +- modules/relay/src/main/RelayDelay.scala | 22 ++++----- modules/relay/src/main/RelayFetch.scala | 23 +++++---- modules/relay/src/main/RelayFormat.scala | 36 ++++++-------- modules/relay/src/main/RelayRound.scala | 18 ++++--- modules/relay/src/main/RelayRoundForm.scala | 55 +++++++++++---------- modules/relay/src/main/ui/FormUi.scala | 38 +++++++------- ui/bits/css/relay/_form.scss | 10 ---- 9 files changed, 100 insertions(+), 109 deletions(-) diff --git a/bin/mongodb/relay-lcc-migrate.js b/bin/mongodb/relay-lcc-migrate.js index e2c53262ca08c..e716fe2167898 100644 --- a/bin/mongodb/relay-lcc-migrate.js +++ b/bin/mongodb/relay-lcc-migrate.js @@ -7,7 +7,7 @@ db.relay.find({ 'sync.upstream.url': /view\.livechesscloud\.com/ }).forEach(rela const id = url.match(regex)[1]; const round = parseInt(url.split(' ')[1]) || 1; if (!id) throw new Error('No id in ' + url); - db.relay.updateOne({ _id: relay._id }, { $set: { 'sync.upstream': { lcc: { id, round } } } }); + db.relay.updateOne({ _id: relay._id }, { $set: { 'sync.upstream': { lcc: id, round } } }); done++; } catch (e) { failed++; diff --git a/modules/relay/src/main/RelayApi.scala b/modules/relay/src/main/RelayApi.scala index 760270c8a686a..ca32574e5ab97 100644 --- a/modules/relay/src/main/RelayApi.scala +++ b/modules/relay/src/main/RelayApi.scala @@ -9,7 +9,7 @@ import scala.util.chaining.* import lila.db.dsl.{ *, given } import lila.memo.{ CacheApi, PicfitApi } -import lila.relay.RelayRound.WithTour +import lila.relay.RelayRound.{ WithTour, Sync } import lila.core.perm.Granter import lila.core.study.data.StudyName import lila.study.{ Settings, Study, StudyApi, StudyId, StudyMaker, StudyRepo, StudyTopic } @@ -244,7 +244,8 @@ final class RelayApi( def requestPlay(id: RelayRoundId, v: Boolean): Funit = WithRelay(id): relay => - relay.sync.upstream.foreach(formatApi.refresh) + relay.sync.upstream.collect: + case f: Sync.FetchableUpstream => formatApi.refresh(f) isOfficial(relay.id).flatMap: official => update(relay): r => if v diff --git a/modules/relay/src/main/RelayDelay.scala b/modules/relay/src/main/RelayDelay.scala index ee90565ae3e9f..81add186ced9d 100644 --- a/modules/relay/src/main/RelayDelay.scala +++ b/modules/relay/src/main/RelayDelay.scala @@ -6,7 +6,7 @@ import scalalib.model.Seconds import lila.db.dsl.{ *, given } import lila.memo.CacheApi -import lila.relay.RelayRound.Sync.UpstreamUrl +import lila.relay.RelayRound.Sync.FetchableUpstream import lila.study.MultiPgn final private class RelayDelay(colls: RelayColls)(using Executor): @@ -14,9 +14,9 @@ final private class RelayDelay(colls: RelayColls)(using Executor): import RelayDelay.* def apply( - url: UpstreamUrl, + url: FetchableUpstream, round: RelayRound, - doFetchUrl: (UpstreamUrl, Max) => Fu[RelayGames] + doFetchUrl: (FetchableUpstream, Max) => Fu[RelayGames] ): Fu[RelayGames] = dedupCache(url, round, () => doFetchUrl(url, RelayFetch.maxChapters)) .flatMap: latest => @@ -31,10 +31,10 @@ final private class RelayDelay(colls: RelayColls)(using Executor): private val cache = CacheApi.scaffeineNoScheduler .initialCapacity(8) .maximumSize(128) - .build[UpstreamUrl, GamesSeenBy]() + .build[FetchableUpstream, GamesSeenBy]() .underlying - def apply(url: UpstreamUrl, round: RelayRound, doFetch: () => Fu[RelayGames]) = + def apply(url: FetchableUpstream, round: RelayRound, doFetch: () => Fu[RelayGames]) = cache.asMap .compute( url, @@ -51,10 +51,10 @@ final private class RelayDelay(colls: RelayColls)(using Executor): private object store: - private def idOf(upstream: UpstreamUrl, at: Instant) = s"${upstream.url} ${at.toSeconds}" - private val longPast = java.time.Instant.ofEpochMilli(0) + private def idOf(upstream: FetchableUpstream, at: Instant) = s"${upstream.url} ${at.toSeconds}" + private val longPast = java.time.Instant.ofEpochMilli(0) - def putIfNew(upstream: UpstreamUrl, games: RelayGames): Funit = + def putIfNew(upstream: FetchableUpstream, games: RelayGames): Funit = val newPgn = RelayGame.iso.from(games).toPgnStr getLatestPgn(upstream).flatMap: case Some(latestPgn) if latestPgn == newPgn => funit @@ -64,14 +64,14 @@ final private class RelayDelay(colls: RelayColls)(using Executor): colls.delay: _.insert.one(doc).void - def get(upstream: UpstreamUrl, delay: Seconds): Fu[Option[RelayGames]] = + def get(upstream: FetchableUpstream, delay: Seconds): Fu[Option[RelayGames]] = getPgn(upstream, delay).map2: pgn => RelayGame.iso.to(MultiPgn.split(pgn, Max(999))) - private def getLatestPgn(upstream: UpstreamUrl): Fu[Option[PgnStr]] = + private def getLatestPgn(upstream: FetchableUpstream): Fu[Option[PgnStr]] = getPgn(upstream, Seconds(0)) - private def getPgn(upstream: UpstreamUrl, delay: Seconds): Fu[Option[PgnStr]] = + private def getPgn(upstream: FetchableUpstream, delay: Seconds): Fu[Option[PgnStr]] = colls.delay: _.find( $doc( diff --git a/modules/relay/src/main/RelayFetch.scala b/modules/relay/src/main/RelayFetch.scala index 98de665690b4d..0271e303e391b 100644 --- a/modules/relay/src/main/RelayFetch.scala +++ b/modules/relay/src/main/RelayFetch.scala @@ -6,6 +6,7 @@ import chess.{ Outcome, Ply } import com.github.blemale.scaffeine.LoadingCache import io.mola.galimatias.URL import play.api.libs.json.* +import scalalib.model.Seconds import lila.core.lilaism.LilaInvalid import lila.common.LilaScheduler @@ -13,10 +14,8 @@ import lila.game.{ GameRepo, PgnDump } import lila.memo.CacheApi import lila.study.{ MultiPgn, StudyPgnImport } import lila.tree.Node.Comments - -import RelayRound.Sync.{ UpstreamIds, UpstreamUrl, UpstreamUrls } -import RelayFormat.CanProxy -import scalalib.model.Seconds +import lila.relay.RelayRound.Sync +import lila.relay.RelayFormat.CanProxy final private class RelayFetch( sync: RelaySync, @@ -142,7 +141,7 @@ final private class RelayFetch( Seconds(60) else round.sync.period | Seconds: - if upstream.asUrl.exists(_.isLcc) && !tour.official then 12 + if upstream.isLcc && !tour.official then 12 else 5 updating: _.withSync: @@ -166,10 +165,12 @@ final private class RelayFetch( ) private def fetchGames(rt: RelayRound.WithTour): Fu[RelayGames] = + given CanProxy = CanProxy(rt.tour.official) rt.round.sync.upstream.so: - case UpstreamIds(ids) => fetchFromGameIds(rt.tour, ids) - case url: UpstreamUrl => delayer(url, rt.round, fetchFromUpstream(using CanProxy(rt.tour.official))) - case UpstreamUrls(urls) => + case Sync.UpstreamIds(ids) => fetchFromGameIds(rt.tour, ids) + case lcc: Sync.UpstreamLcc => fetchFromUpstream(lcc, RelayFetch.maxChapters) + case url: Sync.UpstreamUrl => delayer(url, rt.round, fetchFromUpstream) + case Sync.UpstreamUrls(urls) => urls .traverse: url => delayer(url, rt.round, fetchFromUpstream(using CanProxy(rt.tour.official))) @@ -194,10 +195,12 @@ final private class RelayFetch( } .flatMap(multiPgnToGames(_).toFuture) - private def fetchFromUpstream(using canProxy: CanProxy)(upstream: UpstreamUrl, max: Max): Fu[RelayGames] = + private def fetchFromUpstream(using + canProxy: CanProxy + )(upstream: Sync.FetchableUpstream, max: Max): Fu[RelayGames] = import DgtJson.* formatApi - .get(upstream.withRound) + .get(upstream) .flatMap { case RelayFormat.SingleFile(doc) => doc.format match diff --git a/modules/relay/src/main/RelayFormat.scala b/modules/relay/src/main/RelayFormat.scala index 21d55e19cbd8c..210e34be398e6 100644 --- a/modules/relay/src/main/RelayFormat.scala +++ b/modules/relay/src/main/RelayFormat.scala @@ -29,44 +29,38 @@ final private class RelayFormatApi( )(using Executor): import RelayFormat.* - import RelayRound.Sync.{ Upstream, UpstreamUrl, UpstreamLcc } + import RelayRound.Sync.{ FetchableUpstream, UpstreamUrl, UpstreamLcc } - private val cache = cacheApi[(Upstream, CanProxy), RelayFormat](64, "relay.format"): + private val cache = cacheApi[(FetchableUpstream, CanProxy), RelayFormat](64, "relay.format"): _.expireAfterWrite(5 minutes) .buildAsyncFuture: (url, proxy) => guessFormat(url)(using proxy) - def get(upstream: Upstream)(using proxy: CanProxy): Fu[RelayFormat] = + def get(upstream: FetchableUpstream)(using proxy: CanProxy): Fu[RelayFormat] = cache.get(upstream -> proxy) - def refresh(upstream: Upstream): Unit = + def refresh(upstream: FetchableUpstream): Unit = CanProxy .from(List(false, true)) .foreach: proxy => cache.invalidate(upstream -> proxy) - private def guessFormat(upstream: Upstream)(using CanProxy): Fu[RelayFormat] = { + private def guessFormat(upstream: FetchableUpstream)(using CanProxy): Fu[RelayFormat] = { - // val originalUrl = URL.parse(upstream.url) + def parsedUrl = URL.parse(upstream.url) - // http://view.livechesscloud.com/ed5fb586-f549-4029-a470-d590f8e30c76 - def guessLcc(using CanProxy): Fu[Option[RelayFormat]] = upstream match - case UpstreamLcc(id, round) => - guessManyFiles: - URL.parse: - s"http://1.pool.livechesscloud.com/get/$id/round-$round/index.json" - case _ => fuccess(none) - - def guessSingleFile(url: URL)(using CanProxy): Fu[Option[RelayFormat]] = + def guessSingleFile(url: URL): Fu[Option[RelayFormat]] = List( url.some, - (!url.pathSegments.contains(mostCommonSingleFileName)).option(addPart(url, mostCommonSingleFileName)) + (!url.pathSegments.contains(mostCommonSingleFileName)).option( + addPart(url, mostCommonSingleFileName) + ) ).flatten.distinct .findM(looksLikePgn) .dmap2: (u: URL) => SingleFile(pgnDoc(u)) - def guessManyFiles(url: URL)(using CanProxy): Fu[Option[RelayFormat]] = + def guessManyFiles(url: URL): Fu[Option[RelayFormat]] = (List(url) ::: mostCommonIndexNames .filterNot(url.pathSegments.contains) .map(addPart(url, _))) @@ -87,10 +81,10 @@ final private class RelayFormatApi( ManyFiles(index, _) .dmap(_.orElse(ManyFilesLater(index).some)) - guessLcc(originalUrl) - .orElse(guessSingleFile(originalUrl)) - .orElse(guessManyFiles(originalUrl)) - .orFailWith(LilaInvalid(s"No games found at $originalUrl")) + guessSingleFile(parsedUrl) + .orElse(guessManyFiles(parsedUrl)) + .orFailWith(LilaInvalid(s"No games found at $upstream")) + }.addEffect { format => logger.info(s"guessed format of $upstream: $format") } diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index 84c31915d0699..b200be6266149 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -114,16 +114,18 @@ object RelayRound: override def toString = upstream.toString object Sync: - sealed trait Upstream - case class UpstreamUrl(url: String) extends Upstream - case class UpstreamLcc(id: String, round: Int) extends Upstream: - def url = s"https://view.livechesscloud.com/#$id" + sealed trait Upstream: + def isLcc = false + sealed trait FetchableUpstream extends Upstream: + def url: String + case class UpstreamUrl(url: String) extends FetchableUpstream + case class UpstreamLcc(lcc: String, round: Int) extends FetchableUpstream: + override def isLcc = true + def id = lcc + def url = s"http://1.pool.livechesscloud.com/get/$id/round-$round/index.json" + def viewUrl = s"https://view.livechesscloud.com/#$id" case class UpstreamUrls(urls: List[UpstreamUrl]) extends Upstream case class UpstreamIds(ids: List[GameId]) extends Upstream - val LccRegex = """.*view\.livechesscloud\.com/?#?([0-9a-f\-]+)""".r - def tryLcc(url: String): Option[UpstreamLcc] = url match - case LccRegex(id) => UpstreamLcc(id, 1).some - case _ => none trait AndTour: val tour: RelayTour diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index 8c144eb8788b7..98d972f760c67 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -42,25 +42,29 @@ final class RelayRoundForm(using mode: Mode): _.ids.mkString(" ") ) + val lccMapping = mapping( + "id" -> cleanText(minLength = 10, maxLength = 40), + "round" -> number(min = 1, max = 999) + )(Sync.UpstreamLcc.apply)(unapply) + val roundMapping = mapping( - "name" -> cleanText(minLength = 3, maxLength = 80).into[RelayRound.Name], - "caption" -> optional(cleanText(minLength = 3, maxLength = 80).into[RelayRound.Caption]), - "syncSource" -> optional(stringIn(sourceTypes.map(_._1).toSet)), - "syncUrl" -> optional(of[Sync.UpstreamUrl]), - "syncUrls" -> optional(of[Sync.UpstreamUrls]), - "syncIds" -> optional(of[Sync.UpstreamIds]), - "syncUrlRound" -> optional(number(min = 1, max = 999)), - "startsAt" -> optional(ISOInstantOrTimestamp.mapping), - "finished" -> optional(boolean), - "period" -> optional(number(min = 2, max = 60).into[Seconds]), - "delay" -> optional(number(min = 0, max = RelayDelay.maxSeconds.value).into[Seconds]), - "onlyRound" -> optional(number(min = 1, max = 999)), + "name" -> cleanText(minLength = 3, maxLength = 80).into[RelayRound.Name], + "caption" -> optional(cleanText(minLength = 3, maxLength = 80).into[RelayRound.Caption]), + "syncSource" -> optional(stringIn(sourceTypes.map(_._1).toSet)), + "syncUrl" -> optional(of[Sync.UpstreamUrl]), + "syncUrls" -> optional(of[Sync.UpstreamUrls]), + "syncLcc" -> optional(lccMapping), + "syncIds" -> optional(of[Sync.UpstreamIds]), + "startsAt" -> optional(ISOInstantOrTimestamp.mapping), + "finished" -> optional(boolean), + "period" -> optional(number(min = 2, max = 60).into[Seconds]), + "delay" -> optional(number(min = 0, max = RelayDelay.maxSeconds.value).into[Seconds]), + "onlyRound" -> optional(number(min = 1, max = 999)), "slices" -> optional: nonEmptyText .transform[List[RelayGame.Slice]](RelayGame.Slices.parse, RelayGame.Slices.show) )(Data.apply)(unapply) - .verifying("This source requires a round number. See the new form field below.", !_.roundMissing) def create(trs: RelayTour.WithRounds) = Form( roundMapping @@ -77,6 +81,7 @@ object RelayRoundForm: val sourceTypes = List( "url" -> "Single PGN URL", "urls" -> "Combine several PGN URLs", + "lcc" -> "LiveChessCloud page", "ids" -> "Lichess game IDs", "push" -> "Push local games" ) @@ -117,8 +122,7 @@ object RelayRoundForm: name = RelayRound.Name(guessName | s"Round ${nextNumber}"), caption = prev.flatMap(_.caption), syncSource = prev.map(Data.make).flatMap(_.syncSource), - syncUrl = nextLcc.map(l => Sync.UpstreamUrl(l.url)), - syncUrlRound = nextLcc.map(_.round), + syncLcc = nextLcc, startsAt = guessDate, period = prev.flatMap(_.sync.period), delay = prev.flatMap(_.sync.delay), @@ -184,8 +188,8 @@ object RelayRoundForm: syncSource: Option[String], syncUrl: Option[Sync.UpstreamUrl] = None, syncUrls: Option[Sync.UpstreamUrls] = None, + syncLcc: Option[Sync.UpstreamLcc] = None, syncIds: Option[Sync.UpstreamIds] = None, - syncUrlRound: Option[Int] = None, startsAt: Option[Instant] = None, finished: Option[Boolean] = None, period: Option[Seconds] = None, @@ -194,18 +198,13 @@ object RelayRoundForm: slices: Option[List[RelayGame.Slice]] = None ): def upstream: Option[Sync.Upstream] = syncSource match - case None => syncUrl.orElse(syncUrls).orElse(syncIds) - case Some("url") => - syncUrl.map: u => - (Sync.tryLcc(u.url), syncUrlRound).mapN((l, r) => l.copy(round = r)) | u + case None => syncUrl.orElse(syncUrls).orElse(syncIds) + case Some("url") => syncUrl case Some("urls") => syncUrls + case Some("lcc") => syncLcc case Some("ids") => syncIds case _ => None - def roundMissing = upstream.exists: - case u: Sync.UpstreamUrl => Sync.tryLcc(u.url).isDefined - case _ => false - def update(official: Boolean)(relay: RelayRound)(using me: Me)(using mode: Mode) = val sync = makeSync(me) relay.copy( @@ -252,16 +251,18 @@ object RelayRoundForm: .fold("push"): case _: Sync.UpstreamUrl => "url" case _: Sync.UpstreamUrls => "urls" + case _: Sync.UpstreamLcc => "lcc" case _: Sync.UpstreamIds => "ids" .some, syncUrl = relay.sync.upstream.collect: - case url: Sync.UpstreamUrl => url.withoutRound, + case url: Sync.UpstreamUrl => url, syncUrls = relay.sync.upstream.collect: - case url: Sync.UpstreamUrl => Sync.UpstreamUrls(List(url.withoutRound)) + case url: Sync.UpstreamUrl => Sync.UpstreamUrls(List(url)) case urls: Sync.UpstreamUrls => urls, + syncLcc = relay.sync.upstream.collect: + case lcc: Sync.UpstreamLcc => lcc, syncIds = relay.sync.upstream.collect: case ids: Sync.UpstreamIds => ids, - syncUrlRound = relay.sync.upstream.flatMap(_.asUrl).flatMap(_.withRound.round), startsAt = relay.startsAt, finished = relay.finished.option(true), period = relay.sync.period, diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 15e4b7a97a33d..0000349137d2b 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -129,7 +129,6 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): private def inner(form: Form[RelayRoundForm.Data], url: play.api.mvc.Call, t: RelayTour, create: Boolean)( using ctx: Context ) = - val isLcc = form("syncUrl").value.exists(RelayRound.Sync.UpstreamUrl.LccRegex.matches) postForm(cls := "form3", action := url)( (!Granter.opt(_.StudyAdmin)).option: div(cls := "form-group")( @@ -159,25 +158,26 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): form("syncSource"), "Where do the games come from?" )(form3.select(_, RelayRoundForm.sourceTypes)), - div( - cls := List( - "relay-form__sync relay-form__sync-url" -> true, - "form-split relay-form__sync-lcc" -> isLcc - ) - )( + form3.group( + form("syncUrl"), + trb.sourceUrl(), + help = trb.sourceUrlHelp().some + )(form3.input(_))(cls := "relay-form__sync relay-form__sync-url"), + form3.split(cls := "relay-form__sync relay-form__sync-lcc none")( form3.group( - form("syncUrl"), - trb.sourceUrl(), - help = trb.sourceUrlHelp().some, - half = isLcc + form("syncLcc.id"), + "Tournament ID", + help = frag( + "From the LCC page URL. The ID looks like this: ", + pre("f1943ec6-4992-45d9-969d-a0aff688b404") + ).some, + half = true )(form3.input(_)), - isLcc.option: - form3.group( - form("syncUrlRound"), - trb.roundNumber(), - help = frag("Only for livechesscloud source URLs").some, - half = true - )(form3.input(_, typ = "number")) + form3.group( + form("syncLcc.round"), + trb.roundNumber(), + half = true + )(form3.input(_, typ = "number")) ), form3.group( form("syncUrls"), @@ -205,7 +205,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): "Optional, only keep games from the source that match a round number." ).some, half = true - )(form3.input(_, typ = "number"))(cls := List("none" -> isLcc)), + )(form3.input(_, typ = "number")), form3.group( form("slices"), raw("Select slices of the games"), diff --git a/ui/bits/css/relay/_form.scss b/ui/bits/css/relay/_form.scss index 62030d3ddcda8..48e0c22d62b45 100644 --- a/ui/bits/css/relay/_form.scss +++ b/ui/bits/css/relay/_form.scss @@ -55,16 +55,6 @@ } } -.relay-form__sync-lcc { - .form-half:first-child { - flex: 0 1 76%; - } - .form-half:last-child { - flex: 0 1 20%; - min-width: 3ch; - } -} - @include mq-subnav-side { .relay-form__subnav__group { display: block; From 9950b2e6f669b541ca3f0772d53c1a318a12c44d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 14:51:12 +0200 Subject: [PATCH 144/168] fix lcc format --- modules/relay/src/main/RelayFormat.scala | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/modules/relay/src/main/RelayFormat.scala b/modules/relay/src/main/RelayFormat.scala index 210e34be398e6..6938a708c357e 100644 --- a/modules/relay/src/main/RelayFormat.scala +++ b/modules/relay/src/main/RelayFormat.scala @@ -49,6 +49,8 @@ final private class RelayFormatApi( def parsedUrl = URL.parse(upstream.url) + def guessLcc: Fu[Option[RelayFormat]] = upstream.isLcc.so(guessManyFiles(parsedUrl)) + def guessSingleFile(url: URL): Fu[Option[RelayFormat]] = List( url.some, @@ -81,7 +83,8 @@ final private class RelayFormatApi( ManyFiles(index, _) .dmap(_.orElse(ManyFilesLater(index).some)) - guessSingleFile(parsedUrl) + guessLcc + .orElse(guessSingleFile(parsedUrl)) .orElse(guessManyFiles(parsedUrl)) .orFailWith(LilaInvalid(s"No games found at $upstream")) From 6a83f4fec23e3788605be8b89a1741df1b5f05dc Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 15:00:30 +0200 Subject: [PATCH 145/168] guess lcc sources --- modules/relay/src/main/RelayRound.scala | 6 +++- modules/relay/src/main/RelayRoundForm.scala | 32 ++++++++++++++------- 2 files changed, 26 insertions(+), 12 deletions(-) diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index b200be6266149..3b70161ee0674 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -118,7 +118,10 @@ object RelayRound: def isLcc = false sealed trait FetchableUpstream extends Upstream: def url: String - case class UpstreamUrl(url: String) extends FetchableUpstream + case class UpstreamUrl(url: String) extends FetchableUpstream: + def findLccId: Option[String] = url match + case LccRegex(id) => id.some + case _ => none case class UpstreamLcc(lcc: String, round: Int) extends FetchableUpstream: override def isLcc = true def id = lcc @@ -126,6 +129,7 @@ object RelayRound: def viewUrl = s"https://view.livechesscloud.com/#$id" case class UpstreamUrls(urls: List[UpstreamUrl]) extends Upstream case class UpstreamIds(ids: List[GameId]) extends Upstream + private val LccRegex = """.*view\.livechesscloud\.com/?#?([0-9a-f\-]+)""".r trait AndTour: val tour: RelayTour diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index 98d972f760c67..3140e3f317b41 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -86,15 +86,16 @@ object RelayRoundForm: "push" -> "Push local games" ) + private val roundNumberRegex = """([^\d]*)(\d{1,2})([^\d]*)""".r + val roundNumberIn: String => Option[Int] = + case roundNumberRegex(_, n, _) => n.toIntOption + case _ => none + def fillFromPrevRounds(rounds: List[RelayRound]): Data = val prevs: Option[(RelayRound, RelayRound)] = rounds.reverse match case a :: b :: _ => (a, b).some case _ => none val prev: Option[RelayRound] = rounds.lastOption - val roundNumberRegex = """([^\d]*)(\d{1,2})([^\d]*)""".r - val roundNumberIn: String => Option[Int] = - case roundNumberRegex(_, n, _) => n.toIntOption - case _ => none def replaceRoundNumber(s: String, n: Int): String = roundNumberRegex.replaceAllIn(s, m => s"${m.group(1)}${n}${m.group(3)}") val prevNumber: Option[Int] = prev.flatMap(p => roundNumberIn(p.name.value)) @@ -197,13 +198,22 @@ object RelayRoundForm: onlyRound: Option[Int] = None, slices: Option[List[RelayGame.Slice]] = None ): - def upstream: Option[Sync.Upstream] = syncSource match - case None => syncUrl.orElse(syncUrls).orElse(syncIds) - case Some("url") => syncUrl - case Some("urls") => syncUrls - case Some("lcc") => syncLcc - case Some("ids") => syncIds - case _ => None + def upstream: Option[Sync.Upstream] = syncSource + .match + case None => syncUrl.orElse(syncUrls).orElse(syncIds) + case Some("url") => syncUrl + case Some("urls") => syncUrls + case Some("lcc") => syncLcc + case Some("ids") => syncIds + case _ => None + .map: + case url: Sync.UpstreamUrl => + val foundLcc = for + lccId <- url.findLccId + round <- roundNumberIn(name.value) + yield Sync.UpstreamLcc(lccId, round) + foundLcc | url + case up => up def update(official: Boolean)(relay: RelayRound)(using me: Me)(using mode: Mode) = val sync = makeSync(me) From 59083047e9dbd3a4ca39306fdea93287117d677c Mon Sep 17 00:00:00 2001 From: Jonathan Gamble <101470903+schlawg@users.noreply.github.com> Date: Wed, 19 Jun 2024 08:59:06 -0500 Subject: [PATCH 146/168] better object equivalence --- ui/.build/src/manifest.ts | 59 ++++++++++++++++++++++----------------- 1 file changed, 33 insertions(+), 26 deletions(-) diff --git a/ui/.build/src/manifest.ts b/ui/.build/src/manifest.ts index ddf070021f87c..65df38abc3708 100644 --- a/ui/.build/src/manifest.ts +++ b/ui/.build/src/manifest.ts @@ -29,7 +29,7 @@ export async function css() { const css: { name: string; hash: string }[] = await Promise.all(files.map(hashMove)); const newCssManifest: Manifest = {}; for (const { name, hash } of css) newCssManifest[name] = { hash }; - if (isEquivalent(newCssManifest, current.css)) return; + if (enumerableEquivalence(newCssManifest, current.css)) return; current.css = shallowSort({ ...current.css, ...newCssManifest }); clearTimeout(writeTimer); writeTimer = setTimeout(write, 500); @@ -53,7 +53,7 @@ export async function js(meta: es.Metafile) { } newJsManifest[out.name].imports = imports; } - if (isEquivalent(newJsManifest, current.js) && fs.existsSync(env.manifestFile)) return; + if (enumerableEquivalence(newJsManifest, current.js) && fs.existsSync(env.manifestFile)) return; current.js = shallowSort({ ...current.js, ...newJsManifest }); clearTimeout(writeTimer); writeTimer = setTimeout(write, 500); @@ -94,7 +94,7 @@ async function write() { await Promise.all([ fs.promises.writeFile(path.join(env.jsDir, `manifest.${hash}.js`), clientManifest), fs.promises.writeFile( - path.join(env.jsDir, `manifest.${env.prod ? 'prod' : 'dev'}.json`), + path.join(env.jsonDir, `manifest.${env.prod ? 'prod' : 'dev'}.json`), JSON.stringify(serverManifest, null, env.prod ? undefined : 2), ), ]); @@ -112,29 +112,6 @@ async function hashMove(src: string) { return { name: path.basename(src, '.css'), hash }; } -function shallowSort(obj: { [key: string]: any }): { [key: string]: any } { - // es6 string properties are insertion order, we need more determinism - const sorted: { [key: string]: any } = {}; - for (const key of Object.keys(obj).sort()) sorted[key] = obj[key]; - return sorted; -} - -function parsePath(path: string) { - const match = path.match(/\/public\/compiled\/(.*)\.([A-Z0-9]+)\.js$/); - return match ? { name: match[1], hash: match[2] } : undefined; -} - -function isEquivalent(a: any, b: any) { - // key order does NOT matter - if (typeof a !== typeof b) return false; - if (Array.isArray(a)) return a.length === b.length && a.every(x => b.includes(x)); - if (typeof a !== 'object') return a === b; - for (const key in a) { - if (!(key in b) || !isEquivalent(a[key], b[key])) return false; - } - return true; -} - async function isComplete() { for (const bundle of [...env.modules.values()].map(x => x.bundles ?? []).flat()) { const name = path.basename(bundle, '.ts'); @@ -152,3 +129,33 @@ async function isComplete() { } return true; } + +function shallowSort(obj: { [key: string]: any }): { [key: string]: any } { + // es6 string properties are insertion order, we need more determinism + const sorted: { [key: string]: any } = {}; + for (const key of Object.keys(obj).sort()) sorted[key] = obj[key]; + return sorted; +} + +function parsePath(path: string) { + const match = path.match(/\/public\/compiled\/(.*)\.([A-Z0-9]+)\.js$/); + return match ? { name: match[1], hash: match[2] } : undefined; +} + +function enumerableEquivalence(a: any, b: any) { + if (a === b) return true; + if (typeof a !== typeof b) return false; + if (Array.isArray(a)) + return a.length === b.length && a.every(x => b.find((y: any) => enumerableEquivalence(x, y))); + if (typeof a !== 'object') return false; + for (const [x, y, deep] of [ + [a, b, true], + [b, a, false], + ]) { + for (const key in x) { + if (!Object.prototype.propertyIsEnumerable.call(x, key)) continue; + if (!(key in y) || (deep && !enumerableEquivalence(x[key], y[key]))) return false; + } + } + return true; +} From 05d21993cb73be3ac33c8b7fcec3c13ac3e29cff Mon Sep 17 00:00:00 2001 From: Jonathan Gamble <101470903+schlawg@users.noreply.github.com> Date: Wed, 19 Jun 2024 09:31:57 -0500 Subject: [PATCH 147/168] use Object.keys instead of prototype --- ui/.build/src/manifest.ts | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/ui/.build/src/manifest.ts b/ui/.build/src/manifest.ts index 65df38abc3708..25b1b9c954a51 100644 --- a/ui/.build/src/manifest.ts +++ b/ui/.build/src/manifest.ts @@ -142,20 +142,20 @@ function parsePath(path: string) { return match ? { name: match[1], hash: match[2] } : undefined; } -function enumerableEquivalence(a: any, b: any) { +function enumerableEquivalence(a: any, b: any): boolean { if (a === b) return true; if (typeof a !== typeof b) return false; if (Array.isArray(a)) - return a.length === b.length && a.every(x => b.find((y: any) => enumerableEquivalence(x, y))); + return ( + Array.isArray(b) && + a.length === b.length && + a.every(x => b.find((y: any) => enumerableEquivalence(x, y))) + ); if (typeof a !== 'object') return false; - for (const [x, y, deep] of [ - [a, b, true], - [b, a, false], - ]) { - for (const key in x) { - if (!Object.prototype.propertyIsEnumerable.call(x, key)) continue; - if (!(key in y) || (deep && !enumerableEquivalence(x[key], y[key]))) return false; - } + const [aKeys, bKeys] = [Object.keys(a), Object.keys(b)]; + if (aKeys.length !== bKeys.length) return false; + for (const key of aKeys) { + if (!bKeys.includes(key) || !enumerableEquivalence(a[key], b[key])) return false; } return true; } From f250da197ecaf2d70a3f6d3b3fc34ff836e81dbb Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 18:57:24 +0200 Subject: [PATCH 148/168] document broadcats round filter --- modules/relay/src/main/ui/FormUi.scala | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 0000349137d2b..5d448a55f6a7b 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -202,7 +202,18 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): form("onlyRound"), raw("Filter games by round number"), help = frag( - "Optional, only keep games from the source that match a round number." + "Optional, only keep games from the source that match a round number.", + br, + "It uses the PGN ", + strong("Round"), + " tag. These would match round 3:", + pre( + """[Round "3"] +[Round "3.1"]""" + ), + "Games without a ", + strong("Round"), + " tag are removed." ).some, half = true )(form3.input(_, typ = "number")), From 0445b434a460e30264a27f384c310f2efffa798a Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 20:56:36 +0200 Subject: [PATCH 149/168] broadcast form wording --- modules/coreI18n/src/main/key.scala | 2 +- modules/relay/src/main/ui/FormUi.scala | 6 +++--- translation/source/broadcast.xml | 2 +- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/modules/coreI18n/src/main/key.scala b/modules/coreI18n/src/main/key.scala index 08b853fee5c82..00da5602b6fc9 100644 --- a/modules/coreI18n/src/main/key.scala +++ b/modules/coreI18n/src/main/key.scala @@ -1665,7 +1665,7 @@ object I18nKey: val `tournamentDescription`: I18nKey = "broadcast:tournamentDescription" val `fullDescription`: I18nKey = "broadcast:fullDescription" val `fullDescriptionHelp`: I18nKey = "broadcast:fullDescriptionHelp" - val `sourceUrl`: I18nKey = "broadcast:sourceUrl" + val `sourceSingleUrl`: I18nKey = "broadcast:sourceSingleUrl" val `sourceUrlHelp`: I18nKey = "broadcast:sourceUrlHelp" val `sourceGameIds`: I18nKey = "broadcast:sourceGameIds" val `startDate`: I18nKey = "broadcast:startDate" diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 5d448a55f6a7b..1db723404d330 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -160,7 +160,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): )(form3.select(_, RelayRoundForm.sourceTypes)), form3.group( form("syncUrl"), - trb.sourceUrl(), + trb.sourceSingleUrl(), help = trb.sourceUrlHelp().some )(form3.input(_))(cls := "relay-form__sync relay-form__sync-url"), form3.split(cls := "relay-form__sync relay-form__sync-lcc none")( @@ -211,9 +211,9 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): """[Round "3"] [Round "3.1"]""" ), - "Games without a ", + "If you set a round number, then games without a ", strong("Round"), - " tag are removed." + " tag are dropped." ).some, half = true )(form3.input(_, typ = "number")), diff --git a/translation/source/broadcast.xml b/translation/source/broadcast.xml index d7ddd12a58b7c..daa6188910282 100644 --- a/translation/source/broadcast.xml +++ b/translation/source/broadcast.xml @@ -23,7 +23,7 @@ Short tournament description Full tournament description Optional long description of the tournament. %1$s is available. Length must be less than %2$s characters. - Source URL, or game IDs + PGN Source URL URL that Lichess will check to get PGN updates. It must be publicly accessible from the Internet. Up to 64 Lichess game IDs, separated by spaces. Start date in your own timezone From b549bc25a76b042771a97ef4fdba5b9159e5c8b0 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 21:21:58 +0200 Subject: [PATCH 150/168] broadcast lcc warning --- modules/relay/src/main/package.scala | 3 ++ modules/relay/src/main/ui/FormUi.scala | 42 +++++++++++++++++--------- ui/bits/css/relay/_form.scss | 10 ++++++ ui/common/css/component/_flash.scss | 11 +++++-- 4 files changed, 49 insertions(+), 17 deletions(-) diff --git a/modules/relay/src/main/package.scala b/modules/relay/src/main/package.scala index 1f6b3065e50d2..51d0f38ad876f 100644 --- a/modules/relay/src/main/package.scala +++ b/modules/relay/src/main/package.scala @@ -4,6 +4,9 @@ export lila.core.lilaism.Lilaism.{ *, given } export lila.common.extensions.* export lila.core.id.{ RelayRoundId, RelayTourId } +val broadcasterUrl = + "https://github.com/lichess-org/broadcaster#user-content-lichess-broadcaster" + private val logger = lila.log("relay") private type RelayGames = Vector[RelayGame] diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index 1db723404d330..ce89e092a9d4f 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -163,21 +163,33 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): trb.sourceSingleUrl(), help = trb.sourceUrlHelp().some )(form3.input(_))(cls := "relay-form__sync relay-form__sync-url"), - form3.split(cls := "relay-form__sync relay-form__sync-lcc none")( - form3.group( - form("syncLcc.id"), - "Tournament ID", - help = frag( - "From the LCC page URL. The ID looks like this: ", - pre("f1943ec6-4992-45d9-969d-a0aff688b404") - ).some, - half = true - )(form3.input(_)), - form3.group( - form("syncLcc.round"), - trb.roundNumber(), - half = true - )(form3.input(_, typ = "number")) + div(cls := "relay-form__sync relay-form__sync-lcc none")( + (!Granter.opt(_.Relay)).option( + flashMessage("box")( + p(strong("Please use the ", a(href := broadcasterUrl)("Lichess Broadcaster App"))), + p( + "LiveChessCloud support is deprecated and will be removed soon.", + br, + "If you need help, please contact us at broadcast@lichess.org." + ) + ) + ), + form3.split( + form3.group( + form("syncLcc.id"), + "Tournament ID", + help = frag( + "From the LCC page URL. The ID looks like this: ", + pre("f1943ec6-4992-45d9-969d-a0aff688b404") + ).some, + half = true + )(form3.input(_)), + form3.group( + form("syncLcc.round"), + trb.roundNumber(), + half = true + )(form3.input(_, typ = "number")) + ) ), form3.group( form("syncUrls"), diff --git a/ui/bits/css/relay/_form.scss b/ui/bits/css/relay/_form.scss index 48e0c22d62b45..c3ef02fe7f376 100644 --- a/ui/bits/css/relay/_form.scss +++ b/ui/bits/css/relay/_form.scss @@ -35,6 +35,16 @@ font-family: monospace; } +.flash-box { + border: 5px solid $m-brag_bg--mix-70; + + &::before { + color: $c-brag; + content: $licon-InfoCircle; + font-size: 5em; + } +} + .relay-image { max-width: 100%; height: auto; diff --git a/ui/common/css/component/_flash.scss b/ui/common/css/component/_flash.scss index 2187ef0783da8..6896606ac7bdd 100644 --- a/ui/common/css/component/_flash.scss +++ b/ui/common/css/component/_flash.scss @@ -3,13 +3,12 @@ margin: 1em 0 2em 0; padding: 1em 2em; background: $c-good; - color: $c-over; &::before { @extend %data-icon; content: $licon-Checkmark; - @include margin-inline-end(1em); + @include margin-inline-end(2rem); font-size: 1.5em; } @@ -42,4 +41,12 @@ @extend %box-neat; background: $c-bg-zebra; } + + &-box { + @extend %box-neat; + background: $c-bg-zebra; + a { + color: $c-primary; + } + } } From 20538be05d8413fc7c336f6f25fe53d31d68a8ea Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 21:28:08 +0200 Subject: [PATCH 151/168] fix ui/build by reverting to env.jsDir which is probably wrong reverts this change https://github.com/lichess-org/lila/commit/59083047e9dbd3a4ca39306fdea93287117d677c#diff-b3fbb3c2873779e8f6322c18417dd873d4bf40493285ad3fbd1e514b6b82b322R97 which seems unrelated to its commit and is causing ui/build to fail --- ui/.build/src/manifest.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/.build/src/manifest.ts b/ui/.build/src/manifest.ts index 25b1b9c954a51..074fa5dcd289d 100644 --- a/ui/.build/src/manifest.ts +++ b/ui/.build/src/manifest.ts @@ -94,7 +94,7 @@ async function write() { await Promise.all([ fs.promises.writeFile(path.join(env.jsDir, `manifest.${hash}.js`), clientManifest), fs.promises.writeFile( - path.join(env.jsonDir, `manifest.${env.prod ? 'prod' : 'dev'}.json`), + path.join(env.jsDir, `manifest.${env.prod ? 'prod' : 'dev'}.json`), JSON.stringify(serverManifest, null, env.prod ? undefined : 2), ), ]); From 31480cb7b1b2565d3d6b409f028f8c8ca63b2fd0 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 22:13:33 +0200 Subject: [PATCH 152/168] tweak icon size --- ui/bits/css/relay/_form.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/bits/css/relay/_form.scss b/ui/bits/css/relay/_form.scss index c3ef02fe7f376..ea2fb0d1e2b7f 100644 --- a/ui/bits/css/relay/_form.scss +++ b/ui/bits/css/relay/_form.scss @@ -41,7 +41,7 @@ &::before { color: $c-brag; content: $licon-InfoCircle; - font-size: 5em; + font-size: 4em; } } From 21854f937a7d120d6dc8333dc213bf15f5034aea Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 22:13:44 +0200 Subject: [PATCH 153/168] dimmer c-font-dim --- ui/common/css/theme/_default.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ui/common/css/theme/_default.scss b/ui/common/css/theme/_default.scss index 237e629dc7303..1de52f1820327 100644 --- a/ui/common/css/theme/_default.scss +++ b/ui/common/css/theme/_default.scss @@ -70,7 +70,7 @@ $c-clearer: #fff; --c-metal-top-hover: #{hsl($site-hue, 7%, 25%)}; --c-metal-bottom-hover: #{hsl($site-hue, 5%, 22%)}; - --c-font-dim: #{change-color($c-font, $lightness: 60%)}; + --c-font-dim: #{change-color($c-font, $lightness: 55%)}; --c-font-dimmer: #{change-color($c-font, $lightness: 42%)}; --c-font-clear: #{change-color($c-font, $lightness: 80%)}; --c-font-clearer: #{change-color($c-font, $lightness: 89%)}; From 569e0c15d170e6ad0a06f3a56a945503de8ef086 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Wed, 19 Jun 2024 22:25:16 +0200 Subject: [PATCH 154/168] better broadcaster url --- modules/relay/src/main/package.scala | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/modules/relay/src/main/package.scala b/modules/relay/src/main/package.scala index 51d0f38ad876f..8c146de4836d8 100644 --- a/modules/relay/src/main/package.scala +++ b/modules/relay/src/main/package.scala @@ -4,8 +4,7 @@ export lila.core.lilaism.Lilaism.{ *, given } export lila.common.extensions.* export lila.core.id.{ RelayRoundId, RelayTourId } -val broadcasterUrl = - "https://github.com/lichess-org/broadcaster#user-content-lichess-broadcaster" +val broadcasterUrl = "https://lichess.org/page/broadcaster-app" private val logger = lila.log("relay") private type RelayGames = Vector[RelayGame] From 64dc2545d329bf5677efb7b5bb6eebba2aba72f3 Mon Sep 17 00:00:00 2001 From: Thanh Le Date: Wed, 19 Jun 2024 18:56:35 +0700 Subject: [PATCH 155/168] Bump lila-search RC10 and use it's types --- modules/forumSearch/src/main/ForumSearchApi.scala | 2 +- modules/forumSearch/src/main/package.scala | 2 +- modules/gameSearch/src/main/GameSearchApi.scala | 9 +++++---- modules/gameSearch/src/main/GameSearchForm.scala | 12 ++++++------ modules/gameSearch/src/main/package.scala | 10 ++++++++-- modules/search/src/main/LilaSearchClient.scala | 2 +- modules/search/src/main/model.scala | 7 ------- modules/studySearch/src/main/Env.scala | 2 +- modules/studySearch/src/main/StudySearchApi.scala | 7 ++++--- modules/studySearch/src/main/package.scala | 2 +- modules/teamSearch/src/main/TeamSearchApi.scala | 2 +- modules/teamSearch/src/main/package.scala | 2 +- project/Dependencies.scala | 2 +- 13 files changed, 31 insertions(+), 30 deletions(-) delete mode 100644 modules/search/src/main/model.scala diff --git a/modules/forumSearch/src/main/ForumSearchApi.scala b/modules/forumSearch/src/main/ForumSearchApi.scala index 23bad2e6ae357..576109be9b22e 100644 --- a/modules/forumSearch/src/main/ForumSearchApi.scala +++ b/modules/forumSearch/src/main/ForumSearchApi.scala @@ -10,7 +10,7 @@ final class ForumSearchApi(client: SearchClient)(using Executor) def search(query: Query.Forum, from: From, size: Size) = client - .search(query, from.value, size.value) + .search(query, from, size) .map(res => res.hitIds.map(ForumPostId.apply)) def count(query: Query.Forum) = diff --git a/modules/forumSearch/src/main/package.scala b/modules/forumSearch/src/main/package.scala index 2ad306ac32014..3604898cd0680 100644 --- a/modules/forumSearch/src/main/package.scala +++ b/modules/forumSearch/src/main/package.scala @@ -5,4 +5,4 @@ export lila.common.extensions.* private val logger = lila.log("forumSearch") -val index = lila.search.spec.Index.Forum +val index = lila.search.Index.Forum diff --git a/modules/gameSearch/src/main/GameSearchApi.scala b/modules/gameSearch/src/main/GameSearchApi.scala index f5c687a7b0ed4..6d7c8d194dc39 100644 --- a/modules/gameSearch/src/main/GameSearchApi.scala +++ b/modules/gameSearch/src/main/GameSearchApi.scala @@ -12,9 +12,10 @@ final class GameSearchApi( extends SearchReadApi[Game, Query.Game]: def search(query: Query.Game, from: From, size: Size): Fu[List[Game]] = - client.search(query, from.value, size.value).flatMap { res => - gameRepo.gamesFromSecondary(res.hitIds.map(GameId.apply)) - } + client + .search(query, from, size) + .flatMap: res => + gameRepo.gamesFromSecondary(res.hitIds.map(GameId.apply)) def count(query: Query.Game) = client.count(query).dmap(_.count) @@ -53,7 +54,7 @@ final class GameSearchApi( winnerColor = game.winner.fold(3)(_.color.fold(1, 2)), averageRating = game.averageUsersRating, ai = game.aiLevel, - date = lila.search.spec.SearchDateTime.fromInstant(game.movedAt), + date = lila.search.SearchDateTime.fromInstant(game.movedAt), duration = game.durationSeconds, // for realtime games only clockInit = game.clock.map(_.limitSeconds.value), clockInc = game.clock.map(_.incrementSeconds.value), diff --git a/modules/gameSearch/src/main/GameSearchForm.scala b/modules/gameSearch/src/main/GameSearchForm.scala index fba10520ccd4f..cd1b62aca7cc0 100644 --- a/modules/gameSearch/src/main/GameSearchForm.scala +++ b/modules/gameSearch/src/main/GameSearchForm.scala @@ -91,16 +91,16 @@ private[gameSearch] case class SearchData( source = source, rated = mode.flatMap(Mode.apply).map(_.rated), status = status, - turns = IntRange(turnsMin, turnsMax).some, - averageRating = IntRange(ratingMin, ratingMax).some, + turns = IntRange(turnsMin, turnsMax), + averageRating = IntRange(ratingMin, ratingMax), hasAi = hasAi.map(_ == 1), - aiLevel = IntRange(aiLevelMin, aiLevelMax).some, - date = DateRange(dateMin.map(transform), dateMax.map(transform)).some, - duration = IntRange(durationMin, durationMax).some, + aiLevel = IntRange(aiLevelMin, aiLevelMax), + date = DateRange(dateMin.map(transform), dateMax.map(transform)), + duration = IntRange(durationMin, durationMax), analysed = analysed.map(_ == 1), whiteUser = players.cleanWhite.map(_.value), blackUser = players.cleanBlack.map(_.value), - sorting = SpecSorting(sortOrDefault.field, sortOrDefault.order).some, + sorting = SpecSorting(sortOrDefault.field, sortOrDefault.order), clockInit = clockInit, clockInc = clockInc ) diff --git a/modules/gameSearch/src/main/package.scala b/modules/gameSearch/src/main/package.scala index 7ba8569af0bec..aa3c156c3b59d 100644 --- a/modules/gameSearch/src/main/package.scala +++ b/modules/gameSearch/src/main/package.scala @@ -1,14 +1,20 @@ package lila.gameSearch +import lila.search.spec.{ IntRange, DateRange } + export lila.core.lilaism.Lilaism.{ *, given } export lila.common.extensions.* private val logger = lila.log("gameSearch") -val index = lila.search.spec.Index.Game +val index = lila.search.Index.Game + +extension (range: IntRange) def nonEmpty: Boolean = range.a.nonEmpty || range.b.nonEmpty + +extension (range: DateRange) def nonEmpty: Boolean = range.a.nonEmpty || range.b.nonEmpty extension (query: lila.search.spec.Query.Game) - def nonEmpty = + def nonEmpty: Boolean = query.user1.nonEmpty || query.user2.nonEmpty || query.winner.nonEmpty || diff --git a/modules/search/src/main/LilaSearchClient.scala b/modules/search/src/main/LilaSearchClient.scala index af9dd4a729a44..c20a9cad0e0d6 100644 --- a/modules/search/src/main/LilaSearchClient.scala +++ b/modules/search/src/main/LilaSearchClient.scala @@ -43,7 +43,7 @@ class LilaSearchClient(client: SearchClient, writeable: Boolean)(using Executor) logger.info(s"Count error: query={$query}", e) CountOutput(0) - override def search(query: Query, from: Int, size: Int): Future[SearchOutput] = + override def search(query: Query, from: From, size: Size): Future[SearchOutput] = monitor("search", query.index): client .search(query, from, size) diff --git a/modules/search/src/main/model.scala b/modules/search/src/main/model.scala deleted file mode 100644 index ff7ac084f898f..0000000000000 --- a/modules/search/src/main/model.scala +++ /dev/null @@ -1,7 +0,0 @@ -package lila.search - -opaque type From = Int -object From extends OpaqueInt[From] - -opaque type Size = Int -object Size extends OpaqueInt[Size] diff --git a/modules/studySearch/src/main/Env.scala b/modules/studySearch/src/main/Env.scala index 5348917f1fc9f..47fbaf15c5bcb 100644 --- a/modules/studySearch/src/main/Env.scala +++ b/modules/studySearch/src/main/Env.scala @@ -8,7 +8,7 @@ import scalalib.paginator.* import lila.common.LateMultiThrottler import lila.core.study.RemoveStudy import lila.study.Study -import lila.search.{ From, Size } +import lila.search.* import lila.search.client.SearchClient import lila.search.spec.Query diff --git a/modules/studySearch/src/main/StudySearchApi.scala b/modules/studySearch/src/main/StudySearchApi.scala index c7011a8e0790b..71cb3d867d421 100644 --- a/modules/studySearch/src/main/StudySearchApi.scala +++ b/modules/studySearch/src/main/StudySearchApi.scala @@ -23,9 +23,10 @@ final class StudySearchApi( extends SearchReadApi[Study, Query.Study]: def search(query: Query.Study, from: From, size: Size) = - client.search(query, from.value, size.value).flatMap { res => - studyRepo.byOrderedIds(res.hitIds.map(StudyId(_))) - } + client + .search(query, from, size) + .flatMap: res => + studyRepo.byOrderedIds(res.hitIds.map(StudyId(_))) def count(query: Query.Study) = client.count(query).dmap(_.count) diff --git a/modules/studySearch/src/main/package.scala b/modules/studySearch/src/main/package.scala index c0409a39f590c..91418a10f7308 100644 --- a/modules/studySearch/src/main/package.scala +++ b/modules/studySearch/src/main/package.scala @@ -5,4 +5,4 @@ export lila.common.extensions.* private val logger = lila.log("studySearch") -private val index = lila.search.spec.Index.Study +private val index = lila.search.Index.Study diff --git a/modules/teamSearch/src/main/TeamSearchApi.scala b/modules/teamSearch/src/main/TeamSearchApi.scala index 29dea5cbf57ea..bd7a7c95c14c0 100644 --- a/modules/teamSearch/src/main/TeamSearchApi.scala +++ b/modules/teamSearch/src/main/TeamSearchApi.scala @@ -15,7 +15,7 @@ final class TeamSearchApi( def search(query: Query.Team, from: From, size: Size) = client - .search(query, from.value, size.value) + .search(query, from, size) .map: res => res.hitIds.map(TeamId.apply) diff --git a/modules/teamSearch/src/main/package.scala b/modules/teamSearch/src/main/package.scala index e4fb49b101791..9e3f5b1de4cfb 100644 --- a/modules/teamSearch/src/main/package.scala +++ b/modules/teamSearch/src/main/package.scala @@ -5,4 +5,4 @@ export lila.common.extensions.* private val logger = lila.log("teamSearch") -val index = lila.search.spec.Index.Team +val index = lila.search.Index.Team diff --git a/project/Dependencies.scala b/project/Dependencies.scala index 52412dd976ab2..88f7a4009c0d2 100644 --- a/project/Dependencies.scala +++ b/project/Dependencies.scala @@ -26,7 +26,7 @@ object Dependencies { val lettuce = "io.lettuce" % "lettuce-core" % "6.3.2.RELEASE" val nettyTransport = ("io.netty" % s"netty-transport-native-$notifier" % "4.1.111.Final").classifier(s"$os-$arch") - val lilaSearch = "org.lichess.search" %% "client" % "3.0.0-RC9" + val lilaSearch = "org.lichess.search" %% "client" % "3.0.0-RC10" val munit = "org.scalameta" %% "munit" % "1.0.0" % Test val uaparser = "org.uaparser" %% "uap-scala" % "0.17.0" val apacheText = "org.apache.commons" % "commons-text" % "1.12.0" From 86006cd0e7acb5cedfbb8c7bfc190b51d74d8e99 Mon Sep 17 00:00:00 2001 From: slither77 <132110444+slither77@users.noreply.github.com> Date: Thu, 20 Jun 2024 01:34:25 -0400 Subject: [PATCH 156/168] monarchy piece set (svg format) (#15557) * Create monarchy * Delete public/piece/monarchy * Create test * Delete public/piece/monarchy/test * Add monarchy pieces (svg format) hopefully this works :) * white rook --------- Co-authored-by: Thibault Duplessis --- public/piece/monarchy/bB.svg | 23 +++++++++++++++++++++++ public/piece/monarchy/bK.svg | 21 +++++++++++++++++++++ public/piece/monarchy/bN.svg | 15 +++++++++++++++ public/piece/monarchy/bP.svg | 15 +++++++++++++++ public/piece/monarchy/bQ.svg | 22 ++++++++++++++++++++++ public/piece/monarchy/bR.svg | 15 +++++++++++++++ public/piece/monarchy/wB.svg | 17 +++++++++++++++++ public/piece/monarchy/wK.svg | 17 +++++++++++++++++ public/piece/monarchy/wN.svg | 15 +++++++++++++++ public/piece/monarchy/wP.svg | 15 +++++++++++++++ public/piece/monarchy/wQ.svg | 23 +++++++++++++++++++++++ public/piece/monarchy/wR.svg | 17 +++++++++++++++++ 12 files changed, 215 insertions(+) create mode 100644 public/piece/monarchy/bB.svg create mode 100644 public/piece/monarchy/bK.svg create mode 100644 public/piece/monarchy/bN.svg create mode 100644 public/piece/monarchy/bP.svg create mode 100644 public/piece/monarchy/bQ.svg create mode 100644 public/piece/monarchy/bR.svg create mode 100644 public/piece/monarchy/wB.svg create mode 100644 public/piece/monarchy/wK.svg create mode 100644 public/piece/monarchy/wN.svg create mode 100644 public/piece/monarchy/wP.svg create mode 100644 public/piece/monarchy/wQ.svg create mode 100644 public/piece/monarchy/wR.svg diff --git a/public/piece/monarchy/bB.svg b/public/piece/monarchy/bB.svg new file mode 100644 index 0000000000000..216ea7d1d8814 --- /dev/null +++ b/public/piece/monarchy/bB.svg @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/bK.svg b/public/piece/monarchy/bK.svg new file mode 100644 index 0000000000000..fc80f2a1ac179 --- /dev/null +++ b/public/piece/monarchy/bK.svg @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/bN.svg b/public/piece/monarchy/bN.svg new file mode 100644 index 0000000000000..5371375f793a3 --- /dev/null +++ b/public/piece/monarchy/bN.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/bP.svg b/public/piece/monarchy/bP.svg new file mode 100644 index 0000000000000..e56d92900df61 --- /dev/null +++ b/public/piece/monarchy/bP.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/bQ.svg b/public/piece/monarchy/bQ.svg new file mode 100644 index 0000000000000..cde1027fb5484 --- /dev/null +++ b/public/piece/monarchy/bQ.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/bR.svg b/public/piece/monarchy/bR.svg new file mode 100644 index 0000000000000..c5035059267a3 --- /dev/null +++ b/public/piece/monarchy/bR.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/wB.svg b/public/piece/monarchy/wB.svg new file mode 100644 index 0000000000000..fad56652550d7 --- /dev/null +++ b/public/piece/monarchy/wB.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/wK.svg b/public/piece/monarchy/wK.svg new file mode 100644 index 0000000000000..97d87236f1dca --- /dev/null +++ b/public/piece/monarchy/wK.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/wN.svg b/public/piece/monarchy/wN.svg new file mode 100644 index 0000000000000..b4feb360baaab --- /dev/null +++ b/public/piece/monarchy/wN.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/wP.svg b/public/piece/monarchy/wP.svg new file mode 100644 index 0000000000000..5c4c7bc17d5e0 --- /dev/null +++ b/public/piece/monarchy/wP.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/wQ.svg b/public/piece/monarchy/wQ.svg new file mode 100644 index 0000000000000..4245a53508184 --- /dev/null +++ b/public/piece/monarchy/wQ.svg @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/piece/monarchy/wR.svg b/public/piece/monarchy/wR.svg new file mode 100644 index 0000000000000..21aadda404368 --- /dev/null +++ b/public/piece/monarchy/wR.svg @@ -0,0 +1,17 @@ + + + + + + + + + + + + + + + + + From 9babeae0ef30f75cc51918dc53c07870f0205d67 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 07:32:31 +0200 Subject: [PATCH 157/168] fix gitignore unignore bin/gen --- .gitignore | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index d69e05f00f0ee..5c91d043ca27a 100644 --- a/.gitignore +++ b/.gitignore @@ -22,7 +22,7 @@ data/ dist/ node_modules/ local/ -gen/ +ui/common/css/theme/gen/*.scss ui/common/**/*.js ui/common/**/*.d.ts ui/chess/**/*.js From aabf7abb49620394b22892e3b5ac7b1c34083d9f Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 07:32:44 +0200 Subject: [PATCH 158/168] integrate monarchy piece set --- COPYING.md | 1 + bin/gen/piece-sprite | 1 + modules/pref/src/main/PieceSet.scala | 1 + public/piece-css/monarchy.css | 12 ++++++++++++ public/piece-css/monarchy.external.css | 12 ++++++++++++ 5 files changed, 27 insertions(+) create mode 100644 public/piece-css/monarchy.css create mode 100644 public/piece-css/monarchy.external.css diff --git a/COPYING.md b/COPYING.md index 0362eaf249abe..6531d7f304a28 100644 --- a/COPYING.md +++ b/COPYING.md @@ -59,6 +59,7 @@ public/piece/disguised | danegraphics | [CC BY-NC-SA 4.0](https://creativecommon public/piece/kiwen-suwi | [neverRare](https://github.com/neverRare) | [CC BY 4.0](https://creativecommons.org/licenses/by/4.0/) public/piece/mpchess | [Maxime Chupin](https://github.com/chupinmaxime) | [GPL3v3+](https://www.gnu.org/licenses/quick-guide-gplv3.en.html) public/piece/cooke | [fejfar](https://github.com/fejfar) | [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) +public/piece/monarchy | [slither77](https://github.com/slither77) | [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) public/sounds/futuristic | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ public/sounds/nes | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ public/sounds/piano | [Enigmahack](https://github.com/Enigmahack) | AGPLv3+ diff --git a/bin/gen/piece-sprite b/bin/gen/piece-sprite index afcb9537bb27c..668c5e33e371d 100755 --- a/bin/gen/piece-sprite +++ b/bin/gen/piece-sprite @@ -44,6 +44,7 @@ themes = [ ['kiwen-suwi', 'svg'], ['mpchess','svg'], ['cooke','svg'], + ['monarchy','svg'], ] types = { 'svg' => 'svg+xml;base64,', diff --git a/modules/pref/src/main/PieceSet.scala b/modules/pref/src/main/PieceSet.scala index 55178a1e70e5f..38874ecf48c98 100644 --- a/modules/pref/src/main/PieceSet.scala +++ b/modules/pref/src/main/PieceSet.scala @@ -47,6 +47,7 @@ object PieceSet extends PieceSetObject: PieceSet("tatiana"), PieceSet("staunty"), PieceSet("cooke"), + PieceSet("monarchy"), PieceSet("governor"), PieceSet("dubrovny"), PieceSet("icpieces"), diff --git a/public/piece-css/monarchy.css b/public/piece-css/monarchy.css new file mode 100644 index 0000000000000..ccb6c12e18b60 --- /dev/null +++ b/public/piece-css/monarchy.css @@ -0,0 +1,12 @@ +.is2d .pawn.white {background-image:url('')} +.is2d .knight.white {background-image:url('')} +.is2d .bishop.white {background-image:url('')} +.is2d .rook.white {background-image:url('')} +.is2d .queen.white {background-image:url('')} +.is2d .king.white {background-image:url('')} +.is2d .pawn.black {background-image:url('')} +.is2d .knight.black {background-image:url('')} +.is2d .bishop.black {background-image:url('')} +.is2d .rook.black {background-image:url('')} +.is2d .queen.black {background-image:url('')} +.is2d .king.black {background-image:url('')} diff --git a/public/piece-css/monarchy.external.css b/public/piece-css/monarchy.external.css new file mode 100644 index 0000000000000..4f8779b0e84fe --- /dev/null +++ b/public/piece-css/monarchy.external.css @@ -0,0 +1,12 @@ +.is2d .pawn.white {background-image:url('/assets/piece/monarchy/wP.svg')} +.is2d .knight.white {background-image:url('/assets/piece/monarchy/wN.svg')} +.is2d .bishop.white {background-image:url('/assets/piece/monarchy/wB.svg')} +.is2d .rook.white {background-image:url('/assets/piece/monarchy/wR.svg')} +.is2d .queen.white {background-image:url('/assets/piece/monarchy/wQ.svg')} +.is2d .king.white {background-image:url('/assets/piece/monarchy/wK.svg')} +.is2d .pawn.black {background-image:url('/assets/piece/monarchy/bP.svg')} +.is2d .knight.black {background-image:url('/assets/piece/monarchy/bN.svg')} +.is2d .bishop.black {background-image:url('/assets/piece/monarchy/bB.svg')} +.is2d .rook.black {background-image:url('/assets/piece/monarchy/bR.svg')} +.is2d .queen.black {background-image:url('/assets/piece/monarchy/bQ.svg')} +.is2d .king.black {background-image:url('/assets/piece/monarchy/bK.svg')} From 758f9697918d501c110e6294f76149e06c66f324 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 08:16:43 +0200 Subject: [PATCH 159/168] fix broadcaster url --- modules/relay/src/main/ui/FormUi.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/relay/src/main/ui/FormUi.scala b/modules/relay/src/main/ui/FormUi.scala index ce89e092a9d4f..f423b2fd6be4a 100644 --- a/modules/relay/src/main/ui/FormUi.scala +++ b/modules/relay/src/main/ui/FormUi.scala @@ -205,7 +205,7 @@ final class FormUi(helpers: Helpers, ui: RelayUi, tourUi: RelayTourUi): div(cls := "form-group relay-form__sync relay-form__sync-push none")( p( "Send your local games to Lichess using ", - a(href := "https://github.com/lichess-org/broadcaster")("the Lichess Broadcaster App"), + a(href := "https://github.com/lichess-org/broadcaster")(lila.relay.broadcasterUrl), "." ) ), From d342021d6edc9ce852d3b0633c93be2ebda5c216 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 09:55:12 +0200 Subject: [PATCH 160/168] combine lcc sources --- modules/relay/src/main/BSONHandlers.scala | 16 +++++++++++--- modules/relay/src/main/JsonView.scala | 5 ++--- modules/relay/src/main/RelayDelay.scala | 2 +- modules/relay/src/main/RelayFormat.scala | 2 +- modules/relay/src/main/RelayRound.scala | 24 ++++++++++++++------- modules/relay/src/main/RelayRoundForm.scala | 13 +++++++---- 6 files changed, 42 insertions(+), 20 deletions(-) diff --git a/modules/relay/src/main/BSONHandlers.scala b/modules/relay/src/main/BSONHandlers.scala index 03b4f2377624c..4c6da447cce59 100644 --- a/modules/relay/src/main/BSONHandlers.scala +++ b/modules/relay/src/main/BSONHandlers.scala @@ -10,9 +10,19 @@ object BSONHandlers: given BSONHandler[RelayTeamsTextarea] = stringAnyValHandler(_.text, RelayTeamsTextarea(_)) import RelayRound.Sync - import Sync.{ Upstream, UpstreamIds, UpstreamUrl, UpstreamLcc, UpstreamUrls } - given upstreamUrlHandler: BSONDocumentHandler[UpstreamUrl] = Macros.handler - given upstreamLccHandler: BSONDocumentHandler[UpstreamLcc] = Macros.handler + import Sync.{ Upstream, UpstreamIds, UpstreamUrl, UpstreamLcc, UpstreamUrls, FetchableUpstream } + given upstreamUrlHandler: BSONDocumentHandler[UpstreamUrl] = Macros.handler + given upstreamLccHandler: BSONDocumentHandler[UpstreamLcc] = Macros.handler + given BSONHandler[FetchableUpstream] = tryHandler( + { + case d: BSONDocument if d.contains("url") => upstreamUrlHandler.readTry(d) + case d: BSONDocument if d.contains("lcc") => upstreamLccHandler.readTry(d) + }, + { + case url: UpstreamUrl => upstreamUrlHandler.writeTry(url).get + case lcc: UpstreamLcc => upstreamLccHandler.writeTry(lcc).get + } + ) given upstreamUrlsHandler: BSONDocumentHandler[UpstreamUrls] = Macros.handler given upstreamIdsHandler: BSONDocumentHandler[UpstreamIds] = Macros.handler diff --git a/modules/relay/src/main/JsonView.scala b/modules/relay/src/main/JsonView.scala index 8625ba6c272c4..40110e054f143 100644 --- a/modules/relay/src/main/JsonView.scala +++ b/modules/relay/src/main/JsonView.scala @@ -178,9 +178,8 @@ object JsonView: "log" -> s.log.events ) .add("delay" -> s.delay) ++ - s.upstream.so { + s.upstream.so: case Sync.UpstreamUrl(url) => Json.obj("url" -> url) case Sync.UpstreamLcc(url, round) => Json.obj("url" -> url, "round" -> round) - case Sync.UpstreamUrls(urls) => Json.obj("urls" -> urls.map(_.url)) + case Sync.UpstreamUrls(urls) => Json.obj("urls" -> urls.map(_.formUrl)) case Sync.UpstreamIds(ids) => Json.obj("ids" -> ids) - } diff --git a/modules/relay/src/main/RelayDelay.scala b/modules/relay/src/main/RelayDelay.scala index 81add186ced9d..2666e314861bb 100644 --- a/modules/relay/src/main/RelayDelay.scala +++ b/modules/relay/src/main/RelayDelay.scala @@ -51,7 +51,7 @@ final private class RelayDelay(colls: RelayColls)(using Executor): private object store: - private def idOf(upstream: FetchableUpstream, at: Instant) = s"${upstream.url} ${at.toSeconds}" + private def idOf(upstream: FetchableUpstream, at: Instant) = s"${upstream.formUrl} ${at.toSeconds}" private val longPast = java.time.Instant.ofEpochMilli(0) def putIfNew(upstream: FetchableUpstream, games: RelayGames): Funit = diff --git a/modules/relay/src/main/RelayFormat.scala b/modules/relay/src/main/RelayFormat.scala index 6938a708c357e..0270c7a0d8436 100644 --- a/modules/relay/src/main/RelayFormat.scala +++ b/modules/relay/src/main/RelayFormat.scala @@ -47,7 +47,7 @@ final private class RelayFormatApi( private def guessFormat(upstream: FetchableUpstream)(using CanProxy): Fu[RelayFormat] = { - def parsedUrl = URL.parse(upstream.url) + def parsedUrl = URL.parse(upstream.fetchUrl) def guessLcc: Fu[Option[RelayFormat]] = upstream.isLcc.so(guessManyFiles(parsedUrl)) diff --git a/modules/relay/src/main/RelayRound.scala b/modules/relay/src/main/RelayRound.scala index 3b70161ee0674..6308fe4d0f26c 100644 --- a/modules/relay/src/main/RelayRound.scala +++ b/modules/relay/src/main/RelayRound.scala @@ -117,19 +117,27 @@ object RelayRound: sealed trait Upstream: def isLcc = false sealed trait FetchableUpstream extends Upstream: - def url: String + def fetchUrl: String + def formUrl: String case class UpstreamUrl(url: String) extends FetchableUpstream: - def findLccId: Option[String] = url match - case LccRegex(id) => id.some - case _ => none + def fetchUrl = url + def formUrl = url + case class UpstreamUrls(urls: List[FetchableUpstream]) extends Upstream + case class UpstreamIds(ids: List[GameId]) extends Upstream case class UpstreamLcc(lcc: String, round: Int) extends FetchableUpstream: override def isLcc = true def id = lcc - def url = s"http://1.pool.livechesscloud.com/get/$id/round-$round/index.json" + def fetchUrl = s"http://1.pool.livechesscloud.com/get/$id/round-$round/index.json" def viewUrl = s"https://view.livechesscloud.com/#$id" - case class UpstreamUrls(urls: List[UpstreamUrl]) extends Upstream - case class UpstreamIds(ids: List[GameId]) extends Upstream - private val LccRegex = """.*view\.livechesscloud\.com/?#?([0-9a-f\-]+)""".r + def formUrl = s"$viewUrl $round" + object UpstreamLcc: + private val idRegex = """.*view\.livechesscloud\.com/?#?([0-9a-f\-]+)""".r + def findId(url: UpstreamUrl): Option[String] = url.url match + case idRegex(id) => id.some + case _ => none + def find(url: String): Option[UpstreamLcc] = url.split(' ').map(_.trim).filter(_.nonEmpty) match + case Array(idRegex(id), round) => round.toIntOption.map(UpstreamLcc(id, _)) + case _ => none trait AndTour: val tour: RelayTour diff --git a/modules/relay/src/main/RelayRoundForm.scala b/modules/relay/src/main/RelayRoundForm.scala index 3140e3f317b41..ec0a821b282e2 100644 --- a/modules/relay/src/main/RelayRoundForm.scala +++ b/modules/relay/src/main/RelayRoundForm.scala @@ -19,15 +19,15 @@ final class RelayRoundForm(using mode: Mode): import RelayRoundForm.* import lila.common.Form.ISOInstantOrTimestamp - private given Formatter[Sync.UpstreamUrl] = formatter.stringTryFormatter(validateUpstreamUrl, _.url) + private given Formatter[Sync.UpstreamUrl] = formatter.stringTryFormatter(validateUpstreamUrl, _.fetchUrl) private given Formatter[Sync.UpstreamUrls] = formatter.stringTryFormatter( _.linesIterator.toList .map(_.trim) .filter(_.nonEmpty) - .traverse(validateUpstreamUrl) + .traverse(validateUpstreamUrlOrLcc) .map(_.distinct) .map(Sync.UpstreamUrls.apply), - _.urls.map(_.url).mkString("\n") + _.urls.map(_.formUrl).mkString("\n") ) private given Formatter[Sync.UpstreamIds] = formatter.stringTryFormatter( _.split(' ').toList @@ -147,6 +147,11 @@ object RelayRoundForm: if !subdomain(host, "chess.com") || url.toString.startsWith("https://api.chess.com/pub") yield url.toString.stripSuffix("/") + private def validateUpstreamUrlOrLcc(s: String)(using Mode): Either[String, Sync.FetchableUpstream] = + Sync.UpstreamLcc.find(s) match + case Some(lcc) => Right(lcc) + case None => validateUpstreamUrl(s) + private def validateUpstreamUrl(s: String)(using Mode): Either[String, Sync.UpstreamUrl] = for url <- cleanUrl(s).toRight("Invalid source URL") url <- if !validSourcePort(url) then Left("The source URL cannot specify a port") else Right(url) @@ -209,7 +214,7 @@ object RelayRoundForm: .map: case url: Sync.UpstreamUrl => val foundLcc = for - lccId <- url.findLccId + lccId <- Sync.UpstreamLcc.findId(url) round <- roundNumberIn(name.value) yield Sync.UpstreamLcc(lccId, round) foundLcc | url From e0dda580d93c6affa10cd07f99f0455ff386a22d Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 10:21:06 +0200 Subject: [PATCH 161/168] apply board brightness to custom svgs - closes #15563 --- ui/common/css/theme/board/_chessground.scss | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ui/common/css/theme/board/_chessground.scss b/ui/common/css/theme/board/_chessground.scss index 2691386ed9872..7f0b444e5b4dd 100644 --- a/ui/common/css/theme/board/_chessground.scss +++ b/ui/common/css/theme/board/_chessground.scss @@ -199,7 +199,8 @@ cg-auto-pieces { html:not(.transp) body:not(.simple-board) { &.coords-in coord, cg-board piece, - cg-board square { + cg-board square, + .cg-custom-svgs { filter: brightness(calc(0.3 + 0.0059 * min(120, var(---board-brightness)))); //hue-rotate(calc(var(---board-hue) * 3.6deg)); } From 2ffadc93d9c8013ea3ed9c633e66eeebf0f69ee2 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 10:21:33 +0200 Subject: [PATCH 162/168] New Crowdin updates (#15548) * New translations: site.xml (Chinese Simplified) * New translations: site.xml (Tagalog) * New translations: coordinates.xml (Luxembourgish) * New translations: site.xml (Luxembourgish) * New translations: study.xml (Luxembourgish) * New translations: site.xml (Hebrew) * New translations: site.xml (Arabic) * New translations: faq.xml (Arabic) * New translations: swiss.xml (Arabic) * New translations: appeal.xml (Arabic) * New translations: faq.xml (Arabic) * New translations: arena.xml (Arabic) * New translations: team.xml (Arabic) * New translations: coordinates.xml (Arabic) * New translations: faq.xml (Arabic) * New translations: swiss.xml (Arabic) * New translations: appeal.xml (Arabic) * New translations: ublog.xml (Arabic) * New translations: swiss.xml (Arabic) * New translations: site.xml (Arabic) * New translations: site.xml (Arabic) * New translations: faq.xml (Arabic) * New translations: swiss.xml (Arabic) * New translations: broadcast.xml (Korean) * New translations: broadcast.xml (Arabic) * New translations: broadcast.xml (Romanian) * New translations: broadcast.xml (French) * New translations: broadcast.xml (Spanish) * New translations: broadcast.xml (Afrikaans) * New translations: broadcast.xml (Belarusian) * New translations: broadcast.xml (Bulgarian) * New translations: broadcast.xml (Catalan) * New translations: broadcast.xml (Czech) * New translations: broadcast.xml (Danish) * New translations: broadcast.xml (German) * New translations: broadcast.xml (Greek) * New translations: broadcast.xml (Basque) * New translations: broadcast.xml (Finnish) * New translations: broadcast.xml (Irish) * New translations: broadcast.xml (Gujarati) * New translations: broadcast.xml (Hebrew) * New translations: broadcast.xml (Hungarian) * New translations: broadcast.xml (Italian) * New translations: broadcast.xml (Japanese) * New translations: broadcast.xml (Lithuanian) * New translations: broadcast.xml (Dutch) * New translations: broadcast.xml (Polish) * New translations: broadcast.xml (Portuguese) * New translations: broadcast.xml (Russian) * New translations: broadcast.xml (Slovak) * New translations: broadcast.xml (Slovenian) * New translations: broadcast.xml (Albanian) * New translations: broadcast.xml (Swedish) * New translations: broadcast.xml (Turkish) * New translations: broadcast.xml (Ukrainian) * New translations: broadcast.xml (Chinese Simplified) * New translations: broadcast.xml (Chinese Traditional) * New translations: broadcast.xml (Vietnamese) * New translations: broadcast.xml (Galician) * New translations: broadcast.xml (Portuguese, Brazilian) * New translations: broadcast.xml (Indonesian) * New translations: broadcast.xml (Persian) * New translations: broadcast.xml (Marathi) * New translations: broadcast.xml (Croatian) * New translations: broadcast.xml (Norwegian Nynorsk) * New translations: broadcast.xml (Kazakh) * New translations: broadcast.xml (Estonian) * New translations: broadcast.xml (Latvian) * New translations: broadcast.xml (Azerbaijani) * New translations: broadcast.xml (Hindi) * New translations: broadcast.xml (English, United States) * New translations: broadcast.xml (Faroese) * New translations: broadcast.xml (Esperanto) * New translations: broadcast.xml (Luxembourgish) * New translations: broadcast.xml (Tatar) * New translations: broadcast.xml (Breton) * New translations: broadcast.xml (Rusyn) * New translations: broadcast.xml (Bosnian) * New translations: broadcast.xml (Kannada) * New translations: broadcast.xml (Norwegian Bokmal) * New translations: broadcast.xml (Sorani (Kurdish)) * New translations: broadcast.xml (Swiss German) * New translations: broadcast.xml (German) * New translations: broadcast.xml (Hebrew) * New translations: broadcast.xml (Persian) * New translations: appeal.xml (Hindi) * New translations: broadcast.xml (Spanish) * New translations: broadcast.xml (Danish) * New translations: broadcast.xml (German) * New translations: broadcast.xml (Hebrew) * New translations: broadcast.xml (Portuguese) * New translations: broadcast.xml (Russian) * New translations: broadcast.xml (Galician) * New translations: broadcast.xml (Persian) * New translations: broadcast.xml (Danish) * New translations: broadcast.xml (German) * New translations: broadcast.xml (Hebrew) * New translations: broadcast.xml (Polish) * New translations: broadcast.xml (Norwegian Nynorsk) * New translations: broadcast.xml (Portuguese) * New translations: broadcast.xml (French) * New translations: broadcast.xml (Vietnamese) * New translations: broadcast.xml (English, United States) * New translations: site.xml (Catalan) * New translations: site.xml (Tagalog) * New translations: broadcast.xml (Finnish) --- translation/dest/appeal/ar-SA.xml | 21 ++++++++++++++++++++- translation/dest/appeal/hi-IN.xml | 1 + translation/dest/arena/ar-SA.xml | 1 + translation/dest/broadcast/af-ZA.xml | 2 -- translation/dest/broadcast/ar-SA.xml | 2 -- translation/dest/broadcast/az-AZ.xml | 2 -- translation/dest/broadcast/be-BY.xml | 2 -- translation/dest/broadcast/bg-BG.xml | 2 -- translation/dest/broadcast/br-FR.xml | 1 - translation/dest/broadcast/bs-BA.xml | 2 -- translation/dest/broadcast/ca-ES.xml | 2 -- translation/dest/broadcast/ckb-IR.xml | 2 -- translation/dest/broadcast/cs-CZ.xml | 2 -- translation/dest/broadcast/da-DK.xml | 4 ++-- translation/dest/broadcast/de-DE.xml | 4 ++-- translation/dest/broadcast/el-GR.xml | 2 -- translation/dest/broadcast/en-US.xml | 4 ++-- translation/dest/broadcast/eo-UY.xml | 2 -- translation/dest/broadcast/es-ES.xml | 4 ++-- translation/dest/broadcast/et-EE.xml | 2 -- translation/dest/broadcast/eu-ES.xml | 2 -- translation/dest/broadcast/fa-IR.xml | 3 +-- translation/dest/broadcast/fi-FI.xml | 4 ++-- translation/dest/broadcast/fo-FO.xml | 2 -- translation/dest/broadcast/fr-FR.xml | 4 ++-- translation/dest/broadcast/ga-IE.xml | 2 -- translation/dest/broadcast/gl-ES.xml | 4 ++-- translation/dest/broadcast/gsw-CH.xml | 2 -- translation/dest/broadcast/gu-IN.xml | 5 +---- translation/dest/broadcast/he-IL.xml | 4 ++-- translation/dest/broadcast/hi-IN.xml | 2 -- translation/dest/broadcast/hr-HR.xml | 2 -- translation/dest/broadcast/hu-HU.xml | 2 -- translation/dest/broadcast/id-ID.xml | 2 -- translation/dest/broadcast/it-IT.xml | 2 -- translation/dest/broadcast/ja-JP.xml | 2 -- translation/dest/broadcast/kk-KZ.xml | 2 -- translation/dest/broadcast/kn-IN.xml | 2 -- translation/dest/broadcast/ko-KR.xml | 2 -- translation/dest/broadcast/lb-LU.xml | 2 -- translation/dest/broadcast/lt-LT.xml | 2 -- translation/dest/broadcast/lv-LV.xml | 2 -- translation/dest/broadcast/mr-IN.xml | 2 -- translation/dest/broadcast/nb-NO.xml | 2 -- translation/dest/broadcast/nl-NL.xml | 2 -- translation/dest/broadcast/nn-NO.xml | 4 ++-- translation/dest/broadcast/pl-PL.xml | 4 ++-- translation/dest/broadcast/pt-BR.xml | 2 -- translation/dest/broadcast/pt-PT.xml | 4 ++-- translation/dest/broadcast/ro-RO.xml | 2 -- translation/dest/broadcast/ru-RU.xml | 3 +-- translation/dest/broadcast/ry-UA.xml | 2 -- translation/dest/broadcast/sk-SK.xml | 2 -- translation/dest/broadcast/sl-SI.xml | 2 -- translation/dest/broadcast/sq-AL.xml | 2 -- translation/dest/broadcast/sv-SE.xml | 2 -- translation/dest/broadcast/tr-TR.xml | 2 -- translation/dest/broadcast/tt-RU.xml | 2 -- translation/dest/broadcast/uk-UA.xml | 2 -- translation/dest/broadcast/vi-VN.xml | 4 ++-- translation/dest/broadcast/zh-CN.xml | 2 -- translation/dest/broadcast/zh-TW.xml | 2 -- translation/dest/coordinates/ar-SA.xml | 1 + translation/dest/coordinates/lb-LU.xml | 1 + translation/dest/faq/ar-SA.xml | 9 +++++++++ translation/dest/site/ar-SA.xml | 3 +++ translation/dest/site/ca-ES.xml | 4 ++-- translation/dest/site/he-IL.xml | 6 +++--- translation/dest/site/lb-LU.xml | 6 ++++-- translation/dest/site/tl-PH.xml | 4 ++++ translation/dest/site/zh-CN.xml | 2 ++ translation/dest/study/lb-LU.xml | 2 +- translation/dest/swiss/ar-SA.xml | 14 ++++++++++++++ translation/dest/team/ar-SA.xml | 16 ++++++++++++++++ translation/dest/ublog/ar-SA.xml | 1 + 75 files changed, 110 insertions(+), 128 deletions(-) diff --git a/translation/dest/appeal/ar-SA.xml b/translation/dest/appeal/ar-SA.xml index 3ea04e700dfa8..31f0a3374cb04 100644 --- a/translation/dest/appeal/ar-SA.xml +++ b/translation/dest/appeal/ar-SA.xml @@ -1,2 +1,21 @@ - + + حسابك غير مميز أو مقيد. أنت بخير! + حسابك مشار إليه كمساعدة الخارجية في الألعاب. + نحن نعرف هذا بأنه استخدام أي مساعدة خارجية لتعزيز معرفتك و/أو مهاراتك في الحساب من أجل اكتساب أفضلية غير عادلة على خصمك. راجع صفحة %s لمزيد من التفاصيل. + حسابك محظور من الانضمام إلى الساحات. + تم حظر حسابك من البطولات ذو جوائز حقيقية. + تم تحديد حسابك للتلاعب بالتقييم. + نحن نعرف هذا بأنه التلاعب المتعمد بالتصنيف عن طريق فقدان الأدوار عن قصد أو اللعب ضد حساب آخر يفقد الأدوار عمدا. + تم كتم صوت حسابك. + اقرأ %s الخاص بنا. الفشل في اتباع القواعد الإرشادية للتواصل يمكن أن يؤدي إلى كتم الحسابات. + تم استبعاد حسابك من لوحات المتصدرين. + نحن نعرّف هذا على أنه استخدام أي طريقة غير عادلة للدخول إلى لوحة المتصدرين. + تم إغلاق حسابك من قبل المشرفين. + تم إخفاء مدوناتك من قبل المشرفين. + تأكد من قراءة %s مرة أخرى. + تم إيقافك عن اللعب. + قواعد التواصل + قواعد المدونة + اللعب العادل + diff --git a/translation/dest/appeal/hi-IN.xml b/translation/dest/appeal/hi-IN.xml index 89a4eda072b15..17509bcf792cb 100644 --- a/translation/dest/appeal/hi-IN.xml +++ b/translation/dest/appeal/hi-IN.xml @@ -1,6 +1,7 @@ आपका एकाउंट मॉडरेटर द्वारा बंद कर दिया गया था. + हमारे %s को दोबारा पढ़ना सुनिश्चित करें। संचार दिशानिर्देश फेयर प्ले diff --git a/translation/dest/arena/ar-SA.xml b/translation/dest/arena/ar-SA.xml index 63b5953c3bcfb..3d2246931b54d 100644 --- a/translation/dest/arena/ar-SA.xml +++ b/translation/dest/arena/ar-SA.xml @@ -70,6 +70,7 @@ أي فريق ستمثله في هذه المعركة؟ يجب أن تنضم إلى أحد هذه الفرق للمشاركة! تم الإنشاء + لعب مؤخرًا أفضل النتائج الإحصاءات إحصائيات البطولة diff --git a/translation/dest/broadcast/af-ZA.xml b/translation/dest/broadcast/af-ZA.xml index f98e438c79129..d0e649c0eeae4 100644 --- a/translation/dest/broadcast/af-ZA.xml +++ b/translation/dest/broadcast/af-ZA.xml @@ -14,9 +14,7 @@ Kort beskrywing van die toernooi Volle geleentheid beskrywing Opsionele lang beskrywing van die uitsending. %1$s is beskikbaar. Lengte moet minder as %2$s karakters. - Bronadress, of spel IDs URL wat Lichess sal nagaan vir PGN opdaterings. Dit moet openbaar beskikbaar wees vanaf die Internet. - Of anders, kan jy tot en met 64 Lichess spel IDs bevoeg, gespasieer deur spasie. Begin datum in jou eie tydsone Optioneel, indien jy weet wanner die geleentheid begin Gee krediet aan die bron diff --git a/translation/dest/broadcast/ar-SA.xml b/translation/dest/broadcast/ar-SA.xml index e601455155848..fa854c94789a4 100644 --- a/translation/dest/broadcast/ar-SA.xml +++ b/translation/dest/broadcast/ar-SA.xml @@ -27,9 +27,7 @@ وصف موجز للبطولة الوصف الكامل الوصف الاختياري الطويل للبث. %1$s متوفر. يجب أن لا يتجاوز طول النص %2$s حرفاً. - URL المصدر، أو ID المباراة URL الذي سيتحقق منه Lichess للحصول على تحديثات PGN. يجب أن يكون متاحًا للجميع على الإنترنت. - بدلاً من ذلك، يمكنك إدخال ما يصل إلى 64 ID لمباريات Lichess، مفصولة بمسافات. تاريخ البدء في المنطقة الزمنية الخاصة بك اختياري، إذا كنت تعرف متى يبدأ الحدث ائتمن المصدر diff --git a/translation/dest/broadcast/az-AZ.xml b/translation/dest/broadcast/az-AZ.xml index d0ee3ce6bce32..d02db396eec5b 100644 --- a/translation/dest/broadcast/az-AZ.xml +++ b/translation/dest/broadcast/az-AZ.xml @@ -13,9 +13,7 @@ Qısa turnir açıqlaması Tədbirin tam açıqlaması Tədbirin istəyə bağlı təfsilatlı açıqlaması. %1$s seçimi mövcuddur. Mətnin uzunluğu %2$s simvoldan az olmalıdır. - Mənbə URL və ya oyun ID-si Lichess, verdiyiniz URL ilə PGN-i yeniləyəcək. Bu internetdə hamı tərəfindən əldə edilə bilən olmalıdır. - Alternativ olaraq, boşluqlarla ayrılmış 64-ə qədər Lichess oyun ID-ni daxil edə bilərsiniz. Öz saat qurşağınızdakı başlama tarixi İstəyə bağlı, tədbirin başlama vaxtını bilirsinizsə Mənbəyə bax diff --git a/translation/dest/broadcast/be-BY.xml b/translation/dest/broadcast/be-BY.xml index 2e77ade23cb5f..75968908b127e 100644 --- a/translation/dest/broadcast/be-BY.xml +++ b/translation/dest/broadcast/be-BY.xml @@ -16,9 +16,7 @@ Сціслае апісанне турніру Поўнае апісанне турніру Неабавязковая дасканалае апісанне турніру. Даступны %1$s. Даўжыня павінна быць менш за %2$s сімвалаў. - Зыходная спасылка, або ідэнтыфікатары гульні Спасылка, з якой Lichess паспрабуе атрымоўваць абнаўленні PGN. Яны павінна быць даступнай для кожнай ва Інтэрнэце. - Інакш, вы можаце праз прабел ўвесці 64 сімвальныя ID гульняў на Lichess. Дата пачатаку ў вашым часавым поясе Па жаданні, калі вы ведаеце пачатак падзеі Падзякаваць крыніцы diff --git a/translation/dest/broadcast/bg-BG.xml b/translation/dest/broadcast/bg-BG.xml index 78c2bc702ea71..87a40894b8942 100644 --- a/translation/dest/broadcast/bg-BG.xml +++ b/translation/dest/broadcast/bg-BG.xml @@ -19,9 +19,7 @@ Кратко описание на турнира Пълно описание на събитието Незадължително дълго описание на излъчването. %1$s е налично. Дължината трябва да по-малка от %2$s знака. - Изходен URL, или идентификатор (ID) на партията Уебадресът, който Lichess ще проверява, за да получи осъвременявания на PGN. Той трябва да е публично достъпен от интернет. - Вместо това, може да въведете до 64 идентификатора на Lichess партии, разделени с интервал. Дата на започване във вашата часова зона По избор, ако знаете, кога започва събитието Признателност на източника diff --git a/translation/dest/broadcast/br-FR.xml b/translation/dest/broadcast/br-FR.xml index 268d7569976a2..e65733441ab66 100644 --- a/translation/dest/broadcast/br-FR.xml +++ b/translation/dest/broadcast/br-FR.xml @@ -10,7 +10,6 @@ Deskrivadur an abadenn a-bezh Deskrivadur hir ar skignañ war-eeun ma fell deoc\'h.%1$s zo dijabl. Ne vo ket hiroc\'h evit %2$s sin. An URL a ray Lichess ganti evit kaout hizivadurioù ar PGN. Ret eo dezhi bezañ digor d\'an holl war Internet. - A-hend all e c\'hallit lakaat betek 64 ID partienn Lichess, dispartiet gant spasoù. Eur kregiñ en ho kwerzhid-eur Diret eo, ma ouzit pegoulz e krogo Orin ar vammenn diff --git a/translation/dest/broadcast/bs-BA.xml b/translation/dest/broadcast/bs-BA.xml index 130447e5c09a0..7d537e804b639 100644 --- a/translation/dest/broadcast/bs-BA.xml +++ b/translation/dest/broadcast/bs-BA.xml @@ -19,9 +19,7 @@ Kratak opis turnira Potpuni opis događaja Neobavezni dugi opis događaja koji se emituje. %1$s je dostupan. Dužina mora biti manja od %2$s slova. - URL izvora ili identifikacijski broj partije Link koji će Lichess koristiti kako bi redovno ažurirao PGN. Mora biti javno dostupan na internetu. - Alternativno, možete unijeti do 64 identifikacijska broja partija na Lichessu, odvojena razmacima. Datum početka po Vašoj vremenskoj zoni Neobavezno, ukoliko znate kada počinje događaj Navedite ko je zaslužan diff --git a/translation/dest/broadcast/ca-ES.xml b/translation/dest/broadcast/ca-ES.xml index 9104c81ab6ae7..a57524541db57 100644 --- a/translation/dest/broadcast/ca-ES.xml +++ b/translation/dest/broadcast/ca-ES.xml @@ -23,9 +23,7 @@ Breu descripció del torneig Descripció total de l\'esdeveniment Opció de llarga descripció de l\'esdeveniment. %1$s és disponible. Ha de tenir menys de %2$s lletres. - Enllaç d\'origen, o identificador de la partida URL que Lichess comprovarà per a obtenir actualitzacions PGN. Ha de ser públicament accessible des d\'Internet. - Alternativament, pots introduir fins a 64 identificadors de partida, separats per espais. Data d\'inici en la teva zona horària Opcional, si saps quan comença l\'esdeveniment Cita la font diff --git a/translation/dest/broadcast/ckb-IR.xml b/translation/dest/broadcast/ckb-IR.xml index bce8b67a194fa..5fca9553a022d 100644 --- a/translation/dest/broadcast/ckb-IR.xml +++ b/translation/dest/broadcast/ckb-IR.xml @@ -21,9 +21,7 @@ باسی پاڵەوانێتیەکە بکە بە گشتی باسی پالەوانێتیەکە بکە وەسفێکی درێژی ھەڵبژاردنی پاڵەوانێتییەکە. %1$s بەردەستە. درێژی دەبێت کەمتر بێت لە کاراکتەرەکانی %2$s. - URL سەرچاوە، یا ID یارییەکە URL مافی لی چێسە بۆ بەردەست خستنی تازەکردنەوە PGN پێویستە بۆ ھەموان بەردەست بێت لەسەر ئەنتەرنێت - یان دەتوانیت تا ٦٤ ناسنامەی یاری Lichess دابنێیت، کە بە بۆشایی جیا دەکرێنەوە. بەرواری دەستپێکردن لە چوارچێوەی کاتی خۆتدا بە ھەلبژاردنی خۆتە چ کاتێک دەس پێ بکات لە سەرچاوەکە دڵنیابە diff --git a/translation/dest/broadcast/cs-CZ.xml b/translation/dest/broadcast/cs-CZ.xml index 4183cf222a4e9..5827af04c0b2e 100644 --- a/translation/dest/broadcast/cs-CZ.xml +++ b/translation/dest/broadcast/cs-CZ.xml @@ -24,9 +24,7 @@ Stručný popis turnaje Úplný popis události Volitelný dlouhý popis přenosu. %1$s je k dispozici. Délka musí být menší než %2$s znaků. - Adresa URL či ID hry URL adresa, kterou bude Lichess kontrolovat pro získání PGN aktualizací. Musí být veřejně přístupná z internetu. - Případně můžete zadat až 64 Lichess ID partií oddělených mezerou. Datum a čas zahájení ve vašem časovém pásmu Nepovinné, pokud víte, kdy událost začíná Uveďte zdroj diff --git a/translation/dest/broadcast/da-DK.xml b/translation/dest/broadcast/da-DK.xml index adbfca1eb533e..9214e24fc7640 100644 --- a/translation/dest/broadcast/da-DK.xml +++ b/translation/dest/broadcast/da-DK.xml @@ -23,9 +23,9 @@ Kort beskrivelse af turnering Fuld beskrivelse af begivenheden Valgfri lang beskrivelse af transmissionen. %1$s er tilgængelig. Længde skal være mindre end %2$s tegn. - Kilde URL, eller spil-ID\'er + URL for PGN-kilde URL som Lichess vil trække på for at få PGN updates. Den skal være offentlig tilgængelig fra internettet. - Alternativt kan du indtaste op til 64 Lichess spil-ID\'er, adskilt af mellemrum. + Op til 64 Lichess parti-ID\'er, adskilt af mellemrum. Startdato i din egen tidszone Valgfri, hvis du ved, hvornår begivenheden starter Krediter kilden diff --git a/translation/dest/broadcast/de-DE.xml b/translation/dest/broadcast/de-DE.xml index 9c4fca5d170de..54fcd97d89d31 100644 --- a/translation/dest/broadcast/de-DE.xml +++ b/translation/dest/broadcast/de-DE.xml @@ -23,9 +23,9 @@ Kurze Turnierbeschreibung Vollständige Ereignisbeschreibung Optionale, ausführliche Beschreibung der Übertragung. %1$s ist verfügbar. Die Beschreibung muss kürzer als %2$s Zeichen sein. - Quell-URL oder Partie-IDs + PGN Quell-URL URL die Lichess abfragt um PGN Aktualisierungen zu erhalten. Sie muss öffentlich aus dem Internet zugänglich sein. - Alternativ kannst du bis zu 64 durch Leerzeichen getrennte Lichess-Partie-IDs eingeben. + Bis zu 64 Lichess Partie-IDs, getrennt durch Leerzeichen. Startdatum in deiner eigenen Zeitzone Optional, falls du weißt wann das Ereignis beginnt Erwähne die Quelle diff --git a/translation/dest/broadcast/el-GR.xml b/translation/dest/broadcast/el-GR.xml index a47fe00d23289..5aeb0c835599f 100644 --- a/translation/dest/broadcast/el-GR.xml +++ b/translation/dest/broadcast/el-GR.xml @@ -22,9 +22,7 @@ Σύντομη περιγραφή τουρνουά Πλήρης περιγραφή γεγονότος Προαιρετική αναλυτική περιγραφή της αναμετάδοσης. Η μορφή %1$s είναι διαθέσιμη. Το μήκος πρέπει μικρότερο από %2$s χαρακτήρες. - Διεύθυνση URL πηγής ή ID παιχνιδιού URL για λήψη PGN ενημερώσεων. Πρέπει να είναι δημόσια προσβάσιμο μέσω διαδικτύου. - Εναλλακτικά, μπορείτε να εισάγετε έως 64 IDs παιχνιδιών Lichess, με κενά μεταξύ τους. Ημερομηνία έναρξης στη δική σας ζώνη ώρας Προαιρετικό, εάν γνωρίζετε πότε αρχίζει η εκδήλωση Αναφέρετε την πηγή diff --git a/translation/dest/broadcast/en-US.xml b/translation/dest/broadcast/en-US.xml index 43ffd27d234ad..449b09b6167d1 100644 --- a/translation/dest/broadcast/en-US.xml +++ b/translation/dest/broadcast/en-US.xml @@ -23,9 +23,9 @@ Short tournament description Full event description Optional long description of the broadcast. %1$s is available. Length must be less than %2$s characters. - Source URL, or game IDs + PGN Source URL URL that Lichess will check to get PGN updates. It must be publicly accessible from the Internet. - Alternatively, you can enter up to 64 Lichess game IDs, separated by spaces. + Up to 64 Lichess game IDs, separated by spaces. Start date in your own timezone Optional, if you know when the event starts Credit the source diff --git a/translation/dest/broadcast/eo-UY.xml b/translation/dest/broadcast/eo-UY.xml index 68ed84a1c5993..ff7c8046daf57 100644 --- a/translation/dest/broadcast/eo-UY.xml +++ b/translation/dest/broadcast/eo-UY.xml @@ -23,9 +23,7 @@ Mallonga turnira priskribo Plena eventa priskribo Laŭvola longa priskribo de la elsendo. %1$s haveblas. Longeco devas esti malpli ol %2$s literoj. - Fontaj URL-oj, aŭ ludaj ID-oj URL kiun Lichess kontrolos por akiri PGN ĝisdatigojn. Ĝi devas esti publike alirebla en interreto. - Alternative, vi povas enmeti ĝis 64 Lichess ludajn ID-ojn, apartigitajn per spacoj. Komenca dato en via propra horzono Laŭvola, se vi scias, kiam komenciĝas la evento Citu la fonton diff --git a/translation/dest/broadcast/es-ES.xml b/translation/dest/broadcast/es-ES.xml index cee42009ee332..715a6e1f9aada 100644 --- a/translation/dest/broadcast/es-ES.xml +++ b/translation/dest/broadcast/es-ES.xml @@ -23,9 +23,9 @@ Breve descripción del torneo Descripción completa del evento Descripción larga opcional de la emisión. %1$s está disponible. La longitud debe ser inferior a %2$s caracteres. - Enlace de origen o identificador de la partida + URL origen del archivo PGN URL que Lichess comprobará para obtener actualizaciones PGN. Debe ser públicamente accesible desde Internet. - Alternativamente, puedes introducir hasta 64 identificadores de partida, separados por espacios. + Hasta 64 identificadores de partidas de Lichess, separados por espacios. Fecha de inicio en tu zona horaria Opcional, si sabes cuando comienza el evento Cita la fuente diff --git a/translation/dest/broadcast/et-EE.xml b/translation/dest/broadcast/et-EE.xml index f7f0681108a65..46a035083600f 100644 --- a/translation/dest/broadcast/et-EE.xml +++ b/translation/dest/broadcast/et-EE.xml @@ -13,9 +13,7 @@ Lühike turniiri kirjeldus Sündmuse täielik kirjeldus Valikuline otseülekande kirjeldus. %1$s on saadaval. Pikkus peab olema maksimaalselt %2$s tähemärki. - Allika URL või mängu ID-d URL, kust Lichess saab PGN-i värskenduse. See peab olema Internetist kättesaadav. - Või siis on võimalik lisada 64 Lichess mängu ID-t, vahedega eraldatud. Alguskuupäev sinu ajavööndis Valikuline, kui tead millal sündmus algab Viita allikale diff --git a/translation/dest/broadcast/eu-ES.xml b/translation/dest/broadcast/eu-ES.xml index d9bc2070db07f..e4e2a834d6743 100644 --- a/translation/dest/broadcast/eu-ES.xml +++ b/translation/dest/broadcast/eu-ES.xml @@ -23,9 +23,7 @@ Txapelketaren deskribapen laburra Ekitaldiaren deskribapen osoa Emanaldiaren azalpen luzea, hautazkoa da. %1$s badago. Luzera %2$s karaktere edo laburragoa izan behar da. - Iturburu URLa edo partiden IDak Lichessek PGNaren eguneraketak jasoko dituen URLa. Interneteko helbide bat izan behar da. - Bestela, Lichess partiden 64 id sartu ditzakezu, espazioarekin banatuta. Zure ordu-zonako hasiera data Hautazkoa, ekitaldia noiz hasten den baldin badakizu Jatorria zein den esaiguzu diff --git a/translation/dest/broadcast/fa-IR.xml b/translation/dest/broadcast/fa-IR.xml index ef708cc564d81..9c0bb795b9ed1 100644 --- a/translation/dest/broadcast/fa-IR.xml +++ b/translation/dest/broadcast/fa-IR.xml @@ -23,9 +23,8 @@ توضیحات کوتاه مسابقات توضیحات کامل مسابقات توضیحات بلند و اختیاری پخش همگانی. %1$s قابل‌استفاده است. طول متن باید کمتر از %2$s نویسه باشد. - نشانی منبع، یا شناسه بازی‌ها وب‌نشانی‌ای که Lichess برای دریافت به‌روزرسانی‌های PGN می‌بررسد. آن باید از راه اینترنت در دسترس همگان باشد. - همچنین، می‌توانید تا ۶۴ شناسه بازی Lichess را که با *فاصله* از هم جدا شده‌اند، وارد کنید. + تا ۶۴ نشانه بازی لیچس٬ جداشده با فاصله. تاریخ شروع، در منطقه زمانی خودتان اختیاری است، اگر می‌دانید چه زمانی رویداد شروع می‌شود به منبع اعتبار دهید diff --git a/translation/dest/broadcast/fi-FI.xml b/translation/dest/broadcast/fi-FI.xml index ad045ec9af4c4..6e54bd3b0bdf1 100644 --- a/translation/dest/broadcast/fi-FI.xml +++ b/translation/dest/broadcast/fi-FI.xml @@ -23,9 +23,9 @@ Turnauksen lyhyt kuvaus Täysimittainen kuvaus tapahtumasta Ei-pakollinen pitkä kuvaus lähetyksestä. %1$s-muotoiluja voi käyttää. Pituus voi olla enintään %2$s merkkiä. - Lähteen URL-osoite tai peli ID:t + PGN:n lähde-URL URL, josta Lichess hakee PGN-päivitykset. Sen täytyy olla julkisesti saatavilla internetissä. - Vaihtoehtoisesti, voit syöttää jopa 64 Lichess peli ID:tä, erotettuna välilyönneillä. + Korkeintaan 64 Lichess-pelin tunnistenumeroa välilyönneillä eroteltuna. Alkamispäivämäärä omalla aikavyöhykkeelläsi Ei-pakollinen, laita jos tiedät milloin tapahtuma alkaa Mainitse lähde diff --git a/translation/dest/broadcast/fo-FO.xml b/translation/dest/broadcast/fo-FO.xml index f50d18b0c4eeb..0cb94cce03c13 100644 --- a/translation/dest/broadcast/fo-FO.xml +++ b/translation/dest/broadcast/fo-FO.xml @@ -9,9 +9,7 @@ Nummar á umfari Fullfíggjað lýsing av tiltaki Valfrí long lýsing av sending. %1$s er tøkt. Longdin má vera styttri enn %2$s bókstavir. - Keldu-URL ella talvsamleikar URL-leinki, ið Lichess fer at kanna til tess at fáa PGN dagføringar. Leinkið nýtist at vera alment atkomiligt á alnetinum. - Til ber eisini ístaðin at skriva 64 Lichess-samleikar upp. Teir noyðast at vera skildir sundur við millumrúmum. Byrjanardagfesting í tínum egna tíðarøki Valfrítt, um tú veitst, nær tiltakið byrjar Nevn kelduna diff --git a/translation/dest/broadcast/fr-FR.xml b/translation/dest/broadcast/fr-FR.xml index 39d2deef5c60d..49a3ee11bde87 100644 --- a/translation/dest/broadcast/fr-FR.xml +++ b/translation/dest/broadcast/fr-FR.xml @@ -23,9 +23,9 @@ Brève description du tournoi Description complète de l\'événement Description détaillée et optionnelle de la diffusion. %1$s est disponible. La longueur doit être inférieure à %2$s caractères. - URL source ou IDs de parties + URL source de la partie en PGN URL que Lichess interrogera pour obtenir les mises à jour du PGN. Elle doit être accessible publiquement depuis Internet. - Vous pouvez également entrer jusqu\'à 64 IDs de parties Lichess séparés par un espace. + Jusqu\'à 64 ID de partie Lichess séparés par des espaces. Date de début dans votre fuseau horaire Facultatif, si vous savez quand l\'événement commence Créditer la source diff --git a/translation/dest/broadcast/ga-IE.xml b/translation/dest/broadcast/ga-IE.xml index 4941e6601b6ac..97bab1791832e 100644 --- a/translation/dest/broadcast/ga-IE.xml +++ b/translation/dest/broadcast/ga-IE.xml @@ -13,9 +13,7 @@ Cur síos gairid ar an gcomórtas Cur síos iomlán ar an ócáid Cur síos fada roghnach ar an craoladh. Tá %1$s ar fáil. Caithfidh an fad a bheith níos lú ná %2$s carachtar. - URL foinse, nó cluiche CA URL a seiceálfaidh Lichess chun PGN nuashonruithe a fháil. Caithfidh sé a bheith le féiceáil go poiblí ón Idirlíon. - Nó is féidir suas le 64 ID cluiche Lichess a iontráil, scartha le spásanna. Dáta tosaigh i do chrios ama féin Roghnach, má tá a fhios agat cathain a thosóidh an ócáid Creidmheas an fhoinse diff --git a/translation/dest/broadcast/gl-ES.xml b/translation/dest/broadcast/gl-ES.xml index cd2de28ff4eee..4852c77cbf46b 100644 --- a/translation/dest/broadcast/gl-ES.xml +++ b/translation/dest/broadcast/gl-ES.xml @@ -23,9 +23,9 @@ Breve descrición do torneo Descrición completa do evento Descrición longa opcional da retransmisión. %1$s está dispoñíbel. A lonxitude debe ser menor de %2$s caracteres. - Ligazón de orixe ou identificador da partida + URL de orixe do arquivo PGN Ligazón que Lichess comprobará para obter actualizacións dos PGN. Debe ser publicamente accesíbel desde a Internet. - Alternativamente, podes introducir ata 64 identificadores de partida, separados por espazos. + Até 64 identificadores de partidas de Lichess, separados por espazos. Data de inicio na túa zona horaria Opcional, se sabes cando comeza o evento Cita a fonte diff --git a/translation/dest/broadcast/gsw-CH.xml b/translation/dest/broadcast/gsw-CH.xml index 168ad25abcbd3..7a3b2f16d796b 100644 --- a/translation/dest/broadcast/gsw-CH.xml +++ b/translation/dest/broadcast/gsw-CH.xml @@ -23,9 +23,7 @@ Churzi Turnier Beschribig Vollschtändigi Ereignisbeschribig Optionali, usfüehrlichi Beschribig vu de Überträgig. %1$s isch verfügbar. Die Beschribig muess chürzer als %2$s Zeiche si. - Quell-URL oder Partie-IDs URL wo Lichess abfrögt, für PGN Aktualisierige z\'erhalte. Sie muess öffentlich im Internet zuegänglich si. - Alternativ chasch bis zu 64 dur Leerzeiche trännti Lichess Partie IDs igeh. Startdatum in dinere eigene Zitzone Optional, falls du weisch, wänn das Ereignis afangt Erwähn die Quälle diff --git a/translation/dest/broadcast/gu-IN.xml b/translation/dest/broadcast/gu-IN.xml index 7a6c4fad92c5b..3ea04e700dfa8 100644 --- a/translation/dest/broadcast/gu-IN.xml +++ b/translation/dest/broadcast/gu-IN.xml @@ -1,5 +1,2 @@ - - સ્રોત ચુ. આર. એલ., અથવા રમત આઇ. ડીસ - વૈકલ્પિક રીતે, તમેં 64 લાઇચેસ રમત આઇડીસ નાખિ શકો છો, વચ્ચે સ્પેસ નાખિને. - + diff --git a/translation/dest/broadcast/he-IL.xml b/translation/dest/broadcast/he-IL.xml index 099739d2f7d7b..f938cd4c992a4 100644 --- a/translation/dest/broadcast/he-IL.xml +++ b/translation/dest/broadcast/he-IL.xml @@ -25,9 +25,9 @@ תיאור הטורניר בקצרה תיאור מלא של הטורניר תיאור מפורט של הטורניר (אופציונאלי). %1$s זמין. אורך התיאור לא יעלה על %2$s תווים. - קישור למקור (URL) או מזהה המשחקים (IDs) + קישור המקור של ה-PGN הקישור ש־Lichess יבדוק כדי לקלוט עדכונים ב-PGN. הוא חייב להיות פומבי ונגיש דרך האינטרנט. - לחלופין, ניתן להזין עד 64 IDs של משחקי ליצ׳ס, מופרדים ברווחים. + עד 64 מזהי משחק של Lichess, מופרדים ברווחים. תאריך ההתחלה באזור הזמן שלך אופציונאלי, אם את/ה יודע/ת מתי האירוע צפוי להתחיל תן/י קרדיט למקור diff --git a/translation/dest/broadcast/hi-IN.xml b/translation/dest/broadcast/hi-IN.xml index d97e2c0670f6b..d27142de5a14f 100644 --- a/translation/dest/broadcast/hi-IN.xml +++ b/translation/dest/broadcast/hi-IN.xml @@ -18,9 +18,7 @@ संक्षिप्त प्रतियोगिता वर्णन संक्षिप्त वर्णन प्रसारण का वैकल्पिक लंबा विवरण. %1$s उपलब्ध है. लंबाई %2$s से कम होना चाहिए - स्रोत URL, या गेम आईडी URL जो Lichess PGN अपडेट प्राप्त करने के लिए जाँच करेगा। यह सार्वजनिक रूप से इंटरनेट पर सुलभ होना चाहिए। - वैकल्पिक रूप से, आप रिक्त स्थान द्वारा अलग किए गए 64 Lichess गेम आईडी तक दर्ज कर सकते हैं। अपने स्वयं के समयक्षेत्र में प्रारंभ दिनांक वैकल्पिक, यदि आप जानना चाहते हो की प्रतिस्प्रधा कब शुरू होगी स्रोत को श्रेय दें diff --git a/translation/dest/broadcast/hr-HR.xml b/translation/dest/broadcast/hr-HR.xml index 9abe54e1519b7..5a8a5071df827 100644 --- a/translation/dest/broadcast/hr-HR.xml +++ b/translation/dest/broadcast/hr-HR.xml @@ -18,9 +18,7 @@ Kratak opis turnira Potpuni opis događaja Neobavezni dugi opis prijenosa. %1$s je dostupno. Duljina mora biti manja od %2$s znakova. - URL izvora ili identifikacijski broj igre Link koji će Lichess ispitavati kako bi dobio PGN ažuriranja. Mora biti javno dostupan s interneta. - Možete i unijeti do 64 identifikacijska broja igara Lichessa, odvojenih razmacima. Datum početka u vlastitoj vremenskoj zoni Neobavezno, ako znaš kada događaj počinje Naglasi izvor diff --git a/translation/dest/broadcast/hu-HU.xml b/translation/dest/broadcast/hu-HU.xml index 832da2213ff91..d0adfb37758bc 100644 --- a/translation/dest/broadcast/hu-HU.xml +++ b/translation/dest/broadcast/hu-HU.xml @@ -17,9 +17,7 @@ Verseny rövid leírása Esemény teljes leírása Opcionális a közvetítés még bővebb leírása. %1$s használható. A hossz nem lehet több, mint %2$s karakter. - Forrás URL vagy játszmaazonosítók URL amit a Lichess időnként PGN frissítésekért ellenőriz. Ennek nyilvános internetcímnek kell lennie. - Illetve megadhatsz legfeljebb 64 Lichess játszmaazonosítót, szóközökkel elválasztva. Kezdés időpontja a saját időzónádban Opcionális, ha tudod mikor kezdődik az esemény Forrásmegjelölés diff --git a/translation/dest/broadcast/id-ID.xml b/translation/dest/broadcast/id-ID.xml index 9f04a7fd5527e..80fb0440c0161 100644 --- a/translation/dest/broadcast/id-ID.xml +++ b/translation/dest/broadcast/id-ID.xml @@ -13,9 +13,7 @@ Deskripsi singkat turnamen Keterangan acara secara penuh Deskripsi panjang opsional dari siaran. %1$s tersedia. Panjangnya harus kurang dari %2$s karakter. - Sumber URL, atau IDs game URL yang akan di-polling oleh Lichess untuk mendapatkan pembaruan PGN. Itu harus dapat diakses publik dari Internet. - Kalau tidak, kamu bisa memasukkan lebih dari 64 Lichess IDs game, dipisahkan oleh spasi. Tanggal mulai di zona waktu Anda sendiri Opsional, jika Anda tahu kapan acara dimulai Sertakan sumber diff --git a/translation/dest/broadcast/it-IT.xml b/translation/dest/broadcast/it-IT.xml index bca4380d9abcd..adc748be56885 100644 --- a/translation/dest/broadcast/it-IT.xml +++ b/translation/dest/broadcast/it-IT.xml @@ -23,9 +23,7 @@ Breve descrizione dell\'evento Descrizione completa dell\'evento (Facoltativo) Descrizione completa dell\'evento. %1$s è disponibile. La lunghezza deve essere inferiore a %2$s caratteri. - URL sorgente, o ID gioco L\'URL che Lichess utilizzerà per ottenere gli aggiornamenti dei PGN. Deve essere accessibile pubblicamente su Internet. - In alternativa è possibile inserire fino a 64 ID di partite su Lichess, separati da spazi. Data di inizio nel tuo fuso orario Facoltativo, se sai quando inizia l\'evento Cita la fonte diff --git a/translation/dest/broadcast/ja-JP.xml b/translation/dest/broadcast/ja-JP.xml index 37bcba2c1f51a..53bdbd425abe8 100644 --- a/translation/dest/broadcast/ja-JP.xml +++ b/translation/dest/broadcast/ja-JP.xml @@ -22,9 +22,7 @@ 大会の短い説明 長い説明 内容の詳しい説明(オプション)。%1$s が利用できます。長さは [欧文換算で] %2$s 字まで。 - ソース URLかゲーム ID Lichess が PGN を取得するための URL。インターネット上に公表されているもののみ。 - Lichess ゲーム ID は半角スペースで区切れば最大 64 個まで入力できます。 開始日付(あなたの現地時間) イベント開始時刻(オプション) ソースを表示する diff --git a/translation/dest/broadcast/kk-KZ.xml b/translation/dest/broadcast/kk-KZ.xml index 854b352a05ea4..6406d3b24d3af 100644 --- a/translation/dest/broadcast/kk-KZ.xml +++ b/translation/dest/broadcast/kk-KZ.xml @@ -18,9 +18,7 @@ Жарыстың қысқа сипаттамасы Оқиғаның толық сипаттамасы Көрсетілімнің қосымша үлкен сипаттамасы. %1$s қолданысқа ашық. Ұзындығы %2$s таңбадан кем болуы керек. - Сілтемесі не ойын нөмірлері PGN жаңартуларын алу үшін Личес тексеретін сілтеме. Ол интернетте баршалыққа ашық болуы керек. - Одан бөлек, бос орынмен ажыратып, ең көбі 64 Личес ойындарының нөмірін енгізе аласыз. Басталу күні (өз уақыт белдеуіңізде) Міндетті емес, егер күнін біліп тұрсаңыз Қайнар көзіне сілтеңіз diff --git a/translation/dest/broadcast/kn-IN.xml b/translation/dest/broadcast/kn-IN.xml index 1132b463203e7..132a6c33e8ba4 100644 --- a/translation/dest/broadcast/kn-IN.xml +++ b/translation/dest/broadcast/kn-IN.xml @@ -17,9 +17,7 @@ ಸಣ್ಣ ಪಂದ್ಯಾವಳಿಯ ವಿವರಣೆ ಪಂದ್ಯಾವಳಿಯ ಸಂಪೂರ್ಣ ವಿವರಣೆ ಪಂದ್ಯಾವಳಿಯ ಐಚ್ಛಿಕ ದೀರ್ಘ ವಿವರಣೆ. %1$s ಲಭ್ಯವಿದೆ. ಉದ್ದವು %2$s ಅಕ್ಷರಗಳಿಗಿಂತ ಕಡಿಮೆಯಿರಬೇಕು. - ಮೂಲ URL, ಅಥವಾ ಆಟದ ಐಡಿಗಳು PGN ನವೀಕರಣಗಳನ್ನು ಪಡೆಯಲು Lichess ಪರಿಶೀಲಿಸುವ URL. ಇದನ್ನು ಇಂಟರ್ನೆಟ್‌ನಿಂದ ಸಾರ್ವಜನಿಕವಾಗಿ ಪ್ರವೇಶಿಸಬೇಕು. - ಪರ್ಯಾಯವಾಗಿ, ನೀವು 64 ಲೈಚೆಸ್ ಗೇಮ್ ID ಗಳನ್ನು ನಮೂದಿಸಬಹುದು, ಇದನ್ನು ಸ್ಪೇಸ್‌ಗಳಿಂದ ಬೇರ್ಪಡಿಸಬಹುದು. ನಿಮ್ಮ ಸ್ವಂತ ಸಮಯವಲಯದಲ್ಲಿ ದಿನಾಂಕವನ್ನು ಪ್ರಾರಂಭಿಸಿ ಐಚ್ಛಿಕ, ಈವೆಂಟ್ ಯಾವಾಗ ಪ್ರಾರಂಭವಾಗುತ್ತದೆ ಎಂದು ನಿಮಗೆ ತಿಳಿದಿದ್ದರೆ ಮೂಲವನ್ನು ಕ್ರೆಡಿಟ್ ಮಾಡಿ diff --git a/translation/dest/broadcast/ko-KR.xml b/translation/dest/broadcast/ko-KR.xml index 373ddcf389856..162ee9fefe881 100644 --- a/translation/dest/broadcast/ko-KR.xml +++ b/translation/dest/broadcast/ko-KR.xml @@ -20,9 +20,7 @@ 짧은 토너먼트 설명 전체 이벤트 설명 (선택) 방송에 대한 긴 설명입니다. %1$s 사용이 가능합니다. 길이는 %2$s 글자보다 짧아야 합니다. - 출처 URL 또는 게임 ID Lichess가 PGN 업데이트를 받기 위해 확인할 URL입니다. 인터넷에서 공개적으로 액세스 할 수 있어야 합니다. - 또는, 공백으로 구분하여 최대 64 개의 Lichess 게임 ID를 입력 할 수 있습니다. 본인 시간대 기준 시작일 선택사항. 언제 이벤트가 시작되는지 알고 있는 경우 출처 diff --git a/translation/dest/broadcast/lb-LU.xml b/translation/dest/broadcast/lb-LU.xml index 7e9e837182ac7..fe30e6accdee3 100644 --- a/translation/dest/broadcast/lb-LU.xml +++ b/translation/dest/broadcast/lb-LU.xml @@ -17,9 +17,7 @@ Kuerz Turnéierbeschreiwung Komplett Turnéierbeschreiwung Optional laang Beschreiwung vum Turnéier. %1$s ass disponibel. Längt muss manner wéi %2$s Buschtawen sinn. - Quellen-URL oder Partien IDs URL déi Lichess checkt fir PGN à jour ze halen. Muss ëffentlech iwwer Internet zougänglech sinn. - Alternativ kanns du och bis zu 64 Lichess Partien IDs aginn, getrennt duerch Espacen. Startdatum an denger eegener Zäitzon Optional, wann du wees wéini den Turnéier ufänkt Quell kreditéieren diff --git a/translation/dest/broadcast/lt-LT.xml b/translation/dest/broadcast/lt-LT.xml index 8198580d28b4d..c4c5fcdf314cb 100644 --- a/translation/dest/broadcast/lt-LT.xml +++ b/translation/dest/broadcast/lt-LT.xml @@ -20,9 +20,7 @@ Trumpas turnyro aprašymas Pilnas renginio aprašymas Neprivalomas pilnas transliacijos aprašymas. Galima naudoti %1$s. Ilgis negali viršyti %2$s simbolių. - Šaltinių adresai arba žaidimų id URL, į kurį „Lichess“ kreipsis gauti PGN atnaujinimus. Privalo būti viešai pasiekiamas internete. - Taip pat galite įvesti iki 64 Lichess žaidimų id, atskyrę juos tarpais. Pradžios laikas jūsų laiko juostoje Neprivaloma; tik jeigu žinote, kada prasideda renginys Paminėkite šaltinį diff --git a/translation/dest/broadcast/lv-LV.xml b/translation/dest/broadcast/lv-LV.xml index 8242152a35009..92eac6b60cebb 100644 --- a/translation/dest/broadcast/lv-LV.xml +++ b/translation/dest/broadcast/lv-LV.xml @@ -13,9 +13,7 @@ Īss turnīra apraksts Pilns pasākuma apraksts Neobligāts garš raidījuma apraksts. Pieejams %1$s. Garumam jābūt mazāk kā %2$s rakstzīmēm. - Avota URL, vai spēļu ID URL, ko Lichess aptaujās, lai iegūtu PGN atjauninājumus. Tam jābūt publiski piekļūstamam no interneta. - Varat arī ievadīt līdz pat 64 Lichess spēļu ID, atdalītus ar atstarpēm. Sākuma datums jūsu laika joslā Neobligāts, ja zināt, kad pasākums sākas Kreditējiet avotu diff --git a/translation/dest/broadcast/mr-IN.xml b/translation/dest/broadcast/mr-IN.xml index d48b77fccd3b7..3111fb098a28d 100644 --- a/translation/dest/broadcast/mr-IN.xml +++ b/translation/dest/broadcast/mr-IN.xml @@ -22,9 +22,7 @@ लघु स्पर्धेचे वर्णन संपूर्ण स्पर्धेचे वर्णन स्पर्धेचे पर्यायी संपूर्ण वर्णन. %1$s उपलब्ध आहे. %2$s अक्षरांपेक्षा कमी. - Source URL, or game IDs URL जी लाइसेस PGN अद्यतने मिळविण्यासाठी तपासणी करेल. ते इंटरनेट वरून सार्वजनिकरित्या प्रवेश करण्यायोग्य असणे आवश्यक आहे. - वैकल्पिकरित्या, आपण स्पेस देऊन ६४ पर्यंत लीचेस गेम आयडी प्रविष्ट करू शकता. आपल्या स्वत: च्या टाइमझोनमधील तारीख पर्यायी, इव्हेंट कधी सुरू होतो हे आपल्याला माहिती असल्यास Credit the source diff --git a/translation/dest/broadcast/nb-NO.xml b/translation/dest/broadcast/nb-NO.xml index a2bf0865ecdb8..55d652db79fa9 100644 --- a/translation/dest/broadcast/nb-NO.xml +++ b/translation/dest/broadcast/nb-NO.xml @@ -23,9 +23,7 @@ Kort beskrivelse av turneringen Full beskrivelse av turneringen Valgfri lang beskrivelse av turneringen. %1$s er tilgjengelig. Beskrivelsen må være kortere enn %2$s tegn. - Kilde-URL eller parti-ID-er Lenke som Lichess vil hente PGN-oppdateringer fra. Den må være offentlig tilgjengelig på internett. - Du kan ellers skrive inn opptil 64 ID-er for partier hos Lichess. De må være adskilt med mellomrom. Startdato i din egen tidssone Valgfritt, hvis du vet når arrangementet starter Krediter kilden diff --git a/translation/dest/broadcast/nl-NL.xml b/translation/dest/broadcast/nl-NL.xml index d5c98975e8353..8063d1ddd6a4c 100644 --- a/translation/dest/broadcast/nl-NL.xml +++ b/translation/dest/broadcast/nl-NL.xml @@ -22,9 +22,7 @@ Korte toernooibeschrijving Volledige beschrijving evenement Optionele lange beschrijving van de uitzending. %1$s is beschikbaar. Totale lengte moet minder zijn dan %2$s tekens. - Bron-URL of partij-ID\'s Link die Lichess gebruikt om PGN updates te krijgen. Deze moet openbaar toegankelijk zijn via internet. - U kunt ook maximaal 64 Lichess partij-ID\'s invoeren, gescheiden door spaties. Aanvangsdatum in je eigen tijdzone Optioneel, als je weet wanneer het evenement start Bronvermelding diff --git a/translation/dest/broadcast/nn-NO.xml b/translation/dest/broadcast/nn-NO.xml index e2250cf1ead49..f73a33756d29b 100644 --- a/translation/dest/broadcast/nn-NO.xml +++ b/translation/dest/broadcast/nn-NO.xml @@ -23,9 +23,9 @@ Kortfatta skildring av turneringa Full omtale av arrangementet Valfri lang omtale av overføringa. %1$s er tilgjengeleg. Omtalen må vera kortare enn %2$s teikn. - Kjelde-URL eller parti-ID-ar + PGN kjelde-URL Lenke som Lichess vil hente PGN-oppdateringar frå. Den må vera offentleg tilgjengeleg på internett. - Som eit alternativ kan du skriva opp til 64 Lichess-ID\'ar, skilde frå kvarandre med eit mellomrom. + Opp til 64 Lichess spel-ID\'ar, skilde med mellomrom. Startdato i di eiga tidssone Valfritt, om du veit når arrangementet startar Kreditér kjelda diff --git a/translation/dest/broadcast/pl-PL.xml b/translation/dest/broadcast/pl-PL.xml index 26c2b6f59d383..62130d166a1e2 100644 --- a/translation/dest/broadcast/pl-PL.xml +++ b/translation/dest/broadcast/pl-PL.xml @@ -25,9 +25,9 @@ Krótki opis turnieju Pełny opis wydarzenia Opcjonalny długi opis transmisji. %1$s jest dostępny. Długość musi być mniejsza niż %2$s znaków. - Źródłowy URL lub lista ID partii + Adres URL zapisu PGN Adres URL, który Lichess będzie udostępniał, aby można było uzyskać aktualizacje PGN. Musi być publicznie dostępny z internetu. - Alternatywnie, możesz wprowadzić do 64 ID\'ów gier Lichess, oddzielonych spacjami. + Do 64 identyfikatorów partii, oddzielonych spacjami. Data rozpoczęcia wydarzenia w Twojej strefie czasowej Opcjonalne, jeśli wiesz kiedy wydarzenie się rozpocznie Potwierdź źródło diff --git a/translation/dest/broadcast/pt-BR.xml b/translation/dest/broadcast/pt-BR.xml index 1b6e7587efeb9..827ac3fd43520 100644 --- a/translation/dest/broadcast/pt-BR.xml +++ b/translation/dest/broadcast/pt-BR.xml @@ -23,9 +23,7 @@ Descrição curta do torneio Descrição completa do evento Descrição longa e opcional da transmissão. %1$s está disponível. O tamanho deve ser menor que %2$s caracteres. - URL de origem ou IDs das partidas URL que Lichess irá verificar para obter atualizações PGN. Deve ser acessível ao público a partir da Internet. - Como alternativa, você pode inserir até 64 IDs de jogo do Lichess, separados por espaços. Data de início em seu próprio fuso horário Opcional, se você sabe quando o evento começa Crédito a fonte diff --git a/translation/dest/broadcast/pt-PT.xml b/translation/dest/broadcast/pt-PT.xml index a0bf27cb80744..098522ff13a8d 100644 --- a/translation/dest/broadcast/pt-PT.xml +++ b/translation/dest/broadcast/pt-PT.xml @@ -23,9 +23,9 @@ Breve descrição do torneio Descrição completa do evento Descrição longa do evento opcional da transmissão. %1$s está disponível. Tem de ter menos que %2$s carácteres. - URL de origem ou IDs do jogo + URL da fonte PGN Link que o Lichess vai verificar para obter atualizações da PGN. Deve ser acessível ao público a partir da internet. - Como alternativa, você pode inserir até 64 IDs do jogo Lichess, separados por espaços. + Até 64 IDs de jogo Lichess, separados por espaços. Data de início no teu fuso horário Opcional, se souberes quando começa o evento Credita a fonte diff --git a/translation/dest/broadcast/ro-RO.xml b/translation/dest/broadcast/ro-RO.xml index e5e9696766f27..b1df2049de7e2 100644 --- a/translation/dest/broadcast/ro-RO.xml +++ b/translation/dest/broadcast/ro-RO.xml @@ -23,9 +23,7 @@ O descriere scurtă a turneului Întreaga descriere a evenimentului Descriere lungă, opțională, a difuzării. %1$s este disponibil. Lungimea trebuie să fie mai mică decât %2$s caractere. - Link-ul sursă, sau ID-ul partidei URL-ul pe care Lichess îl va verifica pentru a obține actualizări al PGN-ului. Trebuie să fie public accesibil pe Internet. - Alternativ, poți să adaugi 64 ID-uri de jocuri pe Lichess, separat de spațiu. Data de începere conform fusului tău orar Opțional, dacă știi când va începe evenimentul Creditează sursa diff --git a/translation/dest/broadcast/ru-RU.xml b/translation/dest/broadcast/ru-RU.xml index 972e56f549e36..4146cf59a043b 100644 --- a/translation/dest/broadcast/ru-RU.xml +++ b/translation/dest/broadcast/ru-RU.xml @@ -25,9 +25,8 @@ Краткое описание турнира Полное описание события Необязательное полное описание трансляции. Доступна разметка %1$s. Длина должна быть меньше %2$s символов. - Исходный URL или идентификатор партии URL-адрес, с которого Lichess будет получать обновление PGN. Он должен быть доступен для получения из Интернета. - Кроме того, вы можете ввести до 64 идентификаторов партий, разделённых пробелами. + До 64 идентификаторов (ID) игр Lichess, разделённых пробелами. Дата начала в вашем часовом поясе Дополнительно, если вы знаете, когда событие начнётся Признательность diff --git a/translation/dest/broadcast/ry-UA.xml b/translation/dest/broadcast/ry-UA.xml index 9dfcbfc4d5d68..4625abe83fc1f 100644 --- a/translation/dest/broadcast/ry-UA.xml +++ b/translation/dest/broadcast/ry-UA.xml @@ -13,9 +13,7 @@ Куртый опис турніра Повный опис турніра Немусайно повный опис турніра. %1$s приступный. Довжина має быти менша за %2$s сімболув. - Зачаточноє URL, вадь ідентіфікатор бавкы Одкликованя, через якоє Lichess буде доставати обновеня PGN. Ун має быти приступный из Інтернета. - Попиля того, можете увести 64 ідентіфікаторув бавкы на Lichess, роздїлені пролишами. Дата старта у вашум часовум поясі Опціонално, кідь знаєте коли ся зачинат припад Оддяка diff --git a/translation/dest/broadcast/sk-SK.xml b/translation/dest/broadcast/sk-SK.xml index bdc97f08e9479..75d8978539a7a 100644 --- a/translation/dest/broadcast/sk-SK.xml +++ b/translation/dest/broadcast/sk-SK.xml @@ -21,9 +21,7 @@ Krátky popis turnaja Úplný popis turnaja Voliteľný dlhý popis vysielania. %1$s je dostupný. Dĺžka musí byť menej ako %2$s znakov. - Zdrojová URL alebo ID partií URL, ktorú bude Lichess kontrolovať, aby získal aktualizácie PGN. Musí byť verejne prístupná z internetu. - Prípadne môžete zadať až 64 ID Lichess partií oddelených medzerami. Dátum a čas začiatku, vo vašej časovej zóne Voliteľné, ak viete kedy sa udalosť začne Uveďte zdroj diff --git a/translation/dest/broadcast/sl-SI.xml b/translation/dest/broadcast/sl-SI.xml index c1810a50e6e20..719c85ee31daa 100644 --- a/translation/dest/broadcast/sl-SI.xml +++ b/translation/dest/broadcast/sl-SI.xml @@ -25,9 +25,7 @@ Kratek opis turnirja Polni opis dogodka Neobvezen dolg opis prenosa. %1$s je na voljo. Dolžina mora biti manjša od %2$s znakov. - Izvorni URL ali ID igre URL, ki ga bo Lichess preveril, da bo prejel PGN posodobitve. Javno mora biti dostopen preko interneta. - Lahko pa vnesete do 64 ID-jev lichess iger, ločene s presledki. Datum začetka v vaše časovnem pasu Izbirno, če veste, kdaj se dogodek začne Navedi vir diff --git a/translation/dest/broadcast/sq-AL.xml b/translation/dest/broadcast/sq-AL.xml index 9c9bd8d7845eb..ff873de32923a 100644 --- a/translation/dest/broadcast/sq-AL.xml +++ b/translation/dest/broadcast/sq-AL.xml @@ -23,9 +23,7 @@ Përshkrim i shkurtër i turneut Përshkrim i plotë i turneut Përshkrim i gjatë opsional i turneut. %1$s është e disponueshme. Gjatësia duhet të jetë më pak se %2$s shenja. - URL burimi, ose ID-ra loje URL-ja që do të kontrollojë Lichess-i për të marrë përditësime PGN-sh. Duhet të jetë e përdorshme publikisht që nga Interneti. - Ndryshe, mund të jepni deri në 63 ID-ra lojërash Lichess, ndarë me presje. Datë fillimi në zonën tuaj kohore Opsionale, nëse e dini kur fillon veprimtaria Atriboji merita burimit diff --git a/translation/dest/broadcast/sv-SE.xml b/translation/dest/broadcast/sv-SE.xml index 26a8fc5dcec28..b7507af95de49 100644 --- a/translation/dest/broadcast/sv-SE.xml +++ b/translation/dest/broadcast/sv-SE.xml @@ -22,9 +22,7 @@ Kort beskrivning av turneringen Fullständig beskrivning Valfri längre beskrivning av sändningen. %1$s är tillgänglig. Längden måste vara mindre än %2$s tecken. - Käll-URL eller spel-ID:n URL som Lichess kan använda för att få PGN-uppdateringar. Den måste vara publikt tillgänglig från Internet. - Alternativt kan du ange upp till 64 Lichess spel-ID:n, separerade med mellanslag. Startdatum i din egen tidszon Valfritt, om du vet när händelsen startar Kreditera källan diff --git a/translation/dest/broadcast/tr-TR.xml b/translation/dest/broadcast/tr-TR.xml index c9f823299015d..76e5a6567899d 100644 --- a/translation/dest/broadcast/tr-TR.xml +++ b/translation/dest/broadcast/tr-TR.xml @@ -23,9 +23,7 @@ Turnuvanın kısa tanımı Etkinliğin detaylıca açıklaması Etkinliğin isteğe bağlı detaylı açıklaması. %1$s seçeneği mevcuttur. Metnin uzunluğu azami %2$s karakter olmalıdır. - Kaynak URL ya da oyun ID\'si Lichess, sağladığınız URL yardımıyla PGN\'yi güncelleyecektir. İnternet üzerinden herkese açık bir URL yazmalısınız. - Ayrıca en fazla 64 Lichess oyununun ID\'sini, aralarında boşluk bırakarak, ekleyebilirsiniz. Kendi saat diliminizdeki başlangıç zamanı İsteğe bağlı, etkinliğin ne zaman başladığını biliyorsanız ekleyebilirsiniz. Kaynağı görüntüle diff --git a/translation/dest/broadcast/tt-RU.xml b/translation/dest/broadcast/tt-RU.xml index 1f4995a9a5d90..b4bea29039cc2 100644 --- a/translation/dest/broadcast/tt-RU.xml +++ b/translation/dest/broadcast/tt-RU.xml @@ -10,9 +10,7 @@ Ярыш исеме Вакыйганы тулы тасвирлау Ишетләштереш өстәмә озынлыгы тасвирламасы %1$s бар. Озынлык %2$s символлардан ким булырга тиеш. - Чыганак URL яки уен IDе PGN яңартуларын алу өчен Lichess тикшерәчәк URL. Ул Интернеттан ачык булырга тиеш. - Альтернатив рәвештә, сез буш урыннар белән аерылган 64 Lichess уен ID-ләрен кертә аласыз. Башлагыз сәнә сезнең үзегезнең вакыт җирендән Ихтимал, вакыйганың кайчан башланганын белсәгез Чыганакка кредит diff --git a/translation/dest/broadcast/uk-UA.xml b/translation/dest/broadcast/uk-UA.xml index e32161e348799..d8cbe8d1075de 100644 --- a/translation/dest/broadcast/uk-UA.xml +++ b/translation/dest/broadcast/uk-UA.xml @@ -25,9 +25,7 @@ Короткий опис турніру Повний опис події Необов\'язковий довгий опис трансляції. Наявна розмітка %1$s. Довжина має бути менша ніж %2$s символів. - Посилання на джерело або ідентифікатор гри Посилання, яке Lichess перевірятиме, щоб отримати оновлення PGN. Воно має бути загальнодоступним в Інтернеті. - Крім того, ви можете ввести до 64 ідентифікаторів гри Lichess, розділених пробілами. Дата початку у вашому часовому поясі За бажанням, якщо ви знаєте, коли починається подія Вдячність джерелу diff --git a/translation/dest/broadcast/vi-VN.xml b/translation/dest/broadcast/vi-VN.xml index 3bd49d5b48818..4c1b2fafbcbf0 100644 --- a/translation/dest/broadcast/vi-VN.xml +++ b/translation/dest/broadcast/vi-VN.xml @@ -22,9 +22,9 @@ Mô tả ngắn giải đấu Mô tả đầy đủ giải đấu Tùy chọn mô tả dài về giải đấu. Có thể sử dụng %1$s. Độ dài phải nhỏ hơn %2$s ký tự. - URL nguồn hoặc ID ván cờ + URL Nguồn PGN URL mà Lichess sẽ khảo sát để nhận cập nhật PGN. Nó phải được truy cập công khai từ Internet. - Ngoài ra, bạn có thể nhập 64 ID ván đấu trên Lichess, phân tách bởi dấu cách. + Tối đa 64 ID ván cờ trên Lichess, phân tách bằng dấu cách. Ngày bắt đầu theo múi giờ của bạn Tùy chọn, nếu bạn biết khi nào sự kiện bắt đầu Công nhận nguồn diff --git a/translation/dest/broadcast/zh-CN.xml b/translation/dest/broadcast/zh-CN.xml index b701099df1fb0..975fa3503897a 100644 --- a/translation/dest/broadcast/zh-CN.xml +++ b/translation/dest/broadcast/zh-CN.xml @@ -22,9 +22,7 @@ 锦标赛简短描述 赛事详情 转播内容的详细描述 (可选)。可以使用 %1$s,字数少于 %2$s 个。 - 资源的 URL 链接或对局 ID Lichess 将从该网址搜查 PGN 的更新。它必须是公开的。 - 你也可以输入最长 64 个字符的 Lichess 对局 ID,用空格隔开。 开始日期,在你的本地时区 如果你知道比赛开始时间 (可选) 信任来源 diff --git a/translation/dest/broadcast/zh-TW.xml b/translation/dest/broadcast/zh-TW.xml index 51e54912bbeb7..fbe5362588d1b 100644 --- a/translation/dest/broadcast/zh-TW.xml +++ b/translation/dest/broadcast/zh-TW.xml @@ -13,9 +13,7 @@ 簡短比賽說明 完整比賽說明 直播內容的詳細描述 。可以利用 %1$s。字數限於%2$s個字。 - 對局URL連接,或遊戲ID Lichess 將以該網址更新PGN數據,網址必須公開 - 此外,您可以輸入最長64个字符的 Lichess 遊戲 ID,用空格隔開。 開始日期 (當地時間) 可選,如果知道比賽開始時間 將來源歸因 diff --git a/translation/dest/coordinates/ar-SA.xml b/translation/dest/coordinates/ar-SA.xml index c057b3b1e09c8..339bfab6471cf 100644 --- a/translation/dest/coordinates/ar-SA.xml +++ b/translation/dest/coordinates/ar-SA.xml @@ -13,6 +13,7 @@ لديك 30 ثانية لرسم خريطة صحيحة لأكبر عدد ممكن من المربعات! اذهب طالما تريد، لا يوجد حد زمني! إظهار الإحداثيات + الإحداثيات على كل مربع إظهار القطع إبدأ التدريب ابحث عن مربع diff --git a/translation/dest/coordinates/lb-LU.xml b/translation/dest/coordinates/lb-LU.xml index 69a843ed4efcd..b3512b7a20718 100644 --- a/translation/dest/coordinates/lb-LU.xml +++ b/translation/dest/coordinates/lb-LU.xml @@ -13,6 +13,7 @@ Du hues 30 Sekonnen fir esou vill Felder wéi méiglech korrekt ze wielen! Spill sou lang wéi du wëlls, et gëtt keng Zäitlimite! Koordinaten uweisen + Koordinaten op jiddwer Feld Figuren uweisen Training ufänken Feld fannen diff --git a/translation/dest/faq/ar-SA.xml b/translation/dest/faq/ar-SA.xml index 14d1ab58eb429..013d037f970c7 100644 --- a/translation/dest/faq/ar-SA.xml +++ b/translation/dest/faq/ar-SA.xml @@ -163,4 +163,13 @@ 3. انقر على ملفات تعريف الارتباط و أذونات الموقع 4. مرر للأسفل وانقر على Media Autoplay 5. أضف lichess.org للقائمة المسموح بها + كيف أوقف نفسي عن اللعب؟ + حالة عقلية مستقلة + أنماط المستخدم ليتشيس + تصنيفات أقل + نتلقى بانتظام رسائل من المستخدمين يطلبون منا المساعدة لمنعهم من اللعب كثيرًا. + + على الرغم من أن Lichess لا تقوم بحظر اللاعبين أو حظرهم باستثناء انتهاكات شروط الخدمة، فإننا نوصي باستخدام أدوات خارجية للحد من سلوك اللعب المفرط. تتضمن بعض الاقتراحات الشائعة لأدوات حظر مواقع الويب %1$s و%2$s و%3$s. إذا كنت تريد الاستمرار في استخدام الموقع ولكن لا تنجذب إلى التحكم في الوقت السريع، فقد تكون مهتمًا أيضًا بـ %4$s، إليك واحدة مع %5$s. + + قد يشعر بعض اللاعبين أن سلوكهم في اللعب يتحول إلى إدمان. في الواقع، تصنف منظمة الصحة العالمية اضطراب الألعاب على أنه %6$s، مع ميزات أساسية تتمثل في 1) ضعف التحكم في الألعاب، 2) زيادة الأولوية المعطاة للألعاب، و3) تصاعد الألعاب على الرغم من العواقب السلبية. إذا كنت تعتقد أن سلوكك في لعب الشطرنج يتبع هذا النمط، فإننا نشجعك على التحدث عن ذلك مع صديق أو أحد أفراد العائلة و/أو مع أحد المحترفين. diff --git a/translation/dest/site/ar-SA.xml b/translation/dest/site/ar-SA.xml index f3949ace2acc9..d179135d643be 100644 --- a/translation/dest/site/ar-SA.xml +++ b/translation/dest/site/ar-SA.xml @@ -74,6 +74,8 @@ رفع سلسلة الحركات رفع الى التسلسل الرئيسي احذف من هنا + أعلى + وسع التفريع فرض التسلسل انسخ التفريع بصيغة PGN التقلة @@ -854,6 +856,7 @@ تبديل أسهم النقلات المرشحة الدورة السابقة/التفريع التالي تبديل الرموز التوضيحية + تبديل تحليل الموقف أسمهم النقلات تسمح لك بلعبها دون استخدام قائمة النقلات المرشحة. لعب النقلة المحددة مسابقة جديدة diff --git a/translation/dest/site/ca-ES.xml b/translation/dest/site/ca-ES.xml index 88b4f62d71dd2..3940f593b8bea 100644 --- a/translation/dest/site/ca-ES.xml +++ b/translation/dest/site/ca-ES.xml @@ -70,8 +70,8 @@ Promoure variant Convertir en línia principal Esborrar des d\'aquí - Amagar variacions - Expandir variacions + Amagar variacions + Expandir variacions Forçar variant Copia el PGN de la variació Moviment diff --git a/translation/dest/site/he-IL.xml b/translation/dest/site/he-IL.xml index 5db647b9d37b3..07c1667f44639 100644 --- a/translation/dest/site/he-IL.xml +++ b/translation/dest/site/he-IL.xml @@ -72,8 +72,8 @@ העדפת וריאנט קביעה כוריאנט הראשי מחיקה מכאן והלאה - הסתרת מהלכים חלופיים - הצגת מהלכים חלופיים + הסתרת מהלכים חלופיים + הצגת מהלכים חלופיים וריאנט יחיד העתקת ה-PGN של הוריאנט מסע @@ -460,7 +460,7 @@ או העלו קובץ PGN מעמדה מסוימת המשיכו מכאן - הפכו ללוח למידה + לוח למידה ייבוא משחק כשמדביקים משחק בפורמט PGN מקבלים אפשרות לצפות במשחק ולדפדף בו, ניתוח ממוחשב, צ׳אט וקישור לשיתוף. וריאציות — כלומר רצפי מהלכים שאינם המסעים הראשיים (mainline) — יימחקו. כדי לשמור אותן, ייבאו את ה־PGN כלוח למידה. diff --git a/translation/dest/site/lb-LU.xml b/translation/dest/site/lb-LU.xml index ef08ad14ff253..cc05128a13d49 100644 --- a/translation/dest/site/lb-LU.xml +++ b/translation/dest/site/lb-LU.xml @@ -70,6 +70,8 @@ Variant opwäerten Haaptvariant maachen Vun hei läschen + Varianten zesummeklappen + Varianten opklappen Variant forcéieren PGN-Variant kopéieren Zuch @@ -688,7 +690,7 @@ zeréck/no fir beweegen bei Start/Schluss goen Kommentarer weisen/verstoppen - Variant wiehlen/verloossen + Variant wielen/verloossen Computeranalys ufroen, léier aus denge Feeler Nächst (Aus dengen Feeler léieren) Nächst Gaffe @@ -845,7 +847,7 @@ Mëttelspill Endspill Bedingt Virauszich - Aktuell Variante bäifügen + Aktuell Variant bäifügen Variant spillen fir bedingt Virauszich ze kreéieren Keng bedingt Virauszich Spill %s diff --git a/translation/dest/site/tl-PH.xml b/translation/dest/site/tl-PH.xml index 147466a351696..bbe901f46bdcf 100644 --- a/translation/dest/site/tl-PH.xml +++ b/translation/dest/site/tl-PH.xml @@ -183,12 +183,16 @@ Palitan ang iyong pangalan Ang laki o liit lang ng mga letra ang pwedeng baguhin. Halimbawa, pwedeng gawing \"JohnDoe\" ang \"johndoe\". Palitan ang iyong pangalan. Maaari mo lang itong gawin ng isang beses at maaari mo lang baguhin ang laki o liit ng mga letra sa pangalan mo. + Tiyaking pumili ng pampamilyang username. Hindi mo ito mababago sa ibang pagkakataon at ang anumang mga account na may hindi naaangkop na mga username ay isasara! + Gagamitin lang namin ito para sa pag-reset ng password. Password Palitan ang password Palitan ang email Email Ibalik ang password Nakalimutan mo ba ang iyong password? + Mangyaring huwag gamitin ang iyong username bilang iyong password. + If everything else fails, then send us this email: Ranggo Ranggo: %s diff --git a/translation/dest/site/zh-CN.xml b/translation/dest/site/zh-CN.xml index e526f1204ed56..e899a5ac84c21 100644 --- a/translation/dest/site/zh-CN.xml +++ b/translation/dest/site/zh-CN.xml @@ -69,6 +69,8 @@ 提升变着 做为主线 从此处开始删除 + 折叠变着 + 展开变着 强制作为变着 复制变着的PGN 着法 diff --git a/translation/dest/study/lb-LU.xml b/translation/dest/study/lb-LU.xml index 7612345016498..0f5599ed3fd12 100644 --- a/translation/dest/study/lb-LU.xml +++ b/translation/dest/study/lb-LU.xml @@ -83,7 +83,7 @@ Ugepinnten Kapitelkommentar Kapitel späicheren Annotatiounen läschen - Variatiounen läschen + Variante läschen Kapitel läschen Kapitel läschen? Et gëtt keen zeréck! All Kommentarer, Symboler an Zeechnungsformen an dësem Kapitel läschen? diff --git a/translation/dest/swiss/ar-SA.xml b/translation/dest/swiss/ar-SA.xml index e606cfa288f51..fbca6610f443f 100644 --- a/translation/dest/swiss/ar-SA.xml +++ b/translation/dest/swiss/ar-SA.xml @@ -1,5 +1,6 @@ + النظام السويسري البطولات السويسرية اعرض الجولة%s @@ -66,6 +67,16 @@ الفاصل الزمني بين المباريات أزواج محظورة أسماء المستخدمين للاعبين الذين يجب ألا يلعبوا معا (على سبيل المثال). اسمان مستخدمان لكل سطر، مفصولان بمسافة. + مزاوجة يدوية في الجولة القادمة + حدد يدويًا جميع عمليات الاقتران للجولة التالية. زوج لاعب واحد لكل سطر. مثال: +اللاعب أ اللاعب ب +اللاعب ج اللاعب د +لإعطاء إقصاء (1 نقطة) للاعب بدلا من زوج، إضافة سطر مثل: +اللاعب 1 +اللاعبين المفقودين سيعدون غائبين ويحصلون على نقطة صفر. +اترك هذا الحقل فارغاً للسماح لليتشيس بإنشاء أزواج تلقائياً. + يجب أن يكونوا قد لعبوا آخر مواجهة لهم بالنظام السويسري + اسمح للاعبين بالانضمام فقط إذا كانوا قد لعبوا آخر مباراة لهم في SWISS. إذا أخفقوا في الظهور في حدث SWISS مؤخرًا، فلن يتمكنوا من الدخول إلى حدثك. يؤدي هذا إلى تجربة SWISS أفضل للاعبين الذين يحضرون فعلًا. بطولة جديدة بالنظام السويسري متى تستخدم البطولا السويسرية عوضاً عن البطولات العادية؟ في نظام البطولات السويسرية، يمكن للمشاركين اللعب ضد أحدهم الآخر مرة واحدة فقط، ويلعب الجميع نفس العدد من المباريات. @@ -141,4 +152,7 @@ السماح فقط للمستخدمين المحددين مسبقاً بالانضمام إذا كانت هذه القائمة غير فارغة، فسيمنع اللاعبون غير الواردة أسماءهم في هذه القائمة من الانضمام إلى البطولة. اسم مستخدم واحد لكل سطر. ألعب مبارياتك + أقاصٍ + الغيابات + كسر التعادل diff --git a/translation/dest/team/ar-SA.xml b/translation/dest/team/ar-SA.xml index be798a3abfa64..496b6571c92b2 100644 --- a/translation/dest/team/ar-SA.xml +++ b/translation/dest/team/ar-SA.xml @@ -76,4 +76,20 @@ عدد القادة لكل فريق. مجموع درجاتهم هو نتيجة الفريق. لا يجب عليك تغيير هذه القيمة بعد بدء البطولة! الفريق الداخلي + + معركة %s فريق + معركة فريق واحد + معركة فريقين + معركة %s أفرقة + معركة %s فريقا + معركة %s فريقا + + + %s قائدا لكل فريق + قائد واحد لكل فريق + قائدين للفريق + %s قادة للفريق الواحد + %s قائد للفريق الواحد + %s قائد للفريق الواحد + diff --git a/translation/dest/ublog/ar-SA.xml b/translation/dest/ublog/ar-SA.xml index f59934b716a79..57c3433a2ffad 100644 --- a/translation/dest/ublog/ar-SA.xml +++ b/translation/dest/ublog/ar-SA.xml @@ -64,4 +64,5 @@ اي شيء غير لائق من الممكن أن يتسبب باغلاق حسابك. نصائحنا البسيطة لكتابة مقالة رائعة ناقش منشور المدونة هذا في المنتدى + تم حظرك من قبل مؤلف المدونة. From 92d00e58a5f378a9c7bd56e2e172ca3e10c71376 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 11:20:45 +0200 Subject: [PATCH 163/168] monitor round viewers for 30m after the round stopped syncing --- modules/relay/src/main/RelayStatsApi.scala | 19 ++++++++++++++++--- 1 file changed, 16 insertions(+), 3 deletions(-) diff --git a/modules/relay/src/main/RelayStatsApi.scala b/modules/relay/src/main/RelayStatsApi.scala index e02458ab49fbf..aadfef5ee52f2 100644 --- a/modules/relay/src/main/RelayStatsApi.scala +++ b/modules/relay/src/main/RelayStatsApi.scala @@ -2,6 +2,7 @@ package lila.relay import lila.db.dsl.{ *, given } import reactivemongo.api.bson.BSONInteger +import scalalib.cache.ExpireSetMemo object RelayStats: type Minute = Int @@ -20,7 +21,7 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc def get(id: RelayTourId): Fu[List[RoundStats]] = colls.round - .aggregateList(128): framework => + .aggregateList(RelayTour.maxRelays): framework => import framework.* Match($doc("tourId" -> id)) -> List( Sort(Ascending("createdAt")), @@ -74,13 +75,24 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc _ <- elements.nonEmpty.so(update.many(elements).void) yield () + // keep monitoring rounds for 30m after they stopped syncing + private val syncTail = ExpireSetMemo[RelayRoundId](30 minutes) + private def fetchRoundCrowds: Fu[List[(RelayRoundId, Crowd)]] = val max = 500 colls.round .aggregateList(maxDocs = max, _.sec): framework => import framework.* - Match($doc("sync.until" -> $exists(true), "crowd".$gt(0))) -> - List(Project($doc("_id" -> 1, "crowd" -> 1))) + Match( + $doc( + $or( + $doc("sync.until" -> $exists(true)), + $inIds(syncTail.keys) + ), + "crowd".$gt(0) + ) + ) -> + List(Project($doc("_id" -> 1, "crowd" -> 1, "syncing" -> "$sync.until"))) .map: docs => if docs.size == max then logger.warn(s"RelayStats.fetchRoundCrowds: $max docs fetched") @@ -88,4 +100,5 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc doc <- docs id <- doc.getAsOpt[RelayRoundId]("_id") crowd <- doc.getAsOpt[Crowd]("crowd") + _ = if doc.contains("syncing") then syncTail.put(id) yield (id, crowd) From 0cea482a0a13de77aa4bdf05d809a2e8ed3f7b94 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 11:33:25 +0200 Subject: [PATCH 164/168] also monitor viewers on pushed broadcasts --- modules/relay/src/main/Env.scala | 5 ++--- modules/relay/src/main/RelayPush.scala | 5 +++-- modules/relay/src/main/RelayStatsApi.scala | 12 +++++++----- 3 files changed, 12 insertions(+), 10 deletions(-) diff --git a/modules/relay/src/main/Env.scala b/modules/relay/src/main/Env.scala index c5dbf759d6e33..d63edf02cae0b 100644 --- a/modules/relay/src/main/Env.scala +++ b/modules/relay/src/main/Env.scala @@ -62,6 +62,8 @@ final class Env( lazy val listing: RelayListing = wire[RelayListing] + lazy val stats = wire[RelayStatsApi] + lazy val api: RelayApi = wire[RelayApi] lazy val tourStream: RelayTourStream = wire[RelayTourStream] @@ -90,9 +92,6 @@ final class Env( private lazy val delay = wire[RelayDelay] - // must instanciate eagerly to start the scheduler - val stats = wire[RelayStatsApi] - import SettingStore.CredentialsOption.given val proxyCredentials = settingStore[Option[Credentials]]( "relayProxyCredentials", diff --git a/modules/relay/src/main/RelayPush.scala b/modules/relay/src/main/RelayPush.scala index ce7dd9bab627c..96b45d85c5b75 100644 --- a/modules/relay/src/main/RelayPush.scala +++ b/modules/relay/src/main/RelayPush.scala @@ -4,15 +4,15 @@ import akka.actor.* import akka.pattern.after import chess.format.pgn.{ Parser, PgnStr, San, Std, Tags } import chess.{ ErrorStr, Game, Replay, Square } - import scala.concurrent.duration.* - import scalalib.actor.AsyncActorSequencers + import lila.study.{ MultiPgn, StudyPgnImport } final class RelayPush( sync: RelaySync, api: RelayApi, + stats: RelayStatsApi, irc: lila.core.irc.IrcApi )(using ActorSystem, Executor, Scheduler): @@ -50,6 +50,7 @@ final class RelayPush( .flatMap: event => if !rt.round.hasStarted && !rt.tour.official && event.hasMoves then irc.broadcastStart(rt.round.id, rt.fullName) + stats.setActive(rt.round.id) api .update(rt.round): r1 => val r2 = r1.withSync(_.addLog(event)) diff --git a/modules/relay/src/main/RelayStatsApi.scala b/modules/relay/src/main/RelayStatsApi.scala index aadfef5ee52f2..42fdd86a7eaef 100644 --- a/modules/relay/src/main/RelayStatsApi.scala +++ b/modules/relay/src/main/RelayStatsApi.scala @@ -46,6 +46,11 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc .toList yield RoundStats(round, stats) + def setActive(id: RelayRoundId) = activeRounds.put(id) + + // keep monitoring rounds for 30m after they stopped syncing + private val activeRounds = ExpireSetMemo[RelayRoundId](30 minutes) + private def record(): Funit = for crowds <- fetchRoundCrowds nowMinutes = nowSeconds / 60 @@ -75,9 +80,6 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc _ <- elements.nonEmpty.so(update.many(elements).void) yield () - // keep monitoring rounds for 30m after they stopped syncing - private val syncTail = ExpireSetMemo[RelayRoundId](30 minutes) - private def fetchRoundCrowds: Fu[List[(RelayRoundId, Crowd)]] = val max = 500 colls.round @@ -87,7 +89,7 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc $doc( $or( $doc("sync.until" -> $exists(true)), - $inIds(syncTail.keys) + $inIds(activeRounds.keys) ), "crowd".$gt(0) ) @@ -100,5 +102,5 @@ final class RelayStatsApi(roundRepo: RelayRoundRepo, colls: RelayColls)(using sc doc <- docs id <- doc.getAsOpt[RelayRoundId]("_id") crowd <- doc.getAsOpt[Crowd]("crowd") - _ = if doc.contains("syncing") then syncTail.put(id) + _ = if doc.contains("syncing") then activeRounds.put(id) yield (id, crowd) From 41c3bee552b4a0be39607d81dc4f289e320366a3 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 13:25:33 +0200 Subject: [PATCH 165/168] improve logging --- modules/security/src/main/EmailChange.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/security/src/main/EmailChange.scala b/modules/security/src/main/EmailChange.scala index 910345216ad88..dd0385e9c6a28 100644 --- a/modules/security/src/main/EmailChange.scala +++ b/modules/security/src/main/EmailChange.scala @@ -55,7 +55,7 @@ ${trans.common_orPaste.txt()} _ <- userRepo.setEmail(userId, email).recoverDefault me <- userRepo.me(userId) yield - logger.info(s"Set email for $id: $email") + logger.info(s"Change email for $userId: $previous -> $email") me.map(_ -> previous) } From 53a163889419e4abbeb52790e269c672ad5a8299 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 13:25:48 +0200 Subject: [PATCH 166/168] improve puzzle visual feedback --- ui/puzzle/src/autoShape.ts | 12 +++++++++--- ui/puzzle/src/ctrl.ts | 3 ++- 2 files changed, 11 insertions(+), 4 deletions(-) diff --git a/ui/puzzle/src/autoShape.ts b/ui/puzzle/src/autoShape.ts index 66f9fa129fcc1..103ec70df66fb 100644 --- a/ui/puzzle/src/autoShape.ts +++ b/ui/puzzle/src/autoShape.ts @@ -7,6 +7,7 @@ import { NormalMove } from 'chessops/types'; interface Opts { node: Tree.Node; + nodeList: Tree.Node[]; showComputer(): boolean; ceval: CevalCtrl; ground: CgApi; @@ -71,8 +72,14 @@ export default function (opts: Opts): DrawShape[] { }); } else shapes = shapes.concat(makeAutoShapesFromUci(opposite(color), n.threat.pvs[0].moves[0], 'red')); } + const feedback = feedbackAnnotation(n, opts.nodeList[opts.nodeList.length - 2]); + return shapes.concat(annotationShapes(n)).concat(feedback ? annotationShapes(feedback) : []); +} + +function feedbackAnnotation(n: Tree.Node, prev?: Tree.Node): Tree.Node | undefined { let glyph: Tree.Glyph | undefined; - switch (n.puzzle) { + const node = n.puzzle ? n : prev?.puzzle ? prev : undefined; + switch (node?.puzzle) { case 'good': case 'win': glyph = { id: 7, name: 'good', symbol: '✓' }; @@ -80,6 +87,5 @@ export default function (opts: Opts): DrawShape[] { case 'fail': glyph = { id: 4, name: 'fail', symbol: '✗' }; } - const withPuzzleGlyphs = glyph ? { ...n, glyphs: [glyph] } : n; - return shapes.concat(annotationShapes(withPuzzleGlyphs)); + return node && glyph && { ...node, glyphs: [glyph] }; } diff --git a/ui/puzzle/src/ctrl.ts b/ui/puzzle/src/ctrl.ts index 01d7ecfef6d9b..1c27eef938e33 100755 --- a/ui/puzzle/src/ctrl.ts +++ b/ui/puzzle/src/ctrl.ts @@ -321,6 +321,7 @@ export default class PuzzleCtrl implements ParentCtrl { this.withGround(g => g.playPremove()); const progress = moveTest(this); + this.setAutoShapes(); if (progress === 'fail') site.sound.say('incorrect'); if (progress) this.applyProgress(progress); this.reorderChildren(path); @@ -349,7 +350,7 @@ export default class PuzzleCtrl implements ParentCtrl { revertUserMove = (): void => { if (site.blindMode) this.instantRevertUserMove(); - else setTimeout(this.instantRevertUserMove, 100); + else setTimeout(this.instantRevertUserMove, 300); }; applyProgress = (progress: undefined | 'fail' | 'win' | MoveTest): void => { From 03216e00044b3b69fa30b6152c6af241ff8dd316 Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 16:58:40 +0200 Subject: [PATCH 167/168] eagerly instanciate the challenge socket to make mobile local dev more predictible --- modules/challenge/src/main/Env.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/challenge/src/main/Env.scala b/modules/challenge/src/main/Env.scala index 491cfbf4cb33e..40275ecb1962a 100644 --- a/modules/challenge/src/main/Env.scala +++ b/modules/challenge/src/main/Env.scala @@ -50,7 +50,7 @@ final class Env( lazy val api = wire[ChallengeApi] - private lazy val socket = wire[ChallengeSocket] + private val socket = wire[ChallengeSocket] lazy val granter = wire[ChallengeGranter] From b58a6dab716a8fb2be8a7bdc7e15252574732fec Mon Sep 17 00:00:00 2001 From: Thibault Duplessis Date: Thu, 20 Jun 2024 17:14:20 +0200 Subject: [PATCH 168/168] serve different flavours of challenge to website, lichobile, and API/mobile --- app/controllers/Challenge.scala | 31 ++++++++++++++--------- modules/challenge/src/main/JsonView.scala | 19 +++++++++++--- 2 files changed, 35 insertions(+), 15 deletions(-) diff --git a/app/controllers/Challenge.scala b/app/controllers/Challenge.scala index 04adf129889d5..4c4e8fffbcee3 100644 --- a/app/controllers/Challenge.scala +++ b/app/controllers/Challenge.scala @@ -46,11 +46,11 @@ final class Challenge( else if isForMe(c) then Direction.In.some else none direction.so: dir => - val json = env.challenge.jsonView(dir.some)(c) for fullId <- c.accepted.so(env.round.proxyRepo.game(c.gameId).map2(c.fullIdOf(_, dir))) socketVersion <- ctx.isMobileOauth.so(env.challenge.version(c.id).dmap(some)) - yield JsonOk(json.add("fullId", fullId).add("socketVersion", socketVersion)) + json = env.challenge.jsonView.apiAndMobile(c, socketVersion, dir.some, fullId) + yield JsonOk(json) } protected[controllers] def showId(id: ChallengeId)(using Context): Fu[Result] = @@ -69,7 +69,7 @@ final class Challenge( if mine then Direction.Out.some else if isForMe(c) then Direction.In.some else none - val json = env.challenge.jsonView.show(c, version, direction) + val json = env.challenge.jsonView.websiteAndLichobile(c, version, direction) negotiate( html = val color = get("color").flatMap(Color.fromName) @@ -310,15 +310,22 @@ final class Challenge( JsonBadRequest: jsonError(lila.challenge.ChallengeDenied.translated(denied)) case _ => - env.challenge.api.create(challenge).map { + env.challenge.api.create(challenge).flatMap { if _ then - val json = env.challenge.jsonView - .show(challenge, SocketVersion(0), lila.challenge.Direction.Out.some) - if config.keepAliveStream then - jsOptToNdJson: - ndJson.addKeepAlive(env.challenge.keepAliveStream(challenge, json)) - else JsonOk(json) - else JsonBadRequest(jsonError("Challenge not created")) + ctx.isMobileOauth + .so(env.challenge.version(challenge.id).dmap(some)) + .map: socketVersion => + val json = env.challenge.jsonView + .apiAndMobile( + challenge, + socketVersion, + lila.challenge.Direction.Out.some + ) + if config.keepAliveStream then + jsOptToNdJson: + ndJson.addKeepAlive(env.challenge.keepAliveStream(challenge, json)) + else JsonOk(json) + else JsonBadRequest(jsonError("Challenge not created")).toFuccess } yield res } @@ -358,7 +365,7 @@ final class Challenge( .map: challenge => JsonOk: val url = s"${env.net.baseUrl}/${challenge.id}" - env.challenge.jsonView.show(challenge, SocketVersion(0), none) ++ Json.obj( + env.challenge.jsonView.apiAndMobile(challenge, none, none) ++ Json.obj( "urlWhite" -> s"$url?color=white", "urlBlack" -> s"$url?color=black" ) diff --git a/modules/challenge/src/main/JsonView.scala b/modules/challenge/src/main/JsonView.scala index 18725936bc22a..827a8dfdf6b28 100644 --- a/modules/challenge/src/main/JsonView.scala +++ b/modules/challenge/src/main/JsonView.scala @@ -10,6 +10,7 @@ import lila.game.JsonView.given import lila.core.i18n.I18nKey as trans import lila.core.socket.{ SocketVersion, userLag } import lila.core.i18n.{ Translate, JsDump } +import lila.core.id.GameFullId final class JsonView( baseUrl: lila.core.config.BaseUrl, @@ -45,14 +46,26 @@ final class JsonView( r.key -> JsString(r.trans.txt())) ) - def show(challenge: Challenge, socketVersion: SocketVersion, direction: Option[Direction])(using - Translate - ) = + def websiteAndLichobile( + challenge: Challenge, + socketVersion: SocketVersion, + direction: Option[Direction] + )(using Translate) = Json.obj( "challenge" -> apply(direction)(challenge), "socketVersion" -> socketVersion ) + def apiAndMobile( + challenge: Challenge, + socketVersion: Option[SocketVersion], + direction: Option[Direction], + fullId: Option[GameFullId] = none + )(using Translate) = + apply(direction)(challenge) + .add("socketVersion" -> socketVersion) + .add("fullId" -> fullId) + private given OWrites[Challenge.Open] = Json.writes def apply(direction: Option[Direction])(c: Challenge)(using Translate): JsObject =