원본: https://blog.unity.com/technology/enter-play-mode-faster-in-unity-2019-3

관련 항목

Unity 2019.3에서 더 빨리 Play 모드로 들어가세요.

Play 모드는 Unity를 즐겁게 만드는 핵심 요소입니다. 그러나 프로젝트가 복잡해지면서 시작하는 데 시간이 걸릴 수 있습니다. Play 모드로 들어가고 나갈 수 있는 속도가 빠를수록 변경 사항을 더 빨리 수정하고 테스트할 수 있습니다. 그렇기 때문에 실험적인 기능으로 Unity 2019.3 beta 에서 설정 가능한 Enter Play Mode를 도입하고 있습니다.

현재 Editor에서 Play Mode(재생 모드)를 시작하면, Unity는 스크립팅 상태(도메인 다시 로드)를 재설정하고 씬(scene)을 다시 로드하는 두 가지 작업을 수행합니다. 이 작업은 시간이 걸리고, 프로젝트가 복잡해질수록 재생 모드에서 새로운 변경 사항을 테스트하기 위해 더 오래 기다려야 합니다. 그러나 Unity 2019.3 beta부터는 "도메인 다시 로드" 및 "씬 다시 로드" 작업 중 하나 또는 둘 다를 비활성화할 수 있습니다.

테스트 결과에 따르면, 프로젝트에 따라 대기 시간을 최대 50-90%까지 절약할 수 있습니다.

 

 

우리는 AA 타이틀(Production title), 우리의 FPS 샘플, 메가시티, 그리고 빈 프로젝트로 구성 가능한 플레이 모드를 테스트 했습니다. 그래프는 편집기가 재생 모드로 전환하는 데 걸린 시간(초)을 나타냅니다. 해당 숫자가 작을수록 좋습니다.

File > Project Settings > Editor에서 Enter Play Mode Options를 활성화하면 Domain을 다시 로드하고 Scene을 다시 로드하는 옵션을 사용할 수 있습니다. 좀 더 상세한 내용을 원하면 어떻게 플레이 모드를 설정하나?를 확인하세요.

 

이 옵션을 사용하면 코드 변경이 없을 때 플레이 모드 시작 프로세스에서 Domain 및/또는 Scene 다시 로드를 비활성화할 수 있습니다. 플레이 모드에 들어가기 전에 게임 상태를 재설정하려면 API와 Callback을 통해 이 기능에 액세스할 수도 있습니다.

아래 표는 도메인 리로드 및 Scene 리로드를 비활성화 하기 전과 후의 플레이 모드 시작 프로세스를 보여 줍니다:

 

Unity가 플레이 모드로 전환할 때 수행하는 프로세스에 대한 자세한 내용은 문서를 참조하십시오.

이 기능은 현재 실험적이며 모든 Unity 패키지들이 비활성화된 도메인 및 Scene 리로드와 함께 작동하도록 검증되지는 않습니다. 문제가 생기면 포럼에서 알려주세요!

도메인 리로드를 사용하지 않도록 설정한 경우 스크립트를 올바르게 수정하는 방법

보시다시피 도메인 리로드를 방지하는 것은 매우 간단하지만 비용이 많이 듭니다. 플레이 모드가 시작될 때 스크립트 상태가 올바르게 재설정 되도록 스크립트의 정적 필드 및 정적 이벤트 핸들러를 조정해야 합니다.

다음 코드 예제에는 플레이어가 점프 버튼을 누를 때 올라가는 카운터가 있습니다. 도메인 리로드가 활성화된 경우, 플레이 모드로 들어갈 때 카운터가 자동으로 0으로 재설정 됩니다. 도메인 리로드를 사용하지 않도록 설정한 후에는 카운터가 재설정 되지 않고 값이 플레이 모드에 있거나 플레이 모드에서 해제됩니다. 즉, Editor에서 프로젝트의 두 번째 실행에서 카운터가 이전 실행에서 변경된 경우 카운터가 0이 아닐 수 있습니다.

public class StaticCounterExample : MonoBehaviour
{
    // 도메인 리로드를 사용하지 않도록 설정한 경우 이 카운터가 0으로 재설정 되지 않습니다.
	static int counter = 0;

	// Update is called once per frame
	void Update()
	{
    	if (Input.GetButtonDown("Jump"))
    	{
        	counter++;
        	Debug.Log("Counter: " + counter);
    	}
	}
}

[RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)] 특성을 사용하고, 값을 명시적으로 재설정하여 도메인 리로드가 비활성화된 경우 카운터가 올바르게 재설정되도록 합니다. 예:

using UnityEngine;

public class StaticCounterExampleFixed : MonoBehaviour
{
	static int counter = 0;

	[RuntimeInitializeOnLoadMethod(RuntimeInitializeLoadType.SubsystemRegistration)]
	static void Init()
	{
    	Debug.Log("Counter reset.");
    	counter = 0;
	}

