#include "global.h" #include "entity.h" #include "coord.h" #include "room.h" #include "object.h" #include "functions.h" #include "definitions.h" extern const Hitbox* const gObjectHitboxes[]; u32 LoadObjectSprite(Entity* this, s32 type, const ObjectDefinition* definition); extern const ObjectDefinition gObjectDefinitions[]; extern Entity* sub_0805E744(void); Entity* CreateWaterTrace(Entity*); Entity* CreateSpeechBubble(Entity*, u32, s32, s32); Entity* sub_080A276C(Entity* parent, u32 type, u32 type2) { Entity* e = sub_0805E744(); if (e != NULL) { e->id = 0xC1; e->kind = OBJECT; e->type = type; e->type2 = type2; e->parent = parent; AppendEntityToList(e, 6); PrependEntityToList(e, 6); CopyPosition(&gPlayerEntity, e); gPriorityHandler.sys_priority = 6; gUnk_02034490.unk0 = 1; e->field_0x68.HALF.HI = gPlayerEntity.flags; e->field_0x68.HALF.LO = gPlayerEntity.spriteSettings.draw; e->field_0x6a.HALF.LO = gPlayerEntity.iframes; e->field_0x6a.HALF.HI = gPlayerState.field_0x7; e->field_0x6c.HALF.LO = gPlayerState.keepFacing; e->field_0x6c.HALF.HI = gPlayerState.field_0xa; e->field_0x6e.HALF.LO = gPlayerState.field_0x27[0]; e->field_0x6e.HALF.HI = gPlayerState.mobility; e->field_0x70.WORD = gPlayerState.flags; e->field_0x74.HALF.LO = gPlayerState.field_0x82[8]; gPlayerEntity.flags &= ~ENT_COLLIDE; gPlayerEntity.spriteSettings.draw = 0; } return e; } void ObjectInit(Entity* this) { const ObjectDefinition* definition = &gObjectDefinitions[this->id]; if (LoadObjectSprite(this, this->type, definition) == 2) { UpdateSpriteForCollisionLayer(this); } } u32 LoadObjectSprite(Entity* this, s32 type, const ObjectDefinition* definition) { u16 tmp1; u8 tmp2; if ((this->flags & ENT_DID_INIT) != 0) { // Sprite already loaded return 1; } if ((definition->bitfield.type) == 0) { // Object has no sprite this->flags |= ENT_DID_INIT | definition->bitfield.flags << 2; return 1; } if ((u8)(((*((u8*)definition) & 0xff & 3) - 2)) < 2) { // Multiple forms definition = &definition->data.definition[type]; } tmp1 = definition->bitfield.gfx; switch (definition->bitfield.gfx_type) { case 2: this->spriteVramOffset = definition->bitfield.gfx; break; case 1: if (!LoadSwapGFX(this, tmp1, 0)) { return 0; } break; default: if (!LoadFixedGFX(this, tmp1)) { return 0; } break; } tmp1 = definition->data.sprite.paletteIndex; LoadObjPalette(this, tmp1); tmp2 = 0xff; this->animIndex = tmp2; this->hurtType = 0x48; this->spriteIndex = definition->data.sprite.spriteIndex; this->spriteSettings.shadow = definition->data.sprite.shadow; this->spritePriority.b1 = definition->data.sprite.spritePriority; this->spriteSettings.draw = definition->data.sprite.draw; this->hitbox = (Hitbox*)gObjectHitboxes[definition->bitfield.hitbox]; this->flags |= ENT_DID_INIT | definition->bitfield.flags << 2; return 2; } Entity* CreateObject(u32 subtype, u32 form, u32 type2) { Entity* entity = GetEmptyEntity(); if (entity != NULL) { entity->kind = OBJECT; entity->id = subtype; entity->type = form; entity->type2 = type2; AppendEntityToList(entity, 6); } return entity; } Entity* CreateObjectWithParent(Entity* parent, u32 subtype, u32 form, u32 type2) { Entity* entity = CreateObject(subtype, form, type2); if (entity != NULL) { entity->parent = parent; CopyPosition(parent, entity); } return entity; } Entity* CreateFx(Entity* parentEnt, u32 form, u32 parameter) { return CreateObjectWithParent(parentEnt, SPECIAL_FX, form, parameter); } void CreateDust(Entity* parent) { CreateFx(parent, FX_DEATH, 0); } void CreateDustAt(s32 xOff, s32 yOff, u32 layer) { Entity* ent; ent = CreateObject(SPECIAL_FX, FX_DEATH, 0); if (ent != NULL) { ent->x.HALF.HI = gRoomControls.origin_x + xOff; ent->y.HALF.HI = gRoomControls.origin_y + yOff; ent->collisionLayer = layer; } } void CreateDustSmall(Entity* parent) { CreateFx(parent, FX_DASH, 0); } void CreateExplosionBroken(Entity* parent) { CreateFx(parent, FX_6, 0); } void CreateWaterSplash(Entity* parent) { CreateFx(parent, FX_WATER_SPLASH, 0); } Entity* CreateGroundItem(Entity* parent, u32 form, u32 subtype) { Entity* ent; ent = CreateObjectWithParent(parent, GROUND_ITEM, form, subtype); if (ent != NULL) { ent->actionDelay = 5; } return ent; } Entity* CreateGroundItemWithFlags(Entity* parent, u32 form, u32 subtype, u32 flags) { Entity* ent; ent = CreateObjectWithParent(parent, GROUND_ITEM, form, subtype); if (ent != NULL) { ent->actionDelay = 5; ent->field_0x86.HWORD = flags; } return ent; } Entity* CreateWaterTrace(Entity* parent) { Entity* ent; ent = CreateFx(parent, FX_RIPPLE, 0); if (ent != NULL) { ent->spritePriority.b0 = 7; } return ent; } void CreateRandomWaterTrace(Entity* parent, int range) { s32 sVar1, sVar2, sVar3; Entity* ent; u32 uVar3; ent = CreateWaterTrace(parent); if (ent != NULL) { uVar3 = Random(); sVar1 = (int)uVar3 % (++range); uVar3 >>= 8; if ((uVar3 & 1) != 0) { sVar1 = -sVar1; } ent->x.HALF.HI += sVar1; uVar3 >>= 8; sVar2 = (int)uVar3 % range; uVar3 >>= 8; if ((uVar3 & 1) != 0) { sVar2 = -sVar2; } ent->y.HALF.HI += sVar2; } } Entity* CreateLargeWaterTrace(Entity* parent) { Entity* ent = CreateFx(parent, FX_RIPPLE_LARGE, 0); if (ent != NULL) { ent->spritePriority.b0 = 7; } return ent; } NONMATCH("asm/non_matching/objectUtils/sub_080A2AF4.inc", void sub_080A2AF4(Entity* parent, s32 param_2, s32 param_3)) { Entity* entity; s32 rand; s32 radius; u32 angle; s32 tmp; entity = CreateLargeWaterTrace(parent); if (entity != NULL) { rand = Random(); radius = (rand % (param_3 - param_2 + 1)) + param_2; angle = rand >> 0x10 & 0xff; radius *= 0x100; tmp = FixedMul(gSineTable[angle], radius); tmp = FixedDiv(tmp, 0x100); entity->x.WORD += ((tmp << 0x10) >> 8); tmp = FixedMul(gSineTable[angle + 0x40], radius); tmp = FixedDiv(tmp, 0x100); entity->y.WORD -= ((tmp << 0x10) >> 8); } } END_NONMATCH void CreateSparkle(Entity* entity) { Entity* sparkle; u32 rand; s32 y; s32 x; sparkle = CreateObject(SPECIAL_FX, 0x26, 0); if (sparkle != NULL) { rand = Random(); x = rand & 0xf; if ((rand & 0x80) != 0) { x = -x; } y = rand >> 8 & 0xf; if (4 < y) { y = -y; } PositionRelative(entity, sparkle, x << 0x10, y << 0x10); SortEntityAbove(entity, sparkle); sparkle->spriteOffsetX = entity->spriteOffsetX; sparkle->spriteOffsetY = entity->spriteOffsetY; } } NONMATCH("asm/non_matching/objectUtils/sub_080A2BE4.inc", void sub_080A2BE4(Entity* this, u32 param_2)) { s16 oldValue; s32 diff; u16 newValue; if ((this->direction & 0x80) == 0) { switch (this->direction >> 3) { case 0: oldValue = this->y.HALF_U.HI; LinearMoveUpdate(this); newValue = this->y.HALF_U.HI; diff = ((oldValue - newValue) << 0x10) >> 0x10; if ((diff != 0) && (param_2 != 0)) { sub_080044AE(&gPlayerEntity, diff << 8, 0); } break; case 1: oldValue = this->x.HALF_U.HI; LinearMoveUpdate(this); diff = ((this->x.HALF_U.HI - oldValue) * 0x10000) >> 0x10; if ((diff != 0) && (param_2 != 0)) { sub_080044AE(&gPlayerEntity, diff << 8, 8); } break; case 2: oldValue = this->y.HALF_U.HI; LinearMoveUpdate(this); diff = ((this->y.HALF_U.HI - oldValue) * 0x10000) >> 0x10; if ((diff != 0) && (param_2 != 0)) { sub_080044AE(&gPlayerEntity, diff << 8, 0x10); } break; case 3: oldValue = this->x.HALF_U.HI; LinearMoveUpdate(this); newValue = this->x.HALF_U.HI; diff = ((oldValue - newValue) << 0x10) >> 0x10; if ((diff != 0) && (param_2 != 0)) { sub_080044AE(&gPlayerEntity, diff << 8, 0x18); } break; } } } END_NONMATCH void sub_080A2CC0(Entity* this, u16** param_2, u16* param_3) { u16* ptr = *param_2; if (ptr != NULL) { if (*ptr == 0xfe) { ptr = ptr + (u32)ptr[1] * -3; } if (*ptr == 0xff) { this->direction = 0xff; this->speed = 0; } else { this->direction = (u8)*ptr; this->speed = ptr[2]; } *param_3 = ptr[1]; *param_2 = ptr + 3; } } Entity* CreateSpeechBubbleExclamationMark(Entity* parent, s32 offsetX, s32 offsetY) { return CreateSpeechBubble(parent, 0, offsetX, offsetY); } Entity* CreateSpeechBubbleQuestionMark(Entity* parent, s32 offsetX, s32 offsetY) { return CreateSpeechBubble(parent, 1, offsetX, offsetY); } Entity* CreateSpeechBubbleSleep(Entity* parent, s32 offsetX, s32 offsetY) { return CreateSpeechBubble(parent, 2, offsetX, offsetY); } Entity* CreateSpeechBubble(Entity* parent, u32 type2, s32 xOffset, s32 yOffset) { Entity* obj = CreateObject(THOUGHT_BUBBLE, 0, type2); if (obj != NULL) { CopyPosition(parent, obj); SortEntityAbove(parent, obj); obj->parent = parent; obj->spriteOffsetX = xOffset; obj->spriteOffsetY = yOffset; SetDefaultPriority(obj, parent->updatePriority); } return obj; }