Пакет подключаемых модулей|Инкапсулируйте собственный подключаемый модуль карусели - переключение левой и правой версии

JavaScript
Пакет подключаемых модулей|Инкапсулируйте собственный подключаемый модуль карусели - переключение левой и правой версии

предыдущий постКейс | Родной почерк карусель - блеклая и блеклая версияЕсть еще много недостатков, я хотел бы поблагодарить вас большое здесьcsdokerЦенные мнения и рекомендации даны 🙏, поэтому автор решил повторно улучшить случай карты карусели и планирует сделать простую версию пакета плагина левой и правой карты карусели;

1. Предисловие к упаковке подключаемых модулей (основы упаковки)

карта разума

(1) Идея сменной упаковки

1. Идея построения гибкой платформы

Чтобы инкапсулировать плагин, в первую очередь необходимо проанализировать, проанализировать конечный эффект и элементы конфигурации, которые могут поддерживаться.

2. Основание:

  • => накопление опыта
  • => Исследуйте отличные плагины других людей 😄

3. Цель:

Цель написания плагинов — облегчить дальнейшую разработку, и они должны использоваться большим количеством людей (с открытым исходным кодом).

4. Требования:

  • -> Professional, производительность плагинов, которые мы пишем, лучше (код лучше)
  • -> Простота в использовании, другие очень просты в использовании
  • -> Мощный, насколько это возможно в случае лаконичного кода, поддержка более мощных эффектов
  • -> Гибкая, пишите как можно больше на основе родных JS, не полагайтесь на другие библиотеки или контент (то есть приносят прагматизм)
  • -> Расширяемый, обновляемый, учитывайте обратную совместимость более низкой версии при обновлении и не слишком далеко от исходной версии
  • ......так далее

(2) Идея инкапсуляции левого и правого плагинов карусели

Инкапсуляция плагинов, компонентов, библиотек классов и т. д. в основном основана на объектно-ориентированном мышлении.

  • Объектно-ориентированное мышление может гарантировать отсутствие связи между каждым экземпляром.
  • Основная цель объектно-ориентированного подхода: инкапсуляция подключаемых компонентов.
  • Плагин, который мы инкапсулируем, представляет собой класс
    • Изображение карусели, реализованное на основе каждого вызова плагина, является экземпляром созданного класса.
  • Для каруселей:
    • 1. Основная основная информация должна быть конфиденциальной
    • 2. Некоторые методы работы должны быть общедоступными

(3) Формальная обработка параметров в пакете подключаемых модулей

Способ 1: установите его как формальный параметр и передавайте по одному

  • Функции:
    • -> Порядок, в котором передаются фактические параметры, должен соответствовать порядку, в котором устанавливаются формальные параметры.
    • -> Нехорошо устанавливать значения по умолчанию для формальных параметров
    • -> Если некоторые предметы в середине не будут переданы, предметы, переданные позже, будут перемещены
  • используемые сцены:
    • Как правило, эта ситуация используется только в том случае, если параметров очень мало (обычно не более двух).

Способ 2: обработка на основе пар ключ-значение объекта

  • Функции:
    • -> может быть пройдено или не пройдено
    • -> Порядок доставки также произвольный, потому что все они отмечены на основе имени атрибута, пока атрибут и значение соответствуют, порядок не имеет значения
    • -> легко расширяется
    • -> Для информации, которая не передается, мы устанавливаем для нее значение по умолчанию
  • используемые сцены:
    • В случае передачи нескольких параметров мы обычно имеем дело с парами ключ-значение объекта.

Во-вторых, пакет карты карусели - HTML

В этой части мы можем написать небольшую полку, и пользователи могут напрямую CV, когда они ее используют, заменять свои собственные изображения и настраивать некоторые свои детали;Адрес склада

