/*
videogrid external for Puredata
Copyright (C) 2007 Sergi Lario
sll :: slario-at-gmail.com
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#include
#include
#include
#include
#include
/* incloure les definicions de variables i
prototipus de dades i de funcions de puredata */
#include "m_pd.h"
/* incloure estructures de dades i capceleres de funcions gàfiques bàsiques de pd */
#include "g_canvas.h"
/* incloure estructures de dades i capceleres de funcions per a gestionar una cua */
#include "cua.h"
/* incloure estructures de dades i capceleres de funcions per convertir imatges a diferents formats */
/* #include "magickconverter.h" */
/* incloure estructures de dades i capceleres de funcions per tractar frames de vídeo */
#include "qtconverter.h"
/* incloure estructures de dades i capceleres de funcions per traballar amb threads */
#include "pthread.h"
/* &&&&&&&&&&&&&&&&&&&&&&&&&&&&& VIDEOGRID &&&&&&&&&&&&&&&&&&&&&&&&&&&&& */
/* definició de l'amplada i l'alçada d'una casella */
#define W_CELL 60
#define H_CELL 40
/* crear un apuntador al nou objecte */
static t_class *videogrid_class;
/* indica el nombre de videogrid creats - utilitzat per diferenciar el nom d'instàncies d'objectes del mateix tipus */
static int videogridcount = 0;
/* definició de la classe i la seva estructura de dades */
typedef struct _videogrid {
t_object x_obj;
/* declaració de la sortida de l'objecte */
t_outlet *x_sortida;
/* llista d'objectes gràfics */
t_glist *x_glist;
/* nombre de files */
int x_num_fil;
/* nombre de columnes */
int x_num_col;
/* posició de la última imatge en el tauler */
int x_ultima_img;
/* path del directori actual */
path x_dir_actual;
/* path del directori a canviar */
path x_dir_canvi;
/* posicio ultim al directori actual */
int x_dir_pos;
/* apuntador al primer element posicionat al tauler */
Node *x_tauler_primer;
/* cua d'imatges */
Cua x_cua;
/* nom de l'objecte */
t_symbol *x_name;
/* color de fons */
t_symbol *x_color_fons;
/* color del marge */
t_symbol *x_color_marc;
/* mutex per evitar concurrencia sobre la cua al accedir diferents threads*/
pthread_mutex_t x_lock;
} t_videogrid;
/* calcula la posició x del tauler a partir de la posició de l'element de la cua (d'esquerra a dreta) */
int getX(t_videogrid* x, int posCua){
int c = x->x_num_col;
int xpos = (posCua % c) * W_CELL;
return(xpos + 1);
}
/* calcula la posició y del tauler a partir de la posició de l'element de la cua (de dalt a baix) */
int getY(t_videogrid* x, int posCua){
int c = x->x_num_col;
int ypos = (posCua / c) * H_CELL;
return(ypos + 1);
}
/* elimina les imatges temporals */
void eliminar_imatges_temporals(int maxim){
FILE *fitxer;
path path_total;
int contador = 0;
char contador_str[BYTES_NUM_TEMP];
while(contador < maxim){
strcpy(path_total,PATH_TEMPORAL);
sprintf(contador_str,"%d", contador);
strcat(path_total,contador_str);
strcat(path_total,".");
strcat(path_total,FORMAT_MINIATURA);
/* elimina el fitxer si no hi ha cap problema */
if(unlink(path_total)){
/* post("Imatge temporal %s eliminada\n",path_total); */
}
contador++;
}
post("Videogrid: Imatges temporals eliminades\n",path_total);
}
int format_adequat_v(path nomF){
int retorn = 0;
path ig_path = "";
strcat(ig_path,nomF);
char *t1;
path extensio = "";
for ( t1 = strtok(ig_path,".");
t1 != NULL;
t1 = strtok(NULL,".") )
strcpy(extensio,t1);
if(strcmp(extensio,"mov")==0) retorn = 1;
/*
if(strcmp(extensio,"eps")==0) retorn = 1;
if(strcmp(extensio,"gif")==0) retorn = 1;
if(strcmp(extensio,"jpg")==0) retorn = 1;
if(strcmp(extensio,"jpeg")==0) retorn = 1;
if(strcmp(extensio,"png")==0) retorn = 1;
if(strcmp(extensio,"ppm")==0) retorn = 1;
if(strcmp(extensio,"tif")==0) retorn = 1;
if(strcmp(extensio,"tiff")==0) retorn = 1;
*/
return (retorn);
}
/* afegir una imatge al grid */
void videogrid_afegir_imatge(t_videogrid *x, path entrada)
{
int maxim;
char nNstr[BYTES_NUM_TEMP];
int pos = 0;
/* escriu l'argument entrat */
if (format_adequat_v(entrada) == 1){
/* post("Afegint la imatge %s ...",entrada); */
maxim = x->x_num_fil * x->x_num_col;
path ig_path = PATH_TEMPORAL;
/* si hi ha tants nodes a la cua com el maxim */
if((numNodes(&x->x_cua)) >= maxim){
/* desencua */
int extret;
extret = desencuar(&x->x_cua);
/* obtenir la posició en la cua del nou node */
if(x->x_ultima_img == maxim-1) {
pos = 0;
}else{
pos = x->x_ultima_img+1;
}
sys_vgui(".x%x.c delete %xS%d\n", glist_getcanvas(x->x_glist), x, pos);
}
/* encua el nou node */
encuar(&x->x_cua, entrada);
/* si no és el primer element a encuar incrementem la posicio de la última imatge insertada */
if(numNodes(&x->x_cua) != 1) x->x_ultima_img ++;
/* si assoleix el maxim torna a començar, inicialitzant la posició en el tauler de la última imatge insertada */
if(x->x_ultima_img == maxim) x->x_ultima_img = 0;
/*
Quicktime per les conversions
*/
int nN = x->x_ultima_img;
convertir_img(entrada,FORMAT_MINIATURA, W_CELL, H_CELL, nN);
sprintf(nNstr, "%d", nN);
strcat(ig_path,nNstr);
strcat(ig_path,".");
strcat(ig_path,FORMAT_MINIATURA);
/*printf("Creacio de la imatge %s ...",ig_path);*/
sys_vgui("image create photo img%x%d -file %s\n",x,nN,ig_path);
/* printf("1. Creacio de la imatge %s ...",ig_path); */
sys_vgui(".x%x.c create image %d %d -image img%x%d -tags %xS%d\n",
glist_getcanvas(x->x_glist),
text_xpix(&x->x_obj, x->x_glist) + getX(x,nN) + (W_CELL/2),
text_ypix(&x->x_obj, x->x_glist) + getY(x,nN) + (H_CELL/2),
x,nN,x,nN);
/* printf("2. Creacio de la imatge %s ...",ig_path); */
if(nN == 0){
x->x_tauler_primer = x->x_cua.final;
/* post("Ara el primer del tauler es %s\n",x->x_tauler_primer->pathFitxer); */
}
/* printf("SURT de la creacio de la imatge %s ...",ig_path); */
}else{
post("Videogrid: El format del fitxer %s és incompatible.",entrada);
}
/*
sys_vgui("image create photo img%x -file %s\n",x,entrada);
sys_vgui(".x%x.c create image %d %d -image img%x -tags %xS\n",
glist_getcanvas(glist),text_xpix(&x->x_obj, glist), text_ypix(&x->x_obj, glist),x,x);
*/
}
/* dibuixa videogrid */
void videogrid_drawme(t_videogrid *x, t_glist *glist, int firsttime)
{
/* post("Entra a drawme amb firsttime: %d", firsttime); */
if (firsttime) {
char name[MAXPDSTRING];
canvas_makefilename(glist_getcanvas(x->x_glist), x->x_name->s_name, name, MAXPDSTRING);
sys_vgui(".x%x.c create rectangle %d %d %d %d -fill %s -tags %xGRID -outline %s\n",
glist_getcanvas(glist),
text_xpix(&x->x_obj, glist), text_ypix(&x->x_obj, glist),
text_xpix(&x->x_obj, glist) + (x->x_num_col * W_CELL) + 1, text_ypix(&x->x_obj, glist) + (x->x_num_fil * H_CELL) + 1,
x->x_color_fons->s_name, x,x->x_color_marc->s_name);
canvas_fixlinesfor(glist_getcanvas(glist), (t_text*)x);
/* si hi elements a la cua els afegeix (redimensió) */
if(!cuaBuida(&x->x_cua))
{
path ig_path;
int nN = 0;
char nNstr[BYTES_NUM_TEMP];
Node *actual;
actual=x->x_cua.davanter;
do{
strcpy(ig_path,PATH_TEMPORAL);
sprintf(nNstr, "%d", nN);
strcat(ig_path,nNstr);
strcat(ig_path,".");
strcat(ig_path,FORMAT_MINIATURA);
/* post("reestablint la imatge %s", actual->pathFitxer); */
// videogrid_afegir_imatge(x,actual->pathFitxer);
convertir_img(actual->pathFitxer,FORMAT_MINIATURA, W_CELL, H_CELL, nN);
sys_vgui("image create photo img%x%d -file %s\n",x,nN,ig_path);
sys_vgui(".x%x.c create image %d %d -image img%x%d -tags %xS%d\n",
glist_getcanvas(x->x_glist),text_xpix(&x->x_obj, x->x_glist) + getX(x,nN) + (W_CELL/2), text_ypix(&x->x_obj, x->x_glist) + getY(x,nN) + (H_CELL/2),x,nN,x,nN);
actual = actual->seguent;
nN++;
}while(actual);
}
}
else {
sys_vgui(".x%x.c coords %xGRID %d %d %d %d\n", glist_getcanvas(glist), x, text_xpix(&x->x_obj, glist), text_ypix(&x->x_obj, glist),text_xpix(&x->x_obj, glist) + (x->x_num_col*W_CELL) + 1, text_ypix(&x->x_obj, glist) + (x->x_num_fil*H_CELL) + 1);
if(!cuaBuida(&x->x_cua))
{
int contador = 0;
while(contador < numNodes(&x->x_cua)){
sys_vgui(".x%x.c coords %xS%d \
%d %d\n",
glist_getcanvas(glist), x, contador,
text_xpix(&x->x_obj, x->x_glist) + getX(x,contador) + (W_CELL/2), text_ypix(&x->x_obj, x->x_glist) + getY(x,contador) + (H_CELL/2));
contador++;
}
/* char buf[800];
sprintf(buf, "pdtk_videogrid_table %%s %s %d %d\n", x->x_name->s_name, x->x_num_fil, x->x_num_col);
gfxstub_new(&x->x_obj.ob_pd, x, buf); */
}
}
}
/* borra videogrid */
void videogrid_erase(t_videogrid* x,t_glist* glist)
{
int maxim = x->x_num_fil * x->x_num_col;
path path_total;
char contador_str[2];
/* post("Entra a erase"); */
/* elimina les imatges */
int contador = 0;
while(contador < numNodes(&x->x_cua)){
sys_vgui(".x%x.c delete %xS%d\n", glist_getcanvas(x->x_glist), x, contador);
strcpy(path_total,PATH_TEMPORAL);
sprintf(contador_str,"%d", contador);
strcat(path_total,contador_str);
strcat(path_total,".");
strcat(path_total,FORMAT_MINIATURA);
if(unlink(path_total)){
/* post("Imatge temporal %s eliminada\n",path_total); */
}
contador++;
}
/* elimina el grid */
sys_vgui(".x%x.c delete %xGRID\n", glist_getcanvas(glist), x);
eliminar_imatges_temporals(maxim);
}
/* mètode de la clase que escriu un missatge al rebre un bang */
void videogrid_bang(t_videogrid *x)
{
/* post("Hello videogrid !!"); */
escriuCua(&x->x_cua);
}
/* mètode de la classe que es dispara al rebre una entrada de missatge amb [putvideo +string( com a paràmetre */
void videogrid_putvideo(t_videogrid *x, t_symbol *entrada)
{
/* comprova que existeixi el fitxer */
FILE *fitxer;
path e;
strcpy(e,entrada->s_name);
/* post("putvideo de %s\n", e); */
fitxer = fopen(e,"r");
if (!fitxer) {
post("Videogrid: Problema amb l'obertura del fitxer %s\n",e);
}
else {
/* post("s'encua la imatge %s\n", e); */
videogrid_afegir_imatge(x,e);
/*outlet_symbol(x->x_sortida, entrada);*/
}
/* post("putvideo amb img = %s\n", e); */
}
/* mètode de la classe que es dispara al rebre una entrada de missatge amb [putvideodir +string( com a paràmetre */
void *videogrid_putvideodir_thread(t_videogrid *x)
{
DIR *dirp;
struct dirent * direntp;
path nomImatge, directoriAnterior, pathActual;
int numEncuats = 0, numPosDir = 0;
int maxim;
if ((dirp = opendir(x->x_dir_canvi)) == NULL)
{
post("Videogrid: No es pot obrir el directori %s\n", x->x_dir_canvi);
}else{
maxim = x->x_num_fil * x->x_num_col;
strcpy(directoriAnterior, x->x_dir_actual);
strcpy(x->x_dir_actual, x->x_dir_canvi);
/* si es el mateix directori entrat l'ultim busca la ultima imatge afegida per a seguir a encuant a partir d'ella en endavant */
if(strcmp(directoriAnterior, x->x_dir_actual) == 0){
post("Videogrid: Repeteix directori %s\n", x->x_dir_actual);
while ( (direntp = readdir( dirp )) != NULL ){
/* es descarta el mateix directori, el directori anterior i tot el que no sigui un fitxer regular */
if((strcmp(direntp->d_name,"..") != 0)&&(strcmp(direntp->d_name,".") != 0)&&(direntp->d_type == DT_REG)){
/* incrementa la posició en el directori */
numPosDir++;
/* assolir la posició anterior en el directori */
if(numPosDir > x->x_dir_pos){
/* si el nombre de nodes encuats per aquest directori no supera el màxim encua el nou node */
if(numEncuats < maxim){
/* post("s'encua la imatge %s\n", direntp->d_name); */
/* concatena el path i el nom de la imatge */
strcpy(nomImatge,direntp->d_name);
strcpy(pathActual,x->x_dir_actual);
strcat(pathActual,"/");
strcat(pathActual,nomImatge);
pthread_mutex_lock(&x->x_lock);
videogrid_afegir_imatge(x, pathActual);
pthread_mutex_unlock(&x->x_lock);
/* incrementa en 1 per indicar el nombre de nodes encuats per aquest directori */
numEncuats++;
/* es desa la posició en el directori de l'últim node encuat */
x->x_dir_pos = numPosDir;
}
}
}
}
}else{
/* directori diferent omple la cua començant pel primer fitxer */
post("Videogrid: Nou directori %s \n", x->x_dir_actual);
while ( (direntp = readdir( dirp )) != NULL ){
/* es descarta el mateix directori, el directori anterior i tot el que no sigui un fitxer regular */
if((strcmp(direntp->d_name,"..") != 0)&&(strcmp(direntp->d_name,".") != 0)&&(direntp->d_type == DT_REG)){
/* incrementa la posició en el directori */
numPosDir++;
/* si el nombre de nodes encuats per aquest directori no supera el màxim enca el nou node */
if(numEncuats < maxim){
/* post("s'encua la imatge %s\n", direntp->d_name); */
/* concatena el path i el nom de la imatge */
strcpy(nomImatge,direntp->d_name);
strcpy(pathActual,x->x_dir_actual);
strcat(pathActual,"/");
strcat(pathActual,nomImatge);
pthread_mutex_lock(&x->x_lock);
videogrid_afegir_imatge(x, pathActual);
pthread_mutex_unlock(&x->x_lock);
/* incrementa en 1 per indicar el nombre de nodes encuats per aquest directori */
numEncuats++;
/* es desa la posició en el directori de l'últim node encuat */
x->x_dir_pos = numPosDir;
}
}
}
}
/* si la posicio de l'actual es la de l'utim fitxer del directori, inicialitza la posició */
if(x->x_dir_pos >= numPosDir) x->x_dir_pos = 0;
closedir(dirp);
}
/* escriu l'argument entrat */
/* post("Obtenint imatges del directori: %s ...",x->x_dir_canvi); */
/* envia a la sorida l'argument entrat */
/* outlet_symbol(x->x_sortida, entrada); */
pthread_exit(NULL);
}
void videogrid_putvideodir(t_videogrid *x, t_symbol *entrada)
{
pthread_t unthread;
pthread_attr_t unatribut;
pthread_attr_init( &unatribut );
strcpy(x->x_dir_canvi,entrada->s_name);
// ---------------- THREAD CREAT -------------------------
pthread_mutex_init(&x->x_lock, NULL);
pthread_create(&unthread,&unatribut,(void *)videogrid_putvideodir_thread, x);
pthread_mutex_destroy(&x->x_lock);
}
static int videogrid_click(t_gobj *z, struct _glist *glist, int xpix, int ypix, int shift, int alt, int dbl, int doit)
{
t_videogrid* x = (t_videogrid *)z;
int x_pos = xpix - text_xpix(&x->x_obj, x->x_glist);
int y_pos = ypix - text_ypix(&x->x_obj, x->x_glist);
int xa, ya, postauler, contador, maxim;
path pathSortida;
Node *actual;
if (doit)
{
/* obtenir la posicio en el tauler */
xa = x_pos / W_CELL;
ya = (y_pos / H_CELL) * x->x_num_col;
postauler = ya + xa;
/* obtenir el path per enviar a la sortida */
if((!cuaBuida(&x->x_cua))&&(postauler < numNodes(&x->x_cua))){
contador = 0;
maxim = x->x_num_fil * x->x_num_col;
if(x->x_tauler_primer){
actual = x->x_tauler_primer;
while(contador <= postauler){
if(contador == postauler){
strcpy(pathSortida,actual->pathFitxer);
}
if(actual->seguent == NULL){
actual = x->x_cua.davanter;
}else{
actual = actual->seguent;
}
contador++;
}
outlet_symbol(x->x_sortida, gensym(pathSortida));
}
}
}
return (1);
}
static void videogrid_getrect(t_gobj *z, t_glist *glist,int *xp1, int *yp1, int *xp2, int *yp2)
{
int cols, fils;
t_videogrid* x = (t_videogrid*)z;
cols = x->x_num_col;
fils = x->x_num_fil;
*xp1 = text_xpix(&x->x_obj, glist);
*yp1 = text_ypix(&x->x_obj, glist);
*xp2 = text_xpix(&x->x_obj, glist) + (cols*W_CELL);
*yp2 = text_ypix(&x->x_obj, glist) + (fils*H_CELL);
/* post("Esta amb el ratoli en el punt %d %d %d %d o son els vetexs de la caixa... es/bd", xp1, yp1, xp2, yp2); */
}
static void videogrid_displace(t_gobj *z, t_glist *glist,int dx, int dy)
{
/* post("Entra a displace amb dx %d i dy %d", dx, dy); */
t_videogrid *x = (t_videogrid *)z;
x->x_obj.te_xpix += dx;
x->x_obj.te_ypix += dy;
sys_vgui(".x%x.c coords %xGRID %d %d %d %d\n",
glist_getcanvas(glist), x,
text_xpix(&x->x_obj, glist), text_ypix(&x->x_obj, glist),
text_xpix(&x->x_obj, glist) + (x->x_num_col*W_CELL), text_ypix(&x->x_obj, glist) + (x->x_num_fil*H_CELL));
videogrid_drawme(x, glist, 0);
canvas_fixlinesfor(glist_getcanvas(glist),(t_text*) x);
}
static void videogrid_select(t_gobj *z, t_glist *glist, int state)
{
/* post("Entra select amb state %d", state); */
t_videogrid *x = (t_videogrid *)z;
if (state) {
/* post("Videogrid seleccionat"); */
sys_vgui(".x%x.c itemconfigure %xGRID -outline #0000FF\n", glist_getcanvas(glist), x);
}
else {
/* post("Videogrid deseleccionat"); */
sys_vgui(".x%x.c itemconfigure %xGRID -outline %s\n", glist_getcanvas(glist), x, x->x_color_marc->s_name);
}
}
static void videogrid_delete(t_gobj *z, t_glist *glist)
{
/* post("Entra a delete"); */
t_text *x = (t_text *)z;
canvas_deletelinesfor(glist_getcanvas(glist), x);
}
static void videogrid_vis(t_gobj *z, t_glist *glist, int vis)
{
/* post("Entra a vist amb vis %d", vis); */
t_videogrid* s = (t_videogrid*)z;
if (vis)
videogrid_drawme(s, glist, 1);
else
videogrid_erase(s,glist);
}
static void videogrid_save(t_gobj *z, t_binbuf *b)
{
/* post("Entra a save"); */
t_videogrid *x = (t_videogrid *)z;
/* crea la cadena de paths per desar */
/* 100 possibles paths com a màxim a 512 cada path*/
/* char cadenaPaths[51200];*/
char *cadenaPaths, *cadenaPathsInicials;
path ultimPath = "";
cadenaPaths = (char *)malloc(51200*sizeof(char));
strcpy(cadenaPaths,"");
cadenaPathsInicials = (char *)malloc(51200*sizeof(char));
strcpy(cadenaPathsInicials,"");
/*strcpy(cadenaPaths,(char *)argv[5].a_w.w_symbol->s_name);*/
if(!cuaBuida(&x->x_cua))
{
Node *actual;
int maxim = x->x_num_fil * x->x_num_col;
int contador = x->x_ultima_img + 1;
if (contador > maxim) {
contador = 0;
}
/* printf("\n contador %d i maxim %d i laultimaPOS %d \n", contador, maxim, x->x_ultima_img); */
/*
strcat(cadenaPaths, actual->pathFitxer);
strcat(cadenaPaths, "1|\n");
contador ++;
*/
/* prenem el davanter de la cua */
actual=x->x_cua.davanter;
while(contador < numNodes(&x->x_cua)){
/* afegim els paths del davanter fins a l'ultim node al tauler */
strcat(cadenaPaths, actual->pathFitxer);
strcat(cadenaPaths, "|");
actual = actual->seguent;
contador ++;
}
if(actual != x->x_cua.final){
/* ara resten els de de l'inici del tauler fins al final de la cua */
while(actual != x->x_cua.final){
strcat(cadenaPathsInicials, actual->pathFitxer);
strcat(cadenaPathsInicials, "|");
actual = actual->seguent;
}
/* afegeix l'últim */
strcat(ultimPath, actual->pathFitxer);
strcat(ultimPath, "|");
/* afegeix l'ultim de la cua */
strcat(cadenaPathsInicials, ultimPath);
}else{
if(x->x_ultima_img == 0){
strcat(ultimPath, actual->pathFitxer);
strcat(ultimPath, "|");
strcat(cadenaPathsInicials, ultimPath);
}
}
/* ordena el paths segons aparicio en el tauler */
strcat(cadenaPathsInicials, cadenaPaths);
/* DE MOMENT NO DESA ELS PATHS */
strcat(cadenaPathsInicials, "");
/* printf("%s",cadenaPathsInicials); */
}
binbuf_addv(b, "ssiissiisss", gensym("#X"),gensym("obj"),
x->x_obj.te_xpix, x->x_obj.te_ypix,
gensym("videogrid"),x->x_name,x->x_num_fil,x->x_num_col,x->x_color_fons,x->x_color_marc,gensym(cadenaPathsInicials));
binbuf_addv(b, ";");
}
static void videogrid_properties(t_gobj *z, t_glist *owner)
{
char buf[800];
t_videogrid *x=(t_videogrid *)z;
/* post("Es crida a pdtk_videogrid dialog passant nom = %s\n fils = %d \t cols = %d \t color fons = %s \t color marc = %s\n", x->x_name->s_name, x->x_num_fil, x->x_num_col, x->x_color_fons->s_name, x->x_color_marc->s_name); */
sprintf(buf, "pdtk_videogrid_dialog %%s %s %d %d %s %s\n",
x->x_name->s_name, x->x_num_fil, x->x_num_col, x->x_color_fons->s_name, x->x_color_marc->s_name);
/* post("videogrid_properties : %s", buf ); */
gfxstub_new(&x->x_obj.ob_pd, x, buf);
}
static void videogrid_dialog(t_videogrid *x, t_symbol *s, int argc, t_atom *argv)
{
int maxim, nfil, ncol, maxdigit;
if ( !x ) {
post("Videogrid: error_ intent de modificar le propietats d'un objecte inexistent\n");
}
if ( argc != 5 )
{
post("Videogrid: error_ sobre el nombre d'arguments ( 5 enlloc de %d )\n",argc);
return;
}
if (argv[0].a_type != A_SYMBOL || argv[1].a_type != A_FLOAT || argv[2].a_type != A_FLOAT || argv[3].a_type != A_SYMBOL || argv[4].a_type != A_SYMBOL)
{
post("Videogrid: error_ algun dels arguments no es del tipus adequat\n");
return;
}
x->x_name = argv[0].a_w.w_symbol;
nfil = (int)argv[1].a_w.w_float;
ncol = (int)argv[2].a_w.w_float;
x->x_color_fons = argv[3].a_w.w_symbol;
x->x_color_marc = argv[4].a_w.w_symbol;
/* el màxim es fixa pel nombre de digits utilitzats pel nom de la imatge temporal */
maxdigit = pow(10,BYTES_NUM_TEMP);
if((nfil*ncol) <= maxdigit){
if((nfil*ncol) > 0){
x->x_num_fil = nfil;
x->x_num_col = ncol;
}else{
post("Videogrid: El nombre de files i columnes són inferiors al mímin permès: 1 casella\n",maxdigit);
}
}else{
post("Videogrid: El nombre de files i columnes excedeixen del màxim permès: un total de %d caselles\n",maxdigit);
}
post("Videogrid: Valors modificats_ nom = %s\n fils = %d \t cols = %d\n", x->x_name->s_name, x->x_num_fil, x->x_num_col);
/* elimina els nodes no representables amb la nova configuració */
maxim = x->x_num_fil * x->x_num_col;
int extret;
videogrid_erase(x, x->x_glist);
/* si hi ha més nodes a la cua que el maxim */
while((numNodes(&x->x_cua)) > maxim){
/* desencuem */
extret = desencuar(&x->x_cua);
}
/* al reestablir el tamany del tauler cal saber la posició de l'últim element */
x->x_ultima_img = numNodes(&x->x_cua) - 1;
if (x->x_ultima_img < 0) x->x_ultima_img = 0;
x->x_tauler_primer = x->x_cua.davanter;
videogrid_drawme(x, x->x_glist, 1);
}
t_widgetbehavior videogrid_widgetbehavior;
static void videogrid_setwidget(void)
{
/* post("Entra a setwidget"); */
videogrid_widgetbehavior.w_getrectfn = videogrid_getrect;
videogrid_widgetbehavior.w_displacefn = videogrid_displace;
videogrid_widgetbehavior.w_selectfn = videogrid_select;
videogrid_widgetbehavior.w_activatefn = NULL;
videogrid_widgetbehavior.w_deletefn = videogrid_delete;
videogrid_widgetbehavior.w_visfn = videogrid_vis;
/* clic del ratoli */
videogrid_widgetbehavior.w_clickfn = videogrid_click;
#if PD_MINOR_VERSION < 37
videogrid_widgetbehavior.w_savefn = videogrid_save;
videogrid_widgetbehavior.w_propertiesfn = videogrid_properties;
#endif
}
/* el constructor de la classe*/
static void *videogrid_new(t_symbol* name, int argc, t_atom *argv)
{
/* instanciació del nou objecte */
t_videogrid *x = (t_videogrid *)pd_new(videogrid_class);
/* crea una sortida per l'objecte*/
x->x_sortida = outlet_new(&x->x_obj,&s_symbol);
/* s'obté el canvas de pd */
x->x_glist = (t_glist*) canvas_getcurrent();
/* posició en el tauler de la última imatge afegida */
x->x_ultima_img = 0;
/* posició de l'últim fitxer del directori encuat */
x->x_dir_pos = 0;
/* apuntador al primer element en el tauler */
x->x_tauler_primer = NULL;
/* fixa el nom de l'objecte */
char nom[15];
sprintf(nom, "videogrid%d", ++videogridcount);
name = gensym(nom);
x->x_name = name;
/* amb aquest nom es prepara per poder rebre dades */
pd_bind(&x->x_obj.ob_pd, x->x_name);
/* crea la cua de nodes */
crearCua(&x->x_cua);
post("NOU videogrid: s'han entrat %d arguments\n", argc);
if (argc != 0)
{
post("NOU videogrid: s'obre un objecte existent o amb arguments definits\n");
/* x->x_name */
x->x_num_fil = (int)atom_getintarg(1, argc, argv);
x->x_num_col = (int)atom_getintarg(2, argc, argv);
x->x_color_fons = argv[3].a_w.w_symbol;
x->x_color_marc = argv[4].a_w.w_symbol;
post("!!!NOU videogrid: s'han entrat %d arguments\n", argc);
if(argc == 6){
/* llegir la cadena de paths | afegir els paths a la cua */
char *cadenaPaths;
cadenaPaths = (char *)malloc(51200*sizeof(char));
strcpy(cadenaPaths,(char *)argv[5].a_w.w_symbol->s_name);
/* printf("Es carreguen els paths %s --- %s **** %s\n", cadenaPaths, argv[5].a_w.w_symbol->s_name,argv[3].a_w.w_symbol->s_name); */
/* split */
char *token;
t_symbol *tt;
for ( token = strtok(argv[5].a_w.w_symbol->s_name,"|");
token != NULL;
token = strtok(NULL,"|") ){
tt = gensym(token);
/* printf("AFEGINT CARREGANT %s\n",tt->s_name); */
/* videogrid_putvideo(x,tt); */
/* DE MOMENT NO AFEGEIX ELS PATHS */
/* videogrid_afegir_imatge(x,tt->s_name); */
}
/*
token = strtok(cadenaPaths,"|");
while(token){
tt = gensym(token);
printf("AFEGINT CARREGANT %s\n",tt->s_name);
videogrid_putvideo(x,tt);
token = strtok(NULL,"|");
}
*/
free(cadenaPaths);
}
}else{
/* crea un objecte nou */
post("NOU videogrid: es crea un objecte nou\n");
/* fixa el nombre de files */
x->x_num_fil = 3;
/* fixa el nombre de columnes */
x->x_num_col = 5;
/* colors de fons i de marc*/
x->x_color_fons = gensym("#F0F0F0");
x->x_color_marc = gensym("#0F0F0F");
}
/* printf("S'ha instanciat un videogrid anomenat %s amb les caracteristiques seguents:",x->x_name->s_name);
printf("Nombre de files %d - Nombre de columnes: %d", x->x_num_fil, x->x_num_col); */
return (x);
}
static void videogrid_destroy(t_videogrid *x){
/* elimina el contingut de la cua */
eliminarCua(&x->x_cua);
post("Videogrid eliminat");
}
/* generacio d'una nova classe */
/* al carregar la nova llibreria my_lib pd intenta cridar la funció my_lib_setup */
/* aquesta crea la nova classe i les seves propietats només un sol cop */
void videogrid_setup(void)
{
/* post("Entra a setup per generar la classe videogrid"); */
#include "videogrid.tk2c"
/*
sense pas d'arguments
videogrid_class = class_new(gensym("videogrid"),
(t_newmethod)videogrid_new,
(t_method)videogrid_destroy,
sizeof(t_videogrid),
CLASS_DEFAULT,
A_DEFSYM,
0);
amb pas d'arguments:
*/
videogrid_class = class_new(gensym("videogrid"),
(t_newmethod)videogrid_new,
(t_method)videogrid_destroy,
sizeof(t_videogrid),
CLASS_DEFAULT,
A_GIMME,
0);
/* class_new crea la nova classe retornant un punter al seu prototipus,
el primer argument es el nom simbolic de la classe,
el segon i tercer corresponen al constructor i destructor de la classe respectivament,
(el constructor instancia un objecte i inicialitza les seves dades cada cop que es crea un objecte
el destructor allibera la memoria reservada al destruid l'objecte per qualsevol causa)
el quart correspon a la mida de l'estructura de dades, per tal de poder reservar la memoria necessària,
el cinquè influeix en el mòde de representació gràfica del objectes. Per defecte CLASS_DEFAULT o ``0',
la resta d'arguments defineixen els arguments de l'objecte i el seu tipus, la llista acaba amb 0
*/
/* afegeix el mètode helloworld_bang a la classe helloworld_class */
class_addbang(videogrid_class, videogrid_bang);
/* afegeix el mètode videogrid_putvideo a la classe videogrid per a entrades de missatge
que inicien amb putvideo i una cadena string com a argument */
class_addmethod(videogrid_class,(t_method)videogrid_putvideo,gensym("putvideo"), A_DEFSYMBOL, 0);
/* afegeix el mètode videogrid_putvideodir a la classe videogrid per a entrades de missatge
que inicien amb putvideodir i una cadena string com a argument */
class_addmethod(videogrid_class,(t_method)videogrid_putvideodir,gensym("putvideodir"), A_DEFSYMBOL, 0);
/* afegeix un metode per a modificar el valor de les propietats de l'objecte */
class_addmethod(videogrid_class, (t_method)videogrid_dialog, gensym("dialog"), A_GIMME, 0);
/* afegeix un metode per l'obtencio de la posicio del clic del ratolí */
class_addmethod(videogrid_class, (t_method)videogrid_click, gensym("click"), A_FLOAT, A_FLOAT, A_FLOAT, A_FLOAT, A_FLOAT, 0);
/* inicia el comportament de videogrid */
videogrid_setwidget();
#if PD_MINOR_VERSION >= 37
class_setsavefn(videogrid_class,&videogrid_save);
class_setpropertiesfn(videogrid_class, videogrid_properties);
#endif
/* afegeix el mètode videogrid_widgetbehavior al la classe videogrid per a la creació de l'element visual */
class_setwidget(videogrid_class,&videogrid_widgetbehavior);
class_sethelpsymbol(videogrid_class, gensym("videogrid.pd"));
}