hypertextcpp is an HTML templating system for C++ applications. It provides an .htcpp template file format and a command-line utility that transpiles it to C++ HTML rendering code. Include the generated C++ header file in your project, set up a build system to update it when necessary, and you're all set.
A quick example:
<html>
<body>
<h1>$(cfg.name)'s todo list:</h1>
<p> No tasks found </p>?(cfg.tasks.empty())
<ul>
<li>$(task.name)</li>@(auto task : cfg.tasks)
</ul>
</body>
</html>
Now let's generate the C++ header:
kamchatka-volcano@home:~$ hypertextcpp todolist.htcpp
This command creates the todolist.h
file in the current working directory. All that's left is to use it in our program.
examples/01/todolist_printer.cpp
#include "todolist.h"
#include <string>
#include <vector>
int main()
{
struct PageParams{
struct Task{
std::string name;
};
std::string name = "Bob";
std::vector<Task> tasks = {{"laundry"}, {"cooking"}};
} pageParams;
auto page = todolist{};
page.print(pageParams);
return 0;
}
Compile it with your preferred method, launch it, and you will get this output:
kamchatka-volcano@home:~$ ./todolist_printer
<html>
<body>
<h1>Bob's todo list:</h1>
<ul>
<li>laundry</li><li>cooking</li>
</ul>
</body>
</html>
$(c++ expression
)
Expressions are used to add the application's data to the template. It can be any valid C++ expression, with the only condition that its result must be streamable to the default output stream std::ostream
. Expressions can be placed anywhere in the HTML template, except in tag names.
In our todolist example, $(cfg.name)
is an expression that adds the template config variable name
to the result page.
${c++ statement(s)
}
Statements are used to add any valid C++ code to the template rendering function. For example, you can add variables, class declarations, or lambdas. Let's say you don't like the default name cfg
used for passing data to the template. You can create a reference to it with any name you like and use it later:
${ auto& param = cfg;}
<h1>$(param.name)'s todo list:</h1>
Note, that cfg
and out
are reserved names used for parameters in the generated
rendering function. Also, don't put anything in the htcpp
namespace.
#{c++ statement(s)
}
These statements are used to add any valid C++ code outside the template rendering function. Unlike regular statements, with global ones you can add include directives or function definitions. Global statements can only be placed at the top level of the htcpp
template, outside any HTML element. Please don't put anything in the htcpp
namespace.
If hypertextcpp used a common approach for control flow in HTML template engines, our todolist example would look something like this:
<html>
<body>
<h1>%%cfg.name%%'s todo list:</h1>
%%if cfg.tasks.empty()%%
<p> No tasks found </p>
%%end%%
<ul>
%%for auto task : cfg.tasks%%
<li>$(task.name)</li>
%%end%%
</ul>
</body>
</html>
In our opinion, it significantly hurts the readability of the document tree and makes it hard to choose indentation and keep it consistent — notice how different approaches are used for if and for blocks in the example.
hypertextcpp solves this problem by applying control flow to the HTML elements themselves without adding logic block scopes to the document. It uses just two extensions for tags to make this work:
-
Conditional extension
?(c++ condition
)
HTML elements with this extension are added to the document only when condition is fulfilled. Example of usage from our todolist template:<p> No tasks found </p>?(cfg.tasks.empty())
-
Loop extension
@(c++ init-statement; condition; iteration_expression
)
or
@(c++ range_declaration : range_expression
)
HTML elements with this extension are added to the document multiple times, on each step of the loop or for each element of the iterated range. Example of usage from our todolist template:<li>$(task.name)</li>@(auto task : cfg.tasks)
Let's add another example for another type of
for
loop:<li>Task#$(i)</li>@(auto i = 0 ; i < 3; ++i)
which evaluates to
<li>Task#0</li> <li>Task#1</li> <li>Task#2</li>
Both extensions can be added to the opening or closing tag, but each tag and HTML element can only have one extension.
It's recommended to add the extension to the opening tag for multiline HTML elements:
<div>?(cfg.greet)
<p>Hello world!</p>
</div>
and to the closing tag for the single-line ones:
<div><p>Hello world!</p></div>?(cfg.greet)
Note that while extensions hide control flow block scopes from the template document, they're still present in the generated C++ code and implemented with regular if
and for
control structures. Therefore, a template like this:
<div>@(auto i = 0; i<3; ++i)
${ auto num = i*2;}
<p> Item #$(num)<p>
</div>
<p> Last num is $(num) </p>
won't compile because the num
variable isn't visible outside the for
block scope generated by the loop extension on the div
tag.
[[ text, html elements, statements, expressions or other sections
]]
Sections can contain a part of the template document, and it's possible to attach control flow extensions to them. Their main usage is adding attributes to HTML elements conditionally.
Let's update the todolist example by adding a line-through text style to completed tasks:
examples/02/todolist.htcpp
<html>
<body>
<h1>$(cfg.name)'s todo list:</h1>
<p> No tasks found </p>?(cfg.tasks.empty())
<ul>
<li [[style="text-decoration: line-through;"]]?(task.isCompleted)>$(task.name)</li>@(auto task : cfg.tasks)
</ul>
</body>
</html>
Don't forget to update the C++ template config structure!
examples/02/todolist_printer.cpp
//...
struct PageParams{
struct Task{
std::string name;
bool isCompleted = false;
};
std::string name = "Bob";
std::vector<Task> tasks = {{"laundry", true}, {"cooking", false}};
} pageParams;
//...
Tip of the day: Keep your template tidy and don't introduce sections when it's possible to attach extensions to HTML elements.
#procedureName
(){html elements, statements, expressions or sections
}
Parts of the htcpp
template can be placed inside procedures—parameterless functions capturing the cfg
variable. They are available for call from the C++ application, so if any part of the page needs to be rendered separately from the whole template, procedures are a big help.
Procedures can only be placed at the top level of the htcpp
template, outside any HTML element.
Let's put the list of tasks from the todolist example in the procedure:
#taskList(){
<li [[style="text-decoration: line-through;"]]?(task.isCompleted)>$(task.name)</li>@(auto task : cfg.tasks)
}
<html>
<body>
<h1>$(cfg.name)'s todo list:</h1>
<p> No tasks found </p>?(cfg.tasks.empty())
<ul>
$(taskList())
</ul>
</body>
</html>
Now the tasks list can be output to stdout by itself like this:
examples/03/todolist_printer.cpp
//...
auto page = todolist{};
page.print("taskList", pageParams);
//...
kamchatka-volcano@home:~$ hypertextcpp --help
Usage: hypertextcpp <input> [params] [flags]
Arguments:
<input> (path) .htcpp file to transpile
Parameters:
-o, --output <path> output c++ file path
(if empty, current working directory is used)
(optional, default: "")
-c, --class-name <string> generated class name
(optional)
Flags:
-s, --shared-lib generate result as shared library source
files
--class-pascalcase generate class name by using .htcpp filename
in PascalCase
--class-snakecase generate class name by using .htcpp filename
in snake_case
--class-lowercase generate class name by using .htcpp filename
in lowercase
--help show usage info and exit
Process finished with exit code 0
By default, the hypertextcpp transpiler works in a single header mode and generates a C++ header file that you're supposed to simply include in your project. A generated renderer class has the name of the .htcpp
template file. You can override the name by using the --class-name
parameter, or you can specify one of the following flags: --class-pascalcase
, --class-snakecase
, or --class-lowercase
to use the .htcpp
template's filename converted to the corresponding case as a class name.
Converting the template to C++ code each time you modify it is a laborious task, so it makes sense to add this step to your build process. To do this with CMake you can use hypertextcpp_GenerateHeader
function from the hypertextcpp.cmake
file.
Note that this function launches the hypertextcpp
executable, so it should be installed on your system first.
cmake_minimum_required(VERSION 3.18)
include(../../hypertextcpp.cmake)
hypertextcpp_GenerateHeader(NAME todolist CLASS_NAME TodoList)
set(SRC
todolist_printer.cpp
todolist.h)
add_executable(todolist_printer ${SRC})
target_compile_features(todolist_printer PUBLIC cxx_std_17)
set_target_properties(todolist_printer PROPERTIES CXX_EXTENSIONS OFF)
Now, every time you change the template, the corresponding header will be regenerated on the next build.
It can feel quite wasteful to rebuild your project each time the template file is changed, so hypertextcpp supports the generation of a C++ source file for building templates in the form of shared libraries and linking them dynamically from your application.
It requires duplicating the config declaration in the .htcpp template, registering it with the HTCPP_CONFIG
macro in both the template and the application source, generating the renderer code with the --shared-lib
command line flag, building the library, and loading it using the tiny API installed from the shared_lib_api/
directory. It sounds scarier than it is, so let's quickly update the todolist example to see how it works.
First we need to copy the config structure declaration in the template:
examples/05/todolist.htcpp
#{
#include <vector>
struct PageParams{
struct Task{
std::string name;
bool isCompleted = false;
};
std::string name = "Bob";
std::vector<Task> tasks = {{"laundry", true}, {"cooking", false}};
} pageParams;
HTCPP_CONFIG(PageParams);
}
#taskList(){
<li [[style="text-decoration: line-through;"]]?(task.isCompleted)>$(task.name)</li>@(auto task : cfg.tasks)
}
<html>
<body>
<h1>$(cfg.name)'s todo list:</h1>
<p> No tasks found </p>?(cfg.tasks.empty())
<ul>
$(taskList())
</ul>
</body>
</html>
Be sure to use an exact copy; any mismatch of the config structure between the template and the application can't be handled gracefully. So, if you try to load a template library with a different structure, you'll definitely crash the application and maybe hurt someone as a result.
Next, we need to build our template renderer as a library. It's not possible to bundle multiple template files in one library, so we can build a library from a single .htcpp
file by using the hypertextcpp_BuildSharedLibrary
CMake function from hypertextcpp.cmake
:
cmake_minimum_required(VERSION 3.18)
hypertextcpp_BuildSharedLibrary(
NAME todolist
OUTPUT_DIR ${CMAKE_CURRENT_BINARY_DIR}
)
add_dependencies(${PROJECT_NAME} todolist)
If everything goes right, you'll get libtodolist.so
in the build directory. That's our todolist.htcpp
template compiled as a shared library.
Next, let's modify todolist_printer.cpp
to be able to load it:
examples/05/todolist_printer.cpp
#include <hypertextcpp/templateloader.h>
#include <string>
#include <vector>
struct PageParams{
struct Task{
std::string name;
bool isCompleted = false;
};
std::string name = "Bob";
std::vector<Task> tasks = {{"laundry", true}, {"cooking", false}};
};
HTCPP_CONFIG(PageParams);
int main()
{
auto pageParams = PageParams{};
auto page = htcpp::loadTemplate<PageParams>("libtodolist.so");
page->print(pageParams);
return 0;
}
On Linux, it may be necessary to link the system library dl
to the build config, as we load the template library dynamically.
todolist_printer
should compile and work the same as the previous example using the single-header approach.
git clone https://github.com/kamchatka-volcano/hypertextcpp.git
cd hypertextcpp
cmake -S . -B build
cmake --build build
cmake --install build
cmake --install build --component shared_lib_api
You can skip the installation of shared_lib_api
component if you don't need to load templates in shared libraries form.
cd hypertextcpp
cmake -S . -B build -DENABLE_TESTS=ON
cmake --build build
cd build/tests && ctest
hypertextcpp is licensed under the MS-PL license