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

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'),
  ],
}

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

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

module.exports = (options, ctx) => {
   return {
      // ...
   }
}

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

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

此时我们运行下 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')
}
// mixin.js
export default {
  created () {},
  mounted () {}
}

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

const path = require('path');

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

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

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

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>

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;)
🎜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>
<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>
🎜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)
    }
}
🎜Erstellen Sie einen neuen clientRootMixin.jsunter vuepress-plugin-code-copy >Datei, Code schreiben: 🎜
.vuepress
├─ vuepress-plugin-code-copy 
│  ├─ CodeCopy.vue
│  ├─ clientRootMixin.js
│  ├─ index.js
│  └─ package.json
└─ config.js
🎜Aktualisieren Sie die Seite im Browser und sehen Sie sich dann den Ausdruck an: 🎜🎜🎜🎜

接下来就要思考如何写入按钮元素了。

当然我们可以使用原生 JavaScript 一点点的创建元素,然后插入其中,但我们其实是在一个支持 Vue 语法的项目里,其实我们完全可以创建一个 Vue 组件,然后将组件的实例挂载到元素上。那用什么方法挂载呢?

我们可以在 Vue 的全局 API 里,找到 Vue.extendAPI,看一下使用示例:

// 要挂载的元素
<div id="mount-point"></div>
// 创建构造器
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;)

结果如下:

// 结果为:
<p>Walter White aka Heisenberg</p>

那接下来,我们就创建一个 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>

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

我们修改一下 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)
    }
}

这里注意两点,第一是我们通过 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

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

插件选项

有的时候,为了增加插件的可拓展性,会允许配置可选项,就比如我们不希望按钮的文字是 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;)
    }
 }

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

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

我们 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;)
    }
 }

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

我们修改下 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>
// ...

最终的效果如下:

[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!

Stellungnahme
Dieser Artikel ist reproduziert unter:掘金社区. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen
Vue.js vs. React: Benutzerfreundlichkeit und LernkurveVue.js vs. React: Benutzerfreundlichkeit und LernkurveMay 02, 2025 am 12:13 AM

Vue.js ist einfacher zu bedienen und hat eine reibungslose Lernkurve, die für Anfänger geeignet ist. React hat eine steilere Lernkurve, hat jedoch eine starke Flexibilität, die für erfahrene Entwickler geeignet ist. 1.Vue.js ist einfach mit einfachen Datenbindung und progressivem Design zu beginnen. 2.React erfordert das Verständnis von virtuellem DOM und JSX, bietet jedoch höhere Flexibilität und Leistungsvorteile.

Vue.js vs. React: Welches Framework ist für Sie geeignet?Vue.js vs. React: Welches Framework ist für Sie geeignet?May 01, 2025 am 12:21 AM

Vue.js eignet sich für schnelle Entwicklung und kleine Projekte, während React eher für große und komplexe Projekte geeignet ist. 1.Vue.js ist einfach und leicht zu lernen, geeignet für schnelle Entwicklung und kleine Projekte. 2.React ist leistungsstark und für große und komplexe Projekte geeignet. 3.. 4. Reacts komponentiertes und virtuelles DOM funktioniert gut, wenn es sich um komplexe Benutzeroberflächen- und datenintensive Anwendungen handelt.

Vue.js vs. React: Eine vergleichende Analyse von JavaScript -FrameworksVue.js vs. React: Eine vergleichende Analyse von JavaScript -FrameworksApr 30, 2025 am 12:10 AM

Vue.js und reagieren jeweils ihre eigenen Vor- und Nachteile. Bei der Auswahl müssen Sie Teamfähigkeiten, Projektgröße und Leistungsanforderungen umfassend berücksichtigen. 1) Vue.js eignet sich für schnelle Entwicklung und kleine Projekte mit einer niedrigen Lernkurve, aber tiefe verschachtelte Objekte können Leistungsprobleme verursachen. 2) React ist für große und komplexe Anwendungen mit einem reichhaltigen Ökosystem geeignet, aber häufige Aktualisierungen können zu Leistungs Engpässen führen.

