local gtable = require("gears.table") local gtimer = require("gears.timer") local wibox = require("wibox") local ipairs = ipairs local table = table local math = math local rofi_grid = { mt = {} } local properties = { "entries", "page", "lazy_load_widgets", "widget_template", "entry_template", "sort_fn", "search_fn", "search_sort_fn", "sort_alphabetically","reverse_sort_alphabetically,", "wrap_page_scrolling", "wrap_entry_scrolling" } local function build_properties(prototype, prop_names) for _, prop in ipairs(prop_names) do if not prototype["set_" .. prop] then prototype["set_" .. prop] = function(self, value) if self._private[prop] ~= value then self._private[prop] = value self:emit_signal("widget::redraw_needed") self:emit_signal("property::" .. prop, value) end return self end end if not prototype["get_" .. prop] then prototype["get_" .. prop] = function(self) return self._private[prop] end end end end local function scroll(self, dir, page_dir) local grid = self:get_grid() if #grid.children < 1 then self._private.selected_widget = nil return end local next_widget_index = nil local grid_orientation = grid:get_orientation() if dir == "up" then if grid_orientation == "horizontal" then next_widget_index = grid:index(self:get_selected_widget()) - 1 elseif grid_orientation == "vertical" then next_widget_index = grid:index(self:get_selected_widget()) - grid.forced_num_cols end elseif dir == "down" then if grid_orientation == "horizontal" then next_widget_index = grid:index(self:get_selected_widget()) + 1 elseif grid_orientation == "vertical" then next_widget_index = grid:index(self:get_selected_widget()) + grid.forced_num_cols end elseif dir == "left" then if grid_orientation == "horizontal" then next_widget_index = grid:index(self:get_selected_widget()) - grid.forced_num_rows elseif grid_orientation == "vertical" then next_widget_index = grid:index(self:get_selected_widget()) - 1 end elseif dir == "right" then if grid_orientation == "horizontal" then next_widget_index = grid:index(self:get_selected_widget()) + grid.forced_num_rows elseif grid_orientation == "vertical" then next_widget_index = grid:index(self:get_selected_widget()) + 1 end end local next_widget = grid.children[next_widget_index] if next_widget then next_widget:select() self:emit_signal("scroll", self:get_index_of_entry(next_widget.entry)) else if dir == "up" or dir == "left" then self:page_backward(page_dir or dir) elseif dir == "down" or dir == "right" then self:page_forward(page_dir or dir) end end end local function entry_widget(self, entry) if self._private.entries_widgets_cache[entry] then return self._private.entries_widgets_cache[entry] end local widget = self._private.entry_template(entry, self) local rofi_grid = self function widget:select() if rofi_grid:get_selected_widget() then rofi_grid:get_selected_widget():unselect() end rofi_grid._private.selected_widget = self self.selected = true local index = rofi_grid:get_index_of_entry(entry) self:emit_signal("select", index) rofi_grid:emit_signal("select", index) end function widget:unselect() self.selected = false rofi_grid._private.selected_widget = nil self:emit_signal("unselect") rofi_grid:emit_signal("unselect") end function widget:is_selected() return rofi_grid._private.selected_widget == self end function entry:select() widget:select() end function entry:unselect() widget:unselect() end function entry:is_selected() return widget:is_selected() end entry.widget = widget widget.entry = entry self._private.entries_widgets_cache[entry] = widget return widget end function rofi_grid:set_widget_template(widget_template) self._private.text_input = widget_template:get_children_by_id("text_input_role")[1] self._private.grid = widget_template:get_children_by_id("grid_role")[1] self._private.scrollbar = widget_template:get_children_by_id("scrollbar_role") if self._private.scrollbar then self._private.scrollbar = self._private.scrollbar[1] end widget_template:connect_signal("button::press", function(_, lx, ly, button, mods, find_widgets_result) if button == 4 then if self:get_grid():get_orientation() == "horizontal" then self:scroll_up() else self:scroll_left("up") end elseif button == 5 then if self:get_grid():get_orientation() == "horizontal" then self:scroll_down() else self:scroll_right("down") end end end) self:get_text_input():connect_signal("property::text", function(_, text) if text == self:get_text() then return end self._private.text = text self._private.search_timer:again() end) self:get_text_input():connect_signal("key::release", function(_, mod, key, cmd) if key == "Up" then self:scroll_up() end if key == "Down" then self:scroll_down() end if key == "Left" then self:scroll_left() end if key == "Right" then self:scroll_right() end end) local scrollbar = self:get_scrollbar() if scrollbar then self:connect_signal("scroll", function(self, new_index) scrollbar:set_value(new_index) end) self:connect_signal("page::forward", function(self, new_index) scrollbar:set_value(new_index) end) self:connect_signal("page::backward", function(self, new_index) scrollbar:set_value(new_index) end) self:connect_signal("search", function(self, text, new_index) scrollbar:set_maximum(math.max(2, #self:get_matched_entries())) if new_index then scrollbar:set_value(new_index) end end) self:connect_signal("select", function(self, new_index) scrollbar:set_value(new_index) end) scrollbar:connect_signal("property::value", function(_, value, instant) if instant ~= true then self:set_selected_entry(value) end end) end self._private.max_entries_per_page = self:get_grid().forced_num_cols * self:get_grid().forced_num_rows self._private.entries_per_page = self._private.max_entries_per_page self:set_widget(widget_template) end function rofi_grid:set_entries(entries, sort_fn) self._private.entries = entries self:set_sort_fn(sort_fn) self:reset() local scrollbar = self:get_scrollbar() if scrollbar then scrollbar:set_maximum(#self._private.entries) scrollbar:set_value(1) end if self._private.lazy_load_widgets == false then for _, entry in ipairs(self._private.entries) do self._private.entries_widgets_cache[entry] = entry_widget(self, entry) end end end function rofi_grid:add_entry(entry) table.insert(self._private.entries, entry) self:set_sort_fn() self:reset() end function rofi_grid:set_sort_fn(sort_fn) if sort_fn ~= nil then self._private.sort_fn = sort_fn end if self._private.sort_fn ~= nil then table.sort(self._private.entries, self._private.sort_fn) end end function rofi_grid:refresh() local max_entry_index_to_include = self._private.entries_per_page * self:get_current_page() local min_entry_index_to_include = max_entry_index_to_include - self._private.entries_per_page self:get_grid():reset() for index, entry in ipairs(self:get_matched_entries()) do -- Only add widgets that are between this range (part of the current page) if index > min_entry_index_to_include and index <= max_entry_index_to_include then self:get_grid():add(entry_widget(self, entry)) end end end function rofi_grid:search() local text = self:get_text() local old_pos = self:get_grid():get_widget_position(self:get_selected_widget()) -- Reset all the matched entrys self._private.matched_entries = {} -- Remove all the grid widgets self:get_grid():reset() if text == "" then self._private.matched_entries = self._private.entries else for _, entry in ipairs(self._private.entries) do text = text:gsub( "%W", "" ) if self._private.search_fn(text:lower(), entry) then table.insert(self:get_matched_entries(), entry) end end if self:get_search_sort_fn() then table.sort(self:get_matched_entries(), function(a, b) return self._private.search_sort_fn(text, a, b) end) end end for _, entry in ipairs(self._private.matched_entries) do -- Only add the widgets for entrys that are part of the first page if #self:get_grid().children + 1 <= self._private.max_entries_per_page then self:get_grid():add(entry_widget(self, entry)) end end -- Recalculate the entrys per page based on the current matched entrys self._private.entries_per_page = math.min(#self:get_matched_entries(), self._private.max_entries_per_page) -- Recalculate the pages count based on the current entrys per page self._private.pages_count = math.ceil(math.max(1, #self:get_matched_entries()) / math.max(1, self._private.entries_per_page)) -- Page should be 1 after a search self._private.current_page = 1 -- This is an option to mimic rofi behaviour where after a search -- it will reselect the entry whose index is the same as the entry index that was previously selected -- and if matched_entries.length < current_index it will instead select the entry with the greatest index if self._private.try_to_keep_index_after_searching then local entry_at_old_pos = self:get_grid():get_widgets_at(old_pos.row, old_pos.col) if entry_at_old_pos and entry_at_old_pos[1] then entry_at_old_pos[1]:select() else local widget = self:get_grid().children[#self:get_grid().children] widget:select() end -- Otherwise select the first entry on the list elseif self:get_grid().children[1] then local widget = self:get_grid().children[1] widget:select() end self:emit_signal("search", self:get_text(), self:get_index_of_entry(self:get_selected_widget().entry)) end function rofi_grid:set_selected_entry(index) local selected_widget_index = self:get_grid():index(self:get_selected_widget()) if index == selected_widget_index then return end local page = self:get_page_of_index(index) if self:get_current_page() ~= page then self:set_page(page) end self:get_entry_of_index(index).widget:select() end function rofi_grid:scroll_up(page_dir) scroll(self, "up", page_dir) end function rofi_grid:scroll_down(page_dir) scroll(self, "down", page_dir) end function rofi_grid:scroll_left(page_dir) scroll(self, "left", page_dir) end function rofi_grid:scroll_right(page_dir) scroll(self, "right", page_dir) end function rofi_grid:page_forward(dir) local min_entry_index_to_include = 0 local max_entry_index_to_include = self._private.entries_per_page if self:get_current_page() < self:get_pages_count() then min_entry_index_to_include = self._private.entries_per_page * self:get_current_page() self._private.current_page = self:get_current_page() + 1 max_entry_index_to_include = self._private.entries_per_page * self:get_current_page() elseif self._private.wrap_page_scrolling and #self:get_matched_entries() >= self._private.max_entries_per_page then self._private.current_page = 1 min_entry_index_to_include = 0 max_entry_index_to_include = self._private.entries_per_page elseif self._private.wrap_entry_scrolling then local widget = self:get_grid():get_widgets_at(1, 1)[1] widget:select() self:emit_signal("scroll", self:get_index_of_entry(widget.entry)) return else return end local pos = self:get_grid():get_widget_position(self:get_selected_widget()) -- Remove the current page entrys from the grid self:get_grid():reset() for index, entry in ipairs(self:get_matched_entries()) do -- Only add widgets that are between this range (part of the current page) if index > min_entry_index_to_include and index <= max_entry_index_to_include then self:get_grid():add(entry_widget(self, entry)) end end if self:get_current_page() > 1 or self._private.wrap_page_scrolling then local entry = nil if dir == "down" then entry = self:get_grid():get_widgets_at(1, 1)[1] elseif dir == "right" then entry = self:get_grid():get_widgets_at(pos.row, 1) if entry then entry = entry[1] end if entry == nil then entry = self:get_grid().children[#self:get_grid().children] end end entry:select() end self:emit_signal("page::forward", self:get_index_of_entry(self:get_selected_widget().entry)) end function rofi_grid:page_backward(dir) if self:get_current_page() > 1 then self._private.current_page = self:get_current_page() - 1 elseif self._private.wrap_page_scrolling and #self:get_matched_entries() >= self._private.max_entries_per_page then self._private.current_page = self:get_pages_count() elseif self._private.wrap_entry_scrolling then local widget = self:get_grid().children[#self:get_grid().children] widget:select() self:emit_signal("scroll", self:get_index_of_entry(widget.entry)) return else return end local pos = self:get_grid():get_widget_position(self:get_selected_widget()) -- Remove the current page entrys from the grid self:get_grid():reset() local max_entry_index_to_include = self._private.entries_per_page * self:get_current_page() local min_entry_index_to_include = max_entry_index_to_include - self._private.entries_per_page for index, entry in ipairs(self:get_matched_entries()) do -- Only add widgets that are between this range (part of the current page) if index > min_entry_index_to_include and index <= max_entry_index_to_include then self:get_grid():add(entry_widget(self, entry)) end end local entry = nil if self:get_current_page() < self:get_pages_count() then if dir == "up" then entry = self:get_grid().children[#self:get_grid().children] else -- Keep the same row from last page local _, columns = self:get_grid():get_dimension() entry = self:get_grid():get_widgets_at(pos.row, columns)[1] end elseif self._private.wrap_page_scrolling then entry = self:get_grid().children[#self:get_grid().children] end entry:select() self:emit_signal("page::backward", self:get_index_of_entry(self:get_selected_widget().entry)) end function rofi_grid:set_page(page) self:get_grid():reset() self._private.matched_entries = self._private.entries self._private.entries_per_page = self._private.max_entries_per_page self._private.pages_count = math.ceil(#self._private.entries / self._private.entries_per_page) self._private.current_page = page local max_entry_index_to_include = self._private.entries_per_page * self:get_current_page() local min_entry_index_to_include = max_entry_index_to_include - self._private.entries_per_page for index, entry in ipairs(self:get_matched_entries()) do -- Only add widgets that are between this range (part of the current page) if index > min_entry_index_to_include and index <= max_entry_index_to_include then self:get_grid():add(entry_widget(self, entry)) end end local entry = self:get_grid():get_widgets_at(1, 1) if entry then entry = entry[1] if entry then entry:select() end end end function rofi_grid:reset() self:get_grid():reset() self._private.matched_entries = self._private.entries self._private.entries_per_page = self._private.max_entries_per_page self._private.pages_count = math.ceil(#self._private.entries / self._private.entries_per_page) self._private.current_page = 1 for index, entry in ipairs(self._private.entries) do -- Only add the entrys that are part of the first page if index <= self._private.entries_per_page then self:get_grid():add(entry_widget(self, entry)) else break end end local entry = self:get_grid():get_widgets_at(1, 1) if entry then entry = entry[1] if entry then entry:select() end end self:get_text_input():set_text("") end function rofi_grid:get_scrollbar() return self._private.scrollbar end function rofi_grid:get_text_input() return self._private.text_input end function rofi_grid:get_grid() return self._private.grid end function rofi_grid:get_entries_per_page() return self._private.entries_per_page end function rofi_grid:get_pages_count() return self._private.pages_count end function rofi_grid:get_current_page() return self._private.current_page end function rofi_grid:get_entries() return self._private.entries end function rofi_grid:get_matched_entries() return self._private.matched_entries end function rofi_grid:get_text() return self._private.text end function rofi_grid:get_selected_widget() return self._private.selected_widget end function rofi_grid:get_page_of_entry(entry) return math.floor((self:get_index_of_entry(entry) - 1) / self._private.entries_per_page) + 1 end function rofi_grid:get_page_of_index(index) return math.floor((index - 1) / self._private.entries_per_page) + 1 end function rofi_grid:get_index_of_entry(entry) for index, matched_entry in ipairs(self:get_matched_entries()) do if matched_entry == entry then return index end end end function rofi_grid:get_entry_of_index(index) return self:get_matched_entries()[index] end local function new() local widget = wibox.container.background() gtable.crush(widget, rofi_grid, true) local wp = widget._private wp.entries_widgets_cache = setmetatable({}, { __mode = "v" }) wp.entries = {} wp.sort_fn = nil wp.sort_alphabetically = true wp.reverse_sort_alphabetically = false wp.try_to_keep_index_after_searching = false wp.wrap_page_scrolling = true wp.wrap_entry_scrolling = true wp.search_fn = nil wp.lazy_load_widgets = false wp.text = "" wp.pages_count = 0 wp.current_page = 1 wp.search_timer = gtimer { timeout = 0.35, call_now = false, autostart = false, single_shot = true, callback = function() widget:search() end } return widget end function rofi_grid.mt:__call(...) return new(...) end build_properties(rofi_grid, properties) return setmetatable(rofi_grid, rofi_grid.mt)