source-engine/utils/FileSystemOpenDialog/FileSystemOpenDlg.cpp

998 lines
24 KiB
C++
Raw Permalink Normal View History

2020-04-22 16:56:21 +00:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//
//=============================================================================//
// FileSystemOpenDlg.cpp : implementation file
//
#include "stdafx.h"
#include "FileSystemOpenDlg.h"
#include "jpeglib/jpeglib.h"
#include "utldict.h"
#include "resource.h"
#include "ifilesystemopendialog.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
CFileInfo::CFileInfo()
{
m_pBitmap = NULL;
}
CFileInfo::~CFileInfo()
{
}
/////////////////////////////////////////////////////////////////////////////
// This caches the thumbnail bitmaps we generate to speed up browsing.
/////////////////////////////////////////////////////////////////////////////
class CBitmapCache
{
public:
CBitmapCache()
{
m_CurMemoryUsage = 0;
m_MaxMemoryUsage = 1024 * 1024 * 6;
}
void AddToCache( CBitmap *pBitmap, const char *pName, int memoryUsage, bool bLock )
{
Assert( m_Bitmaps.Find( pName ) == -1 );
m_CurMemoryUsage += memoryUsage;
CBitmapCacheEntry newEntry;
newEntry.m_pBitmap = pBitmap;
newEntry.m_MemoryUsage = memoryUsage;
newEntry.m_bLocked = bLock;
m_Bitmaps.Insert( pName, newEntry );
EnsureMemoryUsage();
}
CBitmap* Find( const char *pName )
{
int i = m_Bitmaps.Find( pName );
if ( i == -1 )
return NULL;
else
return m_Bitmaps[i].m_pBitmap;
}
void UnlockAll()
{
for ( int i=m_Bitmaps.First(); i != m_Bitmaps.InvalidIndex(); i=m_Bitmaps.Next( i ) )
{
m_Bitmaps[i].m_bLocked = false;
}
}
private:
void EnsureMemoryUsage()
{
while ( m_CurMemoryUsage > m_MaxMemoryUsage )
{
// Free something.
bool bFreed = false;
for ( int i=m_Bitmaps.First(); i != m_Bitmaps.InvalidIndex(); i=m_Bitmaps.Next( i ) )
{
if ( !m_Bitmaps[i].m_bLocked )
{
delete m_Bitmaps[i].m_pBitmap;
m_CurMemoryUsage -= m_Bitmaps[i].m_MemoryUsage;
m_Bitmaps.RemoveAt( i );
break;
}
}
// Nothing left to free?
if ( !bFreed )
return;
}
}
private:
class CBitmapCacheEntry
{
public:
CBitmap *m_pBitmap;
int m_MemoryUsage;
bool m_bLocked;
};
CUtlDict<CBitmapCacheEntry,int> m_Bitmaps;
int m_CurMemoryUsage;
int m_MaxMemoryUsage;
};
CBitmapCache g_BitmapCache;
/////////////////////////////////////////////////////////////////////////////
// CFileSystemOpenDlg dialog
CFileSystemOpenDlg::CFileSystemOpenDlg(CreateInterfaceFn factory, CWnd* pParent )
: CDialog(CFileSystemOpenDlg::IDD, pParent)
{
//{{AFX_DATA_INIT(CFileSystemOpenDlg)
//}}AFX_DATA_INIT
m_pFileSystem = (IFileSystem*)factory( FILESYSTEM_INTERFACE_VERSION, NULL );
if ( !m_pFileSystem )
{
Error( "Unable to connect to %s!\n", FILESYSTEM_INTERFACE_VERSION );
}
m_bFilterMdlAndJpgFiles = false;
}
void CFileSystemOpenDlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
//{{AFX_DATA_MAP(CFileSystemOpenDlg)
DDX_Control(pDX, IDC_FILENAME_LABEL, m_FilenameLabel);
DDX_Control(pDX, IDC_FILENAME, m_FilenameControl);
DDX_Control(pDX, IDC_LOOKIN, m_LookInLabel);
DDX_Control(pDX, IDC_FILE_LIST, m_FileList);
//}}AFX_DATA_MAP
}
BEGIN_MESSAGE_MAP(CFileSystemOpenDlg, CDialog)
//{{AFX_MSG_MAP(CFileSystemOpenDlg)
ON_WM_CREATE()
ON_WM_SIZE()
ON_NOTIFY(NM_DBLCLK, IDC_FILE_LIST, OnDblclkFileList)
ON_BN_CLICKED(IDC_UP_BUTTON, OnUpButton)
ON_NOTIFY(LVN_ITEMCHANGED, IDC_FILE_LIST, OnItemchangedFileList)
ON_WM_KEYDOWN()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CFileSystemOpenDlg message handlers
void CFileSystemOpenDlg::OnOK()
{
// Make sure it's a valid filename.
CString testFilename;
m_FilenameControl.GetWindowText( testFilename );
CString fullFilename = m_CurrentDir + "\\" + testFilename;
if ( m_pFileSystem->IsDirectory( fullFilename, GetPathID() ) )
{
m_CurrentDir = fullFilename;
PopulateListControl();
}
else if ( m_pFileSystem->FileExists( fullFilename, GetPathID() ) )
{
m_Filename = fullFilename;
// Translate .jpg to .mdl?
if ( m_bFilterMdlAndJpgFiles )
{
char tempFilename[MAX_PATH];
V_strcpy_safe( tempFilename, fullFilename );
char *pPos = strrchr( tempFilename, '.' );
if ( pPos )
{
if ( Q_stricmp( pPos, ".jpeg" ) == 0 || Q_stricmp( pPos, ".jpg" ) == 0 )
{
pPos[0] = 0;
V_strcat_safe( tempFilename, ".mdl" );
m_Filename = tempFilename;
}
}
}
EndDialog( IDOK );
}
else
{
// No file or directory here.
CString str;
str.FormatMessage( "File %1!s! doesn't exist.", (const char*)fullFilename );
AfxMessageBox( str, MB_OK );
}
}
void CFileSystemOpenDlg::SetInitialDir( const char *pDir, const char *pPathID )
{
m_CurrentDir = pDir;
if ( pPathID )
m_PathIDString = pPathID;
else
m_PathIDString = "";
}
CString CFileSystemOpenDlg::GetFilename() const
{
return m_Filename;
}
BOOL CFileSystemOpenDlg::OnInitDialog()
{
CDialog::OnInitDialog();
// Setup our anchor list.
AddAnchor( IDC_FILE_LIST, 2, 2 );
AddAnchor( IDC_FILE_LIST, 3, 3 );
AddAnchor( IDC_FILENAME, 1, 3 );
AddAnchor( IDC_FILENAME, 3, 3 );
AddAnchor( IDC_FILENAME, 2, 2 );
AddAnchor( IDC_FILENAME_LABEL, 0, 0 );
AddAnchor( IDC_FILENAME_LABEL, 2, 0 );
AddAnchor( IDC_FILENAME_LABEL, 1, 3 );
AddAnchor( IDC_FILENAME_LABEL, 3, 3 );
AddAnchor( IDOK, 0, 2 );
AddAnchor( IDOK, 2, 2 );
AddAnchor( IDOK, 1, 3 );
AddAnchor( IDOK, 3, 3 );
AddAnchor( IDCANCEL, 0, 2 );
AddAnchor( IDCANCEL, 2, 2 );
AddAnchor( IDCANCEL, 1, 3 );
AddAnchor( IDCANCEL, 3, 3 );
AddAnchor( IDC_LOOKIN, 2, 2 );
AddAnchor( IDC_UP_BUTTON, 0, 2 );
AddAnchor( IDC_UP_BUTTON, 2, 2 );
// Setup our image list.
m_ImageList.Create( PREVIEW_IMAGE_SIZE, PREVIEW_IMAGE_SIZE, ILC_COLOR32, 0, 512 );
m_BitmapFolder.LoadBitmap( IDB_LABEL_FOLDER );
m_iLabel_Folder = m_ImageList.Add( &m_BitmapFolder, (CBitmap*)NULL );
m_BitmapMdl.LoadBitmap( IDB_LABEL_MDL );
m_iLabel_Mdl = m_ImageList.Add( &m_BitmapMdl, (CBitmap*)NULL );
m_BitmapFile.LoadBitmap( IDB_LABEL_FILE );
m_iLabel_File = m_ImageList.Add( &m_BitmapFile, (CBitmap*)NULL );
m_FileList.SetImageList( &m_ImageList, LVSIL_NORMAL );
// Populate the list with the contents of our current directory.
PopulateListControl();
return TRUE; // return TRUE unless you set the focus to a control
// EXCEPTION: OCX Property Pages should return FALSE
}
void CFileSystemOpenDlg::GetEntries( const char *pMask, CUtlVector<CString> &entries, GetEntriesMode_t mode )
{
CString searchStr = m_CurrentDir + "\\" + pMask;
// Workaround Steam bug.
if ( searchStr == ".\\*.*" )
searchStr = "*.*";
FileFindHandle_t handle;
const char *pFile = m_pFileSystem->FindFirst( searchStr, &handle );
while ( pFile )
{
bool bIsDir = m_pFileSystem->FindIsDirectory( handle );
if ( (mode == GETENTRIES_DIRECTORIES_ONLY && bIsDir) || (mode == GETENTRIES_FILES_ONLY && !bIsDir) )
{
entries.AddToTail( pFile );
}
pFile = m_pFileSystem->FindNext( handle );
}
m_pFileSystem->FindClose( handle );
}
class CJpegSourceMgr : public jpeg_source_mgr
{
public:
CJpegSourceMgr()
{
this->init_source = &CJpegSourceMgr::imp_init_source;
this->fill_input_buffer = &CJpegSourceMgr::imp_fill_input_buffer;
this->skip_input_data = &CJpegSourceMgr::imp_skip_input_data;
this->resync_to_restart = &CJpegSourceMgr::imp_resync_to_restart;
this->term_source = &CJpegSourceMgr::imp_term_source;
this->next_input_byte = 0;
this->bytes_in_buffer = 0;
}
bool Init( IFileSystem *pFileSystem, FileHandle_t fp )
{
m_Data.SetSize( pFileSystem->Size( fp ) );
return pFileSystem->Read( m_Data.Base(), m_Data.Count(), fp ) == m_Data.Count();
}
static void imp_init_source(j_decompress_ptr cinfo)
{
}
static boolean imp_fill_input_buffer(j_decompress_ptr cinfo)
{
Assert( false ); // They should never need to call these functions since we give them all the data up front.
return 0;
}
static void imp_skip_input_data(j_decompress_ptr cinfo, long num_bytes)
{
AssertOnce( false ); // They should never need to call these functions since we give them all the data up front.
}
static boolean imp_resync_to_restart(j_decompress_ptr cinfo, int desired)
{
Assert( false ); // They should never need to call these functions since we give them all the data up front.
return false;
}
static void imp_term_source(j_decompress_ptr cinfo)
{
}
static void error_exit( j_common_ptr cptr )
{
CJpegSourceMgr *pInstance = (CJpegSourceMgr*)cptr->client_data;
longjmp( pInstance->m_JmpBuf, 1 );
}
public:
jmp_buf m_JmpBuf;
CUtlVector<char> m_Data;
};
bool ReadJpeg( IFileSystem *pFileSystem, const char *pFilename, CUtlVector<unsigned char> &buf, int &width, int &height, const char *pPathID )
{
width = height = 0;
// Read the data.
FileHandle_t fp = pFileSystem->Open( pFilename, "rb", pPathID );
if ( fp == FILESYSTEM_INVALID_HANDLE )
return false;
CJpegSourceMgr sourceMgr;
bool bRet = sourceMgr.Init( pFileSystem, fp );
pFileSystem->Close( fp );
if ( !bRet )
return false;
sourceMgr.bytes_in_buffer = sourceMgr.m_Data.Count();
sourceMgr.next_input_byte = (unsigned char*)sourceMgr.m_Data.Base();
// Load the jpeg.
struct jpeg_decompress_struct jpegInfo;
struct jpeg_error_mgr jerr;
memset( &jpegInfo, 0, sizeof( jpegInfo ) );
jpegInfo.client_data = &sourceMgr;
jpegInfo.err = jpeg_std_error(&jerr);
jerr.error_exit = &CJpegSourceMgr::error_exit;
jpeg_create_decompress(&jpegInfo);
jpegInfo.src = &sourceMgr;
if ( setjmp( sourceMgr.m_JmpBuf ) == 1 )
{
jpeg_destroy_decompress(&jpegInfo);
return false;
}
if (jpeg_read_header(&jpegInfo, TRUE) != JPEG_HEADER_OK)
{
return false;
}
// start the decompress with the jpeg engine.
if (jpeg_start_decompress(&jpegInfo) != TRUE || jpegInfo.output_components != 3)
{
jpeg_destroy_decompress(&jpegInfo);
return false;
}
// now that we've started the decompress with the jpeg lib, we have the attributes of the
// image ready to be read out of the decompress struct.
int row_stride = jpegInfo.output_width * jpegInfo.output_components;
int mem_required = jpegInfo.image_height * jpegInfo.image_width * jpegInfo.output_components;
JSAMPROW row_pointer[1];
int cur_row = 0;
width = jpegInfo.output_width;
height = jpegInfo.output_height;
// allocate the memory to read the image data into.
buf.SetSize( mem_required );
// read in all the scan lines of the image into our image data buffer.
bool working = true;
while (working && (jpegInfo.output_scanline < jpegInfo.output_height))
{
row_pointer[0] = &(buf[cur_row * row_stride]);
if (jpeg_read_scanlines(&jpegInfo, row_pointer, 1) != TRUE)
{
working = false;
}
++cur_row;
}
if (!working)
{
jpeg_destroy_decompress(&jpegInfo);
return false;
}
jpeg_finish_decompress(&jpegInfo);
return true;
}
void DownsampleRGBToRGBAImage(
CUtlVector<unsigned char> &srcData,
int srcWidth,
int srcHeight,
CUtlVector<unsigned char> &destData,
int destWidth,
int destHeight )
{
int srcPixelSize = 3;
int destPixelSize = 4;
destData.SetSize( destWidth * destHeight * destPixelSize );
memset( destData.Base(), 0xFF, destWidth * destHeight * destPixelSize );
// This preserves the aspect ratio of the image.
int scaledDestWidth = destWidth;
int scaledDestHeight = destHeight;
int destOffsetX = 0, destOffsetY = 0;
if ( srcWidth > srcHeight )
{
scaledDestHeight = (srcHeight * destHeight) / srcWidth;
destOffsetY = (destHeight - scaledDestHeight) / 2;
}
else if ( srcHeight > srcWidth )
{
scaledDestWidth = (srcWidth * destWidth) / srcHeight;
destOffsetX = (destWidth - scaledDestWidth) / 2;
}
for ( int destY=0; destY < scaledDestHeight; destY++ )
{
unsigned char *pDestLine = &destData[(destY + destOffsetY) * destWidth * destPixelSize + (destOffsetX * destPixelSize)];
unsigned char *pDestPos = pDestLine;
float destYPercent = (float)destY / (scaledDestHeight-1);
int srcY = (int)( destYPercent * (srcHeight-1) );
for ( int destX=0; destX < scaledDestWidth; destX++ )
{
float destXPercent = (float)destX / (scaledDestWidth-1);
int srcX = (int)( destXPercent * (srcWidth-1) );
unsigned char *pSrcPos = &srcData[(srcY * srcWidth + srcX) * srcPixelSize];
pDestPos[0] = pSrcPos[2];
pDestPos[1] = pSrcPos[1];
pDestPos[2] = pSrcPos[0];
pDestPos[3] = 255;
pDestPos += destPixelSize;
}
}
}
CBitmap* SetupJpegLabel( IFileSystem *pFileSystem, CString filename, int labelSize, const char *pPathID )
{
CBitmap *pBitmap = g_BitmapCache.Find( filename );
if ( pBitmap )
return pBitmap;
CUtlVector<unsigned char> data;
int width, height;
if ( !ReadJpeg( pFileSystem, filename, data, width, height, pPathID ) )
return NULL;
CUtlVector<unsigned char> downsampled;
DownsampleRGBToRGBAImage( data, width, height, downsampled, labelSize, labelSize );
pBitmap = new CBitmap;
if ( pBitmap->CreateBitmap( labelSize, labelSize, 1, 32, downsampled.Base() ) )
{
g_BitmapCache.AddToCache( pBitmap, filename, downsampled.Count(), true );
return pBitmap;
}
else
{
delete pBitmap;
return NULL;
}
}
int CFileSystemOpenDlg::SetupLabelImage( CFileInfo *pInfo, CString name, bool bIsDir )
{
if ( bIsDir )
return m_iLabel_Folder;
CString extension = name.Right( 4 );
extension.MakeLower();
if ( extension == ".jpg" || extension == ".jpeg" )
{
pInfo->m_pBitmap = SetupJpegLabel( m_pFileSystem, m_CurrentDir + "\\" + name, PREVIEW_IMAGE_SIZE, GetPathID() );
if ( pInfo->m_pBitmap )
return m_ImageList.Add( pInfo->m_pBitmap, (CBitmap*)NULL );
else
return m_iLabel_File;
}
else
{
return (extension == ".mdl") ? m_iLabel_Mdl : m_iLabel_File;
}
}
void FilterMdlAndJpgFiles( CUtlVector<CString> &files )
{
// Build a dictionary with all the .jpeg files.
CUtlDict<int,int> jpgFiles;
for ( int i=0; i < files.Count(); i++ )
{
CString extension = files[i].Right( 4 );
extension.MakeLower();
if ( extension == ".jpg" || extension == ".jpeg" )
{
CString base = files[i].Left( files[i].GetLength() - 4 );
jpgFiles.Insert( base, 1 );
}
}
// Now look for all mdls and remove them if they have a jpg.
for ( int i=0; i < files.Count(); i++ )
{
CString extension = files[i].Right( 4 );
extension.MakeLower();
if ( extension == ".mdl" )
{
CString base = files[i].Left( files[i].GetLength() - 4 );
if ( jpgFiles.Find( base ) != -1 )
{
files.Remove( i );
--i;
}
}
}
}
int CALLBACK FileListSortCallback( LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort )
{
CFileSystemOpenDlg *pDlg = (CFileSystemOpenDlg*)lParamSort;
CFileInfo *pInfo1 = &pDlg->m_FileInfos[lParam1];
CFileInfo *pInfo2 = &pDlg->m_FileInfos[lParam2];
if ( pInfo1->m_bIsDir != pInfo2->m_bIsDir )
return pInfo1->m_bIsDir ? -1 : 1;
return Q_stricmp( pInfo1->m_Name, pInfo2->m_Name );
}
void RemoveDuplicates( CUtlVector<CString> &files )
{
CUtlDict<int,int> uniqueFilenames;
for ( int i=0; i < files.Count(); i++ )
{
int iPreviousIndex = uniqueFilenames.Find( files[i] );
if ( iPreviousIndex == -1 )
{
uniqueFilenames.Insert( files[i], i );
}
else
{
files.Remove( i );
--i;
}
}
}
void CFileSystemOpenDlg::PopulateListControl()
{
m_FileList.DeleteAllItems();
g_BitmapCache.UnlockAll();
m_LookInLabel.SetWindowText( CString( "[ROOT]\\" ) + m_CurrentDir );
int iItem = 0;
// First add directories at the top.
CUtlVector<CString> directories;
GetEntries( "*.*", directories, GETENTRIES_DIRECTORIES_ONLY );
RemoveDuplicates( directories );
for ( int i=0; i < directories.Count(); i++ )
{
if ( directories[i] == "." || directories[i] == ".." )
continue;
LVITEM item;
item.mask = LVIF_TEXT | LVIF_IMAGE | LVIF_PARAM;
item.iItem = iItem++;
item.iSubItem = 0;
item.pszText = directories[i].GetBuffer(0);
item.lParam = m_FileInfos.AddToTail();
m_FileInfos[item.lParam].m_bIsDir = true;
m_FileInfos[item.lParam].m_Name = directories[i];
item.iImage = SetupLabelImage( &m_FileInfos[item.lParam], directories[i], true );
m_FileList.InsertItem( &item );
}
CUtlVector<CString> files;
for ( int iMask=0; iMask < m_FileMasks.Count(); iMask++ )
{
GetEntries( m_FileMasks[iMask], files, GETENTRIES_FILES_ONLY );
}
RemoveDuplicates( files );
if ( m_bFilterMdlAndJpgFiles )
FilterMdlAndJpgFiles( files );
for ( int i=0; i < files.Count(); i++ )
{
LVITEM item;
item.mask = LVIF_TEXT | LVIF_IMAGE | LVIF_PARAM;
item.iItem = iItem++;
item.iSubItem = 0;
item.iImage = m_iLabel_Mdl;
item.pszText = files[i].GetBuffer(0);
item.lParam = m_FileInfos.AddToTail();
m_FileInfos[item.lParam].m_bIsDir = false;
m_FileInfos[item.lParam].m_Name = files[i];
item.iImage = SetupLabelImage( &m_FileInfos[item.lParam], files[i], false );
m_FileList.InsertItem( &item );
}
m_FileList.SortItems( FileListSortCallback, (DWORD)this );
}
void CFileSystemOpenDlg::AddFileMask( const char *pMask )
{
m_FileMasks.AddToTail( pMask );
}
BOOL CFileSystemOpenDlg::Create(LPCTSTR lpszClassName, LPCTSTR lpszWindowName, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, CCreateContext* pContext)
{
return CDialog::Create(IDD, pParentWnd);
}
int CFileSystemOpenDlg::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CDialog::OnCreate(lpCreateStruct) == -1)
return -1;
return 0;
}
LONG& GetSideCoord( RECT &rect, int iSide )
{
if ( iSide == 0 )
return rect.left;
else if ( iSide == 1 )
return rect.top;
else if ( iSide == 2 )
return rect.right;
else
return rect.bottom;
}
LONG GetSideScreenCoord( CWnd *pWnd, int iSide )
{
RECT rect;
pWnd->GetWindowRect( &rect );
return GetSideCoord( rect, iSide );
}
void CFileSystemOpenDlg::ProcessAnchor( CWindowAnchor *pAnchor )
{
RECT rect, parentRect;
GetWindowRect( &parentRect );
pAnchor->m_pWnd->GetWindowRect( &rect );
GetSideCoord( rect, pAnchor->m_Side ) = GetSideCoord( parentRect, pAnchor->m_ParentSide ) + pAnchor->m_OriginalDist;
ScreenToClient( &rect );
pAnchor->m_pWnd->MoveWindow( &rect );
}
void CFileSystemOpenDlg::AddAnchor( int iDlgItem, int iSide, int iParentSide )
{
CWnd *pItem = GetDlgItem( iDlgItem );
if ( !pItem )
return;
CWindowAnchor *pAnchor = &m_Anchors[m_Anchors.AddToTail()];
pAnchor->m_pWnd = pItem;
pAnchor->m_Side = iSide;
pAnchor->m_ParentSide = iParentSide;
pAnchor->m_OriginalDist = GetSideScreenCoord( pItem, iSide ) - GetSideScreenCoord( this, iParentSide );
}
void CFileSystemOpenDlg::OnSize(UINT nType, int cx, int cy)
{
CDialog::OnSize(nType, cx, cy);
for ( int i=0; i < m_Anchors.Count(); i++ )
ProcessAnchor( &m_Anchors[i] );
if ( m_FileList.GetSafeHwnd() )
PopulateListControl();
}
void CFileSystemOpenDlg::OnDblclkFileList(NMHDR* pNMHDR, LRESULT* pResult)
{
/*int iSelected = m_FileList.GetNextItem( -1, LVNI_SELECTED );
if ( iSelected != -1 )
{
DWORD iItem = m_FileList.GetItemData( iSelected );
if ( iItem < (DWORD)m_FileInfos.Count() )
{
CFileInfo *pInfo = &m_FileInfos[iItem];
if ( pInfo->m_bIsDir )
{
m_CurrentDir += "\\" + m_FileInfos[iItem].m_Name;
PopulateListControl();
}
else
{
m_Filename = m_CurrentDir + "\\" + m_FileInfos[iItem].m_Name;
EndDialog( IDOK );
}
}
else
{
Assert( false );
}
}*/
OnOK();
*pResult = 0;
}
void CFileSystemOpenDlg::OnUpButton()
{
char str[MAX_PATH];
V_strcpy_safe( str, m_CurrentDir );
Q_StripLastDir( str, sizeof( str ) );
if ( str[0] == 0 )
V_strcpy_safe( str, "." );
if ( str[strlen(str)-1] == '\\' || str[strlen(str)-1] == '/' )
str[strlen(str)-1] = 0;
m_CurrentDir = str;
PopulateListControl();
}
void CFileSystemOpenDlg::OnItemchangedFileList(NMHDR* pNMHDR, LRESULT* pResult)
{
NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
DWORD iItem = m_FileList.GetItemData( pNMListView->iItem );
if ( iItem < (DWORD)m_FileInfos.Count() )
{
CFileInfo *pInfo = &m_FileInfos[iItem];
if ( (pNMListView->uChanged & LVIF_STATE) &&
(pNMListView->uNewState & LVIS_SELECTED) )
{
m_FilenameControl.SetWindowText( pInfo->m_Name );
}
}
*pResult = 0;
}
void CFileSystemOpenDlg::SetFilterMdlAndJpgFiles( bool bFilter )
{
m_bFilterMdlAndJpgFiles = bFilter;
}
const char* CFileSystemOpenDlg::GetPathID()
{
if ( m_PathIDString == "" )
return NULL;
else
return (const char*)m_PathIDString;
}
// ------------------------------------------------------------------------------------------------ //
// Implementation of IFileSystemOpenDialog.
// ------------------------------------------------------------------------------------------------ //
// IFileSystemOpenDialog implementation.
class CFileSystemOpenDialogWrapper : public IFileSystemOpenDialog
{
public:
CFileSystemOpenDialogWrapper()
{
m_pDialog = 0;
m_bLastModalWasWindowsDialog = false;
}
virtual void Release()
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
delete m_pDialog;
delete this;
}
// You must call this first to set the hwnd.
virtual void Init( CreateInterfaceFn factory, void *parentHwnd )
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( !m_pDialog );
m_hParentWnd = (HWND)parentHwnd;
m_pDialog = new CFileSystemOpenDlg( factory, CWnd::FromHandle( m_hParentWnd ) );
}
virtual void AddFileMask( const char *pMask )
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( m_pDialog );
m_pDialog->AddFileMask( pMask );
}
virtual void SetInitialDir( const char *pDir, const char *pPathID )
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( m_pDialog );
m_pDialog->SetInitialDir( pDir, pPathID );
}
virtual void SetFilterMdlAndJpgFiles( bool bFilter )
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( m_pDialog );
m_pDialog->SetFilterMdlAndJpgFiles( bFilter );
}
virtual void GetFilename( char *pOut, int outLen ) const
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( m_pDialog );
if ( m_bLastModalWasWindowsDialog )
{
Q_strncpy( pOut, m_RelativeFilename, outLen );
}
else
{
Q_strncpy( pOut, m_pDialog->GetFilename(), outLen );
}
}
virtual bool DoModal()
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( m_pDialog );
m_bLastModalWasWindowsDialog = false;
return m_pDialog->DoModal() == IDOK;
}
virtual bool DoModal_WindowsDialog()
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
Assert( m_pDialog );
m_bLastModalWasWindowsDialog = true;
// Get the full filename, then make sure it's a relative path.
char defExt[MAX_PATH] = {0};
if ( m_pDialog->m_FileMasks.Count() > 0 )
{
CString ext = m_pDialog->m_FileMasks[m_pDialog->m_FileMasks.Count()-1].Right( 4 );
const char *pStr = ext;
if ( pStr[0] == '.' )
V_strcpy_safe( defExt, pStr+1 );
}
char pFileNameBuf[MAX_PATH];
const char *pFileName = m_pDialog->m_pFileSystem->RelativePathToFullPath( m_pDialog->m_CurrentDir, m_pDialog->m_PathIDString, pFileNameBuf, MAX_PATH );
V_strcat_safe( pFileNameBuf, "\\" );
// Build the list of file filters.
char filters[1024];
if ( m_pDialog->m_FileMasks.Count() == 0 )
{
V_strcpy_safe( filters, "All Files (*.*)|*.*||" );
}
else
{
filters[0] = 0;
for ( int i=0; i < m_pDialog->m_FileMasks.Count(); i++ )
{
if ( i > 0 )
V_strcat_safe( filters, "|" );
V_strcat_safe( filters, m_pDialog->m_FileMasks[i] );
V_strcat_safe( filters, "|" );
V_strcat_safe( filters, m_pDialog->m_FileMasks[i] );
if ( pFileName )
{
V_strcat_safe( pFileNameBuf, m_pDialog->m_FileMasks[i] );
V_strcat_safe( pFileNameBuf, ";" );
}
}
V_strcat_safe( filters, "||" );
}
CFileDialog dlg(
true, // open dialog?
defExt[0]==0 ? NULL : defExt, // default file extension
pFileName, // initial filename
OFN_ENABLESIZING, // flags
filters,
CWnd::FromHandle( m_hParentWnd ) );
while ( dlg.DoModal() == IDOK )
{
// Make sure we can make this into a relative path.
if ( m_pDialog->m_pFileSystem->FullPathToRelativePath( dlg.GetPathName(), m_RelativeFilename, sizeof( m_RelativeFilename ) ) )
{
// Replace .jpg or .jpeg extension with .mdl?
char *pEnd = m_RelativeFilename;
while ( Q_stristr( pEnd+1, ".jpeg" ) || Q_stristr( pEnd+1, ".jpg" ) )
pEnd = max( Q_stristr( pEnd, ".jpeg" ), Q_stristr( pEnd, ".jpg" ) );
if ( pEnd && pEnd != m_RelativeFilename )
Q_strncpy( pEnd, ".mdl", sizeof( m_RelativeFilename ) - (pEnd - m_RelativeFilename) );
return true;
}
else
{
AfxMessageBox( IDS_NO_RELATIVE_PATH );
}
}
return false;
}
private:
CFileSystemOpenDlg *m_pDialog;
HWND m_hParentWnd;
char m_RelativeFilename[MAX_PATH];
bool m_bLastModalWasWindowsDialog;
};
EXPOSE_INTERFACE( CFileSystemOpenDialogWrapper, IFileSystemOpenDialog, FILESYSTEMOPENDIALOG_VERSION );