[Blender] 블렌더 유튜브 튜토리얼 무작정 따라하기 - 모델링

2024.10.17 - [게임만들기/Blender] - [Blender] 블렌더 유튜브 튜토리얼 무작정 따라하기 - 기본조작0. 공부 목표지난 무작정 따라하기에 이어서 2편으로 오브젝트의 크기, 모양, 형태, 질감 커스텀을 학

hostria.tistory.com

 

 

[Blender] 블렌더 유튜브 튜토리얼 무작정 따라하기 - 모델링2

2024.10.18 - [게임만들기/Blender] - [Blender] 블렌더 유튜브 튜토리얼 무작정 따라하기 - 모델링0. 공부 목표지난 모델링의 기본적인 용어와 오브젝트 크기, 모양, 형태, 질감에 대해 학습했다. 이번 무

hostria.tistory.com

0. 공부 목표

지난 모델링한 도넛에  뼈대나 밑바탕에 추가 오브젝트를 붙여서 디테일을 살리는 방법을 학습 했다. 이번 무작정 따라하기 튜토리얼에서는 Sculping 기초 스컬핑의 브러시 조작, 마스킹과 필터를 학습한다. 

 

 

Beginner Blender 4.0 Tutorial - Part 4: Sculpting

지난 모델링이 끝난 이후 Edit Mode에서 Vertex를 확인하니 도넛 오브젝트 내부로 들어가 있는데 이러면 렌더링에 겹쳐지는 문제가 생긴다. 이것을 해결하기 위해 Face Snaping을 활성화하고 Vertex를 살짝 클릭해서 도넛 표면에 붙여준다. 하지만 너무 비효율적인 방법이기 때문에 Shrink wrap Modifier을 사용하여 수정한다.

 

 

1. Sculping Mode

1) 브러시 조작

지금까지 사용했던 Object Mode, Edit Mode에 이어서 사용하는 스컬핑 모드를 사용한다. Vertex 단위의 편집한는 Edit Mode와 다르게 Sculping Mode에서는 작은 디테일을 간편하게 조작할 수 있는 도구를 제공한다.

 

 

아래 사진은 최대 강도 브러시로 2번 클릭한 결과인데 Sculping Mode는 Vertex 하나 하나 클릭해서 얻기에는 어려운 결과를 손쉽게 구할 수 있게 해준다. 

 

브러시는 종류 외에 브러시의 반경과 강도를 원하는 정도에 맞게 수정해주어야 한다. 반경 단축키는 [F] 클릭후 드래그로 조정할 수 있고 강도 단축키는 [Shift] + [F]로 조정할 수 있다.

세세한 브러시 조정을 위해서 Subdvision Surface를 level 1로 적용한다. 블러시를 실습할때는 꼭 저장을 하고 실습한다.

 

이후 Grab 브러시와 Inflating 브러시를 이용하여 지난 시간에 만든 액체가 흘러 굳어진 모양을 만들어 주었다.  

 

2) Mask 브러시

그림 그릴때 마스킹 테이프를 붙이는 거와 같이 다른 브러시의 효과를 받지 않도록 영역을 칠하는 브러시다.

3) Mesh 필터

Sculping 모드 브러시에 거의 마지막 카테고리에 있는 Mesh 필터는 매시에 다양한 필터를 적용한다.

 

 

 

저번 모델링 1편 처럼 기능을 직접 활용하면서 공부할 내용이 주를 이루었다. 내용적 특성때문에 글로 적기보다는 대부분 브러시로 기능을 살펴보는 시간을 가졌다.

개인적으로는 너무 디테일하게 수정할 수 있어서 금손들의 영상을 참고하는 것이 브러시의 특성을 더욱 잘 학습할 수 있다고 판단했다.


2024.10.18 - [게임만들기/Blender] - [Blender] 블렌더 유튜브 튜토리얼 무작정 따라하기 - 모델링

0. 공부 목표

지난 모델링의 기본적인 용어와 오브젝트 크기, 모양, 형태, 질감에 대해 학습했다. 이번 무작정 따라하기 튜토리얼에서는 모델링시 사용하는 모디파이어, 기본 조작을 통해 지난 시간 만든 도넛 위에 아이싱을 올리는 것을 목표로 한다. 이 튜토리얼을 통해서 뼈대나 밑바탕에 추가 오브젝트를 붙여서 디테일을 살리는 방법을 학습한다.

 

Beginner Blender 4.0 Tutorial - Part 3: Modelling the Icing

 

