3D эффект иллюзии с помощью преобразования API - PullRequest
1 голос
/ 10 марта 2019

Я работаю над простой игрой на космическом корабле 2d, и я думал создать эффект эффектной трехмерной иллюзии, используя transform api . Экспериментируя, я закончил настройкой a (Горизонтальное масштабирование) и d (Вертикальное масштабирование).

Так что по умолчанию (без преобразования) мы можем использовать ctx.transform(1, 0, 0, 1, 0, 0);

Моя текущая настройка - это ctx.transform(1.72, 0, 0, 0.65, 0, 0);

При такой настройке корабль выглядит прилично, но только под некоторыми углами. Я подумал, есть ли способ использовать правильную математику, чтобы применить формулу для динамической корректировки этих значений (вероятно, на основе угла поворота корабля под). К сожалению, мои знания по математике довольно плохие, поэтому я прошу более опытных членов помочь разобраться.

Ниже приведен небольшой прототип (нажимая переключатели, вы можете переключаться между значениями по умолчанию и моими жестко заданными значениями):

var mode = "3d";
var toggle = function(ev, item) {
  mode = item.value;
};

var tools = new function() {
  this.rad = a => (Math.PI / 180) * a;
  this.deg = rad => (rad * 180) / Math.PI;
  this.distance = p =>
    Math.sqrt((p.x - p.dx) * (p.x - p.dx) + (p.y - p.dy) * (p.y - p.dy));
  this.rftv = p => Math.atan2(p.dy - p.y, p.dx - p.x);
  this.pfa = function(l, x, y, a) {
    return {
      x: Math.cos(this.rad(a)) * l + x,
      y: Math.sin(this.rad(a)) * l + y
    };
  };
}();

let design = {
  c: [
    { size: 0.7166666666666667, deg: -90 },
    { size: 0.5060742150229658, deg: -107.24145939893998 },
    { size: 0.42196629670573876, deg: -99.09027692082233 },
    { size: 0.08975274678557507, deg: -158.19859051364818 },
    { size: 0.08975274678557507, deg: -21.80140948635181 },
    { size: 0.42196629670573876, deg: -80.90972307917767 },
    { size: 0.5060742150229658, deg: -72.75854060106003 }
  ],
  l1: [
    { size: 0.4552166761249221, deg: -113.74949449286676 },
    { size: 0.3901566636906542, deg: -109.98310652189998 },
    { size: 0.18408935028645435, deg: -174.8055710922652 },
    { size: 0.6324555320336759, deg: 161.565051177078 }
  ],
  r1: [
    { size: 0.3901566636906542, deg: -70.01689347810003 },
    { size: 0.4552166761249221, deg: -66.25050550713325 },
    { size: 0.6324555320336759, deg: 18.43494882292201 },
    { size: 0.18408935028645435, deg: -5.194428907734806 }
  ],
  l2: [
    { size: 0.2608745973749754, deg: 153.434948822922 },
    { size: 0.6262764742685312, deg: 154.79887635452494 },
    { size: 0.6616477747093069, deg: 130.91438322002512 }
  ],
  r2: [
    { size: 0.2608745973749754, deg: 26.56505117707799 },
    { size: 0.6262764742685312, deg: 25.20112364547507 },
    { size: 0.6616477747093069, deg: 49.08561677997487 }
  ]
};

let circle = (x, y, r, fs, ss) => {
  ctx.save();
  ctx.beginPath();
  ctx.arc(x, y, r, 0, Math.PI * 2);
  if (fs !== false) {
    ctx.fillStyle = fs;
    ctx.fill();
  }
  if (ss !== false) {
    ctx.lineWidth = 1;
    ctx.strokeStyle = ss;
    ctx.stroke();
  }
  ctx.restore();
};

var transform = function(zAxis, tilt, scale, x, y) {
	var cs = Math.cos(zAxis), sn = Math.sin(zAxis);
	var h = Math.cos(tilt);
	var a = scale*cs, b = -scale*sn, c = x;
	var d = h*scale*sn, e = h*scale*cs, f = y;
  return { a, d, b, e, c, f };
};

