// 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 = new char[strlen(msg) + 40];
  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;
      }
    }
  delete [] tmp;
}

// 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 
  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;
}