aboutsummaryrefslogtreecommitdiff
path: root/plugins/minimap.lua
blob: 826db8a527c73f50298914d3b2b2af54614ef11c (plain)
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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
-- mod-version:2
local command = require "core.command"
local common = require "core.common"
local config = require "core.config"
local style = require "core.style"
local DocView = require "core.docview"
local Object = require "core.object"

-- General plugin settings
config.plugins.minimap = {
	enabled = true,
	width = 100,
	instant_scroll = false,
	syntax_highlight = true,
	scale = 1,
	-- how many spaces one tab is equivalent to
	tab_width = 4,
	draw_background = true,
	-- if highlight_width is positive, it is drawn on the left
	-- if highlight_width is negative, it is drawn on the right
	-- gutter_width pushes the minimap text to the left to make room for a left-side highlight
	highlight_width = -5,
	gutter_width = 0,
	-- try these values:
	-- full width:
	-- config.plugins.minimap.highlight_width = 100
	-- config.plugins.minimap.gutter_width = 0
	-- left side:
	-- config.plugins.minimap.highlight_width = 3
	-- config.plugins.minimap.gutter_width = 4
	-- right side:
	-- config.plugins.minimap.highlight_width = -5
	-- config.plugins.minimap.gutter_width = 0
}

-- Configure size for rendering each char in the minimap
local char_height = 1 * SCALE * config.plugins.minimap.scale
local char_spacing = 0.8 * SCALE * config.plugins.minimap.scale
local line_spacing = 2 * SCALE * config.plugins.minimap.scale

local MiniMap = Object:extend()

function MiniMap:new()
end

function MiniMap:line_highlight_color(line_index)
	-- other plugins can override this, and return a color
end

local minimap = MiniMap()

-- Overloaded since the default implementation adds a extra x3 size of hotspot for the mouse to hit the scrollbar.
local prev_scrollbar_overlaps_point = DocView.scrollbar_overlaps_point
DocView.scrollbar_overlaps_point = function(self, x, y)
	if not config.plugins.minimap.enabled then
		return prev_scrollbar_overlaps_point(self, x, y)
	end

	local sx, sy, sw, sh = self:get_scrollbar_rect()
	return x >= sx and x < sx + sw and y >= sy and y < sy + sh
end

-- Helper function to determine if current file is too large to be shown fully inside the minimap area.
local function is_file_too_large(self)
	local line_count = #self.doc.lines
	local _, _, _, sh = self:get_scrollbar_rect()

	-- check if line count is too large to fit inside the minimap area
	local max_minmap_lines = math.floor(sh / line_spacing)
	return line_count > 1 and line_count > max_minmap_lines
end

-- Overloaded with an extra check if the user clicked inside the minimap to automatically scroll to that line.
local prev_on_mouse_pressed = DocView.on_mouse_pressed
DocView.on_mouse_pressed = function(self, button, x, y, clicks)
	if not config.plugins.minimap.enabled then
		return prev_on_mouse_pressed(self, button, x, y, clicks)
	end

	-- check if user clicked in the minimap area and jump directly to that line
	-- unless they are actually trying to perform a drag
	local minimap_hit = self:scrollbar_overlaps_point(x, y)
	if minimap_hit then
		local line_count = #self.doc.lines
		local minimap_height = line_count * line_spacing

		-- check if line count is too large to fit inside the minimap area
		local is_too_large = is_file_too_large(self)
		if is_too_large then
			local _, _, _, sh = self:get_scrollbar_rect()
			minimap_height = sh
		end

		-- calc which line to jump to
		local dy = y - self.position.y
		local jump_to_line = math.floor((dy / minimap_height) * line_count) + 1

		local _, cy, _, cy2 = self:get_content_bounds()
		local lh = self:get_line_height()
		local visible_lines_count = math.max(1, (cy2 - cy) / lh)
		local visible_lines_start = math.max(1, math.floor(cy / lh))

		-- calc if user hit the currently visible area
		local hit_visible_area = true
		if is_too_large then

			local visible_height = visible_lines_count * line_spacing
			local scroll_pos = (visible_lines_start - 1) /
							                   (line_count - visible_lines_count - 1)
			scroll_pos = math.min(1.0, scroll_pos) -- 0..1
			local visible_y = self.position.y + scroll_pos *
							                  (minimap_height - visible_height)

			local t = (line_count - visible_lines_start) / visible_lines_count
			if t <= 1 then visible_y = visible_y + visible_height * (1.0 - t) end

			if y < visible_y or y > visible_y + visible_height then
				hit_visible_area = false
			end
		else

			-- If the click is on the currently visible line numbers,
			-- ignore it since then they probably want to initiate a drag instead.
			if jump_to_line < visible_lines_start or jump_to_line > visible_lines_start +
							visible_lines_count then hit_visible_area = false end
		end

		-- if user didn't click on the visible area (ie not dragging), scroll accordingly
		if not hit_visible_area then
			self:scroll_to_line(jump_to_line, false, config.plugins.minimap.instant_scroll)
			return
		end

	end

	return prev_on_mouse_pressed(self, button, x, y, clicks)
