X-Git-Url: http://git.veekun.com/zzz-dywypi.git/blobdiff_plain/5a7c4e0242d0ad85d3e295506f3a971f2a526461..dd70d7ecc6c580aca3271fe2ae8c0951e97f9d5e:/plugins/Pokedex/plugin.py?ds=inline diff --git a/plugins/Pokedex/plugin.py b/plugins/Pokedex/plugin.py index b81ced2..13ce031 100644 --- a/plugins/Pokedex/plugin.py +++ b/plugins/Pokedex/plugin.py @@ -1,3 +1,4 @@ +# encoding: utf8 ### # Copyright (c) 2010, Alex Munroe # All rights reserved. @@ -28,17 +29,265 @@ ### +import supybot.conf as conf import supybot.utils as utils from supybot.commands import * import supybot.plugins as plugins import supybot.ircutils as ircutils import supybot.callbacks as callbacks +import pokedex.db +import pokedex.db.tables as tables +import pokedex.lookup + +import urllib + + +def get_stat_color(stat): + if stat < 41: + return 4 # red + elif stat < 52: + return 7 # orange + elif stat < 61: + return 8 # yellow + elif stat < 71: + return 9 # green + elif stat < 85: + return 11 # cyan + elif stat < 100: + return 12 # blue + else: + return 13 # purple + class Pokedex(callbacks.Plugin): """Add the help for "@plugin help Pokedex" here This should describe *how* to use this plugin.""" - pass + def __init__(self, irc): + self.__parent = super(Pokedex, self) + self.__parent.__init__(irc) + self.db = pokedex.db.connect(self.registryValue('databaseURL')) + self.lookup = pokedex.lookup.PokedexLookup( + directory=conf.supybot.directories.data.dirize('pokedex-index'), + session=self.db, + ) + + def pokedex(self, irc, msg, args, thing): + """ + + Looks up in the veekun Pokédex.""" + + # Fix encoding. Sigh. + if not isinstance(thing, unicode): + ascii_thing = thing + try: + thing = ascii_thing.decode('utf8') + except UnicodeDecodeError: + thing = ascii_thing.decode('latin1') + + # Similar logic to the site, here. + results = self.lookup.lookup(thing) + + # Nothing found + if len(results) == 0: + self._reply(irc, "I don't know what that is.") + return + + # Multiple matches; propose them all + if len(results) > 1: + if results[0].exact: + reply = "Are you looking for" + else: + reply = "Did you mean" + + # For exact name matches with multiple results, use type prefixes + # (item:Metronome). For anything else, omit them + use_prefixes = (results[0].exact + and '*' not in thing + and '?' not in thing) + + result_strings = [] + for result in results: + result_string = result.object.name + + # Prepend, e.g., pokemon: if necessary + if use_prefixes: + # Table classes know their singular names + prefix = result.object.__singlename__ + result_string = prefix + ':' + result_string + + # Identify foreign language names + if result.language: + result_string += u""" ({0}: {1})""".format( + result.iso3166, result.name) + + result_strings.append(result_string) + + self._reply(irc, u"{0}: {1}?".format(reply, '; '.join(result_strings))) + return + + # If we got here, there's an exact match; hurrah! + result = results[0] + obj = result.object + if isinstance(obj, tables.Pokemon): + reply_template = \ + u"""#{id} {name}, {type}-type Pokémon. Has {abilities}. """ \ + """{stats}. """ \ + """http://veekun.com/dex/pokemon/{link_name}""" + + if obj.forme_name: + name = '{form} {name}'.format( + form=obj.forme_name.title(), + name=obj.name + ) + else: + name = obj.name + + if obj.forme_base_pokemon: + # Can't use urllib.quote() on the whole thing or it'll + # catch "?" and "=" where it shouldn't. + # XXX Also we need to pass urllib.quote() things explicitly + # encoded as utf8 or else we get a UnicodeEncodeError. + link_name = '{name}?form={form}'.format( + name=urllib.quote(obj.name.lower().encode('utf8')), + form=urllib.quote(obj.forme_name.lower().encode('utf8')), + ) + else: + link_name = urllib.quote(obj.name.lower().encode('utf8')) + + # a/b/c/d/e/f sucks. Put stats in a more readable format. + # Also, color-code them by good-osity. + colored_stats = [] + stat_total = 0 + for pokemon_stat in obj.stats: + base_stat = pokemon_stat.base_stat + stat_total += base_stat + color = get_stat_color(base_stat) + + colored_stats.append( + "\x03{0:02d}{1}\x0f".format(color, base_stat) + ) + + colored_stat_total = "\x03{0:02d}{1}\x0f".format( + get_stat_color(stat_total / 6), + stat_total, + ) + stats = """{0} HP, {1}/{2} phys, {3}/{4} spec, {5} speed, {total} total""" \ + .format(*colored_stats, total=colored_stat_total) + self._reply(irc, reply_template.format( + id=obj.national_id, + name=name, + type='/'.join(_.name for _ in obj.types), + abilities=' or '.join(_.name for _ in obj.abilities), + stats=stats, + link_name=link_name, + ) + ) + + elif isinstance(obj, tables.Move): + reply_template = \ + u"""{name}, {type}-type {damage_class} move. """ \ + """{power} power; {accuracy}% accuracy; {pp} PP. """ \ + """{effect} """ \ + """http://veekun.com/dex/moves/{link_name}""" + self._reply(irc, reply_template.format( + name=obj.name, + type=obj.type.name, + damage_class=obj.damage_class.name, + power=obj.power, + accuracy=obj.accuracy, + pp=obj.pp, + effect=unicode(obj.short_effect.as_text), + link_name=urllib.quote(obj.name.lower().encode('utf8')), + ) + ) + + elif isinstance(obj, tables.Type): + reply_template = u"""{name}, a type. """ + + offensive_reply_factors = { + 200: u'\x03092×\x0f', + 50: u'\x0304½×\x0f', + 0: u'\x03140×\x0f', + } + + offensive_modifiers = {} + for matchup in obj.damage_efficacies: + if matchup.damage_factor != 100: + offensive_modifiers.setdefault(matchup.damage_factor, []) \ + .append(matchup.target_type.name) + if offensive_modifiers: + reply_template += u"""{offensive_modifiers}. """ + for factor in offensive_modifiers: + offensive_modifiers[factor] = u'{factor} against {types}'.format( + factor=offensive_reply_factors[factor], + types=', '.join(sorted(offensive_modifiers[factor])) + ) + + defensive_reply_factors = { + 200: u'\x03042×\x0f', + 50: u'\x0309½×\x0f', + 0: u'\x03110×\x0f', + } + + defensive_modifiers = {} + for matchup in obj.target_efficacies: + if matchup.damage_factor != 100: + defensive_modifiers.setdefault(matchup.damage_factor, []) \ + .append(matchup.damage_type.name) + if defensive_modifiers: + reply_template += u"""{defensive_modifiers}. """ + for factor in defensive_modifiers: + defensive_modifiers[factor] = u'{factor} from {types}'.format( + factor=defensive_reply_factors[factor], + types=', '.join(sorted(defensive_modifiers[factor])) + ) + + reply_template += u"""http://veekun.com/dex/types/{link_name}""" + + self._reply(irc, reply_template.format( + name=obj.name.capitalize(), + offensive_modifiers='; '.join(offensive_modifiers[_] + for _ in sorted(offensive_modifiers)), + defensive_modifiers='; '.join(defensive_modifiers[_] + for _ in sorted(defensive_modifiers)), + link_name=urllib.quote(obj.name.lower().encode('utf8')), + ) + ) + + elif isinstance(obj, tables.Item): + reply_template = \ + u"""{name}, an item. """ \ + """http://veekun.com/dex/items/{link_name}""" + self._reply(irc, reply_template.format( + name=obj.name, + link_name=urllib.quote(obj.name.lower().encode('utf8')), + ) + ) + + elif isinstance(obj, tables.Ability): + reply_template = \ + u"""{name}, an ability. {effect} """ \ + """http://veekun.com/dex/abilities/{link_name}""" + self._reply(irc, reply_template.format( + name=obj.name, + effect=obj.effect, + link_name=urllib.quote(obj.name.lower().encode('utf8')), + ) + ) + + else: + # This can only happen if lookup.py is upgraded and we are not + self._reply(irc, "Uhh.. I found that, but I don't know what it is. :(") + + pokedex = wrap(pokedex, [rest('something')]) + + def _reply(self, irc, response): + """Wraps irc.reply() to do some Unicode decoding.""" + if isinstance(response, str): + irc.reply(response) + else: + irc.reply(response.encode('utf8')) Class = Pokedex