You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
DiesalLibs/DiesalGUI-2.0/Objects/DropDownPullout.lua

415 lines
15 KiB

7 months ago
---@type Diesal.GUI
local DiesalGUI = LibStub("DiesalGUI-2.0")
---@type Diesal.Tools
local DiesalTools = LibStub("DiesalTools-2.0")
---@type Diesal.Style
local DiesalStyle = LibStub("DiesalStyle-2.0")
-- ~~| Diesal Upvalues |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
local Colors = DiesalStyle.Colors
local HSL, ShadeColor, TintColor = DiesalTools.HSL, DiesalTools.ShadeColor, DiesalTools.TintColor
-- ~~| Lua Upvalues |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
local tsort = table.sort
local sub, format, lower, upper, gsub = string.sub, string.format, string.lower, string.upper, string.gsub
-- ~~| WoW Upvalues |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-- ~~| Dropdown |~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
local Type = "DiesalDropDownPullout"
local Version = 1
---@type backdropInfo
local backdrop = {
bgFile = "Interface\\ChatFrame\\ChatFrameBackground",
edgeFile = "Interface\\Tooltips\\UI-Tooltip-Border",
edgeSize = 8,
tileSize = 8,
tile = true,
insets = { left = 2, right = 2, top = 2, bottom = 2 },
}
local sliderBackdrop = {
bgFile = "Interface\\Buttons\\UI-SliderBar-Background",
edgeFile = "Interface\\Buttons\\UI-SliderBar-Border",
tile = true,
tileSize = 8,
edgeSize = 8,
insets = { left = 3, right = 3, top = 3, bottom = 3 },
}
local Stylesheet = {
["frame-background"] = {
type = "texture",
layer = "BACKGROUND",
color = Colors.UI_100,
alpha = 0.95,
},
["frame-inline"] = {
type = "outline",
layer = "BORDER",
color = "ffffff",
alpha = 0.02,
position = -1,
},
["frame-shadow"] = {
type = "shadow",
},
}
local defaultWidth = 200
local defaultMaxItems = 20
local function fixlevels(parent, ...)
local i = 1
local child = select(i, ...)
while child do
child:SetFrameLevel(parent:GetFrameLevel() + 1)
fixlevels(child, child:GetChildren())
i = i + 1
child = select(i, ...)
end
end
local function fixstrata(strata, parent, ...)
local i = 1
local child = select(i, ...)
parent:SetFrameStrata(strata)
while child do
fixstrata(strata, child, child:GetChildren())
i = i + 1
child = select(i, ...)
end
end
-- HACK: This should be no part of the pullout, but there
-- is no other 'clean' way to response to any item-OnEnter
-- Used to close Submenus when an other item is entered
local function OnEnter(item)
local self = item.pullout
for k, v in ipairs(self.settings.items) do
if v.CloseMenu and v ~= item then
v:CloseMenu()
end
end
end
---@class Diesal.GUI.DropDownPullout.Methods
local methods = {
---@param self Diesal.GUI.Object.DropDownPullout
OnAcquire = function(self)
self.frame:SetParent(DiesalGUI.UIParent)
end,
---@param self Diesal.GUI.Object.DropDownPullout
OnRelease = function(self)
self:ClearAllPoints()
self:Hide()
end,
---@param self Diesal.GUI.Object.DropDownPullout
ApplySettings = function(self) end,
---@param self Diesal.GUI.Object.DropDownPullout
GetItemsHeight = function(self)
local itemHeight = 0
for i, item in ipairs(self.settings.items) do
itemHeight = itemHeight + item.text:GetStringHeight()
end
return itemHeight
end,
---@param self Diesal.GUI.Object.DropDownPullout
GetMaxHeight = function(self)
local maxHeight = 0
for i, item in ipairs(self.settings.items) do
if i <= self.maxItems then
maxHeight = maxHeight + item:GetHeight(false)
end
end
return maxHeight
--return self:GetMaxItemHeight() * min(self.maxItems, #self.settings.items)
end,
---@param self Diesal.GUI.Object.DropDownPullout
---@param item DiesalDropDownItem
---@param setPoints? boolean
AddItem = function(self, item, setPoints)
table.insert(self.settings.items, item)
local maxWidth = 8 + (item.settings.indent * 6) + (16 * 2)
if item.text then
maxWidth = maxWidth + item.text:GetUnboundedStringWidth()
end
if item.check then
maxWidth = maxWidth + item.check:GetWidth() + 4
end
self.settings.maxItemWidth = max(self.settings.maxItemWidth, maxWidth)
--self.frame:SetHeight(min(self:GetItemsHeight() + 28, self:GetMaxHeight() + 28))
if setPoints then
item:SetPoint("LEFT", self.itemFrame, "LEFT", 20, 0)
item:SetPoint("RIGHT", self.itemFrame, "RIGHT", 0, 0)
end
item:SetPullout(self)
--item:SetScript("OnEnter", OnEnter)
end,
---@param self Diesal.GUI.Object.DropDownPullout
---@param value number
SetScroll = function(self, value)
local status = self.scrollStatus
local scrollFrame, itemFrame = self.scrollFrame, self.itemFrame
local height, viewheight = scrollFrame:GetHeight(), itemFrame:GetHeight()
--local minVal, maxVal = self.slider:GetMinMaxValues()
local minVal, maxVal = height, viewheight
local offset
if height > viewheight then
offset = 0
self.slider2:Hide()
else
offset = floor(maxVal * value)
self.slider2:Show()
end
itemFrame:ClearAllPoints()
itemFrame:SetPoint("TOPLEFT", scrollFrame, "TOPLEFT", 0, offset)
itemFrame:SetPoint("TOPRIGHT", scrollFrame, "TOPRIGHT", self.slider2:IsShown() and -6 or 0, offset)
status.offset = offset
status.scrollvalue = (maxVal * value)
end,
---@param self Diesal.GUI.Object.DropDownPullout
---@param value number
MoveScroll = function(self, value)
local status = self.scrollStatus
local frame, child = self.scrollFrame, self.itemFrame
local height, viewheight = frame:GetHeight(), child:GetHeight()
--local minVal, maxVal = self.slider:GetMinMaxValues()
local minVal, maxVal = height, self:GetItemsHeight()
--child:SetHeight(self:GetItemsHeight())
if height > viewheight then
self.slider2:Hide()
else
self.slider2:Show()
local diff = height - viewheight
local delta = 1
if value < 0 then
delta = -1
end
local scrollToVal = min(max(status.scrollvalue + delta * (maxVal / (diff / self.settings.itemHeight)), 0), maxVal)
self.slider2:SetScrollPercentage(scrollToVal/maxVal)
--self.slider:SetValue(scrollToVal)
end
end,
---@param self Diesal.GUI.Object.DropDownPullout
FixScroll = function(self)
local status = self.scrollStatus
local frame, child = self.scrollFrame, self.itemFrame
local height, viewheight = frame:GetHeight(), child:GetHeight()
local offset = status.offset or 0
local minVal, maxVal = height, viewheight
if viewheight < height then
self.slider2:Hide()
child:SetPoint("TOPRIGHT", frame, "TOPRIGHT", 0, offset)
self.scrollFrame:SetVerticalScroll(0)
else
self.slider2:Show()
local value = (offset / (viewheight - height) * maxVal)
--self.slider:SetValue(value)
self:SetScroll(value)
if value < maxVal then
child:ClearAllPoints()
child:SetPoint("TOPLEFT", frame, "TOPLEFT", 0, offset)
child:SetPoint("TOPRIGHT", frame, "TOPRIGHT", 0, offset)
status.offset = offset
end
end
end,
---@param self Diesal.GUI.Object.DropDownPullout
---@param point FramePoint
---@param relFrame Frame
---@param relPoint FramePoint
---@param x number
---@param y number
Open = function(self, point, relFrame, relPoint, x, y)
local items = self.settings.items
local frame = self.frame
local itemFrame = self.itemFrame
frame:SetPoint(point, relFrame, relPoint, x, y)
local maxWidth = self.settings.maxItemWidth + self.slider2:GetWidth()
if self.dropdown.settings.dropdownWidth then
if self.dropdown.settings.dropdownWidth > maxWidth then
frame:SetWidth(self.dropdown.settings.dropdownWidth)
else
frame:SetWidth(maxWidth)
end
else
frame:SetWidth(max(maxWidth, self.dropdown.frame:GetWidth()))
--frame:SetPoint("TOPRIGHT", relFrame, "BOTTOMRIGHT", x, y)
end
local height = 0
for i, item in ipairs(items) do
item:SetPoint("TOPLEFT", i == 1 and itemFrame or items[i - 1].frame, i == 1 and "TOPLEFT" or "BOTTOMLEFT", 0, 0)
item:Show()
height = height + item.text:GetStringHeight()
end
frame:Show()
frame:SetHeight(self:GetMaxHeight())
itemFrame:SetHeight(self:GetItemsHeight())
self.scrollFrame:SetScrollChild(itemFrame)
fixstrata("TOOLTIP", self.scrollFrame, self.scrollFrame:GetChildren())
self:FireEvent("OnOpen")
end,
---@param self Diesal.GUI.Object.DropDownPullout
Close = function(self)
self.frame:Hide()
self:FireEvent("OnClose")
end,
---@param self Diesal.GUI.Object.DropDownPullout
Clear = function(self)
local items = self.settings.items
for i, item in ipairs(items) do
items[i] = nil
end
for i,item in ipairs(self.itemFrame) do
end
end,
---@param self Diesal.GUI.Object.DropDownPullout
IterateItems = function(self)
return ipairs(self.settings.items)
end,
---@param self Diesal.GUI.Object.DropDownPullout
SetHideOnLeave = function(self, val)
self.hideOnLeave = val
end,
---@param self Diesal.GUI.Object.DropDownPullout
GetRightBorderWidth = function(self)
return 6 + (self.slider:IsShown() and 12 or 0)
end,
---@param self Diesal.GUI.Object.DropDownPullout
GetLeftBorderWidth = function(self)
return 6
end,
---@param self Diesal.GUI.Object.DropDownPullout
ClearFocus = function(self)
self:Hide()
end,
---@param self Diesal.GUI.Object.DropDownPullout
---@param height? number
SetItemHeight = function(self, height)
self.settings.itemHeight = height or self.defaults.itemHeight
end,
---@param self Diesal.GUI.Object.DropDownPullout
SetDropDown = function (self, dropdown)
self.dropdown = dropdown
end
}
---@class Diesal.GUI.Object.DropDownPullout.Settings : Diesal.GUI.ObjectBase.Settings
---@field width number
---@field itemHeight number
---@field sliderWidth number
---@field hideOnLeave boolean
---@field items DiesalDropDownItem[]
---@field maxItemWidth number
---@class DiesalDropDownPullout : Diesal.GUI.Object.DropDownPullout
local function Constructor()
---@class Diesal.GUI.Object.DropDownPullout : Diesal.GUI.ObjectBase, Diesal.GUI.DropDownPullout.Methods
---@field settings Diesal.GUI.Object.DropDownPullout.Settings
---@field items DiesalDropDownItem[]
---@field dropdown DiesalDropDown
local self = DiesalGUI:Create(Type, true)
self:SetMethods(methods)
local frame = CreateFrame("Frame", nil, DiesalGUI.UIParent, "BackdropTemplate")
self.frame = frame
self:SetObj(self.frame)
self.defaults = {
width = 200,
itemHeight = 12,
sliderWidth = 10,
hideOnLeave = true,
items = {},
maxItemWidth = 0,
}
self.scrollStatus = {
scrollvalue = 0,
}
self.hideOnLeave = false
self.maxItems = 20
frame:SetBackdrop(backdrop)
frame:SetBackdropColor(0, 0, 0, 1)
frame:SetBackdropBorderColor(0.2, 0.2, 0.2, 1)
frame:SetFrameStrata("FULLSCREEN_DIALOG")
frame:SetClampedToScreen(true)
frame:SetWidth(self.defaults.width)
frame:SetHeight(34)
frame:SetScript("OnHide", function(this)
self.dropdown.open = nil
self.dropdown.arrow:SetRotation(0)
DiesalGUI:ClearFocus()
end)
self:CreateRegion("Frame", "content", frame)
local scrollFrame = CreateFrame("ScrollFrame", nil, frame, "ScrollFrameTemplate")
local itemFrame = CreateFrame("Frame", nil, scrollFrame)
self.scrollFrame = scrollFrame
self.itemFrame = itemFrame
self:SetObj(self.scrollFrame)
self:SetObj(self.itemFrame)
---@type MinimalScrollBar
local slider2 = scrollFrame.ScrollBar
self.slider2 = slider2
self.slider2:SetPoint("TOPLEFT", self.scrollFrame, "TOPRIGHT", -20, -2)
self.slider2:SetPoint("BOTTOMLEFT", self.scrollFrame, "BOTTOMRIGHT", -20, 5)
self.scrollFrame:SetVerticalScroll(0)
self.slider2:SetVisibleExtentPercentage(0.25)
self.slider2:SetThumbExtent(0.25)
self.slider2:RegisterCallback("OnScroll", function(this, percent)
self:SetScroll(percent)
end, self.slider2)
--local slider = CreateFrame("Slider", nil, scrollFrame, "BackdropTemplate")
--slider:SetOrientation("VERTICAL")
--slider:SetHitRectInsets(0, 0, -10, 0)
--slider:SetBackdrop(sliderBackdrop)
--slider:SetWidth(8)
--slider:SetThumbTexture("Interface\\Buttons\\UI-SliderBar-Button-Vertical")
--slider:SetFrameStrata("FULLSCREEN_DIALOG")
--self.slider = slider
--self:SetObj(self.slider)
scrollFrame:SetPoint("TOPLEFT", frame, "TOPLEFT")
scrollFrame:SetPoint("BOTTOMRIGHT", frame, "BOTTOMRIGHT")
scrollFrame:EnableMouseWheel(true)
--[[ scrollFrame:SetScript("OnMouseWheel", function(this, value)
self:MoveScroll(value)
self:FireEvent("OnMouseWheel", value)
end) ]]
scrollFrame:SetToplevel(true)
scrollFrame:SetFrameStrata("FULLSCREEN_DIALOG")
itemFrame:SetPoint("TOPLEFT", scrollFrame, "TOPLEFT", 0, 0)
itemFrame:SetPoint("TOPRIGHT", scrollFrame, "TOPRIGHT", -6, 0)
--itemFrame:SetHeight(400)
itemFrame:SetToplevel(true)
itemFrame:SetFrameStrata("FULLSCREEN_DIALOG")
scrollFrame.ItemFrame = itemFrame
scrollFrame:SetScrollChild(scrollFrame.ItemFrame)
--slider:SetPoint("TOPLEFT", scrollFrame, "TOPRIGHT", -16, 0)
--slider:SetPoint("BOTTOMLEFT", scrollFrame, "BOTTOMRIGHT", -16, 0)
--slider:SetScript("OnValueChanged", function(this, value)
--self:SetScroll(value)
--end)
--slider:SetMinMaxValues(0, 1)
--slider:SetValueStep(1)
--slider:SetValue(0)
scrollFrame:Show()
itemFrame:Show()
--slider:Hide()
self:FixScroll()
return self
end
DiesalGUI:RegisterObjectConstructor(Type, Constructor, Version)