nomsu/examples/how_do_i.nom

276 lines
7.0 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
#.. 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
# Declare a variable?
2018-01-19 18:13:02 -08:00
local %x
# Otherwise, variables are implicitly global (hah, sorry, I know that sucks)
# Set a variable?
set %x = 1
set %str = "Hello world"
2017-10-04 18:04:33 -07:00
# Expressions that are more than just literal values require parentheses:
2018-01-19 18:13:02 -08:00
set %x = (2 + 3)
2017-10-04 18:04:33 -07:00
# Modify a variable?
2018-01-19 18:13:02 -08:00
set %x = 100
2017-10-04 18:04:33 -07:00
%x += 1
# Print stuff?
say "Hello world!"
# Define a mutli-line string?
2018-01-19 18:13:02 -08:00
set %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-19 18:13:02 -08:00
set %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?
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-19 18:13:02 -08:00
set %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-19 18:13:02 -08:00
set %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-19 18:13:02 -08:00
set %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-19 18:13:02 -08:00
set %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
# These do the same thing:
say (%my_list's 1)
say (1 in %my_list)
# List entries can be modified like this:
2018-01-19 18:13:02 -08:00
set (%my_list -> 1) = "ONE!!!"
2017-10-04 18:04:33 -07:00
say (size of %my_list)
# Define a dictionary/hash map?
2018-01-19 18:13:02 -08:00
set %my_dict = {x: 99, y: 101}
set %my_dict = {..}
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
say (%my_dict->"x")
2018-01-19 18:13:02 -08:00
set (%my_dict->"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-19 18:13:02 -08:00
set %list = [1,2,3]
for %x in %list
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
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
say "For %i from 1 to 3 loop #\%i"
2018-01-19 18:13:02 -08:00
for all 1 to 3
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
say "Even #\%even"
2018-01-19 18:13:02 -08:00
for %backwards from 3 to 1 by -1
say "Backwards #\%backwards"
2017-10-04 18:04:33 -07:00
# While loops:
2018-01-19 18:13:02 -08:00
set %x = 1
repeat while: %x <= 3
say "repeat while loop #\%x"
2017-10-04 18:04:33 -07:00
%x += 1
2018-01-19 18:13:02 -08:00
set %x = 1
repeat until: %x > 3
say "repeat until loop #\%x"
2017-10-04 18:04:33 -07:00
%x += 1
# Infinite loop:
2018-01-19 18:13:02 -08:00
set %x = 1
repeat
say "repeat loop #\%x"
2017-10-04 18:04:33 -07:00
%x += 1
2018-01-19 18:13:02 -08:00
if: %x > 3
stop repeat-loop
2017-10-04 18:04:33 -07:00
# GOTOs:
2018-01-19 18:13:02 -08:00
do
set %x = 1
2017-10-04 18:04:33 -07:00
-> %again
say "GOTO loop #\%x"
2017-10-04 18:04:33 -07: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]
set %f1 = 0
set %f2 = 1
repeat
set %tmp = (%f1 + %f2)
set %f1 = %f2
set %f2 = %tmp
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 [..]
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"
#.. 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]
say %what_she_said
2017-10-04 18:04:33 -07:00
say "-- she said"
"Howdy pardner" is what she said
#.. 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 !]
say %foo_bar
say %
2017-10-04 18:04:33 -07:00
say %1
>> "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)
#.. 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-19 18:13:02 -08:00
assume: (5++2) is ( 5 ++ 2 )
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:
# 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> ".."
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
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-19 18:13:02 -08:00
# Or to transform nomsu code into custom lua code using "compile % to code %"
compile [if %condition on opposite day %body] to code ".."
if not (\(%condition as lua)) then
\(%body as lua statements)
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!"