Skip to content

Commit 5e69e20

Browse files
author
Ryan
committed
UI can now get Fighters, LVD finished?
1 parent adb0f53 commit 5e69e20

File tree

5 files changed

+137
-39
lines changed

5 files changed

+137
-39
lines changed

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -191,3 +191,5 @@ LvdSpec/TestMod/groundconfig.prc
191191
LvdSpec/EsnHcxjXIAA7ibb.jpg
192192

193193
*.jpg
194+
195+
LvdSpec/groundconfig.prc

LvdSpec/LvdSpec.py

+6-2
Original file line numberDiff line numberDiff line change
@@ -463,6 +463,10 @@ def SetYaml(automatic=False):
463463
#If it's the same file, do nothing
464464
if (desiredFile == originalYaml):
465465
return
466+
#If accidentally selected the Yaml version instead of the lvd verison, select yaml instead
467+
possibleYaml = desiredFile.replace(".lvd",".yaml")
468+
if (os.path.exists(possibleYaml)):
469+
desiredFile=possibleYaml
466470

467471
desiredFileName = os.path.basename(desiredFile)
468472
extension = os.path.splitext(desiredFileName)[1]
@@ -651,7 +655,7 @@ def exportGroundInfo():
651655
os.makedirs(targetLocation)
652656
targetFile = GetConfigFromYaml()
653657

654-
subcall = [parcel,"diff",sourcePrc,tempPrc,targetFile]
658+
subcall = [parcel,"diff",sourcePrc,tempPrc,targetFile.replace(".prcxml",".prcx")]
655659
with open('output.txt', 'a+') as stdout_file:
656660
process_output = subprocess.run(subcall, stdout=stdout_file, stderr=stdout_file, text=True)
657661
print(process_output.__dict__)
@@ -1180,7 +1184,7 @@ def RefreshValues():
11801184
entry.configure(state = disableSteve)
11811185
else:
11821186
entry.configure(state = "disable")
1183-
root.filemenu.entryconfig("Export Patch File To Mod", state=disableSteve)
1187+
root.filemenu.entryconfig("Export Patch File", state=disableSteve)
11841188

11851189
def Main():
11861190
print("Running main: "+root.stageName)

LvdSpec/README.md

+27-1
Original file line numberDiff line numberDiff line change
@@ -12,4 +12,30 @@ You'll need:
1212

1313
## Usage
1414

