Compare commits

..

6 Commits

Author SHA1 Message Date
Jonas Hahn
57e62d11aa Minors 2025-11-16 23:02:17 +01:00
Jonas Hahn
e0a82ef957 Add deletion for drawings 2025-11-11 15:36:25 +01:00
Jonas Hahn
459ced0c2b Change part 2025-11-03 12:58:14 +01:00
Jonas Hahn
eda6de26bb Removed all the default inserts 2025-11-02 00:49:37 +01:00
Jonas Hahn
dc7fa65a80 Merge branch 'main' into dev 2025-10-30 09:41:08 +01:00
Jonas Hahn
b49e0262a1 Small physics helpers 2025-10-30 09:40:21 +01:00
11 changed files with 184 additions and 120 deletions

1
.gitignore vendored
View File

@@ -3,5 +3,4 @@
.obsidian
assets
result

View File

@@ -156,7 +156,6 @@ require('typstar').setup({ -- depending on your neovim plugin system
</details>
### Snippets
0. The snippets are designed to work with Typst `0.14`. For older versions check out the legacy `typst-0.13` branch.
1. Install [LuaSnip](https://github.com/L3MON4D3/LuaSnip/), set `enable_autosnippets = true` and set a visual mode selection key (e.g. `store_selection_keys = '<Tab>'`) in the configuration
2. Install [jsregexp](https://github.com/kmarius/jsregexp) as described [here](https://github.com/L3MON4D3/LuaSnip/blob/master/DOC.md#transformations) (You will see a warning on startup if jsregexp isn't installed properly)
3. Install [nvim-treesitter](https://github.com/nvim-treesitter/nvim-treesitter) and run `:TSInstall typst`
@@ -176,6 +175,7 @@ require('typstar').setup({ -- depending on your neovim plugin system
4. See comment 4 above at Excalidraw
### Anki
0. Typst version `0.12.0` or higher is required
1. Install [Anki](https://apps.ankiweb.net/#download)
2. Install [Anki-Connect](https://ankiweb.net/shared/info/2055492159) and make sure `http://localhost` is added to `webCorsOriginList` in the Add-on config (should be added by default)
3. Install the typstar python package (I recommend using [pipx](https://github.com/pypa/pipx) via `pipx install git+https://github.com/arne314/typstar`, you will need to have python build tools and clang installed) \[Note: this may take a while\]

18
flake.lock generated
View File

@@ -5,11 +5,11 @@
"nixpkgs-lib": "nixpkgs-lib"
},
"locked": {
"lastModified": 1762040540,
"narHash": "sha256-z5PlZ47j50VNF3R+IMS9LmzI5fYRGY/Z5O5tol1c9I4=",
"lastModified": 1749398372,
"narHash": "sha256-tYBdgS56eXYaWVW3fsnPQ/nFlgWi/Z2Ymhyu21zVM98=",
"owner": "hercules-ci",
"repo": "flake-parts",
"rev": "0010412d62a25d959151790968765a70c436598b",
"rev": "9305fe4e5c2a6fcf5ba6a3ff155720fbe4076569",
"type": "github"
},
"original": {
@@ -20,11 +20,11 @@
},
"nixpkgs": {
"locked": {
"lastModified": 1762156382,
"narHash": "sha256-Yg7Ag7ov5+36jEFC1DaZh/12SEXo6OO3/8rqADRxiqs=",
"lastModified": 1750386251,
"narHash": "sha256-1ovgdmuDYVo5OUC5NzdF+V4zx2uT8RtsgZahxidBTyw=",
"owner": "NixOS",
"repo": "nixpkgs",
"rev": "7241bcbb4f099a66aafca120d37c65e8dda32717",
"rev": "076e8c6678d8c54204abcb4b1b14c366835a58bb",
"type": "github"
},
"original": {
@@ -36,11 +36,11 @@
},
"nixpkgs-lib": {
"locked": {
"lastModified": 1761765539,
"narHash": "sha256-b0yj6kfvO8ApcSE+QmA6mUfu8IYG6/uU28OFn4PaC8M=",
"lastModified": 1748740939,
"narHash": "sha256-rQaysilft1aVMwF14xIdGS3sj1yHlI6oKQNBRTF40cc=",
"owner": "nix-community",
"repo": "nixpkgs.lib",
"rev": "719359f4562934ae99f5443f20aa06c2ffff91fc",
"rev": "656a64127e9d791a334452c6b6606d17539476e2",
"type": "github"
},
"original": {

133
flake.nix
View File

@@ -6,79 +6,78 @@
flake-parts.url = "github:hercules-ci/flake-parts";
};
outputs =
inputs@{
self,
nixpkgs,
flake-parts,
...
}:
flake-parts.lib.mkFlake { inherit inputs; } {
outputs = inputs @ {
self,
nixpkgs,
flake-parts,
...
}:
flake-parts.lib.mkFlake {inherit inputs;} {
systems = [
"x86_64-linux"
"x86_64-darwin"
"aarch64-darwin"
];
perSystem =
{ system, ... }:
let
pkgs = import nixpkgs { inherit system; };
typstar = pkgs.vimUtils.buildVimPlugin {
name = "typstar";
src = self;
buildInputs = [
pkgs.vimPlugins.luasnip
pkgs.vimPlugins.nvim-treesitter-parsers.typst
];
};
in
{
packages = {
default = typstar;
nvim =
let
config = pkgs.neovimUtils.makeNeovimConfig {
customRC = ''
lua << EOF
print("Welcome to Typstar! This is just a demo.")
vim.g.mapleader = " "
require('nvim-treesitter.configs').setup {
highlight = { enable = true },
}
local ls = require('luasnip')
ls.config.set_config({
enable_autosnippets = true,
store_selection_keys = "<Tab>",
})
local typstar = require('typstar')
typstar.setup({})
vim.keymap.set({'n', 'i'}, '<M-t>', '<Cmd>TypstarToggleSnippets<CR>', { silent = true, noremap = true })
vim.keymap.set({'s', 'i'}, '<M-j>', '<Cmd>TypstarSmartJump<CR>', { silent = true, noremap = true })
vim.keymap.set({'s', 'i'}, '<M-k>', '<Cmd>TypstarSmartJumpBack<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>e', '<Cmd>TypstarInsertExcalidraw<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>r', '<Cmd>TypstarInsertRnote<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>o', '<Cmd>TypstarOpenDrawing<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>a', '<Cmd>TypstarAnkiScan<CR>', { silent = true, noremap = true })
EOF
'';
plugins = [
typstar
pkgs.vimPlugins.luasnip
pkgs.vimPlugins.nvim-treesitter
pkgs.vimPlugins.nvim-treesitter-parsers.typst
];
};
in
pkgs.wrapNeovimUnstable pkgs.neovim-unwrapped config;
};
perSystem = {system, ...}: let
pkgs = import nixpkgs {inherit system;};
typstar = pkgs.vimUtils.buildVimPlugin {
name = "typstar";
src = self;
buildInputs = [
pkgs.vimPlugins.luasnip
pkgs.vimPlugins.nvim-treesitter-parsers.typst
];
# TODO: make this check pass instead of skipping
neovimRequireCheckHook = ''
echo "Skipping neovimRequireCheckHook"
'';
};
in {
packages = {
default = typstar;
nvim = let
config = pkgs.neovimUtils.makeNeovimConfig {
customRC = ''
lua << EOF
print("Welcome to Typstar! This is just a demo.")
vim.g.mapleader = " "
require('nvim-treesitter.configs').setup {
highlight = { enable = true },
}
local ls = require('luasnip')
ls.config.set_config({
enable_autosnippets = true,
store_selection_keys = "<Tab>",
})
local typstar = require('typstar')
typstar.setup({})
vim.keymap.set({'n', 'i'}, '<M-t>', '<Cmd>TypstarToggleSnippets<CR>', { silent = true, noremap = true })
vim.keymap.set({'s', 'i'}, '<M-j>', '<Cmd>TypstarSmartJump<CR>', { silent = true, noremap = true })
vim.keymap.set({'s', 'i'}, '<M-k>', '<Cmd>TypstarSmartJumpBack<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>e', '<Cmd>TypstarInsertExcalidraw<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>r', '<Cmd>TypstarInsertRnote<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>o', '<Cmd>TypstarOpenDrawing<CR>', { silent = true, noremap = true })
vim.keymap.set('n', '<leader>a', '<Cmd>TypstarAnkiScan<CR>', { silent = true, noremap = true })
EOF
'';
plugins = [
typstar
pkgs.vimPlugins.luasnip
pkgs.vimPlugins.nvim-treesitter
pkgs.vimPlugins.nvim-treesitter-parsers.typst
];
};
in
pkgs.wrapNeovimUnstable pkgs.neovim-unwrapped config;
};
};
};
}

View File

@@ -104,10 +104,64 @@ local open_drawing = function(prov)
end
end
local function delete_drawing(provider)
local cfg = provider[1]
local line_num = vim.api.nvim_win_get_cursor(0)[1] - 1
local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false)
-- Find the start and end of the #figure() block containing the filename
local start_line, end_line
for i = line_num, 0, -1 do
if lines[i + 1]:match("^#figure%(") then
start_line = i
break
end
end
for i = line_num, #lines - 1 do
if lines[i + 1]:match("%)$") then
end_line = i
break
end
end
if not start_line or not end_line then
print("No #figure() block found on current line")
return
end
-- Extract filename from the block
local block_text = table.concat(vim.api.nvim_buf_get_lines(0, start_line, end_line + 1, false), "\n")
local filename = block_text:match('image%("%s*(.-)%s*"%)')
if not filename then
print("No image filename found in #figure() block")
return
end
local path = vim.fn.expand('%:p:h') .. '/' .. filename
local path_export = vim.fn.expand('%:p:h') .. '/' .. filename:match("^(.*)%.")
-- Delete files if they exist
if vim.fn.filereadable(path) == 1 then os.remove(path) end
if vim.fn.filereadable(path_export) == 1 then os.remove(path_export) end
-- Delete the lines of the #figure() block
vim.api.nvim_buf_set_lines(0, start_line, end_line + 1, false, {})
print("Deleted drawing: " .. filename)
end
function M.delete_obsidian_excalidraw() delete_drawing(excalidraw) end
function M.delete_rnote() delete_drawing(rnote) end
function M.insert_obsidian_excalidraw() insert_drawing(excalidraw) end
function M.insert_rnote() insert_drawing(rnote) end
function M.open_obsidian_excalidraw() open_drawing({ excalidraw }) end
function M.open_rnote() open_drawing({ rnote }) end
function M.open_drawing() open_drawing(providers) end
return M

View File

@@ -14,7 +14,9 @@ M.setup = function(args)
vim.api.nvim_create_user_command('TypstarSmartJumpBack', function() M.smart_jump(-1) end, {})
vim.api.nvim_create_user_command('TypstarInsertExcalidraw', drawings.insert_obsidian_excalidraw, {})
vim.api.nvim_create_user_command('TypstarDeleteExcalidraw', drawings.delete_obsidian_excalidraw, {})
vim.api.nvim_create_user_command('TypstarInsertRnote', drawings.insert_rnote, {})
vim.api.nvim_create_user_command('TypstarDeleteRnote', drawings.delete_rnote, {})
vim.api.nvim_create_user_command('TypstarOpenExcalidraw', drawings.open_obsidian_excalidraw, {})
vim.api.nvim_create_user_command('TypstarOpenRnote', drawings.open_rnote, {})
vim.api.nvim_create_user_command('TypstarOpenDrawing', drawings.open_drawing, {})

View File

@@ -19,14 +19,16 @@ local ctheorems = {
{ 'def', 'definition' },
{ 'exa', 'example' },
{ 'rem', 'remark' },
{ 'nte', 'note' },
{ 'eri', 'experiment' },
}
local wrappings = {
{ 'll', '$', '$', '1+1' },
{ 'BLD', '*', '*', 'abc' },
{ 'ITL', '_', '_', 'abc' },
{ 'HIG', '#highlight[', ']', 'abc' },
{ 'UND', '#underline[', ']', 'abc' },
{ 'll', '$', '$', '' },
{ 'BLD', '*', '*', '' },
{ 'ITL', '_', '_', '' },
{ 'HIG', '#highlight[', ']', '' },
{ 'UND', '#underline[', ']', '' },
}
local document_snippets = {}
@@ -49,6 +51,7 @@ return {
helper.list_snip('dm', '\n$\n\t<>\n$ <>', { helper.visual(1), i(2) }, markup, 1100, { prepend = '\t' }),
start('fla', '#flashcard(0)[<>][\n<>\n<>]', { i(1, 'flashcard'), indent_visual(2), cap(1) }, markup),
start('flA', '#flashcard(0, "<>")[\n<>\n<>]', { i(1, 'flashcard'), indent_visual(2), cap(1) }, markup),
start('TAB', '#table(columns: <>,\n[<>], [<>], <>\n)', { i(1), i(2), i(3), i(4) }, markup),
snip('IMP', '$==>>$ ', {}, markup),
snip('IFF', '$<<==>>$ ', {}, markup),
unpack(document_snippets),

View File

@@ -9,6 +9,7 @@ local cap = helper.cap
return {
snip('fa', 'forall ', {}, math),
snip('ex', 'exists ', {}, math),
snip('ni', 'in.not ', {}, math),
snip('Sq', 'square', {}, math),
-- logical chunks
@@ -18,8 +19,6 @@ return {
snip('een', 'exists epsilon>>0 ', {}, math),
-- boolean logic
snip('and', 'and ', {}, math),
snip('or', 'or ', {}, math),
snip('not', 'not ', {}, math),
snip('ip', '==>> ', {}, math),
snip('ib', '<<== ', {}, math),
@@ -32,31 +31,46 @@ return {
snip('gt', '>> ', {}, math),
snip('le', '<<= ', {}, math),
snip('ne', '!= ', {}, math),
snip('eel', '&= ', {}, math),
snip('ge', '>>= ', {}, math),
-- operators
snip('mak', 'plus.minus ', {}, math),
snip('oak', 'plus.o ', {}, math),
snip('bak', 'plus.square ', {}, math),
snip('osk', 'minus.o ', {}, math),
snip('bsk', 'minus.square ', {}, math),
snip('xx', 'times ', {}, math),
snip('oxx', 'times.o ', {}, math),
snip('bxx', 'times.square ', {}, math),
snip('ff', '(<>) / (<>) <>', { i(1, 'a'), i(2, 'b'), i(3) }, math),
-- custom
snip('mm', '- ', {}, math),
snip('pl', '+ ', {}, math),
snip('nl', '\\ \n<>', {i(1)}, math),
snip('pm', '+- ', {}, math),
snip('nx', '\\, space ', {}, math),
snip('nbx', '\\, quad ', {}, math),
snip('deg', 'degree ', {}, math),
snip('ta', 'star ', {}, math),
snip('del', 'Delta ', {}, math),
snip('apr', 'approx ', {}, math),
snip('pal', 'parallel ', {}, math),
snip('sa', 'space ', {}, math),
-- subscript/superscript
-- operators
snip('ak([^k ])', '+ <>', { cap(1) }, math, 100, { wordTrig = false }),
snip('sk([^k ])', '- <>', { cap(1) }, math, 100, { wordTrig = false }),
snip('oak', 'plus.circle ', {}, math),
snip('bak', 'plus.square ', {}, math),
snip('mak', 'plus.minus ', {}, math),
snip('xx', 'times ', {}, math),
snip('oxx', 'times.circle ', {}, math),
snip('bxx', 'times.square ', {}, math),
snip('ff', '(<>) / (<>) <>', { i(1), i(2), i(3) }, math),
snip('nab', 'arrow(nabla) ', {}, math),
-- exponents
snip('iv', '^(-1) ', {}, math, 500, { wordTrig = false, blacklist = { 'equiv' } }),
snip('tp', '^top ', {}, math, 500, { wordTrig = false }),
snip('cmp', '^complement ', {}, math, 500, { wordTrig = false }),
snip('prp', '^perp ', {}, math, 500, { wordTrig = false }),
snip('sr', '^2 ', {}, math, 500, { wordTrig = false }),
snip('cb', '^3 ', {}, math, 500, { wordTrig = false }),
snip('jj', '_(<>) ', { i(1, 'n') }, math, 500, { wordTrig = false }),
snip('kk', '^(<>) ', { i(1, 'n') }, math, 500, { wordTrig = false }),
snip('ep', 'exp(<>) ', { i(1, '1') }, math),
-- sets
-- 'st' to '{<>}' in ./visual.lua
-- 'st' to '{<>} in ./visual.lua
snip('set', '{<> mid(|) <>}', { i(1), i(2) }, math),
snip('es', 'emptyset ', {}, math),
snip('ses', '{emptyset} ', {}, math),
@@ -69,45 +83,38 @@ return {
snip('bnn', 'inter.big ', {}, math),
snip('buu', 'union.big ', {}, math),
snip('swo', 'without ', {}, math),
snip('ni', 'in.not ', {}, math),
-- misc
snip('to', '->> ', {}, math),
snip('mt', '|->> ', {}, math),
snip('cp', 'compose ', {}, math),
snip('Oo', 'compose ', {}, math),
snip('iso', 'tilde.equiv ', {}, math),
snip('nab', 'nabla ', {}, math),
snip('ep', 'exp(<>) ', { i(1, '1') }, math),
snip('ccs', 'cases(\n\t<>,\n)', { i(1, '1') }, math),
snip('cc', 'cases(\n\t<>\n)\\', { i(1) }, math),
snip('([A-Za-z])o([A-Za-z0-9]) ', '<>(<>) ', { cap(1), cap(2) }, math, 100, {
maxTrigLength = 4,
blacklist = { 'bot ', 'cos ', 'cot ', 'dot ', 'log ', 'mod ', 'not ', 'top ', 'won ', 'xor ' },
blacklist = { 'bot ', 'cos ', 'cot ', 'dot ', 'log ', 'mod ', 'top ', 'won ', 'xor ' },
}),
snip('(K|M|N|Q|R|S|Z)([\\dn]) ', '<><>^<> ', { cap(1), cap(1), cap(2) }, math),
-- derivatives
snip('dx', 'dif / (dif <>) ', { i(1, 'x') }, math),
snip('ddx', '(dif <>) / (dif <>) ', { i(1, 'f'), i(2, 'x') }, math),
snip('DX', 'partial / (partial <>) ', { i(1, 'x') }, math),
snip('DDX', '(partial <>) / (partial <>) ', { i(1, 'f'), i(2, 'x') }, math),
snip('part', 'partial ', {}, math, 1600),
-- integrals
snip('DX', 'diff / (diff <>) ', { i(1, 'x') }, math),
snip('DDX', '(diff <>) / (diff <>) ', { i(1, 'f'), i(2, 'x') }, math),
snip('ppa', 'partial ', {}, math, 1600),
snip('it', 'integral ', {}, math),
snip('iot', 'integral.vol ', {}, math),
snip('ift', 'integral.surf ', {}, math),
snip('int', 'integral_(<>)^(<>) ', { i(1, 'a'), i(2, 'b') }, math),
snip('oit', 'integral.cont_(<>) ', { i(1, 'C') }, math),
snip('dit', 'integral_(<>) ', { i(1, 'Omega') }, math),
-- sums
snip('sm', 'sum ', {}, math),
snip('sum', 'sum_(<>)^(<>) ', { i(1, 'k=1'), i(2, 'oo') }, math),
snip('dsm', 'sum_(<>) ', { i(1, 'Omega') }, math),
-- products
snip('prd', 'product ', {}, math),
snip('prod', 'product_(<>)^(<>) ', { i(1, 'k=1'), i(2, 'n') }, math),
-- limits
snip('lm', 'lim ', {}, math),
snip('lim', 'lim_(<> ->> <>) ', { i(1, 'n'), i(2, 'oo') }, math),
snip('lim (sup|inf)', 'lim<> ', { cap(1) }, math),

View File

@@ -28,20 +28,20 @@ local operations = { -- first boolean: existing brackets should be kept; second
{ 'sQ', '[', ']', false, false }, -- replace with square brackets
{ 'BB', '', '', false, false }, -- remove brackets
{ 'ss', '"', '"', false, false },
{ 'chv', 'lr(chevron.l ', ' chevron.r)', false, false },
{ 'agl', 'lr(angle.l ', ' angle.r)', false, false },
{ 'abs', 'abs', '', true, true },
{ 'ul', 'underline', '', true, true },
{ 'ol', 'overline', '', true, true },
{ 'ub', 'underbrace', '', true, true },
{ 'ob', 'overbrace', '', true, true },
{ 'ht', 'hat', '', true, true },
{ 'ar', 'arrow', '', true, true },
{ 'br', 'macron', '', true, true },
{ 'dt', 'dot', '', true, true },
{ 'dia', 'diaer', '', true, true },
{ 'dou', 'dot.double', '', true, true },
{ 'ci', 'circle', '', true, true },
{ 'td', 'tilde', '', true, true },
{ 'nr', 'norm', '', true, true },
{ 'arr', 'arrow', '', true, true },
{ 'vv', 'vec', '', true, true },
{ 'rt', 'sqrt', '', true, true },
{ 'flo', 'floor', '', true, true },
@@ -98,7 +98,7 @@ local smart_wrap = function(args, parent, old_state, expand)
-- normal snippet
if not visual_disable_normal[trigger] then
return s(nil, { t(expand1), i(1, '1+1'), t(expand2) })
return s(nil, { t(expand1), i(1), t(expand2) })
else
return s(nil, t(trigger))
end

View File

@@ -4,7 +4,7 @@ build-backend = "pdm.backend"
[project]
name = "typstar"
version = "1.4.2"
version = "1.4.1"
description = "Neovim plugin for efficient note taking in Typst"
authors = [
{ name = "arne314" }

2
uv.lock generated
View File

@@ -516,7 +516,7 @@ wheels = [
[[package]]
name = "typstar"
version = "1.4.2"
version = "1.4.1"
source = { editable = "." }
dependencies = [
{ name = "aiohttp" },