수안이의 컴퓨터 연구실

  • Mainpage
  • About Me
  • Tags
  • Metapage
  • Notice
  • Location
  • Keywords
  • Guestbook
  • Admin
  • Write an Article
  • Total | 1693891
  • Today | 242
  • Yesterday | 588

Programming/Win32 API2007/03/19 09:10

Bitmap 이미지 resizing 하기

원본 : http://www.debuglab.com/knowledge/bitmapstretch.html

1.요약

Bitmap 이미지를 resizing 하는 방법에 대해서 알아봅니다.


2.본문

HBITMAP 을 가지고, 너비와 높이를 주면 적당한 HBITMAP 을 넘겨주는 함수를 소개합니다. 유용하게 사용하세요.


3.예제

#include "stdafx.h" 

#include <math.h> 



#ifdef _DEBUG 

#define new DEBUG_NEW 

#undef THIS_FILE 

static char THIS_FILE[] = __FILE__; 

#endif 



///////////////////////////////////////////////// 



// helper function prototypes 

BITMAPINFO *PrepareRGBBitmapInfo(WORD wWidth, 

                                 WORD wHeight); 



void ShrinkData(BYTE *pInBuff, 

                WORD wWidth, 

                WORD wHeight, 

                BYTE *pOutBuff, 

                WORD wNewWidth, 

                WORD wNewHeight); 



void EnlargeData(BYTE *pInBuff, 

                 WORD wWidth, 

                 WORD wHeight, 

                 BYTE *pOutBuff, 

                 WORD wNewWidth, 

                 WORD wNewHeight); 



///////////////////////////////////////////////// 

// Main resize function 



HBITMAP ScaleBitmap(HBITMAP hBmp, 

                    WORD wNewWidth, 

                    WORD wNewHeight) 

{ 

    BITMAP bmp; 

    ::GetObject(hBmp, sizeof(BITMAP), &bmp); 

    

    // check for valid size 

    if((bmp.bmWidth > wNewWidth && bmp.bmHeight < wNewHeight) || 

        (bmp.bmWidth < wNewWidth && bmp.bmHeight > wNewHeight)) 

        return NULL; 

    

    HDC hDC = ::GetDC(NULL); 

    

    BITMAPINFO *pbi = PrepareRGBBitmapInfo((WORD)bmp.bmWidth, 

        (WORD)bmp.bmHeight); 

    

    BYTE *pData = new BYTE[pbi->bmiHeader.biSizeImage]; 

    

    ::GetDIBits(hDC, hBmp, 0, bmp.bmHeight, pData, 

        pbi, DIB_RGB_COLORS); 

    

    delete pbi; 

    

    pbi = PrepareRGBBitmapInfo(wNewWidth, wNewHeight); 

    BYTE *pData2 = new BYTE[pbi->bmiHeader.biSizeImage]; 

    

    if(bmp.bmWidth >= wNewWidth && bmp.bmHeight >= wNewHeight) 

        ShrinkData(pData, (WORD)bmp.bmWidth, (WORD)bmp.bmHeight, 

        pData2, wNewWidth, wNewHeight); 

    else 

        EnlargeData(pData, (WORD)bmp.bmWidth, (WORD)bmp.bmHeight, 

        pData2, wNewWidth, wNewHeight); 

    

    delete pData; 

    

    HBITMAP hResBmp = ::CreateCompatibleBitmap(hDC, 

        wNewWidth, 

        wNewHeight); 

    

    ::SetDIBits(hDC, hResBmp, 0, wNewHeight, pData2, 

        pbi, DIB_RGB_COLORS); 

    

    ::ReleaseDC(NULL, hDC); 

    

    delete pbi; 

    delete pData2; 

    

    return hResBmp; 

} 



///////////////////////////////////////////////// 



BITMAPINFO *PrepareRGBBitmapInfo(WORD wWidth, WORD wHeight) 

