5 Columns have a info dictionary with these keys:
6 - description: The description of the column
7 - official: True if the values appear in games or official material; False if
8 they are fan-created or fan-written. This flag is currently only set for
10 - markup: The format of a text column. Can be one of:
11 - plaintext: Normal Unicode text (widely used in names)
12 - markdown: Veekun's Markdown flavor (generally used in effect descriptions)
13 - gametext: Transcription of in-game text that strives to be both
14 human-readable and represent the original text exactly.
15 - identifier: A fan-made identifier in the [-_a-z0-9]* format. Not intended
17 - latex: A formula in LaTeX syntax.
19 A localizable text column is visible as two properties:
20 The plural-name property (e.g. Pokemon.names) is a language-to-name dictionary:
21 bulbasaur.names['en'] == "Bulbasaur" and bulbasaur.names['de'] == "Bisasam".
22 You can use Pokemon.names['en'] to filter a query.
23 The singular-name property returns the name in the default language, English.
24 For example bulbasaur.name == "Bulbasaur"
25 Setting pokedex.db.tables.default_lang changes the default language.
27 # XXX: Check if "gametext" is set correctly everywhere
31 from sqlalchemy
import Column
, ForeignKey
, MetaData
, PrimaryKeyConstraint
, Table
, UniqueConstraint
32 from sqlalchemy
.ext
.declarative
import (
33 declarative_base
, declared_attr
, DeclarativeMeta
,
35 from sqlalchemy
.ext
.associationproxy
import association_proxy
36 from sqlalchemy
.orm
import (
37 backref
, eagerload_all
, relation
, class_mapper
, synonym
, mapper
,
39 from sqlalchemy
.orm
.session
import Session
, object_session
40 from sqlalchemy
.orm
.collections
import attribute_mapped_collection
41 from sqlalchemy
.ext
.associationproxy
import association_proxy
42 from sqlalchemy
.sql
import and_
43 from sqlalchemy
.sql
.expression
import ColumnOperators
44 from sqlalchemy
.schema
import ColumnDefault
45 from sqlalchemy
.types
import *
46 from inspect
import isclass
48 from pokedex
.db
import markdown
50 # A list of all table classes will live in table_classes
53 class TableMetaclass(DeclarativeMeta
):
54 def __init__(cls
, name
, bases
, attrs
):
55 super(TableMetaclass
, cls
).__init__(name
, bases
, attrs
)
56 if hasattr(cls
, '__tablename__'):
57 table_classes
.append(cls
)
59 class TableSuperclass(object):
60 """Superclass for declarative tables, to give them some generic niceties
63 def __unicode__(self
):
64 """Be as useful as possible. Show the primary key, and an identifier
67 typename
= u
'.'.join((__name__
, type(self
).__name__
))
69 pk_constraint
= self
.__table__
.primary_key
71 return u
"<%s object at %x>" %
(typename
, id(self
))
73 pk
= u
', '.join(unicode(getattr(self
, column
.name
))
74 for column
in pk_constraint
.columns
)
76 return u
"<%s object (%s): %s>" %
(typename
, pk
, self
.identifier
)
77 except AttributeError:
78 return u
"<%s object (%s)>" %
(typename
, pk
)
81 return unicode(self
).encode('utf8')
84 TableBase
= declarative_base(metadata
=metadata
, cls
=TableSuperclass
, metaclass
=TableMetaclass
)
87 class LanguageSpecific(object):
88 """Mixin for prose and text tables"""
91 return Column(Integer
, ForeignKey('languages.id'), primary_key
=True, nullable
=False,
92 info
=dict(description
="The language"))
94 class LanguageSpecificColumn(object):
95 """A column that will not appear in the table it's defined in, but in a related one"""
97 def __init__(self
, *args
, **kwargs
):
99 self
.plural
= kwargs
.pop('plural')
101 self
.order
= self
._ordering
[0]
102 self
._ordering
[0] += 1
104 def makeSAColumn(self
):
105 return Column(*self
.args
, **self
.kwargs
)
107 class ProseColumn(LanguageSpecificColumn
):
108 """A column that will appear in the corresponding _prose table"""
110 class TextColumn(LanguageSpecificColumn
):
111 """A column that will appear in the corresponding _text table"""
114 ### The actual tables
116 class Ability(TableBase
):
117 u
"""An ability a Pokémon can have, such as Static or Pressure.
119 __tablename__
= 'abilities'
120 __singlename__
= 'ability'
121 id = Column(Integer
, primary_key
=True, nullable
=False,
122 info
=dict(description
="This ability's unique ID; matches the games' internal ID"))
123 identifier
= Column(Unicode(24), nullable
=False,
124 info
=dict(description
="An identifier", format
='identifier'))
125 generation_id
= Column(Integer
, ForeignKey('generations.id'), nullable
=False,
126 info
=dict(description
="The ID of the generation this ability was introduced in", detail
=True))
127 effect
= ProseColumn(markdown
.MarkdownColumn(5120), plural
='effects', nullable
=False,
128 info
=dict(description
="A detailed description of this ability's effect", format
='markdown'))
129 short_effect
= ProseColumn(markdown
.MarkdownColumn(255), plural
='short_effects', nullable
=False,
130 info
=dict(description
="A short summary of this ability's effect", format
='markdown'))
131 name
= TextColumn(Unicode(24), nullable
=False, index
=True, plural
='names',
132 info
=dict(description
="The name", format
='plaintext', official
=True))
134 class AbilityChangelog(TableBase
):
135 """History of changes to abilities across main game versions."""
136 __tablename__
= 'ability_changelog'
137 __singlename__
= 'ability_changelog'
138 id = Column(Integer
, primary_key
=True, nullable
=False,
139 info
=dict(description
="This change's unique ID"))
140 ability_id
= Column(Integer
, ForeignKey('abilities.id'), nullable
=False,
141 info
=dict(description
="The ID of the ability that changed"))
142 changed_in_version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), nullable
=False,
143 info
=dict(description
="The ID of the version group in which the ability changed"))
144 effect
= ProseColumn(markdown
.MarkdownColumn(255), plural
='effects', nullable
=False,
145 info
=dict(description
="A description of the old behavior", format
='markdown'))
147 class AbilityFlavorText(TableBase
, LanguageSpecific
):
148 u
"""In-game flavor text of an ability
150 __tablename__
= 'ability_flavor_text'
151 ability_id
= Column(Integer
, ForeignKey('abilities.id'), primary_key
=True, nullable
=False, autoincrement
=False,
152 info
=dict(description
="The ID of the ability"))
153 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), primary_key
=True, nullable
=False, autoincrement
=False,
154 info
=dict(description
="The ID of the version group this flavor text is taken from"))
155 flavor_text
= Column(Unicode(64), nullable
=False,
156 info
=dict(description
="The actual flavor text", official
=True, format
='gametext'))
158 class Berry(TableBase
):
159 u
"""A Berry, consumable item that grows on trees
161 For data common to all items, such as the name, see the corresponding item entry.
163 __tablename__
= 'berries'
164 id = Column(Integer
, primary_key
=True, nullable
=False,
165 info
=dict(description
="This Berry's in-game number"))
166 item_id
= Column(Integer
, ForeignKey('items.id'), nullable
=False,
167 info
=dict(description
="The ID of the item that represents this Berry"))
168 firmness_id
= Column(Integer
, ForeignKey('berry_firmness.id'), nullable
=False,
169 info
=dict(description
="The ID of this Berry's firmness category"))
170 natural_gift_power
= Column(Integer
, nullable
=True,
171 info
=dict(description
="Natural Gift's power when used with this Berry"))
172 natural_gift_type_id
= Column(Integer
, ForeignKey('types.id'), nullable
=True,
173 info
=dict(description
="The ID of the Type that Natural Gift has when used with this Berry"))
174 size
= Column(Integer
, nullable
=False,
175 info
=dict(description
=u
"The size of this Berry, in millimeters"))
176 max_harvest
= Column(Integer
, nullable
=False,
177 info
=dict(description
="The maximum number of these berries that can grow on one tree in Generation IV"))
178 growth_time
= Column(Integer
, nullable
=False,
179 info
=dict(description
="Time it takes the tree to grow one stage, in hours. Berry trees go through four of these growth stages before they can be picked."))
180 soil_dryness
= Column(Integer
, nullable
=False,
181 info
=dict(description
="The speed at which this Berry dries out the soil as it grows. A higher rate means the soil dries more quickly."))
182 smoothness
= Column(Integer
, nullable
=False,
183 info
=dict(description
="The smoothness of this Berry, used in making Pokéblocks or Poffins"))
185 class BerryFirmness(TableBase
):
186 u
"""A Berry firmness, such as "hard" or "very soft".
188 __tablename__
= 'berry_firmness'
189 __singlename__
= 'berry_firmness'
190 id = Column(Integer
, primary_key
=True, nullable
=False,
191 info
=dict(description
="A unique ID for this firmness"))
192 identifier
= Column(Unicode(10), nullable
=False,
193 info
=dict(description
="An identifier", format
='identifier'))
194 name
= TextColumn(Unicode(10), nullable
=False, index
=True, plural
='names',
195 info
=dict(description
="The name", format
='plaintext', official
=True))
197 class BerryFlavor(TableBase
):
198 u
"""A Berry flavor level.
200 __tablename__
= 'berry_flavors'
201 berry_id
= Column(Integer
, ForeignKey('berries.id'), primary_key
=True, nullable
=False, autoincrement
=False,
202 info
=dict(description
="The ID of the berry"))
203 contest_type_id
= Column(Integer
, ForeignKey('contest_types.id'), primary_key
=True, nullable
=False, autoincrement
=False,
204 info
=dict(description
="The ID of the flavor"))
205 flavor
= Column(Integer
, nullable
=False,
206 info
=dict(description
="The level of the flavor in the berry"))
208 class ContestCombo(TableBase
):
209 u
"""Combo of two moves in a Contest.
211 __tablename__
= 'contest_combos'
212 first_move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
213 info
=dict(description
="The ID of the first move in the combo"))
214 second_move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
215 info
=dict(description
="The ID of the second and final move in the combo"))
217 class ContestEffect(TableBase
):
218 u
"""Effect of a move when used in a Contest.
220 __tablename__
= 'contest_effects'
221 __singlename__
= 'contest_effect'
222 id = Column(Integer
, primary_key
=True, nullable
=False,
223 info
=dict(description
="A unique ID for this effect"))
224 appeal
= Column(SmallInteger
, nullable
=False,
225 info
=dict(description
="The base number of hearts the user of this move gets"))
226 jam
= Column(SmallInteger
, nullable
=False,
227 info
=dict(description
="The base number of hearts the user's opponent loses"))
228 flavor_text
= ProseColumn(Unicode(64), plural
='flavor_texts', nullable
=False,
229 info
=dict(description
="The in-game description of this effect", official
=True, format
='gametext'))
230 effect
= ProseColumn(Unicode(255), plural
='effects', nullable
=False,
231 info
=dict(description
="A detailed description of the effect", format
='plaintext'))
233 class ContestType(TableBase
):
234 u
"""A Contest type, such as "cool" or "smart", and their associated Berry flavors and Pokéblock colors.
236 __tablename__
= 'contest_types'
237 __singlename__
= 'contest_type'
238 id = Column(Integer
, primary_key
=True, nullable
=False,
239 info
=dict(description
="A unique ID for this Contest type"))
240 identifier
= Column(Unicode(6), nullable
=False,
241 info
=dict(description
="An identifier", format
='identifier'))
242 flavor
= TextColumn(Unicode(6), nullable
=False, plural
='flavors',
243 info
=dict(description
="The name of the corresponding Berry flavor", official
=True, format
='plaintext'))
244 color
= TextColumn(Unicode(6), nullable
=False, plural
='colors',
245 info
=dict(description
=u
"The name of the corresponding Pokéblock color", official
=True, format
='plaintext'))
246 name
= TextColumn(Unicode(6), nullable
=False, index
=True, plural
='names',
247 info
=dict(description
="The name", format
='plaintext', official
=True))
249 class EggGroup(TableBase
):
250 u
"""An Egg group. Usually, two Pokémon can breed if they share an Egg Group.
252 (exceptions are the Ditto and No Eggs groups)
254 __tablename__
= 'egg_groups'
255 __singlename__
= 'egg_group'
256 id = Column(Integer
, primary_key
=True, nullable
=False,
257 info
=dict(description
="A unique ID for this group"))
258 identifier
= Column(Unicode(16), nullable
=False,
259 info
=dict(description
=u
"An identifier.", format
='identifier'))
260 name
= ProseColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
261 info
=dict(description
="The name", format
='plaintext', official
=False))
263 class Encounter(TableBase
):
264 u
"""Encounters with wild Pokémon.
268 Within a given area in a given game, encounters are differentiated by the
269 "slot" they are in and the state of the game world.
271 What the player is doing to get an encounter, such as surfing or walking
272 through tall grass, is called terrain. Each terrain has its own set of
275 Within a terrain, slots are defined primarily by rarity. Each slot can
276 also be affected by world conditions; for example, the 20% slot for walking
277 in tall grass is affected by whether a swarm is in effect in that area.
278 "Is there a swarm?" is a condition; "there is a swarm" and "there is not a
279 swarm" are the possible values of this condition.
281 A slot (20% walking in grass) and any appropriate world conditions (no
282 swarm) are thus enough to define a specific encounter.
284 Well, okay, almost: each slot actually appears twice.
287 __tablename__
= 'encounters'
288 id = Column(Integer
, primary_key
=True, nullable
=False,
289 info
=dict(description
="A unique ID for this encounter"))
290 version_id
= Column(Integer
, ForeignKey('versions.id'), nullable
=False, autoincrement
=False,
291 info
=dict(description
="The ID of the version this applies to"))
292 location_area_id
= Column(Integer
, ForeignKey('location_areas.id'), nullable
=False, autoincrement
=False,
293 info
=dict(description
="The ID of the location of this encounter"))
294 encounter_slot_id
= Column(Integer
, ForeignKey('encounter_slots.id'), nullable
=False, autoincrement
=False,
295 info
=dict(description
="The ID of the encounter slot, which determines terrain and rarity"))
296 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), nullable
=False, autoincrement
=False,
297 info
=dict(description
=u
"The ID of the encountered Pokémon"))
298 min_level
= Column(Integer
, nullable
=False, autoincrement
=False,
299 info
=dict(description
=u
"The minimum level of the encountered Pokémon"))
300 max_level
= Column(Integer
, nullable
=False, autoincrement
=False,
301 info
=dict(description
=u
"The maxmum level of the encountered Pokémon"))
303 class EncounterCondition(TableBase
):
304 u
"""A conditions in the game world that affects Pokémon encounters, such as time of day.
307 __tablename__
= 'encounter_conditions'
308 __singlename__
= 'encounter_condition'
309 id = Column(Integer
, primary_key
=True, nullable
=False,
310 info
=dict(description
="A unique ID for this condition"))
311 identifier
= Column(Unicode(64), nullable
=False,
312 info
=dict(description
="An identifier", format
='identifier'))
313 name
= ProseColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
314 info
=dict(description
="The name", format
='plaintext', official
=False))
316 class EncounterConditionValue(TableBase
):
317 u
"""A possible state for a condition; for example, the state of 'swarm' could be 'swarm' or 'no swarm'.
320 __tablename__
= 'encounter_condition_values'
321 __singlename__
= 'encounter_condition_value'
322 id = Column(Integer
, primary_key
=True, nullable
=False,
323 info
=dict(description
="A numeric ID"))
324 encounter_condition_id
= Column(Integer
, ForeignKey('encounter_conditions.id'), primary_key
=False, nullable
=False, autoincrement
=False,
325 info
=dict(description
="The ID of the encounter condition this is a value of"))
326 identifier
= Column(Unicode(64), nullable
=False,
327 info
=dict(description
="An identifier", format
='identifier'))
328 is_default
= Column(Boolean
, nullable
=False,
329 info
=dict(description
='Set if this value is the default state for the condition'))
330 name
= ProseColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
331 info
=dict(description
="The name", format
='plaintext', official
=False))
333 class EncounterConditionValueMap(TableBase
):
334 u
"""Maps encounters to the specific conditions under which they occur.
336 __tablename__
= 'encounter_condition_value_map'
337 encounter_id
= Column(Integer
, ForeignKey('encounters.id'), primary_key
=True, nullable
=False, autoincrement
=False,
338 info
=dict(description
="The ID of the encounter"))
339 encounter_condition_value_id
= Column(Integer
, ForeignKey('encounter_condition_values.id'), primary_key
=True, nullable
=False, autoincrement
=False,
340 info
=dict(description
="The ID of the encounter condition value"))
342 class EncounterTerrain(TableBase
):
343 u
"""A way the player can enter a wild encounter, e.g., surfing, fishing, or walking through tall grass.
346 __tablename__
= 'encounter_terrain'
347 __singlename__
= __tablename__
348 id = Column(Integer
, primary_key
=True, nullable
=False,
349 info
=dict(description
="A unique ID for the terrain"))
350 identifier
= Column(Unicode(64), nullable
=False,
351 info
=dict(description
="An identifier", format
='identifier'))
352 name
= ProseColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
353 info
=dict(description
="The name", format
='plaintext', official
=False))
355 class EncounterSlot(TableBase
):
356 u
"""An abstract "slot" within a terrain, associated with both some set of conditions and a rarity.
358 Note that there are two encounters per slot, so the rarities will only add
362 __tablename__
= 'encounter_slots'
363 id = Column(Integer
, primary_key
=True, nullable
=False,
364 info
=dict(description
="A unique ID for this slot"))
365 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), nullable
=False, autoincrement
=False,
366 info
=dict(description
="The ID of the version group this slot is in"))
367 encounter_terrain_id
= Column(Integer
, ForeignKey('encounter_terrain.id'), primary_key
=False, nullable
=False, autoincrement
=False,
368 info
=dict(description
="The ID of the terrain"))
369 slot
= Column(Integer
, nullable
=True,
370 info
=dict(description
="This slot's order for the location and terrain"))
371 rarity
= Column(Integer
, nullable
=False,
372 info
=dict(description
="The chance of the encounter as a percentage"))
374 class EvolutionChain(TableBase
):
375 u
"""A family of Pokémon that are linked by evolution
377 __tablename__
= 'evolution_chains'
378 id = Column(Integer
, primary_key
=True, nullable
=False,
379 info
=dict(description
="A numeric ID"))
380 growth_rate_id
= Column(Integer
, ForeignKey('growth_rates.id'), nullable
=False,
381 info
=dict(description
="ID of the growth rate for this family"))
382 baby_trigger_item_id
= Column(Integer
, ForeignKey('items.id'), nullable
=True,
383 info
=dict(description
="Item that a parent must hold while breeding to produce a baby"))
385 class EvolutionTrigger(TableBase
):
386 u
"""An evolution type, such as "level" or "trade".
388 __tablename__
= 'evolution_triggers'
389 __singlename__
= 'evolution_trigger'
390 id = Column(Integer
, primary_key
=True, nullable
=False,
391 info
=dict(description
="A numeric ID"))
392 identifier
= Column(Unicode(16), nullable
=False,
393 info
=dict(description
="An identifier", format
='identifier'))
394 name
= ProseColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
395 info
=dict(description
="The name", format
='plaintext', official
=False))
397 class Experience(TableBase
):
398 u
"""EXP needed for a certain level with a certain growth rate
400 __tablename__
= 'experience'
401 growth_rate_id
= Column(Integer
, ForeignKey('growth_rates.id'), primary_key
=True, nullable
=False,
402 info
=dict(description
="ID of the growth rate"))
403 level
= Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
404 info
=dict(description
="The level"))
405 experience
= Column(Integer
, nullable
=False,
406 info
=dict(description
="The number of EXP points needed to get to that level"))
408 class Generation(TableBase
):
409 u
"""A Generation of the Pokémon franchise
411 __tablename__
= 'generations'
412 __singlename__
= 'generation'
413 id = Column(Integer
, primary_key
=True, nullable
=False,
414 info
=dict(description
="A numeric ID"))
415 main_region_id
= Column(Integer
, ForeignKey('regions.id'),
416 info
=dict(description
="ID of the region this generation's main games take place in"))
417 canonical_pokedex_id
= Column(Integer
, ForeignKey('pokedexes.id'),
418 info
=dict(description
=u
"ID of the Pokédex this generation's main games use by default"))
419 identifier
= Column(Unicode(16), nullable
=False,
420 info
=dict(description
=u
'An identifier', format
='identifier'))
421 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
422 info
=dict(description
="The name", format
='plaintext', official
=True))
424 class GrowthRate(TableBase
):
425 u
"""Growth rate of a Pokémon, i.e. the EXP → level function.
427 __tablename__
= 'growth_rates'
428 __singlename__
= 'growth_rate'
429 id = Column(Integer
, primary_key
=True, nullable
=False,
430 info
=dict(description
="A numeric ID"))
431 identifier
= Column(Unicode(20), nullable
=False,
432 info
=dict(description
="An identifier", format
='identifier'))
433 formula
= Column(Unicode(500), nullable
=False,
434 info
=dict(description
="The formula", format
='latex'))
435 name
= ProseColumn(Unicode(20), nullable
=False, index
=True, plural
='names',
436 info
=dict(description
="The name", format
='plaintext', official
=False))
438 class Item(TableBase
):
439 u
"""An Item from the games, like "Poké Ball" or "Bicycle".
441 __tablename__
= 'items'
442 __singlename__
= 'item'
443 id = Column(Integer
, primary_key
=True, nullable
=False,
444 info
=dict(description
="A numeric ID"))
445 identifier
= Column(Unicode(20), nullable
=False,
446 info
=dict(description
="An identifier", format
='identifier'))
447 category_id
= Column(Integer
, ForeignKey('item_categories.id'), nullable
=False,
448 info
=dict(description
="ID of a category this item belongs to"))
449 cost
= Column(Integer
, nullable
=False,
450 info
=dict(description
=u
"Cost of the item when bought. Items sell for half this price."))
451 fling_power
= Column(Integer
, nullable
=True,
452 info
=dict(description
=u
"Power of the move Fling when used with this item."))
453 fling_effect_id
= Column(Integer
, ForeignKey('item_fling_effects.id'), nullable
=True,
454 info
=dict(description
=u
"ID of the fling-effect of the move Fling when used with this item. Note that these are different from move effects."))
455 short_effect
= ProseColumn(Unicode(256), plural
='short_effects', nullable
=False,
456 info
=dict(description
="A short summary of the effect", format
='plaintext'))
457 effect
= ProseColumn(markdown
.MarkdownColumn(5120), plural
='effects', nullable
=False,
458 info
=dict(description
=u
"Detailed description of the item's effect.", format
='markdown'))
459 name
= TextColumn(Unicode(20), nullable
=False, index
=True, plural
='names',
460 info
=dict(description
="The name", format
='plaintext', official
=True))
463 def appears_underground(self
):
464 u
"""True if the item appears underground, as specified by the appropriate flag
466 return any(flag
.identifier
== u
'underground' for flag
in self
.flags
)
468 class ItemCategory(TableBase
):
471 # XXX: This is fanon, right?
472 __tablename__
= 'item_categories'
473 __singlename__
= 'item_category'
474 id = Column(Integer
, primary_key
=True, nullable
=False,
475 info
=dict(description
="A numeric ID"))
476 pocket_id
= Column(Integer
, ForeignKey('item_pockets.id'), nullable
=False,
477 info
=dict(description
="ID of the pocket these items go to"))
478 identifier
= Column(Unicode(16), nullable
=False,
479 info
=dict(description
="An identifier", format
='identifier'))
480 name
= ProseColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
481 info
=dict(description
="The name", format
='plaintext', official
=False))
483 class ItemFlag(TableBase
):
484 u
"""An item attribute such as "consumable" or "holdable".
486 __tablename__
= 'item_flags'
487 __singlename__
= 'item_flag'
488 id = Column(Integer
, primary_key
=True, nullable
=False,
489 info
=dict(description
="A numeric ID"))
490 identifier
= Column(Unicode(24), nullable
=False,
491 info
=dict(description
="Identifier of the flag", format
='identifier'))
492 description
= ProseColumn(Unicode(64), plural
='descriptions', nullable
=False,
493 info
=dict(description
="Short description of the flag", format
='plaintext'))
494 name
= ProseColumn(Unicode(24), nullable
=False, index
=True, plural
='names',
495 info
=dict(description
="The name", format
='plaintext', official
=False))
497 class ItemFlagMap(TableBase
):
498 u
"""Maps an item flag to its item.
500 __tablename__
= 'item_flag_map'
501 item_id
= Column(Integer
, ForeignKey('items.id'), primary_key
=True, autoincrement
=False, nullable
=False,
502 info
=dict(description
="The ID of the item"))
503 item_flag_id
= Column(Integer
, ForeignKey('item_flags.id'), primary_key
=True, autoincrement
=False, nullable
=False,
504 info
=dict(description
="The ID of the item flag"))
506 class ItemFlavorText(TableBase
, LanguageSpecific
):
507 u
"""An in-game description of an item
509 __tablename__
= 'item_flavor_text'
510 __singlename__
= 'item_flavor_text'
511 item_id
= Column(Integer
, ForeignKey('items.id'), primary_key
=True, autoincrement
=False, nullable
=False,
512 info
=dict(description
="The ID of the item"))
513 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), primary_key
=True, autoincrement
=False, nullable
=False,
514 info
=dict(description
="ID of the version group that sports this text"))
515 flavor_text
= Column(Unicode(255), nullable
=False,
516 info
=dict(description
="The flavor text itself", official
=True, format
='gametext'))
518 class ItemFlingEffect(TableBase
):
519 u
"""An effect of the move Fling when used with a specific item
521 __tablename__
= 'item_fling_effects'
522 __singlename__
= 'item_fling_effect'
523 id = Column(Integer
, primary_key
=True, nullable
=False,
524 info
=dict(description
="A numeric ID"))
525 effect
= ProseColumn(Unicode(255), plural
='effects', nullable
=False,
526 info
=dict(description
="Description of the effect", format
='plaintext'))
528 class ItemInternalID(TableBase
):
529 u
"""The internal ID number a game uses for an item
531 __tablename__
= 'item_internal_ids'
532 item_id
= Column(Integer
, ForeignKey('items.id'), primary_key
=True, autoincrement
=False, nullable
=False,
533 info
=dict(description
="The database ID of the item"))
534 generation_id
= Column(Integer
, ForeignKey('generations.id'), primary_key
=True, autoincrement
=False, nullable
=False,
535 info
=dict(description
="ID of the generation of games"))
536 internal_id
= Column(Integer
, nullable
=False,
537 info
=dict(description
="Internal ID of the item in the generation"))
539 class ItemPocket(TableBase
):
540 u
"""A pocket that categorizes items
542 __tablename__
= 'item_pockets'
543 __singlename__
= 'item_pocket'
544 id = Column(Integer
, primary_key
=True, nullable
=False,
545 info
=dict(description
="A numeric ID"))
546 identifier
= Column(Unicode(16), nullable
=False,
547 info
=dict(description
="An identifier of this pocket", format
='identifier'))
548 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
549 info
=dict(description
="The name", format
='plaintext', official
=True))
551 class Language(TableBase
):
552 u
"""A language the Pokémon games have been transleted into
554 __tablename__
= 'languages'
555 __singlename__
= 'language'
556 id = Column(Integer
, primary_key
=True, nullable
=False,
557 info
=dict(description
="A numeric ID"))
558 iso639
= Column(Unicode(2), nullable
=False,
559 info
=dict(description
="The two-letter code of the country where this language is spoken. Note that it is not unique.", format
='identifier'))
560 iso3166
= Column(Unicode(2), nullable
=False,
561 info
=dict(description
="The two-letter code of the language. Note that it is not unique.", format
='identifier'))
562 identifier
= Column(Unicode(16), nullable
=False,
563 info
=dict(description
="An identifier", format
='identifier'))
564 official
= Column(Boolean
, nullable
=False, index
=True,
565 info
=dict(description
=u
"True iff games are produced in the language."))
566 order
= Column(Integer
, nullable
=True,
567 info
=dict(description
=u
"Order for sorting in foreign name lists."))
568 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
569 info
=dict(description
="The name", format
='plaintext', official
=True))
571 # Languages compare equal to its identifier, so a dictionary of
572 # translations, with a Language as the key, can be indexed by the identifier
573 def __eq__(self
, other
):
577 self
.identifier
== other
or
578 self
.identifier
== other
.identifier
580 except AttributeError:
581 return NotImplemented
583 def __ne__(self
, other
):
584 return not (self
== other
)
587 return hash(self
.identifier
)
589 class Location(TableBase
):
590 u
"""A place in the Pokémon world
592 __tablename__
= 'locations'
593 __singlename__
= 'location'
594 id = Column(Integer
, primary_key
=True, nullable
=False,
595 info
=dict(description
="A numeric ID"))
596 region_id
= Column(Integer
, ForeignKey('regions.id'),
597 info
=dict(description
="ID of the region this location is in"))
598 identifier
= Column(Unicode(64), nullable
=False,
599 info
=dict(description
="An identifier", format
='identifier'))
600 name
= TextColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
601 info
=dict(description
="The name", format
='plaintext', official
=True))
603 class LocationArea(TableBase
):
604 u
"""A sub-area of a location
606 __tablename__
= 'location_areas'
607 __singlename__
= 'location_area'
608 id = Column(Integer
, primary_key
=True, nullable
=False,
609 info
=dict(description
="A numeric ID"))
610 location_id
= Column(Integer
, ForeignKey('locations.id'), nullable
=False,
611 info
=dict(description
="ID of the location this area is part of"))
612 internal_id
= Column(Integer
, nullable
=False,
613 info
=dict(description
="ID the games ude for this area"))
614 identifier
= Column(Unicode(64), nullable
=True,
615 info
=dict(description
="An identifier", format
='identifier'))
616 name
= ProseColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
617 info
=dict(description
="The name", format
='plaintext', official
=False))
619 class LocationAreaEncounterRate(TableBase
):
620 # XXX: What's this exactly? Someone add the docstring & revise the descriptions
621 __tablename__
= 'location_area_encounter_rates'
622 location_area_id
= Column(Integer
, ForeignKey('location_areas.id'), primary_key
=True, nullable
=False, autoincrement
=False,
623 info
=dict(description
="ID of the area"))
624 encounter_terrain_id
= Column(Integer
, ForeignKey('encounter_terrain.id'), primary_key
=True, nullable
=False, autoincrement
=False,
625 info
=dict(description
="ID of the terrain"))
626 version_id
= Column(Integer
, ForeignKey('versions.id'), primary_key
=True, autoincrement
=False,
627 info
=dict(description
="ID of the version"))
628 rate
= Column(Integer
, nullable
=True,
629 info
=dict(description
="The encounter rate")) # units?
631 class LocationInternalID(TableBase
):
632 u
"""IDs the games use internally for locations
634 __tablename__
= 'location_internal_ids'
635 location_id
= Column(Integer
, ForeignKey('locations.id'), nullable
=False, primary_key
=True,
636 info
=dict(description
="Database ID of the locaion"))
637 generation_id
= Column(Integer
, ForeignKey('generations.id'), nullable
=False, primary_key
=True,
638 info
=dict(description
="ID of the generation this entry to"))
639 internal_id
= Column(Integer
, nullable
=False,
640 info
=dict(description
="Internal game ID of the location"))
642 class Machine(TableBase
):
643 u
"""A TM or HM; numbered item that can teach a move to a Pokémon
645 __tablename__
= 'machines'
646 machine_number
= Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
647 info
=dict(description
="Number of the machine for TMs, or 100 + the munber for HMs"))
648 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), primary_key
=True, nullable
=False, autoincrement
=False,
649 info
=dict(description
="Versions this entry applies to"))
650 item_id
= Column(Integer
, ForeignKey('items.id'), nullable
=False,
651 info
=dict(description
="ID of the corresponding Item"))
652 move_id
= Column(Integer
, ForeignKey('moves.id'), nullable
=False,
653 info
=dict(description
="ID of the taught move"))
657 u
"""True if this machine is a HM, False if it's a TM
659 return self
.machine_number
>= 100
661 class MoveBattleStyle(TableBase
):
662 u
"""A battle style of a move""" # XXX: Explain better
663 __tablename__
= 'move_battle_styles'
664 __singlename__
= 'move_battle_style'
665 id = Column(Integer
, primary_key
=True, nullable
=False,
666 info
=dict(description
="A numeric ID"))
667 identifier
= Column(Unicode(8), nullable
=False,
668 info
=dict(description
="An identifier", format
='identifier'))
669 name
= ProseColumn(Unicode(8), nullable
=False, index
=True, plural
='names',
670 info
=dict(description
="The name", format
='plaintext', official
=False))
672 class MoveEffectCategory(TableBase
):
673 u
"""Category of a move effect
675 __tablename__
= 'move_effect_categories'
676 __singlename__
= 'move_effect_category'
677 id = Column(Integer
, primary_key
=True, nullable
=False,
678 info
=dict(description
="A numeric ID"))
679 identifier
= Column(Unicode(64), nullable
=False,
680 info
=dict(description
="An identifier", format
='identifier'))
681 can_affect_user
= Column(Boolean
, nullable
=False,
682 info
=dict(description
="Set if the user can be affected"))
683 name
= ProseColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
684 info
=dict(description
="The name", format
='plaintext', official
=False))
686 class MoveEffectCategoryMap(TableBase
):
687 u
"""Maps a move effect category to a move effect
689 __tablename__
= 'move_effect_category_map'
690 move_effect_id
= Column(Integer
, ForeignKey('move_effects.id'), primary_key
=True, nullable
=False,
691 info
=dict(description
="ID of the move effect"))
692 move_effect_category_id
= Column(Integer
, ForeignKey('move_effect_categories.id'), primary_key
=True, nullable
=False,
693 info
=dict(description
="ID of the category"))
694 affects_user
= Column(Boolean
, primary_key
=True, nullable
=False,
695 info
=dict(description
="Set if the user is affected"))
697 class MoveDamageClass(TableBase
):
698 u
"""Any of the damage classes moves can have, i.e. physical, special, or non-damaging.
700 __tablename__
= 'move_damage_classes'
701 __singlename__
= 'move_damage_class'
702 id = Column(Integer
, primary_key
=True, nullable
=False,
703 info
=dict(description
="A numeric ID"))
704 identifier
= Column(Unicode(16), nullable
=False,
705 info
=dict(description
="An identifier", format
='identifier'))
706 description
= ProseColumn(Unicode(64), plural
='descriptions', nullable
=False,
707 info
=dict(description
="A description of the class", format
='plaintext'))
708 name
= ProseColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
709 info
=dict(description
="The name", format
='plaintext', official
=False))
711 class MoveEffect(TableBase
):
712 u
"""An effect of a move
714 __tablename__
= 'move_effects'
715 __singlename__
= 'move_effect'
716 id = Column(Integer
, primary_key
=True, nullable
=False,
717 info
=dict(description
="A numeric ID"))
718 short_effect
= ProseColumn(Unicode(256), plural
='short_effects', nullable
=False,
719 info
=dict(description
="A short summary of the effect", format
='plaintext'))
720 effect
= ProseColumn(Unicode(5120), plural
='effects', nullable
=False,
721 info
=dict(description
="A detailed description of the effect", format
='plaintext'))
723 class MoveEffectChangelog(TableBase
):
724 """History of changes to move effects across main game versions."""
725 __tablename__
= 'move_effect_changelog'
726 __singlename__
= 'move_effect_changelog'
727 id = Column(Integer
, primary_key
=True, nullable
=False,
728 info
=dict(description
="A numeric ID"))
729 effect_id
= Column(Integer
, ForeignKey('move_effects.id'), nullable
=False,
730 info
=dict(description
="The ID of the effect that changed"))
731 changed_in_version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), nullable
=False,
732 info
=dict(description
="The ID of the version group in which the effect changed"))
733 effect
= ProseColumn(markdown
.MarkdownColumn(512), plural
='effects', nullable
=False,
734 info
=dict(description
="A description of the old behavior", format
='markdown'))
737 UniqueConstraint(effect_id
, changed_in_version_group_id
),
741 class MoveFlag(TableBase
):
742 u
"""Maps a move flag to a move
744 # XXX: Other flags have a ___Flag class for the actual flag and ___FlagMap for the map,
745 # these, somewhat confusingly, have MoveFlagType and MoveFlag
746 __tablename__
= 'move_flags'
747 move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
748 info
=dict(description
="ID of the move"))
749 move_flag_type_id
= Column(Integer
, ForeignKey('move_flag_types.id'), primary_key
=True, nullable
=False, autoincrement
=False,
750 info
=dict(description
="ID of the flag"))
752 class MoveFlagType(TableBase
):
753 u
"""A Move attribute such as "snatchable" or "contact".
755 __tablename__
= 'move_flag_types'
756 __singlename__
= 'move_flag_type'
757 id = Column(Integer
, primary_key
=True, nullable
=False,
758 info
=dict(description
="A numeric ID"))
759 identifier
= Column(Unicode(32), nullable
=False,
760 info
=dict(description
="A short identifier for the flag", format
='identifier'))
761 description
= ProseColumn(markdown
.MarkdownColumn(128), plural
='descriptions', nullable
=False,
762 info
=dict(description
="A short description of the flag", format
='markdown'))
763 name
= ProseColumn(Unicode(32), nullable
=False, index
=True, plural
='names',
764 info
=dict(description
="The name", format
='plaintext', official
=False))
766 class MoveFlavorText(TableBase
, LanguageSpecific
):
767 u
"""In-game description of a move
769 __tablename__
= 'move_flavor_text'
770 move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
771 info
=dict(description
="ID of the move"))
772 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), primary_key
=True, nullable
=False, autoincrement
=False,
773 info
=dict(description
="ID of the version group this text appears in"))
774 flavor_text
= Column(Unicode(255), nullable
=False,
775 info
=dict(description
="The flavor text", official
=True, format
='gametext'))
777 class MoveMeta(TableBase
):
778 u
"""Metadata for move effects, sorta-kinda ripped straight from the game"""
779 __tablename__
= 'move_meta'
780 move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
781 info
=dict(description
="A numeric ID"))
782 meta_category_id
= Column(Integer
, ForeignKey('move_meta_categories.id'), nullable
=False,
783 info
=dict(description
="ID of the move category"))
784 meta_ailment_id
= Column(Integer
, ForeignKey('move_meta_ailments.id'), nullable
=False,
785 info
=dict(description
="ID of the caused ailment"))
786 min_hits
= Column(Integer
, nullable
=True, index
=True,
787 info
=dict(description
="Minimum number of hits per use"))
788 max_hits
= Column(Integer
, nullable
=True, index
=True,
789 info
=dict(description
="Maximum number of hits per use"))
790 min_turns
= Column(Integer
, nullable
=True, index
=True,
791 info
=dict(description
="Minimum number of turns the user is forced to use the move"))
792 max_turns
= Column(Integer
, nullable
=True, index
=True,
793 info
=dict(description
="Maximum number of turns the user is forced to use the move"))
794 recoil
= Column(Integer
, nullable
=False, index
=True,
795 info
=dict(description
="Recoil damage, in percent of damage done"))
796 healing
= Column(Integer
, nullable
=False, index
=True,
797 info
=dict(description
="Healing, in percent of user's max HP"))
798 crit_rate
= Column(Integer
, nullable
=False, index
=True,
799 info
=dict(description
="Critical hit rate bonus"))
800 ailment_chance
= Column(Integer
, nullable
=False, index
=True,
801 info
=dict(description
="Chance to cause an ailment, in percent"))
802 flinch_chance
= Column(Integer
, nullable
=False, index
=True,
803 info
=dict(description
="Chance to cause flinching, in percent"))
804 stat_chance
= Column(Integer
, nullable
=False, index
=True,
805 info
=dict(description
="Chance to cause a stat change, in percent"))
807 class MoveMetaAilment(TableBase
):
808 u
"""Common status ailments moves can inflict on a single Pokémon, including
809 major ailments like paralysis and minor ailments like trapping.
811 __tablename__
= 'move_meta_ailments'
812 __singlename__
= 'move_meta_ailment'
813 id = Column(Integer
, primary_key
=True, nullable
=False,
814 info
=dict(description
="A numeric ID"))
815 identifier
= Column(Unicode(24), nullable
=False,
816 info
=dict(description
="An identifier", format
='identifier'))
817 name
= TextColumn(Unicode(24), nullable
=False, index
=True, plural
='names',
818 info
=dict(description
="The name", format
='plaintext', official
=True))
820 class MoveMetaCategory(TableBase
):
821 u
"""Very general categories that loosely group move effects."""
822 __tablename__
= 'move_meta_categories'
823 __singlename__
= 'move_meta_category'
824 id = Column(Integer
, primary_key
=True, nullable
=False,
825 info
=dict(description
="A numeric ID"))
826 description
= ProseColumn(Unicode(64), plural
='descriptions', nullable
=False,
827 info
=dict(description
="A description of the category"))
829 class MoveMetaStatChange(TableBase
):
830 u
"""Stat changes moves (may) make."""
831 __tablename__
= 'move_meta_stat_changes'
832 move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
833 info
=dict(description
="ID of the move"))
834 stat_id
= Column(Integer
, ForeignKey('stats.id'), primary_key
=True, nullable
=False, autoincrement
=False,
835 info
=dict(description
="ID of the stat"))
836 change
= Column(Integer
, nullable
=False, index
=True,
837 info
=dict(description
="Amount of increase/decrease, in stages"))
839 class MoveTarget(TableBase
):
840 u
"""Targetting or "range" of a move, e.g. "Affects all opponents" or "Affects user".
842 __tablename__
= 'move_targets'
843 __singlename__
= 'move_target'
844 id = Column(Integer
, primary_key
=True, nullable
=False,
845 info
=dict(description
="A numeric ID"))
846 identifier
= Column(Unicode(32), nullable
=False,
847 info
=dict(description
="An identifier", format
='identifier'))
848 description
= ProseColumn(Unicode(128), plural
='descriptions', nullable
=False,
849 info
=dict(description
="A description", format
='plaintext'))
850 name
= ProseColumn(Unicode(32), nullable
=False, index
=True, plural
='names',
851 info
=dict(description
="The name", format
='plaintext', official
=False))
853 class Move(TableBase
):
854 u
"""A Move: technique or attack a Pokémon can learn to use
856 __tablename__
= 'moves'
857 __singlename__
= 'move'
858 id = Column(Integer
, primary_key
=True, nullable
=False,
859 info
=dict(description
="A numeric ID"))
860 identifier
= Column(Unicode(24), nullable
=False,
861 info
=dict(description
="An identifier", format
='identifier'))
862 generation_id
= Column(Integer
, ForeignKey('generations.id'), nullable
=False,
863 info
=dict(description
="ID of the generation this move first appeared in"))
864 type_id
= Column(Integer
, ForeignKey('types.id'), nullable
=False,
865 info
=dict(description
="ID of the move's elemental type"))
866 power
= Column(SmallInteger
, nullable
=False,
867 info
=dict(description
="Base power of the move"))
868 pp
= Column(SmallInteger
, nullable
=True,
869 info
=dict(description
="Base PP (Power Points) of the move, nullable if not applicable (e.g. Struggle and Shadow moves)."))
870 accuracy
= Column(SmallInteger
, nullable
=True,
871 info
=dict(description
="Accuracy of the move; NULL means it never misses"))
872 priority
= Column(SmallInteger
, nullable
=False,
873 info
=dict(description
="The move's priority bracket"))
874 target_id
= Column(Integer
, ForeignKey('move_targets.id'), nullable
=False,
875 info
=dict(description
="ID of the target (range) of the move"))
876 damage_class_id
= Column(Integer
, ForeignKey('move_damage_classes.id'), nullable
=False,
877 info
=dict(description
="ID of the damage class (physical/special) of the move"))
878 effect_id
= Column(Integer
, ForeignKey('move_effects.id'), nullable
=False,
879 info
=dict(description
="ID of the move's effect"))
880 effect_chance
= Column(Integer
, nullable
=True,
881 info
=dict(description
="The chance for a secondary effect. What this is a chance of is specified by the move's effect."))
882 contest_type_id
= Column(Integer
, ForeignKey('contest_types.id'), nullable
=True,
883 info
=dict(description
="ID of the move's Contest type (e.g. cool or smart)"))
884 contest_effect_id
= Column(Integer
, ForeignKey('contest_effects.id'), nullable
=True,
885 info
=dict(description
="ID of the move's Contest effect"))
886 super_contest_effect_id
= Column(Integer
, ForeignKey('super_contest_effects.id'), nullable
=True,
887 info
=dict(description
="ID of the move's Super Contest effect"))
888 name
= TextColumn(Unicode(24), nullable
=False, index
=True, plural
='names',
889 info
=dict(description
="The name", format
='plaintext', official
=True))
891 class MoveChangelog(TableBase
):
892 """History of changes to moves across main game versions."""
893 __tablename__
= 'move_changelog'
894 __singlename__
= 'move_changelog'
895 move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False,
896 info
=dict(description
="ID of the move that changed"))
897 changed_in_version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), primary_key
=True, nullable
=False,
898 info
=dict(description
="ID of the version group in which the move changed"))
899 type_id
= Column(Integer
, ForeignKey('types.id'), nullable
=True,
900 info
=dict(description
="Prior type of the move, or NULL if unchanged"))
901 power
= Column(SmallInteger
, nullable
=True,
902 info
=dict(description
="Prior base power of the move, or NULL if unchanged"))
903 pp
= Column(SmallInteger
, nullable
=True,
904 info
=dict(description
="Prior base PP of the move, or NULL if unchanged"))
905 accuracy
= Column(SmallInteger
, nullable
=True,
906 info
=dict(description
="Prior accuracy of the move, or NULL if unchanged"))
907 effect_id
= Column(Integer
, ForeignKey('move_effects.id'), nullable
=True,
908 info
=dict(description
="Prior ID of the effect, or NULL if unchanged"))
909 effect_chance
= Column(Integer
, nullable
=True,
910 info
=dict(description
="Prior effect chance, or NULL if unchanged"))
912 class Nature(TableBase
):
913 u
"""A nature a Pokémon can have, such as Calm or Brave
915 __tablename__
= 'natures'
916 __singlename__
= 'nature'
917 id = Column(Integer
, primary_key
=True, nullable
=False,
918 info
=dict(description
="A numeric ID"))
919 identifier
= Column(Unicode(8), nullable
=False,
920 info
=dict(description
="An identifier", format
='identifier'))
921 decreased_stat_id
= Column(Integer
, ForeignKey('stats.id'), nullable
=False,
922 info
=dict(description
="ID of the stat that this nature decreases by 10% (if decreased_stat_id is the same, the effects cancel out)"))
923 increased_stat_id
= Column(Integer
, ForeignKey('stats.id'), nullable
=False,
924 info
=dict(description
="ID of the stat that this nature increases by 10% (if decreased_stat_id is the same, the effects cancel out)"))
925 hates_flavor_id
= Column(Integer
, ForeignKey('contest_types.id'), nullable
=False,
926 info
=dict(description
=u
"ID of the Berry flavor the Pokémon hates (if likes_flavor_id is the same, the effects cancel out)"))
927 likes_flavor_id
= Column(Integer
, ForeignKey('contest_types.id'), nullable
=False,
928 info
=dict(description
=u
"ID of the Berry flavor the Pokémon likes (if hates_flavor_id is the same, the effects cancel out)"))
929 name
= TextColumn(Unicode(8), nullable
=False, index
=True, plural
='names',
930 info
=dict(description
="The name", format
='plaintext', official
=True))
933 def is_neutral(self
):
934 u
"""Returns True iff this nature doesn't alter a Pokémon's stats,
935 bestow taste preferences, etc.
937 return self
.increased_stat_id
== self
.decreased_stat_id
939 class NatureBattleStylePreference(TableBase
):
940 u
"""Battle Palace move preference
942 Specifies how likely a Pokémon with a specific Nature is to use a move of
943 a particular battl style in Battle Palace or Battle Tent
945 __tablename__
= 'nature_battle_style_preferences'
946 nature_id
= Column(Integer
, ForeignKey('natures.id'), primary_key
=True, nullable
=False,
947 info
=dict(description
=u
"ID of the Pokémon's nature"))
948 move_battle_style_id
= Column(Integer
, ForeignKey('move_battle_styles.id'), primary_key
=True, nullable
=False,
949 info
=dict(description
="ID of the battle style"))
950 low_hp_preference
= Column(Integer
, nullable
=False,
951 info
=dict(description
=u
"Chance of using the move, in percent, if HP is under ½"))
952 high_hp_preference
= Column(Integer
, nullable
=False,
953 info
=dict(description
=u
"Chance of using the move, in percent, if HP is over ½"))
955 class NaturePokeathlonStat(TableBase
):
956 u
"""Specifies how a Nature affects a Pokéathlon stat
958 __tablename__
= 'nature_pokeathlon_stats'
959 nature_id
= Column(Integer
, ForeignKey('natures.id'), primary_key
=True, nullable
=False,
960 info
=dict(description
="ID of the nature"))
961 pokeathlon_stat_id
= Column(Integer
, ForeignKey('pokeathlon_stats.id'), primary_key
=True, nullable
=False,
962 info
=dict(description
="ID of the stat"))
963 max_change
= Column(Integer
, nullable
=False,
964 info
=dict(description
="Maximum change"))
966 class PokeathlonStat(TableBase
):
967 u
"""A Pokéathlon stat, such as "Stamina" or "Jump".
969 __tablename__
= 'pokeathlon_stats'
970 __singlename__
= 'pokeathlon_stat'
971 id = Column(Integer
, primary_key
=True, nullable
=False,
972 info
=dict(description
="A numeric ID"))
973 identifier
= Column(Unicode(8), nullable
=False,
974 info
=dict(description
="An identifier", format
='identifier'))
975 name
= TextColumn(Unicode(8), nullable
=False, index
=True, plural
='names',
976 info
=dict(description
="The name", format
='plaintext', official
=True))
978 class Pokedex(TableBase
):
979 u
"""A collection of Pokémon species ordered in a particular way
981 __tablename__
= 'pokedexes'
982 __singlename__
= 'pokedex'
983 id = Column(Integer
, primary_key
=True, nullable
=False,
984 info
=dict(description
="A numeric ID"))
985 region_id
= Column(Integer
, ForeignKey('regions.id'), nullable
=True,
986 info
=dict(description
=u
"ID of the region this Pokédex is used in, or None if it's global"))
987 identifier
= Column(Unicode(16), nullable
=False,
988 info
=dict(description
=u
"An identifier", format
='identifier'))
989 description
= ProseColumn(Unicode(512), plural
='descriptions', nullable
=False,
990 info
=dict(description
=u
"A longer description of the Pokédex", format
='plaintext'))
991 name
= ProseColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
992 info
=dict(description
="The name", format
='plaintext', official
=False))
994 class Pokemon(TableBase
):
995 u
"""A species of Pokémon. The core to this whole mess.
997 __tablename__
= 'pokemon'
998 __singlename__
= 'pokemon'
999 id = Column(Integer
, primary_key
=True, nullable
=False,
1000 info
=dict(description
=u
"A numeric ID"))
1001 identifier
= Column(Unicode(20), nullable
=False,
1002 info
=dict(description
=u
"An identifier", format
='identifier'))
1003 generation_id
= Column(Integer
, ForeignKey('generations.id'),
1004 info
=dict(description
=u
"ID of the generation this species first appeared in"))
1005 evolution_chain_id
= Column(Integer
, ForeignKey('evolution_chains.id'),
1006 info
=dict(description
=u
"ID of the species' evolution chain (a.k.a. family)"))
1007 height
= Column(Integer
, nullable
=False,
1008 info
=dict(description
=u
"The height of the Pokémon, in decimeters (tenths of a meter)"))
1009 weight
= Column(Integer
, nullable
=False,
1010 info
=dict(description
=u
"The weight of the Pokémon, in tenths of a kilogram (decigrams)"))
1011 species
= TextColumn(Unicode(16), nullable
=False, plural
='species_names',
1012 info
=dict(description
=u
'The short flavor text, such as "Seed" or "Lizard"; usually affixed with the word "Pokémon"',
1013 official
=True, format
='plaintext'))
1014 color_id
= Column(Integer
, ForeignKey('pokemon_colors.id'), nullable
=False,
1015 info
=dict(description
=u
"ID of this Pokémon's Pokédex color, as used for a gimmick search function in the games."))
1016 pokemon_shape_id
= Column(Integer
, ForeignKey('pokemon_shapes.id'), nullable
=True,
1017 info
=dict(description
=u
"ID of this Pokémon's body shape, as used for a gimmick search function in the games."))
1018 habitat_id
= Column(Integer
, ForeignKey('pokemon_habitats.id'), nullable
=True,
1019 info
=dict(description
=u
"ID of this Pokémon's habitat, as used for a gimmick search function in the games."))
1020 gender_rate
= Column(Integer
, nullable
=False,
1021 info
=dict(description
=u
"The chance of this Pokémon being female, in eighths; or -1 for genderless"))
1022 capture_rate
= Column(Integer
, nullable
=False,
1023 info
=dict(description
=u
"The base capture rate; up to 255"))
1024 base_experience
= Column(Integer
, nullable
=False,
1025 info
=dict(description
=u
"The base EXP gained when defeating this Pokémon")) # XXX: Is this correct?
1026 base_happiness
= Column(Integer
, nullable
=False,
1027 info
=dict(description
=u
"The tameness when caught by a normal ball"))
1028 is_baby
= Column(Boolean
, nullable
=False,
1029 info
=dict(description
=u
"True iff the Pokémon is a baby, i.e. a lowest-stage Pokémon that cannot breed but whose evolved form can."))
1030 hatch_counter
= Column(Integer
, nullable
=False,
1031 info
=dict(description
=u
"Initial hatch counter: one must walk 255 × (hatch_counter + 1) steps before this Pokémon's egg hatches, unless utilizing bonuses like Flame Body's"))
1032 has_gender_differences
= Column(Boolean
, nullable
=False,
1033 info
=dict(description
=u
"Set iff the species exhibits enough sexual dimorphism to have separate sets of sprites in Gen IV and beyond."))
1034 order
= Column(Integer
, nullable
=False, index
=True,
1035 info
=dict(description
=u
"Order for sorting. Almost national order, except families and forms are grouped together."))
1036 name
= TextColumn(Unicode(20), nullable
=False, index
=True, plural
='names',
1037 info
=dict(description
="The name", format
='plaintext', official
=True))
1039 ### Stuff to handle alternate Pokémon forms
1043 u
"""Returns the Pokémon's form, using its default form as fallback."""
1045 return self
.unique_form
or self
.default_form
1048 def is_base_form(self
):
1049 u
"""Returns True iff the Pokémon is the base form for its species,
1053 return self
.unique_form
is None or self
.unique_form
.is_default
1056 def form_name(self
):
1057 u
"""Returns the Pokémon's form name if it represents a particular form
1058 and that form has a name, or None otherwise.
1061 # If self.unique_form is None, the short-circuit "and" will go ahead
1062 # and return that. Otherwise, it'll return the form's name, which may
1064 return self
.unique_form
and self
.unique_form
.name
1067 def full_name(self
):
1068 u
"""Returns the Pokémon's name, including its form if applicable."""
1071 return u
'{0} {1}'.format(self
.form_name
, self
.name
)
1076 def normal_form(self
):
1077 u
"""Returns the normal form for this Pokémon; i.e., this will return
1078 regular Deoxys when called on any Deoxys form.
1081 if self
.unique_form
:
1082 return self
.unique_form
.form_base_pokemon
1087 def stat(self
, stat_name
):
1088 u
"""Returns a PokemonStat record for the given stat name (or Stat row
1089 object). Uses the normal has-many machinery, so all the stats are
1092 if isinstance(stat_name
, Stat
):
1093 stat_name
= stat_name
.name
1095 for pokemon_stat
in self
.stats
:
1096 if pokemon_stat
.stat
.name
== stat_name
:
1099 raise KeyError(u
'No stat named %s' % stat_name
)
1102 def better_damage_class(self
):
1103 u
"""Returns the MoveDamageClass that this Pokémon is best suited for,
1104 based on its attack stats.
1106 If the attack stats are about equal (within 5), returns None. The
1107 value None, not the damage class called 'None'.
1109 phys
= self
.stat(u
'Attack')
1110 spec
= self
.stat(u
'Special Attack')
1112 diff
= phys
.base_stat
- spec
.base_stat
1115 return phys
.stat
.damage_class
1117 return spec
.stat
.damage_class
1121 class PokemonAbility(TableBase
):
1122 u
"""Maps an ability to a Pokémon that can have it
1124 __tablename__
= 'pokemon_abilities'
1125 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1126 info
=dict(description
=u
"ID of the Pokémon"))
1127 ability_id
= Column(Integer
, ForeignKey('abilities.id'), nullable
=False,
1128 info
=dict(description
=u
"ID of the ability"))
1129 # XXX having both a method and a slot is kind of gross. "slot" is a
1130 # misnomer, anyway: duplicate abilities don't appear in slot 2.
1131 # Probably should replace that with "order".
1132 is_dream
= Column(Boolean
, nullable
=False, index
=True,
1133 info
=dict(description
=u
"Whether this is a Dream World ability"))
1134 slot
= Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
1135 info
=dict(description
=u
"The ability slot, i.e. 1 or 2 for gen. IV"))
1137 class PokemonColor(TableBase
):
1138 u
"""The "Pokédex color" of a Pokémon species. Usually based on the Pokémon's color.
1140 __tablename__
= 'pokemon_colors'
1141 __singlename__
= 'pokemon_color'
1142 id = Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
1143 info
=dict(description
=u
"ID of the Pokémon"))
1144 identifier
= Column(Unicode(6), nullable
=False,
1145 info
=dict(description
=u
"An identifier", format
='identifier'))
1146 name
= TextColumn(Unicode(6), nullable
=False, index
=True, plural
='names',
1147 info
=dict(description
="The name", format
='plaintext', official
=True))
1149 class PokemonDexNumber(TableBase
):
1150 u
"""The number of a Pokémon in a particular Pokédex (e.g. Jigglypuff is #138 in Hoenn's 'dex)
1152 __tablename__
= 'pokemon_dex_numbers'
1153 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1154 info
=dict(description
=u
"ID of the Pokémon"))
1155 pokedex_id
= Column(Integer
, ForeignKey('pokedexes.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1156 info
=dict(description
=u
"ID of the Pokédex"))
1157 pokedex_number
= Column(Integer
, nullable
=False,
1158 info
=dict(description
=u
"Number of the Pokémon in that the Pokédex"))
1160 class PokemonEggGroup(TableBase
):
1161 u
"""Maps an Egg group to a Pokémon; each Pokémon belongs to one or two egg groups
1163 __tablename__
= 'pokemon_egg_groups'
1164 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1165 info
=dict(description
=u
"ID of the Pokémon"))
1166 egg_group_id
= Column(Integer
, ForeignKey('egg_groups.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1167 info
=dict(description
=u
"ID of the egg group"))
1169 class PokemonEvolution(TableBase
):
1170 u
"""A required action ("trigger") and the conditions under which the trigger
1171 must occur to cause a Pokémon to evolve.
1173 Any condition may be null if it does not apply for a particular Pokémon.
1175 __tablename__
= 'pokemon_evolution'
1176 from_pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), nullable
=False,
1177 info
=dict(description
=u
"The ID of the pre-evolution Pokémon."))
1178 to_pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1179 info
=dict(description
=u
"The ID of the post-evolution Pokémon."))
1180 evolution_trigger_id
= Column(Integer
, ForeignKey('evolution_triggers.id'), nullable
=False,
1181 info
=dict(description
=u
"The ID of the evolution trigger."))
1182 trigger_item_id
= Column(Integer
, ForeignKey('items.id'), nullable
=True,
1183 info
=dict(description
=u
"The ID of the item that must be used on the Pokémon."))
1184 minimum_level
= Column(Integer
, nullable
=True,
1185 info
=dict(description
=u
"The minimum level for the Pokémon."))
1186 gender
= Column(Enum('male', 'female', name
='pokemon_evolution_gender'), nullable
=True,
1187 info
=dict(description
=u
"The Pokémon's required gender, or None if gender doesn't matter"))
1188 location_id
= Column(Integer
, ForeignKey('locations.id'), nullable
=True,
1189 info
=dict(description
=u
"The ID of the location the evolution must be triggered at."))
1190 held_item_id
= Column(Integer
, ForeignKey('items.id'), nullable
=True,
1191 info
=dict(description
=u
"The ID of the item the Pokémon must hold."))
1192 time_of_day
= Column(Enum('day', 'night', name
='pokemon_evolution_time_of_day'), nullable
=True,
1193 info
=dict(description
=u
"The required time of day."))
1194 known_move_id
= Column(Integer
, ForeignKey('moves.id'), nullable
=True,
1195 info
=dict(description
=u
"The ID of the move the Pokémon must know."))
1196 minimum_happiness
= Column(Integer
, nullable
=True,
1197 info
=dict(description
=u
"The minimum happiness value the Pokémon must have."))
1198 minimum_beauty
= Column(Integer
, nullable
=True,
1199 info
=dict(description
=u
"The minimum Beauty value the Pokémon must have."))
1200 relative_physical_stats
= Column(Integer
, nullable
=True,
1201 info
=dict(description
=u
"The required relation between the Pokémon's Attack and Defense stats, as sgn(atk-def)."))
1202 party_pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), nullable
=True,
1203 info
=dict(description
=u
"The ID of the Pokémon that must be present in the party."))
1204 trade_pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), nullable
=True,
1205 info
=dict(description
=u
"The ID of the Pokémon for which this Pokémon must be traded."))
1207 class PokemonFlavorText(TableBase
, LanguageSpecific
):
1208 u
"""In-game Pokédex descrption of a Pokémon.
1210 __tablename__
= 'pokemon_flavor_text'
1211 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1212 info
=dict(description
=u
"ID of the Pokémon"))
1213 version_id
= Column(Integer
, ForeignKey('versions.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1214 info
=dict(description
=u
"ID of the version that has this flavor text"))
1215 flavor_text
= Column(Unicode(255), nullable
=False,
1216 info
=dict(description
=u
"ID of the version that has this flavor text", official
=True, format
='gametext'))
1218 class PokemonForm(TableBase
):
1219 u
"""An individual form of a Pokémon.
1221 Pokémon that do not have separate forms are still given a single row to
1222 represent their single form.
1224 __tablename__
= 'pokemon_forms'
1225 __singlename__
= 'pokemon_form'
1226 id = Column(Integer
, primary_key
=True, nullable
=False,
1227 info
=dict(description
=u
'A unique ID for this form.'))
1228 identifier
= Column(Unicode(16), nullable
=True,
1229 info
=dict(description
=u
"An identifier", format
='identifier'))
1230 form_base_pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), nullable
=False, autoincrement
=False,
1231 info
=dict(description
=u
'The ID of the base Pokémon for this form.'))
1232 unique_pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), autoincrement
=False,
1233 info
=dict(description
=u
'The ID of a Pokémon that represents specifically this form, for Pokémon with functionally-different forms like Wormadam.'))
1234 introduced_in_version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), autoincrement
=False,
1235 info
=dict(description
=u
'The ID of the version group in which this form first appeared.'))
1236 is_default
= Column(Boolean
, nullable
=False,
1237 info
=dict(description
=u
'Set for exactly one form used as the default for each species.'))
1238 order
= Column(Integer
, nullable
=False, autoincrement
=False,
1239 info
=dict(description
=u
'The order in which forms should be sorted. Multiple forms may have equal order, in which case they should fall back on sorting by name.'))
1240 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
1241 info
=dict(description
="The name", format
='plaintext', official
=True))
1245 u
"""Returns the Pokémon for this form, using the form base as fallback.
1248 return self
.unique_pokemon
or self
.form_base_pokemon
1251 def full_name(self
):
1252 u
"""Returns the full name of this form, e.g. "Plant Cloak"."""
1256 elif self
.form_group
and self
.form_group
.term
:
1257 return u
'{0} {1}'.format(self
.name
, self
.form_group
.term
)
1262 def pokemon_name(self
):
1263 u
"""Returns the name of this Pokémon with this form, e.g. "Plant
1268 return u
'{0} {1}'.format(self
.name
, self
.form_base_pokemon
.name
)
1270 return self
.form_base_pokemon
.name
1272 class PokemonFormGroup(TableBase
):
1273 u
"""Information about a Pokémon's forms as a group."""
1274 __tablename__
= 'pokemon_form_groups'
1275 __singlename__
= 'pokemon_form_group'
1276 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1277 info
=dict(description
=u
"ID of the base form Pokémon"))
1278 term
= ProseColumn(Unicode(16), plural
='terms', nullable
=True,
1279 info
=dict(description
=u
"The term for this Pokémon's forms, e.g. \"Cloak\" for Burmy or \"Forme\" for Deoxys.", official
=True, format
='plaintext'))
1280 is_battle_only
= Column(Boolean
, nullable
=False,
1281 info
=dict(description
=u
"Set iff the forms only change in battle"))
1282 description
= ProseColumn(markdown
.MarkdownColumn(1024), plural
='descriptions', nullable
=False,
1283 info
=dict(description
=u
"Description of how the forms work", format
='markdown'))
1284 PokemonFormGroup
.id = PokemonFormGroup
.pokemon_id
1286 class PokemonFormPokeathlonStat(TableBase
):
1287 u
"""A Pokémon form's performance in one Pokéathlon stat."""
1288 __tablename__
= 'pokemon_form_pokeathlon_stats'
1289 pokemon_form_id
= Column(Integer
, ForeignKey('pokemon_forms.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1290 info
=dict(description
=u
'The ID of the Pokémon form.'))
1291 pokeathlon_stat_id
= Column(Integer
, ForeignKey('pokeathlon_stats.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1292 info
=dict(description
=u
'The ID of the Pokéathlon stat.'))
1293 minimum_stat
= Column(Integer
, nullable
=False, autoincrement
=False,
1294 info
=dict(description
=u
'The minimum value for this stat for this Pokémon form.'))
1295 base_stat
= Column(Integer
, nullable
=False, autoincrement
=False,
1296 info
=dict(description
=u
'The default value for this stat for this Pokémon form.'))
1297 maximum_stat
= Column(Integer
, nullable
=False, autoincrement
=False,
1298 info
=dict(description
=u
'The maximum value for this stat for this Pokémon form.'))
1300 class PokemonHabitat(TableBase
):
1301 u
"""The habitat of a Pokémon, as given in the FireRed/LeafGreen version Pokédex
1303 __tablename__
= 'pokemon_habitats'
1304 __singlename__
= 'pokemon_habitat'
1305 id = Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
1306 info
=dict(description
=u
"A numeric ID"))
1307 identifier
= Column(Unicode(16), nullable
=False,
1308 info
=dict(description
=u
"An identifier", format
='identifier'))
1309 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
1310 info
=dict(description
="The name", format
='plaintext', official
=True))
1312 class PokemonInternalID(TableBase
):
1313 u
"""The number of a Pokémon a game uses internally
1315 __tablename__
= 'pokemon_internal_ids'
1316 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, autoincrement
=False, nullable
=False,
1317 info
=dict(description
=u
"Database ID of the Pokémon"))
1318 generation_id
= Column(Integer
, ForeignKey('generations.id'), primary_key
=True, autoincrement
=False, nullable
=False,
1319 info
=dict(description
=u
"Database ID of the generation"))
1320 internal_id
= Column(Integer
, nullable
=False,
1321 info
=dict(description
=u
"Internal ID the generation's games use for the Pokémon"))
1323 class PokemonItem(TableBase
):
1324 u
"""Record of an item a Pokémon can hold in the wild
1326 __tablename__
= 'pokemon_items'
1327 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1328 info
=dict(description
=u
"ID of the Pokémon"))
1329 version_id
= Column(Integer
, ForeignKey('versions.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1330 info
=dict(description
=u
"ID of the version this applies to"))
1331 item_id
= Column(Integer
, ForeignKey('items.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1332 info
=dict(description
=u
"ID of the item"))
1333 rarity
= Column(Integer
, nullable
=False,
1334 info
=dict(description
=u
"Chance of the Pokémon holding the item, in percent"))
1336 class PokemonMove(TableBase
):
1337 u
"""Record of a move a Pokémon can learn
1339 __tablename__
= 'pokemon_moves'
1340 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), nullable
=False, index
=True,
1341 info
=dict(description
=u
"ID of the Pokémon"))
1342 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), nullable
=False, index
=True,
1343 info
=dict(description
=u
"ID of the version group this applies to"))
1344 move_id
= Column(Integer
, ForeignKey('moves.id'), nullable
=False, index
=True,
1345 info
=dict(description
=u
"ID of the move"))
1346 pokemon_move_method_id
= Column(Integer
, ForeignKey('pokemon_move_methods.id'), nullable
=False, index
=True,
1347 info
=dict(description
=u
"ID of the method this move is learned by"))
1348 level
= Column(Integer
, nullable
=True, index
=True,
1349 info
=dict(description
=u
"Level the move is learned at, if applicable"))
1350 order
= Column(Integer
, nullable
=True,
1351 info
=dict(description
=u
"A sort key to produce the correct ordering when all else is equal")) # XXX: This needs a better description
1354 PrimaryKeyConstraint('pokemon_id', 'version_group_id', 'move_id', 'pokemon_move_method_id', 'level'),
1358 class PokemonMoveMethod(TableBase
):
1359 u
"""A method a move can be learned by, such as "Level up" or "Tutor".
1361 __tablename__
= 'pokemon_move_methods'
1362 __singlename__
= 'pokemon_move_method'
1363 id = Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
1364 info
=dict(description
=u
"A numeric ID"))
1365 identifier
= Column(Unicode(64), nullable
=False,
1366 info
=dict(description
=u
"An identifier", format
='identifier'))
1367 description
= ProseColumn(Unicode(255), plural
='descriptions', nullable
=False,
1368 info
=dict(description
=u
"A detailed description of how the method works", format
='plaintext'))
1369 name
= ProseColumn(Unicode(64), nullable
=False, index
=True, plural
='names',
1370 info
=dict(description
="The name", format
='plaintext', official
=False))
1372 class PokemonShape(TableBase
):
1373 u
"""The shape of a Pokémon's body, as used in generation IV Pokédexes.
1375 __tablename__
= 'pokemon_shapes'
1376 __singlename__
= 'pokemon_shape'
1377 id = Column(Integer
, primary_key
=True, nullable
=False,
1378 info
=dict(description
=u
"A numeric ID"))
1379 identifier
= Column(Unicode(24), nullable
=False,
1380 info
=dict(description
=u
"An identifier", format
='identifier'))
1381 awesome_name
= ProseColumn(Unicode(16), plural
='awesome_names', nullable
=False,
1382 info
=dict(description
=u
"A splendiferous name of the body shape", format
='plaintext'))
1383 name
= ProseColumn(Unicode(24), nullable
=False, index
=True, plural
='names',
1384 info
=dict(description
="The name", format
='plaintext', official
=False))
1386 class PokemonStat(TableBase
):
1387 u
"""A stat value of a Pokémon
1389 __tablename__
= 'pokemon_stats'
1390 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1391 info
=dict(description
=u
"ID of the Pokémon"))
1392 stat_id
= Column(Integer
, ForeignKey('stats.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1393 info
=dict(description
=u
"ID of the stat"))
1394 base_stat
= Column(Integer
, nullable
=False,
1395 info
=dict(description
=u
"The base stat"))
1396 effort
= Column(Integer
, nullable
=False,
1397 info
=dict(description
=u
"The effort increase in this stat gained when this Pokémon is defeated"))
1399 class PokemonType(TableBase
):
1400 u
"""Maps a type to a Pokémon. Each Pokémon has 1 or 2 types.
1402 __tablename__
= 'pokemon_types'
1403 pokemon_id
= Column(Integer
, ForeignKey('pokemon.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1404 info
=dict(description
=u
"ID of the Pokémon"))
1405 type_id
= Column(Integer
, ForeignKey('types.id'), nullable
=False,
1406 info
=dict(description
=u
"ID of the type"))
1407 slot
= Column(Integer
, primary_key
=True, nullable
=False, autoincrement
=False,
1408 info
=dict(description
=u
"The type's slot, 1 or 2, used to sort types if there are two of them"))
1410 class Region(TableBase
):
1411 u
"""Major areas of the world: Kanto, Johto, etc.
1413 __tablename__
= 'regions'
1414 __singlename__
= 'region'
1415 id = Column(Integer
, primary_key
=True, nullable
=False,
1416 info
=dict(description
=u
"A numeric ID"))
1417 identifier
= Column(Unicode(16), nullable
=False,
1418 info
=dict(description
=u
"An identifier", format
='identifier'))
1419 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
1420 info
=dict(description
="The name", format
='plaintext', official
=True))
1422 class Stat(TableBase
):
1423 u
"""A Stat, such as Attack or Speed
1425 __tablename__
= 'stats'
1426 __singlename__
= 'stat'
1427 id = Column(Integer
, primary_key
=True, nullable
=False,
1428 info
=dict(description
=u
"A numeric ID"))
1429 damage_class_id
= Column(Integer
, ForeignKey('move_damage_classes.id'), nullable
=True,
1430 info
=dict(description
=u
"For offensive and defensive stats, the damage this stat relates to; otherwise None (the NULL value)"))
1431 identifier
= Column(Unicode(16), nullable
=False,
1432 info
=dict(description
=u
"An identifier", format
='identifier'))
1433 name
= TextColumn(Unicode(16), nullable
=False, index
=True, plural
='names',
1434 info
=dict(description
="The name", format
='plaintext', official
=True))
1436 class StatHint(TableBase
):
1437 u
"""Flavor text for genes that appears in a Pokémon's summary. Sometimes
1438 called "characteristics".
1440 __tablename__
= 'stat_hints'
1441 __singlename__
= 'stat_hint'
1442 id = Column(Integer
, primary_key
=True, nullable
=False,
1443 info
=dict(description
=u
"A numeric ID"))
1444 stat_id
= Column(Integer
, ForeignKey('stats.id'), nullable
=False,
1445 info
=dict(description
=u
"ID of the highest stat"))
1446 gene_mod_5
= Column(Integer
, nullable
=False, index
=True,
1447 info
=dict(description
=u
"Value of the highest stat modulo 5"))
1448 message
= TextColumn(Unicode(24), plural
='messages', nullable
=False, index
=True, unique
=True,
1449 info
=dict(description
=u
"The text displayed", official
=True, format
='plaintext'))
1451 class SuperContestCombo(TableBase
):
1452 u
"""Combo of two moves in a Super Contest.
1454 __tablename__
= 'super_contest_combos'
1455 first_move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1456 info
=dict(description
=u
"The ID of the first move in the combo."))
1457 second_move_id
= Column(Integer
, ForeignKey('moves.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1458 info
=dict(description
=u
"The ID of the second and last move."))
1460 class SuperContestEffect(TableBase
):
1461 u
"""An effect a move can have when used in the Super Contest
1463 __tablename__
= 'super_contest_effects'
1464 __singlename__
= 'super_contest_effect'
1465 id = Column(Integer
, primary_key
=True, nullable
=False,
1466 info
=dict(description
=u
"This effect's unique ID."))
1467 appeal
= Column(SmallInteger
, nullable
=False,
1468 info
=dict(description
=u
"The number of hearts the user gains."))
1469 flavor_text
= ProseColumn(Unicode(64), plural
='flavor_texts', nullable
=False,
1470 info
=dict(description
=u
"A description of the effect.", format
='plaintext'))
1472 class TypeEfficacy(TableBase
):
1473 u
"""The damage multiplier used when a move of a particular type damages a
1474 Pokémon of a particular other type.
1476 __tablename__
= 'type_efficacy'
1477 damage_type_id
= Column(Integer
, ForeignKey('types.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1478 info
=dict(description
=u
"The ID of the damaging type."))
1479 target_type_id
= Column(Integer
, ForeignKey('types.id'), primary_key
=True, nullable
=False, autoincrement
=False,
1480 info
=dict(description
=u
"The ID of the defending Pokémon's type."))
1481 damage_factor
= Column(Integer
, nullable
=False,
1482 info
=dict(description
=u
"The multiplier, as a percentage of damage inflicted."))
1484 class Type(TableBase
):
1485 u
"""Any of the elemental types Pokémon and moves can have."""
1486 __tablename__
= 'types'
1487 __singlename__
= 'type'
1488 id = Column(Integer
, primary_key
=True, nullable
=False,
1489 info
=dict(description
=u
"A unique ID for this type."))
1490 identifier
= Column(Unicode(12), nullable
=False,
1491 info
=dict(description
=u
"An identifier", format
='identifier'))
1492 generation_id
= Column(Integer
, ForeignKey('generations.id'), nullable
=False,
1493 info
=dict(description
=u
"The ID of the generation this type first appeared in."))
1494 damage_class_id
= Column(Integer
, ForeignKey('move_damage_classes.id'), nullable
=True,
1495 info
=dict(description
=u
"The ID of the damage class this type's moves had before Generation IV, null if not applicable (e.g. ???)."))
1496 name
= TextColumn(Unicode(12), nullable
=False, index
=True, plural
='names',
1497 info
=dict(description
="The name", format
='plaintext', official
=True))
1499 class VersionGroup(TableBase
):
1500 u
"""A group of versions, containing either two paired versions (such as Red
1501 and Blue) or a single game (such as Yellow.)
1503 __tablename__
= 'version_groups'
1504 id = Column(Integer
, primary_key
=True, nullable
=False,
1505 info
=dict(description
=u
"This version group's unique ID."))
1506 generation_id
= Column(Integer
, ForeignKey('generations.id'), nullable
=False,
1507 info
=dict(description
=u
"The ID of the generation the games in this group belong to."))
1508 pokedex_id
= Column(Integer
, ForeignKey('pokedexes.id'), nullable
=False,
1509 info
=dict(description
=u
"The ID of the regional Pokédex used in this version group."))
1511 class VersionGroupRegion(TableBase
):
1512 u
"""Maps a version group to a region that appears in it."""
1513 __tablename__
= 'version_group_regions'
1514 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), primary_key
=True, nullable
=False,
1515 info
=dict(description
=u
"The ID of the version group."))
1516 region_id
= Column(Integer
, ForeignKey('regions.id'), primary_key
=True, nullable
=False,
1517 info
=dict(description
=u
"The ID of the region."))
1519 class Version(TableBase
):
1520 u
"""An individual main-series Pokémon game."""
1521 __tablename__
= 'versions'
1522 __singlename__
= 'version'
1523 id = Column(Integer
, primary_key
=True, nullable
=False,
1524 info
=dict(description
=u
"A unique ID for this version."))
1525 version_group_id
= Column(Integer
, ForeignKey('version_groups.id'), nullable
=False,
1526 info
=dict(description
=u
"The ID of the version group this game belongs to."))
1527 identifier
= Column(Unicode(32), nullable
=False,
1528 info
=dict(description
=u
'And identifier', format
='identifier'))
1529 name
= TextColumn(Unicode(32), nullable
=False, index
=True, plural
='names',
1530 info
=dict(description
="The name", format
='plaintext', official
=True))
1533 ### Relations down here, to avoid ordering problems
1534 Ability
.changelog
= relation(AbilityChangelog
,
1535 order_by
=AbilityChangelog
.changed_in_version_group_id
.desc(),
1538 Ability
.flavor_text
= relation(AbilityFlavorText
, order_by
=AbilityFlavorText
.version_group_id
, backref
='ability')
1539 Ability
.generation
= relation(Generation
, backref
='abilities')
1540 Ability
.all_pokemon
= relation(Pokemon
,
1541 secondary
=PokemonAbility
.__table__
,
1542 order_by
=Pokemon
.order
,
1543 back_populates
='all_abilities',
1545 Ability
.pokemon
= relation(Pokemon
,
1546 secondary
=PokemonAbility
.__table__
,
1548 PokemonAbility
.ability_id
== Ability
.id,
1549 PokemonAbility
.is_dream
== False
1551 order_by
=Pokemon
.order
,
1552 back_populates
='abilities',
1554 Ability
.dream_pokemon
= relation(Pokemon
,
1555 secondary
=PokemonAbility
.__table__
,
1557 PokemonAbility
.ability_id
== Ability
.id,
1558 PokemonAbility
.is_dream
== True
1560 order_by
=Pokemon
.order
,
1561 back_populates
='dream_ability',
1564 AbilityChangelog
.changed_in
= relation(VersionGroup
, backref
='ability_changelog')
1566 AbilityFlavorText
.version_group
= relation(VersionGroup
)
1568 Berry
.berry_firmness
= relation(BerryFirmness
, backref
='berries')
1569 Berry
.firmness
= association_proxy('berry_firmness', 'name')
1570 Berry
.flavors
= relation(BerryFlavor
, order_by
=BerryFlavor
.contest_type_id
, backref
='berry')
1571 Berry
.natural_gift_type
= relation(Type
)
1573 BerryFlavor
.contest_type
= relation(ContestType
)
1575 ContestCombo
.first
= relation(Move
, primaryjoin
=ContestCombo
.first_move_id
==Move
.id,
1576 backref
='contest_combo_first')
1577 ContestCombo
.second
= relation(Move
, primaryjoin
=ContestCombo
.second_move_id
==Move
.id,
1578 backref
='contest_combo_second')
1580 Encounter
.location_area
= relation(LocationArea
, backref
='encounters')
1581 Encounter
.pokemon
= relation(Pokemon
, backref
='encounters')
1582 Encounter
.version
= relation(Version
, backref
='encounters')
1583 Encounter
.slot
= relation(EncounterSlot
, backref
='encounters')
1585 EncounterConditionValue
.condition
= relation(EncounterCondition
, backref
='values')
1587 Encounter
.condition_value_map
= relation(EncounterConditionValueMap
, backref
='encounter')
1588 Encounter
.condition_values
= association_proxy('condition_value_map', 'condition_value')
1589 EncounterConditionValueMap
.condition_value
= relation(EncounterConditionValue
,
1590 backref
='encounter_map')
1592 EncounterSlot
.terrain
= relation(EncounterTerrain
, backref
='slots')
1593 EncounterSlot
.version_group
= relation(VersionGroup
)
1595 EvolutionChain
.growth_rate
= relation(GrowthRate
, backref
='evolution_chains')
1596 EvolutionChain
.baby_trigger_item
= relation(Item
, backref
='evolution_chains')
1597 EvolutionChain
.pokemon
= relation(Pokemon
, order_by
=Pokemon
.order
, back_populates
='evolution_chain')
1599 Experience
.growth_rate
= relation(GrowthRate
, backref
='experience_table')
1601 Generation
.canonical_pokedex
= relation(Pokedex
, backref
='canonical_for_generation')
1602 Generation
.versions
= relation(Version
, secondary
=VersionGroup
.__table__
)
1603 Generation
.main_region
= relation(Region
)
1605 GrowthRate
.max_experience_obj
= relation(Experience
, primaryjoin
=and_(Experience
.growth_rate_id
== GrowthRate
.id, Experience
.level
== 100), uselist
=False)
1606 GrowthRate
.max_experience
= association_proxy('max_experience_obj', 'experience')
1608 Item
.berry
= relation(Berry
, uselist
=False, backref
='item')
1609 Item
.flags
= relation(ItemFlag
, secondary
=ItemFlagMap
.__table__
)
1610 Item
.flavor_text
= relation(ItemFlavorText
, order_by
=ItemFlavorText
.version_group_id
.asc(), backref
='item')
1611 Item
.fling_effect
= relation(ItemFlingEffect
, backref
='items')
1612 Item
.machines
= relation(Machine
, order_by
=Machine
.version_group_id
.asc())
1613 Item
.category
= relation(ItemCategory
)
1614 Item
.pocket
= association_proxy('category', 'pocket')
1616 ItemCategory
.items
= relation(Item
, order_by
=Item
.identifier
)
1617 ItemCategory
.pocket
= relation(ItemPocket
)
1619 ItemFlavorText
.version_group
= relation(VersionGroup
)
1621 ItemInternalID
.item
= relation(Item
, backref
='internal_ids')
1622 ItemInternalID
.generation
= relation(Generation
)
1624 ItemPocket
.categories
= relation(ItemCategory
, order_by
=ItemCategory
.identifier
)
1626 Location
.region
= relation(Region
, backref
='locations')
1628 LocationArea
.location
= relation(Location
, backref
='areas')
1630 LocationInternalID
.location
= relation(Location
, backref
='internal_ids')
1631 LocationInternalID
.generation
= relation(Generation
)
1633 Machine
.item
= relation(Item
)
1634 Machine
.version_group
= relation(VersionGroup
)
1636 Move
.changelog
= relation(MoveChangelog
,
1637 order_by
=MoveChangelog
.changed_in_version_group_id
.desc(),
1640 Move
.contest_effect
= relation(ContestEffect
, backref
='moves')
1641 Move
.contest_combo_next
= association_proxy('contest_combo_first', 'second')
1642 Move
.contest_combo_prev
= association_proxy('contest_combo_second', 'first')
1643 Move
.contest_type
= relation(ContestType
, backref
='moves')
1644 Move
.damage_class
= relation(MoveDamageClass
, backref
='moves')
1645 Move
.flags
= association_proxy('move_flags', 'flag')
1646 Move
.flavor_text
= relation(MoveFlavorText
, order_by
=MoveFlavorText
.version_group_id
, backref
='move')
1647 Move
.generation
= relation(Generation
, backref
='moves')
1648 Move
.machines
= relation(Machine
, backref
='move')
1649 Move
.meta
= relation(MoveMeta
, uselist
=False, backref
='move')
1650 Move
.meta_stat_changes
= relation(MoveMetaStatChange
)
1651 Move
.move_effect
= relation(MoveEffect
, backref
='moves')
1652 Move
.move_flags
= relation(MoveFlag
, backref
='move')
1653 Move
.super_contest_effect
= relation(SuperContestEffect
, backref
='moves')
1654 Move
.super_contest_combo_next
= association_proxy('super_contest_combo_first', 'second')
1655 Move
.super_contest_combo_prev
= association_proxy('super_contest_combo_second', 'first')
1656 Move
.target
= relation(MoveTarget
, backref
='moves')
1657 Move
.type = relation(Type
, back_populates
='moves')
1659 MoveChangelog
.changed_in
= relation(VersionGroup
, backref
='move_changelog')
1660 MoveChangelog
.move_effect
= relation(MoveEffect
, backref
='move_changelog')
1661 MoveChangelog
.type = relation(Type
, backref
='move_changelog')
1663 MoveEffect
.category_map
= relation(MoveEffectCategoryMap
)
1664 MoveEffect
.categories
= association_proxy('category_map', 'category')
1665 MoveEffect
.changelog
= relation(MoveEffectChangelog
,
1666 order_by
=MoveEffectChangelog
.changed_in_version_group_id
.desc(),
1667 backref
='move_effect',
1669 MoveEffectCategoryMap
.category
= relation(MoveEffectCategory
)
1671 MoveEffectChangelog
.changed_in
= relation(VersionGroup
, backref
='move_effect_changelog')
1673 MoveFlag
.flag
= relation(MoveFlagType
)
1675 MoveFlavorText
.version_group
= relation(VersionGroup
)
1677 MoveMeta
.category
= relation(MoveMetaCategory
, backref
='move_meta')
1678 MoveMeta
.ailment
= relation(MoveMetaAilment
, backref
='move_meta')
1680 MoveMetaStatChange
.stat
= relation(Stat
, backref
='move_meta_stat_changes')
1682 Nature
.decreased_stat
= relation(Stat
, primaryjoin
=Nature
.decreased_stat_id
==Stat
.id,
1683 backref
='decreasing_natures')
1684 Nature
.increased_stat
= relation(Stat
, primaryjoin
=Nature
.increased_stat_id
==Stat
.id,
1685 backref
='increasing_natures')
1686 Nature
.hates_flavor
= relation(ContestType
, primaryjoin
=Nature
.hates_flavor_id
==ContestType
.id,
1687 backref
='hating_natures')
1688 Nature
.likes_flavor
= relation(ContestType
, primaryjoin
=Nature
.likes_flavor_id
==ContestType
.id,
1689 backref
='liking_natures')
1690 Nature
.battle_style_preferences
= relation(NatureBattleStylePreference
,
1691 order_by
=NatureBattleStylePreference
.move_battle_style_id
,
1693 Nature
.pokeathlon_effects
= relation(NaturePokeathlonStat
, order_by
=NaturePokeathlonStat
.pokeathlon_stat_id
)
1695 NatureBattleStylePreference
.battle_style
= relation(MoveBattleStyle
, backref
='nature_preferences')
1697 NaturePokeathlonStat
.pokeathlon_stat
= relation(PokeathlonStat
, backref
='nature_effects')
1699 Pokedex
.region
= relation(Region
, backref
='pokedexes')
1700 Pokedex
.version_groups
= relation(VersionGroup
, order_by
=VersionGroup
.id, back_populates
='pokedex')
1702 Pokemon
.all_abilities
= relation(Ability
,
1703 secondary
=PokemonAbility
.__table__
,
1704 order_by
=PokemonAbility
.slot
,
1706 Pokemon
.abilities
= relation(Ability
,
1707 secondary
=PokemonAbility
.__table__
,
1709 Pokemon
.id == PokemonAbility
.pokemon_id
,
1710 PokemonAbility
.is_dream
== False,
1712 order_by
=PokemonAbility
.slot
,
1714 Pokemon
.dream_ability
= relation(Ability
,
1715 secondary
=PokemonAbility
.__table__
,
1717 Pokemon
.id == PokemonAbility
.pokemon_id
,
1718 PokemonAbility
.is_dream
== True,
1722 Pokemon
.pokemon_color
= relation(PokemonColor
, backref
='pokemon')
1723 Pokemon
.color
= association_proxy('pokemon_color', 'name')
1724 Pokemon
.dex_numbers
= relation(PokemonDexNumber
, order_by
=PokemonDexNumber
.pokedex_id
.asc(), backref
='pokemon')
1725 Pokemon
.egg_groups
= relation(EggGroup
, secondary
=PokemonEggGroup
.__table__
,
1726 order_by
=PokemonEggGroup
.egg_group_id
,
1727 backref
=backref('pokemon', order_by
=Pokemon
.order
))
1728 Pokemon
.evolution_chain
= relation(EvolutionChain
, back_populates
='pokemon')
1729 Pokemon
.child_pokemon
= relation(Pokemon
,
1730 primaryjoin
=Pokemon
.id==PokemonEvolution
.from_pokemon_id
,
1731 secondary
=PokemonEvolution
.__table__
,
1732 secondaryjoin
=PokemonEvolution
.to_pokemon_id
==Pokemon
.id,
1733 backref
=backref('parent_pokemon', uselist
=False),
1735 Pokemon
.flavor_text
= relation(PokemonFlavorText
, order_by
=PokemonFlavorText
.version_id
.asc(), backref
='pokemon')
1736 Pokemon
.forms
= relation(PokemonForm
, primaryjoin
=Pokemon
.id==PokemonForm
.form_base_pokemon_id
,
1737 order_by
=(PokemonForm
.order
.asc(), PokemonForm
.identifier
.asc()))
1738 Pokemon
.default_form
= relation(PokemonForm
,
1739 primaryjoin
=and_(Pokemon
.id==PokemonForm
.form_base_pokemon_id
, PokemonForm
.is_default
==True),
1742 Pokemon
.pokemon_habitat
= relation(PokemonHabitat
, backref
='pokemon')
1743 Pokemon
.habitat
= association_proxy('pokemon_habitat', 'name')
1744 Pokemon
.items
= relation(PokemonItem
, backref
='pokemon')
1745 Pokemon
.generation
= relation(Generation
, backref
='pokemon')
1746 Pokemon
.shape
= relation(PokemonShape
, backref
='pokemon')
1747 Pokemon
.stats
= relation(PokemonStat
, backref
='pokemon', order_by
=PokemonStat
.stat_id
.asc())
1748 Pokemon
.types
= relation(Type
, secondary
=PokemonType
.__table__
,
1749 order_by
=PokemonType
.slot
.asc(),
1750 back_populates
='pokemon')
1752 PokemonDexNumber
.pokedex
= relation(Pokedex
)
1754 PokemonEvolution
.from_pokemon
= relation(Pokemon
,
1755 primaryjoin
=PokemonEvolution
.from_pokemon_id
==Pokemon
.id,
1756 backref
='child_evolutions',
1758 PokemonEvolution
.to_pokemon
= relation(Pokemon
,
1759 primaryjoin
=PokemonEvolution
.to_pokemon_id
==Pokemon
.id,
1760 backref
=backref('parent_evolution', uselist
=False),
1762 PokemonEvolution
.child_evolutions
= relation(PokemonEvolution
,
1763 primaryjoin
=PokemonEvolution
.from_pokemon_id
==PokemonEvolution
.to_pokemon_id
,
1764 foreign_keys
=[PokemonEvolution
.to_pokemon_id
],
1765 backref
=backref('parent_evolution',
1766 remote_side
=[PokemonEvolution
.from_pokemon_id
],
1770 PokemonEvolution
.trigger
= relation(EvolutionTrigger
, backref
='evolutions')
1771 PokemonEvolution
.trigger_item
= relation(Item
,
1772 primaryjoin
=PokemonEvolution
.trigger_item_id
==Item
.id,
1773 backref
='triggered_evolutions',
1775 PokemonEvolution
.held_item
= relation(Item
,
1776 primaryjoin
=PokemonEvolution
.held_item_id
==Item
.id,
1777 backref
='required_for_evolutions',
1779 PokemonEvolution
.location
= relation(Location
, backref
='triggered_evolutions')
1780 PokemonEvolution
.known_move
= relation(Move
, backref
='triggered_evolutions')
1781 PokemonEvolution
.party_pokemon
= relation(Pokemon
,
1782 primaryjoin
=PokemonEvolution
.party_pokemon_id
==Pokemon
.id,
1783 backref
='triggered_evolutions',
1785 PokemonEvolution
.trade_pokemon
= relation(Pokemon
,
1786 primaryjoin
=PokemonEvolution
.trade_pokemon_id
==Pokemon
.id,
1789 PokemonFlavorText
.version
= relation(Version
)
1791 PokemonForm
.form_base_pokemon
= relation(Pokemon
, primaryjoin
=PokemonForm
.form_base_pokemon_id
==Pokemon
.id)
1792 PokemonForm
.unique_pokemon
= relation(Pokemon
, backref
=backref('unique_form', uselist
=False),
1793 primaryjoin
=PokemonForm
.unique_pokemon_id
==Pokemon
.id)
1794 PokemonForm
.version_group
= relation(VersionGroup
)
1795 PokemonForm
.form_group
= association_proxy('form_base_pokemon', 'form_group')
1796 PokemonForm
.pokeathlon_stats
= relation(PokemonFormPokeathlonStat
,
1797 order_by
=PokemonFormPokeathlonStat
.pokeathlon_stat_id
,
1798 backref
='pokemon_form')
1800 PokemonFormGroup
.pokemon
= relation(Pokemon
, backref
=backref('form_group',
1803 PokemonFormPokeathlonStat
.pokeathlon_stat
= relation(PokeathlonStat
)
1805 PokemonItem
.item
= relation(Item
, backref
='pokemon')
1806 PokemonItem
.version
= relation(Version
)
1808 PokemonMove
.pokemon
= relation(Pokemon
, backref
='pokemon_moves')
1809 PokemonMove
.version_group
= relation(VersionGroup
)
1810 PokemonMove
.machine
= relation(Machine
, backref
='pokemon_moves',
1811 primaryjoin
=and_(Machine
.version_group_id
==PokemonMove
.version_group_id
,
1812 Machine
.move_id
==PokemonMove
.move_id
),
1813 foreign_keys
=[Machine
.version_group_id
, Machine
.move_id
],
1815 PokemonMove
.move
= relation(Move
, backref
='pokemon_moves')
1816 PokemonMove
.method
= relation(PokemonMoveMethod
)
1818 PokemonStat
.stat
= relation(Stat
)
1820 # This is technically a has-many; Generation.main_region_id -> Region.id
1821 Region
.generation
= relation(Generation
, uselist
=False)
1822 Region
.version_group_regions
= relation(VersionGroupRegion
, backref
='region',
1823 order_by
='VersionGroupRegion.version_group_id')
1824 Region
.version_groups
= association_proxy('version_group_regions', 'version_group')
1826 Stat
.damage_class
= relation(MoveDamageClass
, backref
='stats')
1828 StatHint
.stat
= relation(Stat
, backref
='hints')
1830 SuperContestCombo
.first
= relation(Move
, primaryjoin
=SuperContestCombo
.first_move_id
==Move
.id,
1831 backref
='super_contest_combo_first')
1832 SuperContestCombo
.second
= relation(Move
, primaryjoin
=SuperContestCombo
.second_move_id
==Move
.id,
1833 backref
='super_contest_combo_second')
1835 Type
.damage_efficacies
= relation(TypeEfficacy
,
1837 ==TypeEfficacy
.damage_type_id
,
1838 backref
='damage_type')
1839 Type
.target_efficacies
= relation(TypeEfficacy
,
1841 ==TypeEfficacy
.target_type_id
,
1842 backref
='target_type')
1844 Type
.generation
= relation(Generation
, backref
='types')
1845 Type
.damage_class
= relation(MoveDamageClass
, backref
='types')
1846 Type
.pokemon
= relation(Pokemon
, secondary
=PokemonType
.__table__
,
1847 order_by
=Pokemon
.order
,
1848 back_populates
='types')
1849 Type
.moves
= relation(Move
, back_populates
='type', order_by
=Move
.id)
1851 Version
.version_group
= relation(VersionGroup
, back_populates
='versions')
1852 Version
.generation
= association_proxy('version_group', 'generation')
1854 VersionGroup
.versions
= relation(Version
, order_by
=Version
.id, back_populates
='version_group')
1855 VersionGroup
.generation
= relation(Generation
, backref
='version_groups')
1856 VersionGroup
.version_group_regions
= relation(VersionGroupRegion
, backref
='version_group')
1857 VersionGroup
.regions
= association_proxy('version_group_regions', 'region')
1858 VersionGroup
.pokedex
= relation(Pokedex
, back_populates
='version_groups')
1861 ### Add text/prose tables
1863 default_lang
= u
'en'
1865 def makeTextTable(object_table
, name_plural
, name_singular
, columns
, lazy
):
1866 # With "Language", we'd have two language_id. So, rename one to 'lang'
1867 safe_name
= object_table
.__singlename__
1868 if safe_name
== 'language':
1871 tablename
= object_table
.__singlename__
+ '_' + name_plural
1872 singlename
= object_table
.__singlename__
+ '_' + name_singular
1874 class Strings(object):
1875 __tablename__
= tablename
1876 __singlename__
= singlename
1877 _attrname
= name_plural
1878 _language_identifier
= association_proxy('language', 'identifier')
1880 for name
, plural
, column
in columns
:
1882 if not column
.nullable
:
1883 # A Python side default value, so that the strings can be set
1884 # one by one without the DB complaining about missing values
1885 column
.default
= ColumnDefault(u
'')
1887 table
= Table(tablename
, metadata
,
1888 Column(safe_name
+ '_id', Integer
, ForeignKey(object_table
.id),
1889 primary_key
=True, nullable
=False),
1890 Column('language_id', Integer
, ForeignKey(Language
.id),
1891 primary_key
=True, index
=True, nullable
=False),
1892 *(column
for name
, plural
, column
in columns
)
1895 mapper(Strings
, table
,
1897 "object_id": synonym(safe_name
+ '_id'),
1898 "language": relation(Language
,
1899 primaryjoin
=table
.c
.language_id
== Language
.id,
1901 safe_name
: relation(object_table
,
1902 primaryjoin
=(object_table
.id == table
.c
[safe_name
+ "_id"]),
1903 backref
=backref(name_plural
,
1904 collection_class
=attribute_mapped_collection('language'),
1911 # The relation to the object
1912 Strings
.object = getattr(Strings
, safe_name
)
1914 # Link the tables themselves, so we can get them if needed
1915 Strings
.object_table
= object_table
1916 setattr(object_table
, name_singular
+ '_table', Strings
)
1918 for colname
, pluralname
, column
in columns
:
1919 # Provide a property with all the names, and an English accessor
1920 # for backwards compatibility
1921 setattr(object_table
, pluralname
, StringProperty(
1922 object_table
, Strings
, colname
,
1924 setattr(object_table
, colname
, DefaultLangProperty(pluralname
))
1926 if colname
== 'name':
1927 object_table
.name_table
= Strings
1931 class StringProperty(object):
1932 def __init__(self
, cls
, stringclass
, colname
):
1934 self
.colname
= colname
1935 self
.stringclass
= stringclass
1937 def __get__(self
, instance
, cls
):
1939 return StringMapping(instance
, self
)
1943 def __getitem__(self
, lang
):
1944 return StringExpression(self
, lang
)
1947 return '<StringDict %s.%s>' %
(self
.cls
, self
.colname
)
1949 class StringMapping(collections
.MutableMapping
):
1950 def __init__(self
, instance
, prop
):
1951 self
.stringclass
= prop
.stringclass
1952 self
.instance
= instance
1953 self
.strings
= getattr(instance
, prop
.stringclass
._attrname
)
1954 self
.colname
= prop
.colname
1957 return len(self
.strings
)
1960 return iter(self
.strings
)
1962 def __contains__(self
, lang
):
1963 return lang
in self
.strings
1965 def __getitem__(self
, lang
):
1966 return getattr(self
.strings
[lang
], self
.colname
)
1968 def __setitem__(self
, lang
, value
):
1970 # Modifying an existing row
1971 row
= self
.strings
[lang
]
1973 # We need do add a whole row for the language
1974 row
= self
.stringclass()
1975 row
.object_id
= self
.instance
.id
1976 session
= object_session(self
.instance
)
1977 if isinstance(lang
, basestring
):
1978 lang
= session
.query(Language
).filter_by(
1979 identifier
=lang
).one()
1981 self
.strings
[lang
] = row
1983 return setattr(row
, self
.colname
, value
)
1985 def __delitem__(self
, lang
):
1986 raise NotImplementedError('Cannot delete a single string. '
1987 'Perhaps you wan to delete all of %s.%s?' %
1988 (self
.instance
, self
.stringclass
._attrname
)
1991 class StringExpression(ColumnOperators
):
1992 def __init__(self
, prop
, lang
):
1994 self
.column
= getattr(prop
.stringclass
, prop
.colname
)
1995 self
.lang_column
= prop
.stringclass
._language_identifier
1996 if isinstance(lang
, basestring
):
1999 self
.lang
= lang
.identifier
2001 def operate(self
, op
, *values
, **kwargs
):
2002 return getattr(self
.prop
.cls
, self
.prop
.stringclass
._attrname
).any(and_(
2003 self
.lang_column
== self
.lang
,
2004 op(self
.column
, *values
, **kwargs
),
2007 class DefaultLangProperty(object):
2008 def __init__(self
, colname
):
2009 self
.colname
= colname
2011 def __get__(self
, instance
, cls
):
2013 return getattr(instance
, self
.colname
)[default_lang
]
2015 return getattr(cls
, self
.colname
)[default_lang
]
2017 def __set__(self
, instance
, value
):
2018 getattr(instance
, self
.colname
)[default_lang
] = value
2020 def __delete__(self
, instance
):
2021 del getattr(instance
, self
.colname
)[default_lang
]
2023 for table
in list(table_classes
):
2024 # Find all the language-specific columns, keeping them in the order they
2027 for colname
in dir(table
):
2028 column
= getattr(table
, colname
)
2029 if isinstance(column
, LanguageSpecificColumn
):
2030 all_columns
.append((colname
, column
))
2031 delattr(table
, colname
)
2032 all_columns
.sort(key
=lambda pair
: pair
[1].order
)
2034 # Break them into text and prose columns
2037 for colname
, column
in all_columns
:
2038 spec
= colname
, column
.plural
, column
.makeSAColumn()
2039 if isinstance(column
, TextColumn
):
2040 text_columns
.append(spec
)
2041 elif isinstance(column
, ProseColumn
):
2042 prose_columns
.append(spec
)
2044 if (text_columns
or prose_columns
) and issubclass(table
, LanguageSpecific
):
2045 raise AssertionError("Language-specific table %s shouldn't have explicit language-specific columns" % table
)
2048 string_table
= makeTextTable(table
, 'texts', 'text', text_columns
, lazy
=False)
2050 string_table
= makeTextTable(table
, 'prose', 'prose', prose_columns
, lazy
=True)
2052 ### Add language relations
2053 for table
in list(table_classes
):
2054 if issubclass(table
, LanguageSpecific
):
2055 table
.language
= relation(Language
, primaryjoin
=table
.language_id
== Language
.id)
2057 Move
.effect
= DefaultLangProperty('effects')
2058 Move
.effects
= markdown
.MoveEffectsProperty('effect')
2059 Move
.short_effect
= DefaultLangProperty('short_effects')
2060 Move
.short_effects
= markdown
.MoveEffectsProperty('short_effect')
2062 MoveChangelog
.effect
= DefaultLangProperty('effects')
2063 MoveChangelog
.effects
= markdown
.MoveEffectsProperty('effect')
2064 MoveChangelog
.short_effect
= DefaultLangProperty('short_effects')
2065 MoveChangelog
.short_effects
= markdown
.MoveEffectsProperty('short_effect')