Невелика колекція ефектів для діалогових вікон з використанням CSS-анімації

17 хв. читання

Стилі і тренди змінюються і це вимагає від нас ініціалізації різних ефектів, які підходять для сучасного інтерфейсу. Цей набір містить деякі основні анімації, а також більш незвичні, з використанням техніки SVG морфінга для діалогів зі Snap.svg.

Зверніть увагу, що тестування відбувалося в останніх версіях браузерів.

Також, зверніть увагу, що IE11 не підтримує vieport одиниць в calc(), який ми використовували в деяких анімаціях перетворення.

Для діалогового вікна ми використовуємо наступну розмітку:

    <div class="dialog">
        <div class="dialog__overlay"></div>
        <div class="dialog__content">
            <h2><strong>Howdy</strong>, I'm a dialog box</h2><
            div><button class="action" data-dialog-close="">Close</button></div>
        </div>

Зверніть увагу, що в майбутньому ми зможемо використовувати нативний <dialog> елемент. Але в даний час підтримка як і раніше дуже слабка, IE, Firefox і Safari не підтримують його.

Як ви можете бачити, у нас є обгортка головного діалогового вікна, яка містить оверлей і контент діалогу.

Основним стилем для діалогу є (префікси постачальників опущені):

    .dialog,
    .dialog__overlay {
        width: 100%;
        height: 100%;
        top: 0;
        left: 0;
    }
     
    .dialog {
        position: fixed;
        display: flex;
        align-items: center;
        justify-content: center;
        pointer-events: none;
    }
     
    .dialog__overlay {
        position: absolute;
        z-index: 1;
        background: rgba(55, 58, 71, 0.9);
        opacity: 0;
        transition: opacity 0.3s;
    }
     
    .dialog--open .dialog__overlay {
        opacity: 1;
        pointer-events: auto;
    }
     
    .dialog__content {
        width: 50%;
        max-width: 560px;
        min-width: 290px;
        background: #fff;
        padding: 4em;
        text-align: center;
        position: relative;
        z-index: 5;
        opacity: 0;
    }
     
    .dialog--open .dialog__content {
        pointer-events: auto;
    }
     
    /* Content */
    .dialog h2 {
        margin: 0;
        font-weight: 400;
        font-size: 2em;
        padding: 0 0 2em;
        margin: 0;
    }

Ми використовуємо flexbox для головного елемента діалогового вікна для того, щоб центрувати вміст діалогу. Накладення з'явиться з переходом. Зверніть увагу, що вказівник подій не працює в IE < 11.

Приклад ефекту (Сандра) наступний:

    .dialog.dialog--open .dialog__content,
    .dialog.dialog--close .dialog__content {
        animation-duration: 0.3s;
        animation-fill-mode: forwards;
    }
     
    .dialog.dialog--open .dialog__content {
        animation-name: anim-open;
    }
     
    .dialog.dialog--close .dialog__content {
        animation-name: anim-close;
    }
     
    @keyframes anim-open {
        0% { opacity: 0; transform: scale3d(1.1, 1.1, 1); }
        100% { opacity: 1; transform: scale3d(1, 1, 1); }
    }
     
    @keyframes anim-close {
        0% { opacity: 1; }
        100% { opacity: 0; transform: scale3d(0.9, 0.9, 1); }
    }

З додаванням класів dialog--open і dialog--close, ми маємо можливість контролювати появу діалогу і його елементів.