let ship = (x, y, size, a, fs) => {
  ctx.save();
  ctx.beginPath();

  ctx.translate(x, y);
  if (mode === "2d") {
    ctx.transform(1, 0, 0, 1, 0, 0);
  }
  if (mode === "3d") {
    var { a, d, b, e, c, f } = transform(tools.rad(a), 45, 1, x, y);
    ctx.setTransform(a, d, b, e, c, f);
  }
  ctx.translate(-x, -y);

  for (let type in design) {
    for (let i = 0; i < design[type].length; i++) {
      let c = design[type][i],
        p = tools.pfa(size * c.size, x, y, c.deg + a + 90);
      if (i === 0) {
        ctx.moveTo(p.x, p.y);
      } else {
        ctx.lineTo(p.x, p.y);
      }
    }
    if (design[type].length > 0) {
      let c = design[type][0],
        p = tools.pfa(size * c.size, x, y, c.deg + a + 90);
      ctx.lineTo(p.x, p.y);
    }
  }

  ctx.fillStyle = fs;
  ctx.fill();
  ctx.restore();

  circle(x, y, size, false, "blue");
};

let cvs = document.createElement("canvas"),
  ctx = cvs.getContext("2d"),
  w = (cvs.width = 400),
  h = (cvs.height = 400),
  cx = w / 2,
  cy = h / 2;

let points = [
  { x: cx - 40, y: 3 },
  { x: cx + 40, y: h - 3 },
  { x: 3, y: cy + 40 },
  { x: w - 3, y: cy - 40 }
];

let shipData = {
  x: cx,
  y: cy,
  r: 40,
  a: 0,
  c: 0,
  dx: points[0].x,
  dy: points[0].y,
  run: function() {
    let d = tools.distance(this);
    if (d < 1) {
      this.c += 1;
      if (this.c > points.length - 1) {
        this.c = 0;
      }
      this.dx = points[this.c].x;
      this.dy = points[this.c].y;
    }
    let rad = tools.rftv(this);
    this.a = tools.deg(rad);
    this.x += Math.cos(rad);
    this.y += Math.sin(rad);
  }
};

let render = () => {
  ctx.clearRect(0, 0, w, h);
  ctx.fillStyle = "#ccc";
  ctx.fillRect(0, 0, w, h);

  /* debug */ circle(points[0].x, points[0].y, 3, "red", false);
  /* debug */ circle(points[1].x, points[1].y, 3, "red", false);
  /* debug */ circle(points[2].x, points[2].y, 3, "red", false);
  /* debug */ circle(points[3].x, points[3].y, 3, "red", false);

  ship(shipData.x, shipData.y, shipData.r, shipData.a, "blue");
  shipData.run();

  requestAnimationFrame(render);
};

document.body.appendChild(cvs);
render();
<div>
	2d <input type="radio" onclick="toggle(event, this);" name="display" value="2d">
	3d <input type="radio" onclick="toggle(event, this);" name="display" value="3d" checked>
</div>

--- редактировать

Я сделал правку, основанную на этой теме , где в ответе кто-то представил формулы для расчетной матрицы, но с моими настройками она не работает.

Я реализовал это так:

var transform = function(angle1, angle2, size1, size2) {
  var cs = Math.cos(angle1), sn = Math.sin(angle1);
  var h = Math.cos(angle2);
  var a = size1*cs, b = -size1*sn, c = size2;
  var d = h*size1*sn, e = h*size1*cs, f = size2;
  return { a, d, b, e, c, f };
};

if (mode === "3d") {
    var { a, d, b, e, c, f } = transform(a, a, size * 0.5, size);
    ctx.setTransform(a, d, b, e, c, f);
}

--- редактировать 2

С помощью комментариев я предположил следующее:

var transform = function(zAxis, tilt, scale, x, y) {
    var cs = Math.cos(zAxis), sn = Math.sin(zAxis);
    var h = Math.cos(tilt);
    var a = scale*cs, b = -scale*sn, c = x;
    var d = h*scale*sn, e = h*scale*cs, f = y;
    return { a, d, b, e, c, f };
};

if(mode === '3d') {
    var {a, d, b, e, c, f} = transform(tools.rad(a), 45, 1, x, y);
    ctx.setTransform(a, d, b, e, c, f);
}

Так что я думаю, что scale 1 должно быть в порядке, и, поскольку мои x, y являются центром корабля, я могу их использовать. Последним фрагментом головоломки будет второй параметр global tilt, я установил его на 45 deg, и он выглядит довольно прилично, но я не знаю, правильно ли он на 100%, я обновил код во фрагменте, чтобы кто-то мог посмотрите.

1 Ответ

1 голос
/ 10 марта 2019

Вы можете легко визуализировать измоетрическую (не перспективную) графику, используя canvas, и те же преобразования можно использовать и с CSS. Коэффициенты матрицы зависят от угла наклона (обычно фиксируется в изометрической игре) и вращения вокруг оси Z (высота).

Для полных формул вы можете увидеть этот ответ: https://stackoverflow.com/a/5186153/320726

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...