/vaunt

A cms with visual editor written in V that you can use to generate static sites.

Primary LanguageVGNU General Public License v3.0GPL-3.0

Vaunt

Important

This repository will no longer be maintained as it's built on vweb, vlang's old web module. It is not compatible with veb.

Current version: 0.2

Vaunt is a static site generator with built-in cms and visual block editor written in V.

Disclamer: Vaunt is still early software. There might be breaking changes until 0.3, but these changes will be minor.

License About the license: any website generated with Vaunt is license free since it doesn't contain any code used in this repository.

vaunt_2

Features

  • Static site generator
  • easy to configure SEO (Search Engine Optimization)
  • The CMS backend is optional
  • Admin panel and visual block editor for content creation
  • User configurable themes
  • Image uploads automatically generate small, medium and large sizes
  • Serve, include, import & edit markdown files

See only use static generator if you only want to convert your vweb application into a static website and don't need the admin panel.

Requirements

Make sure you have V installed. You can check out the documentation to install V.

If you have installed V make sure you have the latest version installed by running v up.

Installation

Run the following command to install Vaunt with the V package manager:

v install --git https://github.com/Casper64/vaunt

Now you are able to import Vaunt directly into your projects with import vaunt!

Database

Vaunt works with all databases in the db module.

Quick Start

Go to the Vaunt default theme to start making your website!

Themes

It is possible to create themes with Vaunt. Go to the Vaunt default theme example to get a quick start with Vaunt.

Usage

In the following example we start a basic Vaunt app.

module main

import vaunt
import vweb
import os
import db.sqlite

const (
	template_dir = os.abs_path('templates') // where you want to store templates
	upload_dir   = os.abs_path('uploads') // where you want to store uploads
	app_secret   = 'my-256-bit-secret' // secret key used to generate secure hashes
)

// Your theme settings
struct Theme {}

// Base app for Vaunt which you can extend
struct App {
	vweb.Context
	vweb.Controller
	vaunt.Util
pub:
	template_dir string [vweb_global]
	upload_dir   string [vweb_global]
pub mut:
	dev   bool      [vweb_global] // used by Vaunt internally
	db    sqlite.DB
	theme Theme // Theme settings
}

fn main() {
	// insert your own database
	db := sqlite.connect('app.db')!

	theme := Theme{}

	// setup database and controllers
	controllers := vaunt.init(db, template_dir, upload_dir, theme, app_secret)!

	// create the app
	mut app := &App{
		template_dir: template_dir
		upload_dir: upload_dir
		db: db
		controllers: controllers
	}

	// serve all css files from 'static' (optional)
	app.handle_static('static', true)
	// start the Vaunt server
	vaunt.start(mut app, 8080, vaunt.GenerateSettings{})!
}

pub fn (mut app App) before_request() {
	// copy database connection to Util
	app.Util.db = app.db
}

pub fn (mut app App) index() vweb.Result {
	content := 'Hello!'
	app.s_html = content
	return app.html(content)
}

Note The App struct showed in the example contains all fields that must be present. They are used by Vaunt internally and your code won't work without them!

As you can see a Vaunt app is just a Vweb app with some predefined properties. And you can add other properties and methods to App as it is a regular Vweb application. The controllers that are generated control the api, admin panel and file uploads.

You can start the application with v watch run main.v for single files. If you have multiple .v files you can put them in the src folder and run v watch run src.

Only static generator

You can also use Vaunt to generate a static version of your vweb app if you don't need the CMS backend.

You can still use the SEO utilities.

You only have to set app.s_html in the routes you want to generate. See custom routes and generate for more information.

Example:

module main

import vaunt
import vweb

struct App {
	vweb.Context
pub mut:
	dev    bool   [vweb_global] // used by Vaunt internally
	s_html string // used by Vaunt to generate html
}

fn main() {
	mut app := &App{}
	vaunt.start(mut app, 8080, vaunt.GenerateSettings{})!
}

