Build skills tab
This commit is contained in:
parent
1a6a006280
commit
d8f842f854
20 changed files with 423 additions and 3 deletions
0
.idea/.gitignore
generated
vendored
Normal file → Executable file
0
.idea/.gitignore
generated
vendored
Normal file → Executable file
0
.idea/misc.xml
generated
Normal file → Executable file
0
.idea/misc.xml
generated
Normal file → Executable file
0
.idea/modules.xml
generated
Normal file → Executable file
0
.idea/modules.xml
generated
Normal file → Executable file
0
.idea/vcs.xml
generated
Normal file → Executable file
0
.idea/vcs.xml
generated
Normal file → Executable file
0
AltSystem.iml
Normal file → Executable file
0
AltSystem.iml
Normal file → Executable file
1
AltSystem.toc
Normal file → Executable file
1
AltSystem.toc
Normal file → Executable file
|
|
@ -8,5 +8,6 @@
|
|||
|
||||
Data.lua
|
||||
Core.lua
|
||||
BuildSkillsUI.lua
|
||||
UI.lua
|
||||
Roll.lua
|
||||
|
|
|
|||
310
BuildSkillsUI.lua
Executable file
310
BuildSkillsUI.lua
Executable file
|
|
@ -0,0 +1,310 @@
|
|||
-- AltSystem Build Skills UI
|
||||
-- Creates the Build Skills tab content with editable skill rows, add/delete functionality, and save to TRP.
|
||||
|
||||
AltSystem = AltSystem or {}
|
||||
|
||||
local PADDING = 12
|
||||
local ROW_HEIGHT = 36
|
||||
local ROW_SPACING = 8
|
||||
local NAME_WIDTH = 340
|
||||
local LEVEL_WIDTH = 140
|
||||
local VALUE_WIDTH = 80
|
||||
local DELETE_WIDTH = 30
|
||||
|
||||
-- Working copy of skills being edited (not saved until user clicks Save)
|
||||
local editableSkills = {}
|
||||
|
||||
-- Pool of created row frames for reuse
|
||||
local skillRowFrames = {}
|
||||
|
||||
-- References set during creation
|
||||
local scrollFrame, scrollChild, addRowButton
|
||||
|
||||
-- ─── Create a single skill row frame ────────────────────────────────────────
|
||||
|
||||
local function CreateSkillRowFrame(index)
|
||||
local row = CreateFrame("Frame", "AltSystemSkillRow" .. index, scrollChild)
|
||||
row:SetHeight(ROW_HEIGHT)
|
||||
|
||||
-- Dark background
|
||||
local bg = row:CreateTexture(nil, "BACKGROUND")
|
||||
bg:SetAllPoints()
|
||||
bg:SetColorTexture(0.1, 0.1, 0.1, 0.8)
|
||||
row.bg = bg
|
||||
|
||||
-- Name EditBox
|
||||
local nameBox = CreateFrame("EditBox", "AltSystemSkillName" .. index, row, "InputBoxTemplate")
|
||||
nameBox:SetPoint("LEFT", row, "LEFT", 8, 0)
|
||||
nameBox:SetSize(NAME_WIDTH, 24)
|
||||
nameBox:SetAutoFocus(false)
|
||||
nameBox:SetFontObject("GameFontHighlight")
|
||||
row.nameBox = nameBox
|
||||
|
||||
-- Level Dropdown
|
||||
local levelDropdown = CreateFrame("DropdownButton", "AltSystemSkillLevel" .. index, row, "WowStyle1DropdownTemplate")
|
||||
levelDropdown:SetPoint("LEFT", nameBox, "RIGHT", 12, 0)
|
||||
levelDropdown:SetWidth(LEVEL_WIDTH)
|
||||
row.levelDropdown = levelDropdown
|
||||
|
||||
-- Value Dropdown
|
||||
local valueDropdown = CreateFrame("DropdownButton", "AltSystemSkillValue" .. index, row, "WowStyle1DropdownTemplate")
|
||||
valueDropdown:SetPoint("LEFT", levelDropdown, "RIGHT", 8, 0)
|
||||
valueDropdown:SetWidth(VALUE_WIDTH)
|
||||
row.valueDropdown = valueDropdown
|
||||
|
||||
-- Delete Button
|
||||
local deleteBtn = CreateFrame("Button", "AltSystemSkillDelete" .. index, row)
|
||||
deleteBtn:SetSize(DELETE_WIDTH, DELETE_WIDTH)
|
||||
deleteBtn:SetPoint("LEFT", valueDropdown, "RIGHT", 8, 0)
|
||||
|
||||
local deleteBg = deleteBtn:CreateTexture(nil, "BACKGROUND")
|
||||
deleteBg:SetAllPoints()
|
||||
deleteBg:SetColorTexture(0.5, 0.1, 0.1, 0.9)
|
||||
deleteBtn.bg = deleteBg
|
||||
|
||||
local deleteText = deleteBtn:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||
deleteText:SetPoint("CENTER")
|
||||
deleteText:SetText("X")
|
||||
deleteText:SetTextColor(1, 0.3, 0.3)
|
||||
|
||||
deleteBtn:SetScript("OnEnter", function(self)
|
||||
self.bg:SetColorTexture(0.7, 0.15, 0.15, 1)
|
||||
end)
|
||||
deleteBtn:SetScript("OnLeave", function(self)
|
||||
self.bg:SetColorTexture(0.5, 0.1, 0.1, 0.9)
|
||||
end)
|
||||
|
||||
row.deleteBtn = deleteBtn
|
||||
|
||||
return row
|
||||
end
|
||||
|
||||
-- ─── Refresh all skill rows ─────────────────────────────────────────────────
|
||||
|
||||
local function RefreshSkillRows()
|
||||
-- Hide all existing row frames
|
||||
for _, row in ipairs(skillRowFrames) do
|
||||
row:Hide()
|
||||
end
|
||||
|
||||
local yPos = 0
|
||||
|
||||
for i, skillData in ipairs(editableSkills) do
|
||||
local row = skillRowFrames[i]
|
||||
if not row then
|
||||
row = CreateSkillRowFrame(i)
|
||||
skillRowFrames[i] = row
|
||||
end
|
||||
|
||||
-- Clear previous anchor points before repositioning
|
||||
row:ClearAllPoints()
|
||||
row:SetPoint("TOPLEFT", scrollChild, "TOPLEFT", 0, -yPos)
|
||||
row:SetPoint("TOPRIGHT", scrollChild, "TOPRIGHT", 0, -yPos)
|
||||
|
||||
-- Populate name
|
||||
row.nameBox:SetText(skillData.name or "")
|
||||
row.nameBox:SetScript("OnTextChanged", function(self)
|
||||
local idx = nil
|
||||
for j, s in ipairs(editableSkills) do
|
||||
if skillRowFrames[j] == row then
|
||||
idx = j
|
||||
break
|
||||
end
|
||||
end
|
||||
if idx then
|
||||
editableSkills[idx].name = self:GetText()
|
||||
end
|
||||
end)
|
||||
|
||||
-- Setup level dropdown
|
||||
local currentRowIndex = i
|
||||
row.levelDropdown:SetupMenu(function(dropdown, rootDescription)
|
||||
for _, levelName in ipairs(AltSystem.Data.SkillLevelOrder) do
|
||||
rootDescription:CreateRadio(
|
||||
levelName,
|
||||
function()
|
||||
return editableSkills[currentRowIndex] and editableSkills[currentRowIndex].level == levelName
|
||||
end,
|
||||
function()
|
||||
if editableSkills[currentRowIndex] then
|
||||
editableSkills[currentRowIndex].level = levelName
|
||||
editableSkills[currentRowIndex].value = AltSystem.Data:GetDefaultValueForLevel(levelName)
|
||||
RefreshSkillRows()
|
||||
end
|
||||
end
|
||||
)
|
||||
end
|
||||
end)
|
||||
|
||||
-- Setup value dropdown based on current level
|
||||
local range = AltSystem.Data.SkillValueRanges[skillData.level]
|
||||
if range then
|
||||
row.valueDropdown:SetupMenu(function(dropdown, rootDescription)
|
||||
for v = range.min, range.max do
|
||||
rootDescription:CreateRadio(
|
||||
tostring(v),
|
||||
function()
|
||||
return editableSkills[currentRowIndex] and editableSkills[currentRowIndex].value == v
|
||||
end,
|
||||
function()
|
||||
if editableSkills[currentRowIndex] then
|
||||
editableSkills[currentRowIndex].value = v
|
||||
end
|
||||
end
|
||||
)
|
||||
end
|
||||
end)
|
||||
end
|
||||
|
||||
-- Bind delete button for current index
|
||||
row.deleteBtn:SetScript("OnClick", function()
|
||||
table.remove(editableSkills, currentRowIndex)
|
||||
RefreshSkillRows()
|
||||
end)
|
||||
|
||||
row:Show()
|
||||
yPos = yPos + ROW_HEIGHT + ROW_SPACING
|
||||
end
|
||||
|
||||
-- Reposition Add A Row button
|
||||
if addRowButton then
|
||||
addRowButton:ClearAllPoints()
|
||||
addRowButton:SetPoint("TOPRIGHT", scrollChild, "TOPRIGHT", -PADDING, -(yPos + 4))
|
||||
end
|
||||
|
||||
-- Update scroll child height
|
||||
local totalHeight = yPos + ROW_HEIGHT + 20 -- extra space for Add A Row button
|
||||
if scrollChild then
|
||||
scrollChild:SetHeight(math.max(totalHeight, 1))
|
||||
end
|
||||
end
|
||||
|
||||
-- ─── Create Build Skills Content ────────────────────────────────────────────
|
||||
|
||||
function AltSystem:CreateBuildSkillsContent(parentFrame)
|
||||
local contentWidth = parentFrame:GetWidth() or 692
|
||||
local contentHeight = parentFrame:GetHeight() or 444
|
||||
|
||||
local yPos = -PADDING
|
||||
|
||||
-- Info text paragraph 1
|
||||
local info1 = parentFrame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||
info1:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING, yPos)
|
||||
info1:SetPoint("TOPRIGHT", parentFrame, "TOPRIGHT", -PADDING, yPos)
|
||||
info1:SetJustifyH("LEFT")
|
||||
info1:SetText("All skills below are directly extracted from your TRP's characteristics sheet. You can view and edit them there at any time.")
|
||||
info1:SetTextColor(0.9, 0.75, 0.2)
|
||||
info1:SetWordWrap(true)
|
||||
|
||||
yPos = yPos - (info1:GetStringHeight() or 16) - 10
|
||||
|
||||
-- Info text paragraph 2
|
||||
local info2 = parentFrame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||
info2:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING, yPos)
|
||||
info2:SetPoint("TOPRIGHT", parentFrame, "TOPRIGHT", -PADDING, yPos)
|
||||
info2:SetJustifyH("LEFT")
|
||||
info2:SetText("This menu serves as an easy alternative for if you want to use this system fast without diving deep into understanding it and/or styling your TRP sheet at this point in time. All your changes made will not be saved to TRP up until you hit the specific button to do so.")
|
||||
info2:SetTextColor(0.9, 0.75, 0.2)
|
||||
info2:SetWordWrap(true)
|
||||
|
||||
yPos = yPos - (info2:GetStringHeight() or 32) - 16
|
||||
|
||||
-- "Skill list" section header
|
||||
local sectionHeader = parentFrame:CreateFontString(nil, "OVERLAY", "GameFontNormalLarge")
|
||||
sectionHeader:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING, yPos)
|
||||
sectionHeader:SetText("Skill list")
|
||||
sectionHeader:SetTextColor(1, 1, 1)
|
||||
|
||||
yPos = yPos - 24
|
||||
|
||||
-- Column headers
|
||||
local nameHeader = parentFrame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||
nameHeader:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING + 8, yPos)
|
||||
nameHeader:SetText("Name")
|
||||
nameHeader:SetTextColor(0.9, 0.75, 0.2)
|
||||
|
||||
local levelHeader = parentFrame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||
levelHeader:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING + 8 + NAME_WIDTH + 12, yPos)
|
||||
levelHeader:SetText("Level")
|
||||
levelHeader:SetTextColor(0.9, 0.75, 0.2)
|
||||
|
||||
local valueHeader = parentFrame:CreateFontString(nil, "OVERLAY", "GameFontNormal")
|
||||
valueHeader:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING + 8 + NAME_WIDTH + 12 + LEVEL_WIDTH + 8, yPos)
|
||||
valueHeader:SetText("Value")
|
||||
valueHeader:SetTextColor(0.9, 0.75, 0.2)
|
||||
|
||||
yPos = yPos - 20
|
||||
|
||||
-- Save button (pinned to bottom, outside scroll frame)
|
||||
local saveButton = CreateFrame("Button", "AltSystemSaveSkillsButton", parentFrame, "UIPanelButtonTemplate")
|
||||
saveButton:SetSize(180, 30)
|
||||
saveButton:SetPoint("BOTTOM", parentFrame, "BOTTOM", 0, PADDING)
|
||||
saveButton:SetText("Save Skills to TRP")
|
||||
|
||||
-- Style the save button with a dark red tint
|
||||
local saveBg = saveButton:CreateTexture(nil, "BACKGROUND")
|
||||
saveBg:SetAllPoints()
|
||||
saveBg:SetColorTexture(0.4, 0.08, 0.08, 0.9)
|
||||
|
||||
saveButton:SetScript("OnClick", function()
|
||||
local success = AltSystem.Data:SaveSkills(editableSkills)
|
||||
if success and AltSystem.RefreshSkillDropdown then
|
||||
AltSystem:RefreshSkillDropdown()
|
||||
end
|
||||
end)
|
||||
|
||||
-- Scrollable skill list area (between column headers and save button)
|
||||
scrollFrame = CreateFrame("ScrollFrame", "AltSystemBuildSkillsScrollFrame", parentFrame, "UIPanelScrollFrameTemplate")
|
||||
scrollFrame:SetPoint("TOPLEFT", parentFrame, "TOPLEFT", PADDING, yPos)
|
||||
scrollFrame:SetPoint("BOTTOMRIGHT", parentFrame, "BOTTOMRIGHT", -PADDING - 22, saveButton:GetHeight() + PADDING + 8)
|
||||
|
||||
scrollChild = CreateFrame("Frame", "AltSystemBuildSkillsScrollChild", scrollFrame)
|
||||
scrollChild:SetWidth(scrollFrame:GetWidth() or (contentWidth - PADDING * 2 - 22))
|
||||
scrollChild:SetHeight(1)
|
||||
scrollFrame:SetScrollChild(scrollChild)
|
||||
|
||||
-- Update scroll child width when frame resizes
|
||||
scrollFrame:SetScript("OnSizeChanged", function(self)
|
||||
scrollChild:SetWidth(self:GetWidth())
|
||||
end)
|
||||
|
||||
-- "Add A Row" button (inside scroll child, below last row)
|
||||
addRowButton = CreateFrame("Button", "AltSystemAddRowButton", scrollChild, "UIPanelButtonTemplate")
|
||||
addRowButton:SetSize(130, 26)
|
||||
addRowButton:SetText("+ Add A Row")
|
||||
addRowButton:SetPoint("TOPRIGHT", scrollChild, "TOPRIGHT", -PADDING, 0)
|
||||
|
||||
-- Style add row button
|
||||
local addBg = addRowButton:CreateTexture(nil, "BACKGROUND")
|
||||
addBg:SetAllPoints()
|
||||
addBg:SetColorTexture(0.4, 0.08, 0.08, 0.9)
|
||||
|
||||
addRowButton:SetScript("OnClick", function()
|
||||
table.insert(editableSkills, {
|
||||
name = "Skillname",
|
||||
level = "Novice",
|
||||
value = 1,
|
||||
icon = "inv_misc_questionmark",
|
||||
isNew = true,
|
||||
})
|
||||
RefreshSkillRows()
|
||||
|
||||
-- Auto-scroll to bottom to show the new row
|
||||
C_Timer.After(0.05, function()
|
||||
if scrollFrame then
|
||||
scrollFrame:SetVerticalScroll(scrollFrame:GetVerticalScrollRange())
|
||||
end
|
||||
end)
|
||||
end)
|
||||
|
||||
AltSystem.BuildSkillsScrollFrame = scrollFrame
|
||||
AltSystem.BuildSkillsScrollChild = scrollChild
|
||||
end
|
||||
|
||||
-- ─── Refresh Build Skills list (called on tab switch) ───────────────────────
|
||||
|
||||
function AltSystem:RefreshBuildSkillsList()
|
||||
-- Reload skills from TRP3 into working copy, discarding any unsaved edits
|
||||
editableSkills = AltSystem.Data:GetEditableSkills()
|
||||
RefreshSkillRows()
|
||||
end
|
||||
0
Core.lua
Normal file → Executable file
0
Core.lua
Normal file → Executable file
109
Data.lua
Normal file → Executable file
109
Data.lua
Normal file → Executable file
|
|
@ -39,6 +39,27 @@ local SKILL_KEYWORD_RANGES = {
|
|||
["Master"] = { min = 20, max = 20 },
|
||||
}
|
||||
|
||||
-- Shared lookup tables for the Build Skills tab
|
||||
AltSystem.Data.SkillValueRanges = {
|
||||
["Inept"] = { min = 0, max = 0 },
|
||||
["Novice"] = { min = 1, max = 5 },
|
||||
["Adept"] = { min = 6, max = 10 },
|
||||
["Expert"] = { min = 11, max = 19 },
|
||||
["Master"] = { min = 20, max = 20 },
|
||||
}
|
||||
|
||||
-- Ordered list of skill levels for dropdown display
|
||||
AltSystem.Data.SkillLevelOrder = { "Inept", "Novice", "Adept", "Expert", "Master" }
|
||||
|
||||
-- Returns the minimum value for a given skill level
|
||||
function AltSystem.Data:GetDefaultValueForLevel(level)
|
||||
local range = AltSystem.Data.SkillValueRanges[level]
|
||||
if range then
|
||||
return range.min
|
||||
end
|
||||
return 1
|
||||
end
|
||||
|
||||
-- Check if the trait's right text field contains a valid skill keyword.
|
||||
-- Returns the matched keyword if found, nil otherwise.
|
||||
local function FindSkillKeyword(rightText)
|
||||
|
|
@ -134,6 +155,94 @@ function AltSystem.Data:RefreshSkills()
|
|||
return skills
|
||||
end
|
||||
|
||||
-- Returns an array of {name, level, value, icon} for each valid skill trait in the TRP3 profile.
|
||||
-- Excludes Base Roll and Unskilled (system-generated). Sorted by level order (Inept first, Master last).
|
||||
function AltSystem.Data:GetEditableSkills()
|
||||
local skills = {}
|
||||
|
||||
if TRP3_API and TRP3_API.profile and TRP3_API.profile.getData then
|
||||
local ok, characteristics = pcall(TRP3_API.profile.getData, "player/characteristics")
|
||||
if ok and characteristics and characteristics.PS then
|
||||
for _, trait in ipairs(characteristics.PS) do
|
||||
local skillName = trait.LT
|
||||
local numericValue = trait.V2 or 0
|
||||
local keyword = FindSkillKeyword(trait.RT)
|
||||
if skillName and skillName ~= "" and keyword then
|
||||
table.insert(skills, {
|
||||
name = skillName,
|
||||
level = keyword,
|
||||
value = numericValue,
|
||||
icon = trait.IC or "inv_misc_questionmark",
|
||||
})
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
-- Sort by level order descending (Master first, Inept last)
|
||||
local levelOrderMap = {}
|
||||
for i, lvl in ipairs(AltSystem.Data.SkillLevelOrder) do
|
||||
levelOrderMap[lvl] = i
|
||||
end
|
||||
table.sort(skills, function(a, b)
|
||||
return (levelOrderMap[a.level] or 99) > (levelOrderMap[b.level] or 99)
|
||||
end)
|
||||
|
||||
return skills
|
||||
end
|
||||
|
||||
-- Saves the edited skills back to the TRP3 profile.
|
||||
-- editedSkills is an array of {name, level, value, icon}.
|
||||
function AltSystem.Data:SaveSkills(editedSkills)
|
||||
if not TRP3_API or not TRP3_API.profile or not TRP3_API.profile.getData then
|
||||
print("|cFFFF0000AltSystem:|r TRP3 API is unavailable. Cannot save skills.")
|
||||
return false
|
||||
end
|
||||
|
||||
local ok, characteristics = pcall(TRP3_API.profile.getData, "player/characteristics")
|
||||
if not ok or not characteristics then
|
||||
print("|cFFFF0000AltSystem:|r Could not access TRP3 profile characteristics. Cannot save skills.")
|
||||
return false
|
||||
end
|
||||
|
||||
-- Ensure PS array exists
|
||||
if not characteristics.PS then
|
||||
characteristics.PS = {}
|
||||
end
|
||||
|
||||
-- Separate non-skill traits from skill traits in the existing PS
|
||||
local nonSkillTraits = {}
|
||||
for _, trait in ipairs(characteristics.PS) do
|
||||
local keyword = FindSkillKeyword(trait.RT)
|
||||
if not keyword then
|
||||
-- This is not a skill trait, preserve it
|
||||
table.insert(nonSkillTraits, trait)
|
||||
end
|
||||
end
|
||||
|
||||
-- Rebuild PS: non-skill traits first, then edited skills
|
||||
local newPS = {}
|
||||
for _, trait in ipairs(nonSkillTraits) do
|
||||
table.insert(newPS, trait)
|
||||
end
|
||||
for _, skill in ipairs(editedSkills) do
|
||||
table.insert(newPS, {
|
||||
LT = skill.name,
|
||||
RT = skill.level,
|
||||
V2 = skill.value,
|
||||
IC = skill.icon or "inv_misc_questionmark",
|
||||
})
|
||||
end
|
||||
|
||||
characteristics.PS = newPS
|
||||
|
||||
-- Refresh the Use Skills tab data
|
||||
AltSystem.Data:RefreshSkills()
|
||||
|
||||
print("|cFF00FF00AltSystem:|r Skills saved to TRP profile successfully.")
|
||||
return true
|
||||
end
|
||||
|
||||
-- Initialize with the default skill list
|
||||
AltSystem.Data.Skills = {}
|
||||
for _, skill in ipairs(DEFAULT_SKILLS) do
|
||||
|
|
|
|||
0
README.md
Normal file → Executable file
0
README.md
Normal file → Executable file
0
Roll.lua
Normal file → Executable file
0
Roll.lua
Normal file → Executable file
6
UI.lua
Normal file → Executable file
6
UI.lua
Normal file → Executable file
|
|
@ -184,9 +184,8 @@ function AltSystem:CreateMainFrame()
|
|||
buildSkillsContent:SetSize(contentWidth, tabContentHeight)
|
||||
buildSkillsContent:Hide()
|
||||
|
||||
local buildPlaceholder = buildSkillsContent:CreateFontString(nil, "OVERLAY", "GameFontHighlight")
|
||||
buildPlaceholder:SetPoint("CENTER")
|
||||
buildPlaceholder:SetText("Coming soon")
|
||||
-- Build Skills tab content (created in BuildSkillsUI.lua)
|
||||
AltSystem:CreateBuildSkillsContent(buildSkillsContent)
|
||||
|
||||
-- Tab switching logic
|
||||
local function SelectTab(tabIndex)
|
||||
|
|
@ -200,6 +199,7 @@ function AltSystem:CreateMainFrame()
|
|||
buildSkillsContent:Show()
|
||||
tabUseSkillsBg:SetColorTexture(0.3, 0.3, 0.3, 1)
|
||||
tabBuildSkillsBg:SetColorTexture(0.15, 0.15, 0.15, 1)
|
||||
AltSystem:RefreshBuildSkillsList()
|
||||
end
|
||||
end
|
||||
|
||||
|
|
|
|||
0
docs/1-interface.md
Normal file → Executable file
0
docs/1-interface.md
Normal file → Executable file
0
docs/2-skills.md
Normal file → Executable file
0
docs/2-skills.md
Normal file → Executable file
0
docs/3-announce.md
Normal file → Executable file
0
docs/3-announce.md
Normal file → Executable file
0
docs/4-redesign.md
Normal file → Executable file
0
docs/4-redesign.md
Normal file → Executable file
0
docs/5-build_skills.md
Normal file → Executable file
0
docs/5-build_skills.md
Normal file → Executable file
0
docs/Changelog.md
Normal file → Executable file
0
docs/Changelog.md
Normal file → Executable file
0
docs/build_skills_tab_design.png
Normal file → Executable file
0
docs/build_skills_tab_design.png
Normal file → Executable file
|
Before Width: | Height: | Size: 98 KiB After Width: | Height: | Size: 98 KiB |
0
docs/roll_tab_design.png
Normal file → Executable file
0
docs/roll_tab_design.png
Normal file → Executable file
|
Before Width: | Height: | Size: 76 KiB After Width: | Height: | Size: 76 KiB |
Loading…
Add table
Add a link
Reference in a new issue