//
// Copyright 1997 Nicholas J. Irias.  All rights reserved.
//
//


#include "stdafx.h"
#include "errno.h"
//#define TRACE_HERE
#include "p4win.h"
#include "DepotTreeCtrl.h"
#include "resource.h"
#include "MainFrm.h"
#include "Historydlg.h"
#include "P4FileStats.h"
#include <process.h>
#include "viewerdlg.h"
#include "FileInfoDlg.h"
#include "FindFilesDlg.h"
#include "ForceSyncDlg.h"
#include "RemoveViewer.h"
#include "integdlg\FileSpecPage.h"
#include "integdlg\OptionsPage.h"
#include "MsgBox.h"
#include "Diff2Dlg.h"
#include "FilterDepotDlg.h"
#include "ImageList.h"
#include "SpecDescDlg.h"
#include "BookmarkAdd.h"

#include "cmd_add.h"
#include "cmd_changes.h"
#include "cmd_delete.h"
#include "cmd_diff.h"
#include "cmd_diff2.h"
#include "cmd_get.h"
#include "cmd_files.h"
#include "cmd_fstat.h"
#include "cmd_depots.h"
#include "cmd_dirstat.h"
#include "cmd_history.h"
#include "cmd_integrate2.h"
#include "cmd_listopstat.h"
#include "cmd_opened.h"
#include "cmd_maxchange.h"
#include "cmd_prepbrowse.h"
#include "cmd_prepedit.h"
#include "cmd_refresh.h"
#include "cmd_revert.h"
#include "cmd_where.h"
#include "strops.h"

// Flags for selection set adjustments durning a drag-drop operation
#define KEEP_SELECTION    0x00
#define DELETE_EXISTING_SELECTION 0x01
#define	ADD_TO_SELECTION  0x02

#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif



//		we use the new method of getting depot files in 98.2
//		use the old way for those lusers that have old servers
//
#define SERVER_IS_982_ORMORE (GET_SERVERLEVEL()>3)
#define NEW_DEPOT_LISTING  SERVER_IS_982_ORMORE

//		used for LPARAM of htree item.
//		the lparam of files is the item count
//		the lparam of subdirs is a negative number, showing
//		whether it is expanded or not.
//		since the code was full of if (item.lparam != -1 )
//		let's take the magic numbers out.
//

#define ITEM_IS_A_FILE_NOT_A_SUBDIR		(item.lParam>-1)
#define ITEM_IS_FILE(x) ((int) GetLParam(x) > -1)


//		to prevent littering the code with slashes
//
const CString g_sSlashes = "//";


//		we tag the directory names with a space so that they can be 
//		sorted, subdirs first, files after. use this const so we 
//		can find those blanks right away if we have to.
//
const CString g_sStupidLeadingBlank = " ";


//	Make sure the user knows why an empty folder of depot is visible.
//  Two possible causes are: 
//  1) only deleted files below, and 
//  2) its a depot that is empty or is outside the client view

/*const */CString g_TrulyEmptyDir;// = LoadStringResource(IDS_CONTAINS_NO_FILES_OR_FOLDERS);

void DepotContextMenu(CView * pDepotView)
{
	CPoint point;
	point.x = point.y = 0;
	((CDepotTreeCtrl *)pDepotView)->Call_OnContextMenu(NULL, point);
}

/////////////////////////////////////////////////////////////////////////////
// CDepotTreeCtrl

IMPLEMENT_DYNCREATE(CDepotTreeCtrl, CMultiSelTreeCtrl)

BEGIN_MESSAGE_MAP(CDepotTreeCtrl, CMultiSelTreeCtrl)
	ON_WM_CONTEXTMENU()
	ON_WM_CREATE()
	ON_WM_LBUTTONDBLCLK()
	ON_WM_LBUTTONDOWN()
	ON_WM_MOUSEMOVE()
	ON_WM_RBUTTONUP()
	ON_WM_RBUTTONDOWN()
	ON_WM_VSCROLL()
	ON_UPDATE_COMMAND_UI(ID_FILE_LOCK, OnUpdateFileLock)
	ON_UPDATE_COMMAND_UI(ID_FILE_UNLOCK, OnUpdateFileUnlock)
	ON_UPDATE_COMMAND_UI(ID_FILE_OPENDELETE, OnUpdateFileOpendelete)
	ON_UPDATE_COMMAND_UI(ID_FILE_OPENEDIT, OnUpdateFileOpenedit)
	ON_UPDATE_COMMAND_UI(ID_FILE_OPENEDITA, OnUpdateFileOpenedit)
	ON_UPDATE_COMMAND_UI(ID_FILE_REVERT, OnUpdateFileRevert)
	ON_UPDATE_COMMAND_UI(ID_FILE_GET, OnUpdateFileGet)
	ON_UPDATE_COMMAND_UI(ID_FILE_GETWHATIF, OnUpdateGetwhatif)
	ON_UPDATE_COMMAND_UI(ID_FILE_REMOVE, OnUpdateFileRemove)
	ON_UPDATE_COMMAND_UI(ID_FILE_ANNOTATIONS, OnUpdateFileTimeLapse)
	ON_UPDATE_COMMAND_UI(ID_FILE_ANNOTATE, OnUpdateFileAnnotate)
	ON_UPDATE_COMMAND_UI(ID_FILE_ANNOTATEALL, OnUpdateFileAnnotate)
	ON_UPDATE_COMMAND_UI(ID_FILE_ANNOTATECHG, OnUpdateFileAnnotate)
	ON_UPDATE_COMMAND_UI(ID_FILE_ANNOTATECHGALL, OnUpdateFileAnnotate)
	ON_UPDATE_COMMAND_UI(ID_FILE_REVISIONHISTORY, OnUpdateFileRevisionhistory)
	ON_UPDATE_COMMAND_UI(ID_FILE_REVISIONTREE, OnUpdateFileRevisiontree)
	ON_UPDATE_COMMAND_UI(ID_FILE_DIFFHEAD, OnUpdateFileDiffhead)
	ON_UPDATE_COMMAND_UI(ID_FILE_DIFF2, OnUpdateFileDiff2)
	ON_UPDATE_COMMAND_UI(ID_POSITIONTOPATTERN, OnUpdatePositionDepot)
	ON_UPDATE_COMMAND_UI(ID_FINDNEXT, OnUpdatePositionDepotNext)
	ON_UPDATE_COMMAND_UI(ID_FINDPREV, OnUpdatePositionDepotNext)
	ON_UPDATE_COMMAND_UI(ID_POSITIONCHGS, OnUpdatePositionChgs)
	ON_UPDATE_COMMAND_UI(ID_WINEXPLORE, OnUpdateWinExplore)
	ON_UPDATE_COMMAND_UI(ID_CMDPROMPT, OnUpdateCmdPrompt)
	ON_COMMAND(ID_FILE_LOCK, OnFileLock)
	ON_COMMAND(ID_FILE_UNLOCK, OnFileUnlock)
	ON_COMMAND(ID_FILE_OPENEDIT, OnFileOpenedit)
	ON_COMMAND(ID_FILE_OPENEDITA, OnFileOpenedit)
	ON_COMMAND(ID_FILE_OPENDELETE, OnFileOpendelete)
	ON_COMMAND(ID_FILE_REVERT, OnFileRevert)
	ON_COMMAND(ID_FILE_GET, OnFileGet)
	ON_COMMAND(ID_FILE_GETWHATIF, OnFileGetwhatif)
	ON_COMMAND(ID_FILE_REMOVE, OnFileRemove)
	ON_COMMAND(ID_FILE_ANNOTATIONS, OnFileTimeLapseView)
	ON_COMMAND(ID_FILE_ANNOTATE, OnFileAnnotate)
	ON_COMMAND(ID_FILE_ANNOTATEALL, OnFileAnnotateAll)
	ON_COMMAND(ID_FILE_ANNOTATECHG, OnFileAnnotateChg)
	ON_COMMAND(ID_FILE_ANNOTATECHGALL, OnFileAnnotateChgAll)
	ON_COMMAND(ID_FILE_REVISIONHISTORY, OnFileRevisionhistory)
	ON_COMMAND(ID_FILE_REVISIONTREE, OnFileRevisionTree)
	ON_COMMAND(ID_POSITIONTOPATTERN, OnPositionDepot)
	ON_COMMAND(ID_FINDNEXT, OnPositionDepotNext)
	ON_COMMAND(ID_FINDPREV, OnPositionDepotPrev)
	ON_COMMAND(ID_POSITIONCHGS, OnPositionChgs)
	ON_COMMAND(ID_WINEXPLORE, OnWinExplore)
	ON_COMMAND(ID_CMDPROMPT, OnCmdPrompt)
	ON_UPDATE_COMMAND_UI(ID_FILE_GETCUSTOM, OnUpdateFileGetcustom)
	ON_COMMAND(ID_FILE_GETCUSTOM, OnFileGetcustom)
	ON_UPDATE_COMMAND_UI(ID_FILE_REFRESH, OnUpdateFileRefresh)
	ON_COMMAND(ID_FILE_REFRESH, OnFileRefresh)
	ON_UPDATE_COMMAND_UI(ID_FILE_FORCESYNCTOHEAD, OnUpdateFileForceToHead)
	ON_COMMAND(ID_FILE_FORCESYNCTOHEAD, OnFileForceToHead)
	ON_COMMAND(ID_FILE_DIFFHEAD, OnFileDiffhead)
	ON_COMMAND(ID_FILE_DIFF2, OnFileDiff2)
	ON_UPDATE_COMMAND_UI(ID_FILE_PROPERTIES, OnUpdateFileInformation)
	ON_COMMAND(ID_FILE_PROPERTIES, OnFileInformation)
	ON_UPDATE_COMMAND_UI(ID_FILEDROP_EDIT, OnUpdateFiledropEdit)
	ON_COMMAND(ID_FILEDROP_EDIT, OnFiledropEdit)
	ON_UPDATE_COMMAND_UI(ID_FILEDROP_DELETE, OnUpdateFiledropDelete)
	ON_COMMAND(ID_FILEDROP_DELETE, OnFiledropDelete)
	ON_UPDATE_COMMAND_UI(ID_FILEDROP_CANCEL, OnUpdateFiledropCancel)
	ON_COMMAND(ID_FILEDROP_CANCEL, OnFiledropCancel)
	ON_UPDATE_COMMAND_UI(ID_FILE_AUTOEDIT, OnUpdateFileAutoedit)
	ON_COMMAND(ID_FILE_AUTOEDIT, OnFileAutoedit)
	ON_COMMAND(ID_FILE_QUICKEDIT, OnFileQuickedit)
	ON_UPDATE_COMMAND_UI(ID_FILE_AUTOBROWSE, OnUpdateFileAutobrowse)
	ON_COMMAND(ID_FILE_AUTOBROWSE, OnFileAutobrowse)
	ON_COMMAND(ID_FILE_QUICKBROWSE, OnFileQuickbrowse)
	ON_UPDATE_COMMAND_UI(ID_JOB_SETFILEFILTER, OnUpdateFilterJobview)
	ON_COMMAND(ID_JOB_SETFILEFILTER, OnFilterJobview)
	ON_UPDATE_COMMAND_UI(ID_JOB_SETFILEFILTERINTEG, OnUpdateFilterJobview)
	ON_COMMAND(ID_JOB_SETFILEFILTERINTEG, OnFilterJobviewInteg)
	ON_UPDATE_COMMAND_UI(ID_FILE_RENAME, OnUpdateFileRename)
	ON_COMMAND(ID_FILE_RENAME, OnFileRename)
	ON_UPDATE_COMMAND_UI(ID_CHANGE_SUBMIT, OnUpdateFileSubmit)
	ON_COMMAND(ID_CHANGE_SUBMIT, OnFileSubmit)
	ON_UPDATE_COMMAND_UI(ID_FILE_INTEGRATE, OnUpdateFileIntegrate)
	ON_COMMAND(ID_FILE_INTEGRATE, OnFileIntegrate)
	ON_COMMAND(ID_FILE_INTEGSPEC, OnFileIntegspec)
	ON_COMMAND(ID_VIEW_UPDATE_LEFT, OnViewUpdate)
	ON_UPDATE_COMMAND_UI(ID_FILE_RECOVER, OnUpdateFileRecover)
	ON_COMMAND(ID_FILE_RECOVER, OnFileRecover)
	ON_UPDATE_COMMAND_UI(ID_EDIT_SELECT_ALL, OnUpdateEditSelectAll)
	ON_COMMAND(ID_EDIT_SELECT_ALL, OnEditSelectAll)
	ON_UPDATE_COMMAND_UI(ID_EDIT_COPY, OnUpdateEditCopy)
	ON_COMMAND(ID_EDIT_COPY, OnEditCopy)
	ON_UPDATE_COMMAND_UI(ID_EDIT_COPYCLIENTPATH, OnUpdateEditCopyclientpath)
	ON_COMMAND(ID_EDIT_COPYCLIENTPATH, OnEditCopyclientpath)
	ON_COMMAND(ID_DIFFSDSE, OnDiff_sd_se)
	ON_UPDATE_COMMAND_UI(ID_DIFFSDSE, OnUpdateDiff_sd_se)
	ON_UPDATE_COMMAND_UI(ID_ADD_TOVIEW, OnUpdateAddToClientView)
	ON_COMMAND(ID_ADD_TOVIEW, OnAddToClientView)
	ON_UPDATE_COMMAND_UI(ID_ADD_REVIEWS, OnUpdateAddReviews)
	ON_COMMAND(ID_ADD_REVIEWS, OnAddReviews)
	ON_UPDATE_COMMAND_UI(ID_FILE_RMVBROWSER, OnUpdateRemoveViewer)
	ON_COMMAND(ID_FILE_RMVBROWSER, OnRemoveViewer)
	ON_UPDATE_COMMAND_UI(ID_FILE_RMVEDITOR, OnUpdateRemoveViewer)
	ON_COMMAND(ID_FILE_RMVEDITOR, OnRemoveViewer)
	ON_UPDATE_COMMAND_UI(ID_FILE_QUICKEDIT, OnUpdateFileAutoedit)
	ON_UPDATE_COMMAND_UI(ID_FILE_QUICKBROWSE, OnUpdateFileAutobrowse)
	ON_UPDATE_COMMAND_UI(ID_FILE_INTEGSPEC, OnUpdateFileIntegrate)
	ON_UPDATE_COMMAND_UI(ID_FINDFILEUNDERFOLDER, OnUpdateFindFileUnderFolder)
	ON_COMMAND(ID_FINDFILEUNDERFOLDER, OnFindFileUnderFolder)
	ON_UPDATE_COMMAND_UI(ID_ADD_BOOKMARK, OnUpdateAddBookmark)
	ON_COMMAND(ID_ADD_BOOKMARK, OnAddBookmark)
	ON_UPDATE_COMMAND_UI(ID_FILE_DELETE, OnUpdateFileDelete)
	ON_COMMAND(ID_FILE_DELETE, OnFileDelete)
	ON_UPDATE_COMMAND_UI(ID_FILE_ADD, OnUpdateFileAdd)
	ON_COMMAND(ID_FILE_ADD, OnFileAdd)
	ON_COMMAND_RANGE(ID_FILE_BROWSER_1, ID_FILE_BROWSER_1+MAX_MRU_VIEWERS-1, OnFileMRUBrowse)
	ON_COMMAND_RANGE(ID_FILE_EDITOR_1, ID_FILE_EDITOR_1+MAX_MRU_VIEWERS-1, OnFileMRUEditor)
	ON_COMMAND(ID_FILE_NEWBROWSER, OnFileNewBrowser)
	ON_COMMAND(ID_FILE_NEWEDITOR, OnFileNewEditor)
	ON_COMMAND(ID_SHOWDELETED, OnShowDeletedFiles )
	ON_UPDATE_COMMAND_UI(ID_VIEW_UPDATE_LEFT, OnUpdateViewUpdate)
	ON_UPDATE_COMMAND_UI(ID_VIEW_FILTEREDVIEW, OnUpdateViewFilteredview)
	ON_COMMAND(ID_VIEW_FILTEREDVIEW, OnViewFilteredview)
	ON_UPDATE_COMMAND_UI(ID_VIEW_CLEARFILTER, OnUpdateViewClearfilter)
	ON_COMMAND(ID_VIEW_CLEARFILTER, OnViewClearfilter)
	ON_UPDATE_COMMAND_UI(ID_FILTER_SETVIEW, OnUpdateFilterSetview)
	ON_COMMAND(ID_FILTER_SETVIEW, OnFilterSetview)
	ON_UPDATE_COMMAND_UI(ID_FILTER_CLEARVIEW, OnUpdateFilterClearview)
	ON_COMMAND(ID_FILTER_CLEARVIEW, OnFilterClearview)
	ON_UPDATE_COMMAND_UI(ID_LABELFILTER_SETVIEW, OnUpdateLabelFilterSetview)
	ON_COMMAND(ID_LABELFILTER_SETVIEW, OnLabelFilterSetview)
	ON_UPDATE_COMMAND_UI(ID_LABELFILTER_SETVIEWREV, OnUpdateLabelFilterSetviewRev)
	ON_COMMAND(ID_LABELFILTER_SETVIEWREV, OnLabelFilterSetviewRev)
	ON_UPDATE_COMMAND_UI(ID_LABELFILTER_CLEARVIEW, OnUpdateLabelFilterClearview)
	ON_COMMAND(ID_LABELFILTER_CLEARVIEW, OnLabelFilterClearview)
	ON_COMMAND(ID_PERFORCE_OPTIONS, OnPerforceOptions)
	ON_MESSAGE(WM_FILEEDITTXT, OnEditFileTxt )
	ON_MESSAGE(WM_FILEEDITBIN, OnEditFileBin )
	ON_MESSAGE(WM_FILEBROWSETXT, OnBrowseFileTxt )  
	ON_MESSAGE(WM_FILEBROWSEBIN, OnBrowseFileBin )  
	ON_MESSAGE(WM_P4LISTOPSTAT, OnP4ListOp )
	ON_MESSAGE(WM_P4DIFF, OnP4Diff )
	ON_MESSAGE(WM_P4ERROR, OnP4Error )
	ON_MESSAGE(WM_P4FSTAT, OnP4FStat )
    ON_MESSAGE(WM_P4DIRSTAT, OnP4DirStat )
	ON_MESSAGE(WM_P4EXPANDSUBDIR, OnP4ExpandTree )
    ON_MESSAGE(WM_P4DEPOTS, OnP4Depots )
	ON_MESSAGE(WM_P4INTEGRATE2, OnP4Integ )
	ON_MESSAGE(WM_P4PREPBROWSE, OnP4PrepBrowse )
	ON_MESSAGE(WM_P4PREPEDIT, OnP4PrepEdit )
	ON_MESSAGE(WM_P4HISTORY, OnP4History )
	ON_MESSAGE(WM_P4ENDHISTORY, OnP4EndHistory )
	ON_MESSAGE(WM_P4GET, OnP4Get )
	ON_MESSAGE(WM_P4REFRESH, OnP4Refresh )
	ON_MESSAGE(WM_P4ADD, OnP4Add )
	ON_MESSAGE(WM_P4RECOVER, OnP4Recover )
	ON_MESSAGE(WM_UPDATEOPEN, OnP4UpdateOpen )
	ON_MESSAGE(WM_GETSELCOUNT, OnGetSelectedCount )
	ON_MESSAGE(WM_GETSELLIST, OnGetSelectedList )
	ON_MESSAGE(WM_BRANCHINTEG, OnBranchIntegrate )
	ON_MESSAGE(WM_CHANGELISTINTEG, OnChangelistIntegrate )
	ON_MESSAGE(WM_DOINTEGRATE1, OnIntegrate1 )
	ON_MESSAGE(WM_DOCUSTOMGET, OnDoGetCustom )
	ON_MESSAGE(WM_P4FILEINFORMATION, OnP4FileInformation )
	ON_MESSAGE(WM_P4ENDFILEINFORMATION, OnP4EndFileInformation )
	ON_MESSAGE(WM_P4FILES, OnP4Files )
	ON_MESSAGE(WM_P4OPENED, OnP4Opened )
	ON_MESSAGE(WM_DROPTARGET, OnDropTarget)
	ON_MESSAGE(WM_VIEWHEAD, OnViewHead )
	ON_MESSAGE(WM_ISFILTEREDONOPEN, IsFilteredOnOpen )
	ON_MESSAGE(WM_REDOOPENEDFILTER, OnRedoOpendList )
	ON_MESSAGE(WM_SETADDFSTATS, OnSetAddFstats )
	ON_MESSAGE(WM_GETADDFSTATS, OnGetAddFstats )
	ON_MESSAGE(WM_P4DIFF2, OnP4Diff2 )
	ON_MESSAGE(WM_P4ENDDESCRIBE, OnP4EndDiff2 )
	ON_MESSAGE(WM_P4ENDDIFF2OUTPUT, OnP4EndDiff2Output )
	ON_MESSAGE(WM_P4ENDFINDFILES, OnEndPositionDepot )
	ON_MESSAGE(WM_P4FILEREVERT, OnP4FileRevert )
	ON_MESSAGE(WM_ADDBOOKMARK, OnAddBookmarkMsg )
	ON_MESSAGE(WM_BROWSECALLBACK1, OnIntegBranchBrowseCallBack)
	ON_MESSAGE(WM_SETVIEWER, OnSetViewer)
END_MESSAGE_MAP()

CDepotTreeCtrl::CDepotTreeCtrl()
{
	m_OLESource.SetTreeCtrl(this);
	m_DepotIsDropTarget = FALSE;
	m_DragDropCtr= 0;

	g_TrulyEmptyDir = LoadStringResource(IDS_CONTAINS_NO_FILES_OR_FOLDERS);

	m_CF_DELTA=RegisterClipboardFormat(LoadStringResource(IDS_DRAGFROMDELTA));
	m_CF_DEPOT=RegisterClipboardFormat(LoadStringResource(IDS_DRAGFROMDEPOT));

	m_Need2Filter = FALSE;
	m_ExpandDepotContinue = m_Add2ExpandItemList = m_JustExpanded = FALSE;
	m_DropTargetFlag = 0;

	m_CustomGetDlg = 0;

	m_FilterDepot = m_P4Files_Deselect = FALSE;
	m_DepotFilterType = -1;
	m_DepotFilterPort = m_DepotFilterClient = "";

	m_ContextPoint.x = m_ContextPoint.y = -1;
	m_InContextMenu = FALSE;
	m_SkipSyncDialog = FALSE;
}

/////////////////////////////////////////////////////////////////////////////
// CDepotTreeCtrl diagnostics

#ifdef _DEBUG
void CDepotTreeCtrl::AssertValid() const
{
	CMultiSelTreeCtrl::AssertValid();
}

void CDepotTreeCtrl::Dump(CDumpContext& dc) const
{
	CMultiSelTreeCtrl::Dump(dc);
}
#endif //_DEBUG

/*
	_________________________________________________________________

	wipe out entire depot tree and start fresh with the depot at the root.
	_________________________________________________________________
*/

void CDepotTreeCtrl::Clear()
{
	XTRACE(_T("Clear()\n"));

    SetRedraw(FALSE);
	DeleteAllItems( );
    SetRedraw(TRUE);
    m_ItemCount = m_DepotCount = 0;
	m_FSColl.DestroyAll( );

	// We also need to empty out the Fstats info
	// for files opened for add
	Empty_FstatsAdds();

	m_LastPath = g_sSlashes;
	m_Root = m_LastPathItem = TVI_ROOT;
	SetAdd2ExpandItemList(FALSE);
}


/////////////////////////////////////////////////////////////////////////////
// CDepotTreeCtrl tree control access functions

// This is a recursive function!
// It searches a whole tree looking for a file with the name given in 'lookingfor'
// Traversing a file tree just begs for a fully resursive routine. 
// But if there are too many files, the stack overflows and we crash. 
// So recurse when we change levels and loop at the same level
// to dramatically reduce the stack usage.
HTREEITEM CDepotTreeCtrl::CheckItem(CString lookingfor, HTREEITEM item, BOOL useRevNum, int revNum)
{
	int lgth = lookingfor.GetLength();
	if (!HasChildren(item))
	{
		while (item)	// loop on items at the same level
		{
			BOOL b;
			HTREEITEM prevItem;

			CP4FileStats *stats= m_FSColl.GetStats(GetLParam(item));
			if (IsBadWritePtr(stats, sizeof(CP4FileStats)))
			{
				b = FALSE;
			}
			else
			{
				LPCTSTR path = lookingfor.GetAt(0) == _T('/') ? stats->GetFullDepotPath() 
					                                          : stats->GetFullClientPath();
				if (IsBadWritePtr((LPVOID)path, lgth))
					b = FALSE;
				else
					b = nCompare(path, lookingfor, lgth) == 0;
				if (b && useRevNum)
					b = revNum == stats->GetHaveRev();
			}
			if (b)
				return item;
			item = TreeView_GetNextSibling(m_hWnd, prevItem = item);
			if (prevItem == item)
				item = NULL;
			else while (item && HasChildren(item))
			{
				item = TreeView_GetChild(m_hWnd, item);
				if (item)
				{
					// recurse to go down a level
					item = CheckItem(lookingfor, item, useRevNum, revNum);
				}
				else
				{
					item = TreeView_GetNextSibling(m_hWnd, prevItem = item);
					if (prevItem == item)
						item = NULL;
				}
			}
		}
	}
	else
	{
		// before exploring the next level, check everything at this level
		HTREEITEM nextitem = TreeView_GetNextSibling(m_hWnd, item);
		if (nextitem)
		{
			// hopefully we can resurse here without blowing the stack
			// since we will only be recursing for directories, not files
			if ((nextitem = CheckItem(lookingfor, nextitem, useRevNum, revNum)) != NULL)
				return nextitem;
		}
		item = TreeView_GetChild(m_hWnd, item);
		if (item)
		{
			// recurse to go down a level
			item = CheckItem(lookingfor, item, useRevNum, revNum);
		}
	}
	return item;
}

HTREEITEM CDepotTreeCtrl::FindItem(CString path, CString fname, BOOL useRevNum)
{
    // If the file is under a folder known to be empty, immediately return NULL for
    // not found.  Note: before each series of FindItem() calls, m_LastChildlessPath 
    // should be set to an impossible path, by calling InitFindItem()    

    if( nCompare( path, m_LastChildlessPath, m_LastChildlessPath.GetLength()) == 0 )
        return NULL;

	HTREEITEM item=m_Root;
	CString temp;
	int start=2;   // Right after '//'
	int last;
	m_LastPath.Empty();
	m_LastPathItem=m_Root;
    BOOL isDepotNode;

	if (FindMBCS(path, m_SlashChar) == -1)
	{
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			long revNum = -5;
			CString findwhat = path + fname;
			if ((last = findwhat.Find(_T('#'))) != -1)
			{
				revNum = _tstol(findwhat.Mid(last+1));
				findwhat = findwhat.Left(last);
			}
			item=TreeView_GetNextItem(m_hWnd, TVI_ROOT, TVGN_ROOT);
			return CheckItem(findwhat, item, useRevNum, revNum);
		}
	}

	while(1)
	{
		////////////
		// Get next chunk of path, break if no more path left
		temp=path.Mid(start);
		last=FindMBCS(temp, m_SlashChar);
		start+=(last+1);
		if(last ==-1)
			break;		
		
		////////////
		// Get the first item at m_LastPath node in tree
		if(item==m_Root)
		{
			item=TreeView_GetNextItem(m_hWnd, TVI_ROOT, TVGN_ROOT);
			temp = g_sSlashes + temp.Left(last);
            isDepotNode=TRUE;
		}
		else
        {
			item=TreeView_GetChild(m_hWnd, item);   // Get first child
            temp = g_sStupidLeadingBlank + temp.Left(last);	
		    isDepotNode=FALSE;
        }

        ///////////
        // If there are NO leaves at this node, save the path to the empty
        // node, so subsequent searches for files under this path can be
        // expedited
        if( item == NULL )
        {
            m_LastChildlessPath= m_LastPath;
            if(m_LastPathItem == m_Root)
				m_LastChildlessPath= g_sSlashes;
        }

		///////////
		// For each leaf at this node, compare path
		while(item != NULL)
        {
            CString itemTxt= GetItemName(item);
            if( Compare(itemTxt,temp) == 0 ||
                            Compare(itemTxt,temp+g_TrulyEmptyDir) == 0 )
                break;
			item= TreeView_GetNextSibling(m_hWnd, item);
        }
		

		//////////
		// A NULL item means we looked at all leaves, and didnt get a match
		// In other words, we didnt find anything, so return a NULL for not found
		if(item==NULL)
		{
			if(m_LastPathItem == m_Root)
				m_LastPath= g_sSlashes;
			return NULL;
		}

		//////////
		// We have a match, at least up to current item, so update
		// last path and parent
        if( isDepotNode )
            m_LastPath= temp;
        else
		    m_LastPath+= temp.Mid(1);

		m_LastPath+=_T("/");
		m_LastPathItem=item;
	}

	// If item is still m_Root, we found nothing
	if(item==m_Root)
	{
		m_LastPath= g_sSlashes;
		m_LastPathItem=m_Root;
	}

	// Should be at the correct directory node
	item=TreeView_GetChild(m_hWnd, item);   // Get first child
	
    ///////////
    // If there are NO leaves at this node, save the path to the empty
    // node, so subsequent searches for files under this path can be
    // expedited
    if( item == NULL )
    {
        m_LastChildlessPath= m_LastPath;
    	return NULL;
    }

	int startRev=fname.ReverseFind(_T('#'));
	if(startRev == -1)
	{
		if(!useRevNum)
			startRev=fname.GetLength();
		else
		{
			ASSERT(0);    // Wanted to search by revision # but didnt provide revision number
			return NULL;
		}
	}

	if(useRevNum)	// do the compare outside the loop for speed - this gets called a lot
	{
		while(1)
		{
#ifdef _DEBUG
			temp=GetItemName(item);
#endif
			if(Compare(fname, GetItemName(item)) == 0)
				break;                      // found it
		
			item= TreeView_GetNextSibling(m_hWnd, item);
			if(item==NULL)
				return NULL;
		}
	}
	else
	{
		while(1)
		{
			temp=GetItemName(item);
			
			// compare everything except the revision
			if(nCompare(fname, temp, max(startRev, temp.GetLength())) == 0)
				break;                      // found it

			item= TreeView_GetNextSibling(m_hWnd, item);
			if(item==NULL)
				return NULL;
		}
	}
	
	return item;
}


void CDepotTreeCtrl::RecordTreeExploration()
{
    // Hose out the lists of explored and expanded files
    m_StringList.RemoveAll();
    m_ExpandedNodeList.RemoveAll();

    // Record the first visible item
    HTREEITEM item= TreeView_GetFirstVisible(m_hWnd);

    if( item != NULL )
        m_FirstVisibleNodeText= GetItemPath( TreeView_GetFirstVisible(m_hWnd) ); 
    else
        m_FirstVisibleNodeText.Empty();

	XTRACE(_T("Recorded first vis node=%s\n"), m_FirstVisibleNodeText);

    // Rummage through the tree, recording all folders that have been explored,
    // and noting those that are still expanded
    RecordFolderExploration(m_Root, _T(""));
}

void CDepotTreeCtrl::RecordFolderExploration(HTREEITEM parentItem, LPCTSTR path)
{
    TCHAR buf[ LONGPATH + 1 ];
    HTREEITEM item;

    if( parentItem == m_Root)
		item=TreeView_GetRoot(m_hWnd);
	else
		item=TreeView_GetChild(m_hWnd, parentItem);

    while(item != NULL)
	{
        TV_ITEM tvItem;
	    tvItem.hItem=item;
        tvItem.pszText = buf;         
        tvItem.cchTextMax = LONGPATH ;  
        tvItem.stateMask= TVIS_EXPANDED ;
	    tvItem.mask=TVIF_STATE | TVIF_PARAM | TVIF_HANDLE | TVIF_TEXT;
	    TreeView_GetItem(m_hWnd, &tvItem );	
	       
        if( tvItem.lParam == FOLDER_ALREADY_EXPANDED && (tvItem.state & TVIS_EXPANDED) == TVIS_EXPANDED )
        {
			// Get the foldername, minus leading spaces and trailing empty dir text
			TCHAR slashChar;
            CString folder;
			if( buf[0] == _T(' ') )
				folder= buf+1;
			else
				folder= buf;
            
			int junkStart= folder.Find( g_TrulyEmptyDir );
			if( junkStart != -1 )
				folder= folder.Left(junkStart);

            CString fullPath= path;

            if( !fullPath.IsEmpty() )
			{
				slashChar = fullPath.GetAt(0) == _T('/') ? _T('/') : _T('\\');
				if ( fullPath.GetLength() > 2 && fullPath.GetAt(2) == _T('\\') )
					TrimRightMBCS(fullPath, _T("\\"));
                fullPath += slashChar;
			}
			else
			{
				slashChar = folder.GetAt(0) == _T('/') ? _T('/') : _T('\\');
				if ( folder.GetLength() > 2 && folder.GetAt(2) == _T('\\') )
					TrimRightMBCS(folder, _T("\\"));
			}

            fullPath += folder;

            m_StringList.AddHead( CString( fullPath + slashChar + _T("*")) );
            m_ExpandedNodeList.AddHead( fullPath );

            RecordFolderExploration( item, fullPath );
		}
        
		item= TreeView_GetNextSibling(m_hWnd, item);
	}
}


/*
	_________________________________________________________________
*/

HTREEITEM CDepotTreeCtrl::FindFolder(LPCTSTR folderName)
{
	HTREEITEM item=m_LastPathItem;
	CString temp=folderName;
	CString temp2= temp+g_TrulyEmptyDir;
		
	if(m_LastPathItem == m_Root)
		item=TreeView_GetNextItem(m_hWnd, m_LastPathItem, TVGN_ROOT);
	else
		item=TreeView_GetNextItem(m_hWnd, m_LastPathItem, TVGN_CHILD);
	
	
	///////////
	// For each leaf compare the folder name

	while(item != NULL)
	{
		CString itemText= GetItemText(item);
		if(Compare(itemText,temp) == 0  || Compare(itemText,temp2) == 0 )
			break;

		item= TreeView_GetNextSibling(m_hWnd, item);
	}

	return item;
}


/*
	_________________________________________________________________

	Same as GetItemText, but strip off revision info
	_________________________________________________________________
*/

CString CDepotTreeCtrl::GetItemName(HTREEITEM curr_item)
{
	CString temp=GetItemText(curr_item);
	int i=temp.ReverseFind(_T('#'));
	if (i > 0 && temp.GetAt(i-1) == _T(' '))
		// the '#' is preceded by a space, so its i-1
 		return temp.Left(i-1); 
	else
		return temp;
}

/*
	_________________________________________________________________

	Same as GetItemName, but return have revision number as a CString
	_________________________________________________________________
*/

CString CDepotTreeCtrl::GetItemRev(HTREEITEM curr_item)
{
	CString temp=GetItemText(curr_item);
	int i = temp.ReverseFind(_T('#'));
	if(i != -1)
	{
 		temp = temp.Mid(i+1);
		i = temp.Find(_T('/'));
		if (i != -1)
			temp = temp.Left(i);
	}
	else
		temp.Empty();
	return temp;
}

/*
	_________________________________________________________________

	Same as GetItemName, but return head revision number as a CString
	_________________________________________________________________
*/

CString CDepotTreeCtrl::GetItemHeadRev(HTREEITEM curr_item)
{
	CString temp=GetItemText(curr_item);
	int i = temp.ReverseFind(_T('#'));
	if(i != -1)
	{
 		temp = temp.Mid(i+1);
		i = temp.Find(_T('/'));
		if (i != -1)
		{
			temp = temp.Mid(i+1);
			i = temp.Find(_T(' '));
			if (i != -1)
				temp = temp.Left(i);
		}
		else
			temp.Empty();
	}
	else
		temp.Empty();
	return temp;
}

/*
	_________________________________________________________________

	Return the item's file type
	_________________________________________________________________
*/

CString CDepotTreeCtrl::GetItemType(HTREEITEM curr_item)
{
	CString type = _T("");
	if (GET_P4REGPTR()->ShowFileType())
	{
		CString temp = GetItemText(curr_item);
		int i = temp.ReverseFind(_T('#'));
		if(i != -1)
		{
 			temp = temp.Mid(i+1);
			i = temp.Find(_T(' '));
			if (i != -1)
			{
				type = temp.Mid(i+1);
				i = type.Find(_T(' '));
				if (i != -1)
					type = type.Left(i);
				type.TrimLeft(_T("< "));
				type.TrimRight(_T("> "));
			}
		}
	}
	if (type.IsEmpty())
	{
		TV_ITEM item;
		item.hItem=curr_item;
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			CP4FileStats *fs=m_FSColl.GetStats((int) item.lParam);
			if (fs && fs->IsKindOf(RUNTIME_CLASS(CP4FileStats)))
				type = fs->GetHeadType();
		}
	}
	if (type.IsEmpty())
		type = _T("unknown");	// don't know
	return type;
}



/*
	_________________________________________________________________
	
	climb all the way up the tree (to "//depot" or whatever it's
	called in the server), getting the path pieces and concatenating 
	them with a single slash between.
	_________________________________________________________________
*/

CString CDepotTreeCtrl::GetItemPath(HTREEITEM item)
{
	ASSERT( item != NULL );

	CString fullPath;
	HTREEITEM htParent = item;

	if(ITEM_IS_FILE(item))
		fullPath = GetItemName( htParent );
	else
	{
		fullPath = GetItemText( htParent ) + m_SlashChar;
		if( fullPath[0] == _T(' ') )
			fullPath = fullPath.Mid(1);
	}

	while ( (htParent = TreeView_GetParent( m_hWnd, htParent ) ) != 0 )
	{
		fullPath = GetItemText( htParent ) + m_SlashChar + fullPath;
		if( fullPath[0] == _T(' ') )
			fullPath = fullPath.Mid(1);
	}

	// if we are doing local view, we need to check for the
	// C:\ case so that we don't wind up with things like 
	// C:\\dirname\file.ext - which is wrong
	if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT
	 && TheApp( )->m_ClientRoot.GetLength( ) == 3
	 && fullPath.GetAt(2) == _T('\\')
	 && fullPath.GetAt(3) == _T('\\'))
	{
		// for some reason, this has to be done
		// in multiple statements using a temp
		// variable (called endpart) - as a
		// single line, the compile does the
		// wrong thing and spits out garbage
		CString endpart = fullPath.Mid(4);
		fullPath = fullPath.GetAt(0);
		fullPath += _T(":\\");
		fullPath += endpart;
	}
	return fullPath;
}

/*	_________________________________________________________________

	Like previous routine, but for current selection rather than a given item.
	_________________________________________________________________
*/	

CString CDepotTreeCtrl::GetCurrentItemPath() 
{
	if( GetSelectedCount() )
	{
		TVITEM	tvItem;
		tvItem.state = 0;         
		tvItem.stateMask = TVIS_EXPANDED;         
		tvItem.mask = TVIF_STATE;
		tvItem.hItem = GetSelectedItem(0);
		CString path = GetItemPath(tvItem.hItem);
		if (TreeView_GetItem(m_hWnd, &tvItem) && !(tvItem.state & TVIS_EXPANDED))	// get the item and its expansion state
			path.TrimRight(_T('/'));
		return path;
	}
	return _T("");
}

// Convert a local syntax path into a depot syntax path
// There are several way to call this routine:
//	1) item == 0 && localPath == 0 -> GetItemDepotSyntax(current selection)
//	2) item != 0 && localPath == 0 -> Slowest, GetItemDepotSyntax(for given item)
//	3) item == 0 && localPath != 0 -> Must be a client file, GetItemDepotSyntax(for given file)
//	4) item != 0 && localPath != 0 -> Fastest, but item must be for localPath
CString CDepotTreeCtrl::GetItemDepotSyntax(HTREEITEM item, CString *localPath)
{
	CString depotPath;
	if (!localPath)
	{
		if (!item)
		{
			depotPath = GetCurrentItemPath();
			item = GetSelectedItem(0);
		}
		else
		{
			depotPath = GetItemPath(item);
		}
	}
	else if (!item)
		depotPath = *localPath;

	if( item && ITEM_IS_FILE(item) )
	{
		CP4FileStats *fs= m_FSColl.GetStats(GetLParam(item));
		if( fs->InClientView() )
			return(fs->GetFullDepotPath());
	}

	if (depotPath.IsEmpty())
		depotPath = GetItemPath(item);
	BOOL bAddSl = FALSE;
	if (depotPath.GetAt(depotPath.GetLength()-1) == _T('\\'))
	{
		depotPath.TrimRight(_T('\\'));
		bAddSl = TRUE;
	}
	if (depotPath.FindOneOf(_T("@#%*")) != -1)
	{
		StrBuf b;
		StrBuf f;
		f << CharFromCString(depotPath);
		StrPtr *p = &f;
		StrOps::WildToStr(*p, b);
		depotPath = CharToCString(b.Value());
	}
	CCmd_Where *pCmd1 = new CCmd_Where;
	pCmd1->Init(NULL, RUN_SYNC);
	if ( pCmd1->Run(depotPath) && !pCmd1->GetError() 
		&& pCmd1->GetDepotFiles()->GetCount() )
	{
		depotPath = pCmd1->GetDepotSyntax();
		if (bAddSl)
			depotPath += _T('/');
	}
	else if (bAddSl)
		depotPath += _T('\\');
	delete pCmd1;
	return depotPath;
}

/*
	_________________________________________________________________
*/

HTREEITEM CDepotTreeCtrl::Insert( LPCTSTR text, int imageIndex
							 , LPARAM lParam, HTREEITEM hParent)
{
    CString temp;
#ifdef _DEBUG 	// Asserts to sniff for the deadly QQ bug
	temp = text;
	ASSERT( temp.Find( QQBUG_JOB000458 ) == -1 );
#endif

	// Make sure the parent is now displayed with the '+' button,
	// since the tree control does a "steel trap" after cChildren
	// is set for any node and will no longer attempt to compute
	// the child count for itself. Also make sure we dont have the
	// bailing wire message, g_sTrulyEmptyDir in the parent's name.
	if(hParent != m_Root)
	{
		SetChildCount(hParent, 1);
		temp= GetItemText(hParent);
		int offendingtxt;
		if( (offendingtxt= temp.Find(g_TrulyEmptyDir)) != -1)
			SetItemText(hParent, temp.Left(offendingtxt));
	}

	TV_INSERTSTRUCT tree_insert;
	// If its a folder or depot we are inserting, make sure it has a '+' sign on the button
	if( imageIndex == CP4ViewImageList::VI_FOLDER ||
		imageIndex == CP4ViewImageList::VI_OPENFOLDER ||
		imageIndex == CP4ViewImageList::VI_DEPOT ||
		imageIndex == CP4ViewImageList::VI_REMOTEDEPOT )
	{
		tree_insert.item.cChildren=1;
		tree_insert.item.mask= TVIF_TEXT | TVIF_PARAM | TVIF_IMAGE | TVIF_SELECTEDIMAGE | TVIF_CHILDREN;
	}
	else
		tree_insert.item.mask= TVIF_TEXT | TVIF_PARAM | TVIF_IMAGE | TVIF_SELECTEDIMAGE;

	if (GET_P4REGPTR()->SortByExtension())
		tree_insert.hInsertAfter = (*text == _T(' ')) ? TVI_SORT : SortItemByExtension(text, hParent);
	else
		tree_insert.hInsertAfter = TVI_SORT;
	tree_insert.hParent = hParent;
	tree_insert.item.iImage = imageIndex;
	tree_insert.item.iSelectedImage = imageIndex;
	tree_insert.item.pszText = const_cast<LPTSTR>(text);
	tree_insert.item.cchTextMax = lstrlen( text );

	//		lparam is used in two different ways:
	//			1. for files, it's the m_ItemCount
	//			2. for subdirs in 98.2, it's either EXPAND_FOLDER
	//				or FOLDER_ALREADY_EXPANDED.
	//				EXPAND_FOLDER is used to trigger a p4 dirs and fstat.
	//
	tree_insert.item.lParam = lParam;   

	return( TreeView_InsertItem( m_hWnd, &tree_insert ) );
}

HTREEITEM CDepotTreeCtrl::SortItemByExtension(LPCTSTR text, HTREEITEM hParent)
{
	HTREEITEM item=TreeView_GetNextItem(m_hWnd, hParent, TVGN_CHILD);
	if (!item)
		return TVI_FIRST;

	CString myText = text;
	int i;
	CString myExt = _T("");
	if ((i = myText.ReverseFind(_T('#'))) > 0)
		myText = myText.Left(i-1);
	if ((i = myText.ReverseFind(_T('.'))) != -1) 
		myExt = myText.Right(myText.GetLength() - i - 1);
	CString itemText;
	CString itemExt;
	HTREEITEM prevItem = TVI_FIRST;
	while(item != NULL)
	{
		itemText= GetItemText(item);
		itemExt = _T("");
		if (itemText.GetAt(0) != _T(' '))
		{
			if ((i = itemText.ReverseFind(_T('#'))) > 0)
				itemText = itemText.Left(i-1);
			if ((i = itemText.ReverseFind(_T('.'))) != -1)
				itemExt = itemText.Right(itemText.GetLength() - i - 1);
			if ((i = itemExt.CompareNoCase(myExt)) > 0)
				return prevItem;
			if (!i && (itemText.CompareNoCase(myText) > 0))
				return prevItem;
		}
		prevItem = item;
		item= TreeView_GetNextSibling(m_hWnd, item);
	}
	return TVI_LAST;
}


HTREEITEM CDepotTreeCtrl::VerifySubdir(CString path, HTREEITEM startItem)
{
	HTREEITEM item=startItem;
	HTREEITEM parent;

	CString temp;
	int start=2;   // Right after '//'
	int last;
	
	while(1)
	{
		// Extract the next subdir name
		temp=path.Mid(start);
		last=temp.Find(_T("/"));
		start+=(last+1);
		if(last ==-1)
			break;  //at end of path
		temp = g_sStupidLeadingBlank + temp.Left(last);
		
		// Search children of current item for matching subdir name
		parent=item;
		item=TreeView_GetChild(m_hWnd, parent);   
		while(item != NULL && GetItemName(item) != temp)
			item= TreeView_GetNextSibling(m_hWnd, item);

		if(item == NULL)  
			// subdir name not found - so add it
			item=Insert ( temp, CP4ViewImageList::VI_FOLDER, (LPARAM) -1, parent);
	}

	return item;
}


/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Error(WPARAM wParam, LPARAM lParam)
{
	return 0;
}


/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Diff(WPARAM wParam, LPARAM lParam)
{
	CCmd_Diff *pCmd= (CCmd_Diff *) wParam;

	if (pCmd->GetSflag() == _T('d') || pCmd->GetSflag() == _T('e'))
		return OnP4Diff_sd_se(wParam, lParam);

	if( !pCmd->GetError() && pCmd->GetDiffRunCount() == 0)
	{
		if (pCmd->GetDiffNbrFiles() == 1)
        {
            CString msg;
            msg.FormatMessage(IDS_CLIENTFILE_s_DOESNOTDIFFER, pCmd->GetDiffFileName());
			AddToStatus(msg, SV_COMPLETION);
        }
		else if (pCmd->GetDiffErrCount() == 0)
			AddToStatus(LoadStringResource(IDS_NONE_OF_THE_SELECTED_CLIENT_FILES_DIFFER), SV_COMPLETION);
	}
	MainFrame()->ClearStatus();         
	delete pCmd;
	return 0;
}


/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Get(WPARAM wParam, LPARAM lParam)
{
	CCmd_Get *pCmd= (CCmd_Get *) wParam;
    ASSERT_KINDOF(CCmd_Get,pCmd);
	BOOL bNeed2Refresh = FALSE;
	BOOL bGet = FALSE;
	BOOL bRecover = FALSE;
	CString txt;
	int i;

	if(!pCmd->GetError())
	{
		CStringList *getlist= pCmd->GetGetList();
		ASSERT_KINDOF(CStringList,getlist);
		CStringList *removelist= pCmd->GetRemoveList();
		ASSERT_KINDOF(CStringList,removelist);

		if(!pCmd->IsWhatIf())
		{
            // Update the trees (unless we've hit max file seeks; then just refresh)
			if ((getlist->GetCount() > MAX_FILESEEKS || removelist->GetCount() > MAX_FILESEEKS)
			 && (!pCmd->IsRunIntegAfterSync() && !pCmd->IsOpenAfterSync() && !pCmd->GetRecover()->GetCount()))
			{
				bNeed2Refresh = TRUE;
			}
			else
			{
				// Temporarily disable redraws for both windows
				SetRedraw(FALSE);
				::SendMessage(m_changeWnd, WM_SETREDRAW, FALSE, 0);

	            // Update the trees
				ProcessGetListResults(WM_P4GET, getlist);
				ProcessGetListResults(WM_P4UNGET, removelist);

				// Re-enable redraws for both windows
				SetRedraw(TRUE);
				::SendMessage(m_changeWnd, WM_SETREDRAW, TRUE, 0);

				// And finally, make sure both windows redraw
				RedrawWindow();
				::RedrawWindow( m_changeWnd, NULL, NULL, RDW_INVALIDATE );
			}

			if(getlist->GetCount() >0)
			{
				txt.FormatMessage(IDS_COPIED_n_FILES_TO_CLIENT, getlist->GetCount());
				if (pCmd->GetNumberAdded())
				{
					CString str;
					str.FormatMessage(IDS_n_WERE_ADDED, pCmd->GetNumberAdded());
					txt += str;
				}
				AddToStatus(txt, SV_COMPLETION);
				if (!bNeed2Refresh)
				{
					for(POSITION pos = getlist->GetHeadPosition(); pos != NULL; )
					{
						if (getlist->GetNext( pos ).Find(_T(" added as ")) >= 0
							&& !pCmd->IsOpeningForEdit())
						{
							// found something new was added - signal to start a refresh
							bNeed2Refresh = TRUE;
							break;
						}
					}
				}
			}
			else
				AddToStatus(LoadStringResource(IDS_SYNC_COPIED_NO_FILES_TO_CLIENT), SV_COMPLETION);
		
			if(removelist->GetCount() >0)
			{
				txt.FormatMessage(IDS_REMOVED_n_FILES_FROM_CLIENT, removelist->GetCount());
				AddToStatus(txt, SV_COMPLETION);
			}
			else
				AddToStatus(LoadStringResource(IDS_REMOVED_NO_FILES_FROM_CLIENT), SV_COMPLETION);
		}
		else
		{
			if(getlist->GetCount() >0)
			{
				txt.FormatMessage(IDS_SYNC_WOULD_COPY_n_FILES_TO_CLIENT, getlist->GetCount());
				if (pCmd->GetNumberAdded())
				{
					CString str;
					str.FormatMessage(IDS_n_WOULD_BE_ADDED, pCmd->GetNumberAdded());
					txt += str;
				}
				AddToStatus(txt, SV_COMPLETION);
			}
			else
				AddToStatus(LoadStringResource(IDS_SYNC_WOULD_COPY_NO_FILES_TO_CLIENT), SV_COMPLETION);
		
			if(removelist->GetCount() >0)
			{
				txt.FormatMessage(IDS_SYNC_WOULD_REMOVE_n_FILES_FROM_CLIENT, removelist->GetCount());
				AddToStatus(txt, SV_COMPLETION);
			}
			else
				AddToStatus(LoadStringResource(IDS_SYNC_WOULD_REMOVE_NO_FILES_FROM_CLIENT), SV_COMPLETION);
		}
		if (((i = pCmd->ReiterateWarnings()) == 0) 
		 || ((pCmd->IsRunIntegAfterSync()) 
		  && (IDYES == AfxMessageBox(IDS_THERE_WERE_WARNINGS_REPORTED_DURING_THE_SYNC, MB_YESNO|MB_ICONQUESTION))))
		{
			if (pCmd->IsRunIntegAfterSync())
			{
				if (OnIntegrate3())
					bNeed2Refresh = FALSE;	// the integration completion routine will do the refresh
			}
			else if (pCmd->IsOpenAfterSync())
			{
				bGet = TRUE;
				if(getlist->GetCount() == 0)
					m_SkipSyncDialog = TRUE;
				// May have to reslect all that were selected when sync command fired.
				// This undoes any fiddling the user might have done while waiting.
				CDWordArray *selset = pCmd->GetSelectionSet();	// get original sel set
				int cnt = selset->GetSize();
				BOOL b = cnt == GetSelectedCount();
				if (b)
				{	// same number in each list; are the lists the same?
					for(int index=-1; ++index < cnt; )
					{
						HTREEITEM item = (HTREEITEM) selset->GetAt(index);
						if (GetSelectedItem(index) != item)
						{
							b = FALSE;	// lists are different - have to reselect
							break;
						}
					}
				}
				if (!b)
				{	// the lists differ; have to reselect
					UnselectAll();
					for(int index=-1; ++index < cnt; )
					{
						HTREEITEM item = (HTREEITEM) selset->GetAt(index);
						SetSelectState(item, TRUE);
					}
				}
			}
			else if (pCmd->GetRecover()->GetCount() > 0)
			{
				m_StringList.RemoveAll();
				CStringList *recoverList = pCmd->GetRecover();
				for(POSITION pos = recoverList->GetHeadPosition(); pos != NULL; )
					m_StringList.AddHead(recoverList->GetNext( pos ));
				bRecover = TRUE;
			}
		}
		else if (i)
		{
			if (pCmd->IsOpenAfterSync())
				AfxMessageBox(IDS_SYNC_WARNINGS_CANTEDIT, MB_ICONSTOP);
			else if (pCmd->GetRecover()->GetCount() > 0)
				AfxMessageBox(IDS_SYNC_WARNINGS_CANTRECOVER, MB_ICONSTOP);
			// We need to do a refresh to update our internal stats and the screen
			bNeed2Refresh = TRUE;
		}
		HWND hwnd = pCmd->GetRevHistWnd();
		if (hwnd)
			::PostMessage(hwnd, WM_UPDATEHAVEREV, 0, pCmd->GetRevReq());
	}
	else if (pCmd->IsRunIntegAfterSync())	// got an error; do we need to re-enable integ dialog?
	{
		EnterCriticalSection(&MainFrame()->CriticalSection); 
		if (m_IntegWizard)
			m_IntegWizard->SendMessage(WM_ENABLEDISABLE, 0, TRUE);
		LeaveCriticalSection(&MainFrame()->CriticalSection);
	}

	MainFrame()->ClearStatus();         
	delete pCmd;
	
	if (bGet)
		OnFiledropEdit();
	else if (bRecover)
		OnFileRecoverProceed();
	else if (bNeed2Refresh)
		OnViewUpdate();
	return 0;
}


/*
	_________________________________________________________________

	wait, p4 refresh was superseded by p4 sync -f -- see too onFilerefresh
	test for a server 98.1 or greater: how? SERVERLEVEL>2?
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Refresh(WPARAM wParam, LPARAM lParam)
{
	CCmd_Refresh *pCmd= (CCmd_Refresh *) wParam;
	
	if(!pCmd->GetError())
		AddToStatus(LoadStringResource(IDS_FILE_REFRESH_COMPLETED), SV_COMPLETION);

	delete pCmd;
	MainFrame()->ClearStatus();         

	return 0;
}


/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4History(WPARAM wParam, LPARAM lParam)
{
	CCmd_History *pCmd= (CCmd_History *) wParam;
	CString txt;

	// if just want rev hist dialog, minimize main window
	if (!TheApp()->m_RevHistPath.IsEmpty())
		MainFrame()->ShowWindow(SW_SHOWMINIMIZED);

	if(!pCmd->GetError())
	{
		// Get the file type for the head rev
		CP4FileStats *fs= pCmd->GetFileStats();
		CString ftype= fs->GetHeadType();
		BOOL isText= fs->IsTextFile();

		CHistory *hist= pCmd->GetHistory();
		if(hist->GetRevisionCount() > 0)
		{
			BOOL enableShowIntegs = pCmd->GetEnableShowIntegs();
			CHistoryDlg *dlg= new CHistoryDlg(this);
			if (pCmd->IsAFile())
			{
				dlg->Init(hist, ftype, isText, m_hWnd, fs->GetHaveRev(), fs->IsMyOpen(),
							pCmd->GetInitialRev(), pCmd->GetInitialName(), 
							pCmd->HaveServerLock() ? pCmd->GetServerKey() : 0);
			}
			else
			{
				dlg->Init(hist, ftype = _T("text"), TRUE, m_hWnd, -1, FALSE,
							pCmd->GetInitialRev(), pCmd->GetInitialName(), 
							pCmd->HaveServerLock() ? pCmd->GetServerKey() : 0);
			}
			dlg->SetEnableShowIntegs(enableShowIntegs);
			dlg->SetCallingWnd(pCmd->GetCallingWnd());
			dlg->SetCallingCommand(pCmd);
			// Display the Rev Hist dialog
			if (!dlg->Create(IDD_HISTORY, this))	// display the description dialog box
			{
				dlg->DestroyWindow();	// some error! clean up
				delete dlg;
			}
		}
		delete fs;
	}
	else
	{
		CString errTxt = pCmd->GetErrorText();
		if ((errTxt.Find(_T("Remote depot 'db.rev' database access failed")) != -1)
		  && GET_P4REGPTR()->GetFetchCompleteHist())
		{
			if(!GET_P4REGPTR()->SetFetchCompleteHist( 0 ) )
				AfxMessageBox( IDS_BAD_REGISTRY,  MB_ICONSTOP );
			else
			{
				TheApp()->StatusAdd(LoadStringResource(IDS_REMOTE_DOWN_TURNING_OFF_INTEGS),SV_WARNING);
				::PostMessage(m_hWnd, WM_COMMAND, ID_FILE_REVISIONHISTORY, 0);
			}
		}
		else if (!TheApp()->m_RevHistPath.IsEmpty())
		{
			CString txt;
			txt.FormatMessage(IDS_CANTRUNREVHIST_s_s, TheApp()->m_RevHistPath, errTxt);
			AfxMessageBox( txt,  MB_ICONSTOP );
		}
	}
	MainFrame()->ClearStatus();
	if(pCmd->GetError())
		::SetFocus(pCmd->GetCallingWnd());
	return 0;
}

LRESULT CDepotTreeCtrl::OnP4EndHistory( WPARAM wParam, LPARAM lParam )
{
	CHistoryDlg *dlg = (CHistoryDlg *)lParam;
	BOOL enableShowIntegs = dlg->GetEnableShowIntegs();
	HWND hRerun = NULL;

	if (dlg->m_Rerun 
		|| (dlg->m_CompleteHist != GET_P4REGPTR()->GetFetchCompleteHist() 
		 && enableShowIntegs))
	{
		hRerun = dlg->GetCallingWnd();
		TheApp()->m_RevHistMore = dlg->m_RevHistCount;
		int n = GET_P4REGPTR()->GetFetchHistCount();
		if (!dlg->m_More && n > 0)
		{
			TheApp()->m_RevHistMore -= n;
			if (TheApp()->m_RevHistMore < 0)
				TheApp()->m_RevHistMore = 0;
		}
		if (enableShowIntegs && !GET_P4REGPTR()->SetFetchCompleteHist( dlg->m_CompleteHist ) )
			AfxMessageBox( IDS_BAD_REGISTRY,  MB_ICONSTOP );
	}
	CCmd_History *pCmd = dlg->GetCallingCommand();
	delete pCmd;
	dlg->DestroyWindow();

	if (hRerun)
		::PostMessage(hRerun, WM_COMMAND, 
					  hRerun == MainFrame()->m_hWnd ? ID_FILE_REVISIONHISTORYCONT 
					                                : ID_FILE_REVISIONHISTORY, 0);
	else if (!TheApp()->m_RevHistPath.IsEmpty())
		::PostMessage(MainFrame()->m_hWnd, WM_COMMAND, ID_APP_EXIT, 0);

	return TRUE;
}

/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Integ(WPARAM wParam, LPARAM lParam)
{
	BOOL bRunInteg2 = FALSE;
	BOOL chainedCommands= FALSE;
	CCmd_Integrate2 *pCmd= (CCmd_Integrate2 *) wParam;

	// Did anything actually get integrated (and this is not a preview)?
	EnterCriticalSection(&MainFrame()->CriticalSection); 
	if (pCmd->GetActualCount() && !pCmd->IsPreview() && !pCmd->HitFatalError())
	{
		if (m_IntegWizard)	// if the dialog is still up, close it
		{
			m_IntegWizard->DestroyWindow();	 // deletes m_IntegWizard
			m_IntegWizard = 0;
			MainFrame()->SetModelessUp(FALSE);
			m_StringListSv.RemoveAll();		// saved list now no longer useful
		}
	}
	else if (m_IntegWizard)
		m_IntegWizard->SendMessage(WM_ENABLEDISABLE, 0, TRUE);
	LeaveCriticalSection(&MainFrame()->CriticalSection);

	if(!pCmd->GetError() && (!pCmd->IsRename() || pCmd->IsOK2rename()))
	{
		CObList *list= pCmd->GetTargetList();
		ASSERT_KINDOF(CObList,list);
		if (list->GetCount() > MAX_FILESEEKS && !pCmd->HitMaxFileSeeks())
		{
			POSITION pos= list->GetHeadPosition();
			while(pos != NULL)
			{
				// delete the filestats info
				CP4FileStats *stats = (CP4FileStats *) list->GetNext(pos);
				ASSERT_KINDOF(CP4FileStats, stats);
				delete stats;
			}
			pCmd->SetHitMaxFileSeeks(TRUE);
		}

		if(!pCmd->IsPreview())
		{
			CString text;
			text.FormatMessage(IDS_OPENED_n_FILES_FOR_INTEGRATE, pCmd->GetActualCount());
			AddToStatus(text, SV_COMPLETION);
		}

		if(!pCmd->IsPreview() && pCmd->HitMaxFileSeeks())
		{
			// Too much was added for seeking out each file for an att update
			// to be efficient.  CCmd_Integrate2 thus did not create a target
			// list for us.  Just start a full update.
			int key= pCmd->GetServerKey();
			chainedCommands= TRUE;
			MainFrame()->UpdateDepotandChangeViews(REDRILL, key);
		}
		else if(pCmd->IsPreview())
		{
			// The list is empty for previews
			ASSERT( list->GetCount() == 0 );
			AddToStatus(LoadStringResource(IDS_INTEGRATE_PREVIEW_COMPLETED), SV_COMPLETION);
			if (pCmd->IsRunSyncAfterPreview())
				bRunInteg2 = TRUE;
		}
		else
		{
			 // Temporarily disable redraws for both windows
            SetRedraw(FALSE);
            ::SendMessage(m_changeWnd, WM_SETREDRAW, FALSE, 0);

			POSITION pos= list->GetHeadPosition();
            InitFindItem();
			while(pos != NULL)
			{
				// Get the filestats info
				CP4FileStats *stats = (CP4FileStats *) list->GetNext(pos);
				ASSERT_KINDOF(CP4FileStats, stats);

				// Find the item
				HTREEITEM item;
				item=FindItem(stats->GetDepotDir(), stats->GetDepotFilename(), FALSE);
				if(item==NULL)
					// It is quite common for get or unget to reference a file
					// that is not in our tree, due to client view or deleted
					// file viewing options
					XTRACE(_T("OnP4Integ() item not found %s\n"), stats->GetFullDepotPath());
				else
				{
					// Update its properties
					int index=GetLParam(item);
					CP4FileStats *fs=m_FSColl.GetStats(index);
					
					// Note that the open action could be a delete, since integ
					// can result in file deletes as well as branch and integ
					// open actions
					fs->SetOpenAction(stats->GetMyOpenAction(), FALSE);
					fs->SetHeadType(stats->GetHeadType());
					fs->SetType(stats->GetType());
					fs->SetOpenChangeNum(stats->GetOpenChangeNum());
							
					// Update the image in this window
					SetImage(item, TheApp()->GetFileImageIndex(fs));
				}

				// Update changes window whether or not we found file here
				::SendMessage(m_changeWnd, WM_UPDATEOPEN, (WPARAM) stats, P4INTEG);

				// Delete the filestats object
				delete stats;

			} // while

			   // Re-enable redraws for both windows
            SetRedraw(TRUE);
            ::SendMessage(m_changeWnd, WM_SETREDRAW, TRUE, 0);

            // And finally, make sure both windows redraw
	        RedrawWindow();
            ::RedrawWindow( m_changeWnd, NULL, NULL, RDW_INVALIDATE );
		}
		
	} // !error
	else if (pCmd->IsRename() && !pCmd->IsOK2rename())
	{
		int chgnbr = pCmd->GetTargetChange();
		CString txt;
		txt.FormatMessage(IDS_RENAME_NOT_POSSIBLE, pCmd->GetErrorText());
		AddToStatus(txt, SV_ERROR);
		if (chgnbr && (chgnbr == pCmd->GetNewChangeNbr()))
		{
			CString msg;
			msg.FormatMessage(IDS_DELETE_NEW_CHGLIST, txt, chgnbr);
			CCmd_Delete *pCmd2= new CCmd_Delete;
			txt.Format(_T("%d"), chgnbr);
			pCmd2->Init( m_changeWnd, RUN_ASYNC, HOLD_LOCK, pCmd->GetServerKey() );
			pCmd2->SetIgnoreActiveItem(TRUE);
			if( pCmd2->Run( P4CHANGE_DEL, txt ) )
			{
				txt.FormatMessage(IDS_DELETING_n, chgnbr);
				MainFrame()->UpdateStatus(txt);
			}	
			else
				delete pCmd2;
		}
	}

	if(!chainedCommands || MainFrame()->IsQuitting())
		pCmd->ReleaseServerLock();

	if (bRunInteg2 && !MainFrame()->IsQuitting())
	{
		EnterCriticalSection(&MainFrame()->CriticalSection); 
		if (m_IntegWizard)
			m_IntegWizard->SendMessage(WM_ENABLEDISABLE, 0, FALSE);
		LeaveCriticalSection(&MainFrame()->CriticalSection);

		OnIntegrate2(pCmd->GetPreviewTargetList());
	}

	delete pCmd;

	return 0;
}

LRESULT CDepotTreeCtrl::OnP4Add(WPARAM wParam, LPARAM lParam)
{
	BOOL chainedCommands= FALSE;
	CCmd_Add *pCmd= (CCmd_Add *) wParam;
	ASSERT_KINDOF(CCmd_Add, pCmd);

	if(!pCmd->GetError() && pCmd->HitMaxFileSeeks() )
	{
		// Too much was added for seeking out each file for an att update
		// to be efficient.  Just start a full update.
		int key= pCmd->GetServerKey();
		chainedCommands= TRUE;
		MainFrame()->UpdateDepotandChangeViews(REDRILL, key);
	}
	else if(!pCmd->GetError())
	{
		// Temporarily disable redraws
		SetRedraw(FALSE);

		// Get the results list, and update any files that are
		// present in the DepotTree
		HTREEITEM item;
		CObList const *list=pCmd->GetList();

		InitFindItem();
		for(POSITION pos= list->GetHeadPosition(); pos!=NULL; )
		{
			CP4FileStats *fs= (CP4FileStats *) list->GetNext(pos);

			// Try to find the item
			item=FindItem(fs->GetDepotDir(), fs->GetDepotFilename(), FALSE);
			if(item==NULL)
				// It is not unusual to NOT find the file, since the user
				// could have initiated the command by clicking a folder
				// that had not yet been explored
				XTRACE(_T("OnP4Recover() item not found %s\n"), fs->GetFullDepotPath());
			else
			{
				// Update its properties in accordance with the command.  Remember
				// that CCmd_ListOpStat will return a sparse set of information.
				// In general only the file name and revision are present.

				CP4FileStats *treefs= m_FSColl.GetStats(GetLParam(item));
						
				treefs->SetNotInDepot(TRUE);
				treefs->SetOpenAction(fs->GetMyOpenAction(), FALSE);
				treefs->SetHaveRev(fs->GetHaveRev());
				treefs->SetOpenChangeNum(fs->GetOpenChangeNum());

				// Update the image in this window
				SetImage(item, TheApp()->GetFileImageIndex(treefs));
			
			}// if found
			
		}//for each file
	
		 // Re-enable redraw
        SetRedraw(TRUE);
        RedrawWindow();

         // And finally, re-send the WM_P4ADD to the changelist pane
	    ::SendMessage(m_changeWnd, WM_P4ADD, wParam, lParam);
    	MainFrame()->ClearStatus();         

		// The server lock release and deletion of the command are
		// both handled by the changes window
		return 0;
	}
	else
    	MainFrame()->ClearStatus();         
	

	if( !chainedCommands || MainFrame()->IsQuitting() )
		pCmd->ReleaseServerLock();
	delete pCmd;
		
	return 0;
}


LRESULT CDepotTreeCtrl::OnP4Recover(WPARAM wParam, LPARAM lParam)
{
	BOOL chainedCommands= FALSE;
	CCmd_Add *pCmd= (CCmd_Add *) wParam;
	ASSERT_KINDOF(CCmd_Add, pCmd);

	if(!pCmd->GetError())
	{
		CString text;
		text.FormatMessage(IDS_RECOVERED_n_FILES, pCmd->GetAddedFileCount());
		AddToStatus(text,SV_COMPLETION);
	}
	
	if(!pCmd->GetError() && pCmd->HitMaxFileSeeks() )
	{
		// Too much was added for seeking out each file for an att update
		// to be efficient.  Just start a full update.
		int key= pCmd->GetServerKey();
		chainedCommands= TRUE;
		MainFrame()->UpdateDepotandChangeViews(REDRILL, key);
	}
	else if(!pCmd->GetError())
	{
		// Temporarily disable redraws
		SetRedraw(FALSE);

		// Get the results list, and update any files that are
		// present in the DepotTree
		HTREEITEM item;
		CObList const *list=pCmd->GetList();

		InitFindItem();
		for(POSITION pos= list->GetHeadPosition(); pos!=NULL; )
		{
			CP4FileStats *fs= (CP4FileStats *) list->GetNext(pos);

			// Try to find the item
			item=FindItem(fs->GetDepotDir(), fs->GetDepotFilename(), FALSE);
			if(item==NULL)
				// It is not unusual to NOT find the file, since the user
				// could have initiated the command by clicking a folder
				// that had not yet been explored
				XTRACE(_T("OnP4Recover() item not found %s\n"), fs->GetFullDepotPath());
			else
			{
				// Update its properties in accordance with the command.  Remember
				// that CCmd_ListOpStat will return a sparse set of information.
				// In general only the file name and revision are present, though the
				// rev is not present for LOCK, UNLOCK.  For EDIT and DELETE we will
				// have additional info as returned by a call to ostat.

				CP4FileStats *treefs= m_FSColl.GetStats(GetLParam(item));
						
				treefs->SetNotInDepot(fs->IsNotInDepot());
				treefs->SetOpenAction(fs->GetMyOpenAction(), FALSE);
				if (fs->GetHaveRev() != 0)
					treefs->SetHaveRev(fs->GetHaveRev());
				treefs->SetOpenChangeNum(fs->GetOpenChangeNum());

				SetImage(item, TheApp()->GetFileImageIndex(treefs));
			
			}// if found
			
		}//for each file
	
		 // Re-enable redraw
        SetRedraw(TRUE);
        RedrawWindow();

         // And finally, re-send the WM_P4ADD to the changelist pane
	    ::SendMessage(m_changeWnd, WM_P4ADD, wParam, lParam);
    	MainFrame()->ClearStatus();         

		// The server lock release and deletion of the command are
		// both handled by the changes window
		return 0;
	}
	else
    	MainFrame()->ClearStatus();         
	

	if( !chainedCommands || MainFrame()->IsQuitting() )
		pCmd->ReleaseServerLock();
	delete pCmd;
		
	return 0;
}


/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4ListOp(WPARAM wParam, LPARAM lParam)
{
	BOOL chainedCommands= FALSE;
	int  iRedoOpenedFilter = 0;
	CCmd_ListOpStat *pCmd= (CCmd_ListOpStat *) wParam;
	
	if(!pCmd->GetError())
	{
		iRedoOpenedFilter = pCmd->GetRedoOpenedFilter();
		if( pCmd->GetOpenAfterDelete() )
		{
			int selectedChange = pCmd->GetSelectedChange();
			int key= pCmd->GetServerKey();
			chainedCommands= TRUE;
			CCmd_ListOpStat *pCmd2= new CCmd_ListOpStat;
			pCmd2->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key );

			if( pCmd2->Run( &m_StringList2, P4EDIT, selectedChange ) )
				MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUEST_OPEN_EDIT) );
			else
				delete pCmd2;
		}
		else if( pCmd->HitMaxFileSeeks() || pCmd->HitMaxFileStats())
		{
			// Too much was added (or the chglist is too big)
			// for seeking out each file for an att update
			// to be efficient.  Just start a full update.
			int key= pCmd->GetServerKey();
			chainedCommands= TRUE;
			if (pCmd->GetRevertUnchgAfter())
			{
				CCmd_ListOpStat *pCmd2= new CCmd_ListOpStat;
				pCmd2->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key );
				pCmd2->SetRedoOpenedFilter(pCmd->GetRedoOpenedFilter());
				pCmd2->SetHitMaxFileSeeks(TRUE);
				if( pCmd2->Run( &m_StringList2, P4REVERTUNCHG ) )
				{
					MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_REVERT) );
					iRedoOpenedFilter = 0;
				}
				else
				{
					delete pCmd2;
					MainFrame()->UpdateDepotandChangeViews(REDRILL, key);
				}
			}
			else
				MainFrame()->UpdateDepotandChangeViews(REDRILL, key);
		}
		else
		{
			 // Temporarily disable redraws for both windows
			SetRedraw(FALSE);
			::SendMessage(m_changeWnd, WM_SETREDRAW, FALSE, 0);

			switch(pCmd->GetCommand())
			{
			case P4REVERT:
			case P4VIRTREVERT:
			case P4REVERTUNCHG:
				chainedCommands = OnP4RevertFile(pCmd);
				if (chainedCommands)
					iRedoOpenedFilter = 0;
				break;
			case P4ADD:
			case P4EDIT:
			case P4DELETE:
			case P4LOCK:
			case P4UNLOCK:
				ProcessStatListResults(wParam);
				break;
			default:
				ASSERT(0);
			}						   

			// Re-enable redraws for both windows
			SetRedraw(TRUE);
			::SendMessage(m_changeWnd, WM_SETREDRAW, TRUE, 0);

			 // And finally, make sure both windows redraw
			RedrawWindow();
			::RedrawWindow( m_changeWnd, NULL, NULL, RDW_INVALIDATE );
		}
	}

	pCmd->DeleteStatList();

	if( !chainedCommands || MainFrame()->IsQuitting() )
		pCmd->ReleaseServerLock();

	BOOL bOutputError = pCmd->GetOutputErrFlag();
	delete pCmd;
	
	if (iRedoOpenedFilter)
		OnRedoOpendList((WPARAM)iRedoOpenedFilter, 0);
	else if (bOutputError)
		::PostMessage(m_hWnd, WM_COMMAND, ID_VIEW_UPDATE, 0);
	return 0;
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::ProcessStatListResults(WPARAM wParam)
{
	CCmd_ListOpStat *pCmd= (CCmd_ListOpStat *) wParam;
	ASSERT_KINDOF(CCmd_ListOpStat, pCmd);
	
	// Get the results list, and update any files that are
	// present in the DepotTree
	HTREEITEM item;
	CObList *list=pCmd->GetStatList();

    InitFindItem();
	for(POSITION pos= list->GetHeadPosition(); pos!=NULL; )
	{
		CP4FileStats *fs= (CP4FileStats *) list->GetNext(pos);

		// Try to find the item
		item=FindItem(fs->GetDepotDir(), fs->GetDepotFilename(), FALSE);
		if(item==NULL)
			// It is not unusual to NOT find the file, since the user
			// could have initiated the command by clicking a folder
			// that had not yet been explored
			XTRACE(_T("ProcessListResults() item not found %s\n"), fs->GetFullDepotPath());
		else
		{
			// Update its properties in accordance with the command.  Remember
			// that CCmd_ListOpStat will return a sparse set of information.
			// In general only the file name and revision are present, though the
			// rev is not present for LOCK, UNLOCK.  For EDIT and DELETE we will
			// have additional info as returned by a call to ostat.

			CP4FileStats *treefs= m_FSColl.GetStats(GetLParam(item));
					
			switch(pCmd->GetCommand())
			{
			case P4ADD:
				treefs->SetNotInDepot(fs->IsNotInDepot());
			case P4EDIT:
			case P4DELETE:
				treefs->SetOpenAction(fs->GetMyOpenAction(), FALSE);
				treefs->SetHaveRev(fs->GetHaveRev());
				treefs->SetOpenChangeNum(fs->GetOpenChangeNum());
				break;
						
			case P4LOCK:
				treefs->SetLocked(TRUE, FALSE);
				break;
			case P4UNLOCK:
				treefs->SetLocked(FALSE, FALSE);
				break;
			default:
				ASSERT(0);
			}

			// Update the image in this window
			if (treefs->GetHeadRev() != fs->GetHeadRev() && fs->GetHeadRev())
			{
				treefs->SetHeadRev(fs->GetHeadRev());
				SetItemText(item, fs->GetFormattedFilename(GET_P4REGPTR()->ShowFileType()));
			}
			SetImage(item, TheApp()->GetFileImageIndex(treefs));
			
		}// if found

		// Always notify the changes window
		::SendMessage(m_changeWnd, WM_UPDATEOPEN, (WPARAM) fs, (LPARAM) pCmd->GetCommand());
		
		MainFrame()->ClearStatus();
	
	}//for each file
}

void CDepotTreeCtrl::ProcessGetListResults(UINT command, CStringList *list)
{
	CString path, name, listRow, fname, fRev;
	long rev;
	DWORD index;
	HTREEITEM item;
	int lastSlash;
	POSITION pos;
	CP4FileStats *fs;
		
	pos=list->GetHeadPosition();
    InitFindItem();
	for(int i=0; i < list->GetCount(); i++)
	{
		BOOL differentRev=FALSE;

		listRow=fname=list->GetNext(pos);

		int separator, pound;
		BOOL rowError=FALSE;

		//////////////
		// Separate the filename from the action description
		// For all operations but lock and unlock, this amounts to a
		// quick search for '#'.  In the case of the lock commands, 
		// there is no revision number, so search for the action text itself
	
		pound= listRow.Find(_T('#'));   
		if(pound == -1)
		{
			rowError=TRUE;
			break;
		}

		separator= pound+1;
		int len= listRow.GetLength();
		for( ; separator < len ; separator++)
		{
			if(listRow[separator]==_T(' ') && listRow[separator+1]==_T('-') && listRow[separator+2]==_T(' '))
			break;
		}
		if(separator==len)
			rowError=TRUE;

		if(rowError)
		{ 
			// doesnt look like a valid row, report it and skip it
			ASSERT(0); 
			AddToStatus(listRow, SV_WARNING);
			continue; 
		}	
					
		fRev=fname.Mid(pound+1);
		rev=_ttol(fRev);
		
		fname=fname.Left(separator);    // full name w/ revision
		lastSlash=fname.ReverseFind(_T('/')); 
		path=fname.Left(lastSlash+1);	// path with trailing /
		name=fname.Mid(lastSlash+1);	// file name with revision

		// Find the item
		item=FindItem(path, name, FALSE);
		if(item==NULL)
		{
			// If the user did a GET from a folder, we wont be finding
			// files unless we already explored the folder
			XTRACE(_T("ProcessGetListResults() item not found %s\n"), listRow);
			fs= (CP4FileStats *) new CP4FileStats;
			fs->SetDepotPath(fname.Left(pound));  
			fs->SetHaveRev(rev);

			// Send the filename to the changes window for update
			::SendMessage(m_changeWnd, WM_SETUNRESOLVED, (WPARAM) fs, 0); 
			delete fs;
		}
		else
		{
			// Update its properties
			index=GetLParam(item);
			fs=m_FSColl.GetStats(index);
			// We need to do this because depot paths are not stored after
			// files are added to the tree
			fs->SetDepotPath(fname.Left(pound));  

			CString digest = _T("");
			switch(command)
			{
			case WM_P4GET:
				if(rev != fs->GetHaveRev())
					differentRev=TRUE;
				fs->SetHaveRev(rev);
				TheApp()->localDigest(fs, &digest);
				fs->SetDigest(&digest);
				break;
			case WM_P4UNGET:
				fs->SetHaveRev(0);
				fs->SetDigest(&digest);
				break;
			default:
				ASSERT(0);
			}

			// Rule: Only show deleted file if user has opted to see deleted files
			// Exception: If the head rev is deleted and user has a rev < head, then
			//            we always show the file
			if(command == WM_P4UNGET && fs->GetHeadAction() == F_DELETE && 
				!GET_P4REGPTR()->ShowDeleted() && fs->GetHaveRev() == 0 ) 
			{
						DeleteLeaf(item);
			}
			else
			{
				// Update the image in this window
				SetImage(item, TheApp()->GetFileImageIndex(fs));
			
				// A get will require update of the file reve and possibly file type info
				SetItemText(item, fs->GetFormattedFilename(GET_P4REGPTR()->ShowFileType()));
				if(differentRev && fs->IsMyOpen())
					// Send the filename to the changes window for update
					::SendMessage(m_changeWnd, WM_SETUNRESOLVED, (WPARAM) fs, 0); 
			}
		} // if item found
	} // for each item
	
	MainFrame()->ClearStatus();
}



/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4UpdateOpen(WPARAM wParam, LPARAM lParam)
{
	XTRACE(_T("OnP4UpdateOpen() wParam=%ld lParam=%ld\n"), wParam, lParam);

	// Changes view sends this after reverting a file
	CString *fileName= (CString *) wParam;

	int lastSlash=fileName->ReverseFind(_T('/'));
	ASSERT(lastSlash != -1);

	CString path=fileName->Left(lastSlash+1);
	CString file=fileName->Mid(lastSlash+1);
	
    InitFindItem();
	HTREEITEM item=FindItem(path, file, FALSE);

	if(item == NULL)
		// F_ADD and some F_INTEG files wont be in this view! 
		return 0;
		
	DWORD index=GetLParam(item);	
	CP4FileStats *fs=m_FSColl.GetStats(index);
	
	if(fs->GetHeadRev() || GET_P4REGPTR()->ShowEntireDepot() == SDF_LOCALTREE)
	{
		fs->SetLocked(FALSE, FALSE);
		fs->SetOpenAction(0, FALSE);
		fs->SetOpenChangeNum(0);
		if (!fs->GetHeadRev() && fs->IsNotInDepot())
			fs->SetHaveRev(0);
		SetImage(item, TheApp()->GetFileImageIndex(fs));
	}
	else
	{
		// A stranded file was reverted, and will no longer be in the 
		// client view of the depot
		ASSERT(GET_P4REGPTR()->ShowEntireDepot() != SDF_DEPOT);
	}
		
	return 0;
}


/*
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::OnP4RevertFile(CCmd_ListOpStat *pCmd)
{
	CStringList *list = pCmd->GetList();
	CStringList reverted;
	CString listRow, fname;
	POSITION pos;
	
	pos=list->GetHeadPosition();
	for(int i=0; i < list->GetCount(); i++)
	{
		listRow=list->GetNext(pos);
		if(listRow.Find(_T("reverted")) != -1 || listRow.Find(_T("was add, abandoned")) != -1)
		{
			fname= listRow.Left(listRow.ReverseFind(_T('#')));
			reverted.AddHead(fname);
					
			// Send the message to update our own window
			SendMessage(WM_UPDATEOPEN, (WPARAM) &fname, 0);
		}
	}
	
	// Make sure changes window gets updated
	if(reverted.GetCount() > 0)
	{
		::SendMessage(m_changeWnd, WM_REVERTLIST, (WPARAM) &reverted, 0);
		::RedrawWindow(m_changeWnd, NULL, NULL, RDW_INVALIDATE);
	}

	if (pCmd->GetRevertUnchgAfter())
	{
		CCmd_ListOpStat *pCmd2= new CCmd_ListOpStat;
		pCmd2->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, pCmd->GetServerKey() );
		pCmd2->SetNbrChgedFilesReverted(list->GetCount());
		pCmd2->SetRedoOpenedFilter(pCmd->GetRedoOpenedFilter());
		if( pCmd2->Run( &m_StringList2, P4REVERTUNCHG ) )
		{
			MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_REVERT) );
			return TRUE;
		}
		else
		{
			delete pCmd2;
			MainFrame()->UpdateDepotandChangeViews(REDRILL, pCmd->GetServerKey());
		}
	}
	else
		MainFrame()->ClearStatus();

	return FALSE;
}


/*
	_________________________________________________________________

	Start the refresh of the changes window, called at the end of
	a CStat operation.  Can be called directly from OnP4Depots for 
	a full refresh, otherwise called after files and folders are fetched
	_________________________________________________________________
*/

void CDepotTreeCtrl::StartChangeWndUpdate(int key)
{
	XTRACE(_T("StartChangeWndUpdate()\n"));

	if( !m_ClearedChangeWnd )
		::SendMessage(m_changeWnd, WM_INITTREE, 0, 0);

	MainFrame()->UpdateStatus(LoadStringResource(IDS_CHECKING_PENDING_CHANGES));

	// And start the refresh of the changes tree
	CString clistr = GET_P4REGPTR()->GetP4Client();
	CCmd_Changes *pCmd= new CCmd_Changes;
	pCmd->Init( m_changeWnd, RUN_ASYNC, HOLD_LOCK, key);
	if( !pCmd->Run( PENDING_CHANGES, 
			(GET_SERVERLEVEL() >= 19 && GET_P4REGPTR()->GetUseLongChglistDesc() > 31) ? 2 : 0, 
			NULL, 0, FALSE, NULL,
			GET_P4REGPTR()->GetEnablePendingChgsOtherClients() 
				&& !GET_P4REGPTR()->FilterPendChgsByMyClient() ? NULL : &clistr ) )
	{
		RELEASE_SERVER_LOCK(key);
		MainFrame()->SetLastUpdateTime(UPDATE_FAILED);
		MainFrame()->ClearStatus();
		delete pCmd;
	}
}



/*
	_________________________________________________________________

	We just got a list of dirs and files, because a tree node is expanding.  So put the
    dirs into the tree and call fstat to get the files at the same level in
    the tree.
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4ExpandTree ( WPARAM wParam, LPARAM lParam )
{
	XTRACE(_T("OnP4Dirs() wParam=%ld lParam=%ld\n"), wParam, lParam);

	CCmd_DirStat *pCmd= ( CCmd_DirStat * ) wParam;
	ASSERT_KINDOF(CCmd_DirStat, pCmd);

	// Data used for triggering an auto expand 
	// of a folder that contains only a single subfolder
	HTREEITEM lastDir = NULL;
	int nbrDirs = 0;
	int children = 0;

	// Set the context
	m_LastPathItem= pCmd->GetItemRef();
	m_LastPath = pCmd->GetTextRef();
	m_UpdateType= UPDATE_EXPAND;

	if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		m_LastPath.Replace(_T('/'), _T('\\'));

	// We passed in a 'new' stringlist, so delete it
	delete pCmd->GetSpecList();

	MainFrame()->ClearStatus();

	if(pCmd->GetError( ) )
	{
		MainFrame()->SetLastUpdateTime(UPDATE_FAILED);
		if( m_LastPathItem != NULL )
		{
			SetLParam( m_LastPathItem, EXPAND_FOLDER );
			SetChildCount( m_LastPathItem, 1 );
		}
		m_ExpandDepotContinue = FALSE;
	}
	else
	{
		POSITION pos;
		SetRedraw(FALSE);

		HTREEITEM iNode = NULL;

		// First insert all directories
		CStringList *dirs= pCmd->GetDirs();
		pos= dirs->GetHeadPosition();
		while(pos != NULL)
		{
			// Get the dir
			CString dir= dirs->GetNext(pos) + m_SlashChar;
			if (InsertDir( dir ))
			{
				iNode = TreeView_GetParent( m_hWnd, m_LastPathItem );
				lastDir = m_LastPathItem;
				children++;
				nbrDirs++;
			}
		}

		// Then insert all files
		CObList const *files= pCmd->GetFiles();
		ASSERT_KINDOF(CObList, files);

		BOOL bSbyE = FALSE;
		if (GET_P4REGPTR()->SortByExtension()
		 && (files->GetCount() > _ttoi(GET_P4REGPTR()->GetExtSortMax())))
		{
			GET_P4REGPTR()->SetSortByExtension( FALSE );
			CString txt;
			txt.FormatMessage(IDS_TOO_MANY_TO_SORT_BYEXT_n, files->GetCount());
			AddToStatus( txt, SV_WARNING );
			bSbyE = TRUE;
		}

		pos= files->GetHeadPosition();
		while(pos != NULL)
		{
			// Get the filestats
			CP4FileStats *stats= (CP4FileStats *) files->GetNext(pos);
			ASSERT_KINDOF(CP4FileStats, stats);

			// Just another successfully retrieved row - do NOT delete pCmd
			if(stats->GetHeadAction() != F_DELETE || GET_P4REGPTR()->ShowDeleted() || stats->GetHaveRev() != 0 )
			{
				InsertFromFstat(stats);
				iNode= m_LastPathItem;
				children++;
			}
			else
				delete stats;
		}

		if (bSbyE)
			GET_P4REGPTR()->SetSortByExtension( TRUE );

		if( children )
		{
			if( iNode != NULL )
			{
				if (TreeView_Expand( m_hWnd, iNode, TVE_EXPAND ))
					SetLParam( iNode, FOLDER_ALREADY_EXPANDED);
			}
			else
				ASSERT(0); // Why didnt iNode get set?
		}
		else
		{
			// Leave the node ready to try expanding
			if( m_LastPathItem != NULL )
			{
				SetLParam( m_LastPathItem, EXPAND_FOLDER );
				CString txt= GetItemText( m_LastPathItem );
				if( txt.Find( g_TrulyEmptyDir ) == -1 )
				{
					txt+= g_TrulyEmptyDir;
					SetItemText( m_LastPathItem, txt );
				}
				SetChildCount( m_LastPathItem, 0 );
			}
			else
				ASSERT(0);
		}

		SetRedraw(TRUE);
		RedrawWindow();
	} // if error

    delete pCmd;

	// if we are in the process of expanding down a branch of the depot tree
	// to a given path (m_ExpandPath), now that we have finished expanding a
	// node, we need to restart ExpandDepotString() with a FALSE to indicate
	// that it should continue from where it left off.
	if (m_ExpandDepotContinue)
	{
		 ExpandDepotString(m_ExpandPath, FALSE);
	}
	// else if there is only a single subfolder in this folder, expand it
	else if (nbrDirs == children && nbrDirs == 1 && GET_P4REGPTR()->AutoTreeExpand())
	{
		if (!TreeView_Expand( m_hWnd, lastDir, TVE_EXPAND ))
			ApplySelectAtts(GetSelectAtts());
	}
	else
	{
		ApplySelectAtts(GetSelectAtts());
	}
	return 0;
}



/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4DirStat(WPARAM wParam, LPARAM lParam)
{
	XTRACE(_T("OnP4DirStat() wParam=%ld lParam=%ld\n"), wParam, lParam);
    CCmd_DirStat *pCmd;

    if(lParam == 0)   // completion
	{
        pCmd= (CCmd_DirStat *) wParam;
	    ASSERT_KINDOF(CCmd_DirStat, pCmd);

		if(pCmd->GetError() || MainFrame()->IsQuitting())
		{
			pCmd->ReleaseServerLock();
			MainFrame()->SetLastUpdateTime(UPDATE_FAILED);
			MainFrame()->ClearStatus();
			delete pCmd;
			return 0;
		}
    }
    else
    {
        // This command does not return intermediate results
        ASSERT(0);
        return 0;
    }

	int key= pCmd->GetServerKey( );
	BOOL bNoExpand = TRUE;
    POSITION pos;
    SetRedraw(FALSE);

    // Set the context
    m_LastPathItem= TVI_ROOT;
    m_LastPath = g_sSlashes;
	m_UpdateType= UPDATE_REDRILL;

    // First insert all directories
    CStringList *dirs= pCmd->GetDirs();
    pos= dirs->GetHeadPosition();
	while(pos != NULL && m_DepotCount)
	{
		// Get the dir
		CString dir= dirs->GetNext(pos) + m_SlashChar;
		InsertDir( dir );
    }

    // Then insert all files
    CObList const *files= pCmd->GetFiles();
    ASSERT_KINDOF(CObList, files);
    		
	BOOL bSbyE = FALSE;
	if (GET_P4REGPTR()->SortByExtension()
	 && (files->GetCount() > _ttoi(GET_P4REGPTR()->GetExtSortMax())))
	{
		GET_P4REGPTR()->SetSortByExtension( FALSE );
		CString txt;
		txt.FormatMessage(IDS_TOO_MANY_TO_SORT_BYEXT_n, files->GetCount());
		AddToStatus( txt, SV_WARNING );
		bSbyE = TRUE;
	}

	pos= files->GetHeadPosition();
	while(pos != NULL && m_DepotCount)
	{
		// Get the filestats
		CP4FileStats *stats= (CP4FileStats *) files->GetNext(pos);
		ASSERT_KINDOF(CP4FileStats, stats);

		// Just another successfully retrieved row - do NOT delete pCmd
		if(stats->GetHeadAction() != F_DELETE || GET_P4REGPTR()->ShowDeleted() || stats->GetHaveRev() != 0 )
		{
			InsertFromFstat(stats);
			if( m_LastPathItem != NULL && m_LastPathItem != TVI_ROOT)
			{
				if (GetLParam( m_LastPathItem) != FOLDER_ALREADY_EXPANDED)
				{
					XTRACE(_T("OnP4DirStat() marking %s expanded\n"), 
							   m_LastPathItem != TVI_ROOT ? GetItemPath(m_LastPathItem) : _T("ROOT"));
					SetLParam( m_LastPathItem, FOLDER_ALREADY_EXPANDED);
				}
			}
		}
		else
			delete stats;
	} 

	if (bSbyE)
		GET_P4REGPTR()->SetSortByExtension( TRUE );
	   
    // Then expand all previously expanded nodes that still exist
    pos= m_ExpandedNodeList.GetHeadPosition();
    while(pos != NULL && m_DepotCount)
	{
        int commonLength;
		CString nodeName= m_ExpandedNodeList.GetNext(pos);
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			if ((nodeName.GetLength() == 2) && (nodeName.GetAt(1) == _T(':')))
				nodeName += _T('\\');
			m_LastPathItem = TreeView_GetRoot(m_hWnd);
			m_LastPath = TheApp()->m_ClientRoot;
			m_LastPath.TrimRight(_T("\\"));	/* to handle c:\ */
			m_LastPath += _T('\\');
			commonLength = m_LastPath.GetLength();
		}
		else
		{
	        m_LastPathItem= TVI_ROOT;
			m_LastPath = g_sSlashes;
			commonLength = 2;
		}
		if (nodeName.GetAt(1) == _T(':'))
			nodeName.Replace(_T('/'), _T('\\'));
		else
			ReplaceMBCS(nodeName, _T('\\'), _T('/'));
		if (((m_SlashChar == _T('/'))  && (nodeName.GetAt(1) == _T(':')))
		 || ((m_SlashChar == _T('\\')) && (nodeName.GetAt(0) == _T('/'))))
		{
			// the node to expand syntax doesn't match our tree syntax now
			// so just bail after setting 'bNoExpand' so that the reselection
			// of any previous selections will know that it must expand nodes
			bNoExpand = FALSE;
			continue;
		}

        FindParentDirectory( nodeName, commonLength );
		if( m_LastPathItem != NULL && m_LastPathItem != TVI_ROOT)
        {
            XTRACE(_T("OnP4DirStat() expanding=%s\n"), nodeName);
			SetLParam( m_LastPathItem, FOLDER_ALREADY_EXPANDED);
            if(!TreeView_Expand( m_hWnd, m_LastPathItem, TVE_EXPAND ))
			{
				XTRACE(_T("OnP4DirStat() expanding failed\n"));
				SetLParam( m_LastPathItem, EXPAND_FOLDER);
			}
        }
	}

    // Then, see if the previous first visible node is in the tree
    // and if so, scroll it into view
    HTREEITEM firstItem= NULL;

    int len= m_FirstVisibleNodeText.GetLength();
    if( len > 0 )
    {
        if( m_FirstVisibleNodeText[len-1] == m_SlashChar )
        {
            // It's a depot or folder, so initialize m_LastPath and m_LastPathItem
            // and then try to find the parent folder
            m_LastPathItem = m_Root;
            m_LastPath= g_sSlashes;
            int commonLength= m_LastPath.GetLength();
            FindParentDirectory( m_FirstVisibleNodeText, commonLength );

            if( Compare( m_LastPath, m_FirstVisibleNodeText ) == 0 )
                firstItem= m_LastPathItem;
        }
        else
        {
            // It's a file, so split apart the path and filename and then
            // try a FindItem() call
            CString path= m_FirstVisibleNodeText.Left(ReverseFindMBCS(m_FirstVisibleNodeText, m_SlashChar)+1);
            CString name= m_FirstVisibleNodeText.Mid(ReverseFindMBCS(m_FirstVisibleNodeText, m_SlashChar)+1);
			
			InitFindItem();
            firstItem= FindItem( path, name, FALSE );
        }
    }

    if( firstItem != NULL )
        ScrollToFirstItem( firstItem );

	// Now if there were any items selected, reslect them
	if  (!m_SavedSelectionSet.IsEmpty())
	{
		int i;
		int count = m_SavedSelectionSet.GetCount();
		CView* pView = MainFrame()->GetActiveView ();
		HTREEITEM item = NULL;
		HTREEITEM itemPrv = NULL;
		CDWordArray treeItems;
		TCHAR   lastSlashChr = m_SavedSelectionSet.GetTail().GetAt(0) == _T('/') ? _T('/') : _T('\\');
		int     lastSlashCur = -2;
		int     lastSlashPrv = -2;
		CString lastSlashStr = "";
		CString prevPath = "";
		do
		{
			BOOL bQuick;
			CString itemPath;
			itemPath = m_SavedSelectionSet.GetHead();
			m_SavedSelectionSet.RemoveHead();

			// Do a quick check to see if we are trying to find
			// it in the same folder - if so, set a flag
			lastSlashCur = ReverseFindMBCS(itemPath, lastSlashChr);
			if (lastSlashPrv == lastSlashCur
			 && itemPath.Left(lastSlashCur) == lastSlashStr)
			{
				bQuick = itemPrv != NULL;
			}
			else
			{
				bQuick = FALSE;
				if (lastSlashCur > 0)
				{
					lastSlashStr = itemPath.Left(lastSlashCur);
					lastSlashPrv = lastSlashCur;
				}
			}

			// Now clear away any trailing ...s
			if (itemPath.Right(3) == _T("..."))
			{
				if (itemPath.Right(4) == _T("/..."))
					itemPath = itemPath.Left(itemPath.GetLength() - 4);
				else if (itemPath.Right(4) == _T("\\..."))
					itemPath = itemPath.Left(itemPath.GetLength() - 4);
			}

			// Can we use the quick method?
			if (bQuick)
			{
				// look for it amongst our siblings
				BOOL bUp = itemPath.CompareNoCase(prevPath) < 0;
				if ((itemPrv = FindDepotSibling(itemPath, itemPrv, bUp)) != NULL)
				{
					treeItems.Add((DWORD)itemPrv);
					prevPath = itemPath;
					continue;
				}
			}

			// Go find it in the depot
			ExpandDepotString(itemPath, TRUE, bNoExpand, key, TRUE);
			if (count == 1)									// If only one thing was selected
				break;										//	whatever we found is good enough
			if (!GetSelectedCount())						// If nothing got selected
			{												//	we don't want to persue this
				ASSERT(0);									//	or the GetSelectedItem() will
				continue;									//	return a NULL
			}
			item = GetSelectedItem(0);
			CString itemText = GetItemText(item);			// Yields last element of found path
			if ((i = itemText.ReverseFind(_T('#'))) != -1)	// If there is a # after the item name
				itemText = itemText.Left(i);				//	grab only the actual name
			itemText.TrimLeft();
			itemText.TrimRight();
			TCHAR slashChar = itemPath.GetAt(0) == _T('/') ? _T('/') : _T('\\');
			CString lastElem;
			if ((i = ReverseFindMBCS(itemPath, slashChar)) != -1)	// If there is a / in the full depot path
				lastElem = itemPath.Mid(i+1);						//	grab only the last element
			else
				lastElem = itemPath;
			lastElem.TrimLeft();
			lastElem.TrimRight();
			if (lastElem == itemText)		 // Did we actually find the thing we were looking for?
			{
				treeItems.Add((DWORD)item ); //  remember it cuz future calls to ExpandDepotString() will deselect it
				itemPrv  = item;
				prevPath = itemPath;
			}
		} while (!m_SavedSelectionSet.IsEmpty());

		// Get the update of the changes window underway before we set all the selections
		// because some code in StartChangeWndUpdate() causes the selections to be reset
		StartChangeWndUpdate( key );
		if (count > 1)
		{
			UnselectAll();
			for (i = treeItems.GetSize(); i--; )
			{
				item = (HTREEITEM)treeItems.GetAt(i);
				SetSelectState( item, TRUE );
			}
			// Make sure selection set is properly displayed
			SetAppearance(FALSE, TRUE, FALSE);
		}
		else if (GetSelectedCount())
		{
			item = GetSelectedItem(0);
		}
	    if( item != NULL )
	        TreeView_EnsureVisible( m_hWnd, item );
		MainFrame()->SetActiveView(pView, TRUE);
	}
	else
	{
		// Finally, get the update of the changes window underway
		StartChangeWndUpdate( key );
	}

	SetRedraw(TRUE);
	MainFrame()->ClearStatus();
	delete pCmd;

	return 0;
}


/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4FStat(WPARAM wParam, LPARAM lParam)
{
	XTRACE(_T("OnP4FStat() wParam=%ld lParam=%ld\n"), wParam, lParam);
    CCmd_Fstat *pCmd;

    if(lParam == 0)   // completion
	{
        pCmd= (CCmd_Fstat *) wParam;
	    ASSERT_KINDOF(CCmd_Fstat, pCmd);

		if(pCmd->GetError() || MainFrame()->IsQuitting())
		{
			pCmd->ReleaseServerLock();
			MainFrame()->SetLastUpdateTime(UPDATE_FAILED);
			MainFrame()->ClearStatus();
			delete pCmd;
			return 0;
		}
   
             	
		// We only run fstat for a full update against a 97.3 server
		ASSERT( m_UpdateType == UPDATE_FULL && GET_SERVERLEVEL() < 4 );
		
        int key= pCmd->GetServerKey( );
		StartChangeWndUpdate( key );
		
		MainFrame()->ClearStatus();
		delete pCmd;
		return 0;
	}
	else
	{
        // For postings of CP4FileStats lists, we need the list 
        // as well as a pointer to the command, so a primitive
        // wrapper is used
        CFstatWrapper *pWrap= (CFstatWrapper *) wParam;
        pCmd= (CCmd_Fstat *) pWrap->pCmd;
	    ASSERT_KINDOF(CCmd_Fstat, pCmd);
		CObList *list= (CObList *) pWrap->pList;
		ASSERT_KINDOF(CObList, list);

        // Get the context from the command
        m_LastPathItem= pCmd->GetItemRef();
        m_LastPath= pCmd->GetTextRef();
        m_UpdateType= pCmd->GetUpdateType();
		
		POSITION pos= list->GetHeadPosition();

        SetRedraw(FALSE);
        InitFindItem();
		while(pos != NULL)
		{
			// Get the filestats and insert it
			CP4FileStats *stats= (CP4FileStats *) list->GetNext(pos);
			ASSERT_KINDOF(CP4FileStats, stats);
		
			// Just another successfully retrieved row - do NOT delete pCmd
			InsertFromFstat(stats);

		} // while row batch not done

        SetRedraw(TRUE);

		delete list;
        delete pWrap;  // do NOT delete pCmd, since it's still running!
		return 0;
	} // a batch of rows, we'll be called again so dont delete pCmd
}


/*
	_________________________________________________________________
*/


// Delete a leaf from the tree, making sure childless parent(s) are not left behind
void CDepotTreeCtrl::DeleteLeaf(HTREEITEM item)
{
	XTRACE(_T("DeleteLeaf()\n"));

	ASSERT(item != NULL);
	ASSERT(item != m_Root);

	HTREEITEM parent=TreeView_GetParent(m_hWnd, item);
	HTREEITEM child;

	// First delete the item number from the filestats obj, so 
    // tree-related functions don't attempt to monkey with a 
    // non-existent tree node

    
    if( ITEM_IS_FILE(item) )
    {
        CP4FileStats *fs= (CP4FileStats *) m_FSColl.GetStats( GetLParam(item) );
        ASSERT_KINDOF(CP4FileStats, fs);
        fs->SetUserParam( NULL );
    }
    else
        ASSERT(0);

	TreeView_DeleteItem(m_hWnd, item);

	// Make sure childless parent directori(es) are not left behind
	while(parent != NULL && TreeView_GetChild(m_hWnd, parent) == NULL )
	{
		child=parent;
		parent=TreeView_GetParent(m_hWnd, child);
		TreeView_DeleteItem(m_hWnd, child);
	}	
}


BOOL CDepotTreeCtrl::FindDirInDepotFilter(LPCTSTR path)
{
	long lgth = lstrlen(path);
	CString str;
	for( POSITION pos= m_DepotFilterList.GetHeadPosition(); pos != NULL; )
	{
		str = m_DepotFilterList.GetNext(pos);
		if (str.GetLength() >= lgth)
		{
			str = str.Left(lgth);
			if (Compare(str, path) == 0)
				return TRUE;
		}
	}
	return FALSE;
}

BOOL CDepotTreeCtrl::InsertDir( CString &path )
{
	if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
	{
		path.TrimRight(_T('/'));
		if (m_LastPathItem == TVI_ROOT)
		{
		    m_LastPathItem = TreeView_GetRoot(m_hWnd);
			m_LastPath = TheApp()->m_ClientRoot;
		}
		TrimRightMBCS(m_LastPath, _T("\\/"));
		m_LastPath += _T('\\');
	}

	XTRACE(_T("InsertDir() path=%s\n"), path);

	int commonLength= nCommon(path, m_LastPath);

	// back up the tree to the last common node
	int end;
	TCHAR slashbuf[2];
	slashbuf[0] = m_SlashChar;
	slashbuf[1] = _T('\0');
	while(m_LastPath.GetLength() > commonLength)
	{
		if (m_LastPath == CString(slashbuf))
			break;
	    m_LastPathItem= TreeView_GetParent(m_hWnd, m_LastPathItem);
		TrimRightMBCS(m_LastPath, slashbuf);
		end = ReverseFindMBCS(m_LastPath, m_SlashChar);
		m_LastPath=m_LastPath.Left(end+1);
        commonLength= nCommon(path, m_LastPath);
	}

	ASSERT( commonLength < path.GetLength() || path == m_LastPath); 

	// If we are filtering the depot, look for the Dir in the filter list
	// Note: have to do this after m_LastPath is set correctly
	if (m_FilterDepot && !FindDirInDepotFilter(path))
		return FALSE;

	// May add to commonLength if folders found.  Will also remove
	// g_TrulyEmptyDir text if found, since presumably it is now
	// a parent directory
	FindParentDirectory(path, commonLength);

    // The search for parent directory should at least find a depot node:
    if(commonLength < path.GetLength())
	{
        // Add subdirs as required
		while( path.GetLength() > m_LastPath.GetLength() )
		{
			// Directories are prefixed w/ a space so they sort to the top
			// Yes, a grim fix, but better than putting all sort info into lParam
			// and using SortChildrenCB() 
			CString temp= path.Mid(m_LastPath.GetLength());
            if( FindMBCS(temp, m_SlashChar) > 0 ) 
			    temp=temp.Left(FindMBCS(temp, m_SlashChar));

            XTRACE(_T("Insert dir: path=%s m_LastPath=%s\n"), path, m_LastPath);

            // The parent of the folder being added clearly has been explored
            SetLParam( m_LastPathItem, FOLDER_ALREADY_EXPANDED);

			if(m_LastPathItem==TVI_ROOT || m_LastPathItem==NULL)
				m_LastPathItem=Insert(_T("//")+temp, CP4ViewImageList::VI_DEPOT, 
                                           EXPAND_FOLDER, TVI_ROOT);
			else
				m_LastPathItem=Insert(_T(" ")+temp, CP4ViewImageList::VI_FOLDER, 
                                           EXPAND_FOLDER, m_LastPathItem);
                
            m_LastPath += temp;
	    	m_LastPath += m_SlashChar;
         } //while
    } // if commonlength < path.GetLength
	return TRUE;
}


BOOL CDepotTreeCtrl::FindFileInDepotFilter(LPCTSTR path)
{
	for( POSITION pos= m_DepotFilterList.GetHeadPosition(); pos != NULL; )
	{
		if (Compare(m_DepotFilterList.GetNext(pos),path) == 0)
			return TRUE;
	}
	return FALSE;
}

// This routine assumes m_LastPath is valid
LRESULT CDepotTreeCtrl::InsertFromFstat( CP4FileStats *stats )
{
    // Under 98.2 API, insertion of files under unexplored folders is avoided, so
    // the folders can be properly explored later.  Under any api, insertion of 
    // files is skipped when the file is deleted and showdeleted is not true.
    BOOL skipFileInsert=FALSE;

	CString temp;
    CString path;

	// full depot paths that are the empty string mean ingnore this record
	if (!stats->GetFullDepotPath()[0])
	{
		skipFileInsert=TRUE;
		goto AfterPath;
	}
	else
	{
		XTRACE(_T("InsertFromFstat() path=%s file=%s\n"), stats->GetDepotDir(), stats->GetDepotFilename());
		XTRACE(_T("InsertFromFstat() deleting=%d\n"), (stats->GetHeadAction()==F_DELETE));
	}

	// Rule: Only show deleted file if user has opted to see deleted files
	// Exception: If the head rev is deleted and user has a rev < head, then
	//            we always show the file
	if (stats->GetHeadAction() == F_DELETE && !GET_P4REGPTR()->ShowDeleted()
		&& stats->GetHaveRev() == 0 ) 
	{
		skipFileInsert=TRUE;
		goto AfterPath;
	}

	// if we are only showing files not in the depot
	// skip any file with a head action
	if (MainFrame()->m_ShowOnlyNotInDepot && stats->GetHeadAction())
	{
		skipFileInsert=TRUE;
		goto AfterPath;
	}

	// If we are filtering the depot, look for the file in the filter list
	if (m_FilterDepot 
	 && !FindFileInDepotFilter(m_SlashChar == _T('/') ? stats->GetFullDepotPath() 
													  : stats->GetFullClientPath()))
	{
		skipFileInsert=TRUE;
		goto AfterPath;
	}

	// Go up the tree from m_LastPath as required to find a common point, and
	// set m_LastParentItem to point to that node.  Then add subdirs as required,
    // except that if m_UpdateType== UPDATE_INC_982API we will add no more than
    // one node deep.

    path = GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT 
		 ? stats->GetClientDir() : stats->GetDepotDir();
		
	if(Compare(path, m_LastPath)!=0)
	{
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			if (m_LastPathItem == TVI_ROOT)
			{
				m_LastPathItem = TreeView_GetRoot(m_hWnd);
				m_LastPath = TheApp()->m_ClientRoot;
			}
			TrimRightMBCS(m_LastPath, _T("\\/"));
			m_LastPath += _T('\\');
		}

		int commonLength= nCommon(path, m_LastPath);

		// back up the tree to the last common node
		int end;
		while(m_LastPath.GetLength() > commonLength)
		{
			m_LastPathItem= TreeView_GetParent(m_hWnd,m_LastPathItem);
			m_LastPath = m_LastPath.Left(m_LastPath.GetLength()-1);
			end = ReverseFindMBCS(m_LastPath, m_SlashChar);
			ASSERT(end != -1);
			m_LastPath=m_LastPath.Left(end+1);
		}

        commonLength= m_LastPath.GetLength();

		if(commonLength < path.GetLength())
			// May add to commonLength if folders found.  Will also remove
			// g_TrulyEmptyDir text if found, since presumably it is now
			// a parent directory
			FindParentDirectory(path, commonLength);

        if(commonLength < path.GetLength())
		{
			// Add subdirs as required
			while( path.GetLength() > m_LastPath.GetLength() )
			{
				// Directories are prefixed w/ a space so they sort to the top
				// Yes, a grim fix, but better than putting all sort info into lParam
				// and using SortChildrenCB() 
				temp=path.Mid(m_LastPath.GetLength());
				temp=temp.Left(FindMBCS(temp, m_SlashChar));

                XTRACE(_T("Insert dir: path=%s m_LastPath=%s\n"), path, m_LastPath);

				if(m_LastPathItem==TVI_ROOT || m_LastPathItem==NULL)
					m_LastPathItem=Insert(_T("//")+temp, CP4ViewImageList::VI_DEPOT, 
                                            EXPAND_FOLDER, TVI_ROOT);
				else
					m_LastPathItem=Insert(_T(" ")+temp, CP4ViewImageList::VI_FOLDER, 
                                            EXPAND_FOLDER, m_LastPathItem);
                
                m_LastPath+=temp;
				m_LastPath+=m_SlashChar;
             
            } //while
        } // if commonlength < path.GetLength
    } // if path mismatch

AfterPath:
    if( skipFileInsert )
    {
        delete stats;
    }
    else
    {
	    //		Add the file. make sure it lives under a subdirectory
	    //
	    m_FSColl.SetStats( m_ItemCount, stats );

	    HTREEITEM newItem = Insert( stats->GetFormattedFilename( GET_P4REGPTR()->ShowFileType( )),
		                    TheApp( )->GetFileImageIndex( stats ), 	m_ItemCount, m_LastPathItem );

        m_ItemCount++;
	    stats->SetUserParam( (LPARAM) newItem );

	    if ( ! NEW_DEPOT_LISTING && m_ItemCount == 1 )
		    TreeView_Expand( m_hWnd, TreeView_GetRoot( m_hWnd ), TVE_EXPAND );

        #ifdef _DEBUG
	        // Asserts to sniff for the deadly QQ bug
	        CString temp2(GetItemText(newItem));
	        ASSERT(temp.Find( QQBUG_JOB000458 ) == -1);
        #endif
    }
	return 0;
}


void CDepotTreeCtrl::FindParentDirectory(CString path, int& commonLength)
{
	// We already backed up the tree till the path was only as long as commonLength
	// Now try to drill down for the rest of the path, since the folders may be
	// present in the tree.  The Compare() function will consider whether the server
	// is or isnt case sensitive.

	HTREEITEM item;

	while( path.GetLength() > m_LastPath.GetLength() )
	{
        // skip over chars till we hit a slash or the end of path
		int i = path.Find(m_SlashChar, commonLength);
		if(i == -1)
			i = path.GetLength();

		CString dirname= path.Mid(commonLength, i-commonLength);

        // Note that Win95 will bite it if you call TreeView_GetChild at the root.  Must
        // be some sort of 16 bit overflow inside brittleware.
        if( m_LastPathItem == m_Root)
		    item=TreeView_GetRoot(m_hWnd);
	    else
		    item=TreeView_GetChild(m_hWnd, m_LastPathItem);

		int skipAmt = (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT) 
					? TheApp( )->m_ClientRoot.GetLength( ) : 2;
		while(item != NULL)
		{
			CString txt;
            if( IsDepot(item) )
                // Skip over leading "//"
                txt=GetItemText(item).Mid(skipAmt);  
            else
                // Skip over the leading space
                txt=GetItemText(item).Mid(1);

			if(Compare(dirname,txt) == 0 || Compare(dirname+g_TrulyEmptyDir,txt)==0)
			{
				// Then store the dir name and tree item
				m_LastPathItem=item;
				m_LastPath+= dirname + m_SlashChar ;

				// Set new commonlength and get out of inner loop
				commonLength=m_LastPath.GetLength();
				break;  
			}
			item= TreeView_GetNextSibling(m_hWnd, item);
		}
                
        // We didnt get a match at this subdir level, so
        // the search is over
        if(item==NULL)
			break;
    } // while
    return;
}


/*
	_________________________________________________________________

	VIEWUPDATE STEP I:  
	Maxchanges is called to find out what the head change number of
	the server is.
	
	the first overloaded function is the protected member, and called when 
	the user hits F5 or the toolbar button.  it will attempt to do a full
    refresh of explored folders, re-expanding folders as required.
    
    the second overload is for when the mainframe calls us directly, and 
    supports incremental refresh for use after a change is submitted, etc
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnViewUpdate() 
{
	XTRACE(_T("OnViewUpdate()\n"));

    if( GET_SERVERLEVEL() == 3)
        OnViewUpdate( NO_REDRILL );
    else
	    OnViewUpdate( REDRILL );
}

void CDepotTreeCtrl::OnViewUpdate( BOOL redrill, int key /*=0*/ )
{
	if (MainFrame()->IsQuitting())
	{
		if (key)
			RELEASE_SERVER_LOCK(key);
		return;
	}

	// Make sure that autopolling gets turned back on if it was off
	MainFrame()->ResumeAutoPoll();

	// Save our selection set
	AssembleStringList( &m_SavedSelectionSet, FALSE, FALSE, TRUE );

	// Don't change the slach char before the selection set is assembled!
	TCHAR oldSlashChar = m_SlashChar;
	m_SlashChar = (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT) ? _T('\\') : _T('/');

	// If we are filtering the depot and changed the port or client,
	// we may need to clear the filter
	if (m_FilterDepot)
	{
		if (m_DepotFilterPort != GET_P4REGPTR()->GetP4Port())
			ClearDepotFilter(FALSE);
		else if ((m_DepotFilterClient != GET_P4REGPTR()->GetP4Client())
			  && (m_DepotFilterType == DFT_MYOPENED))
			ClearDepotFilter(FALSE);
	}

	if (m_SlashChar != oldSlashChar)
	{
		if (m_FilterDepot && (m_DepotFilterType == DFT_LIST))
			ConvertDepotFilterList(key);
	}

	m_ExpandDepotContinue = FALSE;
	if( redrill && GET_SERVERLEVEL() > 3 )
	{
		XTRACE(_T("OnViewUpdate - redrill\n"));
		m_UpdateType= UPDATE_REDRILL;
	}
	else
	{
		XTRACE(_T("OnViewUpdate - full\n"));
		m_UpdateType= UPDATE_FULL;
	}

    m_RunningUpdate= FALSE;
    m_ClearedChangeWnd= FALSE;

	if( m_UpdateType == UPDATE_REDRILL )
	{
		// Only 98.2 and higher servers support 'p4 dirs' to redrill
		ASSERT( GET_SERVERLEVEL() > 3 );
		RecordTreeExploration();
	}
	else
	{
		//	For full updates, we always clear any cached server info
		//  since connect parms may have been changed
		CLEAR_SERVERINFO();
	}

	// Clear both panes now, so if there is an error along the way
	// we dont leave rubble on the screen
	Clear();
	CString client= GET_P4REGPTR()->GetP4Client();
	if( !client.IsEmpty() )
	{
		m_ClearedChangeWnd= TRUE;
		::SendMessage(m_changeWnd, WM_INITTREE, 0, 0);
	}

	// Every refresh starts with fetching the depot list
	CCmd_Depots *pCmd = new CCmd_Depots;
	    
	pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key );
	BOOL ret = pCmd->Run();

	if( ret )
	{
		MainFrame()->UpdateStatus(LoadStringResource(IDS_REQUESTING_DEPOTS));
	}
	else
	{
		if( key )
			RELEASE_SERVER_LOCK( key );        
		delete pCmd;
		MainFrame()->ClearStatus();
	}
}


/*
	_________________________________________________________________
	
	VIEWUPDATE STEP II:
	We now have a list of local depots and a list of remote depots.  We also
	know what serverlevel we are running against.

	Put the depots into the tree and, depending upon serverlevel and update type,
	proceed to fetch more files and folders or jump to the pending changelist pane
	update  
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Depots ( WPARAM wParam, LPARAM lParam )
{
	XTRACE(_T("OnP4Depots() wParam=%ld lParam=%ld\n"), wParam, lParam);

	// Clear the menu if required adter deleting the pCmd
	BOOL bRmvMRUPcu = FALSE;

	// Re-enable folder expansion, now that we have the root of the
	// tree on hand
	m_RunningUpdate= FALSE;
    
	CCmd_Depots *pCmd= ( CCmd_Depots * ) wParam;
	ASSERT_KINDOF(CCmd_Depots, pCmd);

    // Before testing for an error, try to get the mainframe caption set
    MainFrame()->UpdateCaption( ) ;	

	if(pCmd->GetError( ) || MainFrame()->IsQuitting())
	{
		pCmd->ReleaseServerLock();
		MainFrame()->SetLastUpdateTime(UPDATE_FAILED);
		MainFrame()->ClearStatus();
	
        // If PWD is busted, try the update again
        if(GET_PWD_ERROR() && !pCmd->PWDDlgCancelled())
        {
            delete pCmd;
            OnViewUpdate(NO_REDRILL);
            return 0;
        }
        
		if (!LoadStringResource(IDS_UNABLE_TO_GET_CLIENT_DESCRIPTION).Compare(pCmd->GetErrorText()))
			bRmvMRUPcu = TRUE;	// remove the PCU because it has total failure

		goto depotend;
	}
	else
	{
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
#ifdef	_DEBUG
			CString root = TheApp()->m_ClientRoot;
#endif
			if (TheApp()->m_ClientRoot.GetAt(1) != _T(':'))
			{
				int key;
				if( (key = pCmd->GetServerKey( )) != 0 )
					RELEASE_SERVER_LOCK( key );        
				::PostMessage(MainFrame()->m_hWnd, WM_COMMAND, ID_VIEW_CLIENTVIEW, 0);
				AddToStatus( LoadStringResource(IDS_NOLOCALFORNULLROOT), SV_WARNING );
				goto depotend;
			}
		}

        CString base, folderName;
        POSITION pos;

		m_LocalDepotList.RemoveAll();
		CStringList *list= pCmd->GetLocalDepotList();
	    for ( pos= list->GetHeadPosition(); pos != NULL; )
	    {
		    folderName= list->GetNext(pos);
			m_LocalDepotList.AddTail(folderName);
            m_LastPathItem= m_Root;
            m_LastPath= g_sSlashes;
		    if( !FindFolder(folderName) )
		    {
			    // Change the m_LastPathItem so an error dialog
			    // doesnt spoil the fun when Cmd_dirstat finishes
			    m_LastPathItem = Insert( folderName, 
				    CP4ViewImageList::VI_DEPOT, EXPAND_FOLDER, TVI_ROOT );
			    m_DepotCount++;
                m_LastPath= folderName + _T("/");
            }
		}

		m_RemoteDepotList.RemoveAll();
		list= pCmd->GetRemoteDepotList();
		for ( pos= list->GetHeadPosition(); pos != NULL; )
	    {
		    folderName= list->GetNext(pos);
			m_RemoteDepotList.AddTail(folderName);
			// we ignore remote depots for local view other than keeping the
			// above list of them so that we can know what menu items to add/enable
			if (GET_P4REGPTR()->ShowEntireDepot() > SDF_DEPOT)
				continue;
            m_LastPathItem= m_Root;
            m_LastPath= g_sSlashes;
		    if( !FindFolder(folderName) )
		    {
			    // Change the m_LastPathItem so an error dialog
			    // doesnt spoil the fun when Cmd_dirstat finishes
			    m_LastPathItem = Insert( folderName, 
				    CP4ViewImageList::VI_REMOTEDEPOT, EXPAND_FOLDER, TVI_ROOT );
			    m_DepotCount++;
                m_LastPath= folderName + _T("/");
            }
		}

		if( m_DepotCount == 0 && GET_SERVERLEVEL() > 3 )
		{
            //	If no depots were found, there is nothing to display in the depot pane.  
            //  Try to provide a usefull message to the user, and proceed to the
            //  update of the pending changelists pane.
            int msgId;

			if( GET_P4REGPTR()->ShowEntireDepot() > SDF_DEPOT )
			{
				if( GET_P4REGPTR()->ShowDeleted() )
				{
					if( GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT )
						msgId = IDS_THERE_ARE_NO_FILES_IN_THE_DEPOT;
					else
						msgId = IDS_THERE_ARE_NO_FILES_IN_YOUR_CLIENT_VIEW;
				}
				else
				{
					if( GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT )
						msgId = IDS_THERE_ARE_NO_UNDELETED_FILES_IN_THE_DEPOT;
					else
						msgId = IDS_THERE_ARE_NO_UNDELETED_FILES_IN_YOUR_CLIENT_VIEW;
				}
			}
			else
			{
				if( GET_P4REGPTR()->ShowEntireDepot() == SDF_LOCALTREE )
					msgId = IDS_THERE_ARE_NO_FILES_UNDER_YOUR_CLIENT_ROOT;
				else
					msgId = IDS_THERE_ARE_NO_PERFORCE_FILES_UNDER_YOUR_CLIENT_ROOT;
			}
            
            AddToStatus( LoadStringResource(msgId), SV_WARNING ); 
            AfxMessageBox( msgId, MB_ICONINFORMATION );
		}
		
        // Make sure the window redraws
        SetRedraw(TRUE);
        RedrawWindow();

        int key= pCmd->GetServerKey( );

		MainFrame()->ResumeAutoPoll();	// just to clear the flag - we know we can poll now
        switch( m_UpdateType )
        {
		case UPDATE_FULL:
			if( NEW_DEPOT_LISTING )
				StartChangeWndUpdate( key);
			else
				RunCStat( key );
		    break;
        case UPDATE_REDRILL:
			RunRedrill( key );
		    break;
        default:
            ASSERT(0);
		}
       
	} // if error
depotend:
    delete pCmd;
	if (bRmvMRUPcu)	// Must delete pCmd before removing MRU PCU
	{
		CMenu *pMenu= MainFrame()->GetMenu();
		if (pMenu != NULL)
		{
			LPCTSTR p = GET_P4REGPTR()->GetP4Port();
			LPCTSTR c = GET_P4REGPTR()->GetP4Client();
			LPCTSTR u = GET_P4REGPTR()->GetP4User();
			if (p && *p && c && *c && u && *u)
			{
				LPCTSTR q;
				for (q = p; *++q; )
				{
					if (*q == _T(' '))
						break;
				}
				CString txt;
				txt.Format(*q ? _T("\"%s\" %s %s") : _T("%s %s %s"), p, c, u);
				GET_P4REGPTR()->RmvMRUPcu( txt );
			}
			MainFrame()->loadMRUPcuMenuItems(pMenu);
		}
	}
	return 0;
}


void CDepotTreeCtrl::RunCStat(int key)
{
	// VIEWUPDATE STEP III 
	
	ASSERT( m_UpdateType == UPDATE_FULL && GET_SERVERLEVEL() < 4 );

	CString spec;
	// The filespec varies for client-view-only display
	if(GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT)
		spec=_T("//...");
	else
		spec.Format(_T("//%s/..."), GET_P4REGPTR()->GetP4Client());

	CCmd_Fstat *pCmd= new CCmd_Fstat;

    // Store our context, so when this command returns we know where we are
    pCmd->SetItemRef(m_LastPathItem);
    pCmd->SetTextRef(m_LastPath);
    pCmd->SetUpdateType(m_UpdateType);

	pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key);
    if( pCmd->Run( FALSE, LPCTSTR(spec), GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_UPDATING) );
	else
	{
		RELEASE_SERVER_LOCK(key);
		delete pCmd;
	}
}


void CDepotTreeCtrl::RunRedrill( int key )
{
	// VIEWUPDATE STEP III, for UPDATE_REDRILL 
	
	ASSERT( m_UpdateType == UPDATE_REDRILL );

	// If this is local syntax, 
	// we have to have the root as one of our directories
	// or we won't find all the dirs and files
	// We also want to remove any entires in the list
	// that are not under our root
	if (GET_P4REGPTR()->ShowEntireDepot() > SDF_DEPOT)
	{
		BOOL found = FALSE;
		CString rootbare = TheApp()->m_ClientRoot;
		rootbare.TrimRight(_T("\\"));				/* to handle c:\ */
		rootbare += _T("\\");
		int barelgth = rootbare.GetLength();
		CString rootdir = rootbare + _T("*");
		if ( m_StringList.GetCount() )
		{
			CString str;
			CString tmp;
			CStringList templist;
			for (POSITION pos = m_StringList.GetHeadPosition(); pos != NULL; )
			{
				str = m_StringList.GetNext(pos);
				tmp = str.Left(barelgth);
				if (!tmp.CompareNoCase(rootbare))	// under new root?
					templist.AddHead(str);			//	yes - add to list
			}
			m_StringList.RemoveAll();
			if ( templist.GetCount() )
			{
				for (POSITION pos = templist.GetHeadPosition(); pos != NULL; )
				{
					str = templist.GetNext(pos);
					if ( rootdir == str )			// is this the root?
						found = TRUE;				//	yes - remember we found it
					m_StringList.AddHead(str);
				}
			}
		}
		if (!found)
			m_StringList.AddHead( rootdir );
	}

    if( m_StringList.GetCount() == 0 )
    {
		// No need to run dirstat, but need to update changes window anyway
		StartChangeWndUpdate(key);
	}
    else  
    {
        CCmd_DirStat *pCmd= new CCmd_DirStat;

		pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key);
		if( pCmd->Run( &m_StringList, GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_UPDATING));
		else
		{
			RELEASE_SERVER_LOCK(key);
			delete pCmd;
		}
	}
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileDiffhead() 
{
	HTREEITEM item=GetLastSelection();  
	
	if (item == NULL)
		ASSERT(0);
	else
	{
		CString itemStr=GetItemPath(item);
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			CP4FileStats *fs = m_FSColl.GetStats(GetLParam(item));
			itemStr = fs->GetFullDepotPath();
		}
		m_StringList.RemoveAll();
		m_StringList.AddHead(LPCTSTR(itemStr));

		CCmd_Diff *pCmd= new CCmd_Diff;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		if( pCmd->Run( &m_StringList, IsOpened(item) ? NULL : _T("-f") ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_DIFFING_FILE) );
		else
			delete pCmd;
	}	
}

/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileDiff2() 
{
	int i;
	HTREEITEM item1 = GetSelectedItem(0);
	HTREEITEM item2 = (GetSelectedCount() > 1) ? GetSelectedItem(1) : item1;
	i = item1 == item2 ? 0 : 1;
	
	if ((item1 == NULL) || (item2 == NULL))
	{
		ASSERT(0);
		return;
	}
	BOOL isFiles = ITEM_IS_FILE(GetSelectedItem(0)) && ITEM_IS_FILE(GetSelectedItem(i));
	CString txt;
	CString rev1, rev2;
	CString ft1,  ft2;
	CDiff2Dlg dlg;

	dlg.m_Edit1 = GetItemPath(item1);
	dlg.m_Edit2 = GetItemPath(item2);

	if (isFiles)
	{
 		dlg.m_RevNbr1 = dlg.m_HaveRev1 = rev1 = GetItemRev(item1);
 		dlg.m_RevNbr2 = dlg.m_HaveRev2 = rev2 = GetItemRev(item2);

		rev1 = GetItemHeadRev(item1);
		rev2 = GetItemHeadRev(item2);

		CP4FileStats *fs;
		fs = m_FSColl.GetStats(GetLParam(item1));
		dlg.m_HeadRev1 = (fs->GetHeadAction() == F_DELETE) ? 0 : _tstoi(rev1);
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			CString dPath = fs->GetFullDepotPath();
			if (dPath.Find(_T('%')))
				dlg.m_Edit1 = dPath;
		}
		fs = m_FSColl.GetStats(GetLParam(item2));
		dlg.m_HeadRev2 = (fs->GetHeadAction() == F_DELETE) ? 0 : _tstoi(rev2);
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			CString dPath = fs->GetFullDepotPath();
			if (dPath.Find(_T('%')))
				dlg.m_Edit2 = dPath;
		}

		ft1 = GetItemType(item1);
		ft2 = GetItemType(item2);
		// assume all unknown types are text
		if (ft1 == _T("unknown"))
			ft1 = (ft2 == _T("unknown")) ? _T("text") : ft2;
		if (ft2 == _T("unknown"))
			ft2 = ft1;

		if (!GET_P4REGPTR()->GetDiffAppIsBinary())
		{
			if (((ft1.Find(_T("text")) != -1) && (ft2.Find(_T("binary")) != -1))
			 || ((ft1.Find(_T("binary")) != -1) && (ft2.Find(_T("text")) != -1)))
			{
				txt.FormatMessage(IDS_ONLY_DIFF_SAME_TYPES_s_n_s_s_n_s, 
					dlg.m_Edit1, _tstoi(rev1), ft1,
					dlg.m_Edit2, _tstoi(rev2), ft2);
				AfxMessageBox(txt);
				return;
			}
		}
	}
	else	// dirs, not files
	{
		dlg.m_Edit1 += _T("...");
		dlg.m_Edit2 += _T("...");
		dlg.m_IsFolders = TRUE;
	}

	SET_APP_HALTED(TRUE);
	int rc=dlg.DoModal();  
	SET_APP_HALTED(FALSE);
	if (rc == IDOK)
	{
		int r1, r2;

		if (dlg.m_RevRadio1 == 1)
		{
			if (isFiles)
			{
				rev1 = dlg.m_HaveRev1;
				r1 = _tstoi(rev1);
				if ((i = dlg.m_Edit1.Find(_T('#'))) != -1)
					dlg.m_Edit1 = dlg.m_Edit1.Left(i);
			}
			else
			{
				dlg.m_Edit1 += _T("#have");
				r1 = 0;
			}
		}
		else if (dlg.m_RevRadio1 == 2)
		{
			rev1 = dlg.m_RevNbr1;
			r1 = _tstoi(rev1);
			if ((i = dlg.m_Edit1.Find(_T('#'))) != -1)
				dlg.m_Edit1 = dlg.m_Edit1.Left(i);
		}
		else
		{
			r1 = 0 - _tstoi(rev1); // make negative as a signal to only use as part of temp name
			// dlg.m_Edit1 is already set with either nothing or @xxx at the end
		}
		if (dlg.m_RevRadio2 == 1)
		{
			if (isFiles)
			{
				rev2 = dlg.m_HaveRev2;
				r2 = _tstoi(rev2);
				if ((i = dlg.m_Edit2.Find(_T('#'))) != -1)
					dlg.m_Edit2 = dlg.m_Edit2.Left(i);
			}
			else
			{
				dlg.m_Edit2 += _T("#have");
				r2 = 0;
			}
		}
		else if (dlg.m_RevRadio2 == 2)
		{
			rev2 = dlg.m_RevNbr2;
			r2 = _tstoi(rev2);
			if ((i = dlg.m_Edit2.Find(_T('#'))) != -1)
				dlg.m_Edit2 = dlg.m_Edit2.Left(i);
		}
		else
		{
			r2 = 0 - _tstoi(rev2); // make negative as a signal to only use as part of temp name
			// dlg.m_Edit2 is already set with either nothing or @xxx at the end
			// if same basename and rev#, we must change so tempfiles don't step on each other
			if (r2 && (r2 == r1))
			{
				int i;
				CString str1 = _T("");
				CString str2 = _T("");
				if ((i = dlg.m_Edit1.ReverseFind(_T('/'))) != -1)
					str1 = dlg.m_Edit1.Mid(i+1);
				else if ((i = dlg.m_Edit1.ReverseFind(_T('\\'))) != -1)
					str1 = dlg.m_Edit1.Mid(i+1);
				if ((i = dlg.m_Edit2.ReverseFind(_T('/'))) != -1)
					str2 = dlg.m_Edit2.Mid(i+1);
				else if ((i = dlg.m_Edit2.ReverseFind(_T('\\'))) != -1)
					str2 = dlg.m_Edit2.Mid(i+1);
				if (!str1.CompareNoCase(str2))
					r2--;
			}
		}
		BOOL bDoIt;
		if (dlg.m_Edit1.GetAt(0) == _T('@') && dlg.m_Edit2.GetAt(0) == _T('@'))
		{
			dlg.m_Edit1 = _T("//...") + dlg.m_Edit1;
			dlg.m_Edit2 = _T("//...") + dlg.m_Edit2;
			bDoIt = TRUE;
		}
		else if ((dlg.m_Edit1.Find(_T("/..."))  != -1 && dlg.m_Edit2.Find(_T("/..."))  != -1)
			  || (dlg.m_Edit1.Find(_T("\\...")) != -1 && dlg.m_Edit2.Find(_T("\\...")) != -1))
			bDoIt = TRUE;
		else
			bDoIt = FALSE;
		if (bDoIt && (r1 || r2))
		{
			CString str;	// have to use a temp variable or Format() yields garbage
			if (r1)
			{
				str.Format(_T("%s#%d"), dlg.m_Edit1, r1);
				dlg.m_Edit1 = str;
			}
			if (r2)
			{
				str.Format(_T("%s#%d"), dlg.m_Edit2, r2);
				dlg.m_Edit2 = str;
			}
		}
		CCmd_Diff2 *pCmd= new CCmd_Diff2;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		pCmd->SetOutput2Dlg(GET_P4REGPTR()->Diff2InDialog());
		if( pCmd->Run( dlg.m_Edit1, dlg.m_Edit2, r1, r2, ft1, ft2, 
						dlg.m_RevRadio1 == 1, dlg.m_RevRadio2 == 1, bDoIt) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_DIFFING_FILES) );
		else
			delete pCmd;
	}
}

LRESULT CDepotTreeCtrl::OnP4Diff2(WPARAM wParam, LPARAM lParam)
{
	CCmd_Diff2 *pCmd= (CCmd_Diff2 *) wParam;
	CString msg= pCmd->GetInfoText();
	if( ! msg.IsEmpty() )
	{
		if (pCmd->IsOutput2Dlg())
		{
			int key;
			m_Diff2dlg = new CDiff2Output(this);
			m_Diff2dlg->SetKey(key = pCmd->HaveServerLock()? pCmd->GetServerKey() : 0);
			m_Diff2dlg->SetMsg( msg );
			CStringArray names;
			names.Add(pCmd->GetFileName(0));
			names.Add(pCmd->GetFileName(1));
			m_Diff2dlg->SetNames(&names);
			CString caption = LoadStringResource(IDS_DIFF2FOLDERS) + pCmd->GetFileName(0) + _T(" <> ") + pCmd->GetFileName(1);
			m_Diff2dlg->SetCaption( caption );
			m_Diff2dlg->SetFont(GetFont());
			if (!m_Diff2dlg->Create(IDD_DIFF2OUTPUT, this))	// display the diff2 output dialog box
			{
				m_Diff2dlg->DestroyWindow();	// some error! clean up
				delete m_Diff2dlg;
			}
		}
		else
		{
			AfxMessageBox( msg, MB_ICONINFORMATION);
		}
	}
	MainFrame()->UpdateStatus(_T(""));
	delete pCmd;
	return 0;
}

LRESULT CDepotTreeCtrl::OnP4EndDiff2( WPARAM wParam, LPARAM lParam )
{
	CSpecDescDlg *dlg = (CSpecDescDlg *)lParam;
	dlg->DestroyWindow();
	if (m_Diff2dlg)
		m_Diff2dlg->SetFocus();
	return TRUE;
}

LRESULT CDepotTreeCtrl::OnP4EndDiff2Output( WPARAM wParam, LPARAM lParam )
{
	CDiff2Output *dlg = (CDiff2Output *)lParam;
	dlg->DestroyWindow();
	m_Diff2dlg = 0;
	return TRUE;
}

/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileRevisionTree() 
{
	HTREEITEM item=GetLastSelection();  
	
	if (item == NULL)
		ASSERT(0);
	else
	{
		CString itemStr = GetItemPath(item);

		// Are we in local syntax?
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			// use depot syntax
			CP4FileStats *fs=m_FSColl.GetStats(GetLParam(item));
			itemStr = fs->GetFullDepotPath();
		}

		TheApp()->CallP4RevisionTree(itemStr);
	}
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFiledropEdit() 
{
	if(GetSelectedCount() == 0)
	{
		ASSERT(0);
		return;
	}

    int key=0;
	BOOL hitMax = m_SkipSyncDialog;

	// Files were just dropped into a change in the changes window
	if(	AnyNotCurrent() && !m_SkipSyncDialog)
	{
		// Prevent anything from running while dlg is up
        if(SERVER_BUSY() || !GET_SERVER_LOCK(key) )
			return;

		if ( AnyDeleted() )
		{
			AfxMessageBox(IDS_ONE_OR_MORE_FILES_DELETED, MB_ICONEXCLAMATION);
			RELEASE_SERVER_LOCK(key);
			return;
		}
		
		int rc;
		if((rc = MsgBox(IDS_ONE_OR_MORE_FILES_IS_NOT_THE_HEAD_REVISION,
					MB_ICONEXCLAMATION | MB_DEFBUTTON3, IDC_BUTTON3)) == IDC_BUTTON3)
		{
			RELEASE_SERVER_LOCK(key);
			return;
		}
		else if (rc == IDC_BUTTON2)
		{
			RELEASE_SERVER_LOCK(key);
			m_StringList.RemoveAll();

			HTREEITEM cItem;
			CString itemStr;
			CP4FileStats *fs;
			int i, j;

			for(i=GetSelectedCount()-1; i>=0; i--)
			{
				cItem=GetSelectedItem(i);	// Next selected item

				TV_ITEM item;
				item.hItem=cItem;
				item.mask=TVIF_HANDLE| TVIF_CHILDREN;
				TreeView_GetItem(m_hWnd, &item );	
				if(!(ITEM_IS_A_FILE_NOT_A_SUBDIR))
					continue;
				fs=m_FSColl.GetStats((int) item.lParam);
				if(fs->GetHaveRev() < fs->GetHeadRev())
				{
					BOOL b =  TheApp()->m_HasPlusMapping;
					CString dPath = fs->GetFullDepotPath();
					if (!b && dPath.Find(_T('%')) != -1)
						b = TRUE;
					itemStr = b ? dPath : fs->GetFullClientPath();
					m_StringList.AddHead(itemStr);
				}
			}
			if (!m_StringList.IsEmpty())
			{
				CCmd_Get *pCmd= new CCmd_Get;
				pCmd->Init( m_hWnd, RUN_ASYNC);
				pCmd->SetOpenAfterSync(TRUE);
				for(i=-1, j=GetSelectedCount(); ++i < j; )
					pCmd->Add2SelSet(GetSelectedItem(i));	// Save list of selected items in cmd
				if( pCmd->Run( &m_StringList, FALSE ) )
					MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
				else
					delete pCmd;
			}
			return;
		}
	}
	m_SkipSyncDialog = FALSE;

	if(	GET_P4REGPTR()->GetWarnAlreadyOpened() && AnyOtherOpened())
	{
		if(AfxMessageBox(IDS_ONE_OR_MORE_FILES_HAVE_BEEN_OPENED_BY_OTHER_USERS, 
						MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
		{
			RELEASE_SERVER_LOCK(key);
			return;
		}
	}

	// Maybe too many files were dropped?
	int fCount=LeafSelectedCount();
	if(	(fCount > _ttoi(GET_P4REGPTR()->GetWarnLimitOpen())) || (fCount < 0) )
	{
		// Prevent anything from running while dlg is up
		if( key==0 && (SERVER_BUSY() || !GET_SERVER_LOCK(key)) )
			return;
		
		CString txt;
		if (fCount > 0)
		{
			txt.FormatMessage(IDS_YOU_ARE_ATTEMPTING_TO_OPEN_n_FILES, fCount);
		}
		else
		{
			txt = LoadStringResource(IDS_YOU_ARE_ATTEMPTING_TO_OPEN_ALL_THE_FILES);
		}
		if(AfxMessageBox(txt, MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
		{
			RELEASE_SERVER_LOCK(key);
			return;
		}
	}

	if (((GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE)
	 && !AnyHaveChildren() && AnyAddable()) || AnyRecoverable())
	{
		AssembleStringList( &m_StringList, FALSE, FALSE, TRUE );
		m_StringList2.RemoveAll();
		m_SelectionList.RemoveAll();
		for (POSITION pos= m_StringList.GetHeadPosition(); pos!=NULL; )
		{
			CString path = m_StringList.GetNext(pos);
			if (path.GetAt(0) == _T('/'))
				m_SelectionList.AddTail(path);	// if we have depot syntax, don't add it
			else
				m_StringList2.AddTail(path);
		}

		CCmd_Add *pCmd= new CCmd_Add;
		if( key==0 )
			pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
		else
			// Pass in the server lock if we have it
			pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key );
		pCmd->SetOpenAction(2);
		pCmd->SetHitMaxFileSeeks(FALSE);
		if( pCmd->Run( m_OpenUnderChangeNumber, &m_StringList2,  
							m_SelectionList.IsEmpty() ? NULL : &m_SelectionList) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_ADDING_FILES) );
		else
			delete pCmd;
	}
	else
	{
		AssembleStringList( &m_StringList );
		CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;
		if( key==0 )
			pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
		else
			// Pass in the server lock if we have it
			pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key );

		pCmd->SetHitMaxFileSeeks(hitMax);
		if( pCmd->Run( &m_StringList, P4EDIT, m_OpenUnderChangeNumber ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUEST_OPEN_EDIT) );
		else
		{
			if( key != 0 )
				RELEASE_SERVER_LOCK(key);
			delete pCmd;
		}
	}
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileOpenedit() 
{	
	m_OpenUnderChangeNumber=0;
	OnFiledropEdit();	
}


/*
	_________________________________________________________________

	Do nothing - same as hitting escape to make menu go away
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFiledropCancel() 
{
}


void CDepotTreeCtrl::OnFileRecover() 
{
	if(GetSelectedCount() == 0)
	{
		ASSERT(0);
		return;
	}

	// At least one file is selected and user is sure, so proceed with recover
	m_StringList.RemoveAll();
	m_StringList2.RemoveAll();

	HTREEITEM item;
	CString itemStr;
	CP4FileStats *fs;
	int warnCount=0;

	for( int i = GetSelectedCount() - 1; i >= 0; i-- )
	{
		item = GetSelectedItem( i );
		if( HasChildren( item ) )
			ASSERT(0);
		else
		{
			itemStr = GetItemPath( item );
			fs=m_FSColl.GetStats( GetLParam(item) );
			if( fs->GetHeadAction() == F_DELETE && 
				fs->GetMyOpenAction()==0 && 
				fs->GetHeadRev() > 0 &&
				fs->GetHaveRev() < fs->GetHeadRev() &&
				fs->InClientView())
			{
				BOOL b = TheApp()->m_HasPlusMapping;
				CString dPath = fs->GetFullDepotPath();
				if (!b && dPath.Find(_T('%')) != -1)
					b = TRUE;
				m_StringList.AddHead( b ? fs->GetFullClientPath() : itemStr );
				if (fs->GetHaveRev() == 0)
				{
					itemStr.Format(_T("%s#%ld"), 
							b ? dPath : fs->GetFullClientPath(),
							fs->GetHeadRev()-1);
					m_StringList2.AddHead( itemStr );
				}
			}
			else
			{
				CString warning;
				warning.FormatMessage(IDS_UNABLE_TO_RECOVER_FILE_s, itemStr);
				AddToStatus(warning, SV_WARNING);
				warnCount++;
			}
		}
	}

	int proceed= IDYES;
	if(warnCount)
	{
		if (warnCount == GetSelectedCount())
			proceed = IDNO;
		else
		{
			CString message;
			message.FormatMessage( IDS_n_OF_n_SELECTED_FILES_CANNOT_BE_RECOVERED,
							warnCount, GetSelectedCount() );
			proceed= AfxMessageBox( message, MB_ICONEXCLAMATION | MB_YESNO );
		}
	}

	if( proceed == IDYES && !m_StringList.IsEmpty() )
	{
		if (m_StringList2.IsEmpty())
			OnFileRecoverProceed();
		else
		{
			CCmd_Get *pCmd= new CCmd_Get;
			pCmd->Init( m_hWnd, RUN_ASYNC );
			// Save the entire list of files to be recovered in the sync command
			for(POSITION pos = m_StringList.GetHeadPosition(); pos != NULL; )
				pCmd->Add2Recover(m_StringList.GetNext( pos ));
			// Run the sync command on the files that need syncing
			if( pCmd->Run( &m_StringList2, FALSE ) )
				MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
			else
				delete pCmd;
		}
	}
}

void CDepotTreeCtrl::OnFileRecoverProceed()
{
	CCmd_Add *pCmd= new CCmd_Add;
	pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
	pCmd->SetAlternateReplyMsg( WM_P4RECOVER );
	
	if( pCmd->Run( 0, &m_StringList ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_RECOVER) );
	else
		delete pCmd;
}

/*
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::AllOpenedForAdd()
{
	if (GET_P4REGPTR( )->ShowEntireDepot( ) != SDF_LOCALTREE)
		return FALSE;

	BOOL allOpenedForAdd=TRUE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction() && fs->GetMyOpenAction() != F_ADD)
			{
				allOpenedForAdd=FALSE;
				break;
			}
		}
	}
	return allOpenedForAdd;
}

void CDepotTreeCtrl::OnFileRevert() 
{
	if(GetSelectedCount() == 0)
	{
		ASSERT(0);
		return;
	}

	BOOL b;
	if ( ((((b=AllOpenedForAdd())==FALSE) && GET_SERVERLEVEL() < 14)) 
	  || GET_P4REGPTR()->AlwaysWarnOnRevert() )
	{
		if(AfxMessageBox(IDS_REVERTING_FILES_WILL_OVERWRITE_EDITS, MB_ICONQUESTION|MB_YESNO) != IDYES)
			return;
		b = TRUE;
	}

	// At least one file is selected, 
	// and user is sure or running against new server,
	// so proceed with revert.
	m_StringList.RemoveAll();
	AssembleStringList( &m_StringList, TRUE );

	CString itemStr;
	
	if (GET_SERVERLEVEL() >= 14 && !b)
	{
		// Copy m_StringList to m_SelectionList
		// because CCmd_Revert will clear what is passed to it.
		// We need the m_StringList in OnP4FileRevert()
		m_SelectionList.RemoveAll();
		for (POSITION pos= m_StringList.GetHeadPosition(); pos!=NULL; )
			m_SelectionList.AddTail(m_StringList.GetNext(pos));
		
		// Run the p4 revert -an command on all the selected files
		CCmd_Revert *pCmd= new CCmd_Revert;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		pCmd->SetAlternateReplyMsg( WM_P4FILEREVERT );
		if( pCmd->Run( &m_SelectionList, FALSE, TRUE, TRUE, FALSE, TRUE ) )
			MainFrame()->UpdateStatus(LoadStringResource(IDS_RUNNING_DIFF));
		else
			delete pCmd;
	}
	else
	{
		CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;
		pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
		int iFlag;
		IsFilteredOnOpen(0, (LPARAM)&iFlag);
		pCmd->SetRedoOpenedFilter(iFlag);
		if( pCmd->Run( &m_StringList, P4REVERT ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_REVERT) );
		else
			delete pCmd;
	}
}

LRESULT CDepotTreeCtrl::OnP4FileRevert( WPARAM wParam, LPARAM lParam )
{
	BOOL	bBox = TRUE;
	BOOL	bUseDashA = FALSE;
	ASSERT(GET_SERVERLEVEL() >= 14);

	// m_StringList still contains our revert list
	CCmd_Revert *pCmdR= (CCmd_Revert *) wParam;

	if(!pCmdR->GetError() 
		&& pCmdR->GetFileList()->GetCount() + pCmdR->NbrNonEdits() == m_StringList.GetCount())
	{
		bBox = FALSE;
		if (pCmdR->NbrNonEdits() == 0)
			bUseDashA = TRUE;
	}
	delete pCmdR;

	if (bBox)
	{
		if(AfxMessageBox(IDS_REVERTING_FILES_WILL_OVERWRITE_EDITS, MB_ICONQUESTION|MB_YESNO) != IDYES)
		{
			MainFrame()->ClearStatus();
			return (0);
		}
	}

	CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;
	pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
	int iFlag;
	IsFilteredOnOpen(0, (LPARAM)&iFlag);
	pCmd->SetRedoOpenedFilter(iFlag);
	if( pCmd->Run( &m_StringList, bUseDashA ? P4REVERTUNCHG : P4REVERT ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_REVERT) );
	else
		delete pCmd;

	return (0);
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileLock() 
{
	LockAndUnlock( P4LOCK );
}


void CDepotTreeCtrl::OnFileUnlock() 
{
	LockAndUnlock( P4UNLOCK );
}


void CDepotTreeCtrl::LockAndUnlock( int which ) 
{
	if(GetSelectedCount() == 0)
	{
		ASSERT(0);
		return;
	}

	AssembleStringList( &m_StringList );
	
	CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;
	pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );

	if( pCmd->Run( &m_StringList, which ) )
        MainFrame()->UpdateStatus(LoadStringResource(which == P4LOCK ? IDS_REQUESTINGLOCK : IDS_REQUESTINGUNLOCK));
	else
		delete pCmd;
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFiledropDelete() 
{
	if(GetSelectedCount() == 0)
	{
		ASSERT(0);
		return;
	}

    int key=0;

	// Maybe too many files were dropped?
	int fCount=LeafSelectedCount();
	if(	(fCount > 50) || (fCount < 0) )
	{
		// Prevent anything from running while dlg is up
		if( SERVER_BUSY() && !GET_SERVER_LOCK(key) )
			return;
		
		CString txt;
		if (fCount > 0)
		{
			txt.FormatMessage(IDS_YOU_ARE_ATTEMPTING_TO_DELETE_n_FILES, fCount);
		}
		else
		{
			txt = LoadStringResource(IDS_YOU_ARE_ATTEMPTING_TO_DELETE_ALL_THE_FILES);
		}
		if(AfxMessageBox(txt, MB_OKCANCEL | MB_ICONEXCLAMATION) == IDCANCEL)
		{
			RELEASE_SERVER_LOCK(key);
			return;
		}
	}

	AssembleStringList( &m_StringList );

	CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;

    if( key == 0 )
	    pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
    else
        // Pass in the server lock if we have it
        pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK, key);

	if( pCmd->Run( &m_StringList, P4DELETE, m_OpenUnderChangeNumber ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_DELETE) );
	else
    {
        if( key != 0 )
            RELEASE_SERVER_LOCK( key );
		delete pCmd;
    }

}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileOpendelete() 
{
	m_OpenUnderChangeNumber=0;
	OnFiledropDelete();	
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileTimeLapseView() 
{
	HTREEITEM currentItem=GetLastSelection();
	
	if (currentItem == NULL)
		ASSERT(0);
	else
	{
		CString itemStr= GetItemPath(currentItem);
		
		// Are we in local syntax?
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			// use depot syntax
			CP4FileStats *fs=m_FSColl.GetStats(GetLParam(currentItem));
			itemStr = fs->GetFullDepotPath();
		}

		TheApp()->CallP4A(itemStr, _T(""), 0);	// use p4v.exe for annotate
	}
}

void CDepotTreeCtrl::OnFileAnnotate() 
{
	FileAnnotate(FALSE);
}

void CDepotTreeCtrl::OnFileAnnotateAll() 
{
	FileAnnotate(TRUE);
}

void CDepotTreeCtrl::OnFileAnnotateChg() 
{
	FileAnnotate(FALSE, TRUE);
}

void CDepotTreeCtrl::OnFileAnnotateChgAll() 
{
	FileAnnotate(TRUE, TRUE);
}

void CDepotTreeCtrl::FileAnnotate(BOOL bAll, BOOL bChg/*=FALSE*/) 
{
	HTREEITEM currentItem=GetLastSelection();
	
	if (currentItem == NULL)
		ASSERT(0);
	else
	{
		CString itemStr= GetItemPath(currentItem);
		
		// Are we in local syntax?
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT)
		{
			// use depot syntax
			CP4FileStats *fs=m_FSColl.GetStats(GetLParam(currentItem));
			itemStr = fs->GetFullDepotPath();
		}

		DWORD index=GetLParam(currentItem);
		CP4FileStats *fs=m_FSColl.GetStats(index);
		CCmd_PrepBrowse *pCmd= new CCmd_PrepBrowse;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		pCmd->SetFileType(fs->IsTextFile() ? FST_TEXT : FST_BINARY);
		CString fType = fs->GetHeadType();
		if( pCmd->Run( FALSE, itemStr, fType, bAll, bChg, FALSE, fs->GetHeadRev(),
			GET_P4REGPTR()->GetAnnotateWhtSpace(),
			bChg ? GET_P4REGPTR()->GetAnnotateIncInteg() : FALSE) )
		{
			MainFrame()->UpdateStatus( LoadStringResource(IDS_FETCHING_FILE) );
			m_Viewer=GET_P4REGPTR()->GetEditApp();
		}
		else
			delete pCmd;
	}
}

LRESULT CDepotTreeCtrl::OnSetViewer( WPARAM wParam, LPARAM lParam )
{
	m_Viewer = (LPCTSTR)lParam;
	return 0;
}

/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileRevisionhistory() 
{
	HTREEITEM currentItem=GetLastSelection();  
	
	if (currentItem == NULL)
		ASSERT(0);
	else
	{
		BOOL isAfile;
		CString itemStr= GetItemPath(currentItem);
		if (itemStr.GetAt(itemStr.GetLength() -1) == m_SlashChar)
		{
			itemStr += _T("...");
			isAfile = FALSE;
		}
		else
			isAfile = TRUE;
		
		// Are we in local syntax?
		if (GET_P4REGPTR( )->ShowEntireDepot( ) > SDF_DEPOT && isAfile)
		{
			// use depot syntax
			CP4FileStats *fs=m_FSColl.GetStats(GetLParam(currentItem));
			itemStr = fs->GetFullDepotPath();
		}

		CCmd_History *pCmd= new CCmd_History;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		pCmd->SetCallingWnd(m_hWnd);
		if (!isAfile)
		{
			pCmd->OverrideFetchCompleteHist(0);
			pCmd->OverrideFetchHistCount(25);
			pCmd->SetEnableShowIntegs(FALSE);
		}
		pCmd->SetIsAFile(isAfile);
		pCmd->SetInitialRev(-1, itemStr);
		if( pCmd->Run(itemStr) )
		{
			MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_HISTORY) );
		}
		else
			delete pCmd;
		
	}
}

/*	_________________________________________________________________

	Handle "Find in Changelist" menu item.  
	Notify the pending changelist window of the name of
	the file in depot syntax to be positioned to.
	Go via MainFrame since it knows the changelist window ptr
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnPositionChgs() 
{
	HTREEITEM currentItem=GetLastSelection();  
	
	if ((currentItem == NULL) || !ITEM_IS_FILE(currentItem))
	{
		ASSERT(0);
	}
	else
	{
		CP4FileStats *fs=m_FSColl.GetStats( GetLParam(currentItem) );
		ASSERT_KINDOF(CP4FileStats, fs);
		MainFrame()->PositionChgs( fs->GetFullDepotPath(), fs->GetMyOpenAction() );
	}
}

/*	_________________________________________________________________

	Handle "Find in Depot" menu item.  This routine also gets
	called by the completion routine for p4 files (OnP4Files) if p4 files
	returns more that one file.

	Bring up a dialogbox that allows the user to 
	1) enter a possibly ambigeous filename to find in the Depot tree  or
	2) select from a list of files that satisfy a previously given ambigeous filename
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnPositionDepot() 
{
	m_Need2Filter = FALSE;
	CFindFilesDlg *dlg= new CFindFilesDlg(this);
	dlg->Init(m_P4Files_FileSpec, &m_P4Files_List, this, m_P4Files_Deselect);
	if (!dlg->Create(IDD_P4FILES, this))	// display the find files dialog box
	{
		dlg->DestroyWindow();	// some error! clean up
		delete dlg;
	}
}

LRESULT CDepotTreeCtrl::OnEndPositionDepot( WPARAM wParam, LPARAM lParam ) 
{
	CFindFilesDlg *dlg = (CFindFilesDlg *)lParam;

	m_P4Files_Deselect = FALSE;
	switch(dlg->m_ExitCode)
	{
	case 3:
			EmptyDepotFilter();					// empty out any old filter info
			m_Need2Filter = TRUE;
	case 1:	// focus was on the edit field at the top when they clicked the OK button,
		{	//  start a p4 files on the possibly ambigeous filename given
			RunP4Files(dlg->GetEditString());
			break;
		}
	case 2:	// the focus was on the listbox when they clicked the OK button,
		{	// expand the depot treeview to find the item selected in the listbox
			ExpandDepotString(dlg->GetListString(), TRUE);
			if (!dlg->m_CloseAfterFind)
				return TRUE;
			break;
		}
	case 4:	// they clicked on the Filter Depot button while the focus was in the listbox;
		{	// the filter has been loaded, so fire up a refresh
			OnViewUpdate();
			break;
		}
	default: // they canceled
		{
			break;
		}
	}
	dlg->DestroyWindow();
	return TRUE;
}

void CDepotTreeCtrl::OnPositionDepotNext() 
{
	if (m_P4Files_List.IsEmpty())
	{
		MessageBeep(0);
		return;
	}
	CString depotPath = GetCurrentItemPath();
	if (depotPath.GetAt(0) != _T('/'))
	{
		CString str = depotPath;
		str.TrimRight(_T("\\ "));
		if (str == TheApp()->m_ClientRoot)
		{
			ExpandDepotString(m_P4Files_List.GetHead(), TRUE);
			return;
		}
		depotPath = GetItemDepotSyntax(GetSelectedItem(0), &depotPath);
	}
	int i;
	for( POSITION pos= m_P4Files_List.GetHeadPosition(); pos != NULL; )
	{
		CString str = m_P4Files_List.GetNext(pos);
		if ((i = fCompare(depotPath, str, GET_P4REGPTR()->SortByExtension())) <= 0)
		{
			if (i)
			{
				if ((i = str.Find(_T('#'))) != -1)
					str = str.Left(i);
				str.TrimRight();
			}
			if (fCompare(depotPath, str, GET_P4REGPTR()->SortByExtension()) >= 0)
				continue;
			ExpandDepotString(str, TRUE);
			return;
		}
	}
	MessageBeep(0);
}
void CDepotTreeCtrl::OnPositionDepotPrev() 
{
	if (m_P4Files_List.IsEmpty())
	{
		MessageBeep(0);
		return;
	}
	CString depotPath = GetCurrentItemPath();
	if (depotPath.GetAt(0) != _T('/'))
	{
		CString str = depotPath;
		str.TrimRight(_T("\\ "));
		if (str == TheApp()->m_ClientRoot)
		{
			MessageBeep(0);
			return;
		}
		depotPath = GetItemDepotSyntax(GetSelectedItem(0), &depotPath);
	}
	int i;
	for( POSITION pos= m_P4Files_List.GetTailPosition(); pos != NULL; )
	{
		CString str = m_P4Files_List.GetPrev(pos);
		if ((i = fCompare(depotPath, str, GET_P4REGPTR()->SortByExtension())) >= 0)
		{
			ExpandDepotString(str, TRUE);
			return;
		}
	}
	MessageBeep(0);
}

void CDepotTreeCtrl::RunP4Files(CString str)
{
	m_P4Files_FileSpec = str;
	CCmd_Files *pCmdFiles= new CCmd_Files;
	pCmdFiles->Init( m_hWnd, RUN_ASYNC);
	if( pCmdFiles->Run(m_P4Files_FileSpec) )
	{
		MainFrame()->UpdateStatus( LoadStringResource(IDS_RUNNING_P4_FILES) );
		m_pCmdFiles = (void *)pCmdFiles;
	}
	else
	{
		delete pCmdFiles;
		RedrawWindow();
		MainFrame()->ClearStatus();
	}
}

/*	_________________________________________________________________

	Completion routine for p4 files

	1)	no files found - nothing to do
	2)	1 file found - expand the depot treeview to find the file
	3)	more than 1 file found - bring up the "Find in Depot"
		dialogbox again with all the found files in the listbox
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnP4Files(WPARAM wParam, LPARAM lParam)
{
	CCmd_Files *pCmdFiles= (CCmd_Files *)m_pCmdFiles;

	m_P4Files_List.RemoveAll();		// empty the files list
	CStringList *list = pCmdFiles->GetFileList();
	switch(list->GetCount())
	{
	case 0:			//	no matching files found, nothing to do
		break;
	case 1:			//	1 file found - expand the depot treeview to find the file
		if (!m_Need2Filter)
		{
			CString path = list->GetHead();
			int i = path.Find(_T('#'));
			if (i != -1)
			{
				if (!GET_P4REGPTR()->ShowDeleted() && (path.Find(_T(" - delete change "), i) != -1))
				{
					AddToStatus(path, SV_WARNING);
					break;
				}
				path = path.Left(i);
			}
			ExpandDepotString(path, TRUE);
			break;
		}
	default:		//	more than 1 file found - load the file list
		if (GET_P4REGPTR()->ShowDeleted())
		{
			for( POSITION pos = list->GetHeadPosition(); pos != NULL; )
				m_P4Files_List.AddHead(list->GetNext( pos ));
		}
		else
		{
            int i = -1;
            CString path;
			for( POSITION pos = list->GetHeadPosition(); pos != NULL; )
			{
				path = list->GetNext( pos );
				if (((i = path.Find(_T('#'))) == -1) 
				 || (path.Find(_T(" - delete change "), i) == -1))
					m_P4Files_List.AddHead(path);
			}
			if (m_P4Files_List.GetCount() == 1 && !m_Need2Filter)
			{
                // if after removing deletes, there was only one file left,
                // just expand the depot to find it
                path = m_P4Files_List.GetHead();
				m_P4Files_List.RemoveAll();
                i = path.Find(_T('#'));
                if(i != -1)
                    path = path.Left(i);
				ExpandDepotString(path, TRUE);	
			}
			else if (!m_P4Files_List.GetCount())
				AddToStatus(LoadStringResource(IDS_ONLY_DELETED_FILES_WERE_FOUND), SV_WARNING);
		}
		break;
	}
	delete pCmdFiles;
	MainFrame()->ClearStatus();
	if (!m_P4Files_List.IsEmpty())	// if there is anyting in the files list,
	{
		if (m_Need2Filter)				// if they want to filter the depot view
		{
			LoadDepotFilterList(&m_P4Files_List);	// load the depot filter list
			m_FilterDepot=TRUE;
			m_DepotFilterPort = GET_P4REGPTR()->GetP4Port();
			m_DepotFilterClient = GET_P4REGPTR()->GetP4Client();
			if (m_DepotFilterType == DFT_LIST)
				GetNextFilesFromFilterList();		// [which will turn off m_Need2Filter and call OnViewUpdate() when done]
			else
			{
				m_Need2Filter = FALSE;
				OnViewUpdate();							//	and refresh the depot pane
			}
		}
		else
			OnPositionDepot();			// or bring up the "Find in Depot" dialog box again
	}
	return 0;
}

/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnDropTarget(WPARAM wParam, LPARAM lParam)
{
	m_DropTargetFlag = wParam;
	m_DropTargetPt.x = LOWORD(lParam);
	m_DropTargetPt.y = HIWORD(lParam);
	return 0;
}

/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnShowDeletedFiles() 
{
	BOOL showDeleted = !GET_P4REGPTR()->ShowDeleted();
	GET_P4REGPTR()->SetShowDeleted( showDeleted );
	MainFrame()->SetDepotCaption( GET_P4REGPTR()->ShowEntireDepot() );
	OnViewUpdate( TRUE );
}

/*
	_________________________________________________________________
*/


//	If user right clicks on a file and chooses 'Explore', run Windows Explorer in
//	the directory where that file resides on the client machine. If we have any
//	trouble determining the directory, use the client root.
void CDepotTreeCtrl::OnWinExplore() 
{
	CString itemStr;
	CString switches = _T("");
	HTREEITEM currentItem = NULL;

	if (m_ContextPoint.x != -1 && m_ContextPoint.y != -1)
	{
		// find out what item was clicked to generate the last context menu
		TV_HITTESTINFO ht;
		ht.pt=m_ContextPoint;
		ScreenToClient(&ht.pt);
		ht.flags=TVHT_ONITEMLABEL | TVHT_ONITEMICON | TVHT_ONITEMBUTTON;
		currentItem=HitTest( &ht );
	}
	if (!currentItem || !IsSelected(currentItem))
		 currentItem=GetLastSelection();
	
	if (currentItem == NULL)
	{
		itemStr = TheApp()->m_ClientRoot;
	}
	else if (ITEM_IS_FILE(currentItem))
	{
		DWORD index=GetLParam(currentItem);
		CP4FileStats *fs=m_FSColl.GetStats(index);
		itemStr= fs->GetFullClientPath();
		itemStr.Replace(_T('/'), _T('\\'));
		if (GET_P4REGPTR()->GetExplorer()			// not using Win Explorer
		 || (::GetFileAttributes(itemStr) == -1))	// file not found
		{
			int i;
			if ((i = ReverseFindMBCS(itemStr, _T('\\'))) != -1)
				itemStr = itemStr.Left(i);
		}
		else
		{
			switches = _T("/select,");
		}
	}
	else
	{
		itemStr = GetItemPath(currentItem);
		if (itemStr.GetAt(0) == _T('/'))
		{
			// the itemStr is in depot syntax
			itemStr.TrimRight(_T('/'));
			// first check for the root
			if (itemStr.Find(_T('/'), 2) == -1)
			{
				itemStr = TheApp()->m_ClientRoot;
			}
			else
			{
				// if wasn't the root
				// so now run p4 where on the string and see what we get
				BOOL bOK = FALSE;
				CCmd_Where *pCmd1 = new CCmd_Where;
				pCmd1->Init(NULL, RUN_SYNC);
				if ( pCmd1->Run(itemStr) && !pCmd1->GetError() 
				  && pCmd1->GetDepotFiles()->GetCount() )
				{
					itemStr = pCmd1->GetLocalSyntax();
					bOK = TRUE;
				}
				delete pCmd1;
				if (!bOK)
					itemStr = TheApp()->m_ClientRoot;
			}
		}
		if (itemStr.GetLength() > 3)
			TrimRightMBCS(itemStr, _T("\\"));
	}
	if (itemStr.GetLength() < 3)
		itemStr += "\\";
	if (::GetFileAttributes(itemStr) == -1)
		itemStr = TheApp()->m_ClientRoot;
	if (itemStr.IsEmpty() || itemStr == _T("null"))
		itemStr = _T("C:\\");

	if (itemStr.FindOneOf(_T(" &()[]{}^=;!'+,`~")) != -1)
	{
		itemStr.TrimLeft();
		itemStr.TrimRight();
		itemStr = _T('\"') + itemStr + _T('\"');
	}

    STARTUPINFO si;
    memset(&si, 0, sizeof(si));
    si.cb = sizeof(si);
    PROCESS_INFORMATION pi;
    CreateProcess(NULL, const_cast<LPTSTR>((LPCTSTR)(TheApp()->GetExplorer()+switches+itemStr)), 
					NULL, NULL, 
#ifdef UNICODE
					FALSE, DETACHED_PROCESS | NORMAL_PRIORITY_CLASS | CREATE_UNICODE_ENVIRONMENT, 
#else
					FALSE, DETACHED_PROCESS | NORMAL_PRIORITY_CLASS, 
#endif
					MainFrame()->P4GetEnvironmentStrings(), 
					NULL, &si, &pi);
}

/*
	_________________________________________________________________
*/

//	If user right clicks on a file and chooses 'Command Prompt', run a Command Prompt
//	in the directory where that file resides on the client machine.
//	We can only do this for a specific file - not for a directory since it may map
//	to multiple local folders - in that case let MainFrame handle it.

void CDepotTreeCtrl::OnCmdPrompt() 
{
	if (!(GetSelectedCount() && ((!AnyHaveChildren() && AnyInView()) 
								|| GetSelectedCount()==1)))
	{
		MainFrame()->OnCmdPromptPublic();
		return;
	}

	int i;
	CString itemStr;
	HTREEITEM currentItem = NULL;

	if (m_ContextPoint.x != -1 && m_ContextPoint.y != -1)
	{
		// find out what item was clicked to generate the last context menu
		TV_HITTESTINFO ht;
		ht.pt=m_ContextPoint;
		ScreenToClient(&ht.pt);
		ht.flags=TVHT_ONITEMLABEL | TVHT_ONITEMICON | TVHT_ONITEMBUTTON;
		currentItem=HitTest( &ht );
	}
	if (!currentItem || !IsSelected(currentItem))
		 currentItem=GetLastSelection();  

	if (ITEM_IS_FILE(currentItem))
	{
		DWORD index=GetLParam(currentItem);
		CP4FileStats *fs=m_FSColl.GetStats(index);
		itemStr= fs->GetFullClientPath();
		itemStr.Replace(_T('/'), _T('\\'));
		if ((i = ReverseFindMBCS(itemStr, _T('\\'))) != -1)
			itemStr = itemStr.Left(i);
	}
	else
	{
		itemStr = GetItemPath(currentItem);
		if (itemStr.GetAt(0) == _T('/'))
		{
			// the itemStr is in depot syntax
			itemStr.TrimRight(_T('/'));
			// first check for the root
			if (itemStr.Find(_T('/'), 2) == -1)
			{
				itemStr = TheApp()->m_ClientRoot;
			}
			else
			{
				// if wasn't the root
				// so now run p4 where on the string and see what we get
				BOOL bOK = FALSE;
				CCmd_Where *pCmd1 = new CCmd_Where;
				pCmd1->Init(NULL, RUN_SYNC);
				if ( pCmd1->Run(itemStr) && !pCmd1->GetError() 
				  && pCmd1->GetDepotFiles()->GetCount() )
				{
					itemStr = pCmd1->GetLocalSyntax();
					bOK = TRUE;
				}
				delete pCmd1;
				if (!bOK)
					itemStr = TheApp()->m_ClientRoot;
			}
		}
		if (itemStr.GetLength() > 3)
			TrimRightMBCS(itemStr, _T("\\"));
	}
	if (itemStr.GetLength() < 3)
		itemStr += "\\";
	if (::GetFileAttributes(itemStr) == -1)
		itemStr = TheApp()->m_ClientRoot;
	if (itemStr.IsEmpty())
		itemStr = _T("C:\\");

	TCHAR	cmd[MAX_PATH+1];

	GetEnvironmentVariable(_T("ComSpec"), cmd, MAX_PATH);
	STARTUPINFO si;
	memset(&si, 0, sizeof(si));
	si.cb = sizeof(si);
	PROCESS_INFORMATION pi;
	CreateProcess(NULL, cmd, 
					NULL, NULL, 
#ifdef UNICODE
					FALSE, CREATE_NEW_CONSOLE | NORMAL_PRIORITY_CLASS | CREATE_UNICODE_ENVIRONMENT, 
#else
					FALSE, CREATE_NEW_CONSOLE | NORMAL_PRIORITY_CLASS, 
#endif
					MainFrame()->P4GetEnvironmentStrings(), 
					itemStr, &si, &pi);
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileGetwhatif() 
{
	FileGet(TRUE, FALSE, FALSE);
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileGet() 
{
	FileGet(FALSE, FALSE, FALSE);
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileGetcustom() 
{
	if (MainFrame()->IsModlessUp())
		return;

	int cnt;
	if((cnt = GetSelectedCount()) == 0)
	{
		ASSERT(0);
		return;
	}

	m_CustomGetDlg = new CCustomGetDlg(this);
	if (!m_CustomGetDlg)
	{
		ASSERT(0);
		AfxMessageBox(IDS_COULD_NOT_CREATE_CUSTOM_SYNC_DIALOG_BOX, MB_ICONSTOP);
		return;
	}
	MainFrame()->SetModelessUp(TRUE);

	if (cnt == 1 && AnyHaveChildren())
		cnt = 100;
	m_CustomGetDlg->m_NbrSel = cnt;
	if (!m_CustomGetDlg->Create(IDD_CUSTOMGET, this))
	{
		delete m_CustomGetDlg;
		MainFrame()->SetModelessUp(FALSE);
	}
}

LRESULT CDepotTreeCtrl::OnDoGetCustom(WPARAM wParam, LPARAM lParam) 
{
	BOOL preview=FALSE;
	switch(wParam)
	{
	case IDCANCEL:
		break;

	case IDGETFORCEPREVIEW:
		preview=TRUE;
		// fall thru
	case IDGETFORCE:
		ASSERT(lParam);
		FileGet(preview, TRUE, FALSE, (LPCTSTR)lParam); 
		break;

	case IDGETPREVIEW:
		preview=TRUE;
		// fall thru
	case IDGET:
		ASSERT(lParam);
		FileGet(preview, FALSE, FALSE, (LPCTSTR)lParam); 
		break;
	default:
		ASSERT(0);
	}
	if (m_CustomGetDlg && (wParam != IDGETPREVIEW) && (wParam != IDGETFORCEPREVIEW))
	{
		m_CustomGetDlg->DestroyWindow();	// deletes m_CustomGetDlg
		m_CustomGetDlg = 0;
		MainFrame()->SetModelessUp(FALSE);
	}
	return 0;
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::FileGet(BOOL whatIf, BOOL force, BOOL removeFiles, LPCTSTR qualifier)
{
	if(GetSelectedCount()==0)
	{
		ASSERT(0);
		return;
	}

	m_StringList.RemoveAll();

	HTREEITEM cItem;
	CString itemStr;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		cItem=GetSelectedItem(i);  // Next selected item

		TV_ITEM item;
		item.hItem=cItem;
		item.mask=TVIF_HANDLE| TVIF_CHILDREN;
		TreeView_GetItem(m_hWnd, &item );	
		itemStr= GetItemPath(cItem);
		if(item.cChildren == 1) // a directory
			itemStr+=_T("...");
		else
		{
			int i;
			if ((itemStr.GetAt(1) == _T(':')) && ((i = itemStr.Find(g_TrulyEmptyDir)) != -1))
			{
				itemStr = itemStr.Left(i);
				itemStr.TrimRight();
				itemStr += _T("\\...");
			}
			else
			{
				CP4FileStats *stats= m_FSColl.GetStats(GetLParam(cItem));
				BOOL b =  TheApp()->m_HasPlusMapping;
				CString dPath = stats->GetFullDepotPath();
				if (!b && dPath.Find(_T('%')) != -1)
					b = TRUE;
				itemStr = b ? dPath : stats->GetFullClientPath();
			}
		}
		if(removeFiles)
			itemStr+=_T("#none");
		else 
			itemStr+=qualifier;  //  '@label' or '@changenum'

		m_StringList.AddHead(itemStr);
	}
	
	CCmd_Get *pCmd= new CCmd_Get;
	pCmd->Init( m_hWnd, RUN_ASYNC);
	if( pCmd->Run( &m_StringList, whatIf, force ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
	else
		delete pCmd;
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileRefresh()
{
	if(GetSelectedCount()==0)
	{
		ASSERT(0);
		return;
	}

	AssembleStringList ( &m_StringList, TheApp()->m_HasPlusMapping );

	//		98.1 changed the command to recopy. before: p4 refresh. after: p4 sync -f
	//
	if ( GET_SERVERLEVEL( ) > 3 )
	{
        // Add a "#have" to each filespec
        POSITION pos= m_StringList.GetHeadPosition();
        while( pos != NULL )
        {
			int i;
            POSITION oldPos= pos;
            CString filespec= m_StringList.GetNext(pos);
			if (filespec.GetAt(1) == _T(':') && ((i = filespec.Find(g_TrulyEmptyDir)) != -1))
			{
				filespec = filespec.Left(i);
				filespec.TrimRight();
				filespec += _T("\\...");
			}
            m_StringList.SetAt(oldPos, filespec+_T("#have"));
        }

		CCmd_Get *pCmd= new CCmd_Get;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		if( pCmd->Run( &m_StringList, FALSE, TRUE ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
		else
			delete pCmd;
	}
	else
	{
		CCmd_Refresh *pCmd= new CCmd_Refresh;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		if(	pCmd->Run( &m_StringList ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_REFRESH) );
		else
			delete pCmd;
	}
}

void CDepotTreeCtrl::OnFileForceToHead()
{
	if(GetSelectedCount()==0)
	{
		ASSERT(0);
		return;
	}

	AssembleStringList ( &m_StringList, TheApp()->m_HasPlusMapping );

	//		98.1 changed the command to recopy. before: p4 refresh. after: p4 sync -f
	//
	if ( GET_SERVERLEVEL( ) > 3 )
	{
        // Add a "#head" to each filespec
        POSITION pos= m_StringList.GetHeadPosition();
        while( pos != NULL )
        {
			int i;
            POSITION oldPos= pos;
            CString filespec= m_StringList.GetNext(pos);
			if (filespec.GetAt(1) == _T(':') && ((i = filespec.Find(g_TrulyEmptyDir)) != -1))
			{
				filespec = filespec.Left(i);
				filespec.TrimRight();
				filespec += _T("\\...");
			}
            m_StringList.SetAt(oldPos, filespec+_T("#head"));
        }

		CCmd_Get *pCmd= new CCmd_Get;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		if( pCmd->Run( &m_StringList, FALSE, TRUE ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
		else
			delete pCmd;
	}
	else
		MessageBeep(0);
}


/*
	_________________________________________________________________

	removes files you're sync'd to, but don't have opened.
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileRemove() 
{
	FileGet(FALSE, FALSE, TRUE);
}



/*
	_________________________________________________________________

	Called by MSTreeView during OnLButtonDown to see if the user is 
	dragging before letting the mouse button up, which would indicate
	a drag drop operation.
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::TryDragDrop( HTREEITEM currentItem )
{
	// Dont actually send data - clipboard format is all the info target requires
	m_OLESource.DelayRenderData( (unsigned short) m_CF_DEPOT);
	m_OLESource.DelayRenderData( (unsigned short) CF_HDROP);	// for external programs (dragging to an editor)
	m_SelectOnDrag= KEEP_SELECTION;
	m_DropTargetFlag=0;
	m_DragDropCtr++;

	if(m_OLESource.DoDragDrop(DROPEFFECT_COPY, &m_DragSourceRect, NULL) == DROPEFFECT_COPY)
	{
		if (m_DropTargetFlag == PENDINGCHG)
		{
			HTREEITEM item;
			CString itemStr;
			// changes view accepted a drop - so get the target change number
			CPoint point;
			m_OpenUnderChangeNumber= ::SendMessage(m_changeWnd, WM_GETDRAGTOCHANGENUM, (WPARAM) &point, 0);

			// and make a list of args for p4 add to use
			m_StringList.RemoveAll();
			for(int i=GetSelectedCount()-1; i>=0; i--)
			{
				item=GetSelectedItem(i);
				itemStr= GetItemPath(item);
				m_StringList.AddHead(itemStr);
			}
			
			OnFiledropEdit();
			return TRUE;
		}
		else if (m_DropTargetFlag == SUBMITTEDCHG)
		{
			// Post a message to the Submitted Changelist Window so it sets the filter
			::SendMessage(m_oldChgWnd, WM_COMMAND, MAKEWPARAM(ID_FILTER_SETVIEW_DROP, 0), 0);
			return TRUE;
		}
		else if (m_DropTargetFlag == LABELVIEW)
		{
			// Either post a message to the Label Window or pop up a menu
			switch (GET_P4REGPTR()->GetLabelDragDropOption())
			{
			case LDD_ADD:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABEL_SYNC, 0), 0);
				break;
			case LDD_DELETE:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABEL_DELETEFILES, 0), 0);
				break;
			case LDD_SYNC:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABEL_SYNC_CLIENT, 0), 0);
				break;
			case LDD_ADD2VIEW:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABEL_ADDTOLABELVIEW, 0), 0);
				break;
			case LDD_FILTER:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_SETVIEW, 0), 0);
				break;
			case LDD_FILTERREV:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_SETVIEWREV, 0), 0);
				break;
			default:
				LabelDropMenu(TRUE);
				break;
			}
			return TRUE;
		}
		else if (m_DropTargetFlag == LABELNOSEL)
		{
			// Either post a message to the Label Window or pop up a menu
			switch (GET_P4REGPTR()->GetLabelDragDropOption())
			{
			case LDD_ADD:
			case LDD_DELETE:
			case LDD_SYNC:
				break;			// nothing was selected in the labelview
			case LDD_FILTER:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_SETVIEW, 0), 0);
				break;
			case LDD_FILTERREV:
				::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_SETVIEWREV, 0), 0);
				break;
			default:
				LabelDropMenu(FALSE);
				break;
			}
			return TRUE;
		}
		else if (m_DropTargetFlag == CLIENTVIEW)
		{
			// Post a message to the Client Window so it adds the file(s) to the view
			OnAddToClientView();
			return TRUE;
		}
		else if (m_DropTargetFlag == USERVIEW)
		{
			// Post a message to the User Window so it adds the review file(s)
			OnAddReviews();
			return TRUE;
		}
		else if (m_DropTargetFlag == JOBVIEW)
		{
			// Post a message to the Job Window so it sets the filter
			OnFilterJobview();
			return TRUE;
		}
	}
	return FALSE;
}
	

/*
	_________________________________________________________________

	This window never accepts drops - just do this so drags 
	originated here dont have the ugly cant
	_________________________________________________________________
*/

int CDepotTreeCtrl::OnCreate(LPCREATESTRUCT lpCreateStruct) 
{
	if (CMultiSelTreeCtrl::OnCreate(lpCreateStruct) == -1)
		return -1;

	SetIndent(15);
	SetScrollTime(10);
	SetImageList(TheApp()->GetImageList(), TVSIL_NORMAL);

	
	return 0;
}


/*
	_________________________________________________________________
*/

DROPEFFECT CDepotTreeCtrl::OnDragEnter(COleDataObject* pDataObject, DWORD dwKeyState, CPoint point) 
{
	m_DepotIsDropTarget=TRUE;
	m_DropEffect=DROPEFFECT_NONE;
	m_DragDataFormat=0;

	if(pDataObject->IsDataAvailable( (unsigned short) m_CF_DEPOT))
	{
		// Right-drag support

	    // We NEVER come in here with just DELETE_EXISTING_SELECTION, because
	    // that would leave us with an empty selection set
	    ASSERT( m_SelectOnDrag == KEEP_SELECTION ||
			    m_SelectOnDrag == ADD_TO_SELECTION ||
			    m_SelectOnDrag == (DELETE_EXISTING_SELECTION | ADD_TO_SELECTION) );

	    ASSERT( m_DragFromItem != NULL);

		if( m_SelectOnDrag & DELETE_EXISTING_SELECTION)
			UnselectAll();

		if( m_SelectOnDrag & ADD_TO_SELECTION )
			SetSelectState(m_DragFromItem, TRUE);

		m_DropEffect=DROPEFFECT_COPY;
	}
#ifdef UNICODE
	else if(pDataObject->IsDataAvailable( (unsigned short) CF_UNICODETEXT))
	{
		m_DropEffect = DROPEFFECT_COPY;
		m_DragDataFormat = CF_UNICODETEXT;
	}
#else
	else if(pDataObject->IsDataAvailable( (unsigned short) CF_TEXT))
	{
		m_DropEffect = DROPEFFECT_COPY;
		m_DragDataFormat = CF_TEXT;
	}
#endif
	m_DepotIsDropTarget=FALSE;
	return m_DropEffect;
}


/*
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnDragLeave() 
{
	m_PendingDeselect=FALSE;
}


/*
	_________________________________________________________________
*/

DROPEFFECT CDepotTreeCtrl::OnDragOver(COleDataObject* pDataObject, DWORD dwKeyState, CPoint point) 
{
	if(pDataObject->IsDataAvailable( (unsigned short) m_CF_DEPOT))
	{
		// Left-drag support.  Dont clear pending deselect until the cursor
		// actually moves!
		CPoint pt= point;
		ClientToScreen( &pt );
		if( !m_DragSourceRect.PtInRect( pt ) )
			m_PendingDeselect=FALSE;
	}
	return m_DropEffect;
}

/*
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::OnDrop(COleDataObject* pDataObject, DROPEFFECT dropEffect, CPoint point) 
{
	CString fname;
	
	if(SERVER_BUSY())
	{
		// OnDragEnter() and OnDragOver() should avoid a drop at 
		// the wrong time!
		ASSERT(0);
		return FALSE;
	}
	
	m_DepotIsDropTarget=TRUE;
#ifdef UNICODE
	if(m_DragDataFormat == CF_UNICODETEXT)
	{
		HGLOBAL hGlob = pDataObject->GetGlobalData(CF_UNICODETEXT);
#else
	if(m_DragDataFormat == CF_TEXT)
	{
		HGLOBAL hGlob = pDataObject->GetGlobalData(CF_TEXT);
#endif
		LPCTSTR p;

		if ((hGlob != NULL)	&& ((p = (LPCTSTR)::GlobalLock(hGlob)) != NULL))
		{
			CString itemStr = p;
			::GlobalUnlock(hGlob);
			if (itemStr.GetAt(0) != _T('/') && itemStr.GetAt(1) != _T(':'))
			{
				int i;
				if ((i = itemStr.Find(_T("//"))) > 0)
				{
					// we have something like "STATUS: 3:33:26 Sync preview: //depot/main/www/perforce/customers/bycountry.html#16 - updating c:\workspac\main\www\perforce\customers\bycountry.html"
					// so throw away everything before the "//"
					itemStr = itemStr.Right(itemStr.GetLength() - i);
					// if we find a 2nd "//", throw it and the following chars away
					if ((i = itemStr.Find(_T("//"), 2)) > 0)
						itemStr = itemStr.Left(i);
					itemStr.TrimRight();
				}
				else if ((i = itemStr.Find(_T(":\\"))) > 0)
				{
					// we have something like "STATUS: 11:48:04 Executing p4 where c:\workspac\P4CONFIG"
					// so throw away everything before the "c:\"
					itemStr = itemStr.Right(itemStr.GetLength() - i + 1);
					// if we find a 2nd "c:\", throw it and the following chars away
					if ((i = itemStr.Find(_T(":\\"), 3)) > 0)
						itemStr = itemStr.Left(i-1);
					itemStr.TrimRight();
				}
			}
			if ((itemStr.FindOneOf(_T("?*")) != -1) || (itemStr.Find(_T("...")) != -1))
				RunP4Files(itemStr);
			else
				ExpandDepotString( itemStr, TRUE );
		}
		m_DepotIsDropTarget=FALSE;
		return TRUE;
	}
	// Return false, so depot window doesnt start a file-open operation
	m_DepotIsDropTarget=FALSE;
	return FALSE;
}

/*
	_________________________________________________________________

	Context menu implementation.  Due to an apparent MFC bug, 
	two mouse click handlers are included to make this work.  
	Prolly in the next release of MFC, the default OnContextMenu 
	linkage can be used.
	_________________________________________________________________
*/
void CDepotTreeCtrl::OnContextMenu(CWnd* pWnd, CPoint point) 
{
	// Save mouse location for last context menu
	m_ContextPoint = point;

	//		make sure window is active
	//
	GetParentFrame()->ActivateFrame();

	BOOL isDepot = FALSE ;

	m_ExpandDepotContinue = FALSE;

	//		find out what was hit, select it if it hasn't been
	//
	CString text;
	HTREEITEM currentItem;
	if (!pWnd && !point.x && !point.y)
	{
		currentItem = NULL;
		CRect rect;
//		MainFrame()->GetHLSplitter()->GetWindowRect(rect);
		int i = GetSystemMetrics(SM_CXSIZEFRAME) + 2;
		point.x = rect.left+i;
		point.y = rect.top+i;
	}
	else
	{
		SetItemAndPoint( currentItem, point );
		ClientToScreen( &point );

		if ( currentItem != NULL )
		{
			text = GetItemText ( currentItem );
			if ( text.Find( g_sSlashes ) == 0 )
				isDepot = TRUE;
		}

		if( !IsSelected( currentItem ))
		{
			UnselectAll( );
			if( currentItem != NULL )
				SetSelectState( currentItem, TRUE );
		}
	}

	//		create an empty context menu and some submenus
	//
	CP4Menu popMenu;
	popMenu.CreatePopupMenu();

	CP4Menu getMenu;
	getMenu.CreatePopupMenu();

	CP4Menu integMenu;
	integMenu.CreatePopupMenu();

	CP4Menu viewMenu;
	viewMenu.CreatePopupMenu();

	CP4Menu editMenu;
	editMenu.CreatePopupMenu();

	//		start putting in items in the menus
	//
	if( currentItem != NULL && currentItem != m_Root )
	{
		if ( !text.IsEmpty( ) && text.Find ( g_TrulyEmptyDir ) != -1 )
		{
			CP4Menu showdelMenu;
			showdelMenu.CreatePopupMenu();
			if (GET_P4REGPTR( )->ShowEntireDepot( ) != SDF_LOCALTREE )
				showdelMenu.AppendMenu( stringsON, ID_FILE_ADD, LoadStringResource( IDS_FILE_ADD ) );
			if (GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALP4
			 || GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE)
			{
				getMenu.AppendMenu( stringsON, ID_FILE_GET, LoadStringResource( IDS_SYNCTOHEAD ) );
				getMenu.AppendMenu( stringsON, ID_FILE_GETWHATIF, LoadStringResource( IDS_SYNCTOHEADPREVIEW ) );
				getMenu.AppendMenu( stringsON, ID_FILE_GETCUSTOM );
				getMenu.AppendMenu( stringsON, ID_FILE_REFRESH );
				getMenu.AppendMenu( stringsON, ID_FILE_FORCESYNCTOHEAD, LoadStringResource( IDS_FILE_FORCESYNCTOHEAD ) );
				showdelMenu.AppendMenu(MF_POPUP, (UINT) getMenu.GetSafeHmenu(), LoadStringResource( IDS_SYNC ) );
				showdelMenu.AppendMenu(MF_SEPARATOR);
			}
			showdelMenu.AppendMenu( stringsON, ID_SHOWDELETED, LoadStringResource( IDS_SHOWDELITEMS ) );
			showdelMenu.TrackPopupMenu( TPM_LEFTALIGN | TPM_RIGHTBUTTON,point.x, point.y, AfxGetMainWnd() );
			return;
		}

		BOOL anyChildren= AnyHaveChildren();
		BOOL anyInview = AnyInView( );
		BOOL anyAddable  = AnyAddable();
		BOOL anyEditable = AnyEditable();
		BOOL anyNotCurrent = AnyNotCurrent( );
		BOOL allNotInDepot = AllNotInDepot( );
		BOOL needToAddSeparator = FALSE;

		// a single directory
		if( GetSelectedCount()==1 && !ITEM_IS_FILE(currentItem) )
		{
			popMenu.AppendMenu( stringsON, ID_FILE_ADD, LoadStringResource( IDS_FILE_ADD ) );
			needToAddSeparator = TRUE;
			point.y -= GetSystemMetrics(SM_CYMENUSIZE) * 2;
		}
		// a directory or a number of files, but not the root of the depot
		if( anyChildren || ( anyInview && anyNotCurrent )) 
		{
			getMenu.AppendMenu( stringsON, ID_FILE_GET, LoadStringResource( IDS_SYNCTOHEAD ) );
			getMenu.AppendMenu( stringsON, ID_FILE_GETWHATIF, LoadStringResource( IDS_SYNCTOHEADPREVIEW ) );
		}

		if( anyChildren || ( anyInview && !allNotInDepot ))
		{
			getMenu.AppendMenu( stringsON, ID_FILE_GETCUSTOM );
			getMenu.AppendMenu( stringsON, ID_FILE_REFRESH );
			getMenu.AppendMenu( stringsON, ID_FILE_FORCESYNCTOHEAD, LoadStringResource( IDS_FILE_FORCESYNCTOHEAD ) );
		}

		BOOL okToAddSeparator = FALSE;	// only used for 1st Separator
		if(!SERVER_BUSY() && getMenu.GetMenuItemCount() > 0
		 && ( (anyChildren && !m_FilterDepot) || (!anyChildren && (anyInview || AnyRemoveable()))))
		{
			if (needToAddSeparator)
			{
				needToAddSeparator = FALSE;
				popMenu.AppendMenu(MF_SEPARATOR);
			}
			popMenu.AppendMenu(MF_POPUP, (UINT) getMenu.GetSafeHmenu(), LoadStringResource( IDS_SYNC ) );
			okToAddSeparator = TRUE;
		}

		BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));

		if( (anyChildren && !m_FilterDepot && !isRemoteFolder) 
		 || (!anyChildren && (anyEditable || anyAddable || AnyOpenedForInteg()) ) )
		{
			DWORD id;
			okToAddSeparator = TRUE;
			if (anyChildren || GET_P4REGPTR()->ShowEntireDepot() != SDF_LOCALTREE)
				id = IDS_CHKOUTFOREDIT;
			else if (anyAddable && anyEditable)
				id = IDS_CHKOUTFOREDITORADD;
			else
			{
				id = anyAddable ? IDS_CHKOUTFORADD : IDS_CHKOUTFOREDIT;
				okToAddSeparator = anyEditable;
			}
			if (needToAddSeparator)
			{
				needToAddSeparator = FALSE;
				popMenu.AppendMenu(MF_SEPARATOR);
			}
			popMenu.AppendMenu( stringsON, 
				id == IDS_CHKOUTFORADD ? ID_FILE_OPENEDITA : ID_FILE_OPENEDIT, 
				LoadStringResource(id));
		}

		integMenu.AppendMenu( stringsON, ID_FILE_INTEGSPEC, LoadStringResource( IDS_USINGFILESPEC ) );
		integMenu.AppendMenu( stringsON, ID_FILE_INTEGRATE, LoadStringResource( IDS_USINGBRANCHSPEC ) );
		if(!SERVER_BUSY() && ( (anyChildren && !m_FilterDepot) || !anyChildren ) 
						  && ( anyChildren || !allNotInDepot ))
		{
			if (needToAddSeparator)
			{
				needToAddSeparator = FALSE;
				popMenu.AppendMenu(MF_SEPARATOR);
			}
			popMenu.AppendMenu(MF_POPUP, (UINT) integMenu.GetSafeHmenu(), LoadStringResource( IDS_INTEGRATE ) );
			if ((anyChildren || (anyEditable && anyInview)) && !isRemoteFolder)
				popMenu.AppendMenu( stringsON, ID_FILE_RENAME, LoadStringResource( IDS_RENAME_DOTS ) );
		}

		if(GetSelectedCount()==1 && !anyChildren && ITEM_IS_FILE(GetSelectedItem(0)))
		{
		    HTREEITEM item = GetSelectedItem(0);
			int index=GetLParam(item);
			CP4FileStats *fs=m_FSColl.GetStats(index);
			if (fs->GetMyOpenAction())
				popMenu.AppendMenu( stringsON, ID_CHANGE_SUBMIT, LoadStringResource( IDS_SUBMIT_DOTS ) );
		}

		if(popMenu.GetMenuItemCount() > 0 && okToAddSeparator)
			popMenu.AppendMenu(MF_SEPARATOR);

		if( (anyChildren && !m_FilterDepot && !isRemoteFolder) || (!anyChildren && anyEditable) )
		{
			popMenu.AppendMenu( stringsON, ID_FILE_OPENDELETE, LoadStringResource( IDS_CHKOUTFORDEL ) );
		}
		else if ( !anyChildren && AllAddable() )
		{
			popMenu.AppendMenu(MF_SEPARATOR);
			popMenu.AppendMenu( stringsON, ID_FILE_DELETE, LoadStringResource( IDS_FILE_DELETE ) );
			if(GetSelectedCount()==2)
			{
				popMenu.AppendMenu(MF_SEPARATOR);
				popMenu.AppendMenu( stringsON, ID_FILE_DIFF2, LoadStringResource(IDS_FILE_DIFF2) );
			}
		}

		if(!SERVER_BUSY() && ( (anyChildren && !m_FilterDepot) 
			               || ((anyInview || AnyRemoveable()) && !allNotInDepot && !anyChildren)))
		{
			popMenu.AppendMenu( stringsON, ID_FILE_REMOVE, LoadStringResource( IDS_RMVFROMCLIENT ) );
		}

		if( !anyChildren && AnyRecoverable() )
			popMenu.AppendMenu( stringsON, ID_FILE_RECOVER, LoadStringResource( IDS_RECOVERDELFILE ) );

		if( (anyChildren && !m_FilterDepot && !isRemoteFolder)  ||  (!anyChildren && AnyLockable()) )
			popMenu.AppendMenu( stringsON, ID_FILE_LOCK, LoadStringResource( IDS_LOCK ));
		if( (anyChildren && !m_FilterDepot && !isRemoteFolder)  ||  (!anyChildren && AnyUnlockable())  )
			popMenu.AppendMenu( stringsON, ID_FILE_UNLOCK, LoadStringResource( IDS_UNLOCK ));
		if( !anyChildren )  // if no directories
		{
			if(AnyOpened())
				popMenu.AppendMenu( stringsON, ID_FILE_REVERT, LoadStringResource( IDS_REVERT ));
		}
		
		if(GetSelectedCount()==1 && !anyChildren && ITEM_IS_FILE(GetSelectedItem(0)))
		{
			// Only one selected item and its a file
			if(popMenu.GetMenuItemCount() > 0)
				popMenu.AppendMenu(MF_SEPARATOR);

			TV_ITEM item;
			CP4FileStats *fs;
			item.hItem=GetSelectedItem(0);
			item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
			TreeView_GetItem(m_hWnd, &item );
			fs=m_FSColl.GetStats((int) item.lParam);

			if (!allNotInDepot)
			{
				popMenu.AppendMenu( stringsON, ID_FILE_DIFFHEAD, 
					LoadStringResource( AnyOpened() ? IDS_DIFFVSDEPOT : IDS_DIFFVSHEAD ) );
				popMenu.AppendMenu( stringsON, ID_FILE_DIFF2, LoadStringResource(IDS_FILE_DIFF2) );
			}
			
			UINT uFlags = MF_POPUP;
			int bAnn = MainFrame()->HaveTLV();
			if (bAnn && IsInRemoteDepot(&(CString(fs->GetFullDepotPath()))))
				bAnn = 0;
			if(!SERVER_BUSY() && !IsDeleted(currentItem))
			{
				// Build the edit and view submenues simultaneously.  We need two
				// nearly identical menues because the command IDs must be different
				// and there is no simple way to record which main menu item
				// popped the submenu.  Obviously the CMenu could be subclassed and
				// have a member that did something with the parent menu ID, but that
 				// hardly seems simpler than just repeating a few lines of code.
				//
				viewMenu.AppendMenu( stringsON, ID_FILE_QUICKBROWSE, LoadStringResource( IDS_ASSOCVIEWER ) );
				editMenu.AppendMenu( stringsON, ID_FILE_QUICKEDIT, LoadStringResource( IDS_ASSOCEDITOR ) );

				int actualMRUs=0;
				for(int i=0; i < MAX_MRU_VIEWERS; i++)
				{
					if( GET_P4REGPTR()->GetMRUViewerName(i).GetLength() > 0 )
					{
						CString viewer = GET_P4REGPTR()->GetMRUViewerName(i);
						viewMenu.AppendMenu( stringsON, ID_FILE_BROWSER_1+i, CString ( _T("&") + viewer ) );
						editMenu.AppendMenu( stringsON, ID_FILE_EDITOR_1+i, CString ( _T("&") + viewer ) );
						actualMRUs++;
					}
				}

				viewMenu.AppendMenu( stringsON, ID_FILE_NEWBROWSER, LoadStringResource ( IDS_OTHERVIEWER ) );
				editMenu.AppendMenu( stringsON, ID_FILE_NEWEDITOR, LoadStringResource ( IDS_OTHEREDITOR ));
				viewMenu.AppendMenu( stringsON, ID_FILE_RMVBROWSER, LoadStringResource ( IDS_RMVVIEWER ) );
				editMenu.AppendMenu( stringsON, ID_FILE_RMVEDITOR, LoadStringResource ( IDS_RMVEDITOR ));
				if (GET_SERVERLEVEL() >= 14)
				{
					viewMenu.AppendMenu(MF_SEPARATOR);
					viewMenu.AppendMenu( stringsON, ID_FILE_ANNOTATE, LoadStringResource ( IDS_FILE_ANNOTATE ) );
					viewMenu.AppendMenu( stringsON, ID_FILE_ANNOTATEALL, LoadStringResource ( IDS_FILE_ANNOTATEALL ) );
					viewMenu.AppendMenu( stringsON, ID_FILE_ANNOTATECHG, LoadStringResource ( IDS_FILE_ANNOTATECHG ) );
					viewMenu.AppendMenu( stringsON, ID_FILE_ANNOTATECHGALL, LoadStringResource ( IDS_FILE_ANNOTATECHGALL ) );
				}
			}
			else
			{
				if (GET_SERVERLEVEL() >= 14)
				{
					viewMenu.AppendMenu( stringsON, ID_FILE_ANNOTATEALL, LoadStringResource ( IDS_FILE_ANNOTATEALL ) );
					viewMenu.AppendMenu( stringsON, ID_FILE_ANNOTATECHGALL, LoadStringResource ( IDS_FILE_ANNOTATECHGALL ) );
				}
				else
					uFlags = MF_GRAYED | MF_DISABLED | MF_POPUP;
			}

			if(!SERVER_BUSY())
			{
				UINT i = !fs->IsMyOpen() && (fs->GetHaveRev() < fs->GetHeadRev())
					   ? IDS_VIEWDEPOTVERUSING : IDS_VIEWDEPOTVERUSINGHEAD;
				popMenu.AppendMenu(uFlags, (UINT)viewMenu.GetSafeHmenu(), LoadStringResource(i));
			}

			uFlags = IsDeleted(currentItem) ? MF_GRAYED | MF_DISABLED | MF_POPUP : MF_POPUP;
			if(!SERVER_BUSY() && GetSelectedCount()==1 && IsInView() && ITEM_IS_FILE(GetSelectedItem(0)))
			{
				BOOL b = TRUE;
				if (!anyEditable && !anyAddable)
				{
					b = !fs->IsOtherOpenExclusive();
					if (b)
						b = !IsInRemoteDepot(&(CString(fs->GetFullDepotPath())));
				}
				if(b)
					popMenu.AppendMenu(uFlags, (UINT) editMenu.GetSafeHmenu(), 
							LoadStringResource(anyEditable || anyAddable 
												? IDS_OPENANDEDITUSING : IDS_EDITUSING));
			}
			
			popMenu.AppendMenu( stringsON, ID_FILE_PROPERTIES, LoadStringResource( IDS_PROPERTIES ) );
			if (!allNotInDepot)
			{
				if (MainFrame()->HaveP4QTree())
					popMenu.AppendMenu( stringsON, ID_FILE_REVISIONTREE, LoadStringResource( IDS_REVISIONTREE ));
				if (bAnn == 1)
					popMenu.AppendMenu( stringsON, ID_FILE_ANNOTATIONS, LoadStringResource( IDS_ANNOTATIONS ));
				popMenu.AppendMenu( stringsON, ID_FILE_REVISIONHISTORY, LoadStringResource( IDS_REVISIONHISTORY ));
			}
		}
		else if(GetSelectedCount()==2 
			 && ((!anyChildren && ITEM_IS_FILE(GetSelectedItem(0)) && ITEM_IS_FILE(GetSelectedItem(1)))
			  || (anyChildren && !ITEM_IS_FILE(GetSelectedItem(0)) && !ITEM_IS_FILE(GetSelectedItem(1)))))
		{
			// Two selected items and they are both files or both dirs
			if (!allNotInDepot && !IsDepot(GetSelectedItem(0)) && !IsDepot(GetSelectedItem(1)))
			{
				if(popMenu.GetMenuItemCount() > 0)
					popMenu.AppendMenu(MF_SEPARATOR);
				popMenu.AppendMenu( stringsON, ID_FILE_DIFF2, 
					LoadStringResource(anyChildren ? IDS_FILE_DIFF2DIRS : IDS_FILE_DIFF2) );
			}
		}

		if( !anyChildren || GetSelectedCount()==1 )  // if no directories or only a single directory
		{
			if (anyChildren && !IsDepot(GetSelectedItem(0)))
			{
				popMenu.AppendMenu(MF_SEPARATOR);
				popMenu.AppendMenu( stringsON, ID_FILE_DIFF2, LoadStringResource(IDS_FILE_DIFF2DIRS) );
			}
			popMenu.AppendMenu( MF_SEPARATOR);
			if (GetSelectedCount()==1)
			{
				if ((AnyOpened() || AnyOtherOpened()) && ITEM_IS_FILE(GetSelectedItem(0)))
					popMenu.AppendMenu( stringsON, ID_POSITIONCHGS, LoadStringResource( IDS_POSITIONCHGS ) );
                popMenu.AppendMenu( stringsON, ID_ADD_BOOKMARK, LoadStringResource(IDS_ADD_BOOKMARK) );
			}
			popMenu.AppendMenu( stringsON, ID_WINEXPLORE, LoadStringResource( IDS_EXPLORE ) );
			popMenu.AppendMenu( stringsON, ID_CMDPROMPT, LoadStringResource( IDS_CMDPROMPT ) );
		}
	}

	if (GetSelectedCount()==1 && currentItem != NULL)
	{
		if (AnyHaveChildren())  // if is a directory
		{
			if(popMenu.GetMenuItemCount() > 0)
				popMenu.AppendMenu( MF_SEPARATOR );
			popMenu.AppendMenu( stringsON, ID_FINDFILEUNDERFOLDER, LoadStringResource(IDS_FINDFILEUNDERFOLDER) );
		}
	}

	MainFrame()->AddToolsToContextMenu(&popMenu);

	m_InContextMenu = TRUE;
	popMenu.TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON,	point.x, point.y, AfxGetMainWnd());
	m_InContextMenu = FALSE;
}

/*
	_________________________________________________________________

	Right click context menu is broken in CTreeViews - 
	a dbl right click is what it wants.  
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnRButtonUp(UINT nFlags, CPoint point) 
{
	CPoint screenPt=point;
	ClientToScreen(&screenPt);
    m_ContextContext= MOUSEHIT;
	OnContextMenu(NULL, screenPt);
}


/*
	_________________________________________________________________

	Mimic explorer behavior for right-drag operations

	1) Find out what got clicked
	
	2) If there are no current selections, select item
		Else if current item is not selected
			If there are selections and no key flags, 
				replace current selections when drag starts
			If there are selections and ctrl or shift is down, 
				add to current selections when drag starts
				
	3) On drop success, pop a context menu over the target change, 
	to allow open for edit vs open for delete
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnRButtonDown(UINT nFlags, CPoint point) 
{
	// make sure window is active
	GetParentFrame()->ActivateFrame();

	// Step 1: find out what was hit
	TV_HITTESTINFO ht;
	ht.pt=point;
	HTREEITEM currentItem=TreeView_HitTest( m_hWnd, &ht	);
	if(currentItem==NULL)
	{
		UnselectAll();
		return;
	}
		
	// Step 2: make selections or pending selections
	//         Note: m_DragFromItem and m_SelectOnDrag and GetLastSelection()
	//               will be used in OnDragEnter() to make necessary
	//               adjustments to the selection set
	m_SelectOnDrag= KEEP_SELECTION;
	m_DragFromItem= currentItem;

	if(GetSelectedCount()==0)
		SetSelectState(currentItem, TRUE);
	else if(!IsSelected(currentItem))
	{
		if( (nFlags & MK_SHIFT || nFlags & MK_CONTROL) &&
				TreeView_GetParent(m_hWnd, currentItem) == GetLastSelectionParent())
			m_SelectOnDrag= ADD_TO_SELECTION;
		else
			m_SelectOnDrag= ADD_TO_SELECTION | DELETE_EXISTING_SELECTION;
	}

	
	// Step 3: Start possible drag-drop action
	RECT rect;
	TreeView_GetItemRect(m_hWnd, currentItem, &rect, TRUE);
	ClientToScreen(&rect);
		
	// Dont actually send data - clipboard format is all the info target requires
	m_OLESource.DelayRenderData( (unsigned short) m_CF_DEPOT);
		
	if(m_OLESource.DoDragDrop(DROPEFFECT_COPY|DROPEFFECT_MOVE, &rect, NULL) == DROPEFFECT_COPY)
	{
		if (m_DropTargetFlag == PENDINGCHG)
		{
			// changes view accepted a drop - so get the target change number
			CPoint screen;
			m_OpenUnderChangeNumber= ::SendMessage(m_changeWnd, WM_GETDRAGTOCHANGENUM, (WPARAM) &screen, 0);
			
			// create an empty context menu
			CP4Menu popMenu;
			popMenu.CreatePopupMenu();

			popMenu.AppendMenu( stringsON, ID_FILEDROP_EDIT, LoadStringResource( IDS_CHKOUTFOREDIT ) );
			popMenu.AppendMenu( stringsON, ID_FILEDROP_DELETE, LoadStringResource( IDS_CHKOUTFORDEL ) );
			popMenu.AppendMenu(MF_SEPARATOR);
			popMenu.AppendMenu( stringsON, ID_FILEDROP_CANCEL, LoadStringResource( IDS_ampCANCEL ) );

			// Pause the auto refresh timer
			SET_APP_HALTED(TRUE);

			// Finally blast the menu onto the screen
			popMenu.TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON,	screen.x, screen.y, AfxGetMainWnd());

			// Make sure selection set is properly displayed
			SetAppearance(FALSE, TRUE, FALSE);

			//Release the auto refresh timer
			SET_APP_HALTED(FALSE);
		}
		else if (m_DropTargetFlag == SUBMITTEDCHG)
		{
			// submitted changes view accepted a drop
			// create a context menu
			::SendMessage(m_oldChgWnd, WM_COMMAND, MAKEWPARAM(ID_FILTER_SETVIEW, 0), 0);
		}
		else if (m_DropTargetFlag == LABELVIEW)
		{
			LabelDropMenu(TRUE);
		}
		else if (m_DropTargetFlag == LABELNOSEL)
		{
			LabelDropMenu(FALSE);
		}
		else if (m_DropTargetFlag == USERVIEW)
		{
			MessageBeep(0);
		}
		else MessageBeep(0);
	}
	else
		// Drag-drop ate the wm_rbuttonup
		OnRButtonUp(nFlags, point); 
}

void CDepotTreeCtrl::LabelDropMenu(BOOL bLabelSelected)
{
	// label view accepted a drop
	// tell the label view to create a context menu
	::SendMessage(m_labelWnd, WM_LABELDROPMENU, (WPARAM)bLabelSelected, (LPARAM)&m_DropTargetPt);

}

//////////////////////////////////////////////////////////////////////////////////////
// On update UI handlers

void CDepotTreeCtrl::OnUpdateFileLock(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
		&& ((anyChildren && !m_FilterDepot && !isRemoteFolder) || (!anyChildren && AnyLockable()))));
}

void CDepotTreeCtrl::OnUpdateFileUnlock(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
		&& ((anyChildren && !m_FilterDepot && !isRemoteFolder) || (!anyChildren && AnyUnlockable()))));
}

void CDepotTreeCtrl::OnUpdateFileInformation(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(!SERVER_BUSY() && GetSelectedCount()==1 && 
				   !IsSelected(m_Root) && 
				   ITEM_IS_FILE(GetSelectedItem(0)) );	
}

void CDepotTreeCtrl::OnUpdateFileOpendelete(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
		&& ((anyChildren && !m_FilterDepot && !isRemoteFolder) || (!anyChildren && AnyEditable()))));
}

void CDepotTreeCtrl::OnUpdateFileOpenedit(CCmdUI* pCmdUI) 
{
	BOOL anyChildren = AnyHaveChildren();
	BOOL anyEditable = anyChildren ? TRUE : AnyEditable();
	BOOL anyAddable  = anyChildren ? TRUE : AnyAddable();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));

	DWORD id;
	if (anyChildren || GET_P4REGPTR()->ShowEntireDepot() != SDF_LOCALTREE)
		id = m_InContextMenu ? IDS_CHKOUTFOREDIT : IDS_CHKOUTFOREDIT_CTRLE;
	else if (anyAddable && anyEditable)
		id = m_InContextMenu ? IDS_CHKOUTFOREDITORADD : IDS_CHKOUTFOREDITORADD_CTRLE;
	else if (anyAddable)
		id = IDS_CHKOUTFORADD;
	else
		id = m_InContextMenu ? IDS_CHKOUTFOREDIT : IDS_CHKOUTFOREDIT_CTRLE;
	pCmdUI->SetText(LoadStringResource(id));

	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
					&& ((anyChildren && !m_FilterDepot && !isRemoteFolder) 
									|| (!anyChildren && (anyEditable 
													  || anyAddable 
													  || AnyOpenedForInteg())))));	
}

void CDepotTreeCtrl::OnUpdateFiledropEdit(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));
	pCmdUI->Enable(!SERVER_BUSY() && ((anyChildren && !m_FilterDepot && !isRemoteFolder)
								   || (!anyChildren && AnyEditable())));
}

void CDepotTreeCtrl::OnUpdateFiledropDelete(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));
	pCmdUI->Enable(!SERVER_BUSY() && ((anyChildren && !m_FilterDepot && !isRemoteFolder)
								   || (!anyChildren && AnyEditable())));
}

void CDepotTreeCtrl::OnUpdateFiledropCancel(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(TRUE);
}

void CDepotTreeCtrl::OnUpdateFileRecover(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(!SERVER_BUSY() && GET_SERVERLEVEL() > 5 && 
		AnyRecoverable() && !AnyHaveChildren()  );	
}

void CDepotTreeCtrl::OnUpdateFileRevert(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
											&& AnyOpened() && !AnyHaveChildren()));
}

void CDepotTreeCtrl::OnUpdateFileRename(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL isRemoteFolder = anyChildren && IsInRemoteDepot(&(CString(GetCurrentItemPath())));
	pCmdUI->Enable(!SERVER_BUSY() 
		&& ((anyChildren && !m_FilterDepot && !isRemoteFolder)
		 || (!anyChildren && (AnyInView() && !AllNotInDepot())))
		&& !IsSelected(m_Root) 
		&& !MainFrame()->IsModlessUp()
		&& !MainFrame()->IsPendChgEditInProgress()
		&& !AnyInRemoteDepot() );
}

void CDepotTreeCtrl::OnUpdateFileIntegrate(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	pCmdUI->Enable(!SERVER_BUSY() 
		&& ((anyChildren && !m_FilterDepot) || !anyChildren)
		&& GetSelectedCount()
		&& !IsSelected(m_Root) 
		&& !MainFrame()->IsModlessUp()
		&& !MainFrame()->IsPendChgEditInProgress() );	
}

void CDepotTreeCtrl::OnUpdateFileGet(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL b = !SERVER_BUSY() && !IsSelected(m_Root)
		&& ((anyChildren && !m_FilterDepot) || (!anyChildren && AnyNotCurrent() && AnyInView()));
	if (!b && !anyChildren && !ITEM_IS_FILE(GetSelectedItem(0)) 
		&& (GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE 
		 || GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALP4))
	{
		CString text = GetItemText( GetSelectedItem(0) );
		b = text.Find( g_TrulyEmptyDir ) != -1 ? TRUE : FALSE;
	}
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, b));
}

void CDepotTreeCtrl::OnUpdateFileGetcustom(CCmdUI* pCmdUI) 
{
	// Can be called even if all files up to date, since we
	// may be getting at an old change number or label
	//
	pCmdUI->SetText( ( SERVER_IS_982_ORMORE ) 
			? LoadStringResource( IDS_SYNCTORLCD )
			: LoadStringResource( IDS_SYNCTORLC ) );
	BOOL anyChildren= AnyHaveChildren();
	BOOL b = !SERVER_BUSY() 
					&& !IsSelected(m_Root)
					&& GetSelectedCount() > 0 
					&& ((anyChildren && !m_FilterDepot) || (!anyChildren && AnyInView() && !AllNotInDepot()))
					&& !MainFrame()->IsModlessUp();
	if (!b && !anyChildren && !ITEM_IS_FILE(GetSelectedItem(0)) 
		&& (GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE 
		 || GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALP4))
	{
		CString text = GetItemText( GetSelectedItem(0) );
		b = text.Find( g_TrulyEmptyDir ) != -1 ? TRUE : FALSE;
	}
	pCmdUI->Enable(b);
}

void CDepotTreeCtrl::OnUpdateGetwhatif(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	BOOL b = !SERVER_BUSY() && !IsSelected(m_Root) &&
		((anyChildren && !m_FilterDepot) || (!anyChildren && AnyInView() && AnyNotCurrent()));
	if (!b && !anyChildren && !ITEM_IS_FILE(GetSelectedItem(0)) 
		&& (GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE 
		 || GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALP4))
	{
		CString text = GetItemText( GetSelectedItem(0) );
		b = text.Find( g_TrulyEmptyDir ) != -1 ? TRUE : FALSE;
	}
	pCmdUI->Enable(b);
}

void CDepotTreeCtrl::OnUpdateFileRemove(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	pCmdUI->Enable(!SERVER_BUSY() && !IsSelected(m_Root) &&
		((anyChildren && !m_FilterDepot) || (!anyChildren && AnyRemoveable())) );
}

void CDepotTreeCtrl::OnUpdateFileTimeLapse(CCmdUI* pCmdUI) 
{
	UpdateFileAnnotate(pCmdUI, TRUE);
}

void CDepotTreeCtrl::OnUpdateFileAnnotate(CCmdUI* pCmdUI) 
{
	UpdateFileAnnotate(pCmdUI, TRUE);
}

void CDepotTreeCtrl::UpdateFileAnnotate(CCmdUI* pCmdUI, BOOL bUnicodeOK) 
{
	BOOL enable = !SERVER_BUSY() && GET_SERVERLEVEL() >= 14
					&& GetSelectedCount()==1
					&& !IsSelected(m_Root) 
					&& ITEM_IS_FILE(GetSelectedItem(0))
					&& !AllNotInDepot();
	if( enable )
	{
		CP4FileStats *fs=m_FSColl.GetStats(GetLParam(GetSelectedItem(0)));
		CString fileType = fs->GetHeadType();
		enable = ((fileType.Find(_T("text")) != -1) 
			   || (fileType.Find(_T("symlink")) != -1)
			   || (bUnicodeOK && ((fileType.Find(_T("unicode")) != -1) 
							   || (fileType.Find(_T("utf16")) != -1)))) ? TRUE : FALSE;
	}
	pCmdUI->Enable(enable);
}

void CDepotTreeCtrl::OnUpdateFileRevisionhistory(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY()
					&& GetSelectedCount()==1
					&& !IsSelected(m_Root) 
#if 0	// define this to allow Rev Hist of folders
					&& (!ITEM_IS_FILE(GetSelectedItem(0)) || !AllNotInDepot())));
#else
					&& ITEM_IS_FILE(GetSelectedItem(0))
					&& !AllNotInDepot()));
#endif
}

void CDepotTreeCtrl::OnUpdateFileRevisiontree(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY()
					&& GetSelectedCount()==1
					&& !IsSelected(m_Root) 
					&& ITEM_IS_FILE(GetSelectedItem(0))
					&& !AllNotInDepot()));
}

void CDepotTreeCtrl::OnUpdatePositionDepot(CCmdUI* pCmdUI) 
{
	CString txt = LoadStringResource(IDS_POSITIONTOPATTERN);
	pCmdUI->SetText ( txt );
	pCmdUI->Enable(!SERVER_BUSY());
}

// This works for both Next and Prev
void CDepotTreeCtrl::OnUpdatePositionDepotNext(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(!SERVER_BUSY() && !m_P4Files_List.IsEmpty() 
		&& GetSelectedCount() && !GET_P4REGPTR()->SortByExtension());
}

void CDepotTreeCtrl::OnUpdatePositionChgs(CCmdUI* pCmdUI) 
{
	if (!m_InContextMenu)
		pCmdUI->SetText( LoadStringResource(IDS_POSITIONCHGS_DPTPANE) );
	pCmdUI->Enable(!SERVER_BUSY() && GetSelectedCount()==1 && (AnyOpened() || AnyOtherOpened()));
}

void CDepotTreeCtrl::OnUpdateShowDeletedFiles(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(!SERVER_BUSY());
	pCmdUI->SetCheck(GET_P4REGPTR()->ShowDeleted());
}

void CDepotTreeCtrl::OnUpdateWinExplore(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(GetSelectedCount() 
					&& ((!AnyHaveChildren() && AnyInView()) 
					   || GetSelectedCount()==1));
}

void CDepotTreeCtrl::OnUpdateCmdPrompt(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable((GetSelectedCount() 
				&& ((!AnyHaveChildren() && AnyInView()) 
				   || GetSelectedCount()==1))
			|| !SERVER_BUSY());
}

void CDepotTreeCtrl::OnUpdateFileDiffhead(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY()
		&& GetSelectedCount()==1
		&& !IsSelected(m_Root) && !AnyHaveChildren()
		&& (AnyOpened() || AnyRemoveable())));	
}

void CDepotTreeCtrl::OnUpdateFileDiff2(CCmdUI* pCmdUI) 
{
	int cnt = GetSelectedCount();
	pCmdUI->Enable(!SERVER_BUSY() && cnt && !IsSelected(m_Root) && !IsDepot(GetSelectedItem(0))
		        && ((cnt==1)
				 || (cnt==2 && !IsDepot(GetSelectedItem(1))
				  && (ITEM_IS_FILE(GetSelectedItem(0)) == ITEM_IS_FILE(GetSelectedItem(1))))));
}

void CDepotTreeCtrl::OnUpdateFileRefresh(CCmdUI* pCmdUI) 
{
	pCmdUI->SetText( LoadStringResource( IDS_REDOSYNCTOSAME ) );

	BOOL anyChildren= AnyHaveChildren();
	// Can be called for any files
	BOOL b = !SERVER_BUSY() && GetSelectedCount() > 0
		&& ((anyChildren && !m_FilterDepot) || (!anyChildren && AnyInView() && !AllNotInDepot()));
	if (!b && !anyChildren && !ITEM_IS_FILE(GetSelectedItem(0)) 
		&& (GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE 
		 || GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALP4))
	{
		CString text = GetItemText( GetSelectedItem(0) );
		b = text.Find( g_TrulyEmptyDir ) != -1 ? TRUE : FALSE;
	}
	pCmdUI->Enable(b);
}

void CDepotTreeCtrl::OnUpdateFileForceToHead(CCmdUI* pCmdUI) 
{
	BOOL anyChildren= AnyHaveChildren();
	// Can be called for any files
	BOOL b = !SERVER_BUSY() && GET_SERVERLEVEL( ) > 3 && GetSelectedCount() > 0
		&& ((anyChildren && !m_FilterDepot) || (!anyChildren && AnyInView() && !AllNotInDepot()));
	if (!b && !anyChildren && !ITEM_IS_FILE(GetSelectedItem(0)) 
		&& (GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALTREE 
		 || GET_P4REGPTR( )->ShowEntireDepot( ) == SDF_LOCALP4))
	{
		CString text = GetItemText( GetSelectedItem(0) );
		b = text.Find( g_TrulyEmptyDir ) != -1 ? TRUE : FALSE;
	}
	pCmdUI->Enable(b);
}

void CDepotTreeCtrl::OnUpdateFileSubmit(CCmdUI* pCmdUI) 
{
	BOOL b = FALSE;
	HTREEITEM item;
	if(GetSelectedCount()==1 && ITEM_IS_FILE(item = GetSelectedItem(0)) && !AnyHaveChildren())
	{
		int index=GetLParam(item);
		CP4FileStats *fs=m_FSColl.GetStats(index);
		if (fs->GetMyOpenAction())
			b = !SERVER_BUSY();
	}
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, b));
}


////////////////////////////////////////////////////////
// Utility functions

int CDepotTreeCtrl::LeafSelectedCount()
{
	int rc;
	int count=0;

	for(int i=GetSelectedCount()-1; i >= 0; i--)
	{
		HTREEITEM item= GetSelectedItem(i);
		rc= GetLeafCount(item);
		if (rc < 0)
			return(-1);	// count cannot be determined at this time
		count+= rc;
	}
	return count;
}

// Warning: next function recurses
int CDepotTreeCtrl::GetLeafCount(HTREEITEM item)
{
	int rc;
	int count=0;
	if(HasChildren(item))
	{
		HTREEITEM cItem= TreeView_GetChild(m_hWnd, item);
		if (cItem == NULL)
			return(-1);	// count cannot be determined at this time - item never expanded
		while(cItem != NULL)
		{
			rc= GetLeafCount(cItem);
			if (rc < 0)
				return(-1);	// count cannot be determined at this time
			count+= rc;
			cItem= TreeView_GetNextSibling(m_hWnd, cItem);
		}
	}
	else
		count=1;

	return count;
}


BOOL CDepotTreeCtrl::AnyHaveChildren()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgHaveChildren))
		return m_AnyHaveChildren;
	m_FlgHaveChildren = idleFlag;

	m_AnyHaveChildren=FALSE;
	TV_ITEM item;

	for( int i=GetSelectedCount()-1; i>=0; i-- )
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );	
		if( item.cChildren ==1 )   // a directory
		{
			m_AnyHaveChildren=TRUE;
			break;
		}
	}
	return m_AnyHaveChildren;
}

BOOL CDepotTreeCtrl::HasChildren(HTREEITEM currentItem)
{
	BOOL hasChildren=FALSE;
	TV_ITEM item;

	if(currentItem != NULL)
	{
		item.hItem=currentItem;
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		if(TreeView_GetItem(m_hWnd, &item ) && item.cChildren ==1)   // a directory
			hasChildren=TRUE;
	}
	return hasChildren;
}


BOOL CDepotTreeCtrl::AnyRemoveable()
{
	BOOL current=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetHaveRev() > 0 && fs->GetMyOpenAction()==0)
			{
				current=TRUE;
				break;
			}
		}
	}
	return current;
}

BOOL CDepotTreeCtrl::AnyInView()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgInView))
		return m_AnyInView;
	m_FlgInView = idleFlag;

	m_AnyInView=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->InClientView())
			{
				m_AnyInView=TRUE;
				break;
			}
		}
	}
	return m_AnyInView;
}

BOOL CDepotTreeCtrl::AllInView()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgAllInView))
		return m_AllInView;
	m_FlgAllInView = idleFlag;

	m_AllInView=TRUE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(!fs->InClientView())
			{
				m_AllInView=FALSE;
				break;
			}
		}
	}
	return m_AllInView;
}

BOOL CDepotTreeCtrl::AnyNotCurrent()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgNotCurrent))
		return m_AnyNotCurrent;
	m_FlgNotCurrent = idleFlag;

	m_AnyNotCurrent=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetHaveRev() < fs->GetHeadRev())
			{
				m_AnyNotCurrent=TRUE;
				break;
			}
		}
	}
	return m_AnyNotCurrent;
}


BOOL CDepotTreeCtrl::AllNotInDepot()
{
	if (GET_P4REGPTR( )->ShowEntireDepot( ) != SDF_LOCALTREE)
		return FALSE;

	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgAllNotInDepot))
		return m_AllNotInDepot;
	m_FlgAllNotInDepot = idleFlag;

	m_AllNotInDepot=TRUE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(!fs->IsNotInDepot())
			{
				m_AllNotInDepot=FALSE;
				break;
			}
		}
	}
	return m_AllNotInDepot;
}


BOOL CDepotTreeCtrl::AnyInRemoteDepot()
{
	if (m_RemoteDepotList.IsEmpty())
		return FALSE;

	BOOL anyInRemote=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			CString path = fs->GetFullDepotPath();
			if(IsInRemoteDepot(&path))
			{
				anyInRemote=TRUE;
				break;
			}
		}
	}
	return anyInRemote;
}


BOOL CDepotTreeCtrl::AnyAddable()
{
	if (GET_P4REGPTR( )->ShowEntireDepot( ) != SDF_LOCALTREE)
		return FALSE;

	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgAddable))
		return m_AnyAddable;
	m_FlgAddable = idleFlag;

	m_AnyAddable=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction()== 0 && fs->IsNotInDepot())
			{
				m_AnyAddable=TRUE;
				break;
			}
		}
	}
	return m_AnyAddable;
}


BOOL CDepotTreeCtrl::AllAddable()
{
	if (GET_P4REGPTR( )->ShowEntireDepot( ) != SDF_LOCALTREE)
		return FALSE;

	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgAllAddable))
		return m_AllAddable;
	m_FlgAllAddable = idleFlag;

	m_AllAddable=TRUE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction() || !fs->IsNotInDepot())
			{
				m_AllAddable=FALSE;
				break;
			}
		}
		else
		{
			m_AllAddable=FALSE;
			break;
		}
	}
	return m_AllAddable;
}


BOOL CDepotTreeCtrl::AnyEditable()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgEditable))
		return m_AnyEditable;
	m_FlgEditable = idleFlag;

	m_AnyEditable=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction()==0 && fs->GetHeadRev() != 0 
				&& fs->GetHaveRev() != 0 && fs->InClientView()
				&& !fs->IsOtherOpenExclusive()
				&& !IsInRemoteDepot(&(CString(fs->GetFullDepotPath()))))
			{
				m_AnyEditable=TRUE;
				break;
			}
		}
	}
	return m_AnyEditable;
}


BOOL CDepotTreeCtrl::AnyOpenedForInteg()
{
	BOOL editable=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction()==F_INTEGRATE && fs->GetHeadRev() != 0 
				&& fs->GetHaveRev() != 0 && fs->InClientView())
			{
				editable=TRUE;
				break;
			}
		}
	}
	return editable;
}


BOOL CDepotTreeCtrl::AnyLockable()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgLockable))
		return m_AnyLockable;
	m_FlgLockable = idleFlag;

	m_AnyLockable=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction() > 0 && !fs->IsMyLock() && !fs->IsOtherLock() )
			{
				m_AnyLockable=TRUE;
				break;
			}
		}
	}
	return m_AnyLockable;
}

BOOL CDepotTreeCtrl::AnyRecoverable()
{
	BOOL recoverable=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if( fs->GetHeadAction() == F_DELETE && 
				fs->GetMyOpenAction()==0 && 
				fs->GetHeadRev() != 0 && /*fs->GetHaveRev() != 0 &&*/
				fs->GetHaveRev() < fs->GetHeadRev() && 
				fs->InClientView())
			{
				recoverable=TRUE;
				break;
			}
		}
	}
	return recoverable;
}

BOOL CDepotTreeCtrl::AnyUnlockable()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgUnlockable))
		return m_AnyUnlockable;
	m_FlgUnlockable = idleFlag;

	m_AnyUnlockable=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if( fs->IsMyLock() )
			{
				m_AnyUnlockable=TRUE;
				break;
			}
		}
	}
	return m_AnyUnlockable;
}

BOOL CDepotTreeCtrl::AnyOpened()
{
	int idleFlag;
	if (((idleFlag = TheApp()->m_IdleFlag) != 0)
	  && (idleFlag == m_FlgOpened))
		return m_AnyOpened;
	m_FlgOpened = idleFlag;

	m_AnyOpened=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction() > 0)
			{
				m_AnyOpened=TRUE;
				break;
			}
		}
	}
	return m_AnyOpened;
}

BOOL CDepotTreeCtrl::AnyDeleted()
{
	BOOL someDeleted=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if ((fs->GetHeadAction() == F_DELETE) && (fs->GetHaveRev() == 0))
			{
				someDeleted=TRUE;
				break;
			}
		}
	}
	return someDeleted;
}

BOOL CDepotTreeCtrl::IsDeleted(HTREEITEM currentItem)
{
	BOOL deleted=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	if(currentItem != NULL)
	{
		item.hItem=currentItem;
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;

		if(TreeView_GetItem(m_hWnd, &item ) && ITEM_IS_A_FILE_NOT_A_SUBDIR)
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if ((fs->GetHeadAction() == F_DELETE) && (fs->GetHaveRev() == 0))
				deleted=TRUE;
		}
	}
	return deleted;
}

BOOL CDepotTreeCtrl::IsOpened(HTREEITEM currentItem)
{
	BOOL opened=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	if(currentItem != NULL)
	{
		item.hItem=currentItem;
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;

		if(TreeView_GetItem(m_hWnd, &item ) && ITEM_IS_A_FILE_NOT_A_SUBDIR)
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetMyOpenAction() > 0)
				opened=TRUE;
		}
	}
	return opened;
}

BOOL CDepotTreeCtrl::AnyOtherOpened()
{
	BOOL opened=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		item.hItem=GetSelectedItem(i);
		item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
		TreeView_GetItem(m_hWnd, &item );
		if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
		{
			fs=m_FSColl.GetStats((int) item.lParam);
			if(fs->GetOtherOpenAction() > 0)
			{
				opened=TRUE;
				break;
			}
		}
	}
	return opened;
}

BOOL CDepotTreeCtrl::IsInView()
{
	BOOL InView=FALSE;
	TV_ITEM item;
	CP4FileStats *fs;

	// Is the one and only selection a file stranded outside
	// the current client view? 
	if(GetSelectedCount() != 1)
		{ ASSERT(0); return InView; }

	item.hItem=GetSelectedItem(0);
	item.mask=TVIF_HANDLE | TVIF_PARAM;
	TreeView_GetItem(m_hWnd, &item );
	if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
	{
		fs=m_FSColl.GetStats((int) item.lParam);
		if(fs->InClientView())
			InView=TRUE;
		
	}
	return InView;
}


/*
	_________________________________________________________________

	Produce a dialog showing fstat and opened file information for
	the selected file.  This info is all retrieved from the depot
	and pending changes windows.
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnFileInformation() 
{
	// Find the file and get its name 
	HTREEITEM currentItem=GetLastSelection(); 
	if (currentItem == NULL || GetSelectedCount() > 1 || HasChildren( currentItem) )
	{
		ASSERT(0);
		return;
	}

	AssembleStringList( &m_StringList );
	
	CCmd_Opened *pCmd= new CCmd_Opened;
	pCmd->Init( m_hWnd, RUN_ASYNC);
	pCmd->SetItemRef( currentItem );
	pCmd->SetAlternateReplyMsg( WM_P4FILEINFORMATION );

	if( pCmd->Run( TRUE, FALSE, -1, &m_StringList ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_FILE_INFORMATION) );
	else
		delete pCmd;
}


LRESULT CDepotTreeCtrl::OnP4FileInformation( WPARAM wParam, LPARAM lParam )
{
	CCmd_Opened *pCmd= (CCmd_Opened *) wParam;
	
	if(!pCmd->GetError())
	{
		HTREEITEM currentItem= pCmd->GetItemRef();

		// Initialize the file info dialog
		CFileInfoDlg *dlg = new CFileInfoDlg(this);

		CP4FileStats *stats= m_FSColl.GetStats(GetLParam(currentItem));
		dlg->m_DepotPath= stats->GetFullDepotPath();
		dlg->m_ClientPath= stats->GetFullClientPath();
		if(dlg->m_ClientPath.GetLength() == 0)
			dlg->m_ClientPath= LoadStringResource(IDS_NOT_IN_CLIENT_VIEW);
		
		dlg->m_HeadRev.Format(_T("%ld"), stats->GetHeadRev());
		dlg->m_HaveRev.Format(_T("%ld"), stats->GetHaveRev());
		
		dlg->m_HeadAction= stats->GetActionStr(stats->GetHeadAction());
		dlg->m_HeadChange.Format(_T("%ld"), stats->GetHeadChangeNum());
		dlg->m_HeadType= stats->GetHeadType();
		dlg->m_ModTime= stats->GetFormattedHeadTime();
		dlg->m_FileSize= stats->GetFileSize();

		// Check for open/lock by this user
		CString thisuser=GET_P4REGPTR()->GetMyID();
		
		if(stats->IsMyLock())
			dlg->m_LockedBy= thisuser;
		
		// Add to list other users who have this file open
		//
		CObList *list= pCmd->GetList();
		ASSERT_KINDOF(CObList, list);

        POSITION pos= list->GetHeadPosition();
		while(pos != NULL)
		{
			CP4FileStats *fs= (CP4FileStats *) list->GetNext(pos);
			
			CString str;
			CString strUser;
			CString strChange;
			CString strAction;

			if( fs->GetOpenChangeNum() == 0 )
				strChange= LoadStringResource(IDS_DEFAULT_CHANGE);
			else
				strChange.FormatMessage(IDS_CHANGE_n, fs->GetOpenChangeNum()); 

			strUser= fs->GetOtherUsers();
			if( fs->IsMyOpen() && strUser.IsEmpty() )
			{
				strUser= thisuser;
				strAction= fs->GetActionStr(fs->GetMyOpenAction());
			}
			else
				strAction= fs->GetActionStr(fs->GetOtherOpenAction());

			str.Format(_T("%s - %s (%s)"), strUser, strChange, strAction);

			if( fs->IsOtherLock() )
				str += _T(" ") + LoadStringResource(IDS_STAR_LOCKED);
			
			dlg->m_StrList.AddHead( str );
			

			delete fs;
		}
		// Display the info
		if (!dlg->Create(IDD_FILE_INFORMATION, this))	// display the description dialog box
		{
			dlg->DestroyWindow();	// some error! clean up
			delete dlg;
		}
	}

	MainFrame()->ClearStatus();
	delete pCmd;
	
	return 0;
}

LRESULT CDepotTreeCtrl::OnP4EndFileInformation( WPARAM wParam, LPARAM lParam )
{
	CFileInfoDlg *dlg = (CFileInfoDlg *)lParam;
	dlg->DestroyWindow();
	return TRUE;
}

///////////////////////////////////////////////////////////////////////////////////////////////
// Support for file viewing and editing
///////////////////////////////////////////////////////////////////////////////////////////////
//
// Command UI update functions:
//		OnUpdateFileAutoBrowse(CCmdUI* pCmdUI) 
//
//		OnUpdateFileAutoedit(CCmdUI* pCmdUI) 
//		----> AutoEditPossible(BOOL *mustWaitForServer)
//
// Main Menu command handlers
//		OnFileQuickedit()		(will use associated app)
//		----> RunAssocViewer(BOOL editing)
//
//		OnFileAutoedit()		(will start with chooser dialog)
//		OnFileAutobrowse()			(will start with chooser dialog)
//		----> ChooseAndRunViewer(BOOL editing)
//
// Context Menu command handlers
//		OnFileMRUEditor(UINT  nID)
//		OnFileMRUBrowser(UINT  nID)
//		----> RunMRUViewer(UINT  nID, BOOL editing)
//
//		OnFileNewEditor()
//		OnFileNewBrowser()		
//		----> FindAndRunNewViewer(BOOL editing)
//
//		OnFileQuickedit()
//		OnFileQuickbrowse()	(will use associated app)
//		----> RunAssocViewer(BOOL editing)
//
// Command goes to server
//		GetViewerFile(LPCTSTR appPath)
//		----> CP4::PrepBrowse(HWND replywnd, LPCTSTR depotPathRev) (only 1st time)
//		----> CP4::PrepEdit(HWND replywnd, LPCTSTR depotPath) (only if reqd)
//		----> straight to RunViewer() if we already have file on client
//
// Application spawn actually happens here
//		OnP4PrepareForViewer(WPARAM wParam, LPARAM lParam)
//		----> RunViewer()
//			calls viewer dlg if error spawning app
//
///////////////////////////////////////////////////////////////////////////////////////////////


void CDepotTreeCtrl::OnUpdateFileAutobrowse(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY()
		&& GetSelectedCount()==1
		&& !IsSelected(m_Root)
		&& ITEM_IS_FILE(GetSelectedItem(0))
		&& !IsDeleted(GetSelectedItem(0))));
}

void CDepotTreeCtrl::OnUpdateFileAutoedit(CCmdUI* pCmdUI) 
{
	BOOL mustWaitForServer;
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, AutoEditPossible(&mustWaitForServer)
		&& !mustWaitForServer && !SERVER_BUSY()
		&& !IsDeleted(GetSelectedItem(0))));
}

BOOL CDepotTreeCtrl::AutoEditPossible(BOOL *mustWaitForServer)
{
	BOOL canEdit=FALSE;
	
	if(GetSelectedCount() == 1 && !IsSelected(m_Root) && !AnyHaveChildren() )
	{
		// If we already have it open for edit, there should
		// be no trouble editing it
		if(IsOpened(GetSelectedItem(0)))
		{
			canEdit=TRUE;
			*mustWaitForServer=FALSE;
		}
		// If in client view, we should be able to edit. Server must not
		// be busy because an edit and possibly a get will be required.
		// Also must not be +l filetype that is opened by someone else.
		else 
		{
			TV_ITEM item;
			CP4FileStats *fs;

			item.hItem=GetSelectedItem(0);
			item.mask=TVIF_HANDLE| TVIF_CHILDREN | TVIF_PARAM;
			TreeView_GetItem(m_hWnd, &item );
			if ( ITEM_IS_A_FILE_NOT_A_SUBDIR )
			{
				fs=m_FSColl.GetStats((int) item.lParam);
				if(fs->InClientView() && !fs->IsOtherOpenExclusive())
				{
					canEdit=TRUE;
					*mustWaitForServer=SERVER_BUSY();
				}
			}
		}
	}
	return canEdit;
}

///////////
// auto run  means to allow choosing of the viewer app, and then automatically 
//			get the head rev and fire up that viewer app

void CDepotTreeCtrl::OnFileAutoedit() 
{
	BOOL mustWaitForServer;
	if(!AutoEditPossible(&mustWaitForServer) || mustWaitForServer)
		return;

	if(	GET_P4REGPTR()->GetWarnAlreadyOpened() && AnyOtherOpened())
	{
		// Prevent anything from running while dlg is up
		SET_APP_HALTED(TRUE);
		if(AfxMessageBox(IDS_ONE_OR_MORE_FILES_HAVE_BEEN_OPENED_BY_OTHER_USERS, 
						MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
		{
			SET_APP_HALTED(FALSE);
			return;
		}
		SET_APP_HALTED(FALSE);
	}

	m_ViewItem=GetLastSelection();
	m_ViewFilePath.Empty();
	m_Editing=TRUE;
	ChooseAndRunViewer();
}

void CDepotTreeCtrl::OnFileAutobrowse() 
{
	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=FALSE;
	ChooseAndRunViewer();
}

void CDepotTreeCtrl::ChooseAndRunViewer()
{
	if (m_ViewItem == NULL && m_ViewFilePath.IsEmpty())
		ASSERT(0);
	else
	{
		// Ask the user to pick a viewer
		CViewerDlg dlg;
		if(dlg.DoModal() == IDCANCEL)
			return;

		m_Viewer=dlg.GetViewer();
		if(m_Viewer != _T("SHELLEXEC"))
			GET_P4REGPTR()->AddMRUViewer(m_Viewer);

		PrepareForViewer();
	}
}

//////////
// MRU versions will use the selected MRU viewer and will then automatically
//			get the head rev and run the viewer app

void CDepotTreeCtrl::OnFileMRUEditor(UINT  nID)
{
	BOOL mustWaitForServer;
	if(!AutoEditPossible(&mustWaitForServer) || mustWaitForServer)
		return;

	if(	GET_P4REGPTR()->GetWarnAlreadyOpened() && AnyOtherOpened())
	{
		if(AfxMessageBox(IDS_ONE_OR_MORE_FILES_HAVE_BEEN_OPENED_BY_OTHER_USERS, 
						MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
			return;
	}

	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=TRUE;
	RunMRUViewer(nID - ID_FILE_EDITOR_1);
}

void CDepotTreeCtrl::OnFileMRUBrowse(UINT  nID)
{
	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=FALSE;
	RunMRUViewer(nID - ID_FILE_BROWSER_1);
}

void CDepotTreeCtrl::RunMRUViewer(UINT nID)
{
	ASSERT(nID >= 0);

	if (m_ViewItem == NULL && m_ViewFilePath.IsEmpty())
		ASSERT(0);
	else
	{
		m_Viewer= GET_P4REGPTR()->GetMRUViewer( nID );
		GET_P4REGPTR()->AddMRUViewer(m_Viewer);
		PrepareForViewer();
	}
}


//////////
// New Editor versions will run common file dialog to find a viewer, add that
//		viewer to the MRU list, then viewer and will then automatically
//			get the head rev and run the viewer app

void CDepotTreeCtrl::OnFileNewEditor()
{
	BOOL mustWaitForServer;
	if(!AutoEditPossible(&mustWaitForServer) || mustWaitForServer)
		return;

	if(	GET_P4REGPTR()->GetWarnAlreadyOpened() && AnyOtherOpened())
	{
		if(AfxMessageBox(IDS_ONE_OR_MORE_FILES_HAVE_BEEN_OPENED_BY_OTHER_USERS, 
						MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
			return;
	}

	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=TRUE;
	FindAndRunNewViewer();
}

void CDepotTreeCtrl::OnFileNewBrowser()
{
	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=FALSE;
	FindAndRunNewViewer();
}

void CDepotTreeCtrl::FindAndRunNewViewer()
{
	if (m_ViewItem == NULL && m_ViewFilePath.IsEmpty())
		ASSERT(0);
	else
	{
		// Fire up a common dlg to find new file
		CFileDialog fDlg(TRUE, _T("exe"), NULL,  
			OFN_FILEMUSTEXIST | OFN_HIDEREADONLY | OFN_NONETWORKBUTTON | OFN_PATHMUSTEXIST,
			LoadStringResource(IDS_VIEWER_FILTER), this,
			MainFrame()->m_osVer.dwMajorVersion < 5 ? OPENFILENAME_SIZE_VERSION_400 : sizeof(OPENFILENAME)); 

		// Set the dlg caption
        CString title = LoadStringResource(IDS_FIND_FILE_EDITOR_VIEWER);
		fDlg.m_ofn.lpstrTitle = title;
		// We dont need no stinking file title
		fDlg.m_ofn.lpstrFileTitle=NULL;

		if(fDlg.DoModal() == IDOK)
		{
			m_Viewer= fDlg.GetPathName();
			if(m_Viewer.GetLength() > 0)
			{
				// User selected a valid viewer, so try to run it
				GET_P4REGPTR()->AddMRUViewer(m_Viewer);
				RunMRUViewer(0);
			}
		}
	}
}

void CDepotTreeCtrl::OnUpdateRemoveViewer(CCmdUI* pCmdUI)
{
	BOOL b = FALSE;
	for(int i=0; i < MAX_MRU_VIEWERS; i++)
	{
		if( GET_P4REGPTR()->GetMRUViewerName(i).GetLength() > 0 )
		{
			b = TRUE;
			break;
		}
	}
	pCmdUI->Enable( b );
}

void CDepotTreeCtrl::OnRemoveViewer()
{
	CRemoveViewer dlg;
	dlg.DoModal();
}


//////////
// Run the associated viewer on the head rev - view only

LRESULT CDepotTreeCtrl::OnViewHead(WPARAM wParam, LPARAM lParam)
{
	m_Viewer= _T("SHELLEXEC");

	// Fetch the head revision of the file to a temp filename
	CString itemStr= (TCHAR*)wParam;
	CP4FileStats *fs = (CP4FileStats *)lParam;
	CCmd_PrepBrowse *pCmd= new CCmd_PrepBrowse;
	pCmd->Init( m_hWnd, RUN_ASYNC);
	pCmd->SetFileType(fs->IsTextFile() ? FST_TEXT : FST_BINARY);
	CString fType = fs->GetHeadType();
	if( pCmd->Run( itemStr, fType, fs->GetHeadRev(), TheApp()->m_bNoCRLF ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_FETCHING_FILE) );
	else
		delete pCmd;
	return 0;
}

//////////
// QuickEdit versions will automatically get the head rev and run the associated viewer app

void CDepotTreeCtrl::OnFileQuickedit() 
{
	BOOL mustWaitForServer;
	if(!AutoEditPossible(&mustWaitForServer) || mustWaitForServer)
		return;
	
	if(	GET_P4REGPTR()->GetWarnAlreadyOpened() && AnyOtherOpened())
	{
		if(AfxMessageBox(IDS_ONE_OR_MORE_FILES_HAVE_BEEN_OPENED_BY_OTHER_USERS, 
						MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
			return;
	}

	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=TRUE;
	RunAssocViewer();
}

void CDepotTreeCtrl::OnFileQuickbrowse()
{
	m_ViewItem=GetLastSelection(); 
	m_ViewFilePath.Empty();
	m_Editing=FALSE;
	RunAssocViewer();
}

void CDepotTreeCtrl::RunAssocViewer()
{
	if (m_ViewItem == NULL && m_ViewFilePath.IsEmpty())
		ASSERT(0);
	else
	{
		m_Viewer=_T("SHELLEXEC");
		PrepareForViewer();
	}
}


void CDepotTreeCtrl::PrepareForViewer()
{
	ASSERT(!m_Viewer.IsEmpty());

	// Shortcut for edit requests coming from Delta view, because
	// there is no file lookup required, and we know for certain
	// that the file is open; plus we don't have access to 'fs'
	if( m_ViewItem == NULL && !m_ViewFilePath.IsEmpty() )
	{
		if (m_Editing)
			RunViewer();
		else	// it's opened for integrate
		{
			int i;
			DWORD errorCode  = 0;
            CString tempPath = GET_P4REGPTR()->GetTempDir();
            CString fileName = m_ViewFilePath;
            CString filePath;

			if ((i = ReverseFindMBCS(fileName, _T('\\'))) != -1)
				fileName = fileName.Mid(i+1);
	    	for( i=0; i< 100; i++)
	        {
		        filePath.Format(_T("%s\\ReadOnly-%d-Rev-Integ-%s"), tempPath, i, fileName);
				int j;
				while ((j = filePath.Find(':', 2)) != -1)
					filePath.SetAt(j, '_');
				while ((j = filePath.FindOneOf(_T("/*?\"<>|"))) != -1)
					filePath.SetAt(j, '_');
		        if( CopyFile(m_ViewFilePath, filePath, TRUE) )
                {
                    // Verify that it is readonly
					m_ViewFilePath = filePath;
                    SetFileAttributes( m_ViewFilePath, FILE_ATTRIBUTE_READONLY );
                    RunViewer();
                    return;
                };
				errorCode = GetLastError();
				if ((errorCode == ERROR_FILE_NOT_FOUND) 
				 || (errorCode == ERROR_PATH_NOT_FOUND) 
				 || (errorCode == ERROR_HANDLE_DISK_FULL))
					break;
	        }
			if (errorCode == ERROR_HANDLE_DISK_FULL)
			{
				// after 100 tries, we couldn't open the temp file
				CString ErrorTxt;
				ErrorTxt.FormatMessage(IDS_DISKFULL_OPENING_TEMP_FILE_s, m_ViewFilePath);
				AddToStatus(ErrorTxt, SV_ERROR);
			}
		}
		return;
	}

	ASSERT(m_ViewItem != NULL);
	DWORD index=GetLParam(m_ViewItem);
	CP4FileStats *fs=m_FSColl.GetStats(index);
	if ((fs->GetHeadAction() == F_DELETE) && (fs->GetHaveRev() == 0))
	{
		MessageBeep(0);
		return;
	}

	m_ViewFileIsText= fs->IsTextFile();
	CString itemStr= GetItemPath(m_ViewItem);

	if ((m_Editing && fs->IsMyOpen())
	 || ((GET_P4REGPTR()->ShowEntireDepot() == SDF_LOCALTREE) && !m_Editing && !fs->GetHeadRev()))
	{
		if((fs->GetMyOpenAction()==F_INTEGRATE || fs->GetMyOpenAction()==F_BRANCH)
		 && fs->GetHaveRev() != 0)
		{
			m_StringList.RemoveAll();
			m_StringList.AddHead(fs->GetFullClientPath());
			if(!SERVER_BUSY())
			{	
				CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;
				pCmd->Init( m_changeWnd, RUN_ASYNC, HOLD_LOCK );
				if( pCmd->Run( &m_StringList, P4EDIT, 0 ) )
				{
					MainFrame()->UpdateStatus( LoadStringResource(IDS_OPENING_FILES_FOR_EDIT) );
					m_ViewFilePath= fs->GetFullClientPath();
					RunViewer();
				}
				else
					delete pCmd;
			}
			else
				AddToStatus(LoadStringResource(IDS_SERVER_BUSY_UNABLE_TO_CHECK_OUT_FILE), SV_WARNING);	
		}
		else
		{
			// We already have the file open, just set viewfilepath 
			// and fire up the view app
			m_ViewFilePath= fs->GetFullClientPath();
			RunViewer();
		}
	}
	else if(m_Editing)
	{
		m_ViewFilePath= fs->GetFullClientPath();

		int getHead= IDNO;

		// See if the user wants to fetch the 
		// head revision of the file first
		if(fs->GetHeadRev() > fs->GetHaveRev())
			 getHead=AfxMessageBox( 
						IDS_CLIENT_FILE_NOT_HEAD__SYNC_TO_HEAD,
						MB_ICONEXCLAMATION | MB_YESNOCANCEL );

		switch( getHead )
		{
		case IDNO:
		case IDYES:
			if(!SERVER_BUSY())
			{	
				m_OpenUnderChangeNumber=0;
				CCmd_PrepEdit *pCmd= new CCmd_PrepEdit;
				pCmd->Init( m_hWnd, RUN_ASYNC, LOSE_LOCK);
				pCmd->SetWarnIfLocked(TRUE);
				if( pCmd->Run( itemStr, getHead == IDYES, fs->IsNotInDepot() ) )
					MainFrame()->UpdateStatus( LoadStringResource(IDS_EDITING_HEAD_REV) );
				else
					delete pCmd;
			}
			else
				AddToStatus(LoadStringResource(IDS_SERVER_BUSY_UNABLE_TO_CHECK_OUT_FILE), SV_WARNING);	

		case IDCANCEL:
		default:
			// User cancelled edit
			return;
		} //switch
	} 
	else
	{
        // If browsing a file that we HAVE on the client AND IT'S NOT OPEN or IT'S OPEN FOR INTEGRATE, 
		// copy the file to a temp file and then fire up the viewer.
		// Use a temp file to ensure that the browse app gets a readonly file,
		// (browse is a read only activity)

		BOOL bMD5ok = TRUE;
		BOOL bUseLive = FALSE;
        if( fs->GetHaveRev() > 0 && !CString(fs->GetFullClientPath()).IsEmpty() 
			&& (!fs->IsMyOpen() || (fs->GetMyOpenAction() == F_INTEGRATE))
			&& ((bMD5ok = TheApp()->digestIsSame(fs)) == TRUE
			 || ((bUseLive = FileExtUsesLiveFileToView(fs->GetFullClientPath())) == TRUE)
			  && (GET_P4REGPTR()->GetUseTempForView() == 2)))
        {
			if ((GET_P4REGPTR()->GetUseTempForView() == 1) 
			 || ((GET_P4REGPTR()->GetUseTempForView() == 2) 
			  && !FileExtUsesLiveFileToView(fs->GetFullClientPath())))
			{
				DWORD errorCode = 0;
				int fileRev= fs->GetHaveRev();
				CString fileName= GetItemName(m_ViewItem);
				CString tempPath= GET_P4REGPTR()->GetTempDir();

	    		for( int i=0; i< 100; i++)
				{
					m_ViewFilePath.Format(_T("%s\\ReadOnly-%d-Rev-%d-%s"), tempPath, i, fileRev, fileName);
					int j;
					while ((j = m_ViewFilePath.Find(':', 2)) != -1)
						m_ViewFilePath.SetAt(j, '_');
					while ((j = m_ViewFilePath.FindOneOf(_T("/*?\"<>|"))) != -1)
						m_ViewFilePath.SetAt(j, '_');
					if( CopyFile(fs->GetFullClientPath(), m_ViewFilePath, TRUE) )
					{
						// Verify that it is readonly
						SetFileAttributes( m_ViewFilePath, FILE_ATTRIBUTE_READONLY );
						RunViewer();
						return;
					};
					errorCode = GetLastError();
					if ((errorCode == ERROR_FILE_NOT_FOUND) 
					 || (errorCode == ERROR_PATH_NOT_FOUND) 
					 || (errorCode == ERROR_HANDLE_DISK_FULL))
						break;
				}
				if (errorCode == ERROR_HANDLE_DISK_FULL)
				{
					// after 100 tries, we couldn't open the temp file
					CString ErrorTxt;
					ErrorTxt.FormatMessage(IDS_DISKFULL_OPENING_TEMP_FILE_s, m_ViewFilePath);
					AddToStatus(ErrorTxt, SV_ERROR);
					return;
				}
			}
			else
			{
				if (!bMD5ok && bUseLive)
					TheApp()->StatusAdd(LoadStringResource(IDS_USINGLOCALBUTITSCHGED), SV_WARNING);
				m_ViewFilePath = fs->GetFullClientPath();
				RunViewer();
				return;
			}
        }

		// The file is not on the client OR NOT OPENED OR USER WANTS THE DEPOT VERSION,
		// so fetch the head revision of the file to a temp filename
		CCmd_PrepBrowse *pCmd= new CCmd_PrepBrowse;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		pCmd->SetFileType(fs->IsTextFile() ? FST_TEXT : FST_BINARY);
		CString fType = fs->GetHeadType();
		if( pCmd->Run( itemStr, fType, fs->GetHeadRev(), TheApp()->m_bNoCRLF ),
			fs->IsTextFile() ? FST_TEXT : FST_BINARY )
		{
			MainFrame()->UpdateStatus( LoadStringResource(IDS_FETCHING_FILE) );
		}
		else
		{
			delete pCmd;
		}
	}
}


BOOL CDepotTreeCtrl::FileExtUsesLiveFileToView(LPCTSTR path)
{
	int i, j, k;
	CString pathstr = path;
	if ((i = pathstr.ReverseFind(_T('.'))) != -1)
	{
		CString ext;
		ext = pathstr.Mid(i+1);
		if (!ext.IsEmpty())
		{
			CString extList = GET_P4REGPTR()->GetUseTempForExts();
			for (i=0; !extList.IsEmpty(); )
			{
				if ((j = extList.Find(ext)) == -1)
					break;
				k = j + ext.GetLength();
				if (((j-- == 0)                 || extList.GetAt(j) == _T(','))
				 && ((k >= extList.GetLength()) || extList.GetAt(k) == _T(',')))
					return TRUE;
				extList = extList.Mid(k+1);
			}
		}
	}
	return FALSE;
}

// Message handlers
LRESULT CDepotTreeCtrl::OnP4PrepEdit(WPARAM wParam, LPARAM lParam)
{
	CCmd_PrepEdit *pCmd= (CCmd_PrepEdit *) wParam;
	MainFrame()->ClearStatus();
	
	if(!pCmd->GetError())
		RunViewer();
	
	delete pCmd;
	return 0;
}

LRESULT CDepotTreeCtrl::OnP4PrepBrowse(WPARAM wParam, LPARAM lParam)
{
	CCmd_PrepBrowse *pCmd= (CCmd_PrepBrowse *) wParam;
	MainFrame()->ClearStatus();

	if(!pCmd->GetError())
	{
		if (pCmd->IsAnnotating() && pCmd->UseP4A())
		{
			TheApp()->CallP4A(pCmd->GetTempName(), pCmd->GetTempFilelog(), pCmd->GetFileRev());
		}
		else
		{
			m_ViewFilePath= pCmd->GetTempName();
			m_ViewFileIsText= pCmd->GetFileType() == FST_TEXT;
			RunViewer();
		}
	}
	
	delete pCmd;
	return 0;
}
		
		
void CDepotTreeCtrl::RunViewer()
{
	// First, get the file extension, if any, and find out if
	// its a text file
	CString extension;
	int slash= ReverseFindMBCS(m_ViewFilePath, _T('\\'));
	if(slash != -1)
		extension=m_ViewFilePath.Mid(slash+1);
	else
		extension=m_ViewFilePath;

	int dot= extension.ReverseFind(_T('.'));
	if(dot == -1)
		extension.Empty();
	else
		extension=extension.Mid(dot+1);

	CString viewFilePath = m_ViewFilePath;

	// We have the file, viewFilePath, try to display it
	while(1)
	{
		if(m_Viewer == _T("SHELLEXEC"))
		{
			if (extension.IsEmpty())
				viewFilePath += _T('.');	// So Windows won't get confused!

			CString assocViewer;

			// First, see if there a P4win file association
			if(!extension.IsEmpty())
				assocViewer= GET_P4REGPTR()->GetAssociatedApp(extension);
			
			// If we still havent found a viewer, set viewer to default text app
			// if user wishes to ignore windows associations
			if(assocViewer.IsEmpty() && m_ViewFileIsText && GET_P4REGPTR()->GetIgnoreWinAssoc())
				assocViewer= GET_P4REGPTR()->GetEditApp();
			
			// Let windows take a crack at finding a viewer
			if(assocViewer.IsEmpty() /*&& !extension.IsEmpty()*/ )
			{
				// Quick check for executeable extension, 
				// which will make ShellExec try to run the file
				HINSTANCE hinst=0;
				if( extension.CompareNoCase(_T("com")) != 0 && extension.CompareNoCase(_T("exe")) != 0 &&
					extension.CompareNoCase(_T("bat")) != 0 && extension.CompareNoCase(_T("cmd")) != 0)
				{
					// first try the non-standard M$ IDE's - Have to be done first they're so BAAAD
					if(!extension.IsEmpty())
					{	// give VS .NET (non-standard!) a try.
						hinst= ShellExecute( m_hWnd, _T("Open.VisualStudio.7.1"), viewFilePath, NULL, NULL, SW_SHOWNORMAL);
						if( (int) hinst > 32 ) 
							break;  // successfull VS .NET editor launch
						if( (int) hinst == SE_ERR_NOASSOC)	// give MSDEV (non-standard!) a try
						{
							hinst= ShellExecute( m_hWnd, _T("&Open with MSDEV"), viewFilePath, NULL, NULL, SW_SHOWNORMAL);
							if( (int) hinst > 32 ) 
								break;  // successfull MSDEV editor launch
						}
					}
					if( m_Editing && (!GET_P4REGPTR()->GetUseOpenForEdit() 
								   || !extUsesOpen(extension)))
					{
						hinst= ShellExecute( m_hWnd, _T("edit"), viewFilePath, NULL, NULL, SW_SHOWNORMAL);
						if( (int) hinst > 32 ) 
							break;  // successfull editor launch
					}
					if(!extension.IsEmpty() || !m_Editing)
					{
						hinst= ShellExecute( m_hWnd, _T("open"), viewFilePath, NULL, NULL, SW_SHOWNORMAL);
						if( (int) hinst > 32)
							break;  // successfull viewer launch
					}
				}
				else if (!extension.CompareNoCase(_T("exe")))
				{
					// Use ShellExecute() rather than CreateProcess()
					// because it gives better messages in case of an error
					// Also this appears to be what WindosExplorer uses.
					hinst= ShellExecute( m_hWnd, _T("open"), viewFilePath, NULL, NULL, SW_SHOWNORMAL);
					if( (int) hinst > 32)
						break;	// successfully spawned program
				}
			}

			// Remove any '.' we appended so that Windows wouldn't get confused,
			// because some editors get confused if given a trailing '.'
			if (extension.IsEmpty())
				viewFilePath.TrimRight(_T('.'));

			// If windows doesnt have an associated viewer for a text file, we use the 
			// default text editor
			if (assocViewer.IsEmpty() && (m_ViewFileIsText || !extension.CompareNoCase(_T("bat"))))
				assocViewer= GET_P4REGPTR()->GetEditApp();

			if ( TheApp()->RunViewerApp( assocViewer, viewFilePath ) )
				break;  // successfull viewer launch
		}
		else
		{
			if ( TheApp()->RunViewerApp( m_Viewer, viewFilePath ) )
				break;  // successfull viewer launch
		}

		CString msg;
		msg.FormatMessage(IDS_UNABLE_TO_LAUNCH_VIEWER_s, m_ViewFilePath);
		if(AfxMessageBox(msg, MB_YESNO | MB_ICONEXCLAMATION) == IDNO)
			break;

		// Try to find an alternate viewer
		CViewerDlg dlg;
		if(dlg.DoModal() == IDCANCEL)
			break;

		m_Viewer=dlg.GetViewer();
		if(m_Viewer != _T("SHELLEXEC"))
			GET_P4REGPTR()->AddMRUViewer(m_Viewer);
	}
}

BOOL CDepotTreeCtrl::extUsesOpen(CString extension)
{
	if (extension.IsEmpty())
		return FALSE;

	extension.MakeLower();
	CString exts = CString(GET_P4REGPTR()->GetUseOpenForEditExts()) + _T(",");
	int i;
	if ((i = exts.Find(extension)) != -1)
	{
		if ((exts.GetAt(i + extension.GetLength()) == _T(',')) 
		 && (!i || exts.GetAt(i-1) == _T(',')))
			return TRUE;
	}
	return FALSE;
}

/*
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::IsDepot( HTREEITEM hItem )
{
	HTREEITEM p = TreeView_GetParent( m_hWnd, hItem ) ;
	if ( p == TVI_ROOT  || p == NULL )
		return TRUE;
	else
		return FALSE;
}


/*
	_________________________________________________________________
	
	Double-click will edit the local file if its open for
	edit, and will run quick browse for unopened files
	_________________________________________________________________
*/

void CDepotTreeCtrl::OnLButtonDblClk(UINT nFlags, CPoint point) 
{
	// must clear this flag first!
	m_JustExpanded = FALSE;

	CTreeCtrl::OnLButtonDblClk(nFlags, point);
	if( nFlags & (MK_CONTROL | MK_MBUTTON | MK_RBUTTON | MK_SHIFT ))
		return;

	// Handle any expand messages first
	MSG msg;
	while ( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ) )
	{
		//		get out if app is terminating
		//
		if ( msg.message == WM_QUIT )
			return;
		TranslateMessage(&msg);
		DispatchMessage(&msg);
	}
	// if we generated a node expand in the above loop,
	// then there is nothing more for us to do.
	if (m_JustExpanded)
	{
		m_JustExpanded = FALSE;
		return;
	}

	//		find out what was hit
	//
	TV_HITTESTINFO ht;
	ht.pt=point;
	HTREEITEM currentItem=TreeView_HitTest( m_hWnd, &ht	);

	//		if clicked on non-tree part of pane, skip it
	//
	if( currentItem != NULL && (ht.flags & TVHT_ONITEM) )
		OnLButtonDblClk(currentItem);
}

void CDepotTreeCtrl::OnLButtonDblClk(HTREEITEM currentItem)
{
	//		clicked on a depot. expand it, but only for the new 98.2
	//
	if ( NEW_DEPOT_LISTING )
	{
		if ( IsDepot( currentItem ) )
		{
			ExpandTree( currentItem );
			return;
		}	
	}

	//		empty directory? go no further
	//
	CString empty = GetItemText ( currentItem );
	if ( empty.Find ( g_TrulyEmptyDir ) != -1 )
	{
		return;
	}

	//		make a new selection new if reqd
	//
	if(!IsSelected(currentItem))
	{
		UnselectAll();
		if(currentItem != NULL)
			SetSelectState(currentItem, TRUE);
	}

	if(GetSelectedCount() == 1 && GetSelectedItem(0) != m_Root && ITEM_IS_FILE(GetSelectedItem(0)) )
	{
		switch (GET_P4REGPTR()->GetDoubleClickOption())
		{
		case 0:	// view unopened, edit opened
		default:
			// If we already have it open for edit,
			// there should be no trouble editing it
			if(IsOpened(GetSelectedItem(0)) && !AnyOpenedForInteg())
				OnFileQuickedit();
			else if(SERVER_BUSY())
				MessageBeep(MB_OK);
			else
				OnFileQuickbrowse();
			return;

		case 1:	// open
			PostMessage(WM_COMMAND, ID_FILE_OPENEDIT, 0);
			break;

		case 2:	// open and edit
			PostMessage(WM_COMMAND, ID_FILE_QUICKEDIT, 0);
			break;

		case 3:	// view head revision
		{
			CString itemStr= GetItemPath(currentItem);
			DWORD index=GetLParam(currentItem);
			CP4FileStats *fs=m_FSColl.GetStats(index);
			SendMessage(WM_VIEWHEAD, (WPARAM)(itemStr.GetBuffer(32)), (LPARAM)fs);
			itemStr.ReleaseBuffer();
			break;
		}

		case 4:	// sync to head revision
			PostMessage(WM_COMMAND, ID_FILE_GET, 0);
			break;

		case 5:	// diff versus head revision
			PostMessage(WM_COMMAND, ID_FILE_DIFFHEAD, 0);
			break;

		case 6:	// display Properties dialogbox
			PostMessage(WM_COMMAND, ID_FILE_PROPERTIES, 0);
			break;

		case 7:	// display Revision History dialogbox
			PostMessage(WM_COMMAND, ID_FILE_REVISIONHISTORY, 0);
			break;
		}
	}
}


////////////////////////
// Handlers for edit requests from the changes window

LRESULT CDepotTreeCtrl::OnEditFileTxt(WPARAM wParam, LPARAM lParam)
{
	m_ViewFileIsText= TRUE;
	return OnEditFile(wParam, lParam);
}

LRESULT CDepotTreeCtrl::OnEditFileBin(WPARAM wParam, LPARAM lParam)
{
	m_ViewFileIsText= FALSE;
	return OnEditFile(wParam, lParam);
}

LRESULT CDepotTreeCtrl::OnEditFile(WPARAM wParam, LPARAM lParam)
{
	CString *clientPath= (CString *) wParam;
	ASSERT(!clientPath->IsEmpty());

	m_ViewItem=NULL;
	m_ViewFilePath= *clientPath;
	m_Editing=TRUE;


	if(lParam==EDIT_FINDNEWVIEWER)
		FindAndRunNewViewer();
	else if(lParam==EDIT_CHOOSEVIEWER)
		ChooseAndRunViewer();
	else if(lParam==EDIT_ASSOCVIEWER)
		RunAssocViewer();
	else if(lParam >= 0 && lParam < MAX_MRU_VIEWERS)
		RunMRUViewer(lParam);
	else
	{
		ASSERT(0);
		return 0;
	}
	
	return 0;
}


////////////////////////
// Handlers for browse requests from the changes window

LRESULT CDepotTreeCtrl::OnBrowseFileTxt(WPARAM wParam, LPARAM lParam)
{
	m_ViewFileIsText= TRUE;
	return OnBrowseFile(wParam, lParam);
}

LRESULT CDepotTreeCtrl::OnBrowseFileBin(WPARAM wParam, LPARAM lParam)
{
	m_ViewFileIsText= FALSE;
	return OnBrowseFile(wParam, lParam);
}

LRESULT CDepotTreeCtrl::OnBrowseFile(WPARAM wParam, LPARAM lParam)
{
	CString *clientPath= (CString *) wParam;
	ASSERT(!clientPath->IsEmpty());

	m_ViewItem=NULL;
	m_ViewFilePath= *clientPath;
	m_Editing=FALSE;


	if(lParam==EDIT_FINDNEWVIEWER)
		FindAndRunNewViewer();
	else if(lParam==EDIT_CHOOSEVIEWER)
		ChooseAndRunViewer();
	else if(lParam==EDIT_ASSOCVIEWER)
		RunAssocViewer();
	else if(lParam >= 0 && lParam < MAX_MRU_VIEWERS)
		RunMRUViewer(lParam);
	else
	{
		ASSERT(0);
		return 0;
	}
	
	return 0;
}


/*
	_________________________________________________________________

	Message handlers to allow another window to request info
	about our selected files set
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnGetSelectedCount(WPARAM wParam, LPARAM lParam)
{
	return GetSelectedCount();
}
	

/*
	_________________________________________________________________
*/

LRESULT CDepotTreeCtrl::OnGetSelectedList(WPARAM wParam, LPARAM lParam)
{
	CStringList *list= (CStringList *) wParam;
	ASSERT_KINDOF(CStringList, list);

	AssembleStringList( list, FALSE, lParam ? TRUE : FALSE );

	return (LRESULT) list;
}

//////////////////////
// Handle the Pending & Submitted Changelist Set/Clear Filter menu command
void CDepotTreeCtrl::OnUpdateFilterSetview(CCmdUI* pCmdUI) 
{
	int msg;
	BOOL b;
	HWND hwnd = MainFrame()->GetRightHandWnd();
	if (hwnd == ::GetParent(m_changeWnd))
	{
		msg = IDS_FILTER_PCO_SETVIEW;
		b = TRUE;
	}
	else if (hwnd == ::GetParent(m_oldChgWnd))
	{
		msg = IDS_FILTER_SETVIEW;
		b = TRUE;
	}
	else
	{
		msg = IDS_FILTER_SETVIEW;
		b = FALSE;
	}
	pCmdUI->SetText(LoadStringResource(msg));
	pCmdUI->Enable( MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() && b) );
}

void CDepotTreeCtrl::OnFilterSetview() 
{
	HWND hwnd = MainFrame()->GetRightHandWnd();
	if (hwnd == ::GetParent(m_changeWnd))
		hwnd = m_changeWnd;
	else if (hwnd == ::GetParent(m_oldChgWnd))
		hwnd = m_oldChgWnd;
	else
		return;

	ASSERT(IsWindow(hwnd));
	::SendMessage(hwnd, WM_COMMAND, MAKEWPARAM(ID_FILTER_SETVIEW, 0), 0);
}

void CDepotTreeCtrl::OnUpdateFilterClearview(CCmdUI* pCmdUI) 
{
	BOOL b = TRUE;
	HWND hwnd = MainFrame()->GetRightHandWnd();
	if (hwnd == ::GetParent(m_changeWnd))
		hwnd = m_changeWnd;
	else if (hwnd == ::GetParent(m_oldChgWnd))
		hwnd = m_oldChgWnd;
	else
	{
		hwnd = m_oldChgWnd;
		b = FALSE;
	}

	ASSERT(IsWindow(hwnd));
	::SendMessage(hwnd, WM_SUBCHGOUFC, 0, (LPARAM)pCmdUI);
	if (!b)
		pCmdUI->Enable( MainFrame()->SetMenuIcon(pCmdUI, FALSE) );
}

void CDepotTreeCtrl::OnFilterClearview() 
{
	HWND hwnd = MainFrame()->GetRightHandWnd();
	if (hwnd == ::GetParent(m_changeWnd))
		hwnd = m_changeWnd;
	else if (hwnd == ::GetParent(m_oldChgWnd))
		hwnd = m_oldChgWnd;
	else
		return;

	ASSERT(IsWindow(hwnd));
	::SendMessage(hwnd, WM_COMMAND, MAKEWPARAM(ID_FILTER_CLEARVIEW, 0), 0);
}

//////////////////////
// Handle the Label-Set Filter menu command
//
void CDepotTreeCtrl::OnUpdateLabelFilterSetview(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
		&& (GET_SERVERLEVEL() >= 11) && GetSelectedCount()));
}

void CDepotTreeCtrl::OnLabelFilterSetview() 
{
	ASSERT(IsWindow(m_labelWnd));
	::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_SETVIEW, 0), 0);
}

void CDepotTreeCtrl::OnUpdateLabelFilterSetviewRev(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() 
		&& (GET_SERVERLEVEL() >= 11) && GetSelectedCount()));
}

void CDepotTreeCtrl::OnLabelFilterSetviewRev() 
{
	ASSERT(IsWindow(m_labelWnd));
	::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_SETVIEWREV, 0), 0);
}

void CDepotTreeCtrl::OnUpdateLabelFilterClearview(CCmdUI* pCmdUI) 
{
	ASSERT(IsWindow(m_labelWnd));
	::SendMessage(m_labelWnd, WM_LABELOUFC, 0, (LPARAM)pCmdUI);
}

void CDepotTreeCtrl::OnLabelFilterClearview() 
{
	ASSERT(IsWindow(m_labelWnd));
	::SendMessage(m_labelWnd, WM_COMMAND, MAKEWPARAM(ID_LABELFILTER_CLEARVIEW, 0), 0);
}

//////////////////////
// More drag and drop challenged menu items
//
void CDepotTreeCtrl::OnUpdateAddToClientView(CCmdUI* pCmdUI) 
{
	CString txt;
    txt.FormatMessage(IDS_ADD_FILES_TO_CLIENT_s_VIEW, GET_P4REGPTR()->GetP4Client());
	pCmdUI->SetText ( txt );
	pCmdUI->Enable( !SERVER_BUSY() && GetSelectedCount()
		&& GET_P4REGPTR()->ShowEntireDepot() <= SDF_DEPOT );
}

void CDepotTreeCtrl::OnAddToClientView() 
{
	::SendMessage(m_clientWnd, WM_COMMAND, MAKEWPARAM(ID_ADD_TOVIEW, 0), 0);
}

void CDepotTreeCtrl::OnUpdateAddReviews(CCmdUI* pCmdUI) 
{
	CString txt;
    txt.FormatMessage(IDS_ADD_FILES_TO_USER_s_REVIEWS, GET_P4REGPTR()->GetP4User());
	pCmdUI->SetText ( txt );
	pCmdUI->Enable( !SERVER_BUSY() && GetSelectedCount() 
		&& GET_P4REGPTR()->ShowEntireDepot() <= SDF_DEPOT );
}

void CDepotTreeCtrl::OnAddReviews() 
{
	::SendMessage(m_userWnd, WM_COMMAND, MAKEWPARAM(ID_ADD_REVIEWS, 0), 0);
}

void CDepotTreeCtrl::OnUpdateFilterJobview(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable( !SERVER_BUSY() && GetSelectedCount() );
}

void CDepotTreeCtrl::OnFilterJobview() 
{
	ASSERT(IsWindow(m_jobWnd));

	::SendMessage(m_jobWnd, WM_COMMAND, MAKEWPARAM(ID_JOB_SETFILEFILTER, 0), 0);
}

void CDepotTreeCtrl::OnFilterJobviewInteg() 
{
	ASSERT(IsWindow(m_jobWnd));

	::SendMessage(m_jobWnd, WM_COMMAND, MAKEWPARAM(ID_JOB_SETFILEFILTERINTEG, 0), 0);
}


/////////////////////
//
//

void CDepotTreeCtrl::OnFileSubmit()
{
	HTREEITEM item;
	if(ITEM_IS_FILE(item = GetSelectedItem(0)) && !AnyHaveChildren())
	{
		int index=GetLParam(item);
		CP4FileStats *fs=m_FSColl.GetStats(index);
		if (fs->GetMyOpenAction())
		{
			long chgnbr = MainFrame()->PositionChgs( fs->GetFullDepotPath(), fs->GetMyOpenAction() );
			while (SERVER_BUSY())
				Sleep(200);
			if (chgnbr > 0)
				::SendMessage(m_changeWnd, WM_COMMAND, ID_EDIT_SELECT_ALL, 0);
			::SendMessage(m_changeWnd, WM_COMMAND, ID_CHANGE_SUBMIT, 0);
		}
	}
}

void CDepotTreeCtrl::OnUpdateFindFileUnderFolder(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable( !SERVER_BUSY() && GetSelectedCount()==1 && AnyHaveChildren());
}

void CDepotTreeCtrl::OnFindFileUnderFolder() 
{
	m_P4Files_FileSpec = GetItemPath(GetSelectedItem(0)) + "...";
	m_P4Files_List.RemoveAll();
	m_P4Files_Deselect = TRUE;
	OnPositionDepot();
}

void CDepotTreeCtrl::OnFileRename() 
{
	OnIntegrate(FALSE, NULL, FALSE, 0, NULL, TRUE);
}

void CDepotTreeCtrl::OnFileIntegspec() 
{
	OnIntegrate(FALSE, NULL, FALSE, 0, NULL);
}

void CDepotTreeCtrl::OnFileIntegrate() 
{
	OnIntegrate(TRUE, NULL, FALSE, 0, NULL);
}

// Process a request from the Branch View, to integrate an entire branch view
LRESULT CDepotTreeCtrl::OnBranchIntegrate(WPARAM wParam, LPARAM lParam)
{
	LPCTSTR branchName= (LPCTSTR) wParam;
	
	OnIntegrate(TRUE, branchName, TRUE, 0, NULL);
	return 0;
}

// Process a request from the Sumitted Changelist View to integrate a change
LRESULT CDepotTreeCtrl::OnChangelistIntegrate(WPARAM wParam, LPARAM lParam)
{
	INTEGCHG *integChg= (INTEGCHG *) lParam;
	
	OnIntegrate(integChg->useBranch, NULL, FALSE, integChg->changeList, integChg->filelist,
													FALSE, integChg->useBranch);
	return 0;
}

void CDepotTreeCtrl::OnIntegrate(BOOL useBranch, LPCTSTR branchName, BOOL entireView, 
								 int changeList, CStringList *filelist, 
								 BOOL rename/*=FALSE*/, BOOL isChgListInteg/*=FALSE*/) 
{
	if (MainFrame()->IsModlessUp() || MainFrame()->IsPendChgEditInProgress())
		return;

	int branchFlag = useBranch ? ((branchName != NULL && lstrlen(branchName)) 
						? INTEG_USING_BRANCH : INTEG_USING_BRANCH_SPEC) : INTEG_USING_FILE_SPEC;

	if (useBranch && (branchName == NULL || !lstrlen(branchName)))
	{
		m_Save_branchFlag = branchFlag;
		m_Save_useBranch = useBranch;
		m_Save_entireView = entireView;
		m_Save_changeList = changeList;
		m_Save_rename = rename;
		m_Save_isChgListInteg = isChgListInteg;
		// Save the file list
		m_StringListI1.RemoveAll();
		if (filelist)
		{
			for( POSITION pos = filelist->GetHeadPosition(); pos != NULL; )
				m_StringListI1.AddTail(filelist->GetNext( pos ));
		}
		// Display the Branches Browse dialog.
		// If the branches window is empty, a refresh will be initiated.
		::SendMessage(MainFrame()->BranchWnd(), WM_INTEGFETCHOBJECTLIST, 
						(WPARAM)(this->m_hWnd), WM_BROWSECALLBACK1);
	}
	else
		OnIntegrate0(branchFlag, useBranch, branchName, entireView, changeList, filelist, rename);
}

LRESULT CDepotTreeCtrl::OnIntegBranchBrowseCallBack(WPARAM wParam, LPARAM lParam)
{
	CString *str = (CString *)lParam;
	::SendMessage(MainFrame()->BranchWnd(), WM_SELECTTHIS, 0, (LPARAM)str);
	OnIntegrate0(m_Save_branchFlag, m_Save_useBranch, *str, m_Save_entireView, 
				 m_Save_changeList, &m_StringListI1, m_Save_rename, m_Save_isChgListInteg);
	return 0;
}

void CDepotTreeCtrl::OnIntegrate0(int branchFlag, BOOL useBranch, LPCTSTR branchName, 
									BOOL entireView, int changeList, CStringList *filelist, 
									BOOL rename/*=FALSE*/, BOOL isChgListInteg/*=FALSE*/) 
{
	ASSERT( !entireView || useBranch);

	m_IntegWizard = new CIntegFileSpecPage(this);

	if (!m_IntegWizard)
	{
		ASSERT(0);
		AfxMessageBox(IDS_COULD_NOT_CREATE_INTEGRATION_WIZARD, MB_ICONSTOP);
		return;
	}
	MainFrame()->SetModelessUp(TRUE);

	m_IntegWizard->SetIsChgListInteg( isChgListInteg );
	m_IntegWizard->SetIsRename( rename );
	m_IntegWizard->SetIsBranch( useBranch );
	m_IntegWizard->SetBranchFlag( branchFlag );
	m_IntegWizard->SetDepotWnd( m_hWnd );

	///////
	// If this is a branch integ, pass the branch name to the Wizard
	if(useBranch)
		m_IntegWizard->SetBranchName(branchName);

	///////
	// Get a list of my changes, so we can give the dialog a list of possible
	// target change numbers

	m_Changes.RemoveAll();
	::SendMessage(m_changeWnd, WM_GETMYCHANGESLIST, (WPARAM) &m_Changes, 0);
	m_IntegWizard->SetChangesList(&m_Changes);
	
	///////
	// Build the list of my source files or directories
	// target change numbers

	if(changeList)
	{
		if (changeList > 0)
			m_IntegWizard->SetChangeNbr(changeList);
		m_StringList.RemoveAll();
		for( POSITION pos = filelist->GetHeadPosition(); pos != NULL; )
			m_StringList.AddTail(filelist->GetNext( pos ));
	}
	else if(entireView)
	{
		m_StringList.RemoveAll();
		m_StringList.AddHead(_T("//..."));
	}
	else
		AssembleStringList( &m_StringList, TRUE );

	// Make a duplicate of the filelist in case they press Back.
	m_StringListSv.RemoveAll();
	if (!m_StringList.IsEmpty())
	{
		for( POSITION pos = m_StringList.GetHeadPosition(); pos != NULL; )
			m_StringListSv.AddTail(m_StringList.GetNext( pos ));
	}

	if (!m_IntegWizard->SetSpecList(&m_StringList))
	{
		m_IntegWizard->DestroyWindow();	// some error! clean up
		delete m_IntegWizard;
		MainFrame()->SetModelessUp(FALSE);
		m_StringListSv.RemoveAll();
		return;
	}

	// Fire up the dialog to get user's preferences
	if (!m_IntegWizard->Create(IDD_PAGE_INTEGFILESPECS, this))
	{
		m_IntegWizard->DestroyWindow();	// some error! clean up
		delete m_IntegWizard;
		MainFrame()->SetModelessUp(FALSE);
		m_StringListSv.RemoveAll();
	}
}

LRESULT CDepotTreeCtrl::OnIntegrate1(WPARAM wParam, LPARAM lParam)
{
	if ((int)wParam == ID_WIZFINISH)
	{
		// Copy the stringlists for using in the 3 IntegratePreview-Sync-Integrate steps (i-s-i)
		m_StringListI1.RemoveAll();	// will hold source list
		m_StringListI2.RemoveAll();	// will hold target list
		m_StringListI3.RemoveAll();	// will be used to save source list for 3rd step of i-s-i
		m_StringListI4.RemoveAll();	// will be used to save target list for 3rd step of i-s-i

		CStringList *list= m_IntegWizard->GetSourceList();
		POSITION pos;
		for( pos= list->GetHeadPosition(); pos != NULL; )
			m_StringListI1.AddHead( list->GetNext(pos) );

		list= m_IntegWizard->GetTargetList();
		for( pos= list->GetHeadPosition(); pos != NULL; )
			m_StringListI2.AddHead( list->GetNext(pos) );

		// save the parameters for the integrate(s)
		m_integCont3 = CIntegContinue(  m_IntegWizard->GetReference(), 
										m_IntegWizard->GetRevRange(),
										m_IntegWizard->GetCommonPath(),
										m_IntegWizard->IsBranch(), m_IntegWizard->IsReverse(), 
										m_IntegWizard->IsNoCopy(), m_IntegWizard->IsForced(), 
										m_IntegWizard->IsForcedDirect(), 
										m_IntegWizard->IsRename(), m_IntegWizard->IsPreview(),
										m_IntegWizard->GetChangeNum(), 
										m_IntegWizard->IsPermitDelReadd(),
										m_IntegWizard->DelReaddType(),
										m_IntegWizard->IsBaselessMerge(),
										m_IntegWizard->IsIndirectMerge(),
										m_IntegWizard->IsPropagateTypes(),
										m_IntegWizard->IsBaseViaDelReadd(),
										m_IntegWizard->GetBranchFlag(),
										m_IntegWizard->GetBiDirFlag(),
										m_IntegWizard->GetNewChangeNbr() );
		m_IntegWizard->ClrNewChangeNbr();
		// Do they want us to sync the targets (and this is for real - not a preview)?
		if (GET_SERVERLEVEL() >= 13 && !m_IntegWizard->IsRename())	// Do we have support for -h?
		{
			// If we have support for -h, let the server do all the dirty work
			OnIntegrate3(FALSE, !m_IntegWizard->IsAutoSync());
		}
		else if (m_IntegWizard->IsAutoSync() && !m_IntegWizard->IsPreview())
		{
			list= m_IntegWizard->GetSourceList();	// save the source for use during the 2nd integ pass
			for( pos= list->GetHeadPosition(); pos != NULL; )
				m_StringListI3.AddHead( list->GetNext(pos) );

			list= m_IntegWizard->GetTargetList();	// save the target for use during the 2nd integ pass
			for( pos= list->GetHeadPosition(); pos != NULL; )
				m_StringListI4.AddHead( list->GetNext(pos) );

			m_integCont3.m_isPreview = TRUE;	// fire up an integ preview to get the tragets
			OnIntegrate3(TRUE);					// TRUE => run a sync of the targets afterwards
			m_integCont3.m_isPreview = FALSE;	// now set the preview flag back to OFF
		}
		else	// they don't want to sync or this is only a preview
			OnIntegrate3();

		m_IntegWizard->SendMessage(WM_ENABLEDISABLE, 0, FALSE);
		if (m_IntegWizard->IsPreview())
			return 0;	// if it's a preview, we're done.
		
		// Give the user some feedback that we are doing the work
		SET_BUSYCURSOR();
	}
	else
	{
		EnterCriticalSection(&MainFrame()->CriticalSection); 
		if (m_IntegWizard)	// ID_WIZFINISH leaves the dialog up; it's closed later if no error
		{
			m_IntegWizard->DestroyWindow();	 // deletes m_IntegWizard
			m_IntegWizard = 0;
			MainFrame()->SetModelessUp(FALSE);
			if ((int)wParam != ID_WIZBACK)	// Go back, but remember selection
				m_StringListSv.RemoveAll();
		}
		LeaveCriticalSection(&MainFrame()->CriticalSection);
	}
	m_Changes.RemoveAll();

	if ((int)wParam == ID_WIZBACK)	// Go back & let user select branch again
	{
		OnIntegrate(m_Save_useBranch, NULL, m_Save_entireView, 
					m_Save_changeList ? m_Save_changeList : -1, 
					&m_StringListSv, m_Save_rename, m_Save_isChgListInteg);
	}
	else if ((int)wParam != ID_WIZFINISH)
		m_StringListSv.RemoveAll();	// If Finished clicked, don't removed saved list in case integ fails
	return 0;
}

void CDepotTreeCtrl::OnIntegrate2(CStringList *list) // arrive here after the integ preview finishes
{												 //		now do a sync on the targets found
	static	CStringList	svList;		// save the targets from preview here for use in the sync
	POSITION pos;

	svList.RemoveAll();
	m_StringListI1.RemoveAll();		// these are now stale - reload them from lists 3 & 4
	m_StringListI2.RemoveAll();		//	so they can be used in OnIntegrate3() after sync finishes
	if ( list->GetCount() > 0 )		// Did the integ preview actually find any targets?
	{
		for (pos=list->GetHeadPosition(); pos != NULL; )	// copy the integ targets for syncing
			svList.AddHead(list->GetNext(pos));
		for (pos=m_StringListI3.GetHeadPosition(); pos != NULL; )	// reload list 1
			m_StringListI1.AddHead(m_StringListI3.GetNext(pos));
		for (pos=m_StringListI4.GetHeadPosition(); pos != NULL; )	// reload list 2
			m_StringListI2.AddHead(m_StringListI4.GetNext(pos));

		CCmd_Get *pCmd= new CCmd_Get;		// run a sync on the files found by the integ preview
		pCmd->Init( m_hWnd, RUN_ASYNC);
		pCmd->SetRunIntegAfterSync( TRUE );
		if( pCmd->Run( &svList, FALSE ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
		else
			delete pCmd;
	}
	else // there were no files in the target list, so there is nothing to do: no sync or integ
	{
		EnterCriticalSection(&MainFrame()->CriticalSection); 
		if (m_IntegWizard)
			m_IntegWizard->SendMessage(WM_ENABLEDISABLE, 0, TRUE);
		LeaveCriticalSection(&MainFrame()->CriticalSection);
	}

	m_StringListI3.RemoveAll();		// these are now no longer needed - free the resources
	m_StringListI4.RemoveAll();
}

BOOL CDepotTreeCtrl::OnIntegrate3(BOOL bRunSyncAftPreview, BOOL bDontSync)
{
	CString txt = LoadStringResource(( m_integCont3.m_isRename ) ? IDS_FILE_RENAME : IDS_FILE_INTEGRATE);

	// Actually run the integrate.  Hang onto the key for possible use
	// in OnP4Integ()

	CCmd_Integrate2 *pCmd= new CCmd_Integrate2;
	pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
	pCmd->SetRunSyncAfterPreview( bRunSyncAftPreview );
	if( pCmd->Run( &m_StringListI1, &m_StringListI2,
					m_integCont3.m_reference, m_integCont3.m_revRange,  m_integCont3.m_commonPath,
					m_integCont3.m_isBranch,  m_integCont3.m_isReverse, m_integCont3.m_isNoCopy, 
					m_integCont3.m_isForced,  m_integCont3.m_isForcedDirect,  
					m_integCont3.m_isRename,  m_integCont3.m_isPreview, 
					m_integCont3.m_changeNum, m_integCont3.m_isPermitDelReadd, 
					m_integCont3.m_DelReaddType,      m_integCont3.m_isBaselessMerge,
					m_integCont3.m_isIndirectMerge,   m_integCont3.m_isPropagateTypes, 
					m_integCont3.m_isBaseViaDelReadd, m_integCont3.m_BranchFlag, 
					bDontSync,                        m_integCont3.m_BiDir,
					m_integCont3.m_NewChangeNbr ) )
	{
		MainFrame()->UpdateStatus(txt);
		return TRUE;
	}
	delete pCmd;
	return FALSE;
}



/*
	_________________________________________________________________

	user clicked on the little plus sign. start our p4 dirs/fstat 
	adventure if expanding; change folder icon always.
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::ExpandTree( const HTREEITEM hItem )
{
    XTRACE(_T("ExpandTree() already expanded=%d\n"), (GetLParam( hItem ) == FOLDER_ALREADY_EXPANDED));
    if( APP_HALTED() )
        return FALSE;

	// If server is busy and we haven't already expanded this node, ignore the click
    if( SERVER_BUSY() && (GetLParam( hItem ) != FOLDER_ALREADY_EXPANDED) )
	{
		Sleep(0);
		SET_BUSYCURSOR();
		// wait a bit in 1/10 sec intervals to see if the prev request finishes
		int t=GET_P4REGPTR()->BusyWaitTime();
		do
		{
			Sleep(50);
			t -= 50;
		} while (SERVER_BUSY() && t > 0);
		if( SERVER_BUSY() )
			return FALSE;
	}

	m_JustExpanded = TRUE;

	// Set the Open Folder image on Folders
	TVITEM tvitem;
	memset(&tvitem, '\0', sizeof(tvitem));
	tvitem.mask = TVIF_IMAGE|TVIF_SELECTEDIMAGE;
	tvitem.hItem = hItem;
	if (TreeView_GetItem(m_hWnd, &tvitem))
	{
		if (tvitem.iImage == CP4ViewImageList::VI_FOLDER
		 || tvitem.iSelectedImage == CP4ViewImageList::VI_FOLDER)
		{
			tvitem.iImage = CP4ViewImageList::VI_OPENFOLDER;
			tvitem.iSelectedImage = CP4ViewImageList::VI_OPENFOLDER;
			TreeView_SetItem(m_hWnd, &tvitem);
		}
	}

	// if user wants to automatically re-expand depot to previous selection
	// upon reconnection, then we have to make sure to do the selection
	if (GET_P4REGPTR()->GetExpandFlag() == 1)
	{
		UnselectAll();
		SetSelectState( hItem, TRUE );
	}

	//		get out if p4 dirs was already called for this subdir
	//		or if the 98.2 server api is not available
	//
	if ( GetLParam( hItem ) == FOLDER_ALREADY_EXPANDED ||
		 !NEW_DEPOT_LISTING || m_RunningUpdate)
	{
		ApplySelectAtts(GetSelectAtts());
		return TRUE;
	}

	// Identify the type of update.  We set full update so that all files
	// under the node, rather than recent ones, will be retrieved
	m_UpdateType = UPDATE_EXPAND;

	//		set the children count to zero - if any subdirs
	//      or files are found, the count will wind up non-zero
	//      after running dirs and fstat
	SetChildCount(hItem, 0);
	
	//		set the member variables we'll be using later:
	//		the htree item for expanding and inserting items
	//		and the string for when we call p4 fstat
	//
	m_LastPathItem = hItem;
	m_LastPath = GetItemPath( m_LastPathItem ) ;
	XTRACE(_T("ExpandTree got m_LastPath= %s\n"), m_LastPath);

	//		call p4 dirs on the path, and don't let it be called again.
	//
	if( RunDirStat( m_LastPath + _T("*") ) )
		SetLParam( m_LastPathItem, FOLDER_ALREADY_EXPANDED );
	else
		SetChildCount( m_LastPathItem, 1 );

	return TRUE;
}

BOOL CDepotTreeCtrl::CollapseTree( const HTREEITEM hItem )
{
	// Set the Closed Folder image on Folders
	TVITEM tvitem;
	memset(&tvitem, '\0', sizeof(tvitem));
	tvitem.mask = TVIF_IMAGE|TVIF_SELECTEDIMAGE;
	tvitem.hItem = hItem;
	if (TreeView_GetItem(m_hWnd, &tvitem))
	{
		if (tvitem.iImage == CP4ViewImageList::VI_OPENFOLDER
		 || tvitem.iSelectedImage == CP4ViewImageList::VI_OPENFOLDER)
		{
			tvitem.iImage = CP4ViewImageList::VI_FOLDER;
			tvitem.iSelectedImage = CP4ViewImageList::VI_FOLDER;
			TreeView_SetItem(m_hWnd, &tvitem);
		}
	}
	return TRUE;
}


/*
	_________________________________________________________________

	call 'p4 dirs' and get a stringarray of the directories under the 
	expanding leaf.
	_________________________________________________________________
*/

BOOL CDepotTreeCtrl::RunDirStat ( const CString &path )
{
	XTRACE(_T("RunDirStat() for path %s\n"), path);
	ASSERT( m_UpdateType == UPDATE_EXPAND );
	
	int	i;
	CStringList *pList= new CStringList;
	CString str = path;
    CString emptyMarker = g_TrulyEmptyDir + _T("/*");
	if ((i = str.Find( emptyMarker )) > -1)
	{
		str = str.Left(i);
		str.Insert(i, _T("/*"));
	}
	pList->AddHead( str );

	CCmd_DirStat *pCmd= new CCmd_DirStat;
	
    // Store our context, so when this command returns we know where we are
    pCmd->SetItemRef(m_LastPathItem);
    pCmd->SetTextRef(m_LastPath);
        
	pCmd->Init( m_hWnd, RUN_ASYNC );
	pCmd->SetAlternateReplyMsg( WM_P4EXPANDSUBDIR );

	BOOL ret = pCmd->Run( pList, GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT);
	if( ret )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_UPDATING) );
	else
	{
		MainFrame()->ClearStatus();
		delete pList;
		delete pCmd;
	}
	return ret;
}


/*
	_________________________________________________________________

	get this out of the main line.
	_________________________________________________________________
*/


void CDepotTreeCtrl::AssembleStringList(CStringList *list, 
										BOOL bDepotSyntax4Files/*=FALSE*/,
										BOOL bNoRemoteFiles/*=FALSE*/,
										BOOL bWildOK/*=FALSE*/)
{
	list->RemoveAll();

	HTREEITEM cItem;
	CString itemStr;

	for( int i = GetSelectedCount() - 1; i >= 0; i-- )
	{
		cItem = GetSelectedItem( i );
		itemStr = GetItemPath( cItem );
		if (itemStr.GetAt(1) == _T(':') && !bWildOK)
		{
			if (itemStr.FindOneOf(_T("@#%*")) != -1)
			{
				StrBuf b;
				StrBuf f;
				f << CharFromCString(itemStr);
				StrPtr *p = &f;
				StrOps::WildToStr(*p, b);
				itemStr = CharToCString(b.Value());
			}
		}

		if (bNoRemoteFiles && IsInRemoteDepot(&itemStr))
			continue;

		//		if it's a directory, append the wildcard string
		//
		if( HasChildren( cItem ) ) 
			itemStr += _T("...");
		else if (ITEM_IS_FILE(cItem) && !bWildOK)
		{
			CP4FileStats *fs= m_FSColl.GetStats(GetLParam(cItem));
			CString dPath = fs->GetFullDepotPath();
			if (bDepotSyntax4Files || dPath.Find(_T('%')) != -1)
				itemStr = fs->GetFullDepotPath();
		}
	
		list->AddHead( itemStr );
	}
}

BOOL CDepotTreeCtrl::GetSelectedFiles( CStringList *list )
{
	list->RemoveAll();

	HTREEITEM cItem;

	for( int i = GetSelectedCount() - 1; i >= 0; i-- )
	{
		cItem = GetSelectedItem( i );

		if( ITEM_IS_FILE(cItem) )
		{
			CP4FileStats *fs= m_FSColl.GetStats(GetLParam(cItem));
			if( fs->InClientView() )
				list->AddHead( fs->GetFullClientPath() );
			else
				return FALSE;
		}
	}
	return TRUE;
}

BOOL CDepotTreeCtrl::GetSelectedFStats( CObList *list )
{
	list->RemoveAll();

	HTREEITEM cItem;

	for( int i = GetSelectedCount() - 1; i >= 0; i-- )
	{
		cItem = GetSelectedItem( i );

		if( ITEM_IS_FILE(cItem) )
		{
			CP4FileStats *fs= m_FSColl.GetStats(GetLParam(cItem));
			if( fs->InClientView() )
				list->AddHead( fs );
			else
				return FALSE;
		}
	}
	return TRUE;
}

void CDepotTreeCtrl::OnSetFlyoverMessage(HTREEITEM currentItem)
{
    if( ! GET_P4REGPTR()->ShowClientPath() || !currentItem) 
        return;

    LPARAM lParam= 0;
    lParam= GetLParam(currentItem);
   
    // Bail out if its the depot or a folder
    if( lParam < 0 || lParam >= m_ItemCount )
    {
        MainFrame()->SetMessageText(LoadStringResource(IDS_FOR_HELP_PRESS_F1));
        CMultiSelTreeCtrl::SetItemFocus(NULL);
    }
    else
    {
        if( GET_P4REGPTR()->ShowDepotPathHiLite() )
            CMultiSelTreeCtrl::SetItemFocus(currentItem);
        else
            CMultiSelTreeCtrl::SetItemFocus(NULL);

        CP4FileStats *fs= m_FSColl.GetStats((int) lParam);
    
        if( fs->InClientView() ) // status bar will show local path plus file type explanation
        {
            CString msg = fs->GetFullClientPath();
			CString itemStr = GetItemText(currentItem);
			CString type;
			int		baseType;
			int		storeType;
			BOOL	typeK = FALSE;
			BOOL	typeW = FALSE;
			BOOL	typeX = FALSE;
			BOOL	typeO = FALSE;
			BOOL	typeM = FALSE;
			BOOL	typeL = FALSE;
			BOOL	typeS = FALSE;
			int		nbrrevs;
			BOOL	unknown = FALSE;

			// convert the GetItemText() string to many flags
			TheApp()->GetFileType(itemStr, baseType, storeType, typeK, typeW, typeX, 
									typeO, typeM, typeL, typeS, nbrrevs, unknown);
			// determine the base file type
			switch(baseType)
			{
			case 0:
				type = _T("  <text");
				break;
			case 1:
				type = _T("  <binary");
				break;
			case 2:
				type = _T("  <symlink");
				break;
			case 3:
				type = _T("  <resource");
				break;
			case 4:
				type = _T("  <apple");
				break;
			case 5:
				type = _T("  <unicode");
				break;
			case 6:
				type = _T("  <utf16");
				break;
			default:
				type = _T("  <unknown");
				break;
			}
			// determine storage type - if it's the default for the base type, do nothing
			switch(storeType)
			{
			case 1:					// +C
				if (baseType != 1)
					type += LoadStringResource(IDS_comma_FULL_COMPRESSED_VERSION_STORED);
				break;
			case 2:					// +D
				if (baseType != 0)
					type += LoadStringResource(IDS_comma_RCS_DELTAS_STORED);
				break;
			case 3:					// +F
				type += LoadStringResource(IDS_comma_FULL_FILE_STORED_PER_REV);
				break;
			default:
				break;
			}
			// Warning: if more of these modifier types are added, we may need to
			// shorten these strings so that all will fit in the status bar window.
			// Already there is a possible overflow if the local path name is long!
			if (typeO)
				type += LoadStringResource(IDS_comma_LIMITED_RCS_KEYWORD_EXPANSION);
			else if (typeK)
				type += LoadStringResource(IDS_comma_RCS_KEYWORD_EXPANSION);
			if (typeW)
				type += LoadStringResource(IDS_comma_ALWAYS_WRITABLE);
			if (typeX)
				type += LoadStringResource(IDS_comma_EXECUTABLE);
			if (typeL)
				type += LoadStringResource(IDS_comma_LOCKED);
			if (typeS)
				type += LoadStringResource(IDS_comma_ONLY_HEAD_REV_STORED);
			if (unknown)
				type += LoadStringResource(IDS_PLUS_UNKNOWN);
			type += ">";
			msg += (type != LoadStringResource(IDS_ONLY_UNKNOWN)) ? type : _T(" ");

			if(fs->GetHaveRev() == 0)
				msg += fs->GetHeadRev() ? LoadStringResource(IDS_DONT_HAVE)
				                        : LoadStringResource(IDS_NOT_IN_DEPOT);
			else 
			{
				if(fs->GetHaveRev() > fs->GetHeadRev() &&
					fs->GetMyOpenAction() == 0 )
					msg += _T("<") + LoadStringResource(IDS_NOT_IN_CLIENT_VIEW) + _T(">");
				else if(fs->GetHaveRev() >= fs->GetHeadRev() || fs->GetMyOpenAction() == F_ADD)  
					msg += LoadStringResource(IDS_HAVE_CURRENT);  // have head rev or doing file recovery
				else
					msg += LoadStringResource(IDS_NOT_LATEST);  // have rev < head
			}


			if (fs->GetMyOpenAction() > 0)
				msg += _T("<") + fs->GetActionStr( fs->GetMyOpenAction() ) + _T(">");

			if( fs->IsUnresolved() )
				msg += LoadStringResource(IDS_UNRESOLVED);
			if(fs->IsMyLock())
				msg += LoadStringResource(IDS_LOCKED);

			if((fs->GetOtherOpens() > 0) || fs->IsOtherLock())
			{
				msg += LoadStringResource(IDS_OTHERUSER);
				if(fs->GetOtherOpens() > 0)
                {
                    CString otherAction;
                    otherAction.FormatMessage(IDS_OPENFOR_s, fs->GetActionStr( fs->GetOtherOpenAction() ));
					msg += otherAction;
					if (*(fs->GetOtherUsers()))
						msg += CString(_T(" by ")) + fs->GetOtherUsers();
                }
				if(fs->IsOtherLock())
					msg += _T(" ") + LoadStringResource(IDS_LOCKED);
			}

			// write the local path and the file type to the status bar
            MainFrame()->SetMessageText(msg);
        }
        else
            MainFrame()->SetMessageText(LoadStringResource(IDS_FILE_NOT_IN_CLIENT_VIEW));
    }
}

	
// Support for quick copy of depot path or client path to the clipboard
//

void CDepotTreeCtrl::OnUpdateEditCopy(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(GetSelectedCount() >= 1);
}

void CDepotTreeCtrl::OnUpdateEditCopyclientpath(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(GetSelectedCount() >= 1 
		&& (m_SlashChar == _T('\\') || AllInView()));
}

void CDepotTreeCtrl::OnUpdateEditSelectAll(CCmdUI* pCmdUI) 
{
	int i;
	BOOL b = (i = GetSelectedCount()) > 0;
	if (i > 1)
	{
		HTREEITEM parent1 = TreeView_GetParent( m_hWnd, GetSelectedItem(0) );
		HTREEITEM parent2 = TreeView_GetParent( m_hWnd, GetSelectedItem(i-1) );
		b = parent1 == parent2;
	}
	pCmdUI->Enable( b );
}

void CDepotTreeCtrl::OnEditCopyclientpath() 
{
	CString txt;
	for(int i=-1; ++i < GetSelectedCount(); )
	{
		HTREEITEM item= GetSelectedItem(i);
		if (m_SlashChar == _T('/'))
		{
			if( ITEM_IS_FILE(item) )
			{
				CP4FileStats *fs= m_FSColl.GetStats(GetLParam(item));
				if( fs->InClientView() )
				{
					if (i)
						txt += _T("\r\n");
					txt += fs->GetFullClientPath();
				}
			}
			else
			{
				txt.Empty();
				break;
			}
		}
		else
		{
			if (i)
				txt += _T("\r\n");
			txt += GetItemPath(item);
		}
	}
	if (txt.IsEmpty())
		MessageBeep(MB_ICONEXCLAMATION);
	else
		CopyTextToClipboard( txt );
}

void CDepotTreeCtrl::OnEditCopy() 
{
	CString txt;
	for(int i=-1; ++i < GetSelectedCount(); )
	{
		HTREEITEM item= GetSelectedItem(i);
		if (m_SlashChar == _T('\\'))
		{
			if( ITEM_IS_FILE(item) )
			{
				CP4FileStats *fs= m_FSColl.GetStats(GetLParam(item));
				if( fs->InClientView() )
				{
					if (i)
						txt += _T("\r\n");
					txt += fs->GetFullDepotPath();
				}
			}
			else
			{
				txt.Empty();
				break;
			}
		}
		else
		{
			if (i)
				txt += _T("\r\n");
			txt += GetItemPath(item);
		}
	}
	if (txt.IsEmpty())
		MessageBeep(MB_ICONEXCLAMATION);
	else
		CopyTextToClipboard( txt );
}

void CDepotTreeCtrl::OnEditSelectAll() 
{
	HTREEITEM item= GetSelectedItem(0);
	if ( item != NULL )
	{
		UnselectAll();
		SetMultiSelect(TRUE);
		HTREEITEM parent= TreeView_GetParent(m_hWnd, item);
		HTREEITEM child= TreeView_GetChild(m_hWnd, parent);
		while( child != NULL )
		{
			SetSelectState( child, TRUE );
			child= TreeView_GetNextSibling(m_hWnd, child);
		}
		SetMultiSelect(FALSE);
		ShowNbrSelected();
	}
}

void CDepotTreeCtrl::SetToolTipColors(CPoint point)
{
	HTREEITEM item;
	TVHITTESTINFO	htInfo;

	htInfo.pt = point;
	item = HitTest( &htInfo );
	if (item && (htInfo.flags & (TVHT_ONITEMLABEL | TVHT_ONITEMICON)))
	{
		if (IsSelected(item))
		{
			if ((m_ToolState != 1) )
			{
				 m_ToolTip = GetToolTips( );
				 if (m_ToolTip && ::IsWindow(m_ToolTip->m_hWnd))
				 {
					 m_ToolTip->SetTipTextColor(GetSysColor(COLOR_HIGHLIGHTTEXT));
					 m_ToolTip->SetTipBkColor(GetSysColor(COLOR_HIGHLIGHT));
					 m_ToolState = 1;
				 }
			}
		}
		else
		{
			if (m_ToolState != 0)
			{
				 m_ToolTip = GetToolTips( );
				 if (m_ToolTip && ::IsWindow(m_ToolTip->m_hWnd))
				 {
					 m_ToolTip->SetTipTextColor(GetSysColor(COLOR_INFOTEXT));
					 m_ToolTip->SetTipBkColor(GetSysColor(COLOR_INFOBK));
					 m_ToolState = 0;
				 }
			}
		}
	}
}

void CDepotTreeCtrl::OnMouseMove(UINT nFlags, CPoint point) 
{
	SetToolTipColors(point);
	CMultiSelTreeCtrl::OnMouseMove(nFlags, point);
}

//	Routine to expand the Depot Treeview down a branch to a given path.
//	The path to expand down is given in the 'path' variable.
//	Since this routine works by calling other routines that work in
//	an async manner, it must be restartable - the 'newPath' variable
//	indicates whether the path is a new one that starts at the top
//	of the Depot tree, or is a continuation and that expansion down
//	the 'path' should continue from the current location in the
//	Depot treeview.
//
void CDepotTreeCtrl::ExpandDepotString(const CString &path, BOOL newPath, 
								   BOOL noExpand /*= FALSE*/, int key /*=0*/, BOOL noErrMsg /*=FALSE*/)
{
	int  i, j;
    TCHAR buf[ LONGPATH + 2 ] = _T(" ");	// will hold tree item names
	CString node2find;		// we will be looking for this node of the treeview
	BOOL	bLast;			// tells whether this is the last element of the path or not
	HTREEITEM svExpandItem;
	TVITEM	tvItem;
    tvItem.pszText = buf+1;         
	tvItem.mask = TVIF_TEXT;

	if (path.GetLength() == 0)	// no path == nothing to do
	{
		m_ExpandDepotContinue = FALSE;
		m_ExpandPath.Empty();
		return;
	}
	// If there are trailing pathnames, 
	//	trim 'em off cuz we won't be able to find 'em
	// also save the path in a class variable 
	//	so other routines in this class can restart us.
	if ((i = path.Find(_T(" //"))) > 0)
		 m_ExpandPath = path.Left(i);
	else m_ExpandPath = path;

	// since a # can't be part of a file or folder name,
	//	trim off everything after a # (includeing the #)
	if ((i = m_ExpandPath.Find(_T('#'))) > 0)
		 m_ExpandPath = m_ExpandPath.Left(i);

	// now check to see if we have a string in the same syntax as the depot tree
	if (newPath)
	{
		if (((m_SlashChar == _T('/'))  && (m_ExpandPath.GetAt(1) == _T(':')))
		 || ((m_SlashChar == _T('\\')) && (m_ExpandPath.GetAt(0) == _T('/'))))
		{
			// we have a mismatch of tree syntax and syntax of the string to find
			// see if fstat can determine its name in the other syntax
			CStringList list;
			BOOL bGotIt = FALSE;
			BOOL bAddSl = FALSE;
			TCHAR tchar = m_SlashChar == _T('/') ? _T('\\') : _T('/');
			if (m_ExpandPath.GetAt(m_ExpandPath.GetLength()-1) == tchar)
				bAddSl = TRUE;
			TrimRightMBCS(m_ExpandPath, _T("*./\\"));
			// quick check for the special case of the client root
			if (m_SlashChar == _T('/') && m_ExpandPath == TheApp()->m_ClientRoot)
			{
				m_ExpandItem = TreeView_GetRoot(m_hWnd);
				m_ExpandDepotContinue = FALSE;
				m_ExpandPath.Empty();
				goto done;
			}
			list.AddHead(m_ExpandPath);
			CCmd_Fstat *pCmd = new CCmd_Fstat;
			pCmd->Init(NULL, RUN_SYNC, key ? HOLD_LOCK : LOSE_LOCK, key);
			if ( pCmd->Run(FALSE, &list, GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT ) 
			 && !pCmd->GetError() )
			{
				CObList *list= pCmd->GetFileList();
				if( list->GetCount() > 0 )
				{
					CP4FileStats *pFileStats = (CP4FileStats *)list->GetHead();
					m_ExpandPath = m_SlashChar == _T('/') 
								 ? pFileStats->GetFullDepotPath() : pFileStats->GetFullClientPath();
					bGotIt = !m_ExpandPath.IsEmpty();
					delete pFileStats;
				}
#ifdef _DEBUG
				// More than 1 thing in the list will cause memory to leak!
				if ( list->GetCount() > 1 )
					ASSERT(0);
#endif
			}
			delete pCmd;
			if (!bGotIt)
			{
				// if we couldn't find it using p4 fstat,
				// so now run p4 where on the string and see what we get
				if (m_SlashChar == _T('/'))
				{
					if (m_ExpandPath.FindOneOf(_T("@#%*")) != -1)
					{
						StrBuf b;
						StrBuf f;
						f << CharFromCString(m_ExpandPath);
						StrPtr *p = &f;
						StrOps::WildToStr(*p, b);
						m_ExpandPath = CharToCString(b.Value());
					}
				}
				CCmd_Where *pCmd1 = new CCmd_Where;
				pCmd1->Init(NULL, RUN_SYNC, key ? HOLD_LOCK : LOSE_LOCK, key);
				if ( pCmd1->Run(m_ExpandPath) && !pCmd1->GetError() 
				  && pCmd1->GetDepotFiles()->GetCount() )
				{
					m_ExpandPath = m_SlashChar == _T('/') 
								 ? pCmd1->GetDepotSyntax() : pCmd1->GetLocalSyntax();
					if (bAddSl)
						m_ExpandPath += m_SlashChar;
					bGotIt = TRUE;
				}
				delete pCmd1;
				if (!bGotIt)
					return;
			}
		}
	}

	if (newPath && (m_SlashChar == _T('\\')))
	{
		// if new and is local syntax, skip root and trailing backslash
		i = TheApp()->m_ClientRoot.GetLength();
		if (i >= m_ExpandPath.GetLength())
			i = -1;
	}
	else
	{
		// if new depot and is syntax, gotta skip "//", 
		// otherwise gotta skip a single '/' (or '\')
		// then find the end of the 1st element of the path
		i = FindMBCS(m_ExpandPath, m_SlashChar, newPath ?  2 : 1);
	}
	if (i != -1)	// found a '/', now extract the 1st element
	{
		node2find = m_ExpandPath.Left(i);
		bLast = FALSE;
	}
	else		// if didn't find a '/', this is the last element
	{
		node2find = m_ExpandPath;
		bLast = TRUE;
	}

	if (newPath)
	{
		// start at top of treeview
		m_OrigPath = m_ExpandPath;
		svExpandItem = m_ExpandItem = TreeView_GetRoot(m_hWnd);
	}
	else
	{
		// the leading '/' does not show in the treeview - a space appears instead
		ReplaceMBCS(node2find, m_SlashChar, _T(' '));

		// remeber where we started
		svExpandItem = m_ExpandItem;

		// get the first child
		m_ExpandItem = TreeView_GetChild(m_hWnd, m_ExpandItem);
	}

	do	// look thru all the siblings for a name that equals our 1st element (node2find)
	{
		tvItem.hItem = m_ExpandItem;
	    tvItem.cchTextMax = LONGPATH;

		// get the item so we can compare its name
		if (TreeView_GetItem(m_hWnd, &tvItem))
		{
			// if last element, it may be followed by #rev-number; trim it off
			if (bLast)
			{
				LPTSTR	p;

				if ((p = _tcsrchr( buf, _T('#') )) != NULL)
				{
					do
					{
						*p = _T('\0');
					} while (*--p == _T(' '));
				}
			}
			// is this the tree item we are looking for? if could be " name" or "name"
			if (IS_NOCASE())
			{
				if (!_tcsicmp(buf+1, node2find) || !_tcsicmp(buf, node2find))
					break;
			}
			else
			{
				if (!_tcscmp(buf+1, node2find) || !_tcscmp(buf, node2find))
					break;
			}
		}
	} while ((m_ExpandItem = TreeView_GetNextSibling(m_hWnd, m_ExpandItem)) != NULL);

	if (m_ExpandItem)	// did we find it?
	{
		if (!bLast)		// if it is not the last element of the path
		{
			// if we are doing local syntax and the root is the root of a drive (eg. C:\)
			// we need to leave the \ behind for the next round
			if (newPath && i == 3 && m_SlashChar == _T('\\'))
				i = 2;

			// we will need to be restarted if TreeView_Expand() triggers p4 dirs & fstats
			m_ExpandDepotContinue = TRUE;

			// remove the element that we found above from our class path save variable
			m_ExpandPath = m_ExpandPath.Right(m_ExpandPath.GetLength() - i);

			if (noExpand && ( GetLParam( m_ExpandItem ) != FOLDER_ALREADY_EXPANDED ))
			{	// We were told not to start an expand, so we are done.
				// This shouldn't happen unless the folder was deleted
				//	by someone else and we aren't showing deleted files
				// or we have changed clients, are in client view
				//	and the selected files are not under this client
				m_ExpandDepotContinue = FALSE;
				m_ExpandPath.Empty();
			}
			// expand the node we found - this may fail - which will trigger p4 fdris and fstats
			else if (TreeView_Expand(m_hWnd, m_ExpandItem, TVE_EXPAND))
			{
				// node has already been expanded, so just recurse
				m_ExpandDepotContinue = FALSE;
				if ( GetLParam( m_ExpandItem ) == FOLDER_ALREADY_EXPANDED )
					ExpandDepotString(m_ExpandPath, FALSE, noExpand, key, noErrMsg);
				return;
			}
		}
		else	// it is the last element of the path - we are done
		{
			m_ExpandDepotContinue = FALSE;
			m_ExpandPath.Empty();
			if (m_Add2ExpandItemList)
				m_ExpandItemList.AddTail((CObject *)m_ExpandItem);
			if (TheApp()->m_bFindInChg)
			{
				TheApp()->m_bFindInChg = FALSE;
				if (ITEM_IS_FILE(m_ExpandItem))
				{
					CP4FileStats *fs= m_FSColl.GetStats(GetLParam(m_ExpandItem));
					if (fs->GetMyOpenAction())
						MainFrame()->PositionChgs( fs->GetFullDepotPath(), fs->GetMyOpenAction() );
				}
			}
		}
	}
	else if ((j = m_ExpandPath.Find(_T(' '))) > 0)		// it was not found in the tree; 
	{				// see if there is a space in the string and trim everything after it
		m_ExpandPath = m_ExpandPath.Left(j);
		m_ExpandItem = svExpandItem;
		ExpandDepotString(m_ExpandPath, FALSE, noExpand, key, noErrMsg);
		return;
	}
	else	// it was not found in the tree, highlight last found & we are done
	{
		m_ExpandPath.TrimLeft(m_SlashChar);
		m_ExpandPath.TrimLeft();
		m_ExpandPath.TrimRight();
		if (m_ExpandPath.GetLength())
        {
            CString msg;
            msg.FormatMessage(IDS_COULD_NOT_FIND_s_IN_DEPOT_VIEW, m_OrigPath);
			AddToStatus(msg, noErrMsg ? SV_DEBUG : SV_WARNING);
        }
		m_ExpandItem = svExpandItem;	// so we can highlight the last thing we did find
		m_ExpandDepotContinue = FALSE;
	}
done:
	// we get here if 
	//	1)	we found what we were looking for  or
	//	2)	we cannot find what we are looking for  or
	//	3)	TreeView_Expand() failed and has thus triggered p4 dirs & fstats
	// in all 3 cases, select the last thing we did find.
	UnselectAll();
	SetSelectState( m_ExpandItem, TRUE );
	ASSERT(GetSelectedCount());
	MainFrame()->SetActiveView(DYNAMIC_DOWNCAST(CView,GetParent()), TRUE);	// set focus to Depot pane
}

BOOL CDepotTreeCtrl::SelectExpandItemList()
{
	if (m_ExpandItemList.IsEmpty())
		return FALSE;

	UnselectAll();
	for (POSITION pos = m_ExpandItemList.GetHeadPosition(); pos != NULL; )
	{
		HTREEITEM item = (HTREEITEM)m_ExpandItemList.GetNext(pos);
		SetSelectState(item, TRUE);
	}
	// Make sure selection set is properly displayed
	SetAppearance(FALSE, TRUE, FALSE);
	return TRUE;
}

HTREEITEM CDepotTreeCtrl::FindDepotSibling(const CString &path, HTREEITEM item, BOOL bUp)
{
    TCHAR  buf[ LONGPATH + 2 ] = _T(" ");	// will hold tree item names
	TVITEM tvItem;
    tvItem.pszText = buf+1;         
	tvItem.mask = TVIF_TEXT;
	TCHAR tchar = path.GetAt(0) == _T('/') ? _T('/') : _T('\\');
	int lastSlash;
	CString pathnotconst = path;
	if ((lastSlash = ReverseFindMBCS(pathnotconst, tchar)) != -1)
	{
		// Grab the last element of the given path
		CString node2find = path.Mid(lastSlash+1);
		TrimRightMBCS(node2find, _T("\\/"));

		// look thru all the siblings for a name that equals our last element (node2find)
		while ((item = bUp ? TreeView_GetPrevSibling(m_hWnd, item)
			               : TreeView_GetNextSibling(m_hWnd, item)) != NULL)
		{
			tvItem.hItem = item;
			tvItem.cchTextMax = LONGPATH;

			// get the item so we can compare its name
			if (TreeView_GetItem(m_hWnd, &tvItem))
			{
				// The last element may be followed by #rev-number; trim it off
				LPTSTR	p;

				if ((p = _tcsrchr( buf, _T('#') )) != NULL)
				{
					do
					{
						*p = _T('\0');
					} while (*--p == _T(' '));
				}
				// is this the tree item we are looking for? if could be " name" or "name"
				if (IS_NOCASE())
				{
					if (!_tcsicmp(buf+1, node2find) || !_tcsicmp(buf, node2find))
						return item;
				}
				else
				{
					if (!_tcscmp(buf+1, node2find) || !_tcscmp(buf, node2find))
						return item;
				}
			}
		}
	}
	return NULL;
}

void CDepotTreeCtrl::OnUpdateDiff_sd_se(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(!SERVER_BUSY() && GetSelectedCount());
}

void CDepotTreeCtrl::OnDiff_sd_se() 
{
	m_StringList.RemoveAll();
	for(int i=GetSelectedCount()-1; i>=0; i--)
	{
		HTREEITEM cItem=GetSelectedItem(i);
		CString itemStr= GetItemPath(cItem);
		if( HasChildren( cItem ) )
			itemStr += _T("...");
		else if (ITEM_IS_FILE(cItem))
		{
			CP4FileStats *fs= m_FSColl.GetStats(GetLParam(cItem));
			itemStr = fs->GetFullDepotPath();
		}
		m_StringList.AddHead(itemStr);
	}

	CCmd_Diff *pCmd= new CCmd_Diff;
	pCmd->Init( m_hWnd, RUN_ASYNC);
	if( pCmd->Run( &m_StringList, NULL, 'd' ) )
		MainFrame()->UpdateStatus( LoadStringResource(IDS_CHECKING_FOR_MISSING) );
	else
		delete pCmd;
}

LRESULT CDepotTreeCtrl::OnP4Diff_sd_se(WPARAM wParam, LPARAM lParam)
{
	int key = 0;
	CString filename;
	POSITION pos;
	CCmd_Diff *pCmd= (CCmd_Diff *) wParam;
	TCHAR sFlag = pCmd->GetSflag();
		
	MainFrame()->DoNotAutoPoll();
	if( !pCmd->GetError() )
	{
		if (pCmd->GetDiffNbrFiles())
		{
			AddToStatus(LoadStringResource(sFlag == _T('d') ? 
                IDS_THERE_ARE_MISSING_FILES : IDS_THERE_ARE_UNOPENED_FILES_THAT_DIFFER),
				SV_WARNING);
			if (sFlag == _T('d'))
				m_StringList.RemoveAll();
			else
				m_StringList2.RemoveAll();
			for(pos = (pCmd->GetList())->GetHeadPosition(); pos != NULL; )
			{
				filename = (pCmd->GetList())->GetNext(pos);
				if (sFlag == _T('d'))
					m_StringList.AddHead(filename);
				else
					m_StringList2.AddHead(filename);
                CString msg;
                msg.FormatMessage(sFlag == _T('d') ? IDS_s_IS_MISSING : IDS_s_IS_DIFFERENT, filename);
				AddToStatus(msg, SV_WARNING);
			}
		}
		else
			AddToStatus(LoadStringResource(sFlag == _T('d') ? 
                IDS_THERE_ARE_NO_MISSING_FILES : IDS_THERE_ARE_NO_UNOPENED_FILES_THAT_DIFFER),
						SV_COMPLETION);
	}
	delete pCmd;
	if (sFlag == _T('d'))
	{
		m_StringList2.RemoveAll();
		for(int i=GetSelectedCount()-1; i>=0; i--)
		{
			HTREEITEM cItem=GetSelectedItem(i);
			CString itemStr= GetItemPath(cItem);
			if( HasChildren( cItem ) )
				itemStr += _T("...");
			else if (ITEM_IS_FILE(cItem))
			{
				CP4FileStats *fs= m_FSColl.GetStats(GetLParam(cItem));
				itemStr = fs->GetFullDepotPath();
			}
			m_StringList2.AddHead(itemStr);
		}

		CCmd_Diff *pCmd= new CCmd_Diff;
		pCmd->Init( m_hWnd, RUN_ASYNC);
		if( pCmd->Run( &m_StringList2, NULL, 'e' ) )
			MainFrame()->UpdateStatus( LoadStringResource(IDS_CHECKING_FOR_DIFFERENCES) );
		else
			delete pCmd;
	}
	else if (m_StringList.GetCount() || m_StringList2.GetCount())
	{
		MainFrame()->UpdateStatus(LoadStringResource(IDS_ACTION));

		CForceSyncDlg dlg;
		dlg.m_lpCstrListD = &m_StringList;
		dlg.m_lpCstrListC = &m_StringList2;
		dlg.m_Key = key;

		dlg.m_SelChange = LoadStringResource(IDS_DEFAULTCHANGELISTNAME);
		::SendMessage(m_changeWnd, WM_GETMYCHANGESLIST, (WPARAM) &(dlg.m_pChangeList), 0);

		if ((dlg.DoModal() == IDOK) 
		 && (m_StringList.GetCount() || m_StringList2.GetCount()))
		{
			if (dlg.m_Action == 1)
			{
				//	Combine the 2 lists
				//
				for(pos = m_StringList2.GetHeadPosition(); pos != NULL; )
				{
					filename = m_StringList2.GetNext(pos);
					m_StringList.AddHead(filename);
				}
				m_StringList2.RemoveAll();
				//
				//		98.1 changed the command to recopy. before: p4 refresh. after: p4 sync -f
				//
				if ( GET_SERVERLEVEL( ) > 3 )
				{
					// Add a "#have" to each filespec
					POSITION pos= m_StringList.GetHeadPosition();
					while( pos != NULL )
					{
						POSITION oldPos= pos;
						CString filespec= m_StringList.GetNext(pos);
						m_StringList.SetAt(oldPos, filespec+_T("#have"));
					}

					CCmd_Get *pCmd= new CCmd_Get;
					pCmd->Init( m_hWnd, RUN_ASYNC);
					if( pCmd->Run( &m_StringList, FALSE, TRUE ) )
						MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_SYNC) );
					else
						delete pCmd;
				}
				else
				{
					CCmd_Refresh *pCmd= new CCmd_Refresh;
					pCmd->Init( m_hWnd, RUN_ASYNC);
					if(	pCmd->Run( &m_StringList ) )
						MainFrame()->UpdateStatus( LoadStringResource(IDS_FILE_REFRESH) );
					else
						delete pCmd;
				}
			}
			else	// open for delete and edit
			{
				int selectedChange = dlg.m_SelectedChange;
				if (m_StringList.GetCount())	// Anything to open for delete?
				{
					if( SERVER_BUSY() && !GET_SERVER_LOCK(key) )
					{
						AfxMessageBox(IDS_UNABLE_TO_OPEN_TRY_AGAIN, MB_ICONSTOP);
						MainFrame()->ResumeAutoPoll();
						return 0;
					}
					CCmd_ListOpStat *pCmd= new CCmd_ListOpStat;
					pCmd->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );
					if (m_StringList2.GetCount())
						pCmd->SetOpenAfterDelete(TRUE, selectedChange);
					if( pCmd->Run( &m_StringList, P4DELETE, selectedChange ) )
						MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_DELETE) );
					else
						delete pCmd;
				}
				else if (m_StringList2.GetCount())	// Anything to open for edit?
				{
					if( SERVER_BUSY() && !GET_SERVER_LOCK(key) )
					{
						AfxMessageBox(IDS_UNABLE_TO_OPEN_TRY_AGAIN, MB_ICONSTOP);
						MainFrame()->ResumeAutoPoll();
						return 0;
					}
					CCmd_ListOpStat *pCmd2= new CCmd_ListOpStat;
					pCmd2->Init( m_hWnd, RUN_ASYNC, HOLD_LOCK );

					if( pCmd2->Run( &m_StringList2, P4EDIT, selectedChange ) )
						MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUEST_OPEN_EDIT) );
					else
						delete pCmd2;
				}
				else ASSERT(0);
			}
		}
		else
		{
			m_StringList.RemoveAll();
			m_StringList2.RemoveAll();
			MainFrame()->ClearStatus();
		}
	}
	else MainFrame()->ClearStatus();
	MainFrame()->ResumeAutoPoll();
	return 0;
}

void CDepotTreeCtrl::ClearDepotFilter(BOOL bRunUpdate/*=TRUE*/)
{
	m_FilterDepot=FALSE; 
	m_DepotFilterList.RemoveAll();
	m_DepotFilterPort = m_DepotFilterClient = "";
	MainFrame()->SetDepotCaption( GET_P4REGPTR()->ShowEntireDepot() );
	if (bRunUpdate)
		OnViewUpdate(); 
}

void CDepotTreeCtrl::ConvertDepotFilterList(int key)
{
	CCmd_Fstat *pCmd = new CCmd_Fstat;
	pCmd->Init(NULL, RUN_SYNC, key ? HOLD_LOCK : LOSE_LOCK, key);
	if ( pCmd->Run(FALSE, &m_DepotFilterList, GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT ) 
	 && !pCmd->GetError() )
	{
		m_DepotFilterList.RemoveAll();
		CObList *list= pCmd->GetFileList();
		if( list->GetCount() > 0 )
		{
		    for( POSITION pos= list->GetHeadPosition(); pos!= NULL; )
			{
				CP4FileStats *pFileStats = (CP4FileStats *)list->GetNext(pos);
				m_DepotFilterList.AddTail(m_SlashChar == _T('/') 
						 ? pFileStats->GetFullDepotPath() : pFileStats->GetFullClientPath());
				delete pFileStats;
			}
		}
	}
	else
	{
		m_FilterDepot = FALSE;
		m_DepotFilterList.RemoveAll();
		m_DepotFilterPort = m_DepotFilterClient = "";
		MainFrame()->SetDepotCaption( GET_P4REGPTR()->ShowEntireDepot() );
	}
	delete pCmd;
}

LRESULT CDepotTreeCtrl::IsFilteredOnOpen(WPARAM wParam, LPARAM lParam)
{
	int *pInt = (int *)lParam;
	if (m_FilterDepot && (m_DepotFilterType == DFT_ALLOPENED 
					   || m_DepotFilterType == DFT_MYOPENED))
	{
		*pInt = m_DepotFilterType;
	}
	else
		*pInt = 0;
	return *pInt;
}

void CDepotTreeCtrl::LoadDepotFilterList(CStringList *pList)
{
	int i;
	CString str;
	m_FilterDepot=TRUE; 
	for( POSITION pos= pList->GetHeadPosition(); pos != NULL; )
	{
		str = pList->GetNext(pos);
		if ((i = str.Find(_T('#'))) != -1)
			str = str.Left(i);
		m_DepotFilterList.AddHead( str );
	}
	MainFrame()->SetDepotCaption( GET_P4REGPTR()->ShowEntireDepot() );
}

LRESULT CDepotTreeCtrl::OnRedoOpendList(WPARAM wParam, LPARAM lParam)
{
	if (wParam)
		GetOpenedList(wParam == DFT_ALLOPENED ? TRUE : FALSE);
	return 0;
}

void CDepotTreeCtrl::GetOpenedList(BOOL bAll)
{
	CCmd_Opened *pCmd= new CCmd_Opened;
	pCmd->Init( m_hWnd, RUN_ASYNC);
	pCmd->SetItemRef( NULL );
	if( pCmd->Run( bAll, TRUE ) )
	{
		MainFrame()->UpdateStatus( LoadStringResource(IDS_REQUESTING_FILE_INFORMATION) );
		EmptyDepotFilter();					// empty out any old filter info
		SetDepotFilterType(bAll ? DFT_ALLOPENED : DFT_MYOPENED);
		m_DepotFilterPort = GET_P4REGPTR()->GetP4Port();
		m_DepotFilterClient = GET_P4REGPTR()->GetP4Client();

	}
	else
		delete pCmd;
}

LRESULT CDepotTreeCtrl::OnP4Opened(WPARAM wParam, LPARAM lParam)
{
	CCmd_Opened *pCmd= (CCmd_Opened *) wParam;

	if(!pCmd->GetError())
	{
		m_StringList.RemoveAll();
		CObList *list = pCmd->GetList( );
		ASSERT_KINDOF( CObList, list );
		for(POSITION pos= list->GetHeadPosition(); pos!=NULL; )
		{
			CP4FileStats *stats = ( CP4FileStats * )list->GetNext( pos );
			ASSERT_KINDOF( CP4FileStats, stats );
			m_StringList.AddHead(stats->GetFullDepotPath( ));
            delete stats;
		}
		if (!m_StringList.IsEmpty())	// if there is anything in the files list,
		{
			EmptyDepotFilter();
			LoadDepotFilterList(&m_StringList);	// load the depot filter list
			OnViewUpdate();						//	and refresh the depot pane
		}
	}
	delete pCmd;
	return 0;
}

void CDepotTreeCtrl::FilterViaList(CString filelist)
{
	if (filelist.IsEmpty())
	{
		ClearDepotFilter();
		return;
	}

	CString line;
	m_StringList.RemoveAll();
	for( int i=0; i<filelist.GetLength(); i++ )
	{
		switch( filelist[i] )
		{
		case _T('\r'):
			break;
		case _T('\n'):
			line.TrimRight();
			if( !line.IsEmpty() )
			{
				m_StringList.AddHead( line );
				line.Empty();
			}
			break;
		default:
			line+= filelist[i];
		}
	}
	if( !line.IsEmpty() )
		m_StringList.AddHead( line );

	if (m_StringList.IsEmpty())
	{
		ClearDepotFilter();
		return;
	}
	EmptyDepotFilter();					// empty out any old filter info
	SetDepotFilterType(DFT_LIST);
	m_Need2Filter = TRUE;
	m_FilterList  = filelist;
	m_DepotFilterPort = GET_P4REGPTR()->GetP4Port();
	m_DepotFilterClient = GET_P4REGPTR()->GetP4Client();
	m_SaveP4Files_FileSpec = m_P4Files_FileSpec;
	GetNextFilesFromFilterList();
}

void CDepotTreeCtrl::GetNextFilesFromFilterList()
{
	if (!m_StringList.IsEmpty())
	{
		CString str = m_StringList.GetTail();
		m_StringList.RemoveTail();
		if ((str.Find(_T('/')) == -1) && (str.Find(_T('\\')) == -1))
		{
			if (GET_P4REGPTR()->ShowEntireDepot() == SDF_DEPOT)
				str = _T("//...") + str;
			else
			{
				CString client = GET_P4REGPTR()->GetP4Client();
				str	= _T("//") + client + _T("/...") + str;
			}
		}
		RunP4Files(str);
	}
	else
	{
		m_P4Files_FileSpec = m_SaveP4Files_FileSpec;
		m_P4Files_List.RemoveAll();		// empty the files list from the find since it is now out-of-date
		m_Need2Filter = FALSE;
		OnViewUpdate();					//	and refresh the depot pane
	}
}

void CDepotTreeCtrl::OnUpdateAddBookmark(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable( !SERVER_BUSY() && GetSelectedCount()==1 && !IsSelected(m_Root) );	
}

void CDepotTreeCtrl::OnAddBookmark() 
{
	if (GetSelectedCount() != 1)
	{
		ASSERT(0);
		return;
	}
	CString depotPath = GetCurrentItemPath();

	if (depotPath.GetAt(0) != _T('/'))
		GetItemDepotSyntax(GetSelectedItem(0), &depotPath);
	AddBookmark(depotPath);
}

LRESULT CDepotTreeCtrl::OnAddBookmarkMsg(WPARAM wParam, LPARAM lParam)
{
	CString depotPath = *((CString *)lParam);
	AddBookmark(depotPath);
	return 0;
}

void CDepotTreeCtrl::AddBookmark(CString &depotPath) 
{
	BOOL bSM[MAX_BOOKMARKS+1];
	CString str[MAX_BOOKMARKS+1];
	CString temp;
	int		i;
	int 	j = 0;
	int		k = MAX_BOOKMARKS+1;
	int		s = MAX_BOOKMARKS+1;
	CBookMarkAdd dlg;

	dlg.SetTitle(LoadStringResource(IDS_EDIT_BOOKMARK));
	dlg.SetLabelText(LoadStringResource(IDS_PATH));
	dlg.SetNewMenuName(depotPath);
	dlg.SetIsSubMenu(FALSE);
	dlg.SetRadioShow(3);
	dlg.SetCanCr8SubMenu(FALSE);
	if ((dlg.DoModal() != IDOK) || (!(dlg.GetNewMenuName()).GetLength()))
		return;

	depotPath = dlg.GetNewMenuName();

	// Get the strings
	for(i = -1; ++i < MAX_BOOKMARKS; )
	{
		str[i] = GET_P4REGPTR()->GetBkMkMenuName(i);
		if (str[i] == depotPath)
		{
			AddToStatus(depotPath + LoadStringResource(IDS_ALREADY_BOOKMARKED), SV_WARNING);
			MessageBeep(0);
			return;
		}
		bSM[i] = GET_P4REGPTR()->GetBkMkIsSubMenu(i);
		if (bSM[i])
		{
			j = i;
			if (s == MAX_BOOKMARKS+1)
				s = i;
		}
		else if (!(str[i].IsEmpty()))
			j = i;
		else if (k == MAX_BOOKMARKS+1)
			k = i;
	}
	if (k == MAX_BOOKMARKS+1)
	{
		MessageBeep(0);
		return;
	}

	// Reload the strings
	BOOL bAdded = FALSE;
	for(i = j = -1; ++j < MAX_BOOKMARKS; )
	{
		if (j == s)
		{
			GET_P4REGPTR()->SetBkMkMenuName(j, depotPath);
			GET_P4REGPTR()->SetBkMkIsSubMenu(j, FALSE);
			bAdded = TRUE;
		}
		else
		{
			GET_P4REGPTR()->SetBkMkMenuName(j, str[++i]);
			GET_P4REGPTR()->SetBkMkIsSubMenu(j, bSM[i]);
		}
	}
	if (!bAdded)
	{
		GET_P4REGPTR()->SetBkMkMenuName(k, depotPath);
		GET_P4REGPTR()->SetBkMkIsSubMenu(k, FALSE);
	}
	MainFrame()->LoadBkMkMenu();
	AddToStatus(depotPath + LoadStringResource(IDS_ADDED_TO_BOOKMARKS));
}

LRESULT CDepotTreeCtrl::OnSetAddFstats(WPARAM wParam, LPARAM lParam)
{
	CObList *list = (CObList *)lParam;
	ASSERT_KINDOF(CObList, list);
	POSITION pos= list->GetHeadPosition();
    InitFindItem();
	while(pos != NULL)
	{
		// Get the filestats info
		CP4FileStats *stats = (CP4FileStats *) list->GetNext(pos);
		ASSERT_KINDOF(CP4FileStats, stats);

		// Find the item
		HTREEITEM item;
		item=FindItem(stats->GetDepotDir(), stats->GetDepotFilename(), FALSE);
		if(item==NULL)
		{
			// It is quite possible for a file to be opened for add but to not be in the tree
			XTRACE(_T("OnSetAddFstats() item not found %s\n"), stats->GetFullDepotPath());
		}
		else
		{
			// Update its properties
			int index=GetLParam(item);
			CP4FileStats *fs=m_FSColl.GetStats(index);
			
			if (!stats->GetOtherUsers()[0])
				fs->SetOpenAction(stats->GetMyOpenAction(), FALSE);
			else
				fs->SetOpenAction(stats->GetOtherOpenAction(), TRUE);
			fs->SetHeadType(stats->GetHeadType());
			fs->SetType(stats->GetType());
			fs->SetOpenChangeNum(stats->GetOpenChangeNum());
			fs->SetOpenAction(stats->GetOtherOpenAction(), TRUE);
			fs->SetOtherOpens(stats->GetOtherOpens());
					
			// Update the image in this window
			SetImage(item, TheApp()->GetFileImageIndex(fs));
		}

		// If this an fstat from 'p4 opened -a' for another client
		// and they have it opened for add, we might also have it opened for add
		// so search m_FstatsAdds and if we find an entry for it update it.
		BOOL b = TRUE;
		if (stats->GetOtherOpenAction() == F_ADD)
		{
			POSITION pos2 = m_FstatsAdds.GetHeadPosition();
			while(pos2 != NULL)
			{
				CP4FileStats *statsAdd = (CP4FileStats *) m_FstatsAdds.GetNext(pos2);
				// We have to check the depot paths
				// because the client path is empty for 'stats'
				if (!_tcscmp(statsAdd->GetFullDepotPath(), stats->GetFullDepotPath()))
				{
					statsAdd->SetOpenAction(F_ADD, FALSE);
					statsAdd->SetOpenAction(F_ADD, TRUE);
					b = FALSE;
					break;
				}
			}
		}
		if (b)
		{
			// Create a copy of the fstat info
			// and add it to the list off fstat's for files opened for add
			CP4FileStats *newStats;
			newStats= new CP4FileStats;
			newStats->Create(stats);
			m_FstatsAdds.AddTail(newStats);
		}

	} // while
	return 0;
}

LRESULT CDepotTreeCtrl::OnGetAddFstats(WPARAM wParam, LPARAM lParam)
{
	return (LRESULT)&m_FstatsAdds;
}

void CDepotTreeCtrl::Empty_FstatsAdds()
{
	if (m_FstatsAdds.GetCount())
	{
		POSITION pos= m_FstatsAdds.GetHeadPosition();
		while(pos != NULL)
		{
			// Delete the filestats object
			CP4FileStats *stats = (CP4FileStats *) m_FstatsAdds.GetNext(pos);
			delete stats;
		}
		m_FstatsAdds.RemoveAll();
	}
}

void CDepotTreeCtrl::OnUpdateFileDelete(CCmdUI* pCmdUI)
{
	pCmdUI->Enable( AllAddable() && GetSelectedCount() );	
}

void CDepotTreeCtrl::OnFileDelete()
{
	int selcount;
	if ((selcount = GetSelectedCount()) == 0)
		return;

	CString filename;
	if (GetSelectedCount() == 1)
		filename = GetItemPath(GetLastSelection());
	else
		filename.FormatMessage(IDS_THESE_n_FILES, GetSelectedCount());
	CString txt;
	txt.FormatMessage(IDS_ARE_YOU_SURE_YOU_WANT_TO_DELETE_s, filename );
	if (IDYES == AfxMessageBox(txt, MB_YESNO|MB_ICONQUESTION|MB_DEFBUTTON2))
	{
		int i, count;
		CDWordArray unselectionSet;
		unselectionSet.SetSize(selcount);
		for( i = -1, count = 0; ++i < selcount; )
		{
			HTREEITEM hItem = GetSelectedItem( i );
			// can't unlink a directory (should never happen, but...)
			if( HasChildren( hItem ) ) 
				continue;
			filename = GetItemPath( hItem );
			int rc;
			if ((rc = _tunlink(filename)) == 0)
			{
				unselectionSet.SetAt(count++, (DWORD)hItem);
				txt.FormatMessage(IDS_s_DELETED, filename);
				TheApp()->StatusAdd( txt );
			}
			else
			{
				txt.FormatMessage(IDS_s_NOT_DELETED_s, filename, 
					errno == EACCES ? LoadStringResource(IDS_BECAUSE_READONLY) : _T(""));
				TheApp()->StatusAdd( txt, SV_WARNING );
			}
		}
		if (selcount > 1)
		{
			txt.FormatMessage(IDS_n_FILES_DELETED, count);
			TheApp()->StatusAdd( txt, SV_COMPLETION );
		}
		if (count)
		{
			for (i = count; i--; )
				SetSelectState((HTREEITEM)unselectionSet.GetAt(i), FALSE);
			OnViewUpdate();
		}
	}
}

void CDepotTreeCtrl::OnUpdateFileAdd(CCmdUI* pCmdUI)
{
	pCmdUI->Enable( MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY()) );	
}

void CDepotTreeCtrl::OnFileAdd()
{
	CString initDir;
	if ( GetSelectedCount() == 1 )
	{
		HTREEITEM item = GetSelectedItem(0);
		if (!ITEM_IS_FILE(item))
		{
			int i;
			CString itemTxt= GetItemPath(item);

			if ( (i = itemTxt.Find(g_TrulyEmptyDir)) != -1 )
				itemTxt = itemTxt.Left(i);

			if (itemTxt.GetAt(1) == _T(':'))
			{
				initDir = itemTxt;
			}
			else
			{
				itemTxt.TrimRight(_T('/'));
				if (itemTxt.Find(_T('/'), 2) == -1)
					initDir = TheApp()->m_ClientRoot;
				else
				{
					// so now run p4 where on the string and see what we get
					CCmd_Where *pCmd1 = new CCmd_Where;
					pCmd1->Init(NULL, RUN_SYNC);
					if ( pCmd1->Run(itemTxt) && !pCmd1->GetError() 
					  && pCmd1->GetDepotFiles()->GetCount() )
					{
						initDir = pCmd1->GetLocalSyntax();
					}
					delete pCmd1;
				}
			}
			initDir.Replace('/', '\\');
		}
	}
	MainFrame()->OnFileAddSetDir(initDir.IsEmpty() ? (LPTSTR)NULL 
							   : initDir.GetBuffer(initDir.GetLength()));
	if (!initDir.IsEmpty())
		 initDir.ReleaseBuffer(-1);
}

void CDepotTreeCtrl::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
	MainFrame()->WaitAWhileToPoll( );
	CMultiSelTreeCtrl::OnVScroll(nSBCode, nPos, pScrollBar);
}

void CDepotTreeCtrl::SetCaption( int iShowEntiredepot )
{
	m_caption.Empty();
	switch (iShowEntiredepot)
	{
	case SDF_CLIENT:	// Client View of Depot
		m_caption = LoadStringResource( IDS_CLIENTVIEWONLY );
		break;
	case SDF_DEPOT:		// Entire Depot
		m_caption = LoadStringResource( IDS_ENTIREDEPOT );
		break;
	case SDF_LOCALP4:	// Local Perforce Files
		m_caption = LoadStringResource( IDS_LOCALP4FILES );
		break;
	case SDF_LOCALTREE:	// All Local Files in Client Tree
		m_caption = LoadStringResource( MainFrame()->m_ShowOnlyNotInDepot 
				  ? IDS_FILESNOTINDEPOT : IDS_LOCALCLIENTTREE );
		break;
	}
	if (IsDepotFiltered())
		m_caption += LoadStringResource(IDS_CAPTION_DEPOT_FILTERED);
	if (GET_P4REGPTR()->ShowDeleted() && (iShowEntiredepot <= IDS_ENTIREDEPOT))
	{
		CString del = LoadStringResource(IDS_CAPTION_DEPOT_DELETED);
		if (IsDepotFiltered())
		{
			m_caption.TrimRight(_T(')'));
			del.TrimLeft(_T(" ("));
			del = _T(", ") + del;
		}
		m_caption += del;
	}
	GetView()->SetCaption(  );
}

void CDepotTreeCtrl::OnUpdateViewUpdate(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(!SERVER_BUSY() && !MainFrame()->IsModlessUp());
}

void CDepotTreeCtrl::OnUpdateViewFilteredview(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() && !MainFrame()->m_ShowOnlyNotInDepot));
}

void CDepotTreeCtrl::OnViewFilteredview() 
{
	// Note that the filter type is 0 relative in the dlg,
	// but is 1 relative in _dft enum in mainframe.h
	// that because -1 is "not set" for dialogbox radio buttons
	FilterDepotDlg dlg;
	dlg.m_FilterType = GetDepotFilterType()-1;
	dlg.m_FileList   = GetFilterList();
	if (dlg.DoModal() == IDOK)	// get the user's preferences
	{
		switch (dlg.m_FilterType+1)
		{
		case DFT_ALLOPENED:
			GetOpenedList(TRUE);
			break;
		case DFT_MYOPENED:
			GetOpenedList(FALSE);
			break;
		case DFT_LIST:
			FilterViaList(dlg.m_FileList);
			break;
		}
	}
}

void CDepotTreeCtrl::OnUpdateViewClearfilter(CCmdUI* pCmdUI) 
{
	pCmdUI->Enable(MainFrame()->SetMenuIcon(pCmdUI, !SERVER_BUSY() && IsDepotFiltered()));
}

void CDepotTreeCtrl::OnViewClearfilter() 
{
	ClearDepotFilter();
}

BOOL CDepotTreeCtrl::IsInRemoteDepot(CString *depotfilename)
{
	if (!m_RemoteDepotList.IsEmpty())
	{
		int i;
		CString depotname = *depotfilename;
		if ((i = depotname.Find(_T('/'), 2)) != -1)
			depotname = depotname.Left(i);
		POSITION pos;
		for ( pos= m_RemoteDepotList.GetHeadPosition(); pos != NULL; )
		{
			if (depotname == m_RemoteDepotList.GetNext( pos ))
				return TRUE;
		}
	}
	return FALSE;
}

void CDepotTreeCtrl::OnPerforceOptions()
{
	MainFrame()->OnPerforceOptions(TRUE, FALSE, IDS_PAGE_DEPOT);
}

// Render CF_HDROP format drag and drop data.  Note that this routine is NOT a CDepotTreeCtrl
// routine - it is an override of COleDataSource for m_OLESource.  It calls back to
// CDepotTreeCtrl::RenderFileNames() to render the files names - it just sets up the structure
// that will hold the rendered file names.
BOOL CP4DOleDataSource::OnRenderData( LPFORMATETC lpFormatEtc, LPSTGMEDIUM lpStgMedium )
{
	int	lgth;

	if ((lgth = m_depotTree->RenderFileNames(NULL)) == 0)
		return FALSE;
	lpStgMedium->tymed = TYMED_HGLOBAL;
	lpStgMedium->hGlobal = GlobalAlloc(GHND, sizeof(_DROPFILES) + (lgth + 2)*sizeof(TCHAR));
	LPDROPFILES lpdropfiles = (LPDROPFILES)GlobalLock(lpStgMedium->hGlobal);
	lpdropfiles->pFiles = sizeof(_DROPFILES);
	lpdropfiles->pt.x = 0;
	lpdropfiles->pt.y = 0;
	lpdropfiles->fNC = FALSE;
#ifdef UNICODE
	lpdropfiles->fWide = TRUE;
#else
	lpdropfiles->fWide = FALSE;
#endif
	BOOL rc = m_depotTree->RenderFileNames((LPTSTR)((char*)lpdropfiles + lpdropfiles->pFiles)) 
			? TRUE : FALSE;
	GlobalUnlock(lpStgMedium->hGlobal);
	if (!rc)	// if there were no file names rendered, clean up
	{
		GlobalFree(lpStgMedium->hGlobal);
		lpStgMedium->hGlobal = 0;
	}
	return rc;
}

// This rountine provides a list of dropped file names for CF_HDROP format drag and drop
// when the Depot pane is the source of the drag and drop.
// The return value is the length of all the file names plus number of files (this is one less
// than the length of the memory needed to render the file names).  If there is nothing to
// render, the return value is 0.
// Call this routine with a NULL to just obtain the length needed for the buffer; call it
// with a pointer to the addr of a buffer in order to load that buffer with the file names.
int CDepotTreeCtrl::RenderFileNames(LPTSTR p)
{
	static LPTSTR	pFN = 0;	// ptr to buffer to store file names to be rendered
	static DWORD	lFN = 0;	// lgth of buffer at pFN
	static DWORD	uFN = 0;	// amt of buffer at pFN actually in use
	static DWORD	ddCtr = 0;	// counter to validate contents of buffer at pFN
	LPTSTR ptr;

	// If the depot pane is the drop target, don't provide a list of files
	// because files from the depot pane to the depot pane are NOT in CF_HDROP format.
	if ( m_DepotIsDropTarget )
		return 0;

	if (!pFN)
	{
		 pFN = (LPTSTR)::VirtualAlloc(NULL, 4096*sizeof(TCHAR), MEM_COMMIT, PAGE_READWRITE);
		 if (!pFN)
			 return(0);	// out of memory!
		 lFN = 4096;
	}
	else if (ddCtr == m_DragDropCtr)	// is this D&D the same as the one we have stored?
	{
		if (p)
			memcpy(p, pFN, uFN*sizeof(TCHAR));
		return uFN;
	}

	CObList list;
	if (!GetSelectedFStats(&list) || !list.GetCount())
		return(0);

	POSITION pos = list.GetHeadPosition();
	ptr = pFN;
	uFN = 0;
	while(pos != NULL)
	{
		CP4FileStats *fs = (CP4FileStats *)list.GetNext(pos);
		CString clientPath = fs->GetFullClientPath();
		// Check to see if we need to use a temp file rather than the live file
	    if( fs->GetHaveRev() > 0 && !CString(fs->GetFullClientPath()).IsEmpty() 
		 && (!fs->IsMyOpen() || (fs->GetMyOpenAction() == F_INTEGRATE)))
        {
			if ((GET_P4REGPTR()->GetUseTempForView() == 1) 
			 || ((GET_P4REGPTR()->GetUseTempForView() == 2) 
			  && !FileExtUsesLiveFileToView(fs->GetFullClientPath())))
			{
				CString fileName;
				int fileRev= fs->GetHaveRev();
				int i = clientPath.ReverseFind(_T('\\'));
				fileName = (i != -1) ? clientPath.Mid(i+1) : _T("TEMP");
				CString tempPath= GET_P4REGPTR()->GetTempDir();
				DWORD errorCode = 0;
	    		for( int i=0; i< 100; i++)
				{
					clientPath.Format(_T("%s\\ReadOnly-%d-Rev-%d-%s"), tempPath, i, fileRev, fileName);
					int j;
					while ((j = clientPath.Find(':', 2)) != -1)
						clientPath.SetAt(j, '_');
					while ((j = clientPath.FindOneOf(_T("/*?\"<>|"))) != -1)
						clientPath.SetAt(j, '_');
					if( CopyFile(fs->GetFullClientPath(), clientPath, TRUE) )
					{
						// Verify that it is readonly
						SetFileAttributes( clientPath, FILE_ATTRIBUTE_READONLY );
						errorCode = 0;
						break;
					}
					else
					{
						errorCode = GetLastError();
						if ((errorCode == ERROR_FILE_NOT_FOUND) 
						 || (errorCode == ERROR_PATH_NOT_FOUND) 
						 || (errorCode == ERROR_HANDLE_DISK_FULL))
							break;
					}
				}
				if (errorCode == ERROR_HANDLE_DISK_FULL)
				{
					// after 100 tries, we couldn't open the temp file
					CString ErrorTxt;
					ErrorTxt.FormatMessage(IDS_DISKFULL_OPENING_TEMP_FILE_s, clientPath);
					AddToStatus(ErrorTxt, SV_ERROR);
					return(0);
				}
			}
		}
		// clientPath now contains the name of either the live file or the newly created temp file
		if ((ptr + clientPath.GetLength() + 4) >= (pFN + lFN))	// running out of room?
		{
			LPTSTR sav = (LPTSTR)::VirtualAlloc(NULL, (lFN + 4096)*sizeof(TCHAR), MEM_COMMIT, PAGE_READWRITE);
			if (!sav)
				return(0);	// we're in trouble - out of memory!
			memcpy(sav, pFN, uFN*sizeof(TCHAR));
			ptr = sav + (ptr - pFN);
			::VirtualFree(pFN, 0, MEM_RELEASE);
			pFN = sav;
			lFN += 4096;
		}
		lstrcpy(ptr, clientPath);
		ptr += clientPath.GetLength() + 1;
		uFN += clientPath.GetLength() + 1;
	}
	*ptr = _T('\0');
	ddCtr = m_DragDropCtr;
	if (p)
		memcpy(p, pFN, uFN*sizeof(TCHAR));
	return uFN;
}