Differences

This shows you the differences between two versions of the page.

Link to this comparison view

pjv:laboratoare:06 [2018/11/14 21:22]
alexandru.gradinaru
pjv:laboratoare:06 [2019/10/02 12:59] (current)
alexandru.gradinaru [Cerinte]
Line 5: Line 5:
 === NavMesh === === NavMesh ===
  
-Pentru navigarea ​personajelor ​automata pana la un punct pe harta, indiferent ca e vorba de un personaj ​conrolat ​sau un agent inteligent, avem nevoie in primul rand de generarea unor zone in care acestea sa se poata misca. Aceste zone poarta numele de Navigation Mesh sau NavMesh, prescurtat. In Unity se pot genera automat acest NavMesh folosind utilitarul de Navigation Bake: Window > AI > Navigation > Bake.+Pentru navigarea automata ​a personajelor ​pana la un punct pe harta, indiferent ca e vorba de un personaj ​controlat ​sau un agent inteligent, avem nevoie in primul rand de generarea unor zone in care acestea sa se poata misca. Aceste zone poarta numele de Navigation Mesh sau NavMesh, prescurtat. In Unity se pot genera automat acest NavMesh folosind utilitarul de Navigation Bake: Window > AI > Navigation > Bake.
  
 {{ :​pjv:​laboratoare:​bakenavmesh.png?​direct&​300 |}} {{ :​pjv:​laboratoare:​bakenavmesh.png?​direct&​300 |}}
Line 71: Line 71:
   {   {
     target = hit.transform;​     target = hit.transform;​
-    ​//follow target pana la destinatie +    StartCoroutine(FollowTarget()); ​//follow target pana la destinatie
-     +
-    ​StartCoroutine(FollowTarget());​ +
-     +
-    ​+
   }   }
  
Line 110: Line 106:
  
 </​code>​ </​code>​
 +
 +
 +==== Animatia caracterelor ====
 +
 +Animatia caracterelor 3D, se face similar ca in 2D, fiind utilizata animatia bazata pe oase. Astfel, fiecare caracter 3D are definite oase (bones) care controleaza miscarea si animatia acestuia.
 +
 +{{ :​pjv:​laboratoare:​humanoid_bones.png?​direct&​600 |}}
 +
 +Crearea animatiilor se poate face folosind utilitarul de animatie (Window > Animation), similar ca in spatiul 2D. Similar, se poate folosi Animator Controller pentru a gestiona starile.
 +
 +Ca notiune suplimentara,​ se pot defini stari de tranzitie/​interpolare intre anumite animatii. Un exemplu comun este trecerea sau ajustarea animatiei de walk si run in functie de viteza curenta a personajului,​ sau a animatie de alergare stanga - fata - dreapta (daca sunt definite animatii diferite). Acest lucru se poate realiza folosind ''​BlendTrees''​ (Create State > From New Blend Tree).
 +
 +{{ :​pjv:​laboratoare:​mecanimblendtreestatediagramcombined.png?​direct&​600 |}}
  
 ==== Interactiunea cu obiectele ==== ==== Interactiunea cu obiectele ====
  
 +Interactiunea cu obiectele in spatiul 3D poate fi extrem de complexa, intrucat exista foarte multe forme de interactiuni:​ inamici, obiecte de pickup, deschidere de usi, activare de manivele etc. Fiecare dintre aceste interactiuni are specificul ei, dar abstractizand,​ putem deduce ca fiecare interactiune se intampla intr-o anumita raza si cu un anumit punct de interactiune. Pentru a defini usor aceste lucruri, putem crea o clasa generica denumita ''​InteractionObject''​ cu o metoda abstracta (virtuala) ce defineste interactiunea in detaliu.
  
 +<​code>​
 +public class InteractionObject : MonoBehaviour {
  
 +  public float radius = 1f;
 +  public Transform interactionPoint;​
 +  Transform interactionObject;​
 +  bool done = false;
  
-==== Cerinte ====+  //metoda abstracta, speficica fiecarui tip de interactiuni 
 +  public virtual void Interaction () 
 +  { 
 +  
 +  }
  
-Realizarea unui joc 3D RPG+  void Update () 
 +  { 
 +      float distance = Vector3.Distance(interactionObject.position,​ interactionPoint.position);​
  
-  - Configurati scena pentru rulare 3D +      if (distance <= radius && !done// avem interactiune cu obiectul 
-  - Adaugati in scena un caracter TPS animat ​(puteti lua resurse din asset store sau alte locuri: [[pjv:​resurse|]]astfel incat: +      ​{ 
-      ​- camera urmareste caracterul 3rd person +        done = true
-      - se va controla cu mouse-ul pe hartaabilitatile speciale se pot activa si pe tastatura +        ​Interaction(); 
-      - poate lua obiecte in inventar +      ​
-      - poate arunca obiecte din inventar +    } 
-      - poate schimba arma de atac echipata +  ​}
-      - poate consulta quest-ul curent +
-      - poate ataca +
-      - are experienta si level-uri +
-      - in functie de level poata adauga puncte intr-un skilltree +
-      - poate folosi skill-uri/​abilitati speciale +
-  - Adaugati unul sau mai multi NPC care: +
-      - stiu sa converseze ​(textprin raspunsuri la intrebari standard +
-      ​- pot oferi un quest  +
-  ​- Adaugati unul sau mai multi inamici in scena scriptati astfel incat: +
-      - sa fie animati +
-      - sa se plimbe intr-o proximitate +
-      - la apropierea jucatorului,​ sa il atace+
  
-<hidden> +
-(vezi demo UI de la 3dpub) +</code>
-  - coliziuni, proiectile +
-  - interfata grafica pt un fps +
-  - Managing weapons, Navmeshes, navmesh Agents, AI Auto-Attack +
-  - animatie caractere atac, arma, coliziuni atac+
  
-7: 3d RPG inventory systemquest systemskill system, pathfinding,​Grid Snapping, Managing weapons, Navmeshes, navmesh Agents, AI Auto-Attack+Astfeltoate obiectele ce vor avea interactiunivor mosteni aceasta clasa. 
 +Spre exemplu pentru un obiect de pickup putem avea urmatoarea secventa: 
 +<​code>​ 
 +public class PickupObject : InteractionObject {
  
-QuestLog+  public override void Interaction() 
 +  { 
 +    base.Interaction();​ // se apeleaza metoda parinte, in caz ca avem ceva generic 
 +     
 +    //​mecanica 
 +    ... 
 +     
 +    //distrugem obiectul 
 +    Destroy(gameObject);​ 
 +     
 +  } 
 +
 +</​code>​
  
-playerul +Pentru a controla ​mai bine zona de actiune ​(radiussi punctul ​de interes pentru ​un obiect de interactiune (InteractionObject),​ se poate defini o functie de editor, atunci cand obiectul ​este selectat. In exemplul de mai josla selectarea obiectului se va afisa o sfera wireframe de culoare alba.
-are abilitati speciale care consuma mana/puncte de abilitate +
-poate prelua unul sau mai multe quest-uri +
-poate cara obiecte ​(iau un cub dintr-o parte si il pun in alta, ca sa sar pe el) +
-poate interactiona cu NPC (botsi) pentru preluare ​de quest-uri +
-are un scor in functie de numarul de quest-uri rezolvate +
-este animat (atacmiscare)+
  
-NPC +<​code>​ 
-Agenti de la care se pot prelua/​rezolva quest-uri +void OnDrawGizmosSelected () 
-Sunt animati +{ 
-Pot purta o conversatie cu mai multe fire de dialog si mai multe variante de raspuns+  ​Gizmos.color = Color.white;​ 
 +  Gizmos.DrawWireSphere(interactionPoint.position,​ radius); 
 +}  
 +</​code>​
  
-</​hidden>​+{{ :​pjv:​laboratoare:​gizmosselected.png?​direct&​750 |}}
  
 +=== Inventar ===
  
-<​hidden>​ +Pentru ​crea un sistem de inventar avem nevoie in primul rand de date atasate fiecarui obiect, cum ar fi nume, icon, atribute etc. Putem realiza acest lucru usor prin obiecte scriptabile (Scriptable Objects). Obiectele Scriptable sunt containere de date ce nu trebuie sa fie atasate la un GameObject intr-o scena. Ele pot fi salvate ca asset-uri in bibliteca proiectului ca mai apoi sa poata fi utilizate.
-===== Elemente de programare ​jocurilor 3D ===== +
-animatii cu blend tree+
  
 +Obiectele scriptabile se definesc prin crearea unui script ce mosteneste clasa ScriptableObject.
  
 +Pentru a instantia un obiect scriptabil avem doua variante:
 +  * prin script: SciptableObject.CreateInstance<​MyScriptableObjectClass>​()
 +  * din mediu: CreateAssetMenu - adauga o intrare noua in meniul de create asset
 +<​code>​
 +
 +[CreateAssetMenu(fileName = "New Item", menuName = "​Inventory/​Item",​ order = 1)]
 +public class Item : ScriptableObject {
 +    new public string name = "New MyScriptableObject";​ //​suprascrie atributul name
 +    public string objectName = "New MyScriptableObject";​
 +    public bool colorIsRandom = false;
 +    public Color thisColor = Color.white;​
 +    public Sprite icon;
 +    public Vector3[] spawnPoints;​
 +}
 +</​code>​
 +
 +Intrucat exista atribute implicite pentru un obiect scriptabil (e.g. name), putem folosi variabile diferite (e.g. objectName) sau putem suprascrie definirea acestui atribut prin folosirea metodei ''​new''​ (new public string name).
 +
 +Mai departe, pentru un inventar vom avea nevoie de o lista de obiecte gestionabile. Pentru acest lucru vom face un script de gestiune pentru inventar (e.g InventoryManager) care gestioneaza adaugarea, eliminarea si interogarea inventarului. Pentru o accesare mai usoara si mai facila, ideal ar fi ca acest inventorymanager sa fie un Singleton. Pentru a avea un singleton trebuie sa ne asiguram ca avem o singura instanta creata pentru acest script atunci cand e accesat.
 +
 +<​code>​
 +public class InventoryManager : MonoBehaviour {
 +
 +  // singleton
 +  public static InventoryManager instance;
 +  void Awake() {
 +    instance = this;
 +  }
 +  ​
 +  //lista de obiecte
 +  public List<​Item>​ items = new List<​Item>​();​
 +  ​
 +  //metode pentru gestionare
 +  public void Add(Item item) {
 +    items.Add(item);​
 +  }
 +  ​
 +  public void Remove(Item item) {
 +    items.Remove(item);​
 +  }
 +
 +}
 +</​code>​
 +
 +Fiind definit ca un singleton, putem accesa acum foarte usor gestionarea inventarului:​
 +<​code>​
 +Inventory.instance.Add(item);​
 +Inventory.instance.Remove(item);​
 +</​code>​
 +
 +Inca un element util in gestionarea inventarului este definirea unei metode de a notifica atunci cand s-a produs o modificare in inventar. Pentru acest lucru putem folosi ''​Delegates''​. Un Delegate este un pointer la o metode. Aceasta ne permite sa tratam metoda ca o variabila și sa o folosim pentru un callback. Cand este apelata, acesta notifica toate metodele care fac referire la delegate. Astfel putem definit o variabila pentru evenimentul de schimbare.
 +<​code>​
 +  public delegate void OnInventoryChanged();​
 +  public OnInventoryChanged onInventoryChangedCallback;​
 +  ​
 + //​metode pentru gestionare
 +  public void Add(Item item) {
 +    ...
 +    onInventoryChangedCallback.Invoke();​ //notifica despre modificare
 +  }
 +  ​
 +  public void Remove(Item item) {
 +    ...
 +    onInventoryChangedCallback.Invoke();​ //notifica despre ​ modificare
 +  }
 +</​code>​
 +Urmatorul pas este crearea unei interfete grafice si legarea interfetei de functionalitatea InventoryManger-ului.
 +Pentru interfata grafica, putem folosi, ca si pana acum, canvas-ul oferit de Unity, structurat astfel incat sa avem un panou general pentru inventar, si mai multe slot-uri pentru obiectele din acesta. Pe fiecare slot putem defini urmatoarele aspecte:
 +  * un gameobject
 +  * un buton de accesare : pentru folosirea obiectului, echipare etc.
 +  * un buton de eliminare : pentru eliminarea din inventar
 +  * o imagine: icon pentru obiectul in inventar
 +  * script de gestionare: pentru fiecare slot putem avea un script de gestionare care va actualiza interfata slotului respectiv (nume, icon etc).
 +
 +Interfata grafica a inventarului are nevoie si de un script de gestionare. Astfel vom aveam un script care asculta (subscribe) evenimentul definit (''​delegate''​) la actualizarea inventarului,​ si actualizeaza fiecare slot din interfata grafica:
 +
 +<​code>​
 +void Start() {
 +  inventory = Inventory.instance;​
 +  inventory.onInventoryChangedCallback += UpdateUI; //definesc o metoda ca se apeleaza la aparitia unui eveniment delegat
 +  ​
 +  slots = GetComponentsInChildren<​InventorySlot>​();​ //fiecare slot din inventar
 +}
 +
 +void UpdateUI() {
 +  ​
 +  //​actualizare fiecare slot
 +  for(i=0; i < slots.Length;​ i++)
 +  {
 +    if(i<​iventory.items.Count) slots[i].AddItem(..)
 +    else slots[i].RemoveItem(..)
 +  } 
 +
 +}
 +</​code>​
 +
 +<​hidden>​sistem inventar
 +
 +Bineinteles,​ in inventar se pot pune diferse restrictii si interactiuni (cum ar fi dimensiunea maxima a inventarului)
 +<​code>​
 +bool AddItem() {
 +
 +if(items.Count >= space)
 +//no more room
 +return false
 +
 +else return true;
 +}
 +
 +if(Inventory.instance.Add(item)) Destroy(gameObject);​
 +</​code>​
 </​hidden>​ </​hidden>​
 +
 +==== Cerinte ====
 +
 +Realizarea unui joc 3D Top-Down
 +
 +  - Configurati scena pentru rulare 3D
 +  - Adaugati in scena un caracter animat (puteti lua resurse din asset store sau alte locuri: [[pjv:​resurse|]]) astfel incat:
 +      - camera urmareste caracterul de sus
 +      - se va deplasa prin click mouse-ul la o locatie pe harta;
 +      - poate lua obiecte in inventar
 +      - poate arunca obiecte din inventar ​
 +  - Adaugati mai multe tipuri de obiecte interactionabile in scena
pjv/laboratoare/06.1542223336.txt.gz · Last modified: 2018/11/14 21:22 by alexandru.gradinaru
CC Attribution-Share Alike 3.0 Unported
www.chimeric.de Valid CSS Driven by DokuWiki do yourself a favour and use a real browser - get firefox!! Recent changes RSS feed Valid XHTML 1.0