15-
When booting for the first time, the program will ask you to locate ArcExplorer, Yamlvd and Parcel. This is so we can retrieve the original groundconfig.prc, parse level data, and export the changes you make as a patch file. You'll be greated with a blank canvas, with most of the settings greyed out (as there is no level loaded). Go to File>Load Stage Collision FolderThis program will also remember the last used workspace on launch. Make sure your mod has a `stage/[stage_name]/normal/param` folder, and that folder should have an lvd or yaml file in it. If you don't have a yaml file, this program can run yamlvd for you. If yamlvd doesn't work with this stage (ie WarioWare), you can manually enter data for the camera,blastzone,and stage radius data.
15+
When booting for the first time, the program will ask you to locate ArcExplorer, Yamlvd and Parcel. This is so we can retrieve the original groundconfig.prc, parse level data, and export the changes you make as a patch file. You'll be greated with a blank canvas, with most of the settings greyed out (as there is no level loaded). Go to File>Load Stage Collision File and select a `.yaml` file to load. If your stage doesn't have a yaml file, you can select a `.lvd` file and this program will run yamlvd for you. If yamlvd doesn't work with this stage (ie Final Destination, WarioWare), you can manually enter data for the camera,blastzone,and stage data.
16+
17+
Edit the Steve LVD Settings values as you see fit. You can also use the Wizard button to automatically set these values. The Wizard will require that your Stage Data values are accurate, so double check to make sure these values make sense.
18+
19+
When you're finished, go to File>Export Patch File. This will create a patch file in whichever mod directory you are currently in under stage/common/shared/param. The name will be groundconfig + the name of the yaml file. The program will read from this file whenever you load the same yaml file again. Make sure you rename the new patch file after moving it to your mod on your SD Card. All changes will be saved to the groundconfig.prc in the LVDSpec folder, so if you're making a large mod pack, you might want to use this file instead of trying to combine several different patch files.
20+
21+
## Terms
22+
23+
| |
24+
| :- |
25+
| **Steve LVD Settings** |
26+
| **material**: Material type of the weakest block |
27+
| **origin**: Used to offset the steve grid. Should be related to Stage Radius and FloorY|
28+
| **cell sensitivity**: A value between 0 and 1. Not sure what it does|
29+
| **line offset**: A value between 0 and 10. Not sure what it does |
30+
| **Side/Top/Bottom**: Distance from Camera where Steve cannot build a block |
31+
| **Camera Boundaries** |
32+
| **Left/Right/Top/Bottom**: Boundaries for the Camera of the stage |
33+
| **Center**: Center X and Y positions of the camera |
34+
| **Blastzone Boundaries** |
35+
| **Left/Right/Top/Bottom**: Boundaries for the Blastzone of the stage |
36+
| **Stage Data** |
37+
| **Radius**: Width of the stage divided by 2 |
38+
| **Top**: Highest platform; often the highest spawn point |
39+
| **Bottom**: Lowest part of the main stage. Walled stages should have their Bottom value lower than their Camera.Bottom value |
40+
| **FloorY**: Y position of the lowest floor of the stage, often the lowest spawn point |
41+
| **Origin**: Often 0,0; some stages may be shifted up by 200 to avoid hardcoded vertices. Set this to a multiple of 10 so that the Steve Grid shows properly |

LvdSpec/groundconfig.prc

-51 KB
Binary file not shown.

UIRetrieve/UIRetrieve.py

+102-36
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,17 @@
11
import os
22
import os.path
3-
import json
3+
44
from tkinter import *
55
from tkinter import filedialog
66
from tkinter import messagebox
77
from pathlib import Path
8-
import shutil
9-
import glob
8+
109
import sys
10+
import shutil
11+
12+
#import glob
13+
import re
14+
reAlpha='[^0-9]'
1115

1216
import configparser
1317
config = configparser.ConfigParser()
@@ -31,6 +35,8 @@ def CreateConfig():
3135
root.title("UI Retrieve")
3236
root.withdraw()
3337

38+
root.modType = "stage"
39+
3440
#make sure that it is a validated destination folder, otherwise quit
3541
def IsValidArc():
3642
#Is this the directory with ArcExplorer.exe?
@@ -56,13 +62,22 @@ def setarcDir():
5662
def IsValidDestination():
5763
root.destinationDirName = os.path.basename(root.destinationDir)
5864
if (root.destinationDirName == "stage"):
59-
return False
65+
root.modType = "stage"
66+
return True
67+
elif (root.destinationDirName == "fighter"):
68+
root.modType = "fighter"
69+
return True
6070
else:
6171
subfolders = [f.path for f in os.scandir(root.destinationDir) if f.is_dir()]
6272
for dirname in list(subfolders):
6373
if (os.path.basename(dirname) == "stage"):
74+
root.modType = "stage"
75+
return True
76+
elif (os.path.basename(dirname) == "fighter"):
77+
root.modType = "fighter"
6478
return True
6579
return False
80+
6681

6782
#open folder dialogue
6883
def setDestinationDir():
@@ -74,7 +89,7 @@ def setDestinationDir():
7489
if (IsValidDestination() == False):
7590
messagebox.showerror(root.title(),"Please select the root of your mod's folder! This folder should contain a stage or fighter folder within it!")
7691
root.destroy()
77-
sys.exit("Not a stage folder")
92+
sys.exit("Not a stage/fighter folder")
7893

7994
#Set Arc Directory if needed
8095
root.arcDir = config["DEFAULT"]["arcDir"]
@@ -92,26 +107,63 @@ def setDestinationDir():
92107

