Introducción
Con Windows 10 los desarrolladores recibimos la culminación de un gran viaje en la convergencia entre las diferentes plataformas Windows. Ahora podemos desarrollar aplicaciones para gran diversidad de familias de dispositivos como móviles, PCs, tabletas, IoT y otros que están por llegar, compartiendo la mayor cantidad de código, con un mismo proyecto y paquete. Además, contamos con grandes nuevas características como Continuum en teléfonos que permite convertirlo en un PC utilizando Microsoft Display Dock o Miracast.
Sin embargo, hay un detalle claro y obvio. Si contamos con un mismo paquete para todos esos dispositivos diferentes…¿cómo adaptamos la experiencia para ofrecer la mejor opción adaptada posible?
Debemos adaptar la interfaz de usuario en cada familia de plataforma para lograr ofrecer la mejor experiencia posible adaptada a la perfección. Para ello, utilizamos:
- AdaptiveTriggers
- Nuevos controles como RelativePanel y/o SplitView
- Detección de modos de interacción
- Etc
Sin embargo, hay elementos vitales en la mayoría de aplicaciones, que no acaban recibiendo la atención que se merecen. Estoy hablando de las imagenes. La aplicación puede usar una imagen que se visualiza perfectamente en un teléfono pero…¿y si se usa la aplicación con Continuum en una pantalla con una resolución diferente (más elevada)?.
En este artículo, vamos a aprender como organizar y utilizar los recursos de la aplicación para que se utilicen y adapten por DPI.
DisplayInformation
La clase DisplayInformation cuenta con propiedades y eventos que nos permiten verificar y monitorear información relacionada con la pantalla física. Para monitorear detalles como cambios en DPI o la rotación podemos usar la clase DisplayInformation.
En nuestra interfaz vamos a mostrar la siguiente información:
<StackPanel Orientation="Horizontal"> <TextBlock Text="Logical DPI:" /> <TextBlock Text="{Binding LogicalDpi}" /> <TextBlock Text="DPI" /> </StackPanel> <StackPanel Orientation="Horizontal"> <TextBlock Text="Scaling:" /> <TextBlock Text="{Binding Scale}" /> <TextBlock Text="%" /> </StackPanel>
De modo que, en la viewmodel bindeada definiremos dos propiedades:
private string _logicalDpi; public string LogicalDpi { get { return _logicalDpi; } set { _logicalDpi = value; RaisePropertyChanged(); } } private string _scale; public string Scale { get { return _scale; } set { _scale = value; RaisePropertyChanged(); } }
Una para cada valor que deseamos mostrar en pantalla. Utilizaremos el método DpiChanged lanzado cada vez que la propiedad LogicalDpi se modifica, cuando cambian los píxeles por pulgada (PPI) de la pantalla.
private DisplayInformation _displayInformation; _displayInformation = DisplayInformation.GetForCurrentView();
Tras obtener la información física actual de la pantalla utilizando el método GetForCurrentView nos suscribimos al evento DpiChanged:
_displayInformation.DpiChanged += _displayInformation_DpiChanged;
Cada vez que el evento se lanza, actualizamos la información mostrada en pantalla:
private void _displayInformation_DpiChanged(DisplayInformation sender, object args) { DisplayInformation displayInformation = sender as DisplayInformation; UpdateDpi(displayInformation); } private void UpdateDpi(DisplayInformation displayInformation) { if (displayInformation != null) { LogicalDpi = displayInformation.LogicalDpi.ToString(); Scale = (displayInformation.RawPixelsPerViewPixel * 100.0).ToString(); } }
Mostramos los píxeles por pulgada lógica de la pantalla actual utilizando la propiedad LogicalDpi, mientras que para la escala utilizamos la propiedad RawPixelsPerViewPixel que indica el número de píxeles físicos (RAW) por cada pixel mostrado (Layout). Para obtener la escala bastará con multiplicar el valor por cien.
NOTA: En este ejemplo utilizamos la clase DisplayInformation para mostrar información contextual relacionada con el escalado de imágenes utilizado. Sin embargo, utilizando propiedades como DiagonalSizeInInches podemos saber facilmente el tamaño en pulgadas de la pantalla y así adaptar la interfaz en consecuencia. Sumamente útil y sencillo combinado con el uso de AdaptiveTriggers personalizados.
Recursos por DPI
Para optimizar nuestra interfaz en cada posible dispositivo o condición, podemos facilitar diferentes assets para diferentes resoluciones y escalas. Cada dispositivo cuenta con una escala específica resultante de la densidad de píxeles física y la distancia de visión teórica.
La escala es utilizada por el sistema, que realiza una gestión de recursos para determinar que recurso es el más adecuado entre las opciones facilitadas por los desarrolladores en sus aplicaciones.
NOTA: Los teléfonos suelen tener una escala de entre 200 y 400 mientras que dispositivos conectados como monitores y TVs tiene valores de 100 y 150 respectivamente.
Añadiendo recursos por escala
Para soportar el uso de diferentes recursos dependientes de la escala, bastará con añadirlos de la forma adecuada. Contamos con dos formas diferentes para ello.
Por un lado, podemos sencillamente añadir el mismo recurso con diferentes tamaños utilizando la notación .scale-xxx dentro de la carpeta Assets:
Por otro lado, podemos añadir diferentes carpetas con el nombre de la escala, es decir, scale-xxx incluyendo como contenido los recursos.
Utilizando recursos por escala
En nuestra carpeta de Assets contamos con una imágen única llamada Picture-NoScale:
<Image Source="ms-appx:///Assets/Picture-NoScale.png" Height="100" Width="100" HorizontalAlignment="Left"/>
Con el código anterior, usaremos la misma (y única) imágen existente bajo cualquier condición. Si la escala es alta y requiere de recursos con mayor resolución, el resultado será una visualización borrosa de la misma. Proporcionamos una experiencia no idónea.
Contamos con múltiples opciones por escala del recurso llamado Picture, bastará con utilizarlo ignorando scale-xxx de la ruta:
<Image Source="ms-appx:///Assets/Picture.png" Height="100" Width="100" HorizontalAlignment="Left"/>
Utilizando los recursos vistos previamente, donde en un caso usamos una imágen única bajo cualquier condición y en otro una adapada a diferentes escalas, el resultado es el siguiente:
Tenemos 192DPI con una escala de 200%. Podemos ver a simple vista que mientras que la primera imágen se visualiza pixelada, la segunda se aprecia con una calidad alta.
Sencillo, ¿cierto?
Tenéis el código fuente disponible e GitHub:
Recordad que podéis dejar cualquier comentario, sugerencia o duda en los comentarios.
Más información
- Windows Dev Center: DisplayInformation
- Windows Blog: Optimizing apps for Continuum for phone