{ 

    BITMAPINFO *pRes = new BITMAPINFO; 

    ::ZeroMemory(pRes, sizeof(BITMAPINFO)); 

    pRes->bmiHeader.biSize = sizeof(BITMAPINFOHEADER); 

    pRes->bmiHeader.biWidth = wWidth; 

    pRes->bmiHeader.biHeight = wHeight; 

    pRes->bmiHeader.biPlanes = 1; 

    pRes->bmiHeader.biBitCount = 24; 

    

    pRes->bmiHeader.biSizeImage = ((3 * wWidth + 3) & ~3) 

        * wHeight; 

    

    return pRes; 

} 



///////////////////////////////////////////////// 



float *CreateCoeff(int nLen, int nNewLen, BOOL bShrink) 

{ 

    int nSum = 0, nSum2; 

    float *pRes = new float[2 * nLen]; 

    float *pCoeff = pRes; 

    float fNorm = (bShrink)? (float)nNewLen / nLen : 1; 

    int      nDenom = (bShrink)? nLen : nNewLen; 

    

    ::ZeroMemory(pRes, 2 * nLen * sizeof(float)); 

    for(int i = 0; i < nLen; i++, pCoeff += 2) 

    { 

        nSum2 = nSum + nNewLen; 

        if(nSum2 > nLen) 

        { 

            *pCoeff = (float)(nLen - nSum) / nDenom; 

            pCoeff[1] = (float)(nSum2 - nLen) / nDenom; 

            nSum2 -= nLen; 

        } 

        else 

        { 

            *pCoeff = fNorm; 

            if(nSum2 == nLen) 

            { 

                pCoeff[1] = -1; 

                nSum2 = 0; 

            } 

        } 

        nSum = nSum2; 

    } 

    

    return pRes; 

} 



///////////////////////////////////////////////// 



#define F_DELTA        0.0001f 



void ShrinkData(BYTE *pInBuff, WORD wWidth, WORD wHeight, 

                BYTE *pOutBuff, WORD wNewWidth, WORD wNewHeight) 

{ 

    BYTE  *pLine = pInBuff, *pPix; 

    BYTE  *pOutLine = pOutBuff; 

    DWORD dwInLn = (3 * wWidth + 3) & ~3; 

    DWORD dwOutLn = (3 * wNewWidth + 3) & ~3; 

    int   x, y, i, ii; 

    BOOL  bCrossRow, bCrossCol; 

    

    float *pRowCoeff = CreateCoeff(wWidth, 

        wNewWidth, 

        TRUE); 

    float *pColCoeff = CreateCoeff(wHeight, 

        wNewHeight, 

        TRUE); 

    

    float fTmp, *pXCoeff, *pYCoeff = pColCoeff; 

    DWORD dwBuffLn = 3 * wNewWidth * sizeof(float); 

    float *fBuff = new float[6 * wNewWidth]; 

    

    float *fCurrLn = fBuff, *fCurrPix, *fNextLn 

        = fBuff + 3 * wNewWidth, *fNextPix; 

    

    ::ZeroMemory(fBuff, 2 * dwBuffLn); 

    

    y = 0; 

    while(y < wNewHeight) 

    { 

        pPix = pLine; 

        pLine += dwInLn; 

        

        fCurrPix = fCurrLn; 

        fNextPix = fNextLn; 

        

        x = 0; 

        pXCoeff = pRowCoeff; 

        bCrossRow = pYCoeff[1] > F_DELTA; 

        while(x < wNewWidth) 

        { 

            fTmp = *pXCoeff * *pYCoeff; 

            

            for(i = 0; i < 3; i++) 

                fCurrPix[i] += fTmp * pPix[i]; 

            

            bCrossCol = pXCoeff[1] > F_DELTA; 

            

            if(bCrossCol) 

            { 

                fTmp = pXCoeff[1] * *pYCoeff; 

                for(i = 0, ii = 3; i < 3; i++, ii++) 

                    fCurrPix[ii] += fTmp * pPix[i]; 

            } 

            

            if(bCrossRow) 

            { 

                fTmp = *pXCoeff * pYCoeff[1]; 

                for(i = 0; i < 3; i++) 

                    fNextPix[i] += fTmp * pPix[i]; 

                if(bCrossCol) 

                { 

                    fTmp = pXCoeff[1] * pYCoeff[1]; 

                    for(i = 0, ii = 3; i < 3; i++, ii++) 

                        fNextPix[ii] += fTmp * pPix[i]; 

                } 

            } 

            if(fabs(pXCoeff[1]) > F_DELTA) 

            { 

                x++; 

                fCurrPix += 3; 

                fNextPix += 3; 

            } 

            pXCoeff += 2; 

            pPix += 3; 

        } 

        

        if(fabs(pYCoeff[1]) > F_DELTA) 

        { 

            // set result line 

            fCurrPix = fCurrLn; 

            pPix = pOutLine; 

            for(i = 3 * wNewWidth; i > 0; i--, fCurrPix++, pPix++) 

                *pPix = (BYTE)*fCurrPix; 

            

            // prepare line buffers 

            fCurrPix = fNextLn; 

            fNextLn = fCurrLn; 

            fCurrLn = fCurrPix; 

            ::ZeroMemory(fNextLn, dwBuffLn); 

            

            y++; 

            pOutLine += dwOutLn; 

        } 

        pYCoeff += 2; 

    } 

    

    delete pRowCoeff; 

    delete pColCoeff; 

    delete fBuff; 

} 