	// 프레임당 한 번씩 업데이트가 호출됩니다.
	void Update()
	{
    	if (Input.GetButtonDown("Jump"))
    	{
        	counter++;
        	Debug.Log("Counter: " + counter);
    	}
	}
}

도메인 리로드를 사용 불가능으로 설정한 후 플레이 모드를 종료할 때 Unity는 정적 이벤트 핸들러에서 메서드 등록을 취소하지 않습니다. 정적 이벤트 핸들러에 메서드를 등록하는 코드가 있는 경우 이로 인해 문제가 발생할 수 있습니다. 예를 들어, Editor에서 프로젝트의 첫 번째 Play(재생)에서 메소드는 정상으로 등록됩니다. 그러나 프로젝트의 두 번째 재생에서는 이러한 메소드가 첫 번째 메소드 외에 두 번째로 등록되므로 이벤트가 발생할 때 두 번 호출됩니다.

다음 코드는 정적 이벤트 핸들러에 Application.quitting을 등록합니다:

using UnityEngine;
public class StaticEventExample : MonoBehaviour
{
	void Start()
	{
    	Debug.Log("Registering quit function");
    	Application.quitting += Quit;
	}

	static void Quit()
	{
    	Debug.Log("Quitting!");
	}
}

도메인 리로드를 사용하지 않도록 설정한 경우 위의 예에서는 플레이 모드에 들어갈 때마다 '종료' 메서드를 다시 추가합니다. 이렇게 하면 플레이 모드를 종료할 때마다 "종료" 메시지가 추가로 표시됩니다.

[RuntimeInitializeOnLoadMethod] 특성을 사용해, 메서드를 두 번 추가하지 않도록 명시적으로 등록 취소합니다:

using UnityEngine;
public class StaticEventExampleFixed : MonoBehaviour
{
	[RuntimeInitializeOnLoadMethod]
	static void RunOnStart()
	{
    	Debug.Log("Unregistering quit function");
    	Application.quitting -= Quit;
	}

	void Start()
	{
    	Debug.Log("Registering quit function");
    	Application.quitting += Quit;
	}

	static void Quit()
	{
    	Debug.Log("Quitting the Player");
	}
}

 

우리의 문서에서 도메인 리로드가 비활성화 된 경우 올바르게 수행되도록 스크립트를 수정하는 방법에 대한 자세한 내용을 참조하십시오.

Asset Store

인기 있는 Asset Store packages가 비활성화된 도메인 및 Scene 리로드와 함께 작동하는지 확인하려고 합니다. 프로젝트에서 발생하는 문제를 Asset packages의 게시자에게 보고하여 도움을 받을 수 있습니다.

Join Unity 2019.3 beta!

현재 프로젝트가 플레이 모드로 전환되는 속도가 느린 경우 이 기능을 사용하면 작업 속도가 상당히 빨라질 것으로 생각됩니다. Unity 2019.3 beta에 가입하여 사용해 보십시오. 포럼에서 귀하의 의견을 들을 수 있기를 기대합니다! 이 기능은 실험적이기 때문에 사용자의 요구에 맞게 구성하는 데 도움이 될 수 있습니다. 우리는 특히 당신이 마주치는 문제에 대해 듣기를 고대하고 있습니다.

이미 이 기능을 테스트하고 귀중한 피드백을 제공함으로써 전체 커뮤니티를 도와준 포럼 사용자 @Sini, @chrisk, @Peter77 및 @Baste에게 큰 감사를 드립니다.

블로그 이미지

RIsN

,

원본: https://github.com/Unity-Technologies/EntityComponentSystemSamples/blob/master/DOTS_Guide/ecs_tutorial/README.md

변환 설정

베이킹 파이프라인(baking pipeline)을 적절히 설정

이 튜토리얼의 나머지 부분에서는 Preferences > Entites의 "Scene View Mode" 항목이 "Runtime Data"로 되어 있다고 가정합니다:


블로그 이미지

RIsN

,

원본: https://github.com/Unity-Technologies/EntityComponentSystemSamples/blob/master/DOTS_Guide/ecs_tutorial/README.md

프로젝트 설정