pub fn (mut app App) index() vweb.Result {
	// save html in `app.s_html` first before returning it
	app.s_html = 'index'
	return app.html(app.s_html)
}

Tips

If you don't want to add the Util struct to your app it's recommended to copy the url function into your app so when generating your app .html get's added correctly after the urls.

src/util.v

// url adds '.html' after the url if the site is being generated
// usage: `href="@{app.url('/my-page')}"`
pub fn (u &Util) url(url string) vweb.RawHtml {
	if u.dev {
		return '${url}'
	} else {
		if url.ends_with('/') {
			return '${url}index.html'
		}
		return '${url}.html'
	}
}

Admin panel

The admin panel is used to created articles via a visual editor. You can access the admin panel by navigating to "/admin".

Create a user

To be able to access the admin you will need to create a superuser.

v run src --create-superuser

Follow the instructions after which you can log in with the created user and password.

Authentication settings

By default, the API and admin panel can only be accessed when authenticated. Vaunt includes 3 function with which you can protect your routes / app.

Only allow authenticated users, else redirect to login page:

vaunt.login_required(mut app.Context, app_secret)

Only allow authenticated users, else send HTTP 401:

login_required_401(mut app.Context, app_secret)

Or you can check if the current user is a superuser and set vaunt.Util.is_superuser

Example:

app.is_superuser = vaunt.is_superuser(mut app.Context, app_secret)

You can put either one of these functions in pub fn (mut app App) before_request() to enable them for your whole app. Or call them in individual routes.

Authentitcation Caveats

When you generate the site all forms of authentication and middleware (with the exception of before_request)will be skipped, except if you return early.

Usage

The admin panel should be self-explanatory. You can press the create article button to create a new article and get into the block editor. If you hit publish in the right nav the html for your article will be generated.

Markdown

You can import any markdown file into Vaunt and edit it as "blocks". Just hit the import markdown button and follow the steps.

The imported markdown is not properly sanitized! The end user is responsible.

Currently supported markdown elements

  • paragraphs
  • h1-h6
  • Inline elements such as: links, bold/italic text, colored text
  • images (the src and alt attributes will be copied)
  • code blocks and inline code blocks
  • non-nested lists
  • blockquotes
  • tables
  • alerts from github's markdown notes: > **Note**

Some raw html might slip by the basic sanitizer. If so you have to manually remove them in the editor.

Routing

When creating a route the html that is returned must be saved in app.s_html before returning. If you forget to store the generated html in app.s_html the generated output of the static site won't contain your html.

There are three dynamic routes that vaunt can generate html for:

Article with a category

A method with name category_article_page and dynamic route "/articles/:category_name/:article_name". category_article_page is used to generate the html page for each article that belongs to a category.

If you press the publish button in the admin panel the html will be generated and outputted to "[template_dir]/articles/[category_name]/[article_name].html".

Example:

['/articles/:category_name/:article_name']
pub fn (mut app App) category_article_page(category_name string, article_name string) vweb.Result {
	// save html in `app.s_html` first before returning it
	app.s_html = app.category_article_html(category_name, article_name, template_dir) or {
		return app.not_found()
	}
	return app.html(app.s_html)
}

Article without a category

A method with name article_page and dynamic route "/articles/:article_name". article_page is used to generate the html page for each article.

If you press the publish button in the admin panel the html will be generated and outputted to "[template_dir]/articles/[article_name].html".

Example:

['/articles/:article_name']
pub fn (mut app App) article_page(article_name string) vweb.Result {
	// save html in `app.s_html` first before returning it
	app.s_html = app.article_html(article_name, template_dir) or { return app.not_found() }
	return app.html(app.s_html)
}

Tags

You can create tags in the admin panel and generate a html page for every tag. Add atag_page method with dynamic route "/tags/:tag_name".

The html pages are generated in "[template_dir]/tags/[tag_name].html".

Example:

['/tags/:tag_name']
pub fn (mut app App) tag_page(tag_name string) vweb.Result {
	tag := app.get_tag(tag_name) or { return app.not_found() }

	content := 'tag: ${tag_name}'

	// save html in `app.s_html` first before returning it
	app.s_html = content
	return app.html(app.s_html)
}

Custom Routes

All methods on your App that return vweb.Result are considered routes. Vaunt will make sure to output files that are reachable the same way as while running the dev server.

Example: A method with attribute ['/about'] will produce the html file about.html. As expected a method with attribute ['/nested/about'] will put html file at nested/about.html.

// will generate in `about.html`
pub fn (mut app App) about() vweb.Result {
	app.s_html = 'About Vaunt'
	return app.html(app.s_html)
}

Index routes

Index routes (or routes ending with a "/") will have index.html as ending. So the route /nested/ will put the html file at nested/index.html.

Dynamic routes

Vaunt works with dynamic routes, but you have to provide values of the dynamic arguments. See the generating dynamic routes section.

Generate

You can generate the static site by passing the --generate flag or -g for short. All files needed to host your website will be in the generated public directory. Including all static files.

v run [project] --generate

Correct URL's

Typically, you would write a link to another page in vweb like this:

<a href="/my-page">My page</a>

But when the website is generated the link breaks, because browsers expect html files. To fix this you can use the url method on vaunt.Util. This method adds .html after the passed route when the app is being generated.

Example:

<a href="@{app.url('/my-page')}">My page</a>

Result (when generated):

<a href="/my-page.html">My page</a>

Dynamic routes (parameters)

Vaunt distincts two kinds of dynamic routes (vweb routes with parameters e.g. "/path/:arg"). Routes with one argument and routes with multiple arguments.

Single parameter routes

If we have a dynamic route with one parameter on our app.

['/dyn/:dynamic']
pub fn (mut app App) custom_dynamic(dynamic string) vweb.Result {
	app.s_html = dynamic
	return app.html(app.s_html)
}

We have to specify what the values of dynamic will be when the site is being generated.

We provide these settings to vaunt.start

Example:

// main function
// ...

settings := vaunt.GenerateSettings{
	// `dynamic_routes` is a map. The key is the method name and the value is the kind of 
	// dynamic route
	dynamic_routes: {
		'custom_dynamic': vaunt.DynamicRoute{
			arguments: ['a', 'b', 'c']
		}
	}
}

vaunt.start(mut app, 8080, settings)!

This will output 3 files with their content being 'a', 'b' and 'c', respectively.

public/
└── dyn/
    ├── a.html
    ├── b.html
    └── c.html

Multiple paremeter routes

Let's see the case where we have multiple parameters in one route.

['/mult/:a/:b']
pub fn (mut app App) multiple_dynamics(a string, b string) vweb.Result {
	app.s_html = '${a}/${b}'
	return app.html(app.s_html)
}

Now we have two supply the arguments as an array of strings.

Example:

// main function
// ...


settings := vaunt.GenerateSettings{
	dynamic_routes: {
		'multiple_dynamics': vaunt.MultipleDynamicRoute{
			arguments: [['1', 'a'], ['2', 'b'], ['3', 'c']]
		}
	}
}

vaunt.start(mut app, 8080, settings)!

This will produce the following folder structure:

public/
└── mult/
    ├── 1/
    │   └── a.html
    ├── 2/
    │   └── b.html
    └── 3/
        └── c.html

Serving markdown from a folder

It is also possible to serve markdown files from one folder. We can use vaunt.get_html_from_markdown to convert the markdown files to html in our route.

Example:

const (
	// the directory containg the markdown files
	md_dir = 'md'
)

