/**
 * A Column subclass which renders a checkbox in each column cell which toggles the truthiness
 * of the associated data field on click.
 *
 * Example usage:
 *
 *     @example
 *     var store = Ext.create('Ext.data.Store', {
 *         fields: ['name', 'email', 'phone', 'active'],
 *         data: [
 *             { name: 'Lisa', email: 'lisa@simpsons.com', phone: '555-111-1224', active: true },
 *             { name: 'Bart', email: 'bart@simpsons.com', phone: '555-222-1234', active: true },
 *             { name: 'Homer', email: 'homer@simpsons.com', phone: '555-222-1244', active: false },
 *             { name: 'Marge', email: 'marge@simpsons.com', phone: '555-222-1254', active: true }
 *         ]
 *     });
 *
 *     Ext.create('Ext.grid.Panel', {
 *         title: 'Simpsons',
 *         height: 200,
 *         width: 400,
 *         renderTo: Ext.getBody(),
 *         store: store,
 *         columns: [
 *             { text: 'Name', dataIndex: 'name' },
 *             { text: 'Email', dataIndex: 'email', flex: 1 },
 *             { text: 'Phone', dataIndex: 'phone' },
 *             { xtype: 'checkcolumn', text: 'Active', dataIndex: 'active' }
 *         ]
 *     });
 *
 * The check column can be at any index in the columns array.
 */