Vue.js vs. React: Anwendungsfälle und AnwendungenVue.js vs. React: Anwendungsfälle und AnwendungenApr 29, 2025 am 12:36 AM

Vue.js eignet sich für kleine bis mittelgroße Projekte, während React für große Projekte und komplexe Anwendungsszenarien geeignet ist. 1) Vue.js ist einfach zu bedienen und ist für schnelle Prototypen und kleine Anwendungen geeignet. 2) React hat mehr Vorteile beim Umgang mit komplexem Zustandsmanagement und Leistungsoptimierung und ist für große Projekte geeignet.

Vue.js vs. React: Vergleich der Leistung und EffizienzVue.js vs. React: Vergleich der Leistung und EffizienzApr 28, 2025 am 12:12 AM

Vue.js und React haben jeweils eigene Vorteile: Vue.js ist für kleine Anwendungen und schnelle Entwicklung geeignet, während React für große Anwendungen und komplexes Staatsmanagement geeignet ist. 1.Vue.js realisiert automatisches Update über ein reaktionsschnelles System, das für kleine Anwendungen geeignet ist. 2.React verwendet virtuelle DOM- und Diff -Algorithmen, die für große und komplexe Anwendungen geeignet sind. Bei der Auswahl eines Frameworks müssen Sie Projektanforderungen und Teamtechnologie -Stack in Betracht ziehen.

Vue.js vs. React: Gemeinschaft, Ökosystem und UnterstützungVue.js vs. React: Gemeinschaft, Ökosystem und UnterstützungApr 27, 2025 am 12:24 AM

Vue.js und React jeweils haben ihre eigenen Vorteile, und die Auswahl sollte auf Projektanforderungen und Teamtechnologie -Stack beruhen. 1. Vue.js ist gemeinschaftsfreundlich und bietet reichhaltige Lernressourcen an, und das Ökosystem umfasst offizielle Tools wie Vuerouter, die vom offiziellen Team und der Community unterstützt werden. 2. Die React -Community ist in Bezug auf Unternehmensanwendungen mit einem starken Ökosystem voreingenommen und unterstützt von Facebook und seiner Community und hat häufige Aktualisierungen.

Reagieren und Netflix: Erforschen der BeziehungReagieren und Netflix: Erforschen der BeziehungApr 26, 2025 am 12:11 AM

Netflix verwendet React, um die Benutzererfahrung zu verbessern. 1) Die komponentierten Merkmale von React unterstützen die Netflix -Komplex -Benutzeroberfläche in überschaubare Module. 2) Virtual DOM optimiert UI -Updates und verbessert die Leistung. 3) Die Kombination von Redux und GraphQL verwaltet Netflix den Anwendungsstatus und den Datenfluss effizient.

Vue.js vs. Backend Frameworks: Klärung der UnterscheidungVue.js vs. Backend Frameworks: Klärung der UnterscheidungApr 25, 2025 am 12:05 AM

Vue.js ist ein Front-End-Framework, und das Back-End-Framework wird verwendet, um die serverseitige Logik zu verarbeiten. 1) Vue.js konzentriert sich auf den Aufbau von Benutzeroberflächen und vereinfacht die Entwicklung durch komponentierte und reaktionsschnelle Datenbindung. 2) Back-End-Frameworks wie Express- und Django-HTTP-Anforderungen, Datenbankvorgänge und Geschäftslogik und auf dem Server ausgeführt.

See all articles

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

Video Face Swap

Video Face Swap

Tauschen Sie Gesichter in jedem Video mühelos mit unserem völlig kostenlosen KI-Gesichtstausch-Tool aus!

Heiße Werkzeuge

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

VSCode Windows 64-Bit-Download

VSCode Windows 64-Bit-Download

Ein kostenloser und leistungsstarker IDE-Editor von Microsoft

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

Dreamweaver Mac

Dreamweaver Mac

Visuelle Webentwicklungstools

SublimeText3 Linux neue Version

SublimeText3 Linux neue Version

SublimeText3 Linux neueste Version