새 프로젝트를 만들고, 필요한 패키지를 설치하고, 설정을 조정.

  1. 3D(URP) 템플릿을 사용하여 Unity Hub에 새 프로젝트를 만듭니다. 템플릿을 사용해 본 적이 없으면 결국 다운로드해야 합니다. 올바른 에디터 버전을 사용하고 있는지 확인해주세요.
  2. 프로젝트가 로드되면 "URP Empty Template" 인스펙터에서 "Remove Readme Assets" 단추를 클릭합니다. 이렇게 하면 이 자습서에 필요하지 않은 "Assets/TutorialInfo" 폴더가 제거됩니다. 이 인스펙터는 기본적으로 선택되어 있어야 하지만 혹시 선택되지 않았다면 프로젝트 창에서 Assets/Readme.asset를 선택합니다.
  3. 이 튜토리얼에서는 패키지 하나만 명시적으로 설치해야 합니다. 다른 패키지는 종속되어 프로젝트에 포함됩니다. Window > Package Manager로 이동하여 왼쪽 상단의 + 버튼을 클릭하고 "Add package by name"을 선택합니다. "Name" 필드를 "com.unity.entities.graphics"으로 설정하고 "Version" 필드를 비워둔 다음 "Add" 버튼을 클릭하고 패키지와 종속 패키지가 설치될 때까지 기다립니다.
  4. Edit > Project Settings > Editor에서 "Enter Play Mode Options"을 사용 가능으로 설정하고, 하위 선택사항 중 "Reload ___" 관련은 선택하지 않은 상태로 둡니다. 이러한 옵션의 기능에 대한 자세한 내용은 다음 링크를 참조 해주세요.
    이러한 설정이 static 필드(블로그 게시물 등 참조)의 사용에 미치는 영향에 주의 해주세요.
  5.  
  6. 프로젝트 창에서 Assets 폴더의 다음 목록에서 누락된 폴더를 생성 해주세요.
    • Prefabs
    • Scenes (already created)
    • Scripts/Aspects
    • Scripts/Authoring
    • Scripts/Components
    • Scripts/MonoBehaviours
    • Scripts/Systems
    • Settings (already created)

블로그 이미지

RIsN

,

원본: https://github.com/Unity-Technologies/EntityComponentSystemSamples/blob/master/DOTS_Guide/ecs_tutorial/README.md

명령어 사용 및 제공된 코드 사용

이 튜토리얼의 인라인 코드에 관한 설명.

이 문서의 모든 코드 토막은 참조 프로젝트에서 자동으로 추출되고, 코드 토막을 포함하는 전체 파일의 사용 명령어가 복사됩니다. 따라서 명령어가 도입되는 시점에 아직 필요하지 않은 명령어를 사용하게 됩니다.

또한 UnityEngine 유형을 명시적으로 참조하는 방법에 주의하십시오. 이것은 이 튜토리얼에서 고전적인 유니티 유형과 DOTS 관련 유형을 명확하게 설명하기 위해 사용되는 규칙입니다.

일부 파일은 전체적으로 제공되며, 일부 파일은 기존 파일을 수정한 것으로 이 경우에는 + 줄(녹색)이 추가되고 - 줄(빨간색)이 제거되며, 다른 줄은 변경되지 않고 코드 토막만 제공합니다.

마지막으로 제공된 소스 파일을 복사할 때 코드의 주석을 읽으십시오. 이러한 설명에는 각 단계의 목적에 대한 많은 중요한 정보가 포함되어 있으며 이 튜토리얼의 필수적인 부분입니다.

블로그 이미지

RIsN

,

원본: https://github.com/Unity-Technologies/EntityComponentSystemSamples/blob/master/DOTS_Guide/ecs_tutorial/README.md

필요한 유니티 버전

이 튜토리얼은 2022.2.0b8을 사용했습니다.

Unity Hub 자체의 Unity 릴리스 목록에서 해당 버전(또는 최신 패치 릴리스)을 찾습니다.
"설치"(왼쪽 메뉴) → "에디터 설치"(오른쪽 위), "사전 릴리스"(탭).

또는, 다음의 링크에서 유니티 에디터의 2022.2.0b8 버전을 수동으로 다운로드 할 수도 있습니다:

블로그 이미지

RIsN

,

원본: https://github.com/Unity-Technologies/EntityComponentSystemSamples/blob/master/DOTS_Guide/ecs_tutorial/README.md

개요

이 문서에서 기대할 수 있는 내용.

이 튜토리얼에서는 다양한 기본 DOTS(Data-Oriented Technology Stack) 개념을 보여주는 매우 간단한 프로젝트를 만드는 과정을 단계별로 안내합니다. 많은 곳에서, 다양한 개념을 점진적으로 도입하기 위해 초기에 최적이 아니거나 부정확한 방식으로 기능이 사용됩니다. 그러니 먼저 전체 문맥을 이해하지 않고 이 자습서의 일부를 다른 프로젝트로 복사하지 마십시오.

이 튜토리얼은 핵심 엔티티 패키지에 초점을 맞추고 있으며 엔터티 그래픽을 매우 기본적으로 사용합니다. 그리고 physics이나 netcode와 같은 추가 패키지를 사용하지 않습니다.

