Concrete

approved

by apokaliptics

This plugin has not been manually reviewed by Obsidian staff. Reactive variable system for Markdown notes.

27 downloadsUpdated 24d ago0BSD

Concrete Extension

A lightweight text styling, color highlighting, and structural outliner plugin for Obsidian.

Define custom text wrappers, inline color styles, CSS variables, and dash-based outline levels directly inside your notes using a simple, readable syntax — no settings menus required.

You can also enable or disable individual features (bullet points, color variables, text variables) from the plugin settings.

Features

  • Custom text wrappers — Turn (text) red, "text" blue, or ^text^ into a bold header by defining a simple rule.
  • Letter wrappers — Use letter pairs like hh text hh for highlighting. The letters must be grouped together and spaced from the content to avoid false matches with normal words.
  • Delimiter hiding — Wrapper symbols are hidden in the rendered view. You only see the styled content; click into the line to reveal the raw syntax.
  • Nested Wrappers — Apply multiple styles to the same text by nesting them seamlessly (e.g., _&text&_).
  • Combined Styles — Reuse the same wrapper symbol across different sections to stack multiple effects simultaneously (e.g., apply a color and a massive header size with a single wrapper!).
  • Interactive color palette — Every hex color you define gets a clickable swatch in the editor. Click it to open the system color picker and update the color inline.
  • CSS variable injection — Keys like header_size = 24 or text_title_size = 32 become CSS variables you can use in custom snippets.
  • Named text variables — Define any text style name (e.g., title, body, caption) and link it to a wrapper.
  • Dash-level outliner — Lines starting with -, --, --- etc. become indented outline levels with aesthetic bullets, guide lines, and fading opacity.
  • Ghost dash effect — Dashes are hidden on non-active lines and replaced with styled bullets. Click into a line to reveal the raw dashes for editing.
  • Collapsible config block — The :::vars block features an inline toggle. Fold it away when you're done configuring, and it will cleanly display a summary like ▶ [VARS: 4 colors, 2 styles].
  • Live Preview & Reading View — All features work in both editor modes.

How to use

1. Create a :::vars block

Place this block anywhere in your note. It defines all your styling rules.

:::vars
##colors
() = #ef4444
"" = #3b82f6
hh = #10b981
&& = #8b5cf6

##text
header_size = 32
paragraph_size = 14

^^ = header
.. = paragraph
__ = underline
&& = bold

# Or use the new named syntax:
##text
text_title_size = 32
text_body_size = 14

title = ^^
body = ..
__ = underline
:::

2. Sections

Rules are organized under section headers prefixed with ##:

SectionPurpose
##colors (or ##colour, ##colours)Rules here treat values as colors. Wrapped text will be colored.
##textRules here treat values as CSS class names. Wrapped text gets the corresponding .rv-{value} class applied. You can also use the new named syntax: text_name_size = number and name = ^^.

3. Color wrappers (under ##colors)

Define a wrapper symbol and assign it a color value.

##colors
() = #ef4444
"" = #3b82f6

Then use them in your note:

(This text will be red!)
"This text will be blue!"

Result: The wrapper symbols (, ), ", " are hidden. You only see the styled text.

4. Text wrappers (under ##text)

Define wrappers that apply CSS classes instead of colors.

##text
^^ = header
__ = underline

Usage:

^This becomes a header^
_This becomes underlined_

Named text variables (new) You can now name text styles anything you want using the text_{name}_size = {number} convention:

##text
text_title_size = 32
text_body_size = 14

title = ^^
body = ..

Now ^Title text^ uses --text_title_size: 32px and _Body text_ uses --text_body_size: 14px.

Built-in Styles The plugin ships with several out-of-the-box styles you can use immediately under ##text:

ClassEffect
headerBold text, sized by header_size or text_header_size (default 1.5em)
paragraphNormal text, sized by paragraph_size or text_paragraph_size (default 1em)
boldBold text
italicItalic text
underlineUnderlined text
strikethroughStrikethrough text
highlightApplies a background highlight color

(You can define any other value and style it yourself with a CSS snippet targeting .rv-{value})

5. Advanced: Nested & Combined Wrappers

Nesting Wrappers: You can combine multiple text wrappers by nesting them inside each other!

_&This text is bold and underlined!&_

Combining Styles: If you want a single wrapper to do multiple things, just define it in both sections!

:::vars
##colors
&& = #ff0000

##text
&& = header
header_size = 70
:::

Now, writing &Huge red header!& will automatically apply the color #ff0000 and the massive header size simultaneously.

6. Letter wrappers

You can also use letter pairs as wrappers. They must be spaced from the content:

##colors
hh = #10b981

Usage:

hh This text will be green hh

Why spaces? To prevent false matches with normal words that happen to start and end with the same letter. hh text hh matches, but hello does not.

7. CSS variables

Alphanumeric keys with underscores or hyphens become CSS variables:

header_size = 24
paragraph_size = 14

These become --header_size: 24px and --paragraph_size: 14px on the document container. The built-in .rv-header and .rv-paragraph classes reference these variables.

You can also use the new named convention:

text_title_size = 32
text_body_size = 14

These become --text_title_size: 32px and --text_body_size: 14px. When using named text variables (title = ^^), the plugin automatically picks up the matching size variable.

8. Dash-level outliner

Start any line with one or more dashes followed by a space to create outline levels:

- Level 1 item
-- Level 2 sub-item
--- Level 3 deep item
---- Level 4

What happens:

  • Each dash level gets increasing indentation and a guide line on the left.
  • The raw dashes are replaced with aesthetic bullet characters (, , , etc.).
  • Deeper levels automatically fade in opacity for visual hierarchy.
  • Ghost dash effect: Click into a line to reveal the raw dashes for editing. Move away and the bullets return.

9. Wrapper syntax rules

KeyTypeStartEndExample
()Asymmetric symbols()(colored text)
""Symmetric symbols"""colored text"
^^Symmetric symbols^^^header text^
hhLetter wrapperhh hhhh highlighted hh

Asymmetric (2 different chars): first char = start, second char = end. Symmetric (2 same chars): that char = both start and end. Letters (2+ letters): the full key is used, must be surrounded by spaces.

10. Interactive color picker

In the editor, every hex color value in your :::vars block gets a small color swatch next to it. Click the swatch to open your system's native color picker — changing the color automatically updates the hex code in your note.

Settings

The plugin now includes a Feature checklist in its settings tab:

  • Use bullet points — Turn the dash-level outliner on or off.
  • Use colour variables — Enable/disable color wrappers, color CSS variables, and the editor color picker.
  • Use text variables — Enable/disable text wrappers and text size variables.

These toggles let you disable features you don't use, keeping the editor lightweight.

Installation

Manual

  1. Download main.js, manifest.json, and styles.css from the latest release.
  2. Place them in VaultFolder/.obsidian/plugins/concrete-extension/.
  3. Reload Obsidian.
  4. Enable Concrete Extension in Settings → Community plugins.

Development

  1. Clone this repo into your .obsidian/plugins/ folder.
  2. npm install
  3. npm run dev — compiles and watches for changes.
  4. npm run lint — checks for style errors.

For plugin developers

Search results and similarity scores are powered by semantic analysis of your plugin's README. If your plugin isn't appearing for searches you'd expect, try updating your README to clearly describe your plugin's purpose, features, and use cases.