|
@@ -8,19 +8,20 @@ from copy import deepcopy
|
|
|
|
|
|
|
|
|
charsheet = {
|
|
|
- 'bab': 9,
|
|
|
- 'str': 20,
|
|
|
- 'dex': 14,
|
|
|
- 'con': 14,
|
|
|
- 'int': 16,
|
|
|
+ 'bab': 14,
|
|
|
+ 'str': 23,
|
|
|
+ 'dex': 18,
|
|
|
+ 'con': 16,
|
|
|
+ 'int': 18,
|
|
|
'wis': 10,
|
|
|
'cha': 10,
|
|
|
'level': [
|
|
|
- ('fighter', 10)
|
|
|
+ ('fighter', 14)
|
|
|
],
|
|
|
'feat': [
|
|
|
'toughness',
|
|
|
'armor proficiency, medium',
|
|
|
+ 'armor proficiency, light',
|
|
|
'power attack',
|
|
|
'combat stamina',
|
|
|
'improved unarmed strike',
|
|
@@ -53,10 +54,20 @@ charsheet = {
|
|
|
'martial focus',
|
|
|
'weapon training class feature',
|
|
|
'combat reflexes',
|
|
|
+ 'dodge',
|
|
|
+ 'artful dodge',
|
|
|
+ 'cut from the air',
|
|
|
+ 'spellcut',
|
|
|
+ 'greater weapon focus',
|
|
|
+ 'greater weapon focus with selected weapon',
|
|
|
+ 'greater weapon specialization',
|
|
|
+ 'greater weapon specialization with selected weapon',
|
|
|
'you have no levels in a class that has the grit class feature',
|
|
|
'any combat feat',
|
|
|
'any good alignment',
|
|
|
'bravery +1 class feature',
|
|
|
+ 'bravery +2 class feature',
|
|
|
+ 'bravery +3 class feature',
|
|
|
'human',
|
|
|
'medium size',
|
|
|
'no levels in a class that has the favored enemy class feature',
|
|
@@ -64,39 +75,45 @@ charsheet = {
|
|
|
'nonlawful',
|
|
|
'proficiency with selected weapon',
|
|
|
'proficient with all martial weapons',
|
|
|
+ 'martial weapon proficiency',
|
|
|
'proficiency with weapon',
|
|
|
'proficiency with chosen weapon',
|
|
|
'proficiency with armor spikes',
|
|
|
'proficient with armor spikes',
|
|
|
'proficient with scimitar',
|
|
|
'proficient with sing',
|
|
|
+ 'proficient with sling',
|
|
|
'susceptibility to bleed damage',
|
|
|
'and proficiency with the selected weapon.',
|
|
|
'bravery class feature',
|
|
|
+ 'proficiency with medium',
|
|
|
],
|
|
|
'skill': [
|
|
|
- ('knowledge (arcane)', 9),
|
|
|
- ('knowledge (dungeoneering)', 9),
|
|
|
- ('knowledge (local)', 9),
|
|
|
- ('knowledge (nature)', 9),
|
|
|
- ('knowledge (planes)', 9),
|
|
|
- ('knowledge (religion)', 9),
|
|
|
- ('knowledge (engineering)', 1),
|
|
|
+ ('knowledge (arcane)', 14),
|
|
|
+ ('knowledge (dungeoneering)', 14),
|
|
|
+ ('knowledge (local)', 14),
|
|
|
+ ('knowledge (nature)', 14),
|
|
|
+ ('knowledge (planes)', 14),
|
|
|
+ ('knowledge (religion)', 14),
|
|
|
+ ('knowledge (engineering)', 2),
|
|
|
('knowledge (geography)', 2),
|
|
|
- ('knowledge (history)', 4),
|
|
|
- ('acrobatics', 9),
|
|
|
- ('climb', 3),
|
|
|
- ('atealth', 1),
|
|
|
+ ('knowledge (history)', 14),
|
|
|
+ ('acrobatics', 14),
|
|
|
+ ('climb', 4),
|
|
|
+ ('swim', 4),
|
|
|
+ ('stealth', 4),
|
|
|
('disable device', 1),
|
|
|
- ('spellcraft', 1),
|
|
|
+ ('spellcraft', 8),
|
|
|
('intimidate', 1),
|
|
|
('use magic device', 1),
|
|
|
('diplomacy', 1),
|
|
|
('perception', 1),
|
|
|
+ ('survival', 1),
|
|
|
('linguistics', 4),
|
|
|
- ('lore (divine battles)', 4),
|
|
|
+ ('lore (divine battles)', 9),
|
|
|
+ ('lore (mythic founts)', 9),
|
|
|
('art (anatomy)', 1),
|
|
|
- ('perform (oratory)', 2),
|
|
|
+ ('perform (oratory)', 1),
|
|
|
],
|
|
|
}
|
|
|
|
|
@@ -145,9 +162,46 @@ def fillsreqs(pr, charsheet):
|
|
|
if f != '' and f not in charsheet['feat']:
|
|
|
qual = False
|
|
|
|
|
|
+ if 'and' in pr:
|
|
|
+ for andpr in pr['and']:
|
|
|
+ if not fillsreqs(andpr, charsheet):
|
|
|
+ qual = False
|
|
|
+
|
|
|
+ if 'or' in pr:
|
|
|
+ orqual = False
|
|
|
+ for orpr in pr['or']:
|
|
|
+ if anyreqs(orpr, charsheet):
|
|
|
+ orqual = True
|
|
|
+ break
|
|
|
+ if not orqual:
|
|
|
+ qual = False
|
|
|
+
|
|
|
return qual
|
|
|
|
|
|
|
|
|
+def anyreqs(pr, charsheet):
|
|
|
+ for attr in ['str', 'dex', 'con', 'int', 'wis', 'cha', 'bab']:
|
|
|
+ if attr in pr:
|
|
|
+ if int(pr[attr][0]) <= charsheet[attr]:
|
|
|
+ return True
|
|
|
+ if 'level' in pr:
|
|
|
+ for l in pr['level']:
|
|
|
+ for ll in charsheet['level']:
|
|
|
+ if l[0] == ll[0] and int(l[1]) <= ll[1]:
|
|
|
+ return True
|
|
|
+ if 'skill' in pr:
|
|
|
+ for s in pr['skill']:
|
|
|
+ for cs in charsheet['skill']:
|
|
|
+ if s[0] == cs[0] and int(s[1]) <= cs[1]:
|
|
|
+ return True
|
|
|
+ if 'feat' in pr:
|
|
|
+ for f in pr['feat']:
|
|
|
+ if f != '' and f in charsheet['feat']:
|
|
|
+ return True
|
|
|
+
|
|
|
+ return False
|
|
|
+
|
|
|
+
|
|
|
def qualfeat(feat, charsheet):
|
|
|
if feat['prereqs']:
|
|
|
return fillsreqs(feat['prereqs'], charsheet)
|
|
@@ -155,22 +209,33 @@ def qualfeat(feat, charsheet):
|
|
|
return True
|
|
|
|
|
|
|
|
|
-
|
|
|
-
|
|
|
def qualfeats(feats, charsheet):
|
|
|
return [f for f in feats if qualfeat(f, charsheet) and f['name'].lower() not in charsheet['feat']]
|
|
|
|
|
|
|
|
|
+def addchain(feat, charsheet, depth=0):
|
|
|
+ newchar = deepcopy(charsheet)
|
|
|
+ newchar['feat'].append(feat['name'].lower())
|
|
|
+
|
|
|
+ newfeats = []
|
|
|
+
|
|
|
+ for f in qualfeats(allfeats, newchar):
|
|
|
+ if 'feat' in f['prereqs']:
|
|
|
+ if f['name'].lower() not in newchar['feat'] and feat['name'].lower() in f['prereqs']['feat']:
|
|
|
+ if depth >= 1:
|
|
|
+ newfeats.append(f)
|
|
|
+ else:
|
|
|
+ newfeats.append(addchain(f, newchar, depth + 1))
|
|
|
+
|
|
|
+ if len(newfeats) > 0:
|
|
|
+ feat['chain'] = newfeats
|
|
|
+
|
|
|
+ return feat
|
|
|
+
|
|
|
+
|
|
|
def addchains(feats, charsheet):
|
|
|
- featnames = [f['name'] for f in feats]
|
|
|
for f in feats:
|
|
|
- newreqs = deepcopy(charsheet)
|
|
|
- newreqs['feat'].append(f['name'].lower())
|
|
|
- newcandidates = qualfeats(allfeats, newreqs)
|
|
|
- newfeats = [ nf for nf in newcandidates if nf['name'] not in featnames ]
|
|
|
-
|
|
|
- if len(newfeats) > 0:
|
|
|
- f['chain'] = newfeats
|
|
|
+ f = addchain(f, charsheet)
|
|
|
|
|
|
|
|
|
def prefeats(feats):
|
|
@@ -204,16 +269,18 @@ def asfeat(feat):
|
|
|
|
|
|
return featsnip
|
|
|
|
|
|
+
|
|
|
def aschainfeat(feat):
|
|
|
featsnip = '\item \\textbf{%s} %s\n\n' % (feat['name'], feat['benefit'])
|
|
|
if feat['trick']:
|
|
|
featsnip += '\\textbf{Combat Trick:} %s\n\n' % feat['trick']
|
|
|
if feat['special']:
|
|
|
featsnip += '\\textbf{Special:} %s\n\n' % feat['special']
|
|
|
+ if 'chain' in feat:
|
|
|
+ featsnip += chainfeats(feat['chain'])
|
|
|
return featsnip
|
|
|
|
|
|
|
|
|
-
|
|
|
def chainfeats(feats):
|
|
|
chainsnip = '\\begin{itemize}\n\n'
|
|
|
for f in feats:
|
|
@@ -251,12 +318,16 @@ if __name__ == '__main__':
|
|
|
singles = []
|
|
|
chainers = []
|
|
|
for f in cantake:
|
|
|
- print(f['name'])
|
|
|
if 'chain' in f:
|
|
|
chainers.append(f)
|
|
|
else:
|
|
|
singles.append(f)
|
|
|
|
|
|
+ # sort both lists
|
|
|
+
|
|
|
+ singles.sort(key=lambda x: x['name'])
|
|
|
+ chainers.sort(key=lambda x: x['name'])
|
|
|
+
|
|
|
# LaTeXify both lists
|
|
|
|
|
|
singlesnip = ''
|