This keyword is not expanded by Git which means it's not replaced with the correct revision value in the releases made using git-based scripts and it's confusing to have lines with unexpanded "$Id$" in the released files. As expanding them with Git is not that simple (it could be done with git archive and export-subst attribute) and there are not many benefits in having them in the first place, just remove all these lines. If nothing else, this will make an eventual transition to Git simpler. Closes #14487. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@74602 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
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);
|
|
}
|
|
|