Remove ancient trunk folder from svn repository
[synfig.git] / synfig-core / src / modules / mod_gif / trgt_gif.h
diff --git a/synfig-core/src/modules/mod_gif/trgt_gif.h b/synfig-core/src/modules/mod_gif/trgt_gif.h
new file mode 100644 (file)
index 0000000..ae10e31
--- /dev/null
@@ -0,0 +1,213 @@
+/* === S Y N F I G ========================================================= */
+/*!    \file trgt_gif.h
+**     \brief Template Header
+**
+**     $Id$
+**
+**     \legal
+**     Copyright (c) 2002-2005 Robert B. Quattlebaum Jr., Adrian Bentley
+**
+**     This package 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 2 of
+**     the License, or (at your option) any later version.
+**
+**     This package 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.
+**     \endlegal
+**
+** === N O T E S ===========================================================
+**
+** ========================================================================= */
+
+/* === S T A R T =========================================================== */
+
+#ifndef __SYNFIG_TRGT_GIF_H
+#define __SYNFIG_TRGT_GIF_H
+
+/* === H E A D E R S ======================================================= */
+
+#include <synfig/target_scanline.h>
+#include <synfig/string.h>
+#include <synfig/smartfile.h>
+#include <cstdio>
+#include <synfig/surface.h>
+#include <synfig/palette.h>
+
+/* === M A C R O S ========================================================= */
+
+/* === T Y P E D E F S ===================================================== */
+
+/* === C L A S S E S & S T R U C T S ======================================= */
+
+class gif : public synfig::Target_Scanline
+{
+       SYNFIG_TARGET_MODULE_EXT
+private:
+       // Class for abstracting the
+       // output of the codes
+       struct bitstream
+       {
+               synfig::SmartFILE file;
+               unsigned char pool;
+               char curr_bit;
+               bitstream():pool(0),curr_bit(0),curr_pos(0) {}
+               bitstream(synfig::SmartFILE file):file(file),pool(0),curr_bit(0),curr_pos(0) {}
+               unsigned char buffer[256];
+               int curr_pos;
+
+               // Pushes a single bit onto the bit
+               void push_bit(bool bit)
+               {
+                       if(bit)
+                               pool|=(1<<(curr_bit));
+                       curr_bit++;
+                       if(curr_bit==8)
+                               empty();
+               }
+
+               // Empties out the current pool into
+               // the buffer. Calls 'dump()' if the
+               // buffer is full.
+               void empty()
+               {
+                       buffer[curr_pos++]=pool;
+                       curr_bit=0;
+                       pool=0;
+                       if(curr_pos==255)dump();
+               }
+
+               // If there is anything in the
+               // buffer or in the pool, it
+               // dumps it to the filestream.
+               // Buffer and pool are cleared.
+               void dump()
+               {
+                       if(curr_bit)
+                               empty();
+                       if(curr_pos || curr_bit)
+                       {
+                               fputc(curr_pos,file.get());
+                               fwrite(buffer,curr_pos,1,file.get());
+                               curr_pos=0;
+                       }
+               }
+
+               // Pushes a symbol of the given size
+               // onto the bitstream.
+               void push_value(int value, int size)
+               {
+                       int i;
+                       for(i=0;i<size;i++)
+                               push_bit((value>>(i))&1);
+               }
+       };
+
+       // Class for dealing with the LZW codes
+       struct lzwcode
+       {
+               int value; // the data element or character
+               int code; // lzwcode
+               struct lzwcode* kids; // children of this node
+               struct lzwcode* next; // siblings of this node
+
+               lzwcode():value(0),code(0),kids(0),next(0) { }
+
+               lzwcode *FindCode(int value)
+               {
+                       lzwcode *node=this;
+
+                       // check the children (kids) of the node for the value
+                       for (node = node->kids; node != 0; node = node->next)
+                               if (node->value == value)
+                                       return(node);
+                       return(0);
+               }
+
+               void AddNode(unsigned short code, unsigned short value)
+               {
+                       lzwcode *n = new lzwcode;
+
+                       // add a new child to node; the child will have code and value
+                       n->value = value;
+                       n->code = code;
+                       n->kids = 0;
+                       n->next = this->kids;
+                       this->kids = n;
+               }
+
+               static lzwcode * NewTable(int values)
+               {
+                       int i;
+                       lzwcode * table = new lzwcode;
+
+                       table->kids = 0;
+                       for (i = 0; i < values; i++)
+                               table->AddNode( i, i);
+
+                       return(table);
+               }
+
+               // Destructor just deletes any
+               // children and siblings.
+               ~lzwcode()
+               {
+                       if(kids)
+                               delete kids;
+                       if(next)
+                               delete next;
+               }
+       };
+
+private:
+       bitstream bs;
+       synfig::String filename;
+       synfig::SmartFILE file;
+       int
+               i,                      // General-purpose index
+               codesize,       // Current code size
+               rootsize,       // Size of pixel bits (will be recalculated)
+               nextcode;       // Next code to use
+       lzwcode *table,*next,*node;
+
+       synfig::Surface curr_surface;
+       etl::surface<unsigned char> curr_frame;
+       etl::surface<unsigned char> prev_frame;
+
+       int imagecount;
+       int cur_scanline;
+
+
+       // GIF compression parameters
+       bool lossy;
+       bool multi_image;
+       bool dithering;
+       int color_bits;
+       int iframe_density;
+       int loop_count;
+       bool local_palette;
+
+       synfig::Palette curr_palette;
+
+       void output_curr_palette();
+
+public:
+       gif(const char *filename);
+
+       virtual bool set_rend_desc(synfig::RendDesc *desc);
+       virtual bool init();
+       virtual bool start_frame(synfig::ProgressCallback *cb);
+       virtual void end_frame();
+
+       virtual ~gif();
+
+       virtual synfig::Color * start_scanline(int scanline);
+       virtual bool end_scanline(void);
+
+};
+
+/* === E N D =============================================================== */
+
+#endif