// the `...` after the parameter indicates that we want our parameter to match all routes after
// `"/md"` including any '/' characters
['/md/:path...']
pub fn (mut app App) from_markdown_folder(path string) vweb.Result {
	// `get_html_from_markdown` converts a markdown file to vaunt blocks and back into html.
	// It fails if the `path` doesn't have the ".md" extension
	raw_html := vaunt.get_html_from_markdown(md_dir, path) or {
		// markdown file does not exist
		return app.not_found()
	}
	app.s_html = raw_html
	return app.html(app.s_html)
}

We have to indicate to vaunt that from_markdown_folder is a dynamic route that serves markdown files from a folder, so it can be generated accordingly.

Example:

// main function
// ...


settings := vaunt.GenerateSettings{
	dynamic_routes: {
		'from_markdown_folder': vaunt.MarkdownDynamicRoute{
			md_dir: md_dir
		}
	}
}

vaunt.start(mut app, 8080, settings)!

MarkdownDynamicRoute will search the md_dir folder recursively and passes any markdown files (with the ".md" extension) as an argument to from_markdown_folder.

This would be the resulting folder structure when our app is generated:

md/
├── index.md
└── docs/
    └── docs.md
public/
└── md/
    ├── index.html
    └── docs/
        └── docs.html

Search Engine Optimization (SEO)

SEO is used to place your website higher in the rankings of search engines like google. One method to improve SEO is to provide metadata about the pages contents in the form of html meta tags. This method is used by Vaunt.

A common protocol is OpenGraph. You might have noticed that if you share a link via WhatsApp or any other messaging app you sometimes see this card markup with the name of the page from the link with a small description and often an image. This card is generated with the OpenGraph protocol. And it never hurts to enable it!

The integrated SEO settings can be enabled by adding vaunt.SEO to your App struct.

struct App {
// ...
pub mut:
    seo vaunt.SEO [vweb_global] // SEO configuration
// ...
}

And adding it to your html templates along with the OpenGraph prefix.

<html prefix="@app.seo.og.prefix">
<head>
    @{app.seo.html()}
</head>
</html>

Articles

The meta tags for articles can be automatically generated if you use SEO.set_article. Let's modify the article_page to enable SEO support (You can do the same for category_article_page).

['/articles/:article_name']
pub fn (mut app App) article_page(article_name string) vweb.Result {
	// get the article by name
	article := vaunt.get_article_by_name(app.db, article_name) or { return app.not_found() }
	// set seo
	app.seo.set_article(article, app.req.url)
	content := app.article_html(article_name, template_dir) or { return app.not_found() }

	// save html in `app.s_html` first before returning it
	app.s_html = content
	return app.html(content)
}

This will set the following meta tags:

  • meta with name="description"
  • og:title
  • og:description
  • og:image
  • og:url
  • og:type = 'article'
  • article:published_time
  • article:modified_time

Note about twitter: the Twitter API's default behaviour is to fallback on OpenGraph tags, so there's no need to set double meta tags like 'twitter:description'

The most common properties of OpenGraph and Twitter are present in SEO.og and SEO.twitter respectively. If you need any other properties you can add them to other_properties.

Example:

app.seo.og.other_properties['image:alt'] = 'Image Description'

Will result into

<meta property="og:image:alt" content="Image Description">

Routes

In other routes you can modify app.seo to your preferences. Let's add a title and description and set the page url for the about page.

// will generate in `about.html`
pub fn (mut app App) about() vweb.Result {
	app.seo.og.title = 'About Vaunt'
	app.seo.set_description('Vaunt is a cms written in V with a frontend editor in Vue. It was created by Casper Kuethe in 2023')
	app.seo.set_url(app.req.url)
	
	app.s_html = 'About Vaunt'
	return app.html(app.s_html)
}

Note SEO.set_url will prefix SEO.website_url to the passed path.

This will set the following meta tags:

  • meta with name="description"
  • og:title
  • og:description
  • og:url

Providing default options

Sometimes you want to define SEO properties for the whole website like the website url. You can set those properties when you create the app.

Example

