React + Three.js. Создаём собственный 3D шутер. Часть 1
- воскресенье, 1 октября 2023 г. в 00:00:16
Привет, уважаемые участники Хабр!
В современной веб-разработке границы между классическими и веб-приложениями стираются с каждым днём. Сегодня мы можем создавать не только интерактивные сайты, но и полноценные игры прямо в браузере. Одним из инструментов, который делает это возможным, является библиотека React Three Fiber - мощное средство для создания 3D-графики на основе Three.js с использованием технологии React.
React Three Fiber - это обёртка над Three.js, которая использует структуру и принципы React для создания 3D-графики в вебе. Этот стек позволяет разработчикам объединить мощь Three.js с удобством и гибкостью React, делая процесс создания приложения более интуитивным и организованным.
В основе React Three Fiber лежит идея, что всё, что вы создаёте на сцене, является компонентами React. Это позволяет разработчикам применять знакомые паттерны и методологии.
Одним из главных преимуществ React Three Fiber является его простота интеграции с экосистемой React. При использовании этой библиотеки по-прежнему можно легко интегрировать любые другие инструменты React.
Web-GameDev за последние годы пережил серьёзные изменения, превратившись из простых 2D-игр на Flash в сложные 3D-проекты, сопоставимые с настольными приложениями. Этот рост популярности и возможностей делает Web-GameDev областью, которую нельзя игнорировать.
Одним из главных преимуществ веб-игр является их доступность. Игрокам не требуется скачивать и устанавливать дополнительные программы - достаточно просто перейти по ссылке в браузере. Это упрощает распространение и продвижение игр, делая их доступными для широкой аудитории по всему миру.
Наконец, разработка веб-игр может стать отличным способом для разработчиков попробовать свои силы в gamedev, используя знакомые технологии. Благодаря доступным инструментам и библиотекам, даже без опыта в 3D-графике можно создать интересные и качественные проекты!
Современные браузеры прошли долгий путь развития, превратившись из достаточно простых инструментов для просмотра веб-страниц в мощные платформы для запуска сложных приложений и игр. Основные браузеры, такие как Chrome, Firefox, Edge и другие, постоянно оптимизируются и развиваются для обеспечения высокой производительности, что делает их идеальной платформой для разработки сложных приложений.
Одним из ключевых инструментов, который дал толчок развитию игр в браузере, является WebGL. Этот стандарт позволил разработчикам использовать аппаратное ускорение графики, что существенно повысило производительность 3D-игр. Совместно с другими веб-API, WebGL открывает новые возможности для создания впечатляющих веб-приложений прямо в браузере.
Тем не менее при разработке игр для браузера крайне важно учитывать различные аспекты производительности: оптимизация ресурсов, управление памятью и адаптация под разные устройства - всё это ключевые моменты, которые могут повлиять на успешность проекта.
Однако слова и теория - это одно, а практический опыт - совершенно иное. Чтобы действительно понять и оценить весь потенциал Web-GameDev, лучший способ - погрузиться в процесс разработки. Поэтому в качества примера успешности разработки веб-игр мы создадим собственную игру. Этот процесс позволит нам освоить ключевые аспекты разработки, столкнуться с реальными проблемами и найти пути их решения, а также на практике увидеть, насколько мощной и гибкой может быть платформа веб-геймдева.
В серии статей мы рассмотрим, как создать шутер от первого лица, используя возможности данной библиотеки, и погрузимся в увлекательный мир веб-игр!
Финальное демо
Репозиторий на GitHub
А теперь, давайте начинать!
В первую очередь нам понадобится шаблон проекта на React. Поэтому начнём с его установки.
npm create vite@latest
выбираем библиотеку React;
выбираем JavaScript.
Устанавливаем дополнительные npm-пакеты.
npm install three @react-three/fiber @react-three/drei @react-three/rapier zustand @tweenjs/tween.js
Затем удалим всё лишнее из нашего проекта.
В файл main.jsx добавим div-элемент, который будет отображаться на странице в качестве прицела. Вставим компонент Canvas и зададим поле зрения камеры. Внутрь компонента Canvas поместим компонент App.
import React from 'react'
import ReactDOM from 'react-dom/client'
import App from './App.jsx'
import './index.css'
import {Canvas} from "@react-three/fiber";
ReactDOM.createRoot(document.getElementById('root')).render(
<React.StrictMode>
<div id="container">
<div className="aim"></div>
<Canvas camera={{ fov: 45 }}>
<App />
</Canvas>
</div>
</React.StrictMode>,
)
Добавим стили, чтобы растянуть элементы интерфейса на всю высоту экрана и вывести прицела в виде круга по центру экрана.
Добавим стили, чтобы растянуть элементы интерфейса на всю высоту экрана и вывести прицела в виде круга по центру экрана.
В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба. В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба.
* {
margin: 0;
padding: 0;
}
html, body, #root, #container {
height: 100%;
}
.aim {
position: absolute;
top: 50%;
left: 50%;
width: 10px;
height: 10px;
border-radius: 50%;
transform: translate3d(-50%, -50%, 0);
border: 2px solid white;
z-index: 2;
}
В компонент App добавляем компонент Sky, который будет отображаться задним фоном в нашей игровой сцене в виде неба.
import {Sky} from "@react-three/drei";
export const App = () => {
return (
<>
<Sky sunPosition={[100, 20, 100]} />
</>
)
}
export default App
Создадим компонент Ground и поместим его в компонент App.
import {Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
export const App = () => {
return (
<>
<Sky sunPosition={[100, 20, 100]}/>
<Ground />
</>
)
}
export default App
В Ground создадим элемент плоской поверхности. По оси Y переместим вниз, чтобы эта плоскость попала в поле зрения камеры. А также перевернём плоскость по оси Х, чтобы она приняла горизонтальное положение.
export const Ground = () => {
return (
<mesh position={[0, -5, 0]} rotation-x={-Math.PI / 2}>
<planeGeometry args={[500, 500]} />
<meshStandardMaterial color="gray" />
</mesh>
);
}
Несмотря на то что в качестве цвета материала мы указали серый, плоскость отображается полностью чёрной.
По умолчанию в сцене отсутствует освещение, поэтому добавим источник света ambientLight, который освещает объект со всех сторон и не имеет направленного луча. В качестве параметра настроим силу свечения.
import {Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
export const App = () => {
return (
<>
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<Ground />
</>
)
}
export default App
Чтобы поверхность пола не выглядела однородно, добавим текстуру. Сделаем рисунок поверхности пола в виде ячеек, повторяющихся на всём протяжении поверхности.
В папку assets добавим изображение PNG с текстурой.
Для загрузки текстуры на сцену воспользуемся хуком useTexture из пакета @react-three/drei. А в качестве параметра для хука передадим импортированное в файл изображение текстуры. Зададим повторение изображения горизонтальным осям.
import * as THREE from "three";
import { useTexture } from "@react-three/drei";
import floorTexture from "./assets/floor.png";
export const Ground = () => {
const texture = useTexture(floorTexture);
texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
return (
<mesh position={[0, -5, 0]} rotation-x={-Math.PI / 2}>
<planeGeometry args={[500, 500]} />
<meshStandardMaterial color="gray" map={texture} map-repeat={[360, 360]} />
</mesh>
);
}
При помощи компонента PointerLockControls из пакета @react-three/drei зафиксируем курсор на экране, чтобы он не перемещался при движении мышью, а менялось положение камеры на сцене.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<Ground />
</>
)
}
export default App
Сделаем небольшую правку для компонента Ground.
import * as THREE from "three";
import { useTexture } from "@react-three/drei";
import floorTexture from "./assets/floor.png";
export const Ground = () => {
const texture = useTexture(floorTexture);
texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
return (
<mesh position={[0, -5, 0]} rotation-x={-Math.PI / 2}>
<planeGeometry args={[500, 500]} />
<meshStandardMaterial color="gray" map={texture} map-repeat={[100, 100]} />
</mesh>
);
}
Для наглядности добавим на сцену простой куб.
<mesh position={[0, 3, -5]}>
<boxGeometry />
</mesh>
Сейчас он просто висит в пространстве.
При помощи компонента Physics из пакета @react-three/rapier добавим “физику” на сцену. В качестве параметра настроим поле gravity, в котором зададим силы притяжения по осям.
<Physics gravity={[0, -20, 0]}>
<Ground />
<mesh position={[0, 3, -5]}>
<boxGeometry />
</mesh>
</Physics>
Однако, наш куб находится внутри компонента с физикой, но с ним ничего не происходит. Для того чтобы куб вёл себя как реальный физический объект, необходимо обернуть его в компонент RigidBody из пакета @react-three/rapier.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics, RigidBody} from "@react-three/rapier";
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<Physics gravity={[0, -20, 0]}>
<Ground />
<RigidBody>
<mesh position={[0, 3, -5]}>
<boxGeometry />
</mesh>
</RigidBody>
</Physics>
</>
)
}
export default App
После этого сразу же увидим, что при каждой перезагрузке страницы куб под воздействием гравитации падает.
Но теперь появилась другая задача - необходимо сделать пол таким объектом, с которым может взаимодействовать куб, и за пределы которого он не провалится.
Вернёмся к компоненту Ground и добавим компонент RigidBody в качестве обёртки над поверхностью пола.
import * as THREE from "three";
import { useTexture } from "@react-three/drei";
import floorTexture from "./assets/floor.png";
import {RigidBody} from "@react-three/rapier";
export const Ground = () => {
const texture = useTexture(floorTexture);
texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
return (
<RigidBody>
<mesh position={[0, -5, 0]} rotation-x={-Math.PI / 2}>
<planeGeometry args={[500, 500]} />
<meshStandardMaterial color="gray" map={texture} map-repeat={[100, 100]} />
</mesh>
</RigidBody>
);
}
Теперь при падении куб остаётся на полу как на реальном физическом объекте.
Создадим компонент Player, который будет контролировать персонажа на сцене.
Персонаж - это такой же физический объект, как и добавленный куб, поэтому он должен взаимодействовать с поверхностью пола, а также с кубом на сцене. Поэтому добавляем компонент RigidBody. А персонажа сделаем в форме капсулы.
import {RigidBody} from "@react-three/rapier";
export const Player = () => {
return (
<>
<RigidBody position={[0, 1, -2]}>
<mesh>
<capsuleGeometry args={[0.5, 0.5]}/>
</mesh>
</RigidBody>
</>
);
}
Компонент Player поместим внутрь компонента Physics.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics, RigidBody} from "@react-three/rapier";
import {Player} from "./Player.jsx";
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<Physics gravity={[0, -20, 0]}>
<Ground />
<Player />
<RigidBody>
<mesh position={[0, 3, -5]}>
<boxGeometry />
</mesh>
</RigidBody>
</Physics>
</>
)
}
export default App
Теперь на сцене появился наш персонаж.
Управление персонажем будет осуществляться при помощи клавиш WASD, а прыжок при помощи кнопки Пробел.
С помощью собственного react-хука мы реализуем логику перемещения персонажа.
Создадим файл hooks.js и добавим туда новую функцию usePersonControls.
Определим объект в формате {“код клавиши”: “выполняемое действие”}. Далее добавим обработчики событий при нажатии и отпускании клавиш клавиатуры. При срабатывании обработчиков будем определять текущие выполняемые действия и обновлять их активное состояние. В конечном результате из хука вернём объект в формате {“выполняемое действие”: “статус”}.
import { useEffect, useState } from "react";
export const usePersonControls = () => {
const keys = {
KeyW: "forward",
KeyS: "backward",
KeyA: "left",
KeyD: "right",
Space: "jump"
}
const moveFieldByKey = (key) => keys[key];
const [movement, setMovement] = useState({
forward: false,
backward: false,
left: false,
right: false,
jump: false
});
const setMovementStatus = (code, status) => {
setMovement((m) => ({...m, [code]: status}))
}
useEffect(() => {
const handleKeyDown = (ev) => {
setMovementStatus(moveFieldByKey(ev.code), true);
}
const handleKeyUp = (ev) => {
setMovementStatus(moveFieldByKey(ev.code), false);
}
document.addEventListener('keydown', handleKeyDown);
document.addEventListener('keyup', handleKeyUp);
return () => {
document.removeEventListener('keydown', handleKeyDown);
document.removeEventListener('keyup', handleKeyUp);
}
}, []);
return movement;
}
После реализации хука usePersonControls его необходимо использовать при управлении персонажем. В компоненте Player добавим отслеживание состояния движения и будем обновлять вектор направления движения персонажа.
Также определим переменные, которые будут хранить состояния направлений движения.
Для обновления положения персонажа воспользуемся useFrame, предоставляемым пакетом @react-three/fiber. Данный хук работает аналогично requestAnimationFrame и выполняет тело функции примерно 60 раз в секунду.
Пояснение по коду
const playerRef = useRef();
Создаём ссылку для объекта игрока. Эта ссылка позволит напрямую взаимодействовать с объектом игрока на сцене.const { forward, backward, left, right, jump } = usePersonControls();
При использовании хука возвращается объект с булевыми значениями, указывающими, какие кнопки управления в данный момент нажаты игроком.useFrame((state) => { ... });
Хук вызывается на каждом кадре анимации. Внутри этого хука происходит обновление позиции и линейной скорости игрока.if (!playerRef.current) return;
Проверка на наличие объекта игрока. Если объект игрока отсутствует, то, во избежание ошибок, функция прекратит выполнение.const velocity = playerRef.current.linvel();
Получение текущей линейной скорости игрока.frontVector.set(0, 0, backward - forward);
Установка вектора движения вперёд/назад на основе нажатых кнопок.sideVector.set(left - right, 0, 0);
Установка вектора движения влево/вправо.direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED);
Вычисление итогового вектора движения игрока путём вычитания векторов движения, нормализации результата (чтобы длина вектора была равна 1) и умножения на константу скорости движения.playerRef.current.wakeUp();
"Пробуждение" объекта игрока, чтобы убедиться, что он реагирует на изменения. Если не использовать данный метод, то через некоторое время объект “заснёт” и не будет реагировать на изменение позиции.playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
Установка новой линейной скорости игрока на основе вычисленного направления движения и сохранение текущей вертикальной скорости (чтобы не влиять на прыжки или падения).
import * as THREE from "three";
import {RigidBody} from "@react-three/rapier";
import {useRef} from "react";
import {usePersonControls} from "./hooks.js";
import {useFrame} from "@react-three/fiber";
const MOVE_SPEED = 5;
const direction = new THREE.Vector3();
const frontVector = new THREE.Vector3();
const sideVector = new THREE.Vector3();
export const Player = () => {
const playerRef = useRef();
const { forward, backward, left, right, jump } = usePersonControls();
useFrame((state) => {
if (!playerRef.current) return;
const velocity = playerRef.current.linvel();
frontVector.set(0, 0, backward - forward);
sideVector.set(left - right, 0, 0);
direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED);
playerRef.current.wakeUp();
playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
});
return (
<>
<RigidBody position={[0, 1, -2]} ref={playerRef}>
<mesh>
<capsuleGeometry args={[0.5, 0.5]}/>
</mesh>
</RigidBody>
</>
);
}
В результате при нажатии на клавиши WASD персонаж начал перемещаться по сцене. Также он может взаимодействовать с кубом, потому что они оба являются физическими объектами.
Для того чтобы реализовать прыжок, воспользуемся функционалом из пакетов @dimforge/rapier3d-compat и @react-three/rapier. В данном примере проверим, что персонаж находится на земле и была нажата клавиша для прыжка. В таком случае мы задаём персонажу направление и силу ускорения по оси Y.
Ещё для Player добавим массу и заблокируем вращения по всем осям, чтобы он не заваливался в разные стороны при столкновении с другими объектами на сцене.
Пояснение по коду
const world = rapier.world;
Получение доступа к сцене физического движка Rapier. Он содержит все физические объекты и управляет их взаимодействием.const ray = world.castRay(new RAPIER.Ray(playerRef.current.translation(), { x: 0, y: -1, z: 0 }));
В этом месте происходит “лучевое прослеживание” (raycasting). Создаётся луч, который начинается в текущей позиции игрока и направлен вниз по оси Y. Этот луч “бросается” в сцену, чтобы определить, пересекается ли он с каким-либо объектом на сцене.const grounded = ray && ray.collider && Math.abs(ray.toi) <= 1.5;
Проверяется условие, находится ли игрок на земле:
ray - был ли луч создан;
ray.collider - столкнулся ли луч с каким-либо объектом на сцене;
Math.abs(ray.toi) - “время воздействия” луча. Если это значение меньше или равно заданному, это может указывать на то, что игрок находится достаточно близко к поверхности, чтобы считаться “на земле”.
import * as THREE from "three";
import * as RAPIER from "@dimforge/rapier3d-compat"
import {RigidBody, useRapier} from "@react-three/rapier";
import {useRef} from "react";
import {usePersonControls} from "./hooks.js";
import {useFrame} from "@react-three/fiber";
const MOVE_SPEED = 5;
const direction = new THREE.Vector3();
const frontVector = new THREE.Vector3();
const sideVector = new THREE.Vector3();
export const Player = () => {
const playerRef = useRef();
const { forward, backward, left, right, jump } = usePersonControls();
const rapier = useRapier();
useFrame((state) => {
if (!playerRef.current) return;
// moving player
const velocity = playerRef.current.linvel();
frontVector.set(0, 0, backward - forward);
sideVector.set(left - right, 0, 0);
direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED);
playerRef.current.wakeUp();
playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
// jumping
const world = rapier.world;
const ray = world.castRay(new RAPIER.Ray(playerRef.current.translation(), { x: 0, y: -1, z: 0 }));
const grounded = ray && ray.collider && Math.abs(ray.toi) <= 1;
if (jump && grounded) doJump();
});
const doJump = () => {
playerRef.current.setLinvel({x: 0, y: 8, z: 0});
}
return (
<>
<RigidBody position={[0, 1, -2]} mass={1} ref={playerRef} lockRotations>
<mesh>
<capsuleGeometry args={[0.5, 0.5]}/>
</mesh>
</RigidBody>
</>
);
}
Также необходимо изменить компонент Ground, чтобы корректно работал алгоритм с лучами для определения статуса “приземления”, добавив в него физический объект, который будет взаимодействовать с другими объектами на сцене.
import * as THREE from "three";
import { useTexture } from "@react-three/drei";
import floorTexture from "./assets/floor.png";
import {CuboidCollider, RigidBody} from "@react-three/rapier";
export const Ground = () => {
const texture = useTexture(floorTexture);
texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
return (
<RigidBody type="fixed" colliders={false}>
<mesh position={[0, 0, 0]} rotation-x={-Math.PI / 2}>
<planeGeometry args={[500, 500]} />
<meshStandardMaterial color="gray" map={texture} map-repeat={[100, 100]} />
</mesh>
<CuboidCollider args={[500, 2, 500]} position={[0, -2, 0]}/>
</RigidBody>
);
}
Поднимем камеру немного выше для лучшей видимости сцены.
import React from 'react'
import ReactDOM from 'react-dom/client'
import App from './App.jsx'
import './index.css'
import {Canvas} from "@react-three/fiber";
ReactDOM.createRoot(document.getElementById('root')).render(
<React.StrictMode>
<div id="container">
<div className="aim"></div>
<Canvas camera={{ fov: 45, position: [0, 5, 0] }}>
<App />
</Canvas>
</div>
</React.StrictMode>,
)
Код раздела:
Для перемещения камеры мы будем получать текущую позицию игрока и при каждом обновлении кадра менять позицию камеры. А для того, чтобы персонаж двигался именно по траектории, куда направлена камера, необходимо добавить applyEuler.
Пояснение по коду
Метод applyEuler применяет вращение к вектору на основе заданных углов Эйлера. В данном случае вращение камеры применяется к вектору direction. Это используется для соответствия движения относительно ориентации камеры, чтобы игрок двигался в том направлении, куда повёрнута камера.
Немного поправим размер Player и сделаем его выше относительно куба, увеличив размер CapsuleCollider и поправив логику “прыжка”.
Код раздела:
Чтобы сцена не ощущалась совсем пустой, добавим генерацию кубов. В файле json перечислим координаты каждого из кубов и затем отобразим их на сцене. Для этого создадим файл cubes.json, в котором перечислим массив координат.
[
[0, 0, -7],
[2, 0, -7],
[4, 0, -7],
[6, 0, -7],
[8, 0, -7],
[10, 0, -7]
]
В файле Cube.jsx создадим компонент Cubes, в котором будет осуществляться генерация кубов в цикле. А компонент Cube будет являться непосредственно генерируемым объектом.
import {RigidBody} from "@react-three/rapier";
import cubes from "./cubes.json";
export const Cubes = () => {
return cubes.map((coords, index) => <Cube key={index} position={coords} />);
}
const Cube = (props) => {
return (
<RigidBody {...props}>
<mesh castShadow receiveShadow>
<meshStandardMaterial color="white" />
<boxGeometry />
</mesh>
</RigidBody>
);
}
Добавим созданный компонент Cubes в компонент App, удалив предыдущий одиночный куб.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics} from "@react-three/rapier";
import {Player} from "./Player.jsx";
import {Cubes} from "./Cube.jsx";
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<Physics gravity={[0, -20, 0]}>
<Ground />
<Player />
<Cubes />
</Physics>
</>
)
}
export default App
Теперь займёмся добавлением 3D-модели на сцену. Добавим модель оружия для персонажа. К примеру, возьмём вот такую.
Скачаем модель в формате GLTF и распакуем архив в корне проекта.
Для того чтобы получить нужный нам формат для импорта модели на сцену, потребуется установить дополнительный пакет gltf-pipeline.
npm i -D gltf-pipeline
При помощи пакета gltf-pipeline реконвертируем модель из формата GLTF в формат GLB, так как в этом формате все данные модели помещаются в один файл. В качестве выходной директории для сгенерированного файла указываем папку public.
gltf-pipeline -i weapon/scene.gltf -o public/weapon.glb
Затем необходимо сгенерировать react-компонент, который будет содержать разметку этой модели, для добавления её на сцену. Воспользуемся официальным ресурсом от разработчиков @react-three/fiber.
Перейдя в конвертер, потребуется загрузить конвертированный файл weapon.glb. При помощи перетаскивания или поиска через Проводник найдём этот файл и загрузим его.
В конвертере мы увидим сгенерированный react-компонент, код которого перенесём в наш проект в новый файл WeaponModel.jsx, изменив название самого компонента на одноимённое с файлом.
Теперь импортируем созданную модель на сцену. В файле App.jsx добавим компонент WeaponModel.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics} from "@react-three/rapier";
import {Player} from "./Player.jsx";
import {Cubes} from "./Cube.jsx";
import {WeaponModel} from "./WeaponModel.jsx";
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<Physics gravity={[0, -20, 0]}>
<Ground />
<Player />
<Cubes />
</Physics>
<group position={[0, 3, 0]}>
<WeaponModel />
</group>
</>
)
}
export default App
В данный момент на нашей сцене ни один из предметов не отбрасывает тени.
Чтобы включить тени на сцене необходимо добавить атрибут shadows для компонента Canvas.
import React from 'react'
import ReactDOM from 'react-dom/client'
import App from './App.jsx'
import './index.css'
import {Canvas} from "@react-three/fiber";
ReactDOM.createRoot(document.getElementById('root')).render(
<React.StrictMode>
<div id="container">
<div className="aim"></div>
<Canvas camera={{ fov: 45 }} shadows>
<App />
</Canvas>
</div>
</React.StrictMode>,
)
Далее необходимо добавить новый источник освещения. Несмотря на то что мы уже имеем на сцене ambientLight, он не может создавать тени для объектов, так как у него нет направленного луча света. Поэтому добавим новый источник освещения directionalLight и настроим его. Атрибутом для включения режима “отбрасывание” тени является castShadow. Именно добавление этого параметра указывает, что данный объект может отбрасывать тень на другие предметы.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics} from "@react-three/rapier";
import {Player} from "./Player.jsx";
import {Cubes} from "./Cube.jsx";
import {WeaponModel} from "./WeaponModel.jsx";
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<directionalLight
castShadow
intensity={.8}
position={[50, 50, 0]} />
<Physics gravity={[0, -20, 0]}>
<Ground />
<Player />
<Cubes />
</Physics>
<group position={[3, 1, -2]}>
<WeaponModel />
</group>
</>
)
}
export default App
После этого добавим другой атрибут receiveShadow в компоненте Ground, который означает, что компонент на сцене может принимать и отображать тени на себе.
import * as THREE from "three";
import { useTexture } from "@react-three/drei";
import floorTexture from "./assets/floor.png";
import {CuboidCollider, RigidBody} from "@react-three/rapier";
export const Ground = () => {
const texture = useTexture(floorTexture);
texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
return (
<RigidBody type="fixed" colliders={false}>
<mesh receiveShadow position={[0, 0, 0]} rotation-x={-Math.PI / 2}>
<planeGeometry args={[500, 500]} />
<meshStandardMaterial color="gray" map={texture} map-repeat={[100, 100]} />
</mesh>
<CuboidCollider args={[500, 2, 500]} position={[0, -2, 0]}/>
</RigidBody>
);
}
Аналогичные атрибуты необходимо добавить другим объектам на сцене: кубам и игроку. Для кубов мы добавим castShadow и receiveShadow, потому что они могут как отбрасывать, так и принимать тени, а для игрока добавим только castShadow.
Добавим castShadow для Player.
import * as THREE from "three";
import * as RAPIER from "@dimforge/rapier3d-compat"
import {CapsuleCollider, RigidBody, useRapier} from "@react-three/rapier";
import {useRef} from "react";
import {usePersonControls} from "./hooks.js";
import {useFrame} from "@react-three/fiber";
const MOVE_SPEED = 5;
const direction = new THREE.Vector3();
const frontVector = new THREE.Vector3();
const sideVector = new THREE.Vector3();
export const Player = () => {
const playerRef = useRef();
const { forward, backward, left, right, jump } = usePersonControls();
const rapier = useRapier();
useFrame((state) => {
if (!playerRef.current) return;
// moving player
const velocity = playerRef.current.linvel();
frontVector.set(0, 0, backward - forward);
sideVector.set(left - right, 0, 0);
direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED).applyEuler(state.camera.rotation);
playerRef.current.wakeUp();
playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
// jumping
const world = rapier.world;
const ray = world.castRay(new RAPIER.Ray(playerRef.current.translation(), { x: 0, y: -1, z: 0 }));
const grounded = ray && ray.collider && Math.abs(ray.toi) <= 1.5;
if (jump && grounded) doJump();
// moving camera
const {x, y, z} = playerRef.current.translation();
state.camera.position.set(x, y, z);
});
const doJump = () => {
playerRef.current.setLinvel({x: 0, y: 8, z: 0});
}
return (
<>
<RigidBody colliders={false} mass={1} ref={playerRef} lockRotations>
<mesh castShadow>
<capsuleGeometry args={[0.5, 0.5]}/>
<CapsuleCollider args={[0.75, 0.5]} />
</mesh>
</RigidBody>
</>
);
}
Добавим castShadow и receiveShadow для Cube.
import {RigidBody} from "@react-three/rapier";
import cubes from "./cubes.json";
export const Cubes = () => {
return cubes.map((coords, index) => <Cube key={index} position={coords} />);
}
const Cube = (props) => {
return (
<RigidBody {...props}>
<mesh castShadow receiveShadow>
<meshStandardMaterial color="white" />
<boxGeometry />
</mesh>
</RigidBody>
);
}
Если сейчас присмотреться, то можно обнаружить, что площадь поверхности, на которую отбрасывается тень совсем небольшая. И при выходе за пределы этой площади тень просто обрезается.
Происходит это по той причине, что по умолчанию камера захватывает лишь небольшую область отображаемых теней от directionalLight. Мы можем для компонента directionalLight при помощи добавления дополнительных атрибутов shadow-camera-(top, bottom, left, right) расширить эту область видимости. После добавления этих атрибутов, тень станет немного размытой. Для улучшения качества добавим атрибут shadow-mapSize.
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics} from "@react-three/rapier";
import {Player} from "./Player.jsx";
import {Cubes} from "./Cube.jsx";
import {WeaponModel} from "./WeaponModel.jsx";
const shadowOffset = 50;
export const App = () => {
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<directionalLight
castShadow
intensity={1.5}
shadow-mapSize={4096}
shadow-camera-top={shadowOffset}
shadow-camera-bottom={-shadowOffset}
shadow-camera-left={shadowOffset}
shadow-camera-right={-shadowOffset}
position={[100, 100, 0]}
/>
<Physics gravity={[0, -20, 0]}>
<Ground />
<Player />
<Cubes />
</Physics>
<group position={[3, 1, -2]}>
<WeaponModel />
</group>
</>
)
}
export default App
Теперь добавим отображение оружия от первого лица. Создадим новый компонент Weapon, который будет содержать в себе логику поведения оружия и саму 3D-модель.
import {WeaponModel} from "./WeaponModel.jsx";
export const Weapon = (props) => {
return (
<group {...props}>
<WeaponModel />
</group>
);
}
Поместим этот компонент на одном уровне с RigidBody персонажа и в хуке useFrame будем задавать позицию и угол вращения на основе положения значений от камеры.
import * as THREE from "three";
import * as RAPIER from "@dimforge/rapier3d-compat"
import {CapsuleCollider, RigidBody, useRapier} from "@react-three/rapier";
import {useRef} from "react";
import {usePersonControls} from "./hooks.js";
import {useFrame} from "@react-three/fiber";
import {Weapon} from "./Weapon.jsx";
const MOVE_SPEED = 5;
const direction = new THREE.Vector3();
const frontVector = new THREE.Vector3();
const sideVector = new THREE.Vector3();
const rotation = new THREE.Vector3();
export const Player = () => {
const playerRef = useRef();
const { forward, backward, left, right, jump } = usePersonControls();
const objectInHandRef = useRef();
const rapier = useRapier();
useFrame((state) => {
if (!playerRef.current) return;
// moving player
const velocity = playerRef.current.linvel();
frontVector.set(0, 0, backward - forward);
sideVector.set(left - right, 0, 0);
direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED).applyEuler(state.camera.rotation);
playerRef.current.wakeUp();
playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
// jumping
const world = rapier.world;
const ray = world.castRay(new RAPIER.Ray(playerRef.current.translation(), { x: 0, y: -1, z: 0 }));
const grounded = ray && ray.collider && Math.abs(ray.toi) <= 1.5;
if (jump && grounded) doJump();
// moving camera
const {x, y, z} = playerRef.current.translation();
state.camera.position.set(x, y, z);
// moving object in hand for the player
objectInHandRef.current.rotation.copy(state.camera.rotation);
objectInHandRef.current.position.copy(state.camera.position).add(state.camera.getWorldDirection(rotation));
});
const doJump = () => {
playerRef.current.setLinvel({x: 0, y: 8, z: 0});
}
return (
<>
<RigidBody colliders={false} mass={1} ref={playerRef} lockRotations>
<mesh castShadow>
<capsuleGeometry args={[0.5, 0.5]}/>
<CapsuleCollider args={[0.75, 0.5]} />
</mesh>
</RigidBody>
<group ref={objectInHandRef}>
<Weapon position={[0.3, -0.1, 0.3]} scale={0.3} />
</group>
</>
);
}
Чтобы походка персонажа была более естественной, мы добавим небольшое покачивание оружия во время движения. Для создания анимации мы воспользуемся установленной библиотекой tween.js.
Компонент Weapon обернём в тег group, чтобы можно было добавить ссылку на него через хук useRef.
Добавим несколько состояний useState для сохранения анимации.
Создадим функцию для инициализации анимации.
Пояснение по коду
const twSwayingAnimation = new TWEEN.Tween(currentPosition) …
Создание анимации "качания" объекта от его текущей позиции к новой позиции.const twSwayingBackAnimation = new TWEEN.Tween(currentPosition) …
Создание анимации возврата объекта обратно к его начальной позиции после завершения первой анимации.twSwayingAnimation.chain(twSwayingBackAnimation);
Соединение двух анимаций, чтобы после завершения первой анимации автоматически начиналась вторая.
В useEffect вызовем функцию инициализации анимации.
Теперь необходимо определить момент, в течение которого происходит движение. Сделать это можно при помощи определения текущего вектора направление персонажа.
Если движение персонажа происходит, то будем обновлять анимацию и при окончании запускать ещё раз.
Пояснение по коду
const isMoving = direction.length() > 0;
Здесь проверяется состояние движения объекта. Если вектор direction имеет длину больше 0, то это значит, что у объекта есть направление движения.if (isMoving && isSwayingAnimationFinished) { ... }
Это состояние выполняется, если объект двигается, а анимация “качания” завершилась.
import * as THREE from "three";
import * as RAPIER from "@dimforge/rapier3d-compat"
import * as TWEEN from "@tweenjs/tween.js";
import {CapsuleCollider, RigidBody, useRapier} from "@react-three/rapier";
import {useEffect, useRef, useState} from "react";
import {usePersonControls} from "./hooks.js";
import {useFrame} from "@react-three/fiber";
import {Weapon} from "./Weapon.jsx";
const MOVE_SPEED = 5;
const direction = new THREE.Vector3();
const frontVector = new THREE.Vector3();
const sideVector = new THREE.Vector3();
const rotation = new THREE.Vector3();
export const Player = () => {
const playerRef = useRef();
const { forward, backward, left, right, jump } = usePersonControls();
const objectInHandRef = useRef();
const swayingObjectRef = useRef();
const [swayingAnimation, setSwayingAnimation] = useState(null);
const [swayingBackAnimation, setSwayingBackAnimation] = useState(null);
const [isSwayingAnimationFinished, setIsSwayingAnimationFinished] = useState(true);
const rapier = useRapier();
useFrame((state) => {
if (!playerRef.current) return;
// moving player
const velocity = playerRef.current.linvel();
frontVector.set(0, 0, backward - forward);
sideVector.set(left - right, 0, 0);
direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED).applyEuler(state.camera.rotation);
playerRef.current.wakeUp();
playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
// jumping
const world = rapier.world;
const ray = world.castRay(new RAPIER.Ray(playerRef.current.translation(), { x: 0, y: -1, z: 0 }));
const grounded = ray && ray.collider && Math.abs(ray.toi) <= 1.5;
if (jump && grounded) doJump();
// moving camera
const {x, y, z} = playerRef.current.translation();
state.camera.position.set(x, y, z);
// moving object in hand for the player
objectInHandRef.current.rotation.copy(state.camera.rotation);
objectInHandRef.current.position.copy(state.camera.position).add(state.camera.getWorldDirection(rotation));
const isMoving = direction.length() > 0;
if (isMoving && isSwayingAnimationFinished) {
setIsSwayingAnimationFinished(false);
swayingAnimation.start();
}
TWEEN.update();
});
const doJump = () => {
playerRef.current.setLinvel({x: 0, y: 8, z: 0});
}
const initSwayingObjectAnimation = () => {
const currentPosition = new THREE.Vector3(0, 0, 0);
const initialPosition = new THREE.Vector3(0, 0, 0);
const newPosition = new THREE.Vector3(-0.05, 0, 0);
const animationDuration = 300;
const easing = TWEEN.Easing.Quadratic.Out;
const twSwayingAnimation = new TWEEN.Tween(currentPosition)
.to(newPosition, animationDuration)
.easing(easing)
.onUpdate(() => {
swayingObjectRef.current.position.copy(currentPosition);
});
const twSwayingBackAnimation = new TWEEN.Tween(currentPosition)
.to(initialPosition, animationDuration)
.easing(easing)
.onUpdate(() => {
swayingObjectRef.current.position.copy(currentPosition);
})
.onComplete(() => {
setIsSwayingAnimationFinished(true);
});
twSwayingAnimation.chain(twSwayingBackAnimation);
setSwayingAnimation(twSwayingAnimation);
setSwayingBackAnimation(twSwayingBackAnimation);
}
useEffect(() => {
initSwayingObjectAnimation();
}, []);
return (
<>
<RigidBody colliders={false} mass={1} ref={playerRef} lockRotations>
<mesh castShadow>
<capsuleGeometry args={[0.5, 0.5]}/>
<CapsuleCollider args={[0.75, 0.5]} />
</mesh>
</RigidBody>
<group ref={objectInHandRef}>
<group ref={swayingObjectRef}>
<Weapon position={[0.3, -0.1, 0.3]} scale={0.3} />
</group>
</group>
</>
);
}
В компонент App добавим useFrame, в котором будем обновлять tween-анимацию.
TWEEN.update() обновляет все активные анимации в библиотеке TWEEN.js. Этот метод вызывается на каждом кадре анимации для обеспечения плавного выполнения всех анимаций.
import * as TWEEN from "@tweenjs/tween.js";
import {PointerLockControls, Sky} from "@react-three/drei";
import {Ground} from "./Ground.jsx";
import {Physics} from "@react-three/rapier";
import {Player} from "./Player.jsx";
import {Cubes} from "./Cube.jsx";
import {useFrame} from "@react-three/fiber";
const shadowOffset = 50;
export const App = () => {
useFrame(() => {
TWEEN.update();
});
return (
<>
<PointerLockControls />
<Sky sunPosition={[100, 20, 100]}/>
<ambientLight intensity={1.5} />
<directionalLight
castShadow
intensity={1.5}
shadow-mapSize={4096}
shadow-camera-top={shadowOffset}
shadow-camera-bottom={-shadowOffset}
shadow-camera-left={shadowOffset}
shadow-camera-right={-shadowOffset}
position={[100, 100, 0]}
/>
<Physics gravity={[0, -20, 0]}>
<Ground />
<Player />
<Cubes />
</Physics>
</>
)
}
export default App
Код раздела:
Нам нужно определить момент, когда происходит выстрел — то есть при нажатии кнопки мыши. Добавим useState для хранения этого состояния, useRef для сохранения ссылки на объект оружия, а также два обработчика событий для нажатия и отпускания клавиши мыши.
Реализуем анимацию отдачи при клике кнопкой мыши. Будем для этого использовать библиотеку tween.js.
Определим константы для силы отдачи и длительности анимации.
Как и при анимации покачивания оружия, добавим два состояния useState для анимации отдачи и возврата в начальное положение и состояние со статусом окончания анимации.
Сделаем функции для получения случайного вектора анимации отдачи - generateRecoilOffset и generateNewPositionOfRecoil.
Создадим функцию для инициализации анимации отдачи. Также добавим useEffect, в котором в качестве зависимости укажем состояние “выстрела”, чтобы при каждом выстреле заново происходила инициализация анимации и генерировались новые конечные координаты.
А в useFrame добавим проверку “удержания” клавиши мыши для стрельбы, чтобы анимация стрельбы не прекращалась до тех пор, пока клавиша не будет отпущена.
import * as THREE from "three";
import * as TWEEN from "@tweenjs/tween.js";
import {WeaponModel} from "./WeaponModel.jsx";
import {useEffect, useRef, useState} from "react";
import {useFrame} from "@react-three/fiber";
const recoilAmount = 0.03;
const recoilDuration = 100;
const easing = TWEEN.Easing.Quadratic.Out;
export const Weapon = (props) => {
const [recoilAnimation, setRecoilAnimation] = useState(null);
const [recoilBackAnimation, setRecoilBackAnimation] = useState(null);
const [isShooting, setIsShooting] = useState(false);
const weaponRef = useRef();
useEffect(() => {
document.addEventListener('mousedown', () => {
setIsShooting(true);
});
document.addEventListener('mouseup', () => {
setIsShooting(false);
});
}, []);
const generateRecoilOffset = () => {
return new THREE.Vector3(
Math.random() * recoilAmount,
Math.random() * recoilAmount,
Math.random() * recoilAmount,
)
}
const generateNewPositionOfRecoil = (currentPosition) => {
const recoilOffset = generateRecoilOffset();
return currentPosition.clone().add(recoilOffset);
}
const initRecoilAnimation = () => {
const currentPosition = new THREE.Vector3(0, 0, 0);
const initialPosition = new THREE.Vector3(0, 0, 0);
const newPosition = generateNewPositionOfRecoil(currentPosition);
const twRecoilAnimation = new TWEEN.Tween(currentPosition)
.to(newPosition, recoilDuration)
.easing(easing)
.onUpdate(() => {
weaponRef.current.position.copy(currentPosition);
});
const twRecoilBackAnimation = new TWEEN.Tween(currentPosition)
.to(initialPosition, recoilDuration)
.easing(easing)
.onUpdate(() => {
weaponRef.current.position.copy(currentPosition);
});
twRecoilAnimation.chain(twRecoilBackAnimation);
setRecoilAnimation(twRecoilAnimation);
setRecoilBackAnimation(twRecoilBackAnimation);
}
const startShooting = () => {
recoilAnimation.start();
}
useEffect(() => {
initRecoilAnimation();
if (isShooting) {
startShooting();
}
}, [isShooting]);
useFrame(() => {
TWEEN.update();
if (isShooting) {
startShooting();
}
});
return (
<group {...props}>
<group ref={weaponRef}>
<WeaponModel />
</group>
</group>
);
}
Реализуем анимацию “бездействия” для персонажа, чтобы не создавалось ощущения “зависнувшей” игры.
Для этого добавим несколько новых состояний через useState.
Поправим инициализацию анимации “покачивания”, чтобы использовать значения из состояния. Идея заключается в том, что при разных состояниях: ходьба или остановка, - будут задействованы различные значения для анимации и каждый раз анимация будет инициализирована сначала.
import * as THREE from "three";
import * as RAPIER from "@dimforge/rapier3d-compat"
import * as TWEEN from "@tweenjs/tween.js";
import {CapsuleCollider, RigidBody, useRapier} from "@react-three/rapier";
import {useEffect, useRef, useState} from "react";
import {usePersonControls} from "./hooks.js";
import {useFrame} from "@react-three/fiber";
import {Weapon} from "./Weapon.jsx";
const MOVE_SPEED = 5;
const direction = new THREE.Vector3();
const frontVector = new THREE.Vector3();
const sideVector = new THREE.Vector3();
const rotation = new THREE.Vector3();
export const Player = () => {
const playerRef = useRef();
const { forward, backward, left, right, jump } = usePersonControls();
const objectInHandRef = useRef();
const swayingObjectRef = useRef();
const [swayingAnimation, setSwayingAnimation] = useState(null);
const [swayingBackAnimation, setSwayingBackAnimation] = useState(null);
const [isSwayingAnimationFinished, setIsSwayingAnimationFinished] = useState(true);
const [swayingNewPosition, setSwayingNewPosition] = useState(new THREE.Vector3(-0.005, 0.005, 0));
const [swayingDuration, setSwayingDuration] = useState(1000);
const [isMoving, setIsMoving] = useState(false);
const rapier = useRapier();
useFrame((state) => {
if (!playerRef.current) return;
// moving player
const velocity = playerRef.current.linvel();
frontVector.set(0, 0, backward - forward);
sideVector.set(left - right, 0, 0);
direction.subVectors(frontVector, sideVector).normalize().multiplyScalar(MOVE_SPEED).applyEuler(state.camera.rotation);
playerRef.current.wakeUp();
playerRef.current.setLinvel({ x: direction.x, y: velocity.y, z: direction.z });
// jumping
const world = rapier.world;
const ray = world.castRay(new RAPIER.Ray(playerRef.current.translation(), { x: 0, y: -1, z: 0 }));
const grounded = ray && ray.collider && Math.abs(ray.toi) <= 1.5;
if (jump && grounded) doJump();
// moving camera
const {x, y, z} = playerRef.current.translation();
state.camera.position.set(x, y, z);
// moving object in hand for the player
objectInHandRef.current.rotation.copy(state.camera.rotation);
objectInHandRef.current.position.copy(state.camera.position).add(state.camera.getWorldDirection(rotation));
setIsMoving(direction.length() > 0);
if (swayingAnimation && isSwayingAnimationFinished) {
setIsSwayingAnimationFinished(false);
swayingAnimation.start();
}
});
const doJump = () => {
playerRef.current.setLinvel({x: 0, y: 8, z: 0});
}
const setAnimationParams = () => {
if (!swayingAnimation) return;
swayingAnimation.stop();
setIsSwayingAnimationFinished(true);
if (isMoving) {
setSwayingDuration(() => 300);
setSwayingNewPosition(() => new THREE.Vector3(-0.05, 0, 0));
} else {
setSwayingDuration(() => 1000);
setSwayingNewPosition(() => new THREE.Vector3(-0.01, 0, 0));
}
}
const initSwayingObjectAnimation = () => {
const currentPosition = new THREE.Vector3(0, 0, 0);
const initialPosition = new THREE.Vector3(0, 0, 0);
const newPosition = swayingNewPosition;
const animationDuration = swayingDuration;
const easing = TWEEN.Easing.Quadratic.Out;
const twSwayingAnimation = new TWEEN.Tween(currentPosition)
.to(newPosition, animationDuration)
.easing(easing)
.onUpdate(() => {
swayingObjectRef.current.position.copy(currentPosition);
});
const twSwayingBackAnimation = new TWEEN.Tween(currentPosition)
.to(initialPosition, animationDuration)
.easing(easing)
.onUpdate(() => {
swayingObjectRef.current.position.copy(currentPosition);
})
.onComplete(() => {
setIsSwayingAnimationFinished(true);
});
twSwayingAnimation.chain(twSwayingBackAnimation);
setSwayingAnimation(twSwayingAnimation);
setSwayingBackAnimation(twSwayingBackAnimation);
}
useEffect(() => {
setAnimationParams();
}, [isMoving]);
useEffect(() => {
initSwayingObjectAnimation();
}, [swayingNewPosition, swayingDuration]);
return (
<>
<RigidBody colliders={false} mass={1} ref={playerRef} lockRotations>
<mesh castShadow>
<capsuleGeometry args={[0.5, 0.5]}/>
<CapsuleCollider args={[0.75, 0.5]} />
</mesh>
</RigidBody>
<group ref={objectInHandRef}>
<group ref={swayingObjectRef}>
<Weapon position={[0.3, -0.1, 0.3]} scale={0.3} />
</group>
</group>
</>
);
}
В данной части мы реализовали генерацию сцены и перемещение персонажа. Также добавили модель оружия, анимацию отдачи при стрельбе и при бездействии. В следующей части мы продолжим дорабатывать нашу игру, добавляя новый функционал.