Ir al contenido principal

Juegos indie y publishers

Una vez un equipo indie ha terminado un juego o está en la fase de desarrollo y tiene claro los datos clave del juego como el modelo de negocio, plataformas, ventana de lanzamiento y un roadmap bien definido es momento de pensar como querréis lanzar su juego . Evidentemente la auto publicación es una opción, pero obliga al equipo a derivar recursos (tiempo, conocimientos y dinero) en entender como se debe lanzar un juego para que este funcione o al menos recuperemos lo invertido. Como desarrollador de juegos indie, conocer el funcionamiento de los publishers es crucial para decidir si trabajar con uno puede ayudarte a lanzar y comercializar tu juego de manera más efectiva. El punto clave para ello es evaluar lo que necesitamos de él (porting, marqueting, localización, etc.), analizar que recoup tendremos (dinero que habrá que devolver al publisher) y con que condiciones. El recoup (o recoupment) es el proceso por el cual un publisher recupera el dinero invertido en un juego antes d...

Prototipo de juegos móviles en Unity 2018 (Parte 2)

En esta segunda parte querría centrarme en aquellos aspectos visuales y no visuales que necesitamos cuando el jugador empieza la partida de nuestro juego. En la primera parte vimos como configurar la interfaz de Unity parar el desarrollo de aplicaciones móviles y hacer un fade IN y un fade OUT para cambiar de escenas desde la pantalla inicial de splash y la escena del título del juego.


A partir de este punto, cuando el jugador está viendo la pantalla del título debemos hacer algunas comprobaciones y procesos para tener la información de nuestro jugador, el juego y las partidas. Además en este punto podríamos, a partir de la configuración activar el sistema de lenguaje de la aplicación para tener en cuenta la localización del jugador y el idioma que usa. 

Escena del titulo del juego:
  1. Serializar la información del juego/jugador
  2. Cargar/crear fichero JSON de la configuración
  3. Conocer el estado de la aplicación
  4. Mostrar la versión de la aplicación
  5. Activar la localización del juego y su idioma (opcional)
  6. Cargar la escena del menú principal del juego o la escena de configuración inicial



Para guardar la información que deseamos guardar del juego crearemos una clase serializada con la estructura de variables, listas, ... que deseamos guardar:

PlayerInfo
 using System.Collections.Generic;  
 using System;  
 [Serializable]  
 public class PlayerInfo  
 {  
   public List<int> ID = new List<int>();  
   public List<int> Amounts = new List<int>();  
   public int life = 0;  
   public float highScore = 0;  
 }  

Una vez tenemos la clase creada debemos comprobar si en el dispositivo móvil tenemos la información almacenada. Cuando cargamos la escena, en el GameObject que hemos creado añadimos el componente LoadConfig que se encargará de comprobar si tenemos el fichero almacenado y en caso contrario lo creará con los valores por defecto que necesitemos.

LoadConfig
 using UnityEngine;  
 using System.IO;  
 public class LoadConfig : MonoBehaviour {  
   public string configFileName;  
   private string fileName;  
      // Use this for initialization  
      void Awake ()  
   {  
     fileName = Path.Combine(Application.persistentDataPath, "data");  
     fileName = Path.Combine(fileName, configFileName + ".txt");  
     if (!File.Exists(fileName))  
     {  
       PlayerInfo saveData = new PlayerInfo();  
       saveData.life = 100;  
       saveData.highScore = 0;  
       //Save data from PlayerInfo to a file named players  
       DataSaver.saveData(saveData, configFileName);  
       GlobalInfo.gameFirstTime = true;  
     } else  
     {  
       PlayerInfo loadedData = DataSaver.loadData<PlayerInfo>(configFileName);  
       if (loadedData == null)  
       {  
         return;  
       }  
       //Display loaded Data  
       Debug.Log("Life: " + loadedData.life);  
       Debug.Log("High Score: " + loadedData.highScore);  
       for (int i = 0; i < loadedData.ID.Count; i++)  
       {  
         Debug.Log("ID: " + loadedData.ID[i]);  
       }  
       for (int i = 0; i < loadedData.Amounts.Count; i++)  
       {  
         Debug.Log("Amounts: " + loadedData.Amounts[i]);  
       }  
       GlobalInfo.gameFirstTime = false;  
     }  
   }  
 }  