93108
#Set Destination Dir
94109
root.destinationDir = ""
110+
root.modSkins = []
111+
112+
def GetModSkins():
113+
modelFolder = root.destinationDir+"/fighter/"+root.modName+"/model"
114+
print(modelFolder)
115+
bodyFolders = [b.path for b in os.scandir(modelFolder+"/body") if b.is_dir()]
116+
for fname in list(bodyFolders):
117+
fname = os.path.basename(fname).replace("c0","")
118+
fname = re.sub(reAlpha, '', fname)
119+
120+
#If empty, skip
121+
if (fname==""): continue
122+
123+
#Make sure the value is a valid int
124+
validValue=True
125+
try:
126+
skinID = int(fname) % 8
127+
except:
128+
validValue=False
129+
if (not validValue): continue
130+
131+
skinName = "0"+str(skinID)
132+
print("found skin "+skinName)
133+
root.modSkins.append(skinName)
134+
95135

96136
#Get or Set root.destinationDir
97137
if (root.destinationDir == ""):
98138
setDestinationDir()
99139

140+
root.fighterUINames = [0,1,2,3,4,6]
100141
def combUIFolder(target,uiArray,folderUI):
101-
ui_Folder = root.arcDir + r"/export/ui/" + folderUI + r"/stage"
102-
#comb through each stage_N folder to find files
142+
modType = "chara" if root.modType=="fighter" else "stage"
143+
ui_Folder = root.arcDir + r"/export/ui/" + folderUI + r"/"+modType
144+
#comb through each type_N folder to find files
103145
subfolders = [f.path for f in os.scandir(ui_Folder) if f.is_dir()]
104146
for folder in list(subfolders):
105-
targetName = folder+r"/"+os.path.basename(folder)+"_"+target+".bntx"
106-
if (os.path.exists(targetName)):
107-
uiArray.append(targetName)
147+
#For stages, we don't need to account for each skin
148+
if (len(root.modSkins)==0):
149+
targetName = folder+r"/"+os.path.basename(folder)+"_"+target+".bntx"
150+
if (os.path.exists(targetName)):
151+
uiArray.append(targetName)
152+
#Fighters on the other hand...ooooo boy
153+
else:
154+
for skin in root.modSkins:
155+
targetName = folder+r"/"+os.path.basename(folder)+"_"+target+"_"+skin+".bntx"
156+
if (os.path.exists(targetName)):
157+
uiArray.append(targetName)
158+
159+
108160
return uiArray
109161

110162
def getUIDump(target):
111163
uiArray = []
112164
isDLC = False
113165

114-
#comb through each stage_N folder to find files
166+
#comb through each stage/fighter_N folder to find files
115167
uiArray = combUIFolder(target,uiArray,"replace")
116168

117169
if (len(uiArray)==0):
@@ -124,25 +176,32 @@ def getUIDump(target):
124176
def getUI(quitOnFail=False):
125177
#Get source UI
126178
uiArray=[]
127-
uiArray,isDLC = getUIDump(root.stageName)
179+
uiArray,isDLC = getUIDump(root.modName)
128180
if (len(uiArray)==0):
129181
if (quitOnFail==False):
182+
print("Could not find UI for "+root.modName+", searching manually")
130183
return
131184
else:
132-
messagebox.showinfo(root.title(),"Could not find UI for that stage")
185+
messagebox.showinfo(root.title(),"Could not find UI for that "+root.modType)
133186
root.destroy()
134187
sys.exit("no ui found")
135188

136189
folderBase = "replace"
137190
folderDLC = "replace_patch"
138191
folderUI = folderDLC if isDLC else folderBase
139192

193+
if (root.modType == "fighter"):
194+
root.modType = "chara"
140195

