[jira] improve the menu list
This commit is contained in:
parent
ddff9577fc
commit
7ea3eec179
|
@ -1,8 +1,10 @@
|
||||||
# Jira widget
|
# Jira widget
|
||||||
|
|
||||||
The widget shows the number of tickets assigned to the user and when clicked shows them in the list with some additional information. When item in the list is clicked - it opens the issue in browser.
|
The widget shows the number of tickets assigned to the user (or any other result of a JQL query, see customization section) and when clicked shows them in the list, grouped by the ticket status. Left-click on the item opens the issue in the default browser:
|
||||||
|
|
||||||
![git](./out.gif)
|
<p align="center">
|
||||||
|
<img alt="screenshot" src="https://raw.githubusercontent.com/streetturtle/awesome-wm-widgets/master/jira-widget/screenshot/screenshot.png"/>
|
||||||
|
</p>
|
||||||
|
|
||||||
## How it works
|
## How it works
|
||||||
|
|
||||||
|
@ -19,7 +21,7 @@ It is possible to customize widget by providing a table with all or some of the
|
||||||
| `host` | Required | Ex: _http://jira.tmnt.com_ |
|
| `host` | Required | Ex: _http://jira.tmnt.com_ |
|
||||||
| `query` | `jql=assignee=currentuser() AND resolution=Unresolved` | JQL query |
|
| `query` | `jql=assignee=currentuser() AND resolution=Unresolved` | JQL query |
|
||||||
| `icon` | `~/.config/awesome/awesome-wm-widgets/jira-widget/jira-mark-gradient-blue.svg` | Path to the icon |
|
| `icon` | `~/.config/awesome/awesome-wm-widgets/jira-widget/jira-mark-gradient-blue.svg` | Path to the icon |
|
||||||
| `timeout` | 10 | How often in seconds the widget refreshes |
|
| `timeout` | 600 | How often in seconds the widget refreshes |
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
|
@ -36,13 +38,13 @@ Then change file's permissions to 600 (so only you can read/write it):
|
||||||
```bash
|
```bash
|
||||||
chmod 600 ~/.netrc
|
chmod 600 ~/.netrc
|
||||||
```
|
```
|
||||||
And test if it works by calling the API:
|
And test if it works by calling the API (`-n` option is to use the .netrc file for authentication):
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
curl -s -n 'https://turtleninja.com/rest/api/2/search?jql=assignee=currentuser()+AND+resolution=Unresolved'
|
curl -n 'https://turtleninja.com/rest/api/2/search?jql=assignee=currentuser()+AND+resolution=Unresolved'
|
||||||
```
|
```
|
||||||
|
|
||||||
Clone/download repo and use widget in **rc.lua**:
|
Clone/download repo and use the widget in **rc.lua**:
|
||||||
|
|
||||||
```lua
|
```lua
|
||||||
local jira_widget = require("awesome-wm-widgets.jira-widget.jira")
|
local jira_widget = require("awesome-wm-widgets.jira-widget.jira")
|
||||||
|
|
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.1 KiB |
|
@ -5,7 +5,7 @@
|
||||||
-- https://github.com/streetturtle/awesome-wm-widgets/tree/master/jira-widget
|
-- https://github.com/streetturtle/awesome-wm-widgets/tree/master/jira-widget
|
||||||
|
|
||||||
-- @author Pavel Makhov
|
-- @author Pavel Makhov
|
||||||
-- @copyright 2019 Pavel Makhov
|
-- @copyright 2020 Pavel Makhov
|
||||||
-------------------------------------------------
|
-------------------------------------------------
|
||||||
|
|
||||||
local awful = require("awful")
|
local awful = require("awful")
|
||||||
|
@ -36,33 +36,40 @@ local jira_widget = wibox.widget {
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
id = 'c',
|
{
|
||||||
widget = wibox.widget.imagebox
|
id = 'c',
|
||||||
|
widget = wibox.widget.imagebox
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id = 'd',
|
||||||
|
draw = function(_, _, cr, width, height)
|
||||||
|
cr:set_source(color(beautiful.fg_urgent))
|
||||||
|
cr:arc(width - height / 6, height / 6, height / 6, 0, math.pi * 2)
|
||||||
|
cr:fill()
|
||||||
|
end,
|
||||||
|
visible = false,
|
||||||
|
layout = wibox.widget.base.make_widget,
|
||||||
|
},
|
||||||
|
id = 'b',
|
||||||
|
layout = wibox.layout.stack
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id = 'd',
|
id = "txt",
|
||||||
draw = function(_, _, cr, width, height)
|
widget = wibox.widget.textbox
|
||||||
cr:set_source(color(beautiful.fg_urgent))
|
|
||||||
cr:arc(width - height/6, height/6, height/6, 0, math.pi*2)
|
|
||||||
cr:fill()
|
|
||||||
end,
|
|
||||||
visible = false,
|
|
||||||
layout = wibox.widget.base.make_widget,
|
|
||||||
},
|
},
|
||||||
id = 'b',
|
spacing = 4,
|
||||||
layout = wibox.layout.stack
|
layout = wibox.layout.fixed.horizontal,
|
||||||
},
|
},
|
||||||
id = 'a',
|
|
||||||
margins = 4,
|
margins = 4,
|
||||||
layout = wibox.container.margin
|
layout = wibox.container.margin
|
||||||
},
|
},
|
||||||
{
|
shape = function(cr, width, height)
|
||||||
id = "txt",
|
gears.shape.rounded_rect(cr, width, height, 4)
|
||||||
widget = wibox.widget.textbox
|
end,
|
||||||
},
|
widget = wibox.container.background,
|
||||||
layout = wibox.layout.fixed.horizontal,
|
|
||||||
set_text = function(self, new_value)
|
set_text = function(self, new_value)
|
||||||
self.txt.text = new_value
|
self:get_children_by_id('txt')[1]:set_text(new_value)
|
||||||
|
--self.txt.text = new_value
|
||||||
end,
|
end,
|
||||||
set_icon = function(self, path)
|
set_icon = function(self, path)
|
||||||
self:get_children_by_id('c')[1]:set_image(path)
|
self:get_children_by_id('c')[1]:set_image(path)
|
||||||
|
@ -73,7 +80,8 @@ local jira_widget = wibox.widget {
|
||||||
self:get_children_by_id('c')[1]:set_opacity(1)
|
self:get_children_by_id('c')[1]:set_opacity(1)
|
||||||
self:get_children_by_id('c')[1]:emit_signal('widget:redraw_needed')
|
self:get_children_by_id('c')[1]:emit_signal('widget:redraw_needed')
|
||||||
else
|
else
|
||||||
self.txt:set_text('')
|
--self.txt:set_text('')
|
||||||
|
self:get_children_by_id('txt')[1]:set_text('')
|
||||||
self:get_children_by_id('d')[1]:set_visible(true)
|
self:get_children_by_id('d')[1]:set_visible(true)
|
||||||
self:get_children_by_id('c')[1]:set_opacity(0.2)
|
self:get_children_by_id('c')[1]:set_opacity(0.2)
|
||||||
self:get_children_by_id('c')[1]:emit_signal('widget:redraw_needed')
|
self:get_children_by_id('c')[1]:emit_signal('widget:redraw_needed')
|
||||||
|
@ -104,13 +112,20 @@ local function worker(user_args)
|
||||||
|
|
||||||
local args = user_args or {}
|
local args = user_args or {}
|
||||||
|
|
||||||
local icon = args.icon or HOME_DIR .. '/.config/awesome/awesome-wm-widgets/jira-widget/jira-mark-gradient-blue.svg'
|
local icon = args.icon or HOME_DIR .. '/.config/awesome/awesome-wm-widgets/jira-widget/icon/jira-mark-gradient-blue.svg'
|
||||||
local host = args.host or show_warning('Jira host is unknown')
|
local host = args.host or show_warning('Jira host is unknown')
|
||||||
local query = args.query or 'jql=assignee=currentuser() AND resolution=Unresolved'
|
local query = args.query or 'jql=assignee=currentuser() AND resolution=Unresolved'
|
||||||
local timeout = args.timeout or 10
|
local timeout = args.timeout or 600
|
||||||
|
|
||||||
jira_widget:set_icon(icon)
|
jira_widget:set_icon(icon)
|
||||||
|
|
||||||
|
local separator_widget = {
|
||||||
|
orientation = 'horizontal',
|
||||||
|
forced_height = 1,
|
||||||
|
color = beautiful.bg_focus,
|
||||||
|
widget = wibox.widget.separator
|
||||||
|
}
|
||||||
|
|
||||||
local update_widget = function(widget, stdout, stderr, _, _)
|
local update_widget = function(widget, stdout, stderr, _, _)
|
||||||
if stderr ~= '' then
|
if stderr ~= '' then
|
||||||
if not warning_shown then
|
if not warning_shown then
|
||||||
|
@ -142,23 +157,46 @@ local function worker(user_args)
|
||||||
widget:set_visible(true)
|
widget:set_visible(true)
|
||||||
widget:set_text(number_of_issues)
|
widget:set_text(number_of_issues)
|
||||||
|
|
||||||
local rows = {
|
local rows = { layout = wibox.layout.fixed.vertical }
|
||||||
{ widget = wibox.widget.textbox },
|
|
||||||
layout = wibox.layout.fixed.vertical,
|
|
||||||
}
|
|
||||||
|
|
||||||
for i = 0, #rows do rows[i]=nil end
|
for i = 0, #rows do rows[i]=nil end
|
||||||
|
|
||||||
|
-- sort issues based on the status
|
||||||
|
table.sort(result.issues, function(a,b) return a.fields.status.name > b.fields.status.name end)
|
||||||
|
|
||||||
|
local cur_status = ''
|
||||||
for _, issue in ipairs(result.issues) do
|
for _, issue in ipairs(result.issues) do
|
||||||
local path_to_avatar = HOME_DIR ..'/.cache/awmw/jira-widget/avatars/' .. issue.fields.assignee.accountId
|
local path_to_avatar = HOME_DIR ..'/.cache/awmw/jira-widget/avatars/' .. issue.fields.assignee.name
|
||||||
|
|
||||||
if not gfs.file_readable(path_to_avatar) then
|
if not gfs.file_readable(path_to_avatar) then
|
||||||
spawn.easy_async(string.format(
|
spawn.easy_async(string.format(
|
||||||
DOWNLOAD_AVATAR_CMD,
|
DOWNLOAD_AVATAR_CMD,
|
||||||
HOME_DIR,
|
HOME_DIR,
|
||||||
issue.fields.assignee.accountId,
|
issue.fields.assignee.name,
|
||||||
issue.fields.assignee.avatarUrls['48x48']))
|
issue.fields.assignee.avatarUrls['48x48']))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if (cur_status ~= issue.fields.status.name) then
|
||||||
|
-- do not insert separator before first item
|
||||||
|
if (cur_status ~= '') then
|
||||||
|
table.insert(rows, separator_widget)
|
||||||
|
end
|
||||||
|
|
||||||
|
table.insert(rows, wibox.widget {
|
||||||
|
{
|
||||||
|
{
|
||||||
|
markup = "<span foreground='#888888'>" .. issue.fields.status.name .. "</span>",
|
||||||
|
widget = wibox.widget.textbox,
|
||||||
|
},
|
||||||
|
left = 8,
|
||||||
|
layout = wibox.container.margin
|
||||||
|
},
|
||||||
|
bg = beautiful.bg_normal,
|
||||||
|
widget = wibox.container.background
|
||||||
|
})
|
||||||
|
cur_status = issue.fields.status.name
|
||||||
|
end
|
||||||
|
|
||||||
local row = wibox.widget {
|
local row = wibox.widget {
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
|
@ -170,44 +208,63 @@ local function worker(user_args)
|
||||||
forced_height = 40,
|
forced_height = 40,
|
||||||
widget = wibox.widget.imagebox
|
widget = wibox.widget.imagebox
|
||||||
},
|
},
|
||||||
margins = 8,
|
left = 4,
|
||||||
layout = wibox.container.margin
|
layout = wibox.container.margin
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
markup = '<b>' .. issue.key .. '</b>',
|
markup = '<b>' .. issue.fields.summary .. '</b>',
|
||||||
align = 'center',
|
|
||||||
forced_width = 350, -- for horizontal alignment
|
|
||||||
widget = wibox.widget.textbox
|
widget = wibox.widget.textbox
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
text = issue.fields.summary,
|
{
|
||||||
widget = wibox.widget.textbox
|
markup = "<span foreground='#888888'>" .. issue.key .. "</span>",
|
||||||
},
|
widget = wibox.widget.textbox
|
||||||
{
|
},
|
||||||
text = issue.fields.status.name,
|
{
|
||||||
widget = wibox.widget.textbox
|
markup = "<span foreground='#888888'>" .. issue.fields.assignee.displayName .. "</span>",
|
||||||
|
widget = wibox.widget.textbox
|
||||||
|
},
|
||||||
|
spacing = 8,
|
||||||
|
layout = wibox.layout.fixed.horizontal
|
||||||
},
|
},
|
||||||
layout = wibox.layout.align.vertical
|
layout = wibox.layout.align.vertical
|
||||||
},
|
},
|
||||||
spacing = 8,
|
spacing = 8,
|
||||||
layout = wibox.layout.fixed.horizontal
|
layout = wibox.layout.fixed.horizontal
|
||||||
},
|
},
|
||||||
margins = 8,
|
margins = 4,
|
||||||
layout = wibox.container.margin
|
layout = wibox.container.margin
|
||||||
},
|
},
|
||||||
bg = beautiful.bg_normal,
|
bg = beautiful.bg_normal,
|
||||||
widget = wibox.container.background
|
widget = wibox.container.background
|
||||||
}
|
}
|
||||||
|
|
||||||
row:connect_signal("mouse::enter", function(c) c:set_bg(beautiful.bg_focus) end)
|
local old_cursor, old_wibox
|
||||||
row:connect_signal("mouse::leave", function(c) c:set_bg(beautiful.bg_normal) end)
|
row:connect_signal("mouse::enter", function(c)
|
||||||
|
c:set_bg(beautiful.bg_focus)
|
||||||
|
c:set_shape(function(cr, width, height)
|
||||||
|
gears.shape.rounded_rect(cr, width, height, 4)
|
||||||
|
end)
|
||||||
|
local wb = mouse.current_wibox
|
||||||
|
old_cursor, old_wibox = wb.cursor, wb
|
||||||
|
wb.cursor = "hand1"
|
||||||
|
end)
|
||||||
|
row:connect_signal("mouse::leave", function(c)
|
||||||
|
c:set_bg(beautiful.bg_normal)
|
||||||
|
c:set_shape(nil)
|
||||||
|
if old_wibox then
|
||||||
|
old_wibox.cursor = old_cursor
|
||||||
|
old_wibox = nil
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
|
||||||
row:buttons(
|
row:buttons(
|
||||||
awful.util.table.join(
|
awful.util.table.join(
|
||||||
awful.button({}, 1, function()
|
awful.button({}, 1, function()
|
||||||
spawn.with_shell("xdg-open " .. host .. '/browse/' .. issue.key)
|
spawn.with_shell("xdg-open " .. host .. '/browse/' .. issue.key)
|
||||||
popup.visible = false
|
popup.visible = false
|
||||||
|
jira_widget:set_bg('#00000000')
|
||||||
end)
|
end)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
@ -222,14 +279,18 @@ local function worker(user_args)
|
||||||
awful.util.table.join(
|
awful.util.table.join(
|
||||||
awful.button({}, 1, function()
|
awful.button({}, 1, function()
|
||||||
if popup.visible then
|
if popup.visible then
|
||||||
|
jira_widget:set_bg('#00000000')
|
||||||
popup.visible = not popup.visible
|
popup.visible = not popup.visible
|
||||||
else
|
else
|
||||||
|
jira_widget:set_bg(beautiful.bg_focus)
|
||||||
popup:move_next_to(mouse.current_widget_geometry)
|
popup:move_next_to(mouse.current_widget_geometry)
|
||||||
end
|
end
|
||||||
end)
|
end)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
watch(string.format(GET_ISSUES_CMD, host, query:gsub(' ', '+')),
|
watch(
|
||||||
|
'cat /home/pmakhov/.config/JetBrains/IntelliJIdea2020.2/scratches/scratch_27.json',
|
||||||
|
--string.format(GET_ISSUES_CMD, host, query:gsub(' ', '+')),
|
||||||
timeout, update_widget, jira_widget)
|
timeout, update_widget, jira_widget)
|
||||||
return jira_widget
|
return jira_widget
|
||||||
end
|
end
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 120 KiB |
Binary file not shown.
After Width: | Height: | Size: 110 KiB |
Loading…
Reference in New Issue