Esta clase tiene como variables publicas el nombre del archivo que deseamos poner al fichero de configuración (sin extensión). Por defecto la rutina añade la extensión .txt al fichero, que podemos cambiar si lo deseamos. Además nos mostrar en la consola la información cargada para comprobar el correcto almacenaje de los datos.

Todo ello es posible gracias a la clase DataServer que es la encargada de hacer el trabajo de conversión entre formatos.

DataServer
 using UnityEngine;  
 using System;  
 using System.IO;  
 using System.Text;  
 public class DataSaver  
 {  
   //Save Data  
   public static void saveData<T>(T dataToSave, string dataFileName)  
   {  
     string tempPath = Path.Combine(Application.persistentDataPath, "data");  
     tempPath = Path.Combine(tempPath, dataFileName + ".txt");  
     //Convert To Json then to bytes  
     string jsonData = JsonUtility.ToJson(dataToSave, true);  
     byte[] jsonByte = Encoding.ASCII.GetBytes(jsonData);  
     //Create Directory if it does not exist  
     if (!Directory.Exists(Path.GetDirectoryName(tempPath)))  
     {  
       Directory.CreateDirectory(Path.GetDirectoryName(tempPath));  
     }  
     try  
     {  
       File.WriteAllBytes(tempPath, jsonByte);  
       Debug.Log("Saved Data to: " + tempPath.Replace("/", "\\"));  
     }  
     catch (Exception e)  
     {  
       Debug.LogWarning("Failed To PlayerInfo Data to: " + tempPath.Replace("/", "\\"));  
       Debug.LogWarning("Error: " + e.Message);  
     }  
   }  
   //Load Data  
   public static T loadData<T>(string dataFileName)  
   {  
     string tempPath = Path.Combine(Application.persistentDataPath, "data");  
     tempPath = Path.Combine(tempPath, dataFileName + ".txt");  
     //Exit if Directory or File does not exist  
     if (!Directory.Exists(Path.GetDirectoryName(tempPath)))  
     {  
       Debug.LogWarning("Directory does not exist");  
       return default(T);  
     }  
     if (!File.Exists(tempPath))  
     {  
       Debug.Log("File does not exist");  
       return default(T);  
     }  
     //Load saved Json  
     byte[] jsonByte = null;  
     try  
     {  
       jsonByte = File.ReadAllBytes(tempPath);  
       Debug.Log("Loaded Data from: " + tempPath.Replace("/", "\\"));  
     }  
     catch (Exception e)  
     {  
       Debug.LogWarning("Failed To Load Data from: " + tempPath.Replace("/", "\\"));  
       Debug.LogWarning("Error: " + e.Message);  
     }  
     //Convert to json string  
     string jsonData = Encoding.ASCII.GetString(jsonByte);  
     //Convert to Object  
     object resultValue = JsonUtility.FromJson<T>(jsonData);  
     return (T)Convert.ChangeType(resultValue, typeof(T));  
   }  
   public static bool deleteData(string dataFileName)  
   {  
     bool success = false;  
     //Load Data  
     string tempPath = Path.Combine(Application.persistentDataPath, "data");  
     tempPath = Path.Combine(tempPath, dataFileName + ".txt");  
     //Exit if Directory or File does not exist  
     if (!Directory.Exists(Path.GetDirectoryName(tempPath)))  
     {  
       Debug.LogWarning("Directory does not exist");  
       return false;  
     }  
     if (!File.Exists(tempPath))  
     {  
       Debug.Log("File does not exist");  
       return false;  
     }  
     try  
     {  
       File.Delete(tempPath);  
       Debug.Log("Data deleted from: " + tempPath.Replace("/", "\\"));  
       success = true;  
     }  
     catch (Exception e)  
     {  
       Debug.LogWarning("Failed To Delete Data: " + e.Message);  
     }  
     return success;  
   }  
 }  

