Subida de ficheros en CakePHP 1.2 con uploadify y jQuery
Entrada actualizada a 18 de junio de 2009. Los cambios son desde aquí
Existe una entrada más reciente relacionada con este tema: Subida de ficheros con Uploadify y validación Ajax en CakePHP
Después de mucho tiempo liado con el fin de curso y tras mucho investigar con CakePHP por fin escribo algo al respecto. Para los que no lo sepáis, CakePHP es un framework de PHP que nos permite programar más rápido nuestras aplicaciones web PHP ya que nos ofrece las herramientas para que empecemos a escribir el código que realmente necesitamos: la lógica de la aplicación.
Si no conocíais CakePHP y os ha interesado el tema echad un ojo a su página web, descargaos una copia y empezad a hacer pruebas. En su web podréis encontrar la mayor parte de la documentación y todo lo demás, como no, en google.
En este artículo explicaré cómo integrar uploadify, un sistema de carga de ficheros con Flash y JavaScript (utilizando el framework jQuery) a CakePHP para permitirnos cargar múltiples ficheros sin tener que refrescar la página, así como poder cargar ficheros de gran tamaño.
Antes de empezar, sobreentiendo que tenéis conocimientos sobre CakePHP, así como que habéis utilizado algún sistema de carga de ficheros con flash, tipo uploadify o swfupload alguna vez en vuestra vida (aunque no haya sido con Cake). Tampoco estaría de más tener conocimientos de jQuery, para poder gestionar las subidas de ficheros de forma asíncrona. También asumo que ya tenéis CakePHP funcionando en algun servidor.
Dicho esto, empecemos…
Descargamos la última versión de jQuery (en mi caso la v. 1.3.2) y copiamos o movemos el fichero JavaScript a nuestro directorio de ficheros JavaScript, situado en la carpeta webroot (en /app/webroot/js).
Descargamos la última versión de uploadify (en mi caso la v. 1.6.2 GPL). Al descargar uploadify nos daremos cuenta que el paso anterior ha sido inútil dado que en la carpeta de uploadify ya viene jQuery :p estos fallos los tiene cualquiera… u.u
De la carpeta de uploadify nos interesan los siguientes ficheros:
- cancel.png
- jquery.uploadify.js
- uploader.swf
- uploadify.css
el resto eliminadlos (para lo que haremos no son necesarios, no obstante recomiendo que echéis un vistazo a todo ello). Renombrad la carpeta a “uploadify” y movedla dentro del directorio de JavaScripts de webroot (app/webroot/js). Opcionalmente podéis copiar el código del fichero uploadify.css a vuestro fichero css si lo preferís, de este modo podríais descartar también el fichero uploadify.css que apenas contiene 20 líneas.
Ahora que ya tenemos todos los JavaScripts necesarios, pasemos a la parte de los controladores y las vistas.
Para gestionar la subida de ficheros necesitaréis un Component (o bien programar toda la gestión de ficheros “a pelo” en vuestros controladores). Yo he utilizado “Image Upload Component“, que además de gestionarme la carga de cualquier tipo de fichero me permite recortar imágenes, generar miniaturas y otras acciones relacionadas con imágenes. Una vez descargado el Upload Component nos quedamos con el fichero upload.php que hay dentro del directorio /app/controllers/components y lo copiamos en este mismo directorio, pero de nuestro proyecto.
Aquí viene el código del controlador, en mi caso “images_controller”:
<?php
class ImagesController extends AppController
{
var $name = 'Images';
var $components = array('Upload');
var $helpers = array('Uploadify');
function beforeFilter()
{
// Si la acción es subir ficheros
if($this->action == 'upload'){
if (isset($this->params['pass'][0])){
// Iniciamos la sesión con el id de sesión pasado como parámetro
$this->Session->id($this->params['pass'][0]);
$this->Session->start();
}else{
$this->redirect('/');
}
}
// Cargamos el beforeFilter superior (en AppController o Controller)
parent::beforeFilter();
}
function upload()
{
// Desactivamos el rendering de la vista para este método
$this->autoRender = false;
if (isset($this->params['form']['Filedata'])){
// Creamos una miniatura
$thumb = $this->Upload->upload($this->params['form']['Filedata'],'img/thumb/', null, array('type' => 'resizecrop', 'size' => array('150', '150'), 'output' => 'jpg'));
// Si no se crea correctamente
if ($thumb)
// Generamos un log con los errores
$this->log("L'usuari " . $this->Auth->user('username') .
" ha tingut errors intentant crear una miniatura: " .
implode(" | ", $this->Upload->errors),'upload');
else{
// Si la miniatura se ha creado subimos el fichero a tamaño original
$result = $this->Upload->upload($this->params['form']['Filedata'],'img/',$this->Upload->result);
if (!$result){
// Si la imagen se sube correctamente enviamos el nombre de ésta al usuario
echo $this->Upload->result;
exit;
}else{
// En caso contrario generamos un log de error
$this->log("L'usuari " . $this->Auth->user('username') .
" ha tingut errors intentant pujar una imatge: " .
implode(" | ", $this->Upload->errors),'upload');
echo __("Error pujant el fitxer");
exit;
}
}
}
}
function add()
{
// Guardamos los datos en la base de datos
if(!empty($this->data)){
$this->Image->save($this->data);
}
}
}
Si no entendéis algo podéis referiros a la api y al libro de recetas de Cake o, en caso de ser con la generación de imágenes a la ayuda del Image Upload Component.
Como podéis ver he instanciado un helper que se llama “Uploadify”. Lo he creado yo para insertar el código JavaScript de uploadify en las vistas y si lo queréis lo podéis descargar desde aquí. Cuando llegue el momento de explicar las vistas pondré los ejemplos utilizando el código convencional y el código del helper.
Siguiendo con el controlador “images_controller”, el beforeFilter se encarga de iniciar la sesión en caso de que ésta no exista, a partir de un ID que le pasaremos desde la vista como primer parámetro. Sin estas líneas en que se inicia la sesión de nuevo seguramente tendríais problemas al enviar el fichero; éste se enviaría pero al llegar al 100% os daría un IO error (en mac y linux) o bien os incrustará la ventana de login por ahí en medio (en windows). Una vez iniciada la sesión se llama al beforeFilter superior para realizar las tareas pertinentes.
Las tareas del método upload están descritas en los comentarios y no creo que haga falta entrar en más detalles. Para más detalles, como he dicho antes, dirigíos a la ayuda de Cake o del Image Upload Component.
En el método add() hacemos las tareas necesarias para guardar los datos.
Nota: El envío de ficheros debéis hacerlo al mismo controlador. Si intentáis hacer un envío de ficheros desde el controlador “images” al controlador “files” (por ejemplo), NO FUNCIONARÁ. No sé porqué, así que si alguien encuentra el modo de hacerlo o sabe porqué pasa agradecería que me lo comentara por aquí.
Ya tenemos el controlador con su componente, ahora nos quedan únicamente la vista, pero antes explicaré cómo funciona el helper que he creado para aquellos que quieran utilizarlo…
Primero decir que es la primera “versión” (si es que se le puede llamar así), así que veréis que hay algún fallo o cosa sin terminar. Con el tiempo quizás lo mejore, aunque por el momento me funciona perfectamente tal como está.
Una vez iniciado el helper en el controlador únicamente tenéis que llamar a la función “startUploader” de dicho helper. Como primer parámetro debéis pasar un array asociativo con el id de la capa del uploader como clave y el directorio destino como valor de la clave. Lamentablemente el directorio de destino no está implementado en este ejemplo, pero utilizando un poco la cabeza seguro que podréis descubrir cómo utilizarlo ;).
Como segundo parámetro se le pasa un array, también asociativo, con las opciones de uploadify (que podéis mirar desde aquí). El tercer parámetro es por si ponéis varios uploadify en una misma página, para no volver a cargar los ficheros JavaScript y css; pasadle como parámetro “false” a partir del segundo uploader que añadáis.
Aquí tenéis el código que he utilizado para la vista:
<h2><?php __('Pujar imatges') ?></h2>
<?= $form->create('Image',array('action'=>'add')) ?>
<div id="imageFile"><?php __("Necessites JavaScript i Flash per poder pujar fitxers") ?></div>
<div id="uploaded" style="display:none">
<div id="files">
</div>
<?= $form->submit(__("Guardar",true)); ?>
</div>
<?= $html->link(__("Enviar",true),'javascript:$("#imageFile").fileUploadStart()'); ?>
<?=$form->end() ?>
Como veis he creado un formulario que envía a la acción add del controlador. Dentro de este he creado una primera capa donde se cargará uploadify (con la id “imageFile”) seguida de una capa donde irán apareciendo las imágenes (en la capa “files” para ser más exactos) que se vayan subiendo. Esta capa está oculta (style=”display:none”) ya que no nos interesa que el usuario la vea hasta que hayamos recibido el primer fichero.
Ahora que ya tenemos el formulario insertemos el JavaScript para que todo funcione (en la misma vista). Primero os pongo la metodología para el Helper y después el equivalente en HTML / JavaScript:
<?= $uploadify->startUploader(
array('imageFile'=>'img/'),
array('imageFile'=>array(
'buttonText'=>__('Cercar fitxers',true),
'script'=>'upload/' . $session->id(),
'fileExt'=>'*.jpg;*.jpeg;*.png;*.gif',
'fileDesc'=>'Fitxers d\\\'imatge',
'multi'=>'true',
'onError'=>'function (a, b, c, d) {
if (d.status == 404)
alert(\'Could not find upload script. Use a path relative to: ' . getcwd() . '\');
else if (d.type === "HTTP")
alert(\'error \'+d.type+\': \'+d.status);
else if (d.type ==="File Size")
alert(c.name+\' \'+d.type+\' Limit: \'+Math.round(d.sizeLimit/1024)+\'KB\');
else alert(\'error \'+d.type+\': \'+d.text);}',
'onComplete'=>'function(evt, queueID, fileObj, response, data){$("#uploaded").show();$("#uploaded #files").append(\'' . $html->image("thumb/'+response+'") . '<input type="text" value="\'+response+\'" /><input type="hidden" value="\'+response+\'" />\');}',
'onAllComplete'=>'function(){$("input[type=\\\'submit\\\']").removeAttr("disabled");}'))) ?>
Equivalente en JavaScript:
$(document).ready(function() {
$('#imageFile').fileUpload ({
'uploader' : '/js/uploadify/uploader.swf',
'script' : 'upload/<?= $session->id() ?>',
'buttonText' : 'Cercar fitxers',
'onError' : function (a, b, c, d) {
if (d.status == 404)
alert('Could not find upload script. Use a path relative to: <?= getcwd() ?>');
else if (d.type === "HTTP")
alert('error '+d.type+': '+d.status);
else if (d.type ==="File Size")
alert(c.name+' '+d.type+' Limit: '+Math.round(d.sizeLimit/1024)+'KB');
else alert('error '+d.type+': '+d.text);},
'onComplete' : 'function(evt, queueID, fileObj, response, data){$("#uploaded").show();$("#uploaded #files").append('<img src="/img/thumb/'+response+'" alt="" /><input type="text" value="'+response+'" /><input type="hidden" value="'+response+'" />');},
'onAllComplete' : 'function(){$("input[type=\'submit\']").removeAttr("disabled");}',
'cancelImg' : '/js/uploadify/cancel.png',
'fileExt' : '*.jpg;*.jpeg;*.png;*.gif',
'fileDesc' : 'Fitxers d\'imatge',
'multi' : 'true',
'folder': 'img/'
});
});
Sobre todo fijaros en que le paso como segundo parámetro a la url “upload” la id de la sesión actual. Esto junto con el beforeFilter del controlador puede ser vital para que os funcione correctamente uploadify con Cake.
He declarado tres funciones a ejecutar con uploadify: onError, onComplete y onAllComplete. La primera es por si sucede algún error con la carga del fichero, es simplemente para depuración pero nunca está de más tenerlo. Una vez os funcione correctamente la subida de ficheros podéis eliminarlo sin miedo alguno.
La función onComplete se encarga de hacer visible la capa “uploaded” y de ir insertando en ella las miniaturas de las imágenes que se vayan subiendo (ubicadas en la carpeta /webroot/img/thumb/). A demás de insertar las imágenes genera un textbox con el nombre del fichero, para que el usuario pueda poner el nombre deseado y un campo oculto con el nombre del fichero (para poder guardar su ruta en la base de datos).
Finalmente la función onAllComplete se encarga de eliminar el atributo “disabled” del botón de envío del formulario.
Aquí tenéis la vista al completo con el helper:
<?= $uploadify->startUploader(
array('imageFile'=>'img/'),
array('imageFile'=>array(
'buttonText'=>__('Cercar fitxers',true),
'script'=>'upload/' . $session->id(),
'fileExt'=>'*.jpg;*.jpeg;*.png;*.gif',
'fileDesc'=>'Fitxers d\\\'imatge',
'multi'=>'true',
'onError'=>'function (a, b, c, d) {
if (d.status == 404)
alert(\'Could not find upload script. Use a path relative to: ' . getcwd() . '\');
else if (d.type === "HTTP")
alert(\'error \'+d.type+\': \'+d.status);
else if (d.type ==="File Size")
alert(c.name+\' \'+d.type+\' Limit: \'+Math.round(d.sizeLimit/1024)+\'KB\');
else alert(\'error \'+d.type+\': \'+d.text);}',
'onComplete'=>'function(evt, queueID, fileObj, response, data){$("#uploaded").show();$("#uploaded #files").append(\'' . $html->image("thumb/'+response+'") . '<input type="text" value="\'+response+\'" /><input type="hidden" value="\'+response+\'" />\');}',
'onAllComplete'=>'function(){$("input[type=\\\'submit\\\']").removeAttr("disabled");}'))) ?>
<h2><?php __('Pujar imatges') ?></h2>
<?= $form->create('Image',array('action'=>'add')) ?>
<div id="imageFile"><?php __("Necessites JavaScript i Flash per poder pujar fitxers") ?></div>
<div id="uploaded" style="display:none">
<div id="files">
</div>
<?= $form->submit(__("Guardar",true)); ?>
</div>
<?= $html->link(__("Enviar",true),'javascript:$("#imageFile").fileUploadStart()'); ?>
<?=$form->end() ?>
Y con Javascript:
<script type="text/javascript">
$(document).ready(function() {
$('#imageFile').fileUpload ({
'uploader' : '/js/uploadify/uploader.swf',
'script' : 'upload/<?= $session->id() ?>',
'buttonText' : 'Cercar fitxers',
'onError' : function (a, b, c, d) {
if (d.status == 404)
alert('Could not find upload script. Use a path relative to: <?= getcwd() ?>');
else if (d.type === "HTTP")
alert('error '+d.type+': '+d.status);
else if (d.type ==="File Size")
alert(c.name+' '+d.type+' Limit: '+Math.round(d.sizeLimit/1024)+'KB');
else alert('error '+d.type+': '+d.text);},
'onComplete' : 'function(evt, queueID, fileObj, response, data){$("#uploaded").show();$("#uploaded #files").append('<img src="/img/thumb/'+response+'" alt="" /><input type="text" value="'+response+'" /><input type="hidden" value="'+response+'" />');},
'onAllComplete' : 'function(){$("input[type=\'submit\']").removeAttr("disabled");}',
'cancelImg' : '/js/uploadify/cancel.png',
'fileExt' : '*.jpg;*.jpeg;*.png;*.gif',
'fileDesc' : 'Fitxers d\'imatge',
'multi' : 'true',
'folder': 'img/'
});
});
</script>
<h2><?php __('Pujar imatges') ?></h2>
<?= $form->create('Image',array('action'=>'add')) ?>
<div id="imageFile"><?php __("Necessites JavaScript i Flash per poder pujar fitxers") ?></div>
<div id="uploaded" style="display:none">
<div id="files">
</div>
<?= $form->submit(__("Guardar",true)); ?>
</div>
<?= $html->link(__("Enviar",true),'javascript:$("#imageFile").fileUploadStart()'); ?>
<?=$form->end() ?>
Vale, recordemos un poco todo lo que hemos hecho para ver que no nos hemos dejado nada:
- Hemos descargado jQuery y Uploadify
- Hemos guardado los ficheros que nos interesaban de ambas librerías en nuestro proyecto
- Hemos creado el controlador (/app/controllers/images_controller.php)”
- Hemos creado la vista (/app/view/images/add.ctp)
- Algo que no he dicho (pero que es bastante lógico..) es haber creado una carpeta donde se guardarán los ficheros, con permisos de escritura (755, 777… en sistemas UNIX/Linux)
- Tampoco he dicho que hay que crear un modelo (/app/models/image.php), pero si tenéis conocimientos de Cake seguro que ya lo sabíais 😉
Para terminar os explicaré cómo funcionará el proceso de envío. Aquí tendría que hacer un diagrama de estados o algo así pero la verdad es que no apetece nada… :p
- El usuario accederá a la página “images/add” y se le mostrará un botón con el que subir imágenes.
- Una vez seleccionado(s) el/los fichero(s) a subir, el usuario le da a “enviar” y se inicia la transferencia del éste.
- Una vez ha terminado el fichero y ha llegado correctamente al servidor, este/estos se procesan y en caso de éxito se devuelve el/los nombre(s) de fichero(s) resultante(s) al usuario. A partir de este/estos nombre(s) de fichero(s) nos encargaremos de que vea una(s) miniatura(s) de la(s) imagen(es) subida(s) (qué pesadito con el plural, eh?).
- A demás de las miniaturas le mostraremos un textbox al lado de cada una para que pueda poner el nombre si quiere.
- Cuando haya editado todos los nombres le dará al botón “guardar” que aparecerá a partir del primer fichero subido pero que no estará activo hasta que todos los ficheros hayan subido al servidor.
- Aquí hay una cosa que yo no hago en el ejemplo porque no me quiero complicar pero que creo conveniente que hagáis si no queréis llenar vuestro servidor de ficheros sin uso. Sería necesario guardar las rutas de ficheros una vez subidos por si el usuario no le diera a “guardar”. De éste modo podríais hacer un método con el que eliminar ficheros inutilizados fácilmente.
Pues ahí lo tenéis. Creo que me ha salido una guía algo pobre es lo que tiene hacerla con algo de prisa… Cuando tenga algo más de tiempo intentaré colgar un ejemplo sobre esto.
Si has terminado este tutorial satisfactoriamente mírate este otro: Subida de ficheros con Uploadify y validación Ajax en CakePHP
Espero que no tengáis muchas dudas. De todos modos sabéis que estoy abierto a preguntas a través de los comentarios si os surge cualquier duda!
Páginas de referencia:
Hola, muy buen post, antes que nada, la duda que tengo es donde colocar el código Javascript o el codigo php del helper, a partir de ahi me lio. Gracias
@raquel
Me alegra mucho ver que le sirve a alguien ^^
Ahora que me lo comentas veo que no he explicado muy bien ese punto. Tienes que colocar el código en la vista; vuelve a mirarte la entrada que la he ampliado para explicarme mejor. He ampliado desde después de haber insertado el código Javascript.
Confío en que ahora te quede más claro
Salu2 y gracias por ayudarme a mejorar el apunte!
PS. Si aún así tienes alguna duda no tengas reparo en preguntar 😉
Muchisimas gracias, ya lo resolvi ayer. Enhorabuena por ello, essta muy bien y la verdad que se agradece que la gente comparta sus conocimientos.
Un saludo!!
Hola, funciona muy bien, gracias, pero a la hora de editar el logo si uno trata de subir un logo no funciona bn, en vez de insertar la imagen lo que hace es insertar es la misma pagina dentro del div, agradeceria la ayuda que me puedan prestar. Gracias
Hola Christian, mira que no tengas activado el debug en el fichero de configuración core.php
A parte de eso no se me ocurre qué puede ser..
Hola, mira mi problema es el siguiente:
Hice todo como estas en el tutorila pero lo único que me aparece es
“Pujar imatges
Necessites JavaScript i Flash per poder pujar fitxers”
¿Tengo que instalar algún plugin en mi sistema para que ejecute el flash, o es otro el problema?
Gracias.
Eso puede ser por varias razones pero todas ellas son similares. Este mensaje aparece cuando no tienes el plugin de flash player instalado o bien cuando no tienes JavaScript activado (éste último también puede estar ocasionado por no haber cargado la librería jQuery). Mírate el código de la página a ver si has cargado en el head de la página la librería en cuestión- Si está cargada la librería el error puede estar ocasionado por algo en tu código de inicio de Uploadify, revísalo utilizando el plugin firebug para ver que Uploadify no da ningún fallo y éste carga correctamente.
Espero que te sirva
Salud
Muchas gracias por el tuto, me guardé la página en favoritos cuando la ví y ahora que estoy aprendiendo cake me ha servido de gran ayuda.
Felicidades por la web. Un saludo de un ex-compañero de clase 😉
Hola muy bueno el post, aunque ha pasado tiempo de que lo publicaste, espero me puedas ayudar, implemente todo sin mayor problemas, claro que en un formulario con otros datos, la imagen se vizualiza bien y se carga en el server, mi problema es que en la base de datos coloca toda la información del formulario menos la información la foto
espero me puedas ayudar Gracias
Hola @Rene. Primero de todo te pido disculpas por la tardanza. Llevo bastante tiempo tan liado que apenas tengo tiempo de pasarme por aquí (y aun menos de contestar algunas de las preguntas que van cayendo de vez en cuando por estos lares..).
Lo mejor que puedes hacer en estos casos (cuando algo debería guardarse y no lo hace) es impregnar tu código JS de “console.log” (método para mostrar mensajes por consola de Firebug), así como tu código PHP con “$this->log” (método de CakePHP para guardar logs en app/tmp/logs). De esta manera, paso por paso y con paciencia, seguramente averiguarás dónde está fallando tu código.
Como comprenderás, con tan pocos detalles no puedo ayudarte mucho más; espero que tengas suerte y lo soluciones y, de ser así, no estaría de más que me lo comentaras e incluso que pusieras por aquí la solución que encontraste ;).
Por otro lado, te recomiendo que eches un vistazo a un tema al respecto bastante más reciente que este (aunque más complicado..): http://racotecnic.underave.net/2009/10/subida-de-ficheros-con-uploadify-y-validacion-ajax-en-cakephp/
Salud
Oye cuales son los campos de la base de datos me podrias decir porfa……
saludos…..
Dany
Pues los que tú necesites @dany. Éste tutorial es totalmente abierto en este sentido. De hecho si te fijas en el momento de guardar no hago nada porque esa parte se supone que la hacéis vosotros.
De todos modos, hice un tutorial algo más avanzado a éste donde sí muestro los campos de la BD.
Puedes ver ese tutorial desde aquí:
http://racotecnic.underave.net/2009/10/subida-de-ficheros-con-uploadify-y-validacion-ajax-en-cakephp/
Hola, ha pasado un poco de tiempo pero alomejor veis este comentario.
He seguido el tutorial y me funciona perfecto. Ahora quiero limitar el numero de imagenes que puede subir el usuario (he visto que uploadify tiene la opcion queueSizeLimit que limita la cola de imagenes). El problema es que no se donde ponerlo, he probado a añadirlo entre las opciones del helper pero no me hace caso.
Cual seria el sitio idoneo para poner este parametro??
Muchas gracias de antemano
Hola Roger, como este tutorial, el helper lo hice hace mucho tiempo y estoy bastante seguro que no será la mejor forma de insertar uploadify en tu site. Te recomiendo que insertes spotify directamente en tu vista, utilizando los métodos link y scriptBlock del helper de Javascript (script y scriptBlock en el helper Html de Cake 1.3).
También te recomiendo que te leas alguna entrada más reciente sobre el tema, como puede ser esta:
http://www.racotecnic.com/2009/10/subida-de-ficheros-con-uploadify-y-validacion-ajax-en-cakephp/
Muchas gracias, al final he utilizado directamente el script jquery con la nueva version de uploadify y, aunque me ha costado un poco hacerlo funcionar ahora esta perfecto.
Un saludo