/**************************************************************
 * 
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 * 
 *************************************************************/



// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_dbui.hxx"

#ifndef _DBAUI_INDEXFIELDSCONTROL_HXX_
#include "indexfieldscontrol.hxx"
#endif
#ifndef _DBU_DLG_HRC_
#include "dbu_dlg.hrc"
#endif
#ifndef _OSL_DIAGNOSE_H_
#include <osl/diagnose.h>
#endif
#ifndef _DBA_DBACCESS_HELPID_HRC_
#include "dbaccess_helpid.hrc"
#endif

//......................................................................
namespace dbaui
{
//......................................................................

#define BROWSER_STANDARD_FLAGS		BROWSER_COLUMNSELECTION | BROWSER_HLINESFULL | BROWSER_VLINESFULL |	\
									BROWSER_HIDECURSOR | BROWSER_HIDESELECT | BROWSER_AUTO_HSCROLL | BROWSER_AUTO_VSCROLL

#define COLUMN_ID_FIELDNAME		1
#define	COLUMN_ID_ORDER			2

	using namespace ::com::sun::star::uno;
	using namespace ::svt;

	//==================================================================
	//= DbaMouseDownListBoxController
	//==================================================================
	class DbaMouseDownListBoxController : public ListBoxCellController
	{
	protected:
		Link	m_aOriginalModifyHdl;
		Link	m_aAdditionalModifyHdl;

	public:
		DbaMouseDownListBoxController(ListBoxControl* _pParent)
			:ListBoxCellController(_pParent)
		{
		}

		void SetAdditionalModifyHdl(const Link& _rHdl);

	protected:
		virtual sal_Bool WantMouseEvent() const { return sal_True; }
		virtual void SetModifyHdl(const Link& _rHdl);

	private:
		void implCheckLinks();
		DECL_LINK( OnMultiplexModify, void* );
	};

	//------------------------------------------------------------------
	void DbaMouseDownListBoxController::SetAdditionalModifyHdl(const Link& _rHdl)
	{
		m_aAdditionalModifyHdl = _rHdl;
		implCheckLinks();
	}

	//------------------------------------------------------------------
	void DbaMouseDownListBoxController::SetModifyHdl(const Link& _rHdl)
	{
		m_aOriginalModifyHdl = _rHdl;
		implCheckLinks();
	}

	//------------------------------------------------------------------
	IMPL_LINK( DbaMouseDownListBoxController, OnMultiplexModify, void*, _pArg )
	{
		if (m_aAdditionalModifyHdl.IsSet())
			m_aAdditionalModifyHdl.Call(_pArg);
		if (m_aOriginalModifyHdl.IsSet())
			m_aOriginalModifyHdl.Call(_pArg);
		return 0L;
	}

	//------------------------------------------------------------------
	void DbaMouseDownListBoxController::implCheckLinks()
	{
		if (m_aAdditionalModifyHdl.IsSet() || m_aOriginalModifyHdl.IsSet())
			ListBoxCellController::SetModifyHdl(LINK(this, DbaMouseDownListBoxController, OnMultiplexModify));
		else
			ListBoxCellController::SetModifyHdl(Link());
	}

	//==================================================================
	//= IndexFieldsControl
	//==================================================================
DBG_NAME(IndexFieldsControl)
//------------------------------------------------------------------
	IndexFieldsControl::IndexFieldsControl( Window* _pParent, const ResId& _rId ,sal_Int32 _nMaxColumnsInIndex,sal_Bool _bAddIndexAppendix)
		:EditBrowseBox(_pParent, _rId, EBBF_SMART_TAB_TRAVEL | EBBF_ACTIVATE_ON_BUTTONDOWN, BROWSER_STANDARD_FLAGS)
		,m_aSeekRow(m_aFields.end())
		,m_pSortingCell(NULL)
		,m_pFieldNameCell(NULL)
		,m_nMaxColumnsInIndex(_nMaxColumnsInIndex)
        ,m_bAddIndexAppendix(_bAddIndexAppendix)
	{
        DBG_CTOR(IndexFieldsControl,NULL);

		SetUniqueId( UID_DLGINDEX_INDEXDETAILS_BACK );
		GetDataWindow().SetUniqueId( UID_DLGINDEX_INDEXDETAILS_MAIN );
	}

