Logo Search packages:      
Sourcecode: parsec47 version File versions  Download package

EnemyType.d

/*
 * $Id: EnemyType.d,v 1.4 2004/01/01 11:26:41 kenta Exp $
 *
 * Copyright 2003 Kenta Cho. All rights reserved.
 */
module abagames.p47.EnemyType;

private:
import std.math;
import bulletml;
import abagames.util.Vector;
import abagames.util.Rand;
import abagames.p47.BarrageManager;
import abagames.p47.MorphBullet;

/**
 * Barrage pattern.
 */
00019 public class Barrage {
 public:
  BulletMLParser *parser;
  BulletMLParser *morphParser[MorphBullet.MORPH_MAX];
  int morphNum, morphCnt;
  float rank, speedRank, morphRank;
  int shape, color;
  float bulletSize;
  float xReverse;
}

/**
 * Enemys' wing with batteries.
 */
00033 public class BatteryType {
 public:
  static const int WING_SHAPE_POINT_NUM = 3;
  static const int WING_BATTERY_MAX = 3;
  static const int BARRAGE_PATTERN_MAX = 8;
  Vector wingShapePos[WING_SHAPE_POINT_NUM];
  Vector collisionPos, collisionSize;
  Vector batteryPos[WING_BATTERY_MAX];
  int batteryNum;
  float r, g, b;
  Barrage barrage[BARRAGE_PATTERN_MAX];
  bool xReverseAlternate;
  int shield;

  public this() {
    for (int i = 0; i < BARRAGE_PATTERN_MAX; i++) {
      barrage[i] = new Barrage;
    }
    for (int i = 0; i < WING_SHAPE_POINT_NUM; i++) {
      wingShapePos[i] = new Vector;
    }
    collisionPos = new Vector;
    collisionSize = new Vector;
    for (int i = 0; i < WING_BATTERY_MAX; i++) {
      batteryPos[i] = new Vector;
    }
  }
}

/**
 * Enemys' specifications.
 */
