SECST Markup Language
SECST ([S]emantic, [E]xtensible, [C]omputational, [S]tyleable, [T]agged) markup is a more expressive, but easy to learn and use, superset and alternative to regular Markdown. It provides these features and more:
[:value[2 * 2] = .
Use SECST to joyfully create compelling, interactive HTML documents.
This is the source for the introduction above:
:abbr[SECST] (**[S]**emantic, **[E]**xtensible, **[C]**omputational, **[S]**tyleable, **[T]**agged) markup is a more expressive, but easy to learn and use, superset and alternative to regular Markdown. It provides these features and more:
:ul[
- Support for almost all HTML tags :a(.#tags)[and then some...], with :a(.#theme)[themes and styles].
- :a(.#tables-of-contents)[Table of contents], :a(.#footnotes)[footnote], :a(.#hashtags)[hashtag] and :a(.#mentions)[social media] support.
- Reactive :a(.#values-and-formulas)[formulas and data import] that are as simple as Excel formulas, e.g. :code[:value[2 * 2]] = :value[2 * 2].
- SEO optimization via :a(.#server-processing)[server rendering], :a(.#meta-tags)[meta tags], and JSON-LD :a(.#structured-data)[structured data] for both content generation and layout.
]
Use SECST to joyfully create compelling, interactive HTML documents.The SECST language, reference HTML transpiler and runtime support implementations are currently in ALPHA. Most, but not all, documented features are implemented, testing is far from complete and resource sizes have not been fully optimized.
This site's navigation, footnotes, computations, special styles like Key formatting and imported data are built entirely using SECST markup.
You can tap on the icon next to a heading to access a table of contents at any time. You can goto the top of the guide by tapping on SECST in the top left corner.
This guide is comprehensive, which may make it a little overwhelming. However, SECST has been designed so that you can start using it just a few tags and capabilities at a time.
The following order of learning is suggested:
align on img.sheetMusic or Latex.For gentle overviews see the Medium and Hackernoon articles:
<b> for bold is not semantic, but <strong> for indicating importance and rendering as bold is semantic.
static on audio, img, video and value tags.If you only need basic Markdown, then SECST is probably not for you.
If you need extended Markdown, then SECST may be a good option.
If you need more than Markdown, then SECST may be your only option short of HTML, CSS, and JavaScript so long as you are focused on interactive document authoring. If you are building an application use Svelte, VUE, React or some other framework.
See https://www.markdownguide.org/basic-syntax/.
| Document Item | Markdown | SECST | Example |
|---|---|---|---|
| bold | **text** | :strong[text] and **text** | text |
| italics | *text* | :em[text] and *text* | text |
| code | 'the code' | :code[the code] and 'the code' | the code |
| newline | a new line | a new line or :br | |
| paragraph | two new lines, i.e. an empty line | two new lines or :p[<content>...] |
Blockquote
> quoted text or :blockquote[quoted text]
> quoted text line one
> quoted text line twoquoted text line one
quoted text line two
Nested blockquotes are supported using nested :blockquote[], but not using Markdown shorthand.
Lists
Must be wrapped in :ol[] or :ul[]. You can use Markdown shorthand inside the SECST tag content.
:ol[
1. Item One
- Item Two
]
Also see Lists
Headings
# heading text where there can be 1 - 6 #.
:h<num>[heading text] where num is 1 - 6.
Horizontal Rule
*** or :hr
See https://www.markdownguide.org/extended-syntax/.
| Document Item | Markdown | SECST | Example |
|---|---|---|---|
| footnotes | [number] | :footnote[number] or :footnote[] for automatic numbering | |
| heading ids | # My Heading {#my-id} | :h1(#my-id)[My Heading] | |
| definition lists | see definition lists | ||
| strikethrough | ~~struck out~~ | :strike[struck out] and ~~struck out~~ | |
| task lists | see task lists | ||
| emojis | :emoji-name: | :emoji[emoji-name] and :smile: | 😄 |
| highlight | ==marked words== | :mark[marked words] and ==marked words== | marked words |
| subscript | H~2~O | H:sub[2]0 and H~2~O | H2O |
| superscript | X^2^ | X:sup[2] and X^2^ | X2 |
| automatic URL linking | https://www.google.com | :a[https://www.google.com] and https://www.google.com | https://www.google.com |
Code Block
```code block``` or :code[code block]
code block
Although most Markdown is supported, authors should write documents using semantic tags with the format <tag>[<content>] or <tag>(<parameters>)[<content>].
For example:
:strong[bolded important content] produces bolded important content:em[italicized emphasized content] produces italicized emphasized content Almost every semantic tag supported by HTML5 is one of the tags supported by SECST. Most non-semantic tags are excluded from SECST, e.g. <div> and <span>. However, since they are so commonly used :b[] for bold and :i[] for italic are included, as are aliases :bold[] and :italic[].
A few tags of note are documented below to illustrate the use of SECST or highlight special features.
Contents wrapped in :code[] will be inline unless the contents contain a newline, in which case the code will be displayed in block format.
Both `code` and ```code``` are supported.
The :details[] tag supports the conditional display of content. It has one required sub-tag :summary[<short text>] and the rest of the content is conditionally displayed. If a summary tag is missing, then the first word is used as the summary.
:details[:summary[See the content...] Hello there!] renders as See the content...
Hello there!
:details[... Hello there!] renders as ...
Hello there!
Headings automatically get ids you can use for the destination of links. The id is the same as the text content in lower case with spaces and special characters replaced by dashes.
To make it more convenient, you can also provide your own ids, e.g. :h1(#myheading)[My Heading].
Note: All tags can optionally take an id starting with #. If an id is present, it must be the first parameter between the parentheses.
You could target this heading with the hyperlink :a(.#myheading)[Go To My Heading].
Note the use of a shorthand "relative hash" (a hash starting with a period) for the destination of Go To My Heading. This could also be specified as: :a({href:"#myheading"})[Go To My Heading].
There is a special heading h:[] with no number. If this is the first element on a :section[], then it will automatically get a number based on section nesting. This allws the copying and pating of sections from one place to another without worrying about updating headings.
Special characters and symbols can be displayed using either &<character-id>; or :&[<character-id>...]. The second form can display multiple characters or symbols, just leave off the leading "&" and trailing ";" in the listed items.
The code ☻ renders as ☻ .
The code :&[female male #9893] renders as ♀♂⚥.
Toptal has a great reference for for special characters.
Sometimes, e.g. with hyperlinks, you need to specify more than content. This is provided using an attribute block in parentheses, e.g.
:a({href:"./index.txt",target:"_tab"})[SECST] yields SECST , where you can access the source of this document.
For convenience, any tag that can accept a url (either src or href in HTML) does not require the attribute, just provide the URL. And, single quotes can be used for attribute values, e.g.
:a(./index.txt {target:'_tab'})[SECST] yields SECST.
You can also use document ids prefixed with a period as targets, e.g. .#headings for this section. The period prefix is necessary to distinguish from providing an id for the link itself.
You can also specify hyperlinks with just a content value, e.g. :a({target:"_tab"})[./index.txt] renders as ./index.txt.
By default, relative URLS and ids, i.e. those starting with a period, have a target that is the current window or tab. And, those starting with a protocol, e.g. https:// have a target of a new tab.
:ins[inserted] renders as inserted
:del[deleted] renders as deleted
Both ordered and unordered lists are supported using ol and ul tags with li tags.
:ul[:li[LinkedIn] :li[Facebook] :li[Twitter]]
There is also a shorthand:
:ul[
- LinkedIn
- Facebook
- Twitter
] The same shorthand also works for ol. Since SECST already knows the list is ordered, numbers are inserted. The main utility of using numbers, is to reset numbering:
:ol[
2. LinkedIn
- Facebook
- Twitter
]
The standard HTML attributes for configuring ordered lists, reversed, start, type are also supported.
:ol({type:"a"})[
- LinkedIn
- Facebook
- Twitter
]
Definition lists follow the MDN documentation for their parallel HTML tags except non-semantic tags can't be used.
:dl[
:dt[Beast of Bodmin]
:dd[A large feline inhabiting Bodmin Moor.]
:dt[Morgawr]
:dd[A sea serpent.]
:dt[Owlman]
:dd[A giant owl-like creature.]
]
The :meta[] tag inserts values into the head of a document. It takes the form :meta({name:"my-name"})[my value], e.g. :meta({name:"author"})[John Jones] injects the HTML .
Suggested meta tag names include:
index or noindex and follow or nofollow.
Also see the section Structured Data.
Since SECST has a focus on semantics, images must be specified with alt text braces, even if they contain no content.
:img(static ./assets/images/anywhichway_mobius_138.png) will not transpile.
Images can be loaded statically at transpile time and delivered bundled into the output HTML.
:img(static ./assets/images/anywhichway_mobius_138.png)[Test]src attribute of the <img title="Test" alt="Test" src=""> The callout for the generated HTML above is implemented using the details tag.
The :p[] tag wraps paragraph content. It accepts plain text and phrasing content, i.e. most other tags.
Inside the content of a :section[], SECST will treat two newlines in sequence as a paragraph break. Newlines inside a p tag will produce separate HTML <br> elements. A :p[] tag is not strictly necessary for really simple content. A SECST parser/transpiler will do a reasonable job of figuring out where paragraph breaks are required; however behavior may vary from implementation to implementation.
If you get an unexpected document layout, try putting content inside a :p[] tag.
:p[] allows three attributes:
Also see the technical documentation.
The content of the :style[] tag is CSS. The CSS is automatically sanitized.
The tag can tag one attribute selector which will prefix all styles in the style block with the selector to limit their scope of application.
Also see the theme tag, which is much more powerful and Styling Content.
SECST tagging inside of :code[] is automatically escaped. If you wish to escape it elsewhere use the tag :escape[].
:code[:strong[text]] renders as :strong[text].
:escape[:strong[text]] renders as :strong[text].
Emojis can take the form :emoji[<emoji-name>...], where <emoji-name>... is a space separated list of emoji names or the short form, :<emoji-name>: can be used.
There are over 1,800 emojis. If a match for the emoji name can't be found, it is displayed as plain text.
See EmjoiMart to find just the right emoji.
:emoji[smile] and :smile: both render as 😄.
:emoji[smile frowning jumping] renders as 😄 😦 :jumping.
Most emoji's are colored. You can use grayscale emojis by providing the attribute grayscale.
:emoji(grayscale)[smile] renders as 😄.
:emoji(grayscale)[smile frowning jumping] renders as 😄 😦 :jumping.
Footnotes are inserted with the :footnote[<footnote-text>] tag. The content between the braces is used as the footnote and placed at the end of the document. Auto-numbering is done so that authors do not have to keep updating footnotes as the document changes. 1
The source for the above footnote is:
:footnote(#markdown-footnote)[Markdown does not autonumber footnotes, which makes them hard to maintain. Although, it does create back-references.]. To re-use a footnote, give the first occurrence an id. Then, give the subsequent uses an href attribute or shorthand relative URL reference using the id and leave the content section between the braces empty. This will override the anchors for numbered links generated by the SECST transpiler, but not break ordering. 2 If you do provide content when reusing, it will be ignored.
The source for the second footnote is: :footnote(.#markdown-footnote).
And, here is a third, :footnote[Footnote management does not get much easier!], that just gets numbered. 3
The :forEach[] tag supports repeating content based on an iterable object, typically an array or the results of using a CSS selector to gather values from a document.
The variables item, index, iterable are available to string template literals in the content.
:ol[
:forEach({iterable:["a","b","c"]})[
:li[${item} at ${index}]
]
]renders as
:value(#a hidden literal)[a] :value(#b hidden literal)[b] :value(#c hidden litral)[c]
:ol[
:forEach({iterable:"#a, #b, #c"})[
:li[${item} at ${index}]
]
] renders as
Hashtags can take the form #<value>[] or #[<value>...], or :hashtag[<value>...] where <value>... is a space separated list of words to hashtag.
#secst-is-great[] renders as #secst-is-great.
#[secst-is-great javascript-is-great] renders as#secst-is-great, #javascript-is-great.
:hashtag[secst-is-great javascript-is-great] renders as #secst-is-great, #javascript-is-great.
Hashtags are automatically added to the meta tags in the head of a document for indexing and SEO purposes.
Math and chemical formulas can be displayed using the :latex[] tag.
:latex[Lift = \frac{1}{2} \rho v^2 S C_L] renders as
:latex[\ce{CO2 + C -> 2 CO}] renders as
To display the generated formula as a block, end the content between the [] with a newline.
New tags can be defined in the content of any document through the use of macros.
A macro takes a name attribute, which will be the new SECST tag name and a tag attribute which will be the top level HTML tag to use for replacing the macro tag.
The content of the macro is used as a substitute for the macro. The content can contain variables that are populated based on attributes provided to the new tag when it is used.
An optional attribute contentAllowed specifies if instances of the macro tag can provide content to append at the end.
:macro({name:"profile",tag:"p",contentAllowed:true})[
:em[${name}]
:em[${age}]
]
:profile({name:"joe",age:26})[Hi there ${name}!] // Hi there joe! gets appended since contentAllowed is truerenders as
joe
26
Hi there joe!
with the HTML
<p> <em>joe</em><br> <em>26</em><br> Hi there joe!</p>
:mermaidChart[] is a block level element, it belongs outside of paragraphs. Just provide Mermaid notation as the content.
:mermaidChart[
graph LR
A --- B
B-->C[fa:fa-ban forbidden]
B-->D(fa:fa-spinner);
]renders as
graph LR
A --- B
B-->C[fa:fa-ban forbidden]
B-->D(fa:fa-spinner);
There are lots of charts to choose from:
Questions can be used to create a quiz like experience.
:question({type:"number",showanswer:true})[
:text[What is the :code[sqrt(4)]?]
:answer[sqrt(4)]
]renders as:
Clicking on the ✓ or leaving the input field tests the answer. If incorrect, the question will be surrounded with a red box based on the style secst-error.
Questions take most of the same type attribute values as input. They can also take the attribute showanswer:true if the correct answer is supposed to be shown after a user provides the wrong response.
Questions have two required content tags, :text[<question-body>] and :answer[<some-formula>], that must be provided in order. The <question-body> can be just about any content that would normally go in a paragraph. Formulas are documented later. The :answer[] tag can also take the attribute literal, e.g. :answer(literal)[2].
The :repl[] tag supports the insertion of a REPL into a document.
Below is what a REPL looks with all sections turned on plus line numbers.
:repl({head:true,css:true,body:true,javascript:true,linenumbers:true})[]
If you drop sections, they will disappear:
:repl({css:true,body:true,linenumbers:true})[]
Sections can be pre-populated:
:repl({css:true,body:true,linenumbers:true})[
:slot({name:"css"})[h1 { font-size:small}]
:slot({name:"body"})[<h1>Heading One</h1>]
]Sections can be made readonly:
:repl({css:true,body:true,linenumbers:true})[
:slot({name:"css",readonly:true})[h1 { font-size:small}]
:slot({name:"body"})[<h1>Heading One</h1>]
]Sections can be hidden:
:repl({css:true,body:true,linenumbers:true})[
:slot({name:"css",hidden:true})[h1 { font-size:small}]
:slot({name:"body"})[<h1>Heading One</h1>]
]See the following for more info:
:sheetMusic[] is a block level element, it belongs outside of paragraphs. Just provide ABC notation as the content.
:sheetMusic[
X: 1
T: Nokia Tune
M: 3/4
L: 1/8
K: Amaj
|: e'd' f2 g2 | c'b d2 e2 | ba c2 e2 | a6 |
]renders as
If you drop the : from the second position in the last line, the play button will disappear.
:sheetMusic[
X:1
T: Paper
K:D
| DDAA | BBA2|
]renders as
Several platforms provide @mention capability, but only for their users, e.g. GitHub and StackExchange. SECST allows you to mention people on any platform, although they do not get notified.
Mentions take the form @<platform>[<space separated profile-names>]. The target for mentions will always be another browser tab, unless a target attribute is provided.
The mark-up @github[anywhichway] renders as anywhichway@github.
Alternatively you can name the user and provide your own content, e.g. @github({user:"anywhichway"})[AnyWhichWay] renders as AnyWhichWay.
The supported platforms are:
For select tags, SECST can generate JSON-LD directly from SECST markup. Content, layout, and SEO optimization will always be in sync.
This is best covered by an example based on Google's documentation for a news article.
Note: Only those tags documented for NewsArticle are currently supported.
The SECST tag :NewsArticle[] can contain regular SECST as well as tags that parallel established JSON-LD schema like :headline[], :author[] and :Person[]. Where there are identical or trivially equivalent tags and attribute names, e.g. image and img, the native SECST tag is used.
:NewsArticle[
:headline[Pinky Conquers The World]
:img(./assets/images/The_Blue_Marble.jpg {height:50})[Earth]
:p[
Today Pinky finally conquered the world! Next up will be Mars. Watch out billionaires!
]
:author[The Brain]
:datePublished({format:""})[2022-12-20T08:00:00+08:00]
] Where JSON-LD names elements accept sub-types, e.g. :Person[], those can also be used:
:NewsArticle[
:headline[Pinky Conquers The World]
:img(./assets/images/The_Blue_Marble.jpg {height:50,align:"left"})[Earth]
:p[
Today Pinky finally conquered the world! Next up will be Mars. Watch out billionaires!
]
:author[
:Person[
:name[The Brain]
]
]
:datePublished({format:""})[2022-12-20T08:00:00+08:00]
]renders as
Today Pinky finally conquered the world! Next up will be Mars. Watch out billionaires!
SECST knows which tags to collect into arrays for JSON-LD and how to map similar SECST or HTML semantics to JSON-LD. It ignores tags that are not part of JSON-LD schema unless they have been mapped to JSON-LD schema internally.
Some JSON-LD attributes mapped to tags can have attributes, e.g. :heading({level:2})[].
Also note the use of Date formatting below based on Javascript date and time format functionality.
:NewsArticle[
:headline({level:2})[Pinky and The Brain Plan For Mars]
:img(./assets/images/mars.jpg {height:50,align:"left"})[Mars]
:p[
We, Pinky and The Brain, will soon announce plans to partner with one of the world's billionaires to take over Mars.
]
:Author[
:Person[
:name[Pinky]
],
:Person[
:name[The Brain]
]
]
:datePublished({format:"{dateStyle:'full',timeStyle:'long'}"})[2022-12-20T08:00:00+08:00]
]will generate this JSON-LD and insert it into the head of the document:
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@type": "NewsArticle",
"headline": "Pinky and The Brain Plan For Mars",
"image": [
"./assets/images/mars.jpg"
],
"datePublished": "2022-12-20T08:00:00+08:00",
"author": [{
"@type": "Person",
"name": "Pinky",
},{
"@type": "Person",
"name": "The Brain"
}]
}
</script>while rendering this content
We, Pinky and The Brain, will soon announce plans to partner with one of the world's billionaires to take over Mars.
The ordering of the tags inside :NewsArticle[] is only relevant to the display layout. This generates equivalent JSON-LD:
:NewsArticle[
:headline({level:4})[Pinky and The Brain Plan For Mars]
:author[
:Person[:name[Pinky]],
:Person[:name[The Brain]]
]
:datePublished({format:"{dateStyle:'full',timeStyle:'long'}"})[2022-12-20T08:00:00+08:00]
:img(./assets/images/mars.jpg {height:50,align:"left"})[Mars]
:p[
We, Pinky and The Brain, will soon announce plans to partner with one of the world's billionaires to take over Mars.
]
]but renders with a smaller heading and the attribution at the start:
We, Pinky and The Brain, will soon announce plans to partner with one of the world's billionaires to take over Mars.
The generated HTML also contains styles, so a style sheet can be used to modify the displayed content.
<div class="JSON-LD-NewsArticle">
<h1>Pinky and The Brain Plan For Mars</h1>
<span>
<span class="JSON-LD-Person"><name class="JSON-LD-Person-name">Pinky</name></span>,
<span class="JSON-LD-Person"><name class="JSON-LD-Person-name">The Brain</name></span>
</span>
<span data-format="{dateStyle:'full',timeStyle:'long'}">Monday, December 19, 2022 at 4:00:00 PM PST</span>
<p> <img height="50" align="left" title="Mars" alt="Mars" src="./assets/images/mars.jpg"><br> We, Pinky and The Brain, will soon announce plans to partner with one of the world's billionaires to take over Mars.<br> </p>
</div> Tables of contents are placed at the location of the tag toc[<title>]. The :toc[] tag should occur at the same level, i.e. outside of, :section[].
If you do not provide a title, "Table of Contents" is used. All content after the tag is analyzed for sections with a heading as the first child tag. These are used for the table.
Levels are determined based on section nesting. You can use the heading tag :h[] that has no number and the headings will be automatically sized.
Only the first occurrence of toc is used, others are ignored.
After transpilation, each heading in the document will have an ☷ icon preceding it. When this icon is clicked, the table of contents will hover over the heading until something in the document is clicked.
Each heading will have up and down arrows placed after it to facilitate jumping to the previous and next section.
The tag can take the boolean attribute toggle to collapse the table of contents into an HTML details element.
If a :theme[] tag with a URL exists in document, then the default CCS file secst.css is not applied.
The theme ./assets/themes/html.css provides only those styles dependent on the .secst class and a few structural defaults. Default browser HTML styling is used elsewhere. It is kind of ugly!
Only one theme tag is usually used, but it is possible to use multiple theme tags.
The content of a theme can be any tag, except most have a special interpretation. The tag content is the CSS to apply to the tag, e.g. :theme[:p[border:solid black 1px;]] will put a black border around all paragraphs.
The theme content tags permitted but not treated in a special way are :link[], :style[], and :class[] (which is unique to :theme[]).
The tag :class[] takes a name attribute (without a prefixed period) and the content is the CSS, e.g. :theme[:class({name:"italic-paragraph"})[font-style:italic;]] creates a class .italic-paragraph. These classes can be added to the parenthetical part of tags, e.g. :p(.italic-paragraph)[Paragraph in italics].
Theme tags can exist as Body Content or Block Content. They should be placed as close to the top of the document as possible.
Themes can be scoped using the scope attribute, the value of which is a CSS selector string, e.g. :theme({scope:#special-section})[:p[border:solid black 1px;]] will only put borders around paragraphs that are inside a :section(#special-section)[].
Themes associated with a URL can be statically loaded as HTML style tags by using the attribute static.
:theme[:class({name:"italic-paragraph"})[font-style:italic;]]
:theme({scope:"#special-section"})[:p[border:solid black 1px;]]
:section[
:p(.italic-paragraph)[Italic paragraph with no border.]
]
:section(#special-section)[
:p[Paragraph with border.]
]renders as
Italic paragraph with no border
Paragraph with border
The HTML of transpiled SECST can be displayed as plain text, e.g. :transpiled[:strong[test]] renders as <strong>test</strong>.
If you desire the transpiled code to be displayed as a block, end the content between the [] with a newline.
SECST supports the tag :value[] for adding reactive variables and formulas to documents.
:value[] is the content, i.e the text between the []. It should be either a literal value or a formula, e.g. :value[2 * 2] = .
hidden to hide them, e.g. :value(hidden)[2 * 2] = .
plaintext, e.g. :value(plaintext)[2 * 2] = 4.
editable, e.g. code[:value(editable)[2 * 2]] = .
[] is treated as a formula for evaluation, if your readers may be entering formula like text and you do not want it evaluated, use the attribute literal, e.g. :value(literal)[2 *2] = .
title attribute, e.g. :value({title:"Two times two"})[2 * 2], then the content is used as the mouseover title unless the content is provided by a URL, in which case the URL is the title, e.g. :value(./sample.json type="application/json") has a mouseover title of ./sample.json.
Formulas are actually just valid JavaScript expressions that also support standard and symbolic math. SECST attempts to process most values as math expressions first4 and if this fails, as JavaScript.
2 * 3 = ,
2^3 when used as part of a math expression5, is the same as 2**3, which is the same as pow(2,3), which is the same as Math.pow(2,3) = .
To use a value in another formula, the value must be named with an id, e.g.
:value(#v1)[2] + :value(#v2)[2] = :value(plaintext)[$(#v1) + $(#v2)]
+ = 4 (try changing one of the inputs)
The code $(#<some-id>) is used to substitute named values into formulas. See Advanced Formulas for use of more general CSS selectors and arrays.
The :value[] tag supports the type attribute, valid types include those supported by the HTML input element except button, file, hidden, image, search and submit.
The type can also be one of the mime-types text/plain, text/csv, application/json, e.g. :value(editable {type:"application/json"})[{name:"Joe"}] as shown below in Importing Data..
If you provide a type, e.g. :value(editable {type:"number"}), then the type will be enforced when the value is edited: .
If the value is computed, a red border will surround type check failures.
Strings can be combined using the + sign, e.g. :value["Hello" + " " + "world!"] = .
Strings can be manipulated using the standard JavaScript methods, e.g. :value["Hello world!".substring(0,5)] = .
Math formulas include anything that can use the normal operators, ,%,^,*,-,+,<,<=,==,>=,> and any functions available via MathJS.
Formulas can include physical constants, e.g :value[pi * 3^2] = .
This includes these items (most links goto MathJS documentation):
You can ...
:value[derivative('x^2 + x', 'x')]:value[simplify('x^2 + x + 3 + x^2')]:value[solve('Math.pow(r,2)',{r:3})}]Math.pow(r,2)
where r = :value(#v1d)[0]
is :value[solve('Math.pow(r,2)',{r:$(#v1d)})]
renders as
The derivative of :value(#formula literal editable)[x^2 + x]
for :value(#variable literal editable)[x]
is :value[derivative($(#formula),$(#variable))]
renders as
Unit math from MathJS allows for conversions using the word to, as well as units for numbers, e.g. 45 deg.
There are many supported units.
Here are a few examples:
:value[2 inch to cm] =
:value[90 km/h to mi/h] =
:value[90 km/h to m/h] = m/h is meters per hour
:value[(1 week + 1 day) to days] =
:value[cos(45 deg)] =
:value[8.314 m^3 Pa / mol / K] =
Remember, if you have a value that looks like a formula, but is not, you can use the attribute literal, e.g. :value(literal)[2 inch to cm] = .
And, you can use plaintext to display values without any styling, e.g. :value(plaintext)[90 km/h to mi/h] = 90 km/h to mi/h.
Values can included imported data using the src attribute and a mime type provided in the type attribute. The trailing [] are optional.
Values with the boolean attribute static is import data at transpile time. If importing fails, another attempt is made at runtime.
JSON is formatted for readability and JSON5 is supported to make it easier to handle remote data.
:value(https://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/all_hour.geojson static {type:"application/json"})
Imported data is readonly because it is essentially a computation. To make it editable, use the attribute editable.
:value(https://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/all_hour.geojson editable {type:"application/json"})
If an id is added, the data can be used in a computation.
:value(#earthquake-data https://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/all_hour.geojson {type:"application/json"})
:value[$(#earthquake-data).metadata]
You can use almost any CSS selector in place of #id in $(#<id>). The value returned will be the contents of the value property on the element or (if not available) the value attribute or the innerText.
You can use commas to select multiple identifier named elements, e.g. :value[$(#id1, #id2, #id3)] is an array of the values associated with the elements :value(#id1)[3], :value(#id2)[2], :value(#id3)[1] and will be [3,2,1].
A value reference $(css-selector) can end in [] to indicate multiple values should be returned as an array if the selector does not contain a comma. This allows tables to be treated somewhat like spreadsheets.
The SECST formula evaluator supports supports some utility functions.
You can use any JavaScript that is available in WebWorkers inside a macro. A special read-only document object is also made available. This object has title, baseURI, location, and navigator properties along with a property for each meta tag name and value.
Content can be styled using CSS classes loaded using the theme tag.
CSS classes are applied to content using .<class-name> inside the parenthetical configuration part of a tag, e.g. :p(.my-class1 .my-class2)[My content].
Also see the style tag.
Style can be applied at the element level using the style attribute of a tag if the tag supports the style attribute. Currently only the p[] tag supports the style attribute. Using the theme[] or :style[] tag with an id selector is suggested.
The style attribute can be in string or JSON format. If in JSON format, it should use camelCase versions of CSS property names. For example, :p({style:"width:200px;text-align:right;float:right"})[My right content] and :p({style:{width:"200px";textAlign:"right",float:"right"}}')[My right content] produce the same output.
:p({style:"text-align:left;float:right"})[
My right content
]
:p[
Left Content
More left content
]renders as
My right content
Left Content
More left content
Note: the p tag supports the attributes align and textAlign, so using style in the manner above is not actually necessary.
See the file ./preview.html in the root of the project directory.
Install secst as a dev dependency in your project: npm install --save-dev secst.
The command line node ./node_modules/secst/secst.js will convert any files in the root directory with the extension .sct into HTML files.
If a wild card Glob pattern is used, then all matching files with a .sct extension will be transpiled.
The command can be run as part of your build process to generate a static website.
There is not yet an Express route or Service Worker, but the code in ./node_modules/secst/secst.js could be used as a starting point.
An SPD (Single Page Document) capability that takes multiple .sct files in a directory tree that parallels the desired section structure and creates a single HTML file is under development.
A multi-page processor that will split a single .sct files into multiple files based on its section structure and patch links accordingly is also under consideration.
Depending on your use case, the reference transpiler can be set to do one of the following when errors are encountered.
The documentation below is generated directly from SECST source code.
N/A
plain text
href: transform href(value)
target: string
plain text
href: transform href(value)
target: string
plain text
href: transform href(value)
target: string
plain text
href: transform href(value)
target: string
plain text
N/A
article, author, datePublished, footer, forEach, forEntries, h, h1, h2, h3, h4, h5, h6, h7, h8, header, headline, img, link, mermaidChart, p, repl, section, sheetMusic, theme, title, toc
N/A
N/A
N/A
Person, name
N/A
name
N/A
plain text
format: transform format(value)
lang: string
plain text
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, p, pre, q, samp, strike, strong, sub, sup, u, var
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
align: one of [left, right]
style: transform style(value)
textAlign: one of [left, center, right, justify]
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, dl, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
filter: string
grayscale: transform grayscale(value)
greyscale: transform greyscale(value)
plain text
N/A
plain text
N/A
N/A
N/A
plain text
N/A
N/A
align: one of [top, middle, bottom, left, right]
alt: string
decoding: one of [sync, async, auto]
fetchpriority: one of [high, low, auto]
height: number
ismap: boolean
loading: one of [eager, lazy]
referrerpolicy: one of [no-referrer, no-referrer-when-downgrade, origin, origin-when-corss-origin, same-origin, strict-origin, strict-origin-when-cross-origin]
src: isURL(value)
static: boolean
title: string
url: transform url(value)
width: number
N/A
title: custom validation object
type: one of [checkbox, color, date, datetime-local, email, month, number, password, radio, range, tel, text, time, url, week]
value: true
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
N/A
caption, tbody, tfoot, thead, tr
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
N/A
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u
N/A
tr
rowspan: transform rowspan(value)
td, th
colspan: transform colspan(value)
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, dl, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
disabled: true
readonly: true
value: true
plain text
N/A
&, @facebook, @github, @linkedin, @twitter, NewsArticle, a, abbr, article, b, blockquote, bold, br, code, del, details, dl, em, emoji, escape, footer, footnote, forEach, forEntries, h, h1, h2, h3, h4, h5, h6, h7, h8, hashtag, header, hr, i, img, input, ins, italic, kbd, latex, link, listeners, macro, mark, mermaidChart, meta, ol, p, pre, q, question, repl, samp, section, sheetMusic, strike, strong, style, sub, sup, table, textarea, theme, title, toc, transpiled, u, ul, value, var
iterable: transform iterable(value)
&, @facebook, @github, @linkedin, @twitter, NewsArticle, a, abbr, answer, article, b, blockquote, bold, br, caption, class, code, dd, del, details, dl, dt, em, emoji, escape, footer, footnote, forEach, forEntries, h, h1, h2, h3, h4, h5, h6, h7, h8, hashtag, header, hr, i, img, input, ins, italic, kbd, latex, li, link, macro, mark, mermaidChart, meta, ol, p, pre, q, question, repl, samp, section, sheetMusic, slot, strike, strong, style, sub, summary, sup, table, tbody, td, text, textarea, tfoot, th, thead, theme, title, toc, tr, transpiled, u, ul, value, var
literal: transform literal(value)
type: one of [hidden]
value: true
plain text
name: string
plain text
N/A
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
N/A
N/A
default: transform default(value)
disabled: transform disabled(value)
editable: transform editable(value)
extract: transform extract(value)
fitcontent: transform fitcontent(value)
format: transform format(value)
globalThis: transform globalThis(value)
hidden: true
literal: transform literal(value)
plaintext: transform plaintext(value)
readonly: transform readonly(value)
src: transform src(value)
static: true
template: transform template(value)
title: custom validation object
type: custom validation object
url: transform url(value)
value: true
plain text
N/A
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
object: transform object(value)
&, @facebook, @github, @linkedin, @twitter, NewsArticle, a, abbr, answer, article, b, blockquote, bold, br, caption, class, code, dd, del, details, dl, dt, em, emoji, escape, footer, footnote, forEach, forEntries, h, h1, h2, h3, h4, h5, h6, h7, h8, hashtag, header, hr, i, img, input, ins, italic, kbd, latex, li, link, macro, mark, mermaidChart, meta, ol, p, pre, q, question, repl, samp, section, sheetMusic, slot, strike, strong, style, sub, summary, sup, table, tbody, td, text, textarea, tfoot, th, thead, theme, title, toc, tr, transpiled, u, ul, value, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, p, pre, q, samp, strike, strong, sub, sup, u, var
type: transform type(value)
value: transform value(value)
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, dl, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
href: transform href(value)
rel: one of [stylesheet]
url: transform url(value)
N/A
N/A
N/A
N/A
plain text
body: transform body(value)
css: transform css(value)
head: transform head(value)
javascript: transform javascript(value)
linenumbers: transform linenumbers(value)
replstyle: string
slot
hidden: boolean
linenumbers: transform linenumbers(value)
name: string
readonly: boolean
plain text
level: number
NewsArticle, article, footer, forEach, forEntries, h, h1, h2, h3, h4, h5, h6, h7, h8, header, img, link, macro, mermaidChart, p, repl, section, sheetMusic, theme, title, toc
N/A
plain text
N/A
class, link, style
selector: true
plain text
N/A
plain text
toggle: transform toggle(value)
plain text
N/A
a, abbr, b, bold, code, del, em, i, ins, italic, kbd, mark, meta, pre, q, samp, strike, strong, sub, sup, u, var
N/A
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, dl, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
N/A
plain text
colspan: transform colspan(value)
&, @facebook, @github, @linkedin, @twitter, a, abbr, b, blockquote, bold, br, code, del, details, dl, em, emoji, escape, footnote, hashtag, hr, i, img, input, ins, italic, kbd, latex, mark, meta, ol, pre, q, question, samp, strike, strong, sub, sup, table, textarea, transpiled, u, ul, value, var
N/A
tr
N/A
N/A
level: true
plain text
Note, not all design objectives are fully implemented.
All SECST markup takes one of two forms:
:<tag>[<content>...] where code:[<content>] is SECST markup ans text.:<tag>(#<id> .<class>... <unaryattribute>... <attributes>...)[<content>...] where <id>, <class> space separated list, <unaryattribute> space separated list, <attributes> JSON5 object are all optional, but must be provided in that order if present. In some cases, e.g. for :hr, :br, and :value the trailing [] are optional.
The reference implementation uses a PEG parser and this grammar:
The Mozilla Development Network, a.k.a. MDN, documents several HTML tag content categories:
<link>, <style.
<body>.
<section>, <article>.
<h1>, <h2>.
<p>.
<img>, audio, that cab be used in the same places as flow content.
<a>, <input>.
<progress>.
SECST uses these content types to determine what types of tags can exists inside other tags with a few enhancements or exceptions:
<strong>.<table>.
<body>, including <style> and <link> which are usually in a <head> section, but may exist in a <body>.
:article[] and :section[] are both Sectioning Content and Flow Content, so they can be nested.
:progress[] can exist outside a <form>.
The relationship between tags is represented as a cyclical JSON structure. The bulk of of the structure is static, but some of it is dynamic and memoized to address its cyclical nature.
Tags are defined with the following parts:
{
attributesAllowed: { // optional
<attributeName>: true || "<primitive-type>" : function, // the function can validate or transform the attribute
...
},
contentAllowed: boolean || function || {<tagName>: Tag,...}, // optional
transform(node) { // optional
// a function body that can validate and manipulate the parsed tag
// e.g. replace SECST attribute names with valid HTML attribute names or classes, map :code[{language:"javascript"}] to the CSS class :code[.language-javascript] for use by HighlightJS.
return node;
},
beforeMount(node) { // optional
// a function body that can map SECST tag names to HTML tag names that are not valid in SECST
// e.g. :code[transpiled] to :code[span] or :code[div]
// node.tag = "div"
return node;
},
toJSONLD(node) { // optional
const jsonld = ...
// a function body to override built-in JSONLD generation
return jsonld;
}
toInnerHTML(node) { // optional
const html = ...
// a function body to generate the :code[innerHTML] for a tag in place of SECST iterating over parsed content for the tag.
return html;
}
toText(node) { // optional
const text = ...
// a function body to generate text inside a :code[<span>] as a :strong[replacement] for the tag in place of SECST iterating over parsed content for the tag.
}
render(el,node) { // optional
// a function body to manipulate the element based on the parsed node in some custom manner
}
mounted(el,node) { // optional
// called when the element is added to a DOM tree that is not yet connected to a document
// a function body that can make adjustments to the generated element before it is rendered, e.g. call HighlightJS with the :code[innerHTML] and replace it.
},
connected(el,node) { // optional
// a function body that is called after all nodes have been added to a document in the order they were added to the document, e.g. re-compute sizes that can't be done easily with CSS
}
listeners: { // optional, responds to events on all occurrences of the tag
<event-name>(event) {
// function body, event.target will be the occurrence of the HTML element generated by the Tag
},
...
},
... // any other functions are added to the element generated by the tag.
} Functions taking a node as an argument are free to modify the node. Where necessary, copying to support immutability is done prior to calling the function. Functions taking a node as the first argument MUST return a node.
Functions taking a node as a second argument should not modify the node or return a value. If they do, it will be ignored.
Parsed instances of tags, a.k.a. nodes, have a structure similar to that found in a virtual DOM:
{
tag:<tag-name>,
attributes:{<attribute-name>:<attribute-value>,...}
classNames: Array // will be changed to use a Set,
content: [string || <node>, ...]
}Note, not all security features are fully implemented and tested in this release
head, body.:style[] tag.
2023-01-01 v0.0.3a - Started publishing release history. Documented use of command line when secst is installed as a dev dependency.
Copyright 2022, AnyWhichWay LLC
Documentation License: Creative Commons Attribution v4.0
Code License: AGPL-3.0
Visit SECST on GitHub to participate in SECST's further development.
1, 2 Markdown does not autonumber footnotes, which makes them hard to maintain. Although, it does create back-references.
3 Footnote management does not get much easier!
4 A parser is used to match potential math expressions.
5 If the expression cannot be parsed as math, then 2^3 will be treated as XOR as it is in regular JavaScript.