nomsu/examples/how_do_i.nom

342 lines
9.6 KiB
Plaintext
Raw Normal View History

2017-10-04 18:04:33 -07:00
# How do I...
# Write a comment? Put a # and go till the end of the line
2018-05-15 18:55:55 -07:00
# How do I write a multi-line comment?
After a comment line, any indented text
is considered part of the comment
(including any deeper-level indented text)
The comment ends when the indentation ends
2017-10-04 18:04:33 -07:00
# How do I import a file?
use "core/control_flow.nom"
# How do I import all the files in a directory?
use "core"
2017-10-04 18:04:33 -07:00
# How do I print stuff?
say "Hello world!"
# How do I set a variable?
2018-01-25 17:43:57 -08:00
%x <- 1
%str <- "Hello world"
2017-10-04 18:04:33 -07:00
# Expressions that are more than just literal values require parentheses:
2018-01-25 17:43:57 -08:00
%x <- (2 + 3)
%one-two <- 12
say %one-two
2017-10-04 18:04:33 -07:00
# How do I modify a variable?
%x <- (%x + 1)
# Or, as a shorthand, you can do this to increment a variable:
%x +<- 1
2017-10-04 18:04:33 -07:00
# How do I define a mutli-line string?
%mutli-str <- ".."
2017-12-14 13:54:31 -08:00
Start with "..", then put lines below it
that are indented one level.
The string will continue until the indentation ends.
2017-10-04 18:04:33 -07:00
# How do I put values inside a string?
%format-str <- ".."
Strings can contain a backslash followed by a variable, list, dict, or parenthesized
expression. This escaped value will be converted to a readable string, like so:
The value of %x is \%x, isn't that nice?
These are some numbers: \[1+1,2+1,3+1]
The sum of 2 and 4 is \(2 + 4).
2017-12-14 13:54:31 -08:00
If you need to use a plain ol' backslash, you can do \\ <-- that
%format-str2 <- "Single-line strings can contain escape sequences like \", \\, \n, \065, and \x0A"
2017-10-04 18:04:33 -07:00
# How do I define a list?
%my-list <- [1,2,"hello"]
2018-05-15 18:55:55 -07:00
# Really long lists can use [..] followed by a bunch of indented values delimited
2017-10-04 18:04:33 -07:00
by commas and/or newlines
%my-really-long-list <- [..]
2017-10-04 18:04:33 -07:00
1,2,3,4
5,6
7
8,9,10
# How do I use a list?
%my-list <- ["first item", "second item", "third item"]
2017-10-04 18:04:33 -07:00
# Lists are 1-indexed because they're implemented as Lua tables, so this prints "first item"
say %my-list.1
2017-10-04 18:04:33 -07:00
# List entries can be modified like this:
%my-list.1 <- "ONE!!!"
say (length of %my-list)
2017-10-04 18:04:33 -07:00
# How do I define a dictionary/hash map?
%my-dict <- {x: 99, y: 101}
%my-dict <- {..}
2018-01-19 18:13:02 -08:00
x: 101, y: 2
"99 bottles": 99
653: 292
2017-10-04 18:04:33 -07:00
# How do I use a dict?
2017-10-04 18:04:33 -07:00
# Dicts are also implemented as Lua tables, so they're accessed and modified the same way as lists
say %my-dict.x
%my-dict.x <- 9999
2017-10-04 18:04:33 -07:00
# How do I do conditional branching?
2018-01-19 18:13:02 -08:00
if: 1 < 10
2017-10-04 18:04:33 -07:00
say "1 is indeed < 10"
2018-01-19 18:13:02 -08:00
if: 1 > 10
2017-10-04 18:04:33 -07:00
say "this won't print"
2018-01-19 18:13:02 -08:00
..else
2017-10-04 18:04:33 -07:00
say "this will print"
# There's no "elseif", so for longer conditionals, a "when" branch is the best option
2018-01-19 18:13:02 -08:00
when
2017-10-04 18:04:33 -07:00
* (3 > 6)
* (3 > 5)
2018-01-19 18:13:02 -08:00
* (3 > 4)
2017-10-04 18:04:33 -07:00
say "this won't print"
2018-01-19 18:13:02 -08:00
* (3 > 3)
2017-10-04 18:04:33 -07:00
say "this won't print"
2018-01-19 18:13:02 -08:00
* (3 > 2)
2017-10-04 18:04:33 -07:00
say "this will print"
2018-01-19 18:13:02 -08:00
* else
2017-10-04 18:04:33 -07:00
say "this is the default case"
# How do I do a switch statement?
2018-01-19 18:13:02 -08:00
when 3 = ?
2017-10-04 18:04:33 -07:00
* 0
* 1
2018-01-19 18:13:02 -08:00
* 2
2017-10-04 18:04:33 -07:00
say "this won't print"
2018-01-19 18:13:02 -08:00
* 3
2017-10-04 18:04:33 -07:00
say "this will print"
2018-01-19 18:13:02 -08:00
* else
2017-10-04 18:04:33 -07:00
say "this won't print"
# How do I loop over a list (a foreach loop)?
2018-01-25 17:43:57 -08:00
%list <- [1,2,3]
2018-01-19 18:13:02 -08:00
for %x in %list
say "For %x loop #\%x"
2017-10-04 18:04:33 -07:00
# How do I loop over a number range?
2017-10-04 18:04:33 -07:00
# This is inclusive, so it will loop over 1,2, and 3
for %i in 1 to 3
say "For %i in 1 to 3 loop #\%i"
2017-10-04 18:04:33 -07:00
# This will print 0,2, and 4
for %even in 0 to 5 by 2
say "Even #\%even"
for %backwards in 3 to 1 by -1
say "Backwards #\%backwards"
2017-10-04 18:04:33 -07:00
# How do I do a 'while' loop?
2018-01-25 17:43:57 -08:00
%x <- 1
2018-01-19 18:13:02 -08:00
repeat while: %x <= 3
say "repeat while loop #\%x"
%x +<- 1
2017-10-04 18:04:33 -07:00
2018-01-25 17:43:57 -08:00
%x <- 1
2018-01-19 18:13:02 -08:00
repeat until: %x > 3
say "repeat until loop #\%x"
%x +<- 1
2017-10-04 18:04:33 -07:00
# How do I do an infinite loop?
2018-01-25 17:43:57 -08:00
%x <- 1
2018-01-19 18:13:02 -08:00
repeat
say "repeat loop #\%x"
%x +<- 1
2018-01-19 18:13:02 -08:00
if: %x > 3
2018-01-25 17:43:57 -08:00
stop repeating
2017-10-04 18:04:33 -07:00
# How do I do a 'goto'?
2018-01-19 18:13:02 -08:00
do
2018-01-25 17:43:57 -08:00
%x <- 1
=== %again ===
say "GOTO loop #\%x"
%x +<- 1
2018-01-19 18:13:02 -08:00
if: %x <= 3
2017-10-04 18:04:33 -07:00
go to %again
say "finished going to"
# How do I define a function/method/procedure?
# In nomsu, they're called "action"s, and they can be declared like this:
2018-01-19 18:13:02 -08:00
action [say both %first and also %second]
2017-10-04 18:04:33 -07:00
say %first
# Function arguments are accessed just like variables
say %second
# Actions can use "return" to return a value early
2018-01-19 18:13:02 -08:00
action [first fibonacci above %n]
2018-01-25 17:43:57 -08:00
%f1 <- 0
%f2 <- 1
2018-01-19 18:13:02 -08:00
repeat
2018-01-25 17:43:57 -08:00
%tmp <- (%f1 + %f2)
%f1 <- %f2
%f2 <- %tmp
2018-01-19 18:13:02 -08:00
if: %f2 > %n
2017-10-04 18:04:33 -07:00
return %f2
say (first fibonacci above 10)
# Actions can have aliases, which may or may not have the arguments in different order
2018-01-19 18:13:02 -08:00
action [..]
I hate %worse-things more than %better-things
I think %worse-things are worse than %better-things
I like %better-things more than %worse-things
2018-01-19 18:13:02 -08:00
..
say "\(%better-things capitalized) rule and \%worse-things drool!"
2017-10-04 18:04:33 -07:00
I like "dogs" more than "cats"
I think "chihuahuas" are worse than "corgis"
2018-05-15 18:55:55 -07:00
# Actions can have parts of the action's name spread throughout.
Everything that's not a literal value is treated as part of the action's name
2017-10-04 18:04:33 -07:00
say both "Hello" and also "again!"
# Actions can even start with a parameter
action [%what-she-said is what she said]
say %what-she-said
2017-10-04 18:04:33 -07:00
say "-- she said"
"Howdy pardner" is what she said
2018-05-15 18:55:55 -07:00
# The language only reserves []{}().,:;% as special characters, so actions
2017-10-04 18:04:33 -07:00
can have really funky names!
action [>> %foo-bar $$$^ --> % @& _~-^-~_~-^ %1 !]
say %foo-bar
say %
2017-10-04 18:04:33 -07:00
say %1
>> "wow" $$$^ --> "so flexible!" @& _~-^-~_~-^ "even numbers can be variables!" !
# There's also full unicode support
%こんにちは <- "こんにちは"
action [% と言う]
"\(%)世界"
say (%こんにちは と言う)
# Math and logic operations are just treated the same as actions in the syntax
2017-10-04 18:04:33 -07:00
say (2 + 3)
2018-05-15 18:55:55 -07:00
# So you can define your own operators, although they will need to be parenthesized to
play nicely with other operators
2018-01-19 18:13:02 -08:00
action [%a ++ %b]
2017-10-04 18:04:33 -07:00
2 * (%a + %b)
say (1 ++ (2 * 3))
2017-10-04 18:04:33 -07:00
# How do I do grouping?
2017-10-04 18:04:33 -07:00
# Expressions can be grouped by enclosing parentheses:
say (2 + 3)
2018-01-19 18:13:02 -08:00
# Or by an indented region
say
2017-10-04 18:04:33 -07:00
2 + 3
# Or by ":" until the end of the line
say: 2 + 3
2017-10-04 18:04:33 -07:00
# If you need to keep going after an indented region, you can start the next line with ".."
2018-01-19 18:13:02 -08:00
say both
2017-10-04 18:04:33 -07:00
"Very long first argument that needs its own line"
..and also "short second arg"
2018-01-19 18:13:02 -08:00
action [my favorite number]: 21 + 2
2017-10-04 18:04:33 -07:00
# This can be nested:
2018-01-19 18:13:02 -08:00
say both
2017-10-04 18:04:33 -07:00
my favorite
..number
..and also "foo"
# Macros:
# The "lua> %" and "=lua %" macros can be used to write raw lua code:
2018-01-19 18:13:02 -08:00
action [say the time]
lua> ".."
io.write("The OS time is: "..os.time().."\n");
2017-10-04 18:04:33 -07:00
say the time
say "Math expression result is: \(=lua "(1 + 2*3 + 3*4)^2")"
2017-10-04 18:04:33 -07:00
2018-01-19 18:13:02 -08:00
# Variables can be accessed via \%varname
action [square root of %n]
=lua "math.sqrt(\%n)"
2017-10-04 18:04:33 -07:00
say "The square root of 2 is \(square root of 2)"
2018-05-15 18:55:55 -07:00
# The "immediately %" macro forces the indented code below it to run before the rest of
the file finishes compiling, so it's often useful for writing your own macros.
2018-01-19 18:13:02 -08:00
immediately
# Macros can be defined to transform one bit of nomsu code into another using "parse % as %":
parse [if %condition is untrue %body] as
if (not %condition) %body
2017-10-04 18:04:33 -07:00
2018-01-25 17:43:57 -08:00
# Or to transform nomsu code into custom lua code using "compile % to %"
compile [if %condition on opposite day %body] to
Lua ".."
if not \(%condition as lua expr) then
\(%body as lua statements)
end
# Constants can be defined as macros
parse [TWENTY] as: 20
# When they're invoked, they'll need parentheses just like a function call
parse [TWENTY ONE] as: 21
if (1 > (TWENTY)) is untrue
2017-10-04 18:04:33 -07:00
say "Nomsu parsing macros work!"
say "It looks like a keyword, but there's no magic here!"
if (1 > (TWENTY)) on opposite day
2017-10-04 18:04:33 -07:00
say "Lua compiling macros work!"
say "It looks like a keyword, but there's no magic here!"
# How do I use an action as a value?
2018-05-15 18:55:55 -07:00
# Well... it's always *possible* to fall back to Lua behavior for something like this:
action [best of %items according to %key-fn]
<- {%best:nil, %best-key:nil}
for % in %items
%key <- (=lua "\%key-fn(\%)")
if: (%best is (nil)) or (%key > %best-key)
<- {%best:%, %best-key:%key}
return %best
immediately
compile [function %var %body] to
Lua value ".."
(function(\(%var as lua expr))
return \(%body as lua expr)
end)
say: best of [2,-3,4,-8] according to (function %: % * %)
2018-05-15 18:55:55 -07:00
# But nomsu was mostly designed so that you don't *need* to. Instead of creating a
one-off function to pass to another function and get called a bunch of times, you
could use a macro to generate a single block of code that inlines the expression you
want to use:
immediately
parse [best of %items according to %key-expr] as
result of
<- {%best:nil, %best-key:nil}
for % in %items
%key <- %key-expr
if: (%best is (nil)) or (%key > %best-key)
<- {%best:%, %best-key:%key}
return %best
2018-05-15 18:55:55 -07:00
# This results in generated code that is more efficient (no function calls in the
inner loop)
say: best of [2,-3,4,-8] according to (% * %)
2018-05-15 18:55:55 -07:00
# In a functional programming language, you might do something like:
doubled = map(list, function(x) return 2 * x end)
to get a new list with every entry multiplied by 2, but it's *much* more readable to
do something like:
%nums <- [1,2,3,4,5]
%double-nums <- ((2 * %num) for %num in %nums)
2018-05-15 18:55:55 -07:00
# Nomsu comes with built-in list comprehensions, but the flexible macro system makes it
incredibly easy to make similar constructs.
immediately
parse [%expr for %key in %list BACKWARDS!] as
result of
%result <- []
%N <- (length of %list)
for %i = %key in %list
%result.(%N - %i + 1) <- %expr
return %result
%double-nums <- ((2 * %num) for %num in %nums BACKWARDS!)
say %double-nums