view editor.luan @ 14:357fdbf446cb

add show_whitespace
author Franklin Schmidt <fschmidt@gmail.com>
date Sat, 05 Apr 2025 22:50:38 -0600
parents 2baecd73d6bb
children 93e46dadb694
line wrap: on
line source

local Luan = require "luan:Luan.luan"
local error = Luan.error
local ipairs = Luan.ipairs or error()
local Io = require "luan:Io.luan"
local print = Io.print or error()
local new_file = Io.schemes.file or error()
local Math = require "luan:Math.luan"
local min = Math.min or error()
local Swing = require "luan:swing/Swing.luan"
local new_frame = require("luan:swing/Frame.luan").new or error()
local new_label = require("luan:swing/Label.luan").new or error()
local new_text_area = require("luan:swing/Text_area.luan").new or error()
local new_scroll_pane = require("luan:swing/Scroll_pane.luan").new or error()
local new_text_area_line_numbers = require("luan:swing/TextAreaLineNumbers.luan").new or error()
local new_menu_bar = require("luan:swing/Menu_bar.luan").new or error()
local new_menu = require("luan:swing/Menu.luan").new or error()
local new_menu_item = require("luan:swing/Menu_item.luan").new or error()
local new_check_box_menu_item = require("luan:swing/Check_box_menu_item.luan").new or error()
local int_to_color = require("luan:swing/Color.luan").int_to_color or error()
local create_empty_border = require("luan:swing/Border.luan").create_empty_border or error()
local Logging = require "luan:logging/Logging.luan"
local logger = Logging.logger "editor"


local new_window

local function make_menu_bar(window)
	local menu_bar = new_menu_bar()
	do
		local file_menu = new_menu()
		file_menu.text = "File"
		do
			local new_file = new_menu_item()
			new_file.text = "New File"
			new_file.accelerator = "meta N"
			new_file.add_action_listener(new_window)
			file_menu.add(new_file)
		end
		do
			local open = new_menu_item()
			open.text = "Open..."
			open.accelerator = "meta O"
			open.add_action_listener(window.open)
			file_menu.add(open)
		end
		local revert
		do
			local save = new_menu_item()
			save.text = "Save"
			save.accelerator = "meta S"
			save.add_action_listener(function()
				if window.save() then
					revert.set_enabled(true)
				end
			end)
			file_menu.add(save)
		end
		do
			revert = new_menu_item()
			revert.text = "Revert"
			revert.set_enabled(window.has_file)
			revert.add_action_listener(window.revert)
			file_menu.add(revert)
		end
		menu_bar.add(file_menu)
	end
	do
		local edit_menu = new_menu()
		edit_menu.text = "Edit"
		local document = window.text_area.document
		local undo, redo
		do
			undo = new_menu_item()
			undo.text = "Undo"
			undo.accelerator = "meta Z"
			undo.add_action_listener(document.undo)
			edit_menu.add(undo)
		end
		do
			redo = new_menu_item()
			redo.text = "Redo"
			redo.accelerator = "meta shift Z"
			redo.add_action_listener(document.redo)
			edit_menu.add(redo)
		end
		local function update_undo_redo()
			undo.set_enabled(document.can_undo())
			redo.set_enabled(document.can_redo())
		end
		edit_menu.dont_gc(update_undo_redo)
		update_undo_redo()
		document.add_undo_listener(update_undo_redo)
		edit_menu.add_separator()
		do
			local cut = new_menu_item()
			cut.text = "Cut"
			cut.accelerator = "meta X"
			cut.add_action_listener(window.text_area.cut)
			edit_menu.add(cut)
		end
		do
			local copy = new_menu_item()
			copy.text = "Copy"
			copy.accelerator = "meta C"
			copy.add_action_listener(window.text_area.copy)
			edit_menu.add(copy)
		end
		do
			local paste = new_menu_item()
			paste.text = "Paste"
			paste.accelerator = "meta V"
			paste.add_action_listener(window.text_area.paste)
			edit_menu.add(paste)
		end
		edit_menu.add_separator()
		do
			local select_all = new_menu_item()
			select_all.text = "Select All"
			select_all.accelerator = "meta A"
			select_all.add_action_listener(window.text_area.select_all)
			edit_menu.add(select_all)
		end
		menu_bar.add(edit_menu)
	end
	do
		local view_menu = new_menu()
		view_menu.text = "View"
		do
			local word_wrap = new_check_box_menu_item()
			word_wrap.text = "Word Wrap"
			word_wrap.state = window.text_area.line_wrap
			word_wrap.add_action_listener(function()
				window.text_area.line_wrap = word_wrap.state
			end)
			view_menu.add(word_wrap)
		end
		do
			local show_whitespace = new_check_box_menu_item()
			show_whitespace.text = "Show Whitespace"
			show_whitespace.add_action_listener(function()
				window.text_area.show_whitespace(show_whitespace.state)
			end)
			view_menu.add(show_whitespace)
		end
		menu_bar.add(view_menu)
	end
	return menu_bar
