gecko-dev/cmd/macfe/rdfui/CHyperTreeFlexTable.cp

1754 lines
51 KiB
C++
Raw Normal View History

1998-03-28 02:44:41 +00:00
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
//
// Mike Pinkerton, Netscape Communications
//
// Subclass of CStandardFlexTable to handle working with XP RDF Hyper Trees
//
#include "CHyperTreeFlexTable.h"
// PowerPlant
#include <LTableArrayStorage.h>
#include <LDropFlag.h>
#include <Appearance.h>
1998-03-28 02:44:41 +00:00
#include "CHyperTreeHeader.h"
#include "URDFUtilities.h"
#include "uapp.h"
#include "CURLDragHelper.h"
#include "CIconTextDragTask.h"
#include "CNetscapeWindow.h"
#include "resgui.h"
#include "macutil.h"
#include "ufilemgr.h"
#include "CInlineEditField.h"
#include "CContextMenuAttachment.h"
#include "libi18n.h"
1998-03-28 02:44:41 +00:00
#include <vector.h>
#include <algorithm>
extern RDF_NCVocab gNavCenter; // RDF vocab struct for NavCenter
const ResIDT cFolderIconID = kGenericFolderIconResource;
1998-03-28 02:44:41 +00:00
const ResIDT cItemIconID = 15313;
const ResIDT cFileIconID = kGenericDocumentIconResource;
1998-03-28 02:44:41 +00:00
#pragma mark -- class CHyperTreeFlexTable --
// ------------------------------------------------------------
// Construction, Destruction
// ------------------------------------------------------------
CHyperTreeFlexTable::CHyperTreeFlexTable(LStream* inStream)
: CStandardFlexTable(inStream),
mHTView(NULL),
mHTNotificationData(NULL),
mViewBeforeDrag(NULL)
{
mSendDataUPP = NewDragSendDataProc(LDropArea::HandleDragSendData);
ThrowIfNil_(mSendDataUPP);
// turn on spring-loading of folders during d&d
mAllowAutoExpand = true;
1998-09-24 09:23:48 +00:00
mTextDrawingStuff.encoding = TextDrawingStuff::eUTF8;
1998-03-28 02:44:41 +00:00
}
CHyperTreeFlexTable::~CHyperTreeFlexTable()
{
}
// ------------------------------------------------------------
// CStandardFlexTable overrides
// ------------------------------------------------------------
ResIDT CHyperTreeFlexTable::GetIconID(TableIndexT inRow) const
{
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inRow) );
if (node) {
if ( HT_IsContainer(node) )
return cFolderIconID;
else {
char* url = HT_GetNodeURL(node);
if ( url && *url == 'f' ) // is it a file url?
return cFileIconID;
else
return cItemIconID;
}
}
return cItemIconID;
}
//
// SetUpTableHelpers
//
// Call the inherited version to do most of the work but replace the
// selector with our own version to track the selection in the HT_API
//
void
CHyperTreeFlexTable :: SetUpTableHelpers()
{
CStandardFlexTable::SetUpTableHelpers();
// replace selector....
SetTableSelector( new CHyperTreeSelector(this) );
} // SetupTableHelpers
//
// GetHiliteTextRect
//
// Find out what rectangle to draw selected in the currently selected row. This is
// based on the item title.
//
Boolean
CHyperTreeFlexTable::GetHiliteTextRect ( TableIndexT inRow, Boolean /*inOkIfRowHidden*/, Rect& outRect) const
1998-03-28 02:44:41 +00:00
{
STableCell cell(inRow, GetHiliteColumn());
if (!GetLocalCellRect(cell, outRect))
return false;
Rect iconRect;
GetIconRect(cell, outRect, iconRect);
outRect.left = iconRect.right;
// Get cell data for first column
char buffer[1000];
GetHiliteText(inRow, buffer, 1000, false, &outRect);
1998-03-28 02:44:41 +00:00
return true;
} // GetHiliteTextRect
//
// GetMainRowText
//
// Returns the text of the title of the item.
//
void
CHyperTreeFlexTable :: GetMainRowText( TableIndexT inRow, char* outText, UInt16 inMaxBufferLength) const
{
if (!outText || inMaxBufferLength == 0)
return;
void* data;
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header != NULL);
CHyperTreeHeader::ColumnInfo info = header->GetColumnInfo(FindTitleColumnID()-1); // GCI() is 0-based
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inRow) );
if ( node && HT_GetNodeData(node, info.token, info.tokenType, &data) ) {
1998-03-28 02:44:41 +00:00
char* title = static_cast<char*>(data);
strcpy ( outText, title );
}
else
*outText = NULL;
} // GetMainRowText
//----------------------------------------------------------------------------------------
void CHyperTreeFlexTable::SetEditParam(int w, int h, char* utf8, SPoint32& ImagePoint)
1998-03-28 02:44:41 +00:00
{
mNameEditor->ResizeFrameTo(w, h, true);
char* editText = (char*) INTL_ConvertLineWithoutAutoDetect(
CS_UTF8, INTL_GetCharSetID(INTL_DefaultTextWidgetCsidSel),
(unsigned char*)utf8, strlen(utf8));
mNameEditor->UpdateEdit(CStr255(editText), &ImagePoint, nil);
XP_FREEIF(editText);
} // SetEditParam
1998-03-28 02:44:41 +00:00
//
// FindTitleColumnID
//
// Returns the index of the column that contains the item title and icon. This will search out the
// correct column by looking at the column headers.
//
// NOTE: THERE IS CURRENTLY NO BACKEND SUPPORT FOR REORDERING COLUMNS, SO WE MAKE AN EXPLICIT
// ASSUMPTION THAT THE FIRST COLUMN IS THE TITLE COLUMN. AT LEAST THIS SHOULD BE THE ONLY
// ROUTINE THAT HAS TO CHANGE WHEN THAT ASSUMPTION CHANGES.
//
Uint32
CHyperTreeFlexTable :: FindTitleColumnID ( ) const
{
return 1; // for now....
} // FindTitleColumnID
//
// CellInitiatesDrag
//
// Determines if a cell is allowed to start a drag. We disallow this for any cell except for the
// cell with the title and icon. Also disallow d&d when selection is not allowed in this view
// (simple tree).
1998-03-28 02:44:41 +00:00
//
Boolean
CHyperTreeFlexTable :: CellInitiatesDrag ( const STableCell& inCell ) const
{
#ifdef USE_SELECTION_PROP
return inCell.col == FindTitleColumnID() &&
URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->useSelection, true) == true
? true : false;
#else
return inCell.col == FindTitleColumnID();
#endif
1998-03-28 02:44:41 +00:00
} // CellInitiatesDrag
//
// CellSelects
//
// Determines if a cell is allowed to select the row. We disallow this for any cell except for the
// cell with the title and icon. Also disallow selection when selection is not allowed in this view
// (simple tree).
1998-03-28 02:44:41 +00:00
//
Boolean
CHyperTreeFlexTable :: CellSelects ( const STableCell& inCell ) const
{
#ifdef USE_SELECTION_PROP
return inCell.col == FindTitleColumnID() &&
URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->useSelection, true) == true
? true : false;
#else
return inCell.col == FindTitleColumnID();
#endif
1998-03-28 02:44:41 +00:00
} // CellSelects
#ifdef USE_SELECTION_PROP
// (pinkerton)
// This stuff is not needed if we allow the table to always select things. CStdFlexTable
// will try to open the selection automatically if the click count is set correctly, which
// we know it is. Just leaving in this code until we make up our collective minds on if
// selection should always be enabled on tree views.
//
// CellWantsClick
//
// This is used in the flex table to allow the cell a chance at the click even if it is not
// supposed to select anything. This is the case when the tree is in single-click mode because
// we still want to respond to clicks even though the cell does not select.
//
Boolean
CHyperTreeFlexTable :: CellWantsClick( const STableCell & inCell ) const
{
return true;
} // CellWantsClick
//
// ClickCell
//
// If we're in single-click mode, do an open.
//
void
CHyperTreeFlexTable :: ClickCell ( const STableCell & inCell, const SMouseDownEvent & inMouse )
{
1998-08-24 19:53:36 +00:00
if ( inCell.col == FindTitleColumnID() && ClickCountToOpen() == 1 ) {
if ( ! ::WaitMouseMoved(inMouse.macEvent.where) )
OpenRow ( inCell.row );
}
else
CStandardFlexTable::ClickCell ( inCell, inMouse );
} // ClickCell
#endif
1998-03-28 02:44:41 +00:00
//
// DoHiliteRgn
//
// Override to not use the hilite color, since that would be confusing with in-place editing.
// This is called when the user clicks on a row in the table to hilite that particular row.
// The routine below is called when updating the window.
//
void
CHyperTreeFlexTable :: DoHiliteRgn ( RgnHandle inHiliteRgn ) const
{
1998-03-28 02:44:41 +00:00
::InvertRgn ( inHiliteRgn );
} // DoHiliteRgn
//
// DoHiliteRect
//
// Override not to use hilite color, since it doesn't draw very well over the gray background.
//
void
CHyperTreeFlexTable :: DoHiliteRect ( const Rect & inHiliteRect ) const
{
::InvertRect ( &inHiliteRect );
} // DoHiliteRect
//
// HiliteSelection
//
// Overload the LTableView class' version of this routine to hilite the selected items
// without setting the hilite bit before drawing. This is called when the window is updating
// to redraw the selected items.
//
void
CHyperTreeFlexTable :: HiliteSelection( Boolean inActively, Boolean /*inHilite*/)
{
if (FocusExposed()) {
StRegion hiliteRgn;
1998-03-28 02:44:41 +00:00
GetHiliteRgn(hiliteRgn);
StColorPenState saveColorPen; // Preserve color & pen state
1998-03-28 02:44:41 +00:00
StColorPenState::Normalize();
if (inActively)
::InvertRgn(hiliteRgn);
else {
1998-03-28 02:44:41 +00:00
::PenMode(srcXor);
::FrameRgn(hiliteRgn);
}
}
} // HiliteSelection
//
// SyncSelectionWithHT
//
// The selection has somehow changed by the backend's decree. Make sure that we are in
// sync with that.
//
void
CHyperTreeFlexTable :: SyncSelectionWithHT ( )
{
CHyperTreeSelector* sel = dynamic_cast<CHyperTreeSelector*>(GetTableSelector());
Assert_( sel != NULL);
sel->SyncSelectorWithHT();
SelectionChanged();
ScrollSelectionIntoFrame();
} // SyncSelectionWithHT
//
// ImageIsReady
//
// Called when the bg image is done loading. Just refresh and we'll pick it up.
//
void
CHyperTreeFlexTable :: ImageIsReady ( )
{
Refresh();
} // ListenToMessage
//
// DrawStandby
//
// Draw something reasonable when the background image has not yet been loaded or if the bg image
// cannot be found
//
void
CHyperTreeFlexTable :: DrawStandby ( const Point & /*inTopLeft*/, const IconTransformType /*inTransform*/ ) const
{
EraseTableBackground();
} // DrawStandby
//
// EraseTableBackground
//
// Erase the background of the table with the specified color from HT or with the appropriate
// AppearanceManager colors
//
void
CHyperTreeFlexTable :: EraseTableBackground ( ) const
{
// draw the unsorted column color all the way down. The sort column can redraw it cell by cell
size_t viewHeight = max(mImageSize.height, static_cast<Int32>(mFrameSize.height));
Rect backRect = { 0, 0, viewHeight, mImageSize.width };
URDFUtilities::SetupBackgroundColor ( TopNode(), gNavCenter->viewBGColor, kThemeListViewBackgroundBrush );
::EraseRect(&backRect);
} // EraseTableBackground
void
CHyperTreeFlexTable :: DrawIconsSelf( const STableCell& inCell, IconTransformType inTransformType,
const Rect& inIconRect) const
{
HT_Resource cellNode = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inCell.row) );
// draw the gif if specified, otherwise draw the normal way.
char* url = NULL;
PRBool success = HT_GetNodeData ( cellNode, gNavCenter->RDF_smallIcon, HT_COLUMN_STRING, &url );
if ( success && url ) {
// extract the image drawing class out of the HT node. If one is not there yet,
// create it.
CTreeIcon* icon = static_cast<CTreeIcon*>(HT_GetNodeFEData(cellNode));
if ( !icon ) {
icon = new CTreeIcon(url, const_cast<CHyperTreeFlexTable*>(this), cellNode);
if ( !icon )
return;
HT_SetNodeFEData(cellNode, icon);
}
// setup where we should draw
Point topLeft;
topLeft.h = inIconRect.left; topLeft.v = inIconRect.top;
uint16 width = inIconRect.right - inIconRect.left;
uint16 height = inIconRect.bottom - inIconRect.top;
// draw
icon->SetImageURL ( url );
icon->DrawImage ( topLeft, kTransformNone, width, height );
}
else {
// use the "open state" transform if container is open, but only if the triggers are hidden.
// The Finder doesn't use the open state in list view because it uses the triangles
// to show that. However, if they aren't being shown, we need another way to indicate
// an open container to the user.
if ( HT_IsContainerOpen(cellNode) &&
URDFUtilities::PropertyValueBool( TopNode(), gNavCenter->showTreeConnections, true) == false )
inTransformType |= kTransformOpen;
CStandardFlexTable::DrawIconsSelf(inCell, inTransformType, inIconRect);
}
} // DrawIconsSelf
//
// DrawSelf
//
// Overridden to draw the background image, if one is present on the current view
//
void
CHyperTreeFlexTable :: DrawSelf ( )
{
mHasBackgroundImage = false;
Point topLeft = { 0, 0 };
HT_Resource topNode = TopNode();
size_t viewHeight = max(mImageSize.height, static_cast<Int32>(mFrameSize.height));
if ( topNode ) {
char* url = NULL;
PRBool success = HT_GetNodeData ( topNode, gNavCenter->viewBGURL, HT_COLUMN_STRING, &url );
if ( success && url ) {
// draw the background image tiled to fill the whole pane
mHasBackgroundImage = true;
SetImageURL ( url );
DrawImage ( topLeft, kTransformNone, mImageSize.width, viewHeight );
FocusDraw();
}
else
EraseTableBackground();
CStandardFlexTable::DrawSelf();
}
} // DrawSelf
//
// RedrawRow
//
// Redraw the row corresponding to the given HT node
//
void
1998-09-01 20:17:46 +00:00
CHyperTreeFlexTable :: RedrawRow ( HT_Resource inNode ) const
{
TableIndexT row = URDFUtilities::HTRowToPPRow(HT_GetNodeIndex(HT_GetView(inNode), inNode));
1998-08-24 19:53:36 +00:00
RefreshRowRange( row, row );
} // RedrawRow
//
// EraseCellBackground
//
// Make the backdrop of the cell look like the Finder in OS8 or whatever the user has explicitly
// set via HT properties. If there is a background image, of course, don't draw the background. The
// default list background has already been painted, so only do something if there is a difference.
//
void
CHyperTreeFlexTable :: EraseCellBackground ( const STableCell& inCell, const Rect& inLocalRect )
{
StColorPenState saved;
if ( !mHasBackgroundImage )
{
PaneIDT columnPane;
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header != NULL);
// only need to draw if this column in sorted
if ( inCell.col == header->GetSortedColumn(columnPane) ) {
Rect backRect = inLocalRect;
backRect.bottom--; // leave a one pixel line on the bottom as separator
backRect.right++; // cover up vertical dividing line on right side
URDFUtilities::SetupBackgroundColor ( TopNode(), gNavCenter->sortColumnBGColor,
kThemeListViewSortColumnBackgroundBrush );
::EraseRect(&backRect);
} // if this column is sorted
} // if no bg image
// draw the separator line if HT says to. This will draw it even if there is a bg image.
if ( URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->showDivider, true) ) {
URDFUtilities::SetupBackgroundColor ( TopNode(), gNavCenter->dividerColor,
kThemeListViewSeparatorBrush );
Rect divider = { inLocalRect.bottom - 1, inLocalRect.left, inLocalRect.bottom, inLocalRect.right };
::EraseRect ( &divider );
}
} // EraseCellBackground
1998-03-28 02:44:41 +00:00
//
// DrawCellContents
//
// Draw what goes inside each cell, naturally
//
void
CHyperTreeFlexTable::DrawCellContents( const STableCell& inCell, const Rect& inLocalRect)
1998-03-28 02:44:41 +00:00
{
PaneIDT cellType = GetCellDataType(inCell);
// Get info for column
PaneIDT columnPane;
1998-03-28 02:44:41 +00:00
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header != NULL);
CHyperTreeHeader::ColumnInfo info = header->GetColumnInfo(inCell.col - 1);
1998-03-28 02:44:41 +00:00
// Get cell data
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inCell.row) );
if (node) {
1998-03-28 02:44:41 +00:00
if ( HT_IsSeparator(node) ) {
1998-03-28 02:44:41 +00:00
Uint16 left = inLocalRect.left;
if ( inCell.col == FindTitleColumnID() ) {
left = DrawIcons(inCell, inLocalRect);
left += CStandardFlexTable::kDistanceFromIconToText;
1998-03-28 02:44:41 +00:00
}
// setup the color based on if this cell is in the sorted column. Note that while
// HT has the concept of a different fg color for sorted columns, AM does not.
StColorPenState saved;
if ( inCell.col == header->GetSortedColumn(columnPane) )
URDFUtilities::SetupForegroundColor ( TopNode(), gNavCenter->sortColumnFGColor,
kThemeListViewTextColor );
else
URDFUtilities::SetupForegroundColor ( TopNode(), gNavCenter->viewFGColor,
kThemeListViewTextColor );
1998-03-28 02:44:41 +00:00
::MoveTo ( left,
inLocalRect.top + ((inLocalRect.bottom - inLocalRect.top) / 2) );
::PenSize ( 2, 2 );
::PenPat ( &qd.gray );
::Line ( inLocalRect.right - left, 0 );
} // if is a separator
1998-03-28 02:44:41 +00:00
else {
void* data;
char* str;
Rect localRect = inLocalRect;
if (HT_GetNodeData(node, info.token, info.tokenType, &data) && data)
1998-03-28 02:44:41 +00:00
{
switch (info.tokenType) {
case HT_COLUMN_STRING:
case HT_COLUMN_DATE_STRING:
str = static_cast<char*>(data);
if (inCell.col == FindTitleColumnID())
{
localRect.left = DrawIcons(inCell, localRect);
localRect.left += CStandardFlexTable::kDistanceFromIconToText;
}
break;
case HT_COLUMN_DATE_INT:
case HT_COLUMN_INT:
char intStr[32];
sprintf(intStr, "%d", (int)data);
str = intStr;
break;
}
// setup the text color based on if this cell is in the sorted column. Note that while
// HT has the concept of a different fg color for sorted columns, AM does not.
StColorPenState saved;
if ( inCell.col == header->GetSortedColumn(columnPane) )
URDFUtilities::SetupForegroundTextColor ( TopNode(), gNavCenter->sortColumnFGColor,
kThemeListViewTextColor );
else
URDFUtilities::SetupForegroundTextColor ( TopNode(), gNavCenter->viewFGColor,
kThemeListViewTextColor );
DrawTextString(str, mTextDrawingStuff, 0, localRect);
1998-03-28 02:44:41 +00:00
}
} // else a normal item
} // if node valid
} // DrawCellContents
1998-03-28 02:44:41 +00:00
Boolean CHyperTreeFlexTable::CellHasDropFlag(const STableCell& inCell, Boolean& outIsExpanded) const
{
// bail quickly if this cell isn't a title column or tree connections are turned off
if ( FindTitleColumnID() != inCell.col ||
URDFUtilities::PropertyValueBool( TopNode(), gNavCenter->showTreeConnections, true) == false )
1998-03-28 02:44:41 +00:00
return false;
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inCell.row) );
if (node)
{
PRBool result = HT_IsContainer(node);
if (result)
{
PRBool openState;
HT_Error err = HT_GetOpenState(node, &openState);
if (err == HT_NoErr)
outIsExpanded = openState;
}
else
outIsExpanded = false;
return result;
}
return false;
} // CellHasDropFlag
UInt16 CHyperTreeFlexTable::GetNestedLevel(TableIndexT inRow) const
{
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inRow) );
if (node)
{
// subtract one because root node as indentation of 1
return (HT_GetItemIndentation(node) - 1);
}
return CStandardFlexTable::GetNestedLevel(inRow);
}
//
// SetCellExpansion
//
// User clicked on the disclosure triangle, tell HT to open/close the container. This call
// may fail if the user has password protected the folder and they get the password wrong.
// If that is the case, redraw the drop flag to its correct state.
//
void CHyperTreeFlexTable::SetCellExpansion(
const STableCell& inCell,
Boolean inExpand)
{
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inCell.row) );
if (node)
{
if ( HT_SetOpenState(node, (PRBool)inExpand) != noErr )
RefreshCell(inCell);
}
}
1998-08-26 19:04:54 +00:00
//
// SetupColumns
//
// Build columns from HT. Simply pitches existing columns and rebuilds from scratch.
//
void
CHyperTreeFlexTable :: SetupColumns ( )
{
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
if ( !header )
return;
header->SetUpColumns( GetHTView() );
SynchronizeColumnsWithHeader();
} // SetupColumns
1998-03-28 02:44:41 +00:00
// ------------------------------------------------------------
// HT Operations
// ------------------------------------------------------------
//
// OpenView
//
// opens up the given view in the navCenter (such as bookmarks). Make sure the selector for this
// table knows about new view, sync the number of rows in the table with the
// number of rows in the HT and setup the columns.
//
// This routine is called as a result of the HT notification mechanism (not directly from any
// FE events like clicks). That means that the shelf may not be open yet.
//
void
CHyperTreeFlexTable::OpenView( HT_View inHTView )
{
// Hide the inline editor. This will send us a message that the name changed and attempt
// to redraw the entire window. We don't want this because is causes an ugly flash as it
// redraws the old view then draws the new view. Setting the visRgn to empty prevents
// the redraw of the old view.
if ( mNameEditor && mNameEditor->IsVisible() ) {
StVisRgn saved(GetMacPort());
mNameEditor->UpdateEdit(nil, nil, nil);
}
// save existing column information for the next time the user comes back to this
// view. Note we do not reflect any of this into HT, so it will be lost when the user
// quits or opens a new window (I think). We currently save which columns are visible,
// their widths, and sort state.
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header != NULL);
TableIndexT visColumns = header->CountVisibleColumns();
if ( visColumns ) {
//<2F><><EFBFBD> implement this. Recall that currently the FE data is used to store the
//<2F><><EFBFBD> SelectorData stuff. We'll have to create a new structure that wraps both
//<2F><><EFBFBD> this info and that.
}
1998-03-28 02:44:41 +00:00
mHTView = mViewBeforeDrag = inHTView;
CHyperTreeSelector* sel = dynamic_cast<CHyperTreeSelector*>(GetTableSelector());
Assert_( sel != NULL);
sel->TreeView ( inHTView );
// Rebuild the column headers based on the new view contents and which columns the
// user had visible last time.
1998-08-26 19:04:54 +00:00
SetupColumns();
1998-03-28 02:44:41 +00:00
1998-08-26 19:04:54 +00:00
//<2F><><EFBFBD>hack until rjc gets the columns to remember visibility across pane deletions
//<2F><><EFBFBD>this should not go in the shipping product! (pinkerton)
header->SetRightmostVisibleColumn(1);
1998-03-28 02:44:41 +00:00
Uint32 count = HT_GetItemListCount(inHTView);
if (mRows != count)
{
if (count > mRows)
InsertRows(count - mRows, 0, NULL, 0, false);
else
RemoveRows(mRows - count, 0, true);
}
SyncSelectionWithHT();
// redraw the table. If the mouse is down, we're probably in the middle of a
// drag and drop so we want to redraw NOW (not when we get an update event after
// the mouse has been released).
if ( ::StillDown() ) {
UnselectAllCells();
FocusDraw();
Rect localRect;
CalcLocalFrameRect(localRect);
localRect.left++;
::EraseRect(&localRect);
GetSuperView()->Draw(nil); // redraw NOW to get the # of visible rows correct
}
else
Refresh();
1998-08-24 19:53:36 +00:00
// make sure the number of clicks to open a row is correct
if ( URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->useSingleClick, false) == true )
mClickCountToOpen = 1;
1998-03-28 02:44:41 +00:00
}
void CHyperTreeFlexTable::ExpandNode(HT_Resource inHTNode)
{
Int32 index = HT_GetNodeIndex(GetHTView(), inHTNode);
Uint32 count = HT_GetCountVisibleChildren(inHTNode);
if (count > 0)
InsertRows(count, index + 1, NULL, 0, true);
}
void CHyperTreeFlexTable::CollapseNode(HT_Resource inHTNode)
{
Int32 index = HT_GetNodeIndex(GetHTView(), inHTNode);
Uint32 count = HT_GetCountVisibleChildren(inHTNode);
if (count > 0)
RemoveRows(count, index + 2, true);
}
//-----------------------------------
// Commands
//-----------------------------------
//
// FindCommandStatus
//
void
CHyperTreeFlexTable :: FindCommandStatus ( CommandT inCommand, Boolean &outEnabled,
Boolean &outUsesMark, Char16 &outMark, Str255 outName)
{
// safety check
if ( ! mHTView )
return;
HT_Pane thePane = HT_GetPane(mHTView);
outUsesMark = false;
if ( inCommand >= cmd_NavCenterBase && inCommand <= cmd_NavCenterCap ) {
outEnabled = HT_IsMenuCmdEnabled(thePane, (HT_MenuCmd)(inCommand - cmd_NavCenterBase));
return;
}
// process the regular menus...
switch ( inCommand ) {
//
// handle commands that we have to share with other parts of the UI
//
case cmd_Cut:
outEnabled = HT_IsMenuCmdEnabled(thePane, HT_CMD_CUT );
break;
case cmd_Copy:
outEnabled = HT_IsMenuCmdEnabled(thePane, HT_CMD_COPY );
break;
case cmd_Paste:
outEnabled = HT_IsMenuCmdEnabled(thePane, HT_CMD_PASTE );
break;
case cmd_Clear:
outEnabled = HT_IsMenuCmdEnabled(thePane, HT_CMD_DELETE_FILE );
break;
default:
LCommander::FindCommandStatus(inCommand, outEnabled, outUsesMark, outMark, outName);
break;
} // case of command
} // FindCommandStatus
//
// DeleteSelection
//
// Delete the selection in the given view. This is called when the user hits backspace
// in the current view, so we don't need to worry (as below) about the view switching
// during a drag and drop.
//
void
CHyperTreeFlexTable::DeleteSelection ( const EventRecord& /*inEvent*/ )
1998-03-28 02:44:41 +00:00
{
HT_Pane pane = HT_GetPane(mHTView);
if ( HT_IsMenuCmdEnabled(pane, HT_CMD_CUT) ) //<2F><><EFBFBD> these should be HT_CMD_CLEAR
HT_DoMenuCmd ( pane, HT_CMD_CUT );
} // DeleteSelection
//
// DeleteSelectionByDragToTrash
//
// Called when the user drags some stuff in the pane into the trash can to delete
// it. We don't rely on the current view or the PowerPlant selection because it
// may have been wiped out by a temporary drag over another workspace pane on
// the way to the trash.
//
void
CHyperTreeFlexTable :: DeleteSelectionByDragToTrash ( LArray & inItems )
{
//<2F><><EFBFBD> return if we're not allowed to delete these objects!!!
CIconTextSuite* curr = NULL;
LArrayIterator it ( inItems );
while ( it.Next(&curr) ) {
HT_Resource node = curr->GetHTNodeData();
HT_Resource parent = HT_GetParent(node);
Assert_( node != NULL && parent != NULL );
HT_RemoveChild ( parent, node );
}
} // DeleteSelectionByDragToTrash
//
// OpenRow
//
// Called to tell us when the user has clicked on a row in such a way that it should
// open. Whether this is by single click or double click is left up to parameters to
// the CStandardFlexTable, however if triggers (drop flags) are hidden and the click
// is on a container, open the container.
1998-03-28 02:44:41 +00:00
//
void
CHyperTreeFlexTable :: OpenRow ( TableIndexT inRow )
{
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inRow) );
if (node) {
if ( !HT_IsContainer(node) ) {
// click is not in a container. If HT doesn't want it (and it's not
// a separator), launch the url
if ( !HT_IsSeparator(node) && !URDFUtilities::LaunchNode(node) )
1998-08-26 19:04:54 +00:00
CFrontApp::DoGetURL( HT_GetNodeURL(node), NULL, GetTargetFrame() );
}
else {
// we are a container. If tree connections hidden, open up the folder otherwise
// ignore.
if ( URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->showTreeConnections, true) == false ) {
PRBool openState = false;
HT_GetOpenState(node, &openState);
1998-08-26 19:04:54 +00:00
SetCellExpansion(STableCell(inRow, FindTitleColumnID()), !openState);
}
// redraw the container to get the new icon state
RefreshRowRange( inRow, inRow );
}
1998-03-28 02:44:41 +00:00
} // if valid node
} // OpenRow
//
// RowCanAcceptDrop
//
// Check if the given row can accept any of the items dropped on it. We iterate over each item
// being dropped and ask the backend if that item is ok. If any item can be dropped, the entire
// lot can be dropped.
//
Boolean
CHyperTreeFlexTable::RowCanAcceptDrop ( DragReference inDragRef, TableIndexT inDropRow )
{
if ( inDropRow > mRows )
return false;
HT_Resource targetNode = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inDropRow) );
return NodeCanAcceptDrop ( inDragRef, targetNode );
} // CStandardFlexTable::RowCanAcceptDrop
//
// RowCanAcceptDropBetweenAbove
//
// Check if the item can be dropped above the current node. This means we need to ask the parent
// of the node for this row if it can be dropped on.
//
Boolean
CHyperTreeFlexTable::RowCanAcceptDropBetweenAbove( DragReference inDragRef, TableIndexT inDropRow )
{
if ( inDropRow > mRows )
return NodeCanAcceptDrop ( inDragRef, TopNode() );
1998-03-28 02:44:41 +00:00
HT_Resource rowNode = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inDropRow));
HT_Resource targetNode = HT_GetParent(rowNode);
1998-03-28 02:44:41 +00:00
return NodeCanAcceptDrop ( inDragRef, targetNode );
} // CStandardFlexTable::RowCanAcceptDropBetweenAbove
//
// RowIsContainer
//
// Is this row a folder or a leaf? HT will tell us!
//
Boolean
CHyperTreeFlexTable :: RowIsContainer ( const TableIndexT & inRow ) const
{
if ( inRow > mRows )
return false;
return HT_IsContainer ( HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inRow)) );
} // RowIsContainer
//
// RowIsContainer
//
// Same as above, but also tells us if the container is open.
//
Boolean
CHyperTreeFlexTable :: RowIsContainer( const TableIndexT & inRow, Boolean* outIsExpanded ) const
{
Assert_(outIsExpanded != NULL);
if ( outIsExpanded )
*outIsExpanded = false;
else
return false;
if ( RowIsContainer(inRow) ) {
HT_Resource node = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(inRow) );
if ( node ) {
PRBool openState;
if ( HT_GetOpenState(node, &openState) == HT_NoErr )
*outIsExpanded = openState;
}
return true;
}
else
return false;
} // RowIsContainer
1998-03-28 02:44:41 +00:00
//
// HiliteDropRow
//
// Override to handle drawing the "insertion bar" for every row to always indicate to the
// user exactly where the drop will occur. The inherited version of this class only draws
// the insertion bar when the mouse is between two cells, and I really don't like that.
//
void
CHyperTreeFlexTable :: HiliteDropRow ( TableIndexT inRow, Boolean inDrawBarAbove )
{
if (inRow == LArray::index_Bad)
return;
StColorState savedPen;
StColorState::Normalize();
1998-03-28 02:44:41 +00:00
STableCell cell(inRow, GetHiliteColumn());
if ( mIsInternalDrop && CellIsSelected(cell) )
return; // Don't show hiliting feedback for a drag when over the selection.
Rect insertionBar, cellRect;
GetLocalCellRect(cell, cellRect);
insertionBar = cellRect;
// when the mouse is over a cell, there are several possibilities:
// - if the cell is a container, hilite the container and draw the insertion bar
// below the last visible child of the container. The left edge should be indented
// from the left edge of the container.
// - if the cell is a normal row, draw the bar directly above the row at
// the same horizontal indent as the row.
// - if the cell is a container but the mouse is between rows, treat it like
// a normal row and draw the bar directly above at the same level and do not
// hilite the container.
// - if the mouse is off the end, draw the bar directly below the last row
// at the same indent as the 1st row (since they should be siblings).
// The only problem with the behavior occurs when you want to drop something
// directly at the end of an open container. Moving the mouse to where you think
// would do it ends up dropping the item after the parent container at the
// same level as the parent, not as the last item in the container. As a result,
// the only way to add something to the end of a container is to drop it on the
// container itself. The drag feedback illustrates that this will happen when you
// drag on top of a container.
const Uint32 kInsertionBarLength = 35;
Rect iconRect;
GetIconRect ( cell, insertionBar, iconRect );
mDropNode = HT_GetNthItem( GetHTView(), URDFUtilities::PPRowToHTRow(cell.row) );
if ( mDropNode && HT_IsContainer(mDropNode) && !inDrawBarAbove ) {
// place the insertion bar below the last visible row at the appropriate
// indent level for its children
Uint32 children = HT_GetCountVisibleChildren(mDropNode);
Uint32 rowHeight = GetRowHeight(inRow);
insertionBar.bottom = insertionBar.top + rowHeight + 1;
insertionBar.bottom += (children * GetRowHeight(inRow));
insertionBar.left = iconRect.left + kIndentPerLevel; // one indentation
insertionBar.right = insertionBar.left + kInsertionBarLength;
insertionBar.top = insertionBar.bottom - 2;
// hilite the container
DrawIcons(cell, cellRect);
StRegion hiliteRgn;
GetRowHiliteRgn(inRow, hiliteRgn);
::InvertRgn(hiliteRgn);
} // if drop on container
else {
// setup insertion bar rectangle. When we're dropping at the end of the list, use
// the indent of the 1st item in the list, as they will be siblings.
if ( inRow <= mRows ) {
insertionBar.left = iconRect.left - 5;
insertionBar.right = insertionBar.left + kInsertionBarLength;
insertionBar.bottom = insertionBar.top + 1;
insertionBar.top -= 1;
}
else {
// compute the icon rect for the 1st row
Rect iconRect, cellRect;
STableCell firstCell(1, GetHiliteColumn());
GetLocalCellRect ( firstCell, cellRect );
GetIconRect ( firstCell, cellRect, iconRect );
// use the last row to compute where the bar should go vertically
Rect lastRowRect;
GetLocalCellRect ( STableCell(mRows, GetHiliteColumn()), lastRowRect );
insertionBar.left = iconRect.left;
insertionBar.right = insertionBar.left + kInsertionBarLength;
insertionBar.top = lastRowRect.bottom + 1;
insertionBar.bottom = insertionBar.top + 2;
}
} // else drop on item or between rows
DoHiliteRect ( insertionBar );
} // HiliteDropRow
//
// ItemIsAcceptable
//
// Determine if the current item can be dropped in this pane. Just always say "yes" for now
// if the flavor is acceptable to us. It will be verified for real in RowCanAcceptDrop*() routines.
1998-03-28 02:44:41 +00:00
//
Boolean
CHyperTreeFlexTable :: ItemIsAcceptable ( DragReference inDragRef, ItemReference inItemRef )
1998-03-28 02:44:41 +00:00
{
FlavorType ignored;
bool acceptableFlavorFound = FindBestFlavor ( inDragRef, inItemRef, ignored );
return acceptableFlavorFound;
1998-03-28 02:44:41 +00:00
} // ItemIsAcceptable
//
// DragSelection
//
// Overridden to use our own version of the drag task...
//
1998-06-23 01:36:59 +00:00
OSErr
1998-03-28 02:44:41 +00:00
CHyperTreeFlexTable :: DragSelection(
const STableCell& inCell,
const SMouseDownEvent& inMouseDown )
{
FocusDraw();
mViewBeforeDrag = GetHTView();
Assert_(mViewBeforeDrag != NULL);
// add the members of the selection to a list we will pass to the drag task.
LArray selection;
STableCell cell(0, 1);
while (GetNextSelectedRow(cell.row)) {
Rect bounds, iconBounds;
GetLocalCellRect(cell, bounds);
GetIconRect ( cell, bounds, iconBounds );
bounds.left = iconBounds.left;
HT_Resource node = HT_GetNthItem(mViewBeforeDrag, URDFUtilities::PPRowToHTRow(cell.row) );
string finalCaption = CURLDragHelper::MakeIconTextValid ( HT_GetNodeName(node) );
1998-03-28 02:44:41 +00:00
CIconTextSuite* suite = new CIconTextSuite( this, bounds, GetIconID(cell.row), finalCaption, node );
selection.InsertItemsAt ( 1, LArray::index_Last, &suite );
}
// There is a problem with the HT backend that if you tell it to do something with
// both the parent and a child of that parent (say the container is open and the user
// grabs both of them), it will die. Iterate over each item, and for each node, if
1998-03-28 02:44:41 +00:00
// we find any children in the list, remove them. LArrayIterator should be
// smart enough to handle deletions while iterating....
LArrayIterator it(selection);
CIconTextSuite* curr = NULL;
while ( it.Next(&curr) ) {
HT_Resource currentNode = curr->GetHTNodeData();
CIconTextSuite* possibleChildSuite = NULL;
LArrayIterator inner(selection);
Uint32 counter = 1; // LArray counts from 1
while ( inner.Next(&possibleChildSuite) ) {
HT_Resource possibleChild = possibleChildSuite->GetHTNodeData();
if ( HT_GetParent(possibleChild) == currentNode )
selection.RemoveItemsAt(1, counter); // child found, delete it
else
counter++;
}
}
// create the drag task
Rect cellBoundsOfClick;
GetLocalCellRect(inCell, cellBoundsOfClick);
CIconTextDragTask theTask(inMouseDown.macEvent, selection, cellBoundsOfClick);
// setup our special data transfer proc called upon drag completion
OSErr theErr = ::SetDragSendProc ( theTask.GetDragReference(), mSendDataUPP, (LDropArea*) this );
ThrowIfOSErr_(theErr);
theTask.DoDrag();
1998-03-28 02:44:41 +00:00
// remove the items if they went into the trash
if ( theTask.DropLocationIsFinderTrash() )
DeleteSelectionByDragToTrash(selection);
UnselectAllCells();
// cleanup after ourselves
LArrayIterator it2(selection);
curr = NULL;
while ( it.Next(&curr) )
delete curr;
mViewBeforeDrag = GetHTView();
1998-06-23 01:36:59 +00:00
return noErr;
1998-03-28 02:44:41 +00:00
} // DragSelection
//
// DoDragSendData
//
// Called when this window is the originator of the drag, after the drop has succeeded. Since we
// were the ones who created the drag task, we know that one of the flavors is emHTNodeDrag, which
// contains as its data the HT_Resource representing this drag item. Use that resource to get
// the url and title.
//
void
CHyperTreeFlexTable :: DoDragSendData( FlavorType inFlavor, ItemReference inItemRef,
DragReference inDragRef)
{
CNetscapeWindow* theWindow = dynamic_cast<CNetscapeWindow*>(LWindow::FetchWindowObject(GetMacPort()));
ThrowIfNil_(theWindow);
HT_Resource node = NULL;
Size dataSize = sizeof(HT_Resource);
OSErr err = ::GetFlavorData(inDragRef, inItemRef, emHTNodeDrag, &node, &dataSize, 0);
if ( node && !err )
CURLDragHelper::DoDragSendData ( HT_GetNodeURL(node), HT_GetNodeName(node),
inFlavor, inItemRef, inDragRef );
else
DebugStr("\pFlavor data does not contain a valid HT_Resource structure");
} // DoDragSendData
//
// ReceiveDragItem
//
// Pass this along to the implementation in CURLDragMixin.
1998-03-28 02:44:41 +00:00
//
void
CHyperTreeFlexTable :: ReceiveDragItem ( DragReference inDragRef, DragAttributes inDragAttrs,
ItemReference inItemRef, Rect & inItemBounds )
1998-03-28 02:44:41 +00:00
{
CHTAwareURLDragMixin::ReceiveDragItem(inDragRef, inDragAttrs, inItemRef, inItemBounds );
1998-03-28 02:44:41 +00:00
} // ReceiveDragItem
//
// HandleDropOfPageProxy
//
// Called when the user drops the page proxy icon in the NavCenter. The page proxy flavor data
// consists of the url and title, separated by a return. Extract the components and create a
// new node.
//
void
CHyperTreeFlexTable :: HandleDropOfPageProxy ( const char* inURL, const char* inTitle )
1998-03-28 02:44:41 +00:00
{
// cast away constness for HT
char* url = const_cast<char*>(inURL);
char* title = const_cast<char*>(inTitle);
1998-03-28 02:44:41 +00:00
// Extract the node where the drop will occur. If the drop is on a container, put
// it in the container, unless the mouse is actually between the rows which means the
// user wants to put it at the same level as the container. Otherwise just put it
// above (and at the same level) as the item it is dropped on.
bool dropAtEnd = mDropNode == NULL;
if ( dropAtEnd ) {
mDropNode = HT_GetNthItem( GetHTView(), URDFUtilities::PPRowToHTRow(mRows) );
if ( !mDropNode ) {
// the view is empty, do drop on and bail
HT_DropURLAndTitleOn ( TopNode(), url, title );
return;
1998-03-28 02:44:41 +00:00
}
}
if ( HT_IsContainer(mDropNode) && !mIsDropBetweenRows )
HT_DropURLAndTitleOn ( mDropNode, url, title );
else
HT_DropURLAndTitleAtPos ( mDropNode, url, title, dropAtEnd ? PR_FALSE : PR_TRUE );
1998-03-28 02:44:41 +00:00
} // HandleDropOfPageProxy
//
// HandleDropOfHTResource
//
// Called when the user drops something from one RDF-savvy location (navcenter, personal toolbar)
// to another. This may result in a move or a copy, so do the right thing for each.
//
void
CHyperTreeFlexTable :: HandleDropOfHTResource ( HT_Resource dropNode )
{
// Extract the node where the drop will occur. If the drop is on a container, put
// it in the container, unless the mouse is actually between the rows which means the
// user wants to put it at the same level as the container. Otherwise just put it
// above (and at the same level) as the item it is dropped on.
bool dropAtEnd = mDropNode == NULL;
if ( dropAtEnd ) {
mDropNode = HT_GetNthItem( GetHTView(), URDFUtilities::PPRowToHTRow(mRows) );
if ( !mDropNode ) {
// the view is empty, do drop on and bail
HT_DropHTROn ( TopNode(), dropNode );
1998-03-28 02:44:41 +00:00
return;
}
} // if we're dropping at end
if ( HT_IsContainer(mDropNode) && !mIsDropBetweenRows )
HT_DropHTROn ( mDropNode, dropNode );
else
HT_DropHTRAtPos ( mDropNode, dropNode, dropAtEnd ? PR_FALSE : PR_TRUE );
} // HandleDropOfHTResource
//
// HandleDropOfLocalFile
//
// Called when the user drops something from the Finder into the navCenter. Create a
// bookmark with a file URL for that file. This assumes aliases have already been
// resolved
//
void
CHyperTreeFlexTable :: HandleDropOfLocalFile ( const char* inFileURL, const char* inFileName,
const HFSFlavor & /*inFileData*/ )
1998-03-28 02:44:41 +00:00
{
// Extract the node where the drop will occur. If the drop is on a container, put
// it in the container, unless the mouse is actually between the rows which means the
// user wants to put it at the same level as the container. Otherwise just put it
// above (and at the same level) as the item it is dropped on.
bool dropAtEnd = mDropNode == NULL;
if ( dropAtEnd ) {
mDropNode = HT_GetNthItem( GetHTView(), URDFUtilities::PPRowToHTRow(mRows) );
if ( !mDropNode ) {
// the view is empty, do drop on and bail
HT_DropURLAndTitleOn ( TopNode(),
1998-03-28 02:44:41 +00:00
const_cast<char*>(inFileURL), const_cast<char*>(inFileName) );
return;
}
}
if ( HT_IsContainer(mDropNode) && !mIsDropBetweenRows )
HT_DropURLAndTitleOn ( mDropNode, const_cast<char*>(inFileURL), const_cast<char*>(inFileName) );
else
HT_DropURLAndTitleAtPos ( mDropNode, const_cast<char*>(inFileURL), const_cast<char*>(inFileName),
dropAtEnd ? PR_FALSE : PR_TRUE );
} // HandleDropOfLocalFile
//
// HandleDropOfText
1998-03-28 02:44:41 +00:00
//
// Called when user drops a text clipping onto the navCenter. Do nothing for now.
1998-03-28 02:44:41 +00:00
//
void
CHyperTreeFlexTable :: HandleDropOfText ( const char* /*inTextData*/ )
1998-03-28 02:44:41 +00:00
{
DebugStr("\pDropping TEXT here not implemented");
1998-03-28 02:44:41 +00:00
} // HandleDropOfText
1998-03-28 02:44:41 +00:00
//
// InlineEditorDone
//
// Called when the user hits return/enter or clicks outside of the inline editor. Tell RDF about the
// change and update the table.
//
void
CHyperTreeFlexTable :: InlineEditorDone ( )
{
Str255 newName;
mNameEditor->GetDescriptor(newName);
// we need to convert to UTF8 here....
unsigned char* utfText = INTL_ConvertLineWithoutAutoDetect(
INTL_GetCharSetID(INTL_DefaultTextWidgetCsidSel), CS_UTF8, newName+1, newName[0]);
cstring nameAsCString( (char*) utfText);
XP_FREEIF(utfText);
1998-03-28 02:44:41 +00:00
HT_Resource editedNode = HT_GetNthItem(GetHTView(), URDFUtilities::PPRowToHTRow(mRowBeingEdited) );
HT_SetNodeName ( editedNode, nameAsCString );
} // InlineEditorDone
//
// CanDoInlineEditing
//
// While we normally want to be able to do inline editing, we have to turn it off for panes
// that don't allow editing (like History). HT also has the chance to turn it off for the
// current view. Assumes mRowBeingEdited is correctly set.
1998-03-28 02:44:41 +00:00
//
1998-06-23 01:36:59 +00:00
Boolean
CHyperTreeFlexTable :: CanDoInlineEditing ( ) const
1998-03-28 02:44:41 +00:00
{
if ( URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->useInlineEditing, true) ) {
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header);
CHyperTreeHeader::ColumnInfo info = header->GetColumnInfo ( FindTitleColumnID() - 1 );
HT_Resource item = HT_GetNthItem( GetHTView(), URDFUtilities::PPRowToHTRow(mRowBeingEdited) );
return HT_IsNodeDataEditable ( item, info.token, info.tokenType );
}
1998-03-28 02:44:41 +00:00
return false;
1998-03-28 02:44:41 +00:00
} // CanDoInlineEditing
//
// TableDesiresSelectionTracking
//
// Disable marquee selection if HT doesn't want it
//
Boolean
CHyperTreeFlexTable :: TableDesiresSelectionTracking( ) const
{
#ifdef USE_SELECTION_PROP
return URDFUtilities::PropertyValueBool(TopNode(), gNavCenter->useSelection, true) == true;
#else
return true;
#endif
} // TableDesiresSelectionTracking
1998-03-28 02:44:41 +00:00
//
// ChangeSort
//
// called when the user clicks in one of the column headings to change the sort column or
// to change the sort order of the given column.
//
void
CHyperTreeFlexTable :: ChangeSort ( const LTableHeader::SortChange* inSortChange )
{
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header);
// CHyperTreeHeader::GetColumnInfo() is zero-based and PP is one-based.
if ( inSortChange->sortColumn >= 1 ) {
CHyperTreeHeader::ColumnInfo info = header->GetColumnInfo ( inSortChange->sortColumn - 1 );
HT_SetSortColumn ( GetHTView(), info.token, info.tokenType, inSortChange->reverseSort ? PR_TRUE : PR_FALSE );
}
else
HT_SetSortColumn ( GetHTView(), NULL, 0, PR_FALSE ); // return to natural order
Refresh();
} // ChangeSort
//
// AdjustCursorSelf
//
// Handle changing cursor to contextual menu cursor if cmd key is down
//
void
CHyperTreeFlexTable::AdjustCursorSelf( Point /*inPoint*/, const EventRecord& inEvent )
{
ExecuteAttachments(CContextMenuAttachment::msg_ContextMenuCursor,
static_cast<void*>(const_cast<EventRecord*>(&inEvent)));
}
//
// TableSupportsNaturalOrderSort
//
// Ask HT if this view does natural order sorting now (can the user drop in the
// middle of the table).
//
Boolean
CHyperTreeFlexTable :: TableSupportsNaturalOrderSort ( ) const
{
return HT_ContainerSupportsNaturalOrderSort(TopNode());
} // TableSupportsNaturalOrderSort
//
// SetTargetFrame
//
// Set the new target to where url's are dispatched when clicked on
//
void
CHyperTreeFlexTable :: SetTargetFrame ( const char* inFrame )
{
mTargetFrame = inFrame;
} // SetTargetFrame
const char*
CHyperTreeFlexTable :: GetTargetFrame ( ) const
{
if ( mTargetFrame.length() )
return mTargetFrame.c_str();
else
return NULL;
}
//
// CalcToolTipText
//
// Used with the CTableToolTipPane/Attachment to provide a better tooltip experience
// for the table. Will only show the tip when the text is truncated.
//
void
CHyperTreeFlexTable :: CalcToolTipText( const STableCell& inCell,
StringPtr outText,
TextDrawingStuff& outStuff,
Boolean& outTruncationOnly)
{
outText[0] = 0;
outTruncationOnly = true; // Only show tip if truncated.
// never show the tip while inline editing is alive
if ( mRowBeingEdited != LArray::index_Bad )
return;
outStuff = GetTextStyle(inCell.row);
if ( inCell.row <= mRows ) {
CHyperTreeHeader* header = dynamic_cast<CHyperTreeHeader*>(mTableHeader);
Assert_(header != NULL);
CHyperTreeHeader::ColumnInfo info = header->GetColumnInfo(inCell.col - 1);
HT_Resource node = HT_GetNthItem( GetHTView(), URDFUtilities::PPRowToHTRow(inCell.row) );
void* data;
if ( HT_GetNodeData(node, info.token, info.tokenType, &data) && data ) {
switch (info.tokenType) {
case HT_COLUMN_STRING:
if ( ! HT_IsSeparator(node) ) {
const char* str = static_cast<char*>(data);
if ( str ) {
outText[0] = strlen(str);
strcpy ( (char*) &outText[1], str );
}
else
outText[0] = 0;
}
else
outText[0] = 0;
break;
default:
outText[0] = 0; // don't display tooltip for other data types
} // case of column data type
} // if data is valid
} // if row has data
else {
// supply a useful message when the mouse is over some part of the table that
// isn't a real row. I'd like to make it display only if the pane is editable,
// but this is not possible to determine given the current HT APIs =(.
outTruncationOnly = false; // always show this message
::GetIndString ( outText, 10506, 16);
} // else row is empty
} // CalcToolTipText
1998-03-28 02:44:41 +00:00
#pragma mark -- class CHyperTreeSelector --
//
// DoSelect
//
// We need to make sure that the HTView's concept of the selection is in sync with what PP
// believes to be the current selection. Since all modifications to the selection are
// channeled through this method, we just override it to also tell HT about the selection
// change. This also picks up SelectCell() and UnselectCell() calls since the LTableRowSelector
// routes those calls here
//
void
CHyperTreeSelector :: DoSelect( const TableIndexT inRow,
Boolean inSelect,
Boolean inHilite,
Boolean inNotify )
{
HT_Resource node = HT_GetNthItem( mTreeView, URDFUtilities::PPRowToHTRow(inRow) );
if (node) {
// when we update the HT selection, we'll get an update event. We don't want
// this because we know we're updating it.
HT_NotificationMask oldMask;
HT_Pane pane = HT_GetPane ( mTreeView );
HT_GetNotificationMask ( pane, &oldMask );
HT_SetNotificationMask ( pane, 0L );
HT_SetSelectedState ( node, inSelect ? PR_TRUE : PR_FALSE );
HT_SetNotificationMask ( pane, oldMask );
}
LTableRowSelector::DoSelect(inRow, inSelect, inHilite, inNotify);
} // DoSelect
//
// CellIsSelected
//
// Just let HT tell us about what is selected and what isn't.
//
Boolean
CHyperTreeSelector :: CellIsSelected ( const STableCell &inCell ) const
{
HT_Resource node = HT_GetNthItem( mTreeView, URDFUtilities::PPRowToHTRow(inCell.row) );
return HT_IsSelected(node);
} // CellIsSelected
//
// SyncSelectorWithHT
//
// Since HT is the one that retains the selection for the view, when a new view is
// displayed, we will be out of sync. This methods gets us back in sync.
//
// This routine is called a lot (during refresh, when nodes are added, whenever js
// affects the selection, etc) so we want it to be fast when the size of the list is
// large. I had some cool ideas this morning in the shower, but when I came in to
// implement them, I realized that all of them boiled down to having to iterate over
// every element of the list because of the way LTableRowSelector finds which cells are
// selected. Since that class probably won't be changing any time soon, just do the
// stupid thing since it works out to the same in the end. (Note that HT_GetNthItem is O(1)).
//
// A valid wimpout? Doubtful, but at least you know what's going on here when we find
// how slow it is ;)
//
1998-03-28 02:44:41 +00:00
void
CHyperTreeSelector :: SyncSelectorWithHT ( )
{
const char notSelected = 0;
for ( int PPRow = 1; PPRow <= GetCount(); PPRow++ ) {
// get the corresponding row in HT. Set the value in the selector accordingly.
HT_Resource curr = HT_GetNthItem ( mTreeView, URDFUtilities::PPRowToHTRow(PPRow) );
if ( curr ) {
PRBool isSelected = HT_IsSelected ( curr );
LTableRowSelector::DoSelect ( PPRow, isSelected, true, false );
}
1998-03-28 02:44:41 +00:00
}
} // SyncSelectorWithHT
#pragma mark -
HT_Pane CPopdownFlexTable :: sPaneOriginatingDragAndDrop = NULL;
CPopdownFlexTable :: CPopdownFlexTable ( LStream* inStream )
: CHyperTreeFlexTable(inStream)
{
}
CPopdownFlexTable :: ~CPopdownFlexTable ( )
{
}
//
// OpenRow
//
// Do the normal thing, but close up the tree afterwards.
//
void
CPopdownFlexTable :: OpenRow ( TableIndexT inRow )
{
CHyperTreeFlexTable::OpenRow(inRow);
1998-08-24 19:53:36 +00:00
if ( ! RowIsContainer(inRow) )
BroadcastMessage ( msg_ClosePopdownTree, NULL );
} // OpenRow
//
// OpenSelection
//
// The inherited version of this routine iterates over the selection and opens each row in
// turn. Obviously, we only care about the first one in this case because it is meaningless
// to open multiple urls in one window.
//
void
CPopdownFlexTable :: OpenSelection()
{
TableIndexT selectedRow = 0;
if ( GetNextSelectedRow(selectedRow) && !CmdPeriod() )
OpenRow(selectedRow);
}
//
// DragSelection
//
// Wraps the inherited version to save the HT_Pane from where the drag originated so
// that when you drag to other popdowns, it doesn't go away.
//
OSErr
CPopdownFlexTable :: DragSelection(
const STableCell& inCell,
const SMouseDownEvent& inMouseDown )
{
sPaneOriginatingDragAndDrop = HT_GetPane(mHTView);
OSErr retVal = CHyperTreeFlexTable::DragSelection ( inCell, inMouseDown );
// If this tree is visible, then the drop was constrained to this tree (or the
// drop went somewhere other than another popdown). Therefore, don't delete the
// pane. If it is invisible, get rid of it so we don't leak the pane.
if ( !IsVisible() && sPaneOriginatingDragAndDrop )
HT_DeletePane ( sPaneOriginatingDragAndDrop );
sPaneOriginatingDragAndDrop = NULL;
return retVal;
} // DragSelection
#pragma mark -
1998-09-01 20:17:46 +00:00
CTreeIcon :: CTreeIcon ( const string & inURL, const CHyperTreeFlexTable* inParent, HT_Resource inNode )
: CImageIconMixin(inURL), mTree(inParent), mNode(inNode)
{
}
CTreeIcon :: ~CTreeIcon ( )
{
}
//
// ImageIsReady
//
// Tell the tree view to redraw the cell representing this node
//
void
CTreeIcon :: ImageIsReady ( )
{
mTree->RedrawRow(mNode);
}
void
CTreeIcon :: DrawStandby ( const Point & inTopLeft, IconTransformType inTransform ) const
{
Rect where;
where.top = inTopLeft.v; where.left = inTopLeft.h;
where.right = where.left + 16; where.bottom = where.top + 16;
TableIndexT row = URDFUtilities::HTRowToPPRow(HT_GetNodeIndex(HT_GetView(mNode), mNode));
mTree->CStandardFlexTable::DrawIconsSelf(STableCell(row, mTree->FindTitleColumnID()),
inTransform, where);
} // DrawStandby