In Bitmap speichern...



  • Ich habe eine dialog. basierende Anwendung erstellt, welche etwas auf den Bildschirm zeichnet und auch ausdruckt. Nun soll es aber auch möglich sein, das ganze in eine Datei zu speichern, am besten so dass es nicht mehr veränderbar ist. Super wär also pdf, fällt aber wegen der Lizens raus. Nun bin ich auf die Idee gekommen, alles einfach in einem Bitmap zu speichern. Aber wie kann ich das am geschicktesten anstellen?? Oder hat noch jemand ne andere Idee??!
    Big thx!!



  • /*
    WriteBitmap - Speichern einer 24 Bit True-Color Bitmap aus einem Bitmap-Handle.
    szFile		= E:Dateiname
    hbitmap		= E:Handle auf ein Speicherkontext oder eine Bitmap
    memdc		= E:ein Speicherkontext in dem sich eine Kopie der Bitmap befindet
    R: Flag ob erfolgreich oder nicht...
    */
    BOOL WriteBitmap( LPTSTR szFile, HBITMAP hbitmap, HDC memdc)
    {
      BITMAP  bmp;
      //Informationen über die übergebene Bitmap sammeln, bmp ist der Zielpuffer der Infos
      if(GetObject(hbitmap, sizeof(BITMAP), &bmp))   
      {     
        BITMAPINFOHEADER BmpInfoHdr;  //Struktur für Bitmap-Infoheader 
        BITMAPFILEHEADER BmpFileHdr;  //Struktur für Bitmap-Dateiheader
        //jetzt werden die gesammlten Infos in bmp ausgewertet und in die Strukts umgeschrieben
        BmpInfoHdr.biSize = sizeof(BITMAPINFOHEADER);     
        BmpInfoHdr.biWidth = bmp.bmWidth;     
        BmpInfoHdr.biHeight = bmp.bmHeight;     
        BmpInfoHdr.biPlanes = bmp.bmPlanes;     
        BmpInfoHdr.biBitCount = 24;     
        BmpInfoHdr.biCompression	= BI_RGB;     
        BmpInfoHdr.biSizeImage		= bmp.bmWidth*bmp.bmHeight*3;     
        BmpFileHdr.bfType		= 0x4d42;     
        BmpFileHdr.bfReserved1		= 0;     
        BmpFileHdr.bfReserved2		= 0;     
        BmpFileHdr.bfOffBits = sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER);     
        BmpFileHdr.bfSize = BmpFileHdr.bfOffBits+BmpInfoHdr.biSizeImage;  
    
        //Heap-Speicher für die ins DIB-Format zu konvertierende Bitmap organisieren
        bmp.bmBits = (void*)GlobalAlloc(GMEM_FIXED, BmpInfoHdr.biSizeImage); 
    
        //Bitmap ins DIB-Format umwandeln, wenn erfolgreich wird die DIB geschrieben
        if(GetDIBits(memdc, hbitmap, 0, BmpInfoHdr.biHeight, bmp.bmBits, 
            (BITMAPINFO*)&BmpInfoHdr, DIB_RGB_COLORS) == BmpInfoHdr.biHeight)      
        {     
          //Datei erzeugen über Win-API (ist performanter als fstream-Funktionen)
    	HANDLE hFile = CreateFile(szFile, GENERIC_READ | GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,NULL); 
          //Datei erzeugen erfolgreich?
          if(hFile != INVALID_HANDLE_VALUE)  {          
            //hier wird die Anzahl der geschriebenen Bytes abgelegt, aber nicht ausgewertet
            DWORD dwTmp;  
            //Bitmap-Dateiheader schreiben
            WriteFile(hFile, &BmpFileHdr, sizeof(BITMAPFILEHEADER), &dwTmp, NULL);           
            //Bitmap-Infoheader schreiben
            WriteFile(hFile, &BmpInfoHdr, sizeof(BITMAPINFOHEADER), &dwTmp, NULL);           
            //Bitmap-Pixeldaten schreiben
            WriteFile(hFile, bmp.bmBits,  BmpInfoHdr.biSizeImage,   &dwTmp, NULL);         
            }   
          //Datei schliessen
          CloseHandle(hFile);      
        }     
        //Heapspeicher freigeben
        GlobalFree(bmp.bmBits); 
        return TRUE;
      } 
      //Nicht erfolgreich gewesen, leider leider ...
      return FALSE;
    }
    

    gruss



  • Wow, danke!! Werds gleich mal ausprobieren!
    Gruß Christian



  • Sorry, aber leider blick ich noch nicht ganz durch: Muß ich jetzt ein neues Bitmap per CBitmap ertsllen und das was ich auf den Bildschirm zeichne da hinein zeichnen und das ganze dann der Methode übergeben??? Bin noch ziemlich neu in dem Geschäft, würd mich freuen wenn nochmal jemand weiterhelfen könnte...



  • Also da Du schon die Bitmap auf Schirm zeichnest und auch ausdruckst brauchst diese Bitmap auch nur noch an die Funktion übergeben. Eine neue erzeugen ist nicht nötig.

    BOOL WriteBitmap( LPTSTR szFile, HBITMAP hbitmap, HDC memdc);

    hbitmap ist das Handle auf Deine existierende Bitmap (jene, welche Du zum Drucken und zum Zeichnen auf dem Schirm verwendest).
    Sicherlich verwendest Du bereits ein CBitmap-Objekt, das Handle davon bekommst Du über BMPObjekt.operator HBITMAP( ) geliefert.
    memdc ist das Handle auf den aktuellen Gerätekontext, das bekommst von Deiner Dialoginstanz über instanz.m_hDC oder auch über this->m_hDC.

    Also sind diese Informationen und der Zieldateiname nur an die Funktion zu übergeben. 😉

    Gruss



  • Also zum zeichnen auf dem Bildschirm und zum drucken verwende ich kein Bitmap, ich mach da folgendes (auf bildschrim):

    CPaintDC dc(this);
    dc.MoveTo(...);
    dc.LineTo(...);
    

    usw.
    jetzt hab ich folgendes versucht:

    CBitmap bitmap;
    bitmap.CreateBitmap( 1000, 900, 1,24, NULL ); 
    CDC MemDC; 
    CDC* pDC = GetDC(); 
    MemDC.CreateCompatibleDC( pDC ); 
    MemDC.SelectObject(bitmap); 
    HBITMAP hBitmap=bitmap;
    ZeichneEtwas(&MemDC);
    WriteBitmap("c:\test", hBitmap, MemDC);
    

    Die ZeichneEtwas Methode übernimmt den Gerätekontext, also entwerder CPaint dc zum auf den Bildschirm zeichnen oder CDC PrintDc zum drucken. Jetzt wollte ich also einfach in das Bitmap zeichnen. Die Funktion liefert mir ja auch TRUE zurück, nur leider finde ich niergend die Datei "test"?????!!!!
    Bin am verzweifeln!!!!



  • verwende statt:

    WriteBitmap("c:\test", hBitmap, MemDC);

    lieber mal

    WriteBitmap("c:**\**test", hBitmap, MemDC); //fett markiert

    Das Zeichen '\' stellt eine Escape-Steuersequenz dar, will man das Zeichen selbst haben muss es "maskiert" werden, damit der Kompiler kein \t draus macht, was in diesem falle ein Tabulatorzeichen wäre!

    😉

    Gruss



  • Danke, aber leider funktioniert das auch nicht, er erstellt keine Datei!



  • ich komm da echt nicht weiter :(((
    Kannst du mir mal ein bißchen code schreiben, der ein Bitmap erstellt,
    eine Linie rein malt und dann das Bitmap mit der Funktion in eine Datei speichert??
    Das wär echt super!!
    Danke im Vorraus,
    Gruß Christian



  • Also du verwendest bis jetzt folgenden Code:

    CBitmap bitmap; 
    bitmap.CreateBitmap( 1000, 900, 1,24, NULL );  
    CDC MemDC;  
    CDC* pDC = GetDC();  
    MemDC.CreateCompatibleDC( pDC );  
    MemDC.SelectObject(bitmap);  
    HBITMAP hBitmap=bitmap; 
    ZeichneEtwas(&MemDC); 
    WriteBitmap("c:\test", hBitmap, MemDC);
    

    Da fehlt nur noch ein wenig.

    CBitmap bitmap; 
    bitmap.CreateBitmap( 1000, 900, 1,24, NULL );  
    CDC MemDC;  
    CDC* pDC = GetDC();  
    MemDC.CreateCompatibleDC( pDC );  
    MemDC.SelectObject(bitmap);  
    ZeichneEtwas(&MemDC); 
    pDC->SelectObject(bitmap);
    pDC->BitBlt(0,0,1000,900, &MemDC, 0,0, SRCCOPY); //Inhalt des MemDC in Bitmap blitten
    WriteBitmap("c:\\test.bmp", Bitmap.operator HBITMAP(), pDC->m_hDC);
    

    Sollte das nicht gehen wäre es gut mal ins Projekt zu sehen, kannst Du eventl.
    das Projekt zipped irgendwohin stellen? (ohne Debug Verzeichnis!)

    Gruss



  • Leider funktoniert es immer noch nicht. Es passiert einfach gar nichts, keine Datei wird erstellt 😞 Das Projekt kann ich leider nicht zeigen, ist für eine Firma. Ich habe mal eine komplett neue Klasse erstellt, mit einem "Save" Button und will nur eine Linie in das Bitmap zeichnen:

    void CZeichnen::OnSave() 
    {
    
    	CBitmap bitmap;  
    	bitmap.CreateBitmap( 1000, 900, 1,24, NULL );   
    	CDC MemDC;   
    	CDC* pDC = GetDC();   
    	MemDC.CreateCompatibleDC( pDC );   
    	MemDC.SelectObject(bitmap);   
    	MemDC.MoveTo(10,10);
    	MemDC.LineTo(100,100);
    	pDC->SelectObject(bitmap); 
    	pDC->BitBlt(0,0,1000,900, &MemDC, 0,0, SRCCOPY); 
    	WriteBitmap("c:\\test.bmp", bitmap.operator HBITMAP(), pDC->m_hDC);
    
    }
    

    Das müßte doch auf jeden Fall funktionieren, unabhängig vom rest des Codes, oder???



  • Ich kann jetzt erstmal nur noch empfehlen durch die WriteBitmap mit dem Debugger zu gehen und mit die Stelle wo er mit welchem Ergebnis aussteigt...

    Wenn man nicht reinsehen kann ist es schwer ;), aber OK - muss man halt durch

    Also poste mir mal das Zeug, dann sehen wir weiter.



  • Hab es debugt, er führt diese Anweisung nicht aus:

    if(GetDIBits(memdc, hbitmap, 0, BmpInfoHdr.biHeight, bmp.bmBits,  
            (BITMAPINFO*)&BmpInfoHdr, DIB_RGB_COLORS) == BmpInfoHdr.biHeight)
    

    Hier die Ergebnisse:
    - BmpFileHdr {...}
    bfType 19778
    bfSize 2700054
    bfReserved1 0
    bfReserved2 0
    bfOffBits 54
    - BmpInfoHdr {...}
    biSize 40
    biWidth 1000
    biHeight 900
    biPlanes 1
    biBitCount 24
    biCompression 0
    biSizeImage 2700000
    biXPelsPerMeter -858993460
    biYPelsPerMeter -858993460
    biClrUsed 3435973836
    biClrImportant 3435973836
    + this 0x00120648 {CZeichnen hWnd=0x0b8c0720}
    - szFile 0x00602268 "d:\test.bmp"
    100 'd'
    - hbitmap 0x4b050683
    unused CXX0030: Fehler: Ausdruck kann nicht ausgewertet werden
    - memdc 0xa6010215
    unused CXX0030: Fehler: Ausdruck kann nicht ausgewertet werden
    - bmp {...}
    bmType 0
    bmWidth 1000
    bmHeight 900
    bmWidthBytes 3000
    bmPlanes 1
    bmBitsPixel 24
    bmBits 0x01610020

    Hoffe du kannst damit etwas anfangen...
    Nochmal vielen Dank für deine Hilfe!



  • Jetzt hab ich mal die if-Schleife auskommentiert (also NUR das if und == BmpInfoHdr.biHeight), dann erhalte ich eine Datei, allerdings ist sie komplett schwarz!



  • Ich mach nacher (wird leider erst so in 20 Minuten werden) mal ein MFC-Projekt
    wo eine Bitmap erzeugt (male irgendwas rein) und dann gespeichert wird. Das Projekt stell ich dann auf ne HP wo Du es ziehen kannst - also nur die Sourcen (keine EXE). Nimm das als Referenzprojekt und wenn dann noch Fragen sind meld dich halt wieder...



  • Das wär echt Klasse, vielen Dank schon mal im vorraus!!! Macht nix wenn es etwas länger geht, arbeite eh erst morgen dran weiter!
    Viele Grüße
    Christian



  • So nun ist das Projekt fertig, aber dummerweise hab ich meine URL-PWD nicht mit in der Stadt wo ich gerade arbeite. Ich hab ein MFC-SDI Projekt erzeugt und nur die OnPaint gefüllt, bei mir wird sauber eine Bitmap erzeugt. Den Quellcode der WriteBitmap habe ich nicht verändert.

    Also vielleicht hilft die auch der Code der OnPaint:

    void CChildView::OnPaint() 
    {
      CPaintDC dc(this); // Gerätekontext zum Zeichnen
    
      //na dann, malen wir mal irgendwas was ins HDC
      dc.Rectangle(rand()%50,rand()%50,600, 600);
      RECT r;
      CBrush brush;
      r.left=rand()%50;
      r.top=rand()%50;
      r.bottom=rand()%800;
      r.right=rand()%800;
      brush.CreateSolidBrush(RGB(255,255,0));
      //ein gelbes Rechteck
      dc.FillRect(&r, &brush);
      //auch noch einen Text
      dc.TextOut(100,100,"Das ist ein Text in einer Bitmap!");
    
      //jetzt das ganze als Bitmap abspeichern
      CBitmap bmp; //Bitmap-Objekt
      CDC memdc;	//Speicherkontext
      RECT cr;	//für die Grösse des Clients
    
      memdc.CreateCompatibleDC((CDC*)&dc);
      GetClientRect(&cr); //Ausdehnung des Clientsbereichs ermitteln
      //passende Bitmap in der Grösse des Clients anlegen
      bmp.CreateCompatibleBitmap((CDC*)&dc, cr.right,cr.bottom); 
    
      //Speicherkontext mit der Bitmap verbinden
      memdc.SelectObject(bmp);
      //In den Speicherkontext (eigentlich in die Bitmap!) den Inhalt des dc kopieren
      memdc.BitBlt(0,0,cr.right,cr.bottom,(CDC*)&dc,0,0,SRCCOPY);
    
      //Jetzt die Bitmap speichern
      WriteBitmap("c:\\testbitmap.bmp",bmp.operator HBITMAP(), memdc.m_hDC);
    
      // Rufen Sie nicht CWnd::OnPaint() für Nachrichten zum Zeichnen auf
    }
    

    Bei mir funktioniert das wie es soll. Wenn man das Fenster scaliert, wird immer
    eine neue Zeichnung erstellt und in die Bitmap-Datei geschrieben.
    Sollte für ne Demo auch reichen...

    Schreib wieder bei Fragen.
    Gruss René



  • Vielen Dank, werd es gleich morgen ausprobieren, sobald ich auf arbeit bin!
    Werd dann das Ergebnis hier posten!

    Schönen Abend noch und viele Grüße
    Christian



  • Jetzt habe ich mal kurz eine dialogf. Anwendung erstellt mit einem Button "Speichern" in OnClicked() habe ich den Code kopiert. Jetzt wird zwar ein Bitmap erstellt aber in diesem ist ein Screenshot von meinem Dialog!! Liegt das daran dass ich es nicht in OnPaint() oder wie kann denn das kommen??



  • Also gut, dann bis morgen und auch schönen Feierabend!

    Gruss René



  • Guten morgen 🙂
    Das funktioniert jetzt alles echt wunderbar, nur ein Problem besteht noch: Er zeichnte mir das was er in die Datei zeichnet auch auf den Bildschirm!! Dadurch überschreibt er mir meine Grafik! Erst wenn ich das Fenster minimiere und wieder maximiere ist wieder alles beim alten! Hast du dafür eine Lösungsidee??


Anmelden zum Antworten