Munin is an enhancement to the Terminal++ library and provides a set of textual, component-based UI widgets. Windows are composed using layouts that automatically size and position their subcomponents according to the rules you specify:
view(
make_compass_layout(),
view(
// Lay out the chess board and move lists next to each other
make_grid_layout({2, 1}), // two components horizontally
view(
make_compass_layout(),
chess_board,
compass_layout::heading::centre,
make_fill("\\[6\\U2502"_ete), // cyan vertical bar
compass_layout::heading::east),
view(
make_grid_layout({1, 2}), // two components vertically
make_scroll_pane(white_moves_history),
make_scroll_pane(black_moves_history)))),
compass_layout::heading::centre,
// Lay out a view for a button below the chess board and move lists
view(
make_compass_layout(),
make_fill("\\[6\\U2500"_ete), // cyan horizontal bar
compass_layout::heading::north,
make_fill(' '),
compass_layout::heading::centre,
ok_button,
compass_layout::heading::east),
compass_layout::heading::south);
See The Chess Example for an example of how to tie this into a console application.
Munin requires a C++17 compiler and the following libraries:
- Boost (At least version 1.69.0)
- nlohmann_json (At least version 3.3.0)
- Terminal++ (At least version 2.4.0)
- (for Terminal++) libfmt (At least version 5.3) and gsl-lite (at least version 0.38)
- Optionally, Console++ (At least version 0.1.0)
- (for testing only) Google Test and Google Mock
Munin can be installed from source using CMake. This requires Boost, Terminal++ and any other dependencies to have been installed beforehand, using their own instructions, or for the call to cmake --configure
to be adjusted appropriately (e.g. -DBOOST_ROOT=...
or -DGtest_DIR=...
). If you do not wish to install into a system directory, and thus avoid the use of sudo, you can also pass -DCMAKE_INSTALL_PREFIX=...
into the cmake --configure
call.
git clone https://github.com/KazDragon/munin.git && cd munin
mkdir build && cd build
cmake --configure -DCMAKE_BUILD_TYPE=Release ..
cmake --build .
sudo cmake --install .
- Component - the base class of all UI components
- Container/Layout - a component that can contain and arrange other components
- A sampling of common layouts
- Null Layout - a layout that doesn't.
- Vertical Strip Layout - a layout that arranges components in vertical strips
- Horizontal Strip Layout - a layout that arranges components in horizontal strips
- Grid Layout - a layout that arranges components in equally-sized cells.
- Aligned Layout - a layout that arranges components with horizontal and vertical alignment (e.g. top-left, right-center)
- Compass Layout - a layout that arranges components according to compass points (e.g. North, South, Centre)
- A sampling of common components
- Filled Box - a component that simply paints a single repeated character
- Brush - a component that paints a repeated character pattern.
- Image - a component that paints a static "image" once.
- Window - a top-level object that contains and draws components onto a canvas.
Implement the ability to frame components
- Composite Component - a component that is made of many other components, but presents the interface of being just one component.
- Solid Frame - a frame made of solid characters.
- Named Frame - a frame that displays text in addition to the solid characters.
Implement clickable buttons
- Button - a normal click button
- Toggle Button - a button that can be toggled on [x] and off [ ].
Fundamentals for all types of text components:
- Viewport - a component that sees a smaller portion of the underlying component
- Horizontal Scroll Bar
- Vertical Scroll Bar
- Scroll Frame - a frame that includes scroll bars
- Scroll Pane - a component that includes a scroll frame and a viewport
Specific text manipulation components:
- Edit - a single-lined horizontally scrolling text box, with a frame.
- Text Area - a multiple-lined vertically scrolling text box, with a frame.
- List - A vertical list of elements that can be individually selected.
Munin is automatically tested with MSVC 2019 and GCC 7.5. For further information about the working status of Munin to report any bugs, or to make any feature requests, visit the issues page. Feel free to discuss using Github Discussions!
The purpose of the Munin library is to provide a windowed component and layout framework for terminal applications, such as those executed on the command line, but also especially those provided by a Telnet server (e.g. a MUD, or chatter). Integration with either the command line or with a Telnet server is not provided by Munin; that is left to the integrator of the library. Examples can be found nearby this repository. For example, the Paradice9 (a chatter with tabletop roleplaying extensions) and Textray (a raycasting 3D room renderer) projects are both built using Server++, Telnet++, Terminal++, and Munin.
The core abstraction within Munin is the component. Everything that is drawable in one way or another derives from this abstraction so that they can provide knowledge about where they are (position), how large they are (size), how large they would like to be (preferred size), what to do if a user interacts with it (event) and how to represent themselves on a canvas (draw).
Another key abstraction within Munin is the container. A container, when combined with a layout, declares how to arrange a set of components. Some layouts, for example the grid layout, simply place components in known positions according to the order in which they are added to the container. Other layouts, such as the compass and aligned layouts, require the developer to supply hints about where to place each component. The end effect is that the container automatically adjusts the size and position of its components according to how large the container itself is.
At the top of the stack is the window. This is where the underlying application integrates with Munin. A window is not a component, but is instead constructed with a component that represents the user interface to be displayed. From here, events can be sent to the window's components. Ideally, these will be directly from Terminal++, and the default set of components are built with this in mind. But the interface is flexible enough to allow custom messages to be propagated to custom components. Additionally, you can register to receive a callback whenever something in the interface changes that requires repainting, and you can of course command to the window to repaint itself. This returns a string of text that you can send to a terminal to effect the repaint on the client.