Hace ya un tiempo Destroy (a.k.a Axelain) publicó como eliminar este mensaje tan molesto que todos los que tenemos una copia ilegal de windows (es decir, todos) sufrimos o hemos sufrido alguna vez.

Él explicó cómo eliminarlo, algo realmente útil para cuando el mal ya está hecho y no hay vuelta atrás, pero yo creo que estas cosas mejor prevenirlas que curarlas, así que os voy a explicar cómo he evitado siempre estos mensajes:

Como ya he dicho esto es la medida de prevención, por ello necesitaremos una instalación de Windows semi nueva que todavía no esté “infectada” por este mensaje.

Vayamos a Mi PC / Panel de Control / Actualizaciones automáticas y habilitemos la casilla “Notificarme, pero no descargarlas ni instalarlas…”

Actualizaciones de windows

¿Porqué hacemos esto? Porque el mensaje de Windows Original no es más que una actualización del sistema, si evitamos descargarla e instalarla ya no tendremos que preocuparnos más por ello (sólo intenta instalarse una vez).

Así pues tendremos que fijarnos a la hora de instalar las actualizaciones. La actualización en concreto se llama “Herramienta de notificación del programa de ventajas de Windows Original”.

Para Office pasa igual, existe una actualización que si se descarga nos hace igual que con Windows. La actualización se llama igual, pero en lugar de Windows, Office ;)

Cuando hayamos desmarcado las actualizaciones que no queramos le damos a descargar y, en la nueva ventana que nos aparezca, marcamos “No volver a notificarme sobre estas actualizaciones”.

Si, como yo, estáis interesados en editar y leer las etiquetas ID3 de los ficheros mp3 que se suban a vuestro servidor, podéis hacerlo utilizando las funciones propias de PHP para ello (necesitáis que vuestro hosting lo tenga instalado) o bien, si os pasa igual que a mi que en mi hosting no tienen habilitadas estas funciones (y si no tienes un servidor dedicado no te las instalarán para no detener el servicio), podéis utilizar las classes de PHP GetID3.

Si no la conocíais os recomiendo que la descarguéis y hagáis alguna prueba con ella, realmente se le puede sacar mucho jugo ya que con ella no sólo podemos editar etiquetas ID3; esto son algunos de los muchos formatos cuya información podéis consultar y / o editar con este magnífico conjunto de clases:

getID3() is a PHP script that extracts useful information from MP3s & other multimedia file formats:

He eliminado unos cuantos formatos de la lista, si queréis verlos todos id a su página web.

¡Al grano! Ya conocemos las clases ID3 (aunque sea un poco por encima, como yo..) y queremos implementarlas en nuestra aplicación de Cake.

No es muy complicado si utilizamos las demos que nos dan en el zip que nos descargamos de getID3.org. Simplemente debemos tener en cuenta que los “require” que utilicen en las demos debemos substituirlas por App::import(‘vendor’,’rutadelaclase’).

Lo primero que deberemos hacer será descargar getID3. Si queréis editar etiquetas ID3v2 deberéis descargar la versión estable (actualmente la 1.7.9) ya que en la versión beta (actualmente la 2.0.0-b5) no funcionan las clases necesarias.

Una vez descargado, lo descomprimimos y lo subimos a la carpeta vendors de nuestro proyecto Cake (para mayor comodidad renombrar la carpeta a “getid3”). Seguramente habréis visto que existen dos carpetas vendor en vuestra aplicación de Cake. Debéis ponerlo en la que se encuentra en la carpeta raíz (junto con las carpeta “app” y “cake”).

Ahora que ya tenemos nuestro plugin en la aplicación debemos crear un componente que nos sirva de conexión entre las librerías getID3 y nuestro Cake.

Para hacerlo utilizaremos como referencia las demos “demo.basic.php” y “demo.simple.write.php”. Recordad que hay que substituir los require e includes por App::import(). Este es el resultado que he obtenido yo:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
<?php
class Getid3Component extends Object
{
  function __construct() {
    set_time_limit(20 * 3600);
    ignore_user_abort(false);
  }