///////////////////////////////////////////////// 



void EnlargeData(BYTE *pInBuff, WORD wWidth, WORD wHeight, 

        BYTE *pOutBuff, WORD wNewWidth, WORD wNewHeight) 

{ 

    BYTE  *pLine = pInBuff, *pPix 

        = pLine, *pPixOld, *pUpPix, *pUpPixOld; 

    BYTE  *pOutLine = pOutBuff, *pOutPix; 

    DWORD dwInLn = (3 * wWidth + 3) & ~3; 

    DWORD dwOutLn = (3 * wNewWidth + 3) & ~3; 

    int   x, y, i; 

    BOOL  bCrossRow, bCrossCol; 

    float *pRowCoeff = CreateCoeff(wNewWidth, wWidth, FALSE); 

    float *pColCoeff = CreateCoeff(wNewHeight, wHeight, FALSE); 

    float fTmp, fPtTmp[3], *pXCoeff, *pYCoeff = pColCoeff; 

    

    y = 0; 

    while(y < wHeight) 

    { 

        bCrossRow = pYCoeff[1] > F_DELTA; 

        x = 0; 

        pXCoeff = pRowCoeff; 

        pOutPix = pOutLine; 

        pOutLine += dwOutLn; 

        pUpPix = pLine; 

        if(fabs(pYCoeff[1]) > F_DELTA) 

        { 

            y++; 

            pLine += dwInLn; 

            pPix = pLine; 

        } 

        

        while(x < wWidth) 

        { 

            bCrossCol = pXCoeff[1] > F_DELTA; 

            pUpPixOld = pUpPix; 

            pPixOld = pPix; 

            if(fabs(pXCoeff[1]) > F_DELTA) 

            { 

                x++; 

                pUpPix += 3; 

                pPix += 3; 

            } 

            fTmp = *pXCoeff * *pYCoeff; 

            for(i = 0; i < 3; i++) 

                fPtTmp[i] = fTmp * pUpPixOld[i]; 

            if(bCrossCol) 

            { 

                fTmp = pXCoeff[1] * *pYCoeff; 

                for(i = 0; i < 3; i++) 

                    fPtTmp[i] += fTmp * pUpPix[i]; 

            } 

            if(bCrossRow) 

            { 

                fTmp = *pXCoeff * pYCoeff[1]; 

                for(i = 0; i < 3; i++) 

                    fPtTmp[i] += fTmp * pPixOld[i]; 

                if(bCrossCol) 

                { 

                    fTmp = pXCoeff[1] * pYCoeff[1]; 

                    for(i = 0; i < 3; i++) 

                        fPtTmp[i] += fTmp * pPix[i]; 

                } 

            } 

            for(i = 0; i < 3; i++, pOutPix++) 

                *pOutPix = (BYTE)fPtTmp[i]; 

            pXCoeff += 2; 

        } 

        pYCoeff += 2; 

    } 

    

    delete pRowCoeff; 

    delete pColCoeff; 

} 

