Extract the scroller widget into it's own file + add scrollbar

This commit is contained in:
Ksaper 2023-03-12 12:12:49 +02:00
parent 4ba108df92
commit ba8071b13f
2 changed files with 335 additions and 649 deletions

View File

@ -6,6 +6,7 @@ local gtimer = require("gears.timer")
local wibox = require("wibox") local wibox = require("wibox")
local beautiful = require("beautiful") local beautiful = require("beautiful")
local text_input_widget = require(... .. ".text_input") local text_input_widget = require(... .. ".text_input")
local rofi_grid_widget = require(... .. ".rofi_grid")
local dpi = beautiful.xresources.apply_dpi local dpi = beautiful.xresources.apply_dpi
local string = string local string = string
local table = table local table = table
@ -37,64 +38,81 @@ local function has_value(tab, val)
return false return false
end end
local function scroll(self, dir, page_dir) local function build_widget(self)
local grid = self:get_grid() local widget_template = self.widget_template
if #grid.children < 1 then if widget_template == nil then
self._private.selected_app_widget = nil widget_template = wibox.widget
return {
end layout = rofi_grid_widget,
lazy_load_widgets = false,
local next_app_index = nil widget_template = wibox.widget {
local grid_orientation = grid:get_orientation() layout = wibox.layout.fixed.vertical,
forced_width = dpi(1000),
if dir == "up" then forced_height = dpi(1000),
if grid_orientation == "horizontal" then spacing = dpi(15),
next_app_index = grid:index(self:get_selected_app_widget()) - 1 {
elseif grid_orientation == "vertical" then widget = text_input_widget,
next_app_index = grid:index(self:get_selected_app_widget()) - grid.forced_num_cols id = "text_input_role",
end reset_on_stop = self.reset_on_hide,
elseif dir == "down" then placeholder = self.text_input_placeholder,
if grid_orientation == "horizontal" then unfocus_keys = { },
next_app_index = grid:index(self:get_selected_app_widget()) + 1 unfocus_on_clicked_inside = false,
elseif grid_orientation == "vertical" then unfocus_on_clicked_outside = false,
next_app_index = grid:index(self:get_selected_app_widget()) + grid.forced_num_cols unfocus_on_mouse_leave = false,
end unfocus_on_tag_change = false,
elseif dir == "left" then unfocus_on_other_text_input_focus = false,
if grid_orientation == "horizontal" then focus_on_subject_mouse_enter = nil,
next_app_index = grid:index(self:get_selected_app_widget()) - grid.forced_num_rows unfocus_on_subject_mouse_leave = nil,
elseif grid_orientation == "vertical" then widget_template = wibox.widget {
next_app_index = grid:index(self:get_selected_app_widget()) - 1 widget = wibox.container.background,
end forced_height = dpi(120),
elseif dir == "right" then bg = self.text_input_bg_color,
if grid_orientation == "horizontal" then {
next_app_index = grid:index(self:get_selected_app_widget()) + grid.forced_num_rows widget = wibox.container.margin,
elseif grid_orientation == "vertical" then margins = dpi(30),
next_app_index = grid:index(self:get_selected_app_widget()) + 1 {
end widget = wibox.widget.textbox,
end text_color = self.text_input_color,
id = "text_role"
local next_app = grid.children[next_app_index] }
if next_app then }
next_app:select() }
self:emit_signal("scroll", dir) },
else {
if dir == "up" or dir == "left" then layout = wibox.layout.fixed.horizontal,
self:page_backward(page_dir or dir) spacing = dpi(10),
elseif dir == "down" or dir == "right" then {
self:page_forward(page_dir or dir) layout = wibox.layout.grid,
end id = "grid_role",
end orientation = "horizontal",
end homogeneous = true,
spacing = dpi(30),
local function app_widget(self, app) forced_num_cols = self.apps_per_column,
if self._private.apps_widgets_cache[app.name] then forced_num_rows = self.apps_per_row,
return self._private.apps_widgets_cache[app.name] },
end {
layout = wibox.container.rotate,
local widget = nil direction = 'west',
{
if self.app_template == nil then widget = wibox.widget.slider,
widget = wibox.widget id = "scrollbar_role",
forced_width = dpi(5),
minimum = 1,
value = 1,
-- bar_shape = helpers.ui.rrect(),
bar_height= 3,
bar_color = beautiful.colors.transparent,
bar_active_color = beautiful.colors.transparent,
handle_width = dpi(50),
handle_color = beautiful.bg_normal,
-- handle_shape = helpers.ui.rrect(),
handle_color = beautiful.colors.on_background
}
}
}
},
entry_template = function(app)
local widget = wibox.widget
{ {
widget = wibox.container.background, widget = wibox.container.background,
forced_width = dpi(300), forced_width = dpi(300),
@ -146,116 +164,95 @@ local function app_widget(self, app)
end end
end) end)
widget:connect_signal("button::press", function(app, _, __, button) widget:connect_signal("button::press", function(_, __, __, button)
if button == 1 then if button == 1 then
if app:is_selected() or not self.select_before_spawn then if widget:is_selected() or not self.select_before_spawn then
app:run() widget:run()
else else
app:select() widget:select()
end end
end end
end) end)
else
widget = self.app_template(app, self)
end
local app_launcher = self widget:connect_signal("select", function()
function widget:run() widget.bg = self.app_selected_color
if app.terminal == true then local name_widget = widget:get_children_by_id("name_role")[1]
local pid = awful.spawn.with_shell(AWESOME_SENSIBLE_TERMINAL_SCRIPT_PATH .. " -e " .. app.exec) name_widget.markup = string.format("<span foreground='%s'>%s</span>", self.app_name_selected_color, name_widget.text)
local class = app.startup_wm_class or app.name end)
awful.spawn.with_shell(string.format(
[[xdotool search --sync --all --pid %s --name '.*' set_window --classname "%s" set_window --class "%s"]],
pid,
class,
class
))
else
app:launch()
end
if app_launcher.hide_on_launch then widget:connect_signal("unselect", function()
app_launcher:hide() widget.bg = self.app_normal_color
end local name_widget = widget:get_children_by_id("name_role")[1]
end name_widget.markup = string.format("<span foreground='%s'>%s</span>", self.app_name_normal_color, name_widget.text)
end)
function widget:run_or_select()
if self:is_selected() then
self:run()
else
self:select()
end
end
function widget:run_as_root()
if app.terminal == true then
local pid = awful.spawn.with_shell(
AWESOME_SENSIBLE_TERMINAL_SCRIPT_PATH .. " -e " ..
RUN_AS_ROOT_SCRIPT_PATH .. " " ..
app.exec
)
local class = app.startup_wm_class or app.name
awful.spawn.with_shell(string.format(
[[xdotool search --sync --all --pid %s --name '.*' set_window --classname "%s" set_window --class "%s"]],
pid,
class,
class
))
else
awful.spawn(RUN_AS_ROOT_SCRIPT_PATH .. " " .. app.exec)
end
if app_launcher.hide_on_launch then
app_launcher:hide()
end
end
function widget:select()
if app_launcher:get_selected_app_widget() then
app_launcher:get_selected_app_widget():unselect()
end
app_launcher._private.selected_app_widget = self
self:emit_signal("select")
self.selected = true
if app_launcher.app_template == nil then
widget.bg = app_launcher.app_selected_color
local name_widget = self:get_children_by_id("name_role")[1]
name_widget.markup = string.format("<span foreground='%s'>%s</span>", app_launcher.app_name_selected_color, name_widget.text)
end
end
function widget:unselect()
self:emit_signal("unselect")
self.selected = false
app_launcher._private.selected_app_widget = nil
if app_launcher.app_template == nil then
widget.bg = app_launcher.app_normal_color
local name_widget = self:get_children_by_id("name_role")[1]
name_widget.markup = string.format("<span foreground='%s'>%s</span>", app_launcher.app_name_normal_color, name_widget.text)
end
end
function widget:is_selected()
return app_launcher._private.selected_app_widget == self
end
function app:run() widget:run() end
function app:run_or_select() widget:run_or_select() end
function app:run_as_root() widget:run_as_root() end
function app:select() widget:select() end
function app:unselect() widget:unselect() end
function app:is_selected() widget:is_selected() end
self._private.apps_widgets_cache[app.name] = widget
return widget return widget
end
}
end
widget_template:set_search_fn(function(text, app)
local matched_apps = Gio.DesktopAppInfo.search(text:lower())
for _, matched_app in ipairs(matched_apps) do
for _, app_id in ipairs(matched_app) do
if app.id == app_id then
return true
end
end
end
end)
self._private.widget = awful.popup
{
screen = self.screen,
type = self.type,
visible = false,
ontop = true,
placement = self.placement,
border_width = self.border_width,
border_color = self.border_color,
shape = self.shape,
bg = self.bg,
widget = widget_template
}
self:get_text_input():connect_signal("key::press", function(_, mod, key, cmd)
if key == "Escape" then
self:hide()
end
end)
self:get_text_input():connect_signal("key::release", function(_, mod, key, cmd)
if key == "Return" then
if self:get_rofi_grid():get_selected_widget() ~= nil then
self:get_rofi_grid():get_selected_widget():run()
end
end
end)
end
local function default_sort_fn(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 end
local function generate_apps(self) local function generate_apps(self)
self._private.all_apps = {} local entries = {}
self._private.matched_apps = {}
local app_info = Gio.AppInfo local app_info = Gio.AppInfo
local apps = app_info.get_all() local apps = app_info.get_all()
@ -299,353 +296,71 @@ local function generate_apps(self)
app:launch() app:launch()
end end
} }
table.insert(self._private.all_apps, app) table.insert(entries, app)
if self.lazy_load_widgets == false then
self._private.apps_widgets_cache[app.name] = app_widget(self, app)
end
end end
end end
end end
end end
self:sort_apps() self:get_rofi_grid():set_entries(entries, self.sort_fn)
end end
local function build_widget(self) function app_launcher:run(widget)
local widget = self.widget_template if widget.entry.terminal == true then
if widget == nil then local pid = awful.spawn.with_shell(AWESOME_SENSIBLE_TERMINAL_SCRIPT_PATH .. " -e " .. widget.entry.exec)
self._private.text_input = wibox.widget local class = widget.entry.startup_wm_class or widget.entry.name
{ awful.spawn.with_shell(string.format(
widget = text_input_widget, [[xdotool search --sync --all --pid %s --name '.*' set_window --classname "%s" set_window --class "%s"]],
reset_on_stop = self.reset_on_hide, pid,
placeholder = self.text_input_placeholder, class,
widget_template = wibox.widget { class
widget = wibox.container.background, ))
forced_height = dpi(120),
bg = self.text_input_bg_color,
{
widget = wibox.container.margin,
margins = dpi(30),
{
widget = wibox.widget.textbox,
id = "text_role"
}
}
}
}
self._private.grid = wibox.widget
{
layout = wibox.layout.grid,
orientation = "horizontal",
homogeneous = true,
spacing = dpi(30),
forced_num_cols = self.apps_per_column,
forced_num_rows = self.apps_per_row,
}
widget = wibox.widget
{
layout = wibox.layout.fixed.vertical,
self._private.text_input,
{
widget = wibox.container.margin,
margins = dpi(30),
self._private.grid
}
}
else else
self._private.text_input = widget:get_children_by_id("text_input_role")[1] widget.entry:launch()
self._private.grid = widget:get_children_by_id("grid_role")[1]
end end
self._private.widget = awful.popup if self.hide_on_launch then
{ app_launcher:hide()
screen = self.screen,
type = self.type,
visible = false,
ontop = true,
placement = self.placement,
border_width = self.border_width,
border_color = self.border_color,
shape = self.shape,
bg = self.bg,
widget = widget
}
self:get_grid():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 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::press", function(_, mod, key, cmd)
if key == "Escape" then
self:hide()
end
end)
self:get_text_input():connect_signal("key::release", function(_, mod, key, cmd)
if key == "Return" then
if self:get_selected_app_widget() ~= nil then
self:get_selected_app_widget():run()
end
end
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)
self._private.max_apps_per_page = self:get_grid().forced_num_cols * self:get_grid().forced_num_rows
self._private.apps_per_page = self._private.max_apps_per_page
end end
function app_launcher:sort_apps(sort_fn) function app_launcher:run_or_select(widget)
table.sort(self._private.all_apps, sort_fn or self.sort_fn or function(a, b) if self:get_selected_widget() == widget then
local is_a_favorite = has_value(self.favorites, a.id) self:run(widget)
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 else
return true self:select(widget)
end
end
function app_launcher:run_as_root(widget)
if widget.entry.terminal == true then
local pid = awful.spawn.with_shell(
AWESOME_SENSIBLE_TERMINAL_SCRIPT_PATH .. " -e " ..
RUN_AS_ROOT_SCRIPT_PATH .. " " ..
widget.entry.exec
)
local class = widget.entry.startup_wm_class or widget.entry.name
awful.spawn.with_shell(string.format(
[[xdotool search --sync --all --pid %s --name '.*' set_window --classname "%s" set_window --class "%s"]],
pid,
class,
class
))
else
awful.spawn(RUN_AS_ROOT_SCRIPT_PATH .. " " .. widget.entry.exec)
end
if self.hide_on_launch then
app_launcher:hide()
end end
end)
end end
function app_launcher:set_favorites(favorites) function app_launcher:set_favorites(favorites)
self.favorites = favorites self.favorites = favorites
self:sort_apps() self:get_rofi_grid():set_sort_fn(self.sort_fn)
self:refresh() self:refresh()
end end
function app_launcher:refresh()
local max_app_index_to_include = self._private.apps_per_page * self:get_current_page()
local min_app_index_to_include = max_app_index_to_include - self._private.apps_per_page
self:get_grid():reset()
for index, app in ipairs(self._private.matched_apps) do
-- Only add widgets that are between this range (part of the current page)
if index > min_app_index_to_include and index <= max_app_index_to_include then
self:get_grid():add(app_widget(self, app))
end
end
end
function app_launcher:search()
local text = self:get_text()
local old_pos = self:get_grid():get_widget_position(self:get_selected_app_widget())
-- Reset all the matched apps
self._private.matched_apps = {}
-- Remove all the grid widgets
self:get_grid():reset()
if text == "" then
self._private.matched_apps = self._private.all_apps
for _, matched_app in ipairs(self._private.matched_apps) do
if #self:get_grid().children + 1 <= self._private.max_apps_per_page then
self:get_grid():add(app_widget(self, matched_app))
else
break
end
end
else
local matched_apps = Gio.DesktopAppInfo.search(text:lower())
for _, matched_app in ipairs(matched_apps) do
for _, app_id in ipairs(matched_app) do
for _, app in ipairs(self._private.all_apps) do
if app.id == app_id then
table.insert(self._private.matched_apps, app)
-- Only add the widgets for apps that are part of the first page
if #self:get_grid().children + 1 <= self._private.max_apps_per_page then
self:get_grid():add(app_widget(self, app))
end
end
end
end
end
end
-- Recalculate the apps per page based on the current matched apps
self._private.apps_per_page = math.min(#self._private.matched_apps, self._private.max_apps_per_page)
-- Recalculate the pages count based on the current apps per page
self._private.pages_count = math.ceil(math.max(1, #self._private.matched_apps) / math.max(1, self._private.apps_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 app whose index is the same as the app index that was previously selected
-- and if matched_apps.length < current_index it will instead select the app with the greatest index
if self.try_to_keep_index_after_searching then
if self:get_grid():get_widgets_at(old_pos.row, old_pos.col) == nil then
local app = self:get_grid().children[#self:get_grid().children]
app:select()
else
local app = self:get_grid():get_widgets_at(old_pos.row, old_pos.col)[1]
app:select()
end
-- Otherwise select the first app on the list
elseif #self:get_grid().children > 0 then
local app = self:get_grid():get_widgets_at(1, 1)[1]
app:select()
end
self:emit_signal("search", self:get_text(), self:get_current_page(), self:get_pages_count())
end
function app_launcher:scroll_up(page_dir)
scroll(self, "up", page_dir)
end
function app_launcher:scroll_down(page_dir)
scroll(self, "down", page_dir)
end
function app_launcher:scroll_left(page_dir)
scroll(self, "left", page_dir)
end
function app_launcher:scroll_right(page_dir)
scroll(self, "right", page_dir)
end
function app_launcher:page_forward(dir)
local min_app_index_to_include = 0
local max_app_index_to_include = self._private.apps_per_page
if self:get_current_page() < self:get_pages_count() then
min_app_index_to_include = self._private.apps_per_page * self:get_current_page()
self._private.current_page = self:get_current_page() + 1
max_app_index_to_include = self._private.apps_per_page * self:get_current_page()
elseif self.wrap_page_scrolling and #self._private.matched_apps >= self._private.max_apps_per_page then
self._private.current_page = 1
min_app_index_to_include = 0
max_app_index_to_include = self._private.apps_per_page
elseif self.wrap_app_scrolling then
local app = self:get_grid():get_widgets_at(1, 1)[1]
app:select()
return
else
return
end
local pos = self:get_grid():get_widget_position(self:get_selected_app_widget())
-- Remove the current page apps from the grid
self:get_grid():reset()
for index, app in ipairs(self._private.matched_apps) do
-- Only add widgets that are between this range (part of the current page)
if index > min_app_index_to_include and index <= max_app_index_to_include then
self:get_grid():add(app_widget(self, app))
end
end
if self:get_current_page() > 1 or self.wrap_page_scrolling then
local app = nil
if dir == "down" then
app = self:get_grid():get_widgets_at(1, 1)[1]
elseif dir == "right" then
app = self:get_grid():get_widgets_at(pos.row, 1)
if app then
app = app[1]
end
if app == nil then
app = self:get_grid().children[#self:get_grid().children]
end
end
app:select()
end
self:emit_signal("page::forward", dir, self:get_current_page(), self:get_pages_count())
end
function app_launcher:page_backward(dir)
if self:get_current_page() > 1 then
self._private.current_page = self:get_current_page() - 1
elseif self.wrap_page_scrolling and #self._private.matched_apps >= self._private.max_apps_per_page then
self._private.current_page = self:get_pages_count()
elseif self.wrap_app_scrolling then
local app = self:get_grid().children[#self:get_grid().children]
app:select()
return
else
return
end
local pos = self:get_grid():get_widget_position(self:get_selected_app_widget())
-- Remove the current page apps from the grid
self:get_grid():reset()
local max_app_index_to_include = self._private.apps_per_page * self:get_current_page()
local min_app_index_to_include = max_app_index_to_include - self._private.apps_per_page
for index, app in ipairs(self._private.matched_apps) do
-- Only add widgets that are between this range (part of the current page)
if index > min_app_index_to_include and index <= max_app_index_to_include then
self:get_grid():add(app_widget(self, app))
end
end
local app = nil
if self:get_current_page() < self:get_pages_count() then
if dir == "up" then
app = self:get_grid().children[#self:get_grid().children]
else
-- Keep the same row from last page
local _, columns = self:get_grid():get_dimension()
app = self:get_grid():get_widgets_at(pos.row, columns)[1]
end
elseif self.wrap_page_scrolling then
app = self:get_grid().children[#self:get_grid().children]
end
app:select()
self:emit_signal("page::backward", dir, self:get_current_page(), self:get_pages_count())
end
function app_launcher:show() function app_launcher:show()
if self.show_on_focused_screen then if self.show_on_focused_screen then
self:get_widget().screen = awful.screen.focused() self:get_widget().screen = awful.screen.focused()
@ -662,7 +377,7 @@ function app_launcher:hide()
end end
if self.reset_on_hide == true then if self.reset_on_hide == true then
self:reset() self:get_rofi_grid():reset()
end end
self:get_widget().visible = false self:get_widget().visible = false
@ -678,54 +393,16 @@ function app_launcher:toggle()
end end
end end
function app_launcher:reset()
self:get_grid():reset()
self._private.matched_apps = self._private.all_apps
self._private.apps_per_page = self._private.max_apps_per_page
self._private.pages_count = math.ceil(#self._private.all_apps / self._private.apps_per_page)
self._private.current_page = 1
for index, app in ipairs(self._private.all_apps) do
-- Only add the apps that are part of the first page
if index <= self._private.apps_per_page then
self:get_grid():add(app_widget(self, app))
else
break
end
end
local app = self:get_grid():get_widgets_at(1, 1)[1]
app:select()
self:get_text_input():set_text("")
end
function app_launcher:get_widget() function app_launcher:get_widget()
return self._private.widget return self._private.widget
end end
function app_launcher:get_rofi_grid()
return self:get_widget().widget
end
function app_launcher:get_text_input() function app_launcher:get_text_input()
return self._private.text_input return self:get_rofi_grid():get_text_input()
end
function app_launcher:get_grid()
return self._private.grid
end
function app_launcher:get_pages_count()
return self._private.pages_count
end
function app_launcher:get_current_page()
return self._private.current_page
end
function app_launcher:get_text()
return self._private.text
end
function app_launcher:get_selected_app_widget()
return self._private.selected_app_widget
end end
local function new(args) local function new(args)
@ -766,6 +443,7 @@ local function new(args)
args.apps_per_column = default_value(args.apps_per_column, 3) args.apps_per_column = default_value(args.apps_per_column, 3)
args.text_input_bg_color = default_value(args.text_input_bg_color, "#000000") args.text_input_bg_color = default_value(args.text_input_bg_color, "#000000")
args.text_input_color = default_value(args.text_input_bg_color, "#FFFFFF")
args.text_input_placeholder = default_value(args.text_input_placeholder, "Search: ") args.text_input_placeholder = default_value(args.text_input_placeholder, "Search: ")
args.app_normal_color = default_value(args.app_normal_color, "#000000") args.app_normal_color = default_value(args.app_normal_color, "#000000")
@ -781,7 +459,6 @@ local function new(args)
ret._private.text = "" ret._private.text = ""
ret._private.pages_count = 0 ret._private.pages_count = 0
ret._private.current_page = 1 ret._private.current_page = 1
ret._private.apps_widgets_cache = setmetatable({}, { __mode = "v" })
ret._private.search_timer = gtimer { ret._private.search_timer = gtimer {
timeout = 0.05, timeout = 0.05,
call_now = false, call_now = false,
@ -827,7 +504,6 @@ local function new(args)
build_widget(ret) build_widget(ret)
generate_apps(ret) generate_apps(ret)
ret:reset()
return ret return ret
end end

View File

@ -74,7 +74,7 @@ local function scroll(self, dir, page_dir)
local next_widget = grid.children[next_widget_index] local next_widget = grid.children[next_widget_index]
if next_widget then if next_widget then
next_widget:select() next_widget:select()
self:emit_signal("scroll", self:get_index_of_widget(next_widget)) self:emit_signal("scroll", self:get_index_of_entry(next_widget.entry))
else else
if dir == "up" or dir == "left" then if dir == "up" or dir == "left" then
self:page_backward(page_dir or dir) self:page_backward(page_dir or dir)
@ -93,17 +93,37 @@ local function entry_widget(self, entry)
local rofi_grid = self local rofi_grid = self
function widget:select() function widget:select()
rofi_grid:select_widget(self) 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 end
function widget:is_selected() function widget:is_selected()
return rofi_grid:get_selected_widget() == self return rofi_grid._private.selected_widget == self
end end
function entry:select() widget:select() end
function entry:unselect() widget:unselect() end
function entry:is_selected() widget:is_selected() end
entry.widget = widget entry.widget = widget
widget.entry = entry widget.entry = entry
self._private.entries_widgets_cache[entry] = widget self._private.entries_widgets_cache[entry] = widget
return widget return widget
end end
@ -182,7 +202,7 @@ function rofi_grid:set_widget_template(widget_template)
scrollbar:connect_signal("property::value", function(_, value, instant) scrollbar:connect_signal("property::value", function(_, value, instant)
if instant ~= true then if instant ~= true then
self:select_widget_by_index(value) self:set_selected_entry(value)
end end
end) end)
end end
@ -253,12 +273,12 @@ function rofi_grid:search()
else else
for _, entry in ipairs(self._private.entries) do for _, entry in ipairs(self._private.entries) do
text = text:gsub( "%W", "" ) text = text:gsub( "%W", "" )
if self.search_fn(text:lower(), entry, self:get_entries()) then if self._private.search_fn(text:lower(), entry) then
table.insert(self:get_matched_entries(), entry) table.insert(self:get_matched_entries(), entry)
end end
end end
if self:get_sort_fn() then if self:get_search_sort_fn() then
table.sort(self:get_matched_entries(), function(a, b) table.sort(self:get_matched_entries(), function(a, b)
return self._private.search_sort_fn(text, a, b) return self._private.search_sort_fn(text, a, b)
end) end)
@ -284,9 +304,9 @@ function rofi_grid:search()
-- it will reselect the entry whose index is the same as the entry index that was previously selected -- 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 -- 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 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) local entry_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 if entry_at_old_pos and entry_at_old_pos[1] then
widget_at_old_pos[1]:select() entry_at_old_pos[1]:select()
else else
local widget = self:get_grid().children[#self:get_grid().children] local widget = self:get_grid().children[#self:get_grid().children]
widget:select() widget:select()
@ -297,35 +317,21 @@ function rofi_grid:search()
widget:select() widget:select()
end end
self:emit_signal("search", self:get_text(), self:get_index_of_widget(self:get_selected_widget())) self:emit_signal("search", self:get_text(), self:get_index_of_entry(self:get_selected_widget().entry))
end end
function rofi_grid:select_widget(widget) function rofi_grid:set_selected_entry(index)
print(widget) local selected_widget_index = self:get_grid():index(self:get_selected_widget())
self:select_widget_by_index(self:get_index_of_widget(widget)) if index == selected_widget_index then
end return
function rofi_grid:select_widget_by_index(index)
local previous_widget = self:get_selected_widget()
if previous_widget then
self._private.selected_widget = nil
self:emit_signal("unselect")
previous_widget:emit_signal("unselect")
end end
print(index) local page = self:get_page_of_index(index)
if self:get_current_page() ~= page then
self:set_page(page)
end
-- local new_widget = self:get_widget_of_index(index) self:get_entry_of_index(index).widget:select()
-- if new_widget then
-- local page = self:get_page_of_index(index)
-- if self:get_current_page() ~= page then
-- self:set_page(page)
-- end
-- self._private.selected_widget = new_widget
-- self:emit_signal("select", index)
-- new_widget:emit_signal("select", index)
-- end
end end
function rofi_grid:scroll_up(page_dir) function rofi_grid:scroll_up(page_dir)
@ -359,7 +365,7 @@ function rofi_grid:page_forward(dir)
elseif self._private.wrap_entry_scrolling then elseif self._private.wrap_entry_scrolling then
local widget = self:get_grid():get_widgets_at(1, 1)[1] local widget = self:get_grid():get_widgets_at(1, 1)[1]
widget:select() widget:select()
self:emit_signal("scroll", self:get_index_of_widget(widget)) self:emit_signal("scroll", self:get_index_of_entry(widget.entry))
return return
else else
return return
@ -378,22 +384,22 @@ function rofi_grid:page_forward(dir)
end end
if self:get_current_page() > 1 or self._private.wrap_page_scrolling then if self:get_current_page() > 1 or self._private.wrap_page_scrolling then
local widget = nil local entry = nil
if dir == "down" then if dir == "down" then
widget = self:get_grid():get_widgets_at(1, 1)[1] entry = self:get_grid():get_widgets_at(1, 1)[1]
elseif dir == "right" then elseif dir == "right" then
widget = self:get_grid():get_widgets_at(pos.row, 1) entry = self:get_grid():get_widgets_at(pos.row, 1)
if widget then if entry then
widget = widget[1] entry = entry[1]
end end
if widget == nil then if entry == nil then
widget = self:get_grid().children[#self:get_grid().children] entry = self:get_grid().children[#self:get_grid().children]
end end
end end
widget:select() entry:select()
end end
self:emit_signal("page::forward", self:get_index_of_widget(self:get_selected_widget())) self:emit_signal("page::forward", self:get_index_of_entry(self:get_selected_widget().entry))
end end
function rofi_grid:page_backward(dir) function rofi_grid:page_backward(dir)
@ -404,7 +410,7 @@ function rofi_grid:page_backward(dir)
elseif self._private.wrap_entry_scrolling then elseif self._private.wrap_entry_scrolling then
local widget = self:get_grid().children[#self:get_grid().children] local widget = self:get_grid().children[#self:get_grid().children]
widget:select() widget:select()
self:emit_signal("scroll", self:get_index_of_widget(widget)) self:emit_signal("scroll", self:get_index_of_entry(widget.entry))
return return
else else
return return
@ -425,21 +431,21 @@ function rofi_grid:page_backward(dir)
end end
end end
local widget = nil local entry = nil
if self:get_current_page() < self:get_pages_count() then if self:get_current_page() < self:get_pages_count() then
if dir == "up" then if dir == "up" then
widget = self:get_grid().children[#self:get_grid().children] entry = self:get_grid().children[#self:get_grid().children]
else else
-- Keep the same row from last page -- Keep the same row from last page
local _, columns = self:get_grid():get_dimension() local _, columns = self:get_grid():get_dimension()
widget = self:get_grid():get_widgets_at(pos.row, columns)[1] entry = self:get_grid():get_widgets_at(pos.row, columns)[1]
end end
elseif self._private.wrap_page_scrolling then elseif self._private.wrap_page_scrolling then
widget = self:get_grid().children[#self:get_grid().children] entry = self:get_grid().children[#self:get_grid().children]
end end
widget:select() entry:select()
self:emit_signal("page::backward", self:get_index_of_widget(self:get_selected_widget())) self:emit_signal("page::backward", self:get_index_of_entry(self:get_selected_widget().entry))
end end
function rofi_grid:set_page(page) function rofi_grid:set_page(page)
@ -459,11 +465,11 @@ function rofi_grid:set_page(page)
end end
end end
local widget = self:get_grid():get_widgets_at(1, 1) local entry = self:get_grid():get_widgets_at(1, 1)
if widget then if entry then
widget = widget[1] entry = entry[1]
if widget then if entry then
widget:select() entry:select()
end end
end end
end end
@ -484,11 +490,11 @@ function rofi_grid:reset()
end end
end end
local widget = self:get_grid():get_widgets_at(1, 1) local entry = self:get_grid():get_widgets_at(1, 1)
if widget then if entry then
widget = widget[1] entry = entry[1]
if widget then if entry then
widget:select() entry:select()
end end
end end
@ -527,22 +533,6 @@ function rofi_grid:get_matched_entries()
return self._private.matched_entries return self._private.matched_entries
end 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_widget(widget)
for index, matched_widget in ipairs(self:get_matched_entries()) do
if matched_widget.entry == widget.entry then
return index
end
end
end
function rofi_grid:get_widget_of_index(index)
return self:get_matched_entries()[index].widget
end
function rofi_grid:get_text() function rofi_grid:get_text()
return self._private.text return self._private.text
end end
@ -551,6 +541,26 @@ function rofi_grid:get_selected_widget()
return self._private.selected_widget return self._private.selected_widget
end 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 function new()
local widget = wibox.container.background() local widget = wibox.container.background()
gtable.crush(widget, rofi_grid, true) gtable.crush(widget, rofi_grid, true)