local gtable = require("gears.table") local gtimer = require("gears.timer") local helpers = require(tostring(...):match(".*bling") .. ".helpers") local fzy_has_match = helpers.fzy.has_match local fzy_score = helpers.fzy.score local wibox = require("wibox") local ipairs = ipairs local pairs = pairs local table = table local math = math local rofi_grid = { mt = {} } local properties = { "entries", "favorites", "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 has_value(tab, val) for _, value in ipairs(tab) do if val:lower():find(value:lower(), 1, true) then return true end end return false end local function has_entry(entries, name) for _, entry in ipairs(entries) do if entry.name == name then return true end end return false end local function scroll(self, dir, page_dir) local grid = self:get_grid() if #grid.children < 1 then self._private.selected_widget = nil self._private.selected_entry = 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("scroll") self:emit_signal("scroll", self:get_index_of_entry(self:get_selected_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(rofi_grid, entry) if rofi_grid._private.entries_widgets_cache[entry.name] then return rofi_grid._private.entries_widgets_cache[entry.name] end local widget = rofi_grid._private.entry_template(entry, rofi_grid) function widget:select(context) if rofi_grid:get_selected_widget() then rofi_grid:get_selected_widget():unselect() end rofi_grid._private.selected_widget = self rofi_grid._private.selected_entry = entry local index = rofi_grid:get_index_of_entry(entry) self:emit_signal("select", index, context) rofi_grid:emit_signal("select", index, context) end function widget:unselect() rofi_grid._private.selected_widget = nil rofi_grid._private.selected_entry = nil widget:emit_signal("unselect") rofi_grid:emit_signal("unselect") end function widget:is_selected() return rofi_grid._private.selected_widget == self end -- Should be overriden function widget:exec() end function widget:select_or_exec(context) if self:is_selected() then self:exec() else self:select(context) end end rofi_grid:emit_signal("entry_widget::add", widget, entry) rofi_grid._private.entries_widgets_cache[entry.name] = widget return rofi_grid._private.entries_widgets_cache[entry.name] end local function default_search_sort_fn(text, a, b) return fzy_score(text, a.name) > fzy_score(text, b.name) end local function default_search_fn(text, entry) if fzy_has_match(text, entry.name) then return true end return false end local function default_sort_fn(self, a, b) local is_a_favorite = has_value(self.favorites, a.id) local is_b_favorite = has_value(self.favorites, b.id) -- Sort the favorite apps first if is_a_favorite and not is_b_favorite then return true elseif not is_a_favorite and is_b_favorite then return false end -- Sort alphabetically if specified if self.sort_alphabetically then return a.name:lower() < b.name:lower() elseif self.reverse_sort_alphabetically then return b.name:lower() > a.name:lower() else return true end 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 function scrollbar:set_value(value, instant) value = math.min(value, self:get_maximum()) value = math.max(value, self:get_minimum()) local changed = self._private.value ~= value self._private.value = value if changed then self:emit_signal( "property::value", value, instant) self:emit_signal( "widget::redraw_needed" ) end end self:connect_signal("scroll", function(self, new_index) scrollbar:set_value(new_index, true) end) self:connect_signal("page::forward", function(self, new_index) scrollbar:set_value(new_index, true) end) self:connect_signal("page::backward", function(self, new_index) scrollbar:set_value(new_index, true) 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, true) end end) self:connect_signal("select", function(self, new_index) scrollbar:set_value(new_index, true) end) scrollbar:connect_signal("property::value", function(_, value, instant) if instant ~= true then self:scroll_to_index(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:add_entry(entry) table.insert(self._private.entries, entry) self:set_sort_fn() self:reset() end function rofi_grid:set_entries(new_entries, sort_fn) -- Remove old entries that are not in the new entries table for index, entry in pairs(self:get_entries()) do if has_entry(new_entries, entry.name) == false then table.remove(self._private.entries, index) if self._private.entries_widgets_cache[key] then self._private.entries_widgets_cache[key]:emit_signal("removed") self._private.entries_widgets_cache[key] = nil end end end -- Add new entries that are not in the old entries table for _, entry in ipairs(new_entries) do if has_entry(self:get_entries(), entry.name) == false then table.insert(self._private.entries, entry) if self:get_lazy_load_widgets() == false then self._private.entries_widgets_cache[entry.name] = entry_widget(self, entry) end end end self:set_sort_fn(sort_fn) self:reset() end function rofi_grid:set_favorites(favorites) self._private.favorites = favorites if self:get_entries() and #self:get_entries() > 1 then self:set_sort_fn() self:refresh() 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:reset() self:get_grid():reset() self._private.matched_entries = self:get_entries() self._private.entries_per_page = self._private.max_entries_per_page self._private.pages_count = math.ceil(#self:get_entries() / self._private.entries_per_page) self._private.current_page = 1 for index, entry in ipairs(self:get_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 widget = self:get_grid():get_widgets_at(1, 1) if widget then widget = widget[1] if widget then widget:select("new_page") end end local scrollbar = self:get_scrollbar() if scrollbar then if #self:get_grid().children <= 0 then self:get_scrollbar():set_visible(false) else self:get_scrollbar():set_visible(true) scrollbar:set_maximum(#self:get_entries()) scrollbar:set_value(1) end end self:get_text_input():set_text("") 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: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:get_entries() else for _, entry in ipairs(self:get_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 widget_at_old_pos = self:get_grid():get_widgets_at(old_pos.row, old_pos.col) if widget_at_old_pos and widget_at_old_pos[1] then widget_at_old_pos[1]:select("search") else local widget = self:get_grid().children[#self:get_grid().children] widget:select("search") 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("search") end if #self:get_grid().children <= 0 then self:get_scrollbar():set_visible(false) else self:get_scrollbar():set_visible(true) end self:emit_signal("search", self:get_text(), self:get_index_of_entry(self:get_selected_entry())) end function rofi_grid:scroll_to_index(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 local index_within_page = index - (page - 1) * self._private.entries_per_page self:get_grid().children[index_within_page]:select("scroll") 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("new_page") self:emit_signal("scroll", self:get_index_of_entry(self:get_selected_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 widget = nil if dir == "down" then widget = self:get_grid():get_widgets_at(1, 1)[1] elseif dir == "right" then widget = self:get_grid():get_widgets_at(pos.row, 1) if widget then widget = widget[1] end if widget == nil then widget = self:get_grid().children[#self:get_grid().children] end end widget:select("new_page") end self:emit_signal("page::forward", self:get_index_of_entry(self:get_selected_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("new_page") self:emit_signal("scroll", self:get_index_of_entry(self:get_selected_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 widget = nil if self:get_current_page() < self:get_pages_count() then if dir == "up" then widget = self:get_grid().children[#self:get_grid().children] else -- Keep the same row from last page local _, columns = self:get_grid():get_dimension() widget = self:get_grid():get_widgets_at(pos.row, columns)[1] end elseif self._private.wrap_page_scrolling then widget = self:get_grid().children[#self:get_grid().children] end widget:select("new_page") self:emit_signal("page::backward", self:get_index_of_entry(self:get_selected_entry())) end function rofi_grid:set_page(page) self:get_grid():reset() self._private.matched_entries = self:get_entries() self._private.entries_per_page = self._private.max_entries_per_page self._private.pages_count = math.ceil(#self:get_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 widget = self:get_grid():get_widgets_at(1, 1) if widget then widget = widget[1] if widget then widget:select("new_page") end end 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_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_selected_entry() return self._private.selected_entry 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.favorites = {} wp.sort_alphabetically = true wp.reverse_sort_alphabetically = false wp.sort_fn = function(a, b) return default_sort_fn(widget, a, b) end wp.search_fn = function(text, entry) return default_search_fn(text, entry) end wp.search_sort_fn = function(text, a, b) return default_search_sort_fn(text, a, b) end wp.try_to_keep_index_after_searching = false wp.wrap_page_scrolling = true wp.wrap_entry_scrolling = true wp.lazy_load_widgets = false wp.text = "" wp.pages_count = 0 wp.current_page = 1 wp.search_timer = gtimer { timeout = 0.05, 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)