end

local n_windows = 0
local documents = {}

function new_window(file)
	local window = {}
	window.has_file = file~=nil and file.is_file()
	local frame = new_frame()
	local title = file and file.canonical().to_string() or "new"
	frame.add_close_listener(function()
		n_windows = n_windows - 1
		if n_windows == 0 then
			Luan.exit()
		end
	end)
	local text_area = new_text_area()
	window.text_area = text_area
	if file ~= nil then
		local document = documents[title]
		if document == nil then
			documents[title] = text_area.document
		else
			text_area.document = document
		end
		if file.is_file() then
			text_area.text = file.read_text()
		end
	end
	local function set_title()
		local s = title
		if not text_area.document.is_unedited() then
			s = s.." *"
		end
		frame.title = s
	end
	set_title()
	text_area.dont_gc(set_title)
	text_area.document.add_undo_listener(set_title)
	text_area.rows = 10
	text_area.columns = 20
	text_area.wrap_style_word = true
	text_area.line_wrap = true
	text_area.tab_size = 4
	text_area.set_font{ family="Monospaced", size=13 }
	text_area.caret_position = 0
	--print(text_area.line_count)
	local scroll_pane = new_scroll_pane(text_area)
	local line_numbers = new_text_area_line_numbers(text_area)
	line_numbers.foreground_color = int_to_color(0x888888)
	line_numbers.border = create_empty_border(0,8,0,8)
	scroll_pane.set_row_header_view(line_numbers)
	frame.add(scroll_pane)
	function window.open()
		local file_chooser = frame.file_chooser_load()
		if file ~= nil then
			file_chooser.directory = file.parent()
		end
		file_chooser.visible = true
		local new_file = file_chooser.file
		if new_file ~= nil then
			new_window(new_file)
		end
	end
	function window.save()
		if file == nil then
			local file_chooser = frame.file_chooser_save()
			file_chooser.visible = true
			file = file_chooser.file
			if file == nil then
				return false
			end
			title = file.canonical().to_string()
			frame.title = title
			documents[title] = text_area.document
		end
		file.write_text(text_area.text)
		text_area.document.set_unedited()
		return true
	end
	function window.revert()
		local caret_position = text_area.caret_position
		local text = file.read_text()
		text_area.text = text
		text_area.caret_position = min(caret_position,#text)
		text_area.document.set_unedited()
	end
	local menu_bar = make_menu_bar(window)
	frame.set_menu_bar(menu_bar)
	frame.pack()
	frame.visible = true
	text_area.request_focus_in_window()
	n_windows = n_windows + 1
end

Swing.run(function()
	local args = Luan.arg
	if #args == 0 then
		new_window()
	else
		for _, arg in ipairs(args) do
			local file = new_file(arg)
			new_window(file)
		end
	end
end)