diff --git a/src/display/nr-filter-component-transfer.cpp b/src/display/nr-filter-component-transfer.cpp
index f4749de825f25b9a05c91c3ad0190c070f8174a7..87f87c95ab981b4c34ac920648ddb7b6eda2d99c 100644 (file)
*
* Authors:
* Felipe CorrĂȘa da Silva Sanches <felipe.sanches@gmail.com>
+ * Jasper van de Gronde <th.v.d.gronde@hccnet.nl>
*
* Copyright (C) 2007 authors
*
#include "display/nr-filter-utils.h"
#include "libnr/nr-pixblock.h"
#include "libnr/nr-blit.h"
+#include "libnr/nr-pixops.h"
#include <math.h>
-namespace NR {
+namespace Inkscape {
+namespace Filters {
FilterComponentTransfer::FilterComponentTransfer()
{
@@ -42,13 +45,10 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
int y0=in->area.y0;
int y1=in->area.y1;
- NRPixBlock *out = new NRPixBlock;
- nr_pixblock_setup_fast(out, NR_PIXBLOCK_MODE_R8G8B8A8N, x0, y0, x1, y1, true);
-
- // this primitive is defined for non-premultiplied RGBA values,
+ // this primitive is defined for RGBA values,
// thus convert them to that format before blending
bool free_in_on_exit = false;
- if (in->mode != NR_PIXBLOCK_MODE_R8G8B8A8N) {
+ if (in->mode != NR_PIXBLOCK_MODE_R8G8B8A8N && in->mode != NR_PIXBLOCK_MODE_R8G8B8A8P) {
NRPixBlock *original_in = in;
in = new NRPixBlock;
nr_pixblock_setup_fast(in, NR_PIXBLOCK_MODE_R8G8B8A8N,
nr_blit_pixblock_pixblock(in, original_in);
free_in_on_exit = true;
}
+ bool premultiplied = in->mode == NR_PIXBLOCK_MODE_R8G8B8A8P;
+
+ NRPixBlock *out = new NRPixBlock;
+ nr_pixblock_setup_fast(out, in->mode, x0, y0, x1, y1, true);
unsigned char *in_data = NR_PIXBLOCK_PX(in);
unsigned char *out_data = NR_PIXBLOCK_PX(out);
@@ -68,14 +72,14 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
int size = 4 * (y1-y0) * (x1-x0);
int i;
- for (int color=0;color<4;color++){
+ int color=4;
+ while(color-->0) {
int _vsize = tableValues[color].size();
- std::vector<gdouble> _tableValues = tableValues[color];
double _intercept = intercept[color];
double _slope = slope[color];
double _amplitude = amplitude[color];
double _exponent = exponent[color];
- double _offset = offset[color];
+ double _offset = offset[color];
switch(type[color]){
case COMPONENTTRANSFER_TYPE_IDENTITY:
for(i=color;i<size;i+=4){
@@ -83,15 +87,37 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
}
break;
case COMPONENTTRANSFER_TYPE_TABLE:
- if (_vsize==0){
+ if (_vsize<=1){
+ if (_vsize==1) {
+ g_warning("A component transfer table has to have at least two values.");
+ }
for(i=color;i<size;i+=4){
out_data[i]=in_data[i];
}
} else {
- for(i=color;i<size;i+=4){
- int k = (int)(((_vsize-1) * (double)in_data[i])/256);
- double dx = ((_vsize-1) * (double)in_data[i])/256 - k;
- out_data[i] = CLAMP_D_TO_U8(256 * (_tableValues[k] + dx * (_tableValues[k+1] - _tableValues[k]) ));
+ if (!premultiplied || color==3) {
+ std::vector<gdouble> _tableValues(tableValues[color]);
+ // Scale by 255 and add .5 to avoid having to add it later for rounding purposes
+ // Note that this means that CLAMP_D_TO_U8 cannot be used here (as it includes rounding!)
+ for(i=0;i<_vsize;i++) {
+ _tableValues[i] = std::max(0.,std::min(255.,255*_tableValues[i])) + .5;
+ }
+ for(i=color;i<size;i+=4){
+ int k = FAST_DIVIDE<255>((_vsize-1) * in_data[i]);
+ double dx = ((_vsize-1) * in_data[i])/255.0 - k;
+ out_data[i] = static_cast<unsigned char>(_tableValues[k] + dx * (_tableValues[k+1] - _tableValues[k]));
+ }
+ } else {
+ std::vector<gdouble> _tableValues(tableValues[color]);
+ for(i=0;i<_vsize;i++) {
+ _tableValues[i] = std::max(0.,std::min(1.,_tableValues[i]));
+ }
+ for(i=color;i<size;i+=4){
+ if (in_data[i+3-color]==0) continue;
+ int k = ((_vsize-1) * in_data[i]) / in_data[i+3-color];
+ double dx = ((_vsize-1) * in_data[i]) / (double)in_data[i+3-color] - k;
+ out_data[i] = CLAMP_D_TO_U8_ALPHA(out_data[i+3-color] * (_tableValues[k] + dx * (_tableValues[k+1] - _tableValues[k])), out_data[i+3-color]); // CLAMP includes rounding!
+ }
}
}
break;
@@ -101,24 +127,64 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
out_data[i] = in_data[i];
}
} else {
- for(i=color;i<size;i+=4){
- out_data[i] = CLAMP_D_TO_U8(256 * _tableValues[(int)((_vsize-1)*(double)in_data[i]/256)] );
+ if (!premultiplied || color==3) {
+ std::vector<unsigned char> _tableValues(_vsize);
+ // Convert to unsigned char
+ for(i=0;i<_vsize;i++) {
+ _tableValues[i] = CLAMP_D_TO_U8(255*tableValues[color][i]);
+ }
+ for(i=color;i<size;i+=4){
+ int k = FAST_DIVIDE<255>((_vsize-1) * in_data[i]);
+ out_data[i] = _tableValues[k];
+ }
+ } else {
+ std::vector<gdouble> _tableValues(tableValues[color]);
+ for(i=0;i<_vsize;i++) {
+ _tableValues[i] = std::max(0.,std::min(1.,_tableValues[i]));
+ }
+ for(i=color;i<size;i+=4){
+ if (in_data[i+3-color]==0) continue;
+ int k = ((_vsize-1) * in_data[i]) / in_data[i+3-color];
+ out_data[i] = CLAMP_D_TO_U8_ALPHA(out_data[i+3-color] * _tableValues[k], out_data[i+3-color]);
+ }
}
}
break;
case COMPONENTTRANSFER_TYPE_LINEAR:
- for(i=color;i<size;i+=4){
- out_data[i] = CLAMP_D_TO_U8(256 * (_slope * (double)in_data[i]/256 + _intercept));
+ if (!premultiplied || color==3) {
+ _intercept = 255*_intercept;
+ for(i=color;i<size;i+=4){
+ out_data[i] = CLAMP_D_TO_U8(_slope * in_data[i] + _intercept); // CLAMP includes rounding!
+ }
+ } else {
+ for(i=color;i<size;i+=4){
+ if (in_data[i+3-color]==0) continue;
+ double out = _slope * in_data[i] / in_data[i+3-color] + _intercept;
+ out_data[i] = CLAMP_D_TO_U8_ALPHA(out_data[i+3-color] * out, out_data[i+3-color]);
+ }
}
break;
case COMPONENTTRANSFER_TYPE_GAMMA:
- for(i=color;i<size;i+=4){
- out_data[i] = CLAMP_D_TO_U8(256 * (_amplitude * pow((double)in_data[i]/256, _exponent) + _offset));
+ if (!premultiplied || color==3) {
+ _amplitude *= pow(255.0, -_exponent+1); // The input should be divided by 255, then exponentiated and then multiplied by 255 again, instead the amplitude is modified accordingly.
+ _offset = 255*_offset;
+ for(i=color;i<size;i+=4){
+ out_data[i] = CLAMP_D_TO_U8(_amplitude * pow((double)in_data[i], _exponent) + _offset);
+ }
+ } else {
+ for(i=color;i<size;i+=4){
+ if (in_data[i+3-color]==0) continue;
+ double out = _amplitude * pow((double)in_data[i] / in_data[i+3-color], _exponent) + _offset;
+ out_data[i] = CLAMP_D_TO_U8_ALPHA(out_data[i+3-color] * out, out_data[i+3-color]);
+ }
}
break;
case COMPONENTTRANSFER_TYPE_ERROR:
//TODO: report an error here
+ g_warning("Component tranfer type \"error\".");
break;
+ default:
+ g_warning("Invalid tranfer type %d.", type[color]);
}
}
@@ -132,11 +198,12 @@ int FilterComponentTransfer::render(FilterSlot &slot, FilterUnits const &/*units
return 0;
}
-void FilterComponentTransfer::area_enlarge(NRRectL &/*area*/, Matrix const &/*trans*/)
+void FilterComponentTransfer::area_enlarge(NRRectL &/*area*/, Geom::Matrix const &/*trans*/)
{
}
-} /* namespace NR */
+} /* namespace Filters */
+} /* namespace Inkscape */
/*
Local Variables: