Pokedex: Use short ability effects; support natures.
[zzz-dywypi.git] / plugins / Pokedex / plugin.py
index 77de7d9..62f11b6 100644 (file)
@@ -29,6 +29,7 @@
 
 ###
 
+import supybot.conf as conf
 import supybot.utils as utils
 from supybot.commands import *
 import supybot.plugins as plugins
@@ -39,6 +40,26 @@ 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."""
@@ -46,18 +67,30 @@ class Pokedex(callbacks.Plugin):
         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):
         """<thing...>
 
         Looks up <thing> 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 = pokedex.lookup.lookup(thing, session=self.db)
+        results = self.lookup.lookup(thing)
 
         # Nothing found
         if len(results) == 0:
-            irc.reply("I don't know what that is.")
+            self._reply(irc, "I don't know what that is.")
             return
 
         # Multiple matches; propose them all
@@ -75,56 +108,200 @@ class Pokedex(callbacks.Plugin):
 
             result_strings = []
             for result in results:
-                result_string = result.name
+                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)
 
-            irc.reply("{0}: {1}?".format(reply, '; '.join(result_strings)))
+            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]
-        if isinstance(result.object, tables.Pokemon):
-            irc.reply("""{name}, {type}-type Pokémon.""".format(
-                name=result.object.name,
-                type='/'.join(_.name for _ in result.object.types),
+        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(result.object, tables.Move):
-            irc.reply("""{name}, {type}-type move.""".format(
-                name=result.object.name,
-                type=result.object.type.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(result.object, tables.Type):
-            irc.reply("""{name}, a type.""".format(
-                name=result.object.name,
+        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(result.object, tables.Item):
-            irc.reply("""{name}, an item.""".format(
-                name=result.object.name,
+        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(result.object, tables.Ability):
-            irc.reply("""{name}, an ability.""".format(
-                name=result.object.name,
+        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.short_effect.as_text,
+                link_name=urllib.quote(obj.name.lower().encode('utf8')),
+                )
+            )
+
+        elif isinstance(obj, tables.Nature):
+            reply_template = \
+                u"""{name}, a nature.  """ \
+                u"""Raises \x0303{up}\x0f, lowers \x0304{down}\x0f.  """ \
+                u"""http://veekun.com/dex/natures/{link_name}"""
+            self._reply(irc, reply_template.format(
+                name=obj.name,
+                up=obj.increased_stat.name,
+                down=obj.decreased_stat.name,
+                link_name=urllib.quote(obj.name.lower().encode('utf8')),
                 )
             )
 
         else:
             # This can only happen if lookup.py is upgraded and we are not
-            irc.reply("Uhh..  I found that, but I don't know what it is.  :(")
+            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