366 lines
10 KiB
C++
366 lines
10 KiB
C++
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// Name: autocapture.cpp
|
||
|
// Purpose: Implement wxCtrlMaskOut class
|
||
|
// Author: Utensil Candel (UtensilCandel@@gmail.com)
|
||
|
// Licence: wxWindows licence
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
// For compilers that support precompilation, includes "wx/wx.h".
|
||
|
#include "wx/wxprec.h"
|
||
|
|
||
|
#ifdef __BORLANDC__
|
||
|
#pragma hdrstop
|
||
|
#endif
|
||
|
|
||
|
#include "autocapture.h"
|
||
|
|
||
|
#ifndef WX_PRECOMP
|
||
|
#include "wx/wx.h"
|
||
|
#endif
|
||
|
|
||
|
#include "wx/bitmap.h"
|
||
|
#include "wx/filename.h"
|
||
|
#include "wx/notebook.h"
|
||
|
|
||
|
#include <ctime>
|
||
|
|
||
|
#ifdef __WXMAC__
|
||
|
#include <cstring>
|
||
|
#endif
|
||
|
|
||
|
|
||
|
// ----------------------------------------------------------------------------
|
||
|
// AutoCaptureMechanism
|
||
|
// ----------------------------------------------------------------------------
|
||
|
|
||
|
AutoCaptureMechanism::AutoCaptureMechanism(wxNotebook *notebook,
|
||
|
int flag, int margin)
|
||
|
: m_notebook(notebook),
|
||
|
m_flag(flag),
|
||
|
m_margin(margin),
|
||
|
m_grid(NULL)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/* static */
|
||
|
wxString AutoCaptureMechanism::default_dir = wxT("screenshots");
|
||
|
|
||
|
/* static */
|
||
|
wxString AutoCaptureMechanism::GetDefaultDirectoryAbsPath()
|
||
|
{
|
||
|
wxFileName output = wxFileName::DirName(GetDefaultDirectory());
|
||
|
output.MakeAbsolute();
|
||
|
return output.GetFullPath();
|
||
|
}
|
||
|
|
||
|
/* static */
|
||
|
void AutoCaptureMechanism::Delay(int seconds)
|
||
|
{
|
||
|
// TODO: Switch this to use wxTimer.
|
||
|
|
||
|
// Wait for 3 seconds
|
||
|
clock_t start = clock();
|
||
|
while ( clock() - start < (clock_t)CLOCKS_PER_SEC * seconds)
|
||
|
wxYieldIfNeeded();
|
||
|
}
|
||
|
|
||
|
/* static */
|
||
|
bool AutoCaptureMechanism::Capture(wxBitmap* bitmap, int x, int y,
|
||
|
int width, int height, int delay)
|
||
|
{
|
||
|
// Somehow wxScreenDC.Blit() doesn't work under Mac for now. Here is a trick.
|
||
|
#ifdef __WXMAC__
|
||
|
|
||
|
// wxExecute(wxT("screencapture -x ") + tempfile, wxEXEC_SYNC);
|
||
|
|
||
|
char captureCommand[80] =""; // a reasonable max size is 80
|
||
|
sprintf(captureCommand, "sleep %d;%s", delay, "screencapture -x /tmp/wx_screen_capture.png");
|
||
|
system(captureCommand);
|
||
|
|
||
|
if(delay) Delay(delay);
|
||
|
|
||
|
wxBitmap fullscreen;
|
||
|
do
|
||
|
{
|
||
|
fullscreen = wxBitmap(wxT("/tmp/wx_screen_capture.png"), wxBITMAP_TYPE_PNG);
|
||
|
}
|
||
|
while(!fullscreen.IsOk());
|
||
|
|
||
|
*bitmap = fullscreen.GetSubBitmap(wxRect(x, y, width, height));
|
||
|
|
||
|
// to prevent loading the old screenshot next time
|
||
|
system("rm /tmp/wx_screen_capture.png");
|
||
|
|
||
|
return true;
|
||
|
|
||
|
#else // Under other paltforms, take a real screenshot
|
||
|
|
||
|
if(delay) Delay(delay);
|
||
|
|
||
|
// Create a DC for the whole screen area
|
||
|
wxScreenDC dcScreen;
|
||
|
|
||
|
bitmap->Create(width, height);
|
||
|
|
||
|
// Create a memory DC that will be used for actually taking the screenshot
|
||
|
wxMemoryDC memDC;
|
||
|
memDC.SelectObject((*bitmap));
|
||
|
memDC.Clear();
|
||
|
|
||
|
// Blit (in this case copy) the actual screen on the memory DC
|
||
|
// and thus the Bitmap
|
||
|
memDC.Blit( 0, // Copy to this X coordinate
|
||
|
0, // Copy to this Y coordinate
|
||
|
width, // Copy this width
|
||
|
height, // Copy this height
|
||
|
&dcScreen, // From where do we copy?
|
||
|
x, // What's the X offset in the original DC?
|
||
|
y // What's the Y offset in the original DC?
|
||
|
);
|
||
|
|
||
|
// Select the Bitmap out of the memory DC by selecting a new
|
||
|
// uninitialized Bitmap
|
||
|
memDC.SelectObject(wxNullBitmap);
|
||
|
#endif // #ifdef __WXMAC__
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/* static */
|
||
|
bool AutoCaptureMechanism::Capture(wxBitmap* bitmap, wxRect rect, int delay)
|
||
|
{
|
||
|
wxPoint origin = rect.GetPosition();
|
||
|
return Capture(bitmap, origin.x, origin.y, rect.GetWidth(), rect.GetHeight(), delay);
|
||
|
}
|
||
|
|
||
|
/* static */
|
||
|
void AutoCaptureMechanism::Save(wxBitmap* screenshot, const wxString& fileName)
|
||
|
{
|
||
|
// make sure default_dir exists
|
||
|
if (!wxDirExists(default_dir))
|
||
|
wxMkdir(default_dir);
|
||
|
|
||
|
wxFileName fullFileName(default_dir, "appear-" + fileName +
|
||
|
"-" + wxPlatformInfo::Get().GetPortIdShortName() + ".png");
|
||
|
|
||
|
// do not overwrite already existing files with this name
|
||
|
while (fullFileName.FileExists())
|
||
|
fullFileName.SetName(fullFileName.GetName() + "_");
|
||
|
|
||
|
// save the screenshot as a PNG
|
||
|
screenshot->SaveFile(fullFileName.GetFullPath(), wxBITMAP_TYPE_PNG);
|
||
|
}
|
||
|
|
||
|
void AutoCaptureMechanism::CaptureAll()
|
||
|
{
|
||
|
// start from the first page
|
||
|
m_notebook->SetSelection(0);
|
||
|
wxYield();
|
||
|
|
||
|
for (ControlList::iterator it = m_controlList.begin();
|
||
|
it != m_controlList.end();
|
||
|
++it)
|
||
|
{
|
||
|
Control &ctrl = *it;
|
||
|
|
||
|
if (ctrl.flag == AJ_TurnPage) // Turn to next page
|
||
|
{
|
||
|
m_notebook->SetSelection(m_notebook->GetSelection() + 1);
|
||
|
wxYield();
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// create the screenshot
|
||
|
wxBitmap screenshot(1, 1);
|
||
|
Capture(&screenshot, ctrl);
|
||
|
|
||
|
if(ctrl.flag & AJ_Union)
|
||
|
{
|
||
|
// union screenshots until AJ_UnionEnd
|
||
|
do
|
||
|
{
|
||
|
++it;
|
||
|
it->name = ctrl.name; //preserving the name
|
||
|
wxBitmap screenshot2(1, 1);
|
||
|
Capture(&screenshot2, *it);
|
||
|
wxBitmap combined(1, 1);
|
||
|
Union(&screenshot, &screenshot2, &combined);
|
||
|
screenshot = combined;
|
||
|
}
|
||
|
while(!(it->flag & AJ_UnionEnd));
|
||
|
}
|
||
|
|
||
|
// and save it
|
||
|
Save(&screenshot, ctrl.name);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool AutoCaptureMechanism::Capture(wxBitmap* bitmap, Control& ctrl)
|
||
|
{
|
||
|
// no manual specification for the control name
|
||
|
// or name adjustment is disabled globally
|
||
|
if (ctrl.name == wxT("") || m_flag & AJ_DisableNameAdjust)
|
||
|
{
|
||
|
// Get its name from wxRTTI
|
||
|
ctrl.name = ctrl.ctrl->GetClassInfo()->GetClassName();
|
||
|
}
|
||
|
|
||
|
int choice = wxNO;
|
||
|
|
||
|
wxRect rect = GetRect(ctrl.ctrl, ctrl.flag);
|
||
|
|
||
|
if (ctrl.flag & AJ_Dropdown && !(m_flag & AJ_DisableDropdown))
|
||
|
{
|
||
|
// for drop-down controls we need the help of the user
|
||
|
wxString caption = _("Drop-down screenshot...");
|
||
|
wxString msg =
|
||
|
wxString::Format(_("Do you wish to capture the drop-down list of '%s' ?\n\n If YES, please drop down the list of '%s' in 5 seconds after closing this message box.\n If NO, the screenshot for this control won't contain its drop-down list."),
|
||
|
ctrl.name, ctrl.name);
|
||
|
|
||
|
choice = wxMessageBox(msg, caption, wxYES_NO, m_notebook);
|
||
|
|
||
|
if (choice == wxYES)
|
||
|
{
|
||
|
//A little hint
|
||
|
ctrl.ctrl->SetCursor(wxCursor(wxCURSOR_HAND));
|
||
|
|
||
|
// Do some rect adjust so it can include the dropdown list
|
||
|
// This adjust isn't pretty, but it works fine on all three paltforms.
|
||
|
// Looking forward to a better solution
|
||
|
int h = rect.GetHeight();
|
||
|
rect.SetHeight(h * 4);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// cut off "wx" and change the name into lowercase.
|
||
|
// e.g. wxButton will have a name of "button" at the end
|
||
|
ctrl.name.StartsWith(wxT("wx"), &(ctrl.name));
|
||
|
ctrl.name.MakeLower();
|
||
|
|
||
|
// take the screenshot
|
||
|
Capture(bitmap, rect, (choice == wxYES)?5:0);
|
||
|
|
||
|
if (choice == wxYES) ctrl.ctrl->SetCursor(wxNullCursor);
|
||
|
|
||
|
if (ctrl.flag & AJ_RegionAdjust)
|
||
|
PutBack(ctrl.ctrl);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/* static */
|
||
|
bool AutoCaptureMechanism::Union(wxBitmap* top, wxBitmap* bottom, wxBitmap* result)
|
||
|
{
|
||
|
int w1, w2, h1, h2, w, h;
|
||
|
w1 = top->GetWidth();
|
||
|
w2 = bottom->GetWidth();
|
||
|
h1 = top->GetHeight();
|
||
|
h2 = bottom->GetHeight();
|
||
|
|
||
|
const int gap_between = 20;
|
||
|
|
||
|
w = (w1 >= w2) ? w1 : w2;
|
||
|
h = h1 + h2 + gap_between;
|
||
|
|
||
|
result->Create(w, h);
|
||
|
|
||
|
wxMemoryDC dstDC;
|
||
|
dstDC.SelectObject((*result));
|
||
|
|
||
|
dstDC.SetBrush(*wxWHITE_BRUSH);
|
||
|
dstDC.Clear();
|
||
|
dstDC.DrawBitmap((*top), 0, 0);
|
||
|
dstDC.DrawBitmap((*bottom), 0, h1 + gap_between);
|
||
|
|
||
|
dstDC.SelectObject(wxNullBitmap);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
wxRect AutoCaptureMechanism::GetRect(wxWindow* ctrl, int flag)
|
||
|
{
|
||
|
if( (!(m_flag & AJ_DisableRegionAdjust) && (flag & AJ_RegionAdjust))
|
||
|
|| (m_flag & AJ_AlwaysRegionAdjust) )
|
||
|
{
|
||
|
wxWindow * parent = ctrl->GetParent();
|
||
|
wxSizer * sizer = parent->GetSizer();
|
||
|
|
||
|
//The assertion won't fail if controls are still managed by wxSizer, and it's unlikely to
|
||
|
//change in the future.
|
||
|
wxASSERT_MSG(sizer,
|
||
|
"The GUI that AutoCaptureMechanism working with doesn't manage controls with wxSizer");
|
||
|
|
||
|
sizer->Detach(ctrl);
|
||
|
|
||
|
/*
|
||
|
+---------+-----------+---------+
|
||
|
| 0 | label | 1 |
|
||
|
+---------+-----------+---------+
|
||
|
| label | ctrl | label |
|
||
|
+---------+-----------+---------+
|
||
|
| 2 | label | 3 |
|
||
|
+---------+-----------+---------+
|
||
|
*/
|
||
|
|
||
|
m_grid = new wxFlexGridSizer(3, 3, m_margin, m_margin);
|
||
|
|
||
|
wxStaticText* l[4];
|
||
|
|
||
|
for (int i = 0; i < 4; ++i)
|
||
|
l[i] = new wxStaticText(parent, wxID_ANY, wxT(" "));
|
||
|
|
||
|
m_grid->Add(l[0]);
|
||
|
m_grid->Add(new wxStaticText(parent, wxID_ANY, wxT(" ")));
|
||
|
m_grid->Add(l[1]);
|
||
|
m_grid->Add(new wxStaticText(parent, wxID_ANY, wxT(" ")));
|
||
|
m_grid->Add(ctrl, 1, wxEXPAND);
|
||
|
m_grid->Add(new wxStaticText(parent, wxID_ANY, wxT(" ")));
|
||
|
m_grid->Add(l[2]);
|
||
|
m_grid->Add(new wxStaticText(parent, wxID_ANY, wxT(" ")));
|
||
|
m_grid->Add(l[3]);
|
||
|
|
||
|
sizer->Add(m_grid);
|
||
|
parent->SetSizer(sizer);
|
||
|
parent->Layout();
|
||
|
|
||
|
parent->Refresh();
|
||
|
wxYield();
|
||
|
|
||
|
return wxRect(l[0]->GetScreenRect().GetBottomRight(),
|
||
|
l[3]->GetScreenRect().GetTopLeft());
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return ctrl->GetScreenRect().Inflate(m_margin);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void AutoCaptureMechanism::PutBack(wxWindow * ctrl)
|
||
|
{
|
||
|
if(!m_grid) return;
|
||
|
|
||
|
m_grid->Detach(ctrl);
|
||
|
|
||
|
wxSizerItemList children = m_grid->GetChildren();
|
||
|
|
||
|
for (wxSizerItemList::iterator it = children.begin(); it != children.end(); ++it)
|
||
|
{
|
||
|
wxSizerItem* item = *it;
|
||
|
if (item->IsWindow()) delete (*it)->GetWindow();
|
||
|
}
|
||
|
|
||
|
wxSizer * sizer = ctrl->GetParent()->GetSizer();
|
||
|
|
||
|
//The assertion won't fail if controls are still managed by wxSizer, and it's unlikely to
|
||
|
//change in the future.
|
||
|
wxASSERT_MSG(sizer,
|
||
|
"The GUI that AutoCaptureMechanism working with doesn't manage controls with wxSizer");
|
||
|
|
||
|
sizer->Detach(m_grid);
|
||
|
delete m_grid;
|
||
|
m_grid = NULL;
|
||
|
|
||
|
sizer->Add(ctrl);
|
||
|
}
|
||
|
|