X-Git-Url: http://research.m1stereo.tv/gitweb?a=blobdiff_plain;f=src%2Flibrary.c;h=c7b60134f49369526346111bd9467e87521ef7c9;hb=cfbef984de0dc53c8b882d7706556656b9934de2;hp=614fca434766d3131bd160aa989bd38fcf145116;hpb=3477009a022a1f3defec8abd7e2b8c85619628bb;p=melted_gui diff --git a/src/library.c b/src/library.c index 614fca4..c7b6013 100644 --- a/src/library.c +++ b/src/library.c @@ -1,6 +1,6 @@ /* - * playlist.c -- GTK+ 2 omnplay - * Copyright (C) 2011 Maksym Veremeyenko + * library.c -- GTK+ 2 melted gui + * Copyright (C) 2012 Maksym Veremeyenko * * 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 @@ -33,10 +33,344 @@ #include #include -#include "omnplay.h" +#include +#include + +#include "library.h" #include "ui.h" #include "timecode.h" +#include "support.h" + +extern GtkTargetEntry drag_targets[]; + +void library_release(instance_t* app) +{ + mvcp_close(app->library.handle[0]); + mvcp_parser_close(app->library.handle[1]); +}; + +static void library_add_fake(instance_t* app, GtkTreeStore *tree_store, GtkTreeIter* parent) +{ + GtkTreeIter iter; + gtk_tree_store_append(tree_store, &iter, parent); + gtk_tree_store_set(tree_store, &iter, -1); +}; + +static int library_init_load(instance_t* app) +{ + GtkTreeIter iter; + GtkTreeStore *tree_store; + + tree_store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(app->library_tree))); + gtk_tree_store_clear(tree_store); + + gtk_tree_store_append(tree_store, &iter, NULL); + gtk_tree_store_set(tree_store, &iter, + 0, app->library.icons[0], + 1, ">", + 2, "LIBRARY", + 3, NULL, + 4, NULL, + 5, FALSE, + 6, "red", + -1 ); + library_add_fake(app, tree_store, &iter); + + gtk_tree_view_collapse_all(GTK_TREE_VIEW(app->library_tree)); + + return 0; +}; + +static void library_add_item(instance_t* app, GtkTreeStore *treestore, GtkTreeIter *iter, + mvcp_dir_entry e, mvcp_list_entry p) +{ + GtkTreeIter this, child; + + if(e->dir) + { + gtk_tree_store_prepend(treestore, &this, iter); + + gtk_tree_store_set(treestore, &this, + 0, app->library.icons[0], + 1, "", + 2, e->name, + 3, e, + 4, NULL, + 5, FALSE, + 6, "red", + -1 ); + + gtk_tree_store_append(treestore, &child, &this); + gtk_tree_store_set(treestore, &child, -1); + } + else + { + char dur[32]; + + if(p) + frames2tc(p->size, p->fps, dur); + else + strcpy(dur, ""); + + gtk_tree_store_append(treestore, &this, iter); + + gtk_tree_store_set(treestore, &this, + 0, app->library.icons[1], + 1, dur, + 2, e->name, + 3, e, + 4, p, + 5, FALSE, + 6, "red", + -1 ); + }; +}; + +static void on_library_row_expanded +( + GtkTreeView *treeview, + GtkTreeIter *iter, + GtkTreePath *path, + gpointer user_data +) +{ + int i; + char* p; + GtkTreeIter fake; + GtkTreeModel *model; + GdkCursor* cursor; + mvcp_dir dir; + mvcp_dir_entry_t *e; + instance_t* app = (instance_t*)user_data; + +// g_warning("on_library_row_expanded: HERE"); + + /* Set busy cursor */ + cursor = gdk_cursor_new(GDK_WATCH); + gdk_window_set_cursor(gtk_widget_get_toplevel(GTK_WIDGET(treeview))->window, cursor); + gdk_cursor_unref(cursor); + gdk_flush(); + + model = gtk_tree_view_get_model(treeview); + + /* save fake item */ + gtk_tree_model_iter_children(GTK_TREE_MODEL(model), &fake, iter); + + /* request mvcp entry */ + gtk_tree_model_get(GTK_TREE_MODEL(model), iter, + 3, &e, + -1); + + /* setup root path */ + if(!e) + p = "/"; + else + p = e->full; + + /* read dir */ + dir = mvcp_dir_init(app->library.handle[0], p); + for (i = 0; i < mvcp_dir_count(dir); i++) + { + mvcp_dir_entry_t dir_entry; + mvcp_list_entry_t *list_e = NULL, list_entry; + + if(mvcp_ok != mvcp_dir_get(dir, i, &dir_entry)) + continue; + +// g_warning("on_library_row_expanded: path=[%s], entry.dur=[%d], entry.full=[%s], entry.name[%s]", +// p, entry.dir, entry.full, entry.name); + + e = (mvcp_dir_entry_t*)malloc(sizeof(mvcp_dir_entry_t)); + *e = dir_entry; + + if(!e->dir && mvcp_ok == mvcp_probe_clip( app->library.handle[0], e->full, &list_entry)) + { + list_e = (mvcp_list_entry_t*)malloc(sizeof(mvcp_list_entry_t)); + *list_e = list_entry; + }; + + library_add_item(app, GTK_TREE_STORE(model), iter, e, list_e); + }; + + /* restore cursor */ + gdk_window_set_cursor(gtk_widget_get_toplevel(GTK_WIDGET(treeview))->window, NULL); + + /* delete fake item */ + gtk_tree_store_remove(GTK_TREE_STORE(model), &fake); +}; + +static void on_library_row_collapsed +( + GtkTreeView *treeview, + GtkTreeIter *iter, + GtkTreePath *path, + gpointer user_data +) +{ + GtkTreeModel *model; + GtkTreeIter child; +// g_warning("on_library_row_collapsed: HERE"); + + /* delete all items */ + model = gtk_tree_view_get_model(treeview); + while (gtk_tree_model_iter_children(GTK_TREE_MODEL(model), &child, iter)) + { + mvcp_dir_entry_t *e; + mvcp_list_entry_t *l; + + /* request mvcp entry */ + gtk_tree_model_get(GTK_TREE_MODEL(model), &child, + 3, &e, + 4, &l, + -1); + + /* free entry */ + if(e) free(e); + if(l) free(l); + + gtk_tree_store_remove(GTK_TREE_STORE(model), &child); + }; + + /* add a fake element */ + library_add_fake(user_data, GTK_TREE_STORE(model), iter); +}; + +static void library_drag_data_get_cb(GtkWidget *widget, GdkDragContext *context, + GtkSelectionData *selection_data, guint info, guint time, gpointer userdata) +{ + int c; + playlist_item_t* items; + instance_t* app = (instance_t*)userdata; + + g_warning("library_drag_data_get_cb"); + + items = library_get_selected_items(app, &c); + + /* clear item */ + if(items) + { + gtk_selection_data_set(selection_data, selection_data->target, 8, + (const guchar *)items, sizeof(playlist_item_t) * c); + free(items); + }; +}; + +static void library_drag_begin_cb(GtkWidget *widget, GdkDragContext *context, gpointer userdata) +{ + g_warning("library_drag_begin_cb"); + gtk_drag_source_set_icon_stock(widget, GTK_STOCK_DND); +}; + + +void library_init(instance_t* app) +{ + /* connect to library */ + app->library.handle[1] = mvcp_parser_init_remote(app->players.host, app->library.port); + app->library.handle[0] = mvcp_init(app->library.handle[1]); + if(mvcp_connect(app->library.handle[0]) != mvcp_ok) + { + g_warning("library_init: failed to connect to server %s", app->players.host); + return; + }; + + /* setup icons */ + app->library.icons[0] = create_pixbuf("Axialis_Team_playlist_open_16x16.png"); + app->library.icons[1] = create_pixbuf("Axialis_Team_playlist_save_16x16.png"); + + /* load lib */ + library_init_load(app); + + /* allow drag source */ + gtk_drag_source_set(app->library_tree, GDK_BUTTON1_MASK, + drag_targets, 1, (GdkDragAction)(GDK_ACTION_COPY)); + + /* set handlers */ + gtk_signal_connect(GTK_OBJECT(app->library_tree), "row-expanded", + GTK_SIGNAL_FUNC(on_library_row_expanded), app); + gtk_signal_connect(GTK_OBJECT(app->library_tree), "row-collapsed", + GTK_SIGNAL_FUNC(on_library_row_collapsed), app); + g_signal_connect(GTK_OBJECT(app->library_tree), "drag_data_get", + G_CALLBACK(library_drag_data_get_cb), app); + g_signal_connect(GTK_OBJECT(app->library_tree), "drag_begin", + G_CALLBACK(library_drag_begin_cb), app); + +}; + +static void library_get_selected_items_iter +( + GtkTreeModel *model, + GtkTreePath *path, + GtkTreeIter *iter, + gpointer data +) +{ + int l; + mvcp_dir_entry_t *dir; + mvcp_list_entry_t *list; + playlist_item_t** pitems = (playlist_item_t**)data; + playlist_item_t* items = *pitems; + + /* request pointers to list and dir entries of library items */ + gtk_tree_model_get(model, iter, + 3, &dir, + 4, &list, + -1); + + /* check if defined */ + if(dir && list) + { + /* allocate items */ + if(!items) + { + items = (playlist_item_t*)malloc(sizeof(playlist_item_t)); + memset(items, 0, sizeof(playlist_item_t)); + }; + + /* find numbers of items in list */ + for(l = 0; items[l].id[0]; l++); + g_warning("library_get_selected_items_iter: l=%d", l); + + /* realloc items */ + items = (playlist_item_t*)realloc(items, (l + 2) * sizeof(playlist_item_t)); + + /* clean last item */ + memset(&items[l + 1], 0, sizeof(playlist_item_t)); + /* setup items */ + memset(&items[l + 0], 0, sizeof(playlist_item_t)); + strncpy(items[l].id, dir->name, PATH_MAX); + strncpy(items[l].title, dir->full, PATH_MAX); + items[l].dur = list->size; + }; + + *pitems = items; +}; + +playlist_item_t* library_get_selected_items(instance_t* app, int *count) +{ + int l = 0; + playlist_item_t* items = NULL; + + GtkTreeSelection *selection; + + selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(app->library_tree)); + if(selection) + { + gtk_tree_selection_selected_foreach( + selection, + library_get_selected_items_iter, + &items); + + if(items) + for(; items[l].id[0]; l++); + }; + + *count = l; + return items; +}; + + +#if 0 playlist_item_t* omnplay_library_find(omnplay_instance_t* app, char* id) { int i; @@ -57,28 +391,68 @@ int omnplay_library_normalize_item(omnplay_instance_t* app, playlist_item_t* ite { int r = 0; playlist_item_t* lib; + playlist_item_t prev; pthread_mutex_lock(&app->library.lock); + prev = *item; + lib = omnplay_library_find(app, item->id); item->error = 0; if(lib) { - if(!item->title[0]) { strcpy(item->title, lib->title); - r = 1; + r++; }; - if(!item->dur || item->in < lib->in || (item->in + item->dur) > (lib->in + lib->dur)) + if(item->in < lib->in || item->in >= (lib->in + lib->dur)) { - item->dur = lib->dur; item->in = lib->in; - r = 1; + r++; }; + + if(!item->dur || (item->in + item->dur) > (lib->in + lib->dur)) + { + item->dur = lib->in + lib->dur - item->in; + r++; + }; + + if(r) + g_warning("omnplay_library_normalize_item: [%s,%d,%d]->[%s,%d,%d]\n", + prev.title, prev.in, prev.dur, item->title, item->in, item->dur); + } + else + { + r = 1; + item->error = PLAYLIST_ITEM_ERROR_LIB; + }; + + pthread_mutex_unlock(&app->library.lock); + + return r; +}; + +int omnplay_library_relink_item(omnplay_instance_t* app, playlist_item_t* item) +{ + int r = 0; + playlist_item_t* lib; + + pthread_mutex_lock(&app->library.lock); + + lib = omnplay_library_find(app, item->id); + + item->error = 0; + + if(lib) + { + r = 1; + strcpy(item->title, lib->title); + item->dur = lib->dur; + item->in = lib->in; } else { @@ -128,7 +502,7 @@ int omnplay_library_load_file(playlist_item_t* items, int *pcount, char* filenam /* open and process file */ if((f = fopen(filename, "rt"))) { - while( !feof(f) && c < (limit -1)) + while(!feof(f) && c < limit) { char *s, *sp_r, *sp_b; @@ -158,7 +532,9 @@ int omnplay_library_load_file(playlist_item_t* items, int *pcount, char* filenam /* insert item */ items[c++] = item; - }; + } + else + g_warning("omnplay_library_load_file: ignored line [%s]\n", l); } fclose(f); @@ -171,25 +547,11 @@ int omnplay_library_load_file(playlist_item_t* items, int *pcount, char* filenam *pcount = c; + g_warning("omnplay_library_load_file: loaded [%d] items from [%s] file, limit [%d]\n", c, filename, limit); + return r; }; -void omnplay_library_load(omnplay_instance_t* app) -{ - pthread_mutex_lock(&app->library.lock); - - if(app->library.filename[0]) - { - app->library.count = MAX_LIBRARY_ITEMS; - omnplay_library_load_file(app->library.item, &app->library.count, app->library.filename); - }; - - omnplay_library_sort(app); - - pthread_mutex_unlock(&app->library.lock); - - omnplay_library_draw(app); -}; static void omnplay_library_save_file(playlist_item_t* item, int count, char* filename) { @@ -206,8 +568,8 @@ static void omnplay_library_save_file(playlist_item_t* item, int count, char* fi frames2tc(item[i].in, 25.0, tc_in), frames2tc(item[i].dur, 25.0, tc_dur), item[i].title); - fclose(f); + g_warning("omnplay_library_save_file: written [%d] lines to file [%s]\n", count, filename); }; }; @@ -224,10 +586,9 @@ void omnplay_library_save(omnplay_instance_t* app) static void omnplay_get_content_cb(omnplay_instance_t* app, playlist_item_t* item, void* data) { - gdk_threads_enter(); - gtk_label_set_text(GTK_LABEL(app->library.refresh_ui[1]), item->id); - gdk_flush(); - gdk_threads_leave(); + if(!(app->library.id_display_idx % app->library.id_display_rate)) + omnplay_set_status(app, item->id); + app->library.id_display_idx++; }; static void* omnplay_library_refresh_proc(void* data) @@ -236,20 +597,26 @@ static void* omnplay_library_refresh_proc(void* data) int count, i; playlist_item_t* items; + gdk_threads_enter(); + gtk_widget_set_sensitive(app->window, FALSE); + gdk_flush(); + gdk_threads_leave(); + + omnplay_set_status(app, "Updating library..."); + items = (playlist_item_t*)malloc(sizeof(playlist_item_t) * MAX_LIBRARY_ITEMS); count = omnplay_get_content(app, items, MAX_LIBRARY_ITEMS, omnplay_get_content_cb, NULL); if(count > 0) { - gdk_threads_enter(); - gtk_label_set_text(GTK_LABEL(app->library.refresh_ui[1]), "Quering whois..."); - gdk_flush(); - gdk_threads_leave(); + omnplay_set_status(app, "Quering whois..."); if(app->library.whois[0]) omnplay_whois_list(app, items, &count); + omnplay_set_status(app, "Setting library..."); + pthread_mutex_lock(&app->library.lock); for(i = 0; i < count; i++) @@ -267,28 +634,33 @@ static void* omnplay_library_refresh_proc(void* data) gdk_threads_leave(); }; + omnplay_set_status(app, "Normalizing playlist..."); + free(items); gdk_threads_enter(); omnplay_playlist_normalize(app); - gtk_widget_destroy(app->library.refresh_ui[0]); gdk_flush(); gdk_threads_leave(); + omnplay_set_status(app, ""); + + gdk_threads_enter(); + gtk_widget_set_sensitive(app->window, TRUE); + gdk_flush(); + gdk_threads_leave(); + + return NULL; }; void omnplay_library_refresh(omnplay_instance_t* app) { - if(app->library.refresh_ui[0]) - pthread_join(app->library.refresh_thread, NULL); - - /* create UI for monitoring update */ - ui_library_refresh(app, &app->library.refresh_ui[0], &app->library.refresh_ui[1]); - - pthread_create(&app->library.refresh_thread, NULL, - omnplay_library_refresh_proc, app); + if(app->library.refresh_thread) + g_thread_join(app->library.refresh_thread); + app->library.refresh_thread = g_thread_create( + omnplay_library_refresh_proc, app, TRUE, NULL); }; void omnplay_library_draw(omnplay_instance_t* app) @@ -416,7 +788,7 @@ void omnplay_library_search(omnplay_instance_t* app, int next) if(i < app->library.count) { - fprintf(stderr, "found at pos=%d\n", i); + g_warning("found at pos=%d\n", i); /* select */ path = gtk_tree_path_new_from_indices(i, -1); @@ -429,3 +801,5 @@ void omnplay_library_search(omnplay_instance_t* app, int next) pthread_mutex_unlock(&app->library.lock); }; + +#endif