1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
|
// deals with textDocument/signatureHelp LSP requests
void signature_help_open(Ted *ted, char32_t trigger) {
(void)trigger; // for now we don't send context
TextBuffer *buffer = ted->active_buffer;
if (!buffer) return;
LSP *lsp = buffer_lsp(buffer);
LSPRequest request = {.type = LSP_REQUEST_SIGNATURE_HELP};
LSPRequestSignatureHelp *s = &request.data.signature_help;
s->position = buffer_cursor_pos_as_lsp_document_position(buffer);
lsp_send_request(lsp, &request);
}
bool signature_help_is_open(Ted *ted) {
return ted->signature_help.signature_count > 0;
}
static void signature_help_clear(SignatureHelp *help) {
for (int i = 0; i < help->signature_count; ++i) {
Signature sig = help->signatures[i];
free(sig.label_pre);
free(sig.label_active);
free(sig.label_post);
}
memset(help->signatures, 0, sizeof help->signatures);
}
void signature_help_close(Ted *ted) {
signature_help_clear(&ted->signature_help);
}
void signature_help_process_lsp_response(Ted *ted, const LSPResponse *response) {
if (response->request.type != LSP_REQUEST_SIGNATURE_HELP)
return;
SignatureHelp *help = &ted->signature_help;
const LSPResponseSignatureHelp *lsp_help = &response->data.signature_help;
u32 signature_count = arr_len(lsp_help->signatures);
if (signature_count > SIGNATURE_HELP_MAX)
signature_count = SIGNATURE_HELP_MAX;
signature_help_clear(help);
for (u32 s = 0; s < signature_count; ++s) {
Signature *signature = &help->signatures[s];
LSPSignatureInformation *lsp_signature = &lsp_help->signatures[s];
const char *label = lsp_response_string(response, lsp_signature->label);
size_t start = unicode_utf16_to_utf8_offset(label, lsp_signature->active_start);
size_t end = unicode_utf16_to_utf8_offset(label, lsp_signature->active_end);
if (start == (size_t)-1) {
assert(0);
start = 0;
}
if (end == (size_t)-1) {
assert(0);
end = 0;
}
u32 active_start = (u32)start;
u32 active_end = (u32)end;
signature->label_pre = strn_dup(label, active_start);
signature->label_active = strn_dup(label + active_start, active_end - active_start);
signature->label_post = str_dup(label + active_end);
}
help->signature_count = (u16)signature_count;
}
void signature_help_frame(Ted *ted) {
SignatureHelp *help = &ted->signature_help;
u16 signature_count = help->signature_count;
if (!signature_count)
return;
Font *font = ted->font;
Font *font_bold = ted->font_bold;
TextBuffer *buffer = ted->active_buffer;
if (!buffer)
return;
Settings *settings = buffer_settings(buffer);
u32 *colors = settings->colors;
float border = settings->border_thickness;
float width = buffer->x2 - buffer->x1;
float height = FLT_MAX;
// make sure signature help doesn't take up too much space
while (1) {
height = font->char_height * signature_count;
if (height < (buffer->y2 - buffer->y1) * 0.25f)
break;
--signature_count;
if (signature_count == 0) return;
}
float x = buffer->x1, y = buffer->y2 - height;
gl_geometry_rect(rect_xywh(x, y - border, width, border),
colors[COLOR_AUTOCOMPLETE_BORDER]);
gl_geometry_rect(rect_xywh(x, y, width, height),
colors[COLOR_AUTOCOMPLETE_BG]);
// draw the signatures
for (int s = 0; s < signature_count; ++s) {
const Signature *signature = &help->signatures[s];
TextRenderState state = text_render_state_default;
state.x = x;
state.y = y;
state.min_x = x;
state.min_y = y;
state.max_x = buffer->x2;
state.max_y = buffer->y2;
rgba_u32_to_floats(colors[COLOR_TEXT], state.color);
text_utf8_with_state(font, &state, signature->label_pre);
text_utf8_with_state(font_bold, &state, signature->label_active);
text_utf8_with_state(font, &state, signature->label_post);
y += font->char_height;
}
gl_geometry_draw();
text_render(font);
text_render(font_bold);
}
|