<!-- 轮播图整体容器 -->
<div class="xiaozhima-container">
    <!-- 所有图片容器 -->
    <div class="xiaozhima-wrapper">
        <!-- 每个图片容器 -->
        <div class="xiaozhima-slide">
            <!-- 插入您的图片 -->
            <img src=" " alt="">
        </div>
        <div class="xiaozhima-slide">
            <img src=" " alt="">
        </div>
        <div class="xiaozhima-slide">
            <img src=" " alt="">
        </div>
        <div class="xiaozhima-slide">
            <img src=" " alt="">
        </div>
    </div>
    <!-- 分页器容器 -->
    <div class="xiaozhima-pagination"></div>
    <!-- 左右按钮容器 -->
    <a href="javascript:;" class="xiaozhima-arrow-prev"></a>
    <a href="javascript:;" class="xiaozhima-arrow-next"></a>
</div>

3. Инкапсуляция карусели — CSS

В этой части мы также пишем небольшую полку, настраиваем контент, которого нам нужно добиться, а подробный стиль может указать сам пользователь.

1. Как использовать:

  • импортировать по имениbanner-plugin.cssТаблица стилей может быть;Адрес склада(Вы также можете сами создать файл CSS и вставить код прямо в него);

2. Что нужно настроить пользователю:

  • Общий контейнер карусели:
    • width
    • height
  • Вставить в свою картинку: (можно просто загрузить картинку не меняя ее)
    • Картины и стили картинок

3. Стили, которые нужно динамически вычислять в JS

  • Для всех контейнеров изображений:

    • ширина : По количеству СЛАЙДОВ (включая клоны) * ширина контейнера КОНТЕЙНЕРА
    • переход: устанавливается в JS
    • слева: какой СЛАЙД отображается изначально, ЛЕВОЕ значение должно быть рассчитано
  • Для каждого контейнера изображения:

    • ширина: должна соответствовать ширине КОНТЕЙНЕРА, переданного пользователем (обрабатывается в JS).
  • Левая и правая кнопки: (изображения Пекина конвертируются в кодировку BASE64)

    • Лучше не использовать картинки для стилевых ресурсов в плагине (т.к. это требует дополнительного импорта пользователем, что не способствует продвижению и использованию плагина) => Если вам нужно использовать картинки, лучше установить картинки в кодировку BASE64

    • Интернет-сайт, который конвертирует изображения в кодировку BASE64:tool.css-js.com/base64.html

4. Специальный код CSS

.xiaozhima-container {
	/* 
	 * 需要用户后期自己配置的样式
	 *   width
	 *   height
	 */
	position: relative;
	box-sizing: border-box;
	width: 100%;
	overflow: hidden;
}

.xiaozhima-container .xiaozhima-wrapper {
	/* 
	 * 需要在JS中动态计算的样式
	 *   width 根据SLIDE的个数(含克隆的)* CONTAINER容器的宽度
	 *   transition 是在JS中设置的
	 *   left 初始展示哪一个SLIDE,LEFT值就应该是计算好的
	 */
	display: flex;
	box-sizing: border-box;
	position: absolute;
	top: 0;
	height: 100%;
}

.xiaozhima-container .xiaozhima-wrapper .xiaozhima-slide {
	/*
	 * 需要用户后期自己配置的样式
	 *   width 和CONTAINER的宽度保持一致(JS中处理)
	 */
	box-sizing: border-box;
	height: 100%;
	overflow: hidden;
}

.xiaozhima-container .xiaozhima-wrapper .xiaozhima-slide img {
	display: block;
	width: 100%;
	height: 100%;
}

.xiaozhima-container .xiaozhima-pagination {
	position: absolute;
	bottom: 10px;
	left: 50%;
	z-index: 999;
	transform: translateX(-50%);
	padding: 3px 6px;
	background: rgba(255, 255, 255, .3);
	font-size: 0;
	border-radius: 10px;
}

.xiaozhima-container .xiaozhima-pagination span {
	display: inline-block;
	margin: 0 6px;
	width: 10px;
	height: 10px;
	border-radius: 50%;
	background: lightblue;
	cursor: pointer;
}

.xiaozhima-container .xiaozhima-pagination span.active {
	background: lightcoral;
}

.xiaozhima-arrow-prev,
.xiaozhima-arrow-next {
	display: none;
	position: absolute;
	z-index: 999;
	top: 50%;
	margin-top: -22.5px;
	width: 30px;
	height: 45px;
	/* 插件中的样式资源最好不要使用图片(因为这样需要用户额外的导入,不利于插件的推广和使用) => 需要使用图片,我们最好把图片设置为BASE64编码 */
	background: url("") no-repeat 0 0;
}

