如何在uniapp根节点下添加全局水印组件?

在uniapp中我想在根节点下添加一个全局水印组件
image.png
image.png

我尝试在app.vue中添加但是没有生效

阅读 2.2k
3 个回答

App.vue 本身并不是一个页面,而是应用的根组件。要在所有页面中显示全局水印组件,可以使用以下方法:

创建水印组件

components 目录下创建一个 Watermark.vue 文件,并添加以下代码:

<!-- components/Watermark.vue -->
<template>
  <view class="watermark-box" :style="{ backgroundImage: `url(${watermark})`, backgroundSize: `auto ${height}px` }">
    <canvas id="watermark" class="watermark" canvas-id="watermark" />
  </view>
</template>

<script>
export default {
  name: 'Watermark',
  data() {
    return {
      watermark: '',
      width: '',
      height: ''
    };
  },
  mounted() {
    this.initWatermark();
  },
  methods: {
    initWatermark() {
      const watermarkText = '水印';
      const ctx = uni.createCanvasContext('watermark');
      const fontSize = 14;
      const rotateRad = 30 * Math.PI / 180;
      const cosRad = Math.cos(rotateRad);
      const sinRad = Math.sin(rotateRad);
      const { width: fontRealWidth } = ctx.measureText(watermarkText);
      const recWidth = Math.ceil(fontRealWidth * cosRad + fontSize * sinRad);
      const recHeight = Math.ceil(fontSize * cosRad + fontRealWidth * sinRad);
      const width = parseInt(recWidth * 2);
      const height = parseInt(recHeight * 2.5);
      const x = parseInt((width - recWidth) / 2);
      const y = parseInt((height + recHeight) / 2);
      this.width = width;
      this.height = height;
      ctx.width = this.width;
      ctx.height = this.height;
      ctx.setGlobalAlpha(0.3);
      ctx.setFontSize(fontSize);
      ctx.setFillStyle('#c8c8c8');
      ctx.translate(x, y);
      ctx.rotate(-rotateRad);
      ctx.textAlign = 'center';
      ctx.textBaseline = 'middle';
      ctx.fillText(watermarkText, 0, 0);
      ctx.draw(true, () => {
        uni.canvasToTempFilePath({
          canvasId: 'watermark',
          width: this.width,
          height: this.height,
          quality: 1,
          fileType: 'png',
          success: (res) => {
            this.watermark = res.tempFilePath;
          }
        });
      });
    }
  }
};
</script>

<style scoped>
.watermark-box {
  overflow: hidden;
  width: 100%;
  height: 100vh;
  pointer-events: none;
  position: fixed;
  top: 0;
  left: 0;
  background-repeat: repeat;
}
.watermark {
  z-index: -1;
}
</style>

在 App.vue 中引入水印组件

在 App.vue 中使用 router-view: 虽然 App.vue 没有传统的 template 部分,但你可以在 render 函数中使用 router-view 来显示当前匹配的路由组件
App.vue 中引入并使用水印组件:

<!-- App.vue -->
<template>
  <view>
    <Watermark />
    <router-view />
  </view>
</template>

<script>
import Watermark from '@/components/Watermark.vue';

export default {
  components: {
    Watermark
  }
};
</script>

确保全局样式不覆盖水印组件

在全局样式文件中(如 App.vuemain.css),确保没有覆盖水印组件的样式:

/* App.vue 或 main.css 中 */
body, html {
  height: 100%;
  margin: 0;
  padding: 0;
}
新手上路,请多包涵

第一步:创建一个全局的watermark组件

// --snip--

第二步: 新建plugins/insertWaterMarkPlugin.js

import { readFileSync } from 'fs';
import { resolve } from 'path';

// 为所有页面插入<watermark />组件, 再去编译
export default function insertWaterMarkPlugin() {
  return {
    name: 'insert-water-mark-plugin',
    async transform(code, id) {
      if (id.endsWith('.vue')) {
        const filePath = resolve(id);
        if (filePath.includes('src\\packages\\') || filePath.includes('src\\pages\\')) {
          const fileContent = readFileSync(filePath, 'utf-8');
          const modifiedCode = fileContent.replace(
            '<template>',
            '<template><watermark />'
          );

          return {
            code: modifiedCode,
            map: null
          }
        }
      }

      return null
    }
  };
}

第三步: vite.config.ts使用插件

export default defineConfig({
  plugins: [
    insertWaterMarkPlugin(), // 加这一行
    uni()
  ]
})

第四步: 重新运行即可

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