Сценарій для діалогу полягає в наступному:

    ;( function( window ) {
         
     
        var support = { animations : Modernizr.cssanimations },
            animEndEventNames = { 'WebkitAnimation' : 'webkitAnimationEnd', 'OAnimation' : 'oAnimationEnd', 'msAnimation' : 'MSAnimationEnd', 'animation' : 'animationend' },
            animEndEventName = animEndEventNames[ Modernizr.prefixed( 'animation' ) ],
            onEndAnimation = function( el, callback ) {
                var onEndCallbackFn = function( ev ) {
                    if( support.animations ) {
                        if( ev.target != this ) return;
                        this.removeEventListener( animEndEventName, onEndCallbackFn );
                    }
                    if( callback && typeof callback === 'function' ) { callback.call(); }
                };
                if( support.animations ) {
                    el.addEventListener( animEndEventName, onEndCallbackFn );
                }
                else {
                    onEndCallbackFn();
                }
            };
     
        function extend( a, b ) {
            for( var key in b ) { 
                if( b.hasOwnProperty( key ) ) {
                    a[key] = b[key];
                }
            }
            return a;
        }
     
        function DialogFx( el, options ) {
            this.el = el;
            this.options = extend( {}, this.options );
            extend( this.options, options );
            this.ctrlClose = this.el.querySelector( '[data-dialog-close]' );
            this.isOpen = false;
            this._initEvents();
        }
     
        DialogFx.prototype.options = {
            // колбеки
            onOpenDialog : function() { return false; },
            onCloseDialog : function() { return false; }
        }
     
        DialogFx.prototype._initEvents = function() {
            var self = this;
     
            // закриття
            this.ctrlClose.addEventListener( 'click', this.toggle.bind(this) );
     
            // esc закриває діалог
            document.addEventListener( 'keydown', function( ev ) {
                var keyCode = ev.keyCode || ev.which;
                if( keyCode === 27 && self.isOpen ) {
                    self.toggle();
                }
            } );
     
            this.el.querySelector( '.dialog__overlay' ).addEventListener( 'click', this.toggle.bind(this) );
        }
     
        DialogFx.prototype.toggle = function() {
            var self = this;
            if( this.isOpen ) {
                classie.remove( this.el, 'dialog--open' );
                classie.add( self.el, 'dialog--close' );
                 
                onEndAnimation( this.el.querySelector( '.dialog__content' ), function() {
                    classie.remove( self.el, 'dialog--close' );
                } );
     
                // закриття колбеків
                this.options.onCloseDialog( this );
            }
            else {
                classie.add( this.el, 'dialog--open' );
     
                // відкриття колбеків
                this.options.onOpenDialog( this );
            }
            this.isOpen = !this.isOpen;
        };
     
        // додавання до глобального простору імен
        window.DialogFx = DialogFx;
     
    })( window );

Ми можемо викликати діалог наступним чином:

    <script src="js/classie.js"></script>
    <script src="js/dialogFx.js"></script>
    <script>
        (function() {
     
            var dlgtrigger = document.querySelector( '[data-dialog]' ),
                somedialog = document.getElementById( dlgtrigger.getAttribute( 'data-dialog' ) ),
                dlg = new DialogFx( somedialog );
     
            dlgtrigger.addEventListener( 'click', dlg.toggle.bind(dlg) );
     
        })();
    </script>

...де наша кнопка пуску має дата-атрибут data-dialog="somedialog".

Для SVG-ефектів (за винятком креслення ліній у Wilma) ми використовуємо Snap.svg для морфінга контурів SVG. Ми додаємо SVG форму в обгортку прямо в діалоговий вміст, а потім визначаємо шлях для перетворення в data-morph-open.

    (function() {
     
        var dlgtrigger = document.querySelector( '[data-dialog]' ),
     
            somedialog = document.getElementById( dlgtrigger.getAttribute( 'data-dialog' ) ),
            // svg..
            morphEl = somedialog.querySelector( '.morph-shape' ),
            s = Snap( morphEl.querySelector( 'svg' ) ),
            path = s.select( 'path' ),
            initialPath = path.attr('d'),
            steps = { 
                open : morphEl.getAttribute( 'data-morph-open' )
            },
            dlg = new DialogFx( somedialog, {
                onOpenDialog : function( instance ) {
                    // скидання
                    morphEl.querySelector( 'svg > path' ).setAttribute( 'd', initialPath );
                    // анімування
                    path.stop().animate( { 'path' : steps.open }, 300, mina.easein );
                }
            } );
     
        dlgtrigger.addEventListener( 'click', dlg.toggle.bind(dlg) );
     
    })();

Демо

Помітили помилку? Повідомте автору, для цього достатньо виділити текст з помилкою та натиснути Ctrl+Enter
Codeguida 5.6K
Приєднався: 8 місяців тому
Коментарі (0)

    Ще немає коментарів

Щоб залишити коментар необхідно авторизуватися.

Вхід / Реєстрація