	//------------------------------------------------------------------
	IndexFieldsControl::~IndexFieldsControl()
	{
		delete m_pSortingCell;
		delete m_pFieldNameCell;
	
        DBG_DTOR(IndexFieldsControl,NULL);
    }

	//------------------------------------------------------------------
	sal_Bool IndexFieldsControl::SeekRow(long nRow)
	{
		if (!EditBrowseBox::SeekRow(nRow))
			return sal_False;

		if (nRow < 0)
		{
			m_aSeekRow = m_aFields.end();
		}
		else
		{
			m_aSeekRow = m_aFields.begin() + nRow;
			OSL_ENSURE(m_aSeekRow <= m_aFields.end(), "IndexFieldsControl::SeekRow: invalid row!");
		}

		return sal_True;
	}

	//------------------------------------------------------------------
	void IndexFieldsControl::PaintCell( OutputDevice& _rDev, const Rectangle& _rRect, sal_uInt16 _nColumnId ) const
	{
		Point aPos(_rRect.TopLeft());
		aPos.X() += 1;

		String aText = GetRowCellText(m_aSeekRow,_nColumnId);
		Size TxtSize(GetDataWindow().GetTextWidth(aText), GetDataWindow().GetTextHeight());

		// clipping
		if (aPos.X() < _rRect.Right() || aPos.X() + TxtSize.Width() > _rRect.Right() ||
			aPos.Y() < _rRect.Top() || aPos.Y() + TxtSize.Height() > _rRect.Bottom())
			_rDev.SetClipRegion( _rRect );

		// allow for a disabled control ...
		sal_Bool bEnabled = IsEnabled();
		Color aOriginalColor = _rDev.GetTextColor();
		if (!bEnabled)
			_rDev.SetTextColor(GetSettings().GetStyleSettings().GetDisableColor());

		// draw the text
		_rDev.DrawText(aPos, aText);

		// reset the color (if necessary)
		if (!bEnabled)
			_rDev.SetTextColor(aOriginalColor);

		if (_rDev.IsClipRegion())
			_rDev.SetClipRegion();
	}

	//------------------------------------------------------------------
	void IndexFieldsControl::initializeFrom(const IndexFields& _rFields)
	{
		// copy the field descriptions
		m_aFields = _rFields;
		m_aSeekRow = m_aFields.end();

		SetUpdateMode(sal_False);
		// remove all rows
		RowRemoved(1, GetRowCount());
		// insert rows for the fields
		RowInserted(GetRowCount(), m_aFields.size(), sal_False);
		// insert an additional row for a new field for that index
//		if(!m_nMaxColumnsInIndex || GetRowCount() < m_nMaxColumnsInIndex )
		RowInserted(GetRowCount(), 1, sal_False);
		SetUpdateMode(sal_True);

		GoToRowColumnId(0, COLUMN_ID_FIELDNAME);
	}

	//------------------------------------------------------------------
	void IndexFieldsControl::commitTo(IndexFields& _rFields)
	{
		// do not just copy the array, we may have empty field names (which should not be copied)
		_rFields.resize(m_aFields.size());
		ConstIndexFieldsIterator aSource = m_aFields.begin();
		ConstIndexFieldsIterator aSourceEnd = m_aFields.end();
		IndexFieldsIterator aDest = _rFields.begin();
		for (; aSource < aSourceEnd; ++aSource)
			if (0 != aSource->sFieldName.Len())
			{
				*aDest = *aSource;
				++aDest;
			}

		_rFields.resize(aDest - _rFields.begin());
	}

