"use strict";

/* WorksheetView.js
 *
 * Author: Dmitry.Sokolov@avsmedia.net
 * Date:   Nov 21, 2011
 */
(/**
 * @param {jQuery} $
 * @param {Window} window
 * @param {undefined} undefined
 */
    function ( $, window, undefined ) {


    /*
     * Import
     * -----------------------------------------------------------------------------
     */
    var asc = window["Asc"];
    var asc_applyFunction = asc.applyFunction;
    var asc_calcnpt = asc.calcNearestPt;
    var asc_getcvt = asc.getCvtRatio;
    var asc_floor = asc.floor;
    var asc_ceil = asc.ceil;
    var asc_obj2Color = asc.colorObjToAscColor;
    var asc_typeof = asc.typeOf;
    var asc_incDecFonSize = asc.incDecFonSize;
    var asc_debug = asc.outputDebugStr;
    var asc_Range = asc.Range;
    var asc_ActiveRange = asc.ActiveRange;
    var asc_CMM = asc.asc_CMouseMoveData;
    var asc_VR = asc.VisibleRange;

    var asc_CCellFlag = asc.asc_CCellFlag;
    var asc_CFont = asc.asc_CFont;
    var asc_CFill = asc.asc_CFill;
    var asc_CCellInfo = asc.asc_CCellInfo;
    var asc_CHyperlink = asc.asc_CHyperlink;
    var asc_CPageOptions = asc.asc_CPageOptions;
    var asc_CPageSetup = asc.asc_CPageSetup;
    var asc_CPageMargins = asc.asc_CPageMargins;
    var asc_CPagePrint = asc.CPagePrint;
    var asc_CSelectionMathInfo = asc.asc_CSelectionMathInfo;
    var asc_CAutoFilterInfo = asc.asc_CAutoFilterInfo;

    /*
     * Constants
     * -----------------------------------------------------------------------------
     */

    /**
     * header styles
     * @const
     */
    var kHeaderDefault = 0;
    var kHeaderActive = 1;
    var kHeaderHighlighted = 2;

    /**
     * text alignment style
     * @const
     */
    var khaLeft = "left";
    var khaCenter = "center";
    var khaRight = "right";
    var khaJustify = "justify";
    var kvaTop = "top";
    var kvaCenter = "center";
    var kvaBottom = "bottom";

    var kNone = "none";

    /**
     * cursor styles
     * @const
     */
    var kCurDefault = "default";
    var kCurCorner = "pointer";
    var kCurColSelect = "pointer";
    var kCurColResize = "col-resize";
    var kCurRowSelect = "pointer";
    var kCurRowResize = "row-resize";
    // Курсор для автозаполнения
    var kCurFillHandle = "crosshair";
    // Курсор для гиперссылки
    var kCurHyperlink = "pointer";
    // Курсор для перемещения области выделения
    var kCurMove = "move";
    var kCurSEResize = "se-resize";
    var kCurNEResize = "ne-resize";
    var kCurAutoFilter = "pointer";
    var kCurCells = '';
    var kCurFormatPainterExcel = '';
    if ( AscBrowser.isIE ) {
        // Пути указаны относительно html в меню, не надо их исправлять
        // и коммитить на пути относительно тестового меню
        kCurCells = 'url(../../../sdk/Common/Images/plus.cur), pointer';
        kCurFormatPainterExcel = 'url(../../../sdk/Common/Images/plus_copy.cur), pointer';
    }
    else if ( AscBrowser.isOpera ) {
        kCurCells = 'cell';
        kCurFormatPainterExcel = 'pointer';
    }
    else {
        kCurCells = "url('') 6 6, pointer";
        kCurFormatPainterExcel = "url('') 6 12, pointer";
    }

    var kNewLine = "\n";

    var kMaxAutoCompleteCellEdit = 20000;

    function calcDecades( num ) {
        return Math.abs( num ) < 10 ? 1 : 1 + calcDecades( asc_floor( num * 0.1 ) );
    }

    function CacheElement() {
        if ( !(this instanceof CacheElement) ) {
            return new CacheElement();
        }
        this.columnsWithText = {};							// Колонки, в которых есть текст
        this.columns = {};
        this.erased = {};
        return this;
    }

    function Cache() {
        if ( !(this instanceof Cache) ) {
            return new Cache();
        }

        this.rows = {};
        this.sectors = [];

        this.reset = function () {
            this.rows = {};
            this.sectors = [];
        };

        // Structure of cache
        //
        // cache : {
        //
        //   rows : {
        //     0 : {
        //       columns : {
        //         0 : {
        //           text : {
        //             cellHA  : String,
        //             cellVA  : String,
        //             cellW   : Number,
        //             color   : String,
        //             metrics : TextMetrics,
        //             sideL   : Number,
        //             sideR   : Number,
        //             state   : StringRenderInternalState
        //           }
        //         }
        //       },
        //       erased : {
        //         1 : true, 2 : true
        //       }
        //     }
        //   },
        //
        //   sectors: [
        //     0 : Range
        //   ]
        //
        // }
    }

    function CellFlags() {
        this.wrapText = false;
        this.shrinkToFit = false;
        this.merged = null;
        this.textAlign = kNone;
    }

    CellFlags.prototype.clone = function () {
        var oRes = new CellFlags();
        oRes.wrapText = this.wrapText;
        oRes.shrinkToFit = this.shrinkToFit;
        oRes.merged = this.merged ? this.merged.clone() : null;
        oRes.textAlign = this.textAlign;
        return oRes;
    };
    CellFlags.prototype.isMerged = function () {
        return null !== this.merged;
    };

    function CellBorderObject( borders, mergeInfo, col, row ) {
        this.borders = borders;
        this.mergeInfo = mergeInfo;

        this.col = col;
        this.row = row;
    }

    CellBorderObject.prototype.isMerge = function () {
        return null != this.mergeInfo;
    };
    CellBorderObject.prototype.getLeftBorder = function () {
        if ( !this.borders || (this.isMerge() && (this.col !== this.mergeInfo.c1 || this.col - 1 !== this.mergeInfo.c2)) ) {
            return null;
        }
        return this.borders.l;
    };
    CellBorderObject.prototype.getRightBorder = function () {
        if ( !this.borders || (this.isMerge() && (this.col - 1 !== this.mergeInfo.c1 || this.col !== this.mergeInfo.c2)) ) {
            return null;
        }
        return this.borders.r;
    };

    CellBorderObject.prototype.getTopBorder = function () {
        if ( !this.borders || (this.isMerge() && (this.row !== this.mergeInfo.r1 || this.row - 1 !== this.mergeInfo.r2)) ) {
            return null;
        }
        return this.borders.t;
    };
    CellBorderObject.prototype.getBottomBorder = function () {
        if ( !this.borders || (this.isMerge() && (this.row - 1 !== this.mergeInfo.r1 || this.row !== this.mergeInfo.r2)) ) {
            return null;
        }
        return this.borders.b;
    };


    /**
     * Widget for displaying and editing Worksheet object
     * -----------------------------------------------------------------------------
     * @param {Woorksheet} model  Worksheet
     * @param {asc_CHandlersList} handlers  Event handlers
     * @param {Object} buffers    DrawingContext + Overlay
     * @param {StringRender} stringRender    StringRender
     * @param {Number} maxDigitWidth    Максимальный размер цифры
     * @param {CCollaborativeEditing} collaborativeEditing
     * @param {Object} settings  Settings
     *
     * @constructor
     * @memberOf Asc
     */
    function WorksheetView( model, handlers, buffers, stringRender, maxDigitWidth, collaborativeEditing, settings ) {
        this.settings = settings;

        this.vspRatio = 1.275;

        this.handlers = handlers;
        this.model = model;

        this.buffers = buffers;
        this.drawingCtx = this.buffers.main;
        this.overlayCtx = this.buffers.overlay;

        this.drawingGraphicCtx = this.buffers.mainGraphic;
        this.overlayGraphicCtx = this.buffers.overlayGraphic;

        this.shapeCtx = this.buffers.shapeCtx;
        this.shapeOverlayCtx = this.buffers.shapeOverlayCtx;

        this.stringRender = stringRender;

        // Флаг, сигнализирует о том, что мы сделали resize, но это не активный лист (поэтому как только будем показывать, нужно перерисовать и пересчитать кеш)
        this.updateResize = false;
        // Флаг, сигнализирует о том, что мы сменили zoom, но это не активный лист (поэтому как только будем показывать, нужно перерисовать и пересчитать кеш)
        this.updateZoom = false;
        // ToDo Флаг-заглушка, для того, чтобы на mobile не было изменения высоты строк при zoom (по правильному высота просто не должна меняться)
        this.notUpdateRowHeight = false;

        this.cache = new Cache();

        //---member declaration---
        // Максимальная ширина числа из 0,1,2...,9, померенная в нормальном шрифте(дефалтовый для книги) в px(целое)
        // Ecma-376 Office Open XML Part 1, пункт 18.3.1.13
        this.maxDigitWidth = maxDigitWidth;

        this.nBaseColWidth = 8; // Число символов для дефалтовой ширины (по умолчинию 8)
        this.defaultColWidthChars = 0;
        this.defaultColWidth = 0;
        this.defaultRowHeight = 0;
        this.defaultRowDescender = 0;
        this.headersLeft = 0;
        this.headersTop = 0;
        this.headersWidth = 0;
        this.headersHeight = 0;
        this.headersHeightByFont = 0;	// Размер по шрифту (размер без скрытия заголовков)
        this.cellsLeft = 0;
        this.cellsTop = 0;
        this.cols = [];
        this.rows = [];
        this.width_1px = 0;
        this.width_2px = 0;
        this.width_3px = 0;
        this.width_4px = 0;
        this.width_padding = 0;
        this.height_1px = 0;
        this.height_2px = 0;
        this.height_3px = 0;
        this.height_4px = 0;
        this.highlightedCol = -1;
        this.highlightedRow = -1;
        this.topLeftFrozenCell = null;	// Верхняя ячейка для закрепления диапазона
        this.visibleRange = new asc_Range( 0, 0, 0, 0 );
        this.activeRange = new asc_ActiveRange( 0, 0, 0, 0 );
        this.isChanged = false;
        this.isCellEditMode = false;
        this.isFormulaEditMode = false;
        this.isChartAreaEditMode = false;
        this.lockDraw = false;
        this.isSelectOnShape = false;	// Выделен shape

        this.stateFormatPainter = c_oAscFormatPainterState.kOff;

        this.selectionDialogType = c_oAscSelectionDialogType.None;
        this.isSelectionDialogMode = false;
        this.copyActiveRange = null;

        this.startCellMoveResizeRange = null;
        this.startCellMoveResizeRange2 = null;
        this.moveRangeDrawingObjectTo = null;

        // Координаты ячейки начала перемещения диапазона
        this.startCellMoveRange = null;
        // Дипазон перемещения
        this.activeMoveRange = null;
        // Координаты fillHandle ("квадрата" автозаполнения)
        this.fillHandleL = 0;
        this.fillHandleT = 0;
        this.fillHandleR = 0;
        this.fillHandleB = 0;
        // Range fillHandle
        this.activeFillHandle = null;
        // Горизонтальное (0) или вертикальное (1) направление автозаполнения
        this.fillHandleDirection = -1;
        // Зона автозаполнения
        this.fillHandleArea = -1;
        this.nRowsCount = 0;
        this.nColsCount = 0;
        // Массив ячеек для текущей формулы
        this.arrActiveFormulaRanges = [];
        this.arrActiveFormulaRangesPosition = -1;
        this.arrActiveChartsRanges = [];
        //------------------------

        this.collaborativeEditing = collaborativeEditing;

        // Auto filters
        this.autoFilters = new asc.AutoFilters( this );
        this.drawingArea = new DrawingArea( this );
        this.cellCommentator = new CCellCommentator( this );
        this.objectRender = null;

        this._init();

        return this;
    }

    WorksheetView.prototype.getCellVisibleRange = function ( col, row ) {
        var vr, offsetX = 0, offsetY = 0, cFrozen, rFrozen;
        if ( this.topLeftFrozenCell ) {
            cFrozen = this.topLeftFrozenCell.getCol0() - 1;
            rFrozen = this.topLeftFrozenCell.getRow0() - 1;
            if ( col <= cFrozen && row <= rFrozen ) {
                vr = new asc_Range( 0, 0, cFrozen, rFrozen );
            }
            else if ( col <= cFrozen ) {
                vr = new asc_Range( 0, this.visibleRange.r1, cFrozen, this.visibleRange.r2 );
                offsetY -= this.rows[rFrozen + 1].top - this.cellsTop;
            }
            else if ( row <= rFrozen ) {
                vr = new asc_Range( this.visibleRange.c1, 0, this.visibleRange.c2, rFrozen );
                offsetX -= this.cols[cFrozen + 1].left - this.cellsLeft;
            }
            else {
                vr = this.visibleRange;
                offsetX -= this.cols[cFrozen + 1].left - this.cellsLeft;
                offsetY -= this.rows[rFrozen + 1].top - this.cellsTop;
            }
        }
        else {
            vr = this.visibleRange;
        }

        offsetX += this.cols[vr.c1].left - this.cellsLeft;
        offsetY += this.rows[vr.r1].top - this.cellsTop;

        return vr.contains( col, row ) ? new asc_VR( vr, offsetX, offsetY ) : null;
    };

    WorksheetView.prototype.getCellMetrics = function ( col, row ) {
        var vr, nColSize, nRowSize;
        if ( vr = this.getCellVisibleRange( col, row ) ) {
            nColSize = this.getColSize( col );
            nRowSize = this.getRowSize( row );
            if ( nColSize && nRowSize ) {
                return {
                    left  : nColSize.left - vr.offsetX,
                    top   : nRowSize.top - vr.offsetY,
                    width : nColSize.width,
                    height: nRowSize.height
                };
            }
        }
        return null;
    };

    WorksheetView.prototype.getColSize = function ( col ) {
        return (col >= 0 && col < this.cols.length) ? this.cols[col] : null;
    };

    WorksheetView.prototype.getRowSize = function ( row ) {
        return (row >= 0 && row < this.rows.length) ? this.rows[row] : null;
    };

    WorksheetView.prototype.getFrozenCell = function () {
        return this.topLeftFrozenCell;
    };

    WorksheetView.prototype.getVisibleRange = function () {
        return this.visibleRange;
    };

    WorksheetView.prototype.updateVisibleRange = function () {
        return this._updateCellsRange( this.getVisibleRange() );
    };

    WorksheetView.prototype.getFirstVisibleCol = function ( allowPane ) {
        var tmp = 0;
        if ( allowPane && this.topLeftFrozenCell ) {
            tmp = this.topLeftFrozenCell.getCol0();
        }
        return this.visibleRange.c1 - tmp;
    };

    WorksheetView.prototype.getLastVisibleCol = function () {
        return this.visibleRange.c2;
    };

    WorksheetView.prototype.getFirstVisibleRow = function ( allowPane ) {
        var tmp = 0;
        if ( allowPane && this.topLeftFrozenCell ) {
            tmp = this.topLeftFrozenCell.getRow0();
        }
        return this.visibleRange.r1 - tmp;
    };

    WorksheetView.prototype.getLastVisibleRow = function () {
        return this.visibleRange.r2;
    };

    WorksheetView.prototype.getHorizontalScrollRange = function () {
        var ctxW = this.drawingCtx.getWidth() - this.cellsLeft;
        for ( var w = 0, i = this.cols.length - 1; i >= 0; --i ) {
            w += this.cols[i].width;
            if ( w > ctxW ) {
                break;
            }
        }
        return i; // Диапазон скрола должен быть меньше количества столбцов, чтобы не было прибавления столбцов при перетаскивании бегунка
    };

    WorksheetView.prototype.getVerticalScrollRange = function () {
        var ctxH = this.drawingCtx.getHeight() - this.cellsTop;
        for ( var h = 0, i = this.rows.length - 1; i >= 0; --i ) {
            h += this.rows[i].height;
            if ( h > ctxH ) {
                break;
            }
        }
        return i; // Диапазон скрола должен быть меньше количества строк, чтобы не было прибавления строк при перетаскивании бегунка
    };

    WorksheetView.prototype.getCellsOffset = function ( units ) {
        var u = units >= 0 && units <= 3 ? units : 0;
        return {
            left: this.cellsLeft * asc_getcvt( 1/*pt*/, u, this._getPPIX() ),
            top : this.cellsTop * asc_getcvt( 1/*pt*/, u, this._getPPIY() )
        };
    };

    WorksheetView.prototype.getCellLeft = function ( column, units ) {
        if ( column >= 0 && column < this.cols.length ) {
            var u = units >= 0 && units <= 3 ? units : 0;
            return this.cols[column].left * asc_getcvt( 1/*pt*/, u, this._getPPIX() );
        }
        return null;
    };

    WorksheetView.prototype.getCellTop = function ( row, units ) {
        if ( row >= 0 && row < this.rows.length ) {
            var u = units >= 0 && units <= 3 ? units : 0;
            return this.rows[row].top * asc_getcvt( 1/*pt*/, u, this._getPPIY() );
        }
        return null;
    };

    WorksheetView.prototype.getCellLeftRelative = function ( col, units ) {
        if ( col < 0 || col >= this.cols.length ) {
            return null;
        }
        // С учетом видимой области
        var offsetX = 0;
        if ( this.topLeftFrozenCell ) {
            var cFrozen = this.topLeftFrozenCell.getCol0();
            offsetX = (col < cFrozen) ? 0 : this.cols[this.visibleRange.c1].left - this.cols[cFrozen].left;
        }
        else {
            offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft;
        }

        var u = units >= 0 && units <= 3 ? units : 0;
        return (this.cols[col].left - offsetX) * asc_getcvt( 1/*pt*/, u, this._getPPIX() );
    };

    WorksheetView.prototype.getCellTopRelative = function ( row, units ) {
        if ( row < 0 || row >= this.rows.length ) {
            return null;
        }
        // С учетом видимой области
        var offsetY = 0;
        if ( this.topLeftFrozenCell ) {
            var rFrozen = this.topLeftFrozenCell.getRow0();
            offsetY = (row < rFrozen) ? 0 : this.rows[this.visibleRange.r1].top - this.rows[rFrozen].top;
        }
        else {
            offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop;
        }

        var u = units >= 0 && units <= 3 ? units : 0;
        return (this.rows[row].top - offsetY) * asc_getcvt( 1/*pt*/, u, this._getPPIY() );
    };

    WorksheetView.prototype.getColumnWidth = function ( index, units ) {
        if ( index >= 0 && index < this.cols.length ) {
            var u = units >= 0 && units <= 3 ? units : 0;
            return this.cols[index].width * asc_getcvt( 1/*pt*/, u, this._getPPIX() );
        }
        return null;
    };

    WorksheetView.prototype.getSelectedColumnWidthInSymbols = function () {
        var c, res = null;
        for ( c = this.activeRange.c1; c <= this.activeRange.c2 && c < this.cols.length; ++c ) {
            if ( null === res ) {
                res = this.cols[c].charCount;
            }
            else if ( res !== this.cols[c].charCount ) {
                return null;
            }
        }
        // ToDo сравнить с default для проверки выделения всего
        return res;
    };

    WorksheetView.prototype.getSelectedRowHeight = function () {
        var r, res = null;
        for ( r = this.activeRange.r1; r <= this.activeRange.r2 && r < this.rows.length; ++r ) {
            if ( null === res ) {
                res = this.rows[r].heightReal;
            }
            else if ( res !== this.rows[r].heightReal ) {
                return null;
            }
        }
        // ToDo сравнить с default для проверки выделения всего
        return res;
    };

    WorksheetView.prototype.getRowHeight = function ( index, units ) {
        if ( index >= 0 && index < this.rows.length ) {
            var u = units >= 0 && units <= 3 ? units : 0;
            return this.rows[index].height * asc_getcvt( 1/*pt*/, u, this._getPPIY() );
        }
        return null;
    };

    WorksheetView.prototype.getSelectedColumnIndex = function () {
        return this.activeRange.startCol;
    };

    WorksheetView.prototype.getSelectedRowIndex = function () {
        return this.activeRange.startRow;
    };

    WorksheetView.prototype.getSelectedRange = function () {
        return this._getRange( this.activeRange.c1, this.activeRange.r1, this.activeRange.c2, this.activeRange.r2 );
    };

    WorksheetView.prototype.resize = function ( isUpdate ) {
        if ( isUpdate ) {
            this._initCellsArea( true );
            this._normalizeViewRange();
            this._cleanCellsTextMetricsCache();
            this._prepareCellTextMetricsCache();
            this.updateResize = false;

            this.objectRender.resizeCanvas();
        }
        else {
            this.updateResize = true;
        }
        return this;
    };

    WorksheetView.prototype.getZoom = function () {
        return this.drawingCtx.getZoom();
    };

    WorksheetView.prototype.changeZoom = function ( isUpdate ) {
        if ( isUpdate ) {
            this.notUpdateRowHeight = true;
            this.cleanSelection();
            this._initCellsArea( false );
            this._normalizeViewRange();
            this._cleanCellsTextMetricsCache();
            this._shiftVisibleRange();
            this._prepareCellTextMetricsCache();
            this._shiftVisibleRange();
            this.cellCommentator.updateCommentPosition();
            this.handlers.trigger( "onDocumentPlaceChanged" );
            this.objectRender.drawingArea.reinitRanges();
            this.updateZoom = false;
            this.notUpdateRowHeight = false;
        }
        else {
            this.updateZoom = true;
        }
        return this;
    };
    WorksheetView.prototype.changeZoomResize = function () {
        this.cleanSelection();
        this._initCellsArea( true );
        this._normalizeViewRange();
        this._cleanCellsTextMetricsCache();
        this._shiftVisibleRange();
        this._prepareCellTextMetricsCache();
        this._shiftVisibleRange();
        this.cellCommentator.updateCommentPosition();
        this.handlers.trigger( "onDocumentPlaceChanged" );
        this.objectRender.drawingArea.reinitRanges();

        this.updateResize = false;
        this.updateZoom = false;
    };

    WorksheetView.prototype.getCellTextMetrics = function ( col, row ) {
        var ct = this._getCellTextCache( col, row );
        return ct ? $.extend( {}, ct.metrics ) : undefined;
    };

    WorksheetView.prototype.getSheetViewSettings = function () {
        return this.model.getSheetViewSettings();
    };

    WorksheetView.prototype.getFrozenPaneOffset = function ( noX, noY ) {
        var offsetX = 0, offsetY = 0, c = this.cols, r = this.rows;
        if ( this.topLeftFrozenCell ) {
            if ( !noX ) {
                var cFrozen = this.topLeftFrozenCell.getCol0();
                offsetX = c[cFrozen].left - c[0].left;
            }
            if ( !noY ) {
                var rFrozen = this.topLeftFrozenCell.getRow0();
                offsetY = r[rFrozen].top - r[0].top;
            }
        }
        return {offsetX: offsetX, offsetY: offsetY};
    };


    // mouseX - это разница стартовых координат от мыши при нажатии и границы
    WorksheetView.prototype.changeColumnWidth = function ( col, x2, mouseX ) {
        var t = this;

        x2 *= asc_getcvt( 0/*px*/, 1/*pt*/, t._getPPIX() );
        // Учитываем координаты точки, где мы начали изменение размера
        x2 += mouseX;

        var offsetFrozenX = 0;
        var c1 = t.visibleRange.c1;
        if ( this.topLeftFrozenCell ) {
            var cFrozen = this.topLeftFrozenCell.getCol0() - 1;
            if ( 0 <= cFrozen ) {
                if ( col < c1 ) {
                    c1 = 0;
                }
                else {
                    offsetFrozenX = t.cols[cFrozen].left + t.cols[cFrozen].width - t.cols[0].left;
                }
            }
        }
        var offsetX = t.cols[c1].left - t.cellsLeft;
        offsetX -= offsetFrozenX;

        var x1 = t.cols[col].left - offsetX - this.width_1px;
        var w = Math.max( x2 - x1, 0 );
        if ( w === t.cols[col].width ) {
            return;
        }
        var cc = Math.min( t._colWidthToCharCount( w ), c_oAscMaxColumnWidth );
        var cw = t._charCountToModelColWidth( cc );

        var onChangeWidthCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            t.model.setColWidth( cw, col, col );
            t._cleanCache( new asc_Range( 0, 0, t.cols.length - 1, t.rows.length - 1 ) );
            t.changeWorksheet( "update", {reinitRanges: true} );
            t._updateVisibleColsCount();
            if ( t.objectRender ) {
                t.objectRender.updateSizeDrawingObjects( {target: c_oTargetType.ColumnResize, col: col} );
            }
        };
        this._isLockedAll( onChangeWidthCallback );
    };

    // mouseY - это разница стартовых координат от мыши при нажатии и границы
    WorksheetView.prototype.changeRowHeight = function ( row, y2, mouseY ) {
        var t = this;

        y2 *= asc_getcvt( 0/*px*/, 1/*pt*/, t._getPPIY() );
        // Учитываем координаты точки, где мы начали изменение размера
        y2 += mouseY;

        var offsetFrozenY = 0;
        var r1 = t.visibleRange.r1;
        if ( this.topLeftFrozenCell ) {
            var rFrozen = this.topLeftFrozenCell.getRow0() - 1;
            if ( 0 <= rFrozen ) {
                if ( row < r1 ) {
                    r1 = 0;
                }
                else {
                    offsetFrozenY = t.rows[rFrozen].top + t.rows[rFrozen].height - t.rows[0].top;
                }
            }
        }
        var offsetY = t.rows[r1].top - t.cellsTop;
        offsetY -= offsetFrozenY;

        var y1 = t.rows[row].top - offsetY - this.height_1px;
        var newHeight = Math.min( t.maxRowHeight, Math.max( y2 - y1, 0 ) );
        if ( newHeight === t.rows[row].height ) {
            return;
        }

        var onChangeHeightCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            t.model.setRowHeight( newHeight, row, row, true );
            t.autoFilters.reDrawFilter( null, row );
            t._cleanCache( new asc_Range( 0, row, t.cols.length - 1, row ) );
            t.changeWorksheet( "update", {reinitRanges: true} );
            t._updateVisibleRowsCount();
            if ( t.objectRender ) {
                t.objectRender.updateSizeDrawingObjects( {target: c_oTargetType.RowResize, row: row} );
            }
        };

        this._isLockedAll( onChangeHeightCallback );
    };


    // Проверяет, есть ли числовые значения в диапазоне
    WorksheetView.prototype._hasNumberValueInActiveRange = function () {
        var cell, cellType, isNumberFormat, arrCols = null, arrRows = null;
        if ( this._rangeIsSingleCell( this.activeRange ) ) {
            // Для одной ячейки не стоит ничего делать
            return null;
        }
        var mergedRange = this.model.getMergedByCell( this.activeRange.r1, this.activeRange.c1 );
        if ( mergedRange && mergedRange.isEqual( this.activeRange ) ) {
            // Для одной ячейки не стоит ничего делать
            return null;
        }

        for ( var c = this.activeRange.c1; c <= this.activeRange.c2; ++c ) {
            for ( var r = this.activeRange.r1; r <= this.activeRange.r2; ++r ) {
                cell = this._getCellTextCache( c, r );
                if ( cell ) {
                    // Нашли не пустую ячейку, проверим формат
                    cellType = cell.cellType;
                    isNumberFormat = (null == cellType || CellValueType.Number === cellType);
                    if ( isNumberFormat ) {
                        if ( !arrCols ) {
                            arrCols = [];
                            arrRows = [];
                        }
                        arrCols.push( c );
                        arrRows.push( r );
                    }
                }
            }
        }
        if ( arrCols ) {
            // Делаем массивы уникальными и сортируем
            arrCols = arrCols.filter( fOnlyUnique );
            arrRows = arrRows.filter( fOnlyUnique );
            return {arrCols: arrCols.sort( fSortAscending ), arrRows: arrRows.sort( fSortAscending )};
        }
        else {
            return null;
        }
    };

    // Автодополняет формулу диапазоном, если это возможно
    WorksheetView.prototype.autoCompleteFormula = function ( functionName ) {
        var t = this;
        this.activeRange.normalize();
        var ar = this.activeRange;
        var arCopy = null;
        var arHistorySelect = ar.clone( true );
        var vr = this.visibleRange;

        // Первая верхняя не числовая ячейка
        var topCell = null;
        // Первая левая не числовая ячейка
        var leftCell = null;

        var r = ar.startRow - 1;
        var c = ar.startCol - 1;
        var cell, cellType, isNumberFormat;
        var result = {};
        // Проверим, есть ли числовые значения в диапазоне
        var hasNumber = this._hasNumberValueInActiveRange();
        var val, text;

        if ( hasNumber ) {
            var i;
            // Есть ли значения в последней строке и столбце
            var hasNumberInLastColumn = (ar.c2 === hasNumber.arrCols[hasNumber.arrCols.length - 1]);
            var hasNumberInLastRow = (ar.r2 === hasNumber.arrRows[hasNumber.arrRows.length - 1]);

            // Нужно уменьшить зону выделения (если она реально уменьшилась)
            var startCol = hasNumber.arrCols[0];
            var startRow = hasNumber.arrRows[0];
            // Старые границы диапазона
            var startColOld = ar.c1;
            var startRowOld = ar.r1;
            // Нужно ли перерисовывать
            var bIsUpdate = false;
            if ( startColOld !== startCol || startRowOld !== startRow ) {
                bIsUpdate = true;
            }
            if ( true === hasNumberInLastRow && true === hasNumberInLastColumn ) {
                bIsUpdate = true;
            }
            if ( bIsUpdate ) {
                this.cleanSelection();
                ar.c1 = startCol;
                ar.r1 = startRow;
                if ( false === ar.contains( ar.startCol, ar.startRow ) ) {
                    // Передвинуть первую ячейку в выделении
                    ar.startCol = startCol;
                    ar.startRow = startRow;
                }
                if ( true === hasNumberInLastRow && true === hasNumberInLastColumn ) {
                    // Мы расширяем диапазон
                    if ( 1 === hasNumber.arrRows.length ) {
                        // Одна строка или только в последней строке есть значения... (увеличиваем вправо)
                        ar.c2 += 1;
                    }
                    else {
                        // Иначе вводим в строку вниз
                        ar.r2 += 1;
                    }
                }
                this._drawSelection();
            }

            arCopy = ar.clone( true );

            var functionAction = null;
            var changedRange = null;

            if ( false === hasNumberInLastColumn && false === hasNumberInLastRow ) {
                // Значений нет ни в последней строке ни в последнем столбце (значит нужно сделать формулы в каждой последней ячейке)
                changedRange = [new asc_Range( hasNumber.arrCols[0], arCopy.r2, hasNumber.arrCols[hasNumber.arrCols.length - 1], arCopy.r2 ), new asc_Range( arCopy.c2, hasNumber.arrRows[0], arCopy.c2, hasNumber.arrRows[hasNumber.arrRows.length - 1] )];
                functionAction = function () {
                    // Пройдемся по последней строке
                    for ( i = 0; i < hasNumber.arrCols.length; ++i ) {
                        c = hasNumber.arrCols[i];
                        cell = t._getVisibleCell( c, arCopy.r2 );
                        text = t._getCellTitle( c, arCopy.r1 ) + ":" + t._getCellTitle( c, arCopy.r2 - 1 );
                        val = "=" + functionName + "(" + text + ")";
                        // ToDo - при вводе формулы в заголовок автофильтра надо писать "0"
                        cell.setValue( val );
                    }
                    // Пройдемся по последнему столбцу
                    for ( i = 0; i < hasNumber.arrRows.length; ++i ) {
                        r = hasNumber.arrRows[i];
                        cell = t._getVisibleCell( arCopy.c2, r );
                        text = t._getCellTitle( arCopy.c1, r ) + ":" + t._getCellTitle( arCopy.c2 - 1, r );
                        val = "=" + functionName + "(" + text + ")";
                        cell.setValue( val );
                    }
                    // Значение в правой нижней ячейке
                    cell = t._getVisibleCell( arCopy.c2, arCopy.r2 );
                    text = t._getCellTitle( arCopy.c1, arCopy.r2 ) + ":" + t._getCellTitle( arCopy.c2 - 1, arCopy.r2 );
                    val = "=" + functionName + "(" + text + ")";
                    cell.setValue( val );
                };
            }
            else if ( true === hasNumberInLastRow && false === hasNumberInLastColumn ) {
                // Есть значения только в последней строке (значит нужно заполнить только последнюю колонку)
                changedRange = new asc_Range( arCopy.c2, hasNumber.arrRows[0], arCopy.c2, hasNumber.arrRows[hasNumber.arrRows.length - 1] );
                functionAction = function () {
                    // Пройдемся по последнему столбцу
                    for ( i = 0; i < hasNumber.arrRows.length; ++i ) {
                        r = hasNumber.arrRows[i];
                        cell = t._getVisibleCell( arCopy.c2, r );
                        text = t._getCellTitle( arCopy.c1, r ) + ":" + t._getCellTitle( arCopy.c2 - 1, r );
                        val = "=" + functionName + "(" + text + ")";
                        cell.setValue( val );
                    }
                };
            }
            else if ( false === hasNumberInLastRow && true === hasNumberInLastColumn ) {
                // Есть значения только в последнем столбце (значит нужно заполнить только последнюю строчку)
                changedRange = new asc_Range( hasNumber.arrCols[0], arCopy.r2, hasNumber.arrCols[hasNumber.arrCols.length - 1], arCopy.r2 );
                functionAction = function () {
                    // Пройдемся по последней строке
                    for ( i = 0; i < hasNumber.arrCols.length; ++i ) {
                        c = hasNumber.arrCols[i];
                        cell = t._getVisibleCell( c, arCopy.r2 );
                        text = t._getCellTitle( c, arCopy.r1 ) + ":" + t._getCellTitle( c, arCopy.r2 - 1 );
                        val = "=" + functionName + "(" + text + ")";
                        cell.setValue( val );
                    }
                };
            }
            else {
                // Есть значения и в последнем столбце, и в последней строке
                if ( 1 === hasNumber.arrRows.length ) {
                    changedRange = new asc_Range( arCopy.c2, arCopy.r2, arCopy.c2, arCopy.r2 );
                    functionAction = function () {
                        // Одна строка или только в последней строке есть значения...
                        cell = t._getVisibleCell( arCopy.c2, arCopy.r2 );
                        // ToDo вводить в первое свободное место, а не сразу за диапазоном
                        text = t._getCellTitle( arCopy.c1, arCopy.r2 ) + ":" + t._getCellTitle( arCopy.c2 - 1, arCopy.r2 );
                        val = "=" + functionName + "(" + text + ")";
                        cell.setValue( val );
                    };
                }
                else {
                    changedRange = new asc_Range( hasNumber.arrCols[0], arCopy.r2, hasNumber.arrCols[hasNumber.arrCols.length - 1], arCopy.r2 );
                    functionAction = function () {
                        // Иначе вводим в строку вниз
                        for ( i = 0; i < hasNumber.arrCols.length; ++i ) {
                            c = hasNumber.arrCols[i];
                            cell = t._getVisibleCell( c, arCopy.r2 );
                            // ToDo вводить в первое свободное место, а не сразу за диапазоном
                            text = t._getCellTitle( c, arCopy.r1 ) + ":" + t._getCellTitle( c, arCopy.r2 - 1 );
                            val = "=" + functionName + "(" + text + ")";
                            cell.setValue( val );
                        }
                    };
                }
            }

            var onAutoCompleteFormula = function ( isSuccess ) {
                if ( false === isSuccess ) {
                    return;
                }

                History.Create_NewPoint();
                History.SetSelection( arHistorySelect.clone() );
                History.SetSelectionRedo( arCopy.clone() );
                History.StartTransaction();

                asc_applyFunction( functionAction );
                t.handlers.trigger( "selectionMathInfoChanged", t.getSelectionMathInfo() );

                History.EndTransaction();
            };

            // Можно ли применять автоформулу
            this._isLockedCells( changedRange, /*subType*/null, onAutoCompleteFormula );

            result.notEditCell = true;
            return result;
        }

        // Ищем первую ячейку с числом
        for ( ; r >= vr.r1; --r ) {
            cell = this._getCellTextCache( ar.startCol, r );
            if ( cell ) {
                // Нашли не пустую ячейку, проверим формат
                cellType = cell.cellType;
                isNumberFormat = (null === cellType || CellValueType.Number === cellType);
                if ( isNumberFormat ) {
                    // Это число, мы нашли то, что искали
                    topCell = {
                        c: ar.startCol, r: r, isFormula: cell.isFormula
                    };
                    // смотрим вторую ячейку
                    if ( topCell.isFormula && r - 1 >= vr.r1 ) {
                        cell = this._getCellTextCache( ar.startCol, r - 1 );
                        if ( cell && cell.isFormula ) {
                            topCell.isFormulaSeq = true;
                        }
                    }
                    break;
                }
            }
        }
        // Проверим, первой все равно должна быть колонка
        if ( null === topCell || topCell.r !== ar.startRow - 1 || topCell.isFormula && !topCell.isFormulaSeq ) {
            for ( ; c >= vr.c1; --c ) {
                cell = this._getCellTextCache( c, ar.startRow );
                if ( cell ) {
                    // Нашли не пустую ячейку, проверим формат
                    cellType = cell.cellType;
                    isNumberFormat = (null === cellType || CellValueType.Number === cellType);
                    if ( isNumberFormat ) {
                        // Это число, мы нашли то, что искали
                        leftCell = {
                            r: ar.startRow, c: c
                        };
                        break;
                    }
                }
                if ( null !== topCell ) {
                    // Если это не первая ячейка слева от текущей и мы нашли верхнюю, то дальше не стоит искать
                    break;
                }
            }
        }

        if ( leftCell ) {
            // Идем влево до первой не числовой ячейки
            --c;
            for ( ; c >= 0; --c ) {
                cell = this._getCellTextCache( c, ar.startRow );
                if ( !cell ) {
                    // Могут быть еще не закешированные данные
                    this._addCellTextToCache( c, ar.startRow );
                    cell = this._getCellTextCache( c, ar.startRow );
                    if ( !cell ) {
                        break;
                    }
                }
                cellType = cell.cellType;
                isNumberFormat = (null === cellType || CellValueType.Number === cellType);
                if ( !isNumberFormat ) {
                    break;
                }
            }
            // Мы ушли чуть дальше
            ++c;
            // Диапазон или только 1 ячейка
            if ( ar.startCol - 1 !== c ) {
                // Диапазон
                result = new asc_Range( c, leftCell.r, ar.startCol - 1, leftCell.r );
            }
            else {
                // Одна ячейка
                result = new asc_Range( c, leftCell.r, c, leftCell.r );
            }
            result.type = c_oAscSelectionType.RangeCells;
            this._fixSelectionOfMergedCells( result );
            if ( result.c1 === result.c2 && result.r1 === result.r2 ) {
                result.text = this._getCellTitle( result.c1, result.r1 );
            }
            else {
                result.text = this._getCellTitle( result.c1, result.r1 ) + ":" + this._getCellTitle( result.c2, result.r2 );
            }
            return result;
        }

        if ( topCell ) {
            // Идем вверх до первой не числовой ячейки
            --r;
            for ( ; r >= 0; --r ) {
                cell = this._getCellTextCache( ar.startCol, r );
                if ( !cell ) {
                    // Могут быть еще не закешированные данные
                    this._addCellTextToCache( ar.startCol, r );
                    cell = this._getCellTextCache( ar.startCol, r );
                    if ( !cell ) {
                        break;
                    }
                }
                cellType = cell.cellType;
                isNumberFormat = (null === cellType || CellValueType.Number === cellType);
                if ( !isNumberFormat ) {
                    break;
                }
            }
            // Мы ушли чуть дальше
            ++r;
            // Диапазон или только 1 ячейка
            if ( ar.startRow - 1 !== r ) {
                // Диапазон
                result = new asc_Range( topCell.c, r, topCell.c, ar.startRow - 1 );
            }
            else {
                // Одна ячейка
                result = new asc_Range( topCell.c, r, topCell.c, r );
            }
            result.type = c_oAscSelectionType.RangeCells;
            this._fixSelectionOfMergedCells( result );
            if ( result.c1 === result.c2 && result.r1 === result.r2 ) {
                result.text = this._getCellTitle( result.c1, result.r1 );
            }
            else {
                result.text = this._getCellTitle( result.c1, result.r1 ) + ":" + this._getCellTitle( result.c2, result.r2 );
            }
            return result;
        }
    };

    // ----- Initialization -----

    WorksheetView.prototype._init = function () {
        this._initConstValues();
        this._initWorksheetDefaultWidth();
        this._initPane();
        this._initCellsArea( true );
        this.autoFilters.addFiltersAfterOpen();
        this._initConditionalFormatting();
        this._cleanCellsTextMetricsCache();
        this._prepareCellTextMetricsCache();

        // initializing is completed
        this.handlers.trigger( "initialized" );
    };

    WorksheetView.prototype._initConditionalFormatting = function () {
        var oGradient = null;
        var aCFs = this.model.aConditionalFormatting;
        var aRules, oRule;
        var oRuleElement = null;
        var min = Number.MAX_VALUE;
        var max = -Number.MAX_VALUE;
        var tmp;
        var arrayCells = [];
        for ( var i in aCFs ) {
            aRules = aCFs[i].aRules;
            if ( 0 >= aRules.length ) {
                continue;
            }
            for ( var j in aRules ) {
                oRule = aRules[j];
                // ToDo aboveAverage, beginsWith, cellIs, containsBlanks, containsErrors,
                // ToDo containsText, dataBar, duplicateValues, endsWith, expression, iconSet, notContainsBlanks,
                // ToDo notContainsErrors, notContainsText, timePeriod, top10, uniqueValues (page 2679)
                switch ( oRule.Type ) {
                    case Asc.ECfType.colorScale:
                        if ( 1 !== oRule.aRuleElements.length ) {
                            break;
                        }
                        oRuleElement = oRule.aRuleElements[0];
                        // ToDo убрать null === aCFs[i].SqRefRange когда научимся мультиселект обрабатывать (\\192.168.5.2\source\DOCUMENTS\XLSX\Matematika Quantum Sedekah.xlsx)
                        if ( !(oRuleElement instanceof asc.CColorScale) || null === aCFs[i].SqRefRange ) {
                            break;
                        }
                        aCFs[i].SqRefRange._setPropertyNoEmpty( null, null, function ( c ) {
                            if ( CellValueType.Number === c.getType() && false === c.isEmptyTextString() ) {
                                tmp = parseFloat( c.getValueWithoutFormat() );
                                if ( isNaN( tmp ) ) {
                                    return;
                                }
                                arrayCells.push( {cell: c, val: tmp} );
                                min = Math.min( min, tmp );
                                max = Math.max( max, tmp );
                            }
                        } );

                        // ToDo CFVO Type (formula, max, min, num, percent, percentile) (page 2681)
                        // ToDo support 3 colors in rule
                        if ( 0 < arrayCells.length && 2 === oRuleElement.aColors.length ) {
                            oGradient = new asc.CGradient( oRuleElement.aColors[0], oRuleElement.aColors[1] );
                            oGradient.init( min, max );

                            for ( var cell in arrayCells ) {
                                var dxf = new CellXfs();
                                dxf.fill = new Fill( {bg: oGradient.calculateColor( arrayCells[cell].val )} );
                                arrayCells[cell].cell.setConditionalFormattingStyle( dxf );
                            }
                        }

                        arrayCells.splice( 0, arrayCells.length );
                        min = Number.MAX_VALUE;
                        max = -Number.MAX_VALUE;
                        break;
                }
            }
        }
    };

    WorksheetView.prototype._prepareComments = function () {
        // Теперь получение всех комментариев через asc_getWorkbookComments
        var commentList = this.cellCommentator.prepareComments( this.model.aComments );
        if ( 0 < commentList.length ) {
            this.model.workbook.handlers.trigger( "asc_onAddComments", commentList );
        }
    };

    WorksheetView.prototype._prepareDrawingObjects = function () {
        this.objectRender = new DrawingObjects();
        if ( !window["NATIVE_EDITOR_ENJINE"] || window['IS_NATIVE_EDITOR'] || window['DoctRendererMode'] ) {
            this.objectRender.init( this );
        }
    };

    WorksheetView.prototype._initWorksheetDefaultWidth = function () {
        this.nBaseColWidth = this.model.oSheetFormatPr.nBaseColWidth || this.nBaseColWidth;
        // Теперь рассчитываем число px
        var defaultColWidthChars = this._charCountToModelColWidth( this.nBaseColWidth );
        this.defaultColWidthPx = this._modelColWidthToColWidth( defaultColWidthChars ) * asc_getcvt( 1/*pt*/, 0/*px*/, 96 );
        // Делаем кратным 8 (http://support.microsoft.com/kb/214123)
        this.defaultColWidthPx = asc_ceil( this.defaultColWidthPx / 8 ) * 8;
        this.defaultColWidthChars = this._colWidthToCharCount( this.defaultColWidthPx * asc_getcvt( 0/*px*/, 1/*pt*/, 96 ) );

        gc_dDefaultColWidthCharsAttribute = this._charCountToModelColWidth( this.defaultColWidthChars );
        this.defaultColWidth = this._modelColWidthToColWidth( gc_dDefaultColWidthCharsAttribute );

        var defaultFontSize = this.model.getDefaultFontSize();
        // ToDo разобраться со значениями
        this._setFont( undefined, this.model.getDefaultFontName(), defaultFontSize );
        var tm = this._roundTextMetrics( this.stringRender.measureString( "A" ) );
        this.headersHeightByFont = tm.height;

        this.maxRowHeight = asc_calcnpt( c_oAscMaxRowHeight, this._getPPIY() );
        this.defaultRowDescender = this._calcRowDescender( defaultFontSize );
        gc_dDefaultRowHeightAttribute = this.defaultRowHeight = this.model.getDefaultHeight() || Math.max( asc_calcnpt( defaultFontSize * this.vspRatio, this._getPPIY() ) + this.height_1px, this.headersHeightByFont );

        // Инициализируем число колонок и строк (при открытии). Причем нужно поставить на 1 больше,
        // чтобы могли показать последнюю строку/столбец (http://bugzserver/show_bug.cgi?id=23513)
        this.nColsCount = Math.min( this.model.getColsCount() + 1, gc_nMaxCol );
        this.nRowsCount = Math.min( this.model.getRowsCount() + 1, gc_nMaxRow );
    };

    WorksheetView.prototype._initConstValues = function () {
        var ppiX = this._getPPIX();
        var ppiY = this._getPPIY();
        this.width_1px = asc_calcnpt( 0, ppiX, 1/*px*/ );
        this.width_2px = asc_calcnpt( 0, ppiX, 2/*px*/ );
        this.width_3px = asc_calcnpt( 0, ppiX, 3/*px*/ );
        this.width_4px = asc_calcnpt( 0, ppiX, 4/*px*/ );
        this.width_padding = asc_calcnpt( 0, ppiX, this.settings.cells.padding/*px*/ );

        this.height_1px = asc_calcnpt( 0, ppiY, 1/*px*/ );
        this.height_2px = asc_calcnpt( 0, ppiY, 2/*px*/ );
        this.height_3px = asc_calcnpt( 0, ppiY, 3/*px*/ );
        this.height_4px = asc_calcnpt( 0, ppiY, 4/*px*/ );
    };

    WorksheetView.prototype._initCellsArea = function ( fullRecalc ) {
        // calculate rows heights and visible rows
        if ( !(window["NATIVE_EDITOR_ENJINE"] && this.notUpdateRowHeight) ) {
            this._calcHeaderRowHeight();
            this._calcHeightRows( fullRecalc ? 1 : 0 );
        }
        this.visibleRange.r2 = 0;
        this._calcVisibleRows();
        this._updateVisibleRowsCount( /*skipScrolReinit*/true );

        // calculate columns widths and visible columns
        if ( !(window["NATIVE_EDITOR_ENJINE"] && this.notUpdateRowHeight) ) {
            this._calcHeaderColumnWidth();
            this._calcWidthColumns( fullRecalc ? 1 : 0 );
        }
        this.visibleRange.c2 = 0;
        this._calcVisibleColumns();
        this._updateVisibleColsCount( /*skipScrolReinit*/true );
    };

    WorksheetView.prototype._initPane = function () {
        var pane = this.model.sheetViews[0].pane;
        if ( null !== pane && pane.isInit() ) {
            this.topLeftFrozenCell = pane.topLeftFrozenCell;
            this.visibleRange.r1 = this.topLeftFrozenCell.getRow0();
            this.visibleRange.c1 = this.topLeftFrozenCell.getCol0();
        }
    };

    WorksheetView.prototype._fixVisibleRange = function ( range ) {
        var tmp;
        if ( null !== this.topLeftFrozenCell ) {
            tmp = this.topLeftFrozenCell.getRow0();
            if ( range.r1 < tmp ) {
                range.r1 = tmp;
                tmp = this._findVisibleRow( range.r1, +1 );
                if ( 0 < tmp ) {
                    range.r1 = tmp;
                }
            }
            tmp = this.topLeftFrozenCell.getCol0();
            if ( range.c1 < tmp ) {
                range.c1 = tmp;
                tmp = this._findVisibleCol( range.c1, +1 );
                if ( 0 < tmp ) {
                    range.c1 = tmp;
                }
            }
        }
    };

    /**
     * Вычисляет ширину столбца для заданного количества символов
     * @param {Number} count  Количество символов
     * @returns {Number}      Ширина столбца в символах
     */
    WorksheetView.prototype._charCountToModelColWidth = function ( count ) {
        if ( count <= 0 ) {
            return 0;
        }
        return asc_floor( (count * this.maxDigitWidth + this.settings.cells.paddingPlusBorder) / this.maxDigitWidth * 256 ) / 256;
    };

    /**
     * Вычисляет ширину столбца в пунктах
     * @param {Number} mcw  Количество символов
     * @returns {Number}    Ширина столбца в пунктах (pt)
     */
    WorksheetView.prototype._modelColWidthToColWidth = function ( mcw ) {
        var px = asc_floor( ((256 * mcw + asc_floor( 128 / this.maxDigitWidth )) / 256) * this.maxDigitWidth );
        return px * asc_getcvt( 0/*px*/, 1/*pt*/, 96 );
    };

    /**
     * Вычисляет количество символов по ширине столбца
     * @param {Number} w  Ширина столбца в пунктах
     * @returns {Number}  Количество символов
     */
    WorksheetView.prototype._colWidthToCharCount = function ( w ) {
        var px = w * asc_getcvt( 1/*pt*/, 0/*px*/, 96 );
        var pxInOneCharacter = this.maxDigitWidth + this.settings.cells.paddingPlusBorder;
        // Когда меньше 1 символа, то просто считаем по пропорции относительно размера 1-го символа
        return px < pxInOneCharacter ? (1 - asc_floor( 100 * (pxInOneCharacter - px) / pxInOneCharacter + 0.49999 ) / 100) : asc_floor( (px - this.settings.cells.paddingPlusBorder) / this.maxDigitWidth * 100 + 0.5 ) / 100;
    };

    /**
     * Вычисляет ширину столбца для отрисовки
     * @param {Number} w  Ширина столбца в символах
     * @returns {Number}  Ширина столбца в пунктах (pt)
     */
    WorksheetView.prototype._calcColWidth = function ( w ) {
        var t = this;
        var res = {};
        var useDefault = w === undefined || w === null || w === -1;
        var width;
        res.width = useDefault ? t.defaultColWidth : (width = t._modelColWidthToColWidth( w ), (width < t.width_1px ? 0 : width));
        res.innerWidth = Math.max( res.width - this.width_padding * 2 - this.width_1px, 0 );
        res.charCount = t._colWidthToCharCount( res.width );
        return res;
    };

    /**
     * Вычисляет Descender строки
     * @param {Number} fontSize
     * @returns {Number}
     */
    WorksheetView.prototype._calcRowDescender = function ( fontSize ) {
        return asc_calcnpt( fontSize * (this.vspRatio - 1), this._getPPIY() ); // ToDo возможно стоит тоже использовать 96
    };

    /** Вычисляет ширину колонки заголовков (в pt) */
    WorksheetView.prototype._calcHeaderColumnWidth = function () {
        if ( false === this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            this.headersWidth = 0;
        }
        else {
            // Ширина колонки заголовков считается  - max число знаков в строке - перевести в символы - перевести в пикселы
            var numDigit = Math.max( calcDecades( this.visibleRange.r2 + 1 ), 3 );
            var nCharCount = this._charCountToModelColWidth( numDigit );
            this.headersWidth = this._modelColWidthToColWidth( nCharCount );
        }

        //var w = this.emSize * Math.max( calcDecades(this.visibleRange.r2 + 1), 3) * 1.25;
        //this.headersWidth = asc_calcnpt(w, this._getPPIX());

        this.cellsLeft = this.headersLeft + this.headersWidth;
    };

    /** Вычисляет высоту строки заголовков (в pt) */
    WorksheetView.prototype._calcHeaderRowHeight = function () {
        if ( false === this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            this.headersHeight = 0;
        }
        else
        //this.headersHeight = this.model.getDefaultHeight() || this.defaultRowHeight;
        {
            this.headersHeight = this.headersHeightByFont + this.height_1px;
        }

        //this.headersHeight = asc_calcnpt( this.settings.header.fontSize * this.vspRatio, this._getPPIY() );
        this.cellsTop = this.headersTop + this.headersHeight;
    };

    /**
     * Вычисляет ширину и позицию колонок (в pt)
     * @param {Number} fullRecalc  0 - без пересчета; 1 - пересчитываем все; 2 - пересчитываем новые строки
     */
    WorksheetView.prototype._calcWidthColumns = function ( fullRecalc ) {
        var x = this.cellsLeft;
        var visibleW = this.drawingCtx.getWidth();
        var obr = this.objectRender ? this.objectRender.getDrawingAreaMetrics() : {maxCol: 0, maxRow: 0};
        var l = Math.max( this.model.getColsCount() + 1, this.nColsCount, obr.maxCol );
        var i = 0, w, column, isBestFit, hiddenW = 0;

        // Берем дефалтовую ширину документа
        var defaultWidth = this.model.getDefaultWidth();
        defaultWidth = (typeof defaultWidth === "number" && defaultWidth >= 0) ? defaultWidth : -1;

        if ( 1 === fullRecalc ) {
            this.cols = [];
        }
        else if ( 2 === fullRecalc ) {
            i = this.cols.length;
            x = this.cols[i - 1].left + this.cols[i - 1].width;
        }
        for ( ; ((0 !== fullRecalc) ? i < l || x + hiddenW < visibleW : i < this.cols.length) && i < gc_nMaxCol; ++i ) {
            // Получаем свойства колонки
            column = this.model._getColNoEmptyWithAll( i );
            if ( !column ) {
                w = defaultWidth; // Используем дефолтное значение
                isBestFit = true; // Это уже оптимальная ширина
            }
            else if ( column.hd ) {
                w = 0;            // Если столбец скрытый, ширину выставляем 0
                isBestFit = false;
                hiddenW += this._calcColWidth( column.width ).width;
            }
            else {
                w = column.width || defaultWidth;
                isBestFit = !!(column.BestFit || (null === column.BestFit && null === column.CustomWidth));
            }
            this.cols[i] = this._calcColWidth( w );
            this.cols[i].isCustomWidth = !isBestFit;
            this.cols[i].left = x;
            x += this.cols[i].width;
        }

        this.nColsCount = Math.min( Math.max( this.nColsCount, i ), gc_nMaxCol );
    };

    /**
     * Вычисляет высоту и позицию строк (в pt)
     * @param {Number} fullRecalc  0 - без пересчета; 1 - пересчитываем все; 2 - пересчитываем новые строки
     */
    WorksheetView.prototype._calcHeightRows = function ( fullRecalc ) {
        var y = this.cellsTop;
        var visibleH = this.drawingCtx.getHeight();
        var obr = this.objectRender ? this.objectRender.getDrawingAreaMetrics() : {maxCol: 0, maxRow: 0};
        var l = Math.max( this.model.getRowsCount() + 1, this.nRowsCount, obr.maxRow );
        var defaultH = this.defaultRowHeight;
        var i = 0, h, hR, isCustomHeight, row, hiddenH = 0;

        if ( 1 === fullRecalc ) {
            this.rows = [];
        }
        else if ( 2 === fullRecalc ) {
            i = this.rows.length;
            y = this.rows[i - 1].top + this.rows[i - 1].height;
        }
        for ( ; ((0 !== fullRecalc) ? i < l || y + hiddenH < visibleH : i < this.rows.length) && i < gc_nMaxRow; ++i ) {
            row = this.model._getRowNoEmptyWithAll( i );
            if ( !row ) {
                h = -1; // Будет использоваться дефолтная высота строки
                isCustomHeight = false;
            }
            else if ( 0 != (g_nRowFlag_hd & row.flags) ) {
                hR = h = 0;  // Скрытая строка, высоту выставляем 0
                isCustomHeight = true;
                hiddenH += row.h > 0 ? row.h - this.height_1px : defaultH;
            }
            else {
                isCustomHeight = 0 != (g_nRowFlag_CustomHeight & row.flags);
                // Берем высоту из модели, если она custom(баг 15618), либо дефолтную
                if ( row.h > 0 && isCustomHeight ) {
                    hR = row.h;
                    h = hR / 0.75;
                    h = (h | h) * 0.75;			// 0.75 - это размер 1px в pt (можно было 96/72)
                }
                else {
                    h = -1;
                }
            }
            h = h < 0 ? (hR = defaultH) : h;
            this.rows[i] = {
                top            : y,
                height         : h,												// Высота с точностью до 1 px
                heightReal     : hR,											// Реальная высота из файла (может быть не кратна 1 px, в Excel можно выставить через меню строки)
                descender      : this.defaultRowDescender,
                isCustomHeight : isCustomHeight,
                isDefaultHeight: !(row && row.h > 0 && isCustomHeight)  // Высота строки, вычисленная на основе текста
            };
            y += this.rows[i].height;
        }

        this.nRowsCount = Math.min( Math.max( this.nRowsCount, i ), gc_nMaxRow );
    };

    /** Вычисляет диапазон индексов видимых колонок */
    WorksheetView.prototype._calcVisibleColumns = function () {
        var l = this.cols.length;
        var w = this.drawingCtx.getWidth();
        var sumW = this.topLeftFrozenCell ? this.cols[this.topLeftFrozenCell.getCol0()].left : this.cellsLeft;
        for ( var i = this.visibleRange.c1, f = false; i < l && sumW < w; ++i ) {
            sumW += this.cols[i].width;
            f = true;
        }
        this.visibleRange.c2 = i - (f ? 1 : 0);
    };

    /** Вычисляет диапазон индексов видимых строк */
    WorksheetView.prototype._calcVisibleRows = function () {
        var l = this.rows.length;
        var h = this.drawingCtx.getHeight();
        var sumH = this.topLeftFrozenCell ? this.rows[this.topLeftFrozenCell.getRow0()].top : this.cellsTop;
        for ( var i = this.visibleRange.r1, f = false; i < l && sumH < h; ++i ) {
            sumH += this.rows[i].height;
            f = true;
        }
        this.visibleRange.r2 = i - (f ? 1 : 0);
    };

    /** Обновляет позицию колонок (в pt) */
    WorksheetView.prototype._updateColumnPositions = function () {
        var x = this.cellsLeft;
        for ( var l = this.cols.length, i = 0; i < l; ++i ) {
            this.cols[i].left = x;
            x += this.cols[i].width;
        }
    };

    /** Обновляет позицию строк (в pt) */
    WorksheetView.prototype._updateRowPositions = function () {
        var y = this.cellsTop;
        for ( var l = this.rows.length, i = 0; i < l; ++i ) {
            this.rows[i].top = y;
            y += this.rows[i].height;
        }
    };

    /**
     * Добавляет колонки, пока общая ширина листа не превысит rightSide
     * @param {Number} rightSide Правая граница
     */
    WorksheetView.prototype._appendColumns = function ( rightSide ) {
        var i = this.cols.length;
        var lc = this.cols[i - 1];
        var done = false;

        for ( var x = lc.left + lc.width; i < gc_nMaxCol && (x < rightSide || !done); ++i ) {
            if ( x >= rightSide ) {
                // add +1 column at the end and exit cycle
                done = true;
            }
            this.cols[i] = this._calcColWidth( this.model.getColWidth( i ) );
            this.cols[i].left = x;
            x += this.cols[i].width;
            this.isChanged = true;
        }
        this.nColsCount = Math.min( Math.max( this.nColsCount, i ), gc_nMaxCol );
    };

    /** Устанаваливает видимый диапазон ячеек максимально возможным */
    WorksheetView.prototype._normalizeViewRange = function () {
        var t = this;
        var vr = t.visibleRange;
        var w = t.drawingCtx.getWidth() - t.cellsLeft;
        var h = t.drawingCtx.getHeight() - t.cellsTop;
        var c = t.cols;
        var r = t.rows;
        var vw = c[vr.c2].left + c[vr.c2].width - c[vr.c1].left;
        var vh = r[vr.r2].top + r[vr.r2].height - r[vr.r1].top;
        var i;

        var offsetFrozen = t.getFrozenPaneOffset();
        vw += offsetFrozen.offsetX;
        vh += offsetFrozen.offsetY;

        if ( vw < w ) {
            for ( i = vr.c1 - 1; i >= 0; --i ) {
                vw += c[i].width;
                if ( vw > w ) {
                    break;
                }
            }
            vr.c1 = i + 1;
            if ( vr.c1 >= vr.c2 ) {
                vr.c1 = vr.c2 - 1;
            }
            if ( vr.c1 < 0 ) {
                vr.c1 = 0;
            }
        }

        if ( vh < h ) {
            for ( i = vr.r1 - 1; i >= 0; --i ) {
                vh += r[i].height;
                if ( vh > h ) {
                    break;
                }
            }
            vr.r1 = i + 1;
            if ( vr.r1 >= vr.r2 ) {
                vr.r1 = vr.r2 - 1;
            }
            if ( vr.r1 < 0 ) {
                vr.r1 = 0;
            }
        }
    };

    WorksheetView.prototype._shiftVisibleRange = function ( range ) {
        var t = this;
        var vr = t.visibleRange;
        var arn = range ? range : t.activeRange.clone( true );
        var i;

        var cFrozen = 0, rFrozen = 0;
        if ( this.topLeftFrozenCell ) {
            cFrozen = this.topLeftFrozenCell.getCol0();
            rFrozen = this.topLeftFrozenCell.getRow0();
        }

        do {
            if ( arn.r2 > vr.r2 ) {
                i = arn.r2 - vr.r2;
                vr.r1 += i;
                vr.r2 += i;
                t._calcVisibleRows();
                continue;
            }
            if ( t._isRowDrawnPartially( arn.r2, vr.r1 ) ) {
                vr.r1 += 1;
                t._calcVisibleRows();
            }
            if ( arn.r1 < vr.r1 && arn.r1 >= rFrozen ) {
                i = arn.r1 - vr.r1;
                vr.r1 += i;
                vr.r2 += i;
                t._calcVisibleRows();
            }
            break;
        } while ( 1 );

        do {
            if ( arn.c2 > vr.c2 ) {
                i = arn.c2 - vr.c2;
                vr.c1 += i;
                vr.c2 += i;
                t._calcVisibleColumns();
                continue;
            }
            if ( t._isColDrawnPartially( arn.c2, vr.c1 ) ) {
                vr.c1 += 1;
                t._calcVisibleColumns();
            }
            if ( arn.c1 < vr.c1 && arn.c1 >= cFrozen ) {
                i = arn.c1 - vr.c1;
                vr.c1 += i;
                vr.c2 += i;
                if ( vr.c1 < 0 ) {
                    vr.c1 = 0;
                    vr.c2 -= vr.c1;
                }
                t._calcVisibleColumns();
            }
            break;
        } while ( 1 );
    };

    // ----- Drawing for print -----
  WorksheetView.prototype.calcPagesPrint = function(pageOptions, printOnlySelection, indexWorksheet, bFitToWidth, bFitToHeight) {
        var range;
        var maxCols = this.model.getColsCount();
        var maxRows = this.model.getRowsCount();
        var lastC = -1, lastR = -1;
        var activeRange = printOnlySelection ? this.activeRange : null;

        if ( null === activeRange ) {
            range = new asc_Range( 0, 0, maxCols, maxRows );
            this._prepareCellTextMetricsCache( range );
            for ( var c = 0; c < maxCols; ++c ) {
                for ( var r = 0; r < maxRows; ++r ) {
                    if ( !this._isCellEmptyOrMergedOrBackgroundColorOrBorders( c, r ) ) {
                        var rightSide = 0;
                        var ct = this._getCellTextCache( c, r );
                        if ( ct !== undefined ) {
                            if ( !ct.flags.isMerged() && !ct.flags.wrapText ) {
                                rightSide = ct.sideR;
                            }
                        }

                        lastC = Math.max( lastC, c + rightSide );
                        lastR = Math.max( lastR, r );
                    }
                }
            }
            maxCols = lastC + 1;
            maxRows = lastR + 1;

            // Получаем максимальную колонку/строку для изображений/чатов
            var maxObjectsCoord = this.objectRender.getDrawingAreaMetrics();
            if ( maxObjectsCoord ) {
                maxCols = Math.max( maxCols, maxObjectsCoord.maxCol );
                maxRows = Math.max( maxRows, maxObjectsCoord.maxRow );
            }
        }
        else {
            maxCols = activeRange.c2 + 1;
            maxRows = activeRange.r2 + 1;
            range = new asc_Range( 0, 0, maxCols, maxRows );
            this._prepareCellTextMetricsCache( range );
        }

        var pageMargins, pageSetup, pageGridLines, pageHeadings;
        if ( pageOptions instanceof asc_CPageOptions ) {
            pageMargins = pageOptions.asc_getPageMargins();
            pageSetup = pageOptions.asc_getPageSetup();
            pageGridLines = pageOptions.asc_getGridLines();
            pageHeadings = pageOptions.asc_getHeadings();
        }

        var pageWidth, pageHeight, pageOrientation;
        if ( pageSetup instanceof asc_CPageSetup ) {
            pageWidth = pageSetup.asc_getWidth();
            pageHeight = pageSetup.asc_getHeight();
            pageOrientation = pageSetup.asc_getOrientation();
        }

        var pageLeftField, pageRightField, pageTopField, pageBottomField;
        if ( pageMargins instanceof asc_CPageMargins ) {
            pageLeftField = pageMargins.asc_getLeft();
            pageRightField = pageMargins.asc_getRight();
            pageTopField = pageMargins.asc_getTop();
            pageBottomField = pageMargins.asc_getBottom();
        }

        if ( null == pageGridLines ) {
            pageGridLines = c_oAscPrintDefaultSettings.PageGridLines;
        }
        if ( null == pageHeadings ) {
            pageHeadings = c_oAscPrintDefaultSettings.PageHeadings;
        }

        if ( null == pageWidth ) {
            pageWidth = c_oAscPrintDefaultSettings.PageWidth;
        }
        if ( null == pageHeight ) {
            pageHeight = c_oAscPrintDefaultSettings.PageHeight;
        }
        if ( null == pageOrientation ) {
            pageOrientation = c_oAscPrintDefaultSettings.PageOrientation;
        }

        if ( null == pageLeftField ) {
            pageLeftField = c_oAscPrintDefaultSettings.PageLeftField;
        }
        if ( null == pageRightField ) {
            pageRightField = c_oAscPrintDefaultSettings.PageRightField;
        }
        if ( null == pageTopField ) {
            pageTopField = c_oAscPrintDefaultSettings.PageTopField;
        }
        if ( null == pageBottomField ) {
            pageBottomField = c_oAscPrintDefaultSettings.PageBottomField;
        }

        if ( c_oAscPageOrientation.PageLandscape === pageOrientation ) {
            var tmp = pageWidth;
            pageWidth = pageHeight;
            pageHeight = tmp;
        }

        var arrResult = [];
        if ( 0 === maxCols || 0 === maxRows ) {
            // Ничего нет, возвращаем пустой массив
            return null;
        }
        else {
            var pageWidthWithFields = pageWidth - pageLeftField - pageRightField;
            var pageHeightWithFields = pageHeight - pageTopField - pageBottomField;
            var leftFieldInPt = pageLeftField / vector_koef;
            var topFieldInPt = pageTopField / vector_koef;
            var rightFieldInPt = pageRightField / vector_koef;
            var bottomFieldInPt = pageBottomField / vector_koef;

            if ( pageHeadings ) {
                // Рисуем заголовки, нужно чуть сдвинуться
                leftFieldInPt += this.cellsLeft;
                topFieldInPt += this.cellsTop;
            }

            var pageWidthWithFieldsHeadings = (pageWidth - pageRightField) / vector_koef - leftFieldInPt;
            var pageHeightWithFieldsHeadings = (pageHeight - pageBottomField) / vector_koef - topFieldInPt;

            var currentColIndex = (null !== activeRange) ? activeRange.c1 : 0;
            var currentWidth = 0;
            var currentRowIndex = (null !== activeRange) ? activeRange.r1 : 0;
            var currentHeight = 0;
            var isCalcColumnsWidth = true;

            var bIsAddOffset = false;
            var nCountOffset = 0;

            while ( true ) {
                if ( currentColIndex === maxCols && currentRowIndex === maxRows ) {
                    break;
                }

                var newPagePrint = new asc_CPagePrint();

                var colIndex = currentColIndex, rowIndex = currentRowIndex;

                newPagePrint.indexWorksheet = indexWorksheet;

                newPagePrint.pageWidth = pageWidth;
                newPagePrint.pageHeight = pageHeight;
                newPagePrint.pageClipRectLeft = pageLeftField / vector_koef;
                newPagePrint.pageClipRectTop = pageTopField / vector_koef;
                newPagePrint.pageClipRectWidth = pageWidthWithFields / vector_koef;
                newPagePrint.pageClipRectHeight = pageHeightWithFields / vector_koef;

                newPagePrint.leftFieldInPt = leftFieldInPt;
                newPagePrint.topFieldInPt = topFieldInPt;
                newPagePrint.rightFieldInPt = rightFieldInPt;
                newPagePrint.bottomFieldInPt = bottomFieldInPt;

                for ( rowIndex = currentRowIndex; rowIndex < maxRows; ++rowIndex ) {
                    var currentRowHeight = this.rows[rowIndex].height;
          if (!bFitToHeight && currentHeight + currentRowHeight > pageHeightWithFieldsHeadings) {
                        // Закончили рисовать страницу
                        break;
                    }
                    if ( isCalcColumnsWidth ) {
                        for ( colIndex = currentColIndex; colIndex < maxCols; ++colIndex ) {
                            var currentColWidth = this.cols[colIndex].width;
                            if ( bIsAddOffset ) {
                                newPagePrint.startOffset = ++nCountOffset;
                                newPagePrint.startOffsetPt = (pageWidthWithFieldsHeadings * newPagePrint.startOffset);
                                currentColWidth -= newPagePrint.startOffsetPt;
                            }

              if (!bFitToWidth && currentWidth + currentColWidth > pageWidthWithFieldsHeadings && colIndex !== currentColIndex) {
                                break;
                            }

                            currentWidth += currentColWidth;

              if (!bFitToWidth && currentWidth > pageWidthWithFieldsHeadings && colIndex === currentColIndex) {
                                // Смещаем в селедующий раз ячейку
                                bIsAddOffset = true;
                                ++colIndex;
                                break;
                            }
                            else {
                                bIsAddOffset = false;
                            }
                        }
                        isCalcColumnsWidth = false;
                        if ( pageHeadings ) {
                            currentWidth += this.cellsLeft;
                        }

            if (bFitToWidth) {
                            newPagePrint.pageClipRectWidth = Math.max( currentWidth, newPagePrint.pageClipRectWidth );
                            newPagePrint.pageWidth = newPagePrint.pageClipRectWidth * vector_koef + (pageLeftField + pageRightField);
                        }
                        else {
                            newPagePrint.pageClipRectWidth = Math.min( currentWidth, newPagePrint.pageClipRectWidth );
                        }
                    }

                    currentHeight += currentRowHeight;
                    currentWidth = 0;
                }

        if (bFitToHeight) {
          newPagePrint.pageClipRectHeight = Math.max(currentHeight, newPagePrint.pageClipRectHeight);
          newPagePrint.pageHeight = newPagePrint.pageClipRectHeight * vector_koef + (pageTopField + pageBottomField);
        }

                // Нужно будет пересчитывать колонки
                isCalcColumnsWidth = true;

                // Рисуем сетку
                if ( pageGridLines ) {
                    newPagePrint.pageGridLines = true;
                }

                if ( pageHeadings ) {
                    // Нужно отрисовать заголовки
                    newPagePrint.pageHeadings = true;
                }

                newPagePrint.pageRange = new asc_Range( currentColIndex, currentRowIndex, colIndex - 1, rowIndex - 1 );

                if ( bIsAddOffset ) {
                    // Мы еще не дорисовали колонку
                    colIndex -= 1;
                }
                else {
                    nCountOffset = 0;
                }

                if ( colIndex < maxCols ) {
                    // Мы еще не все колонки отрисовали
                    currentColIndex = colIndex;
                    currentHeight = 0;
                }
                else {
                    // Мы дорисовали все колонки, нужна новая строка и стартовая колонка
                    currentColIndex = (null !== activeRange) ? activeRange.c1 : 0;
                    currentRowIndex = rowIndex;
                    currentHeight = 0;
                }

                if ( rowIndex === maxRows ) {
                    // Мы вышли, т.к. дошли до конца отрисовки по строкам
                    if ( colIndex < maxCols ) {
                        currentColIndex = colIndex;
                        currentHeight = 0;
                    }
                    else {
                        // Мы дошли до конца отрисовки
                        currentColIndex = colIndex;
                        currentRowIndex = rowIndex;
                    }
                }

                arrResult.push( newPagePrint );
            }
        }

        return arrResult;
    };
    WorksheetView.prototype.drawForPrint = function ( drawingCtx, printPagesData ) {
        var isAppBridge = (undefined != window['appBridge']);

        if ( null === printPagesData ) {
            // Напечатаем пустую страницу
            drawingCtx.BeginPage( c_oAscPrintDefaultSettings.PageWidth, c_oAscPrintDefaultSettings.PageHeight );
            drawingCtx.EndPage();
        }
        else {
            drawingCtx.BeginPage( printPagesData.pageWidth, printPagesData.pageHeight );
            drawingCtx.AddClipRect( printPagesData.pageClipRectLeft, printPagesData.pageClipRectTop, printPagesData.pageClipRectWidth, printPagesData.pageClipRectHeight );

            if ( isAppBridge ) {
                window['appBridge']['dummyCommandUpdate']();
            }

            var offsetCols = printPagesData.startOffsetPt;
            var range = printPagesData.pageRange;
            var offsetX = this.cols[range.c1].left - printPagesData.leftFieldInPt + offsetCols;
            var offsetY = this.rows[range.r1].top - printPagesData.topFieldInPt;

            var tmpVisibleRange = this.visibleRange;
            // Сменим visibleRange для прохождения проверок отрисовки
            this.visibleRange = range;

            if ( isAppBridge ) {
                window['appBridge']['dummyCommandUpdate']();
            }

            // Нужно отрисовать заголовки
            if ( printPagesData.pageHeadings ) {
                this._drawColumnHeaders( drawingCtx, range.c1, range.c2, /*style*/ undefined, offsetX, printPagesData.topFieldInPt - this.cellsTop );
                this._drawRowHeaders( drawingCtx, range.r1, range.r2, /*style*/ undefined, printPagesData.leftFieldInPt - this.cellsLeft, offsetY );
            }

            if ( isAppBridge ) {
                window['appBridge']['dummyCommandUpdate']();
            }

            // Рисуем сетку
            if ( printPagesData.pageGridLines ) {
                this._drawGrid( drawingCtx, range, offsetX, offsetY, printPagesData.pageWidth / vector_koef, printPagesData.pageHeight / vector_koef );
            }

            if ( isAppBridge ) {
                window['appBridge']['dummyCommandUpdate']();
            }

            // Отрисовываем ячейки и бордеры
            this._drawCellsAndBorders( drawingCtx, range, offsetX, offsetY );

            if ( isAppBridge ) {
                window['appBridge']['dummyCommandUpdate']();
            }

            var drawingPrintOptions = {
                ctx: drawingCtx, printPagesData: printPagesData
            };
            this.objectRender.showDrawingObjectsEx( false, null, drawingPrintOptions );
            this.visibleRange = tmpVisibleRange;

            if ( isAppBridge ) {
                window['appBridge']['dummyCommandUpdate']();
            }

            drawingCtx.RemoveClipRect();
            drawingCtx.EndPage();
        }
    };

    // ----- Drawing -----

    WorksheetView.prototype.draw = function ( lockDraw ) {
        if ( lockDraw || this.model.workbook.bCollaborativeChanges || window['IS_NATIVE_EDITOR'] ) {
            return this;
        }
        this._clean();
        this._drawCorner();
        this._drawColumnHeaders( /*drawingCtx*/ undefined );
        this._drawRowHeaders( /*drawingCtx*/ undefined );
        this._drawGrid( /*drawingCtx*/ undefined );
        this._drawCellsAndBorders( /*drawingCtx*/undefined );
        this._drawFrozenPane();
        this._drawFrozenPaneLines();
        this._fixSelectionOfMergedCells();
        this._drawAutoF();
        this.cellCommentator.drawCommentCells();
        this.objectRender.showDrawingObjectsEx( true );
        if ( this.overlayCtx ) {
            this._drawSelection();
        }

        return this;
    };

    WorksheetView.prototype._clean = function () {
        this.drawingCtx
            .setFillStyle( this.settings.cells.defaultState.background )
            .fillRect( 0, 0, this.drawingCtx.getWidth(), this.drawingCtx.getHeight() );
        if ( this.overlayCtx ) {
            this.overlayCtx.clear();
        }
    };

    WorksheetView.prototype.drawHighlightedHeaders = function ( col, row ) {
        this._activateOverlayCtx();
        if ( col >= 0 && col !== this.highlightedCol ) {
            this._doCleanHighlightedHeaders();
            this.highlightedCol = col;
            this._drawColumnHeaders( /*drawingCtx*/ undefined, col, col, kHeaderHighlighted );
        }
        else if ( row >= 0 && row !== this.highlightedRow ) {
            this._doCleanHighlightedHeaders();
            this.highlightedRow = row;
            this._drawRowHeaders( /*drawingCtx*/ undefined, row, row, kHeaderHighlighted );
        }
        this._deactivateOverlayCtx();
        return this;
    };

    WorksheetView.prototype.cleanHighlightedHeaders = function () {
        this._activateOverlayCtx();
        this._doCleanHighlightedHeaders();
        this._deactivateOverlayCtx();
        return this;
    };

    WorksheetView.prototype._activateOverlayCtx = function () {
        this.drawingCtx = this.buffers.overlay;
    };

    WorksheetView.prototype._deactivateOverlayCtx = function () {
        this.drawingCtx = this.buffers.main;
    };

    WorksheetView.prototype._doCleanHighlightedHeaders = function () {
        var hlc = this.highlightedCol, hlr = this.highlightedRow, arn = this.activeRange.clone( true );
        var hStyle = this.objectRender.selectedGraphicObjectsExists() ? kHeaderDefault : kHeaderActive;
        if ( hlc >= 0 ) {
            if ( hlc >= arn.c1 && hlc <= arn.c2 ) {
                this._drawColumnHeaders( /*drawingCtx*/ undefined, hlc, hlc, hStyle );
            }
            else {
                this._cleanColumnHeaders( hlc );
                if ( hlc + 1 === arn.c1 ) {
                    this._drawColumnHeaders( /*drawingCtx*/ undefined, hlc + 1, hlc + 1, kHeaderActive );
                }
                else if ( hlc - 1 === arn.c2 ) {
                    this._drawColumnHeaders( /*drawingCtx*/ undefined, hlc - 1, hlc - 1, hStyle );
                }
            }
            this.highlightedCol = -1;
        }
        if ( hlr >= 0 ) {
            if ( hlr >= arn.r1 && hlr <= arn.r2 ) {
                this._drawRowHeaders( /*drawingCtx*/ undefined, hlr, hlr, hStyle );
            }
            else {
                this._cleanRowHeades( hlr );
                if ( hlr + 1 === arn.r1 ) {
                    this._drawRowHeaders( /*drawingCtx*/ undefined, hlr + 1, hlr + 1, kHeaderActive );
                }
                else if ( hlr - 1 === arn.r2 ) {
                    this._drawRowHeaders( /*drawingCtx*/ undefined, hlr - 1, hlr - 1, hStyle );
                }
            }
            this.highlightedRow = -1;
        }
    };

    WorksheetView.prototype._drawActiveHeaders = function () {
        var arn = this.activeRange.clone( true ), vr = this.visibleRange, c1 = Math.max( vr.c1, arn.c1 ), c2 = Math.min( vr.c2, arn.c2 ), r1 = Math.max( vr.r1, arn.r1 ), r2 = Math.min( vr.r2, arn.r2 );
        this._activateOverlayCtx();
        this._drawColumnHeaders( /*drawingCtx*/ undefined, c1, c2, kHeaderActive );
        this._drawRowHeaders( /*drawingCtx*/ undefined, r1, r2, kHeaderActive );
        if ( this.topLeftFrozenCell ) {
            var cFrozen = this.topLeftFrozenCell.getCol0() - 1;
            var rFrozen = this.topLeftFrozenCell.getRow0() - 1;
            if ( 0 <= cFrozen ) {
                c1 = Math.max( 0, arn.c1 );
                c2 = Math.min( cFrozen, arn.c2 );
                this._drawColumnHeaders( /*drawingCtx*/ undefined, c1, c2, kHeaderActive );
            }
            if ( 0 <= rFrozen ) {
                r1 = Math.max( 0, arn.r1 );
                r2 = Math.min( rFrozen, arn.r2 );
                this._drawRowHeaders( /*drawingCtx*/ undefined, r1, r2, kHeaderActive );
            }
        }
        this._deactivateOverlayCtx();
    };

    WorksheetView.prototype._drawCorner = function () {
        if ( false === this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            return;
        }
        var x2 = this.headersLeft + this.headersWidth;
        var x1 = x2 - this.headersHeight;
        var y2 = this.headersTop + this.headersHeight;
        var y1 = this.headersTop;

        var dx = 4 * this.width_1px;
        var dy = 4 * this.height_1px;

        this._drawHeader( /*drawingCtx*/ undefined, this.headersLeft, this.headersTop, this.headersWidth, this.headersHeight, kHeaderDefault, true, -1 );
        this.drawingCtx.beginPath()
            .moveTo( x2 - dx, y1 + dy )
            .lineTo( x2 - dx, y2 - dy )
            .lineTo( x1 + dx, y2 - dy )
            .lineTo( x2 - dx, y1 + dy )
            .setFillStyle( this.settings.header.cornerColor )
            .fill();
    };

    /** Рисует заголовки видимых колонок */
    WorksheetView.prototype._drawColumnHeaders = function ( drawingCtx, start, end, style, offsetXForDraw, offsetYForDraw ) {
        if ( undefined === drawingCtx && false === this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            return;
        }
        var vr = this.visibleRange;
        var c = this.cols;
        var offsetX = (undefined !== offsetXForDraw) ? offsetXForDraw : c[vr.c1].left - this.cellsLeft;
        var offsetY = (undefined !== offsetYForDraw) ? offsetYForDraw : this.headersTop;
        if ( undefined === drawingCtx && this.topLeftFrozenCell && undefined === offsetXForDraw ) {
            var cFrozen = this.topLeftFrozenCell.getCol0();
            if ( start < vr.c1 ) {
                offsetX = c[0].left - this.cellsLeft;
            }
            else {
                offsetX -= c[cFrozen].left - c[0].left;
            }
        }

        if ( asc_typeof( start ) !== "number" ) {
            start = vr.c1;
        }
        if ( asc_typeof( end ) !== "number" ) {
            end = vr.c2;
        }
        if ( style === undefined ) {
            style = kHeaderDefault;
        }

        this._setFont( drawingCtx, this.model.getDefaultFontName(), this.model.getDefaultFontSize() );

        // draw column headers
        for ( var i = start; i <= end; ++i ) {
            this._drawHeader( drawingCtx, c[i].left - offsetX, offsetY, c[i].width, this.headersHeight, style, true, i );
        }
    };

    /** Рисует заголовки видимых строк */
    WorksheetView.prototype._drawRowHeaders = function ( drawingCtx, start, end, style, offsetXForDraw, offsetYForDraw ) {
        if ( undefined === drawingCtx && false === this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            return;
        }
        var vr = this.visibleRange;
        var r = this.rows;
        var offsetX = (undefined !== offsetXForDraw) ? offsetXForDraw : this.headersLeft;
        var offsetY = (undefined !== offsetYForDraw) ? offsetYForDraw : r[vr.r1].top - this.cellsTop;
        if ( undefined === drawingCtx && this.topLeftFrozenCell && undefined === offsetYForDraw ) {
            var rFrozen = this.topLeftFrozenCell.getRow0();
            if ( start < vr.r1 ) {
                offsetY = r[0].top - this.cellsTop;
            }
            else {
                offsetY -= r[rFrozen].top - r[0].top;
            }
        }

        if ( asc_typeof( start ) !== "number" ) {
            start = vr.r1;
        }
        if ( asc_typeof( end ) !== "number" ) {
            end = vr.r2;
        }
        if ( style === undefined ) {
            style = kHeaderDefault;
        }

        this._setFont( drawingCtx, this.model.getDefaultFontName(), this.model.getDefaultFontSize() );

        // draw row headers
        for ( var i = start; i <= end; ++i ) {
            this._drawHeader( drawingCtx, offsetX, r[i].top - offsetY, this.headersWidth, r[i].height, style, false, i );
        }
    };

    /**
     * Рисует заголовок, принимает координаты и размеры в pt
     * @param {DrawingContext} drawingCtx
     * @param {Number} x  Координата левого угла в pt
     * @param {Number} y  Координата левого угла в pt
     * @param {Number} w  Ширина в pt
     * @param {Number} h  Высота в pt
     * @param {Number} style  Стиль заголовка (kHeaderDefault, kHeaderActive, kHeaderHighlighted)
     * @param {Boolean} isColHeader  Тип заголовка: true - колонка, false - строка
     * @param {Number} index  Индекс столбца/строки или -1
     */
    WorksheetView.prototype._drawHeader = function ( drawingCtx, x, y, w, h, style, isColHeader, index ) {
        // Для отрисовки невидимого столбца/строки
        var isZeroHeader = false;
        if ( -1 !== index ) {
            if ( isColHeader ) {
                if ( w < this.width_1px ) {
                    if ( style !== kHeaderDefault ) {
                        return;
                    }
                    // Это невидимый столбец
                    isZeroHeader = true;
                    // Отрисуем только границу
                    w = this.width_1px;
                    // Возможно мы уже рисовали границу невидимого столбца (для последовательности невидимых)
                    if ( 0 < index && 0 === this.cols[index - 1].width ) {
                        // Мы уже нарисовали border для невидимой границы
                        return;
                    }
                }
                else if ( 0 < index && 0 === this.cols[index - 1].width ) {
                    // Мы уже нарисовали border для невидимой границы (поэтому нужно чуть меньше рисовать для соседнего столбца)
                    w -= this.width_1px;
                    x += this.width_1px;
                }
            }
            else {
                if ( h < this.height_1px ) {
                    if ( style !== kHeaderDefault ) {
                        return;
                    }
                    // Это невидимая строка
                    isZeroHeader = true;
                    // Отрисуем только границу
                    h = this.height_1px;
                    // Возможно мы уже рисовали границу невидимой строки (для последовательности невидимых)
                    if ( 0 < index && 0 === this.rows[index - 1].height ) {
                        // Мы уже нарисовали border для невидимой границы
                        return;
                    }
                }
                else if ( 0 < index && 0 === this.rows[index - 1].height ) {
                    // Мы уже нарисовали border для невидимой границы (поэтому нужно чуть меньше рисовать для соседней строки)
                    h -= this.height_1px;
                    y += this.height_1px;
                }
            }
        }

        var ctx = (drawingCtx) ? drawingCtx : this.drawingCtx;
        var st = this.settings.header.style[style];
        var x2 = x + w;
        var y2 = y + h;
        var x2WithoutBorder = x2 - this.width_1px;
        var y2WithoutBorder = y2 - this.height_1px;

        // background только для видимых
        if ( !isZeroHeader ) {
            // draw background
            ctx.setFillStyle( st.background )
                .fillRect( x, y, w, h );
        }
        // draw border
        ctx.setStrokeStyle( st.border )
            .setLineWidth( 1 )
            .beginPath();
        if ( style !== kHeaderDefault && !isColHeader ) {
            // Select row (top border)
            ctx.lineHorPrevPx( x, y, x2 );
        }

        // Right border
        ctx.lineVerPrevPx( x2, y, y2 );
        // Bottom border
        ctx.lineHorPrevPx( x, y2, x2 );

        if ( style !== kHeaderDefault && isColHeader ) {
            // Select col (left border)
            ctx.lineVerPrevPx( x, y, y2 );
        }
        ctx.stroke();

        // Для невидимых кроме border-а ничего не рисуем
        if ( isZeroHeader || -1 === index ) {
            return;
        }

        // draw text
        var text = isColHeader ? this._getColumnTitle( index ) : this._getRowTitle( index );
        var sr = this.stringRender;
        var tm = this._roundTextMetrics( sr.measureString( text ) );
        var bl = y2WithoutBorder - (isColHeader ? this.defaultRowDescender : this.rows[index].descender);
        var textX = this._calcTextHorizPos( x, x2WithoutBorder, tm, tm.width < w ? khaCenter : khaLeft );
        var textY = this._calcTextVertPos( y, y2WithoutBorder, bl, tm, kvaBottom );
        if ( drawingCtx ) {
            ctx.AddClipRect( x, y, w, h );
            ctx.setFillStyle( st.color )
                .fillText( text, textX, textY + tm.baseline, undefined, sr.charWidths );
            ctx.RemoveClipRect();
        }
        else {
            ctx.save()
                .beginPath()
                .rect( x, y, w, h )
                .clip()
                .setFillStyle( st.color )
                .fillText( text, textX, textY + tm.baseline, undefined, sr.charWidths )
                .restore();
        }
    };

    WorksheetView.prototype._cleanColumnHeaders = function ( colStart, colEnd ) {
        var offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft;
        var i, cFrozen = 0;
        if ( this.topLeftFrozenCell ) {
            cFrozen = this.topLeftFrozenCell.getCol0();
            offsetX -= this.cols[cFrozen].left - this.cols[0].left;
        }

        if ( colEnd === undefined ) {
            colEnd = colStart;
        }
        var colStartTmp = Math.max( this.visibleRange.c1, colStart );
        var colEndTmp = Math.min( this.visibleRange.c2, colEnd );
        for ( i = colStartTmp; i <= colEndTmp; ++i ) {
            this.drawingCtx.clearRectByX( this.cols[i].left - offsetX, this.headersTop, this.cols[i].width, this.headersHeight );
        }
        if ( 0 !== cFrozen ) {
            offsetX = this.cols[0].left - this.cellsLeft;
            // Почистим для pane
            colStart = Math.max( 0, colStart );
            colEnd = Math.min( cFrozen, colEnd );
            for ( i = colStart; i <= colEnd; ++i ) {
                this.drawingCtx.clearRectByX( this.cols[i].left - offsetX, this.headersTop, this.cols[i].width, this.headersHeight );
            }
        }
    };

    WorksheetView.prototype._cleanRowHeades = function ( rowStart, rowEnd ) {
        var offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop;
        var i, rFrozen = 0;
        if ( this.topLeftFrozenCell ) {
            rFrozen = this.topLeftFrozenCell.getRow0();
            offsetY -= this.rows[rFrozen].top - this.rows[0].top;
        }

        if ( rowEnd === undefined ) {
            rowEnd = rowStart;
        }
        var rowStartTmp = Math.max( this.visibleRange.r1, rowStart );
        var rowEndTmp = Math.min( this.visibleRange.r2, rowEnd );
        for ( i = rowStartTmp; i <= rowEndTmp; ++i ) {
            if ( this.height_1px > this.rows[i].height ) {
                continue;
            }
            this.drawingCtx.clearRectByY( this.headersLeft, this.rows[i].top - offsetY, this.headersWidth, this.rows[i].height );
        }
        if ( 0 !== rFrozen ) {
            offsetY = this.rows[0].top - this.cellsTop;
            // Почистим для pane
            rowStart = Math.max( 0, rowStart );
            rowEnd = Math.min( rFrozen, rowEnd );
            for ( i = rowStart; i <= rowEnd; ++i ) {
                if ( this.height_1px > this.rows[i].height ) {
                    continue;
                }
                this.drawingCtx.clearRectByY( this.headersLeft, this.rows[i].top - offsetY, this.headersWidth, this.rows[i].height );
            }
        }
    };

    WorksheetView.prototype._cleanColumnHeadersRect = function () {
        this.drawingCtx.clearRect( this.cellsLeft, this.headersTop, this.drawingCtx.getWidth() - this.cellsLeft, this.headersHeight );
    };

    /** Рисует сетку таблицы */
    WorksheetView.prototype._drawGrid = function ( drawingCtx, range, leftFieldInPt, topFieldInPt, width, height ) {
        // Возможно сетку не нужно рисовать (при печати свои проверки)
        if ( undefined === drawingCtx && false === this.model.sheetViews[0].asc_getShowGridLines() ) {
            return;
        }

        if ( range === undefined ) {
            range = this.visibleRange;
        }
        var ctx = (drawingCtx) ? drawingCtx : this.drawingCtx;
        var c = this.cols;
        var r = this.rows;
        var widthCtx = (width) ? width : ctx.getWidth();
        var heightCtx = (height) ? height : ctx.getHeight();
        var offsetX = (undefined !== leftFieldInPt) ? leftFieldInPt : c[this.visibleRange.c1].left - this.cellsLeft;
        var offsetY = (undefined !== topFieldInPt) ? topFieldInPt : r[this.visibleRange.r1].top - this.cellsTop;
        if ( undefined === drawingCtx && this.topLeftFrozenCell ) {
            if ( undefined === leftFieldInPt ) {
                var cFrozen = this.topLeftFrozenCell.getCol0();
                offsetX -= c[cFrozen].left - c[0].left;
            }
            if ( undefined === topFieldInPt ) {
                var rFrozen = this.topLeftFrozenCell.getRow0();
                offsetY -= r[rFrozen].top - r[0].top;
            }
        }
        var x1 = c[range.c1].left - offsetX;
        var y1 = r[range.r1].top - offsetY;
        var x2 = Math.min( c[range.c2].left - offsetX + c[range.c2].width, widthCtx );
        var y2 = Math.min( r[range.r2].top - offsetY + r[range.r2].height, heightCtx );
        ctx.setFillStyle( this.settings.cells.defaultState.background )
            .fillRect( x1, y1, x2 - x1, y2 - y1 )
            .setStrokeStyle( this.settings.cells.defaultState.border )
            .setLineWidth( 1 ).beginPath();

        var w, h;
        for ( var i = range.c1, x = x1; i <= range.c2 && x <= x2; ++i ) {
            w = c[i].width;
            x += w;
            if ( w >= this.width_1px ) {
                ctx.lineVerPrevPx( x, y1, y2 );
            }
        }
        for ( var j = range.r1, y = y1; j <= range.r2 && y <= y2; ++j ) {
            h = r[j].height;
            y += h;
            if ( h >= this.height_1px ) {
                ctx.lineHorPrevPx( x1, y, x2 );
            }
        }

        ctx.stroke();
    };

    WorksheetView.prototype._drawCellsAndBorders = function ( drawingCtx, range, offsetXForDraw, offsetYForDraw ) {
        if ( range === undefined ) {
            range = this.visibleRange;
        }

        var left, top, cFrozen, rFrozen;
        var offsetX = (undefined === offsetXForDraw) ? this.cols[this.visibleRange.c1].left - this.cellsLeft : offsetXForDraw;
        var offsetY = (undefined === offsetYForDraw) ? this.rows[this.visibleRange.r1].top - this.cellsTop : offsetYForDraw;
        if ( undefined === drawingCtx && this.topLeftFrozenCell ) {
            if ( undefined === offsetXForDraw ) {
                cFrozen = this.topLeftFrozenCell.getCol0();
                offsetX -= this.cols[cFrozen].left - this.cols[0].left;
            }
            if ( undefined === offsetYForDraw ) {
                rFrozen = this.topLeftFrozenCell.getRow0();
                offsetY -= this.rows[rFrozen].top - this.rows[0].top;
            }
        }

        if ( !drawingCtx && !window['IS_NATIVE_EDITOR'] ) {
            left = this.cols[range.c1].left;
            top = this.rows[range.r1].top;
            // set clipping rect to cells area
            this.drawingCtx.save()
                .beginPath()
                .rect( left - offsetX, top - offsetY, Math.min( this.cols[range.c2].left - left + this.cols[range.c2].width, this.drawingCtx.getWidth() - this.cellsLeft ), Math.min( this.rows[range.r2].top - top + this.rows[range.r2].height, this.drawingCtx.getHeight() - this.cellsTop ) )
                .clip();
        }

        var mergedCells = this._drawCells( drawingCtx, range, offsetX, offsetY );
        this._drawCellsBorders( drawingCtx, range, offsetX, offsetY, mergedCells );

        if ( !drawingCtx && !window['IS_NATIVE_EDITOR'] ) {
            // restore canvas' original clipping range
            this.drawingCtx.restore();
        }
    };

    /** Рисует ячейки таблицы */
    WorksheetView.prototype._drawCells = function ( drawingCtx, range, offsetX, offsetY ) {
        this._prepareCellTextMetricsCache( range );

        var mergedCells = {}, mc, i;
        for ( var row = range.r1; row <= range.r2; ++row ) {
            $.extend( mergedCells, this._drawRowBG( drawingCtx, row, range.c1, range.c2, offsetX, offsetY, null ), this._drawRowText( drawingCtx, row, range.c1, range.c2, offsetX, offsetY ) );
        }
        // draw merged cells at last stage to fix cells background issue
        for ( i in mergedCells ) {
            mc = mergedCells[i];
            this._drawRowBG( drawingCtx, mc.r1, mc.c1, mc.c1, offsetX, offsetY, mc );
            this._drawCellText( drawingCtx, mc.c1, mc.r1, range.c1, range.c2, offsetX, offsetY, true );
        }
        return mergedCells;
    };

    /** Рисует фон ячеек в строке */
    WorksheetView.prototype._drawRowBG = function ( drawingCtx, row, colStart, colEnd, offsetX, offsetY, oMergedCell ) {
        if ( this.rows[row].height < this.height_1px && null === oMergedCell ) {
            return {};
        }

        var mergedCells = {};
        var ctx = (undefined === drawingCtx) ? this.drawingCtx : drawingCtx;
        for ( var col = colStart; col <= colEnd; ++col ) {
            if ( this.cols[col].width < this.width_1px && null === oMergedCell ) {
                continue;
            }

            // ToDo подумать, может стоит не брать ячейку из модели (а брать из кеш-а)
            var c = this._getVisibleCell( col, row );
            if ( !c ) {
                continue;
            }

            var bg = c.getFill();
            var mc = null;
            var mwidth = 0, mheight = 0;

            if ( null === oMergedCell ) {
                mc = this.model.getMergedByCell( row, col );
                if ( null !== mc ) {
                    mergedCells[mc.r1 + "_" + mc.c1] = {c1: mc.c1, r1: mc.r1, c2: mc.c2, r2: mc.r2};
                    col = mc.c2;
                    continue;
                }
            }
            else {
                mc = oMergedCell;
            }

            if ( null !== mc ) {
                if ( col !== mc.c1 || row !== mc.r1 ) {
                    continue;
                }
                for ( var i = mc.c1 + 1; i <= mc.c2 && i < this.cols.length; ++i ) {
                    mwidth += this.cols[i].width;
                }
                for ( var j = mc.r1 + 1; j <= mc.r2 && j < this.rows.length; ++j ) {
                    mheight += this.rows[j].height;
                }
            }
            else {
                if ( bg === null ) {
                    if ( col === colEnd && col < this.cols.length - 1 && row < this.rows.length - 1 ) {
                        var c2 = this._getVisibleCell( col + 1, row );
                        if ( c2 ) {
                            var bg2 = c2.getFill();
                            if ( bg2 !== null ) {
                                ctx.setFillStyle( bg2 )
                                    .fillRect( this.cols[col + 1].left - offsetX - this.width_1px, this.rows[row].top - offsetY - this.height_1px, this.width_1px, this.rows[row].height + this.height_1px );
                            }
                        }
                        var c3 = this._getVisibleCell( col, row + 1 );
                        if ( c3 ) {
                            var bg3 = c3.getFill();
                            if ( bg3 !== null ) {
                                ctx.setFillStyle( bg3 )
                                    .fillRect( this.cols[col].left - offsetX - this.width_1px, this.rows[row + 1].top - offsetY - this.height_1px, this.cols[col].width + this.width_1px, this.height_1px );
                            }
                        }
                    }
                    continue;
                }
            }

            var x = this.cols[col].left - (bg !== null ? this.width_1px : 0);
            var y = this.rows[row].top - (bg !== null ? this.height_1px : 0);
            var w = this.cols[col].width + this.width_1px * (bg !== null ? +1 : -1) + mwidth;
            var h = this.rows[row].height + this.height_1px * (bg !== null ? +1 : -1) + mheight;
            var color = bg !== null ? bg : this.settings.cells.defaultState.background;
            ctx.setFillStyle( color ).fillRect( x - offsetX, y - offsetY, w, h );
        }
        return mergedCells;
    };

    /** Рисует текст ячеек в строке */
    WorksheetView.prototype._drawRowText = function ( drawingCtx, row, colStart, colEnd, offsetX, offsetY ) {
        if ( this.rows[row].height < this.height_1px ) {
            return {};
        }

        var dependentCells = {}, mergedCells = {}, i, mc, col;
        // draw cells' text
        for ( col = colStart; col <= colEnd; ++col ) {
            if ( this.cols[col].width < this.width_1px ) {
                continue;
            }
            mc = this._drawCellText( drawingCtx, col, row, colStart, colEnd, offsetX, offsetY, false );
            if ( mc !== null ) {
                mergedCells[mc.index] = {c1: mc.c1, r1: mc.r1, c2: mc.c2, r2: mc.r2};
            }
            // check if long text overlaps this cell
            i = this._findSourceOfCellText( col, row );
            if ( i >= 0 ) {
                dependentCells[i] = (dependentCells[i] || []);
                dependentCells[i].push( col );
            }
        }
        // draw long text that overlaps own cell's borders
        for ( i in dependentCells ) {
            var arr = dependentCells[i], j = arr.length - 1;
            col = i >> 0;
            // if source cell belongs to cells range then skip it (text has been drawn already)
            if ( col >= arr[0] && col <= arr[j] ) {
                continue;
            }
            // draw long text fragment
            this._drawCellText( drawingCtx, col, row, arr[0], arr[j], offsetX, offsetY, false );
        }
        return mergedCells;
    };

    /** Рисует текст ячейки */
    WorksheetView.prototype._drawCellText = function ( drawingCtx, col, row, colStart, colEnd, offsetX, offsetY, drawMergedCells ) {
        var ct = this._getCellTextCache( col, row );
        if ( ct === undefined ) {
            return null;
        }

        var isMerged = ct.flags.isMerged(), range = undefined, isWrapped = ct.flags.wrapText;
        var ctx = (undefined === drawingCtx) ? this.drawingCtx : drawingCtx;

        if ( isMerged ) {
            range = ct.flags.merged;
            if ( !drawMergedCells ) {
                return {c1: range.c1, r1: range.r1, c2: range.c2, r2: range.r2, index: range.r1 + "_" + range.c1};
            }
            if ( col !== range.c1 || row !== range.r1 ) {
                return null;
            }
        }

        var colL = isMerged ? range.c1 : Math.max( colStart, col - ct.sideL );
        var colR = isMerged ? Math.min( range.c2, this.nColsCount - 1 ) : Math.min( colEnd, col + ct.sideR );
        var rowT = isMerged ? range.r1 : row;
        var rowB = isMerged ? Math.min( range.r2, this.nRowsCount - 1 ) : row;
        var isTrimmedR = !isMerged && colR !== col + ct.sideR;

        if ( !(ct.angle || 0) ) {
            if ( !isMerged && !isWrapped ) {
                this._eraseCellRightBorder( drawingCtx, colL, colR + (isTrimmedR ? 1 : 0), row, offsetX, offsetY );
            }
        }

        var x1 = this.cols[colL].left - offsetX;
        var y1 = this.rows[rowT].top - offsetY;
        var w = this.cols[colR].left + this.cols[colR].width - offsetX - x1;
        var h = this.rows[rowB].top + this.rows[rowB].height - offsetY - y1;
        var x2 = x1 + w - (isTrimmedR ? 0 : this.width_1px);
        var y2 = y1 + h - this.height_1px;
        var bl = !isMerged ? (y2 - this.rows[rowB].descender) : (y2 - ct.metrics.height + ct.metrics.baseline - this.height_1px);
        var x1ct = isMerged ? x1 : this.cols[col].left - offsetX;
        var x2ct = isMerged ? x2 : x1ct + this.cols[col].width - this.width_1px;
        var textX = this._calcTextHorizPos( x1ct, x2ct, ct.metrics, ct.cellHA );
        var textY = this._calcTextVertPos( y1, y2, bl, ct.metrics, ct.cellVA );
        var textW = this._calcTextWidth( x1ct, x2ct, ct.metrics, ct.cellHA );

        var xb1, yb1, wb, hb, colLeft, colRight, i;
        var txtRotX, txtRotW, clipUse = false;

        if ( drawingCtx ) {

            // для печати

            if ( ct.angle || 0 ) {

                xb1 = this.cols[col].left - offsetX;
                yb1 = this.rows[row].top - offsetY;
                wb = this.cols[col].width;
                hb = this.rows[row].height;

                txtRotX = ct.textBound.sx + xb1;
                txtRotW = ct.textBound.sw + xb1;

                if ( isMerged ) {

                    wb = 0;

                    for ( i = colL; i <= colR && i < this.nColsCount; ++i ) {
                        wb += this.cols[i].width;
                    }

                    hb = 0;

                    for ( i = rowT; i <= rowB && i < this.nRowsCount; ++i ) {
                        hb += this.rows[i].height;
                    }

                    ctx.AddClipRect( xb1, yb1, wb, hb );
                    clipUse = true;
                }

                this.stringRender.angle = ct.angle;
                this.stringRender.fontNeedUpdate = true;

                if ( 90 === ct.angle || -90 === ct.angle ) {
                    // клип по ячейке
                    if ( !isMerged ) {
                        ctx.AddClipRect( xb1, yb1, wb, hb );
                        clipUse = true;
                    }
                }
                else {
                    // клип по строке
                    if ( !isMerged ) {
                        ctx.AddClipRect( 0, yb1, this.drawingCtx.getWidth(), h );
                        clipUse = true;
                    }

                    if ( !isMerged && !isWrapped ) {
                        colLeft = col;
                        if ( 0 !== txtRotX ) {
                            while ( true ) {
                                if ( 0 == colLeft ) {
                                    break;
                                }
                                if ( txtRotX >= this.cols[colLeft].left ) {
                                    break;
                                }
                                --colLeft;
                            }
                        }

                        colRight = Math.min( col, this.nColsCount - 1 );
                        if ( 0 !== txtRotW ) {
                            while ( true ) {
                                ++colRight;
                                if ( colRight >= this.nColsCount ) {
                                    --colRight;
                                    break;
                                }
                                if ( txtRotW <= this.cols[colRight].left ) {
                                    --colRight;
                                    break;
                                }
                            }
                        }

                        colLeft = isMerged ? range.c1 : colLeft;
                        colRight = isMerged ? Math.min( range.c2, this.nColsCount - 1 ) : colRight;

                        this._eraseCellRightBorder( drawingCtx, colLeft, colRight + (isTrimmedR ? 1 : 0), row, offsetX, offsetY );
                    }
                }

                this.stringRender.rotateAtPoint( drawingCtx, ct.angle, xb1, yb1, ct.textBound.dx, ct.textBound.dy );
                this.stringRender.restoreInternalState( ct.state ).renderForPrint( drawingCtx, 0, 0, textW, ct.color );
                this.stringRender.resetTransform( drawingCtx );

                if ( clipUse ) {
                    ctx.RemoveClipRect();
                }
            }
            else {
                ctx.AddClipRect( x1, y1, w, h );
                this.stringRender.restoreInternalState( ct.state ).renderForPrint( drawingCtx, textX, textY, textW, ct.color );
                ctx.RemoveClipRect();
            }
        }
        else {

            // для отрисовки

            if ( ct.angle || 0 ) {

                xb1 = this.cols[col].left - offsetX;
                yb1 = this.rows[row].top - offsetY;
                wb = this.cols[col].width;
                hb = this.rows[row].height;

                txtRotX = ct.textBound.sx + xb1;
                txtRotW = ct.textBound.sw + xb1;

                if ( isMerged ) {

                    wb = 0;

                    for ( i = colL; i <= colR && i < this.nColsCount; ++i ) {
                        wb += this.cols[i].width;
                    }

                    hb = 0;

                    for ( i = rowT; i <= rowB && i < this.nRowsCount; ++i ) {
                        hb += this.rows[i].height;
                    }

                    ctx.save().beginPath().rect( xb1, yb1, wb, hb ).clip();
                    clipUse = true;
                }

                this.stringRender.angle = ct.angle;
                this.stringRender.fontNeedUpdate = true;

                if ( 90 === ct.angle || -90 === ct.angle ) {
                    // клип по ячейке
                    if ( !isMerged ) {
                        ctx.save().beginPath().rect( xb1, yb1, wb, hb ).clip();
                        clipUse = true;
                    }
                }
                else {
                    // клип по строке
                    if ( !isMerged ) {
                        ctx.save().beginPath().rect( 0, y1, this.drawingCtx.getWidth(), h ).clip();
                        clipUse = true;
                    }

                    if ( !isMerged && !isWrapped ) {
                        colLeft = col;
                        if ( 0 !== txtRotX ) {
                            while ( true ) {
                                if ( 0 == colLeft ) {
                                    break;
                                }
                                if ( txtRotX >= this.cols[colLeft].left ) {
                                    break;
                                }
                                --colLeft;
                            }
                        }

                        colRight = Math.min( col, this.nColsCount - 1 );
                        if ( 0 !== txtRotW ) {
                            while ( true ) {
                                ++colRight;
                                if ( colRight >= this.nColsCount ) {
                                    --colRight;
                                    break;
                                }
                                if ( txtRotW <= this.cols[colRight].left ) {
                                    --colRight;
                                    break;
                                }
                            }
                        }

                        colLeft = isMerged ? range.c1 : colLeft;
                        colRight = isMerged ? Math.min( range.c2, this.nColsCount - 1 ) : colRight;

                        this._eraseCellRightBorder( drawingCtx, colLeft, colRight + (isTrimmedR ? 1 : 0), row, offsetX, offsetY );
                    }
                }

                this.stringRender.rotateAtPoint( null, ct.angle, xb1, yb1, ct.textBound.dx, ct.textBound.dy );
                this.stringRender.restoreInternalState( ct.state ).render( 0, 0, textW, ct.color );
                this.stringRender.resetTransform( null );

                if ( clipUse ) {
                    ctx.restore();
                }

            }
            else {
                ctx.save().beginPath().rect( x1, y1, w, h ).clip();
                this.stringRender.restoreInternalState( ct.state ).render( textX, textY, textW, ct.color );
                ctx.restore();
            }
        }

        return null;
    };

    /** Удаляет вертикальные границы ячейки, если текст выходит за границы и соседние ячейки пусты */
    WorksheetView.prototype._eraseCellRightBorder = function ( drawingCtx, colBeg, colEnd, row, offsetX, offsetY ) {
        if ( colBeg >= colEnd ) {
            return;
        }
        var nextCell = -1;
        var ctx = (drawingCtx) ? drawingCtx : this.drawingCtx;
        ctx.setFillStyle( this.settings.cells.defaultState.background );
        for ( var col = colBeg; col < colEnd; ++col ) {
            var c = -1 !== nextCell ? nextCell : this._getCell( col, row );
            var bg = null !== c ? c.getFill() : null;
            if ( bg !== null ) {
                continue;
            }

            nextCell = this._getCell( col + 1, row );
            bg = null !== nextCell ? nextCell.getFill() : null;
            if ( bg !== null ) {
                continue;
            }

            ctx.fillRect( this.cols[col].left + this.cols[col].width - offsetX - this.width_1px, this.rows[row].top - offsetY, this.width_1px, this.rows[row].height - this.height_1px );
        }
    };

    /** Рисует рамки для ячеек */
    WorksheetView.prototype._drawCellsBorders = function ( drawingCtx, range, offsetX, offsetY, mergedCells ) {
        //TODO: использовать стили линий при рисовании границ
        var t = this;
        var ctx = (drawingCtx) ? drawingCtx : this.drawingCtx;
        var c = this.cols;
        var r = this.rows;

        var objectMergedCells = {}; // Двумерный map вида строка-колонка {1: {1: range, 4: range}}
        var i, mergeCellInfo, startCol, endRow, endCol, col, row;
        for ( i in mergedCells ) {
            mergeCellInfo = mergedCells[i];
            startCol = Math.max( range.c1, mergeCellInfo.c1 );
            endRow = Math.min( mergeCellInfo.r2, range.r2, this.nRowsCount );
            endCol = Math.min( mergeCellInfo.c2, range.c2, this.nColsCount );
            for ( row = Math.max( range.r1, mergeCellInfo.r1 ); row <= endRow; ++row ) {
                if ( !objectMergedCells.hasOwnProperty( row ) ) {
                    objectMergedCells[row] = {};
                }
                for ( col = startCol; col <= endCol; ++col )
                    objectMergedCells[row][col] = mergeCellInfo;
            }
        }

        var bc = null, bw = -1, isNotFirst = false; // cached border color

        function drawBorder( type, border, x1, y1, x2, y2 ) {
            var isStroke = false, isNewColor = !g_oColorManager.isEqual( bc, border.c ), isNewWidth = bw !== border.w;
            if ( isNotFirst && (isNewColor || isNewWidth) ) {
                ctx.stroke();
                isStroke = true;
            }

            if ( isNewColor ) {
                bc = border.c;
                ctx.setStrokeStyle( bc );
            }
            if ( isNewWidth ) {
                bw = border.w;
                ctx.setLineWidth( border.w );
            }

            if ( isStroke || false === isNotFirst ) {
                isNotFirst = true;
                ctx.beginPath();
            }

            switch ( type ) {
                case c_oAscBorderType.Hor:
                    ctx.lineHor( x1, y1, x2 );
                    break;
                case c_oAscBorderType.Ver:
                    ctx.lineVer( x1, y1, y2 );
                    break;
                case c_oAscBorderType.Diag:
                    ctx.lineDiag( x1, y1, x2, y2 );
                    break;
            }
        }

        function drawVerticalBorder( borderLeftObject, borderRightObject, x, y1, y2 ) {
            var border, borderLeft = borderLeftObject ? borderLeftObject.borders : null, borderRight = borderRightObject ? borderRightObject.borders : null;

            if ( borderLeft && borderLeft.r.w ) {
                border = borderLeft.r;
            }
            else if ( borderRight && borderRight.l.w ) {
                border = borderRight.l;
            }
            if ( !border || border.w < 1 ) {
                return;
            }

            // ToDo переделать рассчет
            var tbw = t._calcMaxBorderWidth( borderLeftObject && borderLeftObject.getTopBorder(), borderRightObject && borderRightObject.getTopBorder() ); // top border width
            var bbw = t._calcMaxBorderWidth( borderLeftObject && borderLeftObject.getBottomBorder(), borderRightObject && borderRightObject.getBottomBorder() ); // bottom border width
            var dy1 = tbw > border.w ? tbw - 1 : (tbw > 1 ? -1 : 0);
            var dy2 = bbw > border.w ? -2 : (bbw > 2 ? 1 : 0);

            drawBorder( c_oAscBorderType.Ver, border, x, y1 + (-1 + dy1) * t.height_1px, x, y2 + (1 + dy2) * t.height_1px );
        }

        function drawHorizontalBorder( borderTopObject, borderBottomObject, x1, y, x2 ) {
            var border, borderTop = borderTopObject ? borderTopObject.borders : null, borderBottom = borderBottomObject ? borderBottomObject.borders : null;

            if ( borderTop && borderTop.b.w ) {
                border = borderTop.b;
            }
            else if ( borderBottom && borderBottom.t.w ) {
                border = borderBottom.t;
            }

            if ( border && border.w > 0 ) {
                // ToDo переделать рассчет
                var lbw = t._calcMaxBorderWidth( borderTopObject && borderTopObject.getLeftBorder(), borderBottomObject && borderBottomObject.getLeftBorder() );
                var rbw = t._calcMaxBorderWidth( borderTopObject && borderTopObject.getRightBorder(), borderTopObject && borderTopObject.getRightBorder() );
                var dx1 = border.w > lbw ? (lbw > 1 ? -1 : 0) : (lbw > 2 ? 2 : 1);
                var dx2 = border.w > rbw ? (rbw > 2 ? 1 : 0) : (rbw > 1 ? -2 : -1);
                drawBorder( c_oAscBorderType.Hor, border, x1 + (-1 + dx1) * t.width_1px, y, x2 + (1 + dx2) * t.width_1px, y );
            }
        }

        var arrPrevRow = [], arrCurrRow = [], arrNextRow = [];
        var objMCPrevRow = null, objMCRow = null, objMCNextRow = null;
        var bCur, bPrev, bNext, bTopCur, bTopPrev, bTopNext, bBotCur, bBotPrev, bBotNext;
        bCur = bPrev = bNext = bTopCur = bTopNext = bBotCur = bBotNext = null;
        row = range.r1 - 1;
        var prevCol = range.c1 - 1;
        // Определим первую колонку (т.к. могут быть скрытые колонки)
        while ( 0 <= prevCol && c[prevCol].width < t.width_1px )
            --prevCol;

        // Сначала пройдемся по верхней строке (над отрисовываемым диапазоном)
        while ( 0 <= row ) {
            if ( r[row].height >= t.height_1px ) {
                objMCPrevRow = objectMergedCells[row];
                for ( col = prevCol; col <= range.c2 && col < t.nColsCount; ++col ) {
                    if ( 0 > col || c[col].width < t.width_1px ) {
                        continue;
                    }
                    arrPrevRow[col] = new CellBorderObject( t._getVisibleCell( col, row ).getBorder(), objMCPrevRow ? objMCPrevRow[col] : null, col, row );
                }
                break;
            }
            --row;
        }

        var mc = null, nextRow, isFirstRow = true;
        var isPrevColExist = (0 <= prevCol);
        for ( row = range.r1; row <= range.r2 && row < t.nRowsCount; row = nextRow ) {
            nextRow = row + 1;
            if ( r[row].height < t.height_1px ) {
                continue;
            }
            // Нужно отсеять пустые снизу
            for ( ; nextRow <= range.r2 && nextRow < t.nRowsCount; ++nextRow )
                if ( r[nextRow].height >= t.height_1px ) {
                    break;
                }

            var isFirstRowTmp = isFirstRow, isLastRow = nextRow > range.r2 || nextRow >= t.nRowsCount;
            isFirstRow = false; // Это уже не первая строка (определяем не по совпадению с range.r1, а по видимости)

            objMCRow = isFirstRowTmp ? objectMergedCells[row] : objMCNextRow;
            objMCNextRow = objectMergedCells[nextRow];

            var rowCache = t._fetchRowCache( row );
            var y1 = r[row].top - offsetY;
            var y2 = y1 + r[row].height - t.height_1px;

            var nextCol, isFirstCol = true;
            for ( col = range.c1; col <= range.c2 && col < t.nColsCount; col = nextCol ) {
                nextCol = col + 1;
                if ( c[col].width < t.width_1px ) {
                    continue;
                }
                // Нужно отсеять пустые справа
                for ( ; nextCol <= range.c2 && nextCol < t.nColsCount; ++nextCol )
                    if ( c[nextCol].width >= t.width_1px ) {
                        break;
                    }

                var isFirstColTmp = isFirstCol, isLastCol = nextCol > range.c2 || nextCol >= t.nColsCount;
                isFirstCol = false; // Это уже не первая колонка (определяем не по совпадению с range.c1, а по видимости)

                mc = objMCRow ? objMCRow[col] : null;

                var x1 = c[col].left - offsetX;
                var x2 = x1 + c[col].width - this.width_1px;

                if ( row === t.nRowsCount ) {
                    bBotPrev = bBotCur = bBotNext = null;
                }
                else {
                    if ( isFirstColTmp ) {
                        bBotPrev = arrNextRow[prevCol] = new CellBorderObject( isPrevColExist ? t._getVisibleCell( prevCol, nextRow ).getBorder() : null, objMCNextRow ? objMCNextRow[prevCol] : null, prevCol, nextRow );
                        bBotCur = arrNextRow[col] = new CellBorderObject( t._getVisibleCell( col, nextRow ).getBorder(), objMCNextRow ? objMCNextRow[col] : null, col, nextRow );
                    }
                    else {
                        bBotPrev = bBotCur;
                        bBotCur = bBotNext;
                    }
                }

                if ( isFirstColTmp ) {
                    bPrev = arrCurrRow[prevCol] = new CellBorderObject( isPrevColExist ? t._getVisibleCell( prevCol, row ).getBorder() : null, objMCRow ? objMCRow[prevCol] : null, prevCol, row );
                    bCur = arrCurrRow[col] = new CellBorderObject( t._getVisibleCell( col, row ).getBorder(), mc, col, row );
                    bTopPrev = arrPrevRow[prevCol];
                    bTopCur = arrPrevRow[col];
                }
                else {
                    bPrev = bCur;
                    bCur = bNext;
                    bTopPrev = bTopCur;
                    bTopCur = bTopNext;
                }

                if ( col === t.nColsCount ) {
                    bNext = null;
                    bTopNext = null;
                }
                else {
                    bNext = arrCurrRow[nextCol] = new CellBorderObject( t._getVisibleCell( nextCol, row ).getBorder(), objMCRow ? objMCRow[nextCol] : null, nextCol, row );
                    bTopNext = arrPrevRow[nextCol];

                    if ( row === t.nRowsCount ) {
                        bBotNext = null;
                    }
                    else {
                        bBotNext = arrNextRow[nextCol] = new CellBorderObject( t._getVisibleCell( nextCol, nextRow ).getBorder(), objMCNextRow ? objMCNextRow[nextCol] : null, nextCol, nextRow );
                    }
                }

                if ( mc && row !== mc.r1 && row !== mc.r2 && col !== mc.c1 && col !== mc.c2 ) {
                    continue;
                }

                // draw diagonal borders
                if ( (bCur.borders.dd || bCur.borders.du) && (!mc || (row === mc.r1 && col === mc.c1)) ) {
                    var x2Diagonal = x2;
                    var y2Diagonal = y2;
                    if ( mc ) {
                        // Merge cells
                        x2Diagonal = c[mc.c2].left + c[mc.c2].width - offsetX - t.width_1px;
                        y2Diagonal = r[mc.r2].top + r[mc.r2].height - offsetY - t.height_1px;
                    }
                    // ToDo Clip diagonal borders
                    /*ctx.save()
                     .beginPath()
                     .rect(x1 + this.width_1px * (lb.w < 1 ? -1 : (lb.w < 3 ? 0 : +1)),
                     y1 + this.width_1px * (tb.w < 1 ? -1 : (tb.w < 3 ? 0 : +1)),
                     c[col].width + this.width_1px * ( -1 + (lb.w < 1 ? +1 : (lb.w < 3 ? 0 : -1)) + (rb.w < 1 ? +1 : (rb.w < 2 ? 0 : -1)) ),
                     r[row].height + this.height_1px * ( -1 + (tb.w < 1 ? +1 : (tb.w < 3 ? 0 : -1)) + (bb.w < 1 ? +1 : (bb.w < 2 ? 0 : -1)) ))
                     .clip();
                     */
                    if ( bCur.borders.dd ) {
                        // draw diagonal line l,t - r,b
                        drawBorder( c_oAscBorderType.Diag, bCur.borders.d, x1 - t.width_1px, y1 - t.height_1px, x2Diagonal, y2Diagonal );
                    }
                    if ( bCur.borders.du ) {
                        // draw diagonal line l,b - r,t
                        drawBorder( c_oAscBorderType.Diag, bCur.borders.d, x1 - t.width_1px, y2Diagonal, x2Diagonal, y1 - t.height_1px );
                    }
                    // ToDo Clip diagonal borders
                    //ctx.restore();
                    // canvas context has just been restored, so destroy border color cache
                    //bc = undefined;
                }

                // draw left border
                if ( isFirstColTmp && !t._isLeftBorderErased( col, rowCache ) ) {
                    drawVerticalBorder( bPrev, bCur, x1 - t.width_1px, y1, y2 );
                    // Если мы в печати и печатаем первый столбец, то нужно напечатать бордеры
//						if (lb.w >= 1 && drawingCtx && 0 === col) {
                    // Иначе они будут не такой ширины
                    // ToDo посмотреть что с этим ? в печати будет обрезка
//							drawVerticalBorder(lb, tb, tbPrev, bb, bbPrev, x1, y1, y2);
//						}
                }
                // draw right border
                if ( (!mc || col === mc.c2) && !t._isRightBorderErased( col, rowCache ) ) {
                    drawVerticalBorder( bCur, bNext, x2, y1, y2 );
                }
                // draw top border
                if ( isFirstRowTmp ) {
                    drawHorizontalBorder( bTopCur, bCur, x1, y1 - t.height_1px, x2 );
                    // Если мы в печати и печатаем первую строку, то нужно напечатать бордеры
//						if (tb.w > 0 && drawingCtx && 0 === row) {
                    // ToDo посмотреть что с этим ? в печати будет обрезка
//							drawHorizontalBorder.call(this, tb, lb, lbPrev, rb, rbPrev, x1, y1, x2);
//						}
                }
                if ( !mc || row === mc.r2 ) {
                    // draw bottom border
                    drawHorizontalBorder( bCur, bBotCur, x1, y2, x2 );
                }
            }

            arrPrevRow = arrCurrRow;
            arrCurrRow = arrNextRow;
            arrNextRow = [];
        }

        if ( isNotFirst ) {
            ctx.stroke();
        }
    };

    /** Рисует закрепленные области областей */
    WorksheetView.prototype._drawFrozenPane = function ( noCells ) {
        if ( this.topLeftFrozenCell ) {
            var row = this.topLeftFrozenCell.getRow0();
            var col = this.topLeftFrozenCell.getCol0();

            var tmpRange, offsetX, offsetY;
            if ( 0 < row && 0 < col ) {
                offsetX = this.cols[0].left - this.cellsLeft;
                offsetY = this.rows[0].top - this.cellsTop;
                tmpRange = new asc_Range( 0, 0, col - 1, row - 1 );
                if ( !noCells ) {
                    this._drawGrid( /*drawingCtx*/ undefined, tmpRange, offsetX, offsetY );
                    this._drawCellsAndBorders( /*drawingCtx*/undefined, tmpRange, offsetX, offsetY );
                }
            }
            if ( 0 < row ) {
                row -= 1;
                offsetX = undefined;
                offsetY = this.rows[0].top - this.cellsTop;
                tmpRange = new asc_Range( this.visibleRange.c1, 0, this.visibleRange.c2, row );
                this._drawRowHeaders( /*drawingCtx*/ undefined, 0, row, kHeaderDefault, offsetX, offsetY );
                if ( !noCells ) {
                    this._drawGrid( /*drawingCtx*/ undefined, tmpRange, offsetX, offsetY );
                    this._drawCellsAndBorders( /*drawingCtx*/undefined, tmpRange, offsetX, offsetY );
                }
            }
            if ( 0 < col ) {
                col -= 1;
                offsetX = this.cols[0].left - this.cellsLeft;
                offsetY = undefined;
                tmpRange = new asc_Range( 0, this.visibleRange.r1, col, this.visibleRange.r2 );
                this._drawColumnHeaders( /*drawingCtx*/ undefined, 0, col, kHeaderDefault, offsetX, offsetY );
                if ( !noCells ) {
                    this._drawGrid( /*drawingCtx*/ undefined, tmpRange, offsetX, offsetY );
                    this._drawCellsAndBorders( /*drawingCtx*/undefined, tmpRange, offsetX, offsetY );
                }
            }
        }
    };

    /** Рисует закрепление областей */
    WorksheetView.prototype._drawFrozenPaneLines = function ( drawingCtx ) {
        // Возможно стоит отрисовывать на overlay, а не на основной канве
        var ctx = drawingCtx ? drawingCtx : this.drawingCtx;
        var lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Object, null, this.model.getId(), c_oAscLockNameFrozenPane );
        var isLocked = this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, false );
        var color = isLocked ? c_oAscCoAuthoringOtherBorderColor : this.settings.frozenColor;
        ctx.setLineWidth( 1 ).setStrokeStyle( color ).beginPath();
        var fHorLine, fVerLine;
        if ( isLocked ) {
            fHorLine = ctx.dashLineCleverHor;
            fVerLine = ctx.dashLineCleverVer;
        }
        else {
            fHorLine = ctx.lineHorPrevPx;
            fVerLine = ctx.lineVerPrevPx;
        }

        if ( this.topLeftFrozenCell ) {
            var row = this.topLeftFrozenCell.getRow0();
            var col = this.topLeftFrozenCell.getCol0();
            if ( 0 < row ) {
                fHorLine.apply( ctx, [0, this.rows[row].top, ctx.getWidth()] );
            }
            else {
                fHorLine.apply( ctx, [0, this.headersHeight, this.headersWidth] );
            }

            if ( 0 < col ) {
                fVerLine.apply( ctx, [this.cols[col].left, 0, ctx.getHeight()] );
            }
            else {
                fVerLine.apply( ctx, [this.headersWidth, 0, this.headersHeight] );
            }
        }
        else if ( this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            fHorLine.apply( ctx, [0, this.headersHeight, this.headersWidth] );
            fVerLine.apply( ctx, [this.headersWidth, 0, this.headersHeight] );
        }

        ctx.stroke();
    };

    WorksheetView.prototype.drawFrozenGuides = function ( x, y, target ) {
        var data, offsetFrozen;
        var ctx = this.overlayCtx;

        ctx.clear();
        this._drawSelection();

        switch ( target ) {
            case c_oTargetType.FrozenAnchorV:
                x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
                data = this._findColUnderCursor( x, true, true );
                if ( data ) {
                    data.col += 1;
                    if ( 0 <= data.col && data.col < this.cols.length ) {
                        var h = ctx.getHeight();
                        var offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft;
                        offsetFrozen = this.getFrozenPaneOffset( false, true );
                        offsetX -= offsetFrozen.offsetX;
                        ctx.setFillPattern( this.settings.ptrnLineDotted1 )
                            .fillRect( this.cols[data.col].left - offsetX - this.width_1px, 0, this.width_1px, h );
                    }
                }
                break;
            case c_oTargetType.FrozenAnchorH:
                y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );
                data = this._findRowUnderCursor( y, true, true );
                if ( data ) {
                    data.row += 1;
                    if ( 0 <= data.row && data.row < this.rows.length ) {
                        var w = ctx.getWidth();
                        var offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop;
                        offsetFrozen = this.getFrozenPaneOffset( true, false );
                        offsetY -= offsetFrozen.offsetY;
                        ctx.setFillPattern( this.settings.ptrnLineDotted1 )
                            .fillRect( 0, this.rows[data.row].top - offsetY - this.height_1px, w, this.height_1px );
                    }
                }
                break;
        }
    };

    WorksheetView.prototype._isFrozenAnchor = function ( x, y ) {

        var result = {result: false, cursor: "move", name: ""};
        if ( false === this.model.sheetViews[0].asc_getShowRowColHeaders() ) {
            return result;
        }

        var _this = this;
        var frozenCell = this.topLeftFrozenCell ? this.topLeftFrozenCell : new CellAddress( 0, 0, 0 );

        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        function isPointInAnchor( x, y, rectX, rectY, rectW, rectH ) {
            var delta = 2 * asc_getcvt( 0/*px*/, 1/*pt*/, _this._getPPIX() );
            return (x >= rectX - delta) && (x <= rectX + rectW + delta) && (y >= rectY - delta) && (y <= rectY + rectH + delta);
        }

        // vertical
        var _x = this.getCellLeft( frozenCell.getCol0(), 1 ) - 0.5;
        var _y = _this.headersTop;
        var w = 0;
        var h = _this.headersHeight;
        if ( isPointInAnchor( x, y, _x, _y, w, h ) ) {
            result.result = true;
            result.name = c_oTargetType.FrozenAnchorV;
        }

        // horizontal
        _x = _this.headersLeft;
        _y = this.getCellTop( frozenCell.getRow0(), 1 ) - 0.5;
        w = _this.headersWidth - 0.5;
        h = 0;
        if ( isPointInAnchor( x, y, _x, _y, w, h ) ) {
            result.result = true;
            result.name = c_oTargetType.FrozenAnchorH;
        }

        return result;
    };

    WorksheetView.prototype.applyFrozenAnchor = function ( x, y, target ) {
        var t = this;
        var onChangeFrozenCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                t.overlayCtx.clear();
                t._drawSelection();
                return;
            }
            var lastCol = 0, lastRow = 0, data;
            if ( t.topLeftFrozenCell ) {
                lastCol = t.topLeftFrozenCell.getCol0();
                lastRow = t.topLeftFrozenCell.getRow0();
            }
            switch ( target ) {
                case c_oTargetType.FrozenAnchorV:
                    x *= asc_getcvt( 0/*px*/, 1/*pt*/, t._getPPIX() );
                    data = t._findColUnderCursor( x, true, true );
                    if ( data ) {
                        data.col += 1;
                        if ( 0 <= data.col && data.col < t.cols.length ) {
                            lastCol = data.col;
                        }
                    }
                    break;
                case c_oTargetType.FrozenAnchorH:
                    y *= asc_getcvt( 0/*px*/, 1/*pt*/, t._getPPIY() );
                    data = t._findRowUnderCursor( y, true, true );
                    if ( data ) {
                        data.row += 1;
                        if ( 0 <= data.row && data.row < t.rows.length ) {
                            lastRow = data.row;
                        }
                    }
                    break;
            }
            t._updateFreezePane( lastCol, lastRow );
        };

        this._isLockedFrozenPane( onChangeFrozenCallback );
    };

    /** Для api закрепленных областей */

    WorksheetView.prototype.freezePane = function () {
        var t = this;
        var ar = this.activeRange.clone();
        var onChangeFreezePane = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }
            var col, row, mc;
            if ( null !== t.topLeftFrozenCell ) {
                col = row = 0;
            }
            else {
                col = ar.startCol;
                row = ar.startRow;

                if ( 0 !== row || 0 !== col ) {
                    mc = t.model.getMergedByCell( row, col );
                    if ( mc ) {
                        col = mc.c1;
                        row = mc.r1;
                    }
                }

                if ( 0 === col && 0 === row ) {
                    col = ((t.visibleRange.c2 - t.visibleRange.c1) / 2) >> 0;
                    row = ((t.visibleRange.r2 - t.visibleRange.r1) / 2) >> 0;
                }
            }
            t._updateFreezePane( col, row );
        };

        return this._isLockedFrozenPane( onChangeFreezePane );
    };

    WorksheetView.prototype._updateFreezePane = function ( col, row, lockDraw ) {
        var lastCol = 0, lastRow = 0;
        if ( this.topLeftFrozenCell ) {
            lastCol = this.topLeftFrozenCell.getCol0();
            lastRow = this.topLeftFrozenCell.getRow0();
        }
        History.Create_NewPoint();
        var oData = new UndoRedoData_FromTo( new UndoRedoData_BBox( new asc_Range( lastCol, lastRow, lastCol, lastRow ) ), new UndoRedoData_BBox( new asc_Range( col, row, col, row ) ), null );
        History.Add( g_oUndoRedoWorksheet, historyitem_Worksheet_ChangeFrozenCell, this.model.getId(), null, oData );

        var isUpdate = false;
        if ( 0 === col && 0 === row ) { // Очистка
            if ( null !== this.topLeftFrozenCell ) {
                isUpdate = true;
            }
            this.topLeftFrozenCell = this.model.sheetViews[0].pane = null;
        }
        else { // Создание
            if ( null === this.topLeftFrozenCell ) {
                isUpdate = true;
            }
            var pane = this.model.sheetViews[0].pane = new asc.asc_CPane();
            this.topLeftFrozenCell = pane.topLeftFrozenCell = new CellAddress( row, col, 0 );
        }
        this.visibleRange.c1 = col;
        this.visibleRange.r1 = row;
        if ( col >= this.nColsCount ) {
            this.expandColsOnScroll( false, true, 0 );
        } // Передаем 0, чтобы увеличить размеры
        if ( row >= this.nRowsCount ) {
            this.expandRowsOnScroll( false, true, 0 );
        } // Передаем 0, чтобы увеличить размеры

        this.visibleRange.r2 = 0;
        this._calcVisibleRows();
        this.visibleRange.c2 = 0;
        this._calcVisibleColumns();
        this.handlers.trigger( "reinitializeScroll" );

        if ( this.objectRender && this.objectRender.drawingArea ) {
            this.objectRender.drawingArea.init();
        }
        if ( !lockDraw ) {
            this.draw();
        }

        // Эвент на обновление
        if ( isUpdate && !this.model.workbook.bUndoChanges && !this.model.workbook.bRedoChanges ) {
            this.handlers.trigger( "updateSheetViewSettings" );
        }
    };

    /** */

    WorksheetView.prototype._drawSelectionElement = function ( visibleRange, offsetX, offsetY, args ) {
        var range = args[0], isDashLine = args[1], lineWidth = args[2], strokeColor = args[3], fillColor = args[4], isAllRange = args[5];
        var ctx = this.overlayCtx, c = this.cols, r = this.rows;
        var oIntersection = range.intersectionSimple( visibleRange );
        if ( !oIntersection ) {
            return;
        }

        var fHorLine, fVerLine;
        if ( isDashLine ) {
            fHorLine = ctx.dashLineCleverHor;
            fVerLine = ctx.dashLineCleverVer;
        }
        else {
            fHorLine = ctx.lineHorPrevPx;
            fVerLine = ctx.lineVerPrevPx;
        }

        var firstCol = oIntersection.c1 === visibleRange.c1 && !isAllRange;
        var firstRow = oIntersection.r1 === visibleRange.r1 && !isAllRange;

        var drawLeftSide = oIntersection.c1 === range.c1;
        var drawRightSide = oIntersection.c2 === range.c2;
        var drawTopSide = oIntersection.r1 === range.r1;
        var drawBottomSide = oIntersection.r2 === range.r2;

        var x1 = c[oIntersection.c1].left - offsetX;
        var x2 = c[oIntersection.c2].left + c[oIntersection.c2].width - offsetX;
        var y1 = r[oIntersection.r1].top - offsetY;
        var y2 = r[oIntersection.r2].top + r[oIntersection.r2].height - offsetY;

        ctx.setLineWidth( lineWidth ).setStrokeStyle( strokeColor );
        if ( fillColor ) {
            ctx.setFillStyle( fillColor );
        }
        ctx.beginPath();

        if ( drawTopSide && !firstRow ) {
            fHorLine.apply( ctx, [x1, y1, x2] );
        }
        if ( drawBottomSide ) {
            fHorLine.apply( ctx, [x1, y2, x2] );
        }
        if ( drawLeftSide && !firstCol ) {
            fVerLine.apply( ctx, [x1, y1, y2] );
        }
        if ( drawRightSide ) {
            fVerLine.apply( ctx, [x2, y1, y2] );
        }

        // Отрисовка квадратов для move/resize
        if ( fillColor ) {
            if ( drawLeftSide && drawTopSide ) {
                ctx.fillRect( x1 - this.width_1px, y1 - this.height_1px, this.width_4px, this.height_4px );
            }
            if ( drawRightSide && drawTopSide ) {
                ctx.fillRect( x2 - this.width_4px, y1 - this.height_1px, this.width_4px, this.height_4px );
            }
            if ( drawRightSide && drawBottomSide ) {
                ctx.fillRect( x2 - this.width_4px, y2 - this.height_4px, this.width_4px, this.height_4px );
            }
            if ( drawLeftSide && drawBottomSide ) {
                ctx.fillRect( x1 - this.width_1px, y2 - this.height_4px, this.width_4px, this.height_4px );
            }
        }

        ctx.closePath().stroke();
    };
    /**Отрисовывает диапазон с заданными параметрами*/
    WorksheetView.prototype._drawElements = function ( thisArg, drawFunction ) {
        var cFrozen = 0, rFrozen = 0, args = Array.prototype.slice.call( arguments, 2 ), c = this.cols, r = this.rows, offsetX = c[this.visibleRange.c1].left - this.cellsLeft, offsetY = r[this.visibleRange.r1].top - this.cellsTop;
        if ( this.topLeftFrozenCell ) {
            cFrozen = this.topLeftFrozenCell.getCol0();
            rFrozen = this.topLeftFrozenCell.getRow0();
            offsetX -= this.cols[cFrozen].left - this.cols[0].left;
            offsetY -= this.rows[rFrozen].top - this.rows[0].top;

            var oFrozenRange;
            cFrozen -= 1;
            rFrozen -= 1;
            if ( 0 <= cFrozen && 0 <= rFrozen ) {
                oFrozenRange = new asc_Range( 0, 0, cFrozen, rFrozen );
                drawFunction.call( thisArg, oFrozenRange, c[0].left - this.cellsLeft, r[0].top - this.cellsTop, args );
            }
            if ( 0 <= cFrozen ) {
                oFrozenRange = new asc_Range( 0, this.visibleRange.r1, cFrozen, this.visibleRange.r2 );
                drawFunction.call( thisArg, oFrozenRange, c[0].left - this.cellsLeft, offsetY, args );
            }
            if ( 0 <= rFrozen ) {
                oFrozenRange = new asc_Range( this.visibleRange.c1, 0, this.visibleRange.c2, rFrozen );
                drawFunction.call( thisArg, oFrozenRange, offsetX, r[0].top - this.cellsTop, args );
            }
        }

        // Можно вместо call попользовать apply, но тогда нужно каждый раз соединять массив аргументов и 3 объекта
        drawFunction.call( thisArg, this.visibleRange, offsetX, offsetY, args );
    };

    /**
     * Рисует выделение вокруг ячеек
     * @param {Asc.Range} [range]
     */
    WorksheetView.prototype._drawSelection = function ( range ) {
        if ( window['IS_NATIVE_EDITOR'] ) {
            return;
        }

        if ( !this.isSelectionDialogMode ) {
            this._drawCollaborativeElements( /*bIsDrawObjects*/true );
        }
        if ((this.isSelectionDialogMode || this.isFormulaEditMode) && !this.handlers.trigger('isActive')) {
            if (this.isSelectionDialogMode) {
                this._drawSelectRange(this.activeRange.clone(true));
            } else if (this.isFormulaEditMode) {
                this._drawFormulaRanges(this.arrActiveFormulaRanges);
            }
        } else {
            this._drawSelectionRange(range);
        }
    };

    WorksheetView.prototype._drawSelectionRange = function ( range, isFrozen ) {
        isFrozen = !!isFrozen;
        if ( asc["editor"].isStartAddShape || this.objectRender.selectedGraphicObjectsExists() ) {
            if ( this.isChartAreaEditMode ) {
                this._drawFormulaRanges( this.arrActiveChartsRanges );
            }
            return;
        }

        if ( c_oAscSelectionType.RangeMax === this.activeRange.type ) {
            this.activeRange.c2 = this.cols.length - 1;
            this.activeRange.r2 = this.rows.length - 1;
        }
        else if ( c_oAscSelectionType.RangeCol === this.activeRange.type ) {
            this.activeRange.r2 = this.rows.length - 1;
        }
        else if ( c_oAscSelectionType.RangeRow === this.activeRange.type ) {
            this.activeRange.c2 = this.cols.length - 1;
        }

        var diffWidth = 0, diffHeight = 0;
        if ( this.topLeftFrozenCell ) {
            var cFrozen = this.topLeftFrozenCell.getCol0();
            var rFrozen = this.topLeftFrozenCell.getRow0();
            diffWidth = this.cols[cFrozen].left - this.cols[0].left;
            diffHeight = this.rows[rFrozen].top - this.rows[0].top;

            if ( !isFrozen ) {
                var oFrozenRange;
                cFrozen -= 1;
                rFrozen -= 1;
                if ( 0 <= cFrozen && 0 <= rFrozen ) {
                    oFrozenRange = new asc_Range( 0, 0, cFrozen, rFrozen );
                    this._drawSelectionRange( oFrozenRange, true );
                }
                if ( 0 <= cFrozen ) {
                    oFrozenRange = new asc_Range( 0, this.visibleRange.r1, cFrozen, this.visibleRange.r2 );
                    this._drawSelectionRange( oFrozenRange, true );
                }
                if ( 0 <= rFrozen ) {
                    oFrozenRange = new asc_Range( this.visibleRange.c1, 0, this.visibleRange.c2, rFrozen );
                    this._drawSelectionRange( oFrozenRange, true );
                }
            }
        }

        var tmpRange = range;
        if ( !this.isSelectionDialogMode ) {
            range = this.activeRange.intersection( range !== undefined ? range : this.visibleRange );
        }
        else {
            range = this.copyActiveRange.intersection( range !== undefined ? range : this.visibleRange );
        }

        // Copy fill Handle
        var aFH = null;
        // Вхождение range
        var aFHIntersection = null;
        if ( this.activeFillHandle !== null ) {
            // Мы в режиме автозаполнения
            aFH = this.activeFillHandle.clone( true );
            aFHIntersection = this.activeFillHandle.intersection( tmpRange !== undefined ? tmpRange : this.visibleRange );
        }

        if ( !range && !aFHIntersection && !this.isFormulaEditMode && !this.activeMoveRange && !this.isChartAreaEditMode ) {
            if ( !isFrozen ) {
                this._drawActiveHeaders();
                if ( this.isSelectionDialogMode ) {
                    this._drawSelectRange( this.activeRange.clone( true ) );
                }
            }
            return;
        }

        var ctx = this.overlayCtx;
        var opt = this.settings;
        var offsetX, offsetY;
        if ( isFrozen ) {
            if ( tmpRange.c1 !== this.visibleRange.c1 ) {
                diffWidth = 0;
            }
            if ( tmpRange.r1 !== this.visibleRange.r1 ) {
                diffHeight = 0;
            }
            offsetX = this.cols[tmpRange.c1].left - this.cellsLeft - diffWidth;
            offsetY = this.rows[tmpRange.r1].top - this.cellsTop - diffHeight;
        }
        else {
            offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft - diffWidth;
            offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop - diffHeight;
        }

        var arn = (!this.isSelectionDialogMode) ? this.activeRange.clone( true ) : this.copyActiveRange.clone( true );
        var x1 = (range) ? (this.cols[range.c1].left - offsetX - this.width_1px) : 0;
        var x2 = (range) ? (this.cols[range.c2].left + this.cols[range.c2].width - offsetX - this.width_1px) : 0;
        var y1 = (range) ? (this.rows[range.r1].top - offsetY) : 0;
        var y2 = (range) ? (this.rows[range.r2].top + this.rows[range.r2].height - offsetY - this.height_1px) : 0;
        var drawLeftSide = (range) ? (range.c1 === arn.c1) : false;
        var drawRightSide = (range) ? (range.c2 === arn.c2 && y2 >= y1) : false;
        var drawTopSide = (range) ? (range.r1 === arn.r1) : false;
        var drawBottomSide = (range) ? (range.r2 === arn.r2 && y2 >= y1) : false;
        var l, t, r, b, cr;
        // Размеры "квадрата" автозаполнения
        var fillHandleWidth = 2 * this.width_2px + this.width_1px;
        var fillHandleHeight = 2 * this.height_2px + this.height_1px;

        // Координаты выделения для автозаполнения
        var xFH1 = 0;
        var xFH2 = 0;
        var yFH1 = 0;
        var yFH2 = 0;
        // Рисуем ли мы стороны автозаполнения
        var drawLeftFillHandle;
        var drawRightFillHandle;
        var drawTopFillHandle;
        var drawBottomFillHandle;

        // set clipping rect to cells area
        ctx.save()
            .beginPath()
            .rect( this.cellsLeft, this.cellsTop, ctx.getWidth() - this.cellsLeft, ctx.getHeight() - this.cellsTop )
            .clip();

        // draw frame around cells range
        l = drawLeftSide ? -this.width_1px : 0;
        r = drawRightSide ? this.width_1px : 0;
        t = drawTopSide ? -this.height_1px : 0;
        b = drawBottomSide ? this.height_2px : 0;

        ctx.setStrokeStyle( opt.activeCellBorderColor )
            .setLineWidth( 3 )
            .beginPath();

        if ( aFHIntersection ) {
            // Считаем координаты автозаполнения
            xFH1 = this.cols[aFHIntersection.c1].left - offsetX - this.width_1px;
            xFH2 = this.cols[aFHIntersection.c2].left + this.cols[aFHIntersection.c2].width - offsetX - this.width_1px;
            yFH1 = this.rows[aFHIntersection.r1].top - offsetY;
            yFH2 = this.rows[aFHIntersection.r2].top + this.rows[aFHIntersection.r2].height - offsetY - this.height_1px;
            drawLeftFillHandle = aFHIntersection.c1 === aFH.c1;
            drawRightFillHandle = aFHIntersection.c2 === aFH.c2;
            drawTopFillHandle = aFHIntersection.r1 === aFH.r1;
            drawBottomFillHandle = aFHIntersection.r2 === aFH.r2;

            // Если мы не в нулевом состоянии, то рисуем обводку автозаполнения (толстой линией)
            if ( aFHIntersection.c1 !== aFHIntersection.c2 || aFHIntersection.r1 !== aFHIntersection.r2 || 2 !== this.fillHandleArea ) {
                if ( drawTopFillHandle ) {
                    ctx.lineHor( xFH1 + l, yFH1 - this.height_1px, xFH2 + this.width_1px + r );
                }
                if ( drawBottomFillHandle ) {
                    ctx.lineHor( xFH1 + l, yFH2, xFH2 + this.width_1px + r );
                }
                if ( drawLeftFillHandle ) {
                    ctx.lineVer( xFH1, yFH1 + t, yFH2 + b );
                }
                if ( drawRightFillHandle ) {
                    ctx.lineVer( xFH2, yFH1 + t, yFH2 + b );
                }
            }

            // Для некоторых вариантов областей нужно дорисовывать обводку для выделенной области
            switch ( this.fillHandleArea ) {
                case 1:
                    switch ( this.fillHandleDirection ) {
                        case 0:
                            // Горизонтальный
                            if ( drawLeftSide ) {
                                ctx.lineVer( x1, y1 + t, y2 + b );
                            }
                            break;
                        case 1:
                            // Вертикальный
                            if ( drawTopSide ) {
                                ctx.lineHor( x1 + l, y1 - this.height_1px, x2 + this.width_1px + r );
                            }
                            break;
                    }
                    break;
                case 2:
                    // Для внутренней области нужны все обводки
                    if ( drawTopSide ) {
                        ctx.lineHor( x1 + l, y1 - this.height_1px, x2 + this.width_1px + r );
                    }
                    if ( drawBottomSide ) {
                        ctx.lineHor( x1 + l, y2, x2 + this.width_1px + r );
                    }
                    if ( drawLeftSide ) {
                        ctx.lineVer( x1, y1 + t, y2 + b );
                    }
                    if ( drawRightSide ) {
                        ctx.lineVer( x2, y1 + t, y2 + b );
                    }
                    break;
                case 3:
                    switch ( this.fillHandleDirection ) {
                        case 0:
                            // Горизонтальный
                            if ( range && aFH.c2 !== range.c2 ) {
                                if ( drawRightSide ) {
                                    ctx.lineVer( x2, y1 + t, y2 + b );
                                }
                            }
                            break;
                        case 1:
                            // Вертикальный
                            if ( range && aFH.r2 !== range.r2 ) {
                                if ( drawBottomSide ) {
                                    ctx.lineHor( x1 + l, y2, x2 + this.width_1px + r );
                                }
                            }
                            break;
                    }
                    break;
            }

            ctx.stroke();
        }
        else {
            // Автозаполнения нет, просто рисуем обводку
            if ( drawTopSide ) {
                ctx.lineHor( x1 + l, y1 - this.height_1px, x2 + this.width_1px + r );
            }
            if ( drawBottomSide ) {
                if ( isFrozen && !drawRightSide ) {
                    fillHandleWidth = 0;
                }
                ctx.lineHor( x1 + l, y2, x2 + this.width_1px + r - fillHandleWidth );
            }
            if ( drawLeftSide ) {
                ctx.lineVer( x1, y1 + t, y2 + b );
            }
            if ( drawRightSide ) {
                if ( isFrozen && !drawBottomSide ) {
                    fillHandleHeight = 0;
                }
                ctx.lineVer( x2, y1 + t, y2 + b - fillHandleHeight );
            }
        }
        ctx.stroke();

        // draw cells overlay
        if ( range ) {
            if ( y2 >= y1 ) {
                var lRect = x1 + (drawLeftSide ? this.width_3px : this.width_1px), rRect = x2 - (drawRightSide ? this.width_2px : 0), tRect = y1 + (drawTopSide ? this.height_2px : 0), bRect = y2 - (drawBottomSide ? this.width_2px : 0);
                ctx.setFillStyle( opt.activeCellBackground )
                    .fillRect( lRect, tRect, rRect - lRect, bRect - tRect );

                var lRect2 = x1 + (drawLeftSide ? this.width_2px : this.width_1px), rRect2 = x2 - (drawRightSide ? this.width_2px : 0), tRect2 = y1 + (drawTopSide ? this.height_1px : 0), bRect2 = y2 - (drawBottomSide ? this.width_2px : 0);
                ctx.setStrokeStyle( opt.activeCellBorderColor2 ).setLineWidth( 1 ).beginPath()
                    .strokeRect( lRect2, tRect2, rRect2 - lRect2, bRect2 - tRect2 );

                var firstCell = (!this.isSelectionDialogMode) ? this.activeRange : this.copyActiveRange;
                cr = this.model.getMergedByCell( firstCell.startRow, firstCell.startCol );
                // Проверяем скрыта ли ячейка выделения
                // Получаем активную ячейку в выделении
                cr = range.intersection( null !== cr ? cr : new asc_Range( firstCell.startCol, firstCell.startRow, firstCell.startCol, firstCell.startRow ) );
                if ( cr !== null ) {
                    ctx.save().beginPath().rect( lRect, tRect, rRect - lRect, bRect - tRect ).clip();
                    var _l = this.cols[cr.c1].left - offsetX - this.width_1px, _r = this.cols[cr.c2].left + this.cols[cr.c2].width - offsetX, _t = this.rows[cr.r1].top - offsetY - this.height_1px, _b = this.rows[cr.r2].top + this.rows[cr.r2].height - offsetY;
                    ctx.clearRect( _l, _t, _r - _l, _b - _t ).restore();
                }

                if ( !(isFrozen && (!drawRightSide || !drawBottomSide)) ) {
                    // Рисуем "квадрат" для автозаполнения (располагается "квадрат" в правом нижнем углу последней ячейки выделения)
                    cr = range.intersection( new asc_Range( range.c2, range.r2, range.c2, range.r2 ) );
                    if ( cr !== null ) {
                        this.fillHandleL = this.cols[cr.c1].left - offsetX + this.cols[cr.c1].width - this.width_1px - this.width_2px;
                        this.fillHandleR = this.fillHandleL + fillHandleWidth;
                        this.fillHandleT = this.rows[cr.r1].top - offsetY + this.rows[cr.r1].height - this.height_1px - this.height_2px;
                        this.fillHandleB = this.fillHandleT + fillHandleHeight;

                        ctx.setFillStyle( opt.activeCellBorderColor ).fillRect( this.fillHandleL, this.fillHandleT, this.fillHandleR - this.fillHandleL, this.fillHandleB - this.fillHandleT );

                        ctx.setStrokeStyle( opt.activeCellBorderColor2 ).setLineWidth( 1 ).beginPath();
                        ctx.lineHorPrevPx( this.fillHandleL, this.fillHandleT, this.fillHandleR );
                        ctx.lineVerPrevPx( this.fillHandleL, this.fillHandleT, this.fillHandleB );
                        ctx.stroke();
                    }
                }
            }
            else {
                this.fillHandleL = this.fillHandleR = this.fillHandleT = this.fillHandleB = -1;
            }
        }

        // draw fill handle select
        if ( this.activeFillHandle !== null ) {
            if ( 2 === this.fillHandleArea && (aFH.c1 !== aFH.c2 || aFH.r1 !== aFH.r2) ) {
                // Для внутренней области мы должны "залить" еще и область автозаполнения
                var lFH = xFH1 + (drawLeftFillHandle ? this.width_3px : this.width_1px), rFH = xFH2 - (drawRightFillHandle ? this.width_2px : 0), tFH = yFH1 + (drawTopFillHandle ? this.height_2px : 0), bFH = yFH2 - (drawBottomFillHandle ? this.width_2px : 0);
                ctx.setFillStyle( opt.activeCellBackground )
                    .fillRect( lFH, tFH, rFH - lFH, bFH - tFH );
            }

            ctx.setStrokeStyle( opt.fillHandleBorderColorSelect ).setLineWidth( 1 ).beginPath();

            if ( aFH.c1 !== aFH.c2 || aFH.r1 !== aFH.r2 || 2 !== this.fillHandleArea ) {
                // Рисуем обводку для области автозаполнения, если мы выделили что-то
                if ( drawTopFillHandle ) {
                    ctx.lineHor( xFH1 + l + this.width_1px, yFH1 - this.height_1px, xFH2 + r );
                }
                if ( drawBottomFillHandle ) {
                    ctx.lineHor( xFH1 + l + this.width_1px, yFH2, xFH2 + r );
                }
                if ( drawLeftFillHandle ) {
                    ctx.lineVer( xFH1, yFH1 + t + this.height_1px, yFH2 + b - this.height_1px );
                }
                if ( drawRightFillHandle ) {
                    ctx.lineVer( xFH2, yFH1 + t + this.height_1px, yFH2 + b - this.height_1px );
                }
            }

            if ( 2 === this.fillHandleArea ) {
                // Если мы внутри, еще рисуем обводку для выделенной области
                if ( drawTopSide ) {
                    ctx.lineHor( x1 + l + this.width_1px, y1 - this.height_1px, x2 + r - this.width_1px );
                }
                if ( drawBottomSide ) {
                    ctx.lineHor( x1 + l + this.width_1px, y2, x2 + r - this.width_1px );
                }
                if ( drawLeftSide ) {
                    ctx.lineVer( x1, y1 + t + this.height_1px, y2 + b - this.height_1px );
                }
                if ( drawRightSide ) {
                    ctx.lineVer( x2, y1 + t + this.height_1px, y2 + b - this.height_1px );
                }
            }

            ctx.stroke();
        }

        if ( !isFrozen && this.isFormulaEditMode ) {
            this._drawFormulaRanges( this.arrActiveFormulaRanges );
        }

        if ( !isFrozen && this.isChartAreaEditMode ) {
            this._drawFormulaRanges( this.arrActiveChartsRanges );
        }

        if ( !isFrozen && this.isSelectionDialogMode ) {
            this._drawSelectRange( this.activeRange.clone( true ) );
        }
        if ( !isFrozen && this.stateFormatPainter ) {
            this._drawFormatPainterRange();
        }

        if ( null !== this.activeMoveRange ) {
            ctx.setStrokeStyle( new CColor( 0, 0, 0 ) )
                .setLineWidth( 1 )
                .beginPath();
            var aActiveMoveRangeIntersection = this.activeMoveRange.intersection( tmpRange !== undefined ? tmpRange : this.visibleRange );
            if ( aActiveMoveRangeIntersection ) {
                var drawLeftSideMoveRange = aActiveMoveRangeIntersection.c1 === this.activeMoveRange.c1;
                var drawRightSideMoveRange = aActiveMoveRangeIntersection.c2 === this.activeMoveRange.c2;
                var drawTopSideMoveRange = aActiveMoveRangeIntersection.r1 === this.activeMoveRange.r1;
                var drawBottomSideMoveRange = aActiveMoveRangeIntersection.r2 === this.activeMoveRange.r2;

                var xMoveRange1 = this.cols[aActiveMoveRangeIntersection.c1].left - offsetX - this.width_1px;
                var xMoveRange2 = this.cols[aActiveMoveRangeIntersection.c2].left + this.cols[aActiveMoveRangeIntersection.c2].width - offsetX - this.width_1px;
                var yMoveRange1 = this.rows[aActiveMoveRangeIntersection.r1].top - offsetY;
                var yMoveRange2 = this.rows[aActiveMoveRangeIntersection.r2].top + this.rows[aActiveMoveRangeIntersection.r2].height - offsetY - this.height_1px;

                if ( drawTopSideMoveRange ) {
                    ctx.lineHor( xMoveRange1, yMoveRange1 - this.height_1px, xMoveRange2 + this.width_1px );
                }
                if ( drawBottomSideMoveRange ) {
                    ctx.lineHor( xMoveRange1, yMoveRange2, xMoveRange2 + this.width_1px );
                }
                if ( drawLeftSideMoveRange ) {
                    ctx.lineVer( xMoveRange1, yMoveRange1, yMoveRange2 );
                }
                if ( drawRightSideMoveRange ) {
                    ctx.lineVer( xMoveRange2, yMoveRange1, yMoveRange2 );
                }
            }
            ctx.stroke();
        }

        // restore canvas' original clipping range
        ctx.restore();

        if ( !isFrozen ) {
            this._drawActiveHeaders();
        }
    };

    WorksheetView.prototype._drawFormatPainterRange = function () {
        var lineWidth = 1, isDashLine = true, strokeColor = new CColor( 0, 0, 0 );
        this._drawElements( this, this._drawSelectionElement, this.copyActiveRange, isDashLine, lineWidth, strokeColor );
    };

    WorksheetView.prototype._drawFormulaRanges = function ( arrRanges ) {
        var i, lineWidth = 1, isDashLine = false, length = c_oAscFormulaRangeBorderColor.length;
        var strokeColor, fillColor, colorIndex, uniqueColorIndex = 0, tmpColors = [];
        for ( i = 0; i < arrRanges.length; ++i ) {
            var oFormulaRange = arrRanges[i].clone( true );

            colorIndex = asc.getUniqueRangeColor( arrRanges, i, tmpColors );
            if ( null == colorIndex ) {
                colorIndex = uniqueColorIndex++;
            }
            tmpColors.push( colorIndex );

            strokeColor = fillColor = c_oAscFormulaRangeBorderColor[colorIndex % length];
            this._drawElements( this, this._drawSelectionElement, oFormulaRange, isDashLine, lineWidth, strokeColor, fillColor );
        }
    };

    WorksheetView.prototype._drawSelectRange = function ( oSelectRange ) {
        var lineWidth = 1, isDashLine = true, strokeColor = c_oAscCoAuthoringOtherBorderColor;
        this._drawElements( this, this._drawSelectionElement, oSelectRange, isDashLine, lineWidth, strokeColor );
    };

    WorksheetView.prototype._drawCollaborativeElements = function ( bIsDrawObjects ) {
        if ( this.collaborativeEditing.getCollaborativeEditing() ) {
            this._drawCollaborativeElementsMeOther( c_oAscLockTypes.kLockTypeMine, bIsDrawObjects );
            this._drawCollaborativeElementsMeOther( c_oAscLockTypes.kLockTypeOther, bIsDrawObjects );
            this._drawCollaborativeElementsAllLock();
        }
    };

    WorksheetView.prototype._drawCollaborativeElementsAllLock = function () {
        var currentSheetId = this.model.getId();
        var nLockAllType = this.collaborativeEditing.isLockAllOther( currentSheetId );
        if ( c_oAscMouseMoveLockedObjectType.None !== nLockAllType ) {
            var lineWidth = 1, isDashLine = true, isAllRange = true, strokeColor = (c_oAscMouseMoveLockedObjectType.TableProperties === nLockAllType) ? c_oAscCoAuthoringLockTablePropertiesBorderColor : c_oAscCoAuthoringOtherBorderColor, oAllRange = new asc_Range( 0, 0, gc_nMaxCol0, gc_nMaxRow0 );
            this._drawElements( this, this._drawSelectionElement, oAllRange, isDashLine, lineWidth, strokeColor, null, isAllRange );
        }
    };

    WorksheetView.prototype._drawCollaborativeElementsMeOther = function ( type, bIsDrawObjects ) {
        var currentSheetId = this.model.getId(), i, lineWidth = 1, isDashLine = true, strokeColor, arrayCells, oCellTmp;
        if ( c_oAscLockTypes.kLockTypeMine === type ) {
            strokeColor = c_oAscCoAuthoringMeBorderColor;
            arrayCells = this.collaborativeEditing.getLockCellsMe( currentSheetId );

            arrayCells = arrayCells.concat( this.collaborativeEditing.getArrayInsertColumnsBySheetId( currentSheetId ) );
            arrayCells = arrayCells.concat( this.collaborativeEditing.getArrayInsertRowsBySheetId( currentSheetId ) );
        }
        else {
            strokeColor = c_oAscCoAuthoringOtherBorderColor;
            arrayCells = this.collaborativeEditing.getLockCellsOther( currentSheetId );
        }

        for ( i = 0; i < arrayCells.length; ++i ) {
            oCellTmp = new asc_Range( arrayCells[i].c1, arrayCells[i].r1, arrayCells[i].c2, arrayCells[i].r2 );
            this._drawElements( this, this._drawSelectionElement, oCellTmp, isDashLine, lineWidth, strokeColor );
        }
    };

    WorksheetView.prototype._drawAutoF = function ( updatedRange, offsetX, offsetY ) {
        if ( updatedRange ) {
            this.autoFilters.drawAutoF( updatedRange, offsetX, offsetY );
        }
        else {
            this._drawElements( this, this._drawAutoF );
        }
    };

    WorksheetView.prototype.cleanSelection = function ( range, isFrozen ) {
        if ( window['IS_NATIVE_EDITOR'] ) {
            return;
        }

        isFrozen = !!isFrozen;
        if ( range === undefined ) {
            range = this.visibleRange;
        }
        var ctx = this.overlayCtx;
        var arn = this.activeRange.clone( true );
        var arnIntersection = arn.intersectionSimple( range );
        var width = ctx.getWidth();
        var height = ctx.getHeight();
        var offsetX, offsetY, diffWidth = 0, diffHeight = 0;
        var x1 = Number.MAX_VALUE;
        var x2 = -Number.MAX_VALUE;
        var y1 = Number.MAX_VALUE;
        var y2 = -Number.MAX_VALUE;
        var i;

        if ( this.topLeftFrozenCell ) {
            var cFrozen = this.topLeftFrozenCell.getCol0();
            var rFrozen = this.topLeftFrozenCell.getRow0();
            diffWidth = this.cols[cFrozen].left - this.cols[0].left;
            diffHeight = this.rows[rFrozen].top - this.rows[0].top;

            if ( !isFrozen ) {
                var oFrozenRange;
                cFrozen -= 1;
                rFrozen -= 1;
                if ( 0 <= cFrozen && 0 <= rFrozen ) {
                    oFrozenRange = new asc_Range( 0, 0, cFrozen, rFrozen );
                    this.cleanSelection( oFrozenRange, true );
                }
                if ( 0 <= cFrozen ) {
                    oFrozenRange = new asc_Range( 0, this.visibleRange.r1, cFrozen, this.visibleRange.r2 );
                    this.cleanSelection( oFrozenRange, true );
                }
                if ( 0 <= rFrozen ) {
                    oFrozenRange = new asc_Range( this.visibleRange.c1, 0, this.visibleRange.c2, rFrozen );
                    this.cleanSelection( oFrozenRange, true );
                }
            }
        }
        if ( isFrozen ) {
            if ( range.c1 !== this.visibleRange.c1 ) {
                diffWidth = 0;
            }
            if ( range.r1 !== this.visibleRange.r1 ) {
                diffHeight = 0;
            }
            offsetX = this.cols[range.c1].left - this.cellsLeft - diffWidth;
            offsetY = this.rows[range.r1].top - this.cellsTop - diffHeight;
        }
        else {
            offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft - diffWidth;
            offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop - diffHeight;
        }

        if ( arnIntersection ) {
            x1 = this.cols[arnIntersection.c1].left - offsetX - this.width_2px;
            x2 = this.cols[arnIntersection.c2].left + this.cols[arnIntersection.c2].width - offsetX + this.width_1px + /* Это ширина "квадрата" для автофильтра от границы ячейки */this.width_2px;
            y1 = this.rows[arnIntersection.r1].top - offsetY - this.height_2px;
            y2 = this.rows[arnIntersection.r2].top + this.rows[arnIntersection.r2].height - offsetY + this.height_1px + /* Это высота "квадрата" для автофильтра от границы ячейки */this.height_2px;
        }

        if ( !isFrozen ) {
            this._activateOverlayCtx();
            this._cleanColumnHeaders( arn.c1, arn.c2 );
            this._cleanRowHeades( arn.r1, arn.r2 );
            this._deactivateOverlayCtx();
        }

        // Если есть активное автозаполнения, то нужно его тоже очистить
        if ( this.activeFillHandle !== null ) {
            var activeFillClone = this.activeFillHandle.clone( true );

            // Координаты для автозаполнения
            var xFH1 = this.cols[activeFillClone.c1].left - offsetX - this.width_2px;
            var xFH2 = this.cols[activeFillClone.c2].left + this.cols[activeFillClone.c2].width - offsetX + this.width_1px + this.width_2px;
            var yFH1 = this.rows[activeFillClone.r1].top - offsetY - this.height_2px;
            var yFH2 = this.rows[activeFillClone.r2].top + this.rows[activeFillClone.r2].height - offsetY + this.height_1px + this.height_2px;

            // Выбираем наибольший range для очистки
            x1 = Math.min( x1, xFH1 );
            x2 = Math.max( x2, xFH2 );
            y1 = Math.min( y1, yFH1 );
            y2 = Math.max( y2, yFH2 );
        }

        if ( this.collaborativeEditing.getCollaborativeEditing() ) {
            var currentSheetId = this.model.getId();

            var nLockAllType = this.collaborativeEditing.isLockAllOther( currentSheetId );
            if ( c_oAscMouseMoveLockedObjectType.None !== nLockAllType ) {
                this.overlayCtx.clear();
            }
            else {
                var arrayElementsMe = this.collaborativeEditing.getLockCellsMe( currentSheetId );
                var arrayElementsOther = this.collaborativeEditing.getLockCellsOther( currentSheetId );
                var arrayElements = arrayElementsMe.concat( arrayElementsOther );
                arrayElements = arrayElements.concat( this.collaborativeEditing.getArrayInsertColumnsBySheetId( currentSheetId ) );
                arrayElements = arrayElements.concat( this.collaborativeEditing.getArrayInsertRowsBySheetId( currentSheetId ) );

                for ( i = 0; i < arrayElements.length; ++i ) {
                    var arFormulaTmp = new asc_Range( arrayElements[i].c1, arrayElements[i].r1, arrayElements[i].c2, arrayElements[i].r2 );

                    var aFormulaIntersection = arFormulaTmp.intersection( range );

                    if ( aFormulaIntersection ) {
                        // Координаты для автозаполнения
                        var xCE1 = this.cols[aFormulaIntersection.c1].left - offsetX - this.width_2px;
                        var xCE2 = this.cols[aFormulaIntersection.c2].left + this.cols[aFormulaIntersection.c2].width - offsetX + this.width_1px + this.width_2px;
                        var yCE1 = this.rows[aFormulaIntersection.r1].top - offsetY - this.height_2px;
                        var yCE2 = this.rows[aFormulaIntersection.r2].top + this.rows[aFormulaIntersection.r2].height - offsetY + this.height_1px + this.height_2px;

                        // Выбираем наибольший range для очистки
                        x1 = Math.min( x1, xCE1 );
                        x2 = Math.max( x2, xCE2 );
                        y1 = Math.min( y1, yCE1 );
                        y2 = Math.max( y2, yCE2 );
                    }
                }
            }
        }

        if ( 0 < this.arrActiveFormulaRanges.length ) {
            for ( i = 0; i < this.arrActiveFormulaRanges.length; ++i ) {
                var activeFormula = this.arrActiveFormulaRanges[i].clone( true );

                activeFormula = activeFormula.intersection( range );
                if ( null === activeFormula ) {
                    // это ссылка из формулы на еще не добавленный рэндж
                    continue;
                }

                // Координаты для range формулы
                var xF1 = this.cols[activeFormula.c1].left - offsetX - this.width_2px;
                var xF2 = activeFormula.c2 > this.cols.length ? width : this.cols[activeFormula.c2].left + this.cols[activeFormula.c2].width - offsetX + this.width_1px;
                var yF1 = this.rows[activeFormula.r1].top - offsetY - this.height_2px;
                var yF2 = activeFormula.r2 > this.rows.length ? height : this.rows[activeFormula.r2].top + this.rows[activeFormula.r2].height - offsetY + this.height_1px;

                // Выбираем наибольший range для очистки
                x1 = Math.min( x1, xF1 );
                x2 = Math.max( x2, xF2 );
                y1 = Math.min( y1, yF1 );
                y2 = Math.max( y2, yF2 );
            }

            // Вышли из редактора, очистим массив
            //TODO: Нужно ли очищать массив ячеек из формулы в отрисовке???
            /*if ( false === this.isFormulaEditMode && !isFrozen ) {
                this.arrActiveFormulaRanges = [];
            }*/
        }

        if ( 0 < this.arrActiveChartsRanges.length ) {
            for ( i = 0; i < this.arrActiveChartsRanges.length; ++i ) {
                var activeFormula = this.arrActiveChartsRanges[i].clone( true );

                activeFormula = activeFormula.intersection( range );
                if ( null === activeFormula ) {
                    // это ссылка из формулы на еще не добавленный рэндж
                    continue;
                }

                // Координаты для range формулы
                var xF1 = this.cols[activeFormula.c1].left - offsetX - this.width_2px;
                var xF2 = activeFormula.c2 > this.cols.length ? width : this.cols[activeFormula.c2].left + this.cols[activeFormula.c2].width - offsetX + this.width_1px;
                var yF1 = this.rows[activeFormula.r1].top - offsetY - this.height_2px;
                var yF2 = activeFormula.r2 > this.rows.length ? height : this.rows[activeFormula.r2].top + this.rows[activeFormula.r2].height - offsetY + this.height_1px;

                // Выбираем наибольший range для очистки
                x1 = Math.min( x1, xF1 );
                x2 = Math.max( x2, xF2 );
                y1 = Math.min( y1, yF1 );
                y2 = Math.max( y2, yF2 );
            }
        }

        if ( null !== this.activeMoveRange ) {
            var activeMoveRangeClone = this.activeMoveRange.clone( true );

            // Увеличиваем, если выходим за область видимости // Critical Bug 17413
            while ( !this.cols[activeMoveRangeClone.c2] ) {
                this.expandColsOnScroll( true );
                this.handlers.trigger( "reinitializeScrollX" );
            }
            while ( !this.rows[activeMoveRangeClone.r2] ) {
                this.expandRowsOnScroll( true );
                this.handlers.trigger( "reinitializeScrollY" );
            }

            // Координаты для перемещения диапазона
            var xMR1 = this.cols[activeMoveRangeClone.c1].left - offsetX - this.width_2px;
            var xMR2 = this.cols[activeMoveRangeClone.c2].left + this.cols[activeMoveRangeClone.c2].width - offsetX + this.width_1px + this.width_2px;
            var yMR1 = this.rows[activeMoveRangeClone.r1].top - offsetY - this.height_2px;
            var yMR2 = this.rows[activeMoveRangeClone.r2].top + this.rows[activeMoveRangeClone.r2].height - offsetY + this.height_1px + this.height_2px;

            // Выбираем наибольший range для очистки
            x1 = Math.min( x1, xMR1 );
            x2 = Math.max( x2, xMR2 );
            y1 = Math.min( y1, yMR1 );
            y2 = Math.max( y2, yMR2 );
        }

        if ( null !== this.copyActiveRange ) {
            // Координаты для перемещения диапазона
            var xCopyAr1 = this.cols[this.copyActiveRange.c1].left - offsetX - this.width_2px;
            var xCopyAr2 = this.cols[this.copyActiveRange.c2].left + this.cols[this.copyActiveRange.c2].width - offsetX + this.width_1px + this.width_2px;
            var yCopyAr1 = this.rows[this.copyActiveRange.r1].top - offsetY - this.height_2px;
            var yCopyAr2 = this.rows[this.copyActiveRange.r2].top + this.rows[this.copyActiveRange.r2].height - offsetY + this.height_1px + this.height_2px;

            // Выбираем наибольший range для очистки
            x1 = Math.min( x1, xCopyAr1 );
            x2 = Math.max( x2, xCopyAr2 );
            y1 = Math.min( y1, yCopyAr1 );
            y2 = Math.max( y2, yCopyAr2 );
        }

        if ( !(Number.MAX_VALUE === x1 && -Number.MAX_VALUE === x2 && Number.MAX_VALUE === y1 && -Number.MAX_VALUE === y2) ) {
            ctx.save()
                .beginPath()
                .rect( this.cellsLeft, this.cellsTop, ctx.getWidth() - this.cellsLeft, ctx.getHeight() - this.cellsTop )
                .clip()
                .clearRect( x1, y1, x2 - x1, y2 - y1 )
                .restore();
        }
        return this;
    };

    WorksheetView.prototype.updateSelection = function () {
        this.cleanSelection();
        this._drawSelection();
    };

    // mouseX - это разница стартовых координат от мыши при нажатии и границы
    WorksheetView.prototype.drawColumnGuides = function ( col, x, y, mouseX ) {
        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        // Учитываем координаты точки, где мы начали изменение размера
        x += mouseX;

        var ctx = this.overlayCtx;
        var offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft;
        var offsetFrozen = this.getFrozenPaneOffset( false, true );
        offsetX -= offsetFrozen.offsetX;

        var x1 = this.cols[col].left - offsetX - this.width_1px;
        var h = ctx.getHeight();
        var widthPt = (x - x1);
        if ( 0 > widthPt ) {
            widthPt = 0;
        }

        ctx.clear();
        this._drawSelection();
        ctx.setFillPattern( this.settings.ptrnLineDotted1 )
            .fillRect( x1, 0, this.width_1px, h )
            .fillRect( x, 0, this.width_1px, h );

        return new asc_CMM( {
            type      : c_oAscMouseMoveType.ResizeColumn,
            sizeCCOrPt: this._colWidthToCharCount( widthPt ),
            sizePx    : widthPt * 96 / 72,
            x         : (x1 + this.cols[col].width) * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIX() ),
            y         : this.cellsTop * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIY() )
        } );
    };

    // mouseY - это разница стартовых координат от мыши при нажатии и границы
    WorksheetView.prototype.drawRowGuides = function ( row, x, y, mouseY ) {
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );
        // Учитываем координаты точки, где мы начали изменение размера
        y += mouseY;

        var ctx = this.overlayCtx;
        var offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop;
        var offsetFrozen = this.getFrozenPaneOffset( true, false );
        offsetY -= offsetFrozen.offsetY;

        var y1 = this.rows[row].top - offsetY - this.height_1px;
        var w = ctx.getWidth();
        var heightPt = (y - y1);
        if ( 0 > heightPt ) {
            heightPt = 0;
        }

        ctx.clear();
        this._drawSelection();
        ctx.setFillPattern( this.settings.ptrnLineDotted1 )
            .fillRect( 0, y1, w, this.height_1px )
            .fillRect( 0, y, w, this.height_1px );

        return new asc_CMM( {
            type      : c_oAscMouseMoveType.ResizeRow,
            sizeCCOrPt: heightPt,
            sizePx    : heightPt * 96 / 72,
            x         : this.cellsLeft * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIX() ),
            y         : (y1 + this.rows[row].height) * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIY() )
        } );
    };

    // --- Cache ---

    WorksheetView.prototype._cleanCache = function ( range ) {
        var r, c, row;

        if ( range === undefined ) {
            range = this.activeRange.clone( true );
        }

        for ( r = range.r1; r <= range.r2; ++r ) {
            row = this.cache.rows[r];
            if ( row !== undefined ) {
                // Должны еще крайнюю удалить
                c = range.c1;
                if ( row.erased[c - 1] ) {
                    delete row.erased[c - 1];
                }
                for ( ; c <= range.c2; ++c ) {
                    if ( row.columns[c] ) {
                        delete row.columns[c];
                    }
                    if ( row.columnsWithText[c] ) {
                        delete row.columnsWithText[c];
                    }
                    if ( row.erased[c] ) {
                        delete row.erased[c];
                    }
                }
            }
        }
    };


    // ----- Cell text cache -----

    /** Очищает кэш метрик текста ячеек */
    WorksheetView.prototype._cleanCellsTextMetricsCache = function () {
        var s = this.cache.sectors = [];
        var vr = this.visibleRange;
        var h = vr.r2 + 1 - vr.r1;
        var rl = this.rows.length;
        var rc = asc_floor( rl / h ) + (rl % h > 0 ? 1 : 0);
        var range = new asc_Range( 0, 0, this.cols.length - 1, h - 1 );
        var j;
        for ( j = rc; j > 0; --j, range.r1 += h, range.r2 += h ) {
            if ( j === 1 && rl % h > 0 ) {
                range.r2 = rl - 1;
            }
            s.push( range.clone() );
        }
    };

    /**
     * Обновляет общий кэш и кэширует метрики текста ячеек для указанного диапазона
     * @param {Asc.Range} [range]  Диапазон кэширования текта
     */
    WorksheetView.prototype._prepareCellTextMetricsCache = function ( range ) {
        var firstUpdateRow = null;
        if ( !range ) {
            range = this.visibleRange;
            if ( this.topLeftFrozenCell ) {
                var row = this.topLeftFrozenCell.getRow0();
                var col = this.topLeftFrozenCell.getCol0();
                if ( 0 < row && 0 < col ) {
                    firstUpdateRow = asc.getMinValueOrNull( firstUpdateRow, this._prepareCellTextMetricsCache2( new Asc.Range( 0, 0, col - 1, row - 1 ) ) );
                }
                if ( 0 < row ) {
                    firstUpdateRow = asc.getMinValueOrNull( firstUpdateRow, this._prepareCellTextMetricsCache2( new Asc.Range( this.visibleRange.c1, 0, this.visibleRange.c2, row - 1 ) ) );
                }
                if ( 0 < col ) {
                    firstUpdateRow = asc.getMinValueOrNull( firstUpdateRow, this._prepareCellTextMetricsCache2( new Asc.Range( 0, this.visibleRange.r1, col - 1, this.visibleRange.r2 ) ) );
                }
            }
        }

        firstUpdateRow = asc.getMinValueOrNull( firstUpdateRow, this._prepareCellTextMetricsCache2( range ) );
        if ( null !== firstUpdateRow || this.isChanged ) {
            // Убрал это из _calcCellsTextMetrics, т.к. вызов был для каждого сектора(добавляло тормоза: баг 20388)
            // Код нужен для бага http://bugzserver/show_bug.cgi?id=13875
            this._updateRowPositions();
            this._calcVisibleRows();

            if ( this.objectRender ) {
                this.objectRender.updateSizeDrawingObjects( {target: c_oTargetType.RowResize, row: firstUpdateRow}, true );
            }
        }
    };

    /**
     * Обновляет общий кэш и кэширует метрики текста ячеек для указанного диапазона (сама реализация, напрямую не вызывать, только из _prepareCellTextMetricsCache)
     * @param {Asc.Range} [range]  Диапазон кэширования текта
     */
    WorksheetView.prototype._prepareCellTextMetricsCache2 = function ( range ) {
        var firstUpdateRow = null;
        var s = this.cache.sectors;
        for ( var i = 0; i < s.length; ) {
            if ( s[i].isIntersect( range ) ) {
                this._calcCellsTextMetrics( s[i] );
                s.splice( i, 1 );
                firstUpdateRow = null !== firstUpdateRow ? Math.min( range.r1, firstUpdateRow ) : range.r1;
                continue;
            }
            ++i;
        }
        return firstUpdateRow;
    };

    /**
     * Кэширует метрики текста для диапазона ячеек
     * @param {Asc.Range} range  description
     */
    WorksheetView.prototype._calcCellsTextMetrics = function ( range ) {
        var colsLength = this.cols.length;
        if ( range === undefined ) {
            range = new Asc.Range( 0, 0, colsLength - 1, this.rows.length - 1 );
        }
        var rowModel, rowCells, cellColl;
        for ( var row = range.r1; row <= range.r2; ++row ) {
            if ( this.height_1px > this.rows[row].height ) {
                continue;
            }
            // Теперь получаем только не пустые ячейки для строки
            rowModel = this.model._getRowNoEmpty( row );
            if ( null === rowModel ) {
                continue;
            }
            rowCells = rowModel.getCells();
            for ( cellColl in rowCells ) {
                cellColl = cellColl - 0;
                if ( colsLength <= cellColl || this.width_1px > this.cols[cellColl].width ) {
                    continue;
                }

                this._addCellTextToCache( cellColl, row );
            }
        }
        this.isChanged = false;
    };

    WorksheetView.prototype._fetchRowCache = function ( row ) {
        return (this.cache.rows[row] = ( this.cache.rows[row] || new CacheElement() ));
    };

    WorksheetView.prototype._fetchCellCache = function ( col, row ) {
        var r = this._fetchRowCache( row );
        return (r.columns[col] = ( r.columns[col] || {} ));
    };

    WorksheetView.prototype._fetchCellCacheText = function ( col, row ) {
        var r = this._fetchRowCache( row );
        return (r.columnsWithText[col] = ( r.columnsWithText[col] || {} ));
    };

    WorksheetView.prototype._getRowCache = function ( row ) {
        return this.cache.rows[row];
    };

    WorksheetView.prototype._getCellCache = function ( col, row ) {
        var r = this.cache.rows[row];
        return r ? r.columns[col] : undefined;
    };

    WorksheetView.prototype._getCellTextCache = function ( col, row, dontLookupMergedCells ) {
        var r = this.cache.rows[row], c = r ? r.columns[col] : undefined;
        if ( c && c.text ) {
            return c.text;
        }
        else if ( !dontLookupMergedCells ) {
            // ToDo проверить это условие, возможно оно избыточно
            var range = this.model.getMergedByCell( row, col );
            return null !== range ? this._getCellTextCache( range.c1, range.r1, true ) : undefined;
        }
        return undefined;
    };

    WorksheetView.prototype._changeColWidth = function ( col, width, pad ) {
        var cc = Math.min( this._colWidthToCharCount( width + pad ), c_oAscMaxColumnWidth );
        var modelw = this._charCountToModelColWidth( cc );
        var colw = this._calcColWidth( modelw );

        if ( colw.width > this.cols[col].width ) {
            this.cols[col].width = colw.width;
            this.cols[col].innerWidth = colw.innerWidth;
            this.cols[col].charCount = colw.charCount;

            History.Create_NewPoint();
            History.StartTransaction();
            // Выставляем, что это bestFit
            this.model.setColBestFit( true, modelw, col, col );
            History.EndTransaction();

            this._updateColumnPositions();
            this.isChanged = true;
        }
    };

    WorksheetView.prototype._addCellTextToCache = function ( col, row, canChangeColWidth ) {
        var self = this;

        function makeFnIsGoodNumFormat( flags, width ) {
            return function ( str ) {
                return self.stringRender.measureString( str, flags, width ).width <= width;
            };
        }

        var c = this._getCell( col, row );
        if ( null === c ) {
            return col;
        }

        var bUpdateScrollX = false;
        var bUpdateScrollY = false;
        // Проверка на увеличение колличества столбцов
        if ( col >= this.cols.length ) {
            bUpdateScrollX = this.expandColsOnScroll( /*isNotActive*/ false, /*updateColsCount*/ true );
        }
        // Проверка на увеличение колличества строк
        if ( row >= this.rows.length ) {
            bUpdateScrollY = this.expandRowsOnScroll( /*isNotActive*/ false, /*updateRowsCount*/ true );
        }
        if ( bUpdateScrollX && bUpdateScrollY ) {
            this.handlers.trigger( "reinitializeScroll" );
        }
        else if ( bUpdateScrollX ) {
            this.handlers.trigger( "reinitializeScrollX" );
        }
        else if ( bUpdateScrollY ) {
            this.handlers.trigger( "reinitializeScrollY" );
        }

        var str, tm, isMerged = false, strCopy;

        // Range для замерженной ячейки
        var fl = this._getCellFlags( c );
        var mc = fl.merged;
        var fMergedColumns = false;	// Замержены ли колонки (если да, то автоподбор ширины не должен работать)
        var fMergedRows = false;	// Замержены ли строки (если да, то автоподбор высоты не должен работать)
        if ( null !== mc ) {
            if ( col !== mc.c1 || row !== mc.r1 ) {
        // Проверим внесена ли первая ячейка в cache (иначе если была скрыта первая строка или первый столбец, то мы не внесем)
        if (undefined === this._getCellTextCache(mc.c1, mc.r1, true)) {
          return this._addCellTextToCache(mc.c1, mc.r1, canChangeColWidth);
        }
                return mc.c2;
            } // skip other merged cell from range
            if ( mc.c1 !== mc.c2 ) {
                fMergedColumns = true;
            }
            if ( mc.r1 !== mc.r2 ) {
                fMergedRows = true;
            }
            isMerged = true;
        }

        var angle = c.getAngle();
        if ( this._isCellEmptyTextString( c ) ) {
            if ( !angle && c.isNotDefaultFont() ) {
                // Пустая ячейка с измененной гарнитурой или размером, учитвается в высоте
                str = c.getValue2();
                if ( 0 < str.length ) {
                    // Без текста не будет толка
                    strCopy = [str[0].clone()];
                    strCopy[0].text = 'A';
                    tm = this._roundTextMetrics( this.stringRender.measureString( strCopy, fl ) );
                    this._updateRowHeight( tm, col, row, isMerged, fMergedRows );
                }
            }

            return mc ? mc.c2 : col;
        }

        var dDigitsCount = 0;
        var colWidth = 0;
        var cellType = c.getType();
        fl.isNumberFormat = (null === cellType || CellValueType.String !== cellType); // Автоподбор делается по любому типу (кроме строки)
        var numFormatStr = c.getNumFormatStr();
        var pad = this.width_padding * 2 + this.width_1px;
        var sstr, sfl, stm;

        if ( !this.cols[col].isCustomWidth && fl.isNumberFormat && !fMergedColumns && (c_oAscCanChangeColWidth.numbers === canChangeColWidth || c_oAscCanChangeColWidth.all === canChangeColWidth) ) {
            colWidth = this.cols[col].innerWidth;
            // Измеряем целую часть числа
            sstr = c.getValue2( gc_nMaxDigCountView, function () {
                return true;
            } );
            if ( "General" === numFormatStr && c_oAscCanChangeColWidth.all !== canChangeColWidth ) {
                // asc.truncFracPart изменяет исходный массив, поэтому клонируем
                var fragmentsTmp = [];
                for ( var k = 0; k < sstr.length; ++k )
                    fragmentsTmp.push( sstr[k].clone() );
                sstr = asc.truncFracPart( fragmentsTmp );
            }
            sfl = fl.clone();
            sfl.wrapText = false;
            stm = this._roundTextMetrics( this.stringRender.measureString( sstr, sfl, colWidth ) );
            // Если целая часть числа не убирается в ячейку, то расширяем столбец
            if ( stm.width > colWidth ) {
                this._changeColWidth( col, stm.width, pad );
            }
            // Обновленная ячейка
            dDigitsCount = this.cols[col].charCount;
            colWidth = this.cols[col].innerWidth;
        }
        else if ( null === mc ) {
            // Обычная ячейка
            dDigitsCount = this.cols[col].charCount;
            colWidth = this.cols[col].innerWidth;
            // подбираем ширину
            if ( !this.cols[col].isCustomWidth && !fMergedColumns && !fl.wrapText && c_oAscCanChangeColWidth.all === canChangeColWidth ) {
                sstr = c.getValue2( gc_nMaxDigCountView, function () {
                    return true;
                } );
                stm = this._roundTextMetrics( this.stringRender.measureString( sstr, fl, colWidth ) );
                if ( stm.width > colWidth ) {
                    this._changeColWidth( col, stm.width, pad );
                    // Обновленная ячейка
                    dDigitsCount = this.cols[col].charCount;
                    colWidth = this.cols[col].innerWidth;
                }
            }
        }
        else {
            // Замерженная ячейка, нужна сумма столбцов
            for ( var i = mc.c1; i <= mc.c2 && i < this.cols.length; ++i ) {
                colWidth += this.cols[i].width;
                dDigitsCount += this.cols[i].charCount;
            }
            colWidth -= pad;
        }

        // ToDo dDigitsCount нужно рассчитывать исходя не из дефалтового шрифта и размера, а исходя из текущего шрифта и размера ячейки
        str = c.getValue2( dDigitsCount, makeFnIsGoodNumFormat( fl, colWidth ) );
        var ha = c.getAlignHorizontalByValue().toLowerCase();
        var va = c.getAlignVertical().toLowerCase();
        var maxW = fl.wrapText || fl.shrinkToFit || isMerged || asc.isFixedWidthCell( str ) ? this._calcMaxWidth( col, row, mc ) : undefined;
        tm = this._roundTextMetrics( this.stringRender.measureString( str, fl, maxW ) );
        var cto = (isMerged || fl.wrapText) ? {
            maxWidth: maxW - this.cols[col].innerWidth + this.cols[col].width, leftSide: 0, rightSide: 0
        } : this._calcCellTextOffset( col, row, ha, tm.width );

        // check right side of cell text and append columns if it exceeds existing cells borders
        if ( !isMerged ) {
            var rside = this.cols[col - cto.leftSide].left + tm.width;
            var lc = this.cols[this.cols.length - 1];
            if ( rside > lc.left + lc.width ) {
                this._appendColumns( rside );
                cto = this._calcCellTextOffset( col, row, ha, tm.width );
            }
        }
        var oFontColor = c.getFontcolor();
        var rowHeight = this.rows[row].height;
        var textBound = {};

        if ( angle ) {
            //  повернутый текст учитывает мерж ячеек по строкам
            if ( fMergedRows ) {
                rowHeight = 0;

                for ( var j = mc.r1; j <= mc.r2 && j < this.nRowsCount; ++j ) {
                    rowHeight += this.rows[j].height;
                }
            }

            textBound = this.stringRender.getTransformBound( angle, 0, 0, colWidth, rowHeight, tm.width, ha, va, maxW );

//  NOTE: надо сделать как в экселе если проекция строчки на Y больше высоты ячейки подставлять # и рисовать все по центру

//                    if (fl.isNumberFormat) {
//                        var prj = Math.abs(Math.sin(angle * Math.PI / 180.0) * tm.width);
//                        if (prj > rowHeight) {
//                            //if (maxW === undefined) {}
//                            maxW = rowHeight / Math.abs(Math.cos(angle * Math.PI / 180.0));
//                            str  =  c.getValue2(gc_nMaxDigCountView, makeFnIsGoodNumFormat(fl, maxW));
//
//                            for (i = 0; i < str.length; ++i) {
//                                var f = str[i].format;
//                                if (f) f.repeat = true;
//                            }
//
//                            tm   =  this._roundTextMetrics(this.stringRender.measureString(str, fl, maxW));
//                        }
//                    }
        }

        this._fetchCellCache( col, row ).text = {
            state    : this.stringRender.getInternalState(),
            flags    : fl,
            color    : (oFontColor || this.settings.cells.defaultState.color),
            metrics  : tm,
            cellW    : cto.maxWidth,
            cellHA   : ha,
            cellVA   : va,
            sideL    : cto.leftSide,
            sideR    : cto.rightSide,
            cellType : cellType,
            isFormula: c.isFormula(),
            angle    : angle,
            textBound: textBound
        };

        this._fetchCellCacheText( col, row ).hasText = true;

        if ( !angle && (cto.leftSide !== 0 || cto.rightSide !== 0) ) {
            this._addErasedBordersToCache( col - cto.leftSide, col + cto.rightSide, row );
        }

        this._updateRowHeight( tm, col, row, isMerged, fMergedRows, va, ha, angle, maxW, colWidth, textBound );

        return mc ? mc.c2 : col;
    };

    WorksheetView.prototype._updateRowHeight = function ( tm, col, row, isMerged, fMergedRows, va, ha, angle, maxW, colWidth, textBound ) {
        var rowInfo = this.rows[row], rowHeight;
        // update row's descender
        if ( va !== kvaTop && va !== kvaCenter && !isMerged ) {
            rowInfo.descender = Math.max( rowInfo.descender, tm.height - tm.baseline );
        }

        rowHeight = rowInfo.height;

        // update row's height
        if ( !rowInfo.isCustomHeight && !(window["NATIVE_EDITOR_ENJINE"] && this.notUpdateRowHeight) ) {
            // Замерженная ячейка (с 2-мя или более строками) не влияет на высоту строк!
            if ( !fMergedRows ) {
                var newHeight = tm.height;
                if ( angle ) {
                    if ( textBound ) {
                        newHeight = textBound.height;
                    }
                }

                rowInfo.heightReal = rowInfo.height = Math.min( this.maxRowHeight, Math.max( rowInfo.height, newHeight ) );
                if ( rowHeight !== rowInfo.height ) {
                    // ToDo нужно по идее всегда выставлять, но тут проблема с тем, что не всегда можно это в историю заносить (открытие, например). А если не заносить в историю, то при сборке ничего не изменится...
                    if ( !rowInfo.isDefaultHeight ) {
                        this.model.setRowHeight( rowInfo.height + this.height_1px, row, row, false );
                    }

                    if ( angle ) {
                        this._fetchCellCache( col, row ).text.textBound = this.stringRender.getTransformBound( angle, 0, 0, colWidth, rowInfo.height, tm.width, ha, va, maxW );
                    }

                    this.isChanged = true;
                }
            }
        }
    };

    WorksheetView.prototype._calcMaxWidth = function ( col, row, mc ) {
        if ( null === mc ) {
            return this.cols[col].innerWidth;
        }

        var width = this.cols[mc.c1].innerWidth;
        for ( var c = mc.c1 + 1; c <= mc.c2 && c < this.cols.length; ++c ) {
            width += this.cols[c].width;
        }
        return width;
    };

    WorksheetView.prototype._calcCellTextOffset = function ( col, row, textAlign, textWidth ) {
        var sideL = [0], sideR = [0], i;
        var maxWidth = this.cols[col].width;
        var ls = 0, rs = 0;
        var pad = this.settings.cells.padding * asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        var textW = textAlign === khaCenter ? (textWidth + maxWidth) * 0.5 : textWidth + pad;

        if ( textAlign === khaRight || textAlign === khaCenter ) {
            sideL = this._calcCellsWidth( col, 0, row );
            // condition (sideL.lenght >= 1) is always true
            for ( i = 0; i < sideL.length && textW > sideL[i]; ++i ) {/* do nothing */
            }
            ls = i !== sideL.length ? i : i - 1;
        }

        if ( textAlign !== khaRight ) {
            sideR = this._calcCellsWidth( col, this.cols.length - 1, row );
            // condition (sideR.lenght >= 1) is always true
            for ( i = 0; i < sideR.length && textW > sideR[i]; ++i ) {/* do nothing */
            }
            rs = i !== sideR.length ? i : i - 1;
        }

        if ( textAlign === khaCenter ) {
            maxWidth = (sideL[ls] - sideL[0]) + sideR[rs];
        }
        else {
            maxWidth = textAlign === khaRight ? sideL[ls] : sideR[rs];
        }

        return {
            maxWidth: maxWidth, leftSide: ls, rightSide: rs
        };
    };

    WorksheetView.prototype._calcCellsWidth = function ( colBeg, colEnd, row ) {
        var inc = colBeg <= colEnd ? 1 : -1, res = [];
        for ( var i = colBeg; (colEnd - i) * inc >= 0; i += inc ) {
            if ( i !== colBeg && !this._isCellEmptyOrMerged( i, row ) ) {
                break;
            }
            res.push( this.cols[i].width );
            if ( res.length > 1 ) {
                res[res.length - 1] += res[res.length - 2];
            }
        }
        return res;
    };

    // Ищет текст в строке (columnsWithText - это колонки, в которых есть текст)
    WorksheetView.prototype._findSourceOfCellText = function ( col, row ) {
        var r = this._getRowCache( row );
        if ( r ) {
            for ( var i in r.columnsWithText ) {
                if ( !r.columns[i] || 0 === this.cols[i].width ) {
                    continue;
                }
                var ct = r.columns[i].text;
                if ( !ct ) {
                    continue;
                }
                i >>= 0;
                var lc = i - ct.sideL, rc = i + ct.sideR;
                if ( col >= lc && col <= rc ) {
                    return i;
                }
            }
        }
        return -1;
    };


    // ----- Merged cells cache -----

    WorksheetView.prototype._isMergedCells = function ( range ) {
        return range.isEqual( this.model.getMergedByCell( range.r1, range.c1 ) );
    };

    // ----- Cell borders cache -----

    WorksheetView.prototype._addErasedBordersToCache = function ( colBeg, colEnd, row ) {
        var rc = this._fetchRowCache( row );
        for ( var col = colBeg; col < colEnd; ++col ) {
            rc.erased[col] = true;
        }
    };

    WorksheetView.prototype._isLeftBorderErased = function ( col, rowCache ) {
        return rowCache.erased[col - 1] === true;
    };
    WorksheetView.prototype._isRightBorderErased = function ( col, rowCache ) {
        return rowCache.erased[col] === true;
    };

    WorksheetView.prototype._calcMaxBorderWidth = function ( b1, b2 ) {
        // ToDo пересмотреть
        return Math.max( b1 && b1.w, b2 && b2.w );
    };


    // ----- Cells utilities -----

    /**
     * Возвращает заголовок колонки по индексу
     * @param {Number} col  Индекс колонки
     * @return {String}
     */
    WorksheetView.prototype._getColumnTitle = function ( col ) {
        return g_oCellAddressUtils.colnumToColstrFromWsView( col + 1 );
    };

    /**
     * Возвращает заголовок строки по индексу
     * @param {Number} row  Индекс строки
     * @return {String}
     */
    WorksheetView.prototype._getRowTitle = function ( row ) {
        return "" + (row + 1);
    };

    /**
     * Возвращает заголовок ячейки по индексу
     * @param {Number} col  Индекс колонки
     * @param {Number} row  Индекс строки
     * @return {String}
     */
    WorksheetView.prototype._getCellTitle = function ( col, row ) {
        return this._getColumnTitle( col ) + this._getRowTitle( row );
    };

    /**
     * Возвращает ячейку таблицы (из Worksheet)
     * @param {Number} col  Индекс колонки
     * @param {Number} row  Индекс строки
     * @return {Range}
     */
    WorksheetView.prototype._getCell = function ( col, row ) {
        if ( this.nRowsCount < this.model.getRowsCount() + 1 ) {
            this.expandRowsOnScroll( false, true, 0 );
        } // Передаем 0, чтобы увеличить размеры
        if ( this.nColsCount < this.model.getColsCount() + 1 ) {
            this.expandColsOnScroll( false, true, 0 );
        } // Передаем 0, чтобы увеличить размеры

        if ( col < 0 || col >= this.nColsCount || row < 0 || row >= this.nRowsCount ) {
            return null;
        }

        return this.model.getCell3( row, col );
    };

    WorksheetView.prototype._getVisibleCell = function ( col, row ) {
        return this.model.getCell3( row, col );
    };

    WorksheetView.prototype._getCellFlags = function ( col, row ) {
        var c = row !== undefined ? this._getCell( col, row ) : col;
        var fl = new CellFlags();
        if ( null !== c ) {
            fl.wrapText = c.getWrap();
            fl.shrinkToFit = c.getShrinkToFit();
            fl.merged = c.hasMerged();
            fl.textAlign = c.getAlignHorizontalByValue().toLowerCase();
        }
        return fl;
    };

    WorksheetView.prototype._isCellEmptyText = function ( col, row ) {
        var c = row !== undefined ? this._getCell( col, row ) : col;
        return null === c || c.isEmptyText();
    };
    WorksheetView.prototype._isCellEmptyTextString = function ( col, row ) {
        var c = row !== undefined ? this._getCell( col, row ) : col;
        return null === c || c.isEmptyTextString();
    };

    WorksheetView.prototype._isCellEmptyOrMerged = function ( col, row ) {
        var c = row !== undefined ? this._getCell( col, row ) : col;
        if ( null === c ) {
            return true;
        }
        if ( !c.isEmptyText() ) {
            return false;
        }
        return (null === c.hasMerged());
    };

    WorksheetView.prototype._isCellEmptyOrMergedOrBackgroundColorOrBorders = function ( col, row ) {
        var c = row !== undefined ? this._getCell( col, row ) : col;
        if ( null === c ) {
            return true;
        }
        if ( !c.isEmptyTextString() ) {
            return false;
        }
        if ( null !== c.hasMerged() ) {
            return false;
        }
        var bg = c.getFill();
        if ( null !== bg ) {
            return false;
        }
        var cb = c.getBorder();
        return !((cb.l && c_oAscBorderStyles.None !== cb.l.s) || (cb.r && c_oAscBorderStyles.None !== cb.r.s) || (cb.t && c_oAscBorderStyles.None !== cb.t.s) || (cb.b && c_oAscBorderStyles.None !== cb.b.s) || (cb.dd && c_oAscBorderStyles.None !== cb.dd.s) || (cb.du && c_oAscBorderStyles.None !== cb.du.s));
    };

    WorksheetView.prototype._getRange = function ( c1, r1, c2, r2 ) {
        return this.model.getRange3( r1, c1, r2, c2 );
    };

    WorksheetView.prototype._selectColumnsByRange = function () {
        var ar = this.activeRange;
        if ( c_oAscSelectionType.RangeMax === ar.type ) {
            return;
        }
        else {
            this.cleanSelection();
            if ( c_oAscSelectionType.RangeRow === ar.type ) {
                ar.assign( 0, 0, this.cols.length - 1, this.rows.length - 1 );
                ar.type = c_oAscSelectionType.RangeMax;
            }
            else {
                ar.type = c_oAscSelectionType.RangeCol;
                ar.assign( ar.c1, 0, ar.c2, this.rows.length - 1 );
            }
            this._drawSelection();
        }
    };

    WorksheetView.prototype._selectRowsByRange = function () {
        var ar = this.activeRange;
        if ( c_oAscSelectionType.RangeMax === ar.type ) {
            return;
        }
        else {
            this.cleanSelection();

            if ( c_oAscSelectionType.RangeCol === ar.type ) {
                ar.assign( 0, 0, this.cols.length - 1, this.rows.length - 1 );
                ar.type = c_oAscSelectionType.RangeMax;
            }
            else {
                ar.type = c_oAscSelectionType.RangeRow;
                ar.assign( 0, ar.r1, this.cols.length - 1, ar.r2 );
            }

            this._drawSelection();
        }
    };

    /**
     * Возвращает true, если диапазон больше видимой области, и операции над ним могут привести к задержкам
     * @param {Asc.Range} range  Диапазон для проверки
     * @returns {Boolean}
     */
    WorksheetView.prototype._isLargeRange = function ( range ) {
        var vr = this.visibleRange;
        return range.c2 - range.c1 + 1 > (vr.c2 - vr.c1 + 1) * 3 || range.r2 - range.r1 + 1 > (vr.r2 - vr.r1 + 1) * 3;
    };

    /**
     * Возвращает true, если диапазон состоит из одной ячейки
     * @param {Asc.Range} range  Диапазон
     * @returns {Boolean}
     */
    WorksheetView.prototype._rangeIsSingleCell = function ( range ) {
        return range.c1 === range.c2 && range.r1 === range.r2;
    };

    WorksheetView.prototype.drawDepCells = function () {
        var ctx = this.overlayCtx, _cc = this.cellCommentator, c, node, that = this;

        ctx.clear();
        this._drawSelection();

        var color = new CColor( 0, 0, 255 );

        function draw_arrow( context, fromx, fromy, tox, toy ) {
            var headlen = 9, showArrow = tox > that.getCellLeft( 0, 0 ) && toy > that.getCellTop( 0, 0 ), dx = tox - fromx, dy = toy - fromy, tox = tox > that.getCellLeft( 0, 0 ) ? tox : that.getCellLeft( 0, 0 ), toy = toy > that.getCellTop( 0, 0 ) ? toy : that.getCellTop( 0, 0 ), angle = Math.atan2( dy, dx ), _a = Math.PI / 18;

            // ToDo посмотреть на четкость moveTo, lineTo
            context.save()
                .setLineWidth( 1 )
                .beginPath()
                .lineDiag
                .moveTo( _cc.pxToPt( fromx ), _cc.pxToPt( fromy ) )
                .lineTo( _cc.pxToPt( tox ), _cc.pxToPt( toy ) );
            // .dashLine(_cc.pxToPt(fromx-.5), _cc.pxToPt(fromy-.5), _cc.pxToPt(tox-.5), _cc.pxToPt(toy-.5), 15, 5)
            if ( showArrow ) {
                context
                    .moveTo( _cc.pxToPt( tox - headlen * Math.cos( angle - _a ) ), _cc.pxToPt( toy - headlen * Math.sin( angle - _a ) ) )
                    .lineTo( _cc.pxToPt( tox ), _cc.pxToPt( toy ) )
                    .lineTo( _cc.pxToPt( tox - headlen * Math.cos( angle + _a ) ), _cc.pxToPt( toy - headlen * Math.sin( angle + _a ) ) )
                    .lineTo( _cc.pxToPt( tox - headlen * Math.cos( angle - _a ) ), _cc.pxToPt( toy - headlen * Math.sin( angle - _a ) ) );
            }

            context
                .setStrokeStyle( color )
                .setFillStyle( color )
                .stroke()
                .fill()
                .closePath()
                .restore();
        }

        function gCM( _this, col, row ) {
            var metrics = {top: 0, left: 0, width: 0, height: 0, result: false}; 	// px

            var fvr = _this.getFirstVisibleRow();
            var fvc = _this.getFirstVisibleCol();
            var mergedRange = _this.model.getMergedByCell( row, col );

            if ( mergedRange && (fvc < mergedRange.c2) && (fvr < mergedRange.r2) ) {

                var startCol = (mergedRange.c1 > fvc) ? mergedRange.c1 : fvc;
                var startRow = (mergedRange.r1 > fvr) ? mergedRange.r1 : fvr;

                metrics.top = _this.getCellTop( startRow, 0 ) - _this.getCellTop( fvr, 0 ) + _this.getCellTop( 0, 0 );
                metrics.left = _this.getCellLeft( startCol, 0 ) - _this.getCellLeft( fvc, 0 ) + _this.getCellLeft( 0, 0 );

                for ( var i = startCol; i <= mergedRange.c2; i++ ) {
                    metrics.width += _this.getColumnWidth( i, 0 )
                }
                for ( var i = startRow; i <= mergedRange.r2; i++ ) {
                    metrics.height += _this.getRowHeight( i, 0 )
                }
                metrics.result = true;
            }
            else {

                metrics.top = _this.getCellTop( row, 0 ) - _this.getCellTop( fvr, 0 ) + _this.getCellTop( 0, 0 );
                metrics.left = _this.getCellLeft( col, 0 ) - _this.getCellLeft( fvc, 0 ) + _this.getCellLeft( 0, 0 );
                metrics.width = _this.getColumnWidth( col, 0 );
                metrics.height = _this.getRowHeight( row, 0 );
                metrics.result = true;
            }

            return metrics
        }

        for ( var id in this.depDrawCells ) {
            c = this.depDrawCells[id].from;
            node = this.depDrawCells[id].to;
            var mainCellMetrics = gCM( this, c.nCol, c.nRow ), nodeCellMetrics, _t1, _t2;
            for ( var id in node ) {
                if ( !node[id].isArea ) {
                    _t1 = gCM( this, node[id].returnCell().nCol, node[id].returnCell().nRow )
                    nodeCellMetrics = {
                        t  : _t1.top,
                        l  : _t1.left,
                        w  : _t1.width,
                        h  : _t1.height,
                        apt: _t1.top + _t1.height / 2,
                        apl: _t1.left + _t1.width / 4
                    };
                }
                else {
                    var _t1 = gCM( _wsV, me[id].getBBox().c1, me[id].getBBox().r1 ), _t2 = gCM( _wsV, me[id].getBBox().c2, me[id].getBBox().r2 );

                    nodeCellMetrics = {
                        t  : _t1.top,
                        l  : _t1.left,
                        w  : _t2.left + _t2.width - _t1.left,
                        h  : _t2.top + _t2.height - _t1.top,
                        apt: _t1.top + _t1.height / 2,
                        apl: _t1.left + _t1.width / 4
                    };
                }

                var x1 = Math.floor( nodeCellMetrics.apl ), y1 = Math.floor( nodeCellMetrics.apt ), x2 = Math.floor( mainCellMetrics.left + mainCellMetrics.width / 4 ), y2 = Math.floor( mainCellMetrics.top + mainCellMetrics.height / 2 );

                if ( x1 < 0 && x2 < 0 || y1 < 0 && y2 < 0 ) {
                    continue;
                }

                if ( y1 < this.getCellTop( 0, 0 ) ) {
                    y1 -= this.getCellTop( 0, 0 );
                }

                if ( y1 < 0 && y2 > 0 ) {
                    var _x1 = Math.floor( Math.sqrt( (x1 - x2) * (x1 - x2) * y1 * y1 / ((y2 - y1) * (y2 - y1)) ) );
                    // x1 -= (x1-x2>0?1:-1)*_x1;
                    if ( x1 > x2 ) {
                        x1 -= _x1;
                    }
                    else if ( x1 < x2 ) {
                        x1 += _x1;
                    }
                }
                else if ( y1 > 0 && y2 < 0 ) {
                    var _x2 = Math.floor( Math.sqrt( (x1 - x2) * (x1 - x2) * y2 * y2 / ((y2 - y1) * (y2 - y1)) ) );
                    // x2 -= (x2-x1>0?1:-1)*_x2;
                    if ( x2 > x1 ) {
                        x2 -= _x2;
                    }
                    else if ( x2 < x1 ) {
                        x2 += _x2;
                    }
                }

                if ( x1 < 0 && x2 > 0 ) {
                    var _y1 = Math.floor( Math.sqrt( (y1 - y2) * (y1 - y2) * x1 * x1 / ((x2 - x1) * (x2 - x1)) ) )
                    // y1 -= (y1-y2>0?1:-1)*_y1;
                    if ( y1 > y2 ) {
                        y1 -= _y1;
                    }
                    else if ( y1 < y2 ) {
                        y1 += _y1;
                    }
                }
                else if ( x1 > 0 && x2 < 0 ) {
                    var _y2 = Math.floor( Math.sqrt( (y1 - y2) * (y1 - y2) * x2 * x2 / ((x2 - x1) * (x2 - x1)) ) )
                    // y2 -= (y2-y1>0?1:-1)*_y2;
                    if ( y2 > y1 ) {
                        y2 -= _y2;
                    }
                    else if ( y2 < y1 ) {
                        y2 += _y2;
                    }
                }

                draw_arrow( ctx, x1 < this.getCellLeft( 0, 0 ) ? this.getCellLeft( 0, 0 ) : x1, y1 < this.getCellTop( 0, 0 ) ? this.getCellTop( 0, 0 ) : y1, x2, y2 );
                // draw_arrow(ctx, x1, y1, x2, y2);

                // ToDo посмотреть на четкость rect
                if ( nodeCellMetrics.apl > this.getCellLeft( 0, 0 ) && nodeCellMetrics.apt > this.getCellTop( 0, 0 ) ) {
                    ctx.save()
                        .beginPath()
                        .arc( _cc.pxToPt( Math.floor( nodeCellMetrics.apl ) ), _cc.pxToPt( Math.floor( nodeCellMetrics.apt ) ), 3, 0, 2 * Math.PI, false, -0.5, -0.5 )
                        .setFillStyle( color )
                        .fill()
                        .closePath()
                        .setLineWidth( 1 )
                        .setStrokeStyle( color )
                        .rect( _cc.pxToPt( nodeCellMetrics.l ), _cc.pxToPt( nodeCellMetrics.t ), _cc.pxToPt( nodeCellMetrics.w - 1 ), _cc.pxToPt( nodeCellMetrics.h - 1 ) )
                        .stroke()
                        .restore();
                }
            }
        }
    };

    WorksheetView.prototype.prepareDepCells = function ( se ) {
        var activeCell = this.activeRange, mc = this.model.getMergedByCell( activeCell.startRow, activeCell.startCol ), c1 = mc ? mc.c1 : activeCell.startCol, r1 = mc ? mc.r1 : activeCell.startRow, c = this._getVisibleCell( c1, r1 ), nodes = (se == c_oAscDrawDepOptions.Master) ? this.model.workbook.dependencyFormulas.getMasterNodes( this.model.getId(), c.getName() ) : this.model.workbook.dependencyFormulas.getSlaveNodes( this.model.getId(), c.getName() );

        if ( !nodes ) {
            return;
        }

        if ( !this.depDrawCells ) {
            this.depDrawCells = {};
        }

        if ( se == c_oAscDrawDepOptions.Master ) {
            c = c.getCells()[0];
            var id = getVertexId( this.model.getId(), c.getName() );
            this.depDrawCells[id] = {from: c, to: nodes};
        }
        else {
            var to = {}, to1, id = getVertexId( this.model.getId(), c.getName() );
            to[getVertexId( this.model.getId(), c.getName() )] = this.model.workbook.dependencyFormulas.getNode( this.model.getId(), c.getName() );
            to1 = this.model.workbook.dependencyFormulas.getNode( this.model.getId(), c.getName() );
            for ( var id2 in nodes ) {
                if ( this.depDrawCells[id2] ) {
                    $.extend( this.depDrawCells[id2].to, to )
                }
                else {
                    this.depDrawCells[id2] = {}
                    this.depDrawCells[id2].from = nodes[id2].returnCell()
                    this.depDrawCells[id2].to = {}
                    this.depDrawCells[id2].to[id] = to1;
                }
            }
        }
        this.drawDepCells();

    };

    WorksheetView.prototype.cleanDepCells = function () {
        this.depDrawCells = null;
        this.drawDepCells();
    };

    // ----- Text drawing -----

    WorksheetView.prototype._getPPIX = function () {
        return this.drawingCtx.getPPIX();
    };

    WorksheetView.prototype._getPPIY = function () {
        return this.drawingCtx.getPPIY();
    };

    WorksheetView.prototype._setFont = function ( drawingCtx, name, size ) {
        var ctx = (drawingCtx) ? drawingCtx : this.drawingCtx;
        ctx.setFont( new asc.FontProperties( name, size ) );
    };

    /**
     * @param {Asc.TextMetrics} tm
     * @return {Asc.TextMetrics}
     */
    WorksheetView.prototype._roundTextMetrics = function ( tm ) {
        tm.width = asc_calcnpt( tm.width, this._getPPIX() );
        tm.height = asc_calcnpt( tm.height, 96 );
        tm.baseline = asc_calcnpt( tm.baseline, 96 );

        if ( tm.centerline !== undefined ) {
            tm.centerline = asc_calcnpt( tm.centerline, 96 );
        }
        return tm;
    };

    WorksheetView.prototype._calcTextHorizPos = function ( x1, x2, tm, halign ) {
        switch ( halign ) {
            case khaCenter:
                return asc_calcnpt( 0.5 * (x1 + x2 + this.width_1px - tm.width), this._getPPIX() );
            case khaRight:
                return x2 + this.width_1px - this.width_padding - tm.width;
            case khaJustify:
            default:
                return x1 + this.width_padding;
        }
    };

    WorksheetView.prototype._calcTextVertPos = function ( y1, y2, baseline, tm, valign ) {
        switch ( valign ) {
            case kvaCenter:
                return asc_calcnpt( 0.5 * (y1 + y2 - tm.height), this._getPPIY() ) - this.height_1px; // ToDo возможно стоит тоже использовать 96
            case kvaTop:
                return y1 - this.height_1px;
            default:
                return baseline - tm.baseline;
        }
    };

    WorksheetView.prototype._calcTextWidth = function ( x1, x2, tm, halign ) {
        switch ( halign ) {
            case khaJustify:
                return x2 + this.width_1px - this.width_padding * 2 - x1;
            default:
                return tm.width;
        }
    };

    // ----- Scrolling -----

    WorksheetView.prototype._calcCellPosition = function ( c, r, dc, dr ) {
        var t = this;
        var vr = t.visibleRange;

        function findNextCell( col, row, dx, dy ) {
            var state = t._isCellEmptyText( col, row );
            var i = col + dx;
            var j = row + dy;
            while ( i >= 0 && i < t.cols.length && j >= 0 && j < t.rows.length ) {
                var newState = t._isCellEmptyText( i, j );
                if ( newState !== state ) {
                    var ret = {};
                    ret.col = state ? i : i - dx;
                    ret.row = state ? j : j - dy;
                    if ( ret.col !== col || ret.row !== row || state ) {
                        return ret;
                    }
                    state = newState;
                }
                i += dx;
                j += dy;
            }
            // Проверки для перехода в самый конец (ToDo пока убрал, чтобы не добавлять тормозов)
            /*if (i === t.cols.length && state)
             i = gc_nMaxCol;
             if (j === t.rows.length && state)
             j = gc_nMaxRow;*/
            return {col: i - dx, row: j - dy};
        }

        function findEnd( col, row ) {
            var nc1, nc2 = col;
            do {
                nc1 = nc2;
                nc2 = findNextCell( nc1, row, +1, 0 ).col;
            } while ( nc1 !== nc2 );
            return nc2;
        }

        function findEOT() {
            var obr = t.objectRender ? t.objectRender.getDrawingAreaMetrics() : {maxCol: 0, maxRow: 0};
            var maxCols = t.model.getColsCount();
            var maxRows = t.model.getRowsCount();
            var lastC = -1, lastR = -1;

            for ( var col = 0; col < maxCols; ++col ) {
                for ( var row = 0; row < maxRows; ++row ) {
                    if ( !t._isCellEmptyText( col, row ) ) {
                        lastC = Math.max( lastC, col );
                        lastR = Math.max( lastR, row );
                    }
                }
            }
            return {col: Math.max( lastC, obr.maxCol ), row: Math.max( lastR, obr.maxRow )};
        }

        var eot = dc > +2.0001 && dc < +2.9999 && dr > +2.0001 && dr < +2.9999 ? findEOT() : null;

        var newCol = (function () {
            if ( dc > +0.0001 && dc < +0.9999 ) {
                return c + (vr.c2 - vr.c1 + 1);
            }        // PageDown
            if ( dc < -0.0001 && dc > -0.9999 ) {
                return c - (vr.c2 - vr.c1 + 1);
            }        // PageUp
            if ( dc > +1.0001 && dc < +1.9999 ) {
                return findNextCell( c, r, +1, 0 ).col;
            }  // Ctrl + ->
            if ( dc < -1.0001 && dc > -1.9999 ) {
                return findNextCell( c, r, -1, 0 ).col;
            }  // Ctrl + <-
            if ( dc > +2.0001 && dc < +2.9999 ) {
                return !eot ? findEnd( c, r ) : eot.col;
            }  // End
            if ( dc < -2.0001 && dc > -2.9999 ) {
                return 0;
            }                              // Home
            return c + dc;
        })();
        var newRow = (function () {
            if ( dr > +0.0001 && dr < +0.9999 ) {
                return r + (vr.r2 - vr.r1 + 1);
            }
            if ( dr < -0.0001 && dr > -0.9999 ) {
                return r - (vr.r2 - vr.r1 + 1);
            }
            if ( dr > +1.0001 && dr < +1.9999 ) {
                return findNextCell( c, r, 0, +1 ).row;
            }
            if ( dr < -1.0001 && dr > -1.9999 ) {
                return findNextCell( c, r, 0, -1 ).row;
            }
            if ( dr > +2.0001 && dr < +2.9999 ) {
                return !eot ? 0 : eot.row;
            }
            if ( dr < -2.0001 && dr > -2.9999 ) {
                return 0;
            }
            return r + dr;
        })();

        if ( newCol >= t.cols.length && newCol <= gc_nMaxCol0 ) {
            t.nColsCount = newCol + 1;
            t._calcWidthColumns( /*fullRecalc*/2 );
        }
        if ( newRow >= t.rows.length && newRow <= gc_nMaxRow0 ) {
            t.nRowsCount = newRow + 1;
            t._calcHeightRows( /*fullRecalc*/2 );
        }

        return {
            col: newCol < 0 ? 0 : Math.min( newCol, t.cols.length - 1 ),
            row: newRow < 0 ? 0 : Math.min( newRow, t.rows.length - 1 )
        };
    };

    WorksheetView.prototype._isColDrawnPartially = function ( col, leftCol, diffWidth ) {
        if ( col <= leftCol || col >= this.nColsCount ) {
            return false;
        }
        diffWidth = diffWidth ? diffWidth : 0;
        var c = this.cols;
        return c[col].left + c[col].width - c[leftCol].left + this.cellsLeft + diffWidth > this.drawingCtx.getWidth();
    };

    WorksheetView.prototype._isRowDrawnPartially = function ( row, topRow, diffHeight ) {
        if ( row <= topRow || row >= this.nRowsCount ) {
            return false;
        }
        diffHeight = diffHeight ? diffHeight : 0;
        var r = this.rows;
        return r[row].top + r[row].height - r[topRow].top + this.cellsTop + diffHeight > this.drawingCtx.getHeight();
    };

    WorksheetView.prototype._isVisibleX = function () {
        var vr = this.visibleRange;
        var c = this.cols;
        var x = c[vr.c2].left + c[vr.c2].width;
        var offsetFrozen = this.getFrozenPaneOffset( false, true );
        x += offsetFrozen.offsetX;
        return x - c[vr.c1].left + this.cellsLeft < this.drawingCtx.getWidth();
    };

    WorksheetView.prototype._isVisibleY = function () {
        var vr = this.visibleRange;
        var r = this.rows;
        var y = r[vr.r2].top + r[vr.r2].height;
        var offsetFrozen = this.getFrozenPaneOffset( true, false );
        y += offsetFrozen.offsetY;
        return y - r[vr.r1].top + this.cellsTop < this.drawingCtx.getHeight();
    };

    WorksheetView.prototype._updateVisibleRowsCount = function ( skipScrollReinit ) {
        var isUpdate = false;
        this._calcVisibleRows();
        while ( this._isVisibleY() && !this.isMaxRow() ) {
            // Добавим еще строки, чтоб не было видно фон под таблицей
            this.expandRowsOnScroll( true );
            this._calcVisibleRows();
            isUpdate = true;
        }
        if ( !skipScrollReinit && isUpdate ) {
            this.handlers.trigger( "reinitializeScrollY" );
        }
    };

    WorksheetView.prototype._updateVisibleColsCount = function ( skipScrollReinit ) {
        var isUpdate = false;
        this._calcVisibleColumns();
        while ( this._isVisibleX() && !this.isMaxCol() ) {
            // Добавим еще столбцы, чтоб не было видно фон под таблицей
            this.expandColsOnScroll( true );
            this._calcVisibleColumns();
            isUpdate = true;
        }
        if ( !skipScrollReinit && isUpdate ) {
            this.handlers.trigger( "reinitializeScrollX" );
        }
    };

    WorksheetView.prototype.isMaxRow = function () {
        var rowsCountCurrent = this.rows.length;
        if ( gc_nMaxRow === rowsCountCurrent ) {
            return true;
        }

        var rowsCount = this.model.getRowsCount() + 1;
        return rowsCount <= rowsCountCurrent && this.model.isDefaultHeightHidden();
    };

    WorksheetView.prototype.isMaxCol = function () {
        var colsCountCurrent = this.cols.length;
        if ( gc_nMaxCol === colsCountCurrent ) {
            return true;
        }

        var colsCount = this.model.getColsCount() + 1;
        return colsCount <= colsCountCurrent && this.model.isDefaultWidthHidden();
    };

    WorksheetView.prototype.scrollVertical = function ( delta, editor ) {
        var vr = this.visibleRange;
        var start = this._calcCellPosition( vr.c1, vr.r1, 0, delta ).row;
        var fixStartRow = new asc_Range( vr.c1, start, vr.c2, start );
        fixStartRow.startCol = vr.c1;
        fixStartRow.startRow = start;
        this._fixSelectionOfHiddenCells( 0, delta >= 0 ? +1 : -1, fixStartRow );
        this._fixVisibleRange( fixStartRow );
        var reinitScrollY = start !== fixStartRow.r1;
        if ( reinitScrollY && 0 > delta ) // Для скролла вверх обычный сдвиг + дорисовка
        {
            delta += fixStartRow.r1 - start;
        }
        start = fixStartRow.r1;

        if ( start === vr.r1 ) {
            if ( reinitScrollY ) {
                this.handlers.trigger( "reinitializeScrollY" );
            }
            return this;
        }

        this.cleanSelection();
        this.cellCommentator.cleanSelectedComment();

        var ctx = this.drawingCtx;
        var ctxW = ctx.getWidth();
        var ctxH = ctx.getHeight();
        var offsetX, offsetY, diffWidth = 0, diffHeight = 0, cFrozen = 0, rFrozen = 0;
        if ( this.topLeftFrozenCell ) {
            cFrozen = this.topLeftFrozenCell.getCol0();
            rFrozen = this.topLeftFrozenCell.getRow0();
            diffWidth = this.cols[cFrozen].left - this.cols[0].left;
            diffHeight = this.rows[rFrozen].top - this.rows[0].top;
        }
        var oldVRE_isPartial = this._isRowDrawnPartially( vr.r2, vr.r1, diffHeight );
        var oldVR = vr.clone();
        var oldStart = vr.r1;
        var oldEnd = vr.r2;

        // ToDo стоит тут переделать весь scroll
        vr.r1 = start;
        this._updateVisibleRowsCount();
        // Это необходимо для того, чтобы строки, у которых высота по тексту, рассчитались
        if ( !oldVR.intersectionSimple( vr ) ) {
            // Полностью обновилась область
            this._prepareCellTextMetricsCache( vr );
        }
        else {
            if ( 0 > delta ) {
                // Идем вверх
                this._prepareCellTextMetricsCache( new asc_Range( vr.c1, start, vr.c2, oldStart - 1 ) );
            }
            else {
                // Идем вниз
                this._prepareCellTextMetricsCache( new asc_Range( vr.c1, oldEnd + 1, vr.c2, vr.r2 ) );
            }
        }

        var oldDec = Math.max( calcDecades( oldEnd + 1 ), 3 );
        var oldW, x, dx;
        var dy = this.rows[start].top - this.rows[oldStart].top;
        var oldH = ctxH - this.cellsTop - Math.abs( dy ) - diffHeight;
        var scrollDown = (dy > 0 && oldH > 0);
        var y = this.cellsTop + (scrollDown ? dy : 0) + diffHeight;
        var lastRowHeight = (scrollDown && oldVRE_isPartial) ? ctxH - (this.rows[oldEnd].top - this.rows[oldStart].top + this.cellsTop + diffHeight) : 0;

        if ( this.isCellEditMode && editor && this.activeRange.r1 >= rFrozen ) {
            editor.move( this.cellsLeft + (this.activeRange.c1 >= cFrozen ? diffWidth : 0), this.cellsTop + diffHeight, ctxW, ctxH );
        }

        var widthChanged = Math.max( calcDecades( vr.r2 + 1 ), 3 ) !== oldDec;
        if ( widthChanged ) {
            x = this.cellsLeft;
            this._calcHeaderColumnWidth();
            this._updateColumnPositions();
            this._calcVisibleColumns();
            this._drawCorner();
            this._cleanColumnHeadersRect();
            this._drawColumnHeaders( /*drawingCtx*/ undefined );

            dx = this.cellsLeft - x;
            oldW = ctxW - x - Math.abs( dx );

            if ( rFrozen ) {
                ctx.drawImage( ctx.getCanvas(), x, this.cellsTop, oldW, diffHeight, x + dx, this.cellsTop, oldW, diffHeight );
                // ToDo Посмотреть с объектами!!!
            }
            this._drawFrozenPane( true );
        }
        else {
            dx = 0;
            x = this.headersLeft;
            oldW = ctxW;
        }

        // Перемещаем область
        var moveHeight = oldH - lastRowHeight;
        if ( moveHeight > 0 ) {
            ctx.drawImage( ctx.getCanvas(), x, y, oldW, moveHeight, x + dx, y - dy, oldW, moveHeight );

            // Заглушка для safari (http://bugzserver/show_bug.cgi?id=25546). Режим 'copy' сначала затирает, а
            // потом рисует (а т.к. мы рисуем сами на себе, то уже картинка будет пустой)
            if ( AscBrowser.isSafari ) {
                this.drawingGraphicCtx.moveImageDataSafari( x, y, oldW, moveHeight, x + dx, y - dy );
            }
            else {
                this.drawingGraphicCtx.moveImageData( x, y, oldW, moveHeight, x + dx, y - dy );
            }
        }
        // Очищаем область
        var clearTop = this.cellsTop + diffHeight + (scrollDown && moveHeight > 0 ? moveHeight : 0);
        var clearHeight = (moveHeight > 0) ? Math.abs( dy ) + lastRowHeight : ctxH - (this.cellsTop + diffHeight);
        ctx.setFillStyle( this.settings.cells.defaultState.background )
            .fillRect( this.headersLeft, clearTop, ctxW, clearHeight );
        this.drawingGraphicCtx.clearRect( this.headersLeft, clearTop, ctxW, clearHeight );

        if ( this.objectRender && this.objectRender.drawingArea ) {
            this.objectRender.drawingArea.reinitRanges();
        }

        // Дорисовываем необходимое
        if ( dy < 0 || vr.r2 !== oldEnd || oldVRE_isPartial || dx !== 0 ) {
            var r1, r2;
            if ( moveHeight > 0 ) {
                if ( scrollDown ) {
                    r1 = oldEnd + (oldVRE_isPartial ? 0 : 1);
                    r2 = vr.r2;
                }
                else {
                    r1 = vr.r1;
                    r2 = vr.r1 - 1 - delta;
                }
            }
            else {
                r1 = vr.r1;
                r2 = vr.r2;
            }
            var range = new asc_Range( vr.c1, r1, vr.c2, r2 );
            if ( dx === 0 ) {
                this._drawRowHeaders( /*drawingCtx*/ undefined, r1, r2 );
            }
            else {
                // redraw all headres, because number of decades in row index has been changed
                this._drawRowHeaders( /*drawingCtx*/ undefined );
                if ( dx < 0 ) {
                    // draw last column
                    var r_;
                    var r1_ = r2 + 1;
                    var r2_ = vr.r2;
                    if ( r2_ >= r1_ ) {
                        r_ = new asc_Range( vr.c2, r1_, vr.c2, r2_ );
                        this._drawGrid( /*drawingCtx*/ undefined, r_ );
                        this._drawCellsAndBorders( /*drawingCtx*/undefined, r_ );
                    }
                    if ( 0 < rFrozen ) {
                        r_ = new asc_Range( vr.c2, 0, vr.c2, rFrozen - 1 );
                        offsetY = this.rows[0].top - this.cellsTop;
                        this._drawGrid( /*drawingCtx*/ undefined, r_, /*offsetXForDraw*/undefined, offsetY );
                        this._drawCellsAndBorders( /*drawingCtx*/undefined, r_, /*offsetXForDraw*/undefined, offsetY );
                    }
                }
            }
            offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft - diffWidth;
            offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop - diffHeight;
            this._drawGrid( /*drawingCtx*/ undefined, range );
            this._drawCellsAndBorders( /*drawingCtx*/undefined, range );
            this._drawAutoF( range, offsetX, offsetY );
            this.objectRender.showDrawingObjectsEx( false, new GraphicOption( this, c_oAscGraphicOption.ScrollVertical, range, {
                offsetX: offsetX, offsetY: offsetY
            } ) );
            if ( 0 < cFrozen ) {
                range.c1 = 0;
                range.c2 = cFrozen - 1;
                offsetX = this.cols[0].left - this.cellsLeft;
                this._drawGrid( /*drawingCtx*/ undefined, range, offsetX );
                this._drawCellsAndBorders( /*drawingCtx*/undefined, range, offsetX );
                this._drawAutoF( range, offsetX, offsetY );
                this.objectRender.showDrawingObjectsEx( false, new GraphicOption( this, c_oAscGraphicOption.ScrollVertical, range, {
                    offsetX: offsetX, offsetY: offsetY
                } ) );
            }
            // Отрисовывать нужно всегда, вдруг бордеры
            this._drawFrozenPaneLines();
            this._fixSelectionOfMergedCells();
            this._drawSelection();

            if ( widthChanged ) {
                this.handlers.trigger( "reinitializeScrollX" );
            }
        }

        if ( reinitScrollY ) {
            this.handlers.trigger( "reinitializeScrollY" );
        }

        this.handlers.trigger( "onDocumentPlaceChanged" );
        //ToDo this.drawDepCells();
        this.cellCommentator.updateCommentPosition();
        this.cellCommentator.drawCommentCells();
        return this;
    };

    WorksheetView.prototype.scrollHorizontal = function ( delta, editor ) {
        var vr = this.visibleRange;
        var start = this._calcCellPosition( vr.c1, vr.r1, delta, 0 ).col;
        var fixStartCol = new asc_Range( start, vr.r1, start, vr.r2 );
        fixStartCol.startCol = start;
        fixStartCol.startRow = vr.r1;
        this._fixSelectionOfHiddenCells( delta >= 0 ? +1 : -1, 0, fixStartCol );
        this._fixVisibleRange( fixStartCol );
        var reinitScrollX = start !== fixStartCol.c1;
        if ( reinitScrollX && 0 > delta ) // Для скролла влево обычный сдвиг + дорисовка
        {
            delta += fixStartCol.c1 - start;
        }
        start = fixStartCol.c1;

        if ( start === vr.c1 ) {
            if ( reinitScrollX ) {
                this.handlers.trigger( "reinitializeScrollX" );
            }
            return this;
        }

        this.cleanSelection();
        this.cellCommentator.cleanSelectedComment();

        var ctx = this.drawingCtx;
        var ctxW = ctx.getWidth();
        var ctxH = ctx.getHeight();
        var dx = this.cols[start].left - this.cols[vr.c1].left;
        var oldStart = vr.c1;
        var oldEnd = vr.c2;
        var offsetX, offsetY, diffWidth = 0, diffHeight = 0;
        var oldW = ctxW - this.cellsLeft - Math.abs( dx );
        var scrollRight = (dx > 0 && oldW > 0);
        var x = this.cellsLeft + (scrollRight ? dx : 0);
        var y = this.headersTop;
        var cFrozen = 0, rFrozen = 0;
        if ( this.topLeftFrozenCell ) {
            rFrozen = this.topLeftFrozenCell.getRow0();
            cFrozen = this.topLeftFrozenCell.getCol0();
            diffWidth = this.cols[cFrozen].left - this.cols[0].left;
            diffHeight = this.rows[rFrozen].top - this.rows[0].top;
            x += diffWidth;
            oldW -= diffWidth;
        }
        var oldVCE_isPartial = this._isColDrawnPartially( vr.c2, vr.c1, diffWidth );
        var oldVR = vr.clone();

        // ToDo стоит тут переделать весь scroll
        vr.c1 = start;
        this._updateVisibleColsCount();
        // Это необходимо для того, чтобы строки, у которых высота по тексту, рассчитались
        if ( !oldVR.intersectionSimple( vr ) ) {
            // Полностью обновилась область
            this._prepareCellTextMetricsCache( vr );
        }
        else {
            if ( 0 > delta ) {
                // Идем влево
                this._prepareCellTextMetricsCache( new asc_Range( start, vr.r1, oldStart - 1, vr.r2 ) );
            }
            else {
                // Идем вправо
                this._prepareCellTextMetricsCache( new asc_Range( oldEnd + 1, vr.r1, vr.c2, vr.r2 ) );
            }
        }

        var lastColWidth = (scrollRight && oldVCE_isPartial) ? ctxW - (this.cols[oldEnd].left - this.cols[oldStart].left + this.cellsLeft + diffWidth) : 0;

        if ( this.isCellEditMode && editor && this.activeRange.c1 >= cFrozen ) {
            editor.move( this.cellsLeft + diffWidth, this.cellsTop + (this.activeRange.r1 >= rFrozen ? diffHeight : 0), ctxW, ctxH );
        }

        // Перемещаем область
        var moveWidth = oldW - lastColWidth;
        if ( moveWidth > 0 ) {
            ctx.drawImage( ctx.getCanvas(), x, y, moveWidth, ctxH, x - dx, y, moveWidth, ctxH );

            // Заглушка для safari (http://bugzserver/show_bug.cgi?id=25546). Режим 'copy' сначала затирает, а
            // потом рисует (а т.к. мы рисуем сами на себе, то уже картинка будет пустой)
            if ( AscBrowser.isSafari ) {
                this.drawingGraphicCtx.moveImageDataSafari( x, y, moveWidth, ctxH, x - dx, y );
            }
            else {
                this.drawingGraphicCtx.moveImageData( x, y, moveWidth, ctxH, x - dx, y );
            }
        }
        // Очищаем область
        var clearLeft = this.cellsLeft + diffWidth + (scrollRight && moveWidth > 0 ? moveWidth : 0);
        var clearWidth = (moveWidth > 0) ? Math.abs( dx ) + lastColWidth : ctxW - (this.cellsLeft + diffWidth);
        ctx.setFillStyle( this.settings.cells.defaultState.background )
            .fillRect( clearLeft, y, clearWidth, ctxH );
        this.drawingGraphicCtx.clearRect( clearLeft, y, clearWidth, ctxH );

        if ( this.objectRender && this.objectRender.drawingArea ) {
            this.objectRender.drawingArea.reinitRanges();
        }

        // Дорисовываем необходимое
        if ( dx < 0 || vr.c2 !== oldEnd || oldVCE_isPartial ) {
            var c1, c2;
            if ( moveWidth > 0 ) {
                if ( scrollRight ) {
                    c1 = oldEnd + (oldVCE_isPartial ? 0 : 1);
                    c2 = vr.c2;
                }
                else {
                    c1 = vr.c1;
                    c2 = vr.c1 - 1 - delta;
                }
            }
            else {
                c1 = vr.c1;
                c2 = vr.c2;
            }
            var range = new asc_Range( c1, vr.r1, c2, vr.r2 );
            offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft - diffWidth;
            offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop - diffHeight;
            this._drawColumnHeaders( /*drawingCtx*/ undefined, c1, c2 );
            this._drawGrid( /*drawingCtx*/ undefined, range );
            this._drawCellsAndBorders( /*drawingCtx*/undefined, range );
            this._drawAutoF( range, offsetX, offsetY );
            this.objectRender.showDrawingObjectsEx( false, new GraphicOption( this, c_oAscGraphicOption.ScrollHorizontal, range, {
                offsetX: offsetX, offsetY: offsetY
            } ) );
            if ( rFrozen ) {
                range.r1 = 0;
                range.r2 = rFrozen - 1;
                offsetY = this.rows[0].top - this.cellsTop;
                this._drawGrid( /*drawingCtx*/ undefined, range, undefined, offsetY );
                this._drawCellsAndBorders( /*drawingCtx*/undefined, range, undefined, offsetY );
                this._drawAutoF( range, offsetX, offsetY );
                this.objectRender.showDrawingObjectsEx( false, new GraphicOption( this, c_oAscGraphicOption.ScrollHorizontal, range, {
                    offsetX: offsetX, offsetY: offsetY
                } ) );
            }
            // Отрисовывать нужно всегда, вдруг бордеры
            this._drawFrozenPaneLines();
            this._fixSelectionOfMergedCells();
            this._drawSelection();
        }

        if ( reinitScrollX ) {
            this.handlers.trigger( "reinitializeScrollX" );
        }

        this.handlers.trigger( "onDocumentPlaceChanged" );
        //ToDo this.drawDepCells();
        this.cellCommentator.updateCommentPosition();
        this.cellCommentator.drawCommentCells();
        return this;
    };

    // ----- Selection -----

    // x,y - абсолютные координаты относительно листа (без учета заголовков)
    WorksheetView.prototype.findCellByXY = function ( x, y, canReturnNull, skipCol, skipRow ) {
        var r = 0, c = 0, tmpRow, tmpCol, result = new CCellObjectInfo();
        if ( canReturnNull ) {
            result.col = result.row = null;
        }

        x += this.cellsLeft;
        y += this.cellsTop;
        if ( !skipCol ) {
            while ( c < this.cols.length ) {
                tmpCol = this.cols[c];
                if ( x <= tmpCol.left + tmpCol.width ) {
                    result.col = c;
                    break;
                }
                ++c;
            }

            if ( null !== result.col ) {
                result.colOff = x - this.cols[result.col].left;
            }
        }
        if ( !skipRow ) {
            while ( r < this.rows.length ) {
                tmpRow = this.rows[r];
                if ( y <= tmpRow.top + tmpRow.height ) {
                    result.row = r;
                    break;
                }
                ++r;
            }

            if ( null !== result.row ) {
                result.rowOff = y - this.rows[result.row].top;
            }
        }

        return result;
    };

    // dX = true - считать с половиной следующей ячейки
    WorksheetView.prototype._findColUnderCursor = function ( x, canReturnNull, dX ) {
        var c = this.visibleRange.c1, offset = this.cols[c].left - this.cellsLeft, c2, x1, x2, cFrozen, widthDiff = 0;
        if ( x >= this.cellsLeft ) {
            if ( this.topLeftFrozenCell ) {
                cFrozen = this.topLeftFrozenCell.getCol0();
                widthDiff = this.cols[cFrozen].left - this.cols[0].left;
                if ( x < this.cellsLeft + widthDiff && 0 !== widthDiff ) {
                    c = 0;
                    widthDiff = 0;
                }
            }
            for ( x1 = this.cellsLeft + widthDiff, c2 = this.cols.length - 1; c <= c2; ++c, x1 = x2 ) {
                x2 = x1 + this.cols[c].width;
                if ( x1 <= x && x < x2 ) {
                    if ( dX ) {
                        // Учитываем половину ячейки
                        if ( x1 <= x && x < x1 + this.cols[c].width / 2.0 ) {
                            // Это предыдущая ячейка
                            --c;
                            // Можем вернуть и -1 (но это только для fillHandle)
                        }
                    }
                    return {col: c, left: x1, right: x2};
                }
            }
            if ( !canReturnNull ) {
                return {col: c2, left: this.cols[c2].left - offset, right: x2};
            }
        }
        else {
            if ( this.topLeftFrozenCell ) {
                cFrozen = this.topLeftFrozenCell.getCol0();
                if ( 0 !== cFrozen ) {
                    c = 0;
                    offset = this.cols[c].left - this.cellsLeft;
                }
            }
            for ( x2 = this.cellsLeft + this.cols[c].width, c2 = 0; c >= c2; --c, x2 = x1 ) {
                x1 = this.cols[c].left - offset;
                if ( x1 <= x && x < x2 ) {
                    if ( dX ) {
                        // Учитываем половину ячейки
                        if ( x1 <= x && x < x1 + this.cols[c].width / 2.0 ) {
                            // Это предыдущая ячейка
                            --c;
                            // Можем вернуть и -1 (но это только для fillHandle)
                        }
                    }
                    return {col: c, left: x1, right: x2};
                }
            }
            if ( !canReturnNull ) {
                if ( dX ) {
                    // Это предыдущая ячейка
                    --c2;
                    // Можем вернуть и -1 (но это только для fillHandle)
                    return {col: c2};
                }
                return {col: c2, left: x1, right: x1 + this.cols[c2].width};
            }
        }
        return null;
    };

    // dY = true - считать с половиной следующей ячейки
    WorksheetView.prototype._findRowUnderCursor = function ( y, canReturnNull, dY ) {
        var r = this.visibleRange.r1, offset = this.rows[r].top - this.cellsTop, r2, y1, y2, rFrozen, heightDiff = 0;
        if ( y >= this.cellsTop ) {
            if ( this.topLeftFrozenCell ) {
                rFrozen = this.topLeftFrozenCell.getRow0();
                heightDiff = this.rows[rFrozen].top - this.rows[0].top;
                if ( y < this.cellsTop + heightDiff && 0 !== heightDiff ) {
                    r = 0;
                    heightDiff = 0;
                }
            }
            for ( y1 = this.cellsTop + heightDiff, r2 = this.rows.length - 1; r <= r2; ++r, y1 = y2 ) {
                y2 = y1 + this.rows[r].height;
                if ( y1 <= y && y < y2 ) {
                    if ( dY ) {
                        // Учитываем половину ячейки
                        if ( y1 <= y && y < y1 + this.rows[r].height / 2.0 ) {
                            // Это предыдущая ячейка
                            --r;
                            // Можем вернуть и -1 (но это только для fillHandle)
                        }
                    }
                    return {row: r, top: y1, bottom: y2};
                }
            }
            if ( !canReturnNull ) {
                return {row: r2, top: this.rows[r2].top - offset, bottom: y2};
            }
        }
        else {
            if ( this.topLeftFrozenCell ) {
                rFrozen = this.topLeftFrozenCell.getRow0();
                if ( 0 !== rFrozen ) {
                    r = 0;
                    offset = this.rows[r].top - this.cellsTop;
                }
            }
            for ( y2 = this.cellsTop + this.rows[r].height, r2 = 0; r >= r2; --r, y2 = y1 ) {
                y1 = this.rows[r].top - offset;
                if ( y1 <= y && y < y2 ) {
                    if ( dY ) {
                        // Учитываем половину ячейки
                        if ( y1 <= y && y < y1 + this.rows[r].height / 2.0 ) {
                            // Это предыдущая ячейка
                            --r;
                            // Можем вернуть и -1 (но это только для fillHandle)
                        }
                    }
                    return {row: r, top: y1, bottom: y2};
                }
            }
            if ( !canReturnNull ) {
                if ( dY ) {
                    // Это предыдущая ячейка
                    --r2;
                    // Можем вернуть и -1 (но это только для fillHandle)
                    return {row: r2};
                }
                return {row: r2, top: y1, bottom: y1 + this.rows[r2].height};
            }
        }
        return null;
    };

    WorksheetView.prototype._getCursorFormulaOrChart = function ( vr, x, y, offsetX, offsetY ) {
        var i, l;
        var cursor, oFormulaRange, oFormulaRangeIn, xFormula1, xFormula2, yFormula1, yFormula2;
        var col = -1, row = -1;
        var arrRanges = this.isFormulaEditMode ? this.arrActiveFormulaRanges : this.arrActiveChartsRanges, targetArr = this.isFormulaEditMode ? 0 : -1;
        for ( i = 0, l = arrRanges.length; i < l; ++i ) {
            oFormulaRange = arrRanges[i].clone( true );
            oFormulaRangeIn = oFormulaRange.intersectionSimple( vr );
            if ( oFormulaRangeIn ) {
                xFormula1 = this.cols[oFormulaRangeIn.c1].left - offsetX;
                xFormula2 = this.cols[oFormulaRangeIn.c2].left + this.cols[oFormulaRangeIn.c2].width - offsetX;
                yFormula1 = this.rows[oFormulaRangeIn.r1].top - offsetY;
                yFormula2 = this.rows[oFormulaRangeIn.r2].top + this.rows[oFormulaRangeIn.r2].height - offsetY;

                if ( (x >= xFormula1 + 5 && x <= xFormula2 - 5) && ((y >= yFormula1 - this.height_2px && y <= yFormula1 + this.height_2px) || (y >= yFormula2 - this.height_2px && y <= yFormula2 + this.height_2px)) || (y >= yFormula1 + 5 && y <= yFormula2 - 5) && ((x >= xFormula1 - this.width_2px && x <= xFormula1 + this.width_2px) || (x >= xFormula2 - this.width_2px && x <= xFormula2 + this.width_2px)) ) {
                    cursor = kCurMove;
                    break;
                }
                else if ( x >= xFormula1 && x < xFormula1 + 5 && y >= yFormula1 && y < yFormula1 + 5 ) {
                    cursor = kCurSEResize;
                    col = oFormulaRange.c2;
                    row = oFormulaRange.r2;
                    break;
                }
                else if ( x > xFormula2 - 5 && x <= xFormula2 && y > yFormula2 - 5 && y <= yFormula2 ) {
                    cursor = kCurSEResize;
                    col = oFormulaRange.c1;
                    row = oFormulaRange.r1;
                    break;
                }
                else if ( x > xFormula2 - 5 && x <= xFormula2 && y >= yFormula1 && y < yFormula1 + 5 ) {
                    cursor = kCurNEResize;
                    col = oFormulaRange.c1;
                    row = oFormulaRange.r2;
                    break;
                }
                else if ( x >= xFormula1 && x < xFormula1 + 5 && y > yFormula2 - 5 && y <= yFormula2 ) {
                    cursor = kCurNEResize;
                    col = oFormulaRange.c2;
                    row = oFormulaRange.r1;
                    break;
                }
            }
        }
        return cursor ? {
            cursor           : cursor,
            target           : c_oTargetType.MoveResizeRange,
            col              : col,
            row              : row,
            formulaRange     : oFormulaRange,
            indexFormulaRange: i,
            targetArr        : targetArr
        } : null;
    };
    WorksheetView.prototype._isCursorOnSelectionBorder = function ( ar, vr, x, y ) {
        var arIntersection = ar.intersectionSimple( vr );
        var left, top, right, bottom, wEps = this.width_2px, hEps = this.height_2px;
        if ( arIntersection ) {
            left = ar.c1 === arIntersection.c1 ? this.cols[ar.c1].left : null;
            right = ar.c2 === arIntersection.c2 ? this.cols[ar.c2].left + this.cols[ar.c2].width : null;
            top = ar.r1 === arIntersection.r1 ? this.rows[ar.r1].top : null;
            bottom = ar.r2 === arIntersection.r2 ? this.rows[ar.r2].top + this.rows[ar.r2].height : null;
            var isLeft = (null !== left && x >= left - wEps && x <= left + wEps), isRight = (null !== right && x >= right - wEps && x <= right + wEps), isTop = (null !== top && y >= top - hEps && y <= top + hEps), isBottom = (null !== bottom && y >= bottom - hEps && y <= bottom + hEps), isHorMiddle = ((null === left || x >= left - wEps) && (null === right || x <= right + wEps)), isVerMiddle = ((null === top || y >= top - hEps) && (null === bottom || y <= bottom + hEps));

            if ( ((isLeft || isRight) && isVerMiddle) || ((isTop || isBottom) && isHorMiddle) ) {
                // Мы навели на границу выделения
                return true;
            }
        }
        return false;
    };

    WorksheetView.prototype.getCursorTypeFromXY = function ( x, y, isViewerMode ) {
        var c, r, f, i, offsetX, offsetY, cellCursor, sheetId = this.model.getId(), userId, lockRangePosLeft, lockRangePosTop, lockInfo, oHyperlink, widthDiff = 0, heightDiff = 0, isLocked = false, ar = this.activeRange, target = c_oTargetType.Cells, row = -1, col = -1, isSelGraphicObject, isNotFirst;

        var frozenCursor = this._isFrozenAnchor( x, y );
        if ( !isViewerMode && frozenCursor.result ) {
            lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Object, null, sheetId, c_oAscLockNameFrozenPane );
            isLocked = this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, false );
            if ( false !== isLocked ) {
                // Кто-то сделал lock
                var frozenCell = this.topLeftFrozenCell ? this.topLeftFrozenCell : new CellAddress( 0, 0, 0 );
                userId = isLocked.UserId;
                lockRangePosLeft = this.getCellLeft( frozenCell.getCol0(), 0 );
                lockRangePosTop = this.getCellTop( frozenCell.getRow0(), 0 );
            }
            return {
                cursor          : frozenCursor.cursor,
                target          : frozenCursor.name,
                col             : -1,
                row             : -1,
                userId          : userId,
                lockRangePosLeft: lockRangePosLeft,
                lockRangePosTop : lockRangePosTop
            };
        }

        var drawingInfo = this.objectRender.checkCursorDrawingObject( x, y );
        if ( asc["editor"].isStartAddShape && CheckIdSatetShapeAdd( this.objectRender.controller.curState ) ) {
            return {cursor: kCurFillHandle, target: c_oTargetType.Shape, col: -1, row: -1};
        }

        if ( drawingInfo && drawingInfo.id ) {
            // Возможно картинка с lock
            lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Object, null, sheetId, drawingInfo.id );
            isLocked = this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, false );
            if ( false !== isLocked ) {
                // Кто-то сделал lock
                userId = isLocked.UserId;
                lockRangePosLeft = drawingInfo.object.getVisibleLeftOffset( true );
                lockRangePosTop = drawingInfo.object.getVisibleTopOffset( true );
            }

            if ( drawingInfo.hyperlink instanceof ParaHyperlink ) {
                oHyperlink = new Hyperlink();
                oHyperlink.Tooltip = drawingInfo.hyperlink.ToolTip;
                var spl = drawingInfo.hyperlink.Value.split( "!" );
                if ( spl.length === 2 ) {
                    oHyperlink.setLocation( drawingInfo.hyperlink.Value );
                }
                else {
                    oHyperlink.Hyperlink = drawingInfo.hyperlink.Value;
                }

                cellCursor = {cursor: drawingInfo.cursor, target: c_oTargetType.Cells, col: -1, row: -1, userId: userId};
                return {
                    cursor    : kCurHyperlink,
                    target    : c_oTargetType.Hyperlink,
                    hyperlink : new asc_CHyperlink( oHyperlink ),
                    cellCursor: cellCursor,
                    userId    : userId
                };
            }

            return {
                cursor          : drawingInfo.cursor,
                target          : c_oTargetType.Shape,
                drawingId       : drawingInfo.id,
                col             : -1,
                row             : -1,
                userId          : userId,
                lockRangePosLeft: lockRangePosLeft,
                lockRangePosTop : lockRangePosTop
            };
        }

        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        if ( this.stateFormatPainter ) {
            if ( x <= this.cellsLeft && y >= this.cellsTop ) {
                r = this._findRowUnderCursor( y, true );
                if ( r !== null ) {
                    target = c_oTargetType.RowHeader;
                    row = r.row;
                }
            }
            if ( y <= this.cellsTop && x >= this.cellsLeft ) {
                c = this._findColUnderCursor( x, true );
                if ( c !== null ) {
                    target = c_oTargetType.ColumnHeader;
                    col = c.col;
                }
            }
            return {cursor: kCurFormatPainterExcel, target: target, col: col, row: row};
        }

        var oResDefault = {cursor: kCurDefault, target: c_oTargetType.None, col: -1, row: -1};
        if ( x < this.cellsLeft && y < this.cellsTop ) {
            return {cursor: kCurCorner, target: c_oTargetType.Corner, col: -1, row: -1};
        }

        var cFrozen = -1, rFrozen = -1;
        offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft;
        offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop;
        if ( this.topLeftFrozenCell ) {
            cFrozen = this.topLeftFrozenCell.getCol0();
            rFrozen = this.topLeftFrozenCell.getRow0();
            widthDiff = this.cols[cFrozen].left - this.cols[0].left;
            heightDiff = this.rows[rFrozen].top - this.rows[0].top;

            offsetX = (x < this.cellsLeft + widthDiff) ? 0 : offsetX - widthDiff;
            offsetY = (y < this.cellsTop + heightDiff) ? 0 : offsetY - heightDiff;
        }

        if ( x <= this.cellsLeft && y >= this.cellsTop ) {
            r = this._findRowUnderCursor( y, true );
            if ( r === null ) {
                return oResDefault;
            }
            isNotFirst = (r.row !== (-1 !== rFrozen ? 0 : this.visibleRange.r1));
            f = !isViewerMode && (isNotFirst && y < r.top + 3 || y >= r.bottom - 3);
            // ToDo В Excel зависимость epsilon от размера ячейки (у нас фиксированный 3)
            return {
                cursor: f ? kCurRowResize : kCurRowSelect,
                target: f ? c_oTargetType.RowResize : c_oTargetType.RowHeader,
                col   : -1,
                row   : r.row + (isNotFirst && f && y < r.top + 3 ? -1 : 0),
                mouseY: f ? ((y < r.top + 3) ? (r.top - y - this.height_1px) : (r.bottom - y - this.height_1px)) : null
            };
        }
        if ( y <= this.cellsTop && x >= this.cellsLeft ) {
            c = this._findColUnderCursor( x, true );
            if ( c === null ) {
                return oResDefault;
            }
            isNotFirst = c.col !== (-1 !== cFrozen ? 0 : this.visibleRange.c1);
            f = !isViewerMode && (isNotFirst && x < c.left + 3 || x >= c.right - 3);
            // ToDo В Excel зависимость epsilon от размера ячейки (у нас фиксированный 3)
            return {
                cursor: f ? kCurColResize : kCurColSelect,
                target: f ? c_oTargetType.ColumnResize : c_oTargetType.ColumnHeader,
                col   : c.col + (isNotFirst && f && x < c.left + 3 ? -1 : 0),
                row   : -1,
                mouseX: f ? ((x < c.left + 3) ? (c.left - x - this.width_1px) : (c.right - x - this.width_1px)) : null
            };
        }

        if ( this.isFormulaEditMode || this.isChartAreaEditMode ) {
            var oFormulaOrChartCursor = this._getCursorFormulaOrChart( this.visibleRange, x, y, offsetX, offsetY );
            if ( oFormulaOrChartCursor ) {
                return oFormulaOrChartCursor;
            }
        }

        var xWithOffset = x + offsetX;
        var yWithOffset = y + offsetY;

        isSelGraphicObject = this.objectRender.selectedGraphicObjectsExists();
        if ( !isViewerMode && !isSelGraphicObject ) {
            // Эпсилон для fillHandle
            var fillHandleEpsilon = this.width_1px;
            if ( !this.isChartAreaEditMode && x >= (this.fillHandleL - fillHandleEpsilon) && x <= (this.fillHandleR + fillHandleEpsilon) && y >= (this.fillHandleT - fillHandleEpsilon) && y <= (this.fillHandleB + fillHandleEpsilon) ) {
                // Мы на "квадрате" для автозаполнения
                return {cursor: kCurFillHandle, target: c_oTargetType.FillHandle, col: -1, row: -1};
            }

            // Навели на выделение (стоит вынести в отдельный метод)
            if ( this._isCursorOnSelectionBorder( ar, this.visibleRange, xWithOffset, yWithOffset ) ) {
                return {cursor: kCurMove, target: c_oTargetType.MoveRange, col: -1, row: -1};
            }
            if ( this.topLeftFrozenCell ) {
                var oFrozenRange;
                cFrozen -= 1;
                rFrozen -= 1;
                if ( 0 <= cFrozen && 0 <= rFrozen ) {
                    oFrozenRange = new asc_Range( 0, 0, cFrozen, rFrozen );
                    if ( this._isCursorOnSelectionBorder( ar, oFrozenRange, x, y ) ) {
                        return {cursor: kCurMove, target: c_oTargetType.MoveRange, col: -1, row: -1};
                    }
                }
                if ( 0 <= cFrozen ) {
                    oFrozenRange = new asc_Range( 0, this.visibleRange.r1, cFrozen, this.visibleRange.r2 );
                    if ( this._isCursorOnSelectionBorder( ar, oFrozenRange, x, yWithOffset ) ) {
                        return {cursor: kCurMove, target: c_oTargetType.MoveRange, col: -1, row: -1};
                    }
                }
                if ( 0 <= rFrozen ) {
                    oFrozenRange = new asc_Range( this.visibleRange.c1, 0, this.visibleRange.c2, rFrozen );
                    if ( this._isCursorOnSelectionBorder( ar, oFrozenRange, xWithOffset, y ) ) {
                        return {cursor: kCurMove, target: c_oTargetType.MoveRange, col: -1, row: -1};
                    }
                }
            }
        }

        if ( x > this.cellsLeft && y > this.cellsTop ) {
            c = this._findColUnderCursor( x, true );
            r = this._findRowUnderCursor( y, true );
            if ( c === null || r === null ) {
                return oResDefault;
            }

            // Проверка на совместное редактирование
            var lockRange = undefined;
            var lockAllPosLeft = undefined;
            var lockAllPosTop = undefined;
            var userIdAllProps = undefined;
            var userIdAllSheet = undefined;
            if ( !isViewerMode && this.collaborativeEditing.getCollaborativeEditing() ) {
                var c1Recalc = null, r1Recalc = null;
                var selectRangeRecalc = new asc_Range( c.col, r.row, c.col, r.row );
                // Пересчет для входящих ячеек в добавленные строки/столбцы
                var isIntersection = this._recalcRangeByInsertRowsAndColumns( sheetId, selectRangeRecalc );
                if ( false === isIntersection ) {
                    lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Range, /*subType*/null, sheetId, new asc.asc_CCollaborativeRange( selectRangeRecalc.c1, selectRangeRecalc.r1, selectRangeRecalc.c2, selectRangeRecalc.r2 ) );
                    isLocked = this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/false );
                    if ( false !== isLocked ) {
                        // Кто-то сделал lock
                        userId = isLocked.UserId;
                        lockRange = isLocked.Element["rangeOrObjectId"];

                        c1Recalc = this.collaborativeEditing.m_oRecalcIndexColumns[sheetId].getLockOther( lockRange["c1"], c_oAscLockTypes.kLockTypeOther );
                        r1Recalc = this.collaborativeEditing.m_oRecalcIndexRows[sheetId].getLockOther( lockRange["r1"], c_oAscLockTypes.kLockTypeOther );
                        if ( null !== c1Recalc && null !== r1Recalc ) {
                            lockRangePosLeft = this.getCellLeft( c1Recalc, /*pt*/1 );
                            lockRangePosTop = this.getCellTop( r1Recalc, /*pt*/1 );
                            // Пересчитываем X и Y относительно видимой области
                            lockRangePosLeft -= offsetX;
                            lockRangePosTop -= offsetY;
                            // Пересчитываем в px
                            lockRangePosLeft *= asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIX() );
                            lockRangePosTop *= asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIY() );
                        }
                    }
                }
                else {
                    lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Range, /*subType*/null, sheetId, null );
                }
                // Проверим не удален ли весь лист (именно удален, т.к. если просто залочен, то не рисуем рамку вокруг)
                lockInfo["type"] = c_oAscLockTypeElem.Sheet;
                isLocked = this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/true );
                if ( false !== isLocked ) {
                    // Кто-то сделал lock
                    userIdAllSheet = isLocked.UserId;
                    lockAllPosLeft = this.cellsLeft * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIX() );
                    lockAllPosTop = this.cellsTop * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIY() );
                }

                // Проверим не залочены ли все свойства листа (только если не удален весь лист)
                if ( undefined === userIdAllSheet ) {
                    lockInfo["type"] = c_oAscLockTypeElem.Range;
                    lockInfo["subType"] = c_oAscLockTypeElemSubType.InsertRows;
                    isLocked = this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/true );
                    if ( false !== isLocked ) {
                        // Кто-то сделал lock
                        userIdAllProps = isLocked.UserId;

                        lockAllPosLeft = this.cellsLeft * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIX() );
                        lockAllPosTop = this.cellsTop * asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIY() );
                    }
                }
            }

            var autoFilterInfo = this.autoFilters.checkCursor( x, y, offsetX, offsetY, {
                cFrozen: cFrozen, rFrozen: rFrozen
            }, r, c );
            if ( autoFilterInfo && !isViewerMode ) {
                return {
                    cursor: kCurAutoFilter, target: c_oTargetType.FilterObject, col: -1, row: -1, idFilter: autoFilterInfo.id
                };
            }

            // Проверим есть ли комменты
            var comments = this.cellCommentator.getComments( c.col, r.row );
            var coords = undefined;
            var indexes = undefined;

            if ( 0 < comments.length ) {
                indexes = [];
                for ( i = 0; i < comments.length; ++i ) {
                    indexes.push( comments[i].asc_getId() );
                }
                coords = this.cellCommentator.getCommentsCoords( comments );
            }

            // Проверим, может мы в гиперлинке
            oHyperlink = this.model.getHyperlinkByCell( r.row, c.col );
            cellCursor = {
                cursor          : kCurCells,
                target          : c_oTargetType.Cells,
                col             : (c ? c.col : -1),
                row             : (r ? r.row : -1),
                userId          : userId,
                lockRangePosLeft: lockRangePosLeft,
                lockRangePosTop : lockRangePosTop,
                userIdAllProps  : userIdAllProps,
                lockAllPosLeft  : lockAllPosLeft,
                lockAllPosTop   : lockAllPosTop,
                userIdAllSheet  : userIdAllSheet,
                commentIndexes  : indexes,
                commentCoords   : coords
            };
            if ( null !== oHyperlink ) {
                return {
                    cursor          : kCurHyperlink,
                    target          : c_oTargetType.Hyperlink,
                    hyperlink       : new asc_CHyperlink( oHyperlink ),
                    cellCursor      : cellCursor,
                    userId          : userId,
                    lockRangePosLeft: lockRangePosLeft,
                    lockRangePosTop : lockRangePosTop,
                    userIdAllProps  : userIdAllProps,
                    userIdAllSheet  : userIdAllSheet,
                    lockAllPosLeft  : lockAllPosLeft,
                    lockAllPosTop   : lockAllPosTop,
                    commentIndexes  : indexes,
                    commentCoords   : coords
                };
            }
            return cellCursor;
        }

        return oResDefault;
    };

    WorksheetView.prototype._fixSelectionOfMergedCells = function ( fixedRange ) {
        var ar = fixedRange ? fixedRange : ((this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange);

        if ( !ar ) {
            return;
        }

        if ( ar.type && ar.type !== c_oAscSelectionType.RangeCells ) {
            return;
        }

        // ToDo - переделать этот момент!!!!
        var res = this.model.expandRangeByMerged( ar.clone( true ) );

        if ( ar.c1 !== res.c1 && ar.c1 !== res.c2 ) {
            ar.c1 = ar.c1 <= ar.c2 ? res.c1 : res.c2;
        }
        ar.c2 = ar.c1 === res.c1 ? res.c2 : (res.c1);
        if ( ar.r1 !== res.r1 && ar.r1 !== res.r2 ) {
            ar.r1 = ar.r1 <= ar.r2 ? res.r1 : res.r2;
        }
        ar.r2 = ar.r1 === res.r1 ? res.r2 : res.r1;
        ar.normalize();
    };

    WorksheetView.prototype._findVisibleCol = function ( from, dc, flag ) {
        var to = dc < 0 ? -1 : this.cols.length, c;
        for ( c = from; c !== to; c += dc ) {
            if ( this.cols[c].width > this.width_1px ) {
                return c;
            }
        }
        return flag ? -1 : this._findVisibleCol( from, dc * -1, true );
    };
    WorksheetView.prototype._findVisibleRow = function ( from, dr, flag ) {
        var to = dr < 0 ? -1 : this.rows.length, r;
        for ( r = from; r !== to; r += dr ) {
            if ( this.rows[r].height > this.height_1px ) {
                return r;
            }
        }
        return flag ? -1 : this._findVisibleRow( from, dr * -1, true );
    };

    WorksheetView.prototype._fixSelectionOfHiddenCells = function ( dc, dr, range ) {
        var ar = (range) ? range : this.activeRange, c1, c2, r1, r2, mc, i, arn = ar.clone( true );

        if ( dc === undefined ) {
            dc = +1;
        }
        if ( dr === undefined ) {
            dr = +1;
        }

        if ( ar.c2 === ar.c1 ) {
            if ( this.cols[ar.c1].width < this.width_1px ) {
                c1 = c2 = this._findVisibleCol( ar.c1, dc );
            }
        }
        else {
            if ( 0 !== dc && this.nColsCount > ar.c2 && this.cols[ar.c2].width < this.width_1px ) {
                // Проверка для одновременно замерженных и скрытых ячеек (A1:C1 merge, B:C hidden)
                for ( mc = null, i = arn.r1; i <= arn.r2; ++i ) {
                    mc = this.model.getMergedByCell( i, ar.c2 );
                    if ( mc ) {
                        break;
                    }
                }
                if ( !mc ) {
                    c2 = this._findVisibleCol( ar.c2, dc );
                }
            }
        }
        if ( c1 < 0 || c2 < 0 ) {
            throw "Error: all columns are hidden";
        }

        if ( ar.r2 === ar.r1 ) {
            if ( this.rows[ar.r1].height < this.height_1px ) {
                r1 = r2 = this._findVisibleRow( ar.r1, dr );
            }
        }
        else {
            if ( 0 !== dr && this.nRowsCount > ar.r2 && this.rows[ar.r2].height < this.height_1px ) {
                //Проверка для одновременно замерженных и скрытых ячеек (A1:A3 merge, 2:3 hidden)
                for ( mc = null, i = arn.c1; i <= arn.c2; ++i ) {
                    mc = this.model.getMergedByCell( ar.r2, i );
                    if ( mc ) {
                        break;
                    }
                }
                if ( !mc ) {
                    r2 = this._findVisibleRow( ar.r2, dr );
                }
            }
        }
        if ( r1 < 0 || r2 < 0 ) {
            throw "Error: all rows are hidden";
        }

        ar.assign( c1 !== undefined ? c1 : ar.c1, r1 !== undefined ? r1 : ar.r1, c2 !== undefined ? c2 : ar.c2, r2 !== undefined ? r2 : ar.r2 );

        // ToDo возможно это избыточные проверки
        if ( c1 >= 0 ) {
            ar.startCol = c1;
        }
        if ( r1 >= 0 ) {
            ar.startRow = r1;
        }

        if ( 0 !== dc && this.cols[ar.startCol].width < this.width_1px ) {
            c1 = this._findVisibleCol( ar.startCol, dc );
            if ( c1 >= 0 ) {
                ar.startCol = c1;
            }
        }
        if ( 0 !== dr && this.rows[ar.startRow].height < this.height_1px ) {
            r1 = this._findVisibleRow( ar.startRow, dr );
            if ( r1 >= 0 ) {
                ar.startRow = r1;
            }
        }
    };

    WorksheetView.prototype._moveActiveCellToXY = function ( x, y ) {
        var c, r;
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;

        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        if ( x < this.cellsLeft && y < this.cellsTop ) {
            ar.assign( 0, 0, this.cols.length - 1, this.rows.length - 1 );
            ar.type = c_oAscSelectionType.RangeMax;
            ar.startCol = 0;
            ar.startRow = 0;
            this._fixSelectionOfHiddenCells();
        }
        else if ( x < this.cellsLeft ) {
            r = this._findRowUnderCursor( y ).row;
            ar.assign( 0, r, this.cols.length - 1, r );
            ar.type = c_oAscSelectionType.RangeRow;
            ar.startCol = 0;
            ar.startRow = r;
            this._fixSelectionOfHiddenCells();
        }
        else if ( y < this.cellsTop ) {
            c = this._findColUnderCursor( x ).col;
            ar.assign( c, 0, c, this.rows.length - 1 );
            ar.type = c_oAscSelectionType.RangeCol;
            ar.startCol = c;
            ar.startRow = 0;
            this._fixSelectionOfHiddenCells();
        }
        else {
            c = this._findColUnderCursor( x ).col;
            r = this._findRowUnderCursor( y ).row;
            ar.assign( c, r, c, r );
            ar.startCol = c;
            ar.startRow = r;
            ar.type = c_oAscSelectionType.RangeCells;
            this._fixSelectionOfMergedCells();
        }
    };

    WorksheetView.prototype._moveActiveCellToOffset = function ( dc, dr ) {
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;
        var mc = this.model.getMergedByCell( ar.startRow, ar.startCol );
        var c = mc ? ( dc < 0 ? mc.c1 : dc > 0 ? Math.min( mc.c2, this.nColsCount - 1 - dc ) : ar.startCol) : ar.startCol;
        var r = mc ? ( dr < 0 ? mc.r1 : dr > 0 ? Math.min( mc.r2, this.nRowsCount - 1 - dr ) : ar.startRow ) : ar.startRow;
        var p = this._calcCellPosition( c, r, dc, dr );
        ar.assign( p.col, p.row, p.col, p.row );
        ar.type = c_oAscSelectionType.RangeCells;
        ar.startCol = p.col;
        ar.startRow = p.row;
        this._fixSelectionOfMergedCells();
        this._fixSelectionOfHiddenCells( dc >= 0 ? +1 : -1, dr >= 0 ? +1 : -1 );
    };

    // Движение активной ячейки в выделенной области
    WorksheetView.prototype._moveActivePointInSelection = function ( dc, dr ) {
        var ar = this.activeRange;
        var arn = this.activeRange.clone( true );

        // Если мы на скрытой строке или ячейке, то двигаться в выделении нельзя (так делает и Excel)
        if ( this.width_1px > this.cols[ar.startCol].width || this.height_1px > this.rows[ar.startRow].height ) {
            return;
        }

        // Set active cell
        ar.startCol += dc;
        ar.startRow += dr;

        do {
            var done = true;

            // Обработка выхода за границы выделения
            if ( ar.startCol < arn.c1 ) {
                ar.startCol = arn.c2;
                ar.startRow -= 1;
                if ( ar.startRow < arn.r1 ) {
                    ar.startRow = arn.r2;
                }
            }
            else if ( ar.startCol > arn.c2 ) {
                ar.startCol = arn.c1;
                ar.startRow += 1;
                if ( ar.startRow > arn.r2 ) {
                    ar.startRow = arn.r1;
                }
            }
            if ( ar.startRow < arn.r1 ) {
                ar.startRow = arn.r2;
                ar.startCol -= 1;
                if ( ar.startCol < arn.c1 ) {
                    ar.startCol = arn.c2;
                }
            }
            else if ( ar.startRow > arn.r2 ) {
                ar.startRow = arn.r1;
                ar.startCol += 1;
                if ( ar.startCol > arn.c2 ) {
                    ar.startCol = arn.c1;
                }
            }

            // Обработка движения active point через merged cells
            var mc = this.model.getMergedByCell( ar.startRow, ar.startCol );

            if ( mc ) {
                if ( dc > 0 && (ar.startCol > mc.c1 || ar.startRow !== mc.r1) ) {
                    // Движение слева направо
                    ar.startCol = mc.c2 + 1;
                    done = false;
                }
                else if ( dc < 0 && (ar.startCol < mc.c2 || ar.startRow !== mc.r1) ) {
                    // Движение справа налево
                    ar.startCol = mc.c1 - 1;
                    done = false;
                }
                if ( dr > 0 && (ar.startRow > mc.r1 || ar.startCol !== mc.c1) ) {
                    // Движение сверху вниз
                    ar.startRow = mc.r2 + 1;
                    done = false;
                }
                else if ( dr < 0 && (ar.startRow < mc.r2 || ar.startCol !== mc.c1) ) {
                    // Движение снизу вверх
                    ar.startRow = mc.r1 - 1;
                    done = false;
                }
            }
            if ( !done ) {
                continue;
            }

            // Обработка движения через скрытые столбцы/строки
            while ( ar.startCol >= arn.c1 && ar.startCol <= arn.c2 && this.cols[ar.startCol].width < this.width_1px ) {
                ar.startCol += dc || (dr > 0 ? +1 : -1);
                done = false;
            }
            if ( !done ) {
                continue;
            }

            while ( ar.startRow >= arn.r1 && ar.startRow <= arn.r2 && this.rows[ar.startRow].height < this.height_1px ) {
                ar.startRow += dr || (dc > 0 ? +1 : -1);
                done = false;
            }
        } while ( !done );
    };

    WorksheetView.prototype._calcSelectionEndPointByXY = function ( x, y ) {
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;
        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        var res = new asc_Range( ar.startCol, ar.startRow, this._findColUnderCursor( x ).col, this._findRowUnderCursor( y ).row, true );
        if ( ar.type === c_oAscSelectionType.RangeCells ) {
            this._fixSelectionOfMergedCells( res );
        }
        return res;
        /*return {
         c2: ar.type === c_oAscSelectionType.RangeCol || ar.type === c_oAscSelectionType.RangeCells ? this._findColUnderCursor(x).col : ar.c2,
         r2: ar.type === c_oAscSelectionType.RangeRow || ar.type === c_oAscSelectionType.RangeCells ? this._findRowUnderCursor(y).row : ar.r2
         };*/
    };

    WorksheetView.prototype._calcSelectionEndPointByOffset = function ( dc, dr ) {
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;
        var startCol = ar.startCol, startRow = ar.startRow;
        var c1, r1, c2, r2, tmp;
        tmp = asc.getEndValueRange( dc, startCol, ar.c1, ar.c2 );
        c1 = tmp.x1;
        c2 = tmp.x2;
        tmp = asc.getEndValueRange( dr, startRow, ar.r1, ar.r2 );
        r1 = tmp.x1;
        r2 = tmp.x2;

        var p1 = this._calcCellPosition( c2, r2, dc, dr ), p2;
        var res = new asc_Range( c1, r1, c2 = p1.col, r2 = p1.row, true );
        dc = Math.sign( dc );
        dr = Math.sign( dr );
        if ( c_oAscSelectionType.RangeCells === ar.type ) {
            this._fixSelectionOfMergedCells( res );
            while ( ar.isEqual( res ) ) {
                p2 = this._calcCellPosition( c2, r2, dc, dr );
                res.assign( c1, r1, c2 = p2.col, r2 = p2.row, true );
                this._fixSelectionOfMergedCells( res );
                if ( p1.c2 === p2.c2 && p1.r2 === p2.r2 ) {
                    break;
                }
                p1 = p2;
            }
        }
        var bIsHidden = false;
        if ( 0 !== dc && this.cols[c2].width < this.width_1px ) {
            c2 = this._findVisibleCol( c2, dc );
            bIsHidden = true;
        }
        if ( 0 !== dr && this.rows[r2].height < this.height_1px ) {
            r2 = this._findVisibleRow( r2, dr );
            bIsHidden = true;
        }
        if ( bIsHidden ) {
            res.assign( c1, r1, c2, r2, true );
        }
        return res;
    };

    WorksheetView.prototype._calcActiveRangeOffsetIsCoord = function ( x, y ) {
        var vr = this.visibleRange;
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;
        if ( this.isFormulaEditMode ) {
            // Для формул нужно сделать ограничение по range (у нас хранится полный диапазон)
            if ( ar.c2 >= this.nColsCount || ar.r2 >= this.nRowsCount ) {
                ar = ar.clone( true );
                ar.c2 = (ar.c2 >= this.nColsCount) ? this.nColsCount - 1 : ar.c2;
                ar.r2 = (ar.r2 >= this.nRowsCount) ? this.nRowsCount - 1 : ar.r2;
            }
        }

        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        var d = {};

        if ( y <= this.cellsTop + this.height_2px /*+ offsetFrozen.offsetY*/ ) {
            d.deltaY = -1;
        }
        else if ( y >= this.drawingCtx.getHeight() - this.height_2px ) {
            d.deltaY = 1;
        }

        if ( x <= this.cellsLeft + this.width_2px /*+ offsetFrozen.offsetX*/ ) {
            d.deltaX = -1;
        }
        else if ( x >= this.drawingCtx.getWidth() - this.width_2px ) {
            d.deltaX = 1;
        }

        if ( ar.type === c_oAscSelectionType.RangeRow ) {
            d.deltaX = 0;
        }
        else if ( ar.type === c_oAscSelectionType.RangeCol ) {
            d.deltaY = 0;
        }
        else if ( ar.type === c_oAscSelectionType.RangeMax ) {
            d.deltaX = 0;
            d.deltaY = 0;
        }

        return d;

    };

    WorksheetView.prototype._calcActiveRangeOffset = function () {
        var vr = this.visibleRange;
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;
        if ( this.isFormulaEditMode ) {
            // Для формул нужно сделать ограничение по range (у нас хранится полный диапазон)
            if ( ar.c2 >= this.nColsCount || ar.r2 >= this.nRowsCount ) {
                ar = ar.clone( true );
                ar.c2 = (ar.c2 >= this.nColsCount) ? this.nColsCount - 1 : ar.c2;
                ar.r2 = (ar.r2 >= this.nRowsCount) ? this.nRowsCount - 1 : ar.r2;
            }
        }
        var arn = ar.clone( true );
        var isMC = this._isMergedCells( arn );
        var adjustRight = ar.c2 >= vr.c2 || ar.c1 >= vr.c2 && isMC;
        var adjustBottom = ar.r2 >= vr.r2 || ar.r1 >= vr.r2 && isMC;
        var incX = ar.c1 < vr.c1 && isMC ? arn.c1 - vr.c1 : ar.c2 < vr.c1 ? ar.c2 - vr.c1 : 0;
        var incY = ar.r1 < vr.r1 && isMC ? arn.r1 - vr.r1 : ar.r2 < vr.r1 ? ar.r2 - vr.r1 : 0;

        var offsetFrozen = this.getFrozenPaneOffset();

        if ( adjustRight ) {
            while ( this._isColDrawnPartially( isMC ? arn.c2 : ar.c2, vr.c1 + incX, offsetFrozen.offsetX ) ) {
                ++incX;
            }
        }
        if ( adjustBottom ) {
            while ( this._isRowDrawnPartially( isMC ? arn.r2 : ar.r2, vr.r1 + incY, offsetFrozen.offsetY ) ) {
                ++incY;
            }
        }
        return {
            deltaX: ar.type === c_oAscSelectionType.RangeCol || ar.type === c_oAscSelectionType.RangeCells ? incX : 0,
            deltaY: ar.type === c_oAscSelectionType.RangeRow || ar.type === c_oAscSelectionType.RangeCells ? incY : 0
        };
    };

    /**
     * @param {ActiveRange} [range]
     * @returns {{deltaX: number, deltaY: number}}
     */
    WorksheetView.prototype._calcActiveCellOffset = function ( range ) {
        var vr = this.visibleRange;
        var ar = range ? range : this.activeRange;
        var mc = this.model.getMergedByCell( ar.startRow, ar.startCol );
        var startCol = mc ? mc.c1 : ar.startCol;
        var startRow = mc ? mc.r1 : ar.startRow;
        var incX = startCol < vr.c1 ? startCol - vr.c1 : 0;
        var incY = startRow < vr.r1 ? startRow - vr.r1 : 0;

        var offsetFrozen = this.getFrozenPaneOffset();
        // adjustRight
        if ( startCol >= vr.c2 ) {
            while ( this._isColDrawnPartially( startCol, vr.c1 + incX, offsetFrozen.offsetX ) ) {
                ++incX;
            }
        }
        // adjustBottom
        if ( startRow >= vr.r2 ) {
            while ( this._isRowDrawnPartially( startRow, vr.r1 + incY, offsetFrozen.offsetY ) ) {
                ++incY;
            }
        }
        return {
            deltaX: ar.type === c_oAscSelectionType.RangeCol || ar.type === c_oAscSelectionType.RangeCells ? incX : 0,
            deltaY: ar.type === c_oAscSelectionType.RangeRow || ar.type === c_oAscSelectionType.RangeCells ? incY : 0
        };
    };

    WorksheetView.prototype._calcFillHandleOffset = function ( range ) {
        var vr = this.visibleRange;
        var ar = range ? range : this.activeFillHandle;
        var arn = ar.clone( true );
        var isMC = this._isMergedCells( arn );
        var adjustRight = ar.c2 >= vr.c2 || ar.c1 >= vr.c2 && isMC;
        var adjustBottom = ar.r2 >= vr.r2 || ar.r1 >= vr.r2 && isMC;
        var incX = ar.c1 < vr.c1 && isMC ? arn.c1 - vr.c1 : ar.c2 < vr.c1 ? ar.c2 - vr.c1 : 0;
        var incY = ar.r1 < vr.r1 && isMC ? arn.r1 - vr.r1 : ar.r2 < vr.r1 ? ar.r2 - vr.r1 : 0;

        var offsetFrozen = this.getFrozenPaneOffset();

        if ( adjustRight ) {
            try {
                while ( this._isColDrawnPartially( isMC ? arn.c2 : ar.c2, vr.c1 + incX, offsetFrozen.offsetX ) ) {
                    ++incX;
                }
            } catch( e ) {
                this.expandColsOnScroll( true );
                this.handlers.trigger( "reinitializeScrollX" );
            }
        }
        if ( adjustBottom ) {
            try {
                while ( this._isRowDrawnPartially( isMC ? arn.r2 : ar.r2, vr.r1 + incY, offsetFrozen.offsetY ) ) {
                    ++incY;
                }
            } catch( e ) {
                this.expandRowsOnScroll( true );
                this.handlers.trigger( "reinitializeScrollY" );
            }
        }
        return {
            deltaX: incX, deltaY: incY
        };
    };

    // Потеряем ли мы что-то при merge ячеек
    WorksheetView.prototype.getSelectionMergeInfo = function ( options ) {
        var arn = this.activeRange.clone( true );
        var notEmpty = false;
        var r, c;

        if ( this.cellCommentator.isMissComments( arn ) ) {
            return true;
        }

        switch ( options ) {
            case c_oAscMergeOptions.Merge:
            case c_oAscMergeOptions.MergeCenter:
                for ( r = arn.r1; r <= arn.r2; ++r ) {
                    for ( c = arn.c1; c <= arn.c2; ++c ) {
                        if ( false === this._isCellEmptyText( c, r ) ) {
                            if ( notEmpty ) {
                                return true;
                            }
                            notEmpty = true;
                        }
                    }
                }
                break;
            case c_oAscMergeOptions.MergeAcross:
                for ( r = arn.r1; r <= arn.r2; ++r ) {
                    notEmpty = false;
                    for ( c = arn.c1; c <= arn.c2; ++c ) {
                        if ( false === this._isCellEmptyText( c, r ) ) {
                            if ( notEmpty ) {
                                return true;
                            }
                            notEmpty = true;
                        }
                    }
                }
                break;
        }

        return false;
    };

    WorksheetView.prototype.getSelectionMathInfo = function () {
        var ar = this.activeRange;
        var range = this.model.getRange3( ar.r1, ar.c1, ar.r2, ar.c2 );
        var tmp;
        var oSelectionMathInfo = new asc_CSelectionMathInfo();
        var sum = 0;
        range._setPropertyNoEmpty( null, null, function ( c ) {
            if ( false === c.isEmptyTextString() ) {
                ++oSelectionMathInfo.count;
                if ( CellValueType.Number === c.getType() ) {
                    tmp = parseFloat( c.getValueWithoutFormat() );
                    if ( isNaN( tmp ) ) {
                        return;
                    }
                    if ( 0 === oSelectionMathInfo.countNumbers ) {
                        oSelectionMathInfo.min = oSelectionMathInfo.max = tmp;
                    }
                    else {
                        oSelectionMathInfo.min = Math.min( oSelectionMathInfo.min, tmp );
                        oSelectionMathInfo.max = Math.max( oSelectionMathInfo.max, tmp );
                    }
                    ++oSelectionMathInfo.countNumbers;
                    sum += tmp;
                }
            }
        } );
        // Показываем только данные для 2-х или более ячеек (http://bugzserver/show_bug.cgi?id=24115)
        if ( 1 < oSelectionMathInfo.countNumbers ) {
            // Мы должны отдавать в формате активной ячейки
            var numFormat = range.getNumFormat();
            if ( c_oAscNumFormatType.Time === numFormat.getType() ) {
                // Для времени нужно отдавать в формате [h]:mm:ss (http://bugzserver/show_bug.cgi?id=26271)
                numFormat = oNumFormatCache.get( '[h]:mm:ss' );
            }

            oSelectionMathInfo.sum = numFormat.formatToMathInfo( sum, CellValueType.Number, this.settings.mathMaxDigCount );
            oSelectionMathInfo.average = numFormat.formatToMathInfo( sum / oSelectionMathInfo.countNumbers, CellValueType.Number, this.settings.mathMaxDigCount );

            oSelectionMathInfo.min = numFormat.formatToMathInfo( oSelectionMathInfo.min, CellValueType.Number, this.settings.mathMaxDigCount );
            oSelectionMathInfo.max = numFormat.formatToMathInfo( oSelectionMathInfo.max, CellValueType.Number, this.settings.mathMaxDigCount );
        }
        return oSelectionMathInfo;
    };

    WorksheetView.prototype.getSelectionName = function ( bRangeText ) {
        if ( this.isSelectOnShape ) {
            return " ";
        }	// Пока отправим пустое имя(с пробелом, пустое не воспринимаем в меню..) ToDo

        var ar = this.activeRange;
        var mc = this.model.getMergedByCell( ar.startRow, ar.startCol );
        var c1 = mc ? mc.c1 : ar.startCol, r1 = mc ? mc.r1 : ar.startRow, ar_norm = ar.normalize(), mc_norm = mc ? mc.normalize() : null, c2 = mc_norm ? mc_norm.isEqual( ar_norm ) ? mc_norm.c1 : ar_norm.c2 : ar_norm.c2, r2 = mc_norm ? mc_norm.isEqual( ar_norm ) ? mc_norm.r1 : ar_norm.r2 : ar_norm.r2, selectionSize = !bRangeText ? "" : (function ( r ) {
            var rc = Math.abs( r.r2 - r.r1 ) + 1;
            var cc = Math.abs( r.c2 - r.c1 ) + 1;
            switch ( r.type ) {
                case c_oAscSelectionType.RangeCells:
                    return rc + "R x " + cc + "C";
                case c_oAscSelectionType.RangeCol:
                    return cc + "C";
                case c_oAscSelectionType.RangeRow:
                    return rc + "R";
                case c_oAscSelectionType.RangeMax:
                    return gc_nMaxRow + "R x " + gc_nMaxCol + "C";
            }
            return "";
        })( ar ), cellName = this._getColumnTitle( c1 ) + this._getRowTitle( r1 ), defName = null, dN = new Asc.Range( ar_norm.c1, ar_norm.r1, c2, r2, true ).normalize();

        /*if( c1==c2 && r1==r2 ){
         defName = this.model.getName() + "!" + dN.getAbsName();
         }
         else{
         defName = this.model.getName() + "!" + dN.getAbsName();
         }*/

        defName = parserHelp.getEscapeSheetName( this.model.getName() ) + "!" + dN.getAbsName();

        defName = defName ? this.model.workbook.findDefinesNames( defName, this.model.getId() ) : null;

        return selectionSize || defName || cellName;
    };

    WorksheetView.prototype.getSelectionRangeValue = function () {
        // ToDo проблема с выбором целого столбца/строки
        var ar = this.activeRange.clone( true );
//			ar.r1Abs = ar.c1Abs = ar.r2Abs = ar.c2Abs = true;
        var sName = ar.getAbsName();
        return (c_oAscSelectionDialogType.FormatTable === this.selectionDialogType) ? sName : parserHelp.get3DRef( this.model.getName(), sName );
    };

    WorksheetView.prototype.getSelectionInfo = function ( bExt ) {
        return this.objectRender.selectedGraphicObjectsExists() ? this._getSelectionInfoObject( bExt ) : this._getSelectionInfoCell( bExt );
    };

    WorksheetView.prototype._getSelectionInfoCell = function ( bExt ) {
        var c_opt = this.settings.cells;
        var activeCell = this.activeRange;
        var mc = this.model.getMergedByCell( activeCell.startRow, activeCell.startCol );
        var c1 = mc ? mc.c1 : activeCell.startCol;
        var r1 = mc ? mc.r1 : activeCell.startRow;
        var c = this._getVisibleCell( c1, r1 );

        if ( c === undefined ) {
            asc_debug( "log", "Unknown cell's info: col = " + c1 + ", row = " + r1 );
            return {};
        }

        var fc = c.getFontcolor();
        var bg = c.getFill();
        var fa = c.getFontAlign().toLowerCase();
        var cellType = c.getType();
        var isNumberFormat = (!cellType || CellValueType.Number === cellType);

        var cell_info = new asc_CCellInfo();
        cell_info.name = this._getColumnTitle( c1 ) + this._getRowTitle( r1 );
        cell_info.formula = c.getFormula();

        cell_info.text = c.getValueForEdit();

        cell_info.halign = c.getAlignHorizontalByValue().toLowerCase();
        cell_info.valign = c.getAlignVertical().toLowerCase();

        var tablePartsOptions = this.autoFilters.searchRangeInTableParts( activeCell );
        var curTablePart = tablePartsOptions >= 0 ? this.model.TableParts[tablePartsOptions] : null;

        cell_info.autoFilterInfo = new asc_CAutoFilterInfo();
        cell_info.autoFilterInfo.tableStyleName = curTablePart ? curTablePart.TableStyleInfo.Name : null;
        cell_info.autoFilterInfo.tableName = curTablePart ? curTablePart.DisplayName : null;
        if ( -2 === tablePartsOptions ) {
            cell_info.autoFilterInfo.isAutoFilter = null;
            cell_info.autoFilterInfo.isApplyAutoFilter = false;
        }
        else {
            var checkApplyFilterOrSort = this.autoFilters.checkApplyFilterOrSort( tablePartsOptions );
            cell_info.autoFilterInfo.isAutoFilter = checkApplyFilterOrSort.isAutoFilter;
            cell_info.autoFilterInfo.isApplyAutoFilter = checkApplyFilterOrSort.isFilterColumns;
        }

        cell_info.styleName = c.getStyleName();
        cell_info.angle = c.getAngle();

        cell_info.flags = new asc_CCellFlag();
        cell_info.flags.shrinkToFit = c.getShrinkToFit();
        cell_info.flags.wrapText = c.getWrap();

        cell_info.flags.selectionType = this.activeRange.type;

        cell_info.flags.lockText = ("" !== cell_info.text && (isNumberFormat || "" !== cell_info.formula));

        cell_info.font = new asc_CFont();
        cell_info.font.name = c.getFontname();
        cell_info.font.size = c.getFontsize();
        cell_info.font.bold = c.getBold();
        cell_info.font.italic = c.getItalic();
        cell_info.font.underline = (Asc.EUnderline.underlineNone !== c.getUnderline()); // ToDo убрать, когда будет реализовано двойное подчеркивание
        cell_info.font.strikeout = c.getStrikeout();
        cell_info.font.subscript = fa === "subscript";
        cell_info.font.superscript = fa === "superscript";
        cell_info.font.color = (fc ? asc_obj2Color( fc ) : new asc_CColor( c_opt.defaultState.color ));

        cell_info.fill = new asc_CFill( (null != bg) ? asc_obj2Color( bg ) : bg );

        cell_info.numFormatType = c.getNumFormatType();

        // Получаем гиперссылку
        var ar = this.activeRange.clone();
        var range = this.model.getRange3( ar.r1, ar.c1, ar.r2, ar.c2 );
        var hyperlink = range.getHyperlink();
        var oHyperlink;
        if ( null !== hyperlink ) {
            // Гиперлинк
            oHyperlink = new asc_CHyperlink( hyperlink );
            oHyperlink.asc_setText( cell_info.text );
            cell_info.hyperlink = oHyperlink;
        }
        else {
            cell_info.hyperlink = null;
        }

        cell_info.comments = this.cellCommentator.getComments( ar.c1, ar.r1 );
        cell_info.flags.merge = null !== range.hasMerged();

        if ( bExt ) {
            cell_info.innertext = c.getValue();
            cell_info.numFormat = c.getNumFormatStr();
        }

        var sheetId = this.model.getId();
        // Пересчет для входящих ячеек в добавленные строки/столбцы
        var isIntersection = this._recalcRangeByInsertRowsAndColumns( sheetId, ar );
        if ( false === isIntersection ) {
            var lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Range, /*subType*/null, sheetId, new asc.asc_CCollaborativeRange( ar.c1, ar.r1, ar.c2, ar.r2 ) );

            if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/false ) ) {
                // Уже ячейку кто-то редактирует
                cell_info.isLocked = true;
            }
        }

        return cell_info;
    };

    WorksheetView.prototype._getSelectionInfoObject = function () {
        var objectInfo = new asc_CCellInfo();

        objectInfo.flags = new asc_CCellFlag();
        var graphicObjects = this.objectRender.getSelectedGraphicObjects();
        if ( graphicObjects.length ) {
            objectInfo.flags.selectionType = this.objectRender.getGraphicSelectionType( graphicObjects[0].Id );
        }

        var textPr = this.objectRender.controller.getParagraphTextPr();
        var theme = this.objectRender.controller.getTheme();
        if ( textPr && theme && theme.themeElements && theme.themeElements.fontScheme ) {
            if ( textPr.FontFamily ) {
                textPr.FontFamily.Name = theme.themeElements.fontScheme.checkFont( textPr.FontFamily.Name );
            }
            if ( textPr.RFonts ) {
                if ( textPr.RFonts.Ascii ) {
                    textPr.RFonts.Ascii.Name = theme.themeElements.fontScheme.checkFont( textPr.RFonts.Ascii.Name );
                }
                if ( textPr.RFonts.EastAsia ) {
                    textPr.RFonts.EastAsia.Name = theme.themeElements.fontScheme.checkFont( textPr.RFonts.EastAsia.Name );
                }
                if ( textPr.RFonts.HAnsi ) {
                    textPr.RFonts.HAnsi.Name = theme.themeElements.fontScheme.checkFont( textPr.RFonts.HAnsi.Name );
                }
                if ( textPr.RFonts.CS ) {
                    textPr.RFonts.CS.Name = theme.themeElements.fontScheme.checkFont( textPr.RFonts.CS.Name );
                }
            }
        }

        var paraPr = this.objectRender.controller.getParagraphParaPr();
        if ( textPr && paraPr ) {
            objectInfo.text = this.objectRender.controller.Get_SelectedText( true );

            var horAlign = "center";
            switch ( paraPr.Jc ) {
                case align_Left    :
                    horAlign = "left";
                    break;
                case align_Right  :
                    horAlign = "right";
                    break;
                case align_Center  :
                    horAlign = "center";
                    break;
                case align_Justify  :
                    horAlign = "justify";
                    break;
            }
            var vertAlign = "center";
            var shape_props = this.objectRender.controller.getDrawingProps().shapeProps;
            var angle = null;
            if ( shape_props ) {
                switch ( shape_props.verticalTextAlign ) {
                    case VERTICAL_ANCHOR_TYPE_BOTTOM:
                        vertAlign = "bottom";
                        break;
                    case VERTICAL_ANCHOR_TYPE_CENTER:
                        vertAlign = "center";
                        break;

                    case VERTICAL_ANCHOR_TYPE_TOP:
                    case VERTICAL_ANCHOR_TYPE_DISTRIBUTED:
                    case VERTICAL_ANCHOR_TYPE_JUSTIFIED:
                        vertAlign = "top";
                        break;
                }
                switch ( shape_props.vert ) {
                    case nVertTTvert:
                        angle = 90;
                        break;
                    case nVertTTvert270:
                        angle = 270;
                        break;
                    default:
                        angle = 0;
                        break;
                }

            }

            objectInfo.halign = horAlign;
            objectInfo.valign = vertAlign;
            objectInfo.angle = angle;

            objectInfo.font = new asc_CFont();
            objectInfo.font.name = textPr.FontFamily ? textPr.FontFamily.Name : "";
            objectInfo.font.size = textPr.FontSize;
            objectInfo.font.bold = textPr.Bold;
            objectInfo.font.italic = textPr.Italic;
            objectInfo.font.underline = textPr.Underline;
            objectInfo.font.strikeout = textPr.Strikeout;
            objectInfo.font.subscript = textPr.VertAlign == vertalign_SubScript;
            objectInfo.font.superscript = textPr.VertAlign == vertalign_SuperScript;
            if ( textPr.Color ) {
                objectInfo.font.color = CreateAscColorCustom( textPr.Color.r, textPr.Color.g, textPr.Color.b );
            }

            var shapeHyperlink = this.objectRender.controller.getHyperlinkInfo();
            if ( shapeHyperlink && (shapeHyperlink instanceof ParaHyperlink) ) {

                var hyperlink = new Hyperlink();
                hyperlink.Tooltip = shapeHyperlink.ToolTip;

                var spl = shapeHyperlink.Value.split( "!" );
                if ( spl.length === 2 ) {
                    hyperlink.setLocation( shapeHyperlink.Value );
                }
                else {
                    hyperlink.Hyperlink = shapeHyperlink.Value;
                }

                objectInfo.hyperlink = new asc_CHyperlink( hyperlink );
                objectInfo.hyperlink.asc_setText( shapeHyperlink.Get_SelectedText( true, true ) );
            }
        }
        else {
            // Может быть не задано текста, поэтому выставим по умолчанию
            objectInfo.font = new asc_CFont();
            objectInfo.font.name = this.model.getDefaultFontName();
            objectInfo.font.size = this.model.getDefaultFontSize();
        }

        // Заливка не нужна как таковая
        objectInfo.fill = new asc_CFill( null );

        // ToDo locks

        return objectInfo;
    };

    // Получаем координаты активной ячейки
    WorksheetView.prototype.getActiveCellCoord = function () {
        var offsetX = 0, offsetY = 0;
        var vrCol = this.visibleRange.c1, vrRow = this.visibleRange.r1;
        if ( this.topLeftFrozenCell ) {
            var offsetFrozen = this.getFrozenPaneOffset();
            var cFrozen = this.topLeftFrozenCell.getCol0();
            var rFrozen = this.topLeftFrozenCell.getRow0();
            if ( this.activeRange.startCol >= cFrozen ) {
                offsetX = offsetFrozen.offsetX;
            }
            else {
                vrCol = 0;
            }

            if ( this.activeRange.startRow >= rFrozen ) {
                offsetY = offsetFrozen.offsetY;
            }
            else {
                vrRow = 0;
            }
        }

        var xL = this.getCellLeft( this.activeRange.startCol, /*pt*/1 );
        var yL = this.getCellTop( this.activeRange.startRow, /*pt*/1 );
        // Пересчитываем X и Y относительно видимой области
        xL -= (this.cols[vrCol].left - this.cellsLeft);
        yL -= (this.rows[vrRow].top - this.cellsTop);
        // Пересчитываем X и Y относительно закрепленной области
        xL += offsetX;
        yL += offsetY;

        // Пересчитываем в px
        xL *= asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIX() );
        yL *= asc_getcvt( 1/*pt*/, 0/*px*/, this._getPPIY() );

        var width = this.getColumnWidth( this.activeRange.startCol, /*px*/0 );
        var height = this.getRowHeight( this.activeRange.startRow, /*px*/0 );

        if ( AscBrowser.isRetina ) {
            xL >>= 1;
            yL >>= 1;
            width >>= 1;
            height >>= 1;
        }

        return new asc_CRect( xL, yL, width, height );
    };

    WorksheetView.prototype._checkSelectionShape = function () {
        var isSelectOnShape = this.isSelectOnShape;
        if ( this.isSelectOnShape ) {
            this.isSelectOnShape = false;
            this.objectRender.unselectDrawingObjects();
        }
        return isSelectOnShape;
    };

    WorksheetView.prototype._updateSelectionNameAndInfo = function () {
        this.handlers.trigger( "selectionNameChanged", this.getSelectionName( /*bRangeText*/false ) );
        this.handlers.trigger( "selectionChanged", this.getSelectionInfo() );
        this.handlers.trigger( "selectionMathInfoChanged", this.getSelectionMathInfo() );
    };

    WorksheetView.prototype.getSelectionShape = function () {
        return this.isSelectOnShape;
    };
    WorksheetView.prototype.setSelectionShape = function ( isSelectOnShape ) {
        this.isSelectOnShape = isSelectOnShape;
        // отправляем евент для получения свойств картинки, шейпа или группы
        this.model.workbook.handlers.trigger( "asc_onHideComment" );
        this._updateSelectionNameAndInfo();
    };
    WorksheetView.prototype.getActiveRangeObj = function () {
        return this.activeRange.clone( true );
    };
    WorksheetView.prototype.setSelection = function ( range, validRange ) {
        // Проверка на валидность range.
        if ( validRange && (range.c2 >= this.nColsCount || range.r2 >= this.nRowsCount) ) {
            if ( range.c2 >= this.nColsCount ) {
                this.expandColsOnScroll( false, true, range.c2 + 1 );
            }
            if ( range.r2 >= this.nRowsCount ) {
                this.expandRowsOnScroll( false, true, range.r2 + 1 );
            }
        }

        this.cleanSelection();
        // Проверка на всякий случай
        if ( !(range instanceof asc_Range) ) {
            range = new asc_Range( range.c1, range.r1, range.c2, range.r2 );
        }
        if ( gc_nMaxCol0 === range.c2 || gc_nMaxRow0 === range.r2 ) {
            range = range.clone();
            if ( gc_nMaxCol0 === range.c2 ) {
                range.c2 = this.cols.length - 1;
            }
            if ( gc_nMaxRow0 === range.r2 ) {
                range.r2 = this.rows.length - 1;
            }
        }

        this.activeRange = new asc_ActiveRange( range );
        this.activeRange.type = c_oAscSelectionType.RangeCells;
        this.activeRange.startCol = range.c1;
        this.activeRange.startRow = range.r1;

        // Нормализуем range
        this.activeRange.normalize();
        this._drawSelection();

        this._updateSelectionNameAndInfo();
        return this._calcActiveCellOffset();
    };

    WorksheetView.prototype.setSelectionUndoRedo = function ( range, validRange ) {
        var ar = (range instanceof asc_ActiveRange) ? range.clone() : new asc_ActiveRange( range );

        // Проверка на валидность range.
        if ( validRange && (ar.c2 >= this.nColsCount || ar.r2 >= this.nRowsCount) ) {
            if ( ar.c2 >= this.nColsCount ) {
                this.expandColsOnScroll( false, true, ar.c2 + 1 );
            }
            if ( ar.r2 >= this.nRowsCount ) {
                this.expandRowsOnScroll( false, true, ar.r2 + 1 );
            }
        }
        var oRes = null;
        var type = ar.type;
        if ( type == c_oAscSelectionType.RangeCells || type == c_oAscSelectionType.RangeCol || type == c_oAscSelectionType.RangeRow || type == c_oAscSelectionType.RangeMax ) {
            this.cleanSelection();
            this.activeRange = ar;
            this._drawSelection();

            this._updateSelectionNameAndInfo();
            oRes = this._calcActiveCellOffset();
        }
        return oRes;
    };

    WorksheetView.prototype.changeSelectionStartPoint = function ( x, y, isCoord, isSelectMode ) {
        var ar = ((this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange).clone();
        var ret = {};
        var isChangeSelectionShape = false;

        this.cleanSelection();

        var commentList = this.cellCommentator.getCommentsXY( x, y );
        if ( !commentList.length ) {
            this.model.workbook.handlers.trigger( "asc_onHideComment" );
            this.cellCommentator.resetLastSelectedId();
        }

        if ( isCoord ) {
            // move active range to coordinates x,y
            this._moveActiveCellToXY( x, y );
            isChangeSelectionShape = this._checkSelectionShape();
        }
        else {
            // move active range to offset x,y
            this._moveActiveCellToOffset( x, y );
            var x1 = this.getCellLeftRelative( this.activeRange.c1, /*pt*/0 ), y1 = this.getCellTopRelative( this.activeRange.r1, /*pt*/0 );
            if ( isCoord ) {
                ret = this._calcActiveRangeOffsetIsCoord( x1, y1 );
            }
            else {
                ret = this._calcActiveRangeOffset();
            }

        }

        if ( this.isSelectionDialogMode ) {
            if ( !this.activeRange.isEqual( ar ) ) {
                // Смена диапазона
                this.handlers.trigger( "selectionRangeChanged", this.getSelectionRangeValue() );
            }
        }
        else if ( !this.isCellEditMode ) {
            if ( isChangeSelectionShape || (!isCoord && !this.activeRange.isEqual( ar )) || (isCoord && (this.activeRange.startCol !== ar.startCol || this.activeRange.startRow !== ar.startRow)) ) {
                this.handlers.trigger( "selectionNameChanged", this.getSelectionName( /*bRangeText*/false ) );
                if ( !isSelectMode ) {
                    this.handlers.trigger( "selectionChanged", this.getSelectionInfo() );
                    this.handlers.trigger( "selectionMathInfoChanged", this.getSelectionMathInfo() );
                }
            }
        }

        if ( !isChangeSelectionShape ) {
            this._drawSelection();
        }

        //ToDo this.drawDepCells();

        return ret;
    };

    // Смена селекта по нажатию правой кнопки мыши
    WorksheetView.prototype.changeSelectionStartPointRightClick = function ( x, y ) {
        var ar = this.activeRange;
        var isChangeSelectionShape = this._checkSelectionShape();
        this.model.workbook.handlers.trigger( "asc_onHideComment" );

        // Получаем координаты левого верхнего угла выделения
        var xL = this.getCellLeft( ar.c1, /*pt*/1 );
        var yL = this.getCellTop( ar.r1, /*pt*/1 );
        // Получаем координаты правого нижнего угла выделения
        var xR = this.getCellLeft( ar.c2, /*pt*/1 ) + this.cols[ar.c2].width;
        var yR = this.getCellTop( ar.r2, /*pt*/1 ) + this.rows[ar.r2].height;

        // Пересчитываем координаты
        var _x = x * asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        var _y = y * asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        var isInSelection = false;
        var offsetX = this.cols[this.visibleRange.c1].left - this.cellsLeft, offsetY = this.rows[this.visibleRange.r1].top - this.cellsTop;
        var offsetFrozen = this.getFrozenPaneOffset();
        offsetX -= offsetFrozen.offsetX;
        offsetY -= offsetFrozen.offsetY;

        // Проверяем попали ли мы в выделение
        if ( (_x < this.cellsLeft || _y < this.cellsTop) && c_oAscSelectionType.RangeMax === ar.type ) {
            // Выделено все
            isInSelection = true;
        }
        else if ( _x > this.cellsLeft && _y > this.cellsTop ) {
            // Пересчитываем X и Y относительно видимой области
            _x += offsetX;
            _y += offsetY;

            if ( xL <= _x && _x <= xR && yL <= _y && _y <= yR ) {
                // Попали в выделение ячеек
                isInSelection = true;
            }
        }
        else if ( _x <= this.cellsLeft && _y >= this.cellsTop && c_oAscSelectionType.RangeRow === ar.type ) {
            // Выделены строки
            // Пересчитываем Y относительно видимой области
            _y += offsetY;

            if ( yL <= _y && _y <= yR ) {
                // Попали в выделение ячеек
                isInSelection = true;
            }
        }
        else if ( _y <= this.cellsTop && _x >= this.cellsLeft && c_oAscSelectionType.RangeCol === ar.type ) {
            // Выделены столбцы
            // Пересчитываем X относительно видимой области
            _x += offsetX;
            if ( xL <= _x && _x <= xR ) {
                // Попали в выделение ячеек
                isInSelection = true;
            }
        }

        if ( !isInSelection ) {
            // Не попали в выделение (меняем первую точку)
            this.cleanSelection();
            this._moveActiveCellToXY( x, y );
            this._drawSelection();

            this._updateSelectionNameAndInfo();
            return false;
        }
        else if ( isChangeSelectionShape ) {
            // Попали в выделение, но были в объекте
            this.cleanSelection();
            this._drawSelection();

            this._updateSelectionNameAndInfo();
        }

        return true;
    };

    /**
     *
     * @param x - координата или прибавка к column
     * @param y - координата или прибавка к row
     * @param isCoord - выделение с помощью мышки (true) или с клавиатуры (false)
     * @param isSelectMode - при выделении с помощью мышки, не нужно отправлять эвенты о смене выделения и информации
     * @returns {*}
     */
    WorksheetView.prototype.changeSelectionEndPoint = function ( x, y, isCoord, isSelectMode ) {
        var isChangeSelectionShape = false;
        if ( isCoord ) {
            isChangeSelectionShape = this._checkSelectionShape();
        }
        var ar = (this.isFormulaEditMode) ? this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition] : this.activeRange;

        var newRange = isCoord ? this._calcSelectionEndPointByXY( x, y ) : this._calcSelectionEndPointByOffset( x, y );
        var isEqual = newRange.isEqual( ar );
        if ( isEqual && !isCoord ) {
            // При движении стрелками можем попасть на замерженную ячейку
        }
        if ( !isEqual || isChangeSelectionShape ) {
            this.cleanSelection();
            ar.assign2( newRange );
            this._drawSelection();

            //ToDo this.drawDepCells();

            if ( !this.isCellEditMode ) {
                if ( !this.isSelectionDialogMode ) {
                    this.handlers.trigger( "selectionNameChanged", this.getSelectionName( /*bRangeText*/true ) );
                    if ( !isSelectMode ) {
                        this.handlers.trigger( "selectionChanged", this.getSelectionInfo( false ) );
                        this.handlers.trigger( "selectionMathInfoChanged", this.getSelectionMathInfo() );
                    }
                }
                else {
                    // Смена диапазона
                    this.handlers.trigger( "selectionRangeChanged", this.getSelectionRangeValue() );
                }
            }
        }

        this.model.workbook.handlers.trigger( "asc_onHideComment" );

        return isCoord ? this._calcActiveRangeOffsetIsCoord( x, y ) : this._calcActiveRangeOffset( this.getCellLeftRelative( x < 0 ? this.activeRange.c1 : this.activeRange.c2, /*pt*/0 ), this.getCellTopRelative( y < 0 ? this.activeRange.r1 : this.activeRange.r2, /*pt*/0 ) );
    };

    // Окончание выделения
    WorksheetView.prototype.changeSelectionDone = function () {
        if ( this.stateFormatPainter ) {
            this.applyFormatPainter();
        }
    };

    // Обработка движения в выделенной области
    WorksheetView.prototype.changeSelectionActivePoint = function ( dc, dr ) {
        var ret;
        var ar = this.activeRange;
        var mc = this.model.getMergedByCell( ar.r1, ar.c1 );

        // Если в выделенной области только одна ячейка, то просто сдвигаемся
        if ( ar.c1 === ar.c2 && ar.r1 === ar.r2 || mc && ar.isEqual( mc ) ) {
            return this.changeSelectionStartPoint( dc, dr, /*isCoord*/false, /*isSelectMode*/false );
        }

        // Очищаем выделение
        this.cleanSelection();
        // Двигаемся по выделенной области
        this._moveActivePointInSelection( dc, dr );
        // Перерисовываем
        this._drawSelection();

        // Смотрим, ушли ли мы за границу видимой области
        ret = this._calcActiveCellOffset();

        // Эвент обновления
        this.handlers.trigger( "selectionNameChanged", this.getSelectionName( /*bRangeText*/false ) );
        this.handlers.trigger( "selectionChanged", this.getSelectionInfo() );

        return ret;
    };


    // ----- Changing cells -----

    WorksheetView.prototype.applyFormatPainter = function () {
        var t = this;
        var from = t.copyActiveRange.getAllRange(), to = t.activeRange.getAllRange();
        var oTmpRange = this._getRange( 0, 0, 0, 0 );
        var onApplyFormatPainterCallback = function ( isSuccess ) {
            // Очищаем выделение
            t.cleanSelection();

            if ( true === isSuccess ) {
                oTmpRange.promoteFromTo( from, to );
            }

            t.expandColsOnScroll( false, true, to.c2 + 1 );
            t.expandRowsOnScroll( false, true, to.r2 + 1 );

            // Сбрасываем параметры
            t._updateCellsRange( to, /*canChangeColWidth*/c_oAscCanChangeColWidth.none, /*lockDraw*/true );
            if ( c_oAscFormatPainterState.kMultiple !== t.stateFormatPainter ) {
                t.formatPainter();
            }
            // Перерисовываем
            t._recalculateAfterUpdate( [to] );
        };

        var result = oTmpRange.preparePromoteFromTo( from, to );
        if ( !result ) {
            // ToDo вывести ошибку
            onApplyFormatPainterCallback( false );
            return;
        }

        this._isLockedCells( to, null, onApplyFormatPainterCallback );
    };
    WorksheetView.prototype.formatPainter = function ( stateFormatPainter ) {
        // Если передали состояние, то выставляем его. Если нет - то меняем на противоположное.
        this.stateFormatPainter = (null != stateFormatPainter) ? stateFormatPainter : ((c_oAscFormatPainterState.kOff !== this.stateFormatPainter) ? c_oAscFormatPainterState.kOff : c_oAscFormatPainterState.kOn);

        if ( this.stateFormatPainter ) {
            this.copyActiveRange = this.activeRange.clone( true );
            this._drawFormatPainterRange();
        }
        else {
            this.cleanSelection();
            this.copyActiveRange = null;
            this._drawSelection();
            this.handlers.trigger( "onStopFormatPainter" );
        }
    };

    /* Функция для работы автозаполнения (selection). (x, y) - координаты точки мыши на области */
    WorksheetView.prototype.changeSelectionFillHandle = function ( x, y ) {
        // Возвращаемый результат
        var ret = null;
        // Если мы только первый раз попали сюда, то копируем выделенную область
        if ( null === this.activeFillHandle ) {
            this.activeFillHandle = this.activeRange.clone( true );
            // Для первого раза нормализуем (т.е. первая точка - это левый верхний угол)
            this.activeFillHandle.normalize();
            return ret;
        }

        // Пересчитываем координаты
        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        // Очищаем выделение, будем рисовать заново
        this.cleanSelection();
        // Копируем выделенную область
        var ar = this.activeRange.clone( true );
        // Получаем координаты левого верхнего угла выделения
        var xL = this.getCellLeft( ar.c1, /*pt*/1 );
        var yL = this.getCellTop( ar.r1, /*pt*/1 );
        // Получаем координаты правого нижнего угла выделения
        var xR = this.getCellLeft( ar.c2, /*pt*/1 ) + this.cols[ar.c2].width;
        var yR = this.getCellTop( ar.r2, /*pt*/1 ) + this.rows[ar.r2].height;

        // range для пересчета видимой области
        var activeFillHandleCopy;

        // Колонка по X и строка по Y
        var colByX = this._findColUnderCursor( x, /*canReturnNull*/false, /*dX*/true ).col;
        var rowByY = this._findRowUnderCursor( y, /*canReturnNull*/false, /*dX*/true ).row;
        // Колонка по X и строка по Y (без половинчатого счета). Для сдвига видимой области
        var colByXNoDX = this._findColUnderCursor( x, /*canReturnNull*/false, /*dX*/false ).col;
        var rowByYNoDY = this._findRowUnderCursor( y, /*canReturnNull*/false, /*dX*/false ).row;
        // Сдвиг в столбцах и строках от крайней точки
        var dCol;
        var dRow;

        // Пересчитываем X и Y относительно видимой области
        x += (this.cols[this.visibleRange.c1].left - this.cellsLeft);
        y += (this.rows[this.visibleRange.r1].top - this.cellsTop);

        // Вычисляем расстояние от (x, y) до (xL, yL)
        var dXL = x - xL;
        var dYL = y - yL;
        // Вычисляем расстояние от (x, y) до (xR, yR)
        var dXR = x - xR;
        var dYR = y - yR;
        var dXRMod;
        var dYRMod;

        // Определяем область попадания и точку
        /*
         (1)					(2)					(3)

         ------------|-----------------------|------------
         |						|
         (4)		|			(5)			|		(6)
         |						|
         ------------|-----------------------|------------

         (7)					(8)					(9)
         */

        // Область точки (x, y)
        var _tmpArea = 0;
        if ( dXR <= 0 ) {
            // Области (1), (2), (4), (5), (7), (8)
            if ( dXL <= 0 ) {
                // Области (1), (4), (7)
                if ( dYR <= 0 ) {
                    // Области (1), (4)
                    if ( dYL <= 0 ) {
                        // Область (1)
                        _tmpArea = 1;
                    }
                    else {
                        // Область (4)
                        _tmpArea = 4;
                    }
                }
                else {
                    // Область (7)
                    _tmpArea = 7;
                }
            }
            else {
                // Области (2), (5), (8)
                if ( dYR <= 0 ) {
                    // Области (2), (5)
                    if ( dYL <= 0 ) {
                        // Область (2)
                        _tmpArea = 2;
                    }
                    else {
                        // Область (5)
                        _tmpArea = 5;
                    }
                }
                else {
                    // Область (3)
                    _tmpArea = 8;
                }
            }
        }
        else {
            // Области (3), (6), (9)
            if ( dYR <= 0 ) {
                // Области (3), (6)
                if ( dYL <= 0 ) {
                    // Область (3)
                    _tmpArea = 3;
                }
                else {
                    // Область (6)
                    _tmpArea = 6;
                }
            }
            else {
                // Область (9)
                _tmpArea = 9;
            }
        }

        // Проверяем, в каком направлении движение
        switch ( _tmpArea ) {
            case 2:
            case 8:
                // Двигаемся по вертикали.
                this.fillHandleDirection = 1;
                break;
            case 4:
            case 6:
                // Двигаемся по горизонтали.
                this.fillHandleDirection = 0;
                break;
            case 1:
                // Сравниваем расстояния от точки до левого верхнего угла выделения
                dXRMod = Math.abs( x - xL );
                dYRMod = Math.abs( y - yL );
                // Сдвиги по столбцам и строкам
                dCol = Math.abs( colByX - ar.c1 );
                dRow = Math.abs( rowByY - ar.r1 );
                // Определим направление позднее
                this.fillHandleDirection = -1;
                break;
            case 3:
                // Сравниваем расстояния от точки до правого верхнего угла выделения
                dXRMod = Math.abs( x - xR );
                dYRMod = Math.abs( y - yL );
                // Сдвиги по столбцам и строкам
                dCol = Math.abs( colByX - ar.c2 );
                dRow = Math.abs( rowByY - ar.r1 );
                // Определим направление позднее
                this.fillHandleDirection = -1;
                break;
            case 7:
                // Сравниваем расстояния от точки до левого нижнего угла выделения
                dXRMod = Math.abs( x - xL );
                dYRMod = Math.abs( y - yR );
                // Сдвиги по столбцам и строкам
                dCol = Math.abs( colByX - ar.c1 );
                dRow = Math.abs( rowByY - ar.r2 );
                // Определим направление позднее
                this.fillHandleDirection = -1;
                break;
            case 5:
            case 9:
                // Сравниваем расстояния от точки до правого нижнего угла выделения
                dXRMod = Math.abs( dXR );
                dYRMod = Math.abs( dYR );
                // Сдвиги по столбцам и строкам
                dCol = Math.abs( colByX - ar.c2 );
                dRow = Math.abs( rowByY - ar.r2 );
                // Определим направление позднее
                this.fillHandleDirection = -1;
                break;
        }

        //console.log(_tmpArea);

        // Возможно еще не определили направление
        if ( -1 === this.fillHandleDirection ) {
            // Проверим сдвиги по столбцам и строкам, если не поможет, то рассчитываем по расстоянию
            if ( 0 === dCol && 0 !== dRow ) {
                // Двигаемся по вертикали.
                this.fillHandleDirection = 1;
            }
            else if ( 0 !== dCol && 0 === dRow ) {
                // Двигаемся по горизонтали.
                this.fillHandleDirection = 0;
            }
            else if ( dXRMod >= dYRMod ) {
                // Двигаемся по горизонтали.
                this.fillHandleDirection = 0;
            }
            else {
                // Двигаемся по вертикали.
                this.fillHandleDirection = 1;
            }
        }

        // Проверяем, в каком направлении движение
        if ( 0 === this.fillHandleDirection ) {
            // Определяем область попадания и точку
            /*
             |						|
             |						|
             (1)		|			(2)			|		(3)
             |						|
             |						|
             */
            if ( dXR <= 0 ) {
                // Область (1) или (2)
                if ( dXL <= 0 ) {
                    // Область (1)
                    this.fillHandleArea = 1;
                }
                else {
                    // Область (2)
                    this.fillHandleArea = 2;
                }
            }
            else {
                // Область (3)
                this.fillHandleArea = 3;
            }

            // Находим колонку для точки
            this.activeFillHandle.c2 = colByX;

            switch ( this.fillHandleArea ) {
                case 1:
                    // Первая точка (xR, yR), вторая точка (x, yL)
                    this.activeFillHandle.c1 = ar.c2;
                    this.activeFillHandle.r1 = ar.r2;

                    this.activeFillHandle.r2 = ar.r1;

                    // Когда идем назад, должна быть колонка на 1 больше
                    this.activeFillHandle.c2 += 1;
                    // Случай, если мы еще не вышли из внутренней области
                    if ( this.activeFillHandle.c2 == ar.c1 ) {
                        this.fillHandleArea = 2;
                    }
                    break;
                case 2:
                    // Первая точка (xR, yR), вторая точка (x, yL)
                    this.activeFillHandle.c1 = ar.c2;
                    this.activeFillHandle.r1 = ar.r2;

                    this.activeFillHandle.r2 = ar.r1;

                    // Когда идем назад, должна быть колонка на 1 больше
                    this.activeFillHandle.c2 += 1;

                    if ( this.activeFillHandle.c2 > this.activeFillHandle.c1 ) {
                        // Ситуация половинки последнего столбца
                        this.activeFillHandle.c1 = ar.c1;
                        this.activeFillHandle.r1 = ar.r1;

                        this.activeFillHandle.c2 = ar.c1;
                        this.activeFillHandle.r2 = ar.r1;
                    }
                    break;
                case 3:
                    // Первая точка (xL, yL), вторая точка (x, yR)
                    this.activeFillHandle.c1 = ar.c1;
                    this.activeFillHandle.r1 = ar.r1;

                    this.activeFillHandle.r2 = ar.r2;
                    break;
            }

            // Копируем в range для пересчета видимой области
            activeFillHandleCopy = this.activeFillHandle.clone();
            activeFillHandleCopy.c2 = colByXNoDX;
        }
        else {
            // Определяем область попадания и точку
            /*
             (1)
             ____________________________


             (2)

             ____________________________

             (3)
             */
            if ( dYR <= 0 ) {
                // Область (1) или (2)
                if ( dYL <= 0 ) {
                    // Область (1)
                    this.fillHandleArea = 1;
                }
                else {
                    // Область (2)
                    this.fillHandleArea = 2;
                }
            }
            else {
                // Область (3)
                this.fillHandleArea = 3;
            }

            // Находим строку для точки
            this.activeFillHandle.r2 = rowByY;

            switch ( this.fillHandleArea ) {
                case 1:
                    // Первая точка (xR, yR), вторая точка (xL, y)
                    this.activeFillHandle.c1 = ar.c2;
                    this.activeFillHandle.r1 = ar.r2;

                    this.activeFillHandle.c2 = ar.c1;

                    // Когда идем назад, должна быть строка на 1 больше
                    this.activeFillHandle.r2 += 1;
                    // Случай, если мы еще не вышли из внутренней области
                    if ( this.activeFillHandle.r2 == ar.r1 ) {
                        this.fillHandleArea = 2;
                    }
                    break;
                case 2:
                    // Первая точка (xR, yR), вторая точка (xL, y)
                    this.activeFillHandle.c1 = ar.c2;
                    this.activeFillHandle.r1 = ar.r2;

                    this.activeFillHandle.c2 = ar.c1;

                    // Когда идем назад, должна быть строка на 1 больше
                    this.activeFillHandle.r2 += 1;

                    if ( this.activeFillHandle.r2 > this.activeFillHandle.r1 ) {
                        // Ситуация половинки последней строки
                        this.activeFillHandle.c1 = ar.c1;
                        this.activeFillHandle.r1 = ar.r1;

                        this.activeFillHandle.c2 = ar.c1;
                        this.activeFillHandle.r2 = ar.r1;
                    }
                    break;
                case 3:
                    // Первая точка (xL, yL), вторая точка (xR, y)
                    this.activeFillHandle.c1 = ar.c1;
                    this.activeFillHandle.r1 = ar.r1;

                    this.activeFillHandle.c2 = ar.c2;
                    break;
            }

            // Копируем в range для пересчета видимой области
            activeFillHandleCopy = this.activeFillHandle.clone();
            activeFillHandleCopy.r2 = rowByYNoDY;
        }

        //console.log ("row1: " + this.activeFillHandle.r1 + " col1: " + this.activeFillHandle.c1 + " row2: " + this.activeFillHandle.r2 + " col2: " + this.activeFillHandle.c2);
        // Перерисовываем
        this._drawSelection();

        // Смотрим, ушли ли мы за границу видимой области
        ret = this._calcFillHandleOffset( activeFillHandleCopy );
        this.model.workbook.handlers.trigger( "asc_onHideComment" );

        return ret;
    };

    /* Функция для применения автозаполнения */
    WorksheetView.prototype.applyFillHandle = function ( x, y, ctrlPress ) {
        var t = this;

        // Текущее выделение (к нему применится автозаполнение)
        var arn = t.activeRange.clone( true );
        arn.normalize();
        var range = t.model.getRange3( arn.r1, arn.c1, arn.r2, arn.c2 );

        // Были ли изменения
        var bIsHaveChanges = false;
        // Вычисляем индекс сдвига
        var nIndex = 0;
        /*nIndex*/
        if ( 0 === this.fillHandleDirection ) {
            // Горизонтальное движение
            nIndex = this.activeFillHandle.c2 - arn.c1;
            if ( 2 === this.fillHandleArea ) {
                // Для внутренности нужно вычесть 1 из значения
                bIsHaveChanges = arn.c2 !== (this.activeFillHandle.c2 - 1);
            }
            else {
                bIsHaveChanges = arn.c2 !== this.activeFillHandle.c2;
            }
        }
        else {
            // Вертикальное движение
            nIndex = this.activeFillHandle.r2 - arn.r1;
            if ( 2 === this.fillHandleArea ) {
                // Для внутренности нужно вычесть 1 из значения
                bIsHaveChanges = arn.r2 !== (this.activeFillHandle.r2 - 1);
            }
            else {
                bIsHaveChanges = arn.r2 !== this.activeFillHandle.r2;
            }
        }

        // Меняли ли что-то
        if ( bIsHaveChanges && (this.activeFillHandle.r1 !== this.activeFillHandle.r2 || this.activeFillHandle.c1 !== this.activeFillHandle.c2) ) {

            // Диапазон ячеек, который мы будем менять
            var changedRange = this.activeRange.clone( true );

            // Очищаем выделение
            this.cleanSelection();
            if ( 2 === this.fillHandleArea ) {
                // Мы внутри, будет удаление, нормируем и cбрасываем первую ячейку
                this.activeRange.normalize();
                // Проверяем, удалили ли мы все (если да, то область не меняется)
                if ( arn.c1 !== this.activeFillHandle.c2 || arn.r1 !== this.activeFillHandle.r2 ) {
                    // Уменьшаем диапазон (мы удалили не все)
                    if ( 0 === this.fillHandleDirection ) {
                        // Горизонтальное движение (для внутренности необходимо вычесть 1)
                        this.activeRange.c2 = this.activeFillHandle.c2 - 1;

                        changedRange.c1 = changedRange.c2;
                        changedRange.c2 = this.activeFillHandle.c2;
                    }
                    else {
                        // Вертикальное движение (для внутренности необходимо вычесть 1)
                        this.activeRange.r2 = this.activeFillHandle.r2 - 1;

                        changedRange.r1 = changedRange.r2;
                        changedRange.r2 = this.activeFillHandle.r2;
                    }
                }
            }
            else {
                // Мы вне выделения. Увеличиваем диапазон
                if ( 0 === this.fillHandleDirection ) {
                    // Горизонтальное движение
                    if ( 1 === this.fillHandleArea ) {
                        this.activeRange.c1 = this.activeFillHandle.c2;

                        changedRange.c2 = changedRange.c1 - 1;
                        changedRange.c1 = this.activeFillHandle.c2;
                    }
                    else {
                        this.activeRange.c2 = this.activeFillHandle.c2;

                        changedRange.c1 = changedRange.c2 + 1;
                        changedRange.c2 = this.activeFillHandle.c2;
                    }
                }
                else {
                    // Вертикальное движение
                    if ( 1 === this.fillHandleArea ) {
                        this.activeRange.r1 = this.activeFillHandle.r2;

                        changedRange.r2 = changedRange.r1 - 1;
                        changedRange.r1 = this.activeFillHandle.r2;
                    }
                    else {
                        this.activeRange.r2 = this.activeFillHandle.r2;

                        changedRange.r1 = changedRange.r2 + 1;
                        changedRange.r2 = this.activeFillHandle.r2;
                    }
                }

                // После увеличения, нужно обновить больший range
                arn = this.activeRange.clone( true );
            }

            changedRange.normalize();

            var applyFillHandleCallback = function ( res ) {
                if ( res ) {
                    // Автозаполняем ячейки
                    if ( range.promote( /*bCtrl*/ctrlPress, /*bVertical*/(1 === t.fillHandleDirection), nIndex ) ) {
                        // Вызываем функцию пересчета для заголовков форматированной таблицы
                        t.autoFilters.renameTableColumn( arn );
                    }
                    else {
                        t.handlers.trigger( "onErrorEvent", c_oAscError.ID.CannotFillRange, c_oAscError.Level.NoCritical );
                        t.activeRange.assign2( range.bbox );
                    }
                }

                // Сбрасываем параметры автозаполнения
                t.activeFillHandle = null;
                t.fillHandleDirection = -1;

                // Обновляем выделенные ячейки
                t.isChanged = true;
                t._updateCellsRange( arn );
            };

            // Можно ли применять автозаполнение ?
            this._isLockedCells( changedRange, /*subType*/null, applyFillHandleCallback );
        }
        else {
            // Ничего не менялось, сбрасываем выделение
            this.cleanSelection();
            // Сбрасываем параметры автозаполнения
            this.activeFillHandle = null;
            this.fillHandleDirection = -1;
            // Перерисовываем
            this._drawSelection();
        }
    };

    /* Функция для работы перемещения диапазона (selection). (x, y) - координаты точки мыши на области
     *  ToDo нужно переделать, чтобы moveRange появлялся только после сдвига от текущей ячейки
     */
    WorksheetView.prototype.changeSelectionMoveRangeHandle = function ( x, y, ctrlKey ) {
        // Возвращаемый результат
        var ret = null;
        // Пересчитываем координаты
        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );

        //если выделена ячейка заголовка ф/т, меняем выделение с ячейки на столбец ф/т
        //если выделена вся видимая часть форматированной таблицы, но не выделены последние скрытые строчки
        if ( null === this.startCellMoveRange ) {
            this.autoFilters.changeSelectionTablePart( this.activeRange );
        }

        var ar = this.activeRange.clone( true );

        // Колонка по X и строка по Y
        var colByX = this._findColUnderCursor( x, /*canReturnNull*/false, /*dX*/false ).col;
        var rowByY = this._findRowUnderCursor( y, /*canReturnNull*/false, /*dY*/false ).row;

        if ( ar.type == c_oAscSelectionType.RangeRow ) {
            colByX = 0;
        }
        if ( ar.type == c_oAscSelectionType.RangeCol ) {
            rowByY = 0;
        }
        if ( ar.type == c_oAscSelectionType.RangeMax ) {
            colByX = 0;
            rowByY = 0;
        }

        // Если мы только первый раз попали сюда, то копируем выделенную область
        if ( null === this.startCellMoveRange ) {
            // Учитываем погрешность (мы должны быть внутри диапазона при старте)
            if ( colByX < ar.c1 ) {
                colByX = ar.c1;
            }
            else if ( colByX > ar.c2 ) {
                colByX = ar.c2;
            }
            if ( rowByY < ar.r1 ) {
                rowByY = ar.r1;
            }
            else if ( rowByY > ar.r2 ) {
                rowByY = ar.r2;
            }
            this.startCellMoveRange = new asc_Range( colByX, rowByY, colByX, rowByY );
            this.startCellMoveRange.isChanged = false;	// Флаг, сдвигались ли мы от первоначального диапазона
            return ret;
        }

        // Разница, на сколько мы сдвинулись
        var colDelta = colByX - this.startCellMoveRange.c1;
        var rowDelta = rowByY - this.startCellMoveRange.r1;

        // Проверяем, нужно ли отрисовывать перемещение (сдвигались или нет)
        if ( false === this.startCellMoveRange.isChanged && 0 === colDelta && 0 === rowDelta ) {
            return ret;
        }
        // Выставляем флаг
        this.startCellMoveRange.isChanged = true;

        // Очищаем выделение, будем рисовать заново
        this.cleanSelection();

        this.activeMoveRange = ar;
        // Для первого раза нормализуем (т.е. первая точка - это левый верхний угол)
        this.activeMoveRange.normalize();

        // Выставляем
        this.activeMoveRange.c1 += colDelta;
        if ( 0 > this.activeMoveRange.c1 ) {
            colDelta -= this.activeMoveRange.c1;
            this.activeMoveRange.c1 = 0;
        }
        this.activeMoveRange.c2 += colDelta;

        this.activeMoveRange.r1 += rowDelta;
        if ( 0 > this.activeMoveRange.r1 ) {
            rowDelta -= this.activeMoveRange.r1;
            this.activeMoveRange.r1 = 0;
        }
        this.activeMoveRange.r2 += rowDelta;
        this.activeMoveRange._updateAdditionalData();

        // Увеличиваем, если выходим за область видимости // Critical Bug 17413
        while ( !this.cols[this.activeMoveRange.c2] ) {
            this.expandColsOnScroll( true );
            this.handlers.trigger( "reinitializeScrollX" );
        }
        while ( !this.rows[this.activeMoveRange.r2] ) {
            this.expandRowsOnScroll( true );
            this.handlers.trigger( "reinitializeScrollY" );
        }

        // Перерисовываем
        this._drawSelection();
        var d = {};
        /*var d = {
         deltaX : this.activeMoveRange.c1 < this.visibleRange.c1 ? this.activeMoveRange.c1-this.visibleRange.c1 :
         this.activeMoveRange.c2>this.visibleRange.c2 ? this.activeMoveRange.c2-this.visibleRange.c2 : 0,
         deltaY : this.activeMoveRange.r1 < this.visibleRange.r1 ? this.activeMoveRange.r1-this.visibleRange.r1 :
         this.activeMoveRange.r2>this.visibleRange.r2 ? this.activeMoveRange.r2-this.visibleRange.r2 : 0
         };
         while ( this._isColDrawnPartially( this.activeMoveRange.c2, this.visibleRange.c1 + d.deltaX) ) {++d.deltaX;}
         while ( this._isRowDrawnPartially( this.activeMoveRange.r2, this.visibleRange.r1 + d.deltaY) ) {++d.deltaY;}*/

        if ( y <= this.cellsTop + this.height_2px ) {
            d.deltaY = -1;
        }
        else if ( y >= this.drawingCtx.getHeight() - this.height_2px ) {
            d.deltaY = 1;
        }

        if ( x <= this.cellsLeft + this.width_2px ) {
            d.deltaX = -1;
        }
        else if ( x >= this.drawingCtx.getWidth() - this.width_2px ) {
            d.deltaX = 1;
        }

        this.model.workbook.handlers.trigger( "asc_onHideComment" );

        if ( this.activeMoveRange.type === c_oAscSelectionType.RangeRow ) {
            d.deltaX = 0;
        }
        else if ( this.activeMoveRange.type === c_oAscSelectionType.RangeCol ) {
            d.deltaY = 0;
        }
        else if ( this.activeMoveRange.type === c_oAscSelectionType.RangeMax ) {
            d.deltaX = 0;
            d.deltaY = 0;
        }

        return d;
    };

    WorksheetView.prototype.changeSelectionMoveResizeRangeHandle = function ( x, y, targetInfo, editor ) {
        // Возвращаемый результат
        if ( !targetInfo ) {
            return null;
        }
        var indexFormulaRange = targetInfo.indexFormulaRange, d = {deltaY: 0, deltaX: 0}, newFormulaRange = null;
        // Пересчитываем координаты
        x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
        y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );
        var ar = 0 == targetInfo.targetArr ? this.arrActiveFormulaRanges[indexFormulaRange].clone( true ) : this.arrActiveChartsRanges[indexFormulaRange].clone( true );

        // Колонка по X и строка по Y
        var colByX = this._findColUnderCursor( x, /*canReturnNull*/false, /*dX*/false ).col;
        var rowByY = this._findRowUnderCursor( y, /*canReturnNull*/false, /*dY*/false ).row;

        // Если мы только первый раз попали сюда, то копируем выделенную область
        if ( null === this.startCellMoveResizeRange ) {
            if ( (targetInfo.cursor == kCurNEResize || targetInfo.cursor == kCurSEResize) ) {
                this.startCellMoveResizeRange = ar.clone( true );
                this.startCellMoveResizeRange2 = new asc_Range( targetInfo.col, targetInfo.row, targetInfo.col, targetInfo.row, true );
            }
            else {
                this.startCellMoveResizeRange = ar.clone( true );
                if ( colByX < ar.c1 ) {
                    colByX = ar.c1;
                }
                else if ( colByX > ar.c2 ) {
                    colByX = ar.c2;
                }
                if ( rowByY < ar.r1 ) {
                    rowByY = ar.r1;
                }
                else if ( rowByY > ar.r2 ) {
                    rowByY = ar.r2;
                }
                this.startCellMoveResizeRange2 = new asc_Range( colByX, rowByY, colByX, rowByY );
            }
            return null;
        }

        // Очищаем выделение, будем рисовать заново
        // this.cleanSelection();
        this.overlayCtx.clear();

        if ( targetInfo.cursor == kCurNEResize || targetInfo.cursor == kCurSEResize ) {

            if ( colByX < this.startCellMoveResizeRange2.c1 ) {
                ar.c2 = this.startCellMoveResizeRange2.c1;
                ar.c1 = colByX;
            }
            else if ( colByX > this.startCellMoveResizeRange2.c1 ) {
                ar.c1 = this.startCellMoveResizeRange2.c1;
                ar.c2 = colByX;
            }
            else {
                ar.c1 = this.startCellMoveResizeRange2.c1;
                ar.c2 = this.startCellMoveResizeRange2.c1
            }

            if ( rowByY < this.startCellMoveResizeRange2.r1 ) {
                ar.r2 = this.startCellMoveResizeRange2.r2;
                ar.r1 = rowByY;
            }
            else if ( rowByY > this.startCellMoveResizeRange2.r1 ) {
                ar.r1 = this.startCellMoveResizeRange2.r1;
                ar.r2 = rowByY;
            }
            else {
                ar.r1 = this.startCellMoveResizeRange2.r1;
                ar.r2 = this.startCellMoveResizeRange2.r1;
            }

        }
        else {
            this.startCellMoveResizeRange.normalize();
            var colDelta = this.startCellMoveResizeRange.type != c_oAscSelectionType.RangeRow && this.startCellMoveResizeRange.type != c_oAscSelectionType.RangeMax ? colByX - this.startCellMoveResizeRange2.c1 : 0;
            var rowDelta = this.startCellMoveResizeRange.type != c_oAscSelectionType.RangeCol && this.startCellMoveResizeRange.type != c_oAscSelectionType.RangeMax ? rowByY - this.startCellMoveResizeRange2.r1 : 0;

            ar.c1 = this.startCellMoveResizeRange.c1 + colDelta;
            if ( 0 > ar.c1 ) {
                colDelta -= ar.c1;
                ar.c1 = 0;
            }
            ar.c2 = this.startCellMoveResizeRange.c2 + colDelta;

            ar.r1 = this.startCellMoveResizeRange.r1 + rowDelta;
            if ( 0 > ar.r1 ) {
                rowDelta -= ar.r1;
                ar.r1 = 0;
            }
            ar.r2 = this.startCellMoveResizeRange.r2 + rowDelta;

        }

        if ( y <= this.cellsTop + this.height_2px ) {
            d.deltaY = -1;
        }
        else if ( y >= this.drawingCtx.getHeight() - this.height_2px ) {
            d.deltaY = 1;
        }

        if ( x <= this.cellsLeft + this.width_2px ) {
            d.deltaX = -1;
        }
        else if ( x >= this.drawingCtx.getWidth() - this.width_2px ) {
            d.deltaX = 1;
        }

        if ( this.startCellMoveResizeRange.type === c_oAscSelectionType.RangeRow ) {
            d.deltaX = 0;
        }
        else if ( this.startCellMoveResizeRange.type === c_oAscSelectionType.RangeCol ) {
            d.deltaY = 0;
        }
        else if ( this.startCellMoveResizeRange.type === c_oAscSelectionType.RangeMax ) {
            d.deltaX = 0;
            d.deltaY = 0;
        }

        if ( 0 == targetInfo.targetArr ) {
            var _p = this.arrActiveFormulaRanges[indexFormulaRange].cursorePos, _l = this.arrActiveFormulaRanges[indexFormulaRange].formulaRangeLength;
            this.arrActiveFormulaRanges[indexFormulaRange] = ar.clone( true );
            this.arrActiveFormulaRanges[indexFormulaRange].cursorePos = _p;
            this.arrActiveFormulaRanges[indexFormulaRange].formulaRangeLength = _l;
            newFormulaRange = this.arrActiveFormulaRanges[indexFormulaRange];
        }
        else {
            this.arrActiveChartsRanges[indexFormulaRange] = ar.clone( true );
            this.moveRangeDrawingObjectTo = ar;
        }
        this._drawSelection();

        if ( newFormulaRange ) {
            editor.changeCellRange( newFormulaRange );
        }

        return d;
    };

    WorksheetView.prototype._cleanSelectionMoveRange = function () {
        // Перерисовываем и сбрасываем параметры
        this.cleanSelection();
        this.activeMoveRange = null;
        this.startCellMoveRange = null;
        this._drawSelection();
    };

    /* Функция для применения перемещения диапазона */
    WorksheetView.prototype.applyMoveRangeHandle = function ( ctrlKey ) {
        if ( null === this.activeMoveRange ) {
            // Сбрасываем параметры
            this.startCellMoveRange = null;
            return;
        }

        var arnFrom = this.activeRange.clone( true );
        var arnTo = this.activeMoveRange.clone( true );
        if ( arnFrom.isEqual( arnTo ) ) {
            this._cleanSelectionMoveRange();
            return;
        }

        var resmove = this.model._prepareMoveRange( arnFrom, arnTo );
        if ( resmove === -2 ) {
            this.handlers.trigger( "onErrorEvent", c_oAscError.ID.CannotMoveRange, c_oAscError.Level.NoCritical );
            this._cleanSelectionMoveRange();
        }
        else if ( resmove === -1 && this.autoFilters.checkMoveRangeIntoApplyAutoFilter( arnTo ) ) {
            var t = this;
            this.model.workbook.handlers.trigger( "asc_onConfirmAction", c_oAscConfirm.ConfirmReplaceRange, function ( can ) {
                if ( can ) {
                    t.moveRangeHandle( arnFrom, arnTo, ctrlKey );
                }
                else {
                    t._cleanSelectionMoveRange();
                }
            } );
        }
        else {
            this.moveRangeHandle( arnFrom, arnTo, ctrlKey );
        }
    };

    WorksheetView.prototype.applyMoveResizeRangeHandle = function ( target ) {
        if ( -1 == target.targetArr && !this.startCellMoveResizeRange.isEqual( this.moveRangeDrawingObjectTo ) ) {
            this.objectRender.moveRangeDrawingObject( this.startCellMoveResizeRange, this.moveRangeDrawingObjectTo );
        }

        this.startCellMoveResizeRange = null;
        this.startCellMoveResizeRange2 = null;
        this.moveRangeDrawingObjectTo = null;
    };

    WorksheetView.prototype.moveRangeHandle = function ( arnFrom, arnTo, copyRange ) {
        var t = this;
        var onApplyMoveRangeHandleCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                t._cleanSelectionMoveRange();
                return;
            }

            var onApplyMoveAutoFiltersCallback = function ( isSuccess ) {
                if ( false === isSuccess ) {
                    t._cleanSelectionMoveRange();
                    return;
                }

                // Очищаем выделение
                t.cleanSelection();

                //ToDo t.cleanDepCells();
                History.Create_NewPoint();
                History.SetSelection( arnFrom.clone() );
                History.SetSelectionRedo( arnTo.clone() );
                History.StartTransaction();

                t.autoFilters._preMoveAutoFilters( arnFrom, arnTo, copyRange );

                t.model._moveRange( arnFrom, arnTo, copyRange );
                t.cellCommentator.moveRangeComments( arnFrom, arnTo );
                t.objectRender.moveRangeDrawingObject( arnFrom, arnTo );

                t.autoFilters._moveAutoFilters( arnTo, arnFrom, null, copyRange, true );
                // Вызываем функцию пересчета для заголовков форматированной таблицы
                t.autoFilters.renameTableColumn( arnFrom );
                t.autoFilters.renameTableColumn( arnTo );
                t.autoFilters.reDrawFilter( arnFrom );

                History.EndTransaction();

                t._updateCellsRange( arnTo, false, true );
                t.activeRange = arnTo.clone( true );
                // Сбрасываем параметры
                t.activeMoveRange = null;
                t.startCellMoveRange = null;
                t._updateCellsRange( arnFrom, false, true );
                // Тут будет отрисовка select-а
                t._recalculateAfterUpdate( [arnFrom, arnTo] );

                // Вызовем на всякий случай, т.к. мы можем уже обновиться из-за формул ToDo возможно стоит убрать это в дальнейшем (но нужна переработка формул) - http://bugzserver/show_bug.cgi?id=24505
                t._updateSelectionNameAndInfo();
            };

            if ( t.autoFilters._searchFiltersInRange( arnFrom ) ) {
                t._isLockedAll( onApplyMoveAutoFiltersCallback );
            }
            else {
                onApplyMoveAutoFiltersCallback();
            }
        };

        if ( this.autoFilters.isCheckMoveRange( arnFrom, arnTo ) ) {
            this._isLockedCells( [arnFrom, arnTo], null, onApplyMoveRangeHandleCallback );
        }
        else {
            this._cleanSelectionMoveRange();
        }
    };

    WorksheetView.prototype.emptySelection = function ( options ) {
        // Удаляем выделенные графичекие объекты
        if ( this.objectRender.selectedGraphicObjectsExists() ) {
            this.objectRender.controller.deleteSelectedObjects();
        }
        else {
            this.setSelectionInfo( "empty", options );
        }
    };

    WorksheetView.prototype.setSelectionInfo = function ( prop, val, onlyActive, isLocal ) {
        // Проверка глобального лока
        if ( this.collaborativeEditing.getGlobalLock() ) {
            return;
        }

        var t = this;
        var checkRange = null;
        var arn = t.activeRange.clone( true );
        if ( onlyActive ) {
            checkRange = new asc_Range( arn.startCol, arn.startRow, arn.startCol, arn.startRow );
        }
        else {
            checkRange = arn.getAllRange();
        }

        var onSelectionCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }
            var range;
            var canChangeColWidth = c_oAscCanChangeColWidth.none;
            var bIsUpdate = true;

            if ( onlyActive ) {
                range = t.model.getRange3( arn.startRow, arn.startCol, arn.startRow, arn.startCol );
            }
            else {
                if ( c_oAscSelectionType.RangeMax === arn.type ) {
                    range = t.model.getRange3( /*arn.r1*/0, /*arn.c1*/0, gc_nMaxRow0, gc_nMaxCol0 );
                }
                else if ( c_oAscSelectionType.RangeCol === arn.type ) {
                    range = t.model.getRange3( /*arn.r1*/0, arn.c1, gc_nMaxRow0, arn.c2 );
                }
                else if ( c_oAscSelectionType.RangeRow === arn.type ) {
                    range = t.model.getRange3( arn.r1, /*arn.c1*/0, arn.r2, gc_nMaxCol0 );
                }
                else {
                    range = t.model.getRange3( arn.r1, arn.c1, arn.r2, arn.c2 );
                }
            }

            var isLargeRange = t._isLargeRange( range.bbox ), callTrigger = false;
            var res;
            var mc, r, c, cell;

            function makeBorder( b ) {
                var border = new BorderProp();
                if ( b === false ) {
                    border.setStyle( c_oAscBorderStyles.None );
                }
                else if ( b ) {
                    if ( b.style !== null && b.style !== undefined ) {
                        border.setStyle( b.style );
                    }
                    if ( b.color !== null && b.color !== undefined ) {
                        if ( b.color instanceof asc_CColor ) {
                            border.c = CorrectAscColor( b.color );
                        }
                    }
                }
                return border;
            }

            History.Create_NewPoint();
            History.StartTransaction();

            switch ( prop ) {
                case "fn":
                    range.setFontname( val );
                    canChangeColWidth = c_oAscCanChangeColWidth.numbers;
                    break;
                case "fs":
                    range.setFontsize( val );
                    canChangeColWidth = c_oAscCanChangeColWidth.numbers;
                    break;
                case "b":
                    range.setBold( val );
                    break;
                case "i":
                    range.setItalic( val );
                    break;
                case "u":
                    range.setUnderline( val );
                    break;
                case "s":
                    range.setStrikeout( val );
                    break;
                case "fa":
                    range.setFontAlign( val );
                    break;
                case "a":
                    range.setAlignHorizontal( val );
                    break;
                case "va":
                    range.setAlignVertical( val );
                    break;
                case "c":
                    range.setFontcolor( val );
                    break;
                case "bc":
                    range.setFill( (val) ? (val) : null );
                    break; // ToDo можно делать просто отрисовку
                case "wrap":
                    range.setWrap( val );
                    break;
                case "shrink":
                    range.setShrinkToFit( val );
                    break;
                case "value":
                    range.setValue( val );
                    break;
                case "format":
                    range.setNumFormat( val );
                    canChangeColWidth = c_oAscCanChangeColWidth.numbers;
                    break;
                case "angle":
                    range.setAngle( val );
                    break;
                case "rh":
                    range.removeHyperlink( null, true );
                    break;
                case "border":
                    if ( isLargeRange ) {
                        callTrigger = true;
                        t.handlers.trigger( "slowOperation", true );
                    }
                    // None
                    if ( val.length < 1 ) {
                        range.setBorder( null );
                        break;
                    }
                    res = new Border();
                    // Diagonal
                    res.d = makeBorder( val[c_oAscBorderOptions.DiagD] || val[c_oAscBorderOptions.DiagU] );
                    res.dd = val[c_oAscBorderOptions.DiagD] ? true : false;
                    res.du = val[c_oAscBorderOptions.DiagU] ? true : false;
                    // Vertical
                    res.l = makeBorder( val[c_oAscBorderOptions.Left] );
                    res.iv = makeBorder( val[c_oAscBorderOptions.InnerV] );
                    res.r = makeBorder( val[c_oAscBorderOptions.Right] );
                    // Horizontal
                    res.t = makeBorder( val[c_oAscBorderOptions.Top] );
                    res.ih = makeBorder( val[c_oAscBorderOptions.InnerH] );
                    res.b = makeBorder( val[c_oAscBorderOptions.Bottom] );
                    // Change border
                    range.setBorder( res );
                    break;
                case "merge":
                    if ( isLargeRange ) {
                        callTrigger = true;
                        t.handlers.trigger( "slowOperation", true );
                    }
                    switch ( val ) {
                        case c_oAscMergeOptions.MergeCenter:
                        case c_oAscMergeOptions.Merge:
                            range.merge( val );
                            t.cellCommentator.mergeComments( range.getBBox0() );
                            break;
                        case c_oAscMergeOptions.Unmerge:
                            range.unmerge();
                            break;
                        case c_oAscMergeOptions.MergeAcross:
                            for ( res = arn.r1; res <= arn.r2; ++res ) {
                                t.model.getRange3( res, arn.c1, res, arn.c2 ).merge( val );
                                cell = new asc_Range( arn.c1, res, arn.c2, res );
                                t.cellCommentator.mergeComments( cell );
                            }
                            break;
                    }
                    break;

                case "sort":
                    if ( isLargeRange ) {
                        callTrigger = true;
                        t.handlers.trigger( "slowOperation", true );
                    }
                    t.cellCommentator.sortComments( range.sort( val, arn.startCol ) );
                    break;

                case "empty":
                    if ( isLargeRange ) {
                        callTrigger = true;
                        t.handlers.trigger( "slowOperation", true );
                    }
                    /* отключаем отрисовку на случай необходимости пересчета ячеек, заносим ячейку, при необходимости в список перерисовываемых */
                    lockDraw( t.model.workbook );

                    // Если нужно удалить автофильтры - удаляем
                    if ( val === c_oAscCleanOptions.All || val === c_oAscCleanOptions.Text ) {
                        t.autoFilters.isEmptyAutoFilters( arn );
                    }
                    else if ( val === c_oAscCleanOptions.Format ) {
                        t.autoFilters.cleanFormat( arn );
                    }

                    if ( val === c_oAscCleanOptions.All ) {
                        range.cleanAll();
                        // Удаляем комментарии
                        t.cellCommentator.deleteCommentsRange( arn );
                    }
                    else if ( val === c_oAscCleanOptions.Text || val === c_oAscCleanOptions.Formula ) {
                        range.cleanText();
                    }
                    else if ( val === c_oAscCleanOptions.Format ) {
                        range.cleanFormat();
                    }
                    else if ( val === c_oAscCleanOptions.Comments ) {
                        t.cellCommentator.deleteCommentsRange( arn );
                    }
                    else if ( val === c_oAscCleanOptions.Hyperlinks ) {
                        range.cleanHyperlinks();
                    }

                    // Вызываем функцию пересчета для заголовков форматированной таблицы
                    t.autoFilters.renameTableColumn( arn );

                    /* возвращаем отрисовку. и перерисовываем ячейки с предварительным пересчетом */
                    buildRecalc( t.model.workbook );
                    unLockDraw( t.model.workbook );
                    break;

                case "changeDigNum":
                    res = t.cols.slice( arn.c1, arn.c2 + 1 ).reduce( function ( r, c ) {
                        r.push( c.charCount );
                        return r;
                    }, [] );
                    range.shiftNumFormat( val, res );
                    canChangeColWidth = c_oAscCanChangeColWidth.numbers;
                    break;
                case "changeFontSize":
                    mc = t.model.getMergedByCell( arn.startRow, arn.startCol );
                    c = mc ? mc.c1 : arn.startCol;
                    r = mc ? mc.r1 : arn.startRow;
                    cell = t._getVisibleCell( c, r );
                    if ( undefined !== cell ) {
                        var oldFontSize = cell.getFontsize();
                        var newFontSize = asc_incDecFonSize( val, oldFontSize );
                        if ( null !== newFontSize ) {
                            range.setFontsize( newFontSize );
                            canChangeColWidth = c_oAscCanChangeColWidth.numbers;
                        }
                    }
                    break;
                case "style":
                    range.setCellStyle( val );
                    canChangeColWidth = c_oAscCanChangeColWidth.numbers;
                    break;
                    break;
                case "paste":
                    // Вставляем текст из локального буфера или нет
                    isLocal ? t._pasteFromLocalBuff( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth ) : t._pasteFromGlobalBuff( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth );
                    bIsUpdate = false;
                    break;
                case "hyperlink":
                    if ( val && val.hyperlinkModel ) {
                        if ( c_oAscHyperlinkType.RangeLink === val.asc_getType() ) {
                            var hyperlinkRangeTmp = t.model.getRange2( val.asc_getRange() );
                            if ( null === hyperlinkRangeTmp ) {
                                bIsUpdate = false;
                                break;
                            }
                        }
                        val.hyperlinkModel.Ref = range;
                        range.setHyperlink( val.hyperlinkModel );
                        // Вставим текст в активную ячейку (а не так, как MSExcel в первую ячейку диапазона)
                        mc = t.model.getMergedByCell( arn.startRow, arn.startCol );
                        c = mc ? mc.c1 : arn.startCol;
                        r = mc ? mc.r1 : arn.startRow;
                        if ( null !== val.asc_getText() ) {
                            t.model.getRange3( r, c, r, c ).setValue( val.asc_getText() );
                            // Вызываем функцию пересчета для заголовков форматированной таблицы
                            t.autoFilters.renameTableColumn( arn );
                        }
                        break;
                    }
                    else {
                        bIsUpdate = false;
                        break;
                    }

                default:
                    bIsUpdate = false;
                    break;
            }

            if ( bIsUpdate ) {
                if ( callTrigger ) {
                    t.handlers.trigger( "slowOperation", false );
                }
                t.isChanged = true;
                t._updateCellsRange( arn, canChangeColWidth );
            }

            //в случае, если вставляем из глобального буфера, транзакцию закрываем внутри функции _pasteFromGlobalBuff на callbacks от загрузки шрифтов и картинок
            if ( prop !== "paste" || (prop === "paste" && isLocal) ) {
                History.EndTransaction();
            }
        };
        if ( "paste" === prop && val.onlyImages !== true ) {
            // Для past свой диапазон
            if ( isLocal === "binary" ) {
                checkRange = t._pasteFromBinary( val, true );
            }
            else {
                checkRange = t._setInfoAfterPaste( val, true );
            }
        }
        if ( "paste" === prop && val.onlyImages === true ) {
            onSelectionCallback();
        }
        else {
            this._isLockedCells( checkRange, /*subType*/null, onSelectionCallback );
        }
    };

    WorksheetView.prototype._pasteFromLocalBuff = function ( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth ) {
        var t = this;
        var callTrigger = false;
        if ( isLargeRange ) {
            callTrigger = true;
            t.handlers.trigger( "slowOperation", true );
        }

        lockDraw( t.model.workbook );
        var selectData;
        if ( isLocal === 'binary' ) {
            selectData = t._pasteFromBinary( val );
        }
        else {
            selectData = t._setInfoAfterPaste( val );
        }

        t.autoFilters.renameTableColumn( t.activeRange );

        if ( !selectData ) {
            bIsUpdate = false;
            buildRecalc( t.model.workbook );
            unLockDraw( t.model.workbook );
            return;
        }
        t.expandColsOnScroll();
        t.expandRowsOnScroll();

        var arrFormula = selectData[1];
        for ( var i = 0; i < arrFormula.length; ++i ) {//!!!
            var rangeF = arrFormula[i].range;
            var valF = arrFormula[i].val;
            if ( rangeF.isOneCell() ) {
                rangeF.setValue( valF, null, true );
            }
            else {
                var oBBox = rangeF.getBBox0();
                t.model._getCell( oBBox.r1, oBBox.c1 ).setValue( valF, null, true );
            }
        }

        buildRecalc( t.model.workbook );
        unLockDraw( t.model.workbook );
        var arn = selectData[0];
        var selectionRange = arn.clone( true );

        //добавляем автофильтры и форматированные таблицы
        if ( isLocal === true && val.lStorage && val.lStorage.autoFilters && val.lStorage.autoFilters.length ) {
            var aFilters = val.lStorage.autoFilters;
            var range;
            for ( var aF = 0; aF < aFilters.length; aF++ ) {
                range = t.model.getRange3( aFilters[aF].range.r1 + selectionRange.r1, aFilters[aF].range.c1 + selectionRange.c1, aFilters[aF].range.r2 + selectionRange.r1, aFilters[aF].range.c2 + selectionRange.c1 );

                //если область вставки содержит форматированную таблицу, которая пересекается с вставляемой форматированной таблицей
                var intersectionRangeWithTableParts = t.autoFilters._intersectionRangeWithTableParts( range.bbox, this.model );
                if ( intersectionRangeWithTableParts )
                    continue;

                if ( aFilters[aF].style ) {
                    range.cleanFormat();
                }
                t.autoFilters.addAutoFilter( aFilters[aF].style, range.bbox, true, true );
                if ( !aFilters[aF].autoFilter ) {
                    t.autoFilters.addAutoFilter( null, range.bbox, true, true );
                }
            }
        }
        else if ( isLocal === 'binary' && val.TableParts && val.TableParts.length ) {
            var aFilters = val.TableParts;
            var range;
            var tablePartRange;
            var activeRange = window["Asc"]["editor"].wb.clipboard.activeRange;
            var refInsertBinary = Asc.g_oRangeCache.getAscRange( activeRange );
            var diffRow;
            var diffCol;
            for ( var aF = 0; aF < aFilters.length; aF++ ) {
                tablePartRange = aFilters[aF].Ref;
                diffRow = tablePartRange.r1 - refInsertBinary.r1;
                diffCol = tablePartRange.c1 - refInsertBinary.c1;
                range = t.model.getRange3( diffRow + selectionRange.r1, diffCol + selectionRange.c1, diffRow + selectionRange.r1 + (tablePartRange.r2 - tablePartRange.r1), diffCol + selectionRange.c1 + (tablePartRange.c2 - tablePartRange.c1) );

                //если область вставки содержит форматированную таблицу, которая пересекается с вставляемой форматированной таблицей
                var intersectionRangeWithTableParts = t.autoFilters._intersectionRangeWithTableParts( range.bbox, this.model );
                if ( intersectionRangeWithTableParts )
                    continue;

                if ( aFilters[aF].style ) {
                    range.cleanFormat();
                }

                var bWithoutFilter = false;
                if ( !aFilters[aF].AutoFilter ) {
                    bWithoutFilter = true;
                }

                t.autoFilters.addAutoFilter( aFilters[aF].TableStyleInfo.Name, range.bbox, true, true, bWithoutFilter );
            }
        }

        //делаем unmerge ф/т
        var arnToRange = t.activeRange;
        var intersectionRangeWithTableParts = t.autoFilters._intersectionRangeWithTableParts( arnToRange, this.model );
        if ( intersectionRangeWithTableParts && intersectionRangeWithTableParts.length ) {
            var tablePart;
            for ( var i = 0; i < intersectionRangeWithTableParts.length; i++ ) {
                tablePart = intersectionRangeWithTableParts[i];
                this.model.getRange3( tablePart.Ref.r1, tablePart.Ref.c1, tablePart.Ref.r2, tablePart.Ref.c2 ).unmerge();
            }
        }

        if ( bIsUpdate ) {
            if ( callTrigger ) {
                t.handlers.trigger( "slowOperation", false );
            }
            t.isChanged = true;
            t._updateCellsRange( arn, canChangeColWidth );
        }

        var oSelection = History.GetSelection();
        if ( null != oSelection ) {
            oSelection = oSelection.clone();
            oSelection.assign( selectionRange.c1, selectionRange.r1, selectionRange.c2, selectionRange.r2 );
            History.SetSelection( oSelection );
            History.SetSelectionRedo( oSelection );
        }
    };

    WorksheetView.prototype._pasteFromGlobalBuff = function ( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth ) {
        var t = this;

        //загрузка шрифтов, в случае удачи на callback вставляем текст
        t._loadFonts( val.fontsNew, function () {

            var api = asc["editor"];
            var isEndTransaction = false;
            if ( val.addImages && val.addImages.length != 0 && !(window["Asc"]["editor"] && window["Asc"]["editor"].isChartEditor) ) {
                if ( val.onlyImages !== true ) {
                    t._pasteFromLocalBuff( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth );
                }
                api.wb.clipboard._insertImagesFromHTML( t, val );
                isEndTransaction = true;
            }
            else if ( val.addImagesFromWord && val.addImagesFromWord.length != 0 && !(window["Asc"]["editor"] && window["Asc"]["editor"].isChartEditor) ) {
                var oObjectsForDownload = GetObjectsForImageDownload( val._aPastedImages );

                //if already load images on server
                if ( api.wb.clipboard.alreadyLoadImagesOnServer === true ) {
                    var oImageMap = {};
                    for ( var i = 0, length = oObjectsForDownload.aBuilderImagesByUrl.length; i < length; ++i ) {
                        var url = oObjectsForDownload.aUrls[i];

                        //get name from array already load on server urls
                        var name = api.wb.clipboard.oImages[url];
                        var aImageElem = oObjectsForDownload.aBuilderImagesByUrl[i];
                        if ( name ) {
                            if ( Array.isArray( aImageElem ) ) {
                                for ( var j = 0; j < aImageElem.length; ++j ) {
                                    var imageElem = aImageElem[j];
                                    if ( null != imageElem ) {
                                        imageElem.SetUrl( name );
                                    }
                                }
                            }
                            oImageMap[i] = name;
                        }
                        else {
                            oImageMap[i] = url;
                        }
                    }

                    if ( val.onlyImages !== true ) {
                        t._pasteFromLocalBuff( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth );
                    }
                    api.wb.clipboard._insertImagesFromBinaryWord( t, val, oImageMap );
                    isEndTransaction = true;
                }
                else {
                    sendImgUrls( api, oObjectsForDownload.aUrls, function ( data ) {
                        var oImageMap = {};
                        ResetNewUrls( data, oObjectsForDownload.aUrls, oObjectsForDownload.aBuilderImagesByUrl, oImageMap );

                        if ( val.onlyImages !== true ) {
                            t._pasteFromLocalBuff( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth );
                        }
                        api.wb.clipboard._insertImagesFromBinaryWord( t, val, oImageMap );
                        //закрываем транзакцию, поскольку в setSelectionInfo она не закроется
                        History.EndTransaction();
                    }, true );
                }

            }
            else if ( val.onlyImages !== true ) {
                t._pasteFromLocalBuff( isLargeRange, isLocal, val, bIsUpdate, canChangeColWidth );
                isEndTransaction = true;
            }

            //закрываем транзакцию, поскольку в setSelectionInfo она не закроется
            if ( isEndTransaction ) {
                History.EndTransaction();
            }
        } );
    };

    WorksheetView.prototype._setInfoAfterPaste = function ( values, isCheckSelection ) {
        var t = this;
        var wb = window["Asc"]["editor"].wb;
        var arn = wb && wb.clipboard && wb.clipboard.activeRange ? wb.clipboard.activeRange : t.activeRange.clone( true );

        var arrFormula = [];
        var numFor = 0;
        var rMax = values.length + values.rowSpanSpCount;
        if ( values.rowCount && values.rowCount !== 0 && values.isOneTable ) {
            rMax = values.rowCount + arn.r1;
        }

        var cMax = values.cellCount + arn.c1;

        var isMultiple = false;
        var firstCell = t.model.getRange3( arn.r1, arn.c1, arn.r1, arn.c1 );
        var isMergedFirstCell = firstCell.hasMerged();
        var rangeUnMerge = t.model.getRange3( arn.r1, arn.c1, rMax - 1, cMax - 1 );
        var isOneMerge = false;


        //если вставляем в мерженную ячейку, диапазон которой больше или равен
        if ( arn.c2 >= cMax - 1 && arn.r2 >= rMax - 1 && isMergedFirstCell && isMergedFirstCell.c1 === arn.c1 && isMergedFirstCell.c2 === arn.c2 && isMergedFirstCell.r1 === arn.r1 && isMergedFirstCell.r2 === arn.r2 && cMax - arn.c1 === values[arn.r1][arn.c1][0].colSpan && rMax - arn.r1 === values[arn.r1][arn.c1][0].rowSpan ) {
            if ( !isCheckSelection ) {
                values[arn.r1][arn.c1][0].colSpan = isMergedFirstCell.c2 - isMergedFirstCell.c1 + 1;
                values[arn.r1][arn.c1][0].rowSpan = isMergedFirstCell.r2 - isMergedFirstCell.r1 + 1;
            }
            isOneMerge = true;
        }
        else if ( arn.c2 >= cMax - 1 && arn.r2 >= rMax - 1 && values.isOneTable ) {
            //если область кратная куску вставки
            var widthArea = arn.c2 - arn.c1 + 1;
            var heightArea = arn.r2 - arn.r1 + 1;
            var widthPasteFr = cMax - arn.c1;
            var heightPasteFr = rMax - arn.r1;
            //если кратны, то обрабатываем
            if ( widthArea % widthPasteFr === 0 && heightArea % heightPasteFr === 0 ) {
                isMultiple = true;
            }
            else if ( firstCell.hasMerged() !== null )//в противном случае ошибка
            {
                if ( isCheckSelection ) {
                    return arn;
                }
                else {
                    this.handlers.trigger( "onError", c_oAscError.ID.PastInMergeAreaError, c_oAscError.Level.NoCritical );
                    return;
                }
            }
        }
        else {
            //проверка на наличие части объединённой ячейки в области куда осуществляем вставку
            for ( var rFirst = arn.r1; rFirst < rMax; ++rFirst ) {
                for ( var cFirst = arn.c1; cFirst < cMax; ++cFirst ) {
                    range = t.model.getRange3( rFirst, cFirst, rFirst, cFirst );
                    var merged = range.hasMerged();
                    if ( merged ) {
                        if ( merged.r1 < arn.r1 || merged.r2 > rMax - 1 || merged.c1 < arn.c1 || merged.c2 > cMax - 1 ) {
                            //ошибка в случае если вставка происходит в часть объедененной ячейки
                            if ( isCheckSelection ) {
                                return arn;
                            }
                            else {
                                this.handlers.trigger( "onErrorEvent", c_oAscError.ID.PastInMergeAreaError, c_oAscError.Level.NoCritical );
                                return;
                            }
                        }
                    }
                }
            }
        }
        var rMax2 = rMax;
        var cMax2 = cMax;
        var rMax = values.length;
        var trueArn = t.activeRange;
        if ( isCheckSelection ) {
            var newArr = arn.clone( true );
            newArr.r2 = rMax2 - 1;
            newArr.c2 = cMax2 - 1;
            if ( isMultiple || isOneMerge ) {
                newArr.r2 = trueArn.r2;
                newArr.c2 = trueArn.c2;
            }
            return newArr;
        }
        //если не возникает конфликт, делаем unmerge
        rangeUnMerge.unmerge();
        if ( !isOneMerge ) {
            arn.r2 = (rMax2 - 1 > 0) ? (rMax2 - 1) : 0;
            arn.c2 = (cMax2 - 1 > 0) ? (cMax2 - 1) : 0;
        }

        var mergeArr = [];

        var n = 0;
        if ( isMultiple )//случай автозаполнения сложных форм
        {
            t.model.getRange3( trueArn.r1, trueArn.c1, trueArn.r2, trueArn.c2 ).unmerge();
            var maxARow = heightArea / heightPasteFr;
            var maxACol = widthArea / widthPasteFr;
            var plRow = (rMax2 - arn.r1);
            var plCol = (arn.c2 - arn.c1) + 1;
        }
        else {
            var maxARow = 1;
            var maxACol = 1;
            var plRow = 0;
            var plCol = 0;
        }
        if ( isMultiple ) {
            if ( values[arn.r1] && values[arn.r1][arn.c1] ) {
                var currentObj = values[arn.r1][arn.c1][0];
                var valFormat = '';
                if ( currentObj[0] !== undefined ) {
                    valFormat = currentObj[0].text;
                }
            }
        }
        for ( var autoR = 0; autoR < maxARow; ++autoR ) {
            for ( var autoC = 0; autoC < maxACol; ++autoC ) {
                for ( var r = arn.r1; r < rMax; ++r ) {
                    for ( var c = arn.c1; c < values[r].length; ++c ) {
                        if ( undefined !== values[r][c] ) {
                            var range = t.model.getRange3( r + autoR * plRow, c + autoC * plCol, r + autoR * plRow, c + autoC * plCol );

                            var currentObj = values[r][c][0];
                            if ( currentObj.length === 1 ) {
                                //if(!isMultiple)
                                //{
                                var valFormat = currentObj[0].text;
                                var nameFormat = false;
                                //}
                                if ( currentObj[0].cellFrom ) {
                                    var offset = range.getCells()[0].getOffset2( currentObj[0].cellFrom ), assemb, _p_ = new parserFormula( currentObj[0].text.substring( 1 ), "", range.worksheet );

                                    if ( _p_.parse() ) {
                                        assemb = _p_.changeOffset( offset ).assemble();
                                        //range.setValue("="+assemb);
                                        arrFormula[numFor] = {};
                                        arrFormula[numFor].range = range;
                                        arrFormula[numFor].val = "=" + assemb;
                                        numFor++;
                                    }
                                }
                                else {
                                    range.setValue( valFormat );
                                }
                                if ( nameFormat ) {
                                    range.setNumFormat( nameFormat );
                                }
                                range.setBold( currentObj[0].format.b );
                                range.setItalic( currentObj[0].format.i );
                                range.setStrikeout( currentObj[0].format.s );
                                if ( !isOneMerge && currentObj[0].format && currentObj[0].format.c != null && currentObj[0].format.c != undefined ) {
                                    range.setFontcolor( currentObj[0].format.c );
                                }
                                range.setUnderline( currentObj[0].format.u );
                                range.setAlignVertical( currentObj.va );
                                range.setFontname( currentObj[0].format.fn );
                                range.setFontsize( currentObj[0].format.fs );
                            }
                            else {
                                range.setValue2( currentObj );
                                range.setAlignVertical( currentObj.va );
                            }

                            if ( currentObj.length === 1 && currentObj[0].format.fs !== '' && currentObj[0].format.fs !== null && currentObj[0].format.fs !== undefined ) {
                                range.setFontsize( currentObj[0].format.fs );
                            }
                            if ( !isOneMerge ) {
                                range.setAlignHorizontal( currentObj.a );
                            }
                            var isMerged = false;
                            for ( var mergeCheck = 0; mergeCheck < mergeArr.length; ++mergeCheck ) {
                                if ( r + 1 + autoR * plRow <= mergeArr[mergeCheck].r2 && r + 1 + autoR * plRow >= mergeArr[mergeCheck].r1 && c + autoC * plCol + 1 <= mergeArr[mergeCheck].c2 && c + 1 + autoC * plCol >= mergeArr[mergeCheck].c1 ) {
                                    isMerged = true;
                                }
                            }

                            //обработка для мерженных ячеек
                            if ( (currentObj.colSpan > 1 || currentObj.rowSpan > 1) && !isMerged ) {
                                range.setOffsetLast( {offsetCol: currentObj.colSpan - 1, offsetRow: currentObj.rowSpan - 1} );
                                mergeArr[n] = {
                                    r1: range.first.row, r2: range.last.row, c1: range.first.col, c2: range.last.col
                                };
                                n++;
                                if ( currentObj[0] == undefined ) {
                                    range.setValue( '' );
                                }
                                range.merge( c_oAscMergeOptions.Merge );
                            }
                            if ( !isOneMerge ) {
                                range.setBorderSrc( currentObj.borders );
                            }
                            range.setWrap( currentObj.wrap );
                            if ( currentObj.bc && currentObj.bc.rgb ) {
                                range.setFill( currentObj.bc );
                            }
                            var link = values[r][c][0].hyperLink;
                            if ( link ) {
                                var newHyperlink = new Hyperlink();
                                if ( values[r][c][0].hyperLink.search( '#' ) === 0 ) {
                                    newHyperlink.setLocation( link.replace( '#', '' ) );
                                }
                                else {
                                    newHyperlink.Hyperlink = link;
                                }
                                newHyperlink.Ref = range;
                                newHyperlink.Tooltip = values[r][c][0].toolTip;
                                range.setHyperlink( newHyperlink );
                            }
                        }
                    }
                }
            }
        }
        if ( isMultiple ) {
            arn.r2 = trueArn.r2;
            arn.c2 = trueArn.c2;
        }

        t.isChanged = true;
        t.activeRange.c2 = arn.c2;
        t.activeRange.r2 = arn.r2;
        var arnFor = [];
        arnFor[0] = arn;
        arnFor[1] = arrFormula;
        return arnFor;
    };

    WorksheetView.prototype._pasteFromBinary = function ( val, isCheckSelection ) {
        var t = this;
        var arn = t.activeRange.clone( true );
        var arrFormula = [];
        var numFor = 0;

        var pasteRange = window["Asc"]["editor"].wb.clipboard.activeRange;
        var activeCellsPasteFragment = Asc.g_oRangeCache.getAscRange( pasteRange );
        var rMax = (activeCellsPasteFragment.r2 - activeCellsPasteFragment.r1) + arn.r1 + 1;
        var cMax = (activeCellsPasteFragment.c2 - activeCellsPasteFragment.c1) + arn.c1 + 1;

        if ( cMax > gc_nMaxCol0 ) {
            cMax = gc_nMaxCol0;
        }
        if ( rMax > gc_nMaxRow0 ) {
            rMax = gc_nMaxRow0;
        }

        var isMultiple = false;
        var firstCell = t.model.getRange3( arn.r1, arn.c1, arn.r1, arn.c1 );
        var isMergedFirstCell = firstCell.hasMerged();
        var isOneMerge = false;


        var startCell = val.getCell3( activeCellsPasteFragment.r1, activeCellsPasteFragment.c1 );
        var isMergedStartCell = startCell.hasMerged();

        var firstValuesCol;
        var firstValuesRow;
        if ( isMergedStartCell != null ) {
            firstValuesCol = isMergedStartCell.c2 - isMergedStartCell.c1;
            firstValuesRow = isMergedStartCell.r2 - isMergedStartCell.r1;
        }
        else {
            firstValuesCol = 0;
            firstValuesRow = 0;
        }

        var rowDiff = arn.r1 - activeCellsPasteFragment.r1;
        var colDiff = arn.c1 - activeCellsPasteFragment.c1;
        var newPasteRange = new Asc.Range( arn.c1 - colDiff, arn.r1 - rowDiff, arn.c2 - colDiff, arn.r2 - rowDiff );
        //если вставляем в мерженную ячейку, диапазон которой больше или меньше, но не равен выделенной области
        if ( isMergedFirstCell && isMergedFirstCell.isEqual( arn ) && cMax - arn.c1 === (firstValuesCol + 1) && rMax - arn.r1 === (firstValuesRow + 1) && !newPasteRange.isEqual( activeCellsPasteFragment ) ) {
            isOneMerge = true;
            rMax = arn.r2 + 1;
            cMax = arn.c2 + 1;
        }
        else if ( arn.c2 >= cMax - 1 && arn.r2 >= rMax - 1 ) {
            //если область кратная куску вставки
            var widthArea = arn.c2 - arn.c1 + 1;
            var heightArea = arn.r2 - arn.r1 + 1;
            var widthPasteFr = cMax - arn.c1;
            var heightPasteFr = rMax - arn.r1;
            //если кратны, то обрабатываем
            if ( widthArea % widthPasteFr === 0 && heightArea % heightPasteFr === 0 ) {
                isMultiple = true;
            }
            else if ( firstCell.hasMerged() !== null )//в противном случае ошибка
            {
                if ( isCheckSelection ) {
                    return arn;
                }
                else {
                    this.handlers.trigger( "onError", c_oAscError.ID.PastInMergeAreaError, c_oAscError.Level.NoCritical );
                    return;
                }
            }
        }
        else {
            //проверка на наличие части объединённой ячейки в области куда осуществляем вставку
            for ( var rFirst = arn.r1; rFirst < rMax; ++rFirst ) {
                for ( var cFirst = arn.c1; cFirst < cMax; ++cFirst ) {
                    range = t.model.getRange3( rFirst, cFirst, rFirst, cFirst );
                    var merged = range.hasMerged();
                    if ( merged ) {
                        if ( merged.r1 < arn.r1 || merged.r2 > rMax - 1 || merged.c1 < arn.c1 || merged.c2 > cMax - 1 ) {
                            //ошибка в случае если вставка происходит в часть объедененной ячейки
                            if ( isCheckSelection ) {
                                return arn;
                            }
                            else {
                                this.handlers.trigger( "onErrorEvent", c_oAscError.ID.PastInMergeAreaError, c_oAscError.Level.NoCritical );
                                return;
                            }
                        }
                    }
                }
            }
        }

        var rangeUnMerge = t.model.getRange3( arn.r1, arn.c1, rMax - 1, cMax - 1 );

        var rMax2 = rMax;
        var cMax2 = cMax;
        //var rMax = values.length;
        var trueArn = t.activeRange;
        if ( isCheckSelection ) {
            var newArr = arn.clone( true );
            newArr.r2 = rMax2 - 1;
            newArr.c2 = cMax2 - 1;
            if ( isMultiple || isOneMerge ) {
                newArr.r2 = trueArn.r2;
                newArr.c2 = trueArn.c2;
            }
            return newArr;
        }
        //если не возникает конфликт, делаем unmerge
        rangeUnMerge.unmerge();
        if ( !isOneMerge ) {
            arn.r2 = rMax2 - 1;
            arn.c2 = cMax2 - 1;
        }

        var mergeArr = [];

        var n = 0;
        if ( isMultiple )//случай автозаполнения сложных форм
        {
            t.model.getRange3( trueArn.r1, trueArn.c1, trueArn.r2, trueArn.c2 ).unmerge();
            var maxARow = heightArea / heightPasteFr;
            var maxACol = widthArea / widthPasteFr;
            var plRow = (rMax2 - arn.r1);
            var plCol = (arn.c2 - arn.c1) + 1;
        }
        else {
            var maxARow = 1;
            var maxACol = 1;
            var plRow = 0;
            var plCol = 0;
        }

        var newVal;
        var curMerge;
        var nRow, nCol;
        for ( var autoR = 0; autoR < maxARow; ++autoR ) {
            for ( var autoC = 0; autoC < maxACol; ++autoC ) {
                for ( var r = arn.r1; r < rMax; ++r ) {
                    for ( var c = arn.c1; c < cMax; ++c ) {
                        var pasteRow = r - arn.r1 + activeCellsPasteFragment.r1;
                        var pasteCol = c - arn.c1 + activeCellsPasteFragment.c1;
                        newVal = val.getCell3( pasteRow, pasteCol );

                        curMerge = newVal.hasMerged();

                        if ( undefined !== newVal ) {
                            var isMerged = false, mergeCheck;

                            nRow = r + autoR * plRow;
                            if ( nRow > gc_nMaxRow0 ) {
                                nRow = gc_nMaxRow0;
                            }
                            nCol = c + autoC * plCol;
                            if ( nCol > gc_nMaxCol0 ) {
                                nCol = gc_nMaxCol0;
                            }

                            var range = t.model.getRange3( nRow, nCol, nRow, nCol );
                            //range может далее изменится в связи с наличием мерженных ячеек, firstRange  - не меняется(ему делаем setValue, как первой ячейке в диапазоне мерженных)
                            var firstRange = range.clone();

                            //****paste comments****
                            if ( val.aComments && val.aComments.length ) {
                                var comment;
                                for ( var i = 0; i < val.aComments.length; i++ ) {
                                    comment = val.aComments[i];
                                    if ( comment.nCol == pasteCol && comment.nRow == pasteRow ) {
                                        var commentData = new asc_CCommentData( comment );
                                        //change nRow, nCol
                                        commentData.asc_putCol( c + autoC * plCol );
                                        commentData.asc_putRow( r + autoR * plRow );
                                        t.cellCommentator.addComment( commentData, true );
                                    }
                                }
                            }

                            if ( !isOneMerge ) {
                                for ( mergeCheck = 0; mergeCheck < mergeArr.length; ++mergeCheck ) {
                                    if ( r + autoR * plRow <= mergeArr[mergeCheck].r2 && r + autoR * plRow >= mergeArr[mergeCheck].r1 && c + autoC * plCol <= mergeArr[mergeCheck].c2 && c + autoC * plCol >= mergeArr[mergeCheck].c1 ) {
                                        isMerged = true;
                                    }
                                }
                                if ( curMerge != null && !isMerged ) {
                                    var offsetCol = curMerge.c2 - curMerge.c1;
                                    if ( offsetCol + c + autoC * plCol >= gc_nMaxCol0 ) {
                                        offsetCol = gc_nMaxCol0 - (c + autoC * plCol);
                                    }

                                    var offsetRow = curMerge.r2 - curMerge.r1;
                                    if ( offsetRow + r + autoR * plRow >= gc_nMaxRow0 ) {
                                        offsetRow = gc_nMaxRow0 - (r + autoR * plRow);
                                    }

                                    range.setOffsetLast( {offsetCol: offsetCol, offsetRow: offsetRow} );
                                    range.merge( c_oAscMergeOptions.Merge );
                                    mergeArr[n] = {
                                        r1: curMerge.r1 + arn.r1 - activeCellsPasteFragment.r1 + autoR * plRow,
                                        r2: curMerge.r2 + arn.r1 - activeCellsPasteFragment.r1 + autoR * plRow,
                                        c1: curMerge.c1 + arn.c1 - activeCellsPasteFragment.c1 + autoC * plCol,
                                        c2: curMerge.c2 + arn.c1 - activeCellsPasteFragment.c1 + autoC * plCol
                                    };
                                    n++;
                                }
                            }
                            else {
                                for ( mergeCheck = 0; mergeCheck < mergeArr.length; ++mergeCheck ) {
                                    if ( r + autoR * plRow <= mergeArr[mergeCheck].r2 && r + autoR * plRow >= mergeArr[mergeCheck].r1 && c + autoC * plCol <= mergeArr[mergeCheck].c2 && c + autoC * plCol >= mergeArr[mergeCheck].c1 ) {
                                        isMerged = true;
                                    }
                                }
                                if ( !isMerged ) {
                                    range.setOffsetLast( {
                                        offsetCol: (isMergedFirstCell.c2 - isMergedFirstCell.c1),
                                        offsetRow: (isMergedFirstCell.r2 - isMergedFirstCell.r1)
                                    } );
                                    range.merge( c_oAscMergeOptions.Merge );
                                    mergeArr[n] = {
                                        r1: isMergedFirstCell.r1,
                                        r2: isMergedFirstCell.r2,
                                        c1: isMergedFirstCell.c1,
                                        c2: isMergedFirstCell.c2
                                    };
                                    n++;
                                }
                            }

                            //set style
                            var cellStyle = newVal.getStyleName();
                            if ( cellStyle && !isOneMerge ) {
                                range.setCellStyle( cellStyle );
                            }

                            //add formula
                            var numFormula = null;
                            var skipFormat = null;
                            var noSkipVal = null;
                            var value2 = newVal.getValue2();
                            for ( var nF = 0; nF < value2.length; nF++ ) {
                                if ( value2[nF] && value2[nF].sId ) {
                                    numFormula = nF;
                                    break;
                                }
                                else if ( value2[nF] && value2[nF].format && value2[nF].format.skip ) {
                                    skipFormat = true;
                                }
                                else if ( value2[nF] && value2[nF].format && !value2[nF].format.skip ) {
                                    noSkipVal = nF;
                                }
                            }


                            if ( !isOneMerge )//settings for cell(format)
                            {
                                //format
                                var numFormat = newVal.getNumFormat();
                                var nameFormat;
                                if ( numFormat && numFormat.sFormat ) {
                                    nameFormat = numFormat.sFormat;
                                }
                                if ( nameFormat ) {
                                    range.setNumFormat( nameFormat );
                                }
                            }

                            //TODO вместо range где возможно использовать cell
                            var cellFrom, cellTo;
                            if ( value2.length == 1 || numFormula != null || (skipFormat != null && noSkipVal != null) ) {
                                if ( numFormula == null ) {
                                    numFormula = 0;
                                }
                                var numStyle = 0;
                                if ( skipFormat != null && noSkipVal != null ) {
                                    numStyle = noSkipVal;
                                }

                                //formula
                                if ( newVal.getFormula() && !isOneMerge ) {
                                    var offset = range.getCells()[numFormula].getOffset2( value2[numFormula].sId ), assemb, _p_ = new parserFormula( value2[numFormula].sFormula, "", range.worksheet );

                                    if ( _p_.parse() ) {
                                        assemb = _p_.changeOffset( offset ).assemble();

                                        arrFormula[numFor] = {};
                                        arrFormula[numFor].range = range;
                                        arrFormula[numFor].val = "=" + assemb;
                                        numFor++;
                                    }
                                }
                                else {
                                    cellFrom = newVal.getCells();
                                    if ( isOneMerge && range && range.bbox ) {
                                        cellTo = this._getCell( range.bbox.c1, range.bbox.r1 ).getCells();
                                    }
                                    else {
                                        cellTo = firstRange.getCells();
                                    }

                                    if ( cellFrom && cellTo && cellFrom[0] && cellTo[0] ) {
                                        cellTo[0].setValueData( cellFrom[0].getValueData() );
                                    }
                                }

                                if ( !isOneMerge )//settings for text
                                {
                                    range.setBold( value2[numStyle].format.b );
                                    range.setItalic( value2[numStyle].format.i );
                                    range.setStrikeout( value2[numStyle].format.s );
                                    if ( value2[numStyle].format && null != value2[numStyle].format.c ) {
                                        range.setFontcolor( value2[numStyle].format.c );
                                    }
                                    range.setUnderline( value2[numStyle].format.u );
                                    range.setFontname( value2[numStyle].format.fn );
                                    range.setFontsize( value2[numStyle].format.fs );
                                }
                            }
                            else {
                                firstRange.setValue2( value2 );
                            }


                            if ( !isOneMerge )//settings for cell
                            {
                                //vertical align
                                range.setAlignVertical( newVal.getAlignVertical() );

                                //horizontal align
                                range.setAlignHorizontal( newVal.getAlignHorizontal() );

                                //borders
                                var fullBorders = newVal.getBorderFull();
                                if ( range.bbox.c2 !== range.bbox.c1 && curMerge && fullBorders ) {
                                    //для мерженных ячеек, правая границу
                                    var endMergeCell = val.getCell3( pasteRow, curMerge.c2 );
                                    var fullBordersEndMergeCell = endMergeCell.getBorderFull();
                                    if ( fullBordersEndMergeCell && fullBordersEndMergeCell.r ) {
                                        fullBorders.r = fullBordersEndMergeCell.r;
                                    }
                                }
                                range.setBorder( fullBorders );


                                //fill
                                range.setFill( newVal.getFill() );

                                //wrap
                                range.setWrap( newVal.getWrap() );

                                //angle
                                range.setAngle( newVal.getAngle() );

                                //hyperLink
                                var hyperLink = newVal.getHyperlink();
                                if ( hyperLink != null ) {
                                    hyperLink.Ref = range;
                                    range.setHyperlink( hyperLink );
                                }
                            }
                            //если замержили range
                            c = range.bbox.c2 - autoC * plCol;
                            if ( c === cMax ) {
                                r = range.bbox.r2 - autoC * plCol;
                            }
                        }
                    }
                }
            }
        }
        if ( isMultiple ) {
            arn.r2 = trueArn.r2;
            arn.c2 = trueArn.c2;
        }

        t.isChanged = true;
        t.activeRange.c2 = arn.c2;
        t.activeRange.r2 = arn.r2;
        var arnFor = [];
        arnFor[0] = arn;
        arnFor[1] = arrFormula;
        return arnFor;
    };

    // Залочена ли панель для закрепления
    WorksheetView.prototype._isLockedFrozenPane = function ( callback ) {
        var sheetId = this.model.getId();
        var lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Object, null, sheetId, c_oAscLockNameFrozenPane );

        if ( false === this.collaborativeEditing.getCollaborativeEditing() ) {
            // Пользователь редактирует один: не ждем ответа, а сразу продолжаем редактирование
            asc_applyFunction( callback, true );
            callback = undefined;
        }
        if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeMine, /*bCheckOnlyLockAll*/false ) ) {
            // Редактируем сами
            asc_applyFunction( callback, true );
            return;
        }
        else if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/false ) ) {
            // Уже ячейку кто-то редактирует
            asc_applyFunction( callback, false );
            return;
        }

        this.collaborativeEditing.onStartCheckLock();
        this.collaborativeEditing.addCheckLock( lockInfo );
        this.collaborativeEditing.onEndCheckLock( callback );
    };

    WorksheetView.prototype._isLockedDefNames = function ( callback, defNameId ) {
        var sheetId = this.model.getId();
        var lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Object, null/*c_oAscLockTypeElemSubType.DefinedNames*/, -1, defNameId );

        if ( false === this.collaborativeEditing.getCollaborativeEditing() ) {
            // Пользователь редактирует один: не ждем ответа, а сразу продолжаем редактирование
            asc_applyFunction( callback, true );
            callback = undefined;
        }
        if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeMine, /*bCheckOnlyLockAll*/false ) ) {
            // Редактируем сами
            asc_applyFunction( callback, true );
            return;
        }
        else if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/false ) ) {
            // Уже ячейку кто-то редактирует
            asc_applyFunction( callback, false );
            return;
        }

        this.collaborativeEditing.onStartCheckLock();
        this.collaborativeEditing.addCheckLock( lockInfo );
        this.collaborativeEditing.onEndCheckLock( callback );
    };

    // Залочен ли весь лист
    WorksheetView.prototype._isLockedAll = function ( callback ) {
        var sheetId = this.model.getId();
        var subType = c_oAscLockTypeElemSubType.ChangeProperties;
        var ar = this.activeRange;

        var lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Range, /*subType*/subType, sheetId, new asc.asc_CCollaborativeRange( ar.c1, ar.r1, ar.c2, ar.r2 ) );

        if ( false === this.collaborativeEditing.getCollaborativeEditing() ) {
            // Пользователь редактирует один: не ждем ответа, а сразу продолжаем редактирование
            asc_applyFunction( callback, true );
            callback = undefined;
        }
        if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeMine, /*bCheckOnlyLockAll*/true ) ) {
            // Редактируем сами
            asc_applyFunction( callback, true );
            return;
        }
        else if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/true ) ) {
            // Уже ячейку кто-то редактирует
            asc_applyFunction( callback, false );
            return;
        }

        this.collaborativeEditing.onStartCheckLock();
        this.collaborativeEditing.addCheckLock( lockInfo );
        this.collaborativeEditing.onEndCheckLock( callback );
    };
    // Пересчет для входящих ячеек в добавленные строки/столбцы
    WorksheetView.prototype._recalcRangeByInsertRowsAndColumns = function ( sheetId, ar ) {
        var isIntersection = false, isIntersectionC1 = true, isIntersectionC2 = true, isIntersectionR1 = true, isIntersectionR2 = true;
        do {
            if ( isIntersectionC1 && this.collaborativeEditing.isIntersectionInCols( sheetId, ar.c1 ) ) {
                ar.c1 += 1;
            }
            else {
                isIntersectionC1 = false;
            }

            if ( isIntersectionR1 && this.collaborativeEditing.isIntersectionInRows( sheetId, ar.r1 ) ) {
                ar.r1 += 1;
            }
            else {
                isIntersectionR1 = false;
            }

            if ( isIntersectionC2 && this.collaborativeEditing.isIntersectionInCols( sheetId, ar.c2 ) ) {
                ar.c2 -= 1;
            }
            else {
                isIntersectionC2 = false;
            }

            if ( isIntersectionR2 && this.collaborativeEditing.isIntersectionInRows( sheetId, ar.r2 ) ) {
                ar.r2 -= 1;
            }
            else {
                isIntersectionR2 = false;
            }


            if ( ar.c1 > ar.c2 || ar.r1 > ar.r2 ) {
                isIntersection = true;
                break;
            }
        } while ( isIntersectionC1 || isIntersectionC2 || isIntersectionR1 || isIntersectionR2 )
            ;

        if ( false === isIntersection ) {
            ar.c1 = this.collaborativeEditing.getLockMeColumn( sheetId, ar.c1 );
            ar.c2 = this.collaborativeEditing.getLockMeColumn( sheetId, ar.c2 );
            ar.r1 = this.collaborativeEditing.getLockMeRow( sheetId, ar.r1 );
            ar.r2 = this.collaborativeEditing.getLockMeRow( sheetId, ar.r2 );
        }

        return isIntersection;
    };
    // Функция проверки lock (возвращаемый результат нельзя использовать в качестве ответа, он нужен только для редактирования ячейки)
    WorksheetView.prototype._isLockedCells = function ( range, subType, callback ) {
        var sheetId = this.model.getId();
        var isIntersection = false;
        var newCallback = callback;
        var t = this;

        this.collaborativeEditing.onStartCheckLock();
        var isArrayRange = Array.isArray( range );
        var nLength = isArrayRange ? range.length : 1;
        var nIndex = 0;
        var ar = null;

        for ( ; nIndex < nLength; ++nIndex ) {
            ar = isArrayRange ? range[nIndex].clone( true ) : range.clone( true );

            if ( c_oAscLockTypeElemSubType.InsertColumns !== subType && c_oAscLockTypeElemSubType.InsertRows !== subType ) {
                // Пересчет для входящих ячеек в добавленные строки/столбцы
                isIntersection = this._recalcRangeByInsertRowsAndColumns( sheetId, ar );
            }

            if ( false === isIntersection ) {
                var lockInfo = this.collaborativeEditing.getLockInfo( c_oAscLockTypeElem.Range, /*subType*/subType, sheetId, new asc.asc_CCollaborativeRange( ar.c1, ar.r1, ar.c2, ar.r2 ) );

                if ( false !== this.collaborativeEditing.getLockIntersection( lockInfo, c_oAscLockTypes.kLockTypeOther, /*bCheckOnlyLockAll*/false ) ) {
                    // Уже ячейку кто-то редактирует
                    asc_applyFunction( callback, false );
                    return false;
                }
                else {
                    if ( c_oAscLockTypeElemSubType.InsertColumns === subType ) {
                        newCallback = function ( isSuccess ) {
                            if ( isSuccess ) {
                                t.collaborativeEditing.addColsRange( sheetId, range.clone( true ) );
                                t.collaborativeEditing.addCols( sheetId, range.c1, range.c2 - range.c1 + 1 );
                            }
                            callback( isSuccess );
                        };
                    }
                    else if ( c_oAscLockTypeElemSubType.InsertRows === subType ) {
                        newCallback = function ( isSuccess ) {
                            if ( isSuccess ) {
                                t.collaborativeEditing.addRowsRange( sheetId, range.clone( true ) );
                                t.collaborativeEditing.addRows( sheetId, range.r1, range.r2 - range.r1 + 1 );
                            }
                            callback( isSuccess );
                        };
                    }
                    else if ( c_oAscLockTypeElemSubType.DeleteColumns === subType ) {
                        newCallback = function ( isSuccess ) {
                            if ( isSuccess ) {
                                t.collaborativeEditing.removeColsRange( sheetId, range.clone( true ) );
                                t.collaborativeEditing.removeCols( sheetId, range.c1, range.c2 - range.c1 + 1 );
                            }
                            callback( isSuccess );
                        };
                    }
                    else if ( c_oAscLockTypeElemSubType.DeleteRows === subType ) {
                        newCallback = function ( isSuccess ) {
                            if ( isSuccess ) {
                                t.collaborativeEditing.removeRowsRange( sheetId, range.clone( true ) );
                                t.collaborativeEditing.removeRows( sheetId, range.r1, range.r2 - range.r1 + 1 );
                            }
                            callback( isSuccess );
                        };
                    }
                    this.collaborativeEditing.addCheckLock( lockInfo );
                }
            }
            else {
                if ( c_oAscLockTypeElemSubType.InsertColumns === subType ) {
                    t.collaborativeEditing.addColsRange( sheetId, range.clone( true ) );
                    t.collaborativeEditing.addCols( sheetId, range.c1, range.c2 - range.c1 + 1 );
                }
                else if ( c_oAscLockTypeElemSubType.InsertRows === subType ) {
                    t.collaborativeEditing.addRowsRange( sheetId, range.clone( true ) );
                    t.collaborativeEditing.addRows( sheetId, range.r1, range.r2 - range.r1 + 1 );
                }
                else if ( c_oAscLockTypeElemSubType.DeleteColumns === subType ) {
                    t.collaborativeEditing.removeColsRange( sheetId, range.clone( true ) );
                    t.collaborativeEditing.removeCols( sheetId, range.c1, range.c2 - range.c1 + 1 );
                }
                else if ( c_oAscLockTypeElemSubType.DeleteRows === subType ) {
                    t.collaborativeEditing.removeRowsRange( sheetId, range.clone( true ) );
                    t.collaborativeEditing.removeRows( sheetId, range.r1, range.r2 - range.r1 + 1 );
                }
            }
        }

        if ( false === this.collaborativeEditing.getCollaborativeEditing() ) {
            // Пользователь редактирует один: не ждем ответа, а сразу продолжаем редактирование
            newCallback( true );
            newCallback = undefined;
        }
        this.collaborativeEditing.onEndCheckLock( newCallback );
        return true;
    };

    WorksheetView.prototype.changeWorksheet = function ( prop, val ) {
        // Проверка глобального лока
        if ( this.collaborativeEditing.getGlobalLock() ) {
            return;
        }

        var t = this;
        var arn = t.activeRange.clone( true );
        var checkRange = arn.getAllRange();

        var range;
        var fullRecalc = false;
        var reinitRanges = false;
        var updateDrawingObjectsInfo = null;
        var cw;
        var isUpdateCols = false, isUpdateRows = false;
        var isCheckChangeAutoFilter;
        var functionModelAction = null;
        var lockDraw = false;	// Параметр, при котором не будет отрисовки (т.к. мы просто обновляем информацию на неактивном листе)
        var oChangeData = new CChangeTableData( null, null, null, null ); // Обновление для диаграмм

        var onChangeWorksheetCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            asc_applyFunction( functionModelAction );

            t._initCellsArea( fullRecalc );
            if ( fullRecalc ) {
                t.cache.reset();
            }
            t._cleanCellsTextMetricsCache();
            t._prepareCellTextMetricsCache();
            if ( reinitRanges && t.objectRender && t.objectRender.drawingArea ) {
                t.objectRender.drawingArea.reinitRanges();
            }
            if ( null !== updateDrawingObjectsInfo && t.objectRender ) {
                t.objectRender.updateSizeDrawingObjects( updateDrawingObjectsInfo );
            }
            t.objectRender.rebuildChartGraphicObjects( oChangeData );
            t.draw( lockDraw );

            t.handlers.trigger( "reinitializeScroll" );

            if ( isUpdateCols ) {
                t._updateVisibleColsCount();
            }
            if ( isUpdateRows ) {
                t._updateVisibleRowsCount();
            }

            t.handlers.trigger( "selectionChanged", t.getSelectionInfo() );
            t.handlers.trigger( "selectionMathInfoChanged", t.getSelectionMathInfo() );
        };

        switch ( prop ) {
            case "colWidth":
                functionModelAction = function () {
                    cw = t._charCountToModelColWidth( val );
                    t.model.setColWidth( cw, checkRange.c1, checkRange.c2 );
                    isUpdateCols = true;
                    fullRecalc = true;
                    reinitRanges = true;
                    updateDrawingObjectsInfo = {target: c_oTargetType.ColumnResize, col: checkRange.c1};
                };
                this._isLockedAll( onChangeWorksheetCallback );
                break;
            case "showCols":
                functionModelAction = function () {
                    t.model.setColHidden( /*bHidden*/false, arn.c1, arn.c2 );
                    fullRecalc = true;
                    reinitRanges = true;
                    updateDrawingObjectsInfo = {target: c_oTargetType.ColumnResize, col: arn.c1};
                };
                oChangeData.hided = new asc_Range( arn.c1, 0, arn.c2, gc_nMaxRow0 );
                this._isLockedAll( onChangeWorksheetCallback );
                break;
            case "hideCols":
                functionModelAction = function () {
                    t.model.setColHidden( /*bHidden*/true, arn.c1, arn.c2 );
                    fullRecalc = true;
                    reinitRanges = true;
                    updateDrawingObjectsInfo = {target: c_oTargetType.ColumnResize, col: arn.c1};
                };
                oChangeData.hided = new asc_Range( arn.c1, 0, arn.c2, gc_nMaxRow0 );
                this._isLockedAll( onChangeWorksheetCallback );
                break;
            case "rowHeight":
                functionModelAction = function () {
                    // Приводим к px (чтобы было ровно)
                    val = val / 0.75;
                    val = (val | val) * 0.75;		// 0.75 - это размер 1px в pt (можно было 96/72)
                    t.model.setRowHeight( Math.min( val, t.maxRowHeight ), checkRange.r1, checkRange.r2, true );
                    isUpdateRows = true;
                    fullRecalc = true;
                    reinitRanges = true;
                    updateDrawingObjectsInfo = {target: c_oTargetType.RowResize, row: checkRange.r1};
                };
                return this._isLockedAll( onChangeWorksheetCallback );
            case "showRows":
                functionModelAction = function () {
                    t.model.setRowHidden( /*bHidden*/false, arn.r1, arn.r2 );
                    t.autoFilters.reDrawFilter( arn );
                    fullRecalc = true;
                    reinitRanges = true;
                    updateDrawingObjectsInfo = {target: c_oTargetType.RowResize, row: arn.r1};
                };
                oChangeData.hided = new asc_Range( 0, arn.r1, gc_nMaxCol0, arn.r2 );
                this._isLockedAll( onChangeWorksheetCallback );
                break;
            case "hideRows":
                functionModelAction = function () {
                    t.model.setRowHidden( /*bHidden*/true, arn.r1, arn.r2 );
                    t.autoFilters.reDrawFilter( arn );
                    fullRecalc = true;
                    reinitRanges = true;
                    updateDrawingObjectsInfo = {target: c_oTargetType.RowResize, row: arn.r1};
                };
                oChangeData.hided = new asc_Range( 0, arn.r1, gc_nMaxCol0, arn.r2 );
                this._isLockedAll( onChangeWorksheetCallback );
                break;
            case "insCell":
                range = t.model.getRange3( arn.r1, arn.c1, arn.r2, arn.c2 );
                switch ( val ) {
                    case c_oAscInsertOptions.InsertCellsAndShiftRight:
                        isCheckChangeAutoFilter = t.autoFilters.isActiveCellsCrossHalfFTable( arn, c_oAscInsertOptions.InsertCellsAndShiftRight, prop );
                        if ( isCheckChangeAutoFilter === false ) {
                            return;
                        }

                        functionModelAction = function () {
                            History.Create_NewPoint();
                            History.StartTransaction();
                            if ( range.addCellsShiftRight() ) {
                                fullRecalc = true;
                                reinitRanges = true;
                                if ( isCheckChangeAutoFilter === true ) {
                                    t.autoFilters.insertColumn( prop, arn, c_oAscInsertOptions.InsertCellsAndShiftRight );
                                }
                                t.cellCommentator.updateCommentsDependencies( true, val, arn );
                                t.objectRender.updateDrawingObject( true, val, arn );
                            }
                            History.EndTransaction();
                        };

                        oChangeData.changedRange = new asc_Range( arn.c1, arn.r1, gc_nMaxCol0, arn.r2 );
                        this._isLockedCells( oChangeData.changedRange, null, onChangeWorksheetCallback );
                        break;
                    case c_oAscInsertOptions.InsertCellsAndShiftDown:
                        isCheckChangeAutoFilter = t.autoFilters.isActiveCellsCrossHalfFTable( arn, c_oAscInsertOptions.InsertCellsAndShiftDown, prop );
                        if ( isCheckChangeAutoFilter === false ) {
                            return;
                        }

                        functionModelAction = function () {
                            History.Create_NewPoint();
                            History.StartTransaction();
                            if ( range.addCellsShiftBottom() ) {
                                fullRecalc = true;
                                reinitRanges = true;
                                if ( isCheckChangeAutoFilter === true ) {
                                    t.autoFilters.insertRows( prop, arn, c_oAscInsertOptions.InsertCellsAndShiftDown );
                                }
                                t.cellCommentator.updateCommentsDependencies( true, val, arn );
                                t.objectRender.updateDrawingObject( true, val, arn );
                            }
                            History.EndTransaction();
                        };

                        oChangeData.changedRange = new asc_Range( arn.c1, arn.r1, arn.c2, gc_nMaxRow0 );
                        this._isLockedCells( oChangeData.changedRange, null, onChangeWorksheetCallback );
                        break;
                    case c_oAscInsertOptions.InsertColumns:
                        isCheckChangeAutoFilter = t.autoFilters.isRangeIntersectionSeveralTableParts( arn, true );
                        if ( isCheckChangeAutoFilter === true ) {
                            return;
                        }

                        functionModelAction = function () {
                            History.Create_NewPoint();
                            History.StartTransaction();
                            fullRecalc = true;
                            reinitRanges = true;
                            t.model.insertColsBefore( arn.c1, arn.c2 - arn.c1 + 1 );
                            t.autoFilters.insertColumn( prop, arn, c_oAscInsertOptions.InsertColumns );
                            t.objectRender.updateDrawingObject( true, val, arn );
                            t.cellCommentator.updateCommentsDependencies( true, val, arn );
                            History.EndTransaction();
                        };

                        oChangeData.added = new asc_Range( arn.c1, 0, arn.c2, gc_nMaxRow0 );
                        this._isLockedCells( oChangeData.added, c_oAscLockTypeElemSubType.InsertColumns, onChangeWorksheetCallback );
                        break;
                    case c_oAscInsertOptions.InsertRows:
                        functionModelAction = function () {
                            fullRecalc = true;
                            reinitRanges = true;
                            t.model.insertRowsBefore( arn.r1, arn.r2 - arn.r1 + 1 );
                            t.autoFilters.insertRows( prop, arn, c_oAscInsertOptions.InsertRows );
                            t.objectRender.updateDrawingObject( true, val, arn );
                            t.cellCommentator.updateCommentsDependencies( true, val, arn );
                        };

                        oChangeData.added = new asc_Range( 0, arn.r1, gc_nMaxCol0, arn.r2 );
                        this._isLockedCells( oChangeData.added, c_oAscLockTypeElemSubType.InsertRows, onChangeWorksheetCallback );
                        break;
                }
                break;
            case "delCell":
                range = t.model.getRange3( checkRange.r1, checkRange.c1, checkRange.r2, checkRange.c2 );
                switch ( val ) {
                    case c_oAscDeleteOptions.DeleteCellsAndShiftLeft:
                        isCheckChangeAutoFilter = t.autoFilters.isActiveCellsCrossHalfFTable( arn, c_oAscDeleteOptions.DeleteCellsAndShiftLeft, prop );
                        if ( isCheckChangeAutoFilter === false ) {
                            return;
                        }

                        functionModelAction = function () {
                            History.Create_NewPoint();
                            History.StartTransaction();
                            if ( isCheckChangeAutoFilter === true ) {
                                t.autoFilters.isEmptyAutoFilters( arn, c_oAscDeleteOptions.DeleteCellsAndShiftLeft );
                            }
                            if ( range.deleteCellsShiftLeft( function () {
                                t._cleanCache( oChangeData.changedRange );
                                t.cellCommentator.updateCommentsDependencies( false, val, checkRange );
                            } ) ) {
                                // ToDo проверить, может перенести в функцию prechange
                                if ( isCheckChangeAutoFilter === true ) {
                                    t.autoFilters.insertColumn( prop, checkRange, c_oAscDeleteOptions.DeleteCellsAndShiftLeft );
                                }
                                t.objectRender.updateDrawingObject( false, val, checkRange );
                            }
                            History.EndTransaction();
                            reinitRanges = true;
                        };

                        oChangeData.changedRange = new asc_Range( checkRange.c1, checkRange.r1, gc_nMaxCol0, checkRange.r2 );
                        this._isLockedCells( oChangeData.changedRange, null, onChangeWorksheetCallback );
                        break;
                    case c_oAscDeleteOptions.DeleteCellsAndShiftTop:
                        isCheckChangeAutoFilter = t.autoFilters.isActiveCellsCrossHalfFTable( arn, c_oAscDeleteOptions.DeleteCellsAndShiftTop, prop );
                        if ( isCheckChangeAutoFilter === false ) {
                            return;
                        }

                        functionModelAction = function () {
                            History.Create_NewPoint();
                            History.StartTransaction();
                            if ( isCheckChangeAutoFilter === true ) {
                                t.autoFilters.isEmptyAutoFilters( arn, c_oAscDeleteOptions.DeleteCellsAndShiftTop );
                            }
                            if ( range.deleteCellsShiftUp( function () {
                                t._cleanCache( oChangeData.changedRange );
                                t.cellCommentator.updateCommentsDependencies( false, val, checkRange );
                            } ) ) {
                                // ToDo проверить, может перенести в функцию prechange
                                if ( isCheckChangeAutoFilter === true ) {
                                    t.autoFilters.insertRows( prop, checkRange, c_oAscDeleteOptions.DeleteCellsAndShiftTop );
                                }
                                t.objectRender.updateDrawingObject( false, val, checkRange );
                            }
                            History.EndTransaction();

                            reinitRanges = true;
                        };

                        oChangeData.changedRange = new asc_Range( checkRange.c1, checkRange.r1, checkRange.c2, gc_nMaxRow0 );
                        this._isLockedCells( oChangeData.changedRange, null, onChangeWorksheetCallback );
                        break;
                    case c_oAscDeleteOptions.DeleteColumns:
                        isCheckChangeAutoFilter = t.autoFilters.isActiveCellsCrossHalfFTable( checkRange, c_oAscDeleteOptions.DeleteColumns, prop );
                        if ( isCheckChangeAutoFilter === false ) {
                            return;
                        }

                        functionModelAction = function () {
                            fullRecalc = true;
                            reinitRanges = true;
                            History.Create_NewPoint();
                            History.StartTransaction();
                            t.cellCommentator.updateCommentsDependencies( false, val, checkRange );
                            t.autoFilters.isEmptyAutoFilters( arn, c_oAscDeleteOptions.DeleteColumns );
                            t.model.removeCols( checkRange.c1, checkRange.c2 );
                            t.autoFilters.insertColumn( prop, arn, c_oAscDeleteOptions.DeleteColumns );
                            t.objectRender.updateDrawingObject( false, val, checkRange );
                            History.EndTransaction();
                        };

                        oChangeData.removed = new asc_Range( checkRange.c1, 0, checkRange.c2, gc_nMaxRow0 );
                        this._isLockedCells( oChangeData.removed, c_oAscLockTypeElemSubType.DeleteColumns, onChangeWorksheetCallback );
                        break;
                    case c_oAscDeleteOptions.DeleteRows:
                        isCheckChangeAutoFilter = t.autoFilters.isActiveCellsCrossHalfFTable( checkRange, c_oAscDeleteOptions.DeleteRows, prop );
                        if ( isCheckChangeAutoFilter === false ) {
                            return;
                        }

                        functionModelAction = function () {
                            fullRecalc = true;
                            reinitRanges = true;
                            History.Create_NewPoint();
                            History.StartTransaction();
                            t.cellCommentator.updateCommentsDependencies( false, val, checkRange );
                            t.autoFilters.isEmptyAutoFilters( arn, c_oAscDeleteOptions.DeleteRows );
                            t.model.removeRows( checkRange.r1, checkRange.r2 );
                            t.autoFilters.insertRows( prop, checkRange, c_oAscDeleteOptions.DeleteRows );
                            t.objectRender.updateDrawingObject( false, val, checkRange );
                            History.EndTransaction();
                        };

                        oChangeData.removed = new asc_Range( 0, checkRange.r1, gc_nMaxCol0, checkRange.r2 );
                        this._isLockedCells( oChangeData.removed, c_oAscLockTypeElemSubType.DeleteRows, onChangeWorksheetCallback );
                        break;
                }
                this.handlers.trigger( "selectionNameChanged", t.getSelectionName( /*bRangeText*/false ) );
                break;
            case "sheetViewSettings":
                functionModelAction = function () {
                    t.model.setSheetViewSettings( val );

                    isUpdateCols = true;
                    isUpdateRows = true;
                    fullRecalc = true;
                    reinitRanges = true;
                };

                this._isLockedAll( onChangeWorksheetCallback );
                break;
            case "update":
                if ( val !== undefined ) {
                    fullRecalc = !!val.fullRecalc;
                    lockDraw = true === val.lockDraw;
                    reinitRanges = !!val.reinitRanges;
                }
                onChangeWorksheetCallback( true );
                break;
        }
    };

    WorksheetView.prototype.expandColsOnScroll = function ( isNotActive, updateColsCount, newColsCount ) {
        var t = this;
        var arn;
        var bIsMaxCols = false;
        var obr = this.objectRender ? this.objectRender.getDrawingAreaMetrics() : {maxCol: 0, maxRow: 0};
        var maxc = Math.max( this.model.getColsCount() + 1, this.cols.length, obr.maxCol );
        if ( newColsCount ) {
            maxc = Math.max( maxc, newColsCount );
        }

        // Сохраняем старое значение
        var nLastCols = this.nColsCount;
        if ( isNotActive ) {
            this.nColsCount = maxc + 1;
        }
        else if ( updateColsCount ) {
            this.nColsCount = maxc;
            if ( this.cols.length < this.nColsCount ) {
                nLastCols = this.cols.length;
            }
        }
        else {
            arn = t.activeRange.clone( true );
            if ( arn.c2 >= t.cols.length - 1 ) {
                this.nColsCount = maxc;
                if ( arn.c2 >= this.nColsCount - 1 ) {
                    this.nColsCount = arn.c2 + 2;
                }
            }
        }
        // Проверяем ограничения по столбцам
        if ( gc_nMaxCol < this.nColsCount ) {
            this.nColsCount = gc_nMaxCol;
            bIsMaxCols = true;
        }

        t._calcWidthColumns( /*fullRecalc*/2 );

        if ( this.objectRender && this.objectRender.drawingArea ) {
            this.objectRender.drawingArea.reinitRanges();
        }
        return (nLastCols !== this.nColsCount || bIsMaxCols);
    };

    WorksheetView.prototype.expandRowsOnScroll = function ( isNotActive, updateRowsCount, newRowsCount ) {
        var t = this;
        var arn;
        var bIsMaxRows = false;
        var obr = this.objectRender ? this.objectRender.getDrawingAreaMetrics() : {maxCol: 0, maxRow: 0};
        var maxr = Math.max( this.model.getRowsCount() + 1, this.rows.length, obr.maxRow );
        if ( newRowsCount ) {
            maxr = Math.max( maxr, newRowsCount );
        }

        // Сохраняем старое значение
        var nLastRows = this.nRowsCount;
        if ( isNotActive ) {
            this.nRowsCount = maxr + 1;
        }
        else if ( updateRowsCount ) {
            this.nRowsCount = maxr;
            if ( this.rows.length < this.nRowsCount ) {
                nLastRows = this.rows.length;
            }
        }
        else {
            arn = t.activeRange.clone( true );
            if ( arn.r2 >= t.rows.length - 1 ) {
                this.nRowsCount = maxr;
                if ( arn.r2 >= this.nRowsCount - 1 ) {
                    this.nRowsCount = arn.r2 + 2;
                }
            }
        }
        // Проверяем ограничения по строкам
        if ( gc_nMaxRow < this.nRowsCount ) {
            this.nRowsCount = gc_nMaxRow;
            bIsMaxRows = true;
        }

        t._calcHeightRows( /*fullRecalc*/2 );
        if ( this.objectRender && this.objectRender.drawingArea ) {
            this.objectRender.drawingArea.reinitRanges();
        }
        return (nLastRows !== this.nRowsCount || bIsMaxRows);
    };

    WorksheetView.prototype.onChangeWidthCallback = function ( col, r1, r2, onlyIfMore ) {
        var width = null;
        var row, ct, c, fl, str, maxW, tm, mc, isMerged, oldWidth, oldColWidth;
        var lastHeight = null;
        var filterButton = null;
        if ( null == r1 ) {
            r1 = 0;
        }
        if ( null == r2 ) {
            r2 = this.rows.length - 1;
        }

        oldColWidth = this.cols[col].charCount;

        this.cols[col].isCustomWidth = false;
        for ( row = r1; row <= r2; ++row ) {
            // пересчет метрик текста
            this._addCellTextToCache( col, row, /*canChangeColWidth*/c_oAscCanChangeColWidth.all );
            ct = this._getCellTextCache( col, row );
            if ( ct === undefined ) {
                continue;
            }
            fl = ct.flags;
            isMerged = fl.isMerged();
            if ( isMerged ) {
                mc = fl.merged;
                // Для замерженных ячеек (с 2-мя или более колонками) оптимизировать не нужно
                if ( mc.c1 !== mc.c2 ) {
                    continue;
                }
            }

            if ( ct.metrics.height > this.maxRowHeight ) {
                if ( isMerged ) {
                    continue;
                }
                // Запоминаем старую ширину (в случае, если у нас по высоте не уберется)
                oldWidth = ct.metrics.width;
                lastHeight = null;
                // вычисление новой ширины столбца, чтобы высота текста была меньше maxRowHeight
                c = this._getCell( col, row );
                str = c.getValue2();
                maxW = ct.metrics.width + this.maxDigitWidth;
                while ( 1 ) {
                    tm = this._roundTextMetrics( this.stringRender.measureString( str, fl, maxW ) );
                    if ( tm.height <= this.maxRowHeight ) {
                        break;
                    }
                    if ( lastHeight === tm.height ) {
                        // Ситуация, когда у нас текст не уберется по высоте (http://bugzserver/show_bug.cgi?id=19974)
                        tm.width = oldWidth;
                        break;
                    }
                    lastHeight = tm.height;
                    maxW += this.maxDigitWidth;
                }
                width = Math.max( width, tm.width );
            }
            else {
                filterButton = this.autoFilters.getSizeButton( col, row );
                if ( null !== filterButton && CellValueType.String === ct.cellType ) {
                    width = Math.max( width, ct.metrics.width + filterButton.width );
                }
                else {
                    width = Math.max( width, ct.metrics.width );
                }
            }
        }

        var pad, cc, cw;
        if ( width > 0 ) {
            pad = this.width_padding * 2 + this.width_1px;
            cc = Math.min( this._colWidthToCharCount( width + pad ), c_oAscMaxColumnWidth );
            cw = this._charCountToModelColWidth( cc );
        }
        else {
            cw = gc_dDefaultColWidthCharsAttribute;
            cc = this.defaultColWidthChars;
        }

        if ( onlyIfMore && cc < oldColWidth ) {
            return -1;
        }

        History.Create_NewPoint();
        if ( !onlyIfMore ) {
            var oSelection = History.GetSelection();
            if ( null != oSelection ) {
                oSelection = oSelection.clone();
                oSelection.assign( col, 0, col, gc_nMaxRow0 );
                oSelection.type = c_oAscSelectionType.RangeCol;
                History.SetSelection( oSelection );
                History.SetSelectionRedo( oSelection );
            }
        }
        History.StartTransaction();
        // Выставляем, что это bestFit
        this.model.setColBestFit( true, cw, col, col );
        History.EndTransaction();
        return oldColWidth !== cc ? cw : -1;
    };

    WorksheetView.prototype.optimizeColWidth = function ( col ) {
        var t = this;
        return this._isLockedAll( function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }
            var w = t.onChangeWidthCallback( col, null, null );
            if ( -1 !== w ) {
                t.cols[col] = t._calcColWidth( w );
                t.cols[col].isCustomWidth = false;

                t._updateColumnPositions();
                t._updateVisibleColsCount();
                t._cleanCache( new asc_Range( col, 0, col, t.rows.length - 1 ) );
                t.changeWorksheet( "update" );
            }
        } );
    };

    WorksheetView.prototype.optimizeRowHeight = function ( row ) {
        var t = this;

        var onChangeHeightCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            var height = t.defaultRowHeight;
            var col, ct, mc;

            for ( col = 0; col < t.rows.length; ++col ) {
                ct = t._getCellTextCache( col, row );
                if ( ct === undefined ) {
                    continue;
                }
                if ( ct.flags.isMerged() ) {
                    mc = ct.flags.merged;
                    // Для замерженных ячеек (с 2-мя или более строками) оптимизировать не нужно
                    if ( mc.r1 !== mc.r2 ) {
                        continue;
                    }
                }

                height = Math.max( height, ct.metrics.height );
            }

            History.Create_NewPoint();
            var oSelection = History.GetSelection();
            if ( null != oSelection ) {
                oSelection = oSelection.clone();
                oSelection.assign( 0, row, gc_nMaxCol0, row );
                oSelection.type = c_oAscSelectionType.RangeRow;
                History.SetSelection( oSelection );
                History.SetSelectionRedo( oSelection );
            }
            History.StartTransaction();
            // Выставляем, что это bestFit
            t.model.setRowBestFit( true, Math.min( height + t.height_1px, t.maxRowHeight ), row, row );
            History.EndTransaction();

            t.nRowsCount = 0;
            t._calcHeightRows( /*fullRecalc*/0 );
            t._updateVisibleRowsCount();
            t._cleanCache( new asc_Range( 0, row, t.cols.length - 1, row ) );
            t.changeWorksheet( "update" );
        };
        return this._isLockedAll( onChangeHeightCallback );
    };


    // ----- Search -----

    WorksheetView.prototype._setActiveCell = function ( col, row ) {
        var ar = this.activeRange, sc = ar.startCol, sr = ar.startRow, offs;

        this.cleanSelection();

        ar.assign( col, row, col, row );
        ar.type = c_oAscSelectionType.RangeCells;
        ar.startCol = col;
        ar.startRow = row;

        this._fixSelectionOfMergedCells();
        this._fixSelectionOfHiddenCells();
        this._drawSelection();

        var x1 = this.getCellLeftRelative( this.activeRange.c1, /*pt*/0 ), y1 = this.getCellTopRelative( this.activeRange.r1, /*pt*/0 );

        offs = this._calcActiveRangeOffset( x1, y1 );
        if ( sc !== ar.startCol || sr !== ar.startRow ) {
            this.handlers.trigger( "selectionNameChanged", this.getSelectionName( /*bRangeText*/false ) );
            this.handlers.trigger( "selectionChanged", this.getSelectionInfo() );
        }
        return offs;
    };

    WorksheetView.prototype._isCellEqual = function ( c, r, options ) {
        var cell, cellText;
        // Не пользуемся RegExp, чтобы не возиться со спец.символами
        var mc = this.model.getMergedByCell( r, c );
        cell = mc ? this._getVisibleCell( mc.c1, mc.r1 ) : this._getVisibleCell( c, r );
        cellText = (options.lookIn === c_oAscFindLookIn.Formulas) ? cell.getValueForEdit() : cell.getValue();
        if ( true !== options.isMatchCase ) {
            cellText = cellText.toLowerCase();
        }
        if ( (cellText.indexOf( options.findWhat ) >= 0) && (true !== options.isWholeCell || options.findWhat.length === cellText.length) ) {
            return (mc ? new asc_Range( mc.c1, mc.r1, mc.c1, mc.r1 ) : new asc_Range( c, r, c, r ));
        }
        return null;
    };
    WorksheetView.prototype.findCellText = function ( options ) {
        var self = this;
        if ( true !== options.isMatchCase ) {
            options.findWhat = options.findWhat.toLowerCase();
        }
        var ar = options.activeRange ? options.activeRange : this.activeRange;
        var c = ar.startCol;
        var r = ar.startRow;
        var minC = 0;
        var minR = 0;
        var maxC = this.cols.length - 1;
        var maxR = this.rows.length - 1;
        var inc = options.scanForward ? +1 : -1;
        var isEqual;

        // ToDo стоит переделать это место, т.к. для поиска не нужны измерения, а нужен только сам текст (http://bugzserver/show_bug.cgi?id=26136)
        this._prepareCellTextMetricsCache( new Asc.Range( 0, 0, this.model.getColsCount(), this.model.getRowsCount() ) );

        function findNextCell() {
            var ct = undefined;
            do {
                if ( options.scanByRows ) {
                    c += inc;
                    if ( c < minC || c > maxC ) {
                        c = options.scanForward ? minC : maxC;
                        r += inc;
                    }
                }
                else {
                    r += inc;
                    if ( r < minR || r > maxR ) {
                        r = options.scanForward ? minR : maxR;
                        c += inc;
                    }
                }
                if ( c < minC || c > maxC || r < minR || r > maxR ) {
                    return undefined;
                }
                ct = self._getCellTextCache( c, r, true );
            } while ( !ct );
            return ct;
        }

        while ( findNextCell() ) {
            isEqual = this._isCellEqual( c, r, options );
            if ( null !== isEqual ) {
                return isEqual;
            }
        }

        // Продолжаем циклический поиск
        if ( options.scanForward ) {
            // Идем вперед с первой ячейки
            minC = 0;
            minR = 0;
            if ( options.scanByRows ) {
                c = -1;
                r = 0;

                maxC = this.cols.length - 1;
                maxR = ar.startRow;
            }
            else {
                c = 0;
                r = -1;

                maxC = ar.startCol;
                maxR = this.rows.length - 1;
            }
        }
        else {
            // Идем назад с последней
            c = this.cols.length - 1;
            r = this.rows.length - 1;
            if ( options.scanByRows ) {
                minC = 0;
                minR = ar.startRow;
            }
            else {
                minC = ar.startCol;
                minR = 0;
            }
            maxC = this.cols.length - 1;
            maxR = this.rows.length - 1;
        }
        while ( findNextCell() ) {
            isEqual = this._isCellEqual( c, r, options );
            if ( null !== isEqual ) {
                return isEqual;
            }
        }
        return null;
    };

    WorksheetView.prototype.replaceCellText = function ( options, lockDraw, callback ) {
        if ( true !== options.isMatchCase ) {
            options.findWhat = options.findWhat.toLowerCase();
        }

        // Очищаем результаты
        options.countFind = 0;
        options.countReplace = 0;

        var t = this;
        var ar = this.activeRange.clone();
        var aReplaceCells = [];
        if ( options.isReplaceAll ) {
            var aReplaceCellsIndex = {};
            options.activeRange = ar;
            var findResult, index;
            while ( true ) {
                findResult = t.findCellText( options );
                if ( null === findResult ) {
                    break;
                }
                index = findResult.c1 + '-' + findResult.r1;
                if ( aReplaceCellsIndex[index] ) {
                    break;
                }
                aReplaceCellsIndex[index] = true;
                aReplaceCells.push( findResult );
                ar.startCol = findResult.c1;
                ar.startRow = findResult.r1;
            }
        }
        else {
            // Попробуем сначала найти
            var isEqual = this._isCellEqual( ar.startCol, ar.startRow, options );
            if ( null === isEqual ) {
                return callback( options );
            }

            aReplaceCells.push( isEqual );
        }

        if ( 0 > aReplaceCells.length ) {
            return callback( options );
        }

        return this._replaceCellsText( aReplaceCells, options, lockDraw, callback );
    };

    WorksheetView.prototype._replaceCellsText = function ( aReplaceCells, options, lockDraw, callback ) {
        var t = this;
        var findFlags = "g"; // Заменяем все вхождения
        if ( true !== options.isMatchCase ) {
            findFlags += "i";
        } // Не чувствителен к регистру

        var valueForSearching = options.findWhat
            .replace( /(\\)/g, "\\\\" ).replace( /(\^)/g, "\\^" )
            .replace( /(\()/g, "\\(" ).replace( /(\))/g, "\\)" )
            .replace( /(\+)/g, "\\+" ).replace( /(\[)/g, "\\[" )
            .replace( /(\])/g, "\\]" ).replace( /(\{)/g, "\\{" )
            .replace( /(\})/g, "\\}" ).replace( /(\$)/g, "\\$" )
            .replace( /(~)?\*/g, function ( $0, $1 ) {
                return $1 ? $0 : '(.*)';
            } )
            .replace( /(~)?\?/g, function ( $0, $1 ) {
                return $1 ? $0 : '.';
            } )
            .replace( /(~\*)/g, "\\*" ).replace( /(~\?)/g, "\\?" );
        valueForSearching = new RegExp( valueForSearching, findFlags );

        options.indexInArray = 0;
        options.countFind = aReplaceCells.length;
        options.countReplace = 0;
        if ( options.isReplaceAll && false === this.collaborativeEditing.getCollaborativeEditing() ) {
            this._isLockedCells( aReplaceCells, /*subType*/null, function () {
                t._replaceCellText( aReplaceCells, valueForSearching, options, lockDraw, callback, true );
            } );
        }
        else {
            this._replaceCellText( aReplaceCells, valueForSearching, options, lockDraw, callback, false );
        }
    };

    WorksheetView.prototype._replaceCellText = function ( aReplaceCells, valueForSearching, options, lockDraw, callback, oneUser ) {
        var t = this;
        if ( options.indexInArray >= aReplaceCells.length ) {
            this.draw( lockDraw );
            return callback( options );
        }

        var onReplaceCallback = function ( isSuccess ) {
            var cell = aReplaceCells[options.indexInArray];
            ++options.indexInArray;
            if ( false !== isSuccess ) {
                ++options.countReplace;

                var c = t._getVisibleCell( cell.c1, cell.r1 );

                if ( c === undefined ) {
                    asc_debug( "log", "Unknown cell's info: col = " + cell.c1 + ", row = " + cell.r1 );
                }
                else {
                    var cellValue = c.getValueForEdit();
                    cellValue = cellValue.replace( valueForSearching, options.replaceWith );

                    var oCellEdit = new asc_Range( cell.c1, cell.r1, cell.c1, cell.r1 );
                    var v, newValue;
                    // get first fragment and change its text
                    v = c.getValueForEdit2().slice( 0, 1 );
                    // Создаем новый массив, т.к. getValueForEdit2 возвращает ссылку
                    newValue = [];
                    newValue[0] = new Fragment( {text: cellValue, format: v[0].format.clone()} );

                    t._saveCellValueAfterEdit( oCellEdit, c, newValue, /*flags*/undefined, /*skipNLCheck*/false, /*isNotHistory*/true, /*lockDraw*/true );
                }
            }

            window.setTimeout( function () {
                t._replaceCellText( aReplaceCells, valueForSearching, options, lockDraw, callback, oneUser );
            }, 1 );
        };

        return oneUser ? onReplaceCallback( true ) : this._isLockedCells( aReplaceCells[options.indexInArray], /*subType*/null, onReplaceCallback );
    };

    WorksheetView.prototype.findCell = function ( reference ) {
        var range = asc.g_oRangeCache.getAscRange( reference );
        if ( !range ) {

            var _C2H50H_ = this.model.workbook.getDefinesNames( reference, this.model.workbook.getActiveWs().getId() ), sheetName, ref;

            if ( !_C2H50H_ ) {

                var actRange = this.getActiveRangeObj(), ascRange, mc = this.model.getMergedByCell( actRange.startRow, actRange.startCol ), c1 = mc ? mc.c1 : actRange.c1, r1 = mc ? mc.r1 : actRange.r1, ar_norm = actRange.normalize(), mc_norm = mc ? mc.normalize() : null, c2 = mc_norm ? ( mc_norm.isEqual( ar_norm ) ? mc_norm.c1 : ar_norm.c2 ) : ar_norm.c2, r2 = mc_norm ? ( mc_norm.isEqual( ar_norm ) ? mc_norm.r1 : ar_norm.r2 ) : ar_norm.r2, defName;

                ascRange = new asc_Range( c1, r1, c2, r2 );
                /*TODO: сделать поиск по названиям автофигур, должен искать до того как вызвать поиск по именованным диапазонам*/
                defName = new Asc.asc_CDefName( reference, this.model.getName() + "!" + ascRange.getAbsName() );

                if ( this.collaborativeEditing.getGlobalLock() || !this.handlers.trigger( "getLockDefNameManagerStatus" ) ) {
                    this.handlers.trigger( "onErrorEvent", c_oAscError.ID.LockCreateDefName, c_oAscError.Level.NoCritical );
                    this._updateSelectionNameAndInfo();
                    return true;
                }

                _C2H50H_ = this.model.workbook.editDefinesNames( null, defName );
            }

            if ( _C2H50H_ ) {
                range = true;
                this._isLockedDefNames( null, _C2H50H_.nodeId );

                if ( _C2H50H_.isTable ) {
                    sheetName = _C2H50H_.Ref.split( "!" );
                    ref = sheetName[1];
                    sheetName = sheetName[0];
                    if ( sheetName[0] == "'" && sheetName[sheetName.length - 1] == "'" ) {
                        sheetName = sheetName.substring( 1, sheetName.length - 1 );
                    }
                    range = asc.g_oRangeCache.getAscRange( ref );
                    sheetName = this.model.workbook.getWorksheetByName( sheetName );
                }
                else if ( _C2H50H_.parsedRef.RefPos.length == 1 && _C2H50H_.parsedRef.outStack.length == 1 ) {
                    ref = _C2H50H_.parsedRef.outStack[0];
                    if ( ref.type == cElementType.cell3D ) {
                        range = ref.range.getBBox0().clone( true );
                        sheetName = ref.getWS();
                    }
                    else if ( ref.type == cElementType.cellsRange3D && ref.wsFrom == ref.wsTo ) {
                        range = ref.getRange()[0].getBBox0().clone( true );
                        sheetName = this.model.workbook.getWorksheetById( ref.wsFrom );
                    }

                }

                if ( range && sheetName ) {
                    ar_norm = range.normalize();
                    mc = sheetName.getMergedByCell( ar_norm.r1, ar_norm.c1 )
                    range = {range: mc ? mc : range, sheet: sheetName.getName()};
                }
            }
        }
        else {
            var ar_norm = range.normalize(), mc = this.model.getMergedByCell( ar_norm.r1, ar_norm.c1 );

            range = {range: mc ? mc : range, sheet: this.model.getName()};
        }
        return range;// ? this.setSelection(range, true) : null;
    };

    /* Ищет дополнение для ячейки */
    WorksheetView.prototype.getCellAutoCompleteValues = function ( col, row, maxCount ) {
        var arrValues = [], objValues = {};
        var range = this.findCellAutoComplete( col, row, 1, maxCount );
        this.getColValues( range, col, arrValues, objValues );
        range = this.findCellAutoComplete( col, row, -1, maxCount );
        this.getColValues( range, col, arrValues, objValues );

        arrValues.sort();
        return arrValues;
    };

    /* Ищет дополнение для ячейки (снизу или сверху) */
    WorksheetView.prototype.findCellAutoComplete = function ( col, row, step, maxCount ) {
        row += step;
        if ( !maxCount ) {
            maxCount = Number.MAX_VALUE;
        }
        var count = 0, cell, end = 0 < step ? this.model.getRowsCount() - 1 : 0, isEnd = true, colsCount = this.model.getColsCount(), range = new asc_Range( col, row, col, row );
        for ( ; row * step <= end && count < maxCount; row += step, isEnd = true, ++count ) {
            for ( col = range.c1; col <= range.c2; ++col ) {
                cell = this.model._getCellNoEmpty( row, col );
                if ( cell && false === cell.isEmptyText() ) {
                    isEnd = false;
                    break;
                }
            }
            // Идем влево по колонкам
            for ( col = range.c1 - 1; col >= 0; --col ) {
                cell = this.model._getCellNoEmpty( row, col );
                if ( null === cell || cell.isEmptyText() ) {
                    break;
                }
                isEnd = false;
            }
            range.c1 = col + 1;
            // Идем вправо по колонкам
            for ( col = range.c2 + 1; col < colsCount; ++col ) {
                cell = this.model._getCellNoEmpty( row, col );
                if ( null === cell || cell.isEmptyText() ) {
                    break;
                }
                isEnd = false;
            }
            range.c2 = col - 1;

            if ( isEnd ) {
                break;
            }
        }
        if ( 0 < step ) {
            range.r2 = row - 1;
        }
        else {
            range.r1 = row + 1;
        }
        return range.r1 <= range.r2 ? range : null;
    };

    /* Формирует уникальный массив */
    WorksheetView.prototype.getColValues = function ( range, col, arrValues, objValues ) {
        if ( null === range ) {
            return;
        }
        var row, cell, value, valueLowCase;
        for ( row = range.r1; row <= range.r2; ++row ) {
            cell = this.model._getCellNoEmpty( row, col );
            if ( cell ) {
                value = cell.getValue();
                if ( !isNumber( value ) ) {
                    valueLowCase = value.toLowerCase();
                    if ( !objValues.hasOwnProperty( valueLowCase ) ) {
                        arrValues.push( value );
                        objValues[valueLowCase] = 1;
                    }
                }
            }
        }
    };

    // ----- Cell Editor -----

    WorksheetView.prototype.setCellEditMode = function ( isCellEditMode ) {
        this.isCellEditMode = isCellEditMode;
    };

    WorksheetView.prototype.setFormulaEditMode = function ( isFormulaEditMode ) {
        this.isFormulaEditMode = isFormulaEditMode;
    };

    WorksheetView.prototype.setSelectionDialogMode = function ( selectionDialogType, selectRange ) {
        if ( selectionDialogType === this.selectionDialogType ) {
            return;
        }
        var oldSelectionDialogType = this.selectionDialogType;
        this.selectionDialogType = selectionDialogType;
        this.isSelectionDialogMode = c_oAscSelectionDialogType.None !== this.selectionDialogType;
        this.cleanSelection();

        if ( false === this.isSelectionDialogMode ) {
            if ( null !== this.copyActiveRange ) {
                this.activeRange = this.copyActiveRange.clone( true );
            }
            this.copyActiveRange = null;
            if ( oldSelectionDialogType === c_oAscSelectionDialogType.Chart ) {
                this.objectRender.controller.checkChartForProps( false );
            }
        }
        else {

            this.copyActiveRange = this.activeRange.clone( true );
            if ( selectRange ) {
                if ( typeof selectRange === 'string' ) {
                    selectRange = this.model.getRange2( selectRange );
                    if ( selectRange ) {
                        selectRange = selectRange.getBBox0();
                    }
                }

                if ( null != selectRange ) {
                    this.activeRange.assign( selectRange.c1, selectRange.r1, selectRange.c2, selectRange.r2 );
                }
            }
            if ( selectionDialogType === c_oAscSelectionDialogType.Chart ) {
                this.objectRender.controller.checkChartForProps( true );
            }
        }
        this._drawSelection();
    };

    // Получаем свойство: редактируем мы сейчас или нет
    WorksheetView.prototype.getCellEditMode = function () {
        return this.isCellEditMode;
    };

    WorksheetView.prototype._isFormula = function ( val ) {
        return (0 < val.length && 1 < val[0].text.length && '=' === val[0].text.charAt( 0 ));
    };

    WorksheetView.prototype.getActiveCell = function ( x, y, isCoord ) {
        var t = this;
        var col, row;
        if ( isCoord ) {
            x *= asc_getcvt( 0/*px*/, 1/*pt*/, t._getPPIX() );
            y *= asc_getcvt( 0/*px*/, 1/*pt*/, t._getPPIY() );
            col = t._findColUnderCursor( x, true );
            row = t._findRowUnderCursor( y, true );
            if ( !col || !row ) {
                return false;
            }
            col = col.col;
            row = row.row;
        }
        else {
            col = t.activeRange.startCol;
            row = t.activeRange.startRow;
        }

        // Проверим замерженность
        var mergedRange = this.model.getMergedByCell( row, col );
        return mergedRange ? mergedRange : new asc_Range( col, row, col, row );
    };

    WorksheetView.prototype._saveCellValueAfterEdit = function ( oCellEdit, c, val, flags, skipNLCheck, isNotHistory, lockDraw ) {
        var t = this;
        var oldMode = t.isFormulaEditMode;
        t.isFormulaEditMode = false;

        if ( !isNotHistory ) {
            History.Create_NewPoint();
            History.StartTransaction();
        }

        var isFormula = t._isFormula( val );

        if ( isFormula ) {
            var ftext = val.reduce( function ( pv, cv ) {
                return pv + cv.text;
            }, "" );
            var ret = true;
            // ToDo - при вводе формулы в заголовок автофильтра надо писать "0"
            c.setValue( ftext, function ( r ) {
                ret = r;
            } );
            if ( !ret ) {
                t.isFormulaEditMode = oldMode;
                History.EndTransaction();
                return false;
            }
            isFormula = c.isFormula();
        }
        else {
            c.setValue2( val );
            // Вызываем функцию пересчета для заголовков форматированной таблицы
            t.autoFilters.renameTableColumn( oCellEdit );
        }

        if ( !isFormula ) {
            // Нужно ли выставлять WrapText (ищем символ новой строки в тексте)
            var bIsSetWrap = false;
            if ( !skipNLCheck ) {
                for ( var i = 0; i < val.length; ++i ) {
                    if ( val[i].text.indexOf( kNewLine ) >= 0 ) {
                        bIsSetWrap = true;
                        break;
                    }
                }
            }
            if ( bIsSetWrap ) {
                c.setWrap( true );
            }

            // Для формулы обновление будет в коде рассчета формулы
            t._updateCellsRange( oCellEdit, /*canChangeColWidth*/c_oAscCanChangeColWidth.numbers, lockDraw );
        }

        if ( !isNotHistory ) {
            History.EndTransaction();
        }

        // если вернуть false, то редактор не закроется
        return true;
    };

    WorksheetView.prototype.openCellEditor = function ( editor, x, y, isCoord, fragments, cursorPos, isFocus, isClearCell, isHideCursor, isQuickInput, activeRange ) {
        var t = this, tc = this.cols, tr = this.rows, col, row, c, fl, mc, bg, isMerged;
        var ar = this.activeRange;
        if ( activeRange ) {
            this.activeRange = activeRange.clone();
        }

        if ( this.objectRender.checkCursorDrawingObject( x, y ) ) {
            return false;
        }

        function getVisibleRangeObject() {
            var vr = t.visibleRange.clone(), offsetX = 0, offsetY = 0;
            if ( t.topLeftFrozenCell ) {
                var cFrozen = t.topLeftFrozenCell.getCol0();
                var rFrozen = t.topLeftFrozenCell.getRow0();
                if ( 0 < cFrozen ) {
                    if ( col >= cFrozen ) {
                        offsetX = tc[cFrozen].left - tc[0].left;
                    }
                    else {
                        vr.c1 = 0;
                        vr.c2 = cFrozen - 1;
                    }
                }
                if ( 0 < rFrozen ) {
                    if ( row >= rFrozen ) {
                        offsetY = tr[rFrozen].top - tr[0].top;
                    }
                    else {
                        vr.r1 = 0;
                        vr.r2 = rFrozen - 1;
                    }
                }
            }
            return {vr: vr, offsetX: offsetX, offsetY: offsetY};
        }

        if ( isCoord ) {
            x *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIX() );
            y *= asc_getcvt( 0/*px*/, 1/*pt*/, this._getPPIY() );
            col = this._findColUnderCursor( x, true );
            row = this._findRowUnderCursor( y, true );
            if ( !col || !row ) {
                return false;
            }
            col = col.col;
            row = row.row;
        }
        else {
            col = ar.startCol;
            row = ar.startRow;
        }

        // Возможно стоит заменить на ячейку из кеша
        c = this._getVisibleCell( col, row );
        if ( !c ) {
            throw "Can not get cell data (col=" + col + ", row=" + row + ")";
        }

        fl = this._getCellFlags( c );
        isMerged = fl.isMerged();
        if ( isMerged ) {
            mc = fl.merged;
            c = this._getVisibleCell( mc.c1, mc.r1 );
            if ( !c ) {
                throw "Can not get merged cell data (col=" + mc.c1 + ", row=" + mc.r1 + ")";
            }
            fl = this._getCellFlags( c );
        }

        // Выставляем режим 'не редактируем' (иначе мы попытаемся переместить редактор, который еще не открыт)
        this.isCellEditMode = false;
        this.handlers.trigger( "onScroll", this._calcActiveCellOffset() );
        this.isCellEditMode = true;

        bg = c.getFill();
        this.isFormulaEditMode = false;
        // Очищаем массив ячеек для текущей формулы
        this.arrActiveFormulaRanges = [];

        var oFontColor = c.getFontcolor();
        // Скрываем окно редактирования комментария
        this.model.workbook.handlers.trigger( "asc_onHideComment" );

        if ( fragments === undefined ) {
            var _fragmentsTmp = c.getValueForEdit2();
            fragments = [];
            for ( var i = 0; i < _fragmentsTmp.length; ++i )
                fragments.push( _fragmentsTmp[i].clone() );
        }

        var arrAutoComplete = this.getCellAutoCompleteValues( col, row, kMaxAutoCompleteCellEdit );
        var arrAutoCompleteLC = asc.arrayToLowerCase( arrAutoComplete );

        editor.open( {
            fragments         : fragments,
            flags             : fl,
            font              : new asc.FontProperties( c.getFontname(), c.getFontsize() ),
            background        : bg || this.settings.cells.defaultState.background,
            textColor         : oFontColor || this.settings.cells.defaultState.color,
            cursorPos         : cursorPos,
            zoom              : this.getZoom(),
            focus             : isFocus,
            isClearCell       : isClearCell,
            isHideCursor      : isHideCursor,
            isQuickInput      : isQuickInput,
            isAddPersentFormat: isQuickInput && c_oAscNumFormatType.Percent === c.getNumFormat().getType(),
            autoComplete      : arrAutoComplete,
            autoCompleteLC    : arrAutoCompleteLC,
            saveValueCallback : function ( val, flags, skipNLCheck ) {
                var oCellEdit = isMerged ? new asc_Range( mc.c1, mc.r1, mc.c1, mc.r1 ) : new asc_Range( col, row, col, row );
                return t._saveCellValueAfterEdit( oCellEdit, c, val, flags, skipNLCheck, /*isNotHistory*/false, /*lockDraw*/false );
            },
            getSides          : function () {
                var _col = !isMerged ? col : mc.c1;
                var _row = !isMerged ? row : mc.r2;
                var vro = getVisibleRangeObject();
                var i, w, h, arrLeftS = [], arrRightS = [], arrBottomS = [];
                var offsX = tc[vro.vr.c1].left - tc[0].left - vro.offsetX;
                var offsY = tr[vro.vr.r1].top - tr[0].top - vro.offsetY;
                var cellX = tc[_col].left - offsX, cellY = tr[!isMerged ? row : mc.r1].top - offsY;
                for ( i = _col; i >= vro.vr.c1; --i ) {
                    if ( t.width_1px < tc[i].width ) {
                        arrLeftS.push( tc[i].left - offsX );
                    }
                }
                arrLeftS.sort( fSortDescending );

                // Для замерженных ячеек, можем уйти за границу
                if ( isMerged && _col > vro.vr.c2 ) {
                    _col = vro.vr.c2;
                }
                for ( i = _col; i <= vro.vr.c2; ++i ) {
                    w = tc[i].width;
                    if ( t.width_1px < w ) {
                        arrRightS.push( tc[i].left + w - offsX );
                    }
                }
                w = t.drawingCtx.getWidth();
                if ( arrRightS[arrRightS.length - 1] > w ) {
                    arrRightS[arrRightS.length - 1] = w;
                }
                arrRightS.sort( fSortAscending );

                // Для замерженных ячеек, можем уйти за границу
                if ( isMerged && _row > vro.vr.r2 ) {
                    _row = vro.vr.r2;
                }
                for ( i = _row; i <= vro.vr.r2; ++i ) {
                    h = tr[i].height;
                    if ( t.height_1px < h ) {
                        arrBottomS.push( tr[i].top + h - offsY );
                    }
                }
                h = t.drawingCtx.getHeight();
                if ( arrBottomS[arrBottomS.length - 1] > h ) {
                    arrBottomS[arrBottomS.length - 1] = h;
                }
                arrBottomS.sort( fSortAscending );
                return {l: arrLeftS, r: arrRightS, b: arrBottomS, cellX: cellX, cellY: cellY};
            }
        } );
        return true;
    };

    WorksheetView.prototype.openCellEditorWithText = function ( editor, text, cursorPos, isFocus, activeRange ) {
        var t = this;
        var ar = (activeRange) ? activeRange : t.activeRange;
        var c = t._getVisibleCell( ar.startCol, ar.startRow );
        var v, copyValue;

        if ( !c ) {
            throw "Can not get cell data (col=" + ar.startCol + ", row=" + ar.startCol + ")";
        }

        // get first fragment and change its text
        v = c.getValueForEdit2().slice( 0, 1 );
        // Создаем новый массив, т.к. getValueForEdit2 возвращает ссылку
        copyValue = [];
        copyValue[0] = new Fragment( {text: text, format: v[0].format.clone()} );

        var bSuccess = t.openCellEditor( editor, 0, 0, /*isCoord*/false, /*fragments*/undefined, /*cursorPos*/undefined, isFocus, /*isClearCell*/true, /*isHideCursor*/false, /*isQuickInput*/false, activeRange );
        if ( bSuccess ) {
            editor.paste( copyValue, cursorPos );
        }
        return bSuccess;
    };

    WorksheetView.prototype.getFormulaRanges = function () {
        return this.arrActiveFormulaRanges;
    };

    /**
     *
     * @param {Object} ranges
     * @param canChangeColWidth
     * @param lockDraw
     * @param updateHeight
     */
    WorksheetView.prototype.updateRanges = function ( ranges, canChangeColWidth, lockDraw, updateHeight ) {
        var arrRanges = [], range;
        for ( var i in ranges ) {
            range = ranges[i];
            this.updateRange( range, canChangeColWidth, true );
            arrRanges.push( range );
        }

        if ( 0 < arrRanges.length ) {
            if ( updateHeight ) {
                this.isChanged = true;
            }

            this._recalculateAfterUpdate( arrRanges, lockDraw );
        }
    };
    // ToDo избавиться от этой функции!!!!Заглушка для принятия изменений
    WorksheetView.prototype._checkUpdateRange = function ( range ) {
        // Для принятия изменения нужно делать расширение диапазона
        if ( this.model.workbook.bCollaborativeChanges ) {
            var bIsUpdateX = false, bIsUpdateY = false;
            if ( range.c2 >= this.nColsCount ) {
                this.expandColsOnScroll( false, true, 0 ); // Передаем 0, чтобы увеличить размеры
                // Проверка, вдруг пришел диапазон за пределами существующей области
                if ( range.c2 >= this.nColsCount ) {
                    if ( range.c1 >= this.nColsCount ) {
                        return;
                    }
                    range.c2 = this.nColsCount - 1;
                }
                bIsUpdateX = true;
            }
            if ( range.r2 >= this.nRowsCount ) {
                this.expandRowsOnScroll( false, true, 0 ); // Передаем 0, чтобы увеличить размеры
                // Проверка, вдруг пришел диапазон за пределами существующей области
                if ( range.r2 >= this.nRowsCount ) {
                    if ( range.r1 >= this.nRowsCount ) {
                        return;
                    }
                    range.r2 = this.nRowsCount - 1;
                }
                bIsUpdateY = true;
            }

            if ( bIsUpdateX && bIsUpdateY ) {
                this.handlers.trigger( "reinitializeScroll" );
            }
            else if ( bIsUpdateX ) {
                this.handlers.trigger( "reinitializeScrollX" );
            }
            else if ( bIsUpdateY ) {
                this.handlers.trigger( "reinitializeScrollY" );
            }
        }
    };
    WorksheetView.prototype.updateRange = function ( range, canChangeColWidth, lockDraw ) {
        this._checkUpdateRange( range );
        this._updateCellsRange( range, canChangeColWidth, lockDraw );
    };

    WorksheetView.prototype._updateCellsRange = function ( range, canChangeColWidth, lockDraw ) {
        var r, c, h, d, ct, isMerged;
        var mergedRange, bUpdateRowHeight;

        if ( range === undefined ) {
            range = this.activeRange.clone( true );
        }
        else {
            // ToDo заглушка..пора уже переделать обновление данных
            if ( range.r1 >= this.nRowsCount || range.c1 >= this.nColsCount ) {
                return;
            }
            range.r2 = Math.min( range.r2, this.nRowsCount - 1 );
            range.c2 = Math.min( range.c2, this.nColsCount - 1 );
        }

        if ( gc_nMaxCol0 === range.c2 || gc_nMaxRow0 === range.r2 ) {
            range = range.clone();
            if ( gc_nMaxCol0 === range.c2 ) {
                range.c2 = this.cols.length - 1;
            }
            if ( gc_nMaxRow0 === range.r2 ) {
                range.r2 = this.rows.length - 1;
            }
        }

        this._cleanCache( range );

        // Если размер диапазона превышает размер видимой области больше чем в 3 раза, то очищаем весь кэш
        if ( this._isLargeRange( range ) ) {
            this.changeWorksheet( "update", {lockDraw: lockDraw} );

            this._updateSelectionNameAndInfo();
            return;
        }

        var cto;
        for ( r = range.r1; r <= range.r2; ++r ) {
            if ( this.height_1px > this.rows[r].height ) {
                continue;
            }
            for ( c = range.c1; c <= range.c2; ++c ) {
                if ( this.width_1px > this.cols[c].width ) {
                    continue;
                }
                c = this._addCellTextToCache( c, r, canChangeColWidth ); // may change member 'this.isChanged'
            }
            for ( h = this.defaultRowHeight, d = this.defaultRowDescender, c = 0; c < this.cols.length; ++c ) {
                ct = this._getCellTextCache( c, r, true );
                if ( !ct ) {
                    continue;
                }

                /**
                 * Пробегаемся по строке и смотрим не продолжается ли ячейка на соседние.
                 * С помощью этой правки уйдем от обновления всей строки при каких-либо действиях
                 */
                if ( (c < range.c1 || c > range.c2) && (0 !== ct.sideL || 0 !== ct.sideR) ) {
                    cto = this._calcCellTextOffset( c, r, ct.cellHA, ct.metrics.width );
                    ct.cellW = cto.maxWidth;
                    ct.sideL = cto.leftSide;
                    ct.sideR = cto.rightSide;
                }

                // Замерженная ячейка (с 2-мя или более строками) не влияет на высоту строк!
                isMerged = ct.flags.isMerged();
                if ( !isMerged ) {
                    bUpdateRowHeight = true;
                }
                else {
                    mergedRange = ct.flags.merged;
                    // Для замерженных ячеек (с 2-мя или более строками) оптимизировать не нужно
                    bUpdateRowHeight = mergedRange.r1 === mergedRange.r2;
                }
                if ( bUpdateRowHeight ) {
                    h = Math.max( h, ct.metrics.height );
                }

                if ( ct.cellVA !== kvaTop && ct.cellVA !== kvaCenter && !isMerged ) {
                    d = Math.max( d, ct.metrics.height - ct.metrics.baseline );
                }
            }
            if ( Math.abs( h - this.rows[r].height ) > 0.000001 && !this.rows[r].isCustomHeight ) {
                if ( !this.rows[r].isDefaultHeight ) {
                    this.rows[r].heightReal = this.rows[r].height = Math.min( h, this.maxRowHeight );
                    this.model.setRowHeight( this.rows[r].height + this.height_1px, r, r, false );
                }
                this.isChanged = true;
            }
            if ( Math.abs( d - this.rows[r].descender ) > 0.000001 ) {
                this.rows[r].descender = d;
                this.isChanged = true;
            }
        }

        if ( !lockDraw ) {
            this._recalculateAfterUpdate( [range] );
        }
    };

    WorksheetView.prototype._recalculateAfterUpdate = function ( arrChanged, lockDraw ) {
        if ( this.isChanged ) {
            this.isChanged = false;
            this._initCellsArea( true );
            this.cache.reset();
            this._cleanCellsTextMetricsCache();
            this._prepareCellTextMetricsCache();
            this.handlers.trigger( "reinitializeScroll" );
        }
        if ( !lockDraw ) {
            this._updateSelectionNameAndInfo();
        }

        this.objectRender.rebuildChartGraphicObjects( new CChangeTableData( null, null, null, null, arrChanged ) );
        this.cellCommentator.updateCommentPosition();
        this.handlers.trigger( "onDocumentPlaceChanged" );
        this.draw( lockDraw );
    };

    WorksheetView.prototype.enterCellRange = function ( editor ) {
        if ( !this.isFormulaEditMode ) {
            return;
        }

        var currentRange = this.arrActiveFormulaRanges[this.arrActiveFormulaRangesPosition].clone();
        var startCol = currentRange.startCol, startRow = currentRange.startRow;
        // Замерженную ячейку должны отдать только левую верхнюю.
        var mergedRange = this.model.getMergedByCell( currentRange.r1, currentRange.c1 );
        if ( mergedRange && currentRange.isEqual( mergedRange ) ) {
            currentRange.r2 = currentRange.r1;
            currentRange.c2 = currentRange.c1;
        }

        /*            var defName = this.model.workbook.findDefinesNames(this.model.getName()+"!"+currentRange.getAbsName(),this.model.getId());
         console.log("defName #2 " + defName);*/
        var sheetName = "", cFEWSO = editor.handlers.trigger( "getCellFormulaEnterWSOpen" );
        if( editor.formulaIsOperator() && cFEWSO && cFEWSO.model.getId() != this.model.getId() ){
            sheetName = parserHelp.getEscapeSheetName(this.model.getName())+"!";
        }
        editor.enterCellRange( /*defName || */sheetName + currentRange.getAllRange().getName() );

        for ( var tmpRange, i = 0; i < this.arrActiveFormulaRanges.length; ++i ) {
            tmpRange = this.arrActiveFormulaRanges[i];
            if ( tmpRange.isEqual( currentRange ) ) {
                tmpRange.startCol = startCol;
                tmpRange.startRow = startRow;
                break;
            }
        }
    };

    WorksheetView.prototype.addFormulaRange = function ( range ) {
        var r = range !== undefined ? range : new asc_ActiveRange( this.activeRange.c1, this.activeRange.r1, this.activeRange.c2, this.activeRange.r2 );
        this.arrActiveFormulaRanges.push( r );
        this.arrActiveFormulaRangesPosition = this.arrActiveFormulaRanges.length - 1;
        this._fixSelectionOfMergedCells();
    };

    WorksheetView.prototype.activeFormulaRange = function ( range ) {
        this.arrActiveFormulaRangesPosition = -1;
        for ( var i = 0; i < this.arrActiveFormulaRanges.length; ++i ) {
            if ( this.arrActiveFormulaRanges[i].isEqual( range ) ) {
                this.arrActiveFormulaRangesPosition = i;
                return;
            }
        }
    };

    WorksheetView.prototype.cleanFormulaRanges = function () {
        // Очищаем массив ячеек для текущей формулы
        this.arrActiveFormulaRanges = [];
    };

    WorksheetView.prototype.addAutoFilter = function ( styleName, addFormatTableOptionsObj ) {
        // Проверка глобального лока
        if ( this.collaborativeEditing.getGlobalLock() ) {
            return;
        }

        if ( !this.handlers.trigger( "getLockDefNameManagerStatus" ) ) {
            this.handlers.trigger( "onErrorEvent", c_oAscError.ID.LockCreateDefName, c_oAscError.Level.NoCritical );
            return;
        }

        var t = this;
        var ar = t.activeRange.clone( true );

        var onChangeAutoFilterCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                t.model.workbook.handlers.trigger( "asc_onError", c_oAscError.ID.LockedAllError, c_oAscError.Level.NoCritical );
                t.handlers.trigger( "selectionChanged", t.getSelectionInfo() );
                return;
            }

            if ( addFormatTableOptionsObj && t.autoFilters.isChangeAutoFilterToTablePart( addFormatTableOptionsObj ) === true ) {
                t.autoFilters.changeAutoFilterToTablePart( styleName, ar, addFormatTableOptionsObj );
            }
            else {
                t.autoFilters.addAutoFilter( styleName, ar, addFormatTableOptionsObj );
            }
        };

        if ( t.autoFilters.checkAddAutoFilter( ar, styleName, addFormatTableOptionsObj ) === true ) {
            this._isLockedAll( onChangeAutoFilterCallback );
            this._isLockedDefNames( null, null );
        }
        else//для того, чтобы в случае ошибки кнопка отжималась!
        {
            t.handlers.trigger( "selectionChanged", t.getSelectionInfo() );
        }
    };

    WorksheetView.prototype.changeAutoFilter = function ( tableName, optionType, val ) {
        // Проверка глобального лока
        if ( this.collaborativeEditing.getGlobalLock() ) {
            return;
        }

        var t = this;
        var ar = t.activeRange.clone( true );

        var onChangeAutoFilterCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                t.handlers.trigger( "selectionChanged", t.getSelectionInfo() );
                return;
            }

            switch ( optionType ) {
                case c_oAscChangeFilterOptions.filter:
                {
                    if ( !val ) {
                        t.autoFilters.deleteAutoFilter( ar, tableName );
                    }
                    else {
                        t.autoFilters.addAutoFilter( null, ar );
                    }

                    break;
                }
                case c_oAscChangeFilterOptions.style:
                {
                    t.autoFilters.changeTableStyleInfo( val, ar, tableName );
                    break;
                }
            }
        };

        this._isLockedAll( onChangeAutoFilterCallback );
    };

    WorksheetView.prototype.applyAutoFilter = function ( type, autoFilterObject ) {
        var t = this;
        var ar = t.activeRange.clone( true );
        var onChangeAutoFilterCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            var rowChange = t.autoFilters.applyAutoFilter( autoFilterObject, ar );
            if ( null !== rowChange ) {
                t.objectRender.updateSizeDrawingObjects( {target: c_oTargetType.RowResize, row: rowChange} );
            }
        };
        this._isLockedAll( onChangeAutoFilterCallback );
    };

    WorksheetView.prototype.sortColFilter = function ( type, cellId, displayName ) {
        var t = this;
        var ar = this.activeRange.clone( true );
        var onChangeAutoFilterCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            t.autoFilters.sortColFilter( type, cellId, ar, null, displayName );
        };
        this._isLockedAll( onChangeAutoFilterCallback );
    };

    WorksheetView.prototype.getAddFormatTableOptions = function ( range ) {
        var ar = this.activeRange.clone( true );
        return this.autoFilters.getAddFormatTableOptions( ar, range );
    };

    WorksheetView.prototype.clearFilter = function () {
        var t = this;
        var ar = this.activeRange.clone( true );
        var onChangeAutoFilterCallback = function ( isSuccess ) {
            if ( false === isSuccess ) {
                return;
            }

            t.autoFilters.isApplyAutoFilterInCell( ar, true );
        };
        this._isLockedAll( onChangeAutoFilterCallback );
    };

    /**
     * Обновление при изменениях форматированной таблицы
     * @param range - обновляемый диапазон (он же диапазон для выделения)
     * @param recalc - делать ли автоподбор по названию столбца
     * @param changeRowsOrMerge - менялись ли строки (скрытие раскрытие) или был unmerge
     * @private
     */
    WorksheetView.prototype._onUpdateFormatTable = function ( range, recalc, changeRowsOrMerge ) {
        //ToDo заглушка, чтобы не падало. Нужно полностью переделывать этот код!!!! (Перенес выше из-за бага http://bugzserver/show_bug.cgi?id=26705)
        this._checkUpdateRange( range );

        if ( !recalc ) {
            // При скрытии/открытии строк стоит делать update всему
            if ( changeRowsOrMerge ) {
                this.isChanged = true;
            }
            // Пока вызовем updateRange, но стоит делать просто draw
            this._updateCellsRange( range );
            return;
        }

        if ( !this.activeRange.isEqual( range ) ) {
            this.setSelection( range );
        }

        var i, r = range.r1, bIsUpdate = false, w;
        for ( i = range.c1; i <= range.c2; ++i ) {
            w = this.onChangeWidthCallback( i, r, r, /*onlyIfMore*/true );
            if ( -1 !== w ) {
                this.cols[i] = this._calcColWidth( w );
                this._cleanCache( new asc_Range( i, 0, i, this.rows.length - 1 ) );
                bIsUpdate = true;
            }
        }

        if ( bIsUpdate ) {
            this._updateColumnPositions();
            this._updateVisibleColsCount();
            this.changeWorksheet( "update" );
        }
        else if ( changeRowsOrMerge ) {
            // Был merge, нужно обновить (ToDo)
            this._initCellsArea( true );
            this.cache.reset();
            this._cleanCellsTextMetricsCache();
            this._prepareCellTextMetricsCache();
            if ( this.objectRender && this.objectRender.drawingArea ) {
                this.objectRender.drawingArea.reinitRanges();
            }
            var oChangeData = new CChangeTableData( null, null, null, null ); // Обновление для диаграмм
            oChangeData.change = new asc_Range( range.c1, 0, range.c2, gc_nMaxRow0 );
            this.objectRender.rebuildChartGraphicObjects( oChangeData );
            this.draw();
            this.handlers.trigger( "reinitializeScroll" );
        }
        else {
            // Просто отрисуем
            this.draw();
        }

    };

    WorksheetView.prototype._loadFonts = function ( fonts, callback ) {
        var api = window["Asc"]["editor"];
        api._loadFonts( fonts, callback );
    };

    WorksheetView.prototype.setData = function ( oData ) {
        History.Clear();
        History.TurnOff();
        var oAllRange = new Range( this.model, 0, 0, this.nRowsCount - 1, this.nColsCount - 1 );
        oAllRange.cleanAll();

        var row, oCell;
        for ( var r = 0; r < oData.length; ++r ) {
            row = oData[r];
            for ( var c = 0; c < row.length; ++c ) {
                if ( row[c] ) {
                    oCell = this._getVisibleCell( c, r );
                    oCell.setValue( row[c] );
                }
            }
        }
        History.TurnOn();
        this._updateCellsRange( oAllRange.bbox ); // ToDo Стоит обновить nRowsCount и nColsCount
    };
    WorksheetView.prototype.getData = function () {
        var arrResult, arrCells = [], cell, c, r, row, lastC = -1, lastR = -1, val;
        var maxCols = Math.min( this.model.getColsCount(), gc_nMaxCol );
        var maxRows = Math.min( this.model.getRowsCount(), gc_nMaxRow );

        for ( r = 0; r < maxRows; ++r ) {
            row = [];
            for ( c = 0; c < maxCols; ++c ) {
                cell = this.model._getCellNoEmpty( r, c );
                if ( cell && '' !== (val = cell.getValue()) ) {
                    lastC = Math.max( lastC, c );
                    lastR = Math.max( lastR, r );
                }
                else {
                    val = '';
                }
                row.push( val );
            }
            arrCells.push( row );
        }

        arrResult = arrCells.slice( 0, lastR + 1 );
        ++lastC;
        if ( lastC < maxCols ) {
            for ( r = 0; r < arrResult.length; ++r ) {
                arrResult[r] = arrResult[r].slice( 0, lastC );
            }
        }
        return arrResult;
    };

    /*
     * Export
     * -----------------------------------------------------------------------------
     */
    window["Asc"].WorksheetView = WorksheetView;


})( jQuery, window );