Merge branch 'master' of C:\Users\emil\Documents\My Dropbox\tinkerbell\gitrep\tinkerbell

This commit is contained in:
Emil Segerås 2012-07-28 21:24:02 +02:00
commit a43c52bbd6
12 changed files with 394 additions and 82 deletions

1
.gitignore vendored
View File

@ -27,5 +27,6 @@ Thumbs.db
Debug
Release
ipch
* gitignore
documentation

View File

@ -123,6 +123,7 @@
<ClCompile Include="..\..\tinkerbell\src\tb_window.cpp" />
<ClCompile Include="..\..\tinkerbell\src\tests\tb_test.cpp" />
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_linklist.cpp" />
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_parser.cpp" />
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_space_allocator.cpp" />
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_style_edit.cpp" />
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_tempbuffer.cpp" />

View File

@ -189,6 +189,9 @@
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_tempbuffer.cpp">
<Filter>Source Files\tinkerbell\tests</Filter>
</ClCompile>
<ClCompile Include="..\..\tinkerbell\src\tests\test_tb_parser.cpp">
<Filter>Source Files\tinkerbell\tests</Filter>
</ClCompile>
</ItemGroup>
<ItemGroup>
<Library Include="..\freeglut.lib">

View File

@ -18,4 +18,15 @@ TBLayout: axis: y, distribution: available
TBTextField: text: Menu
skin TBButton.flat
TBEditField: multiline: 1, styling: 1, gravity: all, id: editfield, autofocus: 1
text <u>Tinkerbell UI Toolkit!</u>\n\nThings to test:\n • resizing\n • right clicking\n • searching\n • keyboard navigation\n • scrolling\n • mouse wheel & dragging.\n • This is a list item that should wrap in a smart way.\n • Embedded content in textfield, such as this button: <widget TBButton: text: ":)">\n • <color #f44>Red</color>, <color #6f6>Green</color>, <color #aaf>Blue</color>
text: "Tinkerbell UI Toolkit\n\n" \
"<color #0794f8>Test zone</color>\n" \
"The menu to the left spawns some examples of what tinkerbell can do.\n\n" \
"The code in Demo/* is more like a developers test zone than organized tutorial-like samples. " \
"This would be good to fix of course (help is welcome! ;)\n\n" \
"<color #0794f8>Some things to try out</color>\n" \
" • All layouts provide panning of content automatically when squashed below the minimal size, so try resizing windows and pan.\n" \
" • Lines starting with a bullet sequence (like this one) should wrap in a smart way.\n" \
" • <u>Underline</u>, <color #f44>Red</color>, <color #6f6>Green</color>, <color #aaf>Blue</color>\n\n" \
"<color #0794f8>Good to know</color>\n" \
" • The text component you're reading in handles styling and embedded content (such as this widget: <widget TBButton: text: ':)'>), but does not handle editing of those fully. Undo/Redo should work fine, but caret movement and position updates doesn't.\n" \
" • All resources are UTF-8."

View File

@ -65,6 +65,7 @@ SRC = tinkerbell/src/tb_layout.cpp \
tinkerbell/src/tests/test_tb_space_allocator.cpp \
tinkerbell/src/tests/test_tb_widget_value.cpp \
tinkerbell/src/tests/test_tb_linklist.cpp \
tinkerbell/src/tests/test_tb_parser.cpp \
tinkerbell/src/tests/test_tb_tempbuffer.cpp \
tinkerbell/src/tests/test_tb_test.cpp \
stb_image/tb_image_loader_stb.cpp \

View File

