BASIC is a popular programming language known for its simplicity and ease of use. It has been used in various applications, especially in the early days of computing.
AGEBasic draws its inspiration from another programming language called Tiny Basic which is a minimalistic version of BASIC. However, AGEBasic does not strictly adhere to all the rules and limitations of Tiny Basic. Instead, it takes inspiration from Tiny Basic and extends or modifies certain features to better suit the requirements and capabilities designed specifically to be executed in Age of Joy.
With AGEBasic the player can develop it’s own functions to run in the simulation. Allows to control parts of the game that there aren’t available from the YAML configuration or the Visual configuration.
read about the new AGEBasic cabinet event system (version >= 0.6)
AGEBasic program storage
The main storage for AGEBasic programs is /sdcard/Android/data/com.curif.AgeOfJoy/AGEBasic. AGEBasic programs must to end with the .bas prefix, like mixcabinets.bas or changecontrols.bas.
Variables
- A variable name can contain only letters and numbers
- A variable name must start with a letter.
- A variable name cannot contain special characters (like
_) - Can’t be a reserved word (a name of a function or command). You can’t name a variable with the same name as a function or command, example musicPath is an invalid name because the MUSICPATH function exists.
AGEBasic variables can contain numbers (double precision), hexadecimal (preceded by a &) and strings.booleans are not variable type, but anything different than 0 is considered true.
Examples:
Correct:
10 LET A = 10 'a number
20 LET A = 10.10 'a number
30 LET A = "AGE of Joy" 'a string
40 LET A = "10" 'a string
50 LET A = 0 'an hex numberIncorrect:
10 LET MY_VARIABLE = 10
20 LET TEXT = "text"
30 A = 10Sentences
Refers to a single, complete instruction or command that the program executes.
Numbered lines
Each line of code should be numbered and be in ascending order. Multiline is supported.
A Command must be the first after a line number. Functions can be called using a CALL function
10 PRINT "Hello world"Line separation
One line number could contain two sentences using : as separator.
10 LET a = 10 : REM value of a is 10
20 PRINTLN a : PRINTLN "Hello world"Commands
Each line must begin with a line number and a command.
LET: assign a value to a variable, ex:LET a=10LETS: assign multiple values, e.g.:LETS a,b=10,20is the same aslet a=10and nextlet b=20DIM: create an array variable.REM: a comment, ex:REM this is a comment. It’s the only way to register comments in the program.END: finish the programIF/THEN/ELSE: conditional, . Ex:IF x=1 THEN LET a=2 ELSE a=3. There aren’t available the usual conditional expressions likeANDandOR, but there are functions to replace them. IMPORTANT: ENDIF doesn’t existsGOTO: to jump to a line number. Ex:GOTO 50GOSUB: to jump to a line, and to back usingRETURN. Ex:GOSUB 5000RETURN: jump back to the next sentence after theGOSUBCALL: to call a function discarding the result. e.g.:CALL CabRoomReplace(0, "pacman")FOR/TO/NEXT/STEP: to create loops. Ex:for x=0 to 10 step 2 ... next x- Initial, end and step values can be expressions.
- Initial value is computed at start of the cycle.
- The end value is computed during the
NEXTsentence execution. - The
NEXTsentence evaluates if the cycle should repeat. At least one cycle is executed always.
SLEEPto sleep a number of seconds, doesn’t work in programs executed in the control cabinet (has no sense). E.g.:SLEEP 1(sleeps the program during a second).SLEEP 0.5sleeps for half of a second. Values must be greater than0.01.
Operators
- Adding, subtraction, etc.:
+,-,*,/ - Comparison:
=,!=,<>,<,>,<=,>= - Logical: and:
&&or:||
Arrays
1. What is an Array?
At its core, an array is a structured collection of data items, all stored under a single variable name. Imagine a series of numbered compartments, where each compartment can hold a piece of data. Instead of creating many individual variables (e.g., ITEM1, ITEM2, ITEM3), an array allows you to store and access these related pieces of data using a single name followed by an index (or subscript) that indicates the specific position of the item you want to access.
This makes it incredibly efficient to store and process lists or tables of information, especially when the number of items is large or varies during program execution.
2. The Dimensional Concept of Arrays
Arrays can be designed with different “dimensions,” which essentially refers to the number of indices required to pinpoint a specific element.
-
One-Dimensional (1D) Arrays: These are like a simple list or a single row of data. You need only one index to access an element. Example:
A[0],A[1],A[2], … Think of it as a column of numbers:+-----+ | A[0]| +-----+ | A[1]| +-----+ | A[2]| +-----+ -
Two-Dimensional (2D) Arrays: These are like a table or a grid (rows and columns). You need two indices to access an element: one for the row and one for the column. Example:
B[0,0],B[0,1],B[1,0],B[1,1], … Think of it as a spreadsheet:+----------+----------+----------+ | B[0,0] | B[0,1] | B[0,2] | +----------+----------+----------+ | B[1,0] | B[1,1] | B[1,2] | +----------+----------+----------+ -
Multi-Dimensional Arrays (3D and beyond): AGEBasic supports arrays with more than two dimensions. A three-dimensional array, for instance, can be visualized as a cube or a stack of tables. You would need three indices to access an element (e.g.,
MULTI[F,G,X]). While higher dimensions are supported, they become increasingly abstract to visualize but are extremely useful for complex data structures.
Important Note on Indexing:
AGEBasic arrays utilize zero-based indexing. This means the first element in any dimension is at index 0, the second at 1, and so on, up to N-1 where N is the declared size for that dimension.
3. Declaring Arrays with DIM
Before you can use an array, you must declare it using the DIM (Dimension) statement. This tells AGEBasic the name of your array and how much space to reserve for it in memory by specifying the size of each dimension.
Syntax:
DIM arrayName[size1 [, size2, ...]]
arrayName: The name of your array. Follows standard variable naming rules (starts with a letter, contains only letters and numbers, not a reserved word).size1, size2, ...: The size of each dimension. This number represents the total count of elements in that specific dimension. For example,DIM A[5]declares an array with 5 elements, accessible fromA[0]toA[4].
Examples:
-
One-Dimensional Array:
10 DIM myNumbers[10] : REM Declares an array named 'myNumbers' with 10 elements (indices 0 to 9) -
Two-Dimensional Array:
20 DIM gridData[3, 5] : REM Declares a 2D array: 3 rows (indices 0-2), 5 columns (indices 0-4) -
Multi-Dimensional Array (e.g., 3D):
30 DIM cubeData[2, 3, 4] : REM Declares a 3D array with dimensions: (indices 0-1, 0-2, 0-3)
4. Accessing and Assigning Array Elements
Once an array is declared, you can access individual elements for reading or writing by specifying the array name followed by the appropriate indices enclosed in square brackets [].
Assigning Values:
Use the LET command to assign a value to a specific array element.
Syntax:
LET arrayName[index1 [, index2, ...]] = value
Example:
10 DIM prices[5]
20 LET prices[0] = 10.50
30 LET prices[1] = 22.99
40 LET prices[4] = 5.00
50 DIM matrix[2,2]
60 LET matrix[0,0] = "Top-Left"
70 LET matrix[1,1] = "Bottom-Right"Dinamic array creation:
You can create an array using the ARRAY() function:
10 LET RGB_BLUE = ARRAY(0,0,255)
20 PRINTLN "B:" = STR(RGB_BLUE[2])Retrieving Values:
You can use array elements just like regular variables in expressions, PRINT statements, or other LET assignments.
Syntax:
arrayName[index1 [, index2, ...]]
Example:
100 DIM myValues[3]
110 LET myValues[0] = 100
120 LET myValues[1] = myValues[0] * 2 : REM myValues[1] becomes 200
130 PRINTLN "Value at index 0: " + STR(myValues[0])
140 PRINTLN "Value at index 1: " + STR(myValues[1])5. Getting Array Length with LEN()
The LEN() function, traditionally used for string lengths, has been extended to provide the total number of elements in an array. This is particularly useful for certain iteration patterns or for determining the overall capacity of an array.
Syntax:
LEN(arrayName)
Return Value: A number representing the total count of elements in the array (product of all dimension sizes).
Example:
10 DIM singleDim[5]
20 LET totalElements1 = LEN(singleDim) : REM totalElements1 will be 5
30 DIM twoDim[2,3]
40 LET totalElements2 = LEN(twoDim) : REM totalElements2 will be 2 * 3 = 6
50 DIM multiDim[2,3,4]
60 LET totalElements3 = LEN(multiDim) : REM totalElements3 will be 2 * 3 * 4 = 246. Mixed-Type Arrays
Unlike some programming languages, AGEBasic arrays are flexible in type. This means a single array can store a mix of numbers (double precision, hexadecimal) and strings within its elements. The TYPE() function can be used to check the specific type of an element if needed during runtime.
Example:
10 DIM mixedData[2]
20 LET mixedData[0] = 123.45 : REM A number
30 LET mixedData[1] = "Hello, AGEBasic!" : REM A string
40 IF TYPE(mixedData[0]) = "NUMBER" THEN PRINTLN "Element 0 is a number."
50 IF TYPE(mixedData[1]) = "STRING" THEN PRINTLN "Element 1 is a string."Short Example of Array Use
Here’s a concise AGEBasic program demonstrating a one-dimensional array to store and print a list of scores.
10 REM Short Array Example: High Scores
20 CLS
30 DIM highScores[5] ' Declare an array to hold 5 scores
40 LET highScores[0] = 1000
50 LET highScores[1] = 750
60 LET highScores[2] = 900
70 LET highScores[3] = 500
80 LET highScores[4] = 1200
90 PRINTLN "--- High Scores ---", 0, 0
100 FOR i = 0 TO 4
110 LOCATE 0, GETY() + 1 ' Move cursor to next line
120 PRINTLN "Score " + STR(i+1) + ": " + STR(highScores[i]), 0, 0
130 NEXT i
140 SHOW
150 SLEEP 3 ' Wait for 3 seconds
160 ENDExplanation:
This program first clears the screen. It then declares a one-dimensional array highScores capable of holding 5 numeric elements. It populates this array with example scores. A FOR loop then iterates from index 0 to 4 (the valid range for a 5-element array), printing each score along with its position. The LOCATE and GETY() functions ensure each score appears on a new line. Finally, SHOW updates the screen and the program pauses before ending.
We hope this comprehensive overview of arrays in AGEBasic helps you leverage this powerful new feature in your programs! Happy coding!
General functions
AGEBasic Functions can receive parameters. Parameters must be enclosed.
Math
ABS,COS,SIN,TAN,MODINT: integer part of a numberMAX,MIN: of two numbers, Ex:LET a = max(10,20)thenais 20RND: a random between two numbers.rnd(2,10)could be8.NOT: The inverse of the boolean expression. Ex:if NOT(0) then goto 100AND: to combine two expressions. It returns1if both of the input conditions are!= 0, otherwise it returns0:AND(a = 0, b = 1)OR: to combine two expressions. It returns1if at least one of the input conditions is!= 0, otherwise it returns0.IIF(condition, value1, value2)returnsvalue1ifconditionistrueelse returnsvalue2HEXTODEC(string): convert from a Hexadecimal string (like"FF") to a number. Remember an hex number is represented by a&also. ExampleHEXTODEC("FF") = &FFVAL(string): to coarse a string to a number, inversed ofSTR(number). Example:VAL("10.5") = 10.5ARRAY(val1, val2[, val3,...]: Create an array with the specified values.
Strings
LEN,UCASE,LCASE,SUBSTR: Ex:susbtr("abc", 1, 2)is “bc”, starting in 1 and getting two characters.TRIM,LTRIM,RTRIM.STR: Ex:str(10)is"10"StringMatch(string val, string pattern)checks if a substring (pattern) exists within a string (val), returning1if found and0if not. For example,StringMatch("Hello, World!", "World")returns1, whileStringMatch("Hello, World!", "Earth")returns0.
List simulation
AGEBasic can’t handle arrays or lists, but you can simulate them using character separated strings like aaa:bbb for example. aaa is the member in the position 0, bbb is the one in the position 1 and the separator is :.
GetMember(string, member #, separator): to get a slice of a string. Can be used to emulate lists. Example to get the first member of a list:GetMember("AGE:of:Joy", 0, ":") = "AGE"CountMembers(string, separator)to count how many members a list have:CountMembers("AGE:of:Joy", ":") = 3IsMember(string list, string member, separator)returnstrueif the second string is a member of the first string list.IndexMember(string list, string member, separator)returns the index position of the string in the string list if found, or-1if not found. Index starts in0and ends inCountMembers()-1RemoveMember(string list, string member, separator)returns a new string list without the specified string member.AddMember(string list, string member, separator)returns a new string list with a new member at the end
Introspection
exists(string): to know if a variable is defined, returns 1 or 0 (true or false). Example:if (exists("myvariable")) then goto 100jumps to the line # 100 if the variable"myvariable"was previously assigned.type(var): returns"STRING"if the variable is a string or"NUMBER"if is a number. Exampleif (type(a) == "STRING") the goto 180jumps to the line # 180 if the variable a is previously assigned with a string likelet a = "test"
Screen
Character functions
PRINTto show text on the screen:PRINT x,y, text [, 0/1 [, 0/1]]x,yscreen coordinates (x: cols, y: rows)text: to print1inversed,0normal. Optional parameter, 0 is the default.1show immediately,0don’t show and wait for theSHOWcommand (recommended)
PRINTLNto show text on the screen with internalx, ycontrol:PRINTLN text [, 0/1 [, 0/1]]text: to print1inversed,0normal. Optional parameter, 0 is the default.1show immediately,0don’t show and wait for theSHOWcommand (recommended)
CLSto clear the screenSHOW: to print in the screen the last executed screen commands.LOCATE X, Y: Moves the text cursor to character columnXand rowY.X: Horizontal character position.0(left) toSCREENWIDTH() - 1(right).Y: Vertical character line.0(top) toSCREENHEIGHT() - 1(bottom).
GETX(): Returns the current horizontal character column of the text cursor.- Return value is a Number, from
0(leftmost) toSCREENWIDTH() - 1(rightmost).
- Return value is a Number, from
GETY(): Returns the current vertical character row (line) of the text cursor.- Return value is a Number, from
0(topmost) toSCREENHEIGHT() - 1(bottommost).
- Return value is a Number, from
PRINTCENTERED Y, TEXT [, INVERTED [, DRAW_FLAG]]: PrintsTEXTcentered horizontally on character rowY.Y(Number): The character row (line) to print on.0is the top row.TEXT(String or Number): The text or number to print. Numbers will be converted to strings.INVERTED(Boolean, optional, default0orFALSE): If1(orTRUE), uses inverted foreground/background colors for this print.DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen is updated immediately after printing. Set to0(orFALSE) to batch multiple drawing operations before callingSHOWor another command that draws.
FGCOLORandBGCOLORcommands to set colors depending on the type of screen.RESETCOLORandINVERTCOLORas variants.SCROLL N_LINES [, FILL_COLOR_SPEC [, DRAW_FLAG]]: Scrolls the entire character display area vertically.N_LINES(Number): The number of character lines to scroll.- Positive values scroll content DOWN (new blank space appears at the top).
- Negative values scroll content UP (new blank space appears at the bottom).
- A value of
0results in no scroll.
FILL_COLOR_SPEC(Color, optional): The color to fill the new blank lines created by scrolling.- Can be a string color name (e.g.,
"black","blue"). - Can be three numbers representing R, G, B values (e.g.,
0,0,0for black). - If omitted, the current default background color .
- Can be a string color name (e.g.,
DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen is updated immediately. Set to0(orFALSE) to defer the screen update.
SCROLLRECT CX, CY, CW, CH, N_LINES [, FILL_COLOR_SPEC [, DRAW_FLAG]]: Scrolls a rectangular sub-region of the character display vertically.CX(Number): The starting character column (X-coordinate) of the rectangle’s top-left corner.CY(Number): The starting character row (Y-coordinate) of the rectangle’s top-left corner.CW(Number): The width of the rectangle in characters. Must be greater than 0.CH(Number): The height of the rectangle in characters. Must be greater than 0.N_LINES(Number): The number of character lines to scroll within the specified rectangle.- Positive values scroll content DOWN within the rectangle.
- Negative values scroll content UP within the rectangle.
- A value of
0results in no scroll.
FILL_COLOR_SPEC(Color, optional): The color to fill the new blank lines created by scrolling within the rectangle.- Can be a string color name (e.g.,
"black","blue"). - Can be three numbers representing R, G, B values (e.g.,
0,0,0for black). - If omitted, the current default background color (
ScreenGenerator.charBackgroundColor) is used.
- Can be a string color name (e.g.,
DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen is updated immediately. Set to0(orFALSE) to defer the screen update.
SETCOLORSPACEallows to simulate a computer type (like “c64”):- c64
- ibmpc
- amstrad
- cpc
- zx
- apple2
- cpc_mono
- msx
- msx_mono
- to7
SCREENWIDTH(): returns the screen width in characters. First is0last isScreenWidth() - 1SCREENHEIGHT(): returns the Height in lines. First is0last isScreenHeight() - 1SCREENSIZE(): returns an array with two positions: Width and Height.
Drawing functions
All of them starts withD.
DCHARPIXELX(CHAR_X, CHAR_Y): Returns the screen pixel X-coordinate of the character cell specified byCHAR_X(column) andCHAR_Y(row).CHAR_X(Number): The character column, from0(left) toSCREENWIDTH() - 1.CHAR_Y(Number): The character row, from0(top) toSCREENHEIGHT() - 1.- Returns a Number. If character coordinates are invalid or out of bounds, may return
-1.
DCHARPIXELY(CHAR_X, CHAR_Y): Returns the screen pixel Y-coordinate of the character cell specified byCHAR_X(column) andCHAR_Y(row).CHAR_X(Number): The character column, from0(left) toSCREENWIDTH() - 1.CHAR_Y(Number): The character row, from0(top) toSCREENHEIGHT() - 1.- Returns a Number. If character coordinates are invalid or out of bounds, may return
-1.
DPSET PX, PY, COLOR_SPEC [, DRAW_FLAG]: Draws a single pixel at the specified pixel coordinates(PX, PY)with the given color.PX(Number): The horizontal pixel coordinate (X-axis).0is typically the leftmost pixel.PY(Number): The vertical pixel coordinate (Y-axis).0is typically the bottommost pixel in Unity’s texture coordinate system (but can vary based on screen setup).COLOR_SPEC(Color): The color for the pixel.- Can be a string color name (e.g.,
"red","white"). - Can be three numbers representing R, G, B values (e.g.,
255,0,0for red).
- Can be a string color name (e.g.,
DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen is updated immediately after setting the pixel. Set to0(orFALSE) to defer the screen update (useful for batching multiplePSETor other drawing calls before a singleSHOW.
DLINE PX1, PY1, PX2, PY2, COLOR_SPEC [, DRAW_FLAG]: Draws a line from pixel coordinates(PX1, PY1)to(PX2, PY2)using the specified color.PX1(Number): The horizontal pixel coordinate (X-axis) of the line’s starting point.PY1(Number): The vertical pixel coordinate (Y-axis) of the line’s starting point.PX2(Number): The horizontal pixel coordinate (X-axis) of the line’s ending point.PY2(Number): The vertical pixel coordinate (Y-axis) of the line’s ending point.COLOR_SPEC(Color): The color for the line.- Can be a string color name (e.g.,
"blue","yellow"). - Can be three numbers representing R, G, B values (e.g.,
0,255,0for green).
- Can be a string color name (e.g.,
DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen is updated immediately after drawing the line. Set to0(orFALSE) to defer the screen update.
DBOX PX, PY, PWIDTH, PHEIGHT, BORDER_COLOR_SPEC [, FILL_FLAG [, FILL_COLOR_SPEC [, DRAW_FLAG]]]: Draws a rectangle (box).PX, PY(Numbers): Pixel coordinates of the top-left corner.PWIDTH, PHEIGHT(Numbers): Width and height of the box in pixels. Must be > 0.BORDER_COLOR_SPEC(Color): Color for the box’s border.- Can be a string color name or 3 numbers (R,G,B).
FILL_FLAG(Boolean, optional, default0orFALSE): If1(orTRUE), the box is filled.FILL_COLOR_SPEC(Color, optional): Color to fill the box ifFILL_FLAGis true.- Can be a string color name or 3 numbers (R,G,B).
- If
FILL_FLAGis true andFILL_COLOR_SPECis omitted, the fill color may default to theBORDER_COLOR_SPEC(behavior defined byScreenGenerator.DrawBox).
DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen updates immediately.
DCIRCLE PCX, PCY, PRADIUS, BORDER_COLOR_SPEC [, FILL_FLAG [, FILL_COLOR_SPEC [, DRAW_FLAG]]]: Draws a circle.PCX, PCY(Numbers): Pixel coordinates of the circle’s center.PRADIUS(Number): Radius of the circle in pixels. Must be > 0.BORDER_COLOR_SPEC(Color): Color for the circle’s border.FILL_FLAG(Boolean, optional, default0orFALSE): If1(orTRUE), the circle is filled.FILL_COLOR_SPEC(Color, optional): Color to fill the circle ifFILL_FLAGis true.DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen updates immediately.
DOVAL PCX, PCY, PRADIUSX, PRADIUSY, BORDER_COLOR_SPEC [, FILL_FLAG [, FILL_COLOR_SPEC [, DRAW_FLAG]]]: Draws an ellipse (oval).PCX, PCY(Numbers): Pixel coordinates of the oval’s center.PRADIUSX(Number): Horizontal radius (half-width) of the oval in pixels. Must be > 0.PRADIUSY(Number): Vertical radius (half-height) of the oval in pixels. Must be > 0.BORDER_COLOR_SPEC(Color): Color for the oval’s border.FILL_FLAG(Boolean, optional, default0orFALSE): If1(orTRUE), the oval is filled.FILL_COLOR_SPEC(Color, optional): Color to fill the oval ifFILL_FLAGis true.DRAW_FLAG(Boolean, optional, default1orTRUE): If1(orTRUE), the screen updates immediately.
DSCREENWIDTH(): returns the screen width in pixels. First is0last isDSCREENWIDTH() - 1
DSCREENHEIGHT(): returns the Height in pixels. First is0last isDSCREENHEIGHT() - 1DSCREENSIZE(): returns an array with two positions: Width and Height in pixels.
Screen SHOW Command
100 SHOWThe SHOW command is responsible for updating the visible screen with any pending graphical changes. Many drawing and text manipulation commands make modifications to an internal screen buffer. These changes are not immediately visible on the screen until SHOW is called, or until a command with an implicit or explicit “draw immediately” flag is executed.
Think of it like an artist painting on a canvas hidden behind a curtain. The SHOW command pulls back the curtain to reveal all the changes made since the last time the curtain was pulled back.
Special characters
You can escape AGEBasic characters map codes in a string only to print it. The rest of the string functions doesn’t take in count the escaped characters, e.g. STR("a\23") = 4.
Room related
To get information about rooms
RoomName(): The name of the room where the player is.RoomCount(): How many Rooms are in the game.RoomGet(number): Get the name of the room. Ex:LET name, desc = RoomGet(0)to get the name and the description of the Room number zero.RoomGetName(number): get the name of the room.RoomGetDesc(number): to get the description.
Cabinets related
Functions for deployed cabinets in rooms
Applies to cabinets deployed in the room where the cabinet controller is loaded.
CabRoomCount(): how many cabinets are in the roomCabRoomGetName(number): get the cabinet name by its index in the room.CabRoomReplace(number, cabinet name): replace a cabinet by another.
Functions for cabinets administration
Applies to cabinet database (registry.yaml) and the Cabinets database storage.
CabDbCount(): how many cabinets registered in the storageCabDbCountInRoom(string): how many cabinets are assigned to one particular room. Ex:LET count = CabDbCountInRoom("Room001")CabDBGetName(number): get a cabinet name using the position in the storage. Ex:CabDBGetName(30)could return “pacman”CabDBSearch(string name, string separator): returns a simulated list separated byseparatorof cabinets that starts withname. Example:CabDBSearch("ju", "|")could return"junofst|jupiter". ifnameis"#"will return all the games starting with special characters.CabDBSearchArray(string name): likeCabDBSearchbut returns an array.CabDBGetAssigned(room, cabinetIndex): returns the cabinet name assigned to a position in a room.CabDBDelete(currentRoomName, cabinetIndex): delete the cabinet assignment to a room in the database (frees the position).CabDBAdd(room, cabinetIndex, newCabinetName): to add a new room/position/cabinet in the database, if the position is taken the program will fail.CabDBAssign(room, cabinetIndex, newCabinetName): assign a cabinet to a existent position in DB.CabDBSave: save the database and its changes.
CabDBDelete,CabDBAssign, CabDBSave and CabDBAdd returns 0 if fails, 1 if not.
Functions that only applies to a cabinet.
In programs related with the cabinet, and packed inside a Cabinet Asset.
Cabinet’s parts are named (see the Configuring cabinet parts) and are identified by its position too. You can use one or each other when you need to call a function who uses a cabinet part, an index is preferred if you will call more than one function for the same part (by performance considerations).
The program fail when you name a part incorrectly or when the index is incorrect. See Debug mode to learn how to debug your program.
CabInsertCoin(): insert a coin in the cabinet.CabPartsCount(): return the cabinet’s parts count.CabPartsName(idx): given a part number (starting in cero), return the name of the part, e.g.:CabPartsName(7)returns “joystick”.CabPartsPosition(name): given the name of a part return it’s position on the Cabinet parts list.CabPartsEnable(idx, enable): given a part number or a part name (idx), and a Boolean (remember Booleans are numbers, a true value is anything different to cero), will disable or enable it. When a part is disabled you can’t see it in VR.CabPartsList(string separator)return a string list of cabinet’s part ordered by its position.- Position in space:
CabPartsGetCoordinate(idx, string type)to get the position in 3D space.typecould be “X”, “Y” or “Z” . Refers to the position of the object starting on the cabinet’s base (local coordinate). You can also useCabPartsGetGlobalCoordinate()to get the part coordinates in the Global 3D space.CabPartsSetCoordinate(idx, string type, number coord), likeCabPartsGetCoordinatebut to set the part’s position relative to the cabinet.CabPartsSetGlobalCoordinate()is also available.CabPartsGetRotation(idx, string type)andCabPartsSetRotation(number part idx, string type, number angle)to get and set the rotation of a cabinet part. NOTE: TheCabPartsRotatecould get better results.CabPartsRotate(idx, string type, number angle)to rotate locally a part. Available in v0.6.CabPartsGetGlobalRotation()andCabPartsSetGlobalRotation()to get set the global rotation.
CabPartsGetTransparency(idx): returns the part’s transparency percentage.CabPartsSetTransparency(idx, percentage): set part’s transparency to a percentage (0 to 100).CabPartsSetEmission(idx, true/false): activate the emissive material on the part if it’s possible. You should probable set an emission color too.CabPartsSetEmissionColor(idx, r, g, b)to set the emission color. The color will blend with the main texture, if any.CabPartsSetColor(idx, r, g, b)to set the color. The color will blend with the main texture, if any.
Examples:
10 let base = CabPartsPosition("joystick-base")
20 lets baseX, baseZ, baseH = CabPartsGetCoordinate(base, "X"), CabPartsGetCoordinate(base, "Z"), CabPartsGetCoordinate(base, "H")
30 let transp = CabPartsGetTransparency("bezel")
40 call CabPartsSetTransparency("bezel", transp + 10)
70 call CabPartsEmission("joystick-button", 1)
80 call CabPartsSetEmissionColor("joystick-button", 190, 20, 20)
90 call CabPartsSetColor("left", 200, 0, 0)Audio parts in cabinets
According to the yaml cabinet configuration you can set a part of a cabinet to be a speaker. You can also change some properties on AGEBasic:
CabPartsAudioPlay(name): Play the audio associated with the specified part. Given the name of a part, it triggers the audio playback. For example,CabPartsAudioPlay("speaker")will start playing the audio from the part named “speaker”.CabPartsAudioStop(name): Stop the audio associated with the specified part. Given the name of a part, it stops the audio playback. For example,CabPartsAudioStop("speaker")will stop the audio from the part named “speaker”.CabPartsAudioPause(name): Pause the audio associated with the specified part. Given the name of a part, it pauses the audio playback. For example,CabPartsAudioPause("speaker")will pause the audio from the part named “speaker”.CabPartsAudioVolume(name, volume): Set the audio volume for the specified part. Given the name of a part and a volume value (0.0 to 1.0), it adjusts the volume. For example,CabPartsAudioVolume("speaker", 0.5)will set the audio volume of the part named “speaker” to 50%.CabPartsAudioDistance(name, minDistance, maxDistance): Set the minimum and maximum distance for 3D audio effects for the specified part. Given the name of a part and the min/max distances, it adjusts the 3D audio settings. For example,CabPartsAudioDistance("speaker", 1.0, 5.0)sets the 3D audio distance for the part named “speaker”.CabPartsAudioFile(name, filePath): Assign an audio file to the specified part. Given the name of a part and the file path, it loads the audio file into the part. For example,CabPartsAudioFile("speaker", "gong.mp3")assigns the “gong.mp3” file to the part named “speaker”.CabPartsAudioLoop(name, loop): Set the looping behavior for the specified part’s audio. Given the name of a part and a boolean value (trueorfalse), it enables or disables audio looping. For example,CabPartsAudioLoop("speaker", true)enables looping for the part named “speaker”.
Read more about cabinet’s programs in AGEBasic in cabinets.
Cabinet events
Functions that interacts with the AGEBasic cabinet event system.
EventTrigger(event name string): activate an event.
Room
Functions related with the loaded rooms.
Room Posters
To replace posters in a Room
PosterRoomCount()returns the poster count of the actual room.PosterRoomReplace(position #, Image path)to replace a poster by an image in disk. Example:PosterRoomReplace(1, CombinePath(ConfigPath(), "posters/myposter.png"))to replace the second poster in the room.
Room Light configuration
Important: Room-specific light configuration is now deprecated. Please use the Global Light Configuration for managing lights.
This section describes the legacy methods for interacting with individual room lights. While still functional, these methods are superseded by the global light system.
GetLights(): Retrieves a string containing the names of all lights in the currently loaded rooms. The light names are concatenated using the pipe (|) character as a separator. Each individual light name follows the format"roomName:lightName"for easy identification.
Example:"livingRoom:ceilingLight|bedroom01:nightLamp|kitchen:overhead"To work with this string:- Use a string processing function (like
GetMember(), if available in your system) to split the string into individual light names using the|delimiter. - Further process each individual light name string to separate the room name from the specific light name (e.g., using the
:delimiter).
- Use a string processing function (like
GetLightIntensity(lightName): Returns the current intensity of the specified light.- Parameter:
lightName(string) - The unique identifier of the light (e.g.,"room001:light1"), obtained from theGetLights()method. - Return Value: A number between
0(off) and10(maximum brightness). - Example:
GetLightIntensity("study:deskLamp")might return7.2.
- Parameter:
SetLightIntensity(lightName, intensity): Sets the intensity of the specified light.- Parameters:
lightName(string) - The unique identifier of the light (e.g.,"room001:light1").intensity(number) - The desired brightness level, ranging from0to10.
- Example:
SetLightIntensity("bedroom02:readingLight", 5)will set the reading light in bedroom 02 to a medium intensity.
- Parameters:
SetLightColor(lightName, R, G, B): Sets the color of the specified light using RGB values.- Parameters:
lightName(string) - The unique identifier of the light (e.g.,"room001:light1").R(number) - The red color component (typically 0-255 or 0.0-1.0, depending on your system’s color model).G(number) - The green color component (typically 0-255 or 0.0-1.0).B(number) - The blue color component (typically 0-255 or 0.0-1.0).
- Return Value:
0if an error occurred during the color setting process. - Example:
SetLightColor("livingRoom:ambientLight", 255, 165, 0)would set the ambient light in the living room to orange (assuming 0-255 color range).
- Parameters:
Configuring Global/Room Lighting
This section details how to manage the global lighting system, the recommended method for controlling illumination within the gallery.
The global lighting system provides the flexibility to adjust the color tint and intensity of light either for individual rooms or uniformly across all spaces.
To execute a program to change the light when a room is loaded utilize the AGEBasic entries in the configuration room yaml (like room001.yaml):
Example room001.yaml
agebasic:
after-load: changelights.bas
active: true
debug: falseGlobal lighting functions
Utilize the following functions to configure the gallery’s ambiance:
-
GetGlobalLight(separator): Retrieves the current global light configuration as a string. The output follows the formatr|g|b|i, wherer,g, andbrepresent the red, green, and blue color components, respectively, andidenotes the light intensity. This format emulates an AGEBasic list. -
SetGlobalLight(r, g, b, intensity): Modifies the intensity and color of the global light.r(number): The red color value, ranging from 0 to 255.g(number): The green color value, ranging from 0 to 255.b(number): The blue color value, ranging from 0 to 255.intensity(number): The desired light intensity level.- Example: Executing
SetGlobalLight(0, 0, 255, 1.7)would set the global light to a blue hue with an intensity of 1.7 (assuming a 0-255 color range).
Read about light configuration in Lights Configuration
Audio
Age of Joy plays two type of sounds: ambience (noise in rooms) and games sound. The volume of the audio is expressed in dB: 0 is normal, 20 as loud max, and --80 as silent. These values affect all the rooms and all the games.
To mute a sound set its volume to -80, and to unmute it simply set it to the previous volume value.
If you write a script for a cabinet (read CDL the Cabinet Description Language) that changes the volume of a game (for example), remember to change it to its previous value in order to not affect other games.
Get/Set Volume
AudioAmbienceGetVolume(),AudioMusicGetVolume()andAudioGameGetVolume()to get the volume indB.AudioAmbienceSetVolume(number volume),AudioMusicSetVolume(number volume)andAudioGameSetVolume(number volume)to set the volume, also indB.
Music
To play music (Jukebox functions).
All the audio files (like mp3 or ogg) should be saved in the /sdcard/Android/data/com.curif.AgeOfJoy/music folder. To easily get that folder during an AGEBasic program execution use the MusicPath() function.
The programmer should add all the audio files that the player want to ear in a queue, and the play it (the play action is in order)
Music functions
MusicAdd(audio file path): add an audio file to the queue.MusicExist(audio file path): true if the file is in the jukebox queue.MusicRemove(audio file path): remove the file from the jukebox queue. Returntrueif removed.MusicClear(): clear the audio queue.MusicGetList(separator): get the list of current audio files in the queue. Returns a simulated list string likesong1.mp3:song2.mp3.MusicAddList(files, separator): to add a group of audio files to the queue. Files is a list simulated string likesong1.mp3:song2.mp3, you can useFileGetto get the files.MusicLoop(true/false): to activate/deactivate the loop function.MusicLoopStatus()to query the music loop status.MusicNext()andMusicPrevious()to jump to the next or previous song.MusicReset()start playing again the music queue.MusicCount()count of files in the music queue.
Player
It’s possible to change the position of the player in the 3D space:
PlayerGetHeight(),PlayerSetHeight(number): to get and set the height of the player. It’s recommended to get the height and changing it by adding or decrementing its value.PlayerGetCoordinate(string coord),PlayerSetCoordinate(string coord, value): to get and set the coordinates in the 3D space (force the player to a new position).coordshould bet “X” or “Z”. “Y” cant be changed, usePlayerSetHeightinstead.PlayerLookAt(cabinet part number): to force the player to look at a part of the cabinet. For example the screen. Can be used only in AGEBasic in cabinets mode.PlayerTeleport(string room): to teleport the player to a room. The string is the room name (like “room001” for example.)
It’s recommended to read the AGEBasic examples - player to look at a screen when insert coin.
Controllers
It’s possible to query the control status, for example, to know if a user is pressing some button on the controller.
ControlActive(id [, port]): to know the status of a control. ReturnsTrue (1)if the control is active on the moment of execution, orFalse (0)if not. Theids of the controls (like buttons) are in the table in the page: Default controllers configuration mapping.[port]is an optional port number (0is the default).ControlActivecan be used in AGEBasic in cabinets or in programs to execute in the Configuration control cabinet. Note:portis available in the0.5version and superior.ControlRumble(id, amplitude, duration): to create a vibration on the controller.durationis a decimal where1means one second.amplitudeis a decimal number too.idshould beJOYPAD_LEFT_RUMBLEorJOYPAD_RIGHT_RUMBLE. Returnstrueif the controller support haptic feedback.
Data manipulation
Sometimes you will need to storage and read information for your programs.
Data - READ - RESTORE combo
To add information to be consumed during the program execution. You could storage information in different “storage” that lives during the program execution. Each storage has its name.
DATA "storage name", x[,y,z, ...]: comma separated list of expressions. Example:DATA "my storage", 10, "x", D + 1. Expressions are evaluated during the line execution not when the storage is read.READ "storage name", var[, var, ...]: to read a storage, Example:READ "my storage", A, B, Cto read the storage of the previous example, result:A=10, B="x", C=D+1. There is an internal pointer to identify which is the next data to be read.RESTORE "storage name"[, offset]: move the pointer to theoffsetposition. Defaults to 0.
File management
GetFiles(path, separator, order)get a list string with the file names of a path, parameters: path to scan, string separator, and order. You could useCountMembers()andGetMember()to process the result. Order:0: alphabetic order1: random2: Creation date from old to new3: Creation date from new to old example:let f = getFiles("path\\to\\files", ":", 3)to get"file1.txt:file2.txt:xxx.bas"thenGetMember(f, 1) = "file2.txt"
- Path management:
CombinePath(path1, path2)given two paths, return a string with the combination. Example:CombinePath("/sdcard", "file.txt")returns/sdcard/file.txtConfigPath()returns the path to the configuration files.AGEBasicPath()returns the path to the AGEBasic programs.CabinetsDBPath()returns the path to the cabinet database.CabinetsPath()returns the path to the new cabinets. (usually empty)CabinetPath()returns the path to the actual cabinet (only available in AGEBasic in cabinets).RootPath()the base path of AGE of Joy. Isn’t the Android root home.MusicPath()the base path to the music folder.
- File management (v0.6 and sup)
FileOpen(string path, string mode): returns an file pointer number to identify the file opened or-1if fails. Mode must be:Rread mode,Wwrite mode (will rewrite the file if it exists),Ato append to the end of the file. Only 256 files can be opened at the same time.FileRead(file pointer number): Read the next file string line. UseFileEOF()to know if you can read a next line. If it is EOF the function return an empty string (""). Use Returns the next line or-1if it fails.type(var)to detect if the result is a number (error) or the read string.FileClose(file pointer number)to close the file.FileEOF(file pointer number):1if it is closed or0if not.-1if the file is not open or the number is invalid.FileWrite(file pointer number, string line): add the line to the file.FileDelete(path): to delete a file, returnsfalseif fail.FileExists(path): returnstrueif the file exists.
CPU control
You can increase the CPU load, but take in consideration that it could affect the overall game performance. The use of CPU is administered internally by using delays in the program execution (interline execution).
GetCPU(): obtain the actual CPU percentage used for program executions.SetCPU(percentage)to set the maximum CPU percentage.100is the max value.
The default CPU percentage is 76%
Debug mode
The AGEBasic developer has the option to enable Debug Mode within a program or a setup, such as in a programming YAML subdocument within a cabinet’s configuration.
Once DebugMode is activated (true), Age of Joy will generate a report upon completion, detailing the most recent error (both compilation and runtime errors), along with the final program status, encompassing variable specifics like names, types, and values.
DebugMode(true/false)
Example:
1000 let i = 100
1010 call DebugMode(1) 'activate the debug mode
1020 let ERROR = "some error message to see in debug"Result:
Program: myprogram.bas
PROGRAM STATUS ---
PROGRAM: myprogram.bas
Last line parsed: 1020 executed: 3
vars:
I: 100 [Number]
ERROR: some error message to see in debug [string]
---You can find the result file in the folder: /sdcard/Android/data/com.curif.AgeOfJoy/AGEBasic
The name of the file should be: myprogram.bas.debug (the file name of the program with the .debug suffix)
Warning
Don’t distribute Cabinet Assets with
DebugModeactive. Age of Joy will write a file every time a program ends. If you are using AGEBasic cabinet event system the situation is worse because some programs runs more than one time (like the ones attached toon-allwaysevents). UseDebugModeonly in your tests.
If you want to use ChatGPT to ask for programs, just follow this AGEBasic prompt for ChatGPT