TL;DR: This Godot Editor plugin imports tables like this and provides access to processed, statically typed data. It can impute defaults, convert floats by specified units, prefix text, convert text enumerations to integers, and more!
This plugin works best in combination with our Units plugin. You can use it without that plugin, but you will need to supply your own unit coversion method to use any units functionality.
(Note: This plugin resulted from splitting the now-depreciated "Table Importer" plugin into "Tables" and "Units".)
Find more detailed instructions at our Developers Page.
The plugin directory ivoyager_tables
should be added directly to your addons directory. You can do this one of two ways:
- Download and extract the plugin, then add it (in its entirety) to your addons directory, creating an "addons" directory in your project if needed.
- (Recommended) Add as a git submodule. From your project directory, use git command:
git submodule add https://github.com/ivoyager/ivoyager_tables addons/ivoyager_tables
This method will allow you to version-control the plugin from within your project rather than moving directories manually. You'll be able to pull updates, checkout any commit, or submit pull requests back to us. This does require some learning to use git submodules. (We use GitKraken to make this easier!)
Then enable "I, Voyager - Tables" from Project/Project Settings/Plugins. The plugin will provide an autoload singleton called "IVTableData" through which you can provide postprocessing intructions and interact with processed table data.
This plugin is maintained for I, Voyager's solar system simulator and associated apps and games. It's powerful but requires specific file formatting. It's not meant to be a general "read any" table utility.
Table features:
- Specification of data Type so that all table values are correctly converted to statically typed internal values.
- Specification of data Default to reduce table clutter. Fields that are mostly a particular value can be left mostly empty.
- Specification of data Prefix to reduce enumeration text size. E.g., shorten "PLANET_MERCURY", "PLANET_VENUS", "PLANET_EARTH" to "MERCURY", "VENUS", "EARTH".
- Specification of float Unit so file data can be entered in the most convenient units while maintaining consistent internal representation. Unit can be specified for a whole float column or within a float cell (e.g., "1000 s" or "1000/s"). The unit parser interprets compound units like "m^3/(kg s^2)" and memoizes them for faster subsequent access.
- Table enumerations that may reference project enums or table-defined entity names in an INT column. For example, in our project, "PLANET_EARTH" resolves to 3 as an integer in any table because "PLANET_EARTH" is row 3 in planets.tsv.
- Table constants that can be used in columns of any type. This is how the plugin handles things like "TRUE" and "-Inf" but also can be used to add arbitrary constants like "MARS_PERIHELION" or "SOME_LARGE_TEXT_ITEM" if needed.
- Tables that modify other tables.
- Modding support that allows your game or project users to replace base tables with modded tables in user://modding/mod_files/.
- Construction of a wiki lookup dictionary from file tables to use for an internal or external wiki.
- [For scientific apps, mostly] Determination of float significant digits from file number text, so precision can be correctly displayed even after unit conversion.
- Easy or optimized access to all processed and statically typed data via the IVTableData singleton.
File and table formats are described below.
All user interface is via an autoload singleton IVTableData added by the plugin. From here you will give postprocessing intructions and access all postprocessed table data. IVTableData also provides functions to build objects, dictionaries, or flag fields directly from table data. See API in table_data.gd.
Postprocessing is called once by the project. The call allows specification of a unit conversion method (from our Units plugin if you are using that), enums, constants, and "missing" values by type. After postprocessing, all table data is strictly typed and read-only.
func postprocess_tables(table_file_paths: Array,
unit_conversion_method := placeholder_unit_conversion_method,
enable_wiki := false, enable_precisions := false, project_enums := [],
merge_overwrite_table_constants := {}, overwrite_missing_values := {})
The plugin imports table files as a custom resource class, but the resource isn't useful except for table debugging. (Really, everything you need is in IVTableData.)
We support only tab-delimited files with extension "tsv".
Any line starting with "@" is read as a table directive, which is used to specify one of several table formats and provide additional format-specific instructions. These can be at any line in the file. It may be convenient to include these at the end as many table viewers (including GitHub web pages) assume field names in the top line.
Table format is specified by one of @DB_ENTITIES
(default), @DB_ENTITIES_MOD
, @DB_ANONYMOUS_ROWS
, @ENUMERATION
, @WIKI_LOOKUP
, or @ENUM_X_ENUM
, optionally followed by "=" and then the table name. If omitted, table name is taken from the base file name (e.g., "planets" for "res://path/planets.tsv" file). Several table formats don't need a table format specifier as the importer can figure it out from other clues. Some table formats allow or require additional specific directives. See details in format sections below. (" = " is always ok in place of "=".)
For debugging or work-in-progress, you can prevent any imported table from being processed using @DONT_PARSE
. (It's still technically imported by the editor but won't be parsed line-by-line or handled by the postprocessor.)
Any line starting with "#" is ignored. Additionally, entire columns are ignored if the column field name begins with "#".
All cells are stripped of double-quotes if they enclose the cell on both sides (these are silently added by some editors).
Spaces are edge-stripped from cells and cell elements ("cell element" refers to comma- or semicolon-delimited data within cells of certain types, such as VECTORx, COLOR, and ARRAY[xxxx]).
Type is specified by column for "db-style" format or for the whole table for "enum_x_enum" format. Each data type has its own "missing" value (e.g., NAN for float) that is used if the cell is empty and there is no column or table default value. Missing values listed here can be respecified in the call to postprocess_tables()
. Allowed types are:
STRING
- Data processing applies Godot escaping such as \n, \t, etc. We also convert unicode "\u" escaping up to \uFFFF, but not "\U" escaping for larger unicodes. Empty cells will be imputed with default value (if specified) or "".STRING_NAME
- No escaping. Empty cells will be imputed with default value (if specified) or &"".BOOL
- Recognized table constants (which can be modified or added to) include: "TRUE", "True", "true", "x" (interpreted as true), "FALSE", "False" and "false". Empty cells will be imputed with default value (if specified) or false. Any other cell values will cause an error.INT
- A valid integer or text "enumeration". Enumerations may include any table entity name (from any table) or project enums specified in thepostprocess_tables()
call. Enumerations that can't be found will cause an error. Empty cells will be imputed with default value (if specified) or -1.FLOAT
- Commas and underscores are allowed and removed before float conversion. "E" or "e" are ok. A "~" prefix is allowed and affects precision (see below) but not float value. Recognized table constants (which can be modified or added to) include: "NAN", "Nan", "nan", "INF", "Inf", "inf", "-INF", "-Inf" and "-inf". Empty cells will be imputed with default value (if specified) or NAN. Inline units: An inline unit can be specified using format "x unit" or "x/unit". E.g., "1000 s" and "1000/s" are valid (the latter is equivilent to "1000 1/s"). If an inline unit is present, it will override the column or table unit (see table formats below).VECTOR2
,VECTOR3
,VECTOR4
- Enter in table cell as a comma-delimited set of 2, 3 or 4 float values. All rules for floats above apply except vector elements cannot have commas. Empty cells will be imputed with default value (if specified) or VectorX(-INF, -INF,...).COLOR
- Cell content is interpretted in a sensible way. Valid representations of red include: "red", "ff0000", "red, 1.0", "1, 0, 0" and "1, 0, 0, 1". Empty cells will be imputed with default value (if specified) or Color(-INF, -INF, -INF, -INF).ARRAY[xxxx]
(where "xxxx" specifies element type and is any of the above types) - Array elements are expected to be delimited by semicolon (;). After splitting by semicolon, each element is interpreted exactly as its type above. ColumnUnit
andPrefix
, if specified, are applied element-wise. Empty cells will be imputed with default value (if specified) or an empty, typed array.
(Why do we use strange "missing" values like Color(-INF, -INF, -INF, -INF)? A "missing" value entered in the file table is exactly equivalent to an empty cell
without Default for the purpose of db_has_value()
and constructor methods such as db_build_object()
. Hence, we avoid using potentially valid values such as Color.BLACK and Vector3.ZERO as "missing" values. In our usage, -1 works well for INT because these columns are most often used for enumerations.)
For scientific or educational apps (but probably not games), it is important to know and correctly represent data precision in GUI. To obtain a float value's original file precision in significant digits, specify enable_precisions = true
in the postprocess_tables()
call. You can then access float precisions via the "precisions" dictionary or "get_precision" methods in IVTableData. It's up to you to use precision in your GUI display. Keep in mind that unit-conversion will cause values like "1.0000001" if you don't do any string formatting.
See warning below about .csv/.tsv editors. If you must use Excel or another "smart" editor, then prefix all floats with underscore (_) to prevent modification!
Significant digits are counted from the left-most non-0 digit to the right-most digit if decimal is present, or to the right-most non-0 digit if decimal is not present, ignoring the exponential. Examples:
- "1e3" (1 significant digit)
- "1.000e3" (4 significant digits)
- "1000" (1 significant digit)
- "1100" (2 significant digits)
- "1000." (4 significant digits)
- "1000.0" (5 significant digits)
- "1.0010" (5 significant digits)
- "0.0010" (2 significant digits)
Additionally, any number that is prefixed with "~" is considered a "zero-precision" value (0 significant digits). We use this in our Planetarium to display GUI text like "~1 km".
Most .csv/.tsv file editors will "interpret" and change (i.e., corrupt) table data without any warning, especially numbers and text that looks (even vaguely) like dates or perhaps other things. For example, Excel will change "1.32712440018e20" to "1.33E+20" and "mar-1" to ####### on saving. One editor that does NOT change data without your input is Rons Data Edit. There is a free version that will let you work with files with up to 1000 rows.
Optional specifier: @DB_ENTITIES[=<table_name>]
(table_name defaults to the base file name)
Optional directive: @DONT_PARSE
This is the default table format assumed by the importer if other conditions (described under formats below) don't exist. The format has database-style entities (as rows) and fields (as columns). The first column of each row is taken as an "entity name" and used to create an implicit "name" field. Entity names are treated as enumerations and are accessible in other tables if the enumeration name appears in a field with Type=INT. Entity names must be globally unique.
Processed data are structured as a dictionary-of-statically-typed-field-arrays. Access the dictionary directly or use "get" methods in IVTableData.
The first non-comment, non-directive line is assumed to hold the field names. The left-most cell must be empty.
After field names and before data, tables can have the following header rows in any order:
Type
(required): See data types above.Default
(optional): Default values must be empty or follow Type rules above. If non-empty, this value is imputed for any empty cells in the column.Unit
(optional; FLOAT fields only): The data processor recognizes a broad set of unit symbols (mostly but not all SI) and, by default, converts table floats to SI base units in the postprocessed "internal" data. Default unit conversions are defined by "unit_multipliers" and "unit_lambdas" dictionaries here. Unit symbols and/or internal representation can be changed by specifying replacement conversion dictionaries in thepostprocess_tables()
call.Prefix
(optional; STRING, STRING_NAME and INT fields only): Prefixes any non-empty cells andDefault
(if specified) with provided prefix text. To prefix the column 0 implicit "name" field, usePrefix/<entity prefix>
. E.g., we usePrefix/PLANET_
in planets.tsv to prefix all entity names with "PLANET_".
The left-most 0-column of each content row specifies an "entity name". Entity names are included in an implicit field called "name" with Type=STRING_NAME. Prefix can be specified for the 0-column using header Prefix/<entity prefix>
. Entity names (after prefixing) must be globally unique. They can be used in any table as an enumeration that evaluates to the row number (INT) in the defining table. You can obtain row_number from the "enumerations" dictionary (index with any entity name) or obtain an enum-like dictionary of entity names from the "enumeration_dicts" dictionary (index with table_name or any entity_name) or obtain a enumeration array of entity names from the "enumeration_arrays" dictionary.
To create a wiki lookup dictionary, specify enable_wiki = true
in the postprocess_tables()
call. The postprocessor will populate the "wiki_lookup" dictionary in IVTableData from any columns named "en.wiki" in your table. (TODO: localization for "fr.wiki", "de.wiki", etc...)
For example usage, our Planetarium uses this feature to create hyperlink text to Wikipedia.org pages for almost all table entities: e.g., "Sun", "Ceres_(dwarf_planet)", "Hyperion_(moon)", etc. Alternatively, the lookup could be used for an internal game wiki.
Optional specifier: @DB_ENTITIES_MOD[=<table_name>]
(table_name defaults to the base file name)
Required directive: @MODIFIES=<table_name>
Optional directive: @DONT_PARSE
This table modifies an existing DB_ENTITIES table. It can add entities or fields or overwrite existing data. There can be any number of DB_ENTITIES_MOD tables that modify a single DB_ENTITIES table. The importer assumes this format if the @MODIFIES
directive is present.
Rules exactly follow DB_ENTITIES except that entity names must be present and they may or may not already exist in the DB_ENTITIES table being modified. If an entity name already exists, the mod table data will overwrite existing values. Otherwise, a new entity/row is added to the existing table. Similarly, field names may or may not already exist. If a new field/column is specified, then all previously existing entities (that are absent in the mod table) will be assigned the default value for this field.
Optional specifier: @DB_ANONYMOUS_ROWS[=<table_name>]
(table_name defaults to the base file name)
Optional directive: @DONT_PARSE
This table is exactly like DB_ENTITIES except that row names (the first column of each content row) are empty. The importer can identify this situation without the specifier directive. (Inconsistent use of row names will cause an import error assert.) This table will not create entity enumerations and does not have a "name" field and cannot be modified by DB_ENTITIES_MOD, but is in other ways like DB_ENTITIES.
Optional specifier: @ENUMERATION[=<table_name>]
(table_name defaults to the base file name)
Optional directive: @DONT_PARSE
This is a single-column "enumeration"-only table. The importer assumes this format if the table has only one column.
This is essentially a DB_ENTITIES format with only the 0-column: it creates entities enumerations with no data. There is no header row for field names and the only header tag that may be used (optionally) is Prefix
. As for DB_ENTITIES, prefixing the 0-column is done by modifying the header tag as Prefix/<entity prefix>
.
As for DB_ENTITIES, you can obtain row_number from the "enumerations" dictionary (index with any entity name) or obtain an enum-like dictionary of entity names from the "enumeration_dicts" dictionary (index with table_name or any entity_name).
Required specifier: @WIKI_LOOKUP[=<table_name>]
(table_name defaults to the base file name)
Optional directive: @DONT_PARSE
This format can add items to the wiki lookup dictionary that were not added by DB_ENTITIES or DB_ENTITIES_MOD tables.
The format is the same as DB_ENTITIES except that fields can include only localization-prefixed ".wiki" (e.g., "en.wiki"), and the only header tag allowed is Prefix
. Prefix the 0-column by entering the header tag as Prefix/<0-column prefix>
. The 0-column may contain any text and is not used to create entity enumerations.
For example usage, our Planetarium uses this table format to create hyperlinks to Wikipedia.org pages for concepts such as "Orbital_eccentricity" and "Longitude_of_the_ascending_node" (i.e., non-entity items that don't exist in a DB_ENTITIES table). Alternatively, the lookup could be used for an internal game wiki.
Required specifier: @ENUM_X_ENUM[=<table_name>]
(table_name defaults to the base file name)
Required directive: @DATA_TYPE=<Type>
Optional directives: @DATA_DEFAULT=<Default>
, @DATA_UNIT=<Unit>
, @TRANSPOSE
, @DONT_PARSE
This format creates an array-of-arrays data structure where data is indexed [row_enumeration][column_enumeration] or the transpose if @TRANSPOSE
is specified. All cells in the table have the same Type, Default and Unit (if applicable) specified by data directives above.
Enumerations must be "known" by the plugin, which means that they were added as row entity names by a DB_ENTITIES, DB_ENTITIES_MOD or ENUMERATION format table, or were added as a "project_enum" dictionary in the postprocess_tables()
call.
The upper-left table cell can either be empty or specify row and column entity prefixes delimited by a backslash. E.g., "RESOURCE_\FACILITY_" prefixes all row names with "RESOURCE_" and all column names with "FACILITY_".
The resulting array-of-arrays structure will always have rows and columns that are sized and ordered according to the enumeration, not the table, if entities are missing or out of order in the table. Data not specified in the table file (omited row or column entities) will be imputed with default value.