Skip to content

Commit ec2d3ab

Browse files
authored
Merge pull request #272 from vim-denops/feat-buffer-decoration
Add `listDecorations` in `buffer/decoration`
2 parents 402dedb + f595083 commit ec2d3ab

File tree

4 files changed

+241
-35
lines changed

4 files changed

+241
-35
lines changed

.scripts/gen-option/parse.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@ export function parse(content: string) {
4040
name: string;
4141
start: number;
4242
block: string;
43-
err: Error;
43+
err: unknown;
4444
}[] = [];
4545
let last = -1;
4646
for (const match of content.matchAll(/\*'(\w+)'\*/g)) {

buffer/decoration.ts

+91-19
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,8 @@ import * as nvimFn from "../function/nvim/mod.ts";
77

88
const cacheKey = "denops_std/buffer/decoration/vimDecorate/rs@1";
99

10+
const prefix = "denops_std:buffer:decoration:decorate";
11+
1012
export interface Decoration {
1113
/**
1214
* Line number
@@ -61,7 +63,7 @@ export interface Decoration {
6163
export function decorate(
6264
denops: Denops,
6365
bufnr: number,
64-
decorations: Decoration[],
66+
decorations: readonly Decoration[],
6567
): Promise<void> {
6668
switch (denops.meta.host) {
6769
case "vim":
@@ -131,17 +133,30 @@ export function undecorate(
131133
}
132134
}
133135

134-
function uniq<T>(array: T[]): T[] {
136+
export function listDecorations(
137+
denops: Denops,
138+
bufnr: number,
139+
): Promise<Decoration[]> {
140+
switch (denops.meta.host) {
141+
case "vim":
142+
return vimListDecorations(denops, bufnr);
143+
case "nvim":
144+
return nvimListDecorations(denops, bufnr);
145+
default:
146+
unreachable(denops.meta.host);
147+
}
148+
}
149+
150+
function uniq<T>(array: readonly T[]): T[] {
135151
return [...new Set(array)];
136152
}
137153

138154
async function vimDecorate(
139155
denops: Denops,
140156
bufnr: number,
141-
decorations: Decoration[],
157+
decorations: readonly Decoration[],
142158
): Promise<void> {
143-
const toPropType = (n: string) =>
144-
`denops_std:buffer:decoration:decorate:${n}`;
159+
const toPropType = (n: string) => `${prefix}:${n}`;
145160
const rs = (denops.context[cacheKey] ?? new Set()) as Set<string>;
146161
denops.context[cacheKey] = rs;
147162
const hs = uniq(decorations.map((v) => v.highlight)).filter((v) =>
@@ -163,8 +178,9 @@ async function vimDecorate(
163178
});
164179
rs.add(highlight);
165180
}
181+
let id = 1;
166182
for (const [type, props] of decoMap.entries()) {
167-
await vimFn.prop_add_list(denops, { bufnr, type }, [...props]);
183+
await vimFn.prop_add_list(denops, { bufnr, type, id: id++ }, [...props]);
168184
}
169185
});
170186
}
@@ -178,11 +194,11 @@ async function vimUndecorate(
178194
const propList = await vimFn.prop_list(denops, start + 1, {
179195
bufnr,
180196
end_lnum: end,
181-
}) as { id: string; type: string }[];
197+
}) as { id: number; type: string }[];
182198
const propIds = new Set(
183-
propList.filter((p) =>
184-
p.type.startsWith("denops_std:buffer:decoration:decorate:")
185-
).map((p) => p.id),
199+
propList
200+
.filter((p) => p.type.startsWith(`${prefix}:`))
201+
.map((p) => p.id),
186202
);
187203
await batch(denops, async (denops) => {
188204
for (const propId of propIds) {
@@ -191,15 +207,39 @@ async function vimUndecorate(
191207
});
192208
}
193209

210+
async function vimListDecorations(
211+
denops: Denops,
212+
bufnr: number,
213+
): Promise<Decoration[]> {
214+
const props = await vimFn.prop_list(denops, 1, {
215+
bufnr,
216+
end_lnum: -1,
217+
}) as {
218+
col: number;
219+
end: number;
220+
id: number;
221+
length: number;
222+
lnum: number;
223+
start: number;
224+
type: string;
225+
type_bufnr: number;
226+
}[];
227+
return props
228+
.filter((prop) => prop.type.startsWith(`${prefix}:`))
229+
.map((prop) => ({
230+
line: prop.lnum,
231+
column: prop.col,
232+
length: prop.length,
233+
highlight: prop.type.split(":").pop() as string,
234+
}));
235+
}
236+
194237
async function nvimDecorate(
195238
denops: Denops,
196239
bufnr: number,
197-
decorations: Decoration[],
240+
decorations: readonly Decoration[],
198241
): Promise<void> {
199-
const ns = await nvimFn.nvim_create_namespace(
200-
denops,
201-
"denops_std:buffer:decoration:decorate",
202-
);
242+
const ns = await nvimFn.nvim_create_namespace(denops, prefix);
203243
for (const chunk of itertools.chunked(decorations, 1000)) {
204244
await batch(denops, async (denops) => {
205245
for (const deco of chunk) {
@@ -223,9 +263,41 @@ async function nvimUndecorate(
223263
start: number,
224264
end: number,
225265
): Promise<void> {
226-
const ns = await nvimFn.nvim_create_namespace(
227-
denops,
228-
"denops_std:buffer:decoration:decorate",
229-
);
266+
const ns = await nvimFn.nvim_create_namespace(denops, prefix);
230267
await nvimFn.nvim_buf_clear_namespace(denops, bufnr, ns, start, end);
231268
}
269+
270+
async function nvimListDecorations(
271+
denops: Denops,
272+
bufnr: number,
273+
): Promise<Decoration[]> {
274+
const ns = await nvimFn.nvim_create_namespace(denops, prefix);
275+
const extmarks = await nvimFn.nvim_buf_get_extmarks(
276+
denops,
277+
bufnr,
278+
ns,
279+
0,
280+
-1,
281+
{ details: true },
282+
) as [
283+
extmark_id: number,
284+
row: number,
285+
col: number,
286+
{
287+
hl_group: string;
288+
hl_eol: boolean;
289+
end_right_gravity: boolean;
290+
priority: number;
291+
right_gravity: boolean;
292+
end_col: number;
293+
ns_id: number;
294+
end_row: number;
295+
},
296+
][];
297+
return extmarks.map((extmark) => ({
298+
line: extmark[1] + 1,
299+
column: extmark[2] + 1,
300+
length: extmark[3].end_col - extmark[2],
301+
highlight: extmark[3].hl_group,
302+
}));
303+
}

buffer/decoration_test.ts

+147-14
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,34 @@
11
import { assertEquals } from "@std/assert";
2+
import { omit } from "@std/collections";
23
import { test } from "@denops/test";
34
import * as fn from "../function/mod.ts";
45
import * as vimFn from "../function/vim/mod.ts";
56
import * as nvimFn from "../function/nvim/mod.ts";
67
import * as buffer from "./buffer.ts";
7-
import { decorate } from "./decoration.ts";
8+
import { decorate, listDecorations, undecorate } from "./decoration.ts";
89

910
test({
1011
mode: "vim",
1112
name: "decorate define highlights as text properties",
1213
fn: async (denops) => {
1314
const collect = async (bufnr: number) => {
14-
const lines = await fn.getbufline(denops, bufnr, 1, "$");
15-
const props = [];
16-
for (let line = 1; line <= lines.length; line++) {
17-
props.push(
18-
...(await vimFn.prop_list(denops, line, { bufnr }) as unknown[]),
19-
);
20-
}
21-
return props;
15+
const props = await vimFn.prop_list(denops, 1, {
16+
bufnr,
17+
end_lnum: -1,
18+
}) as {
19+
col: number;
20+
end: number;
21+
id: number;
22+
length: number;
23+
lnum: number;
24+
start: number;
25+
type: string;
26+
type_bufnr: number;
27+
}[];
28+
return props.map((prop) => omit(prop, ["id"]));
2229
};
2330
const bufnr = await fn.bufnr(denops);
24-
await buffer.append(denops, bufnr, [
31+
await buffer.replace(denops, bufnr, [
2532
"Hello",
2633
"Darkness",
2734
"My",
@@ -44,28 +51,45 @@ test({
4451
assertEquals(await collect(bufnr), [{
4552
col: 1,
4653
end: 1,
47-
id: 0,
48-
length: 1,
54+
length: 5,
55+
lnum: 1,
4956
start: 1,
5057
type: "denops_std:buffer:decoration:decorate:Title",
5158
type_bufnr: 0,
5259
}, {
5360
col: 2,
5461
end: 1,
55-
id: 0,
5662
length: 3,
63+
lnum: 2,
5764
start: 1,
5865
type: "denops_std:buffer:decoration:decorate:Search",
5966
type_bufnr: 0,
6067
}]);
68+
69+
// Re-appling the same decorations is OK
70+
await decorate(denops, bufnr, [
71+
{
72+
line: 1,
73+
column: 1,
74+
length: 5,
75+
highlight: "Title",
76+
},
77+
{
78+
line: 2,
79+
column: 2,
80+
length: 3,
81+
highlight: "Search",
82+
},
83+
]);
6184
},
6285
});
86+
6387
test({
6488
mode: "nvim",
6589
name: "decorate define highlights as extmarks",
6690
fn: async (denops) => {
6791
const bufnr = await fn.bufnr(denops);
68-
await buffer.append(denops, bufnr, [
92+
await buffer.replace(denops, bufnr, [
6993
"Hello",
7094
"Darkness",
7195
"My",
@@ -98,3 +122,112 @@ test({
98122
);
99123
},
100124
});
125+
126+
test({
127+
mode: "all",
128+
name: "listDecorations list decorations defined in the buffer",
129+
fn: async (denops) => {
130+
const bufnr = await fn.bufnr(denops);
131+
await buffer.replace(denops, bufnr, [
132+
"Hello",
133+
"Darkness",
134+
"My",
135+
"Old friend",
136+
]);
137+
await decorate(denops, bufnr, [
138+
{
139+
line: 1,
140+
column: 1,
141+
length: 5,
142+
highlight: "Title",
143+
},
144+
{
145+
line: 2,
146+
column: 2,
147+
length: 3,
148+
highlight: "Search",
149+
},
150+
]);
151+
assertEquals(await listDecorations(denops, bufnr), [
152+
{
153+
line: 1,
154+
column: 1,
155+
length: 5,
156+
highlight: "Title",
157+
},
158+
{
159+
line: 2,
160+
column: 2,
161+
length: 3,
162+
highlight: "Search",
163+
},
164+
]);
165+
},
166+
});
167+
168+
test({
169+
mode: "all",
170+
name: "undecorate clears decorations in the buffer",
171+
fn: async (denops) => {
172+
const bufnr = await fn.bufnr(denops);
173+
await buffer.replace(denops, bufnr, [
174+
"Hello",
175+
"Darkness",
176+
"My",
177+
"Old friend",
178+
]);
179+
await decorate(denops, bufnr, [
180+
{
181+
line: 1,
182+
column: 1,
183+
length: 5,
184+
highlight: "Title",
185+
},
186+
{
187+
line: 2,
188+
column: 2,
189+
length: 3,
190+
highlight: "Search",
191+
},
192+
]);
193+
await undecorate(denops, bufnr);
194+
assertEquals(await listDecorations(denops, bufnr), []);
195+
},
196+
});
197+
198+
test({
199+
mode: "all",
200+
name: "undecorate clears decorations in specified region in the buffer",
201+
fn: async (denops) => {
202+
const bufnr = await fn.bufnr(denops);
203+
await buffer.replace(denops, bufnr, [
204+
"Hello",
205+
"Darkness",
206+
"My",
207+
"Old friend",
208+
]);
209+
await decorate(denops, bufnr, [
210+
{
211+
line: 1,
212+
column: 1,
213+
length: 5,
214+
highlight: "Title",
215+
},
216+
{
217+
line: 2,
218+
column: 2,
219+
length: 3,
220+
highlight: "Search",
221+
},
222+
]);
223+
await undecorate(denops, bufnr, 0, 1);
224+
assertEquals(await listDecorations(denops, bufnr), [
225+
{
226+
line: 2,
227+
column: 2,
228+
length: 3,
229+
highlight: "Search",
230+
},
231+
]);
232+
},
233+
});

0 commit comments

Comments
 (0)