  function extract($filename) {
    // Importamos el fichero getid3.php que contiene la classe getID3
    App::import('vendor', 'getid3/getid3', array('file' => 'getid3.php'));
    // Initialize getID3 engine
    $getID3 = new getID3;

    // Analyze file and store returned data in $ThisFileInfo
    $ThisFileInfo = $getID3->analyze($filename);
    // Devolvemos un array con toda la información del fichero
    return $ThisFileInfo;
  }

  function write($filename, $data)
  {
    App::import('vendor', 'getid3/getid3/getid3');

    // Initialize getID3 engine
    $getID3 = new getID3;
    // Indicamos a getID3 que utilice la codificación de Cake
    $getID3->setOption(array('encoding' => Configure::read('App.encoding')));

    App::import('vendor', 'getid3/getid3', array('file' => 'write.php'));

    // Initialize getID3 tag-writing module
    $tagwriter = new getid3_writetags;

    $tagwriter->filename       = $filename;
    $tagwriter->tagformats     = array('id3v1', 'id3v2.3');

    // set various options (optional)
    $tagwriter->overwrite_tags = true;
    $tagwriter->tag_encoding   = Configure::read('App.encoding');
    $tagwriter->remove_other_tags = true;

    // populate data array
    $TagData['title'][]   = $data['title'];
    $TagData['artist'][]  = $data['artist'];
    $TagData['album'][]   = $data['album'];
    $TagData['year'][]    = $data['year'];
    $TagData['genre'][]   = $data['genre'];
    $TagData['comment'][] = $data['comment'];
    $TagData['track'][]   = $data['track'];

    $tagwriter->tag_data = $TagData;

    // write tags
    if ($tagwriter->WriteTags()) {
      if (!empty($tagwriter->warnings)) {
        return $tagwriter->warnings;
      }
      return true;
    } else {
      return $tagwriter->errors;
    }
  }
}

Recordad que los componentes van en la carpeta app/controllers/components y que el nombre de este fichero deberá ser getid3.php.

Ahora que ya tenemos nuestro componente vayamos al controlador sobre el que queramos utilizarlo y añadamoslo al resto de componentes:

1
2
3
4
5
6
7
<?php
class AudiosController extends AppController
{
  var $name = 'Audios';
  var $components = array('Upload', 'Getid3');
  // [...]
}

Y sólo nos queda saber cómo utilizarlo:

1
2
3
4
5
6
7
8
9
$datos = array(
  'album' => 'Nombre del álbum',
  'title' => 'Título del tema',
  'artist' => 'Artista',
  'year' => 'Año',
  'genre' => 'Estilo',
  'comment' => 'Comentario'
);
$this->Getid2->write('rutadelfichero.mp3', $datos);

Y para leer los datos de un mp3:

1
$this->Getid2->extract('rutadelfichero.mp3')

Que nos dará una salida similar a ésta.

Habéis visto más o menos el método de implementar funciones de getID3 en CakePHP. Ahora es tarea vuestra intentar añadir más funcionalidades a vuestro complemento según vuestras necesidades.

Como siempre, espero que le sirva a alguien!

Páginas de referencia:

Salida de mi fichero de prueba mp3:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
Array
(
    [GETID3_VERSION] => 1.7.9-20090308
    [filesize] => 263776
    [avdataoffset] => 4096
    [avdataend] => 263648
    [fileformat] => mp3
    [audio] => Array
        (
            [dataformat] => mp3
            [channels] => 2
            [sample_rate] => 44100
            [bitrate] => 128000
            [channelmode] => stereo
            [bitrate_mode] => cbr
            [lossless] =>
            [encoder_options] => CBR128
            [compression_ratio] => 0.0907029478458
            [streams] => Array
                (
                    [0] => Array
                        (
                            [dataformat] => mp3
                            [channels] => 2
                            [sample_rate] => 44100
                            [bitrate] => 128000
                            [channelmode] => stereo
                            [bitrate_mode] => cbr
                            [lossless] =>
                            [encoder_options] => CBR128
                            [compression_ratio] => 0.0907029478458
                        )

                )

        )

    [tags] => Array
        (
            [id3v1] => Array
                (
                    [title] => Array
                        (
                            [0] => Títol del tema
                        )

                    [artist] => Array
                        (
                            [0] => Jo mateix
                        )

                    [album] => Array
                        (
                            [0] => Muahahahahah
                        )

                    [year] => Array
                        (
                            [0] => 2009
                        )

                    [comment] => Array
                        (
                            [0] => from www.underave.net
                        )

                    [genre] => Array
                        (
                            [0] => Techno
                        )

                )

            [id3v2] => Array
                (
                    [title] => Array
                        (
                            [0] => Títol del tema
                        )

                    [artist] => Array
                        (
                            [0] => Jo mateix
                        )

                    [album] => Array
                        (
                            [0] => Muahahahahah
                        )

                    [year] => Array
                        (
                            [0] => 2009
                        )

                    [genre] => Array
                        (
                            [0] => Techno
                        )

                    [comments] => Array
                        (
                            [0] => from www.underave.net
                        )

                )

        )

    [encoding] => ISO-8859-1
    [filename] => Bong0.mp3
    [filepath] => /public_html/waste/v3.2/app/webroot/files/mp3
    [filenamepath] => /public_html/waste/v3.2/app/webroot/files/mp3/Bong0.mp3
    [id3v2] => Array
        (
            [header] => 1
            [flags] => Array
                (
                    [unsynch] =>
                    [exthead] =>
                    [experim] =>
                )

            [majorversion] => 3
            [minorversion] => 0
            [headerlength] => 4096
            [tag_offset_start] => 0
            [tag_offset_end] => 4096
            [encoding] => UTF-8
            [comments] => Array
                (
                    [title] => Array
                        (
                            [0] => Títol del tema
                        )

                    [artist] => Array
                        (
                            [0] => Jo mateix
                        )

                    [album] => Array
                        (
                            [0] => Muahahahahah
                        )

                    [year] => Array
                        (
                            [0] => 2009
                        )

                    [genre] => Array
                        (
                            [0] => Techno
                        )

                    [comments] => Array
                        (
                            [0] => from www.underave.net
                        )

                )

            [TIT2] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => TIT2
                            [frame_flags_raw] => 0
                            [data] => ��T���t�o�l� �d�e�l� �t�e�m�a�
                            [datalength] => 31
                            [dataoffset] => 10
                            [framenamelong] => Title/songname/content description
                            [framenameshort] => title
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                        )

                )

            [TPE1] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => TPE1
                            [frame_flags_raw] => 0
                            [data] => ��J�o� �m�a�t�e�i�x�
                            [datalength] => 21
                            [dataoffset] => 51
                            [framenamelong] => Lead performer(s)/Soloist(s)
                            [framenameshort] => artist
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                        )

                )

            [TALB] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => TALB
                            [frame_flags_raw] => 0
                            [data] => ��M�u�a�h�a�h�a�h�a�h�a�h�
                            [datalength] => 27
                            [dataoffset] => 82
                            [framenamelong] => Album/Movie/Show title
                            [framenameshort] => album
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                        )

                )

            [TYER] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => TYER
                            [frame_flags_raw] => 0
                            [data] => ��2�0�0�9�
                            [datalength] => 11
                            [dataoffset] => 119
                            [framenamelong] => Year
                            [framenameshort] => year
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                        )

                )

            [TCON] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => TCON
                            [frame_flags_raw] => 0
                            [data] => ��T�e�c�h�n�o�
                            [datalength] => 15
                            [dataoffset] => 140
                            [framenamelong] => Content type
                            [framenameshort] => genre
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                        )

                )

            [COMM] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => COMM
                            [frame_flags_raw] => 0
                            [data] => ��f�r�o�m� �w�w�w�.�u�n�d�e�r�a�v�e�.�n�e�t�
                            [datalength] => 50
                            [dataoffset] => 165
                            [framenamelong] => Comments
                            [framenameshort] => comments
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                            [language] => eng
                            [languagename] => English
                            [description] =>
                        )

                )

            [TRCK] => Array
                (
                    [0] => Array
                        (
                            [frame_name] => TRCK
                            [frame_flags_raw] => 0
                            [data] =>
                            [datalength] => 1
                            [dataoffset] => 225
                            [framenamelong] => Track number/Position in set
                            [framenameshort] => track_number
                            [flags] => Array
                                (
                                    [TagAlterPreservation] =>
                                    [FileAlterPreservation] =>
                                    [ReadOnly] =>
                                    [compression] =>
                                    [Encryption] =>
                                    [GroupingIdentity] =>
                                )

                            [encodingid] => 1
                            [encoding] => UTF-16
                        )

                )

            [padding] => Array
                (
                    [start] => 236
                    [length] => 3860
                    [valid] => 1
                )

        )

    [id3v1] => Array
        (
            [title] => Títol del tema
            [artist] => Jo mateix
            [album] => Muahahahahah
            [year] => 2009
            [comment] => from www.underave.net
            [genre] => Techno
            [comments] => Array
                (
                    [title] => Array
                        (
                            [0] => Títol del tema
                        )

                    [artist] => Array
                        (
                            [0] => Jo mateix
                        )

                    [album] => Array
                        (
                            [0] => Muahahahahah
                        )

                    [year] => Array
                        (
                            [0] => 2009
                        )

                    [comment] => Array
                        (
                            [0] => from www.underave.net
                        )

                    [genre] => Array
                        (
                            [0] => Techno
                        )

                )

            [padding_valid] => 1
            [tag_offset_end] => 263776
            [tag_offset_start] => 263648
            [encoding] => ISO-8859-1
        )

    [mime_type] => audio/mpeg
    [mpeg] => Array
        (
            [audio] => Array
                (
                    [raw] => Array
                        (
                            [synch] => 4094
                            [version] => 3
                            [layer] => 1
                            [protection] => 0
                            [bitrate] => 9
                            [sample_rate] => 0
                            [padding] => 0
                            [private] => 0
                            [channelmode] => 0
                            [modeextension] => 0
                            [copyright] => 0
                            [original] => 1
                            [emphasis] => 0
                        )

                    [version] => 1
                    [layer] => 3
                    [channelmode] => stereo
                    [channels] => 2
                    [sample_rate] => 44100
                    [protection] => 1
                    [private] =>
                    [modeextension] =>
                    [copyright] =>
                    [original] => 1
                    [emphasis] => none
                    [crc] => 54304
                    [padding] =>
                    [bitrate] => 128000
                    [framelength] => 417
                    [bitrate_mode] => cbr
                )

        )

    [playtime_seconds] => 16.222
    [tags_html] => Array
        (
            [id3v1] => Array
                (
                    [title] => Array
                        (
                            [0] => T&amp;iacute;tol del tema
                        )

                    [artist] => Array
                        (
                            [0] => Jo mateix
                        )

                    [album] => Array
                        (
                            [0] => Muahahahahah
                        )

                    [year] => Array
                        (
                            [0] => 2009
                        )

                    [comment] => Array
                        (
                            [0] => from www.underave.net
                        )

                    [genre] => Array
                        (
                            [0] => Techno
                        )

                )

            [id3v2] => Array
                (
                    [title] => Array
                        (
                            [0] => T&amp;#237;tol del tema
                        )

                    [artist] => Array
                        (
                            [0] => Jo mateix
                        )

                    [album] => Array
                        (
                            [0] => Muahahahahah
                        )

                    [year] => Array
                        (
                            [0] => 2009
                        )

                    [genre] => Array
                        (
                            [0] => Techno
                        )

                    [comments] => Array
                        (
                            [0] => from www.underave.net
                        )

                )

        )

    [bitrate] => 128000
    [playtime_string] => 0:16
)