사실 지난 모델링 참고영상은 실제로 실습한 시간이 길고 어떻게 변하는지 살펴보다가 컴퓨터가 버티지 못하고 블렌더를 강종시켰다. 몇번 세이브를 날려보니 아이들 색연필 칠하듯 속성 슬라이드 바를 드래그질하면 처음부터 다시 할 수 있다는 것, 세이브를 잘하자를 다시한번 새겼다.

 

1. 오브젝트 모양 잡기

1) 오브젝트 복사

지난번 만들어둔 도넛에서 그 위를 덮는 아이싱을 모델링한다. 도넛 위에 덮어지는 형태이기 때문에 해당 도넛을 그대로 복사 [Shift] + [D]하여 기존 도넛 위에 살포시 겹쳐준다.

[Shift] + [D] - [Z]로 살짝 겹치게 올린다,

이후 구분하기 쉽게 각 Torus 오브젝트 이름을 Donut과 Icing으로 변경해 주었다.

 

2) 메시 수정하기

Icing 오브젝트를 모델링하기 위해서 Donut 오브젝트 선택 후 눈 모양 아이콘 클릭 또는 [Enter]로 안 보이게 세팅하고 [Tab]을 눌러 Edit Mode로 들어간다.

 

Icing의 형태는 도넛 위에 뿌려진 형태로 도넛 전체의 형태는 필요 없다. 따라서 도넛의 아래 부분의 Vertex를 전부 날려준다. 저번시간에 잠깐 소개했던 Edge 더블클릭을 포함해서 선택툴을 이용한 드래그 또는 [Ctrl] + [+]/[-] 해서 Vertex를 다수 선택할 수 있지만 이번에는 아래 영상과 같이 X-RAY Toggle을 이용해서 선택한다.

 

X-Ray Toggle의 단축키는 [Alt] + [Z]인데 해당 단축키는 Nvidia experience 오버레이 단축키와 겹치고 이 오버레이 단축키가 우선순위에 있어 Exprience 사용자는 둘 중의 하나 단축키 변경을 추천한다.

 

이후 [Delete]를 눌러 선택한 Vertex를 삭제한다.

 

이후 Donut 오브젝트를 켜고 오브젝트 모드에서 확인해 보니 아래처럼 아래 Donut과 Icing 오브젝트가 겹쳐지면서 무늬와 깜빡거림이 발생했는데 참고한 영상과 학습하면서 발생한 위치는 달랐지만 Z - Fighting이라는 3D 렌더링 현상이 발생했다. 해당 현상은 두 렌더링 대상 오브젝트의 Z-Buffer (오브젝트의 깊이를 추적) 값이 거의 비슷하거나 동일할 때 발생한다고 한다.

겹쳐진 부분의 오브젝트의 어느 부분을 어디까지 렌더링 해야 하는지 구분할 수 없을 때 겉면과 안쪽이 랜덤하게 렌더링 되면서 우리 눈에는 깜빡거리듯 보인다.

 

Z - Fighting

 

2. Solidify Modifier

1) 오브젝트를 두껍게(Thickness) 만들기 

 지난 시간 Subdivision Surfaces Modifier를 통해서 오브젝트의 각진 표면을 깎아보았다. Subdivision 방식은 메시의 수를 늘리지 않고도 자연스러운 부드러움을 표현하게 해 준다.

이번에 사용할 Solidify Modifier는 메시 자체의 Scale 변경 없이 오브젝트를 두껍게 만들어 준다. 

Modifier 생성

 

이 Modifier에는 Simple과 Complex 2가지 모드가 있는데, 매뉴얼에 따르면 뫼비우스의 띠, 클라인 병, 건축 벽 레이아웃 같은 경우가 아니면 연산 최적화를 위하여 Simple 모드를 사용하도록 권장한다. 

 

Simple 모드에서 수정할 수 있는 값은 

  • Offset : -1 ~ 1의 값을 가지고 있고 기존 메시의 내/외부부터의 방향을 정할 수 있다.
  • Thickness : Offset으로부터 두꺼워지는 정도이다.

지금 사용하고 있는 Torus 오브젝트 예시로 동굴이나 파이프 내부에 사용할 경우 Offset을 마이너스 쪽으로 활용하고 이번 따라 하기의 Icing은 외부로 나와야 하니 Offset을 1로, Thickness는 0.025로 설정하였다.



Offset 변경 +1 ~ -1

3. 오브젝트 모델링

1) Snaping

두꺼워진 오브젝트 때문에 Edit Mode에서 Vertex를 확인할 수 없는데 각 Modifier에 있는 아래와 같은 아이콘을 클릭하면 Edit Mode에서 Modifier 적용된 것을 끌 수 있다.

