Dejunk (Sell & Destroy Junk)

Dejunk (Sell & Destroy Junk)

1M Downloads

How does this addon identify the level of items to sell?

JRDK92 opened this issue ยท 3 comments

commented

Curious how it identifies gear/items to sell based on item level. Say I enter 300 ilvl or less, sell it. Where in the script, what LUA file does that?

commented

This code block in the JunkFilter:IsJunkItem(item) method. It only applies to soulbound equipment.

Dejunk/src/junk-filter.lua

Lines 127 to 135 in 359a333

if Items:IsItemBound(item) and Items:IsItemEquipment(item) then
-- Include below item level.
if currentState.includeBelowItemLevel.enabled then
local value = currentState.includeBelowItemLevel.value
if item.itemLevel < value then
local valueText = Colors.Grey("(%s)"):format(Colors.Yellow(value))
return true, concat(L.OPTIONS_TEXT, L.INCLUDE_BELOW_ITEM_LEVEL_TEXT .. " " .. valueText)
end
end

commented

Thanks @moody. Do you have any sort of guide or manifest as to how the add-on works? I'm not trying to copy it per se, but trying to learn from it for similar tasks. I have never programmed in LUA But I've been trying to create something similar as a practice project. Just to get down the fundamentals. This has been a really good exercise in figuring it out, the one thing I was really overwhelmed by was how many different files there are broken up all over the place. For example it seems like you have each individual component of the entire user interface in its own file, and I was trying to understand how they work together

commented

@jsauerland I don't have any additional documentation or guide for the code outside of what you can see in the repo.

Dejunk was my first real coding project when I started learning many years ago. The project has gone through several stages of refactoring and has even been rebuilt from scratch once before; a consequence of sloppy coding as I stumbled my way through learning everything.

I've added a lot more comments and documentation to the code somewhat recently, which hopefully serves useful as you look through it. Still, some sloppy code and confusing design decisions still exist in the project.

As for the UI though, I find it to be one of the most tedious aspects of making an addon. I hate using XML, so I avoided that route and just did everything in Lua. To make the UI code maintainable, I split everything into a separate file as you've noticed. A key part of this is the Widgets module, which is just a collection of methods to create composable pieces of the UI, such as a simple frame with a backdrop. These widgets are then used to construct bigger parts of the UI, by using them as the base frame.

A good example of this would be the MainWindow. Its base frame is a Window widget:

--- @class MainWindowWidget : WindowWidget
local frame = Widgets:Window({
name = ADDON_NAME .. "_MainWindow",
width = TOTAL_FRAME_WIDTH,
height = 600,
titleText = Colors.Blue(ADDON_NAME),
})

The Window widget itself is based from a TitleFrame widget (see line 46):

--- Creates a moveable frame with title text and a close button.
--- @param options WindowWidgetOptions
--- @return WindowWidget frame
function Widgets:Window(options)
-- Defaults.
options.points = Addon:IfNil(options.points, { { "CENTER" } })
options.width = Addon:IfNil(options.width, 675)
options.height = Addon:IfNil(options.height, 500)
options.onUpdateTooltip = nil
options.titleTemplate = "GameFontNormalLarge"
options.titleJustify = "LEFT"
--- @class WindowWidget : TitleFrameWidget
local frame = self:TitleFrame(options)

A TitleFrame widget is based from a simple Frame widget (see line 41):

--- Creates a basic frame with title text.
--- @param options TitleFrameWidgetOptions
--- @return TitleFrameWidget frame
function Widgets:TitleFrame(options)
-- Defaults.
options.frameType = "Frame"
options.titleText = Addon:IfNil(options.titleText, ADDON_NAME)
options.titleTemplate = Addon:IfNil(options.titleTemplate, "GameFontNormal")
options.titleJustify = Addon:IfNil(options.titleJustify, "CENTER")
local onUpdateTooltip = options.onUpdateTooltip
options.onUpdateTooltip = nil
--- @class TitleFrameWidget : FrameWidget
local frame = self:Frame(options)

Finally, a Frame widget is actually just a basic frame created using the WoW API (see line 35):

--- Creates a basic frame with a backdrop.
--- @param options FrameWidgetOptions
--- @return FrameWidget frame
function Widgets:Frame(options)
-- Defaults.
options.frameType = Addon:IfNil(options.frameType, "Frame")
options.parent = Addon:IfNil(options.parent, UIParent)
options.width = Addon:IfNil(options.width, 1)
options.height = Addon:IfNil(options.height, 1)
--- @class FrameWidget : Frame, BackdropTemplate
local frame = CreateFrame(options.frameType, options.name, options.parent)


A core part of these widget methods is having only a single parameter, options, which is a table intended to be passed from top to bottom, containing all of the properties necessary for each widget in the hierarchy.

The layers of abstraction make things easier to manage, and I think of it like Lego; small blocks connected to one another to build something much more complicated.