blob: ab7414e40cc18fb19f14a75f504f1259ed26cafc [file] [log] [blame]
////////////////////////////////////////////////////////////////////////////////
//
// 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.
//
////////////////////////////////////////////////////////////////////////////////
package flashx.textLayout.operations
{
import flashx.textLayout.edit.SelectionManager;
import flashx.textLayout.edit.SelectionState;
import flashx.textLayout.edit.TextFlowEdit;
import flashx.textLayout.edit.TextScrap;
import flashx.textLayout.elements.FlowLeafElement;
import flashx.textLayout.elements.ParagraphElement;
import flashx.textLayout.formats.ITextLayoutFormat;
import flashx.textLayout.formats.TextLayoutFormat;
import flashx.textLayout.formats.TextLayoutFormatValueHolder;
import flashx.textLayout.tlf_internal;
use namespace tlf_internal;
/**
* The DeleteTextOperation class encapsulates the deletion of a range of text.
*
* @see flashx.textLayout.edit.EditManager
* @see flashx.textLayout.events.FlowOperationEvent
*
* @playerversion Flash 10
* @playerversion AIR 1.5
* @langversion 3.0
*/
public class DeleteTextOperation extends FlowTextOperation
{
private var _textScrap:TextScrap;
private var _allowMerge:Boolean;
private var _undoParaFormat:TextLayoutFormatValueHolder;
private var _undoCharacterFormat:TextLayoutFormatValueHolder;
private var _needsOldFormat:Boolean = false;
private var _pendingFormat:TextLayoutFormatValueHolder;
private var _deleteSelectionState:SelectionState = null;
/**
* Creates a DeleteTextOperation operation.
*
* @param operationState The original range of text.
* @param deleteSelectionState The range of text to delete, if different from the range
* described by <code>operationState</code>. (Set to <code>null</code> to delete the range
* described by <code>operationState</code>.)
* @param allowMerge Set to <code>true</code> if this operation can be merged with the next or previous operation.
*
* @playerversion Flash 10
* @playerversion AIR 1.5
* @langversion 3.0
*/
public function DeleteTextOperation(operationState:SelectionState, deleteSelectionState:SelectionState = null, allowMerge:Boolean = false)
{
_deleteSelectionState = deleteSelectionState ? deleteSelectionState : operationState;
super(_deleteSelectionState);
originalSelectionState = operationState;
_allowMerge = allowMerge;
}
/**
* Indicates whether this operation can be merged with operations executed before or after it.
*
* <p>Some delete operations, for example, a sequence of backspace keystrokes, can be fruitfully
* merged into one operation so that undoing the operation reverses the entire sequence.</p>
*
* @playerversion Flash 10
* @playerversion AIR 1.5
* @langversion 3.0
*/
public function get allowMerge():Boolean
{
return _allowMerge;
}
public function set allowMerge(value:Boolean):void
{
_allowMerge = value;
}
/**
* deleteSelectionState The range of text to delete
*
* @playerversion Flash 10
* @playerversion AIR 1.5
* @langversion 3.0
*/
public function get deleteSelectionState():SelectionState
{
return _deleteSelectionState;
}
public function set deleteSelectionState(value:SelectionState):void
{
_deleteSelectionState = value;
}
/** @private */
public override function doOperation():Boolean
{
// Nothing to delete
if (absoluteStart == absoluteEnd)
return false;
_textScrap = TextFlowEdit.createTextScrap(textFlow, absoluteStart, absoluteEnd);
var leafEl:FlowLeafElement = textFlow.findLeaf(absoluteStart);
var paraEl:ParagraphElement = leafEl.getParagraph();
var paraElAbsStart:int = paraEl.getAbsoluteStart();
_pendingFormat = new TextLayoutFormatValueHolder(leafEl.format);
if (_textScrap)
{
if ((_textScrap.textFlow.textLength == 1) &&
((absoluteEnd == (textFlow.textLength - 1)) || (absoluteEnd == (paraElAbsStart + paraEl.textLength))))
{
//special case. Always insert the paragraph
_textScrap.beginMissingArray = new Array();
_textScrap.endMissingArray = new Array();
}
if (_textScrap.textFlow.textLength >= 1)
{
//save off the paragraph format of the next paragraph since we will need to set it back
//on an undo operation
leafEl = textFlow.findLeaf(absoluteEnd);
paraEl = leafEl.getParagraph();
if (absoluteEnd == paraEl.getAbsoluteStart())
{
_undoParaFormat = new TextLayoutFormatValueHolder(paraEl.format);
_undoCharacterFormat = new TextLayoutFormatValueHolder(leafEl.format);
_needsOldFormat = true;
}
}
}
var beforeOpLen:int = textFlow.textLength;
TextFlowEdit.replaceRange(textFlow, absoluteStart, absoluteEnd, null);
if (textFlow.interactionManager)
textFlow.interactionManager.notifyInsertOrDelete(absoluteStart, -(absoluteEnd - absoluteStart));
if (originalSelectionState.selectionManagerOperationState && textFlow.interactionManager)
{
// set pointFormat from leafFormat
var state:SelectionState = textFlow.interactionManager.getSelectionState();
if (state.anchorPosition == state.activePosition)
{
state.pointFormat = new TextLayoutFormatValueHolder(_pendingFormat);
textFlow.interactionManager.setSelectionState(state);
}
}
// nothing deleted???
if (beforeOpLen == textFlow.textLength)
_textScrap = null;
return true;
}
/** @private */
public override function undo():SelectionState
{
if (_textScrap != null) {
TextFlowEdit.replaceRange(textFlow, absoluteStart, absoluteStart, _textScrap);
if (_needsOldFormat)
{
textFlow.normalize();
var leafEl:FlowLeafElement = textFlow.findLeaf(absoluteEnd);
if (leafEl)
{
var paraEl:ParagraphElement = leafEl.getParagraph();
paraEl.format = _undoParaFormat;
leafEl.format = _undoCharacterFormat;
}
}
if (textFlow.interactionManager)
textFlow.interactionManager.notifyInsertOrDelete(absoluteStart, absoluteEnd - absoluteStart);
}
return originalSelectionState;
}
/** @private */
public override function redo():SelectionState
{
TextFlowEdit.replaceRange(textFlow, absoluteStart, absoluteEnd, null);
if (textFlow.interactionManager)
textFlow.interactionManager.notifyInsertOrDelete(absoluteStart, -(absoluteEnd - absoluteStart));
return new SelectionState(textFlow,absoluteStart,absoluteStart,_pendingFormat);
}
/** @private */
tlf_internal override function merge(op2:FlowOperation):FlowOperation
{
if (this.endGeneration != op2.beginGeneration)
return null;
var delOp:DeleteTextOperation = op2 as DeleteTextOperation;
if ((delOp == null) || !delOp.allowMerge || !_allowMerge)
return null;
return new CompositeOperation([this, op2]);
}
}
}