이제 액체가 흘러내리다가 굳어진 Icing을 표현하기 위해 Vertex를 수정해 주어야 한다. Donut의 표면에 흘러내리게 하기 위해 지난 시간에 활용한  Proportional Editing와 Snap을 사용한다.

Snap 설정

Snap으로 Donut 표면을 잘 따라가도 Vertex가 멀어지면 Icing이 안쪽으로 들어간다. 이 문제는 Object Mode에서  Subdivision Surface Modifier를 적용시켜 해결한다.

subdvision 적용 전과 후

 

2) Mesh 가다듬기

Icing은 Snap으로 잘 타고 내려왔지만 액체가 흘러내린듯한 모양이 아니다. 개인적으로는 카툰형 캐릭터 머리에 투구를 올린 것 같이 각이 져 있어서 Subdvision Surfaces Modifier를 새로 추가했고 Solidify에 Edge Data의 Inner Crease를 1로 수정해 주었다.

Solidify와 Subdivision 수정(좌)

3) 오브젝트 돌출 Extrusion

도넛에 아이싱 한 액체가 한 곳에 모였다가 한 방울 흘러내린 것처럼 하기 위해 Vertex를 추가한다. 아래와 같이 와이어 프레임 화면에서 Vertex 2개를 선택하고 [E]를 눌러 Extrusion 한다. 적당한 길이로 추가하면 Modifier가 적용된 모양으로 Vertex가 추가되면서 흘러내린 모양이 완성된다.

 

Vertex 2개 돌출

 

돌출 결과

 

2024.10.16 - [게임만들기/UNITY] - [UNITY] 첫 3D 팽이 게임 만들기 - 각저항 분석 후 팽이 '잘'돌리기

0. 개발 목표

1) 필드에 벽을 세우고 팽이가 벽충돌시의 물리 적용하기

  • 기존 개발 방식은 Raycast의 거리가 땅을 검사하고 거리가 줄어드면 각 저항을 증가시켜 팽이를 멈춤
  • 벽과 땅을 구분하고 벽에는 Bounce + 이동방향 변경, 땅에는 각저항 증가

2) Bounce  구성하기

  • 벽에 팽이 Collider가 충돌하면 OnCollisionEnter()를 통해 진행방향 벡터 변경

1. 테스트 환경 구성

1) 기존 필드에 팽이 대신 테스트용 오브젝트 추가

 기존 팽이는 비활성화하고 3D 오브젝트 구에 RigidBody와 Wall, Ground에 Tag 추가 카메라 Follow 설정 변경

벽은 Kinematic를 활성화하고 충돌과 관련한 모든 오브젝트에 Collision Detection을 Continuous로 변경하여 오브젝트가 프레임 차이시간에 뚫고 나가는 것을 방지.

테스트 환경 구성

2)  ColliderTest 스크립트 임시 추가

  1. 상하좌우 입력받고 벽에 붙을때 Log 출력 추후 다른 오브젝트 충돌과도 고려하여 switch 문으로 작성하고 오브젝트의 Tag를 비교하여 각 오브젝트에 대한 함수 실행
        private void OnCollisionEnter(Collision other)
        {
            switch(other.gameObject.tag){
                case "Floor":
                    Debug.Log("땅");
                break;
    
                case "Wall":
                    Debug.Log("벽");
                    break;
                }
        }

    'Hostria'에서 업로드한 동영상

     



  2. 벽과 충돌시 입력된 벡터의 반사 벡터 방향으로 충돌 오브젝트에 AddForce() 하는 함수 구현.
    private void OnCollisionEnter(Collision other)
    {
        switch(other.gameObject.tag){
            case "Floor":
                Debug.Log("땅");
            break;

            case "Wall":
                Vector3 normal =other.contacts[0].normal;
                Debug.Log("벽");
                DoBounce(normal);
                break;
            }
    }
    void DoBounce(Vector3 normal)
    {
        Debug.Log("Dobounce = " + Vector3.Reflect(inputVec,normal) );
        rigid.AddForce(Vector3.Reflect(inputVec,normal)*speed, ForceMode.VelocityChange);

    }

 

Vector에서 Reflect 매게변수는 입사 벡터, 법선벡터로 벽과 충돌시점의 포인트의 법선벡터 인 normal을 이용하여 DoBounce를 수행한다. 보기 쉽게 임시적으로 반사벡터에 100f 의 힘을 추가했다.

using System.Collections;
using System.Collections.Generic;
using UnityEditor.Callbacks;
using UnityEngine;

public class ColliderTester : MonoBehaviour
{
    public float speed = 10f;
    public float transX;
    public float transY;
    Vector3 inputVec;
    Rigidbody rigid;