// end src 

4.참고

http://codeguru.earthweb.com/bitmap/SmoothBitmapResizing.html


- 2001.08.13 Smile Seo -
"Win32 API" 카테고리의 다른 글
  • 포인터 변수를 검증하기위한 매크로 (0)2007/03/19
  • 프로파일링(Profiling) 기능 활용하기 (0)2007/03/19
  • Bitmap 이미지 resizing 하기 (0)2007/03/19
  • 문자열에서 드라이브,디렉토리,파일이름, 확장자... (0)2007/03/19
  • FTP 로 파일 전송하기 (0)2007/03/15
2007/03/19 09:10 2007/03/19 09:10
Posted by webdizen
No Trackback No Comment

Trackback URL : http://www.webdizen.net/blog/trackback/2697

Leave your greetings.

[로그인][오픈아이디란?]

«Prev  1 ... 551 552 553 554 555 556 557 558 559 ... 3009  Next»

RSS HanRSS
Blog Image
webdizen
이곳은 컴퓨터에 대해 연구하고, 공유하고, 소통하기 위한 연구실입니다. 개인적으로는 OLAP, Data Mining, Semantic Web, Data Modeling에 대해서 연구하고 있습니다.

Categories

전체 (3009)
Webdizen (141)
Life (6)
Diary (16)
Blog (9)
IDEA (2)
Travel (10)
Book (16)
Photo (7)
Movie (8)
Music (14)
Leisure Sports (10)
Funny (6)
Hardware (121)
Software (120)
Windows (5)
Unix & Linux (120)
Installation (5)
Kernel (10)
System (34)
Develop (22)
X-Window (0)
Applicaton (31)
Security (4)
Framework (2)
Hadoop (2)
Programming (804)
Algorithm & Data Structure (1)
Assembly (38)
UNIX/Linux C (95)
C++ (128)
STL (4)
Java (38)
Win32 API (92)
ATL/COM (44)
MFC (151)
.NET (26)
WCF/WPF (4)
C# (28)
Network Programming (17)
Database Programming (12)
OpenGL / DirectX (13)
Multimedia Programming (0)
Game Programming (21)
Parallel Distributed Progra... (0)
Reverse Engineering (0)
Debugging (9)
Python (1)
Ruby (1)
Ruby on Rails (1)
QT (4)
GTK (0)
JSP (0)
PHP (6)
ASP.NET (6)
ASP (2)
Development (28)
Useful Library (2)
Data Modeling (0)
Database (105)
Oracle (4)
MSSQL (41)
MySQL (2)
Data Warehouse (2)
Data Mining (4)
Network (66)
Web (79)
DHTML (4)
XHTML (1)
Javascript (1)
CSS (1)
AJAX (9)
XML (11)
Flex (1)
Silverlight (3)
Security (91)
DoS (1)
Kernel (10)
Scanning (3)
Sniffing (0)
Spoofing (4)
Overflow (28)
Web (11)
Shell (10)
Format String (14)
Window (2)
Embedded (70)
Multimedia (27)
Mobile (14)
Graphic (24)
Management (633)
Knowledge (581)
Hadoop (0)