@ -9,6 +9,65 @@
namespace tinkerbell {
// == Util functions ====================================================================
void UnescapeString(char *str)
{
char *dst = str, *src = str;
while (*src)
{
if (*src == '\\')
{
bool code_found = true;
if (src[1] == 'n')
*dst = '\n';
else if (src[1] == 'r')
*dst = '\r';
else if (src[1] == 't')
*dst = '\t';
else if (src[1] == '\"')
*dst = '\"';
else if (src[1] == '\'')
*dst = '\'';
else if (src[1] == '\\')
*dst = '\\';
else
code_found = false;
if (code_found)
{
src += 2;
dst++;
continue;
}
}
*dst = *src;
dst++;
src++;
}
*dst = 0;
}
bool is_white_space(const char *str)
{
switch (*str)
{
case ' ':
case '\t':
return true;
default:
return false;
}
}
bool is_pending_multiline(const char *str)
{
while (is_white_space(str))
str++;
return str[0] == '\\' && str[1] == 0;
}
// == Parser ============================================================================
Parser::STATUS Parser::Read(ParserStream *stream, ParserTarget *target)
{
TBTempBuffer line, work;
@ -17,6 +76,8 @@ Parser::STATUS Parser::Read(ParserStream *stream, ParserTarget *target)
current_indent = 0;
current_line_nr = 1;
pending_multiline = false;
multi_line_sub_level = 0;
while (int read_len = stream->GetMoreData((char *)work.GetData(), work.GetCapacity()))
{
@ -83,42 +144,6 @@ Parser::STATUS Parser::Read(ParserStream *stream, ParserTarget *target)
return STATUS_OK;
}
void UnescapeString(char *str)
{
char *dst = str, *src = str;
while (*src)
{
if (*src == '\\')
{
bool code_found = true;
if (src[1] == 'n')
*dst = '\n';
else if (src[1] == 'r')
*dst = '\r';
else if (src[1] == 't')
*dst = '\t';
else if (src[1] == '\"')
*dst = '\"';
else if (src[1] == '\'')
*dst = '\'';
else if (src[1] == '\\')
*dst = '\\';
else
code_found = false;
if (code_found)
{
src += 2;
dst++;
continue;
}
}
*dst = *src;
dst++;
src++;
}
*dst = 0;
}
void Parser::OnLine(char *line, ParserTarget *target)
{
if (*line == '#')
@ -126,6 +151,11 @@ void Parser::OnLine(char *line, ParserTarget *target)
target->OnComment(line + 1);
return;
}
if (pending_multiline)
{
OnMultiline(line, target);
return;
}
// Check indent
int indent = 0;
@ -161,11 +191,11 @@ void Parser::OnLine(char *line, ParserTarget *target)
{
char *token = line;
// Read line while consuming it and copy over to token buf
while (*line != ' ' && *line != 0)
while (!is_white_space(line) && *line != 0)
line++;
int token_len = line - token;
// Consume any white space after the token
while (*line == ' ')
while (is_white_space(line))
line++;
bool is_compact_line = token_len && token[token_len - 1] == ':';
@ -175,6 +205,19 @@ void Parser::OnLine(char *line, ParserTarget *target)
{
token_len--;
token[token_len] = 0;
// Check if the first argument is not a child but the value for this token
if (is_number(line) || *line == '[' || *line == '\"' || *line == '\'' || *line == '@')
{
ConsumeValue(value, line);
if (pending_multiline)
{
// The value wrapped to the next line, so we should remember the token and continue.
multi_line_token.Set(token);
return;
}
}
}
else if (token[token_len])
{
@ -192,9 +235,9 @@ void Parser::OnLine(char *line, ParserTarget *target)
/** Check if buf is pointing at a end quote. It may need to iterate
buf backwards toward buf_start to check if any preceding backslashes
make it a escaped quote (which should not be the end quote) */
bool IsEndQuote(const char *buf_start, const char *buf)
bool IsEndQuote(const char *buf_start, const char *buf, const char quote_type)
{
if (*buf != '\"')
if (*buf != quote_type)
return false;
int num_backslashes = 0;
while (buf_start < buf && *(buf-- - 1) == '\\')
@ -208,7 +251,7 @@ void Parser::OnCompactLine(char *line, ParserTarget *target)
while (*line)
{
// consume any whitespace
while (*line == ' ')
while (is_white_space(line))
line++;
// Find token
@ -220,46 +263,19 @@ void Parser::OnCompactLine(char *line, ParserTarget *target)
*line++ = 0;
// consume any whitespace
while (*line == ' ')
while (is_white_space(line))
line++;
TBValue v;
ConsumeValue(v, line);
// Find value (As quoted string, or as auto)
char *value = line;
if (*line == '\"')
if (pending_multiline)
{
// Consume starting quote
line++;
value++;
// Find ending quote or end
while (!IsEndQuote(value, line) && *line != 0)
line++;
// Terminate away the quote
if (*line == '\"')
*line++ = 0;
// consume any whitespace
while (*line == ' ')
line++;
// consume any comma
if (*line == ',')
line++;
UnescapeString(value);
v.SetString(value, TBValue::SET_AS_STATIC);
}
else
{
// Find next comma or end
while (*line != ',' && *line != 0)
line++;
// Terminate away the comma
if (*line == ',')
*line++ = 0;
UnescapeString(value);
v.SetFromStringAuto(value, TBValue::SET_AS_STATIC);
// The value wrapped to the next line, so we should remember the token and continue.
multi_line_token.Set(token);
// Since we need to call target->Leave when the multiline is ready, set multi_line_sub_level.
multi_line_sub_level = 1;
return;
}
// Ready
@ -269,4 +285,77 @@ void Parser::OnCompactLine(char *line, ParserTarget *target)
target->Leave();
}
void Parser::OnMultiline(char *line, ParserTarget *target)
{
// consume any whitespace
while (is_white_space(line))
line++;
TBValue value;
ConsumeValue(value, line);
if (!pending_multiline)
{
// Ready with all lines
value.SetString(multi_line_value.GetData(), TBValue::SET_AS_STATIC);
target->OnToken(multi_line_token, value);
if (multi_line_sub_level)
target->Leave();
// Reset
multi_line_value.SetAppendPos(0);
multi_line_sub_level = 0;
}
}
void Parser::ConsumeValue(TBValue &dst_value, char *&line)
{
// Find value (As quoted string, or as auto)
char *value = line;
if (*line == '\"' || *line == '\'')
{
const char quote_type = *line;
// Consume starting quote
line++;
value++;
// Find ending quote or end
while (!IsEndQuote(value, line, quote_type) && *line != 0)
line++;
// Terminate away the quote
if (*line == quote_type)
*line++ = 0;
// consume any whitespace
while (is_white_space(line))
line++;
// consume any comma
if (*line == ',')
line++;
UnescapeString(value);
dst_value.SetString(value, TBValue::SET_AS_STATIC);
}
else
{
// Find next comma or end
while (*line != ',' && *line != 0)
line++;
// Terminate away the comma
if (*line == ',')
*line++ = 0;
UnescapeString(value);
dst_value.SetFromStringAuto(value, TBValue::SET_AS_STATIC);
}
// Check if we still have pending value data on the following line and set pending_multiline.
bool continuing_multiline = pending_multiline;
pending_multiline = is_pending_multiline(line);
// Append the multi line value to the buffer.
if (continuing_multiline || pending_multiline)
multi_line_value.AppendString(dst_value.GetString());
}
}; // namespace tinkerbell

View File

@ -7,6 +7,7 @@
#define TBParser_H
#include "tb_value.h"
#include "tb_tempbuffer.h"
namespace tinkerbell {
@ -41,8 +42,14 @@ public:
private:
int current_indent;
int current_line_nr;
TBStr multi_line_token;
TBTempBuffer multi_line_value;
int multi_line_sub_level;
bool pending_multiline;
void OnLine(char *line, ParserTarget *target);
void OnCompactLine(char *line, ParserTarget *target);
void OnMultiline(char *line, ParserTarget *target);
void ConsumeValue(TBValue &dst_value, char *&line);
};
}; // namespace tinkerbell

View File

@ -19,14 +19,14 @@ namespace tinkerbell {
// == Helper functions ============================
bool p_is_number(const char *str)
bool is_number(const char *str)
{
if (*str == '-')
if (*str == '-' || *str == '.')
str++;
return *str >= '0' && *str <= '9';
}
bool p_is_number_float(const char *str)
bool is_number_float(const char *str)
{
while (*str) if (*str++ == '.') return true;
return false;
@ -215,7 +215,7 @@ void TBValue::SetFromStringAuto(const char *str, SET set)
{
if (!str)
SetNull();
else if (p_is_number(str))
else if (is_number(str))
{
// If the number has spaces, we'll assume a list of numbers (example: "10 -4 3.5")
if (strstr(str, " "))
@ -238,7 +238,7 @@ void TBValue::SetFromStringAuto(const char *str, SET set)
SetArray(arr, SET_TAKE_OWNERSHIP);
}
}
else if (p_is_number_float(str))
else if (is_number_float(str))
SetFloat((float)atof(str));
else
SetInt(atoi(str));

View File

@ -13,6 +13,13 @@ namespace tinkerbell {
class TBValue;
/** Return true if the given string starts with a number. */
bool is_number(const char *str);
/** Return true if the given number string is a float number.
Should only be called when you've verified it's a number with is_number(). */
bool is_number_float(const char *str);
/** TBValueArray is a array of TBValue */
class TBValueArray
{

View File

@ -0,0 +1,128 @@
// ================================================================================
// == This file is a part of Tinkerbell UI Toolkit. (C) 2011-2012, Emil Segerås ==
// == See tinkerbell.h for more information. ==
// ================================================================================
#include "tb_test.h"
#include "parser/TBNodeTree.h"
#ifdef TB_UNIT_TESTING
using namespace tinkerbell;
TB_TEST_GROUP(tb_parser)
{
TBNode node;
TB_TEST(Init)
{
TB_VERIFY(node.ReadFile(TB_TEST_FILE("test_tb_parser.tb.txt")));
}
TB_TEST(strings)
{
TB_VERIFY_STR(node.GetValueString("strings>string1", ""), "A string");
TB_VERIFY_STR(node.GetValueString("strings>string2", ""), "\"A string\"");
TB_VERIFY_STR(node.GetValueString("strings>string3", ""), "\'A string\'");
TB_VERIFY_STR(node.GetValueString("strings>string4", ""), "\"\'A string\'\"");
TB_VERIFY_STR(node.GetValueString("strings>string5", ""), "Foo\nBar");
}
TB_TEST(strings_compact)
{
TB_VERIFY_STR(node.GetValueString("strings_compact>string1", ""), "");
TB_VERIFY_STR(node.GetValueString("strings_compact>string2", ""), "A string");
TB_VERIFY_STR(node.GetValueString("strings_compact>string3", ""), "A string");
TB_VERIFY_STR(node.GetValueString("strings_compact>string4", ""), "'A string'");
TB_VERIFY_STR(node.GetValueString("strings_compact>string5", ""), "\"A string\"");
TB_VERIFY_STR(node.GetValueString("strings_compact>string6", ""), "\"A string\"");
TB_VERIFY_STR(node.GetValueString("strings_compact>string7", ""), "\\");
TB_VERIFY_STR(node.GetValueString("strings_compact>string8", ""), "\"");
TB_VERIFY_STR(node.GetValueString("strings_compact>string9", ""), "\\\\\\\\");
TB_VERIFY_STR(node.GetValueString("strings_compact>string10", ""), "\\\\\"");
TB_VERIFY_STR(node.GetValueString("strings_compact>string11", ""), "\"\"\'\'");
TB_VERIFY_STR(node.GetValueString("strings_compact>string12", ""), "@language_string_token");
}
TB_TEST(numbers_compact)
{
TB_VERIFY(node.GetValueInt("numbers_compact>integer", 0) == -10);
TB_VERIFY_FLOAT(node.GetValueFloat("numbers_compact>float", 0), 1.0);
}
TB_TEST(compact_with_children)
{
TB_VERIFY_STR(node.GetValueString("compact_with_children>string", ""), "A string");
TB_VERIFY_STR(node.GetValueString("compact_with_children>string>child1", ""), "Child 1");
TB_VERIFY_STR(node.GetValueString("compact_with_children>string>child2", ""), "Child 2");
TB_VERIFY(node.GetValueInt("compact_with_children>integer", 0) == -10);
TB_VERIFY(node.GetValueInt("compact_with_children>integer>child1", 0) == 1);
TB_VERIFY(node.GetValueInt("compact_with_children>integer>child2", 0) == 2);
TB_VERIFY_FLOAT(node.GetValueFloat("compact_with_children>float", 0), 1);
TB_VERIFY_FLOAT(node.GetValueFloat("compact_with_children>float>child1", 0), 1);
TB_VERIFY_FLOAT(node.GetValueFloat("compact_with_children>float>child2", 0), 2);
}
TB_TEST(compact_no_value)
{
TB_VERIFY_STR(node.GetValueString("compact_no_value>string", ""), "A string");
TB_VERIFY(node.GetValueInt("compact_no_value>int", 0) == 42);
TB_VERIFY_FLOAT(node.GetValueFloat("compact_no_value>float", 0), 3.14);
TB_VERIFY_STR(node.GetValueString("compact_no_value>subgroup>string1", ""), "A string, with \"comma\"");
TB_VERIFY_STR(node.GetValueString("compact_no_value>subgroup>string2", ""), "'Another string'");
TB_VERIFY_STR(node.GetValueString("compact_no_value>subgroup>string3", ""), "And another string");
}
TB_TEST(arrays_numbers)
{
TBNode *arr_n = node.GetNode("arrays>numbers");
TB_VERIFY(arr_n);
TB_VERIFY(arr_n->GetValue().GetArrayLength() == 5);
TBValueArray *arr = arr_n->GetValue().GetArray();
TB_VERIFY(arr->GetValue(0)->GetInt() == 1);
TB_VERIFY(arr->GetValue(1)->GetInt() == 2);
TB_VERIFY_FLOAT(arr->GetValue(2)->GetFloat(), 0.5);
TB_VERIFY_FLOAT(arr->GetValue(3)->GetFloat(), 1.0E-8);
TB_VERIFY(arr->GetValue(4)->GetInt() == 1000000000);
}
//FIX: Not supported yet
// TB_TEST(arrays_strings)
// {
// TBNode *arr_n = node.GetNode("arrays>strings");
// TB_VERIFY(arr_n);
// TB_VERIFY(arr_n->GetValue().GetArrayLength() == 5);
// TBValueArray *arr = arr_n->GetValue().GetArray();
// TB_VERIFY_STR(arr->GetValue(0)->GetString(), "Foo");
// TB_VERIFY_STR(arr->GetValue(1)->GetString(), "'Foo'");
// TB_VERIFY_STR(arr->GetValue(2)->GetString(), "Foo");
// TB_VERIFY_STR(arr->GetValue(3)->GetString(), "\"Foo\"");
// TB_VERIFY_STR(arr->GetValue(4)->GetString(), "Foo 'bar'");
// }
//
// TB_TEST(arrays_mixed)
// {
// TBNode *arr_n = node.GetNode("arrays>mixed");
// TB_VERIFY(arr_n);
// TB_VERIFY(arr_n->GetValue().GetArrayLength() == 4);
// TBValueArray *arr = arr_n->GetValue().GetArray();
// TB_VERIFY_STR(arr->GetValue(0)->GetString(), "Foo");
// TB_VERIFY(arr->GetValue(1)->GetInt() == 2);
// TB_VERIFY_STR(arr->GetValue(2)->GetString(), "bar");
// TB_VERIFY(arr->GetValue(3)->GetFloat() == 4.0f);
// }
TB_TEST(strings_multiline)
{
TB_VERIFY_STR(node.GetValueString("strings_multiline>string1", ""), "Line 1\nLine 2\nLine 3");
TB_VERIFY_STR(node.GetValueString("strings_multiline>string2", ""), "abc");
TB_VERIFY_STR(node.GetValueString("strings_multiline>string3", ""), "AB");
TB_VERIFY_STR(node.GetValueString("strings_multiline>string4", ""), "Line 1\nLine 2\nLine 3\n");
TB_VERIFY_STR(node.GetValueString("strings_multiline>subgroup>first", ""), "Foo");
TB_VERIFY_STR(node.GetValueString("strings_multiline>subgroup>second", ""), "AB");
TB_VERIFY_STR(node.GetValueString("strings_multiline>string5", ""), "The last string");
}
}
#endif // TB_UNIT_TESTING

View File

@ -0,0 +1,64 @@
# Test file for the text node system and its parser
# Everything is a tree of nodes with a name and value.
# The values may be strings, numbers or arrays.
# If there is no colon, the entire line will be treated as a value and no quotes is needed around strings.
# Any quotes will be part of the string.
strings
string1 A string
string2 "A string"
string3 'A string'
string4 "'A string'"
string5 Foo\nBar
# If there is a colon (compact mode), there must be quotes around strings.
# First comes the value (optional)
# Then comes other child nodes (separated by comma)
strings_compact
string1: this should fail
string2: "A string"
string3: 'A string'
string4: "'A string'"
string5: '"A string"'
string6: "\"A string\""
string7: "\\"
string8: "\""
string9: "\\\\\\\\"
string10: "\\\\\""
string11: "\"\"\'\'"
string12: @language_string_token
numbers_compact
integer: -10
float: 1.0
compact_with_children
string: "A string", child1: "Child 1", child2: "Child 2"
integer: -10, child1: 1, child2: 2
float: 1.0, child1: 1.0, child2: 2.0
compact_no_value: string: "A string", int: 42, float: 3.14
subgroup: string1: "A string, with \"comma\"", string2: "'Another string'"
string3: "And another string"
arrays
numbers 1 2 .5 1.0E-8 1000000000
# Not supported yet
# strings: "Foo" "'Foo'" 'Foo' '"Foo"' "Foo 'bar'"
# mixed: "foo" 2 "bar" 4.0
# Strings can span over multiple lines by ending with a \
strings_multiline
string1: "Line 1\nLine 2\nLine 3"
string2: "a" \
"b" \
"c"
string3: 'A' \
'B'
string4: 'Line 1\n' \
'Line 2\n'\
'Line 3\n'
subgroup: first: "Foo", second: "A" \
"B"
string5: "The last string"