////////////////////////////////////////////////////////////////////////////////
//
// 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 operationState
. (Set to null
to delete the range
* described by operationState
.)
* @param allowMerge Set to true
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.
*
*
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.
* * @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]); } } }