	//------------------------------------------------------------------
	sal_uInt32 IndexFieldsControl::GetTotalCellWidth(long _nRow, sal_uInt16 _nColId)
	{
		if (COLUMN_ID_ORDER == _nColId)
		{
			sal_Int32 nWidthAsc = GetTextWidth(m_sAscendingText) + GetSettings().GetStyleSettings().GetScrollBarSize();
			sal_Int32 nWidthDesc = GetTextWidth(m_sDescendingText) + GetSettings().GetStyleSettings().GetScrollBarSize();
			// maximum plus some additional space
			return (nWidthAsc > nWidthDesc ? nWidthAsc : nWidthDesc) + GetTextWidth('0') * 2;
		}
		return EditBrowseBox::GetTotalCellWidth(_nRow, _nColId);
	}

	//------------------------------------------------------------------
	void IndexFieldsControl::Init(const Sequence< ::rtl::OUString >& _rAvailableFields)
	{
		RemoveColumns();

        // for the width: both columns together should be somewhat smaller than the whole window (without the scrollbar)
		sal_Int32 nFieldNameWidth = GetSizePixel().Width();

        if ( m_bAddIndexAppendix )
        {
		    m_sAscendingText = String(ModuleRes(STR_ORDER_ASCENDING));
		    m_sDescendingText = String(ModuleRes(STR_ORDER_DESCENDING));

		    // the "sort order" column
		    String sColumnName = String(ModuleRes(STR_TAB_INDEX_SORTORDER));
		    // the width of the order column is the maximum widths of the texts used
		    // (the title of the column)
		    sal_Int32 nSortOrderColumnWidth = GetTextWidth(sColumnName);
		    // ("ascending" + scrollbar width)
		    sal_Int32 nOther = GetTextWidth(m_sAscendingText) + GetSettings().GetStyleSettings().GetScrollBarSize();
		    nSortOrderColumnWidth = nSortOrderColumnWidth > nOther ? nSortOrderColumnWidth : nOther;
		    // ("descending" + scrollbar width)
		    nOther = GetTextWidth(m_sDescendingText) + GetSettings().GetStyleSettings().GetScrollBarSize();
		    nSortOrderColumnWidth = nSortOrderColumnWidth > nOther ? nSortOrderColumnWidth : nOther;
		    // (plus some additional space)
		    nSortOrderColumnWidth += GetTextWidth('0') * 2;
		    InsertDataColumn(COLUMN_ID_ORDER, sColumnName, nSortOrderColumnWidth, HIB_STDSTYLE, 1);

            m_pSortingCell = new ListBoxControl(&GetDataWindow());
		    m_pSortingCell->InsertEntry(m_sAscendingText);
		    m_pSortingCell->InsertEntry(m_sDescendingText);
            m_pSortingCell->SetHelpId( HID_DLGINDEX_INDEXDETAILS_SORTORDER );

            nFieldNameWidth -= nSortOrderColumnWidth;
        }
        StyleSettings aSystemStyle = Application::GetSettings().GetStyleSettings();
	    nFieldNameWidth -= aSystemStyle.GetScrollBarSize();
	    nFieldNameWidth -= 8;
		// the "field name" column
		String sColumnName = String(ModuleRes(STR_TAB_INDEX_FIELD));
		InsertDataColumn(COLUMN_ID_FIELDNAME, sColumnName, nFieldNameWidth, HIB_STDSTYLE, 0);

		// create the cell controllers
		// for the field name cell
		m_pFieldNameCell = new ListBoxControl(&GetDataWindow());
		m_pFieldNameCell->InsertEntry(String());
        m_pFieldNameCell->SetHelpId( HID_DLGINDEX_INDEXDETAILS_FIELD );		
		const ::rtl::OUString* pFields = _rAvailableFields.getConstArray();
		const ::rtl::OUString* pFieldsEnd = pFields + _rAvailableFields.getLength();
		for (;pFields < pFieldsEnd; ++pFields)
			m_pFieldNameCell->InsertEntry(*pFields);
	}

