212 lines
6.6 KiB
Plaintext
212 lines
6.6 KiB
Plaintext
(# Comments use (# ... #), and can be nested #)
|
|
|
|
(# Import files like so: #)
|
|
run file "core.nom"
|
|
|
|
(# Numbers: #)
|
|
23
|
|
4.5
|
|
(# Since this language cross-compiles to lua, integers and floating point numbers are
|
|
both represented using the same primitive. #)
|
|
|
|
(# Strings: #)
|
|
"asdf"
|
|
".."
|
|
|This is a multi-line string
|
|
|that starts with ".." and includes each indented line that starts with a "|"
|
|
|until the indentation ends
|
|
|
|
(# Lists: #)
|
|
[1,2,3]
|
|
[..]
|
|
"like multi-line strings, lists have an indented form", "that can use commas too"
|
|
"or just newlines to separate items"
|
|
5
|
|
6,7,8
|
|
|
|
(# Function calls: #)
|
|
say "Hello world!"
|
|
|
|
(# Function definition: #)
|
|
rule "say both %first and also %second":
|
|
(# Variables use the "%" sign: #)
|
|
say %first
|
|
say %second
|
|
|
|
(# 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: #)
|
|
rule "%what-she-said is what she said":
|
|
say %what-she-said
|
|
say "-- she said"
|
|
"Howdy pardner" is what she said
|
|
|
|
(# The language only reserves []{}().,:;% as special characters, so functions and variables
|
|
can have really funky names! #)
|
|
rule ">> %foo-bar###^ --> %@@& _~-^-~_~-^ %1 !":
|
|
say %foo-bar###^
|
|
say %@@&
|
|
say %1
|
|
>> "wow" --> "so flexible!" _~-^-~_~-^ "even numbers can be variables!" !
|
|
(# Though literals can't be used in function names #)
|
|
|
|
(# 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 #)
|
|
rule "%a ++ %b": 2 * (%a + %b)
|
|
say (2 ++ 3)
|
|
|
|
|
|
(# Code blocks start with ":" and either continue until the end of the line
|
|
or are indented blocks #)
|
|
|
|
(# One liner: #)
|
|
: say "hi"
|
|
|
|
(# Block version: #)
|
|
:
|
|
say "one"
|
|
say "two"
|
|
|
|
(# So the function definitions above are actually just passing a regular string, like
|
|
"say both %first and also %second", and a code block to a function called "rule % %"
|
|
that takes two arguments. #)
|
|
|
|
(# Line continuations work by either ending a line with ".." and continuing with an indented block: #)
|
|
say..
|
|
both "Tom" and
|
|
also
|
|
"Sawyer"
|
|
|
|
(# Or by starting the next line with ".." #)
|
|
say both "Bruce"
|
|
..and also "Lee"
|
|
|
|
(# This can be mixed and matched: #)
|
|
say both..
|
|
"Rick"
|
|
..and also..
|
|
"Moranis"
|
|
|
|
(# And combined with the block forms of literals: #)
|
|
say both ".."
|
|
|Four score and seven years ago our fathers brought forth, upon this continent,
|
|
|a new nation, conceived in liberty, and dedicated to the proposition that
|
|
|"all men are created equal"
|
|
..and also..
|
|
"-- Abraham Lincoln"
|
|
|
|
rule "my favorite number": return 23
|
|
(# Subexpressions are wrapped in parentheses: #)
|
|
(# printf takes a list of bits that are converted to strings and concatenated together, and printed #)
|
|
printf ["My favorite number is ", my favorite number]
|
|
(# There's a multi-line indented block form for subexpressions too: #)
|
|
printf [..]
|
|
"My favorite number is still ", (..)
|
|
my favorite
|
|
number
|
|
|
|
(# There's a few macros in the language for things like conditional branches and logic/math
|
|
operations, but they can be thought of as basically the same as functions.
|
|
There are no keywords in the language! #)
|
|
if (1 < 10):
|
|
say "One is less than ten"
|
|
..else:
|
|
say "One is not less than ten"
|
|
|
|
(# Breakdown of the above: #)
|
|
(# Function call (actually a macro) to "if % % else %" #)
|
|
(# First argument is a subexpression that is a function call (also a macro) to "% < %"
|
|
that performs a comparison on its arguments, 1 and 10 #)
|
|
(# Second argument is a block of code that includes a function call to "say %", the "if" body #)
|
|
(# Third argument is a block of code that includes a different function call to "say %", the "else" body #)
|
|
|
|
(# Line continuations can be used for "elseif" #)
|
|
if (1 > 10):
|
|
say "First condition"
|
|
..else: if (1 > 5):
|
|
say "Second condition"
|
|
..else:
|
|
say "Last condition"
|
|
|
|
(# ^that's the same as: #)
|
|
if (1 > 10):
|
|
say "First condition"
|
|
..else:
|
|
if (1 > 5):
|
|
say "Second condition"
|
|
..else:
|
|
say "Last condition"
|
|
|
|
(# Variables are modified with a macro, "let % = %" #)
|
|
let "numbers" = [5,6,7]
|
|
|
|
(# Looping: #)
|
|
printf ["Looping over: ",%numbers,"!"]
|
|
for "number" in %numbers:
|
|
say (%number + 100)
|
|
|
|
rule "sing %starting-bottles bottles of beer":
|
|
for "n" in (%starting-bottles down through 0):
|
|
printf [..]
|
|
(%n if (%n > 0) else "No more")
|
|
(" bottle" if (%n == 1) else " bottles")
|
|
" of beer on the wall."
|
|
("" if (%n == 0) else " Take one down, pass it around...")
|
|
|
|
sing 9 bottles of beer
|
|
|
|
|
|
(# Note that because math and logic operations are just macros, they require a lot
|
|
of parentheses to disambiguate. There's no PEMDAS. #)
|
|
say (5 + (4 * (- (1 + (6 + 2)))))
|
|
(# For convenience, +,*,"and", and "or" have been hand defined to work with up to 4 operands: #)
|
|
1 + 2 + 3 + 4
|
|
1 * 2 * 3 * 4
|
|
1 and 2 and 3 and 4
|
|
1 or 2 or 3 or 4
|
|
(# Longer lists can use "sum of %", "product of %", "all of %", and "any of %", respectively, or lots of parentheses. #)
|
|
sum of [1,2,3,4,5,6,7]
|
|
product of [1,2,3,4,5,6,7]
|
|
all of [1,1,1,1,0,1,1]
|
|
any of [0,0,0,0,1,0,0]
|
|
(# And 3-operand chained inequality comparisons have been defined: #)
|
|
1 < 2 <= 3
|
|
|
|
|
|
(# Macros: #)
|
|
(# The "lua block %" and "lua expr %" macros can be used to write raw lua code: #)
|
|
rule "say time":
|
|
lua block ".."
|
|
|io.write("The OS time is: ")
|
|
|io.write(tostring(os.time()).."\n")
|
|
say time
|
|
printf ["Math expression: ", lua expr "(1 + 2*3 + 3*4)^2"]
|
|
(# In the lua environment, "vars" can be used to get local variables/function args, and "game"
|
|
can be used to access the compiler, function defs, and other things #)
|
|
rule "square root of %n":
|
|
return (lua expr "math.sqrt(vars.n)")
|
|
printf ["the square root of 2 is ", square root of 2]
|
|
|
|
(# Macros can be defined as functions that take unprocessed syntax trees and return lua code #)
|
|
(# "macro block %" is for defining macros that produce blocks of code, not values #)
|
|
macro block "unless %condition %body":
|
|
concat [..]
|
|
(# "% as lua expr" and "% as lua block" are two useful helper functions here. #)
|
|
"if not (", %condition as lua expr, ") then"
|
|
(# Extract the inner part of the code block's body and insert it: #)
|
|
"\n ", (lua expr "vars.body.value.value") as lua block
|
|
"\nend"
|
|
|
|
unless (1 > 10):
|
|
say "Macros work!"
|
|
say "It looks like a keyword, but there's no magic here!"
|
|
|
|
(# and "macro %" is for defining macros that produce an expression #)
|
|
macro "%value as a boolean":
|
|
concat ["(not not (", %value as lua expr, "))"]
|
|
macro "yep": "true"
|
|
|