mut app := &App{
	// ...
	seo: vaunt.SEO{
		// Provide the website's url
		website_url: 'https://example.com'
		twitter: vaunt.Twitter{
			// twitter:card
			card_type: .summary
			// twitter:site
			site: '@casper_kuethe'
			// twitter:creator
			creator: '@casper_kuethe'
		}
		og: vaunt.OpenGraph{
			// og:site_name
			site_name: 'Vaunt'
			article: vaunt.OpenGraphArticle{
				// article:author
				author: ['Casper Kuethe']
			}
		}
	}
}

For all available options see the SEO api.

Sitemap

The sitemap.xml file is automatically generated if you provide SEO.website_url.

Theme Settings

It is possible to make you theme configurable in the admin panel.

vaunt_theme

All fields of the Theme will be saved in the database and rendered in the admin panel.

Colors

You can add a color option with the type vaunt.Color. For example we could modify the Theme struct from the earlier example to include a background color:

struct Theme {
pub mut:
	background vaunt.Color
}

Class Lists

Let's say we want the option to display navigation links aligned left, centered or right. We can use the ClassList struct for that.

Example:

struct Theme {
pub mut:
    background vaunt.Color
    nav_align  vaunt.ClassList
}

// ...

fn main() {
    // ...
    theme := Theme{
		background: '#ffffff'
		nav_align: vaunt.ClassList{
			name: 'Navigation links aligmnent'
			selected: 'nav-center'
			options: {
				'nav-left':   'left'
				'nav-center': 'center'
				'nav-right':  'right'
			}
		}
	}
    // ...
}

In the example above the default background color is set to white and the options for the navigation links are set as following:

name will be the name displayed in the admin panel. selected will be the default option options is a map where the keys are the class names and the values are the names displayed in the admin panel.

Usage

Using the vweb's before_request middleware you can fetch the latest theme settings before the page is rendered. See the Vaunt default theme for a more comprehensive implementation.

Example:

// fetch the new latest theme before processing a request
pub fn (mut app App) before_request() {
	// copy database connection to Util
	app.Util.db = app.db
	
	// only update when request is a route, assuming all resources contain a "."
	if app.req.url.contains('.') == false {
		app.theme_css = vaunt.update_theme(app.db, mut app.theme)
	}
}

All options are available in templates using app.theme.

Example:

<nav class="@{app.theme.nav_align.selected}"></nav>

Will produce:

<nav class="nav-center"></nav>

The generated css is stored in app.theme_css and is a style tag which contains the css. You can directly include app.theme_css in your templates.

Example:

<head>
    @{app.theme_css}
</head>

Utility

Vaunt offers a few utility functions that you can use in your app for getting articles, categories and other stuff: see util.v

Most of the utility functions are available on the Util struct, and you could use them in templates, except for the functions that return a Result type.

<ul>
    @for article in app.get_all_articles()
        <li>@article.name</li>
    @endfor
</ul>

Templates

// url adds '.html' after the url if the site is being generated 
// usage: `href="@{app.url('/my-page')}"`
pub fn (u &Util) url(url string) vweb.RawHtml

// get the correct url in your templates
// usage: `href="@{app.article_url(article)} "`
pub fn (u &Util) article_url(article Article) vweb.RawHtml

// article_html returns the html for that article
pub fn (u &Util) article_html(article_name string, template_dir string) !vweb.RawHtml

// category_article_html returns the html for that article with category
pub fn (u &Util) category_article_html(category_name string, article_name string, template_dir string) !vweb.RawHtml

// html_picture_from_article_thumbnail returns a `<picture>` tag containing the different
// sizes of the articles thumbnail, if they exist.
// usage: `@{app.html_picture_from_article_thumbnail(article)}`
pub fn (u &Util) html_picture_from_article_thumbnail(article Article) vweb.RawHtml 

Table of Contents

All heading blocks in vaunt have an id in the same way github markdown generates id's for headings. A h1 tag with the text "This is a header" will have an id of "this-is-a-header-${index of block}". We can use these id's to generate a table of contents for each article.