Este artículo proviene del audioblog, clausurado por su desuso y cuyas entradas han sido movidas a este blog para que no se pierdan en el olvido. Entrada escrita originalmente por 909.

¿Quién alguna vez en su vida, como productor, ha cogido un libro sobre audio, síntesis o mastering y no podido pasar de la página diez? (yo soy uno de ellos) Pues la empresa M-Audio nos ha hecho un favor, Y QUE FAVOR!, al desarrollar **Overdub: **Una guia para productores con una estética de cómic.

Este cómic-guia esta pensando sobretodo para newcomers, es decir, si acabas de aterrizar en este mundillo no estaría mal que le dedicaras un rato de tu tiempo. Igualmente queda dicho para la gente con más experiencia/conocimientos en la materia. Ya que, como dicen en su página web: Overdub cubre los conceptos base y también trucos/consejos más avanzados.

El único inconveniente es que esta solo en inglés y que solo han sacado dos números. Pero bueno, tiempo al tiempo, ya irán apareciendo más. También comentaros que Overdub se encuentra tanto en formato digital (más abajo están los links) y en papel. Pero, para que gastar papel y dinero cuando solo ocupa unos pocos bytes?

Aquí os dejo los dos números publicados hasta dia de hoy de Overdub.

Nº1: “Guia sobre monitores de estudio”