.xiaozhima-arrow-prev {
	left: 0;
}

.xiaozhima-arrow-next {
	right: 0;
	background-position: -50px 0;
}

.xiaozhima-container:hover .xiaozhima-arrow-prev,
.xiaozhima-container:hover .xiaozhima-arrow-next {
	display: block;
}

В-четвертых, пакет карты карусели — JS (ключ)

(1), первый шаг - анализ функций и элементов конфигурации

1. Цель:

  • пакет одинbannerPluginспособ, который позволяет указанному контейнеру добиться эффекта левого и правого варианта карусели;

2. Используйте синтаксис:

  • bannerPligin([container],[options]) ;

3. Описание параметра:

  • контейнер: этикетка контейнера
  • options: требуются элементы конфигурации (много параметров настраивается пользователем, поэтому мы используемТипы данных объекта)

Подробное объяснение элемента конфигурации параметров параметров

(2), второй шаг - инициализация параметра

Выше мы упоминали, что есть два способа обработки параметров.Здесь мы выбираем обработку их на основе пар ключ-значение объекта;

Здесь мы используем метод глубокого сравнения и итерации цикла, который мы создали вчера (Глубокое клонирование VS поверхностное клонирование | Глубокое сравнение VS поверхностное сравнение | Функции обратного вызова) и методы определения типа данных, статьяПреимущества и недостатки четырех методов определения типа данных в JSупомянуто в конце, я положил его здесь первым в виде комплекта

1. Инструментарий

Адрес складаВы также можете напрямую резюме

/* 工具包 */
(function () {
    // 数据类型检测
    let class2type = {};
    ["Boolean", "Number", "String", "Function", "Array", "Date", "RegExp", "Object", "Error", "Symbol"].forEach(item => {
        class2type["[object " + item + "]"] = item.toLowerCase();
    });

    function toType(obj) {
        if (obj == null) return obj + "";
        return typeof obj === "object" || typeof obj === "function" ?
            class2type[class2type.toString.call(obj)] || "object" :
            typeof obj;
    }

    // 检测是否为函数
    function isFunction(obj) {
        return typeof obj === "function" && typeof obj.nodeType !== "number";
    }

    // 检测window
    function isWindow(obj) {
        return obj != null && obj === obj.window;
    }

    // 数组/类数组
    function isArrayLike(obj) {
        var length = !!obj && "length" in obj && obj.length,
            type = toType(obj);
        if (isFunction(obj) || isWindow(obj)) return false;
        return type === "array" || length === 0 || typeof length === "number" && length > 0 && (length - 1) in obj;
    }

    // 循环迭代
    function _each(obj, callback, context = window) {
        if (/^(ARRAY|OBJECT)$/i.test(obj.constructor)) {
            obj = _cloneDeep(obj);
        }
        if (isArrayLike(obj)) {
            for (let i = 0; i < obj.length; i++) {
                let res = callback && callback.call(context, obj[i], i);
                if (res === false) break;
                if (res !== undefined) obj[i] = res;
            }
        } else {
            for (let key in obj) {
                if (!obj.hasOwnProperty(key)) break;
                let res = callback && callback.call(context, obj[key], key);
                if (res === false) break;
                if (res !== undefined) obj[key] = res;
            }
        }
        return obj;
    }

    // 深克隆
    function _cloneDeep(obj) {
        if (obj === null) return null;
        if (typeof obj !== "object") return obj;
        if (obj instanceof RegExp) return new RegExp(obj);
        if (obj instanceof Date) return new Date(obj);
        let cloneObj = new obj.constructor;
        for (let key in obj) {
            if (!obj.hasOwnProperty(key)) break;
            cloneObj[key] = _cloneDeep(obj[key]);
        }
        return cloneObj;
    }

    // 深比较
    function _assignDeep(obj1, obj2) {
        let obj = _cloneDeep(obj1);
        for (let key in obj2) {
            if (!obj2.hasOwnProperty(key)) break;
            let v2 = obj2[key],
                v1 = obj[key];
            if ((v1 !== null && typeof v1 === "object") && (v2 !== null && typeof v2 === "object")) {
                obj[key] = _assignDeep(v1, v2);
                continue;
            }
            obj[key] = v2;
        }
        return obj;
    }

    ['_each', '_cloneDeep', '_assignDeep', 'toType', 'isFunction', 'isWindow', 'isArrayLike'].forEach(item => {
        window[item] = eval(item);
    });
})();