The raw id without the block index is available as data attribute rawid.

Generated example:

<h1 id="this-is-a-header-0" data-rawid="this-is-a-header">This is a header</h1>

You can either get a default template, or the Block data to generate a table of contents yourself.

pub struct TOCBlock {
pub:
	// the header level: h1, h2, h3 etc.
	level int
	text string
	// the id of the article element
	link string
}

[params]
pub struct TOCParams {
	// by default show h1-h3
	min_level int = 1
	max_level int = 3
}

// get_toc returns a table of contents markup for the headers of `article`
pub fn (u &Util) get_toc(article Article, params TOCParams) vweb.RawHtml

// get_toc_blocks returns a list of the header blocks that can be used to generate a
// table of contents
pub fn (u &Util) get_toc_blocks(article Article, params TOCParams) []TOCBlock

Example:

<p>Table of contents:</p>
@{app.get_toc(article)}

Database

pub fn (u &Util) get_all_articles() []Article

pub fn (u &Util) get_articles_by_category(category int) []Article

pub fn (u &Util) get_articles_by_tag(name string) []Article
    
pub fn (u &Util) get_article_by_name(name string) !Article

pub fn (u &Util) get_article_by_id(id int) !Article

pub fn (u &Util) get_all_categories() []Category

pub fn (u &Util) get_category_by_id(id int) !Category

pub fn (u &Util) get_image_by_id(id int) !Image
    
pub fn (u &Util) get_all_tags() []Tag 

pub fn (u &Util) get_tags_from_article(article_id int) []Tag

pub fn (u &Util) get_tag(name string) !Tag

pub fn (u &Util) get_tag_by_id(id int) !Tag

Api

Blocks

See blocks.v for the JSON representation of each block.

// generate returns the html form of `blocks`.
pub fn generate(blocks []Block) string

Markdown

"Why would I use Vaunt if I could just import the mardown module?"

Vaunt wraps the html in it's own blocks, which are easier to style and it adds more metadeta to the blocks, like an id on heading tags. So you can keep the same style for imported markdown files in the editor and markdown files you serve as raw html.

// get_html_from_markdown converts the markdown file at `folder/path` into html
pub fn get_html_from_markdown(md_dir string, path string) !string

// get_blocks_from_markdown converts the markdown string `md` to Vaunt blocks.
// These blocks can be added to an article if they are JSON encoded.
pub fn get_blocks_from_markdown(md string) []Block

Including markdown files

You can include markdown files into your templates with Util.include_md

// include_md returns the html for the markdown in `file`
// usage: `@{app.include('path/to/file.md')}`
pub fn (u &Util) include_md(file string) vweb.RawHtml

Example:

@{app.include_md('test.md')}

Converting markdown to html

Example workflow to convert a markdown file to Vaunt blocks and convert that into html.

module main

import os
import vaunt

fn main() {
	markdown_file := 'vaunt.md'
	// get the file contents
	md := os.read_file(markdown_file)!
	// get the blocks
	blocks := vaunt.get_blocks_from_markdown(md)
	// get html from markdown
	html := vaunt.generate(blocks)
	html_file := 'vaunt.html'
	os.write_file(html_file, html)!
}

You could substite the code in the main function into a vweb route. See generating from markdown for an example.

Vweb Config

You can edit the vweb configuration in vaunt.start

Example:

vaunt.start(mut app, 8080, vaunt.GenerateSettings{}, host: '0.0.0.0', nr_workers: 4)
[params]
pub struct RunParams {
	family               net.AddrFamily = .ip
	host                 string = '127.0.0.1'
	nr_workers           int    = 1
	pool_channel_slots   int    = 1000
	show_startup_message bool   = true
}

Articles

The []Article type has a couple of built-in functions to filter the array.

pub fn (a []Article) no_category() []Article {
	return a.filter(it.category_id == 0)
}

pub fn (a []Article) category(id int) []Article {
	return a.filter(it.category_id == id)
}

