def to_mse(self, print_raw=False, vdump=False): outstr = '' # need a 'card' string first outstr += 'card:\n' cardname = titlecase( transforms.name_unpass_1_dashes(self.__dict__[field_name])) outstr += '\tname: ' + cardname + '\n' if self.__dict__[field_rarity]: if self.__dict__[field_rarity] in utils.json_rarity_unmap: rarity = utils.json_rarity_unmap[self.__dict__[field_rarity]] else: rarity = self.__dict__[field_rarity] outstr += '\trarity: ' + rarity.lower() + '\n' if not self.__dict__[field_cost].none: outstr += ('\tcasting cost: ' + self.__dict__[field_cost].format().replace( '{', '').replace('}', '') + '\n') outstr += '\tsuper type: ' + ' '.join( self.__dict__[field_supertypes] + self.__dict__[field_types]).title() + '\n' if self.__dict__[field_subtypes]: outstr += '\tsub type: ' + ' '.join( self.__dict__[field_subtypes]).title() + '\n' if self.__dict__[field_pt]: ptstring = utils.from_unary(self.__dict__[field_pt]).split('/') if (len(ptstring) > 1 ): # really don't want to be accessing anything nonexistent. outstr += '\tpower: ' + ptstring[0] + '\n' outstr += '\ttoughness: ' + ptstring[1] + '\n' if self.__dict__[field_text].text: mtext = self.__dict__[field_text].text mtext = transforms.text_unpass_1_choice(mtext, delimit=False) mtext = transforms.text_unpass_2_counters(mtext) mtext = transforms.text_unpass_3_uncast(mtext) mtext = transforms.text_unpass_4_unary(mtext) mtext = transforms.text_unpass_5_symbols(mtext, False, False) mtext = sentencecase(mtext) # I don't really want these MSE specific passes in transforms, # but they could be pulled out separately somewhere else in here. mtext = mtext.replace( utils.this_marker, '<atom-cardname><nospellcheck>' + utils.this_marker + '</nospellcheck></atom-cardname>') mtext = transforms.text_unpass_6_cardname(mtext, cardname) mtext = transforms.text_unpass_7_newlines(mtext) mtext = transforms.text_unpass_8_unicode(mtext) newtext = Manatext('') newtext.text = mtext newtext.costs = self.__dict__[field_text].costs newtext = newtext.format() # See, the thing is, I think it's simplest and easiest to just leave it like this. # What could possibly go wrong? newtext = newtext.replace('{', '<sym-auto>').replace( '}', '</sym-auto>') else: newtext = '' # Annoying special case for bsides; # This could be improved by having an intermediate function that returned # all of the formatted fields in a data structure and a separate wrapper # that actually packed them into the MSE format. if self.bside: newtext = newtext.replace('\n', '\n\t\t') outstr += '\trule text:\n\t\t' + newtext + '\n' outstr += '\tstylesheet: new-split\n' cardname2 = titlecase( transforms.name_unpass_1_dashes( self.bside.__dict__[field_name])) outstr += '\tname 2: ' + cardname2 + '\n' if self.bside.__dict__[field_rarity]: if self.bside.__dict__[ field_rarity] in utils.json_rarity_unmap: rarity2 = utils.json_rarity_unmap[ self.bside.__dict__[field_rarity]] else: rarity2 = self.bside.__dict__[field_rarity] outstr += '\trarity 2: ' + rarity2.lower() + '\n' if not self.bside.__dict__[field_cost].none: outstr += ('\tcasting cost 2: ' + self.bside.__dict__[field_cost].format().replace( '{', '').replace('}', '') + '\n') outstr += ('\tsuper type 2: ' + ' '.join(self.bside.__dict__[field_supertypes] + self.bside.__dict__[field_types]).title() + '\n') if self.bside.__dict__[field_subtypes]: outstr += ('\tsub type 2: ' + ' '.join( self.bside.__dict__[field_subtypes]).title() + '\n') if self.bside.__dict__[field_pt]: ptstring2 = utils.from_unary( self.bside.__dict__[field_pt]).split('/') if ( len(ptstring2) > 1 ): # really don't want to be accessing anything nonexistent. outstr += '\tpower 2: ' + ptstring2[0] + '\n' outstr += '\ttoughness 2: ' + ptstring2[1] + '\n' if self.bside.__dict__[field_text].text: mtext2 = self.bside.__dict__[field_text].text mtext2 = transforms.text_unpass_1_choice(mtext2, delimit=False) mtext2 = transforms.text_unpass_2_counters(mtext2) mtext2 = transforms.text_unpass_3_uncast(mtext2) mtext2 = transforms.text_unpass_4_unary(mtext2) mtext2 = transforms.text_unpass_5_symbols(mtext2, False, False) mtext2 = sentencecase(mtext2) mtext2 = mtext2.replace( utils.this_marker, '<atom-cardname><nospellcheck>' + utils.this_marker + '</nospellcheck></atom-cardname>') mtext2 = transforms.text_unpass_6_cardname(mtext2, cardname2) mtext2 = transforms.text_unpass_7_newlines(mtext2) mtext2 = transforms.text_unpass_8_unicode(mtext2) newtext2 = Manatext('') newtext2.text = mtext2 newtext2.costs = self.bside.__dict__[field_text].costs newtext2 = newtext2.format() newtext2 = newtext2.replace('{', '<sym-auto>').replace( '}', '</sym-auto>') newtext2 = newtext2.replace('\n', '\n\t\t') outstr += '\trule text 2:\n\t\t' + newtext2 + '\n' # Need to do Special Things if it's a planeswalker. # This code mostly works, but it won't get quite the right thing if the planeswalker # abilities don't come before any other ones. Should be fixed. elif "planeswalker" in str(self.__dict__[field_types]): outstr += '\tstylesheet: m15-planeswalker\n' # set up the loyalty cost fields using regex to find how many there are. i = 0 lcost_regex = r'[-+]?\d+: ' # 1+ figures, might be 0. for lcost in re.findall(lcost_regex, newtext): i += 1 outstr += '\tloyalty cost ' + str(i) + ': ' + lcost + '\n' # sub out the loyalty costs. newtext = re.sub(lcost_regex, '', newtext) # We need to uppercase again, because MSE won't magically capitalize for us # like it does after semicolons. # Abusing passes like this is terrible, should really fix sentencecase. newtext = transforms.text_pass_9_newlines(newtext) newtext = sentencecase(newtext) newtext = transforms.text_unpass_7_newlines(newtext) if self.__dict__[field_loyalty]: outstr += '\tloyalty: ' + utils.from_unary( self.__dict__[field_loyalty]) + '\n' newtext = newtext.replace('\n', '\n\t\t') outstr += '\trule text:\n\t\t' + newtext + '\n' else: newtext = newtext.replace('\n', '\n\t\t') outstr += '\trule text:\n\t\t' + newtext + '\n' # now append all the other useless fields that the setfile expects. outstr += '\thas styling: false\n\ttime created:2015-07-20 22:53:07\n\ttime modified:2015-07-20 22:53:08\n\textra data:\n\timage:\n\tcard code text:\n\tcopyright:\n\timage 2:\n\tcopyright 2:\n\tnotes:' return outstr
def format(self, gatherer=False, for_forum=False, vdump=False, for_html=False): linebreak = '\n' if for_html: linebreak = '<hr>' + linebreak outstr = '' if for_html: outstr += '<div class="card-text">\n' if gatherer: cardname = titlecase( transforms.name_unpass_1_dashes(self.__dict__[field_name])) if vdump and not cardname: cardname = '_NONAME_' # in general, for_html overrides for_forum if for_html: outstr += '<b>' elif for_forum: outstr += '[b]' outstr += cardname if for_html: outstr += '</b>' elif for_forum: outstr += '[/b]' coststr = self.__dict__[field_cost].format(for_forum=for_forum, for_html=for_html) if vdump or not coststr == '_NOCOST_': outstr += ' ' + coststr if for_html and for_forum: #force for_html to false to create tootip with forum spoiler outstr += ( '<div class="hover_img"><a href="#">[F]</a> <span><p>' + self.format(gatherer=gatherer, for_forum=for_forum, for_html=False, vdump=vdump).replace('\n', '<br>') + '</p></span></div><a href="#top" style="float: right;">back to top</a>' ) if self.__dict__[field_rarity]: if self.__dict__[field_rarity] in utils.json_rarity_unmap: rarity = utils.json_rarity_unmap[ self.__dict__[field_rarity]] else: rarity = self.__dict__[field_rarity] outstr += ' (' + rarity + ')' if vdump: if not self.parsed: outstr += ' _UNPARSED_' if not self.valid: outstr += ' _INVALID_' outstr += linebreak basetypes = map(str.capitalize, self.__dict__[field_types]) if vdump and len(basetypes) < 1: basetypes = ['_NOTYPE_'] outstr += ' '.join( map(str.capitalize, self.__dict__[field_supertypes]) + basetypes) if self.__dict__[field_subtypes]: outstr += (' ' + utils.dash_marker + ' ' + ' '.join(self.__dict__[field_subtypes]).title()) if self.__dict__[field_pt]: outstr += ' (' + utils.from_unary( self.__dict__[field_pt]) + ')' if self.__dict__[field_loyalty]: outstr += ' ((' + utils.from_unary( self.__dict__[field_loyalty]) + '))' if self.__dict__[field_text].text: outstr += linebreak mtext = self.__dict__[field_text].text mtext = transforms.text_unpass_1_choice(mtext, delimit=False) mtext = transforms.text_unpass_2_counters(mtext) #mtext = transforms.text_unpass_3_uncast(mtext) mtext = transforms.text_unpass_4_unary(mtext) mtext = transforms.text_unpass_5_symbols( mtext, for_forum, for_html) mtext = sentencecase(mtext) mtext = transforms.text_unpass_6_cardname(mtext, cardname) mtext = transforms.text_unpass_7_newlines(mtext) #mtext = transforms.text_unpass_8_unicode(mtext) newtext = Manatext('') newtext.text = mtext newtext.costs = self.__dict__[field_text].costs outstr += newtext.format(for_forum=for_forum, for_html=for_html) if vdump and self.__dict__[field_other]: outstr += linebreak if for_html: outstr += '<i>' elif for_forum: outstr += '[i]' else: outstr += utils.dash_marker * 2 first = True for idx, value in self.__dict__[field_other]: if for_html: if not first: outstr += '<br>\n' else: first = False else: outstr += linebreak outstr += '(' + str(idx) + ') ' + str(value) if for_html: outstr += '</i>' if for_forum: outstr += '[/i]' else: cardname = self.__dict__[field_name] #cardname = transforms.name_unpass_1_dashes(self.__dict__[field_name]) if vdump and not cardname: cardname = '_NONAME_' outstr += cardname coststr = self.__dict__[field_cost].format(for_forum=for_forum, for_html=for_html) if vdump or not coststr == '_NOCOST_': outstr += ' ' + coststr if vdump: if not self.parsed: outstr += ' _UNPARSED_' if not self.valid: outstr += ' _INVALID_' if for_html and for_forum: #force for_html to false to create tootip with forum spoiler outstr += ( '<div class="hover_img"><a href="#">[F]</a> <span><p>' + self.format(gatherer=gatherer, for_forum=for_forum, for_html=False, vdump=vdump).replace('\n', '<br>') + '</p></span></div><a href="#top" style="float: right;">back to top</a>' ) outstr += linebreak outstr += ' '.join(self.__dict__[field_supertypes] + self.__dict__[field_types]) if self.__dict__[field_subtypes]: outstr += ' ' + utils.dash_marker + ' ' + ' '.join( self.__dict__[field_subtypes]) if self.__dict__[field_rarity]: if self.__dict__[field_rarity] in utils.json_rarity_unmap: rarity = utils.json_rarity_unmap[ self.__dict__[field_rarity]] else: rarity = self.__dict__[field_rarity] outstr += ' (' + rarity.lower() + ')' if self.__dict__[field_text].text: outstr += linebreak mtext = self.__dict__[field_text].text mtext = transforms.text_unpass_1_choice(mtext, delimit=True) #mtext = transforms.text_unpass_2_counters(mtext) #mtext = transforms.text_unpass_3_uncast(mtext) mtext = transforms.text_unpass_4_unary(mtext) mtext = transforms.text_unpass_5_symbols( mtext, for_forum, for_html) #mtext = transforms.text_unpass_6_cardname(mtext, cardname) mtext = transforms.text_unpass_7_newlines(mtext) #mtext = transforms.text_unpass_8_unicode(mtext) newtext = Manatext('') newtext.text = mtext newtext.costs = self.__dict__[field_text].costs outstr += newtext.format(for_forum=for_forum, for_html=for_html) if self.__dict__[field_pt]: outstr += linebreak outstr += '(' + utils.from_unary(self.__dict__[field_pt]) + ')' if self.__dict__[field_loyalty]: outstr += linebreak outstr += '((' + utils.from_unary( self.__dict__[field_loyalty]) + '))' if vdump and self.__dict__[field_other]: outstr += linebreak if for_html: outstr += '<i>' else: outstr += utils.dash_marker * 2 first = True for idx, value in self.__dict__[field_other]: if for_html: if not first: outstr += '<br>\n' else: first = False else: outstr += linebreak outstr += '(' + str(idx) + ') ' + str(value) if for_html: outstr += '</i>' if self.bside: if for_html: outstr += '\n' # force for_forum to false so that the inner div doesn't duplicate the forum # spoiler of the bside outstr += self.bside.format(gatherer=gatherer, for_forum=False, for_html=for_html, vdump=vdump) else: outstr += linebreak outstr += utils.dash_marker * 8 outstr += linebreak outstr += self.bside.format(gatherer=gatherer, for_forum=for_forum, for_html=for_html, vdump=vdump) # if for_html: # if for_forum: # outstr += linebreak # # force for_html to false to create a copyable forum spoiler div # outstr += ('<div>' # + self.format(gatherer=gatherer, for_forum=for_forum, for_html=False, vdump=vdump).replace('\n', '<br>') # + '</div>') if for_html: outstr += "</div>" return outstr
def fields_from_format(src_text, fmt_ordered, fmt_labeled, fieldsep): parsed = True valid = True fields = {} if fmt_labeled: labels = {fmt_labeled[k]: k for k in fmt_labeled} field_label_regex = '[' + ''.join(labels.keys()) + ']' def addf(fields, fkey, fval): # make sure you pass a pair if fval and fval[1]: if fkey in fields: fields[fkey] += [fval] else: fields[fkey] = [fval] textfields = src_text.split(fieldsep) idx = 0 true_idx = 0 for textfield in textfields: # ignore leading or trailing empty fields due to seps if textfield == '': if true_idx == 0 or true_idx == len(textfields) - 1: true_idx += 1 continue # count the field index for other empty fields but don't add them else: idx += 1 true_idx += 1 continue lab = None if fmt_labeled: labs = re.findall(field_label_regex, textfield) # use the first label if we saw any at all if len(labs) > 0: lab = labs[0] textfield = textfield.replace(lab, '', 1) # try to use the field label if we got one if lab and lab in labels: fname = labels[lab] # fall back to the field order specified elif idx < len(fmt_ordered): fname = fmt_ordered[idx] # we don't know what to do with this field: call it other else: fname = field_other parsed = False valid = False # specialized handling if fname in [field_cost]: fval = Manacost(textfield) parsed = parsed and fval.parsed valid = valid and fval.valid addf(fields, fname, (idx, fval)) elif fname in [field_text]: fval = Manatext(textfield) valid = valid and fval.valid addf(fields, fname, (idx, fval)) elif fname in [field_supertypes, field_types, field_subtypes]: addf(fields, fname, (idx, textfield.split())) else: addf(fields, fname, (idx, textfield)) idx += 1 true_idx += 1 # again, bsides are handled by the constructor return parsed, valid and fields_check_valid(fields), fields
def __init__(self, src, fmt_ordered=fmt_ordered_default, fmt_labeled=fmt_labeled_default, fieldsep=utils.fieldsep, linetrans=True): # source fields, exactly one will be set self.json = None self.raw = None # flags self.parsed = True self.valid = True # doesn't record that much # placeholders to fill in with expensive distance metrics self.nearest_names = [] self.nearest_cards = [] # default values for all fields self.__dict__[field_name] = '' self.__dict__[field_rarity] = '' self.__dict__[field_cost] = Manacost('') self.__dict__[field_supertypes] = [] self.__dict__[field_types] = [] self.__dict__[field_subtypes] = [] self.__dict__[field_loyalty] = '' self.__dict__[field_loyalty + '_value'] = None self.__dict__[field_pt] = '' self.__dict__[field_pt + '_p'] = None self.__dict__[field_pt + '_p_value'] = None self.__dict__[field_pt + '_t'] = None self.__dict__[field_pt + '_t_value'] = None self.__dict__[field_text] = Manatext('') self.__dict__[field_text + '_lines'] = [] self.__dict__[field_text + '_words'] = [] self.__dict__[field_text + '_lines_words'] = [] self.__dict__[field_other] = [] self.bside = None # format-independent view of processed input self.fields = None # will be reset later # looks like a json object if isinstance(src, dict): self.json = src if utils.json_field_bside in src: self.bside = Card(src[utils.json_field_bside], fmt_ordered=fmt_ordered, fmt_labeled=fmt_labeled, fieldsep=fieldsep, linetrans=linetrans) p_success, v_success, parsed_fields = fields_from_json( src, linetrans=linetrans) self.parsed = p_success self.valid = v_success self.fields = parsed_fields # otherwise assume text encoding else: self.raw = src sides = src.split(utils.bsidesep) if len(sides) > 1: self.bside = Card(utils.bsidesep.join(sides[1:]), fmt_ordered=fmt_ordered, fmt_labeled=fmt_labeled, fieldsep=fieldsep, linetrans=linetrans) p_success, v_success, parsed_fields = fields_from_format( sides[0], fmt_ordered, fmt_labeled, fieldsep) self.parsed = p_success self.valid = v_success self.fields = parsed_fields # amusingly enough, both encodings allow infinitely deep nesting of bsides... # python name hackery if self.fields: for field in self.fields: # look for a specialized set function if hasattr(self, '_set_' + field): getattr(self, '_set_' + field)(self.fields[field]) # otherwise use the default one elif field in self.__dict__: self.set_field_default(field, self.fields[field]) # If we don't recognize the field, fail. This is a totally artificial # limitation; if we just used the default handler for the else case, # we could set arbitrarily named fields. else: raise ValueError( 'python name mangling failure: unknown field for Card(): ' + field) else: # valid but not parsed indicates that the card was apparently empty self.parsed = False
def fields_from_json(src_json, linetrans=True): parsed = True valid = True fields = {} # we hardcode in what the things are called in the mtgjson format if 'name' in src_json: name_val = src_json['name'].lower() name_orig = name_val name_val = transforms.name_pass_1_sanitize(name_val) name_val = utils.to_ascii(name_val) fields[field_name] = [(-1, name_val)] else: name_orig = '' parsed = False # return the actual Manacost object if 'manaCost' in src_json: cost = Manacost(src_json['manaCost'], fmt='json') valid = valid and cost.valid parsed = parsed and cost.parsed fields[field_cost] = [(-1, cost)] if 'supertypes' in src_json: fields[field_supertypes] = [(-1, map(lambda s: utils.to_ascii(s.lower()), src_json['supertypes']))] if 'types' in src_json: fields[field_types] = [(-1, map(lambda s: utils.to_ascii(s.lower()), src_json['types']))] else: parsed = False if 'subtypes' in src_json: fields[field_subtypes] = [( -1, map( lambda s: utils.to_ascii(s.lower()) # urza's lands... .replace('"', "'").replace('-', utils.dash_marker), src_json['subtypes']))] if 'rarity' in src_json: if src_json['rarity'] in utils.json_rarity_map: fields[field_rarity] = [ (-1, utils.json_rarity_map[src_json['rarity']]) ] else: fields[field_rarity] = [(-1, src_json['rarity'])] parsed = False else: parsed = False if 'loyalty' in src_json: fields[field_loyalty] = [(-1, utils.to_unary(str(src_json['loyalty']))) ] p_t = '' parsed_pt = True if 'power' in src_json: p_t = utils.to_ascii(utils.to_unary( src_json['power'])) + '/' # hardcoded parsed_pt = False if 'toughness' in src_json: p_t = p_t + utils.to_ascii(utils.to_unary(src_json['toughness'])) parsed_pt = True elif 'toughness' in src_json: p_t = '/' + utils.to_ascii(utils.to_unary( src_json['toughness'])) # hardcoded parsed_pt = False if p_t: fields[field_pt] = [(-1, p_t)] parsed = parsed and parsed_pt # similarly, return the actual Manatext object if 'text' in src_json: text_val = src_json['text'].lower() text_val = transforms.text_pass_1_strip_rt(text_val) text_val = transforms.text_pass_2_cardname(text_val, name_orig) text_val = transforms.text_pass_3_unary(text_val) text_val = transforms.text_pass_4a_dashes(text_val) text_val = transforms.text_pass_4b_x(text_val) text_val = transforms.text_pass_5_counters(text_val) text_val = transforms.text_pass_6_uncast(text_val) text_val = transforms.text_pass_7_choice(text_val) text_val = transforms.text_pass_8_equip(text_val) text_val = transforms.text_pass_9_newlines(text_val) text_val = transforms.text_pass_10_symbols(text_val) if linetrans: text_val = transforms.text_pass_11_linetrans(text_val) text_val = utils.to_ascii(text_val) text_val = text_val.strip() mtext = Manatext(text_val, fmt='json') valid = valid and mtext.valid fields[field_text] = [(-1, mtext)] # we don't need to worry about bsides because we handle that in the constructor return parsed, valid and fields_check_valid(fields), fields