    private void Awake() 
    {
        rigid = GetComponent<Rigidbody>();    
    }
    private void Update() 
    {
        transX = Input.GetAxisRaw("Horizontal");
        transY = Input.GetAxisRaw("Vertical");
        inputVec = new Vector3(transX, 0, transY).normalized  * Time.deltaTime * speed ;

    }
    private void FixedUpdate() 
    {
        transform.Translate(inputVec);

    }
    private void OnCollisionEnter(Collision other) {
        switch(other.gameObject.tag){
            case "Floor":
                Debug.Log("땅");
            break;

            case "Wall":
                Debug.Log("벽");
                Vector3 vector3 =other.contacts[0].normal;
                DoBounce(vector3);
                break;
            }
    }
    void DoBounce(Vector3 inNor){
        Debug.Log("Dobounce = " + Vector3.Reflect(inputVec,inNor) );
        rigid.AddForce(Vector3.Reflect(inputVec,inNor)*100,ForceMode.VelocityChange);

    }
}

 

 

 

 

테스트를 하고나니 오래 돌게할려고 꺼둔 중력이 문제다.

벽에 충돌하는 순간 팽이에 Y 값이 충돌한 벡터 만큼 변경되어 중구난방으로 튀어버리는 것.

모양이 큐브이거나 구체인 경우 덜컹거리는(?) 느낌의 조정으로 다시 회전시키면 괜찮은데 팽이가 진행방향과 다른 방향의 기울기로 변경되고 중력이 꺼지면서 매우 이상하게 회전했다. 다음에는 해당 내용을 좀 연구해서 충돌을 추가하려 한다.

2024.10.17 - [게임만들기/Blender] - [Blender] 블렌더 유튜브 튜토리얼 무작정 따라하기 - 기본조작

0. 공부 목표

지난 무작정 따라하기에 이어서 2편으로 오브젝트의 크기, 모양, 형태, 질감 커스텀을 학습한다.

오늘의 참고 영상 - 도넛 모양 모델링 하기

1. 오브젝트 크기와 모양 잡기

1) 오브젝트 Segments 와 Dimensions

도넛 모양을 만들기 위해서 토러스 메쉬를 추가한다. 단축키는 [Shift] + [A]

 

오브젝트를 만들고 나면 오브젝트 초기상태를 확정하는 창이 좌하단에 나타난다.

다른 행동을 하면 해당 창이 없어진다.

다시 불러오고 싶다면 [F9] 를 눌러서 불러온다.

 

토러스 메쉬를 추가했으니 Add Torus 클릭. 각 오브젝트에 Segments를 나눌 수 있는데 너무 많이 나누게 되면 엄청 세밀하지만 세밀한 만큼 렌더링과 작업 시간에 불이익이 있다고 한다.세그먼트를 잘 조정해서 적당한 폴리곤을 형성해야 한다. 

 

뚱도넛

2) Smooth Shading

스무스 셰이딩은 전체 오브젝트의 폴리곤을 부드럽게 만들어주는 기능이다.  오브젝트 전체를 부드럽게 만들어주기 때문에 각진면을 유지하기 위해서는 추가 기능을 사용해야 한다.

 

 

Smooth Shading

 

2) Subdivision Modifier

Smooth Shading을 적용해도 오브젝트에 폴리곤이 보인다. 이를 수정하기 위해 Modifier 랜치 모양 아이콘을 클릭하고 Add Modifier - Generate - Subdvision Surface를 추가하면 더 부드럽게 만들어 준다.

 

2. Edit Mode로 오브젝트 디테일 잡기 

Edit Mode는 Blender 좌상단의 드롭다운 리스트에서 선택하거나 Modeling 버튼을 통해 Edit Mode로 변경할 수 있다.

단축키는 [Tab]으로 오브젝트 모드와 에디트 모드로 변경한다.

오브젝트에 생성된 '점'은 Vertex라고 한다. 각 버텍스를 선택해서 (Edge, 선)와 (Face, 면)를 선택해 디테일하게 수정할 수 있다.

하지만, 복잡한 형태의 모델링에서 점을 한 개 한개 수정하면서 작업하면 비효율적 이기 때문에 블렌더에서는 Proportional Editing이라는 기능을 제공한다.

Edit Mode 첫화면

에딧모드에서 우상단의 원 아이콘을 클릭해서 사용할 수 있고 Vertex 선택 후 [G] 그리고 [스크롤 업 / 다운]을 통해 Vertex가 영향받는 범위를 정할 수 있다. 

Proportional Editing

Modeling은 Vertex의 변경을 통해 오브젝트의 모양을 만드는 방법이므로, 다양한 Vertex 관련 기능을 제공한다.

Vertex 선택 후 우클릭

 

0. 3D 모델링 공부하기

1) 공부 목적

  • 게임 만들기를 하다 보니 이왕이면 보기 좋은 게 당연하다는 생각이 들고
  • 기본 도형들로 만들려고 하니 속터지고 그렇다고 디자인 관련 전공도 아니고
  • 취미에 낙서도 없는 사람이라, '미술'은 전문가들의 영역으로 느껴지는데

이러한 생각들이 에셋 장터의 금손들은 어떻게 이렇게 만들지?라는 생각이 들었다.

그럼 궁금하면 해봐야지.  에셋을 살려고 해도 리깅이 있고 없고 폴리곤이 어쩌고 정성 들여 만든 건지 대충 만든 건지!

3D 모델링 프로그램 튜토리얼을 따라 하면서 용어와 기술들을 알아보려고 한다.

 

좋은 에셋을 골라내려면 나도 공부해 봐야겠다는 생각이 들었다.

2) 공부 방법

참고 영상 : Blender Tutorial for Compete Beginners - Part 1 by Blender Guru

해당 영상에서 사용된 모델링 용어, 기술 블렌더 단축키 등에 대하여 직접 따라 하면서 정리

1. 블렌더 오브젝트 추가하기

첫 프로젝트 생성

처음 일반 프로젝트를 생성하고 마주한 화면에서는 Camera, Cube, Light 3가지의 오브젝트를 볼 수 있다. 프로젝트 생성 시 기본적으로 포함되며 오른쪽 위의 씬 컬렉션에서 확인할 수 있다.

시작 목표는 오브젝트 만들기이니 Cube는 [Delete] 키로 삭제한다.

큐브 삭제

블렌더에서 제공하는 오브젝트 생성하는 방법은 2가지

  • Layout 탭 - Add - 오브젝트 타입 선택 - 모양 선택
  • Add 단축기 사용 : [shift] + [a]

[Shift] + [A] - Mesh의 원...숭이?

원숭이가 너무 궁금해서 눌러보니 수잔이란 이름의 갸름한 원숭이 머리가 나타났다.

어째서 이런 게 있는지 궁금했는데 3D 소프트웨어마다 복잡한 형태의 오브젝트를 가지고 있고 원숭이가 블렌더의 유용한 테스트용 오브젝트라고 한다.

Jay and Silent Bob Strikes Back 출신의 원숭이 수잔

 

2. Navigation 시점 조작하기

 1) 시점 Orbiting

  • 마우스 [휠 클릭] + 드래그 : 자유 시점 변경
  • [Alt] + [휠 클릭] + 드래그 : 드래그한 방향의 축으로 시점을 변경.
  • 우상단 3차원 축 클릭 또는 드래그 : ± X, ± Y, ±Z 축의 시점으로 변경. 단축키는 [Numpad 1,3,7]
  • [Alt] + [클릭]  : Edit탭 - Preferences - Input에서 Emulate 3 Button Mouse 활성화하여 사용

 2) 시점 ZoomIn, ZoomOut

  • [휠 스크롤]
  • [Ctrl] + [휠 클릭] + 위/아래 드래그 
  • 우상단 돋보기 클릭 후 위/아래 드래그

ZoomIn and ZoomOut

 2) 시점 Panning

  • [Shift] + [휠 클릭] + 드래그 
  • 우상단 손바닥 클릭후 드래그

3. Viewport Shading

뷰포트 셰이딩은 씬 화면에서 오브젝트 보기 모드이다. 각 모드는 모델링이나 조명에 적합한 셰이딩을 선택할 수 있게 해 준다. 

뷰포트셰이딩은 

Viewport shading 모드

 

각 모드는 와이어프레임, 솔리드, 매테리얼 미리 보기, 렌더리드 모드가 있으며 [Z]와 [Shift] + [Z]로 모드를 변경할 수 있다.

  • [Z] : 마우스 커서 위치에 파이메뉴

단축키 [Z]

  • [Shift] + [Z] : 현 셰이딩 모드 ↔ 와이어 프레임 모드 전환.
뷰포트셰이딩 전환과 단축키 실습

 

4. Objects Moving, Rotate  오브젝트 이동과 회전

블렌더에서는 유니티의 Transform같이 수치를 입력하거나 이동 툴과 회전 툴을 선택하여 축 방향으로 오브젝트를 이동 회전시킬 수 있다.

