notcurses/tests/layout.cpp

618 lines
20 KiB
C++
Raw Normal View History

#include "main.h"
#include "internal.h"
TEST_CASE("TextLayout") {
auto nc_ = testing_notcurses();
if(!nc_){
return;
}
ncplane* n_ = notcurses_stdplane(nc_);
REQUIRE(n_);
const char str[] = "this is going to be broken up";
SUBCASE("LayoutLeft") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 20,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, str, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(str));
2020-06-16 06:12:11 +00:00
char* line = ncplane_contents(sp, 0, 0, 2, 20);
REQUIRE(line);
CHECK(0 == strcmp(line, "this is going to be broken up"));
free(line);
ncplane_destroy(sp);
}
SUBCASE("LayoutRight") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 20,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_RIGHT, str, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(str));
2020-06-16 06:12:11 +00:00
char* line = ncplane_contents(sp, 0, 0, 2, 20);
REQUIRE(line);
CHECK(0 == strcmp(line, "this is going to be broken up"));
free(line);
ncplane_destroy(sp);
}
SUBCASE("LayoutCenter") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 20,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, str, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(str));
2020-06-16 06:12:11 +00:00
char* line = ncplane_contents(sp, 0, 0, 2, 20);
REQUIRE(line);
CHECK(0 == strcmp(line, "this is going to be broken up"));
free(line);
ncplane_destroy(sp);
}
// lay out text where a word ends on the boundary
SUBCASE("LayoutOnBoundary") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 10,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "my nuclear arms";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
2020-06-16 06:12:11 +00:00
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "my nuclear arms"));
free(line);
ncplane_destroy(sp);
}
// lay out text where a word crosses the boundary
SUBCASE("LayoutCrossBoundary") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 10,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "my grasping arms";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
2020-06-16 06:12:11 +00:00
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "my grasping arms"));
free(line);
2020-06-16 06:12:11 +00:00
ncplane_destroy(sp);
}
2020-08-18 03:53:22 +00:00
// ensure we're honoring newlines
SUBCASE("LayoutNewlines") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 5,
.cols = 5,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
2020-08-18 03:53:22 +00:00
REQUIRE(sp);
size_t bytes;
2020-08-18 05:41:27 +00:00
const char boundstr[] = "a\nb\nc\nd\ne";
2020-08-18 03:53:22 +00:00
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "abcde")); // FIXME should have newlines
2020-08-18 03:53:22 +00:00
free(line);
ncplane_destroy(sp);
}
2020-08-18 05:41:27 +00:00
// ensure we're honoring newlines at the start/end of rows
SUBCASE("LayoutNewlinesAtBorders") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 5,
.cols = 3,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
2020-08-18 05:41:27 +00:00
REQUIRE(sp);
const char boundstr[] = "ab\ncde\nfgh";
size_t bytes;
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "abcdefgh"));
2020-08-18 05:41:27 +00:00
free(line);
ncplane_destroy(sp);
}
// lay out text where a wide word crosses the boundary
SUBCASE("LayoutCrossBoundaryWide") {
if(notcurses_canutf8(nc_)){
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 7,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "a 血的神";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "a 血的神"));
free(line);
ncplane_destroy(sp);
}
}
// a long word (one requiring a split no matter what) ought not force the
// next line, but instead be printed where it starts
SUBCASE("LayoutTransPlanar") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 10,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
2020-06-16 06:12:11 +00:00
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "my thermonuclear arms";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "my thermonuclear arms"));
free(line);
ncplane_destroy(sp);
}
// a long word (one requiring a split no matter what) ought not force the
// next line, but instead be printed where it starts
SUBCASE("LayoutTransPlanarWide") {
if(notcurses_canutf8(nc_)){
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 10,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "1 我能吞下玻璃";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "1 我能吞下玻璃"));
free(line);
ncplane_destroy(sp);
}
}
SUBCASE("LayoutLeadingSpaces") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 18,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = " \t\n my thermonuclear arms";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_CENTER, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, " my thermonuclear arms"));
free(line);
ncplane_destroy(sp);
}
// create a plane of two rows, and fill exactly one with one word
SUBCASE("LayoutFills1DPlane") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 15,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "quarkgluonfart ";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "quarkgluonfart "));
free(line);
ncplane_destroy(sp);
}
// create a plane of two rows, and fill exactly one with words
SUBCASE("LayoutFills1DPlaneWords") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 17,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "quark gluon fart ";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "quark gluon fart "));
free(line);
ncplane_destroy(sp);
}
// create a plane of two rows, and exactly fill the first line
SUBCASE("LayoutFillsSingleLine") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 13,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "quantum balls";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "quantum balls"));
free(line);
ncplane_destroy(sp);
}
// create a plane of three rows, and exactly fill two with regular ol' words
SUBCASE("LayoutFillsPlane") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 14,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "quantum balls scratchy no?! ";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "quantum balls scratchy no?! "));
free(line);
ncplane_destroy(sp);
}
// create a plane of three rows, and exactly fill two, with no spaces
SUBCASE("LayoutFillsPlaneNoSpaces") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 6,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "0123456789AB";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "0123456789AB"));
free(line);
ncplane_destroy(sp);
}
// create a plane of three rows, and exactly fill two with wide chars
SUBCASE("LayoutFillsPlaneWide") {
if(notcurses_canutf8(nc_)){
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 3,
.cols = 7,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "我能吞 下玻璃 ";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "我能吞 下玻璃 "));
free(line);
ncplane_destroy(sp);
}
}
// if we don't have scrolling enabled, puttext() with more text than will
// fit on the plane ought return error, but print what it can.
SUBCASE("LayoutLongNoScroll") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 14,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
const char boundstr[] = "quantum balls scratchy no?! truly! arrrrp";
int res = ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes);
CHECK(0 > res);
CHECK(0 == notcurses_render(nc_));
CHECK(bytes < strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "quantum balls scratchy no?! "));
free(line);
ncplane_destroy(sp);
}
SUBCASE("LayoutLongScroll") {
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = 2,
.cols = 13,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
ncplane_set_scrolling(sp, true);
size_t bytes;
const char boundstr[] = "quantum balls scratchy?! true! arrrrp";
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, boundstr, &bytes));
CHECK(0 == notcurses_render(nc_));
CHECK(bytes == strlen(boundstr));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "scratchy?! true! arrrrp"));
free(line);
ncplane_destroy(sp);
}
SUBCASE("LayoutLongLines") {
// straight from the zoo demo, which didn't work at first
const int READER_COLS = 71; // equal to longest line
const int READER_ROWS = 4;
const char text[] =
"Lorem ipsum dolor sit amet, consectetur adipiscing elit. Proin ornare "
"neque ac ipsum viverra, vestibulum hendrerit leo consequat. Integer "
"velit, pharetra sed nisl quis, porttitor ornare purus. Cras ac "
"sollicitudin dolor, eget elementum dolor. Quisque lobortis sagittis.";
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = READER_ROWS,
.cols = READER_COLS,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
ncplane_home(sp);
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, text, &bytes));
CHECK(bytes == strlen(text));
CHECK(0 == notcurses_render(nc_));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Proin ornare neque ac ipsum viverra, vestibulum hendrerit leo consequat. Integer velit, pharetra sed nisl quis, porttitor ornare purus. Cras ac sollicitudin dolor, eget elementum dolor. Quisque lobortis sagittis."));
free(line);
ncplane_destroy(sp);
}
SUBCASE("LayoutZooText") {
// straight from the zoo demo, which didn't work at first
const int READER_COLS = 64;
const int READER_ROWS = 8;
const char text[] =
"Notcurses provides several widgets to quickly build vivid TUIs.\n\n"
"This NCReader widget facilitates free-form text entry complete with readline-style bindings. " "NCSelector allows a single option to be selected from a list. " "NCMultiselector allows 0..n options to be selected from a list of n items. "
"NCFdplane streams a file descriptor, while NCSubproc spawns a subprocess and streams its output. "
"A variety of plots are supported, and menus can be placed along the top and/or bottom of any plane.\n\n"
"Widgets can be controlled with the keyboard and/or mouse. They are implemented atop ncplanes, and these planes can be manipulated like all others.";
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = READER_ROWS,
.cols = READER_COLS,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
ncplane_set_scrolling(sp, true);
size_t bytes;
ncplane_home(sp);
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, text, &bytes));
CHECK(bytes == strlen(text));
CHECK(0 == notcurses_render(nc_));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "to be selected from a list of n items. NCFdplane streams a file descriptor, while NCSubproc spawns a subprocess and streams its output. A variety of plots are supported, and menus can be placed along the top and/or bottom of any plane.Widgets can be controlled with the keyboard and/or mouse. They are implemented atop ncplanes, and these planes can be manipulated like all others."));
free(line);
ncplane_destroy(sp);
}
SUBCASE("LayoutZooTextNoScroll") {
// straight from the zoo demo, which didn't work at first
const int READER_COLS = 64;
const int READER_ROWS = 15;
const char text[] =
"Notcurses provides several widgets to quickly build vivid TUIs.\n\n"
"This NCReader widget facilitates free-form text entry complete with readline-style bindings. "
"NCSelector allows a single option to be selected from a list. "
"NCMultiselector allows 0..n options to be selected from a list of n items. "
"NCFdplane streams a file descriptor, while NCSubproc spawns a subprocess and streams its output. "
"A variety of plots are supported, and menus can be placed along the top and/or bottom of any plane.\n\n"
"Widgets can be controlled with the keyboard and/or mouse. They are implemented atop ncplanes, and these planes can be manipulated like all others.";
struct ncplane_options nopts = {
.y = 0,
2020-11-15 02:42:37 +00:00
.x = 0,
.rows = READER_ROWS,
.cols = READER_COLS,
.userptr = nullptr,
.name = nullptr,
.resizecb = nullptr,
.flags = 0,
};
auto sp = ncplane_create(n_, &nopts);
REQUIRE(sp);
size_t bytes;
ncplane_home(sp);
CHECK(0 < ncplane_puttext(sp, 0, NCALIGN_LEFT, text, &bytes));
CHECK(bytes == strlen(text));
CHECK(0 == notcurses_render(nc_));
char* line = ncplane_contents(sp, 0, 0, -1, -1);
REQUIRE(line);
CHECK(0 == strcmp(line, "Notcurses provides several widgets to quickly build vivid TUIs.This NCReader widget facilitates free-form text entry complete with readline-style bindings. NCSelector allows a single option to be selected from a list. NCMultiselector allows 0..n options to be selected from a list of n items. NCFdplane streams a file descriptor, while NCSubproc spawns a subprocess and streams its output. A variety of plots are supported, and menus can be placed along the top and/or bottom of any plane.Widgets can be controlled with the keyboard and/or mouse. They are implemented atop ncplanes, and these planes can be manipulated like all others."));
free(line);
ncplane_destroy(sp);
}
CHECK(0 == notcurses_stop(nc_));
}