Resolve a TODO and fix another ancient bug, #386. Print Screen is now handled
by BWindow, no longer by the app_server. This should stop the "screen freeze" effect. This adds a dependency on libpng.so and libz.so to libbe.so. The same dependencies and the PNGDump code added here can be removed from the app_server. I am just waiting for a code review of this before doing that. This implementation still does not give the client a chance to handle it differently. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@25269 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
2428097297
commit
6aede71c0c
@ -35,6 +35,7 @@ SharedLibrary libbe.so :
|
||||
libagg.a
|
||||
|
||||
libroot.so # make sure it links against our libroot.so
|
||||
libpng.so libz.so # For dumping the screen to PNG in the Interface Kit
|
||||
$(TARGET_LIBSTDC++)
|
||||
;
|
||||
|
||||
|
@ -26,7 +26,7 @@ if ! $(TARGET_PLATFORM_HAIKU_COMPATIBLE) {
|
||||
SetSubDirSupportedPlatforms haiku libbe_test ;
|
||||
|
||||
UsePrivateHeaders app input interface shared tracker ;
|
||||
UseLibraryHeaders icon ;
|
||||
UseLibraryHeaders icon png zlib ;
|
||||
|
||||
|
||||
SEARCH_SOURCE += [ FDirName $(SUBDIR) textview_support ] ;
|
||||
@ -78,6 +78,7 @@ MergeObject <libbe>interface_kit.o :
|
||||
PictureButton.cpp
|
||||
PictureDataWriter.cpp
|
||||
PicturePlayer.cpp
|
||||
PNGDump.cpp
|
||||
Point.cpp
|
||||
Polygon.cpp
|
||||
PopUpMenu.cpp
|
||||
|
125
src/kits/interface/PNGDump.cpp
Normal file
125
src/kits/interface/PNGDump.cpp
Normal file
@ -0,0 +1,125 @@
|
||||
/*
|
||||
* Copyright 2001-2006, Haiku.
|
||||
* Distributed under the terms of the MIT License.
|
||||
*
|
||||
* Authors:
|
||||
* DarkWyrm <bpmagic@columbus.rr.com>
|
||||
* Stephan Aßmus <superstippi@gmx.de>
|
||||
*/
|
||||
|
||||
/** Function for saving a generic framebuffer to a PNG file */
|
||||
|
||||
#include "PNGDump.h"
|
||||
|
||||
#include <NodeInfo.h>
|
||||
#include <Rect.h>
|
||||
|
||||
#include <png.h>
|
||||
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
|
||||
#define TRACE_PNGDUMP
|
||||
#ifdef TRACE_PNGDUMP
|
||||
# define TRACE(x) printf x
|
||||
#else
|
||||
# define TRACE(x) ;
|
||||
#endif
|
||||
|
||||
|
||||
status_t
|
||||
SaveToPNG(const char* filename, const BRect& bounds, color_space space,
|
||||
const void* bits, int32 bitsLength, int32 bytesPerRow)
|
||||
{
|
||||
int32 width = bounds.IntegerWidth() + 1;
|
||||
int32 height = bounds.IntegerHeight() + 1;
|
||||
|
||||
TRACE(("SaveToPNG: %s (%ldx%ld)\n", filename, width, height));
|
||||
|
||||
FILE *file = fopen(filename, "wb");
|
||||
if (file == NULL) {
|
||||
TRACE(("Couldn't open file: %s\n", strerror(errno)));
|
||||
return errno;
|
||||
}
|
||||
|
||||
png_structp png = png_create_write_struct(PNG_LIBPNG_VER_STRING,
|
||||
NULL, NULL, NULL);
|
||||
if (png == NULL) {
|
||||
TRACE(("Couldn't create write struct\n"));
|
||||
fclose(file);
|
||||
return B_NO_MEMORY;
|
||||
}
|
||||
|
||||
png_infop info = png_create_info_struct(png);
|
||||
if (info == NULL) {
|
||||
TRACE(("Couldn't create info struct\n"));
|
||||
png_destroy_write_struct(&png, NULL);
|
||||
fclose(file);
|
||||
return B_NO_MEMORY;
|
||||
}
|
||||
|
||||
if (setjmp(png->jmpbuf)) {
|
||||
png_destroy_write_struct(&png, NULL);
|
||||
fclose(file);
|
||||
return B_ERROR;
|
||||
}
|
||||
|
||||
png_init_io(png, file);
|
||||
png_set_bgr(png);
|
||||
|
||||
// TODO: support other color spaces if needed
|
||||
|
||||
switch (space) {
|
||||
case B_RGB32:
|
||||
case B_RGBA32:
|
||||
{
|
||||
// create file without alpha channel
|
||||
png_set_IHDR(png, info, width, height, 8, PNG_COLOR_TYPE_RGB,
|
||||
PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_DEFAULT,
|
||||
PNG_FILTER_TYPE_DEFAULT);
|
||||
png_write_info(png, info);
|
||||
|
||||
// convert from 32 bit RGB to 24 bit RGB while saving
|
||||
png_byte* src = (png_byte*)bits;
|
||||
int srcRowBytes = width * 4;
|
||||
int dstRowBytes = width * 3;
|
||||
int srcRowOffset = bytesPerRow - srcRowBytes;
|
||||
png_byte tempRow[dstRowBytes];
|
||||
for (int row = 0; row < height; row++) {
|
||||
for (int i = 0; i < dstRowBytes; i += 3, src += 4) {
|
||||
tempRow[i] = src[0];
|
||||
tempRow[i + 1] = src[1];
|
||||
tempRow[i + 2] = src[2];
|
||||
}
|
||||
src += srcRowOffset;
|
||||
png_write_row(png, tempRow);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
default:
|
||||
{
|
||||
TRACE(("Unsupported color space\n"));
|
||||
png_destroy_write_struct(&png, NULL);
|
||||
fclose(file);
|
||||
return B_ERROR;
|
||||
}
|
||||
}
|
||||
|
||||
png_write_end(png, info);
|
||||
png_destroy_write_struct(&png, NULL);
|
||||
|
||||
fclose(file);
|
||||
|
||||
// Set the file type manually, so that it doesn't have to be
|
||||
// picked up by the registrar or Tracker, first
|
||||
BNode node(filename);
|
||||
BNodeInfo nodeInfo(&node);
|
||||
if (nodeInfo.InitCheck() == B_OK)
|
||||
nodeInfo.SetType("image/png");
|
||||
|
||||
return B_OK;
|
||||
}
|
20
src/kits/interface/PNGDump.h
Normal file
20
src/kits/interface/PNGDump.h
Normal file
@ -0,0 +1,20 @@
|
||||
/*
|
||||
* Copyright 2001-2005, Haiku.
|
||||
* Distributed under the terms of the MIT License.
|
||||
*
|
||||
* Authors:
|
||||
* DarkWyrm <bpmagic@columbus.rr.com>
|
||||
*/
|
||||
#ifndef PNGDUMP_H
|
||||
#define PNGDUMP_H
|
||||
|
||||
|
||||
#include <GraphicsDefs.h>
|
||||
|
||||
class BRect;
|
||||
|
||||
|
||||
status_t SaveToPNG(const char* filename, const BRect& bounds, color_space space,
|
||||
const void* bits, int32 bitsLength, int32 bytesPerRow);
|
||||
|
||||
#endif
|
@ -17,6 +17,7 @@
|
||||
|
||||
#include <Application.h>
|
||||
#include <Autolock.h>
|
||||
#include <Bitmap.h>
|
||||
#include <Button.h>
|
||||
#include <MenuBar.h>
|
||||
#include <MenuItem.h>
|
||||
@ -41,6 +42,8 @@
|
||||
#include <tracker_private.h>
|
||||
#include <WindowPrivate.h>
|
||||
|
||||
#include "PNGDump.h"
|
||||
|
||||
|
||||
//#define DEBUG_WIN
|
||||
#ifdef DEBUG_WIN
|
||||
@ -3239,6 +3242,39 @@ BWindow::_HandleKeyDown(BMessage* event)
|
||||
PostMessage(&message);
|
||||
return true;
|
||||
}
|
||||
|
||||
if (key == B_FUNCTION_KEY) {
|
||||
// Check for Print Screen
|
||||
int32 rawKey;
|
||||
if (event->FindInt32("key", &rawKey) == B_OK && rawKey == B_PRINT_KEY) {
|
||||
// Get filename
|
||||
char filename[128];
|
||||
BEntry entry;
|
||||
|
||||
int32 index = 1;
|
||||
do {
|
||||
// TODO: Use find_directory(B_USER_DIRECTORY, ...)
|
||||
sprintf(filename, "/boot/home/screen%ld.png", index++);
|
||||
entry.SetTo(filename);
|
||||
} while(entry.Exists());
|
||||
|
||||
// Get the screen bitmap
|
||||
BScreen screen(this);
|
||||
BBitmap* screenDump;
|
||||
screen.GetBitmap(&screenDump, false);
|
||||
|
||||
// Dump to PNG
|
||||
SaveToPNG(filename, screen.Frame(), screenDump->ColorSpace(),
|
||||
screenDump->Bits(),
|
||||
screenDump->BitsLength(),
|
||||
screenDump->BytesPerRow());
|
||||
|
||||
// Free the bitmap allocated by BScreen.GetBitmap
|
||||
delete screenDump;
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
// Handle shortcuts
|
||||
if ((modifiers & B_COMMAND_KEY) != 0) {
|
||||
|
@ -173,25 +173,6 @@ KeyboardFilter::Filter(BMessage* message, EventTarget** _target,
|
||||
return B_SKIP_MESSAGE;
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: this should be moved client side!
|
||||
// (that's how it is done in BeOS, clients could need this key for
|
||||
// different purposes - also, it's preferrable to let the client
|
||||
// write the dump within his own environment)
|
||||
if (key == 0xe) {
|
||||
// screen dump, PrintScreen
|
||||
char filename[128];
|
||||
BEntry entry;
|
||||
|
||||
int32 index = 1;
|
||||
do {
|
||||
sprintf(filename, "/boot/home/screen%ld.png", index++);
|
||||
entry.SetTo(filename);
|
||||
} while(entry.Exists());
|
||||
|
||||
fDesktop->GetDrawingEngine()->DumpToFile(filename);
|
||||
return B_SKIP_MESSAGE;
|
||||
}
|
||||
}
|
||||
|
||||
if (message->what == B_KEY_DOWN
|
||||
|
Loading…
x
Reference in New Issue
Block a user