pub fn (a []Article) visible() []Article {
	return a.filter(it.show == true)
}

pub fn (a []Article) hidden() []Article {
	return a.filter(it.show == false)
}

You can use these functions in your templates.

Example:

<!-- List all visible articles. See admin panel -->
<ul>
    @for article in app.get_all_articles().visible()
        <li>@article.name</li>
    @endfor
</ul>

Tags

Create and edit tags in the admin panel. See the Utility section for helper functions.

Database Models

[table: 'categories']
pub struct Category {
pub mut:
	id   int    [primary; sql: serial]
	name string [unique]
}

[table: 'articles']
pub struct Article {
pub mut:
	id          int    [primary; sql: serial]
	name        string [unique]
	category_id int
	description string
	show        bool
	thumbnail   int
	image_src   string // need this in json, but there is no skip_sql yet
	block_data  string [nonull]
	created_at  string [default: 'CURRENT_TIMESTAMP'; sql_type: 'TIMESTAMP']
	updated_at  string [default: 'now()'; sql_type: 'TIMESTAMP']
}

[table: 'images']
pub struct Image {
pub mut:
	id         int    [primary; sql: serial]
	name       string [nonull]
	src        string [nonull]
	article_id int    [nonull]
}

[table: 'tags']
pub struct Tag {
pub mut:
	id         int    [primary; sql: serial]
	article_id int
	name       string [nonull]
	color      string [nonull]
}

Theme Types

pub type Color = string

// options of classes
pub struct ClassList {
pub:
	name string
	// The value is the option name displayed in the editor 
	// and the key is the class name
	options map[string]string
pub mut:
	selected string
}

SEO

pub struct SEO {
pub mut:
	// twitter card configuration
	twitter Twitter
	// Open Graph configuration
	og OpenGraph
	// your websites url
	website_url string
	// your pages description. Is automatically set by `set_article`
	description string
	// If you need any other meta tags. The map key is the property attribute
	// and the map value is the content attribute.
	other_properties map[string]string
}

Twitter

Implementation from twitter card docs.

// TwitterCardType implement the meta `twitter:card` types 
pub enum TwitterCardType {
	summary
	summary_large_image
	app
}

pub struct Twitter {
pub mut:
	card_type TwitterCardType [name: 'card'] = .summary
	// @username for the website used in the card footer.
	site string
	// @username for the content creator / author.
	creator string
	// all other properties of twitter. The map key is the property attribute
	// and the map value is the content attribute.
	// All properties are prefixed with 'twitter:'
	other_properties map[string]string
}

OpenGraph

OpenGraph meta tags, implementation from the OpenGraph website.

pub struct OpenGraph {
pub:
	// add this as `prefix="@app.seo.og.prefix"` attribute to the `html` tag in your page
	prefix string [skip] = 'og: https://ogp.me/ns#'
pub mut:
	title            string
	og_type          string           [name: 'type']
	image_url        string           [name: 'image']
	url              string
	description      string
	audio            string
	determiner       string
	locale           string
	locale_alternate []string         [name: 'locale:alternate']
	site_name        string
	video            string
	article          OpenGraphArticle
	// all other properties of opengraph. The map key is the property attribute
	// and the map value is the content attribute.
	// All properties are prefixed with 'og:'
	other_properties map[string]string
}

OpenGraph article:

// OpenGraph article attributes; are filled in automatically for each article
// time fields must follow ISO_8601
pub struct OpenGraphArticle {
pub mut:
	published_time  string
	modified_time   string
	expiration_time string
	author          []string
	section         string
	tag             []string
}

Extensibility

The frontend editor is made with Vue and Editorjs. You can already extend the ditor with custom blocks.

In the future custom blocks will be able to be registerd in Vaunt by passing a function that takes a EditorJs block as input and outputs html.

The goal of this project is to provide a backend cms which developers can extend to create their own themes and extensions.