Skip to content

Commit c6fad3b

Browse files
authored
Merge pull request #1098 from japgolly/custom-routes
Allow custom route parsers
2 parents a1aa1bb + 83ad0d1 commit c6fad3b

File tree

11 files changed

+194
-40
lines changed

11 files changed

+194
-40
lines changed

.envrc

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
use flake
2+
layout node
3+
eval "$shellHook"

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -18,3 +18,4 @@ metals.sbt
1818
node_modules/
1919
package-lock.json
2020
target
21+
.direnv/

downstream-tests/scalafix.sbt

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ ThisBuild / scalacOptions ++= {
1010

1111
ThisBuild / semanticdbEnabled := true
1212

13-
ThisBuild / semanticdbVersion := "4.5.9"
13+
ThisBuild / semanticdbVersion := "4.12.0"
1414

1515
ThisBuild / scalafixScalaBinaryVersion := "2.13"
1616

flake.lock

+109
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

flake.nix

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
{
2+
inputs = {
3+
typelevel-nix.url = "github:typelevel/typelevel-nix";
4+
nixpkgs.follows = "typelevel-nix/nixpkgs";
5+
flake-utils.follows = "typelevel-nix/flake-utils";
6+
};
7+
8+
outputs = { self, nixpkgs, flake-utils, typelevel-nix }:
9+
flake-utils.lib.eachDefaultSystem (system:
10+
let
11+
pkgs-x86_64 = import nixpkgs { system = "x86_64-darwin"; };
12+
scala-cli-overlay = final: prev: { scala-cli = pkgs-x86_64.scala-cli; };
13+
pkgs = import nixpkgs {
14+
inherit system;
15+
overlays = [ typelevel-nix.overlays.default scala-cli-overlay];
16+
};
17+
in
18+
{
19+
devShell = pkgs.devshell.mkShell {
20+
imports = [ typelevel-nix.typelevelShell ];
21+
packages = [
22+
pkgs.nodePackages.typescript-language-server
23+
pkgs.nodePackages.vscode-langservers-extracted
24+
pkgs.nodePackages.prettier
25+
pkgs.nodePackages.typescript
26+
pkgs.nodePackages.graphqurl
27+
pkgs.hasura-cli
28+
];
29+
typelevelShell = {
30+
nodejs.enable = true;
31+
jdk.package = pkgs.jdk11;
32+
};
33+
env = [
34+
{
35+
name = "NODE_OPTIONS";
36+
value = "--max-old-space-size=8192";
37+
}
38+
];
39+
};
40+
}
41+
42+
);
43+
}

library/callback/src/main/scala-3/japgolly/scalajs/react/callback/CallbackOption.scala

+2-2
Original file line numberDiff line numberDiff line change
@@ -183,7 +183,7 @@ final class CallbackOption[+A](val underlyingRepr: CallbackOption.UnderlyingRepr
183183
def asCallback: CallbackTo[Option[A]] =
184184
CallbackTo lift cbfn
185185

186-
inline def map[B](f: A => B)(using inline ev: MapGuard[B]): CallbackOption[ev.Out] =
186+
inline def map[B](f: A => B)(using ev: MapGuard[B]): CallbackOption[ev.Out] =
187187
unsafeMap(f)
188188

189189
private[react] def unsafeMap[B](f: A => B): CallbackOption[B] =
@@ -192,7 +192,7 @@ final class CallbackOption[+A](val underlyingRepr: CallbackOption.UnderlyingRepr
192192
/**
193193
* Alias for `map`.
194194
*/
195-
inline def |>[B](f: A => B)(using inline ev: MapGuard[B]): CallbackOption[ev.Out] =
195+
inline def |>[B](f: A => B)(using ev: MapGuard[B]): CallbackOption[ev.Out] =
196196
map(f)
197197

198198
def flatMapOption[B](f: A => Option[B]): CallbackOption[B] =

library/callback/src/main/scala-3/japgolly/scalajs/react/callback/CallbackTo.scala

+3-3
Original file line numberDiff line numberDiff line change
@@ -306,11 +306,11 @@ final class CallbackTo[+A] /*private[react]*/ (private[CallbackTo] val trampolin
306306
inline def runNow(): A =
307307
trampoline.run
308308

309-
inline def map[B](f: A => B)(using inline ev: MapGuard[B]): CallbackTo[ev.Out] =
309+
inline def map[B](f: A => B)(using ev: MapGuard[B]): CallbackTo[ev.Out] =
310310
new CallbackTo(trampoline.map(f))
311311

312312
/** Alias for `map`. */
313-
inline def |>[B](inline f: A => B)(using inline ev: MapGuard[B]): CallbackTo[ev.Out] =
313+
inline def |>[B](inline f: A => B)(using ev: MapGuard[B]): CallbackTo[ev.Out] =
314314
map(f)
315315

316316
inline def flatMap[B](f: A => CallbackTo[B]): CallbackTo[B] =
@@ -589,7 +589,7 @@ final class CallbackTo[+A] /*private[react]*/ (private[CallbackTo] val trampolin
589589

590590

591591
/** Convenience-method to run additional code after this callback. */
592-
inline def thenRun[B](inline runNext: B)(using inline ev: MapGuard[B]): CallbackTo[ev.Out] =
592+
inline def thenRun[B](inline runNext: B)(using ev: MapGuard[B]): CallbackTo[ev.Out] =
593593
this >> CallbackTo(runNext)
594594

595595
/** Convenience-method to run additional code before this callback. */

library/extra/src/main/scala/japgolly/scalajs/react/extra/router/Dsl.scala

+19-21
Original file line numberDiff line numberDiff line change
@@ -87,7 +87,7 @@ object StaticDsl {
8787
val / = literal("/")
8888
}
8989

90-
abstract class RouteCommon[R[X] <: RouteCommon[R, X], A] {
90+
trait RouteCommon[R[X] <: RouteCommon[R, X], A] {
9191

9292
def parseThen(f: Option[A] => Option[A]): R[A]
9393

@@ -159,7 +159,7 @@ object StaticDsl {
159159
// val g = p.matcher("").groupCount
160160
// if (g != matchGroups)
161161
// sys.error(s"Error in regex: /${p.pattern}/. Expected $matchGroups match groups but detected $g.")
162-
new Route(p, m => parse(i => m.group(i + 1)), a => Path(build(a)))
162+
Route.forPattern(p, m => parse(i => m.group(i + 1)), a => Path(build(a)))
163163
}
164164
}
165165

@@ -192,30 +192,28 @@ object StaticDsl {
192192
}
193193

194194
/**
195-
* A complete route.
195+
* A `Route` translates a `Path` into an instance of model `A` and vice versa.
196196
*/
197-
final class Route[A](pattern: Pattern,
198-
parseFn: Matcher => Option[A],
199-
buildFn: A => Path) extends RouteCommon[Route, A] with RouterMacros.ForRoute[A] {
200-
override def toString =
201-
s"Route($pattern)"
202-
197+
case class Route[A](parse: Path => Option[A], pathFor: A => Path) extends RouteCommon[Route, A] {
203198
override def parseThen(f: Option[A] => Option[A]): Route[A] =
204-
new Route(pattern, f compose parseFn, buildFn)
199+
Route(f compose parse, pathFor)
205200

206201
override def pmap[B](b: A => Option[B])(a: B => A): Route[B] =
207-
new Route(pattern, parseFn(_) flatMap b, buildFn compose a)
208-
209-
def parse(path: Path): Option[A] = {
210-
val m = pattern.matcher(path.value)
211-
if (m.matches)
212-
parseFn(m)
213-
else
214-
None
215-
}
202+
Route(parse(_) flatMap b, pathFor compose a)
203+
}
216204

217-
def pathFor(a: A): Path =
218-
buildFn(a)
205+
object Route {
206+
def forPattern[A](pattern: Pattern, parseFn: Matcher => Option[A], buildFn: A => Path): Route[A] =
207+
new Route(
208+
p => {
209+
val m = pattern.matcher(p.value)
210+
if (m.matches)
211+
parseFn(m)
212+
else
213+
None
214+
},
215+
buildFn
216+
)
219217
}
220218

221219
// ===================================================================================================================

library/project/Dependencies.scala

+10-10
Original file line numberDiff line numberDiff line change
@@ -9,23 +9,23 @@ object Dependencies {
99
object Ver {
1010

1111
// Externally observable
12-
val cats = "2.7.0"
13-
val catsEffect = "3.3.11"
14-
val microlibs = "4.2.1"
12+
val cats = "2.12.0"
13+
val catsEffect = "3.5.4"
14+
val microlibs = "4.1.0"
1515
val monocle2 = "2.1.0"
16-
val monocle3 = "3.1.0"
17-
val scala2 = "2.13.8"
18-
val scala3 = "3.1.2"
19-
val scalaJsDom = "2.0.0"
20-
val sourcecode = "0.2.8"
16+
val monocle3 = "3.2.0"
17+
val scala2 = "2.13.15"
18+
val scala3 = "3.3.0"
19+
val scalaJsDom = "2.8.0"
20+
val sourcecode = "0.4.2"
2121

2222
// Internal
2323
val betterMonadicFor = "0.3.1"
2424
val catsTestkitScalaTest = "2.1.5"
2525
val disciplineScalaTest = "2.1.5"
2626
val fastTextEncoding = "1.0.6"
27-
val kindProjector = "0.13.2"
28-
val macrotaskExecutor = "1.0.0"
27+
val kindProjector = "0.13.3"
28+
val macrotaskExecutor = "1.1.1"
2929
val nyaya = "1.0.0"
3030
val reactJs = "18.2.0"
3131
val scalaJsJavaTime = "1.0.0"

library/project/plugins.sbt

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ libraryDependencies ++= Seq(
22
"org.scala-js" %% "scalajs-env-jsdom-nodejs" % "1.1.0",
33
"org.scala-js" %% "scalajs-env-selenium" % "1.1.1")
44

5-
addSbtPlugin("ch.epfl.scala" % "sbt-scalafix" % "0.10.1")
6-
addSbtPlugin("com.github.sbt" % "sbt-ci-release" % "1.5.10")
5+
addSbtPlugin("ch.epfl.scala" % "sbt-scalafix" % "0.12.1")
6+
addSbtPlugin("com.github.sbt" % "sbt-ci-release" % "1.5.12")
77
addSbtPlugin("org.scala-js" % "sbt-jsdependencies" % "1.0.2")
88
addSbtPlugin("org.scala-js" % "sbt-scalajs" % "1.16.0")

library/scalafix.sbt

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,4 +8,4 @@ ThisBuild / scalacOptions ++= {
88
ThisBuild / semanticdbEnabled := true
99

1010
// NOTE: Upgrade downstream-tests/scalafix.sbt too!
11-
ThisBuild / semanticdbVersion := "4.5.9"
11+
ThisBuild / semanticdbVersion := "4.12.0"

0 commit comments

Comments
 (0)