Hasta ahora hemos descrito el lenguaje JavaScript própiamente dicho. Como ya sabemos, se trata de un lenguaje interpretado que se ejecuta en un entorno de ejecución. En este tema estudiaremos la ejecución de JavaScript en el navegador y, por tanto, los objetos de plataforma que proporciona dicho contexto.
Nota: el objetivo de este tema es conocer el contexto de ejecución de navegador y los API que proporciona a los programas escritos con JavaScript. No entraremos en detalle en la escritura de código ya que es preferible utilizar librerías que facilitan su uso, como veremos en el siguiente tema.
Si observamos los sitios web nos daremos cuenta que se pueden clasificar en dos grandes categorias:
Documentos web. Presentan información estática.
Se usa JavaScript para mejorar la experiencia de usuario (interactividad)
Aplicaciones web. Gestionan información dinámica.
Se usa JavaScript para implementar una aplicación completa que gestiona datos remotos. Se trata de una aplicación cliente/servidor que sustituye el sistema operativo como entorno de ejecución por el navegador.
<script></script>
src
del tag <script></script>
<html>
<head>
<script src="micodigo.js"></script>
<script>
funtion comprobarNavegador() {
if (navigator.appName == 'MSIE') alert('Cuidado!!');
}
</script>
</head>
<body onLoad="comprobarNavegador();">
<h1>Lista</h1>
<script>
for (var= 0; i < lista.lenght; i++) document.write(lista[i].nombre + '<br>');
</script>
<a href="javascript:void open('./docs/'+doc+'.pdf');">Abrir documentación</a>
</body>
</html>
El BOM es un API que expone el contexto de ejecución de un programa JavaScript cuando se ejecuta en un navegador. El principal inconveniente es que, al contrario que el propio lenguaje (ECMAScript) y el DOM, el BOM no tiene un estándar oficial. Algunas características del BOM se estandarizan en HTML5, pero no todas.
Su objetivo es que el programa pueda interactuar con el navegador.
El objeto window
es el elemento central del contexto de ejecución en el navegador. Representa una ventana (o pestaña o frame) del navegador. Define el ámbito global de la aplicación.
Principales propiedades del objeto window
:
Propiedad | Descripción |
---|---|
location |
Recupera/establece la url. Devuelve una referencia al objeto Location |
history |
Manipular el historial de la sesión de navegación. Devuelve una referencia al objeto History |
navigator |
Información del navegador. Devuelve una referencia al objeto Navigator |
screen |
Propiedades dela pantalla. Devuelve una referencia al objeto Screen |
frames |
Lista de frames, cada frame es un objeto window |
document |
Devuelve una referencia al documento cargado en el navegador, el document object model (DOM) |
Algunas propiedades de la propia ventana:
Propiedad | Descripción |
---|---|
fullScreen |
Si el navegador está en pantalla completa |
innerHeight, innerWidth |
Alto y ancho, respectivamente, de área de trabajo de la ventana |
outerHeight, outerWidth |
Alto y ancho, respectivamente, de área externa de la ventana |
menubar, toolbar, statusbar |
Devuelve sendos objetos para controlar su visibilidad |
name |
Recupera/asigna el título de la ventana |
scrollX, scrollY |
Devuelve el desplazamiento del documento |
status |
Recupera/estable el texto de la barra de estado |
Principales métodos:
Método | Descripción |
---|---|
alert(), confirm() |
Muestra un mensaje, el segundo con confirmación |
prompt() |
Pide información al usuario |
open(), close() |
Abre/cierra una nueva ventana |
print() |
Abre el cuadro de diálogo de impresión |
scroll(), scrollBy(), scrollByLine(), scrollByPage(), scrollTo() |
Diferentes métodos para actualizar la posición de visualización del documento |
resizeBy(), ResizeTo() |
Modifica el tamaño de la ventana |
setInterval(), setTimeout() |
Ejecuta una función cada intérvalo de tiempo o después de un tiempo, respectivamente |
Nota: al tratarse del objeto global no es necesario escribir
window
al acceder a sus propiedades o invocar sus métodos
A través del objeto window
se accede al nuevo conjunto de Web APIs definidas al amparo de HTML5, como por ejemplo:
El DOM es el API de los documentos HTML, XML y SVG. Proporciona una representación estructurada (en forma de árbol) del documento HTML que genera el navegador cuando carga una página, y define la forma en que los programas pueden acceder a esta estructura para interactuar con ella: añadir, modificar o eliminar componentes.
El DOM se ha diseñado para ser independiente del lenguaje de programación que manipula sus componentes; nosotros accederemos mediante JavaScript pero existen implementaciones en diferentes lenguajes, como Python, Java, C++...
Ejemplo:
<html lang="es">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
</head>
<body>
<h1>Título</h1>
<p>Primer párrafo del documento <span class="clase1">de prueba</span></p>
<ul>
<li>Uno</li>
<li>Dos</li>
<li>Tres</li>
</ul>
<table class="tg">
<tr>
<th>Cabecera 1</th>
<th>Cabecera 2</th>
<th>Cabecera 3</th>
<th>Cabecera 4</th>
</tr>
<tr>
<td>1-1</td>
<td>1-2</td>
<td id="miCelda">1-3</td>
<td>1-4</td>
</tr>
<tr>
<td>2-1</td>
<td class="clase1">2-2</td>
<td>2-3</td>
<td>2-4</td>
</tr>
</table>
</body>
</html>
Todos los componentes del documento son nodos en el árbol, cada tag html es un nodo de tipo elemento y cada fragmento de texto es un nodo de tipo texto. Los nodos y elementos son objetos con propiedades y métodos que nos permiten:
Vamos a ver el DOM utilizando las herramientas para desarrolladores que implementan los navegadores (generalmente la pestaña elements). Abrir ejemplo en el navegador...
La raiz del documento es document.documentElement
y de cuerpo del documento document.body
. Este último puede devolver null
si no hay cuerpo de documento.
Nota: como regla general no se puede referenciar un elemento que aún no se ha renderizado
En la consola del navegador escribir:
document.documentElement
document.body
Principales propiedades y métodos:
Propiedad | Descripción |
---|---|
children, chilNodes |
Devuelve un array con todos los elementos hijos o sus nodos, respectivamente |
firstChild, lastChild |
Devuelve el primer y último nodo hijo |
parentNode |
Devuelve el nodo padre |
previousSibling, nextSibling |
Devuelve el hermano anterior y el siguiente, respectivamente |
En la consola:
document.body.children
document.body.childNodes
document.body.firstChild
document.body.firstChild.nextSibling
document.body.lastChild.previousSibling
Método | Descripción | Interfaz |
---|---|---|
getElementById() |
Recupera un elemento por su id | documento |
getElementsByTagName() |
Devuelve un array con los elementos de un determinado tag | documento y nodo |
getElementsByName() |
Para elementos con el atributo name |
documento |
getElementsByClassName() |
Devuelve array con elementos de una clase (estilo CSS) | documento y nodo |
querySelector(), querySelectorAll() |
Busca por consulta CSS3, devuelve el primer elemento encontrado o todos ellos | documento y nodo |
En la consola:
document.getElementById('miCelda')
document.getElementsByTagName('th')
document.getElementsByClassName('clase1')
document.querySelector('ul > li')
Propiedad | Descripción |
---|---|
nodeType |
Tipo de nodo, devuelve 1 si tipo elemento o 3 si texto |
nodeName/tagName |
Devuelven el nombre del nodo |
innerHTML, nodeValue |
Devuelven/establecen el valor de un elemento o nodo de texto, respectivamente |
En la consola:
document.body.firstChild.nodeType
document.body.firstChild.nextSibling.nodeType
document.getElementById('miCelda').nodeName
document.getElementById('miCelda').tagName
document.getElementById('miCelda').innerHTML
Nota: como cualquier objeto JavaScript podemos añadir dinámicamente propiedades a los objetos del DOM, que serán visibles únicamente en el programa
Métodos | Descripción |
---|---|
hasAttribute() |
Comprueba si el elemento tiene un determinado atributo |
getAttribute()/setAttribute() |
Devuelven/establece el valor de un atributo |
removeAttribute() |
Elimina un atributo |
En la consola:
document.getElementById('miCelda').hasAttribute('style')
document.getElementById('miCelda').setAttribute('style', 'text-align: center')
document.getElementById('miCelda').removeAttribute('style')
Método | Descripción | Interfaz |
---|---|---|
createElement(), createTextNode() |
Crea un nuevo elemento o nodo de texto, respectivamente | documento |
cloneNode() |
Copia un nodo | nodo |
appendChild(), removeChild() |
Añade/elimina un elemento | nodo |
insertBefore() |
Inserta un elemento antes que un determinado nodo hijo | nodo |
replaceChild() |
Reemplaza un nodo hijo por otro elemento | nodo |
En la consola:
miImg= document.createElement('img')
miImg.setAttribute('src', 'https://www.google.es/images/srpr/logo11w.png')
document.body.appendChild(miImg)
document.getElementById('miCelda').innerHTML= 'Nuevo contenido de miCelda'
El método document.write
escribe html en un documento abierto (antes de que esté completamente cargado en el navegador). Si el documento ya está cerrado (cargado completamente) y se utiliza el método se crea un nuevo documento.
document.write('<h1>Nuevo documento</h1>')
for (i= 1; i <= 10; i++) document.write(i + '. Línea de texto<br>')
Las aplicaciones web usan un modelo de programación basado en eventos. Los programas esperan, sin realizar ninguna tarea, hasta que se produce un evento: se carga la página, se pulsa un botón, se selecciona un elemento, se mueve el ratón sobre un elemento...
Un evento es una señal que envia un objeto de que algo ha pasado.
Podemos dividir los eventos en dos grandes grupos:
click
, select
, load
, mousedown
, keypress
, focus
...loadstart
, progress
(XMLHttpRequest), success
(GeoLocation), play
, volumechange
(video, audio)...gesturestart
, touchstart
, touchend
...zoomend
, movestart
, layeradd
, baselayerchange
...Las funciones que responden a los eventos se denominan events handlers o events listener (manejadores de eventos). Generalmente el nombre de la función es on
+nombre de evento
(onChange
, onLayerAdd
...).
Hay tres formas de asignar manejadores de eventos a elementos:
<img id="img1" src='logo.png' onclick="alert('Mensaje')">
document.getElementById('img1').onclick = function() {
alert('Mensaje')
};
addEventListener(evento, funcion, false)
:document.getElementById('img1').addEventListener('click', function() {
alert('Mensaje')
}, false);
removeEventListener()
elimina un manejador asignado a un elemento con addEventListener()
.
La función manejador recibe un parámetro, el objeto event
. Este objeto siempre tiene una propiedad type
que nos informa del tipo de evento. Dependiendo del tipo de evento puede tener propiedades adicionales. Por ejemplo el evento click
añade las propiedades target
(qué elemento ha recibido el click), clientX/clientY
(coordenadas del ratón en el momento del click)...
Los eventos se propagan (bubbling) de las hojas del árbol del DOM hacia la raiz. Mientras el evento se propaga hacia arriba todos los manejadores responderán al evento.
Para interrumpir la propagación, el manejador puede invocar el método stopPropagation()
.
La captura de un evento implica invertir el orden de propagación, de la raiz hacia las hojas. Se captura un evento con el método addEventListener()
definiendo el tercer parámetro a true
.
En la consola:
document.getElementById('miCelda').addEventListener('click', function() {
alert('Manejador de evento click en miCelda')
}, false);
Ajax (Asynchronous JavaScript and XML) es una técnica (NO una tecnología) que hace referencia al uso del objeto XMLHttpRequest
para comunicarse con servicios web del servidor. Es una colección de tecnologías (HTML, CSS, JavaScript, DOM, XMLHttpRequest, JSON/XML) para crear aplicaciones web asíncronas.
El término lo acuñó en el 2005 Jesse James Garrett en un artículo titulado Ajax: A New Approach to Web Applications y los primeros en utilizar estas técnicas en aplicaciones reales fueron Google con su GMail (2004) y, sobre todo, Maps (2005).
Modelo de aplicación clásico frente al uso de técnicas ajax:
Nota: este motor ajax es el objeto
XMLHttpRequest
,
Flujo de datos de un modelo síncrono frente a uno asíncrono:
XMLHttpRequest
Es un objeto de plataforma (implementado por el navegador) que permite realizar peticiones a servidores en segundo plano (de forma asíncrona, aunque se puede realizar también de forma síncrona) sin necesidad de recargar la página.
El procedimiento básico de uso es el siguiente:
XMLHttpRequest
En breve veremos un ejemplo.
Sus principales propiedades y métodos son:
Propiedades | Descripción |
---|---|
readyState |
Estado de la petición (4 completado) |
status |
Código HTTP devuelto por el servidor (200 ok) |
responseText |
Respuesta en formato texto |
responseXML |
Respuesta en formato XML (objeto DOM) |
Métodos | Descripción |
---|---|
onreadystatechange |
Referencia a función que se encarga de manejar el evento de cambio de estado de la petición |
open(GET/POST, url) |
Establecer parámetros de la petición |
send(data) |
Realiza la petición |
abort() |
Aborta la petición en curso |
Define una politica de seguridad de los navegadores que previene que el documento o código cargado de un sitio web (origen) pueda interactuar con un recurso de un origen distinto. Por origen se entiende la misma combinación de protocolo, host y puerto.
Qué recurso es lícito cargar desde otros orígenes?
<script src="..."></script>
<link rel="stylesheet" href="...">
<img>
<video>
y <audio>
<object>
, <embed>
y <applet>
@font-face
<frame>
y <iframe>
El objeto XMLHttpRequest
se ve afectado por esta política de seguridad, por lo tanto las aplicaciones Ajax se ven afectadas.
Procedimientos más comunes para realizar peticiones a otros orígenes (y saltarse la política de seguridad):
<script src="..."></script>
de distintos orígenes para pedir al servidor la respesta JSON encerrada en una función de callbackOrigin
(que indica el origen de la petición) y otra nueva de respuesta Access-Control-Allow-Origin
(que indica que orígenes son válidos), según la cual el navegador actúa en consecuenciaJSON es un estándard para el intercambio de datos entre aplicaciones web y servidores alternativo a XML, sencillo de leer y escribir para las personas, y sencillo de interpretar y generar para los ordenadores. Es el formato de intercambio más utilizado en las técnicas Ajax.
Originalmente deriva de JavaScript pero es un formato independiente del lenguaje de programación. Se utiliza también para serializar (guardar el estado) de objetos. Bases de datos como como MongoDb o CouchDB utilizan BJSON (JSON binario) para almacenar documentos.
A partir de JSON se derivan nuevos estándares como GeoJSON o TopoJSON, especificaciones para codificar colecciones de elementos geográficos, la segunda extensión de la primera a la que añade topología geoespacial.
Soporta únicamente los siguientes tipos de datos:
"
[]
{}
. El nombre debe encerrarse entre comillas dobles "
, separarse del valor por dos puntos :
y éste puede ser cualquier tipo soportado{
"message": "accurate",
"cod": "200",
"count": 1,
"list": [{
"id": 2512989,
"name": "Palma",
"coord": {
"lon": 2.65024,
"lat": 39.569389
},
"main": {
"temp": 15.39,
"pressure": 1016,
"humidity": 93,
"temp_min": 15,
"temp_max": 15.7
},
"dt": 1416944545,
"wind": {
"speed": 2.77,
"deg": 179
},
"sys": {
"country": "ES"
},
"clouds": {
"all": 20
},
"weather": [{
"id": 801,
"main": "Clouds",
"description": "algo de nubes",
"icon": "02n"
}]
}]
}
JavaScript implementa el objeto predefinido JSON
para convertir objetos a JSON y viceversa:
Métodos | Descripción |
---|---|
parse() |
Transforma texto JSON en sus correspondientes valores |
stringify() |
Transforma valores en texto JSON |
OpenWeatherMap es un servicio que proporciona datos meteorológicos abiertos de más de 200.000 ciudades del mundo desde su web y a través de un API.
Utilizaremos el API de OpenWeatherMap para este ejemplo y el proyecto de aplicación web del curso.
Contenido de la página HTML:
<body>
<p>
Ciudad: <input type="text" id="ciudad"> <input type="button" value="Buscar" onclick="getWeather()">
</p>
<div id="div_resp"></div>
</body>
La función getWeather
realiza la petición XMLHttpRequest
:
function getWeather() {
var query= document.getElementById('ciudad').value;
if (query === '') {
alert('Texto en blanco');
return;
}
var req = new XMLHttpRequest();
req.open('get', 'http://api.openweathermap.org/data/2.5/find?type=like&lang=es&units=metric&q=' + query);
// cada vez que cambia el estado de la petición...
req.onreadystatechange = function () {
if (req.readyState === 4) { // readyState 4 cuando ha finalizado la petición
if (req.status === 200) { // 200 ok
evalResp(req.responseText); // evalúa la respuesta
} else {
alert('Error: ' + req.status); // Ha ocurrido un error
}
}
}
req.send(); // envía petición
}
La función evalResp
trata la respuesta y actualiza parte del contenido de la página:
function evalResp(resp) {
var result = JSON.parse(resp),
div_resp= document.getElementById('div_resp'),
newHtml= '';
if (result.cod !== '200' || result.count === 0) {
div_resp.innerHTML= 'Ningún resultado';
} else {
for (var i= 0; i < result.list.length; i++) newHtml+= '<b>'+result.list[i].name+' ('+result.list[i].sys.country+'):</b> '+result.list[i].main.temp+' ºC<br>';
div_resp.innerHTML= newHtml;
}
}