/*************************************************************************
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* Copyright 2000, 2010 Oracle and/or its affiliates.
*
* OpenOffice.org - a multi-platform office productivity suite
*
* This file is part of OpenOffice.org.
*
* OpenOffice.org is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License version 3
* only, as published by the Free Software Foundation.
*
* OpenOffice.org is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License version 3 for more details
* (a copy is included in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with OpenOffice.org. If not, see
*
* for a copy of the LGPLv3 License.
*
************************************************************************/
#ifndef SD_OUTLINER_HXX
#define SD_OUTLINER_HXX
#include
#include
#include "pres.hxx"
#include "OutlinerIterator.hxx"
#include
#include
#include
class Dialog;
class SdPage;
class SdrObject;
class SdrTextObj;
class SdDrawDocument;
class SfxStyleSheetPool;
class SdrObjListIter;
namespace sd {
class DrawViewShell;
class View;
class ViewShell;
class Window;
/** The main purpose of this class is searching and replacing as well as
spelling of impress documents. The main part of both tasks lies in
iterating over the pages and view modes of a document and apply the
respective function to all objects containing text on those pages.
Relevant objects: There are two sets of objects to search/spell
check. One is the set of all selected objects. The other consists of
all objects on all pages in draw-, notes-, and handout view as well as
slide- and background view (draw pages and master pages).
Iteration: Search/replace and spelling functions operate on shapes
containing text. To cover all relevant objects an order has to be
defined on the objects. For the set of all selected objects this order
is simply the order in which they can be retrieved from the selection
object.
When there is no selection the order is nested. The three modes of the
draw view are on the outer level: draw mode, notes mode, handout mode.
The inner level switches between draw pages and master pages. This
leads to the following order:
- draw pages of draw mode
- master pages of draw mode
- draw pages of notes mode
- master pages of notes mode
- draw pages of handout mode
- master pages of handout mode
Iteration starts at the top of the current page. When reaching the end
of the document, i.e. the last master page of the handout mode, it jumps
to the first draw page of draw mode. In backward searches this order is
reversed. When doing a replace all then the whole document is
searched for matches starting at the first page of the draw/slide view
(or last page of handout/background view even though search
direction).
The start position is restored after finishing spell checking or
replacing all matches in a document.
Some related pieces of information:
The search dialog (SvxSearchDialog) can be controlled in
more than one way:
- A set of option flags returned by the slot call
SID_SEARCH_OPTIONS handled by the
SdDrawDocument::GetState() method.
- The contents of the search item of type
SvxSearchItem.
- The HasSelection() view shell method that returns
whether or not a selection exists. However, it is called from the
search dialog with an argument so that only text selections are
queried. This is only sufficient for searching the outline view.
*/
class Outliner
: public SdrOutliner
{
public:
friend class ::sd::outliner::OutlinerContainer;
/** Create a new sd outliner object.
@param pDoc
The draw document from which to take the content.
@param nMode
The valid values OUTLINERMODE_DONTKNOW,
OUTLINERMODE_TEXTOBJECT,
OUTLINERMODE_TITLEOBJECT,
OUTLINERMODE_OUTLINEOBJECT, and
OUTLINERMODE_OUTLINEVIEW are defined in
editeng/outliner.hxx.
*/
Outliner( SdDrawDocument* pDoc, USHORT nMode );
virtual ~Outliner();
/** Despite the name this method is called prior to spell cheking *and*
searching and replacing. The position of current view
mode/page/object/caret position is remembered and, depending on the
search mode, may be restored after finishing searching/spell
checking.
*/
void PrepareSpelling (void);
/** Initialize a spell check but do not start it yet. This method
is a better candiate for the name PrepareSpelling.
*/
void StartSpelling (void);
/** Proxy for method from base class to avoid compiler warning */
void StartSpelling(EditView&, unsigned char);
/** Initiate a find and/or replace on the next relevant text object.
@return
Returns when the search/replace is finished (as
indicated by user input to the search dialog). A value
indicates that another call to this method is required.
*/
bool StartSearchAndReplace (const SvxSearchItem* pSearchItem);
/** Iterate over the sentences in all text shapes and stop at the
next sentence with spelling errors. While doing so the view
mode may be changed and text shapes are set into edit mode.
*/
::svx::SpellPortions GetNextSpellSentence (void);
/** Release all resources that have been created during the find&replace
or spell check.
*/
void EndSpelling (void);
/** callback for textconversion */
sal_Bool ConvertNextDocument (void);
/** Starts the text conversion (hangul/hanja or Chinese simplified/traditional)
for the current viewshell */
void StartConversion( INT16 nSourceLanguage, INT16 nTargetLanguage,
const Font *pTargetFont, INT32 nOptions, BOOL bIsInteractive );
/** This is called internaly when text conversion is started.
The position of current view mode/page/object/caret position
is remembered and will be restored after conversion.
*/
void BeginConversion (void);
/** Release all resources that have been created during the conversion */
void EndConversion (void);
DECL_LINK( SpellError, void * );
enum ChangeHint { CH_VIEW_SHELL_INVALID, CH_VIEW_SHELL_VALID };
/** Handle a change outside the outliner which may affect the outliner.
At the moment this is restricted to changes of the main view shell
but later may include any changes that are detected now by
DetectChange().
*/
void HandleOutsideChange (ChangeHint eHint);
int GetIgnoreCurrentPageChangesLevel() const { return mnIgnoreCurrentPageChangesLevel; };
void IncreIgnoreCurrentPageChangesLevel() { mnIgnoreCurrentPageChangesLevel++; };
void DecreIgnoreCurrentPageChangesLevel() { mnIgnoreCurrentPageChangesLevel--; };
private:
class Implementation;
::std::auto_ptr mpImpl;
/// Specifies whether to search and replace, to spell check or to do a
/// text conversion.
enum mode {SEARCH, SPELL, TEXT_CONVERSION} meMode;
/// The view which displays the searched objects.
::sd::View* mpView;
/// The view shell containing the view.
::boost::shared_ptr mpViewShell;
/// This window contains the view.
::sd::Window* mpWindow;
/// The document on whose objects and pages this class operates.
SdDrawDocument* mpDrawDocument;
/** this is the language that is used for current text conversion.
Only valid if meMode is TEXT_CONVERSION.
*/
INT16 mnConversionLanguage;
/** While the value of this flag is greater than 0 changes of the current page
do not lead to selecting the corresponding text in the outliner.
*/
int mnIgnoreCurrentPageChangesLevel;
/// Specifies whether the search string has been found so far.
bool mbStringFound;
/** This flag indicates whether there may exist a match of the search
string before/after the current position in the document. It can be
set to only when starting from the beginning/end of the
document. When reaching the end/beginning with it still be set to
then there exists no match and the search can be terminated.
*/
bool mbMatchMayExist;
/// The number of pages in the current view.
USHORT mnPageCount;
/// Number of objects on the current page / in the current selection.
INT32 mnObjectCount;
/** A value indicates that the end of the find&replace or spell
check has been reached.
*/
bool mbEndOfSearch;
/** Set to when an object has been prepared successfully for
searching/spell checking. This flag directs the internal iteration
which stops when set to .
*/
bool mbFoundObject;
/** When set to this flag indicates that an error has occured
that should terminate the iteration over the objects to search/spell
check.
*/
bool mbError;
/** This flag indicates whether to search forward or backwards.
*/
bool mbDirectionIsForward;
/** This flag indicates that only the selected objects are to be
searched.
*/
bool mbRestrictSearchToSelection;
/** When the search is restricted to the current selection then
this list contains pointers to all the objects of the
selection. This copy is necessary because during the search
process the mark list is modified.
*/
::std::vector maMarkListCopy;
/** This flag inidcates that only the current view is to be used for
searching and spelling. Automatically switching to other view does
not take place when this flag is set.
*/
bool mbProcessCurrentViewOnly;
/** Current object that may be a text object. The object pointer to
corresponds to mnObjIndex. While iterating over the
objects on a page mpObj will point to every object
while mpTextObj will be set only to valid text
objects.
*/
SdrObject* mpObj;
/** this stores the first object that is used for text conversion.
Conversion automaticly wraps around the document and stops when it
finds this object again.
*/
SdrObject* mpFirstObj;
/// Candidate for being searched/spell checked.
SdrTextObj* mpTextObj;
/// Current text to be searched/spelled inside the current text object
sal_Int32 mnText;
/// Paragraph object of mpTextObj.
OutlinerParaObject* mpParaObj;
/// The view mode that was active when starting to search/spell check.
PageKind meStartViewMode;
/// The master page mode that was active when starting to search/spell check.
EditMode meStartEditMode;
/// The current page index on starting to search/spell check.
USHORT mnStartPageIndex;
/// The object in edit mode when searching /spell checking was started
/// (if any).
SdrObject* mpStartEditedObject;
/// The position of the caret when searching /spell checking was started.
ESelection maStartSelection;
/** The search item contains various attributes that define the type of
search. It is set every time the
SearchAndReplaceAll method is called.
*/
const SvxSearchItem* mpSearchItem;
/// The actual object iterator.
::sd::outliner::Iterator maObjectIterator;
/// The current position of the object iterator.
::sd::outliner::IteratorPosition maCurrentPosition;
/// The position when the search started. Corresponds largely to the
/// m?Start* members.
::sd::outliner::Iterator maSearchStartPosition;
/** The last valid position desribes where the last text object has been
found. This position is restored when some dialogs are shown. The
position is initially set to the where the search begins.
*/
::sd::outliner::IteratorPosition maLastValidPosition;
/** This flag remebers a selection change between a call to the
selection change listener callback and the next
DetectChange() method call.
*/
bool mbSelectionHasChanged;
/** This flag indicates whether a selection change event is expected due
to a programatical change of the selection.
*/
bool mbExpectingSelectionChangeEvent;
/** This flag is set to true when the whole document has been
processed once 'officially', i.e. a message box has been shown
that tells the user so.
*/
bool mbWholeDocumentProcessed;
/** When this flag is true then a PrepareSpelling() is executed when
StartSearchAndReplace() is called the next time.
*/
bool mbPrepareSpellingPending;
/** In this flag we store whether the view shell is valid and may be
accessed.
*/
bool mbViewShellValid;
/** Initialize the object iterator. Call this method after being
invoked from the search or spellcheck dialog. It creates a new
iterator pointing at the current object when this has not been done
before. It reverses the direction of iteration if the given flag
differs from the current direction.
@param bDirectionIsForward
This flag specifies in which direction to iterator over the
objects. If it differs from the current direction the iterator
is reversed.
*/
void Initialize (bool bDirectionIsForward);
/** Do search and replace for whole document.
*/
bool SearchAndReplaceAll (void);
/** Do search and replace for next match.
@return
The return value specifies whether the search ended () or
another call to this method is required ().
*/
bool SearchAndReplaceOnce (void);
/** Detect changes of the document or view and react accordingly. Such
changes may occur because different calls to
SearchAndReplace() there usually is user
interaction. This is at least the press of the search or replace
button but may include any other action some of which affect the
search.
*/
void DetectChange (void);
/** Detect whether the selection has changed.
@return
Return when the selection has been changed since the
last call to this method.
*/
bool DetectSelectionChange (void);
/** Remember the current edited object/caret position/page/view mode
when starting to search/spell check so that it can be restored on
termination.
*/
void RememberStartPosition (void);
/** Restore the position stored in the last call of
RememberStartPositiony.
*/
void RestoreStartPosition (void);
/** Provide next object to search or spell check as text object in edit
mode on the current page. This skips all objects that do not
match or are no text object.
*/
void ProvideNextTextObject (void);
/** Handle the situation that the iterator has reached the last object.
This may result in setting the mbEndOfSearch flag
back to . This method may show either the end-of-search
dialog or the wrap-arround dialog.
*/
void EndOfSearch (void);
/** Prepare to commence searching/spell checking on the specified page.
This also creates a proper page iterator.
@param nPageIndex
Index of the page to prepare for searching. The default value
of -1 is a flag for using the first/last page of the current
edit mode/view mode depending on the search direction.
*/
void InitPage (USHORT nPageIndex = (USHORT)-1);
/** Show a dialog that tells the user that the search has ended either
because there are no more matches after finding at least one or that
no match has been found at all.
*/
void ShowEndOfSearchDialog (void);
/** Show a dialog that asks the user whether to wrap arround to the
beginning/end of the document and continue with the search/spell
check.
*/
bool ShowWrapArroundDialog (void);
/** Check whether the object pointed to by the iterator is a valid text
object.
@param aPosition
The object for which to test whether it is a valid text object.
*/
bool IsValidTextObject (const ::sd::outliner::IteratorPosition& rPosition);
/** Put text of current text object into outliner so that the text can
be searched/spell checked.
*/
void PutTextIntoOutliner (void);
/** Prepare to do spell checking on the current text object. This
includes putting it into edit mode. Under certain conditions this
method sets mbEndOfSearch to .
*/
void PrepareSpellCheck (void);
/** Prepare to search and replace on the current text object. This
includes putting it into edit mode.
*/
void PrepareSearchAndReplace (void);
/** Prepare to do a text conversion on the current text
object. This includes putting it into edit mode.
*/
void PrepareConversion (void);
/** Switch to a new view mode. Try to restore the original edit mode
before doing so.
@param ePageKind
Specifies the new view mode.
*/
void SetViewMode (PageKind ePageKind);
/** Switch to the page or master page specified by the
mnPage index. Master page mode is specified by
meEditMode.
@param eEditMode
The new edit mode.
@param nPageIndex
The new page index.
*/
void SetPage (EditMode eEditMode, USHORT nPageIndex);
/** Switch on edit mode for the currently selected text object.
*/
void EnterEditMode (BOOL bGrabFocus=TRUE);
/** Return the position at which a new search is started with respect to
the search direction as specified by the argument.
@return
The position mentioned above in form of a selection with start
equals end.
*/
ESelection GetSearchStartPosition (void);
/** Detect whether there exists a previous match. Note that only the
absence of such a match can be detected reliably. An existing match
is assumed when the search started not at the beginning/end of the
presentation. This does not have to be true. The user can have set
the cursor at the middle of the text without a prior search.
@return
Returns when there is no previous match and
when there may be one.
*/
bool HasNoPreviousMatch (void);
/** Handle a failed search (with or without replace) for the outline
mode. Show message boxes when the search failed completely,
i.e. there is no match in the whole presentation, or when no further
match exists.
@return
The returned value indicates whether another (wrapped arround)
search shall take place. If that is so, then it is the caller's
responsibility to set the cursor position accordingly.
*/
bool HandleFailedSearch (void);
/** Return a parent window for a dialog. The returned window depends on
the search mode.
*/
::Window* GetParentForDialog (void);
/** Take a position as returned by an object iterator and switch to the
view and page on which the object specified by this position is
located.
@param rPosition
This position points to a SdrObject object and
contains the view and page where it is located.
@return
Return a pointer to the SdrObject.
*/
SdrObject* SetObject (const ::sd::outliner::IteratorPosition& rPosition);
/** Use this method when the view shell in which to search has changed.
It handles i.e. registering at the associated view as selection
change listener.
*/
void SetViewShell (const ::boost::shared_ptr& rpViewShell);
/** Activate or deactivate the search in the current selection. Call
this method whenever the selection has changed. This method creates
a copy of the current selection and reassings the object iterator to
the current() iterator.
*/
void HandleChangedSelection (void);
/** Initiate the spell check of the next relevant text object.
When the outline view is active then this method is called
after a wrap arround to continue at the beginning of the document.
@return
Returns to indicate that another call to this method is
required. When all text objects have been processed then
is returned.
*/
virtual BOOL SpellNextDocument (void);
/** Show the given message box and make it modal. It is assumed that
the parent of the given dialog is NULL, i.e. the application
window. This function makes sure that the otherwise non-modal
search dialog, if visible, is locked, too.
*/
USHORT ShowModalMessageBox (Dialog& rMessageBox);
};
} // end of namespace sd
#endif