forked from Toufool/AutoSplit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsplit_parser.py
227 lines (174 loc) · 7.36 KB
/
split_parser.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
from __future__ import annotations
import os
from typing import TYPE_CHECKING, TypeVar
import error_messages
from AutoSplitImage import RESET_KEYWORD, START_KEYWORD, AutoSplitImage, ImageType
from utils import is_valid_image
if TYPE_CHECKING:
from AutoSplit import AutoSplit
[
DUMMY_FLAG,
BELOW_FLAG,
PAUSE_FLAG,
*_,
] = [1 << i for i in range(31)] # 32 bits of flags
T = TypeVar("T", str, int, float)
# Note, the following symbols cannot be used in a filename:
# / \ : * ? " < > |
def __value_from_filename(
filename: str,
delimiters: str,
default_value: T,
) -> T:
if len(delimiters) != 2:
raise ValueError("delimiters parameter must contain exactly 2 characters")
try:
value_type = type(default_value)
return value_type(filename.split(delimiters[0], 1)[1].split(delimiters[1])[0])
except (IndexError, ValueError):
return default_value
def threshold_from_filename(filename: str):
"""
Retrieve the threshold from the filename, if there is no threshold or the threshold
doesn't meet the requirements of being [0, 1], then None is returned.
@param filename: String containing the file's name
@return: A valid threshold, if not then None
"""
# Check to make sure there is a valid floating point number between
# parentheses of the filename
value = __value_from_filename(filename, "()", -1.0)
# Check to make sure if it is a valid threshold
return value if 0.0 <= value <= 1.0 else None
def pause_from_filename(filename: str):
"""
Retrieve the pause time from the filename, if there is no pause time or the pause time
isn't a valid positive number or 0, then None is returned.
@param filename: String containing the file's name
@return: A valid pause time, if not then None
"""
# Check to make sure there is a valid pause time between brackets
# of the filename
value = __value_from_filename(filename, "[]", -1.0)
# Pause times should always be positive or zero
return value if value >= 0.0 else None
def delay_time_from_filename(filename: str):
"""
Retrieve the delay time from the filename, if there is no delay time or the delay time
isn't a valid positive number or 0 number, then None is returned.
@param filename: String containing the file's name
@return: A valid delay time, if not then none
"""
# Check to make sure there is a valid delay time between brackets
# of the filename
value = __value_from_filename(filename, "##", -1)
# Delay times should always be positive or zero
return value if value >= 0 else None
def loop_from_filename(filename: str):
"""
Retrieve the number of loops from filename, if there is no loop number or the loop number isn't valid,
then 1 is returned.
@param filename: String containing the file's name
@return: A valid loop number, if not then 1
"""
# Check to make sure there is a valid delay time between brackets
# of the filename
value = __value_from_filename(filename, "@@", 1)
# Loop should always be positive
return value if value >= 1 else 1
def comparison_method_from_filename(filename: str):
"""
Retrieve the comparison method index from filename, if there is no comparison method or the index isn't valid,
then None is returned.
@param filename: String containing the file's name
@return: A valid comparison method index, if not then none
"""
# Check to make sure there is a valid delay time between brackets
# of the filename
value = __value_from_filename(filename, "^^", -1)
# Comparison method should always be positive or zero
return value if value >= 0 else None
def flags_from_filename(filename: str):
"""
Retrieve the flags from the filename, if there are no flags then 0 is returned
@param filename: String containing the file's name
@return: The flags as an integer, if invalid flags are found it returns 0
list of flags:
"d" = dummy, do nothing when this split is found
"b" = below threshold, after threshold is met, split when it goes below the threhsold.
"p" = pause, hit pause key when this split is found
"""
# Check to make sure there are flags between curly braces
# of the filename
flags_str = __value_from_filename(filename, "{}", "")
if not flags_str:
return 0
flags = 0x00
for flag_str in flags_str:
character = flag_str.upper()
if character == "D":
flags |= DUMMY_FLAG
elif character == "B":
flags |= BELOW_FLAG
elif character == "P":
flags |= PAUSE_FLAG
# Legacy flags
elif character == "M":
continue
else:
# An invalid flag was caught, this filename was written incorrectly
# return 0. We don't want to interpret any misleading filenames
return 0
# Check for any conflicting flags that were set
# For instance, we can't have a dummy split also pause
if (flags & DUMMY_FLAG == DUMMY_FLAG) and (flags & PAUSE_FLAG == PAUSE_FLAG):
return 0
return flags
def __pop_image_type(split_image: list[AutoSplitImage], image_type: ImageType):
for image in split_image:
if image.image_type == image_type:
split_image.remove(image)
return image
return None
def parse_and_validate_images(autosplit: AutoSplit):
# Get split images
all_images = [
AutoSplitImage(os.path.join(autosplit.settings_dict["split_image_directory"], image_name))
for image_name
in os.listdir(autosplit.settings_dict["split_image_directory"])
]
# Find non-split images and then remove them from the list
autosplit.start_image = __pop_image_type(all_images, ImageType.START)
autosplit.reset_image = __pop_image_type(all_images, ImageType.RESET)
autosplit.split_images = all_images
# Make sure that each of the images follows the guidelines for correct format
# according to all of the settings selected by the user.
for image in autosplit.split_images:
# Test for image without transparency
if not is_valid_image(image.byte_array):
autosplit.gui_changes_on_reset()
return False
# error out if there is a {p} flag but no pause hotkey set and is not auto controlled.
if (
not autosplit.settings_dict["pause_hotkey"]
and image.check_flag(PAUSE_FLAG)
and not autosplit.is_auto_controlled
):
autosplit.gui_changes_on_reset()
error_messages.pause_hotkey()
return False
# Check that there's only one reset image
if image.image_type == ImageType.RESET:
# If there is no reset hotkey set but a reset image is present, and is not auto controlled, throw an error.
if not autosplit.settings_dict["reset_hotkey"] and not autosplit.is_auto_controlled:
autosplit.gui_changes_on_reset()
error_messages.reset_hotkey()
return False
autosplit.gui_changes_on_reset()
error_messages.multiple_keyword_images(RESET_KEYWORD)
return False
# Check that there's only one start image
if image.image_type == ImageType.START:
autosplit.gui_changes_on_reset()
error_messages.multiple_keyword_images(START_KEYWORD)
return False
return True