Make WordPress Core

Ticket #47069: 47069.13.diff

File 47069.13.diff, 16.5 KB (added by dinhtungdu, 5 years ago)
  • src/js/_enqueues/lib/admin-bar.js

     
    44/**
    55 * Admin bar with Vanilla JS, no external dependencies.
    66 *
     7 * @since 5.3.1
     8 *
    79 * @param {Object} document  The document object.
    810 * @param {Object} window    The window object.
    911 * @param {Object} navigator The navigator object.
    1012 *
    1113 * @return {void}
    1214 */
    13 /* global hoverintent */
    1415( function( document, window, navigator ) {
    1516        document.addEventListener( 'DOMContentLoaded', function() {
    1617                var adminBar = document.getElementById( 'wpadminbar' ),
    17                         topMenuItems = adminBar.querySelectorAll( 'li.menupop' ),
    18                         allMenuItems = adminBar.querySelectorAll( '.ab-item' ),
    19                         adminBarLogout = document.getElementById( 'wp-admin-bar-logout' ),
    20                         adminBarSearchForm = document.getElementById( 'adminbarsearch' ),
    21                         shortlink = document.getElementById( 'wp-admin-bar-get-shortlink' ),
    22                         skipLink = adminBar.querySelector( '.screen-reader-shortcut' ),
    23                         mobileEvent = /Mobile\/.+Safari/.test( navigator.userAgent ) ? 'touchstart' : 'click',
     18                        topMenuItems,
     19                        allMenuItems,
     20                        adminBarLogout,
     21                        adminBarSearchForm,
     22                        shortlink,
     23                        skipLink,
     24                        mobileEvent,
     25                        fontFaceRegex,
    2426                        adminBarSearchInput,
    2527                        i;
    2628
    27                 /**
    28                  * Remove nojs class after the DOM is loaded.
    29                  */
    30                 adminBar.classList.remove( 'nojs' );
     29                if ( ! adminBar || ! ( 'querySelectorAll' in adminBar ) ) {
     30                        return;
     31                }
     32
     33                topMenuItems = adminBar.querySelectorAll( 'li.menupop' );
     34                allMenuItems = adminBar.querySelectorAll( '.ab-item' );
     35                adminBarLogout = document.getElementById( 'wp-admin-bar-logout' );
     36                adminBarSearchForm = document.getElementById( 'adminbarsearch' );
     37                shortlink = document.getElementById( 'wp-admin-bar-get-shortlink' );
     38                skipLink = adminBar.querySelector( '.screen-reader-shortcut' );
     39                mobileEvent = /Mobile\/.+Safari/.test( navigator.userAgent ) ? 'touchstart' : 'click';
     40                fontFaceRegex = /Android (1.0|1.1|1.5|1.6|2.0|2.1)|Nokia|Opera Mini|w(eb)?OSBrowser|webOS|UCWEB|Windows Phone OS 7|XBLWP7|ZuneWP7|MSIE 7/;
     41
     42                // Remove nojs class after the DOM is loaded.
     43                removeClass( adminBar, 'nojs' );
    3144
    3245                if ( 'ontouchstart' in window ) {
    33                         /**
    34                          * Remove hover class when the user touches outside the menu items.
    35                          */
     46                        // Remove hover class when the user touches outside the menu items.
    3647                        document.body.addEventListener( mobileEvent, function( e ) {
    3748                                if ( ! getClosest( e.target, 'li.menupop' ) ) {
    3849                                        removeAllHoverClass( topMenuItems );
    3950                                }
    4051                        } );
    4152
    42                         /**
    43                          * Add listener for menu items to toggle hover class by touches.
    44                          * Remove the callback later for better performance.
    45                          */
     53                        // Add listener for menu items to toggle hover class by touches.
     54                        // Remove the callback later for better performance.
    4655                        adminBar.addEventListener( 'touchstart', function bindMobileEvents() {
    4756                                for ( var i = 0; i < topMenuItems.length; i++ ) {
    4857                                        topMenuItems[i].addEventListener( 'click', mobileHover.bind( null, topMenuItems ) );
     
    5261                        } );
    5362                }
    5463
    55                 /**
    56                  * Scroll page to top when clicking on the admin bar.
    57                  */
     64                // Scroll page to top when clicking on the admin bar.
    5865                adminBar.addEventListener( 'click', scrollToTop );
    5966
    6067                for ( i = 0; i < topMenuItems.length; i++ ) {
    61                         /**
    62                          * Adds or removes the hover class based on the hover intent.
    63                          */
    64                         hoverintent(
     68                        // Adds or removes the hover class based on the hover intent.
     69                        window.hoverintent(
    6570                                topMenuItems[i],
    66                                 addHoverClass.bind( null, topMenuItems[i] ),
    67                                 removeHoverClass.bind( null, topMenuItems[i] )
     71                                addClass.bind( null, topMenuItems[i], 'hover' ),
     72                                removeClass.bind( null, topMenuItems[i], 'hover' )
    6873                        ).options( {
    6974                                timeout: 180
    7075                        } );
    7176
    72                         /**
    73                          * Toggle hover class if the enter key is pressed.
    74                          */
     77                        // Toggle hover class if the enter key is pressed.
    7578                        topMenuItems[i].addEventListener( 'keydown', toggleHoverIfEnter );
    7679                }
    7780
    78                 /**
    79                  * Remove hover class if the escape key is pressed.
    80                  */
     81                // Remove hover class if the escape key is pressed.
    8182                for ( i = 0; i < allMenuItems.length; i++ ) {
    8283                        allMenuItems[i].addEventListener( 'keydown', removeHoverIfEscape );
    8384                }
     
    8586                if ( adminBarSearchForm ) {
    8687                        adminBarSearchInput = document.getElementById( 'adminbar-search' );
    8788
    88                         /**
    89                          * Adds the adminbar-focused class on focus.
    90                          */
     89                        // Adds the adminbar-focused class on focus.
    9190                        adminBarSearchInput.addEventListener( 'focus', function() {
    92                                 adminBarSearchForm.classList.add( 'adminbar-focused' );
     91                                addClass( adminBarSearchForm, 'adminbar-focused' );
    9392                        } );
    9493
    95                         /**
    96                          * Removes the adminbar-focused class on blur.
    97                          */
     94                        // Removes the adminbar-focused class on blur.
    9895                        adminBarSearchInput.addEventListener( 'blur', function() {
    99                                 adminBarSearchForm.classList.remove( 'adminbar-focused' );
     96                                removeClass( adminBarSearchForm, 'adminbar-focused' );
    10097                        } );
    10198                }
    10299
    103                 /**
    104                  * Focus the target of skip link after pressing Enter.
    105                  */
    106                 skipLink.addEventListener( 'keydown', focusTargetAfterEnter );
     100                if ( skipLink ) {
     101                        // Focus the target of skip link after pressing Enter.
     102                        skipLink.addEventListener( 'keydown', focusTargetAfterEnter );
     103                }
    107104
    108105                if ( shortlink ) {
    109106                        shortlink.addEventListener( 'click', clickShortlink );
    110107                }
    111108
    112                 /**
    113                  * Prevents the toolbar from covering up content when a hash is present
    114                  * in the URL.
    115                  */
     109                // Prevents the toolbar from covering up content when a hash is present in the URL.
    116110                if ( window.location.hash ) {
    117111                        window.scrollBy( 0, -32 );
    118112                }
    119113
    120                 /**
    121                  * Add no-font-face class to body if needed.
    122                  */
    123                 if ( navigator.userAgent && document.body.className.indexOf( 'no-font-face' ) === -1 &&
    124                         /Android (1.0|1.1|1.5|1.6|2.0|2.1)|Nokia|Opera Mini|w(eb)?OSBrowser|webOS|UCWEB|Windows Phone OS 7|XBLWP7|ZuneWP7|MSIE 7/.test( navigator.userAgent ) ) {
    125                         document.body.className += ' no-font-face';
     114                // Add no-font-face class to body if needed.
     115                if (
     116                        navigator.userAgent &&
     117                        fontFaceRegex.test( navigator.userAgent ) &&
     118                        ! hasClass( document.body, 'no-font-face' )
     119                ) {
     120                        addClass( document.body, 'no-font-face' );
    126121                }
    127122
    128                 /**
    129                  * Clear sessionStorage on logging out.
    130                  */
    131                 adminBarLogout.addEventListener( 'click', emptySessionStorage );
     123                // Clear sessionStorage on logging out.
     124                if ( adminBarLogout ) {
     125                        adminBarLogout.addEventListener( 'click', emptySessionStorage );
     126                }
    132127        } );
    133128
    134129        /**
    135130         * Remove hover class for top level menu item when escape is pressed.
    136131         *
    137          * @since 5.3.0
     132         * @since 5.3.1
    138133         *
    139          * @param {Event} e The keydown event.
     134         * @param {Event} event The keydown event.
    140135         */
    141         function removeHoverIfEscape( e ) {
     136        function removeHoverIfEscape( event ) {
    142137                var wrapper;
    143138
    144                 if ( e.which != 27 ) {
     139                if ( event.which !== 27 ) {
    145140                        return;
    146141                }
    147142
    148                 wrapper = getClosest( e.target, '.menupop' );
     143                wrapper = getClosest( event.target, '.menupop' );
    149144
    150145                if ( ! wrapper ) {
    151146                        return;
    152147                }
    153148
    154149                wrapper.querySelector( '.menupop > .ab-item' ).focus();
    155                 removeHoverClass( wrapper );
     150                removeClass( wrapper, 'hover' );
    156151        }
    157152
    158153        /**
    159154         * Toggle hover class for top level menu item when enter is pressed.
    160155         *
    161          * @since 5.3.0
     156         * @since 5.3.1
    162157         *
    163          * @param {Event} e The keydown event.
     158         * @param {Event} event The keydown event.
    164159         */
    165         function toggleHoverIfEnter( e ) {
     160        function toggleHoverIfEnter( event ) {
    166161                var wrapper;
    167162
    168                 if ( e.which != 13 ) {
     163                if ( event.which !== 13 ) {
    169164                        return;
    170165                }
    171166
    172                 if ( !! getClosest( e.target, '.ab-sub-wrapper' ) ) {
     167                if ( !! getClosest( event.target, '.ab-sub-wrapper' ) ) {
    173168                        return;
    174169                }
    175170
    176                 wrapper = getClosest( e.target, '.menupop' );
     171                wrapper = getClosest( event.target, '.menupop' );
    177172
    178173                if ( ! wrapper ) {
    179174                        return;
    180175                }
    181176
    182                 e.preventDefault();
    183                 if ( hasHoverClass( wrapper ) ) {
    184                         removeHoverClass( wrapper );
     177                event.preventDefault();
     178
     179                if ( hasClass( wrapper, 'hover' ) ) {
     180                        removeClass( wrapper, 'hover' );
    185181                } else {
    186                         addHoverClass( wrapper );
     182                        addClass( wrapper, 'hover' );
    187183                }
    188184        }
    189185
    190186        /**
    191187         * Focus the target of skip link after pressing Enter.
    192188         *
    193          * @since 5.3.0
     189         * @since 5.3.1
    194190         *
    195          * @param {Event} e The keydown event.
     191         * @param {Event} event The keydown event.
    196192         */
    197         function focusTargetAfterEnter( e ) {
     193        function focusTargetAfterEnter( event ) {
    198194                var id, userAgent;
    199195
    200                 if ( 13 !==     e.which ) {
     196                if ( event.which !== 13 ) {
    201197                        return;
    202198                }
    203199
    204                 id = e.target.getAttribute( 'href' );
     200                id = event.target.getAttribute( 'href' );
    205201                userAgent = navigator.userAgent.toLowerCase();
    206202
    207                 if ( userAgent.indexOf( 'applewebkit' ) != -1 && id && id.charAt( 0 ) == '#' ) {
     203                if ( userAgent.indexOf( 'applewebkit' ) > -1 && id && id.charAt( 0 ) === '#' ) {
    208204                        setTimeout( function() {
    209205                                var target = document.getElementById( id.replace( '#', '' ) );
    210206
    211                                 target.setAttribute( 'tabIndex', '0' );
    212                                 target.focus();
     207                                if ( target ) {
     208                                        target.setAttribute( 'tabIndex', '0' );
     209                                        target.focus();
     210                                }
    213211                        }, 100 );
    214212                }
    215213        }
     
    217215        /**
    218216         * Toogle hover class for mobile devices.
    219217         *
    220          * @since 5.3.0
     218         * @since 5.3.1
    221219         *
    222220         * @param {NodeList} topMenuItems All menu items.
    223          * @param {Event} e The click event.
     221         * @param {Event} event The click event.
    224222         */
    225         function mobileHover( topMenuItems, e ) {
     223        function mobileHover( topMenuItems, event ) {
    226224                var wrapper;
    227225
    228                 if ( !! getClosest( e.target, '.ab-sub-wrapper' ) ) {
     226                if ( !! getClosest( event.target, '.ab-sub-wrapper' ) ) {
    229227                        return;
    230228                }
    231229
    232                 e.preventDefault();
     230                event.preventDefault();
    233231
    234                 wrapper = getClosest( e.target, '.menupop' );
     232                wrapper = getClosest( event.target, '.menupop' );
    235233
    236234                if ( ! wrapper ) {
    237235                        return;
    238236                }
    239237
    240                 if ( hasHoverClass( wrapper ) ) {
    241                         removeHoverClass( wrapper );
     238                if ( hasClass( wrapper, 'hover' ) ) {
     239                        removeClass( wrapper, 'hover' );
    242240                } else {
    243241                        removeAllHoverClass( topMenuItems );
    244                         addHoverClass( wrapper );
     242                        addClass( wrapper, 'hover' );
    245243                }
    246244        }
    247245
     
    249247         * Handles the click on the Shortlink link in the adminbar.
    250248         *
    251249         * @since 3.1.0
    252          * @since 5.3.0 Use querySelector to clean up the function.
    253          *
    254          * @param {Event} e The click event.
     250         * @since 5.3.1 Use querySelector to clean up the function.
    255251         *
     252         * @param {Event} event The click event.
    256253         * @return {boolean} Returns false to prevent default click behavior.
    257254         */
    258         function clickShortlink( e ) {
    259                 var wrapper = e.target.parentNode,
     255        function clickShortlink( event ) {
     256                var wrapper = event.target.parentNode,
     257                        input;
     258
     259                if ( wrapper ) {
    260260                        input = wrapper.querySelector( '.shortlink-input' );
     261                }
     262
     263                if ( ! input ) {
     264                        return;
     265                }
    261266
    262                 // IE doesn't support preventDefault, and does support returnValue
    263                 if ( e.preventDefault ) {
    264                         e.preventDefault();
     267                // (Old) IE doesn't support preventDefault, and does support returnValue
     268                if ( event.preventDefault ) {
     269                        event.preventDefault();
    265270                }
    266                 e.returnValue = false;
    267271
    268                 wrapper.classList.add( 'selected' );
     272                event.returnValue = false;
     273
     274                addClass( wrapper, 'selected' );
     275
    269276                input.focus();
    270277                input.select();
    271278                input.onblur = function() {
    272                         wrapper.classList.remove( 'selected' );
     279                        removeClass( wrapper, 'selected' );
    273280                };
    274281
    275282                return false;
     
    278285        /**
    279286         * Clear sessionStorage on logging out.
    280287         *
    281          * @since 5.3.0
     288         * @since 5.3.1
    282289         */
    283290        function emptySessionStorage() {
    284291                if ( 'sessionStorage' in window ) {
    285292                        try {
    286293                                for ( var key in sessionStorage ) {
    287                                         if ( key.indexOf( 'wp-autosave-' ) != -1 ) {
     294                                        if ( key.indexOf( 'wp-autosave-' ) > -1 ) {
    288295                                                sessionStorage.removeItem( key );
    289296                                        }
    290297                                }
    291                         } catch ( e ) {}
     298                        } catch ( er ) {}
    292299                }
    293300        }
    294301
    295302        /**
    296          * Check if menu item has hover class.
     303         * Check if element has class.
    297304         *
    298          * @since 5.3.0
     305         * @since 5.3.1
    299306         *
    300          * @param {HTMLElement} item Menu item Element.
     307         * @param {HTMLElement} element The HTML element.
     308         * @param {String}      className The class name.
     309         * @return {bool} Whether the element has the className.
    301310         */
    302         function hasHoverClass( item ) {
    303                 return item.classList.contains( 'hover' );
     311        function hasClass( element, className ) {
     312                var classNames;
     313
     314                if ( element ) {
     315                        if ( element.classList && element.classList.contains ) {
     316                                return element.classList.contains( className );
     317                        } else if ( element.className ) {
     318                                classNames = element.className.split( ' ' );
     319                                return classNames.indexOf( className ) > -1;
     320                        }
     321                }
     322
     323                return false;
    304324        }
    305325
    306326        /**
    307          * Add hover class for menu item.
     327         * Add class to an element.
    308328         *
    309          * @since 5.3.0
     329         * @since 5.3.1
    310330         *
    311          * @param {HTMLElement} item Menu item Element.
     331         * @param {HTMLElement} element The HTML element.
     332         * @param {String}      className The class name.
    312333         */
    313         function addHoverClass( item ) {
    314                 item.classList.add( 'hover' );
     334        function addClass( element, className ) {
     335                if ( element ) {
     336                        if ( element.classList && element.classList.add ) {
     337                                element.classList.add( className );
     338                        } else if ( ! hasClass( element, className ) ) {
     339                                if ( element.className ) {
     340                                        element.className += ' ';
     341                                }
     342
     343                                element.className += className;
     344                        }
     345                }
    315346        }
    316347
    317348        /**
    318349         * Remove hover class for menu item.
    319350         *
    320          * @since 5.3.0
     351         * @since 5.3.1
    321352         *
    322          * @param {HTMLElement} item Menu item Element.
     353         * @param {HTMLElement} element The HTML element.
     354         * @param {String}      className The class name.
    323355         */
    324         function removeHoverClass( item ) {
    325                 item.classList.remove( 'hover' );
     356        function removeClass( element, className ) {
     357                var testName,
     358                        classes;
     359
     360                if ( element && hasClass( element, className ) ) {
     361                        if ( element.classList && element.classList.remove ) {
     362                                element.classList.remove( className );
     363                        } else {
     364                                testName = ' ' + className + ' ';
     365                                classes = ' ' + element.className + ' ';
     366
     367                                while ( classes.indexOf( testName ) > -1 ) {
     368                                        classes = classes.replace( testName, '' );
     369                                }
     370
     371                                element.className = classes.replace( /^[\s]+|[\s]+$/g, '' );
     372                        }
     373                }
     374
    326375        }
    327376
    328377        /**
    329378         * Remove hover class for all menu items.
    330379         *
    331          * @since 5.3.0
     380         * @since 5.3.1
    332381         *
    333382         * @param {NodeList} topMenuItems All menu items.
    334383         */
    335384        function removeAllHoverClass( topMenuItems ) {
    336                 for ( var i = 0; i < topMenuItems.length; i++ ) {
    337                         if ( hasHoverClass( topMenuItems[i] ) ) {
    338                                 removeHoverClass( topMenuItems[i] );
     385                if ( topMenuItems && topMenuItems.length ) {
     386                        for ( var i = 0; i < topMenuItems.length; i++ ) {
     387                                removeClass( topMenuItems[i], 'hover' );
    339388                        }
    340389                }
    341390        }
     
    354403                if (
    355404                        event.target &&
    356405                        event.target.id &&
    357                         event.target.id != 'wpadminbar' &&
    358                         event.target.id != 'wp-admin-bar-top-secondary'
     406                        event.target.id !== 'wpadminbar' &&
     407                        event.target.id !== 'wp-admin-bar-top-secondary'
    359408                ) {
    360409                        return;
    361410                }
     
    374423        /**
    375424         * Get closest Element.
    376425         *
    377          * @since 5.3.0
     426         * @since 5.3.1
    378427         *
    379428         * @param {HTMLElement} el Element to get parent.
    380429         * @param {string} selector CSS selector to match.
    381430         */
    382431        function getClosest( el, selector ) {
    383                 if ( ! Element.prototype.matches ) {
    384                         Element.prototype.matches =
    385                                 Element.prototype.matchesSelector ||
    386                                 Element.prototype.mozMatchesSelector ||
    387                                 Element.prototype.msMatchesSelector ||
    388                                 Element.prototype.oMatchesSelector ||
    389                                 Element.prototype.webkitMatchesSelector ||
    390                                 function( s ) {
    391                                         var matches = ( this.document || this.ownerDocument ).querySelectorAll( s ),
    392                                                 i = matches.length;
    393                                         while ( --i >= 0 && matches.item( i ) !== this ) { }
    394                                         return i > -1;
    395                                 };
    396                 }
    397 
    398                 // Get the closest matching elent
    399                 for ( ; el && el !== document; el = el.parentNode ) {
    400                         if ( el.matches( selector ) ) {
    401                                 return el;
     432                if ( window.Element ) {
     433                        if ( ! window.Element.prototype.matches ) {
     434                                // Polyfill from https://developer.mozilla.org/en-US/docs/Web/API/Element/matches.
     435                                window.Element.prototype.matches =
     436                                        window.Element.prototype.matchesSelector ||
     437                                        window.Element.prototype.mozMatchesSelector ||
     438                                        window.Element.prototype.msMatchesSelector ||
     439                                        window.Element.prototype.oMatchesSelector ||
     440                                        window.Element.prototype.webkitMatchesSelector ||
     441                                        function( s ) {
     442                                                var matches = ( this.document || this.ownerDocument ).querySelectorAll( s ),
     443                                                        i = matches.length;
     444
     445                                                while ( --i >= 0 && matches.item( i ) !== this ) { }
     446
     447                                                return i > -1;
     448                                        };
     449                        }
     450
     451                        // Get the closest matching elent
     452                        for ( ; el && el !== document; el = el.parentNode ) {
     453                                if ( el.matches( selector ) ) {
     454                                        return el;
     455                                }
    402456                        }
    403457                }
     458
    404459                return null;
    405460        }
    406461} )( document, window, navigator );