Por lo que se refiere a la información tenemos una ultima clase que se encarga de mostrar la información de la versión en la pantalla a través de la variable pública de la clase.

AppVersion
 using UnityEngine;  
 using UnityEngine.UI;  
 public class AppVersion : MonoBehaviour {  
   public Text TextVersion;   
      // Use this for initialization  
      void Start ()  
   {  
     TextVersion.text = "Version: " + Application.version;  
   }  
 }  

Ejemplo:
 
Pantalla Splash
A través de un fade pasamos a la pantalla de título.




Pantalla de título
Carga de la configuración. 
La versión del juego. 
Establece el punto de estado de la aplicación.
Saltamos al menú principal con una barra de progreso.

Cuando saltamos a la siguiente pantalla a través de una variable global sabremos si es la primera vez que ejecutamos la aplicación o no. Algunas veces nos puede interesar que en el menú principal podamos forzar una pantalla de configuración para establecer los parámetros iniciales del juego (idioma, nombre, país, ...).

 using UnityEngine;  
 public class GlobalInfo : MonoBehaviour  
 {  
   public static bool gameFirstTime;  
 }  

Comentarios

Entradas populares de este blog

Back to work!

¡Hola! Estoy emocionado de anunciar que he retomado la actividad en mi blog "Jugando Haciendo Juegos" después de una pausa. En esta nueva etapa, el blog se centrará en proporcionar información valiosa desde dentro de la industria de los videojuegos, incluyendo trucos, curiosidades, herramientas, recursos y consejos sobre cómo crear videojuegos desde la comodidad de tu habitación utilizando Unity. El blog ya cuenta con varios artículos útiles que pueden servir como recursos iniciales. Por ejemplo, hay guías detalladas sobre cómo hacer copias de seguridad de tu código de Unity con GitHub, lo que es crucial para la gestión de versiones y el trabajo colaborativo. También encontrarás tutoriales sobre el diseño de interfaces de usuario, el desarrollo de controladores de personajes y cómo crear un PressKit para tu juego. Mi objetivo es compartir conocimientos prácticos y experiencias personales para ayudar tanto a principiantes como a desarrolladores más avanzados a mejorar sus hab...

El diseño de la interfaz de usuario

El estudio del diseño de interfaz de usuario en videojuegos es un tema que se ha estudiado en profundidad pero que muchos desarrolladores que empiezan no prestan mucha atención centrando su energía en las mecánicas del juego y especialmente el arte ya que muchas veces el éxito o el fracaso de un juego dependen de ello. Pero más lejos de la realidad la capacidad lúdica de un juego muchas veces también viene determinada por el diseño de la interfaz que hace de dialogo entre el jugador y el juego. Uno de los mejores análisis de las interfaces de usuario en videojuegos lo encontramos en los estudios realizados por Anthony Stonehouse y Marcus Andrews . El diseño de la interfaz de usuario en los juegos difiere de otro diseño de interfaz de usuario porque implica un elemento adicional: la ficción. La ficción involucra un avatar del usuario real, o jugador. El jugador se convierte en un elemento invisible, pero clave de la historia, como un narrador en una novela o película. Esta ficc...

Preparando un evento de videojuegos

Participar en un evento de videojuegos con un stand puede ser una oportunidad única para dar a conocer tus juegos, interactuar directamente con tu público objetivo, y generar un gran impacto en la comunidad. En este artículo, exploraremos cómo preparar un stand de 4x2 metros de manera efectiva, maximizar el uso de la cartelería de fondo, y atraer a los visitantes con técnicas sencillas y económicas de marketing. 1. Diseño y uso eficiente de la cartelería de fondo 1.1. La importancia de una lona reutilizable : La lona que utilices para el fondo de tu stand es una de las inversiones más importantes. Una lona de buena calidad y diseño profesional no solo refuerza tu marca, sino que también puede ser reutilizada en futuros eventos, lo que la convierte en una opción sostenible y económica. Asegúrate de que la lona cubra completamente el fondo de tu stand, ya que esto crea un ambiente inmersivo y profesional. 1.2. Elementos esenciales en la lona: En la cartelería de fondo, incluye: El logoti...