Code

fix compositing for premultiplication and non-alpha cases
[inkscape.git] / src / sp-gaussian-blur.cpp
index 0e26fbbf4ef13a623a854752c255636f8ec453db..8b7578fe8f058b751e54715cb95c412725487624 100644 (file)
@@ -6,9 +6,10 @@
  */
 /*
  * Authors:
- *   hugo Rodrigues <haa.rodrigues@gmail.com>
+ *   Hugo Rodrigues <haa.rodrigues@gmail.com>
+ *   Niko Kiirala <niko@kiirala.com>
  *
- * Copyright (C) 2006 Hugo Rodrigues
+ * Copyright (C) 2006,2007 Authors
  *
  * Released under GNU GPL, read the file 'COPYING' for more information
  */
 #include "sp-gaussian-blur.h"
 #include "xml/repr.h"
 
+#include "display/nr-filter.h"
+#include "display/nr-filter-primitive.h"
+#include "display/nr-filter-gaussian.h"
+#include "display/nr-filter-types.h"
+
 //#define SP_MACROS_SILENT
 //#include "macros.h"
 
-#define DEBUG_GAUSSIANBLUR
-#ifdef DEBUG_GAUSSIANBLUR
-# define debug(f, a...) { g_print("%s(%d) %s:", \
-                                  __FILE__,__LINE__,__FUNCTION__); \
-                          g_print(f, ## a); \
-                          g_print("\n"); \
-                        }
-#else
-# define debug(f, a...) /**/
-#endif
-
 /* GaussianBlur base class */
 
 static void sp_gaussianBlur_class_init(SPGaussianBlurClass *klass);
@@ -46,8 +41,9 @@ static void sp_gaussianBlur_release(SPObject *object);
 static void sp_gaussianBlur_set(SPObject *object, unsigned int key, gchar const *value);
 static void sp_gaussianBlur_update(SPObject *object, SPCtx *ctx, guint flags);
 static Inkscape::XML::Node *sp_gaussianBlur_write(SPObject *object, Inkscape::XML::Node *repr, guint flags);
+static void sp_gaussianBlur_build_renderer(SPFilterPrimitive *primitive, NR::Filter *filter);
 
-static SPObjectClass *gaussianBlur_parent_class;
+static SPFilterPrimitiveClass *gaussianBlur_parent_class;
 
 GType
 sp_gaussianBlur_get_type()
@@ -65,7 +61,7 @@ sp_gaussianBlur_get_type()
             (GInstanceInitFunc) sp_gaussianBlur_init,
             NULL,    /* value_table */
         };
-        gaussianBlur_type = g_type_register_static(SP_TYPE_OBJECT, "SPGaussianBlur", &gaussianBlur_info, (GTypeFlags)0);
+        gaussianBlur_type = g_type_register_static(SP_TYPE_FILTER_PRIMITIVE, "SPGaussianBlur", &gaussianBlur_info, (GTypeFlags)0);
     }
     return gaussianBlur_type;
 }
