static void CheckGravityMovement(struct PlayerInfo *);
static void KillHeroUnlessProtected(int, int);
+void PlaySoundLevel(int, int, int);
+void PlaySoundLevelAction(int, int, int);
+void PlaySoundLevelElementAction(int, int, int, int);
+
static void MapGameButtons();
static void HandleGameButtons(struct GadgetInfo *);
static struct GadgetInfo *game_gadget[NUM_GAME_BUTTONS];
-static boolean is_loop_sound[NUM_SOUND_EFFECTS];
-static boolean is_loop_sound_initialized = FALSE;
-static int loop_sounds[] =
+#define SND_ACTION_UNKNOWN 0
+#define SND_ACTION_WAITING 1
+#define SND_ACTION_MOVING 2
+#define SND_ACTION_DIGGING 3
+#define SND_ACTION_COLLECTING 4
+#define SND_ACTION_PASSING 5
+#define SND_ACTION_IMPACT 6
+#define SND_ACTION_PUSHING 7
+#define SND_ACTION_ACTIVATING 8
+#define SND_ACTION_BURNING 9
+
+#define NUM_SND_ACTIONS 10
+
+static struct
+{
+ char *text;
+ int value;
+ boolean is_loop;
+} sound_action_properties[] =
{
- SND_BD_MAGIC_WALL_RUNNING,
- SND_BD_BUTTERFLY_MOVING,
- SND_BD_FIREFLY_MOVING,
- SND_SP_SNIKSNAK_MOVING,
- SND_SP_ELECTRON_MOVING,
- SND_DYNAMITE_BURNING,
- SND_BUG_MOVING,
- SND_SPACESHIP_MOVING,
- SND_YAMYAM_MOVING,
- SND_YAMYAM_WAITING,
- SND_ROBOT_WHEEL_RUNNING,
- SND_MAGIC_WALL_RUNNING,
- SND_BALLOON_MOVING,
- SND_MOLE_MOVING,
- SND_TIMEGATE_WHEEL_RUNNING,
- SND_CONVEYOR_BELT_RUNNING,
- SND_DYNABOMB_BURNING,
- SND_PACMAN_MOVING,
- SND_PENGUIN_MOVING,
- SND_PIG_MOVING,
- SND_DRAGON_MOVING,
- SND_DRAGON_BREATHING_FIRE
+ /* insert _all_ loop sound actions here */
+ { ".waiting", SND_ACTION_WAITING, TRUE },
+ { ".moving", SND_ACTION_MOVING, TRUE }, /* continuos moving */
+ { ".running", SND_ACTION_UNKNOWN, TRUE },
+ { ".burning", SND_ACTION_BURNING, TRUE },
+ { ".growing", SND_ACTION_UNKNOWN, TRUE },
+ { ".attacking", SND_ACTION_UNKNOWN, TRUE },
+
+ /* other (non-loop) sound actions are optional */
+ { ".stepping", SND_ACTION_MOVING, FALSE }, /* discrete moving */
+ { ".digging", SND_ACTION_DIGGING, FALSE },
+ { ".collecting", SND_ACTION_COLLECTING, FALSE },
+ { ".passing", SND_ACTION_PASSING, FALSE },
+ { ".impact", SND_ACTION_IMPACT, FALSE },
+ { ".pushing", SND_ACTION_PUSHING, FALSE },
+ { ".activating", SND_ACTION_ACTIVATING, FALSE },
+ { NULL, 0, 0 },
};
+static int element_action_sound[NUM_LEVEL_ELEMENTS][NUM_SND_ACTIONS];
+static boolean is_loop_sound[NUM_SOUND_EFFECTS];
#define IS_LOOP_SOUND(x) (is_loop_sound[x])
-
#ifdef DEBUG
#if 0
static unsigned int getStateCheckSum(int counter)
int2str(TimeLeft, 3), FS_SMALL, FC_YELLOW);
}
+void InitGameEngine()
+{
+ static int sound_effect_properties[NUM_SOUND_EFFECTS];
+ int i, j;
+
+#if 0
+ debug_print_timestamp(0, NULL);
+#endif
+
+ for (i=0; i<NUM_SND_ACTIONS; i++)
+ for (j=0; j<NUM_LEVEL_ELEMENTS; j++)
+ element_action_sound[j][i] = -1;
+
+ for (i=0; i<NUM_SOUND_EFFECTS; i++)
+ {
+ int len_effect_text = strlen(sound_effects[i].text);
+
+ sound_effect_properties[i] = SND_ACTION_UNKNOWN;
+ is_loop_sound[i] = FALSE;
+
+ /* determine all loop sounds and identify certain sound classes */
+
+ j = 0;
+ while (sound_action_properties[j].text)
+ {
+ int len_action_text = strlen(sound_action_properties[j].text);
+
+ if (len_action_text < len_effect_text &&
+ strcmp(&sound_effects[i].text[len_effect_text - len_action_text],
+ sound_action_properties[j].text) == 0)
+ {
+ sound_effect_properties[i] = sound_action_properties[j].value;
+
+ if (sound_action_properties[j].is_loop)
+ is_loop_sound[i] = TRUE;
+ }
+
+ j++;
+ }
+
+ /* associate elements and some selected sound actions */
+
+ for (j=0; j<NUM_LEVEL_ELEMENTS; j++)
+ {
+ if (element_info[j].sound_class_name)
+ {
+ int len_class_text = strlen(element_info[j].sound_class_name);
+
+ if (len_class_text + 1 < len_effect_text &&
+ strncmp(sound_effects[i].text,
+ element_info[j].sound_class_name, len_class_text) == 0 &&
+ sound_effects[i].text[len_class_text] == '.')
+ {
+ int sound_action_value = sound_effect_properties[i];
+
+ element_action_sound[j][sound_action_value] = i;
+ }
+ }
+ }
+ }
+
+#if 0
+ debug_print_timestamp(0, "InitGameEngine");
+#endif
+
+#if 0
+ /* TEST ONLY */
+ {
+ int element = EL_ERDREICH;
+ int sound_action = SND_ACTION_DIGGING;
+ int j = 0;
+
+ while (sound_action_properties[j].text)
+ {
+ if (sound_action_properties[j].value == sound_action)
+ printf("element %d, sound action '%s' == %d\n",
+ element, sound_action_properties[j].text,
+ element_action_sound[element][sound_action]);
+ j++;
+ }
+ }
+#endif
+}
+
void InitGame()
{
int i, j, x, y;
}
}
- /* initialize sound effect properties */
- if (!is_loop_sound_initialized)
- {
- int i;
-
- for (i=0; i<NUM_SOUND_EFFECTS; i++)
- is_loop_sound[i] = FALSE;
-
- for (i=0; i<SIZEOF_ARRAY_INT(loop_sounds); i++)
- is_loop_sound[loop_sounds[i]] = TRUE;
-
- is_loop_sound_initialized = TRUE;
- }
-
game.version = (tape.playing ? tape.game_version : level.game_version);
game.emulation = (emulate_bd ? EMU_BOULDERDASH :
emulate_sb ? EMU_SOKOBAN :
MovDelay[x][y]--;
if (MovDelay[x][y])
{
-#if 0
- if (!(MovDelay[x][y] % 12))
-#else
if (!(MovDelay[x][y] % 6))
-#endif
- {
- if (Feld[x][y] == EL_DYNAMITE_ACTIVE)
- PlaySoundLevel(x, y, SND_DYNAMITE_BURNING);
- else
- PlaySoundLevel(x, y, SND_DYNABOMB_BURNING);
- }
+ PlaySoundLevelAction(x, y, SND_ACTION_BURNING);
if (IS_ACTIVE_BOMB(Feld[x][y]))
{
/* play sound of object that hits the ground */
if (lastline || object_hit)
- {
- int sound;
-
- switch (element)
- {
- case EL_EDELSTEIN_BD:
- sound = SND_BD_DIAMOND_IMPACT;
- break;
- case EL_EDELSTEIN:
- case EL_EDELSTEIN_GELB:
- case EL_EDELSTEIN_ROT:
- case EL_EDELSTEIN_LILA:
- sound = SND_EMERALD_IMPACT;
- break;
- case EL_DIAMANT:
- sound = SND_DIAMOND_IMPACT;
- break;
- case EL_PEARL:
- sound = SND_PEARL_IMPACT;
- break;
- case EL_CRYSTAL:
- sound = SND_CRYSTAL_IMPACT;
- break;
- case EL_SP_INFOTRON:
- sound = SND_SP_INFOTRON_IMPACT;
- break;
- case EL_KOKOSNUSS:
- sound = SND_NUT_IMPACT;
- break;
- case EL_BD_ROCK:
- sound = SND_BD_ROCK_IMPACT;
- break;
- case EL_FELSBROCKEN:
- sound = SND_ROCK_IMPACT;
- break;
- case EL_SP_ZONK:
- sound = SND_SP_ZONK_IMPACT;
- break;
- case EL_ZEIT_VOLL:
- sound = SND_TIME_ORB_FULL_IMPACT;
- break;
- case EL_ZEIT_LEER:
- sound = SND_TIME_ORB_EMPTY_IMPACT;
- break;
- case EL_SPRING:
- sound = SND_SPRING_IMPACT;
- break;
- default:
- sound = -1;
- break;
- }
-
- if (sound >= 0)
- PlaySoundLevel(x, y, sound);
- }
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_IMPACT);
}
void TurnRound(int x, int y)
if (element!=EL_MAMPFER && element!=EL_MAMPFER2 && element!=EL_PACMAN)
{
TurnRound(x, y);
+
if (MovDelay[x][y] && (element == EL_KAEFER ||
element == EL_FLIEGER ||
element == EL_SP_SNIKSNAK ||
}
}
- if (MovDelay[x][y])
+ if (MovDelay[x][y]) /* element still has to wait some time */
+ {
+ PlaySoundLevelAction(x, y, SND_ACTION_WAITING);
+
return;
+ }
}
- if (element == EL_KAEFER)
- PlaySoundLevel(x, y, SND_BUG_MOVING);
- else if (element == EL_FLIEGER)
- PlaySoundLevel(x, y, SND_SPACESHIP_MOVING);
- else if (element == EL_BUTTERFLY)
- PlaySoundLevel(x, y, SND_BD_BUTTERFLY_MOVING);
- else if (element == EL_FIREFLY)
- PlaySoundLevel(x, y, SND_BD_FIREFLY_MOVING);
- else if (element == EL_SP_SNIKSNAK)
- PlaySoundLevel(x, y, SND_SP_SNIKSNAK_MOVING);
- else if (element == EL_SP_ELECTRON)
- PlaySoundLevel(x, y, SND_SP_ELECTRON_MOVING);
- else if (element == EL_MAMPFER)
- PlaySoundLevel(x, y, SND_YAMYAM_MOVING);
- else if (element == EL_MAMPFER2)
- PlaySoundLevel(x, y, SND_DARK_YAMYAM_MOVING);
- else if (element == EL_BALLOON)
- PlaySoundLevel(x, y, SND_BALLOON_MOVING);
- else if (element == EL_SPRING_MOVING)
- PlaySoundLevel(x, y, SND_SPRING_MOVING);
- else if (element == EL_MOLE)
- PlaySoundLevel(x, y, SND_MOLE_MOVING);
- else if (element == EL_SONDE)
- PlaySoundLevel(x, y, SND_SATELLITE_MOVING);
- else if (element == EL_PACMAN)
- PlaySoundLevel(x, y, SND_PACMAN_MOVING);
- else if (element == EL_PINGUIN)
- PlaySoundLevel(x, y, SND_PENGUIN_MOVING);
- else if (element == EL_SCHWEIN)
- PlaySoundLevel(x, y, SND_PIG_MOVING);
- else if (element == EL_DRACHE)
- PlaySoundLevel(x, y, SND_DRAGON_MOVING);
-
/* now make next step */
Moving2Blocked(x, y, &newx, &newy); /* get next screen position */
else
DrawLevelField(x, y);
- PlaySoundLevel(x, y, SND_DRAGON_BREATHING_FIRE);
+ PlaySoundLevel(x, y, SND_DRAGON_ATTACKING);
MovDelay[x][y] = 50;
Feld[newx][newy] = EL_BURNING;
if (DONT_TOUCH(element))
TestIfBadThingTouchesHero(x, y);
+ PlaySoundLevelAction(x, y, SND_ACTION_WAITING);
+
return;
}
- if (element == EL_ROBOT && IN_SCR_FIELD(x, y))
- PlaySoundLevel(x, y, SND_ROBOT_MOVING);
-
InitMovingField(x, y, MovDir[x][y]);
+
+ PlaySoundLevelAction(x, y, SND_ACTION_MOVING);
}
if (MovDir[x][y])
if (DelayReached(&sound_delay, sound_delay_value))
{
if (Store[x][y] == EL_AMOEBE_BD)
- PlaySoundLevel(x, y, SND_BD_AMOEBA_GROWING);
+ PlaySoundLevel(x, y, SND_BD_AMOEBA_CREATING);
else
- PlaySoundLevel(x, y, SND_AMOEBA_GROWING);
+ PlaySoundLevel(x, y, SND_AMOEBA_CREATING);
sound_delay_value = 30;
}
}
}
if (changed)
- PlaySoundLevel(ax, ay, element == EL_LIFE ? SND_GAMEOFLIFE_GROWING :
- SND_BIOMAZE_GROWING);
+ PlaySoundLevel(ax, ay, element == EL_LIFE ? SND_GAMEOFLIFE_CREATING :
+ SND_BIOMAZE_CREATING);
}
void RobotWheel(int x, int y)
case EL_SP_BASE:
case EL_SP_BUG:
RemoveField(x, y);
-
- if (element == EL_LEERRAUM)
- PlaySoundLevel(x, y, SND_EMPTY_SPACE_DIGGING);
- else if (element == EL_ERDREICH)
- PlaySoundLevel(x, y, SND_SAND_DIGGING);
- else if (element == EL_SAND_INVISIBLE)
- PlaySoundLevel(x, y, SND_SAND_INVISIBLE_DIGGING);
- else if (element == EL_TRAP_INACTIVE)
- PlaySoundLevel(x, y, SND_TRAP_INACTIVE_DIGGING);
- else if (element == EL_SP_BASE)
- PlaySoundLevel(x, y, SND_SP_BASE_DIGGING);
- else if (element == EL_SP_BUG)
- PlaySoundLevel(x, y, SND_SP_BUGGY_BASE_DIGGING);
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_DIGGING);
break;
case EL_EDELSTEIN:
DrawText(DX_EMERALDS, DY_EMERALDS,
int2str(local_player->gems_still_needed, 3),
FS_SMALL, FC_YELLOW);
-
- if (element == EL_EDELSTEIN_BD)
- PlaySoundLevel(x, y, SND_BD_DIAMOND_COLLECTING);
- else if (element == EL_DIAMANT)
- PlaySoundLevel(x, y, SND_DIAMOND_COLLECTING);
- else if (element == EL_SP_INFOTRON)
- PlaySoundLevel(x, y, SND_SP_INFOTRON_COLLECTING);
- else if (element == EL_PEARL)
- PlaySoundLevel(x, y, SND_PEARL_COLLECTING);
- else if (element == EL_CRYSTAL)
- PlaySoundLevel(x, y, SND_CRYSTAL_COLLECTING);
- else /* EL_EDELSTEIN style element */
- PlaySoundLevel(x, y, SND_EMERALD_COLLECTING);
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_COLLECTING);
break;
case EL_SPEED_PILL:
DrawText(DX_DYNAMITE, DY_DYNAMITE,
int2str(local_player->dynamite, 3),
FS_SMALL, FC_YELLOW);
- if (element == EL_SP_DISK_RED)
- PlaySoundLevel(x, y, SND_SP_DISK_RED_COLLECTING);
- else
- PlaySoundLevel(x, y, SND_DYNAMITE_COLLECTING);
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_COLLECTING);
break;
case EL_DYNABOMB_NR:
element == EL_BALLOON_SEND_UP ? MV_UP :
element == EL_BALLOON_SEND_DOWN ? MV_DOWN :
MV_NO_MOVING);
+ PlaySoundLevel(x, y, SND_BALLOON_SWITCH_ACTIVATING);
return MF_ACTION;
break;
player->push_delay_value = (element == EL_SPRING ? 0 : 2 + RND(8));
DrawLevelField(x+dx, y+dy);
- if (element == EL_FELSBROCKEN)
- PlaySoundLevel(x+dx, y+dy, SND_ROCK_PUSHING);
- else if (element == EL_BD_ROCK)
- PlaySoundLevel(x+dx, y+dy, SND_BD_ROCK_PUSHING);
- else if (element == EL_BOMBE)
- PlaySoundLevel(x+dx, y+dy, SND_BOMB_PUSHING);
- else if (element == EL_DX_SUPABOMB)
- PlaySoundLevel(x+dx, y+dy, SND_DX_BOMB_PUSHING);
- else if (element == EL_KOKOSNUSS)
- PlaySoundLevel(x+dx, y+dy, SND_NUT_PUSHING);
- else if (element == EL_ZEIT_LEER)
- PlaySoundLevel(x+dx, y+dy, SND_TIME_ORB_EMPTY_PUSHING);
- else if (element == EL_SP_ZONK)
- PlaySoundLevel(x+dx, y+dy, SND_SP_ZONK_PUSHING);
- else if (element == EL_SP_DISK_ORANGE)
- PlaySoundLevel(x+dx, y+dy, SND_SP_DISK_ORANGE_PUSHING);
- else if (element == EL_SPRING)
- PlaySoundLevel(x+dx, y+dy, SND_SPRING_PUSHING);
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_PUSHING);
break;
case EL_PFORTE1:
DOUBLE_PLAYER_SPEED(player);
PlaySoundLevel(x, y, SND_GATE_PASSING);
-
break;
case EL_EM_GATE_1X:
DOUBLE_PLAYER_SPEED(player);
PlaySoundLevel(x, y, SND_GATE_PASSING);
-
break;
case EL_SWITCHGATE_OPEN:
player->programmed_action = move_direction;
DOUBLE_PLAYER_SPEED(player);
- if (element == EL_SWITCHGATE_OPEN)
- PlaySoundLevel(x, y, SND_SWITCHGATE_PASSING);
- else
- PlaySoundLevel(x, y, SND_TIMEGATE_PASSING);
-
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_PASSING);
break;
case EL_SP_PORT1_LEFT:
if (!(tube_enter_directions[i][1] & move_direction))
return MF_NO_ACTION; /* tube has no opening in this direction */
+
+ PlaySoundLevel(x, y, SND_TUBE_PASSING);
}
break;
{
RemoveField(x, y);
Feld[x+dx][y+dy] = element;
+ PlaySoundLevelElementAction(x, y, element, SND_ACTION_PUSHING);
}
player->push_delay_value = (element == EL_BALLOON ? 0 : 2);
DrawLevelField(x, y);
DrawLevelField(x+dx, y+dy);
- if (element == EL_SONDE)
- PlaySoundLevel(x+dx, y+dy, SND_SATELLITE_PUSHING);
- else if (element == EL_SP_DISK_YELLOW)
- PlaySoundLevel(x+dx, y+dy, SND_SP_DISK_YELLOW_PUSHING);
- else if (element == EL_BALLOON)
- PlaySoundLevel(x+dx, y+dy, SND_BALLOON_PUSHING);
if (IS_SB_ELEMENT(element) &&
local_player->sokobanfields_still_needed == 0 &&
PlaySoundExt(nr, volume, stereo_position, type);
}
+void PlaySoundLevelAction(int x, int y, int sound_action)
+{
+ PlaySoundLevelElementAction(x, y, Feld[x][y], sound_action);
+}
+
+void PlaySoundLevelElementAction(int x, int y, int element, int sound_action)
+{
+ int sound_effect = element_action_sound[element][sound_action];
+
+ if (sound_effect != -1)
+ PlaySoundLevel(x, y, sound_effect);
+}
+
void RaiseScore(int value)
{
local_player->score += value;