Descargar nº1 en PDF

Nº2: “Todo sobre las interfaces de audio”

Descargar nº2 en PDF

Para visualizar estos documentos online y descargar los tonos de prueba que hace mención el primer capitulo de este cómic, visita la web: Overdub web

Espero que os ayude un poco a aclarar alguna duda o que os de alguna que otra sorpresilla, de estas que te quedas: Ostia! Esto no lo sabía.

Zalú’!

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”:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
<?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:

1
2
3
4
5
6
7
8
9
10
<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:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
<?=
$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:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
$(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:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
<?= $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:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
<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:

Com tots sabreu, constantment van apareixent i desapareixen maneres de burlar les restriccions de pàgines de videos i descàrregues. Avui vinc a presentar Illimitux, un complement per a firefox que ens ajudarà a saltar algunes restriccions de Megavideo, Megaupload, Veoh i Rapidshare; així com FreeRapid Downloader que ens farà més còmodes les descàrregues.

Ara anem a parlar de Illimitux. Abans de començar, cal avisar que és un complement en fase de proves i que no és del tot estable, intentarem aclarir en aquest post com configurar-lo perquè ens funcioni el més bé possible. Les indicacions són per a la versió actual (2.0.2) esperem que en un futur pròxim en millorin l’estabilitat.

Per instal·lar-lo accedim a la pàgina de descàrrega, seleccionem conforme som conscients que és experimental i l’afegim al nostre Firefox. Una vegada s’hagi descarregat, haurem de reiniciar el programa.

Una vegada instal·lat ja podeu accedir a les diferents pàgines de descàrrega o streaming i saltar-vos les limitacions. A continuació explico quines limitacions elimina:

Megavideo: Per a megavideo redirecciona les pàgines a illimitux.net i les pots visualitzar sense problemes i descarregar l’arxiu original. Permet la visualització amb el reproductor de illimitux o amb el de Megavideo.

Veoh: Anula la restricció de temps de la web i permet visualitzar les pel·lícules sense el programa Veoh TV. No permet, però. avançar fins al punt desitjat. La visualització és del tirón i només pots pausar/reanudar.

Megaupload: Permet les descàrregues sense limitació de temps i sense límit de descàrregues. Per funcionar correctament no pots tenir activada la opció de “unbridle videos automatically” ja que no funcionarà correctament.

Rapidshare: Enllaça directament les pàgines i no has de fer la combinació de click’s, però realment no evita cap limitació.

Amb tot això us apareixerà a la part inferior de la pàgina una finestra similar a la següent:

Tan sols heu de clicar a “Remove limitation” i us rediccionarà a una pàgina on gaudireu dels avantatges ja citats.

Ara li toca el torn a FreeRapid Downloader, un software lliure disponible per a Windows, Mac i Linux. Programa, que és disponible en castellà i al que se li poden afegir pluggins.

Aquesta aplicació tampoc elimina cap limitació real, però ens permet fer una llista de url’s que ens descarregarà de forma automàtica (respectant les limitacions de temps de cada pàgina). Per a descarregar accedim a la pàgina oficial de descàrrega i cliquem (usuaris de windows) a l’icona del disc dur amb la fletxeta.

Descarreguem el fitxer i el descomprimim. Per executar-lo solament hem de obrir el fitxer frd.exe

Com veureu, bo i que pot ser un programa molt complert també és molt intuïtiu.

Cliquem a “añadir url” i obrira una finestra com la següent; només hem de seleccionar la carpeta on descarregarà els fitxers, enganxar la llista de enllaços i clicar a “empezar”

I la cosa quedarà finalment així:

A aquest enllaç teniu la llista de servidor compatibles amb el servei.

Espero que tot aixó us sigui de utilitat ;) fins haviat!