@@ -74,22 +70,22 @@ static void
 sp_gaussianBlur_class_init(SPGaussianBlurClass *klass)
 {
     SPObjectClass *sp_object_class = (SPObjectClass *)klass;
+    SPFilterPrimitiveClass *sp_primitive_class = (SPFilterPrimitiveClass *)klass;
 
-    gaussianBlur_parent_class = (SPObjectClass*)g_type_class_peek_parent(klass);
+    gaussianBlur_parent_class = (SPFilterPrimitiveClass *)g_type_class_peek_parent(klass);
 
     sp_object_class->build = sp_gaussianBlur_build;
     sp_object_class->release = sp_gaussianBlur_release;
     sp_object_class->write = sp_gaussianBlur_write;
     sp_object_class->set = sp_gaussianBlur_set;
     sp_object_class->update = sp_gaussianBlur_update;
+
+    sp_primitive_class->build_renderer = sp_gaussianBlur_build_renderer;
 }
 
 static void
 sp_gaussianBlur_init(SPGaussianBlur *gaussianBlur)
 {
-    debug("0x%p",gaussianBlur);
-
-//    gaussianBlur->stdDeviation = 1;
 }
 
 /**
@@ -100,7 +96,6 @@ sp_gaussianBlur_init(SPGaussianBlur *gaussianBlur)
 static void
 sp_gaussianBlur_build(SPObject *object, SPDocument *document, Inkscape::XML::Node *repr)
 {
-    debug("0x%p",object);
     if (((SPObjectClass *) gaussianBlur_parent_class)->build) {
         ((SPObjectClass *) gaussianBlur_parent_class)->build(object, document, repr);
     }
@@ -115,7 +110,6 @@ sp_gaussianBlur_build(SPObject *object, SPDocument *document, Inkscape::XML::Nod
 static void
 sp_gaussianBlur_release(SPObject *object)
 {
-    debug("0x%p",object);
 
     if (((SPObjectClass *) gaussianBlur_parent_class)->release)
         ((SPObjectClass *) gaussianBlur_parent_class)->release(object);
@@ -127,13 +121,12 @@ sp_gaussianBlur_release(SPObject *object)
 static void
 sp_gaussianBlur_set(SPObject *object, unsigned int key, gchar const *value)
 {
-    debug("0x%p %s(%u): '%s'",object,
-            sp_attribute_name(key),key,value);
     SPGaussianBlur *gaussianBlur = SP_GAUSSIANBLUR(object);
-
+    
     switch(key) {
-       case SP_ATTR_STDDEVIATION:
-               gaussianBlur->stdDeviation.set(value);
+        case SP_ATTR_STDDEVIATION:
+            gaussianBlur->stdDeviation.set(value);
+            object->parent->requestModified(SP_OBJECT_MODIFIED_FLAG);
             break;
         default:
             if (((SPObjectClass *) gaussianBlur_parent_class)->set)
@@ -149,13 +142,8 @@ sp_gaussianBlur_set(SPObject *object, unsigned int key, gchar const *value)
 static void
 sp_gaussianBlur_update(SPObject *object, SPCtx *ctx, guint flags)
 {
-    debug("0x%p",object);
-
-    if (flags & (SP_OBJECT_MODIFIED_FLAG | SP_OBJECT_STYLE_MODIFIED_FLAG |
-                 SP_OBJECT_VIEWPORT_MODIFIED_FLAG)) {
-
-        /* do something to trigger redisplay, updates? */
-
+    if (flags & SP_OBJECT_MODIFIED_FLAG) {
+        sp_object_read_attr(object, "stdDeviation");
     }
 
     if (((SPObjectClass *) gaussianBlur_parent_class)->update) {
@@ -169,15 +157,14 @@ sp_gaussianBlur_update(SPObject *object, SPCtx *ctx, guint flags)
 static Inkscape::XML::Node *
 sp_gaussianBlur_write(SPObject *object, Inkscape::XML::Node *repr, guint flags)
 {
-    debug("0x%p",object);
 
     // Inkscape-only object, not copied during an "plain SVG" dump:
     if (flags & SP_OBJECT_WRITE_EXT) {
         if (repr) {
             // is this sane?
-            repr->mergeFrom(SP_OBJECT_REPR(object), "id");
+            // repr->mergeFrom(SP_OBJECT_REPR(object), "id");
         } else {
-            repr = SP_OBJECT_REPR(object)->duplicate();
+            repr = SP_OBJECT_REPR(object)->duplicate(NULL); // FIXME
         }
     }
 
@@ -189,6 +176,35 @@ sp_gaussianBlur_write(SPObject *object, Inkscape::XML::Node *repr, guint flags)
 }
 
 
+void  sp_gaussianBlur_setDeviation(SPGaussianBlur *blur, float num)
+{
+    blur->stdDeviation.setNumber(num);
+}
+void  sp_gaussianBlur_setDeviation(SPGaussianBlur *blur, float num, float optnum)
+{
+    blur->stdDeviation.setNumber(num);
+    blur->stdDeviation.setOptNumber(optnum);
+}
+
+static void sp_gaussianBlur_build_renderer(SPFilterPrimitive *primitive, NR::Filter *filter) {
+    SPGaussianBlur *sp_blur = SP_GAUSSIANBLUR(primitive);
+
+    int handle = filter->add_primitive(NR::NR_FILTER_GAUSSIANBLUR);
+    NR::FilterPrimitive *nr_primitive = filter->get_primitive(handle);
+    NR::FilterGaussian *nr_blur = dynamic_cast<NR::FilterGaussian*>(nr_primitive);
+
+    sp_filter_primitive_renderer_common(primitive, nr_primitive);
+
+    gfloat num = sp_blur->stdDeviation.getNumber();
+    if (num >= 0.0) {
+        gfloat optnum = sp_blur->stdDeviation.getOptNumber();
+        if(optnum >= 0.0)
+            nr_blur->set_deviation((double) num, (double) optnum);
+        else
+            nr_blur->set_deviation((double) num);
+    }
+}
+
 /*
   Local Variables:
   mode:c++