aboutsummaryrefslogtreecommitdiff
path: root/register.c
blob: 37a30cef4e4115d368d239905d68149eb92f0cc0 (plain) (blame)
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
#include <stdlib.h>
#include <string.h>

#include "vis.h"
#include "text.h"
#include "util.h"
#include "register.h"

typedef struct {
	Buffer *stdout;
	Buffer *stderr;
} Clipboard;

static ssize_t read_stdout(void *context, char *data, size_t len) {
	Buffer *buf = ((Clipboard*)context)->stdout;
	buffer_append(buf, data, len);
	return len;
}

static ssize_t read_stderr(void *context, char *data, size_t len) {
	Buffer *buf = ((Clipboard*)context)->stderr;
	buffer_append(buf, data, len);
	return len;
}

void register_release(Register *reg) {
	buffer_release(&reg->buf);
}

const char *register_get(Vis *vis, Register *reg, size_t *len) {
	switch (reg->type) {
	case REGISTER_NORMAL:
		*len = buffer_length0(&reg->buf);
		return reg->buf.data;
	case REGISTER_CLIPBOARD:
	{
		Buffer stderr;
		buffer_init(&stderr);
		buffer_clear(&reg->buf);
		Clipboard clipboard = {
			.stdout = &reg->buf,
			.stderr = &stderr,
		};

		int status = vis_pipe(vis, &clipboard,
			&(Filerange){ .start = 0, .end = 0 },
			(const char*[]){ "vis-paste", "vis-paste", NULL },
			read_stdout, read_stderr);
		if (status != 0)
			vis_info_show(vis, "Command failed %s", stderr.len > 0 ? stderr.data : "");
		*len = reg->buf.len;
		return reg->buf.data;
	}
	case REGISTER_BLACKHOLE:
	default:
		*len = 0;
		return NULL;
	}
}

bool register_put_range(Vis *vis, Register *reg, Text *txt, Filerange *range) {
	if (reg->append)
		return register_append_range(reg, txt, range);
	switch (reg->type) {
	case REGISTER_NORMAL:
	{
		size_t len = text_range_size(range);
		if (!buffer_grow(&reg->buf, len))
			return false;
		reg->buf.len = text_bytes_get(txt, range->start, len, reg->buf.data);
		return true;
	}
	case REGISTER_CLIPBOARD:
	{
		Buffer stderr;
		buffer_init(&stderr);
		Clipboard clipboard = {
			.stderr = &stderr,
		};

		int status = vis_pipe(vis, &clipboard, range,
			(const char*[]){ "vis-copy", "vis-copy", NULL },
			NULL, read_stderr);

		if (status != 0)
			vis_info_show(vis, "Command failed %s", stderr.len > 0 ? stderr.data : "");
		return status == 0;
	}
	case REGISTER_BLACKHOLE:
		return true;
	default:
		return false;
	}
}

bool register_append_range(Register *reg, Text *txt, Filerange *range) {
	switch (reg->type) {
	case REGISTER_NORMAL:
	{
		size_t len = text_range_size(range);
		if (!buffer_grow(&reg->buf, reg->buf.len + len))
			return false;
		reg->buf.len += text_bytes_get(txt, range->start, len, reg->buf.data + reg->buf.len);
		return true;
	}
	default:
		return false;
	}
}