2. Цель инициализации параметра

  • Настройте информацию о параметре по умолчанию и перейдите к информации по умолчанию, если пользователь не передает соответствующий параметр.
function bannerPlugin(container, options = {}) {
    // 参数初始化:默认配置信息
    let defaultParams = {
        initialSlide: 0,
        autoplay: 3000,
        speed: 300,
        pagination: {
            el: '.xiaozhima-pagination',
            clickable: true
        },
        navigation: {
            nextEl: '.xiaozhima-arrow-next',
            prevEl: '.xiaozhima-arrow-prev'
        },
        on: {
            init() {},
            transitionStart() {},
            transitionEnd() {}
        }
    };
    // 利用深比较,把用户传递进来的参数和默认配置合并(达到替换默认配置的效果)
    options = _assignDeep(defaultParams, options);

    // 判断用户传递进来的第一个参数是否为元素容器,支持直接传递标签
    typeof container === "string" ? container = document.querySelector(container) : null;
    if (!container || container.nodeType !== 1) {
        throw new TypeError('container must be an element!');
    }  
    return new Banner(container, options);
}

(3), третий шаг - настроить приватные свойства каждой карусели

1. Используйте метод класса и экземпляра, чтобы понять, что каждое изображение карусели не влияет друг на друга.

  • Когда пользователь использует его, каждая информация о конфигурации должна быть частной для текущей карусели, поэтому настроенные нами параметры по умолчанию должны быть частными свойствами каждого экземпляра.
class Banner {
    constructor(container, options) {
        // 把传递进来的信息都挂载到当前类的实例上
        // 1.信息都作为他的私有属性(这样每一个实例之间互不影响)
        // 2.挂载到实例上,以后在当前类的其它方法中,只要保证THIS是实例,都可以基于THIS.XXX获取和操作
        _each(options, (item, key) => {
            this[key] = item;
        });
        this.container = container;
        this.activeIndex = this.initialSlide;
    }
}

(4), четвертый шаг - настроить публичный метод (ключ)

  • Выше мы добавили приватные свойства к каждому экземпляру карусели, а следующие общедоступные методы являются ключевым содержимым.

1. Вычислительная структура и функции стиля

  • После завершения каждой карусели необходимой операцией является получение элемента
computed() {
    let {
        container,
        pagination,
        navigation
    } = this;

    // 轮播图
    this.wrapper = container.querySelector('.xiaozhima-wrapper');
    this.slidesTrue = container.querySelectorAll('.xiaozhima-slide');
    // 克隆第一张到容器的末尾
    this.wrapper.appendChild(this.slidesTrue[0].cloneNode(true));
    this.slides = container.querySelectorAll('.xiaozhima-slide');

    // 分页器
    this.paginationBox = null;
    this.paginationList = null;
    if (toType(pagination) === "object") {
        let el = pagination.el;
        if (el) {
            this.paginationBox = container.querySelector(el);
            // 创建SPAN
            let str = ``;
            _each(this.slidesTrue, item => {
                str += `<span></span>`;
            });
            this.paginationBox.innerHTML = str;
            this.paginationList = this.paginationBox.querySelectorAll('span');
        }
    }

    // 左右切换
    this.arrowPrev = null;
    this.arrowNext = null;
    if (toType(navigation) === "object") {
        navigation.prevEl ? this.arrowPrev = container.querySelector(navigation.prevEl) : null;
        navigation.nextEl ? this.arrowNext = container.querySelector(navigation.nextEl) : null;
    }

    // 控制元素的样式(包含初始展示哪一个)
    this.changeWidth = parseFloat(getComputedStyle(container).width);
    this.activeIndex = this.activeIndex < 0 ? 0 : (this.activeIndex > this.slides.length - 1 ? this.slides.length - 1 : this.activeIndex);
    this.wrapper.style.width = `${this.changeWidth*this.slides.length}px`;
    this.wrapper.style.transition = `left ${this.speed}ms`;
    this.wrapper.style.left = `${-this.activeIndex*this.changeWidth}px`;
    _each(this.slides, item => {
        item.style.width = `${this.changeWidth}px`;
    });
}

2, функция переключения, реализованная на фиг.8

// 实现轮播图切换
change(now = false) {
    //now 传值是立即切换,不传是默认的有动画切换
    let {
        wrapper,
        speed,
        activeIndex,
        changeWidth,
        on
    } = this;
    let isO = toType(on) === "object" ? true : false,
        transitionStart = isO ? on.transitionStart : null,
        transitionEnd = isO ? on.transitionEnd : null;
    
    // 切换之前触发的钩子函数
    !now && transitionStart ? transitionStart.call(this, this) : null;

    // 如果传了now 立即切换,则不需要有动画
    wrapper.style.transitionDuration = `${now?0:speed}ms`;
    wrapper.style.left = `${-activeIndex*changeWidth}px`;
    if (now) {
        // 如果立即切换我需要让上面先渲染一次,利用读写分离;
        wrapper.offsetWidth;
    } 

    // 切换之后触发的钩子函数
    let fn = () => {
        !now && transitionEnd ? transitionEnd.call(this, this) : null;
        // 每一次都会重新监听,所以监听完需要把上一次监听的移除掉
        wrapper.removeEventListener('transitionend', fn);
    };
    wrapper.addEventListener('transitionend', fn);
}

3. Функция автоматического вращения

// 自动轮播
autoMove() {
    if (this.activeIndex === this.slides.length - 1) {
        this.activeIndex = 0;
        this.change(true);
    }
    this.activeIndex++;
    this.change();
}

4, функция выравнивания фокуса

// 实现焦点对齐
autoFocus() {
    let {
        paginationList,
        activeIndex,
        slides
    } = this;
    if (!paginationList) return;
    activeIndex === slides.length - 1 ? activeIndex = 0 : null;
    _each(paginationList, (item, index) => {
        if (index === activeIndex) {
            item.className = 'active';
            return;
        }
        item.className = '';
    });
}

(5) Шаг 5 — Последовательность выполнения и условия общедоступных методов (выделено)

  • Теперь мы инкапсулировали метод для реализации функции карусели, но этот метод мертв, и ценно только его выполнение.
  • В настоящее время мы должны учитывать, что, хотя мы создали эти методы, параметры, переданные пользователем, могут не нуждаться в их использовании, поэтому нам также необходимо судить, нужны ли они пользователю, и если пользователю это нужно, то что наш исполнительный лист? ;
init() {
    // THIS:当前类的实例
    // 入口,在这里控制代码执行的逻辑顺序
    
    // 先基于它获取元素后,才可以获取实例上的信息
    this.computed();

    let {
        autoplay,
        autoMove,
        container,
        pagination,
        paginationList,
        arrowNext,
        arrowPrev
    } = this;

    // 控制是否自动切换
    if (autoplay) {
        this.autoTimer = setInterval(autoMove.bind(this), autoplay);
        // 箭头函数中的THIS是上下文中的,也就是实例
        container.onmouseenter = () => clearInterval(this.autoTimer);
        container.onmouseleave = () => this.autoTimer = setInterval(autoMove.bind(this), autoplay);
    }

    // 控制焦点切换
    if (toType(pagination) === "object" && pagination.clickable === true && paginationList) {
        _each(paginationList, (item, index) => {
            item.onclick = () => {
                let {
                    activeIndex,
                    slides
                } = this;
                if ((index === activeIndex) || (index === 0 && activeIndex === slides.length - 1)) return;
                this.activeIndex = index;
                this.change();
            };
        });
    }

    // 控制左右按钮切换
    arrowNext ? arrowNext.onclick = autoMove.bind(this) : null;
    arrowPrev ? arrowPrev.onclick = () => {
        if (this.activeIndex === 0) {
            this.activeIndex = this.slides.length - 1;
            this.change(true);
        }
        this.activeIndex--;
        this.change();
    } : null;

    // 初始化完成,触发INIT回调函数
    if (toType(this.on) === "object" && isFunction(this.on.init)) {
        // 把回调函数执行,让方法中的THIS是实例,并且传递的第一个参数也是实例
        this.on.init.call(this, this);
    }
}

