2017-10-04 18:04:33 -07:00
|
|
|
# How do I...
|
|
|
|
# Write a comment? Put a # and go till the end of the line
|
|
|
|
|
|
|
|
#.. How do I write a multi-line comment?
|
|
|
|
Put a #.. on its own line and
|
|
|
|
write whatever you want
|
|
|
|
in an
|
|
|
|
indented area.
|
|
|
|
|
|
|
|
# How do I import files?
|
2018-01-19 18:13:02 -08:00
|
|
|
use "lib/core.nom"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-01-19 18:13:02 -08:00
|
|
|
# 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)
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# Modify a variable?
|
2018-01-25 17:43:57 -08:00
|
|
|
%foobar <- 100
|
|
|
|
# As a shorthand, you can type:
|
|
|
|
<- %foobar + 1
|
|
|
|
# which does the same thing as:
|
|
|
|
%foobar <- (%foobar + 1)
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# Print stuff?
|
|
|
|
say "Hello world!"
|
|
|
|
|
|
|
|
# Define a mutli-line string?
|
2018-01-25 17:43:57 -08:00
|
|
|
%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
|
|
|
|
|
|
|
# Format a string?
|
2018-01-25 17:43:57 -08:00
|
|
|
%format_str <- ".."
|
2018-01-03 19:26:41 -08:00
|
|
|
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?
|
|
|
|
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
|
2018-01-25 17:43:57 -08:00
|
|
|
%format_str2 <- "Single-line strings can contain escape sequences like \", \\, \n, \065, and \x0A"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# Define a list?
|
2018-01-25 17:43:57 -08:00
|
|
|
%my_list <- [1,2,"hello"]
|
2017-10-04 18:04:33 -07:00
|
|
|
#.. Really long lists can use [..] followed by a bunch of indented values delimited
|
|
|
|
by commas and/or newlines
|
2018-01-25 17:43:57 -08:00
|
|
|
%my_really_long_list <- [..]
|
2017-10-04 18:04:33 -07:00
|
|
|
1,2,3,4
|
|
|
|
5,6
|
|
|
|
7
|
|
|
|
8,9,10
|
|
|
|
|
|
|
|
# Use a list?
|
2018-01-25 17:43:57 -08:00
|
|
|
%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"
|
2018-01-25 17:43:57 -08:00
|
|
|
say (1st in %my_list)
|
2017-10-04 18:04:33 -07:00
|
|
|
# These do the same thing:
|
|
|
|
say (%my_list's 1)
|
|
|
|
say (1 in %my_list)
|
|
|
|
# List entries can be modified like this:
|
2018-01-25 17:43:57 -08:00
|
|
|
(1st in %my_list) <- "ONE!!!"
|
2017-10-04 18:04:33 -07:00
|
|
|
say (size of %my_list)
|
|
|
|
|
|
|
|
# Define a dictionary/hash map?
|
2018-01-25 17:43:57 -08:00
|
|
|
%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
|
|
|
|
|
|
|
# Use a dict?
|
|
|
|
# Dicts are also implemented as Lua tables, so they're accessed and modified the same way as lists
|
2018-01-25 17:43:57 -08:00
|
|
|
say (%my_dict's "x")
|
|
|
|
(%my_dict's "x") <- 9999
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# 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"
|
|
|
|
|
|
|
|
# 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"
|
|
|
|
|
|
|
|
# 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
|
2018-01-03 19:26:41 -08:00
|
|
|
say "For %x loop #\%x"
|
2017-10-04 18:04:33 -07:00
|
|
|
# There's also a slightly more concise version that automatically populates a loop variable "%"
|
2018-01-19 18:13:02 -08:00
|
|
|
for all %list
|
2018-01-03 19:26:41 -08:00
|
|
|
say "For all loop #\%"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# Loop over a number range?
|
|
|
|
# This is inclusive, so it will loop over 1,2, and 3
|
2018-01-19 18:13:02 -08:00
|
|
|
for %i from 1 to 3
|
2018-01-03 19:26:41 -08:00
|
|
|
say "For %i from 1 to 3 loop #\%i"
|
2018-01-19 18:13:02 -08:00
|
|
|
for all 1 to 3
|
2018-01-03 19:26:41 -08:00
|
|
|
say "For all 1 to 3 loop #\%"
|
2017-10-04 18:04:33 -07:00
|
|
|
# This will print 0,2, and 4
|
2018-01-19 18:13:02 -08:00
|
|
|
for %even from 0 to 5 by 2
|
2018-01-03 19:26:41 -08:00
|
|
|
say "Even #\%even"
|
2018-01-19 18:13:02 -08:00
|
|
|
for %backwards from 3 to 1 by -1
|
2018-01-03 19:26:41 -08:00
|
|
|
say "Backwards #\%backwards"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# While loops:
|
2018-01-25 17:43:57 -08:00
|
|
|
%x <- 1
|
2018-01-19 18:13:02 -08:00
|
|
|
repeat while: %x <= 3
|
2018-01-03 19:26:41 -08:00
|
|
|
say "repeat while loop #\%x"
|
2018-01-25 17:43:57 -08:00
|
|
|
<- %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
|
2018-01-03 19:26:41 -08:00
|
|
|
say "repeat until loop #\%x"
|
2018-01-25 17:43:57 -08:00
|
|
|
<- %x + 1
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
# Infinite loop:
|
2018-01-25 17:43:57 -08:00
|
|
|
%x <- 1
|
2018-01-19 18:13:02 -08:00
|
|
|
repeat
|
2018-01-03 19:26:41 -08:00
|
|
|
say "repeat loop #\%x"
|
2018-01-25 17:43:57 -08:00
|
|
|
<- %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
|
|
|
|
|
|
|
# GOTOs:
|
2018-01-19 18:13:02 -08:00
|
|
|
do
|
2018-01-25 17:43:57 -08:00
|
|
|
%x <- 1
|
|
|
|
=== %again ===
|
2018-01-03 19:26:41 -08:00
|
|
|
say "GOTO loop #\%x"
|
2018-01-25 17:43:57 -08:00
|
|
|
<- %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"
|
|
|
|
|
|
|
|
|
|
|
|
# Function definition:
|
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
|
|
|
|
|
|
|
|
# Functions 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)
|
|
|
|
|
|
|
|
# Functions can have aliases, which may or may not have the arguments in different order
|
2018-01-19 18:13:02 -08:00
|
|
|
action [..]
|
2017-10-13 19:41:58 -07:00
|
|
|
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
|
|
|
..
|
2018-01-03 19:26:41 -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"
|
|
|
|
|
|
|
|
|
|
|
|
#.. Function calls can have parts of the function's name spread throughout.
|
|
|
|
Everything that's not a literal value is treated as part of the function's name
|
|
|
|
say both "Hello" and also "again!"
|
|
|
|
|
|
|
|
# Functions can even have their name at the end:
|
2018-01-19 18:13:02 -08:00
|
|
|
action [%what_she_said is what she said]
|
2017-10-13 19:41:58 -07:00
|
|
|
say %what_she_said
|
2017-10-04 18:04:33 -07:00
|
|
|
say "-- she said"
|
|
|
|
|
|
|
|
"Howdy pardner" is what she said
|
|
|
|
|
2017-10-13 19:41:58 -07:00
|
|
|
#.. The language only reserves []{}().,:;% as special characters, so functions
|
2017-10-04 18:04:33 -07:00
|
|
|
can have really funky names!
|
2018-01-19 18:13:02 -08:00
|
|
|
action [>> %foo_bar $$$^ --> % @& _~-^-~_~-^ %1 !]
|
2017-10-13 19:41:58 -07:00
|
|
|
say %foo_bar
|
2017-10-30 14:08:23 -07:00
|
|
|
say %
|
2017-10-04 18:04:33 -07:00
|
|
|
say %1
|
|
|
|
|
2017-10-13 19:41:58 -07:00
|
|
|
>> "wow" $$$^ --> "so flexible!" @& _~-^-~_~-^ "even numbers can be variables!" !
|
|
|
|
|
2017-10-04 18:04:33 -07:00
|
|
|
# Math and logic operations are just treated the same as function calls in the syntax
|
|
|
|
say (2 + 3)
|
|
|
|
# So it's easy to define your own operators
|
2018-01-19 18:13:02 -08:00
|
|
|
action [%a ++ %b]
|
2017-10-04 18:04:33 -07:00
|
|
|
2 * (%a + %b)
|
|
|
|
|
2018-01-03 19:26:41 -08:00
|
|
|
#.. The following are characters won't "stick" to their neighbors, so the
|
|
|
|
compiler treats them as solitary single-character tokens: '~`!@$^&*-+=|<>?/
|
|
|
|
which means you can jam things together:
|
2018-01-25 17:43:57 -08:00
|
|
|
assume ((5++2) is ( 5 ++ 2 )) or barf "ugh"
|
2017-10-04 18:04:33 -07:00
|
|
|
|
|
|
|
|
|
|
|
# Do grouping?
|
|
|
|
# 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
|
|
|
|
# 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:
|
2017-10-19 17:00:10 -07:00
|
|
|
# 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]
|
2017-10-19 17:00:10 -07:00
|
|
|
lua> ".."
|
2017-12-14 13:54:31 -08:00
|
|
|
nomsu:writeln("The OS time is: "..os.time());
|
2017-10-04 18:04:33 -07:00
|
|
|
say the time
|
2017-10-19 17:00:10 -07:00
|
|
|
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-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
|
|
|
|
%body_lua <- (%body as lua)
|
|
|
|
return {..}
|
|
|
|
locals: %body_lua's "locals"
|
|
|
|
statements: ".."
|
|
|
|
if not \(%condition as lua expr) then
|
|
|
|
\((%body_lua's "statements") or "\(%body_lua's "expr");")
|
|
|
|
end
|
2017-10-04 18:04:33 -07:00
|
|
|
|
2018-01-19 18:13:02 -08:00
|
|
|
if (1 > 10) 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!"
|
|
|
|
|
2018-01-19 18:13:02 -08:00
|
|
|
if (1 > 10) 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!"
|
|
|
|
|