end

-- Overloaded with pretty much the same logic as original DocView implementation,
-- with the exception of the dragging scrollbar delta. We want it to behave a bit snappier
-- since the "scrollbar" essentially represents the lines visible in the content view.
local prev_on_mouse_moved = DocView.on_mouse_moved
DocView.on_mouse_moved = function(self, x, y, dx, dy)
	if not config.plugins.minimap.enabled then
		return prev_on_mouse_moved(self, x, y, dx, dy)
	end

	if self.dragging_scrollbar then
		local line_count = #self.doc.lines
		local lh = self:get_line_height()
		local delta = lh / line_spacing * dy

		if is_file_too_large(self) then
			local _, sy, _, sh = self:get_scrollbar_rect()
			delta = (line_count * lh) / sh * dy
		end

		self.scroll.to.y = self.scroll.to.y + delta
	end

	-- we need to "hide" that the scrollbar is dragging so that View doesnt does its own scrolling logic
	local t = self.dragging_scrollbar
	self.dragging_scrollbar = false
	local r = prev_on_mouse_moved(self, x, y, dx, dy)
	self.dragging_scrollbar = t
	return r
end

-- Overloaded since we want the mouse to interact with the full size of the minimap area,
-- not juse the scrollbar.
local prev_get_scrollbar_rect = DocView.get_scrollbar_rect
DocView.get_scrollbar_rect = function(self)
	if not config.plugins.minimap.enabled then return prev_get_scrollbar_rect(self) end

	return self.position.x + self.size.x - config.plugins.minimap.width * SCALE,
	       self.position.y, config.plugins.minimap.width * SCALE, self.size.y
end

-- Overloaded so we can render the minimap in the "scrollbar area".
local prev_draw_scrollbar = DocView.draw_scrollbar
DocView.draw_scrollbar = function(self)
	if not config.plugins.minimap.enabled then return prev_draw_scrollbar(self) end

	local x, y, w, h = self:get_scrollbar_rect()

	local highlight = self.hovered_scrollbar or self.dragging_scrollbar
	local visual_color = highlight and style.scrollbar2 or style.scrollbar

	local _, cy, _, cy2 = self:get_content_bounds()
	local lh = self:get_line_height()
	local visible_lines_count = math.max(1, (cy2 - cy) / lh)
	local visible_lines_start = math.max(1, math.floor(cy / lh))
	local scroller_height = visible_lines_count * line_spacing
	local line_count = #self.doc.lines

	local visible_y = self.position.y + (visible_lines_start - 1) * line_spacing

	-- check if file is too large to fit inside the minimap area
	local max_minmap_lines = math.floor(h / line_spacing)
	local minimap_start_line = 1
	if is_file_too_large(self) then

		local scroll_pos = (visible_lines_start - 1) /
						                   (line_count - visible_lines_count - 1)
		scroll_pos = math.min(1.0, scroll_pos) -- 0..1, procent of visual area scrolled

		local scroll_pos_pixels = scroll_pos * (h - scroller_height)
		visible_y = self.position.y + scroll_pos_pixels

		-- offset visible area if user is scrolling past end
		local t = (line_count - visible_lines_start) / visible_lines_count
		if t <= 1 then visible_y = visible_y + scroller_height * (1.0 - t) end

		minimap_start_line = visible_lines_start -
						                     math.floor(scroll_pos_pixels / line_spacing)
		minimap_start_line = math.max(1, math.min(minimap_start_line,
		                                          line_count - max_minmap_lines))
	end

	if config.plugins.minimap.draw_background then
		renderer.draw_rect(x, y, w, h, style.minimap_background or style.background)
	end
	-- draw visual rect
	renderer.draw_rect(x, visible_y, w, scroller_height, visual_color)

	local highlight_width = config.plugins.minimap.highlight_width
	local gutter_width = config.plugins.minimap.gutter_width

	-- time to draw the actual code, setup some local vars that are used in both highlighted and plain renderind.
	local line_y = y

	-- when not using syntax highlighted rendering, just use the normal color but dim it 50%.
	local color = style.syntax["normal"]
	color = {color[1], color[2], color[3], color[4] * 0.5}

	-- we try to "batch" characters so that they can be rendered as just one rectangle instead of one for each.
	local batch_width = 0
	local batch_start = x
	local minimap_cutoff_x = x + config.plugins.minimap.width * SCALE
	local batch_syntax_type = nil
	local function flush_batch(type)
		local old_color = color
		color = style.syntax[batch_syntax_type]
		if config.plugins.minimap.syntax_highlight and color ~= nil then
			-- fetch and dim colors
			color = {color[1], color[2], color[3], color[4] * 0.5}
		else
			color = old_color
		end
		if batch_width > 0 then
			renderer.draw_rect(batch_start, line_y, batch_width, char_height, color)
		end
		batch_syntax_type = type
		batch_start = batch_start + batch_width
		batch_width = 0
	end

	local function render_highlight(idx, line_y)
		local highlight_color = minimap:line_highlight_color(idx)
		if highlight_color then
			if highlight_width > 0 then
				renderer.draw_rect(x, line_y, highlight_width, line_spacing, highlight_color)
			else
				renderer.draw_rect(x + w + highlight_width, line_y, -highlight_width, line_spacing, highlight_color)
			end
		end
	end

	-- render lines with syntax highlighting
	if config.plugins.minimap.syntax_highlight then

		-- keep track of the highlight type, since this needs to break batches as well
		batch_syntax_type = nil

		-- per line
		local endidx = minimap_start_line + max_minmap_lines
		endidx = math.min(endidx, line_count)
		for idx = minimap_start_line, endidx do
			batch_syntax_type = nil
			batch_start = x + gutter_width
			batch_width = 0

			render_highlight(idx, line_y)

			-- per token
			for _, type, text in self.doc.highlighter:each_token(idx) do
				-- flush prev batch
				if not batch_syntax_type then batch_syntax_type = type end
				if batch_syntax_type ~= type then flush_batch(type) end

				-- per character
				for char in common.utf8_chars(text) do
					if char == " " or char == "\n" then
						flush_batch(type)
						batch_start = batch_start + char_spacing
					elseif char == "	" then
						flush_batch(type)
						batch_start = batch_start + (char_spacing * config.plugins.minimap.tab_width)
					elseif batch_start + batch_width > minimap_cutoff_x then
						flush_batch(type)
						break
					else
						batch_width = batch_width + char_spacing
					end

				end
			end
			flush_batch(nil)
			line_y = line_y + line_spacing
		end

	else -- render lines without syntax highlighting
		for idx = 1, line_count - 1 do
			batch_start = x + gutter_width
			batch_width = 0

			render_highlight(idx, line_y)

			for char in common.utf8_chars(self.doc.lines[idx]) do
				if char == " " or char == "\n" then
					flush_batch()
					batch_start = batch_start + char_spacing
				elseif batch_start + batch_width > minimap_cutoff_x then
					flush_batch()
				else
					batch_width = batch_width + char_spacing
				end
			end
			flush_batch()
			line_y = line_y + line_spacing
		end

	end

end

local prev_update = DocView.update
DocView.update = function (self)
	if not config.plugins.minimap.enabled then return prev_update(self) end
	self.size.x = self.size.x - config.plugins.minimap.width * SCALE
	return prev_update(self)
end

command.add(nil, {
	["minimap:toggle-visibility"] = function()
		config.plugins.minimap.enabled = not config.plugins.minimap.enabled
	end,
	["minimap:toggle-syntax-highlighting"] = function()
		config.plugins.minimap.syntax_highlight = not config.plugins.minimap.syntax_highlight
	end
})

return minimap