Heim > Web-Frontend > View.js > Hauptteil

[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins

青灯夜游
Freigeben: 2022-01-27 18:18:34
nach vorne
4519 Leute haben es durchsucht

In diesem Artikel erfahren Sie mehr über den eigentlichen Kampf von VuePress und erfahren, wie Sie ein VuePress-Plugin (Code-Kopier-Plugin) von Grund auf entwickeln. Ich hoffe, dass es für alle hilfreich ist!

[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins

Beim Erstellen eines VuePress-Blogs können nicht alle Plug-Ins die Anforderungen erfüllen. Daher nehmen wir in diesem Artikel die Implementierung eines Code-Kopier-Plug-Ins als Beispiel, um Ihnen beizubringen, wie Sie ein VuePress implementieren Plug-in von Grund auf.

Lokale Entwicklung

Das erste Problem, das bei der Entwicklung von Plug-Ins gelöst werden muss, ist, wie man sie lokal entwickelt. Wir haben uns das Kapitel „Plug-In entwickeln“ im offiziellen Dokument von VuePress 1.0 angesehen und keine Lösung gefunden. Aber im Abschnitt „“ des offiziellen Dokuments „Lokales Plug-in “ von VuePress 2.0 steht geschrieben:

Es wird empfohlen, die Konfigurationsdatei direkt als Plug-in zu verwenden, da fast alle Plug-ins vorhanden sind APIs können in der Konfigurationsdatei verwendet werden, was in den meisten Szenarien praktischer ist.

Aber wenn Sie in der Konfigurationsdatei zu viele Dinge zu tun haben, ist es besser, sie in separate Plugins zu extrahieren und sie zu verwenden, indem Sie absolute Pfade festlegen oder über require:

module.exports = {
  plugins: [
    path.resolve(__dirname, './path/to/your-plugin.js'),
    require('./another-plugin'),
  ],
}
Nach dem Login kopieren

Dann lassen Sie uns loslegen!

Initialisieren Sie das Projekt

Wir erstellen einen neuen Ordner vuepress-plugin-code-copy unter dem Ordner .vuepress, um Plug-in-bezogenen Code zu speichern, und geben dann die Eingabetaste ein Öffnen Sie den Ordner, führen Sie npm init aus und erstellen Sie package.json. Zu diesem Zeitpunkt lautet das Verzeichnis der Datei: .vuepress 文件夹下新建一个 vuepress-plugin-code-copy 的文件夹,用于存放插件相关的代码,然后命令行进入到该文件夹,执行 npm init,创建 package.json,此时文件的目录为:

.vuepress
├─ vuepress-plugin-code-copy 
│  └─ package.json
└─ config.js
Nach dem Login kopieren

我们在 vuepress-plugin-code-copy下新建一个 index.js 文件,参照官方文档插件示例中的写法,我们使用返回对象的函数形式,这个函数接受插件的配置选项作为第一个参数、包含编译期上下文的 ctx 对象作为第二个参数:

module.exports = (options, ctx) => {
   return {
      // ...
   }
}
Nach dem Login kopieren

再参照官方文档 Option API 中的 name,以及生命周期函数中的 ready 钩子,我们写一个初始的测试代码:

module.exports = (options, ctx) => {
    return {
        name: 'vuepress-plugin-code-copy',
        async ready() {
            console.log('Hello World!');
        }
    }
 }
Nach dem Login kopieren

此时我们运行下 yarn run docs:dev,可以在运行过程中看到我们的插件名字和打印结果:

[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins

插件设计

现在我们可以设想下我们的代码复制插件的效果了,我想要实现的效果是:

在代码块的右下角有一个 Copy 文字按钮,点击后文字变为 Copied!然后一秒后文字重新变为 Copy,而代码块里的代码则在点击的时候复制到剪切板中,期望的表现效果如下:

[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins

插件开发

如果是在 Vue 组件中,我们很容易实现这个效果,在根组件 mounted 或者 updated的时候,使用 document.querySelector获取所有的代码块,插入一个按钮元素,再在按钮元素上绑定点击事件,当触发点击事件的时候,代码复制到剪切板,然后修改文字,1s 后再修改下文字。

那 VuePress 插件有方法可以控制根组件的生命周期吗?我们查阅下 VuePress 官方文档的 Option API,可以发现 VuePress 提供了一个 clientRootMixin 方法:

指向 mixin 文件的路径,它让你可以控制根组件的生命周期

看下示例代码:

// 插件的入口
const path = require('path')

module.exports = {
  clientRootMixin: path.resolve(__dirname, 'mixin.js')
}
Nach dem Login kopieren
// mixin.js
export default {
  created () {},
  mounted () {}
}
Nach dem Login kopieren

这不就是我们需要的吗?那我们动手吧,修改 index.js的内容为:

const path = require('path');

module.exports = (options, ctx) => {
    return {
        name: 'vuepress-plugin-code-copy',
        clientRootMixin: path.resolve(__dirname, 'clientRootMixin.js')
    }
 }
Nach dem Login kopieren
Nach dem Login kopieren

vuepress-plugin-code-copy下新建一个 clientRootMixin.js

export default {
    updated() {
        setTimeout(() => {
            document.querySelectorAll('div[class*="language-"] pre').forEach(el => {
								console.log('one code block')
            })
        }, 100)
    }
}
Nach dem Login kopieren

Wir befinden uns in vuepress-plugin -code Erstellen Sie eine neue <code>index.js-Datei unter -copy. Siehe die Schreibmethode im offiziellen Dokument-Plug-in-Beispiel. Wir verwenden die Funktionsform des Rückgabeobjekts Die Funktion akzeptiert die Plug-in-Konfigurationsoption als ersten Parameter, das ctx-Objekt, das den Kontext zur Kompilierungszeit enthält, wird als zweiter Parameter verwendet:

// 要挂载的元素
<div id="mount-point"></div>
Nach dem Login kopieren
Nach dem Login kopieren

Dann beziehen Sie sich auf name

und ready Hook [VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins schreiben wir einen ersten Testcode:

// 创建构造器
var Profile = Vue.extend({
  template: &#39;<p>{{firstName}} {{lastName}} aka {{alias}}</p>&#39;,
  data: function () {
    return {
      firstName: &#39;Walter&#39;,
      lastName: &#39;White&#39;,
      alias: &#39;Heisenberg&#39;
    }
  }
})
// 创建 Profile 实例,并挂载到一个元素上。
new Profile().$mount(&#39;#mount-point&#39;)
Nach dem Login kopieren
Nach dem Login kopieren
🎜Zu diesem Zeitpunkt führen wir yarn run docs:dev aus. Sie können den Namen unseres Plug-Ins sehen und die Ergebnisse während des laufenden Vorgangs drucken: 🎜🎜[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins🎜

Plug-in-Design 🎜🎜Jetzt können wir uns vorstellen, dass unser Code den Effekt des Plug-Ins kopiert: 🎜🎜In der unteren rechten Ecke des Codeblocks befindet sich die Schaltfläche „Text kopieren“. ! Dann ändert sich der Text nach einer Sekunde erneut in „Kopieren“ und der Code im Codeblock wird beim Klicken in die Zwischenablage kopiert. Der erwartete Leistungseffekt ist wie folgt: 🎜🎜[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins🎜

Plug-in-Entwicklung🎜🎜 Wenn dies der Fall ist In der Vue-Komponente können wir diesen Effekt leicht erzielen. Wenn die Stammkomponente gemountet oder aktualisiert ist, verwenden Sie document.querySelector Holen Sie sich alle Codes, fügen Sie ein Schaltflächenelement ein und binden Sie dann ein Klickereignis an das Schaltflächenelement. Wenn das Klickereignis ausgelöst wird, wird der Code in die Zwischenablage kopiert und dann der Text geändert nach 1 Sekunde. 🎜🎜Hat das VuePress-Plug-in also eine Möglichkeit, den Lebenszyklus der Root-Komponente zu steuern? Schauen wir uns die offizielle VuePress-Dokumentation an Option API🎜 finden Sie, dass VuePress eine clientRootMixin-Methode bereitstellt: 🎜🎜🎜Zeigt auf den Pfad der Mixin-Datei, mit der Sie den Lebenszyklus der Root-Komponente steuern können🎜🎜🎜Sehen Sie sich den Beispielcode an: 🎜
// 结果为:
<p>Walter White aka Heisenberg</p>
Nach dem Login kopieren
Nach dem Login kopieren
<template>
    <span class="code-copy-btn" @click="copyToClipboard">{{ buttonText }}</span>
</template>

<script>
export default {
    data() {
        return {
            buttonText: &#39;Copy&#39;
        }
    },
    methods: {
        copyToClipboard(el) {
            this.setClipboard(this.code, this.setText);
        },
        setClipboard(code, cb) {
            if (navigator.clipboard) {
                navigator.clipboard.writeText(code).then(
                    cb,
                    () => {}
                )
            } else {
                let copyelement = document.createElement(&#39;textarea&#39;)
                document.body.appendChild(copyelement)
                copyelement.value = code
                copyelement.select()
                document.execCommand(&#39;Copy&#39;)
                copyelement.remove()
                cb()
            }
        },
        setText() {
            this.buttonText = &#39;Copied!&#39;

            setTimeout(() => {
                this.buttonText = &#39;Copy&#39;
            }, 1000)
        }
    }
}
</script>

<style scoped>
.code-copy-btn {
    position: absolute;
    bottom: 10px;
    right: 7.5px;
    opacity: 0.75;
    cursor: pointer;
    font-size: 14px;
}

.code-copy-btn:hover {
    opacity: 1;
}
</style>
Nach dem Login kopieren
Nach dem Login kopieren
🎜Ist das nicht so? Brauchen wir das? Dann machen wir es und ändern den Inhalt von index.js wie folgt: 🎜
import CodeCopy from &#39;./CodeCopy.vue&#39;
import Vue from &#39;vue&#39;

export default {
    updated() {
        // 防止阻塞
        setTimeout(() => {
            document.querySelectorAll(&#39;div[class*="language-"] pre&#39;).forEach(el => {
              	// 防止重复写入
                if (el.classList.contains(&#39;code-copy-added&#39;)) return
                let ComponentClass = Vue.extend(CodeCopy)
                let instance = new ComponentClass()
                instance.code = el.innerText
                instance.$mount()
                el.classList.add(&#39;code-copy-added&#39;)
                el.appendChild(instance.$el)
            })
        }, 100)
    }
}
Nach dem Login kopieren
Nach dem Login kopieren
🎜Erstellen Sie einen neuen clientRootMixin.js</code unter <code>vuepress-plugin-code-copy</code > >Datei, Code schreiben: 🎜<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:js;toolbar:false;">.vuepress ├─ vuepress-plugin-code-copy │ ├─ CodeCopy.vue │ ├─ clientRootMixin.js │ ├─ index.js │ └─ package.json └─ config.js</pre><div class="contentsignin">Nach dem Login kopieren</div></div><div class="contentsignin">Nach dem Login kopieren</div></div>🎜Aktualisieren Sie die Seite im Browser und sehen Sie sich dann den Ausdruck an: 🎜🎜🎜🎜<p>接下来就要思考如何写入按钮元素了。</p><p>当然我们可以使用原生 JavaScript 一点点的创建元素,然后插入其中,但我们其实是在一个支持 Vue 语法的项目里,其实我们完全可以创建一个 Vue 组件,然后将组件的实例挂载到元素上。那用什么方法挂载呢?</p><p>我们可以在 Vue 的<a href="vuepress.vuejs.org" target="_blank" rel="nofollow noopener noreferrer" ref="nofollow noopener noreferrer">全局 API </a>里,找到 <code>Vue.extendAPI,看一下使用示例:

// 要挂载的元素
<div id="mount-point"></div>
Nach dem Login kopieren
Nach dem Login kopieren
// 创建构造器
var Profile = Vue.extend({
  template: &#39;<p>{{firstName}} {{lastName}} aka {{alias}}</p>&#39;,
  data: function () {
    return {
      firstName: &#39;Walter&#39;,
      lastName: &#39;White&#39;,
      alias: &#39;Heisenberg&#39;
    }
  }
})
// 创建 Profile 实例,并挂载到一个元素上。
new Profile().$mount(&#39;#mount-point&#39;)
Nach dem Login kopieren
Nach dem Login kopieren

结果如下:

// 结果为:
<p>Walter White aka Heisenberg</p>
Nach dem Login kopieren
Nach dem Login kopieren

那接下来,我们就创建一个 Vue 组件,然后通过 Vue.extend 方法,挂载到每个代码块元素中。

vuepress-plugin-code-copy下新建一个 CodeCopy.vue 文件,写入代码如下:

<template>
    <span class="code-copy-btn" @click="copyToClipboard">{{ buttonText }}</span>
</template>

<script>
export default {
    data() {
        return {
            buttonText: &#39;Copy&#39;
        }
    },
    methods: {
        copyToClipboard(el) {
            this.setClipboard(this.code, this.setText);
        },
        setClipboard(code, cb) {
            if (navigator.clipboard) {
                navigator.clipboard.writeText(code).then(
                    cb,
                    () => {}
                )
            } else {
                let copyelement = document.createElement(&#39;textarea&#39;)
                document.body.appendChild(copyelement)
                copyelement.value = code
                copyelement.select()
                document.execCommand(&#39;Copy&#39;)
                copyelement.remove()
                cb()
            }
        },
        setText() {
            this.buttonText = &#39;Copied!&#39;

            setTimeout(() => {
                this.buttonText = &#39;Copy&#39;
            }, 1000)
        }
    }
}
</script>

<style scoped>
.code-copy-btn {
    position: absolute;
    bottom: 10px;
    right: 7.5px;
    opacity: 0.75;
    cursor: pointer;
    font-size: 14px;
}

.code-copy-btn:hover {
    opacity: 1;
}
</style>
Nach dem Login kopieren
Nach dem Login kopieren

该组件实现了按钮的样式和点击时将代码写入剪切版的效果,整体代码比较简单,就不多叙述了。

我们修改一下 clientRootMixin.js

import CodeCopy from &#39;./CodeCopy.vue&#39;
import Vue from &#39;vue&#39;

export default {
    updated() {
        // 防止阻塞
        setTimeout(() => {
            document.querySelectorAll(&#39;div[class*="language-"] pre&#39;).forEach(el => {
              	// 防止重复写入
                if (el.classList.contains(&#39;code-copy-added&#39;)) return
                let ComponentClass = Vue.extend(CodeCopy)
                let instance = new ComponentClass()
                instance.code = el.innerText
                instance.$mount()
                el.classList.add(&#39;code-copy-added&#39;)
                el.appendChild(instance.$el)
            })
        }, 100)
    }
}
Nach dem Login kopieren
Nach dem Login kopieren

这里注意两点,第一是我们通过 el.innerText 获取要复制的代码内容,然后写入到实例的 code 属性,在组件中,我们是通过 this.code获取的。

第二是我们没有使用 $mount(element),直接传入一个要挂载的节点元素,这是因为 $mount() 的挂载会清空目标元素,但是这里我们需要添加到元素中,所以我们在执行 instance.$mount()后,通过 instance.$el获取了实例元素,然后再将其 appendChild 到每个代码块中。关于 $el的使用可以参考官方文档的 el 章节

此时,我们的文件目录如下:

.vuepress
├─ vuepress-plugin-code-copy 
│  ├─ CodeCopy.vue
│  ├─ clientRootMixin.js
│  ├─ index.js
│  └─ package.json
└─ config.js
Nach dem Login kopieren
Nach dem Login kopieren

至此,其实我们就已经实现了代码复制的功能。

插件选项

有的时候,为了增加插件的可拓展性,会允许配置可选项,就比如我们不希望按钮的文字是 Copy,而是中文的「复制」,复制完后,文字变为 「已复制!」,该如何实现呢?

前面讲到,我们的 index.js导出的函数,第一个参数就是 options 参数:

const path = require(&#39;path&#39;);

module.exports = (options, ctx) => {
    return {
        name: &#39;vuepress-plugin-code-copy&#39;,
        clientRootMixin: path.resolve(__dirname, &#39;clientRootMixin.js&#39;)
    }
 }
Nach dem Login kopieren
Nach dem Login kopieren

我们在 config.js先写入需要用到的选项:

module.exports = {
    plugins: [
      [
        require(&#39;./vuepress-plugin-code-copy&#39;),
        {
          &#39;copybuttonText&#39;: &#39;复制&#39;,
          &#39;copiedButtonText&#39;: &#39;已复制!&#39;
        }
      ]
    ]
}
Nach dem Login kopieren

我们 index.js中通过 options参数可以接收到我们在 config.js 写入的选项,但我们怎么把这些参数传入 CodeCopy.vue 文件呢?

我们再翻下 VuePress 提供的 Option API,可以发现有一个 define API,其实这个 define 属性就是定义我们插件内部使用的全局变量。我们修改下 index.js

const path = require(&#39;path&#39;);

module.exports = (options, ctx) => {
    return {
        name: &#39;vuepress-plugin-code-copy&#39;,
        define: {
            copybuttonText: options.copybuttonText || &#39;copy&#39;,
            copiedButtonText: options.copiedButtonText || "copied!"
        },
        clientRootMixin: path.resolve(__dirname, &#39;clientRootMixin.js&#39;)
    }
 }
Nach dem Login kopieren

现在我们已经写入了两个全局变量,组件里怎么使用呢?答案是直接使用!

我们修改下 CodeCopy.vue 的代码:

// ...
<script>
export default {
    data() {
        return {
            buttonText: copybuttonText
        }
    },
    methods: {
        copyToClipboard(el) {
            this.setClipboard(this.code, this.setText);
        },
        setClipboard(code, cb) {
            if (navigator.clipboard) {
                navigator.clipboard.writeText(code).then(
                    cb,
                    () => {}
                )
            } else {
                let copyelement = document.createElement(&#39;textarea&#39;)
                document.body.appendChild(copyelement)
                copyelement.value = code
                copyelement.select()
                document.execCommand(&#39;Copy&#39;)
                copyelement.remove()
                cb()
            }
        },
        setText() {
            this.buttonText = copiedButtonText

            setTimeout(() => {
                this.buttonText = copybuttonText
            }, 1000)
        }
    }
}
</script>
// ...
Nach dem Login kopieren

最终的效果如下:

[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins

代码参考

完整的代码查看:https://github.com/mqyqingfeng/Blog/tree/master/demos/VuePress/vuepress-plugin-code-copy

【相关推荐:vue.js视频教程

Das obige ist der detaillierte Inhalt von[VuePress in Aktion] Wir führen Sie Schritt für Schritt durch die Entwicklung eines Code-Kopier-Plug-Ins. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:juejin.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage