Descent3/legacy/editor/TextureDialog.cpp
2024-05-23 23:07:26 -04:00

1004 lines
22 KiB
C++

/*
* Descent 3
* Copyright (C) 2024 Parallax Software
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
--- HISTORICAL COMMENTS FOLLOW ---
* $Logfile: /DescentIII/Main/editor/TextureDialog.cpp $
* $Revision: 1.1.1.1 $
* $Date: 2003-08-26 03:57:39 $
* $Author: kevinb $
*
* Texture Dialog keypad
*
* $Log: not supported by cvs2svn $
*
* 26 5/06/99 12:43p Matt
* Fixed a bug when did a texture grab on the terrain with no cells
* selected.
*
* 25 4/19/99 6:45p Jason
* fixed satellite grabbing bug
*
* 24 4/08/99 5:47p Jason
* added face finding function
*
* 23 2/23/99 3:07p Jason
* added grabbing of terrain textures
*
* 22 12/04/98 1:14p Jason
* added "Currently Used" texture flag
*
* 21 8/20/98 4:09p Jason
* added facemap feature
*
* 20 8/05/98 8:44p Jason
* added uniform scaling of textures
*
* 19 8/03/98 2:38p Matt
* Changed the replace texture function to replace all instances of a
* texture in the current room *or* in the whole mine, at the user's
* discretion.
*
* 18 6/23/98 2:43p Matt
* Changed calls to OutrageMessageBox() & Debug_MessageBox() to deal with
* int return value (instead of bool).
*
* 17 6/15/98 4:00p Jason
* replaced monochromatic polymodel lighting with rgb lighting
*
* 16 6/10/98 1:57p Samir
* fixed some listbox problems.
*
* 15 6/10/98 12:21p Matt
* Revamped system that controls what textures are displayed on the
* texture tab. Should work correctly now, and will now save the settings
* to the registry. Also, textures now default to no type set, not all
* types set.
*
* 14 5/20/98 1:08p Jason
* more stuff for adding bumpmaps
*
* 13 3/03/98 5:39p Jason
* added swatch to texture tab
*
* 12 11/14/97 12:41p Jason
* Added ReplaceTexture function
*
* 11 9/18/97 12:32p Samir
* If currently selected texture is not in the list, default to 0 texture.
*
* 10 9/17/97 1:01p Matt
* Ripped out segment code
*
* 9 9/17/97 12:06p Jason
* ripped out segment engine
*
* 8 9/16/97 6:08p Jason
* made texture names appear
*
* 7 9/15/97 1:18p Jason
* got selectable tmap2s working
*
* 6 9/12/97 3:47p Matt
* Fixed grab texture for room engine
*
* 5 8/29/97 1:52p Samir
* Added bitmap buttons for Texture Keypad.
*
* 4 8/25/97 7:39p Samir
* Using new GrListBox from old editorPictListBox
*
* 3 8/13/97 9:53a Matt
* Show selected texture with magenta box around it.
* Use new texture rotate & scale functions
*
* 15 6/03/97 4:55p Mark
*
* 15 6/3/97 4:14p Jeff
* Added Context Sensitive Help
*
* 14 5/28/97 6:15p Samir
* Fixups for missing textures in scrollbox.
*
* 13 5/27/97 5:29p Samir
* Took out mprintf.
*
* 12 5/27/97 3:11p Samir
* Allow 'masking' of textures to be displayed in listbox.
*
* 11 4/11/97 4:53p Samir
* Grab texture button works for tmap 1s
*
* 10 3/13/97 1:42p Matt
* Don't show texture 0 (the error texture) in the texture list
*
* 9 2/26/97 4:00p Samir
* Added most keypad functions.
*
* 8 2/21/97 6:25p Samir
* Set properly the max items for the listbox.
*
* 7 2/21/97 5:09p Samir
* Uses new PictListBox class for texture listbox here.
*
* 6 2/20/97 6:05p Samir
* Allow for clean activation and deactivation of surfaces in editor when
* going into fullscreen mode.
*
* 5 2/19/97 7:29p Samir
* Got scrolling texture view to work.
*
* 4 1/27/97 11:38a Samir
* Added horizontal scrolling of keypad modeless dialog.
*
* $NoKeywords: $
*/
#include "stdafx.h"
#include "editor.h"
#include "TextureDialog.h"
#include "HTexture.h"
#include "texture.h"
#include "gametexture.h"
#include "gr.h"
#include "descent.h"
#include "erooms.h"
#include "room.h"
#include "terrain.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/////////////////////////////////////////////////////////////////////////////
// CTextureDialog dialog
// Initialization and Destruction of the Dialog.
CTextureDialog::CTextureDialog(CWnd* pParent /*=NULL*/)
: CKeypadDialog(CTextureDialog::IDD, pParent)
{
//{{AFX_DATA_INIT(CTextureDialog)
// NOTE: the ClassWizard will add member initialization here
//}}AFX_DATA_INIT
}
void CTextureDialog::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
//{{AFX_DATA_MAP(CTextureDialog)
// NOTE: the ClassWizard will add DDX and DDV calls here
//}}AFX_DATA_MAP
}
void CTextureDialog::RunKeypadFunction(int code)
{
switch (code)
{
case VK_NUMPAD0: OnTexpadRotate90(); break;
case VK_NUMPAD1: OnTexpadRotLeft(); break;
case VK_NUMPAD2: OnTexpadSlideDown(); break;
case VK_NUMPAD3: OnTexpadRotRight(); break;
case VK_NUMPAD4: OnTexpadSlideLeft(); break;
case VK_NUMPAD5: OnTexpadSetDefault(); break;
case VK_NUMPAD6: OnTexpadSlideRight(); break;
case VK_NUMPAD7: OnTexpadStretchLess(); break;
case VK_NUMPAD8: OnTexpadSlideUp(); break;
case VK_NUMPAD9: OnTexpadStretchMore(); break;
case VK_DELETE: OnTexpadNextEdge(); break;
}
}
// selects a texture in the texture dialog
void CTextureDialog::SelectTexture(int texnum)
{
m_TexLB.SelectItem(texnum);
}
BEGIN_MESSAGE_MAP(CTextureDialog, CDialog)
//{{AFX_MSG_MAP(CTextureDialog)
ON_WM_KEYDOWN()
ON_WM_SIZE()
ON_WM_VSCROLL()
ON_WM_HSCROLL()
ON_WM_LBUTTONDOWN()
ON_WM_PAINT()
ON_BN_CLICKED(IDC_TEXPAD_FLIPX, OnTexpadFlipX)
ON_BN_CLICKED(IDC_TEXPAD_FLIPY, OnTexpadFlipY)
ON_BN_CLICKED(IDC_TEXPAD_SLIDELEFT, OnTexpadSlideLeft)
ON_BN_CLICKED(IDC_TEXPAD_SLIDERIGHT, OnTexpadSlideRight)
ON_BN_CLICKED(IDC_TEXPAD_ROTLEFT, OnTexpadRotLeft)
ON_BN_CLICKED(IDC_TEXPAD_ROTRIGHT, OnTexpadRotRight)
ON_BN_CLICKED(IDC_TEXPAD_ROTATE90, OnTexpadRotate90)
ON_BN_CLICKED(IDC_TEXPAD_STRETCHLESS, OnTexpadStretchLess)
ON_BN_CLICKED(IDC_TEXPAD_STRETCHMORE, OnTexpadStretchMore)
ON_BN_CLICKED(IDC_TEXPAD_SLIDEUP, OnTexpadSlideUp)
ON_BN_CLICKED(IDC_TEXPAD_SLIDEDOWN, OnTexpadSlideDown)
ON_BN_CLICKED(IDC_TEXPAD_SETDEFAULT, OnTexpadSetDefault)
ON_BN_CLICKED(IDC_TEXPAD_NEXTEDGE, OnTexpadNextEdge)
ON_BN_CLICKED(IDC_TEXPAD_GRAB, OnTexpadGrab)
ON_BN_CLICKED(IDC_SHOW_MINE_TEXTURES2, OnShowMineTextures2)
ON_BN_CLICKED(IDC_SHOW_OBJ_TEXTURES, OnShowObjTextures)
ON_BN_CLICKED(IDC_SHOW_LIGHT_TEXTURES, OnShowLightTextures)
ON_BN_CLICKED(IDC_SHOW_TERRAIN_TEXTURES, OnShowTerrainTextures)
ON_WM_HELPINFO()
ON_BN_CLICKED(IDC_RESET_ALL_TEXTURES, OnResetAllTextures)
ON_WM_DESTROY()
ON_BN_CLICKED(IDC_SHOW_TMAP2_TEXTURES, OnShowTmap2Textures)
ON_BN_CLICKED(IDC_REPLACE_TEXTURE, OnReplaceTexture)
ON_EN_KILLFOCUS(IDC_TEXSCALE_EDIT, OnKillfocusTexscaleEdit)
ON_BN_CLICKED(IDC_TEXPAD_EXPAND_U, OnTexpadExpandU)
ON_BN_CLICKED(IDC_TEXPAD_EXPAND_V, OnTexpadExpandV)
ON_BN_CLICKED(IDC_TEXPAD_CONTRACT_U, OnTexpadContractU)
ON_BN_CLICKED(IDC_TEXPAD_CONTRACT_V, OnTexpadContractV)
ON_BN_CLICKED(IDC_FACE_MAP, OnFaceMap)
ON_BN_CLICKED(IDC_CURRENT_USED, OnCurrentUsed)
ON_BN_CLICKED(IDC_FIND_TEXTURE, OnFindTexture)
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CTextureDialog message handlers
static CDialog *ParentDialog=NULL;
static CTextureDialog *ParentTextureDialog=NULL;
uint8_t CurrentlyUsedTextures[MAX_TEXTURES];
bool Show_used=0;
BOOL CTextureDialog::OnInitDialog()
{
CDialog::OnInitDialog();
RECT rect;
CWnd *wnd;
CEdit *ebox;
char str[20];
memset (CurrentlyUsedTextures,0,MAX_TEXTURES);
ParentDialog=this;
ParentTextureDialog=this;
wnd = GetDlgItem(IDC_TEXTURE_LISTBOX);
wnd->GetClientRect(&rect);
if (!GameTextures[D3EditState.texdlg_texture].used) D3EditState.texdlg_texture = 0;
m_TexLB.Create(wnd, IDC_TEXTURE_LISTBOX, rect, 36);
m_TexLB.SelectItem(D3EditState.texdlg_texture);
if (D3EditState.texture_display_flags & TF_TERRAIN)
((CButton *)GetDlgItem(IDC_SHOW_TERRAIN_TEXTURES))->SetCheck(1);
if (D3EditState.texture_display_flags & TF_MINE)
((CButton *)GetDlgItem(IDC_SHOW_MINE_TEXTURES2))->SetCheck(1);
if (D3EditState.texture_display_flags & TF_OBJECT)
((CButton *)GetDlgItem(IDC_SHOW_OBJ_TEXTURES))->SetCheck(1);
if (D3EditState.texture_display_flags & TF_TMAP2)
((CButton *)GetDlgItem(IDC_SHOW_TMAP2_TEXTURES))->SetCheck(1);
if (D3EditState.texture_display_flags & TF_LIGHT)
((CButton *)GetDlgItem(IDC_SHOW_LIGHT_TEXTURES))->SetCheck(1);
// Do bitmap buttons
m_bmUp.LoadBitmap(IDB_NORTH);
m_bmDown.LoadBitmap(IDB_SOUTH);
m_bmLeft.LoadBitmap(IDB_WEST);
m_bmRight.LoadBitmap(IDB_EAST);
m_bmRotLeft.LoadBitmap(IDB_ROTATELEFT);
m_bmRotRight.LoadBitmap(IDB_ROTATERIGHT);
m_bmStretchLess.LoadBitmap(IDB_INARROW);
m_bmStretchMore.LoadBitmap(IDB_OUTARROW);
// Set bitmap buttons
CButton *btn;
btn = (CButton*)GetDlgItem(IDC_TEXPAD_SLIDEUP);
btn->SetBitmap((HBITMAP)m_bmUp);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_SLIDEDOWN);
btn->SetBitmap((HBITMAP)m_bmDown);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_SLIDELEFT);
btn->SetBitmap((HBITMAP)m_bmLeft);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_SLIDERIGHT);
btn->SetBitmap((HBITMAP)m_bmRight);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_ROTLEFT);
btn->SetBitmap((HBITMAP)m_bmRotLeft);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_ROTRIGHT);
btn->SetBitmap((HBITMAP)m_bmRotRight);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_STRETCHLESS);
btn->SetBitmap((HBITMAP)m_bmStretchLess);
btn = (CButton*)GetDlgItem(IDC_TEXPAD_STRETCHMORE);
btn->SetBitmap((HBITMAP)m_bmStretchMore);
ebox=(CEdit *) GetDlgItem (IDC_TEXSCALE_EDIT);
sprintf (str,"%f",D3EditState.texscale);
ebox->SetWindowText (str);
return TRUE; // return TRUE unless you set the focus to a control
// EXCEPTION: OCX Property Pages should return FALSE
}
void CTextureDialog::OnDestroy()
{
CDialog::OnDestroy();
// kill all bitmaps
m_bmUp.DeleteObject();
m_bmDown.DeleteObject();
m_bmLeft.DeleteObject();
m_bmRight.DeleteObject();
m_bmRotLeft.DeleteObject();
m_bmRotRight.DeleteObject();
m_bmStretchLess.DeleteObject();
m_bmStretchMore.DeleteObject();
}
void CTextureDialog::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
{
// TODO: Add your message handler code here and/or call default
CDialog::OnKeyDown(nChar, nRepCnt, nFlags);
}
void CTextureDialog::OnSize(UINT nType, int cx, int cy)
{
CKeypadDialog::OnSize(nType, cx, cy);
}
void CTextureDialog::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
CKeypadDialog::OnVScroll(nSBCode, nPos, pScrollBar);
}
void CTextureDialog::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
// TODO: Add your message handler code here and/or call default
CKeypadDialog::OnHScroll(nSBCode, nPos, pScrollBar);
}
void CTextureDialog::OnLButtonDown(UINT nFlags, CPoint point)
{
CDialog::OnLButtonDown(nFlags, point);
}
void CTextureDialog::OnPaint()
{
CPaintDC dc(this); // device context for painting
if (!m_Active) return;
}
// Keypad Functions!!
void CTextureDialog::OnTexpadFlipX()
{
HTextureFlipX();
}
void CTextureDialog::OnTexpadFlipY()
{
HTextureFlipY();
}
void CTextureDialog::OnTexpadSlideLeft()
{
HTextureSlide(Curroomp, Curface, (float) -1.0*D3EditState.texscale, 0.0f);
}
void CTextureDialog::OnTexpadSlideRight()
{
HTextureSlide(Curroomp, Curface, (float) 1.0*D3EditState.texscale, 0.0f);
}
void CTextureDialog::OnTexpadSlideUp()
{
HTextureSlide(Curroomp, Curface, 0.0f, (float) 1.0*D3EditState.texscale);
}
void CTextureDialog::OnTexpadSlideDown()
{
HTextureSlide(Curroomp, Curface, 0.0f, (float) -1.0*D3EditState.texscale);
}
void CTextureDialog::OnTexpadRotLeft()
{
HTextureRotate(Curroomp, Curface, (-512*D3EditState.texscale));
}
void CTextureDialog::OnTexpadRotRight()
{
HTextureRotate(Curroomp, Curface, (512*D3EditState.texscale));
}
void CTextureDialog::OnTexpadRotate90()
{
HTextureRotate(Curroomp, Curface, 0x4000);
}
void CTextureDialog::OnTexpadStretchLess()
{
HTextureStretchLess(Curroomp, Curface, Curedge);
}
void CTextureDialog::OnTexpadStretchMore()
{
HTextureStretchMore(Curroomp, Curface, Curedge);
}
void CTextureDialog::OnTexpadSetDefault()
{
HTextureSetDefault(Curroomp);
}
void CTextureDialog::OnTexpadNextEdge()
{
HTextureNextEdge();
}
void CTextureDialog::OnTexpadGrab()
{
int texnum=-1;
if (Editor_view_mode==VM_TERRAIN)
{
for (int i=0;i<TERRAIN_WIDTH*TERRAIN_DEPTH;i++)
{
if (TerrainSelected[i])
{
if (texnum != -1) {
OutrageMessageBox("Cannot grab texture: More than one terrain cell selected.");
return;
}
texnum=Terrain_tex_seg[Terrain_seg[i].texseg_index].tex_index;
}
}
if (texnum == -1) {
OutrageMessageBox("Cannot grab texture: No terrain cell selected.");
return;
}
}
else
{
if (!Curroomp || Curroomp->used==0 || Curface>=Curroomp->num_faces)
return;
texnum = Curroomp->faces[Curface].tmap;
}
CTextureDialog::SelectTexture(texnum);
EditorStatus("Texture %d, \"%s\", selected.",texnum,GameTextures[texnum].name);
}
void CTextureDialog::OnShowMineTextures2()
{
D3EditState.texture_display_flags ^= TF_MINE;
Invalidate();
}
void CTextureDialog::OnShowObjTextures()
{
D3EditState.texture_display_flags ^= TF_OBJECT;
Invalidate();
}
void CTextureDialog::OnShowLightTextures()
{
D3EditState.texture_display_flags ^= TF_LIGHT;
Invalidate();
}
void CTextureDialog::OnShowTerrainTextures()
{
D3EditState.texture_display_flags ^= TF_TERRAIN;
Invalidate();
}
BOOL CTextureDialog::OnHelpInfo(HELPINFO* pHelpInfo)
{
// TODO: Add your message handler code here and/or call default
WinHelp(HID_TEXTURETAB,HELP_CONTEXT);
return TRUE;
//return CDialog::OnHelpInfo(pHelpInfo);
}
extern void AssignDefaultUVsToRoom(room *rp);
void CTextureDialog::OnResetAllTextures()
{
int answer=OutrageMessageBox (MBOX_YESNO,"Are you sure you wish to reset all textures for this room? This change cannot be undone!");
if (answer!=IDYES)
return;
AssignDefaultUVsToRoom (Curroomp);
World_changed=1;
}
/////////////////////////////////////////////////////////////////////
// CGrTextureListBox implementation
// enumerates items maximum in the list box. this is here to insure that any changes in
// the user-managed list.
int CGrTextureListBox::ListEnumerate()
{
int count = 0,i;
for (i=0;i<MAX_TEXTURES;i++)
{
bool countit=true;;
if (!GameTextures[i].used)
countit=false;
if (!(GameTextures[i].flags & D3EditState.texture_display_flags))
countit=false;
if (Show_used && !CurrentlyUsedTextures[i])
countit=false;
if (countit)
count++;
}
return count;
}
// Specify the first item to be drawn in the box by DrawItem. returns the item #
// return -1 if there is no first item (list is empty, maybe)
int CGrTextureListBox::ListFirstItem(int firsttex)
{
// skip the error texture
if (firsttex < 0) firsttex = 0;
if (firsttex == 0) firsttex++;
// find first texture that is used if the texture passed became invalid.
if (!GameTextures[firsttex].used || !(GameTextures[firsttex].flags & D3EditState.texture_display_flags) || (Show_used && !CurrentlyUsedTextures[firsttex])) {
int new_first = ListNextItem(firsttex);
if (new_first == firsttex)
return -1;
else
return new_first;
}
else
return firsttex;
}
// Specify the next/prev item to be drawn in the box by DrawItem.
// Returns the item # following or preceding the passed the item #
int CGrTextureListBox::ListNextItem(int curtex)
{
int next_texture = -1;
if (curtex >= 0)
next_texture = GetNextTexture(curtex);
if (next_texture <= curtex)
return curtex;
while ((Show_used && !CurrentlyUsedTextures[next_texture]) || !(GameTextures[next_texture].flags & D3EditState.texture_display_flags) )
{
next_texture = GetNextTexture(next_texture);
// mprintf((0, "next_tex = %d\n", next_texture));
if (next_texture == curtex) {
break;
}
}
return next_texture;
}
int CGrTextureListBox::ListPrevItem(int curtex)
{
int prev_texture = -1;
if (curtex >=0)
prev_texture = GetPreviousTexture(curtex);
if (prev_texture >= curtex)
return curtex;
while ((Show_used && !CurrentlyUsedTextures[prev_texture]) || !(GameTextures[prev_texture].flags & D3EditState.texture_display_flags) )
{
prev_texture = GetPreviousTexture(prev_texture);
// mprintf((0, "prev_tex = %d\n", prev_texture));
if (prev_texture == curtex) {
break;
}
}
return prev_texture;
}
// draw the item
void CGrTextureListBox::DrawItem(int x, int y, grHardwareSurface *itemsurf, int item)
{
int bm_handle;
bm_handle = GetTextureBitmap (item,0);
itemsurf->load(bm_handle);
Desktop_surf->blt(x, y, itemsurf);
}
// if item was selected from the list box, this function is invoked.
void CGrTextureListBox::OnItemSelected(int tex)
{
D3EditState.texdlg_texture = tex;
if (ParentDialog!=NULL)
{
PrintToDlgItem (ParentDialog,IDC_CURRENT_TEXTURE_NAME,"Texture: %s",GameTextures[tex].name);
ParentTextureDialog->DrawSwatch (IDC_TEXTURE_SWATCH,GameTextures[tex].r,GameTextures[tex].g,GameTextures[tex].b);
}
}
void CTextureDialog::OnShowTmap2Textures()
{
// TODO: Add your control notification handler code here
D3EditState.texture_display_flags ^= TF_TMAP2;
Invalidate();
}
//Returns the number of changes
int ReplaceTextureInRoom(room *rp,int old_tmap,int new_tmap)
{
int n_changes = 0;
for (int i=0;i<rp->num_faces;i++) {
if (rp->faces[i].tmap == old_tmap) {
rp->faces[i].tmap = new_tmap;
n_changes++;
}
}
return n_changes;
}
void CTextureDialog::OnReplaceTexture()
{
ASSERT (Curroomp!=NULL);
int n_changes;
int answer = OutrageMessageBox(MBOX_YESNOCANCEL,"Do you wish to change this texture for the whole mine?\n\n(Answer NO to change only the current room.)");
if (answer == IDCANCEL)
return;
int selected_texture=Curroomp->faces[Curface].tmap;
if (answer == IDYES) {
n_changes = 0;
for (int r=0;r<=Highest_room_index;r++)
if (Rooms[r].used)
n_changes += ReplaceTextureInRoom(&Rooms[r],selected_texture,D3EditState.texdlg_texture);
}
else {
n_changes = ReplaceTextureInRoom(Curroomp,selected_texture,D3EditState.texdlg_texture);
}
EditorStatus("Texture %d replaced with texture %d on %d faces.",selected_texture,D3EditState.texdlg_texture,n_changes);
World_changed=1;
}
void CTextureDialog::DrawSwatch(int handle,float r,float g,float b)
{
CWnd *texwnd;
RECT rect;
int w,h;
float rmax=max(r,g);
rmax=max(rmax,b);
if (rmax>=.001)
{
r/=rmax;
g/=rmax;
b/=rmax;
}
texwnd = GetDlgItem(handle);
texwnd->GetWindowRect(&rect);
ScreenToClient(&rect);
Desktop_surf->attach_to_window((unsigned)m_hWnd);
w=rect.right-rect.left;
h=rect.bottom-rect.top;
ddgr_color color=GR_RGB(r*255,g*255,b*255);
Desktop_surf->clear(rect.left,rect.top,w,h,color);
Desktop_surf->attach_to_window(NULL);
}
void CTextureDialog::OnKillfocusTexscaleEdit()
{
CEdit *ebox;
char str[20];
float val;
ebox=(CEdit *) GetDlgItem (IDC_TEXSCALE_EDIT);
ebox->GetWindowText (str,20);
val = atof (str);
if (val<0)
val=0;
if (val>64)
val=64;
D3EditState.texscale=val;
ebox=(CEdit *) GetDlgItem (IDC_TEXSCALE_EDIT);
sprintf (str,"%f",D3EditState.texscale);
ebox->SetWindowText (str);
}
void CTextureDialog::OnTexpadExpandU()
{
if (Curroomp==NULL)
return;
face *fp=&Curroomp->faces[Curface];
int i;
float mid=0;
for (i=0;i<fp->num_verts;i++)
mid+=fp->face_uvls[i].u;
mid/=fp->num_verts;
for (i=0;i<fp->num_verts;i++)
{
float diff=fp->face_uvls[i].u-mid;
diff*=1.0/D3EditState.texscale;
fp->face_uvls[i].u=mid+diff;
}
World_changed=1;
}
void CTextureDialog::OnTexpadExpandV()
{
if (Curroomp==NULL)
return;
face *fp=&Curroomp->faces[Curface];
int i;
float mid=0;
for (i=0;i<fp->num_verts;i++)
mid+=fp->face_uvls[i].v;
mid/=fp->num_verts;
for (i=0;i<fp->num_verts;i++)
{
float diff=fp->face_uvls[i].v-mid;
diff*=1.0/D3EditState.texscale;
fp->face_uvls[i].v=mid+diff;
}
World_changed=1;
}
void CTextureDialog::OnTexpadContractU()
{
if (Curroomp==NULL)
return;
face *fp=&Curroomp->faces[Curface];
int i;
float mid=0;
for (i=0;i<fp->num_verts;i++)
mid+=fp->face_uvls[i].u;
mid/=fp->num_verts;
for (i=0;i<fp->num_verts;i++)
{
float diff=fp->face_uvls[i].u-mid;
diff*=D3EditState.texscale;
fp->face_uvls[i].u=mid+diff;
}
World_changed=1;
}
void CTextureDialog::OnTexpadContractV()
{
if (Curroomp==NULL)
return;
face *fp=&Curroomp->faces[Curface];
int i;
float mid=0;
for (i=0;i<fp->num_verts;i++)
mid+=fp->face_uvls[i].v;
mid/=fp->num_verts;
for (i=0;i<fp->num_verts;i++)
{
float diff=fp->face_uvls[i].v-mid;
diff*=D3EditState.texscale;
fp->face_uvls[i].v=mid+diff;
}
World_changed=1;
}
void CTextureDialog::OnFaceMap()
{
if (Curroomp==NULL)
return;
face *fp=&Curroomp->faces[Curface];
if (fp->num_verts!=4)
{
OutrageMessageBox ("Face must have four vertices to use face mapping.");
return;
}
// Simply square off this face
fp->face_uvls[0].u=0.0;
fp->face_uvls[0].v=0.0;
fp->face_uvls[1].u=1.0;
fp->face_uvls[1].v=0.0;
fp->face_uvls[2].u=1.0;
fp->face_uvls[2].v=1.0;
fp->face_uvls[3].u=0.0;
fp->face_uvls[3].v=1.0;
World_changed=1;
}
void CTextureDialog::OnCurrentUsed()
{
// TODO: Add your control notification handler code here
int c=IsDlgButtonChecked(IDC_CURRENT_USED);
// If we just turned this on, build a list of textures used
if (c)
{
Show_used=1;
memset (CurrentlyUsedTextures,0,MAX_TEXTURES);
for (int i=0;i<=Highest_room_index;i++)
{
room *rp=&Rooms[i];
if (!rp->used)
continue;
for (int t=0;t<rp->num_faces;t++)
{
CurrentlyUsedTextures[rp->faces[t].tmap]=1;
}
}
}
else
Show_used=0;
Invalidate();
}
void CTextureDialog::OnFindTexture()
{
int i,t;
mprintf ((0,"Searching...please wait...\n"));
int found=0,found_room,found_face;
for (i=0;i<=Highest_room_index && !found;i++)
{
room *rp=&Rooms[i];
if (rp->used==0)
continue;
for (t=0;t<rp->num_faces && !found;t++)
{
if (rp->faces[t].tmap==D3EditState.texdlg_texture)
{
if (rp!=Curroomp || t!=Curface)
{
found_room=i;
found_face=t;
found=1;
}
}
}
}
mprintf ((0,"Done searching!\n"));
if (found==0)
{
mprintf ((0,"Couldn't find another face with that texture!\n"));
}
else
{
Curroomp=&Rooms[found_room];
Curface=found_face;
mprintf ((0,"Found that texture in room %d face %d.\n",found_room,found_face));
mprintf ((0,"Press C to center on that face\n"));
}
}