mirror of
https://github.com/leozide/leocad
synced 2025-01-03 23:03:55 +01:00
261 lines
6.9 KiB
C++
261 lines
6.9 KiB
C++
#include "lc_global.h"
|
|
#include "leocad.h"
|
|
#include "Terrain.h"
|
|
#include "TerrWnd.h"
|
|
#include "TerrDlg.h"
|
|
#include "TerOpDlg.h"
|
|
|
|
#ifdef _DEBUG
|
|
#define new DEBUG_NEW
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CTerrainDlg dialog
|
|
|
|
|
|
CTerrainDlg::CTerrainDlg(Terrain* pTerrain, bool bLinear, CWnd* pParent /*=NULL*/)
|
|
: CDialog(CTerrainDlg::IDD, pParent)
|
|
{
|
|
//{{AFX_DATA_INIT(CTerrainDlg)
|
|
//}}AFX_DATA_INIT
|
|
|
|
m_pTerrainWnd = NULL;
|
|
m_pTerrain = pTerrain;
|
|
m_bLinear = bLinear;
|
|
}
|
|
|
|
CTerrainDlg::~CTerrainDlg()
|
|
{
|
|
delete m_pTerrainWnd;
|
|
}
|
|
|
|
void CTerrainDlg::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CTerrainDlg)
|
|
//}}AFX_DATA_MAP
|
|
|
|
DDX_Control(pDX, IDC_GRID, m_Grid);
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CTerrainDlg, CDialog)
|
|
//{{AFX_MSG_MAP(CTerrainDlg)
|
|
ON_WM_SIZE()
|
|
ON_WM_SHOWWINDOW()
|
|
//}}AFX_MSG_MAP
|
|
ON_MESSAGE(WM_LC_EDIT_CLOSED, OnGridChange)
|
|
END_MESSAGE_MAP()
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CTerrainDlg message handlers
|
|
|
|
BOOL CTerrainDlg::OnInitDialog()
|
|
{
|
|
CDialog::OnInitDialog();
|
|
|
|
// Add the ToolBar.
|
|
if (!m_wndToolBar.Create(this) || !m_wndToolBar.LoadToolBar(IDR_TERRAIN))
|
|
{
|
|
TRACE0("Failed to create toolbar\n");
|
|
return -1; // fail to create
|
|
}
|
|
|
|
m_wndToolBar.SetBarStyle(m_wndToolBar.GetBarStyle() | CBRS_TOOLTIPS | CBRS_FLYBY);
|
|
|
|
// We need to resize the dialog to make room for control bars.
|
|
// First, figure out how big the control bars are.
|
|
CRect rcClientStart;
|
|
CRect rcClientNow;
|
|
GetClientRect(rcClientStart);
|
|
RepositionBars(AFX_IDW_CONTROLBAR_FIRST, AFX_IDW_CONTROLBAR_LAST, 0, reposQuery, rcClientNow);
|
|
|
|
// Now move all the controls so they are in the same relative
|
|
// position within the remaining client area as they would be
|
|
// with no control bars.
|
|
CPoint ptOffset(rcClientNow.left - rcClientStart.left, rcClientNow.top - rcClientStart.top);
|
|
|
|
// Adjust the dialog window dimensions
|
|
CRect rcWindow;
|
|
GetWindowRect(rcWindow);
|
|
rcWindow.right += rcClientStart.Width() - rcClientNow.Width();
|
|
rcWindow.bottom += rcClientStart.Height() - rcClientNow.Height();
|
|
MoveWindow(rcWindow, FALSE);
|
|
|
|
// And position the control bars
|
|
RepositionBars(AFX_IDW_CONTROLBAR_FIRST, AFX_IDW_CONTROLBAR_LAST, 0);
|
|
|
|
m_pTerrainWnd = new CTerrainWnd(m_pTerrain);
|
|
m_pTerrainWnd->Create (NULL, NULL, WS_BORDER | WS_CHILD | WS_VISIBLE, CRect (0,0,20,20), this, 501);
|
|
m_pTerrainWnd->LoadTexture(m_bLinear);
|
|
|
|
m_Grid.SetControlPoints(m_pTerrain->GetCountU(), m_pTerrain->GetCountV(), m_pTerrain->GetControlPoints());
|
|
|
|
PostMessage(WM_COMMAND, ID_TERDLG_EDIT_ZOOM);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
LRESULT CTerrainDlg::OnGridChange(WPARAM /*wParam*/, LPARAM /*lParam*/)
|
|
{
|
|
m_Grid.InvalidateRect(NULL, FALSE);
|
|
m_pTerrain->SetControlPoints();
|
|
m_pTerrain->Tesselate();
|
|
|
|
m_pTerrainWnd->InvalidateRect(NULL, FALSE);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL CTerrainDlg::OnCommand(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case ID_TERDLG_FILE_OPEN:
|
|
{
|
|
CFileDialog dlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
|
|
"LeoCAD Terrain Files (*.ltr)|*.ltr|All Files (*.*)|*.*||", this);
|
|
|
|
if (dlg.DoModal() == IDOK)
|
|
{
|
|
CWaitCursor wait;
|
|
CFile f (dlg.GetPathName(), CFile::modeRead | CFile::typeBinary | CFile::shareDenyWrite);
|
|
if (f.GetLength() != 0)
|
|
{
|
|
f.Seek(32, CFile::begin);
|
|
CArchive ar (&f, CArchive::load | CArchive::bNoFlushOnDelete);
|
|
Serialize(ar);
|
|
ar.Close();
|
|
}
|
|
f.Close();
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_FILE_SAVE:
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_FILE_SAVEAS:
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_EDIT_RANDOM:
|
|
{
|
|
m_pTerrain->GenerateRandom();
|
|
m_pTerrain->m_nOptions &= ~LC_TERRAIN_FLAT;
|
|
m_Grid.SetControlPoints(m_pTerrain->GetCountU(), m_pTerrain->GetCountV(), m_pTerrain->GetControlPoints());
|
|
m_pTerrainWnd->InvalidateRect(NULL, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_EDIT_PREFERENCES:
|
|
{
|
|
CTerrainOptionsDlg dlg;
|
|
dlg.SetOptions(m_pTerrain);
|
|
|
|
if (dlg.DoModal() == IDOK)
|
|
{
|
|
dlg.GetOptions(m_pTerrain);
|
|
m_Grid.SetControlPoints(m_pTerrain->GetCountU(), m_pTerrain->GetCountV(), m_pTerrain->GetControlPoints());
|
|
m_pTerrain->Tesselate();
|
|
m_pTerrainWnd->LoadTexture(m_bLinear);
|
|
|
|
m_pTerrainWnd->InvalidateRect(NULL, FALSE);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_EDIT_RESETCAMERA:
|
|
{
|
|
m_pTerrainWnd->ResetCamera();
|
|
m_pTerrainWnd->InvalidateRect(NULL, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_EDIT_ZOOM:
|
|
case ID_TERDLG_EDIT_PAN:
|
|
case ID_TERDLG_EDIT_ROTATE:
|
|
{
|
|
CheckControl(m_pTerrainWnd->m_nAction + ID_TERDLG_EDIT_ZOOM, FALSE);
|
|
m_pTerrainWnd->m_nAction = LOWORD(wParam) - ID_TERDLG_EDIT_ZOOM;
|
|
CheckControl(m_pTerrainWnd->m_nAction + ID_TERDLG_EDIT_ZOOM, TRUE);
|
|
|
|
GetMenu()->GetSubMenu(1)->CheckMenuRadioItem(ID_TERDLG_EDIT_ZOOM, ID_TERDLG_EDIT_ROTATE, m_pTerrainWnd->m_nAction + ID_TERDLG_EDIT_ZOOM, MF_BYCOMMAND);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
case ID_TERDLG_HELP:
|
|
{
|
|
MessageBox("The terrain is a Bezier surface divided in cubic (4x4) patches, "
|
|
"this means that the points along the black lines will always be part of the surface but the others probably won't.\n"
|
|
"For better results check the box \"Smooth\" or select a texture in the options dialog.\n"
|
|
"If you want to have an infinite plane (like in versions before 0.64), check the box \"Flat\".\n"
|
|
"The grid control has some minor drawing bugs, I'll fix them later. :)\n\n"
|
|
"I'll add something better to the help system later.", "Quick Help", MB_OK|MB_ICONINFORMATION);
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return CDialog::OnCommand(wParam, lParam);
|
|
}
|
|
|
|
void CTerrainDlg::CheckControl(UINT nID, BOOL bEnable)
|
|
{
|
|
// GetMenu()->GetSubMenu(1)->CheckMenuItem(nID, MF_BYCOMMAND | (bEnable ? MF_ENABLED : (MF_DISABLED | MF_GRAYED)));
|
|
int state = m_wndToolBar.GetToolBarCtrl().GetState(nID);
|
|
if (bEnable)
|
|
state |= TBSTATE_CHECKED;
|
|
else
|
|
state &= ~TBSTATE_CHECKED;
|
|
|
|
m_wndToolBar.GetToolBarCtrl().SetState(nID, state);
|
|
}
|
|
|
|
BOOL CTerrainDlg::PreTranslateMessage(MSG* pMsg)
|
|
{
|
|
if (pMsg->message == WM_KEYDOWN)
|
|
if ((pMsg->wParam == VK_ESCAPE) || (pMsg->wParam == VK_RETURN))
|
|
return TRUE;
|
|
|
|
return CDialog::PreTranslateMessage(pMsg);
|
|
}
|
|
|
|
void CTerrainDlg::OnSize(UINT nType, int cx, int cy)
|
|
{
|
|
CDialog::OnSize(nType, cx, cy);
|
|
|
|
// Fix the toolbar and menu.
|
|
RepositionBars(AFX_IDW_CONTROLBAR_FIRST, AFX_IDW_CONTROLBAR_LAST, 0);
|
|
|
|
if (m_pTerrainWnd)
|
|
{
|
|
CRect rcClient;
|
|
|
|
// Find out how much space is left.
|
|
RepositionBars(AFX_IDW_CONTROLBAR_FIRST, AFX_IDW_CONTROLBAR_LAST, 0, reposQuery, rcClient);
|
|
|
|
m_Grid.MoveWindow (rcClient.left + 5, rcClient.top + 5, rcClient.Width()/2 - 7, rcClient.Height() - 10);
|
|
m_pTerrainWnd->MoveWindow (rcClient.left + rcClient.Width()/2 + 2, rcClient.top + 5, rcClient.Width()/2 - 7, rcClient.Height() - 10);
|
|
}
|
|
}
|
|
|
|
void CTerrainDlg::OnShowWindow(BOOL bShow, UINT nStatus)
|
|
{
|
|
CDialog::OnShowWindow(bShow, nStatus);
|
|
|
|
if (bShow)
|
|
{
|
|
CRect rcWindow;
|
|
GetWindowRect(rcWindow);
|
|
OnSize(SIZE_RESTORED, rcWindow.Width(), rcWindow.Height());
|
|
}
|
|
}
|