	//------------------------------------------------------------------
	CellController* IndexFieldsControl::GetController(long _nRow, sal_uInt16 _nColumnId)
	{
		if (!IsEnabled())
			return NULL;

		ConstIndexFieldsIterator aRow;
		sal_Bool bNewField = !implGetFieldDesc(_nRow, aRow);

		DbaMouseDownListBoxController* pReturn = NULL;
		switch (_nColumnId)
		{
			case COLUMN_ID_ORDER:
				if (!bNewField && m_pSortingCell && 0 != aRow->sFieldName.Len())
					pReturn = new DbaMouseDownListBoxController(m_pSortingCell);
				break;

			case COLUMN_ID_FIELDNAME:
				pReturn = new DbaMouseDownListBoxController(m_pFieldNameCell);
				break;

			default:
				OSL_ENSURE(sal_False, "IndexFieldsControl::GetController: invalid column id!");
		}

		if (pReturn)
			pReturn->SetAdditionalModifyHdl(LINK(this, IndexFieldsControl, OnListEntrySelected));

		return pReturn;
	}

	//------------------------------------------------------------------
	sal_Bool IndexFieldsControl::implGetFieldDesc(long _nRow, ConstIndexFieldsIterator& _rPos)
	{
		_rPos = m_aFields.end();
		if ((_nRow < 0) || (_nRow >= (sal_Int32)m_aFields.size()))
			return sal_False;
		_rPos = m_aFields.begin() + _nRow;
		return sal_True;
	}

	//------------------------------------------------------------------
	sal_Bool IndexFieldsControl::IsModified() const
	{
		return EditBrowseBox::IsModified();
	}

	//------------------------------------------------------------------
	sal_Bool IndexFieldsControl::SaveModified()
	{
		if (!IsModified())
			return sal_True;

		switch (GetCurColumnId())
		{
			case COLUMN_ID_FIELDNAME:
			{
				String sFieldSelected = m_pFieldNameCell->GetSelectEntry();
				sal_Bool bEmptySelected = 0 == sFieldSelected.Len();
				if (isNewField())
				{
					if (!bEmptySelected)
					{
						// add a new field to the collection
						OIndexField aNewField;
						aNewField.sFieldName = sFieldSelected;
						m_aFields.push_back(aNewField);
						RowInserted(GetRowCount(), 1, sal_True);
					}
				}
				else
				{
					sal_Int32 nRow = GetCurRow();
					OSL_ENSURE(nRow < (sal_Int32)m_aFields.size(), "IndexFieldsControl::SaveModified: invalid current row!");
					if (nRow >= 0)	// may be -1 in case the control was empty
					{
						// remove the field from the selection
						IndexFieldsIterator aPos = m_aFields.begin() + nRow;

						if (bEmptySelected)
						{
							aPos->sFieldName = String();

							// invalidate the row to force repaint
							Invalidate(GetRowRectPixel(nRow));
							return sal_True;
						}

						if (sFieldSelected == aPos->sFieldName)
							// nothing changed
							return sal_True;

						aPos->sFieldName = sFieldSelected;
					}
				}

				Invalidate(GetRowRectPixel(GetCurRow()));
			}
			break;
			case COLUMN_ID_ORDER:
			{
				OSL_ENSURE(!isNewField(), "IndexFieldsControl::SaveModified: why the hell ...!!!");
				// selected entry
				sal_uInt16 nPos = m_pSortingCell->GetSelectEntryPos();
				OSL_ENSURE(LISTBOX_ENTRY_NOTFOUND != nPos, "IndexFieldsControl::SaveModified: how did you get this selection??");
				// adjust the sort flag in the index field description
				OIndexField& rCurrentField = m_aFields[GetCurRow()];
				rCurrentField.bSortAscending = (0 == nPos);

			}
			break;
			default:
				OSL_ENSURE(sal_False, "IndexFieldsControl::SaveModified: invalid column id!");
		}
		return sal_True;
	}

