Skip to content

Commit

Permalink
warn-stranded: Add GUI to allow persistently ignoring units in
Browse files Browse the repository at this point in the history
  • Loading branch information
azrazalea committed Sep 11, 2023
1 parent a31d95d commit dd78906
Show file tree
Hide file tree
Showing 2 changed files with 158 additions and 34 deletions.
11 changes: 9 additions & 2 deletions docs/warn-stranded.rst
Original file line number Diff line number Diff line change
Expand Up @@ -11,17 +11,24 @@ they get overly stressed or start starving.

You can enable ``warn-stranded`` notifications in `gui/control-panel` on the "Maintenance" tab.

If you ignore a unit, either call ``warn-stranded clear`` in the dfhack console or if you have multiple
stranded you can toggle/clear all units in the warning dialog.

Usage
-----

::

warn-stranded
warn-stranded [clear]

Examples
--------

``warn-stranded``
``warn-stranded clear``
Clear all ignored units and then check for ones that are stranded.

Options
-------

``clear``
Will clear all ignored units so that warnings will be displayed again.
181 changes: 149 additions & 32 deletions warn-stranded.lua
Original file line number Diff line number Diff line change
@@ -1,13 +1,24 @@
-- Detects and alerts when a citizen is stranded
-- by Azrazalea
-- Heavily based off of warn-starving
-- Logic heavily based off of warn-starving
-- GUI heavily based off of autobutcher
-- Thanks myk002 for telling me about pathability groups!
--@ module = true

local gui = require 'gui'
local utils = require 'utils'
local widgets = require 'gui.widgets'

local function clear()
dfhack.persistent.delete('warnStrandedIgnore')
end

local args = utils.invert({...})
if args.clear then
clear()
end


warning = defclass(warning, gui.ZScreen)
warning.ATTRS = {
focus_path='warn-stranded',
Expand All @@ -16,24 +27,36 @@ warning.ATTRS = {
}

function warning:init(info)
local main = widgets.Window{
frame={w=80, h=18},
frame_title='Stranded Citizen Warning',
resizable=true,
autoarrange_subviews=true
}

main:addviews{
widgets.WrappedLabel{
text_to_wrap=table.concat(info.messages, NEWLINE),
}
self:addviews{
widgets.Window{
view_id = 'main',
frame={w=80, h=18},
frame_title='Stranded Citizen Warning',
resizable=true,
subviews = {
widgets.Label{
frame = { l = 0, t = 0},
text_pen = COLOR_CYAN,
text = 'Number Stranded: '..#info.units,
},
widgets.List{
view_id = 'list',
frame = { t = 3, b = 5 },
text_pen = { fg = COLOR_GREY, bg = COLOR_BLACK },
cursor_pen = { fg = COLOR_BLACK, bg = COLOR_GREEN },
},
widgets.Label{
view_id = 'bottom_ui',
frame = { b = 0, h = 1 },
text = 'filled by updateBottom()'
}
}
}
}

self:addviews{main}
end

function warning:onDismiss()
view = nil
self.units = info.units
self:initListChoices()
self:updateBottom()
end

local function getSexString(sex)
Expand All @@ -44,6 +67,113 @@ local function getSexString(sex)
return "("..sym..")"
end

local function getUnitDescription(unit)
return '['..dfhack.units.getProfessionName(unit)..'] '..dfhack.TranslateName(dfhack.units.getVisibleName(unit))..
' '..getSexString(unit.sex)..' Stress category: '..dfhack.units.getStressCategory(unit)
end


local function unitIgnored(unit)
local currentIgnore = dfhack.persistent.get('warnStrandedIgnore')
if currentIgnore == nil then return false end

local tbl = string.gmatch(currentIgnore['value'], '%d+')
local index = 1
for id in tbl do
if tonumber(id) == unit.id then
return true, index
end
index = index + 1
end

return false
end

local function toggleUnitIgnore(unit)
local currentIgnore = dfhack.persistent.get('warnStrandedIgnore')
local tbl = {}

if currentIgnore == nil then
currentIgnore = { key = 'warnStrandedIgnore' }
else
local index = 1
for v in string.gmatch(currentIgnore['value'], '%d+') do
tbl[index] = v
index = index + 1
end
end

local ignored, index = unitIgnored(unit)

if ignored then
table.remove(tbl, index)
else
table.insert(tbl, unit.id)
end

dfhack.persistent.delete('warnStrandedIgnore')
currentIgnore.value = table.concat(tbl, ' ')
dfhack.persistent.save(currentIgnore)
end

function warning:initListChoices()
local choices = {}
for _, unit in pairs(self.units) do
local text = ''

dfhack.printerr('Ignored: ', unitIgnored(unit))

if unitIgnored(unit) then
text = '[IGNORED] '
end

text = text..getUnitDescription(unit)
table.insert(choices, { text = text, unit = unit })
end
local list = self.subviews.list
list:setChoices(choices, 1)
end

function warning:updateBottom()
self.subviews.bottom_ui:setText(
{
{ key = 'SELECT', text = ': Toggle ignore unit', on_activate = self:callback('onIgnore') }, ' ',
{ key = 'CUSTOM_SHIFT_I', text = ': Ignore all', on_activate = self:callback('onIgnoreAll') }, ' ',
{ key = 'CUSTOM_SHIFT_C', text = ': Clear all ignored', on_activate = self:callback('onClear') },
}
)
end

function warning:onIgnore()
local index, choice = self.subviews.list:getSelected()
local unit = choice.unit

toggleUnitIgnore(unit)
self:initListChoices()
end

function warning:onIgnoreAll()
local choices = self.subviews.list:getChoices()

for _, choice in pairs(choices) do
if not unitIgnored(choice.unit) then
toggleUnitIgnore(choice.unit)
end
end

self:dismiss()
end

function warning:onClear()
clear()
self:initListChoices()
self:updateBottom()
end

function warning:onDismiss()
view = nil
end

function doCheck()
local grouped = {}
local citizens = dfhack.units.getCitizens()
Expand All @@ -66,26 +196,13 @@ function doCheck()


for _, units in pairs(grouped) do
if #units == 1 then
if #units == 1 and not unitIgnored(units[1]) then
table.insert(strandedUnits, units[1])
end
end

if #strandedUnits > 0 then
dfhack.color(COLOR_LIGHTMAGENTA)

local messages = {}
local preface = "Number of stranded: "..#strandedUnits
print(dfhack.df2console(preface))
table.insert(messages, preface)
for _, unit in pairs(strandedUnits) do
local unitString = '['..dfhack.units.getProfessionName(unit)..'] '..dfhack.TranslateName(dfhack.units.getVisibleName(unit))..' '..getSexString(unit.sex)..' Stress category: '..dfhack.units.getStressCategory(unit)
print(dfhack.df2console(unitString))
table.insert(messages, unitString)
end

dfhack.color()
return warning{messages=messages}:show()
return warning{units=strandedUnits}:show()
end
end

Expand Down

0 comments on commit dd78906

Please sign in to comment.