Initial commit
This commit is contained in:
commit
83052a0240
3 changed files with 343 additions and 0 deletions
7
README.md
Normal file
7
README.md
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
# SIF Tools
|
||||||
|
### a set of Python script to help players of [Love Live School Idol Festival](http://www.school-fes.klabgames.net)
|
||||||
|
### By Donald Burr <dburr@DonaldBurr.com>
|
||||||
|
|
||||||
|
## What is this?
|
||||||
|
|
||||||
|
This is a set of Python scripts to help players who play the mobile rhythm game [Love Live School Idol Festival](http://www.school-fes.klabgames.net).
|
158
cardlevelcalc.py
Executable file
158
cardlevelcalc.py
Executable file
|
@ -0,0 +1,158 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# Card level-up requirements (EXP) calculator
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import getopt
|
||||||
|
|
||||||
|
# The EXP tables
|
||||||
|
# data from: http://www59.atwiki.jp/lovelive-sif/pages/32.html
|
||||||
|
# these have -1 as the first value because python arrays are 0-based and I am using <level> as the index
|
||||||
|
# and am too lazy to do <level-1> to account for the 0-based-ness :P
|
||||||
|
exp_table_n = [-1, 0, 6, 18, 28, 40, 51, 61, 72, 82, 93, 104, 114, 124, 135, 145, 156, 165, 176, 187, 196, 207, 217, 226, 238, 247, 257, 268, 277, 288, 297, 308, 317, 328, 337, 348, 358, 367, 377, 388, 397]
|
||||||
|
exp_table_r = [-1, 0, 14, 31, 45, 55, 67, 76, 85, 94, 103, 110, 119, 125, 134, 140, 148, 155, 161, 168, 174, 181, 187, 193, 199, 206, 211, 217, 223, 228, 235, 240, 245, 251, 256, 262, 267, 272, 277, 283, 288, 292, 298, 303, 308, 313, 317, 323, 327, 332, 337, 342, 346, 351, 356, 360, 365, 370, 374, 378, 383]
|
||||||
|
exp_table_sr = [-1, 0, 54, 98, 127, 150, 169, 187, 203, 218, 232, 245, 257, 269, 281, 291, 302, 311, 322, 331, 340, 349, 358, 366, 374, 383, 391, 398, 406, 413, 421, 428, 435, 442, 449, 456, 462, 469, 475, 482, 488, 494, 500, 507, 512, 518, 524, 530, 536, 541, 547, 552, 558, 563, 568, 574, 579, 584, 590, 594, 600, 605, 609, 615, 619, 625, 629, 634, 639, 643, 648, 653, 657, 662, 667, 670, 676, 680, 684, 689, 693]
|
||||||
|
exp_table_ur = [-1, 0, 201, 294, 345, 382, 411, 438, 460, 481, 499, 517, 532, 547, 561, 574, 587, 598, 611, 621, 631, 642, 651, 661, 670, 679, 687, 696, 704, 712, 720, 727, 734, 742, 749, 755, 763, 769, 775, 782, 788, 794, 800, 806, 812, 818, 823, 829, 834, 840, 845, 850, 856, 860, 866, 870, 875, 880, 885, 890, 894, 899, 903, 908, 912, 917, 921, 925, 930, 933, 938, 942, 946, 950, 954, 959, 961, 966, 970, 974, 977, 981, 985, 988, 992, 996, 999, 1003, 1006, 1010, 1013, 1017, 1020, 1024, 1027, 1030, 1034, 1037, 1040, 1043, 1047]
|
||||||
|
|
||||||
|
# max levels
|
||||||
|
# note we don't check whether the card is idolized or not
|
||||||
|
# we assume that the user knows what they're doing
|
||||||
|
level_cap_n = 40
|
||||||
|
level_cap_r = 60
|
||||||
|
level_cap_sr = 80
|
||||||
|
level_cap_ur = 100
|
||||||
|
|
||||||
|
def check_level_cap(rarity, level):
|
||||||
|
return_value = False
|
||||||
|
if level >= 1:
|
||||||
|
if rarity == "N":
|
||||||
|
return_value = (level <= level_cap_n)
|
||||||
|
elif rarity == "R":
|
||||||
|
return_value = (level <= level_cap_r)
|
||||||
|
elif rarity == "SR":
|
||||||
|
return_value = (level <= level_cap_sr)
|
||||||
|
elif rarity == "UR":
|
||||||
|
return_value = (level <= level_cap_ur)
|
||||||
|
return return_value
|
||||||
|
|
||||||
|
def check_valid_exp(rarity, level, exp):
|
||||||
|
return_value = False
|
||||||
|
if rarity == "N" and check_level_cap(rarity, level):
|
||||||
|
return_value = (exp >= 0 and exp < exp_table_n[level])
|
||||||
|
elif rarity == "R" and check_level_cap(rarity, level):
|
||||||
|
return_value = (exp >= 0 and exp < exp_table_r[level])
|
||||||
|
elif rarity == "SR" and check_level_cap(rarity, level):
|
||||||
|
return_value = (exp >= 0 and exp < exp_table_sr[level])
|
||||||
|
elif rarity == "UR" and check_level_cap(rarity, level):
|
||||||
|
return_value = (exp >= 0 and exp < exp_table_ur[level])
|
||||||
|
return return_value
|
||||||
|
|
||||||
|
def calc(rarity, starting_level, starting_exp, desired_level):
|
||||||
|
# assumes that all values fed into it have been checked already
|
||||||
|
required_exp = 0
|
||||||
|
# desired_level+1 because python ranges are not inclusive :P
|
||||||
|
for level in range(starting_level+1, desired_level+1):
|
||||||
|
if rarity == "N":
|
||||||
|
required_exp = required_exp + exp_table_n[level]
|
||||||
|
elif rarity == "R":
|
||||||
|
required_exp = required_exp + exp_table_r[level]
|
||||||
|
elif rarity == "SR":
|
||||||
|
required_exp = required_exp + exp_table_sr[level]
|
||||||
|
elif rarity == "UR":
|
||||||
|
required_exp = required_exp + exp_table_ur[level]
|
||||||
|
#print "WE ARE AT LEVEL %d and we need %d exp" % (level, required_exp)
|
||||||
|
# subtract what we already have
|
||||||
|
required_exp = required_exp - starting_exp
|
||||||
|
# now tell the user
|
||||||
|
print "To get a %s card from level %d (with %d EXP) to %d requires %d EXP." % (rarity, starting_level, starting_exp, desired_level, required_exp)
|
||||||
|
# calculate equivalent N cards
|
||||||
|
number_of_n_cards = required_exp // 100
|
||||||
|
if number_of_n_cards == 0:
|
||||||
|
number_of_n_cards = 1
|
||||||
|
print "(the equivalent of approx. %d level-1 N cards fed to it)" % number_of_n_cards
|
||||||
|
|
||||||
|
def usage():
|
||||||
|
print "Usage: %s [options]" % os.path.basename(__file__)
|
||||||
|
print "where [options] can be one or more of:"
|
||||||
|
print "[-H | --help] Print this help message"
|
||||||
|
print "[-r | --rarity] Card's rarity (REQUIRED, must be one of: N, R, SR, UR)"
|
||||||
|
print "[-l | --starting-level] Card's starting level (REQUIRED)"
|
||||||
|
print "[-e | --starting-exp] Card's starting EXP (optional, defaults to 0)"
|
||||||
|
print "[-L | --desired-level] Card's desired level (REQUIRED)"
|
||||||
|
|
||||||
|
def main(argv):
|
||||||
|
rarity = None
|
||||||
|
starting_level = None
|
||||||
|
desired_level = None
|
||||||
|
starting_exp = 0
|
||||||
|
try:
|
||||||
|
options, remainder = getopt.getopt(argv, "Hr:l:e:L:", ["help", "rarity=", "starting-level=", "starting-exp=", "desired-level="])
|
||||||
|
except getopt.GetoptError:
|
||||||
|
usage()
|
||||||
|
sys.exit(2)
|
||||||
|
for opt, arg in options:
|
||||||
|
if opt in ('-H', '--help'):
|
||||||
|
usage()
|
||||||
|
sys.exit(0)
|
||||||
|
elif opt in ('-r', '--rarity'):
|
||||||
|
rarity = arg
|
||||||
|
elif opt in ('-l', '--starting-level'):
|
||||||
|
starting_level = int(arg)
|
||||||
|
elif opt in ('-e', '--starting-exp'):
|
||||||
|
starting_exp = int(arg)
|
||||||
|
elif opt in ('-L', '--desired-level'):
|
||||||
|
desired_level = int(arg)
|
||||||
|
|
||||||
|
# first validate rarity
|
||||||
|
if rarity is None:
|
||||||
|
print "Error: must specify rarity"
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
# canonicalize it to uppercase
|
||||||
|
rarity = rarity.upper()
|
||||||
|
if rarity != "N" and rarity != "R" and rarity != "SR" and rarity != "UR":
|
||||||
|
print "Error: invalid rarity specified (%s)" % rarity
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# now validate starting level
|
||||||
|
if starting_level is None:
|
||||||
|
print "Error: must specify starting level"
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
elif not check_level_cap(rarity, starting_level):
|
||||||
|
print "Error: invalid starting level: %d" % starting_level
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# now validate starting level
|
||||||
|
if desired_level is None:
|
||||||
|
print "Error: must specify desired level"
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
elif not check_level_cap(rarity, desired_level):
|
||||||
|
print "Error: invalid desired level: %d" % desired_level
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# now do start+desired levels make sense?
|
||||||
|
if desired_level <= starting_level:
|
||||||
|
print "Error: desired level must be greater than starting level"
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# finally check to see if exp makes sense (can't be >= the number of exp for the next level)
|
||||||
|
if not check_valid_exp(rarity, desired_level, starting_exp):
|
||||||
|
print "Error: invalid EXP (%d)" % starting_exp
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
# all is well, go for it
|
||||||
|
calc(rarity, starting_level, starting_exp, desired_level)
|
||||||
|
|
||||||
|
### main script starts here
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main(sys.argv[1:])
|
178
eventcalc.py
Executable file
178
eventcalc.py
Executable file
|
@ -0,0 +1,178 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
#
|
||||||
|
# https://docs.python.org/2/library/curses.html
|
||||||
|
|
||||||
|
import time
|
||||||
|
import datetime
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import getopt
|
||||||
|
import curses
|
||||||
|
|
||||||
|
scr = None
|
||||||
|
width = None
|
||||||
|
height = None
|
||||||
|
|
||||||
|
def center(string):
|
||||||
|
global width
|
||||||
|
global height
|
||||||
|
string_length = len(string)
|
||||||
|
horizontal_space = (width // 2) - (string_length // 2)
|
||||||
|
print "%s%s" % ((" " * horizontal_space), string)
|
||||||
|
|
||||||
|
def getTerminalSize():
|
||||||
|
import os
|
||||||
|
env = os.environ
|
||||||
|
def ioctl_GWINSZ(fd):
|
||||||
|
try:
|
||||||
|
import fcntl, termios, struct, os
|
||||||
|
cr = struct.unpack('hh', fcntl.ioctl(fd, termios.TIOCGWINSZ,
|
||||||
|
'1234'))
|
||||||
|
except:
|
||||||
|
return
|
||||||
|
return cr
|
||||||
|
cr = ioctl_GWINSZ(0) or ioctl_GWINSZ(1) or ioctl_GWINSZ(2)
|
||||||
|
if not cr:
|
||||||
|
try:
|
||||||
|
fd = os.open(os.ctermid(), os.O_RDONLY)
|
||||||
|
cr = ioctl_GWINSZ(fd)
|
||||||
|
os.close(fd)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
if not cr:
|
||||||
|
cr = (env.get('LINES', 25), env.get('COLUMNS', 80))
|
||||||
|
|
||||||
|
### Use get(key[, default]) instead of a try/catch
|
||||||
|
#try:
|
||||||
|
# cr = (env['LINES'], env['COLUMNS'])
|
||||||
|
#except:
|
||||||
|
# cr = (25, 80)
|
||||||
|
return int(cr[1]), int(cr[0])
|
||||||
|
|
||||||
|
def calc(year, month, day, hour, minute, fullscreen = False):
|
||||||
|
global height
|
||||||
|
today = datetime.datetime.utcnow()
|
||||||
|
today_hour = today.hour
|
||||||
|
today_minute = today.minute
|
||||||
|
today_seconds = today.second
|
||||||
|
delta = datetime.datetime(year, month, day, hour, minute, 0) - today
|
||||||
|
s = delta.total_seconds()
|
||||||
|
days, days_remainder = divmod(s, 86400)
|
||||||
|
days_hours, days_hours_remainder = divmod(days_remainder, 3600)
|
||||||
|
days_minutes, days_seconds = divmod(days_hours_remainder, 60)
|
||||||
|
hours, remainder = divmod(s, 3600)
|
||||||
|
minutes, seconds = divmod(remainder, 60)
|
||||||
|
if fullscreen:
|
||||||
|
absolutely_unused_variable = os.system("clear")
|
||||||
|
number_of_blank_lines = (height // 2) - (8 // 2) + 1
|
||||||
|
for x in range(1, number_of_blank_lines):
|
||||||
|
print ""
|
||||||
|
center("CURRENT TIME")
|
||||||
|
center(today.strftime("%Y-%m-%d %H:%M:%S UTC"))
|
||||||
|
print("")
|
||||||
|
center("EVENT ENDS")
|
||||||
|
center("%04d-%02d-%02d %02d:%02d:%02d UTC" % (year, month, day, hour, minute, 0))
|
||||||
|
print("")
|
||||||
|
center("TIME REMAINING")
|
||||||
|
if s <= 0:
|
||||||
|
center("*** EVENT HAS ENDED ***")
|
||||||
|
else:
|
||||||
|
center("%d:%02d:%02d (%dd%dh%dm)" % (hours, minutes, seconds, days, days_hours, days_minutes))
|
||||||
|
skip_lines = number_of_blank_lines
|
||||||
|
if today_minute == 35:
|
||||||
|
skip_lines = skip_lines - 3
|
||||||
|
print ""
|
||||||
|
center("Check @sifen_trackbot for a tier update:")
|
||||||
|
center("https://twitter.com/sifen_trackbot")
|
||||||
|
if hours > 0 and hours < 6:
|
||||||
|
skip_lines = skip_lines - 2
|
||||||
|
print ""
|
||||||
|
center("*** THE RUSH IS ON!!! ***")
|
||||||
|
if today_hour == 0 and today_minute == 0:
|
||||||
|
skip_lines = skip_lines - 2
|
||||||
|
print ""
|
||||||
|
if today_seconds % 2 == 0:
|
||||||
|
print ""
|
||||||
|
else:
|
||||||
|
absolutely_unused_variable = os.system("afplay /System/Library/Sounds/Glass.aiff 2>/dev/null &")
|
||||||
|
center("*** Be sure and claim your daily Login Gift ***")
|
||||||
|
for x in range(1, skip_lines):
|
||||||
|
print ""
|
||||||
|
else:
|
||||||
|
print " Current time: %s" % today.strftime("%Y-%m-%d %H:%M:%S UTC")
|
||||||
|
print " Event ends: %04d-%02d-%02d %02d:%02d:%02d UTC" % (year, month, day, hour, minute, 0)
|
||||||
|
if s <= 0:
|
||||||
|
print "Time remaining: *** EVENT HAS ENDED ***"
|
||||||
|
else:
|
||||||
|
print "Time remaining: %d:%02d:%02d (%dd%dh%dm)" % (hours, minutes, seconds, days, days_hours, days_minutes)
|
||||||
|
if hours < 6:
|
||||||
|
print "*** THE RUSH IS ON!!! ***"
|
||||||
|
|
||||||
|
def usage():
|
||||||
|
print "Usage: %s [options]" % os.path.basename(__file__)
|
||||||
|
print "where [options] can be one or more of:"
|
||||||
|
print " [-H | --help] Print this help message"
|
||||||
|
print " [-y | --year] Event year (optional, defaults to current year)"
|
||||||
|
print " [-M | --month] Event month (optional, defaults to current month)"
|
||||||
|
print " [-d | --day] Event day (REQUIRED)"
|
||||||
|
print " [-h | --hour] Event hour (REQUIRED)"
|
||||||
|
print " [-m | --minute] Event minute (optional, defaults to 0)"
|
||||||
|
print " [-c | --continuous] Continuously updating display (defaults to off)"
|
||||||
|
print "Note: event time should be specified in UTC."
|
||||||
|
|
||||||
|
def main(argv):
|
||||||
|
global scr
|
||||||
|
global width
|
||||||
|
global height
|
||||||
|
year = datetime.datetime.utcnow().year
|
||||||
|
month = datetime.datetime.utcnow().month
|
||||||
|
day = None
|
||||||
|
hour = None
|
||||||
|
minute = 0
|
||||||
|
continuous_mode = False
|
||||||
|
#scr = curses.initscr()
|
||||||
|
(width, height) = getTerminalSize()
|
||||||
|
#(width, height) = getmaxyx()
|
||||||
|
try:
|
||||||
|
options, remainder = getopt.getopt(argv, "Hy:M:d:h:m:c", ["help", "year=", "month=", "day=", "hour=", "minute=", "continuous"])
|
||||||
|
except getopt.GetoptError:
|
||||||
|
usage()
|
||||||
|
sys.exit(2)
|
||||||
|
for opt, arg in options:
|
||||||
|
if opt in ('-H', '--help'):
|
||||||
|
usage()
|
||||||
|
sys.exit(0)
|
||||||
|
elif opt in ('-y', '--year'):
|
||||||
|
year = int(arg)
|
||||||
|
elif opt in ('-M', '--month'):
|
||||||
|
month = int(arg)
|
||||||
|
elif opt in ('-d', '--day'):
|
||||||
|
day = int(arg)
|
||||||
|
elif opt in ('-h', '--hour'):
|
||||||
|
hour = int(arg)
|
||||||
|
elif opt in ('-m', '--minute'):
|
||||||
|
minute = int(arg)
|
||||||
|
elif opt in ('-c', '--continuous'):
|
||||||
|
continuous_mode = True
|
||||||
|
|
||||||
|
if day is None:
|
||||||
|
print "Error: must specify day"
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
elif hour is None:
|
||||||
|
print "Error: must specify hour"
|
||||||
|
usage()
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
if continuous_mode:
|
||||||
|
while True:
|
||||||
|
absolutely_unused_variable = os.system("clear")
|
||||||
|
calc(year, month, day, hour, minute, True)
|
||||||
|
time.sleep(1)
|
||||||
|
else:
|
||||||
|
calc(year, month, day, hour, minute)
|
||||||
|
|
||||||
|
### main script starts here
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main(sys.argv[1:])
|
Loading…
Add table
Add a link
Reference in a new issue