diff --git a/github-activity-widget/github-activity-widget.lua b/github-activity-widget/github-activity-widget.lua
new file mode 100644
index 0000000..bf1b579
--- /dev/null
+++ b/github-activity-widget/github-activity-widget.lua
@@ -0,0 +1,310 @@
+-------------------------------------------------
+-- GitHub Widget for Awesome Window Manager
+-- Shows the number of currently assigned issues
+-- More details could be found here:
+-- https://github.com/streetturtle/awesome-wm-widgets/tree/master/github-activity-widget
+
+-- @author Pavel Makhov
+-- @copyright 2020 Pavel Makhov
+-------------------------------------------------
+
+local awful = require("awful")
+local wibox = require("wibox")
+local json = require("json")
+local spawn = require("awful.spawn")
+local naughty = require("naughty")
+local gears = require("gears")
+local beautiful = require("beautiful")
+local gfs = require("gears.filesystem")
+
+
+local HOME_DIR = os.getenv("HOME")
+local WIDGET_DIR = HOME_DIR .. '/.config/awesome/awesome-wm-widgets/github-activity-widget/'
+local ICONS_DIR = WIDGET_DIR .. 'icons/'
+local CACHE_DIR = HOME_DIR .. '/.cache/awmw/github-activity-widget/'
+
+local GET_ISSUES_CMD = [[bash -c "cat /home/pmakhov/.cache/awmw/github-activity-widget/activity.json | jq '.[:%d] | [.[] | {type: .type, actor: .actor, repo: .repo, action: .payload.action, issue_url: .payload.issue.html_url, pr_url: .payload.pull_request.html_url, created_at: .created_at}]'"]]
+local DOWNLOAD_AVATAR_CMD = [[bash -c "curl -n --create-dirs -o %s/avatars/%s %s"]]
+local UPDATE_EVENTS_CMD = [[bash -c "curl -s --show-error https://api.github.com/users/%s/received_events > ~/.cache/awmw/github-activity-widget/activity.json"]]
+
+--- Utility function to show warning messages
+local function show_warning(message)
+ naughty.notify{
+ preset = naughty.config.presets.critical,
+ title = 'GitHub Activity Widget',
+ text = message}
+end
+
+--- Converts string representation of date to date
+local function parse_date(date_str)
+ local pattern = "(%d+)%-(%d+)%-(%d+)T(%d+):(%d+):(%d+)%Z"
+ local y, m, d, h, min, sec, mil = date_str:match(pattern)
+
+ return os.time{year = y, month = m, day = d, hour = h, min = min, sec = sec}
+end
+
+--- Converts seconds to "time ago" represenation, like '1 hour ago'
+local function to_time_ago(seconds)
+ local days = seconds / 86400
+ if days > 1 then
+ days = math.floor(days + 0.5)
+ return days .. (days == 1 and ' day' or ' days') .. ' ago'
+ end
+
+ local hours = (seconds % 86400) / 3600
+ if hours > 1 then
+ hours = math.floor(hours + 0.5)
+ return hours .. (hours == 1 and ' hour' or ' hours') .. ' ago'
+ end
+
+ local minutes = ((seconds % 86400) % 3600) / 60
+ if minutes > 1 then
+ minutes = math.floor(minutes + 0.5)
+ return minutes .. (minutes == 1 and ' minute' or ' minutes') .. ' ago'
+ end
+end
+
+
+local popup = awful.popup{
+ ontop = true,
+ visible = false,
+ shape = gears.shape.rounded_rect,
+ border_width = 1,
+ border_color = beautiful.bg_focus,
+ maximum_width = 350,
+ offset = { y = 5 },
+ widget = {}
+}
+
+local type_to_text_mapping = {
+ WatchEvent = 'starred',
+ CommitCommentEvent = '',
+ CreateEvent = 'created',
+ DeleteEvent = '',
+ ForkEvent = 'forked',
+ GollumEvent = '',
+ IssueCommentEvent = '',
+ IssuesEvent = '',
+ MemberEvent = '',
+ PublicEvent = '',
+ PullRequestEvent = '',
+ PullRequestReviewCommentEvent = '',
+ PushEvent = 'pushed',
+ ReleaseEvent = '',
+ SponsorshipEvent = ''
+}
+
+local function generate_action_string(event)
+ local action_string = type_to_text_mapping[event.type]
+ local icon = 'repo.svg'
+ local link = 'http://github.com/' .. event.repo.name
+
+ if (event.type == "PullRequestEvent") then
+ action_string = event.action .. ' a pull request in'
+ link = event.pr_url
+ icon = 'pr.svg'
+ elseif (event.type == "IssuesEvent") then
+ action_string = event.action .. ' an issue in'
+ link = event.issue_url
+ icon = 'issue.svg'
+ elseif (event.type == "IssueCommentEvent") then
+ action_string = event.action == 'created' and 'commented in issue' or event.action .. ' a comment in'
+ link = event.issue_url
+ icon = 'comment.svg'
+ elseif (event.type == "WatchEvent") then
+ action_string = 'starred'
+ link = 'http://github.com/' .. event.repo.name
+ icon = 'star.svg'
+ elseif (event.type == "ForkEvent") then
+ action_string = 'forked'
+ link = 'http://github.com/' .. event.repo.name
+ icon = 'fork.svg'
+ end
+
+ return { action_string = action_string, link = link, icon = icon }
+end
+
+local github_widget = wibox.widget {
+ {
+ {
+ id = 'icon',
+ widget = wibox.widget.imagebox
+ },
+ id = "m",
+ margins = 4,
+ layout = wibox.container.margin
+ },
+ {
+ id = "txt",
+ widget = wibox.widget.textbox
+ },
+ {
+ id = "new_rev",
+ widget = wibox.widget.textbox
+ },
+ layout = wibox.layout.fixed.horizontal,
+ set_icon = function(self, new_icon)
+ self.m.icon.image = new_icon
+ end,
+ set_text = function(self, new_value)
+ self.txt.text = new_value
+ end,
+ set_unseen_review = function(self, is_new_review)
+ self.new_rev.text = is_new_review and '*' or ''
+ end
+}
+
+
+local function worker(args)
+
+ local args = args or {}
+
+ local icon = args.icon or ICONS_DIR .. 'github.png'
+ local username = args.username or show_warning('No username provided')
+ local number_of_events = args.number_of_events or 10
+
+ github_widget:set_icon(icon)
+
+ local current_number_of_reviews
+ local previous_number_of_reviews = 0
+
+ local rows = {
+ { widget = wibox.widget.textbox },
+ layout = wibox.layout.fixed.vertical,
+ }
+
+ local update_widget = function(widget, stdout, stderr, _, _)
+ if stderr ~= '' then
+ show_warning(stderr)
+ return
+ end
+
+ local current_time = os.time(os.date("!*t"))
+
+ local result = json.decode(stdout)
+
+ current_number_of_reviews = rawlen(result)
+
+ if current_number_of_reviews == 0 then
+ widget:set_visible(false)
+ return
+ end
+
+ widget:set_visible(true)
+ -- widget:set_text(current_number_of_reviews)
+
+ for i = 0, #rows do rows[i]=nil end
+ for _, issue in ipairs(result) do
+ local path_to_avatar = CACHE_DIR .. '/avatars/' .. issue.actor.id
+
+ if not gfs.file_readable(path_to_avatar) then
+ spawn.easy_async(string.format(
+ DOWNLOAD_AVATAR_CMD,
+ CACHE_DIR,
+ issue.actor.id,
+ issue.actor.avatar_url))
+ end
+
+ local action_and_link = generate_action_string(issue)
+
+ local row = wibox.widget {
+ {
+ {
+ {
+ {
+ resize = true,
+ image = path_to_avatar,
+ forced_width = 40,
+ forced_height = 40,
+ widget = wibox.widget.imagebox
+ },
+ margins = 8,
+ layout = wibox.container.margin
+ },
+ {
+ {
+ markup = ' ' .. issue.actor.display_login .. ' ' .. action_and_link.action_string .. ' ' .. issue.repo.name .. '',
+ wrap = 'word',
+ widget = wibox.widget.textbox
+ },
+ {
+ {
+ {
+ image = ICONS_DIR .. action_and_link.icon,
+ resize = true,
+ forced_height = 16,
+ forced_width = 16,
+ widget = wibox.widget.imagebox
+ },
+ valign = 'center',
+ layout = wibox.container.place
+ },
+ {
+ markup = to_time_ago(os.difftime(current_time, parse_date(issue.created_at))),
+ widget = wibox.widget.textbox
+ },
+ spacing = 4,
+ layout = wibox.layout.fixed.horizontal,
+ },
+ layout = wibox.layout.align.vertical
+ },
+ spacing = 4,
+ layout = wibox.layout.fixed.horizontal
+ },
+ margins = 4,
+ layout = wibox.container.margin
+ },
+ bg = beautiful.bg_normal,
+ widget = wibox.container.background
+ }
+
+ row:connect_signal("mouse::enter", function(c) c:set_bg(beautiful.bg_focus) end)
+ row:connect_signal("mouse::leave", function(c) c:set_bg(beautiful.bg_normal) end)
+
+ row:buttons(
+ awful.util.table.join(
+ awful.button({}, 1, function()
+ spawn.with_shell("xdg-open " .. action_and_link.link)
+ popup.visible = false
+ end)
+ )
+ )
+
+ table.insert(rows, row)
+ end
+
+ popup:setup(rows)
+ end
+
+ github_widget:buttons(
+ awful.util.table.join(
+ awful.button({}, 1, function()
+ if popup.visible then
+ popup.visible = not popup.visible
+ else
+ spawn.easy_async(string.format(GET_ISSUES_CMD, number_of_events), function (stdout, stderr)
+ update_widget(github_widget, stdout, stderr)
+ popup:move_next_to(mouse.current_widget_geometry)
+ end)
+ end
+ end)
+ )
+ )
+
+ gears.timer {
+ timeout = 600,
+ call_now = true,
+ autostart = true,
+ callback = function()
+ spawn.easy_async(string.format(UPDATE_EVENTS_CMD, username), function(stdout, stderr)
+ if stderr ~= '' then
+ show_warning(stderr)
+ return
+ end
+ end)
+ end
+ }
+ return github_widget
+end
+
+return setmetatable(github_widget, { __call = function(_, ...) return worker(...) end })
diff --git a/github-activity-widget/icons/comment.svg b/github-activity-widget/icons/comment.svg
new file mode 100644
index 0000000..5cb54bf
--- /dev/null
+++ b/github-activity-widget/icons/comment.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/github-activity-widget/icons/fork.svg b/github-activity-widget/icons/fork.svg
new file mode 100644
index 0000000..4c722dc
--- /dev/null
+++ b/github-activity-widget/icons/fork.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/github-activity-widget/icons/github.png b/github-activity-widget/icons/github.png
new file mode 100644
index 0000000..628da97
Binary files /dev/null and b/github-activity-widget/icons/github.png differ
diff --git a/github-activity-widget/icons/issue.svg b/github-activity-widget/icons/issue.svg
new file mode 100644
index 0000000..b47543a
--- /dev/null
+++ b/github-activity-widget/icons/issue.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/github-activity-widget/icons/pr.svg b/github-activity-widget/icons/pr.svg
new file mode 100644
index 0000000..412be1a
--- /dev/null
+++ b/github-activity-widget/icons/pr.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/github-activity-widget/icons/repo.svg b/github-activity-widget/icons/repo.svg
new file mode 100644
index 0000000..f74a595
--- /dev/null
+++ b/github-activity-widget/icons/repo.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/github-activity-widget/icons/star.svg b/github-activity-widget/icons/star.svg
new file mode 100644
index 0000000..7ac51ac
--- /dev/null
+++ b/github-activity-widget/icons/star.svg
@@ -0,0 +1 @@
+
\ No newline at end of file