diff --git a/example/shared/src/commonMain/composeResources/files/test.json b/example/shared/src/commonMain/composeResources/files/test.json index e69de29b..ff29422a 100644 --- a/example/shared/src/commonMain/composeResources/files/test.json +++ b/example/shared/src/commonMain/composeResources/files/test.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.5.6","a":"","k":"","d":"","tc":""},"fr":29.9700012207031,"ip":0,"op":900.000036657751,"w":2000,"h":2000,"nm":"Voice circle","ddd":0,"assets":[{"id":"image_0","w":985,"h":987,"u":"","p":"","e":1},{"id":"image_1","w":1613,"h":1776,"u":"","p":"","e":1},{"id":"image_2","w":1557,"h":1734,"u":"","p":"","e":1},{"id":"image_3","w":1501,"h":1638,"u":"","p":"","e":1},{"id":"image_4","w":1609,"h":1552,"u":"","p":"","e":1},{"id":"image_5","w":1854,"h":1539,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"Circle 1","refId":"image_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":1,"s":[0]},{"t":899.000036617021,"s":[360]}],"ix":10},"p":{"a":0,"k":[988.709,1008.372,0],"ix":2},"a":{"a":0,"k":[492.251,493.332,0],"ix":1},"s":{"a":0,"k":[86,86,100],"ix":6}},"ao":0,"ef":[{"ty":5,"nm":"Turbulent Displace","np":16,"mn":"ADBE Turbulent Displace","ix":1,"en":1,"ef":[{"ty":7,"nm":"Displacement","mn":"ADBE Turbulent Displace-0001","ix":1,"v":{"a":0,"k":3,"ix":1}},{"ty":0,"nm":"Amount","mn":"ADBE Turbulent Displace-0002","ix":2,"v":{"a":0,"k":-14,"ix":2}},{"ty":0,"nm":"Size","mn":"ADBE Turbulent Displace-0003","ix":3,"v":{"a":0,"k":258,"ix":3}},{"ty":3,"nm":"Offset (Turbulence)","mn":"ADBE Turbulent Displace-0004","ix":4,"v":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[0,493.5],"to":[0,0],"ti":[0,0]},{"t":899.000036617021,"s":[5209,493.5]}],"ix":4,"x":"var $bm_rt;\n$bm_rt = loopOut('cycle');"}},{"ty":0,"nm":"Complexity","mn":"ADBE Turbulent Displace-0005","ix":5,"v":{"a":0,"k":4.18,"ix":5}},{"ty":0,"nm":"Evolution","mn":"ADBE Turbulent Displace-0006","ix":6,"v":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[0]},{"t":899.000036617021,"s":[7200]}],"ix":6}},{"ty":6,"nm":"Evolution Options","mn":"ADBE Turbulent Displace-0007","ix":7,"v":0},{"ty":7,"nm":"Cycle Evolution","mn":"ADBE Turbulent Displace-0008","ix":8,"v":{"a":0,"k":0,"ix":8}},{"ty":0,"nm":"Cycle (in Revolutions)","mn":"ADBE Turbulent Displace-0009","ix":9,"v":{"a":0,"k":1,"ix":9}},{"ty":0,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0010","ix":10,"v":{"a":0,"k":0,"ix":10}},{"ty":6,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0011","ix":11,"v":0},{"ty":7,"nm":"Pinning","mn":"ADBE Turbulent Displace-0012","ix":12,"v":{"a":0,"k":3,"ix":12}},{"ty":7,"nm":"Resize Layer","mn":"ADBE Turbulent Displace-0013","ix":13,"v":{"a":0,"k":0,"ix":13}},{"ty":7,"nm":"Antialiasing for Best Quality","mn":"ADBE Turbulent Displace-0014","ix":14,"v":{"a":0,"k":1,"ix":14}}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"Layer 5","refId":"image_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":1,"s":[0]},{"t":899.000036617021,"s":[360]}],"ix":10},"p":{"a":0,"k":[992.145,1013.856,0],"ix":2},"a":{"a":0,"k":[806.265,887.706,0],"ix":1},"s":{"a":0,"k":[86,86,100],"ix":6}},"ao":0,"ef":[{"ty":5,"nm":"Turbulent Displace","np":16,"mn":"ADBE Turbulent Displace","ix":1,"en":1,"ef":[{"ty":7,"nm":"Displacement","mn":"ADBE Turbulent Displace-0001","ix":1,"v":{"a":0,"k":3,"ix":1}},{"ty":0,"nm":"Amount","mn":"ADBE Turbulent Displace-0002","ix":2,"v":{"a":0,"k":86,"ix":2}},{"ty":0,"nm":"Size","mn":"ADBE Turbulent Displace-0003","ix":3,"v":{"a":0,"k":254,"ix":3}},{"ty":3,"nm":"Offset (Turbulence)","mn":"ADBE Turbulent Displace-0004","ix":4,"v":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":-1,"s":[0,493.5],"to":[0,0],"ti":[0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":899,"s":[5000,493.5],"to":[0,0],"ti":[0,0]},{"t":961.000039142332,"s":[4802,493.5]}],"ix":4}},{"ty":0,"nm":"Complexity","mn":"ADBE Turbulent Displace-0005","ix":5,"v":{"a":0,"k":2.15,"ix":5}},{"ty":0,"nm":"Evolution","mn":"ADBE Turbulent Displace-0006","ix":6,"v":{"a":0,"k":0,"ix":6}},{"ty":6,"nm":"Evolution Options","mn":"ADBE Turbulent Displace-0007","ix":7,"v":0},{"ty":7,"nm":"Cycle Evolution","mn":"ADBE Turbulent Displace-0008","ix":8,"v":{"a":0,"k":0,"ix":8}},{"ty":0,"nm":"Cycle (in Revolutions)","mn":"ADBE Turbulent Displace-0009","ix":9,"v":{"a":0,"k":1,"ix":9}},{"ty":0,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0010","ix":10,"v":{"a":0,"k":0,"ix":10}},{"ty":6,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0011","ix":11,"v":0},{"ty":7,"nm":"Pinning","mn":"ADBE Turbulent Displace-0012","ix":12,"v":{"a":0,"k":3,"ix":12}},{"ty":7,"nm":"Resize Layer","mn":"ADBE Turbulent Displace-0013","ix":13,"v":{"a":0,"k":0,"ix":13}},{"ty":7,"nm":"Antialiasing for Best Quality","mn":"ADBE Turbulent Displace-0014","ix":14,"v":{"a":0,"k":1,"ix":14}}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"Layer 6","refId":"image_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":1,"s":[0]},{"t":899.000036617021,"s":[360]}],"ix":10},"p":{"a":0,"k":[1018.494,993.418,0],"ix":2},"a":{"a":0,"k":[778.026,866.548,0],"ix":1},"s":{"a":0,"k":[86,86,100],"ix":6}},"ao":0,"ef":[{"ty":5,"nm":"Turbulent Displace","np":16,"mn":"ADBE Turbulent Displace","ix":1,"en":1,"ef":[{"ty":7,"nm":"Displacement","mn":"ADBE Turbulent Displace-0001","ix":1,"v":{"a":0,"k":3,"ix":1}},{"ty":0,"nm":"Amount","mn":"ADBE Turbulent Displace-0002","ix":2,"v":{"a":0,"k":80,"ix":2}},{"ty":0,"nm":"Size","mn":"ADBE Turbulent Displace-0003","ix":3,"v":{"a":0,"k":254,"ix":3}},{"ty":3,"nm":"Offset (Turbulence)","mn":"ADBE Turbulent Displace-0004","ix":4,"v":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":-1,"s":[0,493.5],"to":[0,0],"ti":[0,0]},{"t":961.000039142332,"s":[4802,493.5]}],"ix":4}},{"ty":0,"nm":"Complexity","mn":"ADBE Turbulent Displace-0005","ix":5,"v":{"a":0,"k":1,"ix":5}},{"ty":0,"nm":"Evolution","mn":"ADBE Turbulent Displace-0006","ix":6,"v":{"a":0,"k":0,"ix":6}},{"ty":6,"nm":"Evolution Options","mn":"ADBE Turbulent Displace-0007","ix":7,"v":0},{"ty":7,"nm":"Cycle Evolution","mn":"ADBE Turbulent Displace-0008","ix":8,"v":{"a":0,"k":0,"ix":8}},{"ty":0,"nm":"Cycle (in Revolutions)","mn":"ADBE Turbulent Displace-0009","ix":9,"v":{"a":0,"k":1,"ix":9}},{"ty":0,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0010","ix":10,"v":{"a":0,"k":0,"ix":10}},{"ty":6,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0011","ix":11,"v":0},{"ty":7,"nm":"Pinning","mn":"ADBE Turbulent Displace-0012","ix":12,"v":{"a":0,"k":3,"ix":12}},{"ty":7,"nm":"Resize Layer","mn":"ADBE Turbulent Displace-0013","ix":13,"v":{"a":0,"k":0,"ix":13}},{"ty":7,"nm":"Antialiasing for Best Quality","mn":"ADBE Turbulent Displace-0014","ix":14,"v":{"a":0,"k":1,"ix":14}}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":2,"nm":"Layer 7","refId":"image_3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":1,"s":[0]},{"t":899.000036617021,"s":[360]}],"ix":10},"p":{"a":0,"k":[988.304,908.883,0],"ix":2},"a":{"a":0,"k":[750.336,818.773,0],"ix":1},"s":{"a":0,"k":[86,86,100],"ix":6}},"ao":0,"ef":[{"ty":5,"nm":"Turbulent Displace","np":16,"mn":"ADBE Turbulent Displace","ix":1,"en":1,"ef":[{"ty":7,"nm":"Displacement","mn":"ADBE Turbulent Displace-0001","ix":1,"v":{"a":0,"k":3,"ix":1}},{"ty":0,"nm":"Amount","mn":"ADBE Turbulent Displace-0002","ix":2,"v":{"a":0,"k":44,"ix":2}},{"ty":0,"nm":"Size","mn":"ADBE Turbulent Displace-0003","ix":3,"v":{"a":0,"k":254,"ix":3}},{"ty":3,"nm":"Offset (Turbulence)","mn":"ADBE Turbulent Displace-0004","ix":4,"v":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":-1,"s":[0,493.5],"to":[0,0],"ti":[0,0]},{"t":961.000039142332,"s":[4802,493.5]}],"ix":4}},{"ty":0,"nm":"Complexity","mn":"ADBE Turbulent Displace-0005","ix":5,"v":{"a":0,"k":1,"ix":5}},{"ty":0,"nm":"Evolution","mn":"ADBE Turbulent Displace-0006","ix":6,"v":{"a":0,"k":7200,"ix":6}},{"ty":6,"nm":"Evolution Options","mn":"ADBE Turbulent Displace-0007","ix":7,"v":0},{"ty":7,"nm":"Cycle Evolution","mn":"ADBE Turbulent Displace-0008","ix":8,"v":{"a":0,"k":0,"ix":8}},{"ty":0,"nm":"Cycle (in Revolutions)","mn":"ADBE Turbulent Displace-0009","ix":9,"v":{"a":0,"k":1,"ix":9}},{"ty":0,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0010","ix":10,"v":{"a":0,"k":0,"ix":10}},{"ty":6,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0011","ix":11,"v":0},{"ty":7,"nm":"Pinning","mn":"ADBE Turbulent Displace-0012","ix":12,"v":{"a":0,"k":3,"ix":12}},{"ty":7,"nm":"Resize Layer","mn":"ADBE Turbulent Displace-0013","ix":13,"v":{"a":0,"k":0,"ix":13}},{"ty":7,"nm":"Antialiasing for Best Quality","mn":"ADBE Turbulent Displace-0014","ix":14,"v":{"a":0,"k":1,"ix":14}}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":2,"nm":"Layer 8","refId":"image_4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":1,"s":[0]},{"t":899.000036617021,"s":[360]}],"ix":10},"p":{"a":0,"k":[1101.51,938.656,0],"ix":2},"a":{"a":0,"k":[804.19,775.886,0],"ix":1},"s":{"a":0,"k":[86,86,100],"ix":6}},"ao":0,"ef":[{"ty":5,"nm":"Turbulent Displace","np":16,"mn":"ADBE Turbulent Displace","ix":1,"en":1,"ef":[{"ty":7,"nm":"Displacement","mn":"ADBE Turbulent Displace-0001","ix":1,"v":{"a":0,"k":6,"ix":1}},{"ty":0,"nm":"Amount","mn":"ADBE Turbulent Displace-0002","ix":2,"v":{"a":0,"k":93,"ix":2}},{"ty":0,"nm":"Size","mn":"ADBE Turbulent Displace-0003","ix":3,"v":{"a":0,"k":254,"ix":3}},{"ty":3,"nm":"Offset (Turbulence)","mn":"ADBE Turbulent Displace-0004","ix":4,"v":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":-1,"s":[0,493.5],"to":[0,0],"ti":[0,0]},{"t":961.000039142332,"s":[4802,493.5]}],"ix":4}},{"ty":0,"nm":"Complexity","mn":"ADBE Turbulent Displace-0005","ix":5,"v":{"a":0,"k":1,"ix":5}},{"ty":0,"nm":"Evolution","mn":"ADBE Turbulent Displace-0006","ix":6,"v":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[0]},{"t":899.000036617021,"s":[7200]}],"ix":6}},{"ty":6,"nm":"Evolution Options","mn":"ADBE Turbulent Displace-0007","ix":7,"v":0},{"ty":7,"nm":"Cycle Evolution","mn":"ADBE Turbulent Displace-0008","ix":8,"v":{"a":0,"k":0,"ix":8}},{"ty":0,"nm":"Cycle (in Revolutions)","mn":"ADBE Turbulent Displace-0009","ix":9,"v":{"a":0,"k":1,"ix":9}},{"ty":0,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0010","ix":10,"v":{"a":0,"k":0,"ix":10}},{"ty":6,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0011","ix":11,"v":0},{"ty":7,"nm":"Pinning","mn":"ADBE Turbulent Displace-0012","ix":12,"v":{"a":0,"k":3,"ix":12}},{"ty":7,"nm":"Resize Layer","mn":"ADBE Turbulent Displace-0013","ix":13,"v":{"a":0,"k":0,"ix":13}},{"ty":7,"nm":"Antialiasing for Best Quality","mn":"ADBE Turbulent Displace-0014","ix":14,"v":{"a":0,"k":1,"ix":14}}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":2,"nm":"Layer 9","refId":"image_5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":1,"s":[0]},{"t":899.000036617021,"s":[360]}],"ix":10},"p":{"a":0,"k":[928.107,1005.845,0],"ix":2},"a":{"a":0,"k":[926.633,769.215,0],"ix":1},"s":{"a":0,"k":[86,86,100],"ix":6}},"ao":0,"ef":[{"ty":5,"nm":"Turbulent Displace","np":16,"mn":"ADBE Turbulent Displace","ix":1,"en":1,"ef":[{"ty":7,"nm":"Displacement","mn":"ADBE Turbulent Displace-0001","ix":1,"v":{"a":0,"k":5,"ix":1}},{"ty":0,"nm":"Amount","mn":"ADBE Turbulent Displace-0002","ix":2,"v":{"a":0,"k":23.5,"ix":2}},{"ty":0,"nm":"Size","mn":"ADBE Turbulent Displace-0003","ix":3,"v":{"a":0,"k":72,"ix":3}},{"ty":3,"nm":"Offset (Turbulence)","mn":"ADBE Turbulent Displace-0004","ix":4,"v":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":-1,"s":[0,493.5],"to":[0,0],"ti":[0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":899,"s":[6231.514,493.5],"to":[0,0],"ti":[0,0]},{"t":961.000039142332,"s":[4802,493.5]}],"ix":4}},{"ty":0,"nm":"Complexity","mn":"ADBE Turbulent Displace-0005","ix":5,"v":{"a":0,"k":1.569,"ix":5}},{"ty":0,"nm":"Evolution","mn":"ADBE Turbulent Displace-0006","ix":6,"v":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[229]},{"t":899.000036617021,"s":[7429]}],"ix":6}},{"ty":6,"nm":"Evolution Options","mn":"ADBE Turbulent Displace-0007","ix":7,"v":0},{"ty":7,"nm":"Cycle Evolution","mn":"ADBE Turbulent Displace-0008","ix":8,"v":{"a":0,"k":0,"ix":8}},{"ty":0,"nm":"Cycle (in Revolutions)","mn":"ADBE Turbulent Displace-0009","ix":9,"v":{"a":0,"k":1,"ix":9}},{"ty":0,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0010","ix":10,"v":{"a":0,"k":0,"ix":10}},{"ty":6,"nm":"Random Seed","mn":"ADBE Turbulent Displace-0011","ix":11,"v":0},{"ty":7,"nm":"Pinning","mn":"ADBE Turbulent Displace-0012","ix":12,"v":{"a":0,"k":3,"ix":12}},{"ty":7,"nm":"Resize Layer","mn":"ADBE Turbulent Displace-0013","ix":13,"v":{"a":0,"k":0,"ix":13}},{"ty":7,"nm":"Antialiasing for Best Quality","mn":"ADBE Turbulent Displace-0014","ix":14,"v":{"a":0,"k":1,"ix":14}}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":1,"nm":"Pale Gray-Blue Solid 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":360,"ix":10},"p":{"a":0,"k":[1000,1000,0],"ix":2},"a":{"a":0,"k":[1000,1000,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"sw":2000,"sh":2000,"sc":"#383847","ip":0,"op":900.000036657751,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/example/shared/src/commonMain/kotlin/App.kt b/example/shared/src/commonMain/kotlin/App.kt index 538562cf..31caf15a 100644 --- a/example/shared/src/commonMain/kotlin/App.kt +++ b/example/shared/src/commonMain/kotlin/App.kt @@ -132,7 +132,7 @@ public suspend fun LottieCompositionSpec.Companion.ResourceString( public fun App() { // return InteractiveControlsScreen() - return LottieFilesScreen() +// return LottieFilesScreen() // return LottieFontExample() diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Expression.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Expression.kt index 141ae3fa..85ae3156 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Expression.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Expression.kt @@ -3,20 +3,11 @@ package io.github.alexzhirkevich.skriptie import io.github.alexzhirkevich.skriptie.common.OpGetVariable import io.github.alexzhirkevich.skriptie.common.OpIndex -public interface Expression { +public fun interface Expression { public operator fun invoke(context: C): Any? } -internal fun Expression( - block : (C) -> Any? -) : Expression = object : Expression { - - override fun invoke(context: C): Any? { - return block(context) - } -} - internal fun Expression<*>.isAssignable() : Boolean { return this is OpGetVariable && assignmentType == null || this is OpIndex && variable is OpGetVariable diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ExpressionContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ExpressionContext.kt new file mode 100644 index 00000000..ef77edb8 --- /dev/null +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ExpressionContext.kt @@ -0,0 +1,55 @@ +package io.github.alexzhirkevich.skriptie + +import io.github.alexzhirkevich.skriptie.javascript.JSInterpretationContext +import kotlin.math.min + +public class ExpressionContext : JSInterpretationContext() { + + override fun sum(a: Any?, b: Any?): Any? { + return when { + a is List<*> && b is List<*> -> { + a as List + b as List + + List(min(a.size, b.size)) { + a[it].toDouble() + b[it].toDouble() + } + } + + a is List<*> && b is Number -> { + if (a is MutableList<*>) { + a as MutableList + a[0] = a[0].toDouble() + b.toDouble() + a + } else { + listOf((a as List).first().toDouble() + b.toDouble()) + a.drop(1) + } + } + + a is Number && b is List<*> -> { + if (b is MutableList<*>) { + b as MutableList + b[0] = b[0].toDouble() + a.toDouble() + b + } else { + listOf(a.toDouble() + (b as List).first().toDouble()) + b.drop(1) + } + } + + else -> super.sum(a, b) + } + } + + override fun sub(a: Any?, b: Any?): Any? { + return when { + a is List<*> && b is List<*> -> { + a as List + b as List + List(min(a.size, b.size)) { + a[it].toDouble() - b[it].toDouble() + } + } + else -> super.sub(a, b) + } + } +} \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/GlobalContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/GlobalContext.kt index 40be9039..41af3ec4 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/GlobalContext.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/GlobalContext.kt @@ -2,6 +2,8 @@ package io.github.alexzhirkevich.skriptie public interface GlobalContext : InterpretationContext { + public fun isFalse(a : Any?) : Boolean + public fun sum(a : Any?, b : Any?) : Any? public fun sub(a : Any?, b : Any?) : Any? public fun mul(a : Any?, b : Any?) : Any? @@ -12,4 +14,5 @@ public interface GlobalContext : InterpretationContext { public fun dec(a : Any?) : Any public fun neg(a : Any?) : Any? + public fun pos(a : Any?) : Any? } \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/InterpretationContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/InterpretationContext.kt index 9e658d13..51620651 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/InterpretationContext.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/InterpretationContext.kt @@ -3,11 +3,13 @@ package io.github.alexzhirkevich.skriptie import io.github.alexzhirkevich.skriptie.common.OpAssign -public interface InterpretationContext : Expression { +public interface InterpretationContext : ExtensionContext, Expression { override fun invoke(context: C): Any? = this public fun interpret(callable: String?, args: List>?): Expression? + + public override fun interpret(parent: Expression, op: String?, args: List>?): Expression? = null } internal fun List>.argForNameOrIndex( @@ -15,13 +17,14 @@ internal fun List>.argForNameOrIndex( vararg name : String, ) : Expression? { - forEach { op -> - if (op is OpAssign && name.any { op.variableName == it }) { - return op.assignableValue - } - } - return argAtOrNull(index) +// forEach { op -> +// if (op is OpAssign && name.any { op.variableName == it }) { +// return op.assignableValue +// } +// } +// +// return argAtOrNull(index) } internal fun List>.argAt( @@ -39,9 +42,10 @@ internal fun List>.argAtOrNull( index : Int, ) : Expression? { - return getOrNull(index).let { - if (it is OpAssign) - it.assignableValue - else it - } + return getOrNull(index) +// /**/.let { +// if (it is OpAssign) +// it.assignableValue +// else it +// } } \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Script.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Script.kt index 16bf878a..ad13af75 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Script.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/Script.kt @@ -1,8 +1,8 @@ package io.github.alexzhirkevich.skriptie public fun interface Script { - public operator fun invoke(engine: ScriptEngine): Any? + public operator fun invoke(context: C): Any? } -public fun Expression.asScript(): Script = Script { invoke(it.context) } +public fun Expression.asScript(): Script = Script { invoke(it) } diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptContext.kt index c3f1f86d..0c833ea1 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptContext.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptContext.kt @@ -1,5 +1,7 @@ package io.github.alexzhirkevich.skriptie +import io.github.alexzhirkevich.skriptie.common.SyntaxError +import io.github.alexzhirkevich.skriptie.common.TypeError import io.github.alexzhirkevich.skriptie.common.unresolvedReference @@ -19,13 +21,12 @@ public interface ScriptContext { extraVariables: Map> = emptyMap(), block: (ScriptContext) -> Any? ): Any? - public fun reset() } private class BlockScriptContext( private val parent : ScriptContext -) : BaseScriptContext() { +) : EcmascriptContext() { override fun getVariable(name: String): Any? { return if (name in variables) { @@ -48,7 +49,7 @@ private class BlockScriptContext( } } -public abstract class BaseScriptContext : ScriptContext { +public abstract class EcmascriptContext : ScriptContext { protected val variables: MutableMap> = mutableMapOf() @@ -65,10 +66,10 @@ public abstract class BaseScriptContext : ScriptContext { unresolvedReference(name) } if (type != null && name in variables) { - error("Identifier '$name' is already declared") + throw SyntaxError("Identifier '$name' is already declared") } if (type == null && variables[name]?.first == VariableType.Const) { - error("TypeError: Assignment to constant variable ('$name')") + throw TypeError("Assignment to constant variable ('$name')") } variables[name] = (type ?: variables[name]?.first)!! to value } @@ -91,4 +92,4 @@ public abstract class BaseScriptContext : ScriptContext { override fun reset(){ variables.clear() } -} \ No newline at end of file +} diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptEngine.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptEngine.kt index aa14d846..a3606c31 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptEngine.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptEngine.kt @@ -1,14 +1,22 @@ package io.github.alexzhirkevich.skriptie -public interface ScriptEngine { +public interface ScriptEngine : ScriptInterpreter { public val context : C - public fun compile(script : String) : Script - - public fun reset() + public fun reset() { + context.reset() + } } public fun ScriptEngine.invoke(script: String) : Any? { - return compile(script).invoke(this) + return interpret(script).invoke(context) } + +public fun ScriptEngine( + context: C, + interpreter: ScriptInterpreter +): ScriptEngine = object : ScriptEngine, ScriptInterpreter by interpreter { + override val context: C + get() = context +} \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptInterpreter.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptInterpreter.kt index fd688657..983881e4 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptInterpreter.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ScriptInterpreter.kt @@ -1,5 +1,6 @@ package io.github.alexzhirkevich.skriptie -internal interface ScriptInterpreter { - fun interpret() : Script +public interface ScriptInterpreter { + + public fun interpret(script : String) : Script } \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/Erorrs.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/Erorrs.kt new file mode 100644 index 00000000..cc6f809e --- /dev/null +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/Erorrs.kt @@ -0,0 +1,14 @@ +package io.github.alexzhirkevich.skriptie.common + +public sealed class SkriptieError(message : String?, cause : Throwable?) : Exception(message, cause) + +public class SyntaxError(message : String? = null, cause : Throwable? = null) : SkriptieError(message, cause) + +public class TypeError(message : String? = null, cause : Throwable? = null) : SkriptieError(message, cause) + +public class ReferenceError(message : String? = null, cause : Throwable? = null) : SkriptieError(message, cause) + +internal fun unresolvedReference(ref : String, obj : String? = null) : Nothing = + if (obj != null) + throw ReferenceError("Unresolved reference '$ref' for $obj") + else throw ReferenceError("$ref is not defined") diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBlock.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBlock.kt index 7ffeb102..58af798a 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBlock.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBlock.kt @@ -3,6 +3,7 @@ package io.github.alexzhirkevich.skriptie.common import io.github.alexzhirkevich.skriptie.Expression import io.github.alexzhirkevich.skriptie.ScriptContext + internal class OpBlock( val expressions: List>, private val scoped : Boolean, @@ -25,14 +26,34 @@ internal class OpBlock( if (expressions.size > 1) { repeat(expressions.size - 1) { - val expr = expressions[it] - val res = expr(context) - - if (expr is OpReturn<*>) { - return res - } + invoke(expressions[it], context) } } - return expressions.last().invoke(context) + + return invoke(expressions.last(), context) + } + + private fun invoke(expression: Expression, context: C) : Any? { + val res = expression(context) + return when(expression){ + is OpReturn -> throw BlockReturn(res) + is OpContinue -> throw BlockContinue + is OpBreak -> throw BlockBreak + else -> res + } } -} \ No newline at end of file +} + + +internal sealed class BlockException : Throwable() +internal data object BlockContinue : BlockException() +internal data object BlockBreak : BlockException() +internal class BlockReturn(val value: Any?) : BlockException() + +internal class OpReturn( + val value : Expression +) : Expression by value + +internal class OpContinue : Expression by OpConstant(Unit) +internal class OpBreak : Expression by OpConstant(Unit) + diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBooleans.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBooleans.kt index b28cb6c7..9332a5ac 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBooleans.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpBooleans.kt @@ -4,16 +4,18 @@ import io.github.alexzhirkevich.skriptie.Expression import io.github.alexzhirkevich.skriptie.ScriptContext internal fun OpNot( - condition : Expression -) = Expression{ - !(condition(it) as Boolean) + condition : Expression, + isFalse : (Any?) -> Boolean, +) = Expression { + isFalse(condition(it)) } internal fun OpBoolean( a : Expression, b : Expression, + isFalse : (Any?) -> Boolean, op : (Boolean, Boolean) -> Boolean, ) = Expression { - op(!a(it).isFalse(), !b(it).isFalse()) + op(!isFalse(a(it)), !(isFalse(b(it)))) } diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpFunction.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpFunction.kt index e04c4b91..c9af43c3 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpFunction.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpFunction.kt @@ -1,55 +1,78 @@ package io.github.alexzhirkevich.skriptie.common -import io.github.alexzhirkevich.skriptie.ScriptContext import io.github.alexzhirkevich.skriptie.Expression +import io.github.alexzhirkevich.skriptie.ScriptContext import io.github.alexzhirkevich.skriptie.VariableType import io.github.alexzhirkevich.skriptie.argForNameOrIndex +import io.github.alexzhirkevich.skriptie.ecmascript.Object internal class FunctionParam( val name : String, - val default : Expression? + val isVararg : Boolean = false, + val default : Expression? = null ) +internal infix fun String.with(default: Expression?) : FunctionParam { + return FunctionParam(this, false, default) +} + + internal class OpFunction( val name : String, private val parameters : List>, private val body : Expression ) { + init { + val varargs = parameters.count { it.isVararg } + + if (varargs > 1 || varargs == 1 && !parameters.last().isVararg){ + throw SyntaxError("Rest parameter must be last formal parameter") + } + } fun invoke( args: List>, context: C, ): Any? { - val arguments = buildMap { - parameters.fastForEachIndexed { i, p -> - this[p.name] = Pair( - VariableType.Local, - requireNotNull(args.argForNameOrIndex(i, p.name) ?: p.default) { - "'${p.name}' argument of '$name' function is missing" - }.invoke(context) - ) + try { + val arguments = buildMap { + parameters.fastForEachIndexed { i, p -> + val value = if (p.isVararg){ + args.drop(i).fastMap { it(context) } + } else { + requireNotNull(args.argForNameOrIndex(i, p.name) ?: p.default) { + "'${p.name}' argument of '$name' function is missing" + }.invoke(context) + } + this[p.name] = Pair( + VariableType.Local, + value + ) + } } - } - - return context.withScope(arguments){ - body.invoke(it as C) + return context.withScope(arguments) { + body.invoke(it as C) + } + } catch (ret: BlockReturn) { + return ret.value } } } internal fun OpFunctionExec( name : String, + receiver : Expression?, parameters : List>, -) = Expression { - val function = it.getVariable(name) as? OpFunction - ?: unresolvedReference(name) +) = Expression { ctx -> + + val function = when (val res = receiver?.invoke(ctx)) { + null -> ctx.getVariable(name) + is Object -> res[name] + else -> null + } as? OpFunction ?: unresolvedReference(name) function.invoke( args = parameters, - context = it, + context = ctx, ) } - -internal class OpReturn( - val value : Expression -) : Expression by value \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpGetVariable.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpGetVariable.kt index d41e2faf..bed50055 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpGetVariable.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpGetVariable.kt @@ -3,9 +3,11 @@ package io.github.alexzhirkevich.skriptie.common import io.github.alexzhirkevich.skriptie.Expression import io.github.alexzhirkevich.skriptie.ScriptContext import io.github.alexzhirkevich.skriptie.VariableType +import io.github.alexzhirkevich.skriptie.ecmascript.Object internal class OpGetVariable( val name : String, + val receiver : Expression?, val assignmentType : VariableType? = null ) : Expression { @@ -15,10 +17,14 @@ internal class OpGetVariable( return if (assignmentType != null) { context.setVariable(name, 0f, assignmentType) } else { - if (context.hasVariable(name)){ - context.getVariable(name) - } else { - Unit + when (val res = receiver?.invoke(context)) { + is Object -> if (name in res) res[name] else Unit + + else -> if (context.hasVariable(name)) { + context.getVariable(name) + } else { + unresolvedReference(name) + } } } } diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpIncDec.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpIncDec.kt new file mode 100644 index 00000000..0d508822 --- /dev/null +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpIncDec.kt @@ -0,0 +1,40 @@ +package io.github.alexzhirkevich.skriptie.common + +import io.github.alexzhirkevich.skriptie.Expression +import io.github.alexzhirkevich.skriptie.ScriptContext + +internal fun OpIncDecAssign( + variable: Expression, + preAssign : Boolean, + op: (Any?) -> Any? +) : Expression { + + val value = Expression { op(variable(it)) } + val assignment = when { + variable is OpGetVariable && variable.assignmentType == null -> + OpAssign( + variableName = variable.name, + assignableValue = value, + merge = null + ) + + variable is OpIndex && variable.variable is OpGetVariable -> + OpAssignByIndex( + variableName = variable.variable.name, + index = variable.index, + assignableValue = value, + scope = null, + merge = null + ) + + else -> error("$variable is not assignable") + } + + if (preAssign) { + return assignment + } + + return Expression { ctx -> + variable(ctx).also { assignment(ctx) } + } +} diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpLoops.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpLoops.kt index 610444ab..9279e04c 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpLoops.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpLoops.kt @@ -4,39 +4,25 @@ import io.github.alexzhirkevich.skriptie.Expression import io.github.alexzhirkevich.skriptie.ScriptContext import io.github.alexzhirkevich.skriptie.VariableType + internal class OpForLoop( private val assignment : OpAssign?, private val increment: Expression?, private val comparison : Expression?, + private val isFalse : (Any?) -> Boolean, private val body: Expression ) : Expression { - override fun invoke( - context: C - ): Any? { - if (comparison == null) { - loop( - condition = true, - context = context, - ) - } else { - TODO("for loop") - } - - return Unit + private val condition: (C) -> Boolean = if (comparison == null) { + { true } + } else { + { !isFalse(comparison.invoke(it)) } } - private fun loop( - condition: Boolean, - context: C, - ) { - val block = { ctx: C -> - while (condition) { - body.invoke(ctx) - increment?.invoke(ctx) - } - } + override fun invoke( + context: C + ): Any { if (assignment?.type == VariableType.Local || assignment?.type == VariableType.Const) { context.withScope( @@ -54,34 +40,54 @@ internal class OpForLoop( assignment?.invoke(context) context.withScope { block(it as C) } } + return Unit + } + + private fun block(ctx: C) { + while (condition(ctx)) { + try { + body.invoke(ctx) + } catch (_: BlockContinue) { + continue + } catch (_: BlockBreak) { + break + } finally { + increment?.invoke(ctx) + } + } } } internal fun OpDoWhileLoop( condition : Expression, - body : OpBlock + body : OpBlock, + isFalse : (Any?) -> Boolean ) = Expression { do { - body.invoke(it) - } while (!condition.invoke(it).isFalse()) + try { + body.invoke(it) + } catch (_: BlockContinue) { + continue + } catch (_: BlockBreak) { + break + } + } while (!isFalse(condition.invoke(it))) } internal fun OpWhileLoop( condition : Expression, - body : Expression + body : Expression, + isFalse : (Any?) -> Boolean ) = Expression { - while (!condition.invoke(it).isFalse()){ - body.invoke(it) + while (!isFalse(condition.invoke(it))) { + try { + body.invoke(it) + } catch (_: BlockContinue) { + continue + } catch (_: BlockBreak) { + break + } } } - -internal fun Any?.isFalse() : Boolean { - return this == null - ||this == false - || this is Number && toDouble() == 0.0 - || this is CharSequence && isEmpty() - || this is Unit -} - diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpVar.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpVar.kt index 02b63598..3919b1e9 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpVar.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/OpVar.kt @@ -10,9 +10,15 @@ internal class OpVar( ) : Expression, InterpretationContext { override fun interpret(callable: String?, args: List>?): Expression? { - return if (callable == null) + return if (callable == null) { OpConstant(Unit) - else OpGetVariable(callable, assignmentType = scope) + } else { + OpGetVariable( + name = callable, + receiver = null, + assignmentType = scope + ) + } } } diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/ScriptUtil.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/ScriptUtil.kt index d6c1651a..3550f7bd 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/ScriptUtil.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/common/ScriptUtil.kt @@ -23,11 +23,6 @@ internal fun checkNotEmpty(value : T?) : T { return value } -internal fun unresolvedReference(ref : String, obj : String? = null) : Nothing = - if (obj != null) - error("Unresolved reference '$ref' for $obj") - else error("Unresolved reference: $ref") - internal fun Expression.cast(block: (T) -> R) : Expression = Expression { block(invoke(it) as T) } diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreter.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreter.kt index 4655b2ff..d089d464 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreter.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreter.kt @@ -1,769 +1,14 @@ package io.github.alexzhirkevich.skriptie.ecmascript -import io.github.alexzhirkevich.skriptie.Expression -import io.github.alexzhirkevich.skriptie.ExtensionContext import io.github.alexzhirkevich.skriptie.GlobalContext -import io.github.alexzhirkevich.skriptie.InterpretationContext import io.github.alexzhirkevich.skriptie.Script import io.github.alexzhirkevich.skriptie.ScriptContext import io.github.alexzhirkevich.skriptie.ScriptInterpreter -import io.github.alexzhirkevich.skriptie.VariableType -import io.github.alexzhirkevich.skriptie.asScript -import io.github.alexzhirkevich.skriptie.common.FunctionParam -import io.github.alexzhirkevich.skriptie.common.OpBlock -import io.github.alexzhirkevich.skriptie.common.OpBoolean -import io.github.alexzhirkevich.skriptie.common.OpDoWhileLoop -import io.github.alexzhirkevich.skriptie.common.OpEquals -import io.github.alexzhirkevich.skriptie.common.OpFunction -import io.github.alexzhirkevich.skriptie.common.OpFunctionExec -import io.github.alexzhirkevich.skriptie.common.OpIfCondition -import io.github.alexzhirkevich.skriptie.common.OpNot -import io.github.alexzhirkevich.skriptie.common.OpReturn -import io.github.alexzhirkevich.skriptie.common.OpTryCatch -import io.github.alexzhirkevich.skriptie.common.OpWhileLoop -import io.github.alexzhirkevich.skriptie.common.unresolvedReference -import io.github.alexzhirkevich.skriptie.common.OpAssign -import io.github.alexzhirkevich.skriptie.common.OpAssignByIndex -import io.github.alexzhirkevich.skriptie.common.OpCompare -import io.github.alexzhirkevich.skriptie.common.OpConstant -import io.github.alexzhirkevich.skriptie.common.OpEqualsComparator -import io.github.alexzhirkevich.skriptie.common.OpGreaterComparator -import io.github.alexzhirkevich.skriptie.common.OpIndex -import io.github.alexzhirkevich.skriptie.common.OpLessComparator -import io.github.alexzhirkevich.skriptie.common.OpMakeArray -import io.github.alexzhirkevich.skriptie.common.OpVar -import io.github.alexzhirkevich.skriptie.common.Delegate -import io.github.alexzhirkevich.skriptie.common.OpGetVariable -import io.github.alexzhirkevich.skriptie.isAssignable -import kotlin.contracts.ExperimentalContracts -import kotlin.contracts.contract -internal val EXPR_DEBUG_PRINT_ENABLED = false - -internal enum class LogicalContext { - And, Or, Compare -} - - -internal class EcmascriptInterpreter( - private val expr : String, - private val scriptContext: C, - private val globalContext : GlobalContext, - private val extensionContext : ExtensionContext +public class EcmascriptInterpreter( + private val context : GlobalContext, ) : ScriptInterpreter { - - private var pos = -1 - private var ch: Char = ' ' - - override fun interpret(): Script { - val expressions = buildList { - pos = -1 - ch = ' ' - if (EXPR_DEBUG_PRINT_ENABLED) { - println("Parsing $expr") - } - nextChar() - do { - while (eat(';')) { - } - if (pos >= expr.length) { - break - } - - add(parseAssignment(globalContext)) - } while (pos < expr.length) - - require(pos <= expr.length) { - "Unexpected Lottie expression $expr" - } - } - return OpBlock( - expressions = expressions, - scoped = false - ).asScript().also { - pos = -1 - ch = ' ' - if (EXPR_DEBUG_PRINT_ENABLED) { - println("Expression parsed: $expr") - } - } - } - - private fun prepareNextChar(){ - while (ch.skip() && pos < expr.length){ - nextChar() - } - } - - private fun nextChar() { - ch = if (++pos < expr.length) expr[pos] else ' ' - } - - private fun prevChar() { - ch = if (--pos > 0 && pos < expr.length) expr[pos] else ' ' - } - - private fun Char.skip() : Boolean = this == ' ' || this == '\n' - - private fun eat(charToEat: Char): Boolean { - while (ch.skip() && pos < expr.length) - nextChar() - - if (ch == charToEat) { - nextChar() - return true - } - return false - } - - private fun nextCharIs(condition: (Char) -> Boolean): Boolean { - var i = pos - - while (i < expr.length) { - if (condition(expr[i])) - return true - if (expr[i].skip()) - i++ - else return false - } - return false - } - - private fun eatSequence(seq : String): Boolean { - - val p = pos - val c = ch - - if (seq.isEmpty()) - return true - - if (!eat(seq[0])) { - return false - } - - return if (expr.indexOf(seq, startIndex = pos-1) == pos-1){ - pos += seq.length -1 - ch = expr[pos.coerceIn(expr.indices)] - true - } else { - pos = p - ch = c - false - } - } - - private fun nextSequenceIs(seq : String): Boolean { - - val p = pos - val c = ch - - if (seq.isEmpty()) - return true - - if (!eat(seq[0])) { - return false - } - - return if (expr.indexOf(seq, startIndex = pos-1) == pos-1){ - pos = p - ch = c - true - } else { - pos = p - ch = c - false - } - } - - private fun parseAssignment(context: Expression): Expression { - var x = parseExpressionOp(context) - if (EXPR_DEBUG_PRINT_ENABLED){ - println("Parsing assignment for $x") - } - while (true) { - prepareNextChar() - x = when { - eatSequence("+=") -> parseAssignmentValue(x, globalContext::sum) - eatSequence("-=") -> parseAssignmentValue(x, globalContext::sub) - eatSequence("*=") -> parseAssignmentValue(x, globalContext::mul) - eatSequence("/=") -> parseAssignmentValue(x, globalContext::div) - eatSequence("%=") -> parseAssignmentValue(x, globalContext::mod) - eat('=') -> parseAssignmentValue(x, null) - x.isAssignable() && eatSequence("++") -> Delegate(x, globalContext::inc) - x.isAssignable() && eatSequence("--") -> Delegate(x, globalContext::dec) - else -> return x - } - } - } - - private fun parseAssignmentValue(x : Expression, merge : ((Any?, Any?) -> Any?)? = null) = when { - x is OpIndex && x.variable is OpGetVariable -> OpAssignByIndex( - variableName = x.variable.name, - scope = x.variable.assignmentType, - index = x.index, - assignableValue = parseAssignment(globalContext), - merge = merge - ).also { - if (EXPR_DEBUG_PRINT_ENABLED) { - println("parsing assignment with index for ${x.variable.name}") - } - } - - x is OpGetVariable -> OpAssign( - variableName = x.name, - assignableValue = parseAssignment(globalContext), - type = x.assignmentType, - merge = merge - ).also { - if (EXPR_DEBUG_PRINT_ENABLED) { - println("parsing assignment for ${x.name} in ${it.type} scope") - } - } - - else -> error("Invalid assignment") - } - - private fun parseExpressionOp(context: Expression, logicalContext: LogicalContext? = null): Expression { - var x = parseTermOp(context) - while (true) { - prepareNextChar() - x = when { - logicalContext != LogicalContext.Compare && eatSequence("&&") -> - OpBoolean(parseExpressionOp(globalContext, LogicalContext.And),x, Boolean::and) - logicalContext == null && eatSequence("||") -> - OpBoolean(parseExpressionOp(globalContext, LogicalContext.Or),x, Boolean::or) - eatSequence("<=") -> OpCompare(x, parseExpressionOp(globalContext, LogicalContext.Compare)) { a, b -> - OpLessComparator(a, b) || OpEqualsComparator(a, b) - } - eatSequence("<") -> OpCompare(x, parseExpressionOp(globalContext, LogicalContext.Compare), OpLessComparator) - eatSequence(">=") -> OpCompare(x, parseExpressionOp(globalContext, LogicalContext.Compare)) { a, b -> - OpGreaterComparator(a, b) || OpEqualsComparator(a, b) - } - eatSequence(">") -> OpCompare(x, parseExpressionOp(globalContext, LogicalContext.Compare), OpGreaterComparator) - eatSequence("===") -> OpEquals(x, parseExpressionOp(globalContext, LogicalContext.Compare), true) - eatSequence("==") -> OpEquals(x, parseExpressionOp(globalContext, LogicalContext.Compare), false) - eatSequence("!==") -> OpNot(OpEquals(x, parseExpressionOp(globalContext, LogicalContext.Compare), false)) - eatSequence("!=") -> OpNot(OpEquals(x, parseExpressionOp(globalContext, LogicalContext.Compare), true)) - !nextSequenceIs("++") && !nextSequenceIs("+=") && eat('+') -> - Delegate(x, parseTermOp(globalContext), globalContext::sum) - !nextSequenceIs("--") && !nextSequenceIs("-=") && eat('-') -> - Delegate(x, parseTermOp(globalContext),globalContext::sub) - else -> return x - } - } - } - - private fun parseTermOp(context: Expression): Expression { - var x = parseFactorOp(context) - while (true) { - prepareNextChar() - x = when { - !nextSequenceIs("*=") && eat('*') -> Delegate( - x, - parseFactorOp(globalContext), - globalContext::mul - ) - - !nextSequenceIs("/=") && eat('/') -> Delegate( - x, - parseFactorOp(globalContext), - globalContext::div - ) - - !nextSequenceIs("%=") && eat('%') -> Delegate( - x, - parseFactorOp(globalContext), - globalContext::mod - ) - - else -> return x - } - } - } - - private fun parseFactorOp(context: Expression): Expression { - val parsedOp = when { - context is GlobalContext && eatSequence("++") -> { - val start = pos - val variable = parseFactorOp(globalContext) - require(variable.isAssignable()){ - "Unexpected '++' as $start" - } - Delegate(variable, globalContext::inc) - } - - context is GlobalContext && eatSequence("--") -> { - val start = pos - val variable = parseFactorOp(globalContext) - require(variable.isAssignable()){ - "Unexpected '--' as $start" - } - Delegate(variable, globalContext::dec) - } - - context is GlobalContext && eat('+') -> - Delegate(parseFactorOp(context)) { it } - - context is GlobalContext && eat('-') -> - Delegate(parseFactorOp(context), globalContext::neg) - - context is GlobalContext && !nextSequenceIs("!=") && eat('!') -> - OpNot(parseExpressionOp(context)) - - context is GlobalContext && eat('(') -> { - parseExpressionOp(context).also { - require(eat(')')) { - "Bad expression: Missing ')'" - } - } - } - - context is GlobalContext && nextCharIs { it.isDigit() || it == '.' } -> { - if (EXPR_DEBUG_PRINT_ENABLED) { - print("making const number... ") - } - var numberFormat = NumberFormat.Dec - var isFloat = nextCharIs { it == '.' } - val startPos = pos - do { - nextChar() - when(ch.lowercaseChar()){ - '.' -> { - if (isFloat) { - break - } - isFloat = true - } - NumberFormat.Hex.prefix -> { - check(numberFormat == NumberFormat.Dec && !isFloat) { - "Invalid number at pos $startPos" - } - numberFormat = NumberFormat.Hex - } - NumberFormat.Oct.prefix -> { - check(numberFormat == NumberFormat.Dec && !isFloat) { - "Invalid number at pos $startPos" - } - numberFormat = NumberFormat.Oct - } - NumberFormat.Bin.prefix -> { - if (numberFormat == NumberFormat.Hex) { - continue - } - check(numberFormat == NumberFormat.Dec && !isFloat) { - "Invalid number at pos $startPos" - } - numberFormat = NumberFormat.Bin - } - } - } while (ch.lowercaseChar().let { - it in numberFormat.alphabet || it in NumberFormatIndicators - }) - - val num = expr.substring(startPos, pos).let { - if (it.endsWith('.')) { - prevChar() - isFloat = false - } - if (isFloat) { - it.toDouble() - } - else { - it.trimEnd('.') - .let { n -> numberFormat.prefix?.let(n::substringAfter) ?: n } - .toULong(numberFormat.radix) - .toLong() - } - } - if (EXPR_DEBUG_PRINT_ENABLED) { - println(num) - } - OpConstant(num) - } - - context is GlobalContext && nextCharIs('\''::equals) || nextCharIs('"'::equals) -> { - if (EXPR_DEBUG_PRINT_ENABLED) { - print("making const string... ") - } - val c = ch - val startPos = pos - do { - nextChar() - } while (!eat(c)) - val str = expr.substring(startPos, pos).drop(1).dropLast(1) - if (EXPR_DEBUG_PRINT_ENABLED) { - println(str) - } - OpConstant(str) - } - - context is GlobalContext && eat('[') -> { // make array - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making array... ") - } - val arrayArgs = buildList { - do { - if (eat(']')) { // empty list - return@buildList - } - add(parseExpressionOp(context)) - } while (eat(',')) - require(eat(']')) { - "Bad expression: missing ]" - } - } - OpMakeArray(arrayArgs) - } - - context !is GlobalContext && eat('[') -> { // index - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making index... ") - } - OpIndex(context, parseExpressionOp(globalContext)).also { - require(eat(']')) { - "Bad expression: Missing ']'" - } - } - } - - - ch.isFun() -> { - - val startPos = pos - do { - nextChar() - } while ( - pos < expr.length && ch.isFun() && !(isReserved(expr.substring(startPos, pos)) && ch == ' ') - ) - - val func = expr.substring(startPos, pos).trim() - - parseFunction(context, func) - } - - else -> error("Unsupported Lottie expression: $expr") - } - - return parsedOp.finish() - } - - private fun Expression.finish() : Expression { - return when { - // inplace function invocation - this is InterpretationContext<*> && nextCharIs { it == '(' } -> { - parseFunction(this, null) - } - // begin condition || property || index - this is OpVar<*> - || eat('.') - || nextCharIs('['::equals) -> - parseFactorOp(this) // continue with receiver - - else -> this - } - } - - private fun parseFunctionArgs(name : String?): List>? { - - if (!nextCharIs('('::equals)){ - return null - } - return buildList { - when { - eat('(') -> { - if (eat(')')){ - return@buildList //empty args - } - do { - add(parseAssignment(globalContext)) - } while (eat(',')) - - require(eat(')')) { - "Bad expression:Missing ')' after argument to $name" - } - } - } - } - } - - private fun parseFunction(context: Expression, func : String?) : Expression { - - return when (func) { - "var", "let", "const" -> { - OpVar( - when (func) { - "var" -> VariableType.Global - "let" -> VariableType.Local - else -> VariableType.Const - } - ) - } - "undefined" -> OpConstant(Unit) - "null" -> OpConstant(null) - "true" -> OpConstant(true) - "false" -> OpConstant(false) - "function" -> parseFunctionDefinition() - "while" -> { - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making while loop") - } - - OpWhileLoop( - condition = parseWhileCondition(), - body = parseBlock() - ) - } - "do" -> { - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making do/while loop") - } - - val body = parseBlock() - - check(body is OpBlock){ - "Invalid do/while syntax" - } - - check(eatSequence("while")){ - "Missing while condition in do/while block" - } - val condition = parseWhileCondition() - - OpDoWhileLoop( - condition = condition, - body = body - ) - } - - "if" -> { - - if (EXPR_DEBUG_PRINT_ENABLED) { - print("parsing if...") - } - - val condition = parseExpressionOp(globalContext) - - val onTrue = parseBlock() - - val onFalse = if (eatSequence("else")) { - parseBlock() - } else null - - OpIfCondition( - condition = condition, - onTrue = onTrue, - onFalse = onFalse - ) - } - - "return" -> { - val expr = parseExpressionOp(globalContext) - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making return with $expr") - } - OpReturn(expr) - } - "try" -> { - val tryBlock = parseBlock(requireBlock = true) - val catchBlock = if (eatSequence("catch")) { - - if (eat('(')) { - val start = pos - while (!eat(')') && pos < expr.length) { - //nothing - } - expr.substring(start, pos).trim() to parseBlock( - scoped = false, - requireBlock = true - ) - } else { - null to parseBlock(requireBlock = true) - } - } - else null - - val finallyBlock = if (eatSequence("finally")){ - parseBlock(requireBlock = true) - } else null - - OpTryCatch( - tryBlock = tryBlock, - catchVariableName = catchBlock?.first, - catchBlock = catchBlock?.second, - finallyBlock = finallyBlock - ) - } - - else -> { - val args = parseFunctionArgs(func) - - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making fun $func") - } - - return when (context) { - is InterpretationContext -> context.interpret(func, args) - ?: (if (args != null && func != null && this.scriptContext.getVariable(func) is OpFunction<*>) { - if (EXPR_DEBUG_PRINT_ENABLED) { - println("parsed call for defined function $func") - } - OpFunctionExec(func, args) - } else null) - ?: run { - if (args == null && func != null) { - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making GetVariable $func...") - } - OpGetVariable(func) - } else { - null - } - } - ?: unresolvedReference( - ref = func ?: "null", - obj = context::class.simpleName - ?.substringAfter("Op") - ?.substringBefore("Context") - ) - - else -> extensionContext.interpret(context, func,args) - ?: unresolvedReference(func ?: "null") - } - } - } - } - - private fun parseWhileCondition(): Expression { - check(eat('(')){ - "Missing while loop condition" - } - - val condition = parseExpressionOp(globalContext) - - check(eat(')')){ - "Missing closing ')' in loop condition" - } - return condition - } - - private fun parseFunctionDefinition() : Expression { - val start = pos - - while (ch != '(') { - nextChar() - } - - val name = expr.substring(start, pos).trim() - - if (EXPR_DEBUG_PRINT_ENABLED) { - println("making defined function $name") - } - - val args = parseFunctionArgs(name).let { args -> - args?.map { - when (it) { - is OpGetVariable -> FunctionParam(name = it.name, default = null) - is OpAssign -> FunctionParam( - name = it.variableName, - default = it.assignableValue - ) - - else -> error("Invalid function declaration at $start") - } - } - } - - checkNotNull(args){ - "Missing function args" - } - - - check(nextCharIs('{'::equals)) { - "Missing function body at $pos" - } - - - val block = parseBlock( - scoped = false // function scope will be used - ) - - this.scriptContext.setVariable( - name = name, - value = OpFunction( - name = name, - parameters = args, - body = block - ), - type = VariableType.Const - ) - if (EXPR_DEBUG_PRINT_ENABLED) { - println("registered function $name") - } - return OpConstant(Unit) - } - - private fun parseBlock(scoped : Boolean = true, requireBlock : Boolean = false): Expression { - val list = buildList { - if (eat('{')) { - while (!eat('}') && pos < expr.length) { - add(parseAssignment(globalContext)) - eat(';') - } - } else { - if (requireBlock){ - error("Unexpected token at $pos: block start was expected") - } - add(parseAssignment(globalContext)) - } - } - return OpBlock(list, scoped) - } -} - - -@OptIn(ExperimentalContracts::class) -internal fun checkArgsNotNull(args : List<*>?, func : String) { - contract { - returns() implies (args != null) - } - checkNotNull(args){ - "$func call was missing" - } -} - -@OptIn(ExperimentalContracts::class) -internal fun checkArgs(args : List<*>?, count : Int, func : String) { - contract { - returns() implies (args != null) - } - checkNotNull(args){ - "$func call was missing" - } - require(args.size == count){ - "$func takes $count arguments, but ${args.size} got" + override fun interpret(script: String): Script { + return EcmascriptInterpreterImpl(script, context).interpret() } } - -private val funMap = (('a'..'z').toList() + ('A'..'Z').toList() + '$' + '_' ).associateBy { it } - -private fun Char.isFun() = isDigit() || funMap[this] != null - - -private enum class NumberFormat( - val radix : Int, - val alphabet : String, - val prefix : Char? -) { - Dec(10, ".0123456789", null), - Hex(16, "0123456789abcdef", 'x'), - Oct(8, "01234567", 'o'), - Bin(2, "01", 'b') -} - -private val NumberFormatIndicators = NumberFormat.entries.mapNotNull { it.prefix } - -private val reservedKeywords = setOf( - "function","return","do","while","for" -) - -private fun isReserved(keyword : String) = keyword in reservedKeywords diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreterImpl.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreterImpl.kt new file mode 100644 index 00000000..1ae52bf2 --- /dev/null +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/EcmascriptInterpreterImpl.kt @@ -0,0 +1,1002 @@ +package io.github.alexzhirkevich.skriptie.ecmascript + +import io.github.alexzhirkevich.skriptie.Expression +import io.github.alexzhirkevich.skriptie.GlobalContext +import io.github.alexzhirkevich.skriptie.InterpretationContext +import io.github.alexzhirkevich.skriptie.Script +import io.github.alexzhirkevich.skriptie.ScriptContext +import io.github.alexzhirkevich.skriptie.VariableType +import io.github.alexzhirkevich.skriptie.asScript +import io.github.alexzhirkevich.skriptie.common.Delegate +import io.github.alexzhirkevich.skriptie.common.FunctionParam +import io.github.alexzhirkevich.skriptie.common.OpAssign +import io.github.alexzhirkevich.skriptie.common.OpAssignByIndex +import io.github.alexzhirkevich.skriptie.common.OpBlock +import io.github.alexzhirkevich.skriptie.common.OpBoolean +import io.github.alexzhirkevich.skriptie.common.OpBreak +import io.github.alexzhirkevich.skriptie.common.OpCompare +import io.github.alexzhirkevich.skriptie.common.OpConstant +import io.github.alexzhirkevich.skriptie.common.OpContinue +import io.github.alexzhirkevich.skriptie.common.OpDoWhileLoop +import io.github.alexzhirkevich.skriptie.common.OpEquals +import io.github.alexzhirkevich.skriptie.common.OpEqualsComparator +import io.github.alexzhirkevich.skriptie.common.OpForLoop +import io.github.alexzhirkevich.skriptie.common.OpFunction +import io.github.alexzhirkevich.skriptie.common.OpFunctionExec +import io.github.alexzhirkevich.skriptie.common.OpGetVariable +import io.github.alexzhirkevich.skriptie.common.OpGreaterComparator +import io.github.alexzhirkevich.skriptie.common.OpIfCondition +import io.github.alexzhirkevich.skriptie.common.OpIncDecAssign +import io.github.alexzhirkevich.skriptie.common.OpIndex +import io.github.alexzhirkevich.skriptie.common.OpLessComparator +import io.github.alexzhirkevich.skriptie.common.OpMakeArray +import io.github.alexzhirkevich.skriptie.common.OpNot +import io.github.alexzhirkevich.skriptie.common.OpReturn +import io.github.alexzhirkevich.skriptie.common.OpTryCatch +import io.github.alexzhirkevich.skriptie.common.OpVar +import io.github.alexzhirkevich.skriptie.common.OpWhileLoop +import io.github.alexzhirkevich.skriptie.common.SyntaxError +import io.github.alexzhirkevich.skriptie.common.unresolvedReference +import io.github.alexzhirkevich.skriptie.isAssignable +import kotlin.contracts.ExperimentalContracts +import kotlin.contracts.contract + +internal val EXPR_DEBUG_PRINT_ENABLED = false + +internal enum class LogicalContext { + And, Or, Compare +} + +internal enum class BlockContext { + None, Loop, Function +} + + +internal class EcmascriptInterpreterImpl( + private val expr : String, + private val globalContext : GlobalContext, +) { + + private var pos = -1 + private var ch: Char = ' ' + private var blockFunctionRegistrar = mutableListOf>() + + fun interpret(): Script { + val expressions = buildList { + pos = -1 + ch = ' ' + if (EXPR_DEBUG_PRINT_ENABLED) { + println("Parsing $expr") + } + nextChar() + do { + while (eat(';')) { + } + if (pos >= expr.length) { + break + } + + add(parseAssignment(globalContext, emptyList())) + } while (pos < expr.length) + + require(pos <= expr.length) { + "Unexpected Lottie expression $expr" + } + } + val block = OpBlock( + expressions = blockFunctionRegistrar + expressions, + scoped = false + ) + pos = -1 + ch = ' ' + if (EXPR_DEBUG_PRINT_ENABLED) { + println("Expression parsed: $expr") + } + return block.asScript() + } + + private fun prepareNextChar() { + while (ch.skip() && pos < expr.length) { + nextChar() + } + } + + private fun nextChar() { + ch = if (++pos < expr.length) expr[pos] else ' ' + } + + private fun prevChar() { + ch = if (--pos > 0 && pos < expr.length) expr[pos] else ' ' + } + + private fun Char.skip(): Boolean = this == ' ' || this == '\n' + + private fun eat(charToEat: Char): Boolean { + while (ch.skip() && pos < expr.length) + nextChar() + + if (ch == charToEat) { + nextChar() + return true + } + return false + } + + private fun nextCharIs(condition: (Char) -> Boolean): Boolean { + var i = pos + + while (i < expr.length) { + if (condition(expr[i])) + return true + if (expr[i].skip()) + i++ + else return false + } + return false + } + + private fun eatSequence(seq: String): Boolean { + + val p = pos + val c = ch + + if (seq.isEmpty()) + return true + + if (!eat(seq[0])) { + return false + } + + return if (expr.indexOf(seq, startIndex = pos - 1) == pos - 1) { + pos += seq.length - 1 + ch = expr[pos.coerceIn(expr.indices)] + true + } else { + pos = p + ch = c + false + } + } + + private fun nextSequenceIs(seq: String): Boolean { + + val p = pos + val c = ch + + if (seq.isEmpty()) + return true + + if (!eat(seq[0])) { + return false + } + + return if (expr.indexOf(seq, startIndex = pos - 1) == pos - 1) { + pos = p + ch = c + true + } else { + pos = p + ch = c + false + } + } + + private fun parseAssignment(context: Expression, blockContext : List): Expression { + var x = parseExpressionOp(context, blockContext = blockContext) + if (EXPR_DEBUG_PRINT_ENABLED) { + println("Parsing assignment for $x") + } + while (true) { + prepareNextChar() + x = when { + eatSequence("+=") -> parseAssignmentValue(x, globalContext::sum) + eatSequence("-=") -> parseAssignmentValue(x, globalContext::sub) + eatSequence("*=") -> parseAssignmentValue(x, globalContext::mul) + eatSequence("/=") -> parseAssignmentValue(x, globalContext::div) + eatSequence("%=") -> parseAssignmentValue(x, globalContext::mod) + eat('=') -> parseAssignmentValue(x, null) + eatSequence("++") -> { + check(x.isAssignable()) { + "Not assignable" + } + OpIncDecAssign( + variable = x, + preAssign = false, + op = globalContext::inc + ) + } + + eatSequence("--") -> { + check(x.isAssignable()) { + "Not assignable" + } + OpIncDecAssign( + variable = x, + preAssign = false, + op = globalContext::dec + ) + } + + else -> return x + } + } + } + + private fun parseAssignmentValue(x: Expression, merge: ((Any?, Any?) -> Any?)? = null) = + when { + x is OpIndex && x.variable is OpGetVariable -> OpAssignByIndex( + variableName = x.variable.name, + scope = x.variable.assignmentType, + index = x.index, + assignableValue = parseAssignment(globalContext, emptyList()), + merge = merge + ).also { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("parsed assignment with index for ${x.variable.name}") + } + } + + x is OpGetVariable -> OpAssign( + variableName = x.name, + assignableValue = parseAssignment(globalContext, emptyList()), + type = x.assignmentType, + merge = merge + ).also { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("parsed assignment for ${x.name}") + } + } + + else -> error("Invalid assignment") + } + + private fun parseExpressionOp( + context: Expression, + logicalContext: LogicalContext? = null, + blockContext : List + ): Expression { + var x = parseTermOp(context, blockContext) + while (true) { + prepareNextChar() + x = when { + logicalContext != LogicalContext.Compare && eatSequence("&&") -> + OpBoolean( + parseExpressionOp(globalContext, LogicalContext.And,blockContext), + x, + globalContext::isFalse, Boolean::and + ) + + logicalContext == null && eatSequence("||") -> + OpBoolean( + parseExpressionOp(globalContext, LogicalContext.Or,blockContext), + x, + globalContext::isFalse, Boolean::or + ) + + eatSequence("<=") -> OpCompare( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext) + ) { a, b -> + OpLessComparator(a, b) || OpEqualsComparator(a, b) + } + + eatSequence("<") -> OpCompare( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext), + OpLessComparator + ) + + eatSequence(">=") -> OpCompare( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext) + ) { a, b -> + OpGreaterComparator(a, b) || OpEqualsComparator(a, b) + } + + eatSequence(">") -> OpCompare( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext), + OpGreaterComparator + ) + + eatSequence("===") -> OpEquals( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext), + true + ) + + eatSequence("==") -> OpEquals( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext), + false + ) + + eatSequence("!==") -> OpNot( + OpEquals( + x, + parseExpressionOp(globalContext, LogicalContext.Compare,blockContext), + false + ), + globalContext::isFalse + ) + + eatSequence("!=") -> OpNot( + OpEquals( + x, + parseExpressionOp(globalContext, LogicalContext.Compare, blockContext), + true + ), + globalContext::isFalse + ) + + !nextSequenceIs("++") && !nextSequenceIs("+=") && eat('+') -> + Delegate(x, parseTermOp(globalContext, blockContext), globalContext::sum) + + !nextSequenceIs("--") && !nextSequenceIs("-=") && eat('-') -> + Delegate(x, parseTermOp(globalContext, blockContext), globalContext::sub) + + else -> return x + } + } + } + + private fun parseTermOp(context: Expression, blockContext : List): Expression { + var x = parseFactorOp(context, blockContext) + while (true) { + prepareNextChar() + x = when { + !nextSequenceIs("*=") && eat('*') -> Delegate( + x, + parseFactorOp(globalContext,blockContext), + globalContext::mul + ) + + !nextSequenceIs("/=") && eat('/') -> Delegate( + x, + parseFactorOp(globalContext,blockContext), + globalContext::div + ) + + !nextSequenceIs("%=") && eat('%') -> Delegate( + x, + parseFactorOp(globalContext, blockContext), + globalContext::mod + ) + + else -> return x + } + } + } + + private fun parseFactorOp(context: Expression, blockContext : List): Expression { + val parsedOp = when { + + nextCharIs('{'::equals) -> parseBlock(context = emptyList()) + + context is GlobalContext && eatSequence("++") -> { + val start = pos + val variable = parseFactorOp(globalContext, blockContext) + require(variable.isAssignable()) { + "Unexpected '++' as $start" + } + OpIncDecAssign( + variable = variable, + preAssign = true, + op = globalContext::inc + ) + } + + context is GlobalContext && eatSequence("--") -> { + val start = pos + val variable = parseFactorOp(globalContext, blockContext) + require(variable.isAssignable()) { + "Unexpected '--' as $start" + } + OpIncDecAssign( + variable = variable, + preAssign = true, + op = globalContext::dec + ) + } + + context is GlobalContext && eat('+') -> + Delegate(parseFactorOp(context, blockContext), globalContext::pos) + + context is GlobalContext && eat('-') -> + Delegate(parseFactorOp(context, blockContext), globalContext::neg) + + context is GlobalContext && !nextSequenceIs("!=") && eat('!') -> + OpNot(parseExpressionOp(context, blockContext = blockContext), globalContext::isFalse) + + context is GlobalContext && eat('(') -> { + parseExpressionOp(context, blockContext = blockContext).also { + require(eat(')')) { + "Bad expression: Missing ')'" + } + } + } + + context is GlobalContext && nextCharIs { it.isDigit() || it == '.' } -> { + if (EXPR_DEBUG_PRINT_ENABLED) { + print("making const number... ") + } + var numberFormat = NumberFormat.Dec + var isFloat = nextCharIs { it == '.' } + val startPos = pos + do { + nextChar() + when (ch.lowercaseChar()) { + '.' -> { + if (isFloat) { + break + } + isFloat = true + } + + NumberFormat.Hex.prefix -> { + check(numberFormat == NumberFormat.Dec && !isFloat) { + "Invalid number at pos $startPos" + } + numberFormat = NumberFormat.Hex + } + + NumberFormat.Oct.prefix -> { + check(numberFormat == NumberFormat.Dec && !isFloat) { + "Invalid number at pos $startPos" + } + numberFormat = NumberFormat.Oct + } + + NumberFormat.Bin.prefix -> { + if (numberFormat == NumberFormat.Hex) { + continue + } + check(numberFormat == NumberFormat.Dec && !isFloat) { + "Invalid number at pos $startPos" + } + numberFormat = NumberFormat.Bin + } + } + } while (ch.lowercaseChar().let { + it in numberFormat.alphabet || it in NumberFormatIndicators + }) + val num = try { + expr.substring(startPos, pos).let { + if (it.endsWith('.')) { + prevChar() + isFloat = false + } + if (isFloat) { + it.toDouble() + } else { + it.trimEnd('.') + .let { n -> numberFormat.prefix?.let(n::substringAfter) ?: n } + .toULong(numberFormat.radix) + .toLong() + } + } + } catch (t: NumberFormatException) { + throw SyntaxError("Unexpected token ${expr[startPos]} at $startPos") + } + if (EXPR_DEBUG_PRINT_ENABLED) { + println(num) + } + OpConstant(num) + } + + context is GlobalContext && nextCharIs('\''::equals) || nextCharIs('"'::equals) -> { + if (EXPR_DEBUG_PRINT_ENABLED) { + print("making const string... ") + } + val c = ch + val startPos = pos + do { + nextChar() + } while (!eat(c)) + val str = expr.substring(startPos, pos).drop(1).dropLast(1) + if (EXPR_DEBUG_PRINT_ENABLED) { + println(str) + } + OpConstant(str) + } + + context is GlobalContext && eat('[') -> { // make array + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making array... ") + } + val arrayArgs = buildList { + do { + if (eat(']')) { // empty list + return@buildList + } + add(parseExpressionOp(context,blockContext = blockContext)) + } while (eat(',')) + require(eat(']')) { + "Bad expression: missing ]" + } + } + OpMakeArray(arrayArgs) + } + + context !is GlobalContext && eat('[') -> { // index + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making index... ") + } + OpIndex(context, parseExpressionOp(globalContext, blockContext = blockContext)).also { + require(eat(']')) { + "Bad expression: Missing ']'" + } + } + } + + + ch.isFun() -> { + + val startPos = pos + do { + nextChar() + } while ( + pos < expr.length && ch.isFun() && !(isReserved( + expr.substring( + startPos, + pos + ) + ) && ch == ' ') + ) + + val func = expr.substring(startPos, pos).trim() + + parseFunction(context, func, blockContext) + } + + else -> throw SyntaxError("Unexpected token $ch at pos $pos") + } + + return parsedOp.finish(blockContext) + } + + private fun Expression.finish(blockContext : List): Expression { + return when { + // inplace function invocation + this is InterpretationContext<*> && nextCharIs { it == '(' } -> { + parseFunction(this, null, blockContext) + } + // begin condition || property || index + this is OpVar<*> + || eat('.') + || nextCharIs('['::equals) -> + parseFactorOp(this, blockContext) // continue with receiver + + else -> this + } + } + + private fun parseFunctionArgs(name: String?): List>? { + + if (!nextCharIs('('::equals)) { + return null + } + return buildList { + when { + eat('(') -> { + if (eat(')')) { + return@buildList //empty args + } + do { + add(parseAssignment(globalContext, emptyList())) + } while (eat(',')) + + require(eat(')')) { + "Bad expression:Missing ')' after argument to $name" + } + } + } + } + } + + private fun parseFunction( + context: Expression, + func: String?, + blockContext : List + ): Expression { + + return when (func) { + "var", "let", "const" -> { + OpVar( + when (func) { + "var" -> VariableType.Global + "let" -> VariableType.Local + else -> VariableType.Const + } + ) + } + + "undefined" -> OpConstant(Unit) + "null" -> OpConstant(null) + "true" -> OpConstant(true) + "false" -> OpConstant(false) + "function" -> { + blockFunctionRegistrar += parseFunctionDefinition(blockContext = blockContext) + OpConstant(Unit) + } + "for" -> { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making for loop") + } + parseForLoop(blockContext) + } + + "while" -> { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making while loop") + } + + OpWhileLoop( + condition = parseWhileCondition(), + body = parseBlock(context = blockContext + BlockContext.Loop), + isFalse = globalContext::isFalse + ) + } + + "do" -> { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making do/while loop") + } + + val body = parseBlock(context = blockContext + BlockContext.Loop) + + check(body is OpBlock) { + "Invalid do/while syntax" + } + + check(eatSequence("while")) { + "Missing while condition in do/while block" + } + val condition = parseWhileCondition() + + OpDoWhileLoop( + condition = condition, + body = body, + isFalse = globalContext::isFalse + + ) + } + + "if" -> { + + if (EXPR_DEBUG_PRINT_ENABLED) { + print("parsing if...") + } + + val condition = parseExpressionOp(globalContext, blockContext = blockContext) + + val onTrue = parseBlock(context = blockContext) + + val onFalse = if (eatSequence("else")) { + parseBlock(context = blockContext) + } else null + + OpIfCondition( + condition = condition, + onTrue = onTrue, + onFalse = onFalse + ) + } + + "continue" -> { + if(BlockContext.Loop in blockContext){ + if (EXPR_DEBUG_PRINT_ENABLED) { + println("parsing loop continue") + } + OpContinue() + } else{ + throw SyntaxError("Illegal continue statement: no surrounding iteration statement") + } + } + + "break" -> { + if(BlockContext.Loop in blockContext){ + if (EXPR_DEBUG_PRINT_ENABLED) { + println("parsing loop break") + } + OpBreak() + } else{ + throw SyntaxError("Illegal break statement") + } + } + + "return" -> { + if (BlockContext.Function in blockContext) { + val expr = parseExpressionOp(globalContext,blockContext = blockContext) + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making return with $expr") + } + OpReturn(expr) + } else { + throw SyntaxError("Illegal return statement") + } + } + + "try" -> { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making try $expr") + } + parseTryCatch(blockContext) + } + + else -> { + val args = parseFunctionArgs(func) + + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making fun $func") + } + + return when (context) { + is InterpretationContext -> context.interpret(func, args) + ?: run { + if (args != null && func != null) { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("parsed call for defined function $func") + } + OpFunctionExec(func, null, args) + } else null + } + ?: run { + if (args == null && func != null) { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making GetVariable $func...") + } + OpGetVariable(name = func, receiver = null) + } else { + null + } + } + ?: unresolvedReference( + ref = func ?: "null", + obj = context::class.simpleName + ?.substringAfter("Op") + ?.substringBefore("Context") + ) + + else -> { + globalContext.interpret(context, func, args) + ?: run { + if (args != null && func != null) { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("parsed call for function $func with receiver $context") + } + OpFunctionExec( + name = func, + receiver = context, + parameters = args + ) + } else null + } + ?: run { + if (args == null && func != null) { + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making GetVariable $func with receiver $context... ") + } + OpGetVariable(name = func, receiver = context) + } else { + null + } + } + ?: unresolvedReference(func ?: "null") + } + } + } + } + } + + private fun parseWhileCondition(): Expression { + check(eat('(')) { + "Missing while loop condition" + } + + val condition = parseExpressionOp(globalContext, blockContext = emptyList()) + + check(eat(')')) { + "Missing closing ')' in loop condition" + } + return condition + } + + private fun parseTryCatch(blockContext : List) : Expression { + val tryBlock = parseBlock(requireBlock = true, context = blockContext) + val catchBlock = if (eatSequence("catch")) { + + if (eat('(')) { + val start = pos + while (!eat(')') && pos < expr.length) { + //nothing + } + expr.substring(start, pos).trim() to parseBlock( + scoped = false, + requireBlock = true, + context = blockContext + ) + } else { + null to parseBlock(requireBlock = true, context = blockContext) + } + } else null + + val finallyBlock = if (eatSequence("finally")) { + parseBlock(requireBlock = true, context = blockContext) + } else null + + return OpTryCatch( + tryBlock = tryBlock, + catchVariableName = catchBlock?.first, + catchBlock = catchBlock?.second, + finallyBlock = finallyBlock + ) + } + + private fun parseForLoop(parentBlockContext: List): Expression { + check(eat('(')) + + val assign = if (eat(';')) null else parseAssignment(globalContext, emptyList()) + check(assign is OpAssign?) + if (assign != null) { + check(eat(';')) + } + println(assign) + val comparison = if (eat(';')) null else parseAssignment(globalContext, emptyList()) + if (comparison != null) { + check(eat(';')) + } + println(comparison) + val increment = if (eat(')')) null else parseAssignment(globalContext, emptyList()) + + println(increment) + if (increment != null) { + check(eat(')')) + } + + val body = parseBlock(scoped = false, context = parentBlockContext + BlockContext.Loop) + + return OpForLoop( + assignment = assign, + increment = increment, + comparison = comparison, + isFalse = globalContext::isFalse, + body = body + ) + } + + private fun parseFunctionDefinition(name: String? = null, blockContext: List): Expression { + + val start = pos + + val actualName = name ?: run { + + while (ch != '(') { + nextChar() + } + + expr.substring(start, pos).trim() + } + if (EXPR_DEBUG_PRINT_ENABLED) { + println("making defined function $actualName") + } + + val args = parseFunctionArgs(actualName).let { args -> + args?.map { + when (it) { + is OpGetVariable -> FunctionParam(name = it.name, default = null) + is OpAssign -> FunctionParam( + name = it.variableName, + default = it.assignableValue + ) + + else -> error("Invalid function declaration at $start") + } + } + } + + checkNotNull(args) { + "Missing function args" + } + + + check(nextCharIs('{'::equals)) { + "Missing function body at $pos" + } + + + val block = parseBlock( + scoped = false, + context = blockContext + BlockContext.Function + ) + + val function = OpFunction( + name = actualName, + parameters = args, + body = block + ) + + return OpAssign( + type = VariableType.Const, + variableName = actualName, + assignableValue = OpConstant(function), + merge = null + ) + } + + private fun parseBlock( + scoped: Boolean = true, + requireBlock: Boolean = false, + context: List + ): Expression { + + val oldRegistrar = blockFunctionRegistrar + try { + blockFunctionRegistrar = mutableListOf() + val list = buildList { + if (eat('{')) { + while (!eat('}') && pos < expr.length) { + add(parseAssignment(globalContext, context)) + eat(';') + } + } else { + if (requireBlock) { + throw SyntaxError("Unexpected token at $pos: block start was expected") + } + add(parseAssignment(globalContext, context)) + } + } + return OpBlock(blockFunctionRegistrar + list, scoped) + } finally { + blockFunctionRegistrar = oldRegistrar + } + } +} + + +@OptIn(ExperimentalContracts::class) +internal fun checkArgsNotNull(args : List<*>?, func : String) { + contract { + returns() implies (args != null) + } + checkNotNull(args){ + "$func call was missing" + } +} + +@OptIn(ExperimentalContracts::class) +internal fun checkArgs(args : List<*>?, count : Int, func : String) { + contract { + returns() implies (args != null) + } + checkNotNull(args){ + "$func call was missing" + } + require(args.size == count){ + "$func takes $count arguments, but ${args.size} got" + } +} + +private val funMap = (('a'..'z').toList() + ('A'..'Z').toList() + '$' + '_' ).associateBy { it } + +private fun Char.isFun() = isDigit() || funMap[this] != null + + +private enum class NumberFormat( + val radix : Int, + val alphabet : String, + val prefix : Char? +) { + Dec(10, ".0123456789", null), + Hex(16, "0123456789abcdef", 'x'), + Oct(8, "01234567", 'o'), + Bin(2, "01", 'b') +} + +private val NumberFormatIndicators = NumberFormat.entries.mapNotNull { it.prefix } + +private val reservedKeywords = setOf( + "function","return","do","while","for" +) + +private fun isReserved(keyword : String) = keyword in reservedKeywords diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/Object.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/Object.kt new file mode 100644 index 00000000..bd074664 --- /dev/null +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/ecmascript/Object.kt @@ -0,0 +1,65 @@ +package io.github.alexzhirkevich.skriptie.ecmascript + +import io.github.alexzhirkevich.skriptie.Expression +import io.github.alexzhirkevich.skriptie.ScriptContext +import io.github.alexzhirkevich.skriptie.common.FunctionParam +import io.github.alexzhirkevich.skriptie.common.OpFunction +import io.github.alexzhirkevich.skriptie.common.OpGetVariable + +internal class Object( + private val name : String, + private val map : MutableMap = mutableMapOf() +) : MutableMap by map { + override fun toString(): String { + return if (name.isNotBlank()){ + "[object $name]" + } else { + "[object]" + } + } +} + + +internal sealed interface ObjectScope { + + infix fun String.eq(value: Any?) + + fun String.func( + vararg args: FunctionParam, + body: (args: List>) -> Expression + ) + + fun String.func( + vararg args: String, + params: (String) -> FunctionParam = { FunctionParam(it) }, + body: (args: List>) -> Expression + ) { + func( + args = args.map(params).toTypedArray(), + body = body + ) + } +} + +private class ObjectScopeImpl(name: String) : ObjectScope { + val o = Object(name) + + override fun String.func( + vararg args: FunctionParam, + body: (args: List>) -> Expression + ) { + this eq OpFunction( + this, + parameters = args.toList(), + body = body(args.map { OpGetVariable(it.name, null) }) + ) + } + + override fun String.eq(value: Any?) { + o[this] = value + } +} + +internal fun obj(name: String, builder : ObjectScope.() -> Unit) : Object { + return ObjectScopeImpl(name).also(builder).o +} \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JS.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JS.kt deleted file mode 100644 index 576c37c4..00000000 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JS.kt +++ /dev/null @@ -1,25 +0,0 @@ -package io.github.alexzhirkevich.skriptie.javascript - -import io.github.alexzhirkevich.skriptie.Script -import io.github.alexzhirkevich.skriptie.ScriptEngine -import io.github.alexzhirkevich.skriptie.ecmascript.EcmascriptInterpreter - -public class JS( - override val context: JSScriptContext = JSScriptContext(), - private val globalContext: JsGlobalContext = JsGlobalContext(), - private val extensionContext : JsExtensionContext = JsExtensionContext(), -) : ScriptEngine { - - override fun compile(script: String): Script { - return EcmascriptInterpreter( - expr = script, - scriptContext = context, - globalContext = globalContext, - extensionContext = extensionContext - ).interpret() - } - - override fun reset() { - context.reset() - } -} \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JsGlobalContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSInterpretationContext.kt similarity index 55% rename from skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JsGlobalContext.kt rename to skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSInterpretationContext.kt index 7e8b966d..06371ddc 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JsGlobalContext.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSInterpretationContext.kt @@ -1,98 +1,111 @@ package io.github.alexzhirkevich.skriptie.javascript -import io.github.alexzhirkevich.skriptie.javascript.math.JsInfinity -import io.github.alexzhirkevich.skriptie.javascript.math.JsMath import io.github.alexzhirkevich.skriptie.Expression import io.github.alexzhirkevich.skriptie.GlobalContext +import io.github.alexzhirkevich.skriptie.argAt import io.github.alexzhirkevich.skriptie.common.fastMap -import kotlin.math.min +import io.github.alexzhirkevich.skriptie.ecmascript.checkArgs +import io.github.alexzhirkevich.skriptie.javascript.iterable.JsIndexOf +import io.github.alexzhirkevich.skriptie.javascript.number.JsNumberContext +import io.github.alexzhirkevich.skriptie.javascript.string.JsStringContext +import kotlin.math.absoluteValue -public open class JsGlobalContext : GlobalContext { +public open class JSInterpretationContext : GlobalContext { override fun interpret( callable: String?, args: List>? + ): Expression? = null + + override fun interpret( + parent: Expression, + op: String?, + args: List>? ): Expression? { - return when (callable) { - "Infinity" -> JsInfinity - "Math" -> JsMath - else -> null + return if (args != null){ + when (op) { + "toString" -> Expression { parent(it).toString() } + "indexOf", "lastIndexOf" -> { + checkArgs(args, 1, op) + JsIndexOf( + value = parent, + search = args.argAt(0), + last = op == "lastIndexOf" + ) + } + + else -> JsNumberContext.interpret(parent, op, args) + ?: JsStringContext.interpret(parent, op, args) + } + } else { + JsNumberContext.interpret(parent, op, args) + ?: JsStringContext.interpret(parent, op, args) } } + override fun isFalse(a: Any?): Boolean { + return a == null + || a == false + || a is Number && a.toDouble() == 0.0 + || a is CharSequence && a.isEmpty() + || a is Unit + } + override fun sum(a: Any?, b: Any?): Any? { - return jssum(a?.validateJsNumber(), b?.validateJsNumber()) + return jssum( + a?.numberOrThis(false), + b?.numberOrThis(false) + ) } override fun sub(a: Any?, b: Any?): Any? { - return jssub(a?.validateJsNumber(), b?.validateJsNumber()) + return jssub(a?.numberOrThis(), b?.numberOrThis()) } override fun mul(a: Any?, b: Any?): Any? { - return jsmul(a?.validateJsNumber(), b?.validateJsNumber()) + return jsmul(a?.numberOrThis(), b?.numberOrThis()) } override fun div(a: Any?, b: Any?): Any? { - return jsdiv(a?.validateJsNumber(), b?.validateJsNumber()) + return jsdiv(a?.numberOrThis(), b?.numberOrThis()) } override fun mod(a: Any?, b: Any?): Any { - return jsmod(a?.validateJsNumber(), b?.validateJsNumber()) + return jsmod(a?.numberOrThis(), b?.numberOrThis()) } override fun inc(a: Any?): Any { - return jsinc(a?.validateJsNumber()) + return jsinc(a?.numberOrThis()) } override fun dec(a: Any?): Any { - return jsdec(a?.validateJsNumber()) + return jsdec(a?.numberOrThis()) } override fun neg(a: Any?): Any { - return jsneg(a?.validateJsNumber()) + return jsneg(a?.numberOrThis()) + } + + override fun pos(a: Any?): Any { + return jspos(a?.numberOrThis()) } } private fun jssum(a : Any?, b : Any?) : Any? { + val a = if (a is List<*>) + a.joinToString(",") + else a + val b = if (b is List<*>) + b.joinToString(",") + else b return when { a == null && b == null -> 0L a == null && b is Number || a is Number && b == null -> a ?: b b is Unit || a is Unit -> Double.NaN a is Long && b is Long -> a + b a is Number && b is Number -> a.toDouble() + b.toDouble() - a is List<*> && b is List<*> -> { - a as List - b as List - - List(min(a.size, b.size)) { - a[it].toDouble() + b[it].toDouble() - } - } - - a is List<*> && b is Number -> { - if (a is MutableList<*>) { - a as MutableList - a[0] = a[0].toDouble() + b.toDouble() - a - } else { - listOf((a as List).first().toDouble() + b.toDouble()) + a.drop(1) - } - } - - a is Number && b is List<*> -> { - if (b is MutableList<*>) { - b as MutableList - b[0] = b[0].toDouble() + a.toDouble() - b - } else { - listOf(a.toDouble() + (b as List).first().toDouble()) + b.drop(1) - } - } - - a is CharSequence || b is CharSequence -> a.toString() + b.toString() - - else -> error("Cant calculate the sum of $a and $b") + else -> a.toString() + b.toString() } } @@ -103,42 +116,7 @@ private fun jssub(a : Any?, b : Any?) : Any? { a is Long? && b is Long? -> (a ?: 0L) - (b ?: 0L) a is Double? && b is Double? ->(a ?: 0.0) - (b ?: 0.0) a is Number? && b is Number? ->(a?.toDouble() ?: 0.0) - (b?.toDouble() ?: 0.0) - a is List<*> && b is List<*> -> { - a as List - b as List - List(min(a.size, b.size)) { - a[it].toDouble() - b[it].toDouble() - } - } - a is CharSequence || b is CharSequence -> { - stringMath(a?.toString(), b?.toString(), Long::minus, Double::minus) - } - else -> error("Cant subtract $b from $a") - } -} - -private fun stringMath( - a : String?, - b : String?, - long : (Long, Long) -> Long, - double: (Double,Double) -> Double -) : Any { - val sa = a ?: "0" - val sb = b ?: "0" - - val la = sa.toLongOrNull() - val lb = sb.toLongOrNull() - - return if (la != null && lb != null) { - long(la, lb) - } else { - val da = sa.toDoubleOrNull() - val db = sb.toDoubleOrNull() - if (da != null && db != null) { - double(da, db) - } else { - Double.NaN - } + else -> Double.NaN } } @@ -160,10 +138,7 @@ private fun jsmul(a : Any?, b : Any?) : Any? { val af = a.toDouble() b.fastMap { it.toDouble() * af } } - a is CharSequence || b is CharSequence -> { - stringMath(a.toString(), b.toString(), Long::times, Double::times) - } - else -> error("Cant multiply $a by $b") + else -> Double.NaN } } @@ -189,21 +164,18 @@ private fun jsdiv(a : Any?, b : Any?) : Any { a.fastMap { it.toDouble() / bf } } - a is CharSequence || b is CharSequence -> { - stringMath(a?.toString(), b?.toString(), Long::div, Double::div) - } - - else -> error("Cant divide $a by $b") + else -> Double.NaN } } private fun jsmod(a : Any?, b : Any?) : Any { return when { + b == null || a == Unit || b == Unit -> Double.NaN + (b as? Number)?.toDouble()?.absoluteValue?.let { it < Double.MIN_VALUE } == true -> Double.NaN a == null -> 0L - b == null -> Double.NaN a is Long && b is Long -> a % b a is Number && b is Number -> a.toDouble() % b.toDouble() - else -> error("Can't get mod of $a and $b") + else -> Double.NaN } } @@ -214,7 +186,7 @@ private fun jsinc(v : Any?) : Any { is Long -> v + 1 is Double -> v + 1 is Number -> v.toDouble() + 1 - else -> error("can't increment $v") + else -> Double.NaN } } @@ -224,7 +196,7 @@ private fun jsdec(v : Any?) : Any { is Long -> v - 1 is Double -> v - 1 is Number -> v.toDouble() - 1 - else -> error("can't decrement $v") + else -> Double.NaN } } @@ -238,11 +210,20 @@ private fun jsneg(v : Any?) : Any { v.fastMap { -it.toDouble() } } - else -> error("Cant apply unary minus to $v") + else -> Double.NaN } } -internal fun Any.validateJsNumber() = when(this) { +private fun jspos(v : Any?) : Any { + return when (v) { + null -> 0 + is Number -> v + else -> Double.NaN + } +} + + +public fun Any.numberOrNull(withNaNs : Boolean = true) : Any? = when(this) { is Byte -> toLong() is UByte -> toLong() is Short -> toLong() @@ -251,6 +232,19 @@ internal fun Any.validateJsNumber() = when(this) { is UInt -> toLong() is ULong -> toLong() is Float -> toDouble() - else -> this + is Long, is Double -> this + is String -> if (withNaNs) { + toLongOrNull() ?: toDoubleOrNull() + } else null + is List<*> -> { + if (withNaNs) { + singleOrNull()?.numberOrNull(withNaNs) + } else{ + null + } + } + else -> null } + +public fun Any.numberOrThis(withMagic : Boolean = true) : Any = numberOrNull(withMagic) ?: this diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSScriptContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSScriptContext.kt index 6de11cd8..fda45916 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSScriptContext.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JSScriptContext.kt @@ -1,7 +1,22 @@ package io.github.alexzhirkevich.skriptie.javascript -import io.github.alexzhirkevich.skriptie.BaseScriptContext -import io.github.alexzhirkevich.skriptie.GlobalContext -import io.github.alexzhirkevich.skriptie.ScriptContext +import io.github.alexzhirkevich.skriptie.EcmascriptContext +import io.github.alexzhirkevich.skriptie.VariableType +import io.github.alexzhirkevich.skriptie.javascript.math.JsMath -public open class JSScriptContext() : BaseScriptContext() \ No newline at end of file +public open class JSScriptContext : EcmascriptContext() { + + init { + recreate() + } + + override fun reset() { + super.reset() + recreate() + } + + private fun recreate(){ + variables["Math"] = VariableType.Const to JsMath() + variables["Infinity"] = VariableType.Const to Double.POSITIVE_INFINITY + } +} \ No newline at end of file diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JsExtensionContext.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JsExtensionContext.kt deleted file mode 100644 index ae1b8567..00000000 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/JsExtensionContext.kt +++ /dev/null @@ -1,39 +0,0 @@ -package io.github.alexzhirkevich.skriptie.javascript - -import io.github.alexzhirkevich.skriptie.javascript.number.JsNumberContext -import io.github.alexzhirkevich.skriptie.javascript.string.JsStringContext -import io.github.alexzhirkevich.skriptie.Expression -import io.github.alexzhirkevich.skriptie.ExtensionContext -import io.github.alexzhirkevich.skriptie.argAt -import io.github.alexzhirkevich.skriptie.ecmascript.checkArgs -import io.github.alexzhirkevich.skriptie.javascript.iterable.JsIndexOf - -public open class JsExtensionContext: ExtensionContext { - - override fun interpret( - parent: Expression, - op: String?, - args: List>? - ): Expression? { - return if (args != null){ - when (op) { - "toString" -> Expression { parent(it).toString() } - "indexOf", "lastIndexOf" -> { - checkArgs(args, 1, op) - JsIndexOf( - value = parent, - search = args.argAt(0), - last = op == "lastIndexOf" - ) - } - - else -> JsNumberContext.interpret(parent, op, args) - ?: JsStringContext.interpret(parent, op, args) - } - } else { - JsNumberContext.interpret(parent, op, args) - ?: JsStringContext.interpret(parent, op, args) - } - } -} - diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/math/JsMath.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/math/JsMath.kt index 1c7c525f..55ae26e9 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/math/JsMath.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/math/JsMath.kt @@ -1,161 +1,232 @@ package io.github.alexzhirkevich.skriptie.javascript.math import io.github.alexzhirkevich.skriptie.Expression -import io.github.alexzhirkevich.skriptie.InterpretationContext +import io.github.alexzhirkevich.skriptie.ScriptContext import io.github.alexzhirkevich.skriptie.argAt -import io.github.alexzhirkevich.skriptie.common.OpConstant +import io.github.alexzhirkevich.skriptie.common.FunctionParam import io.github.alexzhirkevich.skriptie.common.fastMap import io.github.alexzhirkevich.skriptie.common.fastSumBy +import io.github.alexzhirkevich.skriptie.ecmascript.Object import io.github.alexzhirkevich.skriptie.ecmascript.checkArgs +import io.github.alexzhirkevich.skriptie.ecmascript.obj import io.github.alexzhirkevich.skriptie.javascript.JSScriptContext -import io.github.alexzhirkevich.skriptie.javascript.validateJsNumber +import io.github.alexzhirkevich.skriptie.javascript.numberOrThis import kotlin.math.* import kotlin.random.Random -internal val JsInfinity = OpConstant(Double.POSITIVE_INFINITY) - -internal object JsMath : InterpretationContext { - - override fun invoke(context: JSScriptContext): Any = this - - override fun interpret( - callable: String?, - args: List>? - ): Expression { - return if (args == null){ - interpretVar(callable) - } else { - interpretFun(callable, args) - } - } - - private fun interpretVar( - op: String?, - ): Expression { - return when (op) { - "PI" -> PI - "E" -> E - "LN10" -> LN10 - "LN2" -> LN2 - "LOG10E" -> LOG10E - "LOG2E" -> LOG2E - "SQRT1_2" -> SQRT1_2 - "SQRT2" -> SQRT2 - else -> OpConstant(Unit) +internal fun JsMath() : Object { + + return obj("Math") { + + "PI" eq PI + "E" eq E + "LN10" eq 2.302585092994046 + "LN2" eq 0.6931471805599453 + "LOG10E" eq 0.4342944819032518 + "LOG2E" eq 1.4426950408889634 + "SQRT1_2" eq 0.7071067811865476 + "SQRT2" eq 1.4142135623730951 + + "abs".func("x") { op1(it, ::acos) } + "asoc".func("x") { op1(it, ::acos) } + "asoch".func("x") { op1(it, ::acosh) } + "asin".func("x") { op1(it, ::asin) } + "asinh".func("x") { op1(it, ::asinh) } + "atan".func("x") { op1(it, ::atan) } + "atan2".func("y", "x") { op2(it, ::atan2) } + "atanh".func("x") { op1(it, ::atanh) } + "cbrt".func("x") { op1(it, ::cbrt) } + "ceil".func("x") { op1(it, ::ceil) } + "cos".func("x") { op1(it, ::cos) } + "cosh".func("x") { op1(it, ::cosh) } + "exp".func("x") { op1(it, ::exp) } + "expm1".func("x") { op1(it, ::expm1) } + "floor".func("x") { op1(it, ::floor) } + "hypot".func( + "values", + params = { FunctionParam(it, isVararg = true) } + ) { opVararg(it, ::hypotN) } + "imul".func("x", "y",) { op2(it, ::imul) } + "log".func("x") { op1(it, ::ln) } + "log10".func("x") { op1(it, ::log10) } + "log1p ".func("x") { op1(it, ::ln1p) } + "log2".func("x") { op1(it, ::log2) } + "max".func( + "values", + params = { FunctionParam(it, isVararg = true) } + ) { + opVararg(it, List::max) } + "min".func( + "values", + params = { FunctionParam(it, isVararg = true) } + ) { opVararg(it, List::min) } + "pow".func("x", "y",) { op2(it, Double::pow) } + "random".func("x") { Expression { Random.nextDouble() } } + "round".func("x") { op1(it, ::round) } + "sign".func("x") { op1(it, ::sign) } + "sin".func("x") { op1(it, ::sin) } + "sinh".func("x") { op1(it, ::sinh) } + "sqrt".func("x") { op1(it, ::sqrt) } + "tan".func("x") { op1(it, ::tan) } + "tanh".func("x") { op1(it, ::tanh) } + "trunc".func("x") { op1(it, ::truncate) } } - private fun interpretFun( - op: String?, - args: List> - ): Expression { - return when (op) { - "abs" -> op1(args, ::abs, op) - "asoc" -> op1(args, ::acos, op) - "asoch" -> op1(args, ::acosh, op) - "asin" -> op1(args, ::asin, op) - "asinh" -> op1(args, ::asinh, op) - "atan" -> op1(args, ::atan, op) - "atan2" -> op2(args, ::atan2, op) - "atanh" -> op1(args, ::atanh, op) - "cbrt" -> op1(args, ::cbrt, op) - "ceil" -> op1(args, ::ceil, op) - "cos" -> op1(args, ::cos, op) - "cosh" -> op1(args, ::cosh, op) - "exp" -> op1(args, ::exp, op) - "expm1" -> op1(args, ::expm1, op) - "floor" -> op1(args, ::floor, op) - "hypot" -> opN(args, ::hypotN, op) - "imul" -> op2(args, ::imul, op) - "log" -> op2(args, ::log, op) - "log10" -> op1(args, ::log10, op) - "log1p " -> op1(args, ::ln1p, op) - "log2" -> op1(args, ::log2, op) - "max" -> opN(args, List::max, op) - "min" -> opN(args, List::min, op) - "pow" -> op2(args, Double::pow, op) - "random" -> Expression { Random.nextDouble() } - "round" -> op1(args, Double::roundToInt, op) - "sign" -> op1(args, Double::sign, op) - "sin" -> op1(args, ::sin, op) - "sinh" -> op1(args, ::sinh, op) - "sqrt" -> op1(args, ::sqrt, op) - "tan" -> op1(args, ::tan, op) - "tanh" -> op1(args, ::tanh, op) - "trunc" -> op1(args, ::truncate, op) - - else -> OpConstant(Unit) +} + +//internal object JsMath : InterpretationContext { +// +// override fun invoke(context: JSScriptContext): Any = this +// +// override fun interpret( +// callable: String?, +// args: List>? +// ): Expression { +// return if (args == null){ +// interpretVar(callable) +// } else { +// interpretFun(callable, args) +// } +// } +// +// private fun interpretVar( +// op: String?, +// ): Expression { +// return when (op) { +// "PI" -> PI +// "E" -> E +// "LN10" -> LN10 +// "LN2" -> LN2 +// "LOG10E" -> LOG10E +// "LOG2E" -> LOG2E +// "SQRT1_2" -> SQRT1_2 +// "SQRT2" -> SQRT2 +// else -> OpConstant(Unit) +// } +// } +// private fun interpretFun( +// op: String?, +// args: List> +// ): Expression { +// return when (op) { +// "abs" -> op1(args, ::abs, op) +// "asoc" -> op1(args, ::acos, op) +// "asoch" -> op1(args, ::acosh, op) +// "asin" -> op1(args, ::asin, op) +// "asinh" -> op1(args, ::asinh, op) +// "atan" -> op1(args, ::atan, op) +// "atan2" -> op2(args, ::atan2, op) +// "atanh" -> op1(args, ::atanh, op) +// "cbrt" -> op1(args, ::cbrt, op) +// "ceil" -> op1(args, ::ceil, op) +// "cos" -> op1(args, ::cos, op) +// "cosh" -> op1(args, ::cosh, op) +// "exp" -> op1(args, ::exp, op) +// "expm1" -> op1(args, ::expm1, op) +// "floor" -> op1(args, ::floor, op) +// "hypot" -> opN(args, ::hypotN, op) +// "imul" -> op2(args, ::imul, op) +// "log" -> op2(args, ::log, op) +// "log10" -> op1(args, ::log10, op) +// "log1p " -> op1(args, ::ln1p, op) +// "log2" -> op1(args, ::log2, op) +// "max" -> opN(args, List::max, op) +// "min" -> opN(args, List::min, op) +// "pow" -> op2(args, Double::pow, op) +// "random" -> Expression { Random.nextDouble() } +// "round" -> op1(args, Double::roundToInt, op) +// "sign" -> op1(args, Double::sign, op) +// "sin" -> op1(args, ::sin, op) +// "sinh" -> op1(args, ::sinh, op) +// "sqrt" -> op1(args, ::sqrt, op) +// "tan" -> op1(args, ::tan, op) +// "tanh" -> op1(args, ::tanh, op) +// "trunc" -> op1(args, ::truncate, op) +// +// else -> OpConstant(Unit) +// } +// } +// +// +// +// +// private val PI = OpConstant(kotlin.math.PI) +// private val E = OpConstant(kotlin.math.E) +// private val LN10 = OpConstant(2.302585092994046) +// private val LN2 = OpConstant(0.6931471805599453) +// private val LOG10E = OpConstant(0.4342944819032518) +// private val LOG2E = OpConstant(1.4426950408889634) +// private val SQRT1_2 = OpConstant(0.7071067811865476) +// private val SQRT2 = OpConstant(1.4142135623730951) +//} + +private fun op1( + args: List>, + func: (Double) -> Number +): Expression { + checkArgs(args, 1, "") + + val a = args.argAt(0) + + return Expression { + var a = a(it)?.numberOrThis() ?: 0.0 + + if (a !is Number){ + a = a.toString().toDoubleOrNull() ?: return@Expression Double.NaN } - } - private fun op1( - args: List>, - func: (Double) -> Number, - name: String - ): Expression { - checkArgs(args, 1, name) - - val a = args.argAt(0) - return Expression { - val a = a(it)?.validateJsNumber() ?: 0.0 - require(a is Number) { - "Can't get Math.$name of $a" - } + if (a is Number){ func(a.toDouble()) + } else { + Double.NaN } } +} - private fun op2( - args: List>, - func: (Double, Double) -> Number, - name: String - ): Expression { - checkArgs(args, 2, name) - - val a = args.argAt(0) - val b = args.argAt(1) - return Expression { - val a = a(it)?.validateJsNumber() ?: 0 - val b = b(it)?.validateJsNumber() ?: 0 - require(a is Number && b is Number) { - "Can't get Math.$name of ($a,$b)" - } +private fun op2( + args: List>, + func: (Double, Double) -> Number, +): Expression { + checkArgs(args, 2, "") + + val a = args.argAt(0) + val b = args.argAt(1) + return Expression { + var a = a(it)?.numberOrThis() ?: 0.0 + var b = b(it)?.numberOrThis() ?: 0.0 + if (a !is Number){ + a = a.toString().toDoubleOrNull() ?: return@Expression Double.NaN + } + if (b !is Number){ + b = b.toString().toDoubleOrNull() ?: return@Expression Double.NaN + } + if (a is Number && b is Number){ func(a.toDouble(), b.toDouble()) + } else { + Double.NaN } } +} - private fun opN( - args: List>, - func: (List) -> Number, - name: String - ): Expression { - check(args.isNotEmpty()){ - "Math.$name must have at least 1 argument" - } - return Expression { context -> - - val a = args.fastMap { - val n = it(context)?.validateJsNumber().also { - check(it is Number?) { - "Illegal arguments for Math.$name" - } - } as Number? +private fun opVararg( + args: List>, + func: (List) -> Number, +): Expression { + check(args.isNotEmpty()){ - n?.toDouble() ?: 0.0 + } + return Expression { context -> + val a = (args[0].invoke(context) as List<*>).fastMap { + when (it){ + is Number -> it.toDouble() + null -> 0.0 + else -> { + it.toString().toDoubleOrNull() ?: return@Expression Double.NaN + } } - - func(a) } + func(a) } - - - private val PI = OpConstant(kotlin.math.PI) - private val E = OpConstant(kotlin.math.E) - private val LN10 = OpConstant(2.302585092994046) - private val LN2 = OpConstant(0.6931471805599453) - private val LOG10E = OpConstant(0.4342944819032518) - private val LOG2E = OpConstant(1.4426950408889634) - private val SQRT1_2 = OpConstant(0.7071067811865476) - private val SQRT2 = OpConstant(1.4142135623730951) } private fun hypotN(args : List): Double { diff --git a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/number/JsToPrecision.kt b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/number/JsToPrecision.kt index d8049bd5..0297fb8b 100644 --- a/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/number/JsToPrecision.kt +++ b/skriptie/src/commonMain/kotlin/io/github/alexzhirkevich/skriptie/javascript/number/JsToPrecision.kt @@ -2,9 +2,8 @@ package io.github.alexzhirkevich.skriptie.javascript.number import io.github.alexzhirkevich.skriptie.Expression import io.github.alexzhirkevich.skriptie.common.checkNotEmpty -import io.github.alexzhirkevich.skriptie.common.unresolvedReference import io.github.alexzhirkevich.skriptie.javascript.JSScriptContext -import io.github.alexzhirkevich.skriptie.javascript.validateJsNumber +import io.github.alexzhirkevich.skriptie.javascript.numberOrThis import kotlin.math.pow import kotlin.math.roundToInt import kotlin.math.roundToLong @@ -14,9 +13,9 @@ internal fun JsToPrecision( digits : Expression? = null ) = Expression { context -> - val number = checkNotEmpty(number(context)?.validateJsNumber() as? Number?).toDouble() + val number = checkNotEmpty(number(context)?.numberOrThis() as? Number?).toDouble() - val digits = (digits?.invoke(context)?.validateJsNumber() as Number?)?.toInt() + val digits = (digits?.invoke(context)?.numberOrThis() as Number?)?.toInt() ?.takeIf { it > 0 } ?: return@Expression number @@ -27,9 +26,9 @@ internal fun JsToFixed( number : Expression, digits : Expression? ) = Expression { context -> - val number = checkNotEmpty(number(context)?.validateJsNumber() as? Number?).toDouble() + val number = checkNotEmpty(number(context)?.numberOrThis() as? Number?).toDouble() - val digits = (digits?.invoke(context)?.validateJsNumber() as Number?)?.toInt() ?: 0 + val digits = (digits?.invoke(context)?.numberOrThis() as Number?)?.toInt() ?: 0 if (digits == 0) { return@Expression number.roundToLong().toString() diff --git a/skriptie/src/commonTest/kotlin/js/AssignExpressionTest.kt b/skriptie/src/commonTest/kotlin/js/AssignExpressionTest.kt deleted file mode 100644 index 12d74df1..00000000 --- a/skriptie/src/commonTest/kotlin/js/AssignExpressionTest.kt +++ /dev/null @@ -1,26 +0,0 @@ -//package js -// -//import expressions.assertExprReturns -//import expressions.ret -//import io.github.alexzhirkevich.compottie.internal.animation.Vec2 -//import kotlin.test.Test -// -// -//class AssignExpressionTest { -// -// @Test -// fun add_sub_mull_div_assign() { -// -// "var $ret = 13; $ret += 17".assertExprReturns(30f) -// "var $ret = 56; $ret -=17".assertExprReturns(39f) -// "var $ret = 5; $ret *=2".assertExprReturns(10f) -// "var $ret = 13; $ret *= -2*2".assertExprReturns(-52f) -// "var $ret = 144; $ret /=6".assertExprReturns(24f) -// -// "var $ret = []; $ret[0] = 5; $ret[1] = 10; $ret[(5-5)] += 10-3; $ret[5-4] += (4*2);" -// .assertExprReturns(Vec2(12f, 18f)) -// -// "var $ret = []\n $ret[0] = 5\n $ret[1] = 10\n $ret[(5-5)] += 10-3\n $ret[5-4] += (4*2);" -// .assertExprReturns(Vec2(12f, 18f)) -// } -//} diff --git a/skriptie/src/commonTest/kotlin/js/AssignTest.kt b/skriptie/src/commonTest/kotlin/js/AssignTest.kt new file mode 100644 index 00000000..1bc2309e --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/AssignTest.kt @@ -0,0 +1,23 @@ +package js + +import kotlin.test.Test + + +class AssignTest { + + @Test + fun add_sub_mull_div_assign() { + + "var x = 13; x += 17".eval().assertEqualsTo(30L) + "var x = 56; x -=17".eval().assertEqualsTo(39L) + "var x = 5; x *=2".eval().assertEqualsTo(10L) + "var x = 13; x *= -2*2".eval().assertEqualsTo(-52L) + "var x = 144; x /=6".eval().assertEqualsTo(24L) + + "var x = []; x[0] = 5; x[1] = 10; x[(5-5)] += 10-3; x[5-4] += (4*2); x" + .eval().assertEqualsTo(listOf(12L, 18L)) + + "var x = []\n x[0] = 5\n x[1] = 10\n x[(5-5)] += 10-3\n x[5-4] += (4*2); x" + .eval().assertEqualsTo(listOf(12L, 18L)) + } +} diff --git a/skriptie/src/commonTest/kotlin/js/BooleanExpressionsTest.kt b/skriptie/src/commonTest/kotlin/js/BooleanExpressionsTest.kt deleted file mode 100644 index 6ba1e329..00000000 --- a/skriptie/src/commonTest/kotlin/js/BooleanExpressionsTest.kt +++ /dev/null @@ -1,60 +0,0 @@ -//package js -// -//import expressions.assertSimpleExprEquals -//import kotlin.test.Test -// -//class BooleanExpressionsTest { -// -// @Test -// fun and() { -// "true && true".assertSimpleExprEquals(true) -// "true && true && true".assertSimpleExprEquals(true) -// -// "false && false".assertSimpleExprEquals(false) -// "true && false".assertSimpleExprEquals(false) -// "false && true".assertSimpleExprEquals(false) -// -// "true && true && false".assertSimpleExprEquals(false) -// "false && true && true".assertSimpleExprEquals(false) -// } -// -// @Test -// fun or() { -// "true || true".assertSimpleExprEquals(true) -// "true || true || true".assertSimpleExprEquals(true) -// -// "true || false".assertSimpleExprEquals(true) -// "false || true".assertSimpleExprEquals(true) -// "false || false".assertSimpleExprEquals(false) -// -// "true || true || false".assertSimpleExprEquals(true) -// "false || true || true".assertSimpleExprEquals(true) -// "false || false || true".assertSimpleExprEquals(true) -// "false || false || false".assertSimpleExprEquals(false) -// } -// -// @Test -// fun and_or_order() { -// "true && true || false".assertSimpleExprEquals(true) -// "false || true && true".assertSimpleExprEquals(true) -// -// "(false && true) || (true && true)".assertSimpleExprEquals(true) -// "false && true || true && true".assertSimpleExprEquals(true) -// "(false && true || true) && true".assertSimpleExprEquals(true) -//// -// "true || false && false".assertSimpleExprEquals(true) -// "(true || false) && false".assertSimpleExprEquals(false) -// } -// -// @Test -// fun with_different_source() { -// "false || 1 == 1".assertSimpleExprEquals(true) -// "true && 1 == 2".assertSimpleExprEquals(false) -// -// "(1 == 2) || false || (1+1) == 2".assertSimpleExprEquals(true) -// "1 == 2 || false || (1+1) == 2".assertSimpleExprEquals(true) -// "1 == 1 && 2 == 2".assertSimpleExprEquals(true) -// -// "1 == 2 && 2 == 1 || 2 * 2 == 4".assertSimpleExprEquals(true) -// } -//} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/BooleansTest.kt b/skriptie/src/commonTest/kotlin/js/BooleansTest.kt new file mode 100644 index 00000000..d7496d48 --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/BooleansTest.kt @@ -0,0 +1,59 @@ +package js + +import kotlin.test.Test + +class BooleansTest { + + @Test + fun and() { + "true && true".eval().assertEqualsTo(true) + "true && true && true".eval().assertEqualsTo(true) + + "false && false".eval().assertEqualsTo(false) + "true && false".eval().assertEqualsTo(false) + "false && true".eval().assertEqualsTo(false) + + "true && true && false".eval().assertEqualsTo(false) + "false && true && true".eval().assertEqualsTo(false) + } + + @Test + fun or() { + "true || true".eval().assertEqualsTo(true) + "true || true || true".eval().assertEqualsTo(true) + + "true || false".eval().assertEqualsTo(true) + "false || true".eval().assertEqualsTo(true) + "false || false".eval().assertEqualsTo(false) + + "true || true || false".eval().assertEqualsTo(true) + "false || true || true".eval().assertEqualsTo(true) + "false || false || true".eval().assertEqualsTo(true) + "false || false || false".eval().assertEqualsTo(false) + } + + @Test + fun and_or_order() { + "true && true || false".eval().assertEqualsTo(true) + "false || true && true".eval().assertEqualsTo(true) + + "(false && true) || (true && true)".eval().assertEqualsTo(true) + "false && true || true && true".eval().assertEqualsTo(true) + "(false && true || true) && true".eval().assertEqualsTo(true) + + "true || false && false".eval().assertEqualsTo(true) + "(true || false) && false".eval().assertEqualsTo(false) + } + + @Test + fun with_different_source() { + "false || 1 == 1".eval().assertEqualsTo(true) + "true && 1 == 2".eval().assertEqualsTo(false) + + "(1 == 2) || false || (1+1) == 2".eval().assertEqualsTo(true) + "1 == 2 || false || (1+1) == 2".eval().assertEqualsTo(true) + "1 == 1 && 2 == 2".eval().assertEqualsTo(true) + + "1 == 2 && 2 == 1 || 2 * 2 == 4".eval().assertEqualsTo(true) + } +} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/ConditionExpressionTest.kt b/skriptie/src/commonTest/kotlin/js/ConditionExpressionTest.kt deleted file mode 100644 index ad1b37a1..00000000 --- a/skriptie/src/commonTest/kotlin/js/ConditionExpressionTest.kt +++ /dev/null @@ -1,39 +0,0 @@ -//package js -// -//import expressions.assertExprReturns -//import expressions.ret -//import kotlin.test.Test -// -//class ConditionExpressionTest { -// -// @Test -// fun if_with_else() { -// "var $ret = 1; if (true) { $ret = $ret+1 }".assertExprReturns(2f) -// "var $ret = 1; if (true) $ret+=1".assertExprReturns(2f) -// "var $ret = 1; if (1==1) { $ret = $ret+1 }".assertExprReturns(2f) -// "var $ret = 1; if (1==1) $ret +=1".assertExprReturns(2f) -// "var $ret = 1; if (true) { $ret = $ret+1;$ret = $ret+1 }".assertExprReturns(3f) -// "var $ret = 1; if (true) { $ret = $ret+1\n$ret +=1 }".assertExprReturns(3f) -// -// "var $ret = 1; if (false) { $ret = $ret } else { $ret = $ret+1 }" -// .assertExprReturns(2f) -// -// "var $ret = 1.0; if (false) $ret +=1" -// .assertExprReturns(1f) -// -// "var $ret = 1; if (1 != 1) $ret = 0 else { $ret = $ret+1;$ret = $ret+1 }" -// .assertExprReturns(3f) -// -// "var $ret = 1; if (!(1 == 1)) $ret = 0 else { $ret +=1;$ret = $ret+1 }" -// .assertExprReturns(3f) -// -// "var $ret = 0; if (true) { if (true) { $ret +=1 } }" -// .assertExprReturns(1f) -// -// "var $ret = 0; if (true) { if (false) { $ret = 0 } else { $ret += 1 } }" -// .assertExprReturns(1f) -// -// "var $ret = 0; if (true) if (false) $ret = 0 else $ret += 1 " -// .assertExprReturns(1f) -// } -//} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/ConditionTest.kt b/skriptie/src/commonTest/kotlin/js/ConditionTest.kt new file mode 100644 index 00000000..f897c7d7 --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/ConditionTest.kt @@ -0,0 +1,37 @@ +package js + +import kotlin.test.Test + +class ConditionTest { + + @Test + fun if_with_else() { + "var x = 1; if (true) { x = x+1 }; x".eval().assertEqualsTo(2L) + "var x = 1; if (true) x+=1; x".eval().assertEqualsTo(2L) + "var x = 1; if (1==1) { x = x+1 }; x".eval().assertEqualsTo(2L) + "var x = 1; if (1==1) x +=1; x".eval().assertEqualsTo(2L) + "var x = 1; if (true) { x = x+1;x = x+1 }; x".eval().assertEqualsTo(3L) + "var x = 1; if (true) { x = x+1\nx +=1 }; x".eval().assertEqualsTo(3L) + + "var x = 1; if (false) { x = x } else { x = x+1 }; x" + .eval().assertEqualsTo(2L) + + "var x = 1.0; if (false) x +=1; x" + .eval().assertEqualsTo(1.0) + + "var x = 1; if (1 != 1) x = 0 else { x = x+1;x = x+1 }; x" + .eval().assertEqualsTo(3L) + + "var x = 1; if (!(1 == 1)) x = 0 else { x +=1;x = x+1 }; x" + .eval().assertEqualsTo(3L) + + "var x = 0; if (true) { if (true) { x +=1 } }; x" + .eval().assertEqualsTo(1L) + + "var x = 0; if (true) { if (false) { x = 0 } else { x += 1 } }; x" + .eval().assertEqualsTo(1L) + + "var x = 0; if (true) if (false) x = 0 else x += 1; x " + .eval().assertEqualsTo(1L) + } +} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/CustomFunctionsTest.kt b/skriptie/src/commonTest/kotlin/js/CustomFunctionsTest.kt deleted file mode 100644 index 5a001ae3..00000000 --- a/skriptie/src/commonTest/kotlin/js/CustomFunctionsTest.kt +++ /dev/null @@ -1,78 +0,0 @@ -//package js -// -//import expressions.assertExprReturns -//import expressions.ret -//import kotlin.test.Test -// -//class CustomFunctionsTest { -// -// @Test -// fun creation() { -// """ -// var $ret; -// function test(a,b) { return sum(a,b) } -// $ret = test(1,2) -// """.trimIndent().assertExprReturns(3f) -// -// """ -// var $ret; -// function test(a,b) { -// return sum(a,b) -// } -// $ret = test(1,2) -// """.trimIndent().assertExprReturns(3f) -// -// """ -// var $ret; -// function test(a,b) -// { -// let x = b + 1 -// return sum(a,x) -// } -// $ret = test(1,2) -// """.trimIndent().assertExprReturns(4f) -// } -// -// -// @Test -// fun defaultArgs(){ -// """ -// var $ret; -// function test(a, b = 2) -// { -// return sum(a,b) -// } -// $ret = test(1) -// """.trimIndent().assertExprReturns(3f) -// -// """ -// var $ret; -// function test(a, b = 2) -// { -// return sum(a,b) -// } -// $ret = test(2,3) -// """.trimIndent().assertExprReturns(5f) -// -// """ -// var $ret; -// function test(a = 1, b = 2) -// { -// return sum(a,b) -// } -// $ret = test() -// """.trimIndent().assertExprReturns(3f) -// } -// -// @Test -// fun namedArgs(){ -// """ -// var $ret; -// function test(a, b) -// { -// return sum(a,b) -// } -// $ret = test(b = 2, a = 1) -// """.trimIndent().assertExprReturns(3f) -// } -//} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/DivTest.kt b/skriptie/src/commonTest/kotlin/js/DivTest.kt index 7726d269..cd55afae 100644 --- a/skriptie/src/commonTest/kotlin/js/DivTest.kt +++ b/skriptie/src/commonTest/kotlin/js/DivTest.kt @@ -7,45 +7,44 @@ class DivTest { @Test fun numbers() { - "26 / 2.0".runJs().assertEqualsTo(13.0) - "26 / 2".runJs().assertEqualsTo(13L) - "-26 / 2.0".runJs().assertEqualsTo(-13.0) - "-26 / 2".runJs().assertEqualsTo(-13L) - - "-26 / -2.0".runJs().assertEqualsTo(13.0) - "-26 / -2".runJs().assertEqualsTo(13L) - "-52 / -2.0 / 2".runJs().assertEqualsTo(13.0) - - "10/0".runJs().assertEqualsTo(Double.POSITIVE_INFINITY) - "10.0/0".runJs().assertEqualsTo(Double.POSITIVE_INFINITY) - "10/0.0".runJs().assertEqualsTo(Double.POSITIVE_INFINITY) + "26 / 2.0".eval().assertEqualsTo(13.0) + "26 / 2".eval().assertEqualsTo(13L) + "-26 / 2.0".eval().assertEqualsTo(-13.0) + "-26 / 2".eval().assertEqualsTo(-13L) + + "-26 / -2.0".eval().assertEqualsTo(13.0) + "-26 / -2".eval().assertEqualsTo(13L) + "-52 / -2.0 / 2".eval().assertEqualsTo(13.0) + + "10/0".eval().assertEqualsTo(Double.POSITIVE_INFINITY) + "10.0/0".eval().assertEqualsTo(Double.POSITIVE_INFINITY) + "10/0.0".eval().assertEqualsTo(Double.POSITIVE_INFINITY) } @Test - fun string(){ - "'30' / '3'".runJs().assertEqualsTo(10L) - "30.0 / '3'".runJs().assertEqualsTo(10.0) - "'30' / 3.0".runJs().assertEqualsTo(10.0) + fun string() { + "'30' / '3'".eval().assertEqualsTo(10L) + "30.0 / '3'".eval().assertEqualsTo(10.0) + "'30' / 3.0".eval().assertEqualsTo(10.0) - "'30' / 0".runJs().assertEqualsTo(Double.POSITIVE_INFINITY) - "'30' / 0.0".runJs().assertEqualsTo(Double.POSITIVE_INFINITY) - - "'qsd' / 3".runJs().assertEqualsTo(Double.NaN) - "'0' / null".runJs().assertEqualsTo(Double.NaN) - "null / '0'".runJs().assertEqualsTo(Double.NaN) + "'30' / 0".eval().assertEqualsTo(Double.POSITIVE_INFINITY) + "'30' / 0.0".eval().assertEqualsTo(Double.POSITIVE_INFINITY) } @Test - fun null_undefined(){ - "null / 5".runJs().assertEqualsTo(0L) - "5 / null".runJs().assertEqualsTo(Double.POSITIVE_INFINITY) - "0 / null".runJs().assertEqualsTo(Double.NaN) - "null / 0".runJs().assertEqualsTo(Double.NaN) - "null / null".runJs().assertEqualsTo(Double.NaN) - - "undefined / null".runJs().assertEqualsTo(Double.NaN) - "5 / undefined".runJs().assertEqualsTo(Double.NaN) - "undefined / undefined".runJs().assertEqualsTo(Double.NaN) + fun null_undefined() { + "null / 5".eval().assertEqualsTo(0L) + "5 / null".eval().assertEqualsTo(Double.POSITIVE_INFINITY) + "0 / null".eval().assertEqualsTo(Double.NaN) + "null / 0".eval().assertEqualsTo(Double.NaN) + "null / null".eval().assertEqualsTo(Double.NaN) + + "undefined / null".eval().assertEqualsTo(Double.NaN) + "5 / undefined".eval().assertEqualsTo(Double.NaN) + "undefined / undefined".eval().assertEqualsTo(Double.NaN) + + "'qsd' / 3".eval().assertEqualsTo(Double.NaN) + "'0' / null".eval().assertEqualsTo(Double.NaN) + "null / '0'".eval().assertEqualsTo(Double.NaN) } -} - +} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/FunctionsTest.kt b/skriptie/src/commonTest/kotlin/js/FunctionsTest.kt new file mode 100644 index 00000000..37274795 --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/FunctionsTest.kt @@ -0,0 +1,90 @@ +package js + +import io.github.alexzhirkevich.skriptie.common.ReferenceError +import kotlin.test.Test +import kotlin.test.assertFailsWith + +class FunctionsTest { + + @Test + fun creation() { + """ + var x; + function test(a,b) { return a + b } + x = test(1,2) + """.trimIndent().eval().assertEqualsTo(3L) + + """ + function test(a,b) { + return a + b + } + test(1,2) + """.trimIndent().eval().assertEqualsTo(3L) + + """ + function test(a,b) + { + let x = b + 1 + return a + x + } + test(1,2) + """.trimIndent().eval().assertEqualsTo(4L) + } + + @Test + fun defaultArgs(){ + """ + function test(a, b = 2) + { + return a + b + } + test(1) + """.trimIndent().eval().assertEqualsTo(3L) + + """ + var x; + function test(a, b = 2) + { + return a + b + } + x = test(2,3) + """.trimIndent().eval().assertEqualsTo(5L) + + """ + function test(a = 1, b = 2) + { + return a + b + } + test() + """.trimIndent().eval().assertEqualsTo(3L) + } + + @Test + fun scope(){ + """ + let x = 1 + + function fun1(){ + function fun2() { + x = 2 + } + + fun2() + } + fun1() + + x + """.trimIndent().eval().assertEqualsTo(2L) + + assertFailsWith { + """ + function fun1(){ + function fun2() { + + } + } + fun2() + """.trimIndent().eval().assertEqualsTo(2L) + } + } +} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/JsTestUtil.kt b/skriptie/src/commonTest/kotlin/js/JsTestUtil.kt index 6cb8d77d..12a3df3a 100644 --- a/skriptie/src/commonTest/kotlin/js/JsTestUtil.kt +++ b/skriptie/src/commonTest/kotlin/js/JsTestUtil.kt @@ -1,29 +1,20 @@ package js +import io.github.alexzhirkevich.skriptie.ScriptEngine +import io.github.alexzhirkevich.skriptie.ecmascript.EcmascriptInterpreter import io.github.alexzhirkevich.skriptie.invoke -import io.github.alexzhirkevich.skriptie.javascript.JS +import io.github.alexzhirkevich.skriptie.javascript.JSInterpretationContext +import io.github.alexzhirkevich.skriptie.javascript.JSScriptContext import kotlin.test.assertEquals internal fun Any?.assertEqualsTo(other : Any?) = assertEquals(other,this) -internal fun Any?.assertEqualsTo(other : Double, tolerance: Double = 0.0001) = - assertEquals(other,this as Double, tolerance) +internal fun Any?.assertEqualsTo(other : Double, tolerance: Double = 0.0001) { + assertEquals(other, this as Double, tolerance) +} -internal fun String.runJs() : Any? { - return JS().invoke(this) +internal fun String.eval() : Any? { + return ScriptEngine( + JSScriptContext(), + EcmascriptInterpreter(JSInterpretationContext()) + ).invoke(this) } -// -//internal fun String.assertSimpleExprEquals(expected : Any) { -// "var $ret=$this".assertSimpleExprReturns(expected) -//} -// -//internal fun String.assertSimpleExprReturns(expected : Any) { -// assertEquals(expected, runJs()) -//} -// -//internal fun String.assertSimpleExprEquals(expected : Double) { -// "var $ret=$this".assertSimpleExprReturns(expected) -//} -// -//internal fun String.assertSimpleExprReturns(expected : Double) { -// assertEquals(expected, runJs() as Double, 0.00001) -//} diff --git a/skriptie/src/commonTest/kotlin/js/LoopExpressionsTest.kt b/skriptie/src/commonTest/kotlin/js/LoopExpressionsTest.kt deleted file mode 100644 index 7b982877..00000000 --- a/skriptie/src/commonTest/kotlin/js/LoopExpressionsTest.kt +++ /dev/null @@ -1,35 +0,0 @@ -package js - -import kotlin.test.Test - -class LoopExpressionsTest { - - @Test - fun whileLoop() { - """ - var x = 0 - while(x != 3) { - x += 1 - } - x - """.trimIndent().runJs().assertEqualsTo(3L) - - """ - var x = 0 - while(x < 3) - x += 1 - x - """.trimIndent().runJs().assertEqualsTo(3L) - } - - @Test - fun doWhileLoop() { - """ - var x = 0 - do { - x+=1 - } while(x != 3) - x - """.trimIndent().runJs().assertEqualsTo(3L) - } -} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/LoopsTest.kt b/skriptie/src/commonTest/kotlin/js/LoopsTest.kt new file mode 100644 index 00000000..b29e862f --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/LoopsTest.kt @@ -0,0 +1,150 @@ +package js + +import kotlin.test.Test + +class LoopsTest { + + @Test + fun whileLoop() { + """ + var x = 0 + while(x != 3) { + x += 1 + } + x + """.trimIndent().eval().assertEqualsTo(3L) + + """ + var x = 0 + while(x < 3) + x += 1 + x + """.trimIndent().eval().assertEqualsTo(3L) + } + + @Test + fun doWhileLoop() { + """ + var x = 0 + do { + x+=1 + } while(x != 3) + x + """.trimIndent().eval().assertEqualsTo(3L) + } + + @Test + fun forLoop(){ + """ + var x = 0 + for(let i = 0; i<3;i++){ + x+=1 + } + x + """.trimIndent().eval().assertEqualsTo(3L) + + """ + let i = 0 + for(i = 0; i<3;i++){ + + } + i + """.trimIndent().eval().assertEqualsTo(3L) + + """ + let i = 0 + for(i = 0; i<3;){ + i++ + } + i + """.trimIndent().eval().assertEqualsTo(3L) + + """ + let i = 0 + for(; i<3;){ + i++ + } + i + """.trimIndent().eval().assertEqualsTo(3L) + + """ + let i = 0 + for(;;){ + i++ + if (i >= 3) + break + } + i + """.trimIndent().eval().assertEqualsTo(3L) + } + + @Test + fun early_return(){ + """ + var x = 0 + for(let i = 0; i<3;i++){ + break + x+=1 + } + x + """.trimIndent().eval().assertEqualsTo(0L) + + """ + var x = 0 + for(let i = 0; i<3;i++){ + if (i % 2 == 1) + continue + x+=1 + } + x + """.trimIndent().eval().assertEqualsTo(2L) + + """ + var i = 0 + var x = 0 + while(x < 3) { + i++ + if (i == 1) + break + x += 1 + } + i + """.trimIndent().eval().assertEqualsTo(1L) + + """ + var i = 0 + var x = 0 + do { + i++ + if (i % 2 == 1) + continue + x += 1 + } while(x < 3) + i + """.trimIndent().eval().assertEqualsTo(6L) + + """ + var i = 0 + var x = 0 + while(x < 3) { + i++ + if (i == 1) + break + x += 1 + } + i + """.trimIndent().eval().assertEqualsTo(1L) + + """ + var i = 0 + var x = 0 + do { + i++ + if (i % 2 == 1) + continue + x += 1 + } while(x < 3) + i + """.trimIndent().eval().assertEqualsTo(6L) + } +} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/MathTest.kt b/skriptie/src/commonTest/kotlin/js/MathTest.kt index 9f23eabe..a539b2f2 100644 --- a/skriptie/src/commonTest/kotlin/js/MathTest.kt +++ b/skriptie/src/commonTest/kotlin/js/MathTest.kt @@ -6,26 +6,26 @@ class MathTest { @Test fun math() { - "Math.sin(Math.PI/2)".runJs().assertEqualsTo(1.0) - "Math.sin(Math.PI)".runJs().assertEqualsTo(0.0) - "Math.sin(0)".runJs().assertEqualsTo(0.0) - "Math.sin(0.0)".runJs().assertEqualsTo(0.0) + "Math.sin(Math.PI/2)".eval().assertEqualsTo(1.0) + "Math.sin(Math.PI)".eval().assertEqualsTo(0.0) + "Math.sin(0)".eval().assertEqualsTo(0.0) + "Math.sin(0.0)".eval().assertEqualsTo(0.0) - "Math.cos(Math.PI)".runJs().assertEqualsTo(-1.0) - "Math.cos(0)".runJs().assertEqualsTo(1.0) - "Math.cos(0.0)".runJs().assertEqualsTo(1.0) + "Math.cos(Math.PI)".eval().assertEqualsTo(-1.0) + "Math.cos(0)".eval().assertEqualsTo(1.0) + "Math.cos(0.0)".eval().assertEqualsTo(1.0) - "Math.sqrt(16)".runJs().assertEqualsTo(4.0) - "Math.sqrt(16.0)".runJs().assertEqualsTo(4.0) + "Math.sqrt(16)".eval().assertEqualsTo(4.0) + "Math.sqrt(16.0)".eval().assertEqualsTo(4.0) - "Math.imul(3,4)".runJs().assertEqualsTo(12L) - "Math.imul(-5,12)".runJs().assertEqualsTo(-60L) - "Math.imul(0xffffffff, 5)".runJs().assertEqualsTo(-5L) - "Math.imul(0xfffffffe, 5)".runJs().assertEqualsTo(-10L) + "Math.imul(3,4)".eval().assertEqualsTo(12L) + "Math.imul(-5,12)".eval().assertEqualsTo(-60L) + "Math.imul(0xffffffff, 5)".eval().assertEqualsTo(-5L) + "Math.imul(0xfffffffe, 5)".eval().assertEqualsTo(-10L) - "Math.hypot(3, 4)".runJs().assertEqualsTo(5.0) - "Math.hypot(5, 12)".runJs().assertEqualsTo(13.0) - "Math.hypot(3, 4, 5)".runJs().assertEqualsTo(7.071068) - "Math.hypot(-5)".runJs().assertEqualsTo(5.0) + "Math.hypot(3, 4)".eval().assertEqualsTo(5.0) + "Math.hypot(5, 12)".eval().assertEqualsTo(13.0) + "Math.hypot(3, 4, 5)".eval().assertEqualsTo(7.071068) + "Math.hypot(-5)".eval().assertEqualsTo(5.0) } } \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/ModExpressionTest.kt b/skriptie/src/commonTest/kotlin/js/ModExpressionTest.kt deleted file mode 100644 index dd8756db..00000000 --- a/skriptie/src/commonTest/kotlin/js/ModExpressionTest.kt +++ /dev/null @@ -1,15 +0,0 @@ -//package js -// -//import expressions.assertExprEquals -//import kotlin.test.Test -// -// -//class ModExpressionTest { -// -// @Test -// fun mod_num(){ -// "mod(25,4)".assertExprEquals(1f) -// "mod(25.1,4)".assertExprEquals(1.1f) -// } -//} -// diff --git a/skriptie/src/commonTest/kotlin/js/ModTest.kt b/skriptie/src/commonTest/kotlin/js/ModTest.kt new file mode 100644 index 00000000..b504470c --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/ModTest.kt @@ -0,0 +1,39 @@ +package js + +import kotlin.test.Test + + +class ModTest { + + @Test + fun numbers() { + "13 % 7".eval().assertEqualsTo(6L) + "13 % -7".eval().assertEqualsTo(6L) + "-13 % 7".eval().assertEqualsTo(-6L) + "-13 % -7".eval().assertEqualsTo(-6L) + + "13 % 0".eval().assertEqualsTo(Double.NaN) + "0 % 0".eval().assertEqualsTo(Double.NaN) + "0 % 13".eval().assertEqualsTo(0L) + } + + @Test + fun string() { + "'13' % '7'".eval().assertEqualsTo(6L) + "'13' % 7".eval().assertEqualsTo(6L) + "13 % '7'".eval().assertEqualsTo(6L) + } + + @Test + fun null_undefined() { + "null % 5".eval().assertEqualsTo(0L) + "5 % null".eval().assertEqualsTo(Double.NaN) + "0 % null".eval().assertEqualsTo(Double.NaN) + "null % 0".eval().assertEqualsTo(Double.NaN) + "null % null".eval().assertEqualsTo(Double.NaN) + + "undefined % null".eval().assertEqualsTo(Double.NaN) + "5 % undefined".eval().assertEqualsTo(Double.NaN) + "undefined % undefined".eval().assertEqualsTo(Double.NaN) + } +} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/MulTest.kt b/skriptie/src/commonTest/kotlin/js/MulTest.kt index 8708345e..5169ed0d 100644 --- a/skriptie/src/commonTest/kotlin/js/MulTest.kt +++ b/skriptie/src/commonTest/kotlin/js/MulTest.kt @@ -7,32 +7,32 @@ class MulTest { @Test fun numbers() { - "13 * 17.0".runJs().assertEqualsTo(221.0) - "-13 * -17".runJs().assertEqualsTo(221L) - "-13.0 * 17 * 2".runJs().assertEqualsTo(-442.0) + "13 * 17.0".eval().assertEqualsTo(221.0) + "-13 * -17".eval().assertEqualsTo(221L) + "-13.0 * 17 * 2".eval().assertEqualsTo(-442.0) } @Test fun string(){ - "'10' * '3'".runJs().assertEqualsTo(30L) - "10 * '3'".runJs().assertEqualsTo(30L) - "'10' * 3".runJs().assertEqualsTo(30L) - - "'10.5' * '3'".runJs().assertEqualsTo(31.5) - "10.5 * '3'".runJs().assertEqualsTo(31.5) - "'10.5' * 3".runJs().assertEqualsTo(31.5) - "'qsd' * 3".runJs().assertEqualsTo(Double.NaN) + "'10' * '3'".eval().assertEqualsTo(30L) + "10 * '3'".eval().assertEqualsTo(30L) + "'10' * 3".eval().assertEqualsTo(30L) + + "'10.5' * '3'".eval().assertEqualsTo(31.5) + "10.5 * '3'".eval().assertEqualsTo(31.5) + "'10.5' * 3".eval().assertEqualsTo(31.5) + "'qsd' * 3".eval().assertEqualsTo(Double.NaN) } @Test fun null_undefined(){ - "null * 5".runJs().assertEqualsTo(0L) - "5 * null".runJs().assertEqualsTo(0L) - "null * null".runJs().assertEqualsTo(0L) + "null * 5".eval().assertEqualsTo(0L) + "5 * null".eval().assertEqualsTo(0L) + "null * null".eval().assertEqualsTo(0L) - "undefined * null".runJs().assertEqualsTo(Double.NaN) - "5 * undefined".runJs().assertEqualsTo(Double.NaN) - "undefined * undefined".runJs().assertEqualsTo(Double.NaN) + "undefined * null".eval().assertEqualsTo(Double.NaN) + "5 * undefined".eval().assertEqualsTo(Double.NaN) + "undefined * undefined".eval().assertEqualsTo(Double.NaN) } } diff --git a/skriptie/src/commonTest/kotlin/js/NumberFormatTest.kt b/skriptie/src/commonTest/kotlin/js/NumberFormatTest.kt index 3a12a607..98e8bd6b 100644 --- a/skriptie/src/commonTest/kotlin/js/NumberFormatTest.kt +++ b/skriptie/src/commonTest/kotlin/js/NumberFormatTest.kt @@ -6,12 +6,12 @@ class NumberFormatTest { @Test fun test(){ - "123".runJs().assertEqualsTo(123L) - "123.1".runJs().assertEqualsTo(123.1) - ".1".runJs().assertEqualsTo(.1) + "123".eval().assertEqualsTo(123L) + "123.1".eval().assertEqualsTo(123.1) + ".1".eval().assertEqualsTo(.1) - "0xff".runJs().assertEqualsTo(255L) - "0b11".runJs().assertEqualsTo(3L) - "0o123".runJs().assertEqualsTo(83L) + "0xff".eval().assertEqualsTo(255L) + "0b11".eval().assertEqualsTo(3L) + "0o123".eval().assertEqualsTo(83L) } } \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/SubTest.kt b/skriptie/src/commonTest/kotlin/js/SubTest.kt index 7bf568d8..601d78bb 100644 --- a/skriptie/src/commonTest/kotlin/js/SubTest.kt +++ b/skriptie/src/commonTest/kotlin/js/SubTest.kt @@ -5,39 +5,44 @@ import kotlin.test.Test class SubTest { + @Test + fun randomJsPizdec(){ + "10 - [[[[[[[[[[5]]]]]]]]]]".eval().assertEqualsTo(5L) + } + @Test fun numbers() { - "13-17".runJs().assertEqualsTo(-4L) - "13 - 17".runJs().assertEqualsTo(-4L) - "-13-17".runJs().assertEqualsTo(-30L) - "13.0-17.0".runJs().assertEqualsTo(-4.0) - "13 - 17.0".runJs().assertEqualsTo(-4.0) - "-13.0 -17".runJs().assertEqualsTo(-30.0) + "13-17".eval().assertEqualsTo(-4L) + "13 - 17".eval().assertEqualsTo(-4L) + "-13-17".eval().assertEqualsTo(-30L) + "13.0-17.0".eval().assertEqualsTo(-4.0) + "13 - 17.0".eval().assertEqualsTo(-4.0) + "-13.0 -17".eval().assertEqualsTo(-30.0) } @Test fun string(){ - "'10' - '3'".runJs().assertEqualsTo(7L) - "10 - '3'".runJs().assertEqualsTo(7L) - "'10' - 3".runJs().assertEqualsTo(7L) - - "'10.5' - '3'".runJs().assertEqualsTo(7.5) - "10.5 - '3'".runJs().assertEqualsTo(7.5) - "'10.5' - 3".runJs().assertEqualsTo(7.5) - "'qsd' - 3".runJs().assertEqualsTo(Double.NaN) + "'10' - '3'".eval().assertEqualsTo(7L) + "10 - '3'".eval().assertEqualsTo(7L) + "'10' - 3".eval().assertEqualsTo(7L) + + "'10.5' - '3'".eval().assertEqualsTo(7.5) + "10.5 - '3'".eval().assertEqualsTo(7.5) + "'10.5' - 3".eval().assertEqualsTo(7.5) + "'qsd' - 3".eval().assertEqualsTo(Double.NaN) } @Test fun null_undefined(){ - "null - 5".runJs().assertEqualsTo(-5L) - "5 - null".runJs().assertEqualsTo(5L) - "null - null".runJs().assertEqualsTo(0L) - "null - '123'".runJs().assertEqualsTo(-123L) - "null - '123.5'".runJs().assertEqualsTo(-123.5) - - "undefined - 5".runJs().assertEqualsTo(Double.NaN) - "5 - undefined".runJs().assertEqualsTo(Double.NaN) - "undefined - undefined".runJs().assertEqualsTo(Double.NaN) + "null - 5".eval().assertEqualsTo(-5L) + "5 - null".eval().assertEqualsTo(5L) + "null - null".eval().assertEqualsTo(0L) + "null - '123'".eval().assertEqualsTo(-123L) + "null - '123.5'".eval().assertEqualsTo(-123.5) + + "undefined - 5".eval().assertEqualsTo(Double.NaN) + "5 - undefined".eval().assertEqualsTo(Double.NaN) + "undefined - undefined".eval().assertEqualsTo(Double.NaN) } } diff --git a/skriptie/src/commonTest/kotlin/js/SumTest.kt b/skriptie/src/commonTest/kotlin/js/SumTest.kt index 72d8c7ae..645e5a76 100644 --- a/skriptie/src/commonTest/kotlin/js/SumTest.kt +++ b/skriptie/src/commonTest/kotlin/js/SumTest.kt @@ -1,35 +1,41 @@ package js import kotlin.test.Test -import kotlin.test.assertEquals class SumTest { + @Test + fun randomJsPizdec(){ + "'b' + 'a' + + 'a' + 'a'".eval().assertEqualsTo("baNaNa") + "[5] + 1".eval().assertEqualsTo("51") + "[1,2] + [3,3]".eval().assertEqualsTo("1,23,3") + } + @Test fun numbers() { - "13+17".runJs().assertEqualsTo(30L) - "-13+ 17".runJs().assertEqualsTo(4L) - "-13+ -17".runJs().assertEqualsTo(-30L) - "-13+ -17.0 + 10 - 4.0".runJs().assertEqualsTo(-24.0) + "13+17".eval().assertEqualsTo(30L) + "-13+ 17".eval().assertEqualsTo(4L) + "-13+ -17".eval().assertEqualsTo(-30L) + "-13+ -17.0 + 10 - 4.0".eval().assertEqualsTo(-24.0) } @Test fun string(){ - "'10' + '5'".runJs().assertEqualsTo("105") - "'10' + 5".runJs().assertEqualsTo("105") - "10 + '5'".runJs().assertEqualsTo("105") + "'10' + '5'".eval().assertEqualsTo("105") + "'10' + 5".eval().assertEqualsTo("105") + "10 + '5'".eval().assertEqualsTo("105") } @Test fun null_undefined(){ - "null + 5".runJs().assertEqualsTo(5L) - "5 + null".runJs().assertEqualsTo(5L) - "null + null".runJs().assertEqualsTo(0L) + "null + 5".eval().assertEqualsTo(5L) + "5 + null".eval().assertEqualsTo(5L) + "null + null".eval().assertEqualsTo(0L) - "undefined + 5".runJs().assertEqualsTo(Double.NaN) - "5 + undefined".runJs().assertEqualsTo(Double.NaN) - "undefined + undefined".runJs().assertEqualsTo(Double.NaN) + "undefined + 5".eval().assertEqualsTo(Double.NaN) + "5 + undefined".eval().assertEqualsTo(Double.NaN) + "undefined + undefined".eval().assertEqualsTo(Double.NaN) } } diff --git a/skriptie/src/commonTest/kotlin/js/SyntaxExpressionsTest.kt b/skriptie/src/commonTest/kotlin/js/SyntaxExpressionsTest.kt deleted file mode 100644 index 236d5ff8..00000000 --- a/skriptie/src/commonTest/kotlin/js/SyntaxExpressionsTest.kt +++ /dev/null @@ -1,85 +0,0 @@ -package js - -import kotlin.test.Test -import kotlin.test.assertFails - -class SyntaxExpressionsTest { - - @Test - fun newline_property() { - """ - Math - .imul(3,4) - .toString() - """.trimIndent().runJs().assertEqualsTo("12") - - """ - Math - - .imul(3,4) - - .toString() - """.trimIndent().runJs().assertEqualsTo("12") - - """ - Math - - .imul(3,4) - - .toString() - ; - """.trimIndent().runJs().assertEqualsTo("12") - - - assertFails { - """ - Math - .imul(3,4); - .toString() - """.trimIndent().runJs() - } - } - - @Test - fun increment_decrement() { - "let x = 5; x++".runJs().assertEqualsTo(6L) - "let x = 5; ++x".runJs().assertEqualsTo(6L) - - "let x = 5; x--".runJs().assertEqualsTo(4L) - "let x = 5; --x".runJs().assertEqualsTo(4L) - - "let x = 5; --x".runJs().assertEqualsTo(4L) - "let x = 5; ++x".runJs().assertEqualsTo(6L) - "let x = 5; x--".runJs().assertEqualsTo(4L) - "let x = 5; x++".runJs().assertEqualsTo(6L) - } - - @Test - fun variable_scopes() { - """ - var x; - if(true) { - x = 5 - } - x - """.trimIndent().runJs().assertEqualsTo(5L) - - """ - var x = 1; - if(true){ - let x = 5 - } - x - """.trimIndent().runJs().assertEqualsTo(1L) - } - - @Test - fun constMutating() { - assertFails { - """ - const x = 1; - x++ - """.trimIndent().runJs() - } - } -} \ No newline at end of file diff --git a/skriptie/src/commonTest/kotlin/js/SyntaxTest.kt b/skriptie/src/commonTest/kotlin/js/SyntaxTest.kt new file mode 100644 index 00000000..dc94149f --- /dev/null +++ b/skriptie/src/commonTest/kotlin/js/SyntaxTest.kt @@ -0,0 +1,142 @@ +package js + +import io.github.alexzhirkevich.skriptie.common.ReferenceError +import io.github.alexzhirkevich.skriptie.common.SyntaxError +import io.github.alexzhirkevich.skriptie.common.TypeError +import kotlin.test.Test +import kotlin.test.assertFailsWith + +class SyntaxTest { + + @Test + fun newline_property() { + """ + Math + .imul(3,4) + .toString() + """.trimIndent().eval().assertEqualsTo("12") + + """ + Math + + .imul(3,4) + + .toString() + """.trimIndent().eval().assertEqualsTo("12") + + """ + Math + + .imul(3,4) + + .toString() + ; + """.trimIndent().eval().assertEqualsTo("12") + + + assertFailsWith { + """ + Math + .imul(3,4); + .toString() + """.trimIndent().eval() + } + } + + @Test + fun increment_decrement() { + "let x = 5; x++".eval().assertEqualsTo(5L) + "let x = 5; ++x".eval().assertEqualsTo(6L) + + "let x = 5; x--".eval().assertEqualsTo(5L) + "let x = 5; --x".eval().assertEqualsTo(4L) + + "let x = 0; let y = x++; y".eval().assertEqualsTo(0L) + "let x = 0; let y = x++; x".eval().assertEqualsTo(1L) + "let x = 0; let y = ++x; y".eval().assertEqualsTo(1L) + "let x = 0; let y = ++x; x".eval().assertEqualsTo(1L) + + "let x = 1; let y = x--; y".eval().assertEqualsTo(1L) + "let x = 1; let y = x--; x".eval().assertEqualsTo(0L) + "let x = 1; let y = --x; y".eval().assertEqualsTo(0L) + "let x = 1; let y = --x; x".eval().assertEqualsTo(0L) + } + + @Test + fun variable_scopes() { + """ + var x; + if(true) { + x = 5 + } + x + """.trimIndent().eval().assertEqualsTo(5L) + + """ + var x = 1; + if(true){ + let x = 5 + } + x + """.trimIndent().eval().assertEqualsTo(1L) + + + """ + if(true){ + var x = 5 + } + x + """.trimIndent().eval().assertEqualsTo(5L) + + assertFailsWith { + """ + if(true){ + let x = 5 + } + x + """.trimIndent().eval() + } + assertFailsWith { + """ + if(true){ + const x = 5 + } + x + """.trimIndent().eval() + } + } + + @Test + fun constMutating() { + assertFailsWith { "const x = 1; x++".eval() } + } + + @Test + fun variable_redeclaration() { + assertFailsWith { "const x = 1; const x = 2".eval() } + assertFailsWith { "const x = 1; let x = 2".eval() } + assertFailsWith { "const x = 1; var x = 2".eval() } + assertFailsWith { "var x = 1; const x = 2".eval() } + assertFailsWith { "var x = 1; let x = 2".eval() } + assertFailsWith { "var x = 1; var x = 2".eval() } + assertFailsWith { "let x = 1; const x = 2".eval() } + assertFailsWith { "let x = 1; var x = 2".eval() } + assertFailsWith { "let x = 1; let x = 2".eval() } + + "const x = 1; { const x = 2 }; x".eval().assertEqualsTo(1L) + "const x = 1; { let x = 2 }; x".eval().assertEqualsTo(1L) + assertFailsWith { + "const x = 1; { var x = 2 }; x".eval().assertEqualsTo(1L) + } + "var x = 1; { const x = 2 }; x".eval().assertEqualsTo(1L) + "var x = 1; { let x = 2 }; x".eval().assertEqualsTo(1L) + assertFailsWith { + "var x = 1; { var x = 2 }; x".eval().assertEqualsTo(1L) + } + "let x = 1; { const x = 2 }; x".eval().assertEqualsTo(1L) + assertFailsWith { + "let x = 1; { var x = 2 }; x".eval().assertEqualsTo(1L) + } + "let x = 1; { let x = 2 }; x".eval().assertEqualsTo(1L) + } +} \ No newline at end of file