image.cpp
上传用户:yhdzpy8989
上传日期:2007-06-13
资源大小:13604k
文件大小:9k
源码类别:

生物技术

开发平台:

C/C++

  1. /*
  2.  * ===========================================================================
  3.  * PRODUCTION $Log: image.cpp,v $
  4.  * PRODUCTION Revision 1000.2  2004/06/01 19:41:18  gouriano
  5.  * PRODUCTION PRODUCTION: UPGRADED [GCC34_MSVC7] Dev-tree R1.6
  6.  * PRODUCTION
  7.  * ===========================================================================
  8.  */
  9. /*  $Id: image.cpp,v 1000.2 2004/06/01 19:41:18 gouriano Exp $
  10.  * ===========================================================================
  11.  *
  12.  *                            PUBLIC DOMAIN NOTICE
  13.  *               National Center for Biotechnology Information
  14.  *
  15.  *  This software/database is a "United States Government Work" under the
  16.  *  terms of the United States Copyright Act.  It was written as part of
  17.  *  the author's official duties as a United States Government employee and
  18.  *  thus cannot be copyrighted.  This software/database is freely available
  19.  *  to the public for use. The National Library of Medicine and the U.S.
  20.  *  Government have not placed any restriction on its use or reproduction.
  21.  *
  22.  *  Although all reasonable efforts have been taken to ensure the accuracy
  23.  *  and reliability of the software and data, the NLM and the U.S.
  24.  *  Government do not and cannot warrant the performance or results that
  25.  *  may be obtained by using this software or data. The NLM and the U.S.
  26.  *  Government disclaim all warranties, express or implied, including
  27.  *  warranties of performance, merchantability or fitness for any particular
  28.  *  purpose.
  29.  *
  30.  *  Please cite the author in any work or product based on this material.
  31.  *
  32.  * ===========================================================================
  33.  *
  34.  * Authors:  Mike DiCuccio
  35.  *
  36.  * File Description:
  37.  *    CImage -- interface for manipulating image data
  38.  */
  39. #include <ncbi_pch.hpp>
  40. #include <util/image/image.hpp>
  41. #include <util/image/image_exception.hpp>
  42. BEGIN_NCBI_SCOPE
  43. CImage::CImage()
  44.     : m_Width(0),
  45.       m_Height(0),
  46.       m_Depth(3)
  47. {
  48.     return;
  49. }
  50. CImage::CImage(size_t width, size_t height, size_t depth)
  51.     : m_Width(0),
  52.       m_Height(0),
  53.       m_Depth(3)
  54. {
  55.     Init(width, height, depth);
  56. }
  57. void CImage::Init(size_t width, size_t height, size_t depth)
  58. {
  59.     _TRACE("CImage::Init(): " << width << ", " << height << ", " << depth);
  60.     switch (depth) {
  61.     case 3:
  62.     case 4:
  63.         m_Data.resize(width * height * depth);
  64.         break;
  65.     default:
  66.         {{
  67.              string msg("CImage::Init(): depth = ");
  68.              msg += NStr::IntToString(depth);
  69.              msg += " not implemented";
  70.              NCBI_THROW(CImageException, eInvalidDimension, msg);
  71.         }}
  72.     }
  73.     m_Width  = width;
  74.     m_Height = height;
  75.     m_Depth  = depth;
  76. }
  77. const unsigned char* CImage::GetData(void) const
  78. {
  79.     _ASSERT(m_Data.size());
  80.     return &(m_Data[0]);
  81. }
  82. unsigned char* CImage::SetData(void)
  83. {
  84.     _ASSERT(m_Data.size());
  85.     return &(m_Data[0]);
  86. }
  87. //
  88. // GetAspectRatio()
  89. // This computes the aspect ratio for the image (width / height)
  90. //
  91. float CImage::GetAspectRatio(void) const
  92. {
  93.     return float (m_Width) / float (m_Height);
  94. }
  95. //
  96. // set the alpha channel to a given value.  The red channel is channel 3.
  97. // This will optionnally add the channel if it doesn't exist.
  98. //
  99. void CImage::SetAlpha(unsigned char val, bool add_if_unavailable)
  100. {
  101.     if (GetDepth() == 3) {
  102.         if (add_if_unavailable) {
  103.             SetDepth(4);
  104.         } else {
  105.             NCBI_THROW(CImageException, eInvalidDimension,
  106.                        "CImage::SetAlpha(): attempt to set alpha in "
  107.                        "24-bit image");
  108.         }
  109.     }
  110.     SetChannel(eAlpha, val);
  111. }
  112. //
  113. // SetDepth()
  114. // This will force the image to be 24-bit or 32-bit
  115. //
  116. void CImage::SetDepth(size_t depth)
  117. {
  118.     if ( !m_Data.size() ) {
  119.         return;
  120.     }
  121.     switch (depth) {
  122.     case 3:
  123.         if (m_Depth == 4) {
  124.             // squash our data.  we can do this without explicitly reallocating
  125.             // our buffer - we just have junk on the end
  126.             TImageStrip::const_iterator from_data = m_Data.begin();
  127.             TImageStrip::iterator       to_data   = m_Data.begin();
  128.             for ( ;  from_data != m_Data.end();  ) {
  129.                 *to_data++ = *from_data++;
  130.                 *to_data++ = *from_data++;
  131.                 *to_data++ = *from_data++;
  132.                 // skip the alpha channel
  133.                 ++from_data;
  134.             }
  135.             m_Data.resize(m_Width * m_Height * 3);
  136.             m_Depth = 3;
  137.         }
  138.         break;
  139.     case 4:
  140.         if (m_Depth == 3) {
  141.             // expand our data.
  142.             // we do this by expanding in-place
  143.             m_Data.resize(m_Width * m_Height * 4);
  144.             m_Depth = 4;
  145.             // now we need to expand our data
  146.             // we can do this by marching backwares through the data
  147.             TImageStrip::const_reverse_iterator from_data(m_Data.end());
  148.             TImageStrip::const_reverse_iterator end_data (m_Data.begin());
  149.             TImageStrip::reverse_iterator       to_data  (m_Data.end());
  150.             // march to the actual end of the data
  151.             from_data += m_Width * m_Height;
  152.             for ( ;  from_data != end_data;  ) {
  153.                 // insert an alpha channel - this is the first because we're
  154.                 // marching backwards
  155.                 *to_data++ = 255;
  156.                 // copy (in reverse order) BGR data
  157.                 *to_data++ = *from_data++;
  158.                 *to_data++ = *from_data++;
  159.                 *to_data++ = *from_data++;
  160.             }
  161.         }
  162.         break;
  163.         
  164.     default:
  165.         {{
  166.             string msg("CImage::SetDepth(): invalid depth: ");
  167.             msg += NStr::IntToString(depth);
  168.             NCBI_THROW(CImageException, eInvalidDimension, msg);
  169.         }}
  170.     }
  171. }
  172. //
  173. // SetChannel()
  174. // Sets a given channel to a certain value
  175. //
  176. void CImage::SetChannel(size_t channel, unsigned char val)
  177. {
  178.     if ( !m_Data.size() ) {
  179.         return;
  180.     }
  181.     if (channel > 3) {
  182.         NCBI_THROW(CImageException, eInvalidDimension,
  183.                    "CImage::SetChannel(): channel out of bounds");
  184.     }
  185.     TImageStrip::iterator data = m_Data.begin() + channel;
  186.     TImageStrip::iterator end  =
  187.         m_Data.begin() + m_Width * m_Height * m_Depth + channel;
  188.     for ( ;  data != end;  data += m_Depth) {
  189.         *data = val;
  190.     }
  191. }
  192. //
  193. // GetSubImage()
  194. // This function returns a new image that is a subimage of the current image
  195. // the width and height will be clamped to the left / bottom margins; the x and
  196. // y positions must be valid.
  197. //
  198. CImage* CImage::GetSubImage(size_t x, size_t y, size_t w, size_t h) const
  199. {
  200.     if ( !m_Data.size() ) {
  201.         NCBI_THROW(CImageException, eInvalidImage,
  202.                    "CImage::GetSubImage(): image is empty");
  203.     }
  204.     if (x >= m_Width  ||  y >= m_Height) {
  205.         string msg("CImage::GetSubImage(): invalid starting pos: ");
  206.         msg += NStr::IntToString(x);
  207.         msg += ", ";
  208.         msg += NStr::IntToString(y);
  209.         NCBI_THROW(CImageException, eInvalidImage, msg);
  210.     }
  211.     if (x + w >= m_Width) {
  212.         LOG_POST(Warning << "CImage::GetSubImage(): clamping width to " << w);
  213.         w = m_Width - x;
  214.     }
  215.     if (y + h >= m_Width) {
  216.         LOG_POST(Warning << "CImage::GetSubImage(): clamping width to " << w);
  217.         h = m_Height - y;
  218.     }
  219.     CRef<CImage> image(new CImage(w, h, m_Depth));
  220.     const unsigned char* from_data = GetData() +
  221.                                      y * m_Width * m_Depth + x * m_Depth;
  222.     unsigned char*       to_data   = image->SetData();
  223.     size_t from_stride = m_Width * m_Depth;
  224.     size_t to_stride   = w * m_Depth;
  225.     for (size_t i = 0;  i < h;  ++i) {
  226.         memcpy(to_data, from_data, w * m_Depth);
  227.         to_data   += to_stride;
  228.         from_data += from_stride;
  229.     }
  230.     return image.Release();
  231. }
  232. void CImage::Flip(void)
  233. {
  234.     if ( !m_Data.size() ) {
  235.         return;
  236.     }
  237.     unsigned char* from = SetData();
  238.     unsigned char* to   = from + (m_Height - 1) * m_Width * m_Depth;
  239.     size_t stride = m_Width * m_Depth;
  240.     while (to > from) {
  241.         for (size_t pos = 0;  pos < stride;  ++pos) {
  242.             std::swap(from[pos], to[pos]);
  243.         }
  244.         from += stride;
  245.         to   -= stride;
  246.     }
  247. }
  248. END_NCBI_SCOPE
  249. /*
  250.  * ===========================================================================
  251.  * $Log: image.cpp,v $
  252.  * Revision 1000.2  2004/06/01 19:41:18  gouriano
  253.  * PRODUCTION: UPGRADED [GCC34_MSVC7] Dev-tree R1.6
  254.  *
  255.  * Revision 1.6  2004/05/17 21:07:58  gorelenk
  256.  * Added include of PCH ncbi_pch.hpp
  257.  *
  258.  * Revision 1.5  2003/12/16 15:49:35  dicuccio
  259.  * Large re-write of image handling.  Added improved error-handling and support
  260.  * for streams-based i/o (via hooks into each client library).
  261.  *
  262.  * Revision 1.4  2003/10/02 15:37:33  ivanov
  263.  * Get rid of compilation warnings
  264.  *
  265.  * Revision 1.3  2003/06/12 19:49:56  dicuccio
  266.  * Added Flip() to flip an image along the y-axis
  267.  *
  268.  * Revision 1.2  2003/06/09 19:27:53  dicuccio
  269.  * Added GetAspectRatio()
  270.  *
  271.  * Revision 1.1  2003/06/03 15:17:13  dicuccio
  272.  * Initial revision of image library
  273.  *
  274.  * ===========================================================================
  275.  */