(6), шестой шаг - оптимизация интеграции (ключ)

Резюме приведенного выше кода JS

  • Вы можете использовать JS для заливки;Адрес склада
    • Автор не указал здесь адрес, и вам также необходимо создатьbanner-plugin.jsJS файл, вставьте код прямо в него;
(function () {
    class Banner {
        constructor(container, options) {
            // 把传递进来的信息都挂载到当前类的实例上
            // 1.信息都作为他的私有属性(这样每一个实例之间互不影响)
            // 2.挂载到实例上,以后在当前类的其它方法中,只要保证THIS是实例,都可以基于THIS.XXX获取和操作
            _each(options, (item, key) => {
                this[key] = item;
            });
            this.container = container;
            this.activeIndex = this.initialSlide;
            this.init();
        }
        init() {
            // THIS:当前类的实例
            // 入口,在这里控制代码执行的逻辑顺序

            // 先基于它获取元素后,才可以获取实例上的信息
            this.computed();

            let {
                autoplay,
                autoMove,
                container,
                pagination,
                paginationList,
                arrowNext,
                arrowPrev
            } = this;

            // 控制是否自动切换
            if (autoplay) {
                this.autoTimer = setInterval(autoMove.bind(this), autoplay);
                // 箭头函数中的THIS是上下文中的,也就是实例
                container.onmouseenter = () => clearInterval(this.autoTimer);
                container.onmouseleave = () => this.autoTimer = setInterval(autoMove.bind(this), autoplay);
            }

            // 控制焦点切换
            if (toType(pagination) === "object" && pagination.clickable === true && paginationList) {
                _each(paginationList, (item, index) => {
                    item.onclick = () => {
                        let {
                            activeIndex,
                            slides
                        } = this;
                        if ((index === activeIndex) || (index === 0 && activeIndex === slides.length - 1)) return;
                        this.activeIndex = index;
                        this.change();
                    };
                });
            }

            // 控制左右按钮切换
            arrowNext ? arrowNext.onclick = autoMove.bind(this) : null;
            arrowPrev ? arrowPrev.onclick = () => {
                if (this.activeIndex === 0) {
                    this.activeIndex = this.slides.length - 1;
                    this.change(true);
                }
                this.activeIndex--;
                this.change();
            } : null;

            // 初始化完成,触发INIT回调函数
            if (toType(this.on) === "object" && isFunction(this.on.init)) {
                // 把回调函数执行,让方法中的THIS是实例,并且传递的第一个参数也是实例
                this.on.init.call(this, this);
            }
        }


        // 计算结构和样式
        computed() {
            let {
                container,
                pagination,
                navigation
            } = this;

            // 轮播图
            this.wrapper = container.querySelector('.xiaozhima-wrapper');
            this.slidesTrue = container.querySelectorAll('.xiaozhima-slide');
            // 克隆第一张到容器的末尾
            this.wrapper.appendChild(this.slidesTrue[0].cloneNode(true));
            this.slides = container.querySelectorAll('.xiaozhima-slide');

            // 分页器
            this.paginationBox = null;
            this.paginationList = null;
            if (toType(pagination) === "object") {
                let el = pagination.el;
                if (el) {
                    this.paginationBox = container.querySelector(el);
                    // 创建SPAN
                    let str = ``;
                    _each(this.slidesTrue, item => {
                        str += `<span></span>`;
                    });
                    this.paginationBox.innerHTML = str;
                    this.paginationList = this.paginationBox.querySelectorAll('span');
                }
            }

            // 左右切换
            this.arrowPrev = null;
            this.arrowNext = null;
            if (toType(navigation) === "object") {
                navigation.prevEl ? this.arrowPrev = container.querySelector(navigation.prevEl) : null;
                navigation.nextEl ? this.arrowNext = container.querySelector(navigation.nextEl) : null;
            }

            // 控制元素的样式(包含初始展示哪一个)
            this.changeWidth = parseFloat(getComputedStyle(container).width);
            this.activeIndex = this.activeIndex < 0 ? 0 : (this.activeIndex > this.slides.length - 1 ? this.slides.length - 1 : this.activeIndex);
            this.wrapper.style.width = `${this.changeWidth * this.slides.length}px`;
            this.wrapper.style.transition = `left ${this.speed}ms`;
            this.wrapper.style.left = `${-this.activeIndex * this.changeWidth}px`;
            _each(this.slides, item => {
                item.style.width = `${this.changeWidth}px`;
            });
            this.autoFocus();
        }


        // 自动轮播
        autoMove() {
            if (this.activeIndex === this.slides.length - 1) {
                this.activeIndex = 0;
                this.change(true);
            }
            this.activeIndex++;
            this.change();
        }

        // 实现轮播图切换
        change(now = false) {
            //now 传值是立即切换,不传是有动画切换
            let {
                wrapper,
                speed,
                activeIndex,
                changeWidth,
                on
            } = this;
            let isO = toType(on) === "object" ? true : false,
                transitionStart = isO ? on.transitionStart : null,
                transitionEnd = isO ? on.transitionEnd : null;
            // 切换之前触发的钩子函数
            !now && transitionStart ? transitionStart.call(this, this) : null;

            // 如果传了now 立即切换,则不需要有动画
            wrapper.style.transitionDuration = `${now ? 0 : speed}ms`;
            wrapper.style.left = `${-activeIndex * changeWidth}px`;
            if (now) {
                // 如果立即切换我需要让上面先渲染一次,利用读写分离;
                wrapper.offsetWidth;
            } else {
                this.autoFocus();
            }

            // 切换之后触发的钩子函数
            let fn = () => {
                !now && transitionEnd ? transitionEnd.call(this, this) : null;
                // 每一次都会重新监听,所以监听完需要把上一次监听的移除掉
                wrapper.removeEventListener('transitionend', fn);
            };
            wrapper.addEventListener('transitionend', fn);
        }
        // 实现焦点对齐
        autoFocus() {
            let {
                paginationList,
                activeIndex,
                slides
            } = this;
            if (!paginationList) return;
            activeIndex === slides.length - 1 ? activeIndex = 0 : null;
            _each(paginationList, (item, index) => {
                if (index === activeIndex) {
                    item.className = 'active';
                    return;
                }
                item.className = '';
            });
        }
    }

    function bannerPlugin(container, options = {}) {
        let defaultParams = {
            initialSlide: 0,
            autoplay: 3000,
            speed: 300,
            pagination: {
                el: '.xiaozhima-pagination',
                clickable: true
            },
            navigation: {
                nextEl: '.xiaozhima-arrow-next',
                prevEl: '.xiaozhima-arrow-prev'
            },
            on: {
                init() { },
                transitionStart() { },
                transitionEnd() { }
            }
        };
        options = _assignDeep(defaultParams, options);

        typeof container === "string" ? container = document.querySelector(container) : null;
        if (!container || container.nodeType !== 1) {
            throw new TypeError('container must be an element!');
        }

        return new Banner(container, options);
    }

    window.bannerPlugin = bannerPlugin;
})();

5. Инструкция по применению

  • стиль, который должен быть установлен
    • Ширина и высота большого контейнера
    • Ваша собственная фотография
  • Синтаксис использования JS
    • BannerPlugin (большой контейнер, информация о параметрах требуется пользователям)
      • контейнер: этикетка контейнера
      • опции: необходимые элементы конфигурации
        • Информацию о параметрах см. в разделе Описание информации о параметрах.

Автор тоже постоянно учится и пробует на фронтенде.Если что-то не так в содержании текста,просветите,буду очень признателен🙏;

Автор хочет и дальше развиваться в Наггетс.На данном этапе цель поднята на 3 уровень.Надеюсь у всех получится😄,спасибо большое;