	//------------------------------------------------------------------
	void IndexFieldsControl::InitController(CellControllerRef& /*_rController*/, long _nRow, sal_uInt16 _nColumnId)
	{
		ConstIndexFieldsIterator aFieldDescription;
		sal_Bool bNewField = !implGetFieldDesc(_nRow, aFieldDescription);

		switch (_nColumnId)
		{
			case COLUMN_ID_FIELDNAME:
				m_pFieldNameCell->SelectEntry(bNewField ? String() : aFieldDescription->sFieldName);
				m_pFieldNameCell->SaveValue();
				break;

			case COLUMN_ID_ORDER:
				m_pSortingCell->SelectEntry(aFieldDescription->bSortAscending ? m_sAscendingText : m_sDescendingText);
				m_pSortingCell->SaveValue();
				break;

			default:
				OSL_ENSURE(sal_False, "IndexFieldsControl::InitController: invalid column id!");
		}
	}

	//------------------------------------------------------------------
	IMPL_LINK( IndexFieldsControl, OnListEntrySelected, ListBox*, _pBox )
	{
		if (!_pBox->IsTravelSelect() && m_aModifyHdl.IsSet())
			m_aModifyHdl.Call(this);

		if (_pBox == m_pFieldNameCell)
		{	// a field has been selected
			if (GetCurRow() >= GetRowCount() - 2)
			{	// and we're in one of the last two rows
				String sSelectedEntry = m_pFieldNameCell->GetSelectEntry();
				sal_Int32 nCurrentRow = GetCurRow();
				sal_Int32 rowCount = GetRowCount();

				OSL_ENSURE(((sal_Int32)(m_aFields.size() + 1)) == rowCount, "IndexFieldsControl::OnListEntrySelected: inconsistence!");

				if (sSelectedEntry.Len() && (nCurrentRow == rowCount - 1) /*&& (!m_nMaxColumnsInIndex || rowCount < m_nMaxColumnsInIndex )*/ )
				{	// in the last row, an non-empty string has been selected
					// -> insert a new row
					m_aFields.push_back(OIndexField());
					RowInserted(GetRowCount(), 1);
					Invalidate(GetRowRectPixel(nCurrentRow));
				}
				else if (!sSelectedEntry.Len() && (nCurrentRow == rowCount - 2))
				{	// in the (last-1)th row, an empty entry has been selected
					// -> remove the last row
					m_aFields.erase(m_aFields.end() - 1);
					RowRemoved(GetRowCount() - 1, 1);
					Invalidate(GetRowRectPixel(nCurrentRow));
				}
			}

			SaveModified();
		}
		return 0L;
	}
	//------------------------------------------------------------------
	String IndexFieldsControl::GetCellText(long _nRow,sal_uInt16 nColId) const
	{
		ConstIndexFieldsIterator aRow = m_aFields.end();
		if ( _nRow >= 0 )
		{
			aRow = m_aFields.begin() + _nRow;
			OSL_ENSURE(aRow <= m_aFields.end(), "IndexFieldsControl::SeekRow: invalid row!");
		}
		return GetRowCellText(aRow,nColId);
	}
	//------------------------------------------------------------------
	String IndexFieldsControl::GetRowCellText(const ConstIndexFieldsIterator& _rRow,sal_uInt16 nColId) const
	{
		if (_rRow < m_aFields.end())
		{
			switch (nColId)
			{
				case COLUMN_ID_FIELDNAME:
					return _rRow->sFieldName;
				case COLUMN_ID_ORDER:
					if (0 == _rRow->sFieldName.Len())
						return String();
					else
						return _rRow->bSortAscending ? m_sAscendingText : m_sDescendingText;
				default:
					OSL_ENSURE(sal_False, "IndexFieldsControl::GetCurrentRowCellText: invalid column id!");
			}
		}
		return String();
	}
	//------------------------------------------------------------------
	sal_Bool IndexFieldsControl::IsTabAllowed(sal_Bool /*bForward*/) const
	{
		return sal_False;
	}
	//------------------------------------------------------------------

//......................................................................
}	// namespace dbaui
//......................................................................