141196
destinationFolders = []
142-
desitnationParent = root.destinationDir+r"/ui/"+folderUI+ r"/stage"
197+
desitnationParent = root.destinationDir+r"/ui/"+folderUI+ r"/"+root.modType
198+
199+
folderNames = range(0,5)
200+
if (root.modType == "chara"):
201+
folderNames = [0,1,2,3,4,6]
143202
#Populate folders as necessary
144-
for i in range(0,5):
145-
path = desitnationParent+r"/stage_"+str(i)
203+
for i in folderNames:
204+
path = desitnationParent+r"/"+root.modType+"_"+str(i)
146205
destinationFolders.append(path)
147206
os.makedirs(path, exist_ok=True)
148207

@@ -164,7 +223,7 @@ def getUI(quitOnFail=False):
164223
shutil.copy(file,d+r"/"+os.path.basename(file))
165224
break
166225

167-
messagebox.showinfo(root.title(),"UI Retrieved for "+root.stageName+"!")
226+
messagebox.showinfo(root.title(),"UI Retrieved for "+root.modName+"!")
168227
#open folder
169228
import webbrowser
170229
webbrowser.open(desitnationParent)
@@ -173,34 +232,41 @@ def getUI(quitOnFail=False):
173232
root.destroy()
174233
sys.exit("success!")
175234

176-
def manualUI():
177-
root.stageName = root.e.get()
178-
root.withdraw()
179-
print (root.stageName )
180-
if (root.stageName==None or root.stageName == ""):
181-
root.destroy()
182-
sys.exit("no input")
183-
getUI(True)
184235

185236

186-
root.stageName = ""
237+
root.modName = ""
187238
subfolders = [f.path for f in os.scandir(root.destinationDir) if f.is_dir()]
188239
for dirname in list(subfolders):
189240
if (os.path.basename(dirname) == "stage"):
190-
stagesubfolder = [s.path for s in os.scandir(dirname) if s.is_dir()]
191-
root.stageName = os.path.basename(stagesubfolder[0])
192-
if (root.stageName == "common" and len(stagesubfolder)>1):
193-
print("skip common")
194-
root.stageName = os.path.basename(stagesubfolder[1])
195-
196-
print (root.stageName )
197-
if (root.stageName!=""):
241+
modsubfolder = [s.path for s in os.scandir(dirname) if s.is_dir()]
242+
root.modName = os.path.basename(modsubfolder[0])
243+
if (root.modName == "common" and len(modsubfolder)>1):
244+
root.modName = os.path.basename(modsubfolder[1])
245+
elif (os.path.basename(dirname) == "fighter"):
246+
modsubfolder = [s.path for s in os.scandir(dirname) if s.is_dir()]
247+
root.modName = os.path.basename(modsubfolder[0])
248+
if (root.modName == "common" and len(modsubfolder)>1):
249+
root.modName = os.path.basename(modsubfolder[1])
250+
251+
print ("Selected mod: "+root.modName )
252+
if (root.modType == "fighter"):
253+
GetModSkins()
254+
if (root.modName!=""):
198255
getUI()
256+
#GetUI exists system on completion
199257

200258
#Create UI for manually if no stage/fighter name was found
201-
messagebox.showinfo(root.title(),"Could not find stage associated with this mod, please type in the stage name manually on the next window")
259+
def manualUI():
260+
root.modName = root.e.get()
261+
root.withdraw()
262+
if (root.modName==None or root.modName == ""):
263+
root.destroy()
264+
sys.exit("no input")
265+
getUI(True)
266+
267+
messagebox.showinfo(root.title(),"Could not find "+root.modType+" associated with this mod, please type in the "+root.modType+" name manually on the next window")
202268
root.deiconify()
203-
root.label = Label(root, text="Type in the name of the stage you want to search (ie battlefield_s)", anchor=N)
269+
root.label = Label(root, text="Type in the name of the "+root.modType+" you want to destination (ie ridley,battlefield_s,etc)", anchor=N)
204270
root.label.pack(side = TOP)
205271
root.e = Entry(root,width =50)
206272
root.e.pack()

0 commit comments

Comments
 (0)