Mercurial > audlegacy-plugins
view src/Visualization/paranormal/pn/pnvis.c @ 0:13389e613d67 trunk
[svn] - initial import of audacious-plugins tree (lots to do)
author | nenolod |
---|---|
date | Mon, 18 Sep 2006 01:11:49 -0700 |
parents | |
children |
line wrap: on
line source
/* Paranormal - A highly customizable audio visualization library * Copyright (C) 2001 Jamie Gennis <jgennis@mindspring.com> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the Free * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include <config.h> #include <glib.h> #include "pnvis.h" #include "pnactuatorfactory.h" #include "pnerror.h" static void pn_vis_class_init (PnVisClass *class); static void pn_vis_init (PnVis *vis, PnVisClass *class); /* GObject signals */ static void pn_vis_finalize (GObject *gobject); /* PnObject signals */ static void pn_vis_destroy (PnObject *object); /* PnUserObject methods */ static void pn_vis_save_thyself (PnUserObject *user_object, xmlNodePtr node); static void pn_vis_load_thyself (PnUserObject *user_object, xmlNodePtr node); static PnUserObjectClass *parent_class = NULL; GType pn_vis_get_type (void) { static GType vis_type = 0; if (! vis_type) { static const GTypeInfo vis_info = { sizeof (PnVisClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc) pn_vis_class_init, NULL, /* class_finalize */ NULL, /* class_data */ sizeof (PnVis), 0, /* n_preallocs */ (GInstanceInitFunc) pn_vis_init }; /* FIXME: should this be dynamic? */ vis_type = g_type_register_static (PN_TYPE_USER_OBJECT, "PnVis", &vis_info, 0); } return vis_type; } static void pn_vis_class_init (PnVisClass *class) { GObjectClass *gobject_class; PnObjectClass *object_class; PnUserObjectClass *user_object_class; parent_class = g_type_class_peek_parent (class); gobject_class = (GObjectClass *) class; object_class = (PnObjectClass *) class; user_object_class = (PnUserObjectClass *) class; /* GObject signals */ gobject_class->finalize = pn_vis_finalize; /* PnObject signals */ object_class->destroy = pn_vis_destroy; /* PnUserObject methods */ user_object_class->save_thyself = pn_vis_save_thyself; user_object_class->load_thyself = pn_vis_load_thyself; } static void pn_vis_init (PnVis *vis, PnVisClass *class) { /* Set a name for xml stuffs */ pn_user_object_set_name (PN_USER_OBJECT (vis), "Paranormal_Visualization"); /* Create a new image context */ vis->root_image = pn_image_new (); pn_object_ref (PN_OBJECT (vis->root_image)); pn_object_sink (PN_OBJECT (vis->root_image)); } static void pn_vis_destroy (PnObject *object) { PnVis *vis = (PnVis *) object; if (vis->root_actuator) pn_object_unref (PN_OBJECT (vis->root_actuator)); if (vis->root_image) pn_object_unref (PN_OBJECT (vis->root_image)); } static void pn_vis_finalize (GObject *gobject) { PnVis *vis; vis = (PnVis *) gobject; } static void pn_vis_save_thyself (PnUserObject *user_object, xmlNodePtr node) { PnVis *vis; xmlNsPtr ns; xmlNodePtr actuators_node, actuator_node; g_return_if_fail (user_object != NULL); g_return_if_fail (PN_IS_VIS (user_object)); g_return_if_fail (node != NULL); vis = (PnVis *) user_object; ns = xmlNewNs (node, PN_XML_NS_HREF, (xmlChar *) "pn"); xmlSetNs (node, ns); actuators_node = xmlNewChild (node, NULL, (xmlChar *) "Actuators", NULL); actuator_node = xmlNewChild (actuators_node, NULL, (xmlChar *) user_object->name, NULL); if (vis->root_actuator) pn_user_object_save_thyself (PN_USER_OBJECT (vis->root_actuator), actuator_node); if (parent_class->save_thyself) parent_class->save_thyself (user_object, node); } static void pn_vis_load_thyself (PnUserObject *user_object, const xmlNodePtr node) { PnVis *vis; xmlNodePtr actuators_node, actuator_node; #if 0 , tptr; #endif PnActuator *root_actuator = NULL; g_return_if_fail (user_object != NULL); g_return_if_fail (PN_IS_VIS (user_object)); g_return_if_fail (node != NULL); vis = (PnVis *) user_object; /* find the 'Actuators' node */ for (actuators_node = node->xmlChildrenNode; actuators_node; actuators_node = actuators_node->next) if (g_strcasecmp ((gchar *) actuators_node->name, "Actuators") == 0) break; if (! actuators_node) { pn_error ("unable to load a PnVis from xml node \"%s\"", node->name); return; } /* get the root actuator's node */ for (actuator_node = actuators_node->xmlChildrenNode; actuator_node != NULL && !g_strcasecmp((gchar *) actuator_node->name, "text"); actuator_node = actuator_node->next); if (! actuator_node) goto done; /* Create a new actuator (reworked by nenolod) */ #if 0 for (tptr = actuator_node; tptr != NULL && root_actuator == NULL; tptr = tptr->next) { xmlNodePtr ttptr; for (ttptr = tptr; ttptr != NULL; ttptr = ttptr->xmlChildrenNode) } #endif root_actuator = pn_actuator_factory_new_actuator_from_xml (actuator_node); if (!root_actuator) pn_error ("Unknown actuator \"%s\" -> '%s'", actuator_node->name, actuator_node->content); pn_vis_set_root_actuator (vis, root_actuator); done: if (parent_class->load_thyself) parent_class->load_thyself (user_object, node); } PnVis* pn_vis_new (guint image_width, guint image_height) { PnVis *vis; g_return_val_if_fail (image_width > 0, NULL); g_return_val_if_fail (image_height > 0, NULL); vis = (PnVis *) g_object_new (PN_TYPE_VIS, NULL); pn_vis_set_image_size (vis, image_width, image_height); return vis; } void pn_vis_set_root_actuator (PnVis *vis, PnActuator *actuator) { g_return_if_fail (vis != NULL); g_return_if_fail (PN_IS_VIS (vis)); g_return_if_fail (actuator != NULL); g_return_if_fail (PN_IS_ACTUATOR (actuator)); if (vis->root_actuator) pn_object_unref (PN_OBJECT (vis->root_actuator)); vis->root_actuator = actuator; pn_object_ref (PN_OBJECT (actuator)); pn_object_sink (PN_OBJECT (actuator)); pn_user_object_set_owner (PN_USER_OBJECT (actuator), PN_USER_OBJECT (vis)); pn_actuator_prepare (actuator, vis->root_image); } PnActuator* pn_vis_get_root_actuator (PnVis *vis) { g_return_val_if_fail (vis != NULL, NULL); g_return_val_if_fail (PN_IS_VIS (vis), NULL); return vis->root_actuator; } void pn_vis_set_image_size (PnVis *vis, guint width, guint height) { g_return_if_fail (vis != NULL); g_return_if_fail (PN_IS_VIS (vis)); pn_image_set_size (vis->root_image, width, height); if (vis->root_actuator) pn_actuator_prepare (vis->root_actuator, vis->root_image); } gboolean pn_vis_save_to_file (PnVis *vis, const gchar *fname) { xmlDocPtr doc; doc = xmlNewDoc ((xmlChar *) "1.0"); doc->xmlRootNode = xmlNewDocNode (doc, NULL, (xmlChar *) "BUG", NULL); pn_user_object_save_thyself (PN_USER_OBJECT (vis), doc->xmlRootNode); if ( xmlSaveFile (fname, doc) == -1) { xmlFreeDoc (doc); return FALSE; } xmlFreeDoc (doc); return TRUE; } gboolean pn_vis_load_from_file (PnVis *vis, const gchar *fname) { xmlDocPtr doc; xmlNodePtr root_node; xmlNsPtr ns; doc = xmlParseFile (fname); if (! doc) { pn_error ("unable to parse file \"%s\"", fname); return FALSE; } root_node = xmlDocGetRootElement (doc); if (! root_node) { pn_error ("no root element for file \"%s\"", fname); return FALSE; } ns = xmlSearchNsByHref (doc, root_node, PN_XML_NS_HREF); if (! ns) { pn_error ("invalid file format: paranormal namespace not found"); return FALSE; } if (g_strcasecmp ((gchar *) root_node->name, pn_user_object_get_name (PN_USER_OBJECT (vis)))) { pn_error ("invalid file format: this file does not contain a Paranormal visualization"); return FALSE; } pn_user_object_load_thyself (PN_USER_OBJECT (vis), root_node); return TRUE; } PnImage* pn_vis_render (PnVis *vis, PnAudioData *audio_data) { g_return_val_if_fail (vis != NULL, NULL); g_return_val_if_fail (PN_IS_VIS (vis), NULL); g_return_val_if_fail (audio_data != NULL, NULL); /* g_return_val_if_fail (PN_IS_AUDIO_DATA (audio_data), NULL); */ if (vis->root_actuator) pn_actuator_execute (vis->root_actuator, vis->root_image, audio_data); return vis->root_image; }