완료된 프로젝트에는 다음과 같은 "기능"이 포함되어 있습니다:

  • 색깔이 입혀진 작은 탱크들이 필드 위를 움직이며 포탑을 돌리고 있습니다.
  • 본래 위치에서 충분히 멀리 떨어진 곳에서, 탱크들은 색깔이 입혀진 투사체를 쏘고 잇습니다.
  • 투사체는 지면에 닿아 튀기고 잠시 후 사라집니다.
  • 카메라는 임의의 탱크를 따라 움직이고 스페이스 바를 누를 때마다 다른 탱크로 전환됩니다.

이 자료에서 문제가 발견되면 보고해주세요. 혼란스러운 모든 것, 부딪히는 모든 오류 등. 이 튜토리얼을 개선하는 데 도움이 될 수 있는 모든 것을 환영합니다. 이를 위해 DOTS Forum을 사용해주세요.


참고:

블로그 이미지

RIsN

,

해결: Editor 클래스를 Editor 폴더에 넣어 놓으면 사라지는 에러

발생장소: apk 빌드 시

블로그 이미지

RIsN

,

[DPad] 제작중

Unity 2022. 4. 14. 22:58
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;

public class Attach_PadArea : MonoBehaviour
{
    [Header("Camera")]
    [SerializeField]
    private Camera iUICamera;
    [Header("Pad")]
    [SerializeField]
    private Attach_DPad iDPad;
    private void Awake() {
        this.iDPad.Init();
        this.iDPad.SetCamera(this.iUICamera);
    }
    private void OnMouseDown()
    {
        // :: 마우스 포지션 Convert
        var mousePosition = 
            this.iUICamera.ScreenToWorldPoint(Input.mousePosition); 
        this.iDPad.transform.position = mousePosition;
        var LPos = this.iDPad.transform.localPosition;
        this.iDPad.transform.localPosition = new Vector3(LPos.x, LPos.y, 0);
    }
    private void OnMouseDrag() {
        this.iDPad.UpdateStick();
    }
    private void OnMouseUp() {
        this.iDPad.ResetStick();
    }
}
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.AI;

public class Attach_DPad : MonoBehaviour
{
    [Header("Stick")]
    [SerializeField]
    private Transform iStick;
    private Camera iUICamera;
    private float iRadius;
    public void Init()
    {
        // :: 반지름 구하기
        this.iRadius = this.GetComponent<RectTransform>().rect.width * 0.5f;
    }
    public void SetCamera(Camera _camera)
    {
        this.iUICamera = _camera;
    }
    [Header("Character")]
    [SerializeField]
    private NavMeshAgent iNavCharacter;
    public void UpdateStick()
    {
        // :: 드래그 포지션 확인
        var transformPosition =
            this.iUICamera.WorldToScreenPoint(this.transform.position);
        Vector2 positionDiff = new Vector2(
            Input.mousePosition.x - transformPosition.x,
            Input.mousePosition.y - transformPosition.y);

        // :: 반지름 이상 못 가게 처리
        // :: ClampMagnitude(벡터, 최대 길이)
        // :: >> 최대 길이로 한정된 벡터를 Return
        positionDiff = Vector2.ClampMagnitude(positionDiff, this.iRadius);

        // :: 스틱 위치 변경
        this.iStick.localPosition = positionDiff;

        // :: 스틱 위치 정규화
        Vector3 moveDirection = positionDiff.normalized;

        // :: 거리 비율 확인
        float distance = Vector3.Distance(Vector3.zero, 
            this.iStick.localPosition) / this.iRadius;

        // :: EXIT : 거리가 짧을 때
        if (distance <= 0.1f) return;

        // :: 로테이션 이동
        // :: Atan2(점1, 점2)
        // :: >> 두 점 사이의 각도를 Radian으로 출력
        // :: Rad2Deg
        // :: >> Radian에 곱하면 각도를 출력
        float currentAngle // : 현재 각도
            = this.iNavCharacter.transform.localRotation.eulerAngles.y;
        float degree = currentAngle +
                ((Mathf.Atan2(moveDirection.x, moveDirection.y)
                * Mathf.Rad2Deg) * Time.deltaTime * this.iNavCharacter.speed);
        this.iNavCharacter.transform.localRotation
            = Quaternion.Euler(new Vector3(0f, degree, 0));

        // :: 포지션 이동
        Vector3 movePos = new Vector3(moveDirection.x, 0, moveDirection.y);

        // :: 내비게이션 이동
        this.iNavCharacter.Move(
            this.iNavCharacter.transform.forward 
            * movePos.normalized.z * Time.deltaTime 
            * this.iNavCharacter.speed);
    }
    public void ResetStick()
    {
        this.iStick.localPosition = Vector3.zero;
    }
}

블로그 이미지

RIsN

,

Polygon Arsenal

블로그 이미지

RIsN

,

GUI PRO Kit - Fantasy RPG
GUI Kit - SciFi Blue
GUI PRO Kit - Casual Game

 

블로그 이미지

RIsN

,