Elixir bindings for the Vize Vue.js toolchain — compile, lint, and analyze Vue SFCs via Rust NIFs https://vizejs.dev
  • Rust 59%
  • Elixir 41%
Find a file
2026-06-11 13:52:37 +03:00
.cargo Add Rustler precompiled NIF support 2026-03-12 09:18:57 +03:00
.github/workflows Use shared Rustler release workflow 2026-06-07 01:30:12 +03:00
lib Add CSS AST bindings 2026-05-19 16:37:38 +03:00
native/vize_ex_nif Release v0.11.1 2026-05-20 15:38:28 +03:00
test Release v0.11.1 2026-05-20 15:38:28 +03:00
.credo.exs Initial Vize bindings: compile, SSR, Vapor IR, lint 2026-03-11 18:30:24 +03:00
.formatter.exs Initial Vize bindings: compile, SSR, Vapor IR, lint 2026-03-11 18:30:24 +03:00
.gitignore chore: ignore local audit output 2026-06-11 13:52:37 +03:00
Cargo.lock Release v0.11.1 2026-05-20 15:38:28 +03:00
Cargo.toml Initial Vize bindings: compile, SSR, Vapor IR, lint 2026-03-11 18:30:24 +03:00
CHANGELOG.md Release v0.11.1 2026-05-20 15:38:28 +03:00
checksum-Elixir.Vize.Native.exs Add v0.11.1 precompiled NIF checksums 2026-05-20 16:29:56 +03:00
LICENSE Initial Vize bindings: compile, SSR, Vapor IR, lint 2026-03-11 18:30:24 +03:00
mix.exs Use shared Rustler CI workflow 2026-06-07 00:39:34 +03:00
mix.lock Add CSS AST bindings 2026-05-19 16:37:38 +03:00
README.md README: ecosystem footer linking org and Building Blocks standard 2026-06-11 12:21:37 +03:00

Vize

Elixir bindings for the Vize Vue.js toolchain via Rust NIFs.

Compile, lint, and analyze Vue Single File Components at native speed — including Vapor mode IR for BEAM-native SSR.

Features

  • Compile Vue SFCs to JavaScript + CSS (DOM, Vapor, SSR modes)
  • Template compilation — standalone template → render function
  • Vapor IR — get the intermediate representation as Elixir maps for BEAM-native rendering
  • SSR — server-side rendering compilation with _push() codegen
  • Lint Vue SFCs with built-in rules
  • Content hashes — template, script, and style hashes for HMR change detection
  • CSS compilation — standalone LightningCSS pipeline with autoprefixing, minification, and Vue scoped styles
  • CSS AST tooling — parse, traverse, transform, and print LightningCSS-backed ASTs

Installation

def deps do
  [
    {:vize, "~> 0.11.0"}
  ]
end

Requires a Rust toolchain (rustup recommended). The NIF compiles automatically on mix compile.

Usage

Compile SFC

{:ok, result} = Vize.compile_sfc("""
<template>
  <button @click="count++">{{ count }}</button>
</template>

<script setup>
import { ref } from 'vue'
const count = ref(0)
</script>

<style scoped>
button { color: blue; }
</style>
""")

result.code     # Generated JavaScript
result.css      # Compiled CSS
result.errors   # []

Pass a filename for stable scoped CSS data-v-xxxx attributes and content hashes:

{:ok, result} = Vize.compile_sfc(source, filename: "App.vue")

result.template_hash  # "de5ddf78a0f8d31a"
result.style_hash     # "3efafd39ec9747f9"
result.script_hash    # "1a8dae0fef50c189"

Vapor Mode

{:ok, result} = Vize.compile_vapor("<div>{{ msg }}</div>")
result.code       # Vapor JS (no virtual DOM)
result.templates  # Static HTML templates

Vapor IR (for BEAM-native SSR)

{:ok, ir} = Vize.vapor_ir("<div :class=\"cls\">{{ msg }}</div>")

ir.templates             # ["<div> </div>"]
ir.element_template_map  # [{0, 0}]  — element ID → template index
ir.block                 # %{operations: [...], effects: [...], returns: [...]}

The IR exposes every Vue construct as Elixir maps with :kind atoms:

Kind Vue Feature
:set_text {{ expr }}
:set_prop :attr="expr"
:set_html v-html
:set_dynamic_props v-bind="obj"
:set_event @event="handler"
:if_node v-if / v-else-if / v-else
:for_node v-for
:create_component <Component />
:directive v-show, v-model, custom

Static expressions are tagged as {:static, "value"} tuples, dynamic expressions are plain strings.

Vapor Split (for Phoenix LiveView)

{:ok, split} = Vize.vapor_split("<div :class=\"cls\"><p>{{ msg }}</p></div>")

split.statics  # ["<div class=\"", "\"><p>", "</p></div>"]
split.slots    # [%{kind: :set_prop, values: ["cls"]}, %{kind: :set_text, values: ["msg"]}]

Produces a statics/slots split ready for %Phoenix.LiveView.Rendered{}. All HTML manipulation (tag tree parsing, marker injection, splitting) happens in the NIF. Sub-blocks for v-if / v-for are recursively split.

Used by PhoenixVapor to render Vue templates as native LiveView output.

SSR Compilation

{:ok, result} = Vize.compile_ssr("<div>{{ msg }}</div>")
result.code      # JS with _push() calls
result.preamble  # Import statements

Template Compilation

{:ok, result} = Vize.compile_template("<div v-if=\"show\">{{ msg }}</div>")
result.code     # Render function
result.helpers  # ["createElementVNode", "toDisplayString", ...]

Lint

{:ok, diagnostics} = Vize.lint("<template><img></template>", "App.vue")

Parse SFC

{:ok, descriptor} = Vize.parse_sfc(source)
descriptor.template      # %{content: "...", lang: nil, ...}
descriptor.script_setup  # %{content: "...", setup: true, ...}
descriptor.styles        # [%{content: "...", scoped: true, ...}]

CSS Compilation

Standalone CSS compilation via LightningCSS — parse, autoprefix, and minify CSS independently of SFC compilation:

{:ok, result} = Vize.CSS.compile(".foo { color: red; user-select: none }", minify: true)
result.code
# ".foo{color:red;-webkit-user-select:none;user-select:none}"

With Vue scoped styles:

{:ok, result} = Vize.CSS.compile(".foo { color: red }", scoped: true, scope_id: "data-v-abc123")
result.code
# ".foo[data-v-abc123] { color: red }"

Browser targeting:

{:ok, result} = Vize.CSS.compile(css, targets: %{chrome: 80, firefox: 78, safari: 14})

Parser-backed CSS tooling:

{:ok, parsed} = Vize.CSS.parse_ast(".foo { background: url('./logo.svg') }")

ast =
  Vize.CSS.postwalk(parsed.ast, fn
    %{"url" => "./logo.svg"} = node -> %{node | "url" => "/assets/logo.svg"}
    node -> node
  end)

{:ok, result} = Vize.CSS.print_ast(ast)

Part of Elixir Volt

vize compiles and analyzes Vue single-file components from Elixir, including Vapor-mode IR for BEAM-native rendering.

It is part of a frontend stack that runs inside the BEAM — builds, JS runtimes, icons, and Vue-to-LiveView compilation as supervised parts of the application instead of external toolchain processes. See the Elixir Volt organization for the rest, and Building Blocks for the Future Web for the thesis, architecture, and roadmap that tie them together.

License

MIT