Notice

  • 메타 블로그 사이트에 등록
  • 새해 맞이 블로그의 변화
  • 블로그 명칭 변경
  • 도메인(www.webdizen.net) 구...
  • TEXTCUBE 1.6.1로 업그레이드...

Tags

  • 백록관
  • Cartesian 곱
  • NULL
  • 웹디즌
  • 홍보동영상
  • 컴퓨터과학
  • Resource
  • 데이터와 형
  • 관찰
  • ODBC
  • 퍼즐
  • w3m
  • Huffman
  • 법과대학
  • Multithreading
  • 아바이 벌꿀 소주
  • 노테르담
  • 윈도우즈 강제 종료
  • PHP
  • 종이

Recent Articles

  • 트위터(Twitter)의 시작!.
  • 청년 리더의 조건.
  • 애플의 타블렛 PC - 아이패드....
  • 미래의 인터페이스 - 육감 기....
  • 기초발성법 동영상 강좌.

Recent Comments

  • 학교 과제물중 쓰레드에 대하....
    장진혁 03/17
  • 관리자만 볼 수 있는 댓글입....
    비밀방문자 03/12
  • 상대방의 이야기를 열심히 경....
    DoNuts 03/03
  • Lots of students know techn....
    Bobbi35Shannon 02/25
  • 좋은글 잘 보고 갑니다..
    Und_hacker 01/08

Recent Trackbacks

  • printf,scanf를 이용한 형식....
    yundream의 프로그래밍 이야기 03/10
  • 파일 열기/저장하기 CFileDialog.
    은마군의 나태블록 2009
  • World IT Show 2008.
    상우 :: Oranzie's BLOG 2008
  • cvs서버 설치하기.
    3인3색 2008
  • 속속 공개되는 Google Chart....
    PHP와 Web 2.0 2007

Archive

  • 2010/02 (1)
  • 2010/01 (6)
  • 2009/12 (5)
  • 2009/09 (3)
  • 2009/08 (1)

Calendar

«   2010/03   »
일 월 화 수 목 금 토
  1 2 3 4 5 6
7 8 9 10 11 12 13
14 15 16 17 18 19 20
21 22 23 24 25 26 27
28 29 30 31      

Bookmarks

    • Administration
      • IIS.NET
      • NTFAQ
      • OS의 모든 것
      • 리눅스포털
    • Database
      • SQL Server Central
      • SQL Team
    • Development
      • .NET Heaven
      • ASP Alliance
      • ASP.NET 2.0
      • Bullog.net
      • C# Corner
      • C++ (C PlusPlus.com)
      • C++ Reference
      • CodeGuru
      • CodePlex
      • DebugLab
      • Dev Articles
      • Devpia
      • DotNet Junkies
      • DotNet Zone
      • Driver Online
      • GOSU.NET
      • HOONS 닷넷
      • Joinc 팀블로그
      • KOSR
      • MSDN Home Page
      • OSR Online
      • Sky.ph - 개발자 커뮤니...
      • TAEYO.NET
      • The Code Project
      • WindowsClient.net
      • 김상욱의 개발자 Side
      • 조인시 위키
    • Human Networks
      • belief21c's e-space
      • I think I can
      • Invisible Rover's Blog :D
      • Rodman®
      • ■ Feel So Good~! ■
      • 까만 나비
      • 나를 가꾸는 시간.
      • 나만의 즐거움~~!
      • 단녕
      • 상우 :: Oranzie's BLOG
    • Information Technology
      • Microsoft TechNet
      • 지디넷코리아 - 글로벌...
    • Security
      • FoundStone
      • milw0rm
      • NewOrder
      • OpenRCE
      • Phrack.org
      • Reverse Engineering b1...
      • Reverse Engineering Team
      • RootKit
      • SecurityFocus
      • SecurityXploded by Nag...
      • Wow Hacker
      • Zone-H
Textcube
Louice Studio Inc.
Powered by Textcube. Original designed by Tistory.