00065 public class EnemyType {
 public:
  static const int BARRAGE_PATTERN_MAX = BatteryType.BARRAGE_PATTERN_MAX;
  static const int BODY_SHAPE_POINT_NUM = 4;
  static const int BATTERY_MAX = 4;
  static const int ENEMY_TYPE_MAX = 32;
  // Whether each type of the enemy is exist or not.
  static bool isExist[ENEMY_TYPE_MAX];
  Barrage barrage[BARRAGE_PATTERN_MAX];
  Vector bodyShapePos[BODY_SHAPE_POINT_NUM];
  Vector collisionSize;
  bool wingCollision;
  float r, g, b;
  float retroSize;
  BatteryType batteryType[BATTERY_MAX];
  int batteryNum;
  int shield;
  int fireInterval, firePeriod, barragePatternNum;
  int id;
  static enum {
    SMALL, MIDDLE, LARGE, MIDDLEBOSS, LARGEBOSS,
  }
  int type;
  // To avoid the forward reference.
  static enum {
    ROLL, LOCK,
  }
 private:
  static Rand rand;
  static BarrageManager barrageManager;
  static int idCnt;

  public static void init(BarrageManager manager) {
    rand = new Rand;
    barrageManager = manager;
    idCnt = 0;
  }

  public static void clearIsExistList() {
    for (int i = 0; i < idCnt; i++)
      isExist[i] = false;
  }

  public this() {
    for (int i = 0; i < BODY_SHAPE_POINT_NUM; i++) {
      bodyShapePos[i] = new Vector;
    }
    collisionSize = new Vector;
    for (int i = 0; i < BARRAGE_PATTERN_MAX; i++) {
      barrage[i] = new Barrage;
    }
    for (int i = 0; i < BATTERY_MAX; i++) {
      batteryType[i] = new BatteryType;
    }
    assert(idCnt < ENEMY_TYPE_MAX);
    id = idCnt;
    idCnt++;
  }

  // To avoid using the same morph pattern.
  private static bool usedMorphParser[BarrageManager.BARRAGE_MAX];
  
  private void setBarrageType(Barrage br, int btn, int mode) {
    br.parser = barrageManager.parser
      [btn]
      [rand.nextInt(barrageManager.parserNum[btn])];
    for (int i = 0; i< BarrageManager.BARRAGE_MAX; i++)
      usedMorphParser[i] = false;
    int mpn;
    if (mode == ROLL)
      mpn = barrageManager.parserNum[BarrageManager.MORPH];
    else
      mpn = barrageManager.parserNum[BarrageManager.MORPH_LOCK];
    for (int i = 0; i< br.morphParser.length; i++) {
      int mi = rand.nextInt(mpn);
      for (int j = 0; j < mpn; j++) {
      if (!usedMorphParser[mi])
        break;
      mi++;
      if (mi >= mpn) mi = 0;
      }
      if (mode == ROLL)
      br.morphParser[i] = barrageManager.parser[BarrageManager.MORPH][mi];
      else
      br.morphParser[i] = barrageManager.parser[BarrageManager.MORPH_LOCK][mi];
      usedMorphParser[mi] = true;
    }
    br.morphNum = br.morphParser.length;
  }

  // Barrage intense.
  enum {
    NORMAL, WEAK, VERYWEAK, MORPHWEAK
  }

  private void setBarrageRank(Barrage br, float rank, int intense, int mode) {
    if (rank <= 0) {
      br.rank = 0;
      return;
    }
    br.rank = sqrt(rank) / (8 - rand.nextInt(3));
    if (br.rank > 0.8)
      br.rank = rand.nextFloat(0.2) + 0.8;
    rank /= (br.rank + 2);
    if (intense == WEAK)
      br.rank /= 2;
    if (mode == ROLL)
      br.speedRank = sqrt(rank) * (rand.nextFloat(0.2) + 1);
    else
      br.speedRank = sqrt(rank * 0.66) * (rand.nextFloat(0.2) + 0.8);
    if (br.speedRank < 1)
      br.speedRank = 1;
    if (br.speedRank > 2)
      br.speedRank = sqrt(br.speedRank) + 0.27;
    br.morphRank = rank / br.speedRank;
    br.morphCnt = 0;
    while (br.morphRank > 1) {
      br.morphCnt++;
      br.morphRank /= 3;
    }
    if (intense == VERYWEAK) {
      br.morphRank /= 2;
      br.morphCnt /= 1.7f;
    } else if (intense == MORPHWEAK) {
      br.morphRank /= 2;
    } else if (intense == WEAK) {
      br.morphRank /= 1.5f;
    }
  }

  private void setBarrageRankSlow(Barrage br, float rank, int intense, int mode, float slow) {
    setBarrageRank(br, rank, intense, mode);
    br.speedRank *= slow;
  }

  public static const int BULLET_SHAPE_NUM = 7;
  public static const int BULLET_COLOR_NUM = 4;

  private void setBarrageShape(Barrage br, float size) {
    // To avoid the forward reference.
    //br.shape = rand.nextInt(BulletActor.BULLET_SHAPE_NUM);
    //br.color = rand.nextInt(BulletActor.BULLET_COLOR_NUM);
    br.shape = rand.nextInt(BULLET_SHAPE_NUM);
    br.color = rand.nextInt(BULLET_COLOR_NUM);
    br.bulletSize = (1.0 + rand.nextSignedFloat(0.1)) * size;
  }

  private float er, eg, eb;
  private int ect;

  private void setEnemyColorType() {
    ect = rand.nextInt(3);
  }

  private void createEnemyColor() {
    switch (ect) {
    case 0:
      er = 1;
      eg = rand.nextFloat(0.7) + 0.3;
      eb = rand.nextFloat(0.7) + 0.3;
      break;
    case 1:
      er = rand.nextFloat(0.7) + 0.3;
      eg = 1;
      eb = rand.nextFloat(0.7) + 0.3;
      break;
    case 2:
      er = rand.nextFloat(0.7) + 0.3;
      eg = rand.nextFloat(0.7) + 0.3;
      eb = 1;
      break;
    }
  }

  private static const float[][] enemySize = 
    [
     [0.3, 0.3, 0.3, 0.1,  0.1,  1.0, 0.4, 0.6, 0.9],
     [0.4, 0.2, 0.4, 0.1,  0.15,  2.2, 0.2, 1.6, 1.0],
     [0.6, 0.3, 0.5, 0.1,  0.2,  3.0, 0.3, 1.4, 1.2],
     [0.9, 0.3, 0.7, 0.2,  0.25,  5.0, 0.6, 3.0, 1.5],
     [1.2, 0.2, 0.9, 0.1,  0.3,  7.0, 0.8, 4.5, 1.5],
     ];
  private enum {
    SMALL, MIDDLE, LARGE, MIDDLEBOSS, LARGEBOSS,
  }

  // Set the shepe of the BatteryType.
  private void setEnemyShapeAndWings(int size) {
    createEnemyColor();
    r = er;
    g = eg;
    b = eb;
    float x1 = enemySize[size][0] + rand.nextSignedFloat(enemySize[size][1]);
    float y1 = enemySize[size][2] + rand.nextSignedFloat(enemySize[size][3]);
    float x2 = enemySize[size][0] + rand.nextSignedFloat(enemySize[size][1]);
    float y2 = enemySize[size][2] + rand.nextSignedFloat(enemySize[size][3]);
    bodyShapePos[0].x = -x1;
    bodyShapePos[0].y = y1;
    bodyShapePos[1].x = x1;
    bodyShapePos[1].y = y1;
    bodyShapePos[2].x = x2;
    bodyShapePos[2].y = -y2;
    bodyShapePos[3].x = -x2;
    bodyShapePos[3].y = -y2;
    retroSize = enemySize[size][4];
    switch (size) {
    case SMALL:
    case MIDDLE:
    case MIDDLEBOSS:
      batteryNum = 2;
      break;
    case LARGE:
    case LARGEBOSS:
      batteryNum = 4;
      break;
    }
    float px, py, mpx, mpy;
    int bsl;
    if (x1 > x2)
      collisionSize.x = x1;
    else
      collisionSize.x = x2;
    if (y1 > y2)
      collisionSize.y = y1;
    else
      collisionSize.y = y2;
    for (int i = 0; i < batteryNum; i++) {
      BatteryType bt = batteryType[i];
      int wrl = 1;
      if (i % 2 == 0) {
      px = enemySize[size][5] + rand.nextFloat(enemySize[size][6]);
      if (batteryNum <= 2) {
        py = rand.nextSignedFloat(enemySize[size][7]);
      } else {
        if (i < 2) {
          py = rand.nextFloat(enemySize[size][7] / 2) + enemySize[size][7] / 2;
        } else {
          py = -rand.nextFloat(enemySize[size][7] / 2) - enemySize[size][7] / 2;
        }
      }
      float md;
      if (rand.nextInt(2) == 0)
        md = rand.nextFloat(std.math.PI / 2) - std.math.PI / 4;
      else
        md = rand.nextFloat(std.math.PI / 2) + std.math.PI / 4 * 3;
      mpx = px / 2 + sin(md) * (enemySize[size][8] / 2 + rand.nextFloat(enemySize[size][8]/2));
      mpy = py / 2 + cos(md) * (enemySize[size][8] / 2 + rand.nextFloat(enemySize[size][8]/2));
      switch (size) {
      case SMALL:
      case MIDDLE:
      case LARGE:
        bsl = 1;
        break;
      case MIDDLEBOSS:
        bsl = 150 + rand.nextInt(30);
        break;
      case LARGEBOSS:
        bsl = 200 + rand.nextInt(50);
        break;
      }
      createEnemyColor();
      wrl = -1;
      if (!wingCollision) {
        if (px > collisionSize.x)
          collisionSize.x = px;
        float cpy = fabs(py);
        if (cpy > collisionSize.y)
          collisionSize.y = cpy;
        cpy = fabs(mpy);
        if (cpy > collisionSize.y)
          collisionSize.y = cpy;
      }
      }
      switch (wrl) {
      case 1:
      bt.wingShapePos[0].x = px / 4 * wrl;
      bt.wingShapePos[0].y = py / 4;
      bt.wingShapePos[1].x = px * wrl;
      bt.wingShapePos[1].y = py;
      bt.wingShapePos[2].x = mpx * wrl;
      bt.wingShapePos[2].y = mpy;
      break;
      case -1:
      bt.wingShapePos[0].x = px / 4 * wrl;
      bt.wingShapePos[0].y = py / 4;
      bt.wingShapePos[1].x = px * wrl;
      bt.wingShapePos[1].y = py;
      bt.wingShapePos[2].x = mpx * wrl;
      bt.wingShapePos[2].y = mpy;
      break;
      }
      bt.collisionPos.x = (px + px / 4) / 2 * wrl;
      bt.collisionPos.y = (py + mpy + py / 4) / 3;
      bt.collisionSize.x = px / 4 * 3 / 2;
      float sy1 = fabs(py - mpy) / 2;
      float sy2 = fabs(py - py / 4) / 2;
      if (sy1 > sy2)
      bt.collisionSize.y = sy1;
      else
      bt.collisionSize.y = sy2;
      bt.r = er;
      bt.g = eg;
      bt.b = eb;
      bt.shield =bsl;
    }
  }

  // Set the barrage of the BatteryType.
  private void setBattery(float rank, int n, int barrageType, int barrageIntense, 
                    int idx, int ptnIdx, float slow, int mode) {
    BatteryType bt = batteryType[idx];
    BatteryType bt2 = batteryType[idx + 1];
    Barrage br = bt.barrage[ptnIdx];
    Barrage br2 = bt2.barrage[ptnIdx];
    setBarrageType(br, barrageType, mode);
    setBarrageRankSlow(br, rank / n, barrageIntense, mode, slow);
    setBarrageShape(br, 0.8);
    br.xReverse = rand.nextInt(2) * 2 - 1;
    br2.parser = br.parser;
    for (int i = 0; i < MorphBullet.MORPH_MAX; i++) {
      br2.morphParser[i] = br.morphParser[i];
    }
    br2.morphNum = br.morphNum;
    br2.morphCnt = br.morphCnt;
    br2.rank = br.rank;
    br2.speedRank = br.speedRank;
    br2.morphRank = br.morphRank;
    br2.shape = br.shape;
    br2.color = br.color;
    br2.bulletSize = br.bulletSize;
    br2.xReverse = -br.xReverse;
    if (rand.nextInt(4) == 0)
      bt.xReverseAlternate = bt2.xReverseAlternate = true;
    else
      bt.xReverseAlternate = bt2.xReverseAlternate = false;
    float px = bt.wingShapePos[1].x, py = bt.wingShapePos[1].y;
    float mpx = bt.wingShapePos[2].x, mpy = bt.wingShapePos[2].y;
    for (int i = 0; i < n; i++) {
      bt.batteryPos[i].x = px;
      bt.batteryPos[i].y = py;
      bt2.batteryPos[i].x = -px;
      bt2.batteryPos[i].y = py;
      px += (mpx - px) / (n - 1);
      py += (mpy - py) / (n - 1);
    }
    bt.batteryNum = bt2.batteryNum = n;
  }

  public void setSmallEnemyType(float rank, int mode) {
    type = SMALL;
    barragePatternNum = 1;
    wingCollision = false;
    setEnemyColorType();
    Barrage br = barrage[0];
    if (mode == ROLL)
      setBarrageType(br, BarrageManager.SMALL, mode);
    else
      setBarrageType(br, BarrageManager.SMALL_LOCK, mode);
    setBarrageRank(br, rank, VERYWEAK, mode);
    setBarrageShape(br, 0.7);
    br.xReverse = rand.nextInt(2) * 2 - 1;
    setEnemyShapeAndWings(SMALL);
    setBattery(0, 0, 0, NORMAL, 0, 0, 1, mode);
    shield = 1;
    fireInterval = 99999;
    firePeriod = 150 + rand.nextInt(40);
    if (rank < 10)
      firePeriod /= (2 - rank * 0.1);
  }

  public void setMiddleEnemyType(float rank, int mode) {
    type = MIDDLE;
    barragePatternNum = 1;
    wingCollision = false;
    setEnemyColorType();
    Barrage br = barrage[0];
    setBarrageType(br, BarrageManager.MIDDLE, mode);
    float cr, sr;
    if (mode == ROLL) {
      switch (rand.nextInt(6)) {
      case 0:
      case 1:
      cr = rank / 3 * 2;
      sr = 0;
      break;
      case 2:
      cr = rank / 4;
      sr = rank / 4;
      break;
      case 3:
      case 4:
      case 5:
      cr = 0;
      sr = rank / 2;
      break;
      }
    } else {
      switch (rand.nextInt(6)) {
      case 0:
      case 1:
      cr = rank / 5;
      sr = rank / 4;
      break;
      case 2:
      case 3:
      case 4:
      case 5:
      cr = 0;
      sr = rank / 2;
      break;
      }
    }
    setBarrageRank(br, cr, MORPHWEAK, mode);
    setBarrageShape(br, 0.75);
    br.xReverse = rand.nextInt(2) * 2 - 1;
    setEnemyShapeAndWings(MIDDLE);
    if (mode == ROLL) {
      shield = 40 + rand.nextInt(10);
      setBattery(sr, 1, BarrageManager.MIDDLESUB, NORMAL, 0, 0, 1, mode);
      fireInterval = 100 + rand.nextInt(60);
      firePeriod = fireInterval / (1.8 + rand.nextFloat(0.7));
    } else {
      shield = 30 + rand.nextInt(8);
      setBattery(sr, 1, BarrageManager.MIDDLESUB_LOCK, NORMAL, 0, 0, 1, mode);
      fireInterval = 72 + rand.nextInt(30);
      firePeriod = fireInterval / (1.2 + rand.nextFloat(0.2));
    }
    if (rank < 10)
      firePeriod /= (2 - rank * 0.1);
  }

  public void setLargeEnemyType(float rank, int mode) {
    type = LARGE;
    barragePatternNum = 1;
    wingCollision = false;
    setEnemyColorType();
    Barrage br = barrage[0];
    setBarrageType(br, BarrageManager.LARGE, mode);
    float cr, sr1, sr2;
    if (mode == ROLL) {
      switch (rand.nextInt(9)) {
      case 0:
      case 1:
      case 2:
      case 3:
      cr = rank;
      sr1 = sr2 = 0;
      break;
      case 4:
      cr = rank / 3 * 2;
      sr1 = rank / 3 * 2;
      sr2 = 0;
      break;
      case 5:
      cr = rank / 3 * 2;
      sr1 = 0;
      sr2 = rank / 3 * 2;
      break;
      case 6:
      case 7:
      case 8:
      cr = 0;
      sr1 = rank / 3 * 2;
      sr2 = rank / 3 * 2;
      break;
      }
    } else {
      switch (rand.nextInt(9)) {
      case 0:
      cr = rank / 4 * 3;
      sr1 = sr2 = 0;
      break;
      case 1:
      case 2:
      cr = rank / 4 * 2;
      sr1 = rank / 3 * 2;
      sr2 = 0;
      break;
      case 3:
      case 4:
      cr = rank / 4 * 2;
      sr1 = 0;
      sr2 = rank / 3 * 2;
      break;
      case 5:
      case 6:
      case 7:
      case 8:
      cr = 0;
      sr1 = rank / 3 * 2;
      sr2 = rank / 3 * 2;
      break;
      }
    }
    setBarrageRank(br, cr, WEAK, mode);
    setBarrageShape(br, 0.8);
    br.xReverse = rand.nextInt(2) * 2 - 1;
    setEnemyShapeAndWings(LARGE);
    if (mode == ROLL) {
      shield = 60 + rand.nextInt(10);
      setBattery(sr1, 1, BarrageManager.MIDDLESUB, NORMAL, 0, 0, 1, mode);
      setBattery(sr2, 1, BarrageManager.MIDDLESUB, NORMAL, 2, 0, 1, mode);
      fireInterval = 150 + rand.nextInt(60);
      firePeriod = fireInterval / (1.3 + rand.nextFloat(0.8));
    } else {
      shield = 45 + rand.nextInt(8);
      setBattery(sr1, 1, BarrageManager.MIDDLESUB_LOCK, NORMAL, 0, 0, 1, mode);
      setBattery(sr2, 1, BarrageManager.MIDDLESUB_LOCK, NORMAL, 2, 0, 1, mode);
      fireInterval = 100 + rand.nextInt(50);
      firePeriod = fireInterval / (1.2 + rand.nextFloat(0.2));
    }
    if (rank < 10)
      firePeriod /= (2 - rank * 0.1);
  }

  public void setMiddleBossEnemyType(float rank, int mode) {
    type = MIDDLEBOSS;
    barragePatternNum = 2 + rand.nextInt(2);
    wingCollision = true;
    setEnemyColorType();
    int bn = 1 + rand.nextInt(2);
    for (int i = 0; i < barragePatternNum; i++) {
      Barrage br = barrage[i];
      setBarrageType(br, BarrageManager.LARGE, mode);
      float cr, sr;
      switch (rand.nextInt(3)) {
      case 0:
      cr = rank;
      sr = 0;
      break;
      case 1:
      cr = rank / 3;
      sr = rank / 3;
      break;
      case 2:
      cr = 0;
      sr = rank;
      break;
      }
      setBarrageRankSlow(br, cr, NORMAL, mode, 0.9);
      setBarrageShape(br, 0.9);
      br.xReverse = rand.nextInt(2) * 2 - 1;
      setEnemyShapeAndWings(MIDDLEBOSS);
      setBattery(sr, bn, BarrageManager.MIDDLE, WEAK, 0, i, 0.9, mode);
    }
    shield = 300 + rand.nextInt(50);
    fireInterval = 200 + rand.nextInt(40);
    firePeriod = fireInterval / (1.2 + rand.nextFloat(0.4));
    if (rank < 10)
      firePeriod /= (2 - rank * 0.1);
  }

  public void setLargeBossEnemyType(float rank, int mode) {
    type = LARGEBOSS;
    barragePatternNum = 2 + rand.nextInt(3);
    wingCollision = true;
    setEnemyColorType();
    int bn1 = 1 + rand.nextInt(3);
    int bn2 = 1 + rand.nextInt(3);
    for (int i = 0; i < barragePatternNum; i++) {
      Barrage br = barrage[i];
      setBarrageType(br, BarrageManager.LARGE, mode);
      float cr, sr1, sr2;
      switch (rand.nextInt(3)) {
      case 0:
      cr = rank;
      sr1 = sr2 = 0;
      break;
      case 1:
      cr = rank / 3;
      sr1 = rank / 3;
      sr2 = 0;
      break;
      case 2:
      cr = rank / 3;
      sr1 = 0;
      sr2 = rank / 3;
      break;
      }
      setBarrageRankSlow(br, cr, NORMAL, mode, 0.9);
      setBarrageShape(br, 1.0);
      br.xReverse = rand.nextInt(2) * 2 - 1;
      setEnemyShapeAndWings(LARGEBOSS);
      setBattery(sr1, bn1, BarrageManager.MIDDLE, NORMAL, 0, i, 0.9, mode);
      setBattery(sr2, bn2, BarrageManager.MIDDLE, NORMAL, 2, i, 0.9, mode);
    }
    shield = 400 + rand.nextInt(50);
    fireInterval = 220 + rand.nextInt(60);
    firePeriod = fireInterval / (1.2 + rand.nextFloat(0.3));
    if (rank < 10)
      firePeriod /= (2 - rank * 0.1);
  }
}

Generated by  Doxygen 1.6.0   Back to index