그리고 추가적으로 오브젝트 그랩키[G] (이동) / [R] (회전)를 이용할 수 있다.

  • 오브젝트 선택 후 [G] / [R]로 그랩 후 : 좌클릭으로 변경, 우클릭으로 변경 취소
  • 오브젝트 선택 후 [G] / [R] 로 그랩 후 : [X], [Y], [Z] 버튼으로 해당 축 방향으로 이동 / 회전
  • 오브젝트 선택 후 [G] / [R] 로 그랩 후 : [휠 클릭] 3 축 방향 중 가까운 축에 붙음
  • [Shift] + [Space], [G] / [R] : 이동 회전 툴 선택

 

그랩으로 오브젝트 이동

 

5. 카메라 이동하기

카메라는 Transform으로 수정할 수 있지만, 2.Navigation에서 조작한 방법으로 카메라가 움직이지는 않는다.

앞서 학습한 방법으로 카메라를 이동시키기 위해서는 카메라 오브젝트 선택 - [N] - View 박스 선택 - View Lock에서 Camera to View를 활성화시켜주면 카메라를 마우스와 키보드 단축키로 이동할 수 있다.

 

또한 카메라 시점으로 전환은 씬화면 우상단의 카메라 아이콘 클릭 또는 [Numpad 0]으로 전환할 수 있다.

카메라 뷰 설정

 

6.  매테리얼 변경하기

원숭이의 새로운 매테리얼을 만들기 

오브젝트 선택 후 매테리얼 아이콘 선택 -> New 버튼 클릭 후 색상, 특성을 조절

작성자 컴퓨터는 슬라이드 바 드래그로 변경 때마다 굉음을 내서 얌전히 숫자 입력하면서 변경해 보았다.

매터리얼 생성하기

 

뷰포트셰이딩 모드 매터리얼 미리 보기를 통해 매터리얼이 적용되는 것을 실시간으로 확인할 수 있고 미리보기 탭에서 다양한 모양의 오브젝트에 매터리얼이 적용된 모습을 확인할 수 있었다.

매끈하게 만들었더니 풍경이 반사되는 것을 발견

 

다음 시간에는 Basic Modelling 편으로 오브젝트 조작을 따라 해보겠다.

0.AddTorque()를 이용해서 팽이 Rotation 구현하기

2024.10.15 - [UNITY] - [UNITY] 첫 3D 팽이 게임 만들기 - 팽이 프로토타입 만들기

 

[UNITY] 첫 3D 팽이 게임 만들기 - 팽이 프로토타입 만들기

유니티에서 게임 오브젝트를 만들면서 기본적으로 제공되는 도형들이 많이 불편했다. Assets을 찾아봤지만 교통 꼬깔형태의 Cone이 대부분이고 폴리곤 형태에서 원뿔에 가까운 형태는 직접 모델

hostria.tistory.com

어제 글을 작성하고 인스펙터 창에 이것저것 실험해 보았다.

회전력을 크게 주어도 생각한 만큼 회전이 나오지 않았다.

이에 최대한 고정프레임에서 가장 많이 그리고 빠르게 회전하는 팽이를 만들기 위해 물리 연산에 변수들을 정리 분석해 보았다.

 

1. AddTorque() 분석을 위한 임시 코드 수정

using System.Collections;
using System.Collections.Generic;
using System.Numerics;
using UnityEngine;
using Vector3 = UnityEngine.Vector3;

public class SpinController : MonoBehaviour
{
    [SerializeField] private float initRotateSpeed; // 초기회전력
    [SerializeField] private float speed;   //이동속도
    private Vector3 inputVec; //상대 이동벡터

    [SerializeField]private float speedDamper; //속도 감소률
    [SerializeField]private float rotateSpeed; //회전력
    private Rigidbody rb;
    
    private void Awake() {
        Application.targetFrameRate = 60; //고정프레임 60FPS 
        rb = GetComponent<Rigidbody>();
        initRotateSpeed = 1f;
        speed = initRotateSpeed*0f;
        speedDamper = 0.99f;
        rotateSpeed = initRotateSpeed;

    }

    private void Update()
    {
        //float inputX = Input.GetAxis("Horizontal");
        
        //inputVec = new Vector3(inputX, 0, 0) * speed *Time.deltaTime;

    }
    void FixedUpdate()
    {
        //transform.Translate(inputVec, Space.World);
        rb.AddTorque(transform.up * initRotateSpeed * Time.fixedDeltaTime);
    }

}

이번 분석을 위해 회전과 관련 없는 코드는 삭제 수정했으며, 고정 프레임을 추가하였다.

고정 프레임을 작성한 이유는 매번 돌릴 때마다 200~350 FPS 차이의 변동으로 회전과 이동이 변했기 때문이다.

 

 

[결과] 금방 땅에 쓰러지고 남은 회전으로 굴러다니면서 원뿔이 탈출함

하도 바닥에서 뺑뺑 돌기만 해서 바닥에서 얼마나 굴렀는지 확인하는 스크립트를 추가했다.

실험. 초기회전력 100f

 

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class CheckerLogger : MonoBehaviour
{

    private void OnTriggerEnter(Collider other) {
        if(other.CompareTag("Floor")){
                    Debug.Log("쓰러짐");
            return;
        }

    }
}

 

2.RigidBody의 AddTorque(Vector3 vector, ForceMode force);

파라미터 vector = 회전축과 회전력,

파라미터 force =  RigidBody의 1. 질량(M), 2. 프레임당 시간(DT), 3. 토크(Tor)의 계산방식을 결정한다.

force의 변수

ForceMode.Force = Tor*DT/M

ForceMode.Acceleration = Tor * DT

ForceMode.Impulse =Tor/M

ForceMode.VelocityChange = T

 

ForceMode.Force

 

 

RigidBody의 Mass(질량), Drag(저항), Angular Drag(각 저항)은 모두 물리연산에 영향을 준다. 

 

이번 팽이 돌리기에 활용 변수는 각저항을 이용하고자 한다.

 

값을 변경하면서 실험해 본결과 의미 있는 내용 몇 가지를 추려보았다.

 

[실험 1] M:D:A  = 1 : 0 : 0.05

 

[결과 1]

약 2바퀴 회전 후 쓰러짐, 이후 남은 회전에 의해 밖으로 탈출.

쓰러지는 것을 체크할 필요가 있음.

 

[실험 2] M:D:A = 1 : 0 : ~50

팽이가 멈출 때까지 각저항을 증가시킴.

 

[결과 2]

Angular Drag = 50 될 때부터 회전 멈춤.

각저항을 점진적으로 증가시켜서 팽이가 쓰러지고 완전히 멈추게 해야 함.

 

착지하자마자 각 저항이 오른다면 회전력 유지 못함.

따라서

  1. rotationSpeed와 initRotateSpeed를 이용해서 회전 유지 시간을 확보 중력을 끄고 키면서 오래 돌아가는 것처럼 보이게 설정.
  2. Raycast를 이용해 바닥을 체크하고 쓰러질 때의 거리를 측정하여 각저항과 중력을 통제. 
using System.Collections;
using System.Collections.Generic;
using System.Numerics;
using JetBrains.Annotations;
using Unity.VisualScripting;
using UnityEngine;
using Vector3 = UnityEngine.Vector3;

public class SpinController : MonoBehaviour
{
    [SerializeField] private float initRotateSpeed; // 초기회전력
    [SerializeField] private float speed;   //이동속도
    private Vector3 inputVec; //상대 이동벡터

    [SerializeField]private float speedDamper; //속도 감소률
    [SerializeField]private float rotateSpeed; //회전력
    private Rigidbody rb;
    private RaycastHit hit;
    private bool isGrounded;
    private bool isStoped;

    private void Awake() {
        Application.targetFrameRate = 60; //고정프레임 60FPS 
        rb = GetComponent<Rigidbody>();
        initRotateSpeed = 100f;
        speed = initRotateSpeed * 0.1f;
        speedDamper = 0.99f;
        rotateSpeed = initRotateSpeed;
        isGrounded = false;
        isStoped = false;
    }

    private void Update()
    {
        StateCheck();
        
        float inputX = Input.GetAxis("Horizontal");
        
        inputVec = new Vector3(inputX, 0, 1) * speed * Time.deltaTime;
    }
    void FixedUpdate()
    {
        transform.Translate(inputVec, Space.World);
        rb.AddTorque(new Vector3 (0,100000,0) * rotateSpeed, ForceMode.VelocityChange);

    }