Ext.define('Ext.grid.column.Check', {
    extend: 'Ext.grid.column.Column',
    alternateClassName: ['Ext.ux.CheckColumn', 'Ext.grid.column.CheckColumn'],
    alias: 'widget.checkcolumn',
 
    /**
     * @property {Boolean} isCheckColumn
     * `true` in this class to identify an object as an instantiated Check column,
     * or subclass thereof.
     */
    isCheckColumn: true,
 
    config: {
        /**
         * @cfg {Boolean} [headerCheckbox=false]
         * Configure as `true` to display a checkbox below the header text.
         *
         * Clicking the checkbox will check/uncheck all records.
         */
        headerCheckbox: false
    },
 
    /**
     * @cfg
     * @hide
     * Overridden from base class. Must center to line up with editor.
     */
    align: 'center',
 
    /**
     * @cfg {String} [triggerEvent=click]
     * The mouse event which triggers the toggle of a single cell.
     */
    triggerEvent: 'click',
 
    /**
     * @cfg {Boolean} invert
     * Use `true` to display a check when the value is `false` instead of when the value
     * is `true`.
     */
    invert: false,
 
    /**
     * @cfg {String} tooltip
     * The tooltip text to show upon hover of a unchecked cell.
     */
 
    /**
     * @cfg {String} checkedTooltip
     * The tooltip text to show upon hover of an checked cell.
     */
 
    ignoreExport: true,
 
    /**
     * @cfg {Boolean} [stopSelection=true]
     * Prevent grid selection upon mousedown.
     */
    stopSelection: true,
 
    /**
     * @private
     */
    headerCheckedCls: Ext.baseCSSPrefix + 'grid-hd-checker-on',
 
    /**
     * @private
     * The CSS class used to style and select the header checkbox.
     */
    headerCheckboxCls: Ext.baseCSSPrefix + 'column-header-checkbox',
 
    checkboxCls: Ext.baseCSSPrefix + 'grid-checkcolumn',
 
    checkboxCheckedCls: Ext.baseCSSPrefix + 'grid-checkcolumn-checked',
 
    innerCls: Ext.baseCSSPrefix + 'grid-checkcolumn-cell-inner',
 
    clickTargetName: 'el',
 
    defaultFilterType: 'boolean',
 
    checkboxAriaRole: 'checkbox',
 
    /**
     * @event beforecheckchange
     * Fires when the UI requests a change of check status.
     * The change may be vetoed by returning `false` from a listener.
     * @param {Ext.grid.column.Check} this CheckColumn.
     * @param {Number} rowIndex The row index.
     * @param {Boolean} checked `true` if the box is to be checked.
     * @param {Ext.data.Model} record The record to be updated.
     * @param {Ext.event.Event} e The underlying event which caused the check change.
     * @param {Ext.grid.CellContext} e.position {@link Ext.grid.CellContext CellContext} object
     * containing all contextual information about where the event was triggered.
     */
 
    /**
     * @event checkchange
     * Fires when the UI has successfully changed the checked state of a row.
     * @param {Ext.grid.column.Check} this CheckColumn.
     * @param {Number} rowIndex The row index.
     * @param {Boolean} checked `true` if the box is now checked.
     * @param {Ext.data.Model} record The record which was updated.
     * @param {Ext.event.Event} e The underlying event which caused the check change.
     * @param {Ext.grid.CellContext} e.position {@link Ext.grid.CellContext CellContext} object
     */
 
    /**
     * @event beforeheadercheckchange
     * Fires when the header is clicked and before the mass check/uncheck takes place.
     * The change may be vetoed by returning `false` from a listener.
     * @param {Ext.grid.column.Check} this CheckColumn.
     * @param {Boolean} checked `true` if all boxes are to be checked.
     * @param {Ext.event.Event} e The underlying event which caused the check change.
     */
 
    /**
     * @event headercheckchange
     * Fires after the header is clicked and a mass check/uncheck operation has been completed.
     * @param {Ext.grid.column.Check} this CheckColumn.
     * @param {Boolean} checked `true` if all boxes are now checked.
     * @param {Ext.event.Event} e The underlying event which caused the check change.
     */
 
    constructor: function(config) {
        this.scope = this;
        this.callParent([config]);
    },
 
    afterComponentLayout: function() {
        var me = this;
 
        me.callParent(arguments);
 
        if (me.useAriaElements && me.headerCheckbox) {
            me.updateHeaderAriaDescription(me.areAllChecked());
        }
 
        // Only do this once
        if (!me.storeListeners) {
            // Ensure initial rendered state is correct.
            // This will update the header state on the next animation frame
            // after all rows have been rendered.
            me.updateHeaderState();
 
            // We need to listen to data changed. This includes add and remove as well as reload.
            // We cannot rely on the renderer or updater to kick off an updateHeaderState call
            // because buffered rendering may mean that the UI does not process the entire dataset.
            me.storeListeners = me.getView().dataSource.on({
                datachanged: me.onDataChanged,
                scope: me,
                destroyable: true
            });
        }
    },
 
    onRemoved: function() {
        this.callParent(arguments);
        this.storeListeners = Ext.destroy(this.storeListeners);
    },
 
    onDataChanged: function(store, records) {
        // If any records are added or removed, we need up to date the header state.
        this.updateHeaderState();
    },
 
    updateHeaderCheckbox: function(headerCheckbox) {
        var me = this,
            cls = Ext.baseCSSPrefix + 'column-header-checkbox';
 
        if (headerCheckbox) {
            me.addCls(cls);
 
            // So that SPACE/ENTER does not sort, but routes to the checkbox
            me.sortable = false;
 
            if (me.useAriaElements) {
                me.updateHeaderAriaDescription(me.areAllChecked());
            }
        }
        else {
            me.removeCls(cls);
 
            if (me.useAriaElements && me.ariaEl.dom) {
                me.ariaEl.dom.removeAttribute('aria-describedby');
            }
        }
 
        // Keep the header checkbox up to date
        me.updateHeaderState();
    },
 
    /**
     * @private
     * Process and refire events routed from the GridView's processEvent method.
     */
    processEvent: function(type, view, cell, recordIndex, cellIndex, e, record, row) {
        var me = this,
            key = type === 'keydown' && e.getKey(),
            isClick = type === me.triggerEvent,
            disabled = me.disabled,
            ret,
            checked;
 
        // Flag event to tell SelectionModel not to process it.
        e.stopSelection = !key && me.stopSelection;
 
        if (!disabled && (isClick || (key === e.ENTER || key === e.SPACE))) {
            checked = !me.isRecordChecked(record);
 
            // Allow apps to hook beforecheckchange
            if (me.fireEvent('beforecheckchange', me, recordIndex, checked, record, e) !== false) {
 
                me.setRecordCheck(record, recordIndex, checked, cell, e);
 
                // Do not allow focus to follow from this mousedown unless the grid
                // is already in actionable mode
                if (isClick && !view.actionableMode) {
                    e.preventDefault();
                }
 
                if (me.hasListeners.checkchange) {
                    me.fireEvent('checkchange', me, recordIndex, checked, record, e);
                }
            }
        }
        else {
            ret = me.callParent(arguments);
        }
 
        return ret;
    },
 
    onTitleElClick: function(e, t, sortOnClick) {
        var me = this;
 
        // Toggle if no text, or it's activated by SPACE key,
        // or the click is on the checkbox element.
        if (!me.disabled &&
            (e.keyCode || !me.text || (Ext.fly(e.target).hasCls(me.headerCheckboxCls)))) {
            me.toggleAll(e);
        }
        else {
            return me.callParent([e, t, sortOnClick]);
        }
    },
 
    toggleAll: function(e) {
        var me = this,
            view = me.getView(),
            store = view.getStore(),
            checked = !me.allChecked;
 
        if (me.fireEvent('beforeheadercheckchange', me, checked, e) !== false) {
            // Only create and maintain a CellContext if there are consumers
            // in the form of event listeners. The event is a click on a 
            // column header and will have no position property.
            if (me.hasListeners.checkchange || me.hasListeners.beforecheckchange) {
                e.position = new Ext.grid.CellContext(view);
            }
 
            store.each(function(record, recordIndex) {
                me.setRecordCheck(record, recordIndex, checked, view.getCell(record, me));
            });
 
            me.setHeaderStatus(checked, e);
            me.fireEvent('headercheckchange', me, checked, e);
        }
    },
 
    setHeaderStatus: function(checked, e) {
        var me = this;
 
        // Will fire initially due to allChecked being undefined and using !==
        if (me.allChecked !== checked) {
            me.allChecked = checked;
 
            if (me.headerCheckbox) {
                me[checked ? 'addCls' : 'removeCls'](me.headerCheckedCls);
 
                if (me.useAriaElements) {
                    me.updateHeaderAriaDescription(checked);
                }
            }
        }
    },
 
    updateHeaderState: function(e) {
        var me = this;
 
        if (!me.headerStateTimer) {
            me.headerStateTimer = Ext.raf(me.doUpdateHeaderState, me);
        }
    },
 
    doUpdateHeaderState: function(e) {
        var me = this;
 
        me.headerStateTimer = null;
 
        // This is called on a timer, so ignore if it fires after destruction
        if (!me.destroyed && me.headerCheckbox) {
            me.setHeaderStatus(me.areAllChecked(), e);
        }
    },
 
    /**
     * Enables this CheckColumn.
     */
    onEnable: function() {
        this.callParent(arguments);
        this._setDisabled(false);
    },
 
    /**
     * Disables this CheckColumn.
     */
    onDisable: function() {
        this._setDisabled(true);
    },
 
    // Don't want to conflict with the Component method
    _setDisabled: function(disabled) {
        var me = this,
            cls = me.disabledCls,
            items;
 
        items = me.up('tablepanel').el.select(me.getCellSelector());
 
        if (disabled) {
            items.addCls(cls);
        }
        else {
            items.removeCls(cls);
        }
    },
 
    defaultRenderer: function(value, cellValues) {
        var me = this,
            cls = me.checkboxCls,
            tip = '',
            ariaElAttributes = {},
            ariaRenderConfigs = "",
            ariaAttr,
            ariaLabelledBy,
            ariaDescribedBy,
            spanElId;
 
        if (me.invert) {
            value = !value;
        }
 
        if (me.disabled) {
            cellValues.tdCls += ' ' + me.disabledCls;
        }
 
        if (value) {
            cls += ' ' + me.checkboxCheckedCls;
            tip = me.checkedTooltip;
        }
        else {
            tip = me.tooltip;
        }
 
        if (tip) {
            cellValues.tdAttr += ' data-qtip="' + Ext.htmlEncode(tip) + '"';
        }
 
        spanElId = me.id + '-spanEl-' + cellValues.rowIndex + cellValues.columnIndex;
        // User can change the state of checkbox (span element) by clicking anywhere in cell. Hence
        // making span as an active descendant to cell to guide the screenreader while focusing.
        cellValues.tdAttr += 'aria-activedescendant="' + spanElId + '"';
 
        if (me.useAriaElements) {
            // Selection column cannot have other aria attributes as it will
            // hinder the row selection announcement which we have binded 
            // through aria-describedby attribute below
            ariaElAttributes["aria-describedby"] = me.id + '-cell-description' +
            (!value ? '-not' : '') + '-selected';
            ariaElAttributes['aria-rowindex'] = Ext.Number.from(cellValues.rowIndex, 0) + 1;
        }
        else {
            ariaLabelledBy = me.getAriaLabelEl(me.ariaLabelledBy);
            ariaDescribedBy = me.getAriaLabelEl(me.ariaDescribedBy);
 
            if (ariaLabelledBy) {
                ariaElAttributes["aria-labelledby"] = ariaLabelledBy;
            }
            else if (me.ariaLabel) {
                ariaElAttributes["aria-label"] = me.ariaLabel || me.text;
            }
 
            if (ariaDescribedBy) {
                ariaElAttributes["aria-describedby"] = ariaDescribedBy;
            }
 
            // No need to set aria-checked here if cell is already rendered
            // We are handling it on 'setRecordCheck' method
            if (!this.getView().getCell(cellValues.record, cellValues.column)) {
                ariaElAttributes['aria-checked'] = !!value;
            }
 
            ariaElAttributes = Ext.apply(ariaElAttributes, me.getAriaAttributes());
        }
 
        for (ariaAttr in ariaElAttributes) {
            ariaRenderConfigs += ariaAttr + '="' + ariaElAttributes[ariaAttr] + '"';
        }
 
        // This will update the header state on the next animation frame
        // after all rows have been rendered.
        me.updateHeaderState();
 
        return '<span id="' + spanElId + '"' +
               'class="' + cls + '" role="' + me.checkboxAriaRole + '' +
                ariaRenderConfigs +
                (!me.ariaStaticRoles[me.checkboxAriaRole] ? ' tabIndex="0"' : '') +
               '></span>';
    },
 
    isRecordChecked: function(record) {
        var prop = this.property;
 
        if (prop) {
            return record[prop];
        }
 
        return record.get(this.dataIndex);
    },
 
    areAllChecked: function() {
        var me = this,
            store = me.getView().getStore(),
            records, len, i;
 
        if (!store.isBufferedStore && store.getCount() > 0) {
            records = store.getData().items;
            len = records.length;
 
            for (= 0; i < len; ++i) {
                if (!me.isRecordChecked(records[i])) {
                    return false;
                }
            }
 
            return true;
        }
    },
 
    setRecordCheck: function(record, recordIndex, checked, cell, e) {
        var me = this,
            prop = me.property,
            checkEl;
 
        // Only proceed if we NEED to change
        // eslint-disable-next-line eqeqeq
        if ((prop ? record[prop] : record.get(me.dataIndex)) != checked) {
            if (prop) {
                record[prop] = checked;
                me.updater(cell, checked);
            }
            else {
                record.set(me.dataIndex, checked);
            }
 
            checkEl = cell.querySelector('.' + me.checkboxCls);
 
            if (checkEl) {
                // On click, screenreader is announcing the state inconsistently when 
                // checkbox is accessed using mouse/cursor. Defer will force reader to
                // announce both the preceding and the concluding state of the checkbox.
                if (&& e.type === 'click') {
                    Ext.defer(function() {
                        checkEl.setAttribute('aria-checked', checked);
                    }, 500);
                }
                else {
                    checkEl.setAttribute('aria-checked', checked);
                }
            }
        }
    },
 
    updater: function(cell, value) {
        var me = this,
            tip;
 
        if (me.invert) {
            value = !value;
        }
 
        if (value) {
            tip = me.checkedTooltip;
        }
        else {
            tip = me.tooltip;
        }
 
        if (tip) {
            cell.setAttribute('data-qtip', tip);
        }
        else {
            cell.removeAttribute('data-qtip');
        }
 
        if (me.useAriaElements) {
            me.updateCellAriaDescription(null, value, cell);
        }
 
        cell = Ext.fly(cell);
 
        cell[me.disabled ? 'addCls' : 'removeCls'](me.disabledCls);
 
        // eslint-disable-next-line max-len
        Ext.fly(cell.down(me.getView().innerSelector, true).firstChild)[value ? 'addCls' : 'removeCls'](Ext.baseCSSPrefix + 'grid-checkcolumn-checked');
 
        // This will update the header state on the next animation frame
        // after all rows have been updated.
        me.updateHeaderState();
    },
 
    /**
     * @private
     */
    updateHeaderAriaDescription: function(isSelected) {
        var me = this;
 
        if (me.useAriaElements && me.ariaEl.dom) {
            me.ariaEl.dom.setAttribute('aria-describedby', me.id + '-header-description' +
                                       (!isSelected ? '-not' : '') + '-selected');
        }
    },
 
    /**
     * @private
     */
    updateCellAriaDescription: function(record, isSelected, cell) {
        var me = this,
            cellSpanEl, ariaRowIdx, rowDescribedNode, rowDescribedText;
 
        if (me.useAriaElements) {
            cell = cell || me.getView().getCell(record, me);
 
            if (cell) {
                cellSpanEl = cell.querySelector('.' + me.checkboxCls);
 
                if (cellSpanEl) {
                    rowDescribedNode = me.id + '-cell-description' +
                        (!isSelected ? '-not' : '') + '-selected';
                    ariaRowIdx = cellSpanEl.getAttribute('aria-rowindex');
                    rowDescribedText = !isSelected ? me.rowSelectText : me.rowDeselectText;
 
                    Ext.fly(rowDescribedNode).setText(
                        rowDescribedText.replace('{rowIdx}', ariaRowIdx)
                    );
                    cellSpanEl.setAttribute('aria-describedby', rowDescribedNode);
                }
            }
        }
    },
 
    doDestroy: function() {
        Ext.unraf(this.headerStateTimer);
        this.callParent();
    },
 
    privates: {
        /**
         * A method called by the render template to allow extra content after the header text.
         * Needs to be a seperate element to carry this. Cannot be a :after pseudo element
         * on one of the textual elements because we need to filter the click target to this
         * element for header checkbox clicking.
         * @private
         */
        afterText: function(out, values) {
            var me = this,
                id = me.id;
 
            out.push('<span role="presentation" class="', me.headerCheckboxCls, '"></span>');
 
            if (me.useAriaElements) {
                out.push(
                    '<span id="' + id + '-header-description-selected" class="' +
                        Ext.baseCSSPrefix + 'hidden-offsets">' + me.headerDeselectText + '</span>' +
                    '<span id="' + id + '-header-description-not-selected" class="' +
                        Ext.baseCSSPrefix + 'hidden-offsets">' + me.headerSelectText + '</span>' +
                    '<span id="' + id + '-cell-description-selected" class="' +
                        Ext.baseCSSPrefix + 'hidden-offsets">' + me.rowDeselectText +
                    '</span>' +
                    '<span id="' + id + '-cell-description-not-selected" class="' +
                        Ext.baseCSSPrefix + 'hidden-offsets">' + me.rowSelectText +
                    '</span>'
                );
            }
        }
    }
});