diff --git a/docs/warn-stranded.rst b/docs/warn-stranded.rst index b5db2dd496..6242068f87 100644 --- a/docs/warn-stranded.rst +++ b/docs/warn-stranded.rst @@ -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. diff --git a/warn-stranded.lua b/warn-stranded.lua index b08b0be080..5b23d1ac50 100644 --- a/warn-stranded.lua +++ b/warn-stranded.lua @@ -1,6 +1,7 @@ -- 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 @@ -8,6 +9,16 @@ 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', @@ -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) @@ -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() @@ -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