1492 lines
43 KiB
C++
1492 lines
43 KiB
C++
// pcbuilderdialogDlg.cpp : implementation file
|
|
//
|
|
|
|
#include "stdafx.h"
|
|
#include "shellapi.h"
|
|
// a fun undef for DOT NET
|
|
#undef DEBUG
|
|
#include "CMakeSetup.h"
|
|
#include "MakeHelp.h"
|
|
#include "PathDialog.h"
|
|
#include "CMakeSetupDialog.h"
|
|
#include "CMakeCommandLineInfo.h"
|
|
#include "../cmListFileCache.h"
|
|
#include "../cmCacheManager.h"
|
|
#include "../cmake.h"
|
|
#include "../cmGlobalGenerator.h"
|
|
#include "../cmDynamicLoader.h"
|
|
#ifdef _DEBUG
|
|
#define new DEBUG_NEW
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CAboutDlg dialog used for App About
|
|
|
|
class CAboutDlg : public CDialog
|
|
{
|
|
public:
|
|
CAboutDlg();
|
|
|
|
// Dialog Data
|
|
//{{AFX_DATA(CAboutDlg)
|
|
enum { IDD = IDD_ABOUTBOX };
|
|
//}}AFX_DATA
|
|
|
|
// ClassWizard generated virtual function overrides
|
|
//{{AFX_VIRTUAL(CAboutDlg)
|
|
protected:
|
|
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support
|
|
//}}AFX_VIRTUAL
|
|
|
|
// Implementation
|
|
protected:
|
|
//{{AFX_MSG(CAboutDlg)
|
|
//}}AFX_MSG
|
|
DECLARE_MESSAGE_MAP()
|
|
};
|
|
|
|
CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD)
|
|
{
|
|
//{{AFX_DATA_INIT(CAboutDlg)
|
|
//}}AFX_DATA_INIT
|
|
}
|
|
|
|
void CAboutDlg::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CAboutDlg)
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
|
|
//{{AFX_MSG_MAP(CAboutDlg)
|
|
// No message handlers
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP();
|
|
|
|
|
|
void MFCMessageCallback(const char* m, const char* title, bool& nomore, void*)
|
|
{
|
|
std::string message = m;
|
|
message += "\n\n(Press Cancel to suppress any further messages.)";
|
|
if(::MessageBox(0, message.c_str(), title,
|
|
MB_OKCANCEL|MB_TASKMODAL) == IDCANCEL)
|
|
{
|
|
nomore = true;
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CMakeSetupDialog dialog
|
|
void updateProgress(const char *msg, float prog, void *cd)
|
|
{
|
|
char tmp[1024];
|
|
if (prog >= 0)
|
|
{
|
|
sprintf(tmp,"%s %i%%",msg,(int)(100*prog));
|
|
}
|
|
else
|
|
{
|
|
sprintf(tmp,"%s",msg);
|
|
}
|
|
CMakeSetupDialog *self = (CMakeSetupDialog *)cd;
|
|
self->SetDlgItemText(IDC_PROGRESS, tmp);
|
|
CWnd* cancel = self->GetDlgItem(IDCANCEL);
|
|
//
|
|
// Retrieve and dispatch any waiting messages.
|
|
//
|
|
MSG wmsg;
|
|
while (::PeekMessage (&wmsg, NULL, 0, 0, PM_REMOVE))
|
|
{
|
|
switch(wmsg.message)
|
|
{
|
|
case WM_LBUTTONDOWN:
|
|
case WM_LBUTTONUP:
|
|
case WM_LBUTTONDBLCLK:
|
|
{
|
|
if(wmsg.hwnd == cancel->m_hWnd)
|
|
{
|
|
::DispatchMessage(&wmsg);
|
|
}
|
|
}
|
|
break;
|
|
case WM_COMMAND:
|
|
case WM_SETCURSOR:
|
|
case WM_PAINT:
|
|
::DispatchMessage(&wmsg);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Convert to Win32 path (slashes). This calls the system tools one and then
|
|
// removes the spaces. It is not in system tools because we don't want any
|
|
// generators accidentally use it
|
|
std::string ConvertToWindowsPath(const char* path)
|
|
{
|
|
// Convert to output path.
|
|
// Remove the "" around it (if any) since it's an output path for
|
|
// the shell. If another shell-oriented feature is not designed
|
|
// for a GUI use, then we are in trouble.
|
|
// save the value of the force to unix path option
|
|
bool saveForce = cmSystemTools::GetForceUnixPaths();
|
|
// make sure we get windows paths no matter what for the GUI
|
|
cmSystemTools::SetForceUnixPaths(false);
|
|
std::string s = cmSystemTools::ConvertToOutputPath(path);
|
|
// now restore the force unix path to its previous value
|
|
cmSystemTools::SetForceUnixPaths(saveForce);
|
|
if (s.size())
|
|
{
|
|
std::string::iterator i = s.begin();
|
|
if (*i == '\"')
|
|
{
|
|
s.erase(i, i + 1);
|
|
}
|
|
i = s.begin() + s.length() - 1;
|
|
if (*i == '\"')
|
|
{
|
|
s.erase(i, i + 1);
|
|
}
|
|
}
|
|
return s;
|
|
}
|
|
|
|
CMakeSetupDialog::CMakeSetupDialog(const CMakeCommandLineInfo& cmdInfo,
|
|
CWnd* pParent /*=NULL*/)
|
|
: CDialog(CMakeSetupDialog::IDD, pParent)
|
|
{
|
|
m_GeneratorPicked = false;
|
|
m_Cursor = LoadCursor(NULL, IDC_ARROW);
|
|
m_RunningConfigure = false;
|
|
cmSystemTools::SetRunCommandHideConsole(true);
|
|
cmSystemTools::SetErrorCallback(MFCMessageCallback);
|
|
m_RegistryKey = "Software\\Kitware\\CMakeSetup\\Settings\\StartPath";
|
|
m_CacheEntriesList.m_CMakeSetupDialog = this;
|
|
|
|
m_CMakeInstance = new cmake;
|
|
m_CMakeInstance->SetProgressCallback(updateProgress, (void *)this);
|
|
|
|
//{{AFX_DATA_INIT(CMakeSetupDialog)
|
|
//}}AFX_DATA_INIT
|
|
|
|
// Get the parameters from the command line info
|
|
// If an unknown parameter is found, try to interpret it too, since it
|
|
// is likely to be a file dropped on the shortcut :)
|
|
if (cmdInfo.m_LastUnknownParameter.IsEmpty())
|
|
{
|
|
this->m_WhereSource = cmdInfo.m_WhereSource;
|
|
this->m_WhereBuild = cmdInfo.m_WhereBuild;
|
|
this->m_GeneratorDialog.m_GeneratorChoiceString =
|
|
cmdInfo.m_GeneratorChoiceString;
|
|
this->m_AdvancedValues = cmdInfo.m_AdvancedValues;
|
|
}
|
|
else
|
|
{
|
|
this->m_WhereSource = _T("");
|
|
this->m_WhereBuild = _T("");
|
|
this->m_AdvancedValues = FALSE;
|
|
this->m_GeneratorDialog.m_GeneratorChoiceString =
|
|
cmdInfo.m_GeneratorChoiceString;
|
|
this->ChangeDirectoriesFromFile((LPCTSTR)cmdInfo.m_LastUnknownParameter);
|
|
}
|
|
|
|
// Note that LoadIcon does not require a subsequent DestroyIcon in Win32
|
|
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
|
|
m_BuildPathChanged = false;
|
|
// Find the path to the cmake.exe executable
|
|
char fname[1024];
|
|
::GetModuleFileName(NULL,fname,1023);
|
|
// extract just the path part
|
|
m_PathToExecutable = cmSystemTools::GetProgramPath(fname).c_str();
|
|
// add the cmake.exe to the path
|
|
m_PathToExecutable += "/cmake.exe";
|
|
|
|
m_oldCX = -1;
|
|
m_deltaXRemainder = 0;
|
|
}
|
|
|
|
CMakeSetupDialog::~CMakeSetupDialog()
|
|
{
|
|
delete m_CMakeInstance;
|
|
// clean up globals
|
|
cmListFileCache::GetInstance()->ClearCache();
|
|
cmDynamicLoader::FlushCache();
|
|
}
|
|
|
|
void CMakeSetupDialog::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CMakeSetupDialog)
|
|
DDX_Control(pDX, IDC_AdvancedValues, m_AdvancedValuesControl);
|
|
DDX_Control(pDX, IDC_BROWSE_SOURCE, m_BrowseSource);
|
|
DDX_Control(pDX, IDC_BROWSE_BUILD, m_BrowseBuild);
|
|
DDX_Control(pDX, IDC_DELETE_BUTTON, m_DeleteButton);
|
|
DDX_Control(pDX, IDC_HELP_BUTTON, m_HelpButton);
|
|
DDX_Control(pDX, IDC_OK, m_OKButton);
|
|
DDX_Control(pDX, IDCANCEL, m_CancelButton);
|
|
DDX_CBStringExact(pDX, IDC_WhereSource, m_WhereSource);
|
|
DDX_CBStringExact(pDX, IDC_WhereBuild, m_WhereBuild);
|
|
DDX_Control(pDX, IDC_FRAME, m_ListFrame);
|
|
DDX_Control(pDX, IDC_WhereSource, m_WhereSourceControl);
|
|
DDX_Control(pDX, IDC_WhereBuild, m_WhereBuildControl);
|
|
DDX_Control(pDX, IDC_LIST2, m_CacheEntriesList);
|
|
DDX_Control(pDX, IDC_MouseHelpCaption, m_MouseHelp);
|
|
DDX_Control(pDX, IDC_PROGRESS, m_StatusDisplay);
|
|
DDX_Control(pDX, IDC_BuildProjects, m_Configure);
|
|
DDX_Check(pDX, IDC_AdvancedValues, m_AdvancedValues);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CMakeSetupDialog, CDialog)
|
|
//{{AFX_MSG_MAP(CMakeSetupDialog)
|
|
ON_WM_SYSCOMMAND()
|
|
ON_WM_PAINT()
|
|
ON_WM_QUERYDRAGICON()
|
|
ON_BN_CLICKED(IDC_BUTTON2, OnBrowseWhereSource)
|
|
ON_BN_CLICKED(IDC_BuildProjects, OnConfigure)
|
|
ON_BN_CLICKED(IDC_BUTTON3, OnBrowseWhereBuild)
|
|
ON_CBN_EDITCHANGE(IDC_WhereBuild, OnChangeWhereBuild)
|
|
ON_CBN_SELCHANGE(IDC_WhereBuild, OnSelendokWhereBuild)
|
|
ON_CBN_EDITCHANGE(IDC_WhereSource, OnChangeWhereSource)
|
|
ON_CBN_SELENDOK(IDC_WhereSource, OnSelendokWhereSource)
|
|
ON_WM_SIZE()
|
|
ON_WM_GETMINMAXINFO()
|
|
ON_BN_CLICKED(IDC_OK, OnOk)
|
|
ON_BN_CLICKED(IDC_DELETE_BUTTON, OnDeleteButton)
|
|
ON_BN_CLICKED(IDC_HELP_BUTTON, OnHelpButton)
|
|
ON_BN_CLICKED(IDC_AdvancedValues, OnAdvancedValues)
|
|
ON_BN_DOUBLECLICKED(IDC_AdvancedValues, OnDoubleclickedAdvancedValues)
|
|
ON_WM_DROPFILES()
|
|
ON_BN_CLICKED(IDCANCEL, OnCancel)
|
|
ON_WM_SETCURSOR()
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CMakeSetupDialog message handlers
|
|
|
|
BOOL CMakeSetupDialog::OnInitDialog()
|
|
{
|
|
CDialog::OnInitDialog();
|
|
this->DragAcceptFiles(true);
|
|
|
|
// Add "Create shortcut" menu item to system menu.
|
|
|
|
// IDM_CREATESHORTCUT must be in the system command range.
|
|
ASSERT((IDM_CREATESHORTCUT & 0xFFF0) == IDM_CREATESHORTCUT);
|
|
ASSERT(IDM_CREATESHORTCUT < 0xF000);
|
|
|
|
// Add "About..." menu item to system menu.
|
|
|
|
// IDM_ABOUTBOX must be in the system command range.
|
|
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
|
|
ASSERT(IDM_ABOUTBOX < 0xF000);
|
|
|
|
CMenu* pSysMenu = GetSystemMenu(FALSE);
|
|
if (pSysMenu != NULL)
|
|
{
|
|
CString strCreateShortcutMenu;
|
|
strCreateShortcutMenu.LoadString(IDS_CREATESHORTCUT);
|
|
if (!strCreateShortcutMenu.IsEmpty())
|
|
{
|
|
pSysMenu->AppendMenu(MF_SEPARATOR);
|
|
pSysMenu->AppendMenu(MF_STRING,
|
|
IDM_CREATESHORTCUT,
|
|
strCreateShortcutMenu);
|
|
}
|
|
|
|
CString strAboutMenu;
|
|
strAboutMenu.LoadString(IDS_ABOUTBOX);
|
|
if (!strAboutMenu.IsEmpty())
|
|
{
|
|
pSysMenu->AppendMenu(MF_SEPARATOR);
|
|
pSysMenu->AppendMenu(MF_STRING,
|
|
IDM_ABOUTBOX,
|
|
strAboutMenu);
|
|
}
|
|
}
|
|
|
|
// Set the icon for this dialog. The framework does this automatically
|
|
// when the application's main window is not a dialog
|
|
SetIcon(m_hIcon, TRUE); // Set big icon
|
|
SetIcon(m_hIcon, FALSE); // Set small icon
|
|
// Load source and build dirs from registry
|
|
this->LoadFromRegistry();
|
|
|
|
// try to load the cmake cache from disk
|
|
this->LoadCacheFromDiskToGUI();
|
|
m_WhereBuildControl.LimitText(2048);
|
|
m_WhereSourceControl.LimitText(2048);
|
|
|
|
// Set the version number
|
|
char tmp[1024];
|
|
sprintf(tmp,"CMake %d.%d - %s", cmake::GetMajorVersion(),
|
|
cmake::GetMinorVersion(), cmake::GetReleaseVersion());
|
|
SetDlgItemText(IDC_PROGRESS, "");
|
|
this->SetWindowText(tmp);
|
|
this->UpdateData(FALSE);
|
|
return TRUE; // return TRUE unless you set the focus to a control
|
|
}
|
|
|
|
|
|
// About dialog invoke
|
|
void CMakeSetupDialog::OnSysCommand(UINT nID, LPARAM lParam)
|
|
{
|
|
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
|
|
{
|
|
CAboutDlg dlgAbout;
|
|
dlgAbout.DoModal();
|
|
}
|
|
else if ((nID & 0xFFF0) == IDM_CREATESHORTCUT)
|
|
{
|
|
CreateShortcut();
|
|
}
|
|
else
|
|
{
|
|
CDialog::OnSysCommand(nID, lParam);
|
|
}
|
|
}
|
|
|
|
// If you add a minimize button to your dialog, you will need the code below
|
|
// to draw the icon. For MFC applications using the document/view model,
|
|
// this is automatically done for you by the framework.
|
|
|
|
void CMakeSetupDialog::OnPaint()
|
|
{
|
|
if (IsIconic())
|
|
{
|
|
CPaintDC dc(this); // device context for painting
|
|
|
|
SendMessage(WM_ICONERASEBKGND, (WPARAM) dc.GetSafeHdc(), 0);
|
|
|
|
// Center icon in client rectangle
|
|
int cxIcon = GetSystemMetrics(SM_CXICON);
|
|
int cyIcon = GetSystemMetrics(SM_CYICON);
|
|
CRect rect;
|
|
GetClientRect(&rect);
|
|
int x = (rect.Width() - cxIcon + 1) / 2;
|
|
int y = (rect.Height() - cyIcon + 1) / 2;
|
|
|
|
// Draw the icon
|
|
dc.DrawIcon(x, y, m_hIcon);
|
|
}
|
|
else
|
|
{
|
|
CDialog::OnPaint();
|
|
}
|
|
}
|
|
|
|
// The system calls this to obtain the cursor to display while the user drags
|
|
// the minimized window.
|
|
HCURSOR CMakeSetupDialog::OnQueryDragIcon()
|
|
{
|
|
return (HCURSOR) m_hIcon;
|
|
}
|
|
|
|
|
|
|
|
// Browse button
|
|
bool CMakeSetupDialog::Browse(CString &result, const char *title)
|
|
{
|
|
CString initialDir = result;
|
|
initialDir.Replace("/", "\\");
|
|
CPathDialog dlg("Select Path", title, initialDir);
|
|
if(dlg.DoModal()==IDOK)
|
|
{
|
|
result = dlg.GetPathName();
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
void CMakeSetupDialog::SaveToRegistry()
|
|
{
|
|
HKEY hKey;
|
|
DWORD dwDummy;
|
|
|
|
if(RegCreateKeyEx(HKEY_CURRENT_USER,
|
|
m_RegistryKey,
|
|
0, "", REG_OPTION_NON_VOLATILE, KEY_READ|KEY_WRITE,
|
|
NULL, &hKey, &dwDummy) != ERROR_SUCCESS)
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// save some values
|
|
CString regvalue;
|
|
this->ReadRegistryValue(hKey, &(regvalue),"WhereSource1","C:\\");
|
|
int shiftEnd = 9;
|
|
if(m_WhereSource != regvalue)
|
|
{
|
|
char keyName[1024];
|
|
char keyName2[1024];
|
|
int i;
|
|
for (i = 2; i < 10; ++i)
|
|
{
|
|
regvalue = "";
|
|
sprintf(keyName,"WhereSource%i",i);
|
|
this->ReadRegistryValue(hKey, &(regvalue),keyName,"");
|
|
// check for short circuit, if the new value is already in
|
|
// the list then we stop
|
|
if (m_WhereSource == regvalue)
|
|
{
|
|
shiftEnd = i - 1;
|
|
}
|
|
}
|
|
|
|
for (i = shiftEnd; i; --i)
|
|
{
|
|
regvalue = "";
|
|
sprintf(keyName,"WhereSource%i",i);
|
|
sprintf(keyName2,"WhereSource%i",i+1);
|
|
|
|
this->ReadRegistryValue(hKey, &(regvalue),keyName,"");
|
|
if (strlen(regvalue))
|
|
{
|
|
RegSetValueEx(hKey, _T(keyName2), 0, REG_SZ,
|
|
(CONST BYTE *)(const char *)regvalue,
|
|
regvalue.GetLength());
|
|
}
|
|
}
|
|
RegSetValueEx(hKey, _T("WhereSource1"), 0, REG_SZ,
|
|
(CONST BYTE *)(const char *)m_WhereSource,
|
|
m_WhereSource.GetLength());
|
|
}
|
|
|
|
this->ReadRegistryValue(hKey, &(regvalue),"WhereBuild1","C:\\");
|
|
if(m_WhereBuild != regvalue)
|
|
{
|
|
int i;
|
|
char keyName[1024];
|
|
char keyName2[1024];
|
|
for (i = 2; i < 10; ++i)
|
|
{
|
|
regvalue = "";
|
|
sprintf(keyName,"WhereBuild%i",i);
|
|
this->ReadRegistryValue(hKey, &(regvalue),keyName,"");
|
|
// check for short circuit, if the new value is already in
|
|
// the list then we stop
|
|
if (m_WhereBuild == regvalue)
|
|
{
|
|
shiftEnd = i - 1;
|
|
}
|
|
}
|
|
for (i = shiftEnd; i; --i)
|
|
{
|
|
regvalue = "";
|
|
sprintf(keyName,"WhereBuild%i",i);
|
|
sprintf(keyName2,"WhereBuild%i",i+1);
|
|
|
|
this->ReadRegistryValue(hKey, &(regvalue),keyName,"");
|
|
if (strlen(regvalue))
|
|
{
|
|
RegSetValueEx(hKey, _T(keyName2), 0, REG_SZ,
|
|
(CONST BYTE *)(const char *)regvalue,
|
|
regvalue.GetLength());
|
|
}
|
|
}
|
|
RegSetValueEx(hKey, _T("WhereBuild1"), 0, REG_SZ,
|
|
(CONST BYTE *)(const char *)m_WhereBuild,
|
|
m_WhereBuild.GetLength());
|
|
}
|
|
}
|
|
RegCloseKey(hKey);
|
|
}
|
|
|
|
|
|
void CMakeSetupDialog::ReadRegistryValue(HKEY hKey,
|
|
CString *val,
|
|
const char *key,
|
|
const char *adefault)
|
|
{
|
|
DWORD dwType, dwSize;
|
|
char *pb;
|
|
|
|
dwType = REG_SZ;
|
|
pb = val->GetBuffer(MAX_PATH);
|
|
dwSize = MAX_PATH;
|
|
if(RegQueryValueEx(hKey,_T(key), NULL, &dwType,
|
|
(BYTE *)pb, &dwSize) != ERROR_SUCCESS)
|
|
{
|
|
val->ReleaseBuffer();
|
|
*val = _T(adefault);
|
|
}
|
|
else
|
|
{
|
|
val->ReleaseBuffer();
|
|
}
|
|
}
|
|
|
|
|
|
void CMakeSetupDialog::LoadFromRegistry()
|
|
{
|
|
HKEY hKey;
|
|
if(RegOpenKeyEx(HKEY_CURRENT_USER,
|
|
m_RegistryKey,
|
|
0, KEY_READ, &hKey) != ERROR_SUCCESS)
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// load some values
|
|
if (m_WhereSource.IsEmpty())
|
|
{
|
|
this->ReadRegistryValue(hKey, &(m_WhereSource),"WhereSource1","C:\\");
|
|
}
|
|
if (m_WhereBuild.IsEmpty())
|
|
{
|
|
this->ReadRegistryValue(hKey, &(m_WhereBuild),"WhereBuild1","C:\\");
|
|
}
|
|
m_WhereSourceControl.AddString(m_WhereSource);
|
|
m_WhereBuildControl.AddString(m_WhereBuild);
|
|
|
|
char keyname[1024];
|
|
CString regvalue;
|
|
int i;
|
|
for (i = 2; i <= 10; ++i)
|
|
{
|
|
sprintf(keyname,"WhereSource%i",i);
|
|
regvalue = "";
|
|
this->ReadRegistryValue(hKey, &(regvalue),keyname,"C:\\");
|
|
if (strcmp("C:\\",regvalue))
|
|
{
|
|
m_WhereSourceControl.AddString(regvalue);
|
|
}
|
|
sprintf(keyname,"WhereBuild%i",i);
|
|
regvalue = "";
|
|
this->ReadRegistryValue(hKey, &(regvalue),keyname,"C:\\");
|
|
if (strcmp("C:\\",regvalue))
|
|
{
|
|
m_WhereBuildControl.AddString(regvalue);
|
|
}
|
|
}
|
|
}
|
|
RegCloseKey(hKey);
|
|
}
|
|
|
|
|
|
|
|
// Callback for browse source button
|
|
void CMakeSetupDialog::OnBrowseWhereSource()
|
|
{
|
|
this->UpdateData();
|
|
Browse(m_WhereSource, "Enter Path to Source");
|
|
this->UpdateData(false);
|
|
this->OnChangeWhereSource();
|
|
}
|
|
|
|
// Callback for browser build button
|
|
void CMakeSetupDialog::OnBrowseWhereBuild()
|
|
{
|
|
this->UpdateData();
|
|
Browse(m_WhereBuild, "Enter Path to Build");
|
|
this->UpdateData(false);
|
|
this->OnChangeWhereBuild();
|
|
}
|
|
|
|
void CMakeSetupDialog::RunCMake(bool generateProjectFiles)
|
|
{
|
|
if(!cmSystemTools::FileExists(m_WhereBuild))
|
|
{
|
|
std::string message =
|
|
"Build directory does not exist, should I create it?\n\n"
|
|
"Directory: ";
|
|
message += (const char*)m_WhereBuild;
|
|
if(MessageBox(message.c_str(), "Create Directory", MB_OKCANCEL) == IDOK)
|
|
{
|
|
cmSystemTools::MakeDirectory(m_WhereBuild);
|
|
}
|
|
else
|
|
{
|
|
MessageBox("Build Project aborted, nothing done.");
|
|
return;
|
|
}
|
|
}
|
|
// set the wait cursor
|
|
m_Cursor = LoadCursor(NULL, IDC_WAIT);
|
|
::SetCursor(m_Cursor);
|
|
m_RunningConfigure = true;
|
|
|
|
// get all the info from the dialog
|
|
this->UpdateData();
|
|
// always save the current gui values to disk
|
|
this->SaveCacheFromGUI();
|
|
// Make sure we are working from the cache on disk
|
|
this->LoadCacheFromDiskToGUI();
|
|
m_OKButton.EnableWindow(false);
|
|
|
|
// setup the cmake instance
|
|
if (generateProjectFiles)
|
|
{
|
|
if(m_CMakeInstance->Generate() != 0)
|
|
{
|
|
cmSystemTools::Error(
|
|
"Error in generation process, project files may be invalid");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_CMakeInstance->SetHomeDirectory(m_WhereSource);
|
|
m_CMakeInstance->SetStartDirectory(m_WhereSource);
|
|
m_CMakeInstance->SetHomeOutputDirectory(m_WhereBuild);
|
|
m_CMakeInstance->SetStartOutputDirectory(m_WhereBuild);
|
|
m_CMakeInstance->SetGlobalGenerator(
|
|
m_CMakeInstance->CreateGlobalGenerator(m_GeneratorDialog.m_GeneratorChoiceString));
|
|
m_CMakeInstance->SetCMakeCommand(m_PathToExecutable);
|
|
m_CMakeInstance->LoadCache();
|
|
if(m_CMakeInstance->Configure() != 0)
|
|
{
|
|
cmSystemTools::Error(
|
|
"Error in configuration process, project files may be invalid");
|
|
}
|
|
// update the GUI with any new values in the caused by the
|
|
// generation process
|
|
this->LoadCacheFromDiskToGUI();
|
|
}
|
|
|
|
// save source and build paths to registry
|
|
this->SaveToRegistry();
|
|
// path is up-to-date now
|
|
m_BuildPathChanged = false;
|
|
// put the cursor back
|
|
m_Cursor = LoadCursor(NULL, IDC_ARROW);
|
|
::SetCursor(m_Cursor);
|
|
m_RunningConfigure = false;
|
|
cmSystemTools::ResetErrorOccuredFlag();
|
|
}
|
|
|
|
|
|
// Callback for build projects button
|
|
void CMakeSetupDialog::OnConfigure()
|
|
{
|
|
if(!m_GeneratorPicked)
|
|
{
|
|
m_GeneratorDialog.m_CMakeInstance = this->m_CMakeInstance;
|
|
if(m_GeneratorDialog.DoModal() != IDOK)
|
|
{
|
|
return;
|
|
}
|
|
// save the generator choice in the registry
|
|
HKEY hKey;
|
|
DWORD dwDummy;
|
|
|
|
if(RegCreateKeyEx(HKEY_CURRENT_USER,
|
|
m_RegistryKey,
|
|
0, "", REG_OPTION_NON_VOLATILE, KEY_READ|KEY_WRITE,
|
|
NULL, &hKey, &dwDummy) == ERROR_SUCCESS)
|
|
{
|
|
// save some values
|
|
RegSetValueEx(hKey, _T("LastGenerator"), 0, REG_SZ,
|
|
(CONST BYTE *)(const char *)m_GeneratorDialog.m_GeneratorChoiceString,
|
|
m_GeneratorDialog.m_GeneratorChoiceString.GetLength());
|
|
}
|
|
}
|
|
|
|
// enable error messages each time configure is pressed
|
|
cmSystemTools::EnableMessages();
|
|
this->RunCMake(false);
|
|
}
|
|
|
|
|
|
|
|
|
|
// callback for combo box menu where build selection
|
|
void CMakeSetupDialog::OnSelendokWhereBuild()
|
|
{
|
|
m_WhereBuildControl.GetLBText(m_WhereBuildControl.GetCurSel(),
|
|
m_WhereBuild);
|
|
m_WhereBuildControl.SetWindowText( m_WhereBuild);
|
|
this->UpdateData(FALSE);
|
|
this->OnChangeWhereBuild();
|
|
}
|
|
|
|
// callback for combo box menu where source selection
|
|
void CMakeSetupDialog::OnSelendokWhereSource()
|
|
{
|
|
m_WhereSourceControl.GetLBText(m_WhereSourceControl.GetCurSel(),
|
|
m_WhereSource);
|
|
this->UpdateData(FALSE);
|
|
this->OnChangeWhereSource();
|
|
}
|
|
|
|
// callback for chaing source directory
|
|
void CMakeSetupDialog::OnChangeWhereSource()
|
|
{
|
|
}
|
|
|
|
// callback for changing the build directory
|
|
void CMakeSetupDialog::OnChangeWhereBuild()
|
|
{
|
|
this->UpdateData();
|
|
|
|
// The build dir has changed, check if there is a cache, and
|
|
// grab the source dir from it
|
|
|
|
std::string path = this->m_WhereBuild;
|
|
cmSystemTools::ConvertToUnixSlashes(path);
|
|
|
|
// adjust the cmake instance
|
|
m_CMakeInstance->SetHomeOutputDirectory(m_WhereBuild);
|
|
m_CMakeInstance->SetStartOutputDirectory(m_WhereBuild);
|
|
|
|
std::string cache_file = path;
|
|
cache_file += "/CMakeCache.txt";
|
|
|
|
cmCacheManager *cachem = this->m_CMakeInstance->GetCacheManager();
|
|
cmCacheManager::CacheIterator it = cachem->NewIterator();
|
|
|
|
m_GeneratorPicked = false;
|
|
|
|
// make sure we have a normal cache file, specifically if one exists make
|
|
// sure it can be read
|
|
if (cmSystemTools::FileExists(cache_file.c_str()))
|
|
{
|
|
if (cachem->LoadCache(path.c_str()))
|
|
{
|
|
if (it.Find("CMAKE_HOME_DIRECTORY"))
|
|
{
|
|
path = ConvertToWindowsPath(it.GetValue());
|
|
this->m_WhereSource = path.c_str();
|
|
this->m_WhereSourceControl.SetWindowText(this->m_WhereSource);
|
|
this->OnChangeWhereSource();
|
|
m_GeneratorPicked = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//file exists but cqnnot be read
|
|
cmSystemTools::Error("There is a CMakeCache.txt file for the current binary tree but cmake does not have permission to read it. Please check the permissions of the directory you are trying to run CMake on.");
|
|
return;
|
|
}
|
|
}
|
|
|
|
m_CacheEntriesList.RemoveAll();
|
|
m_CacheEntriesList.ShowWindow(SW_SHOW);
|
|
this->LoadCacheFromDiskToGUI();
|
|
m_BuildPathChanged = true;
|
|
}
|
|
|
|
|
|
// copy from the cache manager to the cache edit list box
|
|
void CMakeSetupDialog::FillCacheGUIFromCacheManager()
|
|
{
|
|
cmCacheManager *cachem = this->m_CMakeInstance->GetCacheManager();
|
|
cmCacheManager::CacheIterator it = cachem->NewIterator();
|
|
size_t size = m_CacheEntriesList.GetItems().size();
|
|
// if there are already entries in the cache, then
|
|
// put the new ones in the top, so they show up first
|
|
bool reverseOrder = false;
|
|
// all the current values are not new any more
|
|
std::set<CPropertyItem*> items = m_CacheEntriesList.GetItems();
|
|
for(std::set<CPropertyItem*>::iterator i = items.begin();
|
|
i != items.end(); ++i)
|
|
{
|
|
// first check to see if it is still in the cache
|
|
CPropertyItem* item = *i;
|
|
if ( !it.Find((const char*)item->m_propName) )
|
|
{
|
|
m_CacheEntriesList.RemoveProperty((const char*)item->m_propName);
|
|
}
|
|
else
|
|
{
|
|
// if it is still in the cache then it is no longer new
|
|
item->m_NewValue = false;
|
|
}
|
|
}
|
|
for(cmCacheManager::CacheIterator i = cachem->NewIterator();
|
|
!i.IsAtEnd(); i.Next())
|
|
{
|
|
const char* key = i.GetName();
|
|
|
|
// if value has trailing space or tab, enclose it in single quotes
|
|
// to enforce the fact that it has 'invisible' trailing stuff
|
|
std::string value = i.GetValue();
|
|
if (value.size() &&
|
|
(value[value.size() - 1] == ' ' ||
|
|
value[value.size() - 1] == '\t'))
|
|
{
|
|
value = '\'' + value + '\'';
|
|
}
|
|
bool advanced = i.GetPropertyAsBool("ADVANCED");
|
|
switch(i.GetType() )
|
|
{
|
|
case cmCacheManager::BOOL:
|
|
if(cmSystemTools::IsOn(value.c_str()))
|
|
{
|
|
m_CacheEntriesList.AddProperty(key,
|
|
"ON",
|
|
i.GetProperty("HELPSTRING"),
|
|
CPropertyList::COMBO,"ON|OFF",
|
|
reverseOrder,
|
|
advanced
|
|
);
|
|
}
|
|
else
|
|
{
|
|
m_CacheEntriesList.AddProperty(key,
|
|
"OFF",
|
|
i.GetProperty("HELPSTRING"),
|
|
CPropertyList::COMBO,"ON|OFF",
|
|
reverseOrder, advanced
|
|
);
|
|
}
|
|
break;
|
|
case cmCacheManager::PATH:
|
|
m_CacheEntriesList.AddProperty(key,
|
|
value.c_str(),
|
|
i.GetProperty("HELPSTRING"),
|
|
CPropertyList::PATH,"",
|
|
reverseOrder, advanced
|
|
);
|
|
break;
|
|
case cmCacheManager::FILEPATH:
|
|
m_CacheEntriesList.AddProperty(key,
|
|
value.c_str(),
|
|
i.GetProperty("HELPSTRING"),
|
|
CPropertyList::FILE,"",
|
|
reverseOrder, advanced
|
|
);
|
|
break;
|
|
case cmCacheManager::STRING:
|
|
m_CacheEntriesList.AddProperty(key,
|
|
value.c_str(),
|
|
i.GetProperty("HELPSTRING"),
|
|
CPropertyList::EDIT,"",
|
|
reverseOrder, advanced
|
|
);
|
|
break;
|
|
case cmCacheManager::INTERNAL:
|
|
m_CacheEntriesList.RemoveProperty(key);
|
|
break;
|
|
}
|
|
}
|
|
if(m_CacheEntriesList.GetShowAdvanced())
|
|
{
|
|
m_CacheEntriesList.ShowAdvanced();
|
|
}
|
|
else
|
|
{
|
|
m_CacheEntriesList.HideAdvanced();
|
|
}
|
|
|
|
m_OKButton.EnableWindow(false);
|
|
if(cachem->GetSize() > 0 && !cmSystemTools::GetErrorOccuredFlag())
|
|
{
|
|
bool enable = true;
|
|
items = m_CacheEntriesList.GetItems();
|
|
for(std::set<CPropertyItem*>::iterator i = items.begin();
|
|
i != items.end(); ++i)
|
|
{
|
|
CPropertyItem* item = *i;
|
|
if(item->m_Advanced )
|
|
{
|
|
if(item->m_NewValue && m_CacheEntriesList.GetShowAdvanced())
|
|
{
|
|
enable = false;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(item->m_NewValue)
|
|
{
|
|
// if one new value then disable to OK button
|
|
enable = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(enable)
|
|
{
|
|
m_OKButton.EnableWindow(true);
|
|
}
|
|
}
|
|
|
|
// redraw the list
|
|
m_CacheEntriesList.SetTopIndex(0);
|
|
m_CacheEntriesList.Invalidate();
|
|
}
|
|
|
|
// copy from the list box to the cache manager
|
|
void CMakeSetupDialog::FillCacheManagerFromCacheGUI()
|
|
{
|
|
cmCacheManager *cachem = this->m_CMakeInstance->GetCacheManager();
|
|
std::set<CPropertyItem*> items = m_CacheEntriesList.GetItems();
|
|
cmCacheManager::CacheIterator it = cachem->NewIterator();
|
|
for(std::set<CPropertyItem*>::iterator i = items.begin();
|
|
i != items.end(); ++i)
|
|
{
|
|
CPropertyItem* item = *i;
|
|
if ( it.Find((const char*)item->m_propName) )
|
|
{
|
|
// if value is enclosed in single quotes ('foo') then remove them
|
|
// they were used to enforce the fact that it had 'invisible'
|
|
// trailing stuff
|
|
if (item->m_curValue.GetLength() >= 2 &&
|
|
item->m_curValue[0] == '\'' &&
|
|
item->m_curValue[item->m_curValue.GetLength() - 1] == '\'')
|
|
{
|
|
it.SetValue(item->m_curValue.Mid(
|
|
1, item->m_curValue.GetLength() - 2));
|
|
}
|
|
else
|
|
{
|
|
it.SetValue(item->m_curValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//! Load cache file from m_WhereBuild and display in GUI editor
|
|
void CMakeSetupDialog::LoadCacheFromDiskToGUI()
|
|
{
|
|
cmCacheManager *cachem = this->m_CMakeInstance->GetCacheManager();
|
|
if(m_WhereBuild != "")
|
|
{
|
|
if (!cachem->LoadCache(m_WhereBuild))
|
|
{
|
|
// if it does exist, but isn;t readable then warn the user
|
|
std::string cacheFile = m_WhereBuild;
|
|
cacheFile += "/CMakeCache.txt";
|
|
if(cmSystemTools::FileExists(cacheFile.c_str()))
|
|
{
|
|
cmSystemTools::Error("There is a CMakeCache.txt file for the current binary tree but cmake does not have permission to read it. Please check the permissions of the directory you are trying to run CMake on.");
|
|
return;
|
|
}
|
|
}
|
|
cmCacheManager::CacheIterator itm = cachem->NewIterator();
|
|
if ( itm.Find("CMAKE_HOME_DIRECTORY"))
|
|
{
|
|
std::string path = ConvertToWindowsPath(itm.GetValue());
|
|
this->m_WhereSource = path.c_str();
|
|
this->m_WhereSourceControl.SetWindowText(this->m_WhereSource);
|
|
this->OnChangeWhereSource();
|
|
}
|
|
m_CMakeInstance->SetHomeDirectory(m_WhereSource);
|
|
m_CMakeInstance->SetStartDirectory(m_WhereSource);
|
|
m_CMakeInstance->SetHomeOutputDirectory(m_WhereBuild);
|
|
m_CMakeInstance->SetStartOutputDirectory(m_WhereBuild);
|
|
m_CMakeInstance->PreLoadCMakeFiles();
|
|
this->FillCacheGUIFromCacheManager();
|
|
cmCacheManager::CacheIterator it =
|
|
cachem->GetCacheIterator("CMAKE_GENERATOR");
|
|
if(!it.IsAtEnd())
|
|
{
|
|
m_GeneratorPicked = true;
|
|
std::string curGen = it.GetValue();
|
|
if(m_GeneratorDialog.m_GeneratorChoiceString != curGen.c_str())
|
|
{
|
|
m_GeneratorDialog.m_GeneratorChoiceString = curGen.c_str();
|
|
this->UpdateData(FALSE);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//! Save GUI values to cmCacheManager and then save to disk.
|
|
void CMakeSetupDialog::SaveCacheFromGUI()
|
|
{
|
|
cmCacheManager *cachem = this->m_CMakeInstance->GetCacheManager();
|
|
this->FillCacheManagerFromCacheGUI();
|
|
if(m_WhereBuild != "")
|
|
{
|
|
cachem->SaveCache(m_WhereBuild);
|
|
}
|
|
}
|
|
|
|
|
|
void CMakeSetupDialog::OnSize(UINT nType, int cx, int cy)
|
|
{
|
|
if (nType == SIZE_MINIMIZED)
|
|
{
|
|
CDialog::OnSize(nType, cx, cy);
|
|
return;
|
|
}
|
|
if (m_oldCX == -1)
|
|
{
|
|
m_oldCX = cx;
|
|
m_oldCY = cy;
|
|
}
|
|
int deltax = cx - m_oldCX;
|
|
int deltay = cy - m_oldCY;
|
|
|
|
m_oldCX = cx;
|
|
m_oldCY = cy;
|
|
|
|
CDialog::OnSize(nType, cx, cy);
|
|
|
|
if (deltax == 0 && deltay == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if(m_CacheEntriesList.m_hWnd)
|
|
{
|
|
// get the original sizes/positions
|
|
CRect cRect;
|
|
m_AdvancedValuesControl.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_AdvancedValuesControl.SetWindowPos(&wndTop, cRect.left + deltax,
|
|
cRect.top,
|
|
0, 0,
|
|
SWP_NOCOPYBITS |
|
|
SWP_NOSIZE | SWP_NOZORDER);
|
|
m_BrowseSource.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_BrowseSource.SetWindowPos(&wndTop, cRect.left + deltax,
|
|
cRect.top,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE | SWP_NOZORDER);
|
|
m_BrowseBuild.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_BrowseBuild.SetWindowPos(&wndTop, cRect.left + deltax,
|
|
cRect.top,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE | SWP_NOZORDER);
|
|
|
|
m_WhereSourceControl.GetWindowRect(&cRect);
|
|
m_WhereSourceControl.SetWindowPos(&wndTop, cRect.left, cRect.top,
|
|
cRect.Width() + deltax,
|
|
cRect.Height(),
|
|
SWP_NOCOPYBITS |
|
|
SWP_NOMOVE | SWP_NOZORDER);
|
|
m_WhereBuildControl.GetWindowRect(&cRect);
|
|
m_WhereBuildControl.SetWindowPos(&wndTop, cRect.left, cRect.top,
|
|
cRect.Width() + deltax,
|
|
cRect.Height(),
|
|
SWP_NOCOPYBITS |
|
|
SWP_NOMOVE | SWP_NOZORDER);
|
|
m_ListFrame.GetWindowRect(&cRect);
|
|
m_ListFrame.SetWindowPos(&wndTop, cRect.left, cRect.top,
|
|
cRect.Width() + deltax,
|
|
cRect.Height() + deltay,
|
|
SWP_NOCOPYBITS | SWP_NOMOVE | SWP_NOZORDER);
|
|
m_CacheEntriesList.GetWindowRect(&cRect);
|
|
m_CacheEntriesList.SetWindowPos(&wndTop, cRect.left, cRect.top,
|
|
cRect.Width() + deltax,
|
|
cRect.Height() + deltay,
|
|
SWP_NOCOPYBITS | SWP_NOMOVE | SWP_NOZORDER);
|
|
|
|
m_StatusDisplay.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_StatusDisplay.SetWindowPos(&wndBottom, cRect.left,
|
|
cRect.top + deltay,
|
|
cRect.Width() + deltax, cRect.Height(),
|
|
SWP_NOCOPYBITS);
|
|
|
|
m_MouseHelp.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_MouseHelp.SetWindowPos(&wndTop, cRect.left ,
|
|
cRect.top + deltay,
|
|
cRect.Width() + deltax, cRect.Height(),
|
|
SWP_NOCOPYBITS | SWP_NOZORDER);
|
|
|
|
deltax = int(deltax + m_deltaXRemainder);
|
|
m_deltaXRemainder = float(deltax%2);
|
|
|
|
|
|
m_Configure.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_Configure.SetWindowPos(&wndTop, cRect.left + deltax/2,
|
|
cRect.top + deltay,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE);
|
|
m_CancelButton.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_CancelButton.SetWindowPos(&wndTop, cRect.left + deltax/2,
|
|
cRect.top + deltay,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE);
|
|
m_OKButton.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_OKButton.SetWindowPos(&wndTop, cRect.left + deltax/2,
|
|
cRect.top + deltay,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE);
|
|
m_DeleteButton.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_DeleteButton.SetWindowPos(&wndTop, cRect.left + deltax/2,
|
|
cRect.top + deltay,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE);
|
|
m_HelpButton.GetWindowRect(&cRect);
|
|
this->ScreenToClient(&cRect);
|
|
m_HelpButton.SetWindowPos(&wndTop, cRect.left + deltax/2,
|
|
cRect.top + deltay,
|
|
0, 0,
|
|
SWP_NOCOPYBITS | SWP_NOSIZE);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
void CMakeSetupDialog::OnGetMinMaxInfo( MINMAXINFO FAR* lpMMI )
|
|
{
|
|
lpMMI->ptMinTrackSize.x = 550;
|
|
lpMMI->ptMinTrackSize.y = 272;
|
|
}
|
|
|
|
void CMakeSetupDialog::OnCancel()
|
|
{
|
|
if(m_RunningConfigure)
|
|
{
|
|
if(MessageBox("You are in the middle of a Configure.\n"
|
|
"If you Cancel now the configure information will be lost.\n"
|
|
"Are you sure you want to Cancel?", "Confirm Exit",
|
|
MB_YESNO) == IDYES)
|
|
{
|
|
cmSystemTools::SetFatalErrorOccured();
|
|
}
|
|
return;
|
|
}
|
|
if(m_CacheEntriesList.IsDirty())
|
|
{
|
|
if(MessageBox("You have changed options but not rebuilt, "
|
|
"are you sure you want to exit?", "Confirm Exit",
|
|
MB_YESNO) == IDYES)
|
|
{
|
|
CDialog::OnOK();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CDialog::OnOK();
|
|
}
|
|
}
|
|
|
|
void CMakeSetupDialog::OnOk()
|
|
{
|
|
// enable error messages each time configure is pressed
|
|
cmSystemTools::EnableMessages();
|
|
m_CacheEntriesList.ClearDirty();
|
|
this->RunCMake(true);
|
|
if (!(::GetKeyState(VK_SHIFT) & 0x1000))
|
|
{
|
|
CDialog::OnOK();
|
|
}
|
|
}
|
|
|
|
// Create a shortcut on the desktop with the current Source/Build dir.
|
|
int CMakeSetupDialog::CreateShortcut()
|
|
{
|
|
// Find the desktop folder and create the link name
|
|
|
|
HKEY hKey;
|
|
if(RegOpenKeyEx(HKEY_CURRENT_USER,
|
|
"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders",
|
|
0, KEY_READ, &hKey) != ERROR_SUCCESS)
|
|
{
|
|
AfxMessageBox ("Create shortcut: unable to find 'Shell Folders' key in registry!");
|
|
return 1;
|
|
}
|
|
|
|
DWORD dwType, dwSize;
|
|
#define MAXPATH 1024
|
|
char link_name[MAXPATH];
|
|
dwSize = MAXPATH;
|
|
if(RegQueryValueEx(hKey,
|
|
(LPCTSTR)"Desktop",
|
|
NULL,
|
|
&dwType,
|
|
(BYTE *)link_name,
|
|
&dwSize) != ERROR_SUCCESS)
|
|
{
|
|
AfxMessageBox ("Create shortcut: unable to find 'Desktop' registry value in 'Shell Folders' key!");
|
|
return 1;
|
|
}
|
|
|
|
if(dwType != REG_SZ)
|
|
{
|
|
AfxMessageBox ("Create shortcut: 'Desktop' registry value in 'Shell Folders' key has wrong type!");
|
|
return 1;
|
|
}
|
|
|
|
strcat(link_name, "\\CMake - ");
|
|
std::string current_dir = cmSystemTools::GetFilenameName((LPCTSTR)m_WhereSource);
|
|
strcat(link_name, current_dir.c_str());
|
|
strcat(link_name, ".lnk");
|
|
|
|
// Find the path to the current executable
|
|
|
|
char path_to_current_exe[MAXPATH];
|
|
::GetModuleFileName(NULL, path_to_current_exe, MAXPATH);
|
|
|
|
// Create the shortcut
|
|
|
|
HRESULT hres;
|
|
IShellLink *psl;
|
|
|
|
// Initialize the COM library
|
|
|
|
hres = CoInitialize(NULL);
|
|
|
|
if (! SUCCEEDED (hres))
|
|
{
|
|
AfxMessageBox ("Create shortcut: unable to initialize the COM library!");
|
|
return 1;
|
|
}
|
|
|
|
// Create an IShellLink object and get a pointer to the IShellLink
|
|
// interface (returned from CoCreateInstance).
|
|
|
|
hres = CoCreateInstance(CLSID_ShellLink,
|
|
NULL,
|
|
CLSCTX_INPROC_SERVER,
|
|
IID_IShellLink,
|
|
(void **)&psl);
|
|
|
|
if (! SUCCEEDED (hres))
|
|
{
|
|
AfxMessageBox ("Create shortcut: unable to create IShellLink instance!");
|
|
return 1;
|
|
}
|
|
|
|
IPersistFile *ppf;
|
|
|
|
// Query IShellLink for the IPersistFile interface for
|
|
// saving the shortcut in persistent storage.
|
|
|
|
hres = psl->QueryInterface(IID_IPersistFile, (void **)&ppf);
|
|
|
|
if (SUCCEEDED (hres))
|
|
{
|
|
// Set the path to the shortcut target.
|
|
hres = psl->SetPath(path_to_current_exe);
|
|
|
|
if (! SUCCEEDED (hres))
|
|
{
|
|
AfxMessageBox ("Create shortcut: SetPath failed!");
|
|
}
|
|
|
|
// Set the arguments of the shortcut.
|
|
CString args = " /H=\"" + m_WhereSource + "\" /B=\"" + m_WhereBuild + "\" /G=\"" + m_GeneratorDialog.m_GeneratorChoiceString + "\" /A=\"" + (m_AdvancedValues ? "TRUE" : "FALSE") + "\"";
|
|
|
|
hres = psl->SetArguments(args);
|
|
|
|
if (! SUCCEEDED (hres))
|
|
{
|
|
AfxMessageBox ("Create shortcut: SetArguments failed!");
|
|
}
|
|
|
|
// Set the description of the shortcut.
|
|
hres = psl->SetDescription("Shortcut to CMakeSetup");
|
|
|
|
if (! SUCCEEDED (hres))
|
|
{
|
|
AfxMessageBox ("Create shortcut: SetDescription failed!");
|
|
}
|
|
|
|
// Ensure that the string consists of ANSI characters.
|
|
WORD wszAr[MAX_PATH];
|
|
LPWSTR wsz = (LPWSTR)wszAr;
|
|
MultiByteToWideChar(CP_ACP, 0, link_name, -1, (LPWSTR)(wsz), MAX_PATH);
|
|
|
|
// Save the shortcut via the IPersistFile::Save member function.
|
|
hres = ppf->Save(wsz, TRUE);
|
|
|
|
if (! SUCCEEDED (hres))
|
|
{
|
|
AfxMessageBox ("Create shortcut: Save failed!");
|
|
}
|
|
|
|
// Release the pointer to IPersistFile.
|
|
ppf->Release ();
|
|
}
|
|
// Release the pointer to IShellLink.
|
|
psl->Release ();
|
|
|
|
return 0;
|
|
}
|
|
|
|
void CMakeSetupDialog::OnHelpButton()
|
|
{
|
|
CMakeHelp dialog;
|
|
dialog.DoModal();
|
|
}
|
|
|
|
void CMakeSetupDialog::OnDeleteButton()
|
|
{
|
|
std::string message = "Are you sure you want to delete the CMakeCache.txt file for:\n";
|
|
message += m_WhereBuild;
|
|
if(::MessageBox(0, message.c_str(), "Delete Cache?",
|
|
MB_YESNO|MB_TASKMODAL) == IDNO)
|
|
{
|
|
return;
|
|
}
|
|
m_GeneratorPicked = false;
|
|
|
|
if(m_WhereBuild != "" && this->m_CMakeInstance)
|
|
{
|
|
this->m_CMakeInstance->GetCacheManager()->DeleteCache(m_WhereBuild);
|
|
}
|
|
|
|
// Make sure we are working from the cache on disk
|
|
this->LoadCacheFromDiskToGUI();
|
|
|
|
m_OKButton.EnableWindow(false);
|
|
}
|
|
|
|
void CMakeSetupDialog::ShowAdvancedValues()
|
|
{
|
|
m_CacheEntriesList.ShowAdvanced();
|
|
}
|
|
|
|
void CMakeSetupDialog::RemoveAdvancedValues()
|
|
{
|
|
m_CacheEntriesList.HideAdvanced();
|
|
}
|
|
|
|
|
|
void CMakeSetupDialog::OnAdvancedValues()
|
|
{
|
|
this->UpdateData();
|
|
if(m_AdvancedValues)
|
|
{
|
|
this->ShowAdvancedValues();
|
|
}
|
|
else
|
|
{
|
|
this->RemoveAdvancedValues();
|
|
}
|
|
}
|
|
|
|
void CMakeSetupDialog::OnDoubleclickedAdvancedValues()
|
|
{
|
|
this->OnAdvancedValues();
|
|
}
|
|
|
|
// Handle param or single dropped file.
|
|
void CMakeSetupDialog::ChangeDirectoriesFromFile(const char* arg)
|
|
{
|
|
// Check if the argument refers to a CMakeCache.txt or
|
|
// CMakeLists.txt file.
|
|
std::string listPath;
|
|
std::string cachePath;
|
|
bool argIsFile = false;
|
|
if(cmSystemTools::FileIsDirectory(arg))
|
|
{
|
|
std::string path = cmSystemTools::CollapseFullPath(arg);
|
|
cmSystemTools::ConvertToUnixSlashes(path);
|
|
std::string cacheFile = path;
|
|
cacheFile += "/CMakeCache.txt";
|
|
std::string listFile = path;
|
|
listFile += "/CMakeLists.txt";
|
|
if(cmSystemTools::FileExists(cacheFile.c_str()))
|
|
{
|
|
cachePath = path;
|
|
}
|
|
if(cmSystemTools::FileExists(listFile.c_str()))
|
|
{
|
|
listPath = path;
|
|
}
|
|
}
|
|
else if(cmSystemTools::FileExists(arg))
|
|
{
|
|
argIsFile = true;
|
|
std::string fullPath = cmSystemTools::CollapseFullPath(arg);
|
|
std::string name = cmSystemTools::GetFilenameName(fullPath.c_str());
|
|
name = cmSystemTools::LowerCase(name);
|
|
if(name == "cmakecache.txt")
|
|
{
|
|
cachePath = cmSystemTools::GetFilenamePath(fullPath.c_str());
|
|
}
|
|
else if(name == "cmakelists.txt")
|
|
{
|
|
listPath = cmSystemTools::GetFilenamePath(fullPath.c_str());
|
|
}
|
|
}
|
|
|
|
// If there is a CMakeCache.txt file, use its settings.
|
|
if(cachePath.length() > 0)
|
|
{
|
|
cmCacheManager* cachem = m_CMakeInstance->GetCacheManager();
|
|
cmCacheManager::CacheIterator it = cachem->NewIterator();
|
|
if(cachem->LoadCache(cachePath.c_str()) && it.Find("CMAKE_HOME_DIRECTORY"))
|
|
{
|
|
std::string path = ConvertToWindowsPath(cachePath.c_str());
|
|
m_WhereBuild = path.c_str();
|
|
|
|
path = ConvertToWindowsPath(it.GetValue());
|
|
m_WhereSource = path.c_str();
|
|
|
|
m_GeneratorDialog.m_GeneratorChoiceString = _T("");
|
|
return;
|
|
}
|
|
}
|
|
|
|
// If there is a CMakeLists.txt file, use it as the source tree.
|
|
if(listPath.length() > 0)
|
|
{
|
|
std::string path = ConvertToWindowsPath(listPath.c_str());
|
|
m_WhereSource = path.c_str();
|
|
|
|
if(argIsFile)
|
|
{
|
|
// Source CMakeLists.txt file given. It was probably dropped
|
|
// onto the window or executable. Default to an in-source
|
|
// build.
|
|
m_WhereBuild = path.c_str();
|
|
}
|
|
else
|
|
{
|
|
// Source directory given on command line. Use current working
|
|
// directory as build tree.
|
|
std::string cwd = cmSystemTools::GetCurrentWorkingDirectory();
|
|
path = ConvertToWindowsPath(cwd.c_str());
|
|
m_WhereBuild = path.c_str();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// The framework calls this member function when the user releases the
|
|
// left mouse button over a window that has registered itself as the
|
|
// recipient of dropped files.
|
|
|
|
void CMakeSetupDialog::OnDropFiles(HDROP hDropInfo)
|
|
{
|
|
UINT nb_files = DragQueryFile(hDropInfo, 0xFFFFFFFF, NULL, 0);
|
|
if (nb_files > 0)
|
|
{
|
|
UINT buffer_size = DragQueryFile(hDropInfo, 0, NULL, 0);
|
|
char *buffer = new char [buffer_size + 1];
|
|
DragQueryFile(hDropInfo, 0, buffer, buffer_size + 1);
|
|
|
|
this->ChangeDirectoriesFromFile(buffer);
|
|
delete [] buffer;
|
|
|
|
this->m_WhereSourceControl.SetWindowText(this->m_WhereSource);
|
|
this->m_WhereBuildControl.SetWindowText(this->m_WhereBuild);
|
|
|
|
this->UpdateData(FALSE);
|
|
|
|
this->OnChangeWhereSource();
|
|
this->OnChangeWhereBuild();
|
|
}
|
|
|
|
DragFinish(hDropInfo);
|
|
}
|
|
|
|
BOOL CMakeSetupDialog::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
|
|
{
|
|
CDialog::OnSetCursor(pWnd, nHitTest, message);
|
|
if(m_Cursor == LoadCursor(NULL, IDC_WAIT))
|
|
{
|
|
::SetCursor(m_Cursor);
|
|
}
|
|
return true;
|
|
}
|