    public void StateCheck(){
        if(isGrounded){
            if (Physics.Raycast(rb.transform.position, transform.right, out hit))            {
                Debug.DrawRay(rb.transform.position, transform.right * hit.distance, Color.red);
                if(hit.distance < 1f && rb.angularDrag < 50){
                    Debug.Log("쓰러지이인다");
                    rb.angularDrag += 0.65f;
                }
                if(rb.angularDrag >=50){
                    Debug.Log("멈춤");
                    isGrounded = false;
                    isStoped = true;
                }
             else
                {
                Debug.DrawRay(rb.transform.position, -transform.up * 1000f, Color.red);
                }
            }
        }
      
        if(!isStoped){
            if (Physics.Raycast(rb.transform.position, -transform.up, out hit))
            {
                Debug.DrawRay(rb.transform.position, -transform.up * hit.distance, Color.red);
                if(hit.distance < 0.25f){
                    Debug.Log("착지");
                    isGrounded = true; 
                    rb.useGravity = false;
                }
            }
            else
            {
                Debug.DrawRay(rb.transform.position, transform.right * 1000f, Color.red);
            }
            if(rotateSpeed < 1){
                rb.useGravity = true;
            }else{
                rotateSpeed *= speedDamper;
                speed *= speedDamper;
            }
        }
        else {
            speed = 0;
            rotateSpeed =0;
        }

    }
}

 

실행 결과 어제보다는 쪼금 더 잘 도는 거 같다.

 

0.유니티 프로빌더와 원뿔같은 Cone 만들기

유니티에서 게임 오브젝트를 만들면서 기본적으로 제공되는 도형들이 많이 불편했다.

 

Assets을 찾아봤지만 교통 꼬깔형태의 Cone이 대부분이고 폴리곤 형태에서 원뿔에 가까운 형태는 직접 모델링해야 한다.

 

Blender 써볼까? 했지만 우선 '3D 게임 공부!' 목적이므로 프로빌더 패키지를 활용하기로 했다.

 

1. 프로빌더 설치.

Unity에서 제공하는 3D 모델링 패키지로 Package Manager 에서 바로 설치할 수 있다.

 

[Tools] -> [ProBuilder] -> [ProBuilderWindow] 로 바로 윈도우를 추가해 주었다.

프로빌더 설치후 윈도우 추가

 

2.원뿔 만들기

New Shape를 선택해서 씬창에 생긴 도구를 이용해 Cone을 만들어 보았다.

Create Shape와 Cone 선택

 

Cone을 선택하고 [CTRL] + [Shift] 버튼을 이용해서 원뿔에 가까운 오브젝트를 생성해 주었다. 딸깍!x3

Side Count는 64로 설정하였다.

Cone 오브젝트 생성

팽이처럼 뒤집기 위해서 Transform을 수정하기 위해 오브젝트를 클릭했더니 중심이 아닌 피봇을 잡고 있었다. ProBuilder 창에서 [Center Pivot]을 클릭하여 축을 오브젝트 중앙으로 변경해 주었다.

피봇 설정전
피봇 설정 후

팽이를 돌리기 위해 땅과 색을 추가했다.

땅과 색 추가
생성한 GameObjects

 

3.RigidBody 와 Collider 컴포넌트

Cone은 Spinner로, Plane으로 만든 땅은 Ground로 이름을 변경해 주었다. 둘 다 ProBuilder로 생성하면서 Mesh Collider는 자동으로 추가되어 있다.

 

[Spinner]

Transform.position (0,1,0)

Mesh Collider - [Convex] 설정

RigidBody - 추가

 

[Ground] 

대충 Spinner 밑에 Z축으로 길게

 

4.스크립트 추가. SpinController.cs

구현하고자 하는 기능은 팽이가 회전하면서 이동하고 점점 힘을 잃어 바닥에 구르는 것.

using System.Collections;
using System.Collections.Generic;
using System.Numerics;
using UnityEngine;
using Vector3 = UnityEngine.Vector3;

public class SpinController : MonoBehaviour
{
    [SerializeField] private float rotateSpeed; // 회전력
    [SerializeField] private float speed;   //이동속도
    private Vector3 playerVec; //Player 절대 이동 벡터
    private Vector3 inputVec; //상대 이동벡터

    private float speedDamper; //속도 감소
    private Rigidbody rb;
    
    private void Awake() {
        rb = GetComponent<Rigidbody>();
        rotateSpeed = 10000;
        speed = 100;
        speedDamper = 0.999f;
    }
    void Start()
    {
        
    }
    private void Update() {
        float inputX = Input.GetAxis("Horizontal");
        float inputY = Input.GetAxis("Vertical");

        inputVec = new Vector3(inputX, 0, inputY)*Time.deltaTime;

        rotateSpeed *= speedDamper;
        speed *=speedDamper;
    }
    void FixedUpdate()
    {
        transform.Translate(inputVec*speed,Space.World);
        rb.AddTorque(new Vector3(0,1,0)*rotateSpeed*Time.fixedDeltaTime,0);
    }

}

 

Spinner에 스크립트 추가 후 테스트 해보았다.

잘 돈다.

이동도 구현했으나 뭔가 팽이스럽지 않아서 다음시간에 추가해보고자 한다.

+ Recent posts