BLANG
OVERVIEW
blang is a desktop rpn calculator.
DOWNLOAD
blang is now open source. i am not updating it at the moment, because i want to focus on other things (robots)!
read the README.md for steps to get blang installed and set up.
once you have blang set up, dont forget to......rtfm (read the friendly manual)!
THE PROGRAM
when you open the program, you will see a few panels.
*** since the creation of this page, i added context menu interaction, so some tabs look different.
THE STACK
the stack is the core of any RPN calculator. instead of typing expressions and having the result of their evaluation displayed, values are pushed to the stack and operated on. you can enter values onto the stack using the lineedit underneath it (see 'entering values').
THE LOG
this is where you will see error messages. occasionally, you will see a python error message. this either means you've discovered a bug or i haven't implemented a proper error message for that particular error.
VARW
varw is the variable writer. it's where you can save and retrieve values for use in calculations. some common constants come built-in. double click on a value to push it to the stack. to define a new value, either enter its name and a value (which can be any of blangs 3 types, more on that later) or its name and an index on the stack. the corresponding value on the stack will be saved under that name. in the furure, you will be able to right click on variables in the stack to save them. to delete a value, select it, and click the small x button at the bottom. in the future, you will be able to right click on variables to delete them.
FNWRTR
fnwrtr is the function writer. on the left, folders of saved functions are listed in a tree. you can double click on the functions in the tree to edit them. the larger panel on the right is where you do so. it is also where you write new functions to save. more on writing functions later...
MATRW
my favorite calculator is the hp50g. one of my favorite features it has is the 'matrw', or matrix writer. matrices are very useful structures for data, but they are usually somewhat unweildly to create in calculators. the way the matrix writer in the 50g handled it was brilliant. this is my attempt at faithfully recreating it. you can compose your matrices in the table and send them to the stack when they're done.
note: be sure to hit enter or click out of a cell once you've entered a value before hitting the done button. if you do not, that value may not be saved in the matrix.
ENTERING VALUES
this program is fairly flexible. you can enter integers (which get turned into floats), decimal numbers, or numbers in scientific notation. you can also enter lists and strings. finally, you can enter the names of variables that are saved and their values will be pushed to the stack. in the future, you will be able to type values with units attached to them to create unit objects.
ENTERING FUNCTIONS
to modify the values that are on the stack, you enter functions. functions in blang are prefixed with a '.'.
> .add // adds the x and y registers
functions can have arguments, like so:
> .iota(8) // constructs a list of integers (not inclusive)
arguments can be numbers, strings, or lists:
> .int('name' 0 2) // this integrates 'name' from 0 to 2
> .example([1,2,3]) // no builtin functions currently take lists
most builtin functions pops two values, performs an operation on them, and pushes the result. builtin functions are listed below.
*pop is a stack operation -- it means to take the top element (most recently added) off the stack.
*push is like pop, but it places a value on the top (the position with the smallest index) of the stack.
ENTERING VALUES AND FUNCTIONS -- IMPORTANT NOTE
in this program, you can enter multiple values or functions at once. they must be seperated with spaces. they will be interpreted from left to right.
for example:
> 2 8 .add
will push a 2 to the stack, push an 8 to the stack, and add them together -- leaving 10.
WRITING FUNCTIONS
this program allows you to write simple functions. the functions you write can be executed just like builtin functions. they can have arguments of any type, but they dont have to have any arguments at all.
in fnwrtr, begin by writing your function as you would like to call it. if i wanted a function called 'acircle' that takes a radius and computes the area of a circle, i would write:
.acircle(r)
*note: if your function doesn't take any arguments, you don't need the parentheses.
then, on the lines following, you can type values or commands just like you would regularly. note that you can now type the functions inputs as if they are program variables. for example:
.acircle(r)
r .sq pi .x
this program pushes the value of r to the stack, squares it (.sq), pushes pi to the stack, and then multiplies those two values together.
*note: the commands and values after the function definition line don't have to be on the same line. it doesn't matter, since functions are parsed with whitespace as a delimiter.
to save your function, choose the folder you would like it saved to and click the check mark. you can now find it in the tree to the left of the text-edit and can double click on it to edit it if you need.
to call your function, type it in the input line-edit.
> .acircle(2) -> 12.57
a cool feature: this function can take in a list of radiuses and will compute the area of their respective circles, element-wise.
> .acircle([1,2,3]) -> [3.14, 12.57, 28.27]
*note: if you use a list as an input to a function, you cannot separate the values with spaces. only commas. i'm working on it--this has been a nightmare bug for a while now. however, you can put all the spaces you want in lists that you enter directly into the standard input.
CONDITIONALS
you can execute functions and push values conditionally. this is mostly for use in functions that you write, although it works just fine in the standard input.
how it works: there are some functions you can execute to compare things. for example:
> .teq(a b) // teq stands for test-equals: it will test if a is equal to b.
there functions for all of the standard comparison operators, and they all take two arguments:
{ teq: ==, tne: !=, tgt: >, tlt: <, tle: <=, tge: >= }
these functions set the runstate of the interpreter. the runstate of the interpreter determines which prefixed functions will be executed.
COMMAND PREFIXING
in order to use conditionals, you prefix functions or values with a 't' or an 'f' (not case sensitive). for example, the following will execute if the runstate of the interpreter is set to true:
T.acircle(2)
t5.82
t[1,2,3]
but if the runstate of the interpreter is false, only the first two lines below will execute:
f.print('false!')
f['false','f']
t.print('true!')
if you've played shenzhen i/o, you'll be familiar with this.
*note: in the future, you will be able to prefix entire lines instead of individual commands.
*note2: in the future, there will be other prefixes for things such as loops.
*note3: in the future, i may eventually realize that this is a really dumb way of doing conditionals and implement a more standard block-based if-else branch style thing.
REGISTERS
a quick terminology thing: the x register refers to index 0 of the stack (the most recently added value). the y register refers to index 1 of the stack.
WRAPPING UP
that should be all you need to know for now. there are many more features planned, such as saving your defined functions instead of them dissapearing when you close the program (i'll get to that soon, i promise). i also plan to implement a more fully-fledged syntax for writing functions. also, i'm definitely going to add units and unit conversions.
*note: if you want to see what blang looked like in its earlier versions:
www.youtube.com/watch?v=qq4OFrT4vvE&t=35s
(wasn't the syntax so clunky?)
FUNCTIONS
*note: it is implied that functions without arguments consume the values on the stack.
.pop // pops the x register off the stack
.add // adds the x and y register
.sub // subtracts the x register from the y register
.x // multiplies the x and y registers
.div // divides the y register by the x register
.sqrt // calculates the square root of the x register
.sq // squares the x register
.ln // calculates the natural logarithm of the x register
.lby // calculates the logarithm of the x register with base y
.sin // calculates the sine of the x register in radians
.sind // calculates the sine of the x register in degrees
.cos // calculates the cosine of the x register in radians
.cosd // calculates the cosine of the x register in degrees
.tan // calculates the tangent of the x register in radians
.tand // calculates the tangent of the x register in degrees
.asin // calculates the inverse sine (arcsine) of the x register in radians
.asind // calculates the inverse sine (arcsine) of the x register in degrees
.acos // calculates the inverse cosine (arccosine) of the x register in radians
.acosd // calculates the inverse cosine (arccosine) of the x register in degrees
.atan // calculates the inverse tangent (arctangent) of the x register in radians
.atand // calculates the inverse tangent (arctangent) of the x register in degrees
.chs // changes the sign of the x register
.rec // calculates the reciprocal of the x register
.exp // calculates e raised to the power of the x register
.xry // takes the x'th root of the y register
.xty // raises x to the power of y
.dot // calculates the dot product of two vectors (lists)
.norm // normalizes a vector (turns it into a unit vector)
.cross // calculates the cross product of two vectors
.swap // swaps the x and y registers
.up // rolls the stack up
.down // rolls the stack down
.dup // duplicates the x register and pushes it onto the stack
.get // retrieves the value at a specific stack index and pushes it onto the stack
.det // calculates the determinant of a matrix
.inv // calculates the inverse of a matrix
.trans // calculates the transpose of a matrix
.solve // solves a system of linear equations -- x register is A and y register is b
.sum // sums the stack
.prod // calculates the product of the stack
.int('name' a b) // calculates the definite integral of the function 'name'
.dif('name' a) // calculates the derivative of the function 'name' at a
.roots('name' a b) // calculates the root of function 'name' between a and b
.c // copy!
.v // paste!
.teq(a b) // tests if a and b are equal
.tne(a b) // tests if a and b are not equal
.tgt(a b) // tests if a is greater than b
.tlt(a b) // tests if a is less than b
.tge(a b) // tests if a is greater than or equal to b
.tle(a b) // tests if a is less than or equal to b
.clear // clears the stack
.return // clears the stack, leaving the top value (index 0)
.iota(a) // creates a list of whole numbers up to a
.unpack // unpacks a list onto the stack
.pack // packs the stack into a list
.reverse // reverses the stack
.eye(a) // creates a diagonal a-by-a matrix of ones
.help // opens this page :)
the following are not yet doccumented and soon to be deprecated:
.var *
.varind *
.pvar *
.pvarind *
.getpvar *
.rm *
.clearvars *
.clearpvars *
.clearbfunctions *
.clearlog *
.clearalldata *