普通视图

发现新文章,点击刷新页面。
今天 — 2026年4月11日首页

VSCode 插件推荐 Copy Filename Pro,快速复制文件、目录和路径的首选

2026年4月11日 12:30

大家好,我是笨笨狗吞噬者,uni-app、varlet、nrm 等众多知名仓库的核心开发,专注于分享 前端技术 和 AI 实践知识,欢迎关注我的微信公众号 前端笨笨狗

问题背景

大家平时写代码时,经常会遇到一些很碎的小动作,比如:

  • 想快速拿到组件名、页面名、模块名
  • 想复制路径写 import、写文档或者发消息
  • 想批量整理多个文件名或路径

这些操作不难,但做得多了就很烦,尤其是在目录结构比较复杂的工程里。我也被这类问题折腾了很久,试了插件市场里的很多插件,却总不如意,于是,我就自己写了插件 Copy Filename Pro

插件功能

Copy Filename Pro 主要提供下面几个功能:

复制带文件后缀的文件名

比如我想复制某个 vue 文件的完整文件名

with-filename.gif

复制不带扩展名的文件名

比如我想复制某个 vue 文件的不包含文件后缀的文件名

no-filename.gif

复制目录名

比如我想复制某个文件夹名称

dictory.gif

复制不带拓展名的绝对路径或者相对路径

由于 VSCode 本身有复制路径和相对路径的功能了,所以这里演示如何得到不包含文件后缀的路径

path.gif

一次复制多个文件或目录的信息

比如我想一次复制多个文件名

mul.gif

下载安装

大家可以参考下面的图片安装此插件

zhinan.png

另外,此插件的源码是完全免费公开的,访问 https://github.com/chouchouji/copy-filename-pro 即可获取,如果你有更好的想法和建议,也可以留言给我。

昨天 — 2026年4月10日首页

代理的妙用:uni-app 小程序是怎样用 `Proxy` 和 `wrapper` 抹平平台差异的

2026年4月10日 15:24

前言

大家好,我是 笨笨狗吞噬者,uni-app、varlet、nrm 等众多知名仓库的核心开发,专注于分享前端技术和 AI实践知识,本篇文章是 uni-app 小程序源码解读的第一期,欢迎关注我的微信公众号 前端笨笨狗

很多人在使用 uni-app 时,会很自然地写出 uni.showToast()uni.login() 这样的代码,但很少会继续追问一个问题:为什么同样是 uni.xxx,到了微信能落到 wx.xxx,到了支付宝又能落到 my.xxx

这个问题背后,藏着一个巧妙的设计思路。它既不是简单的字符串替换,也不是把所有平台 API 硬编码映射一遍,接下来我们慢慢揭秘。

问题背景

在 uni-app 小程序端,开发者写的是:

uni.showToast({ title: 'Hello' })
uni.login()

但真正运行时:

  • 微信里调用的是 wx.xxx
  • 支付宝里调用的是 my.xxx

这一层统一,核心就靠两样东西:

  • Proxy:决定 uni.xxx 这次到底取哪个实现
  • wrapper:把不同平台之间的参数、方法名、返回值差异包起来

对应源码入口在 https://github.com/dcloudio/uni-app/blob/uni-app-vue3-dev/packages/uni-mp-core/src/api/index.ts#L61

Proxy:统一入口分发

先看核心代码:

export function initUni(api, protocols, platform = __GLOBAL__) {
  const wrapper = initWrapper(protocols)
  const UniProxyHandlers = {
    get(target, key) {
      if (hasOwn(api, key)) {
        return promisify(key, api[key])
      }
      if (hasOwn(baseApis, key)) {
        return promisify(key, baseApis[key])
      }
      return promisify(key, wrapper(key, platform[key]))
    },
  }

  return new Proxy({}, UniProxyHandlers)
}

这段代码的意思很简单:

uni 不是一个提前写死所有方法的对象,而是一个“访问属性时再决定返回什么”的代理对象。

比如你写:

uni.showToast

这时会触发 get(target, key),其中:

  • key 就是 showToast
  • 代理会临时判断这个方法应该从哪里来

也就是说,Proxy 的作用不是执行 API,而是做一次统一分发。

为什么这里一定要用 Proxy

因为 uni 的目标不是“绑定某个平台”,而是“对外提供稳定名字,对内按平台切换实现”。

如果不用 Proxy,那就只能:

  • 提前把所有 API 全部挂到 uni
  • 或者每个平台都维护一套映射逻辑

这样会很笨重。

用了 Proxy 之后,框架就可以做到:

  1. 开发者永远写 uni.xxx
  2. 真正访问到 uni.xxx 时,再动态决定用谁
  3. 同一个入口,底层可以切到不同平台对象

这正是代理最适合的场景:入口统一,底层实现可变。

这段 get 其实分了三层

源码里的分发顺序很清楚:

1. 先看 api

if (hasOwn(api, key)) {
  return promisify(key, api[key])
}

如果这个方法 uni 自己实现过,就优先返回 uni 自己的实现。

2. 再看 baseApis

if (hasOwn(baseApis, key)) {
  return promisify(key, baseApis[key])
}

像事件总线、拦截器这类基础能力,不一定来自小程序原生对象,也统一挂在 uni 上。

3. 最后兜底到平台对象

return promisify(key, wrapper(key, platform[key]))

这里的 platform 默认是 __GLOBAL__,可以理解成当前平台全局对象:

  • 微信环境下接近 wx
  • 支付宝环境下接近 my

所以 Proxy 最终做成了这件事:

uni.showToast -> 当前平台的 showToast
uni.login -> 当前平台的 login

wrapper:翻译平台差异

如果只是:

return platform[key]

表面上也能调用,但问题是不同小程序平台并不完全一致,常见差异有三种:

  1. 方法名不一致
  2. 参数名不一致
  3. 返回值结构不一致

所以 Proxy 只能解决“分发给谁”,解决不了“怎么适配差异”。

这就是 wrapper 的职责。

wrapper 在这里干了什么

看最关键的一句:

const returnValue = __GLOBAL__[options.name || methodName].apply(
  __GLOBAL__,
  args
)

这里能看出两件事。

第一,wrapper 允许改方法名。

  • 默认调用 methodName
  • 如果协议里配置了 options.name,就改成另一个平台方法名

第二,wrapper 会先处理参数,再调用平台方法。

前面的 processArgs、后面的 processReturnValue,本质上都在做一件事:

把 uni 暴露给开发者的统一接口,翻译成当前平台真正认识的接口。

所以可以把 wrapper 理解成一个“翻译层”。

举一个最容易理解的例子

假设 uni 对外想统一成这样:

uni.showToast({
  title: '保存成功'
})

但某个平台底层不是 title,而是 content

Proxy 只能做到:

uni.showToast -> 找到这个平台的方法

真正把参数从:

{ title: '保存成功' }

改成:

{ content: '保存成功' }

这一步必须由 wrapper 做。

也就是说:

  • Proxy 负责“找到人”
  • wrapper 负责“翻译话”

这两个配合起来,开发者才能始终写统一的 uni.xxx

简易实现

下面写一个极简版,只保留 Proxywrapper 两层核心思想。

const wx = {
  showToast(options) {
    console.log('wx.showToast', options)
  },
}

function wrapper(name, method) {
  if (name === 'showToast') {
    return function (options) {
      const newOptions = {
        content: options.title,
      }
      return method(newOptions)
    }
  }
  return method
}

function initUni(platform) {
  return new Proxy(
    {},
    {
      get(target, key) {
        const method = platform[key]
        if (typeof method !== 'function') {
          return undefined
        }
        return wrapper(key, method)
      },
    }
  )
}

const uni = initUni(wx)

uni.showToast({
  title: '保存成功',
})

执行时虽然外部写的是:

uni.showToast({ title: '保存成功' })

但实际到平台方法时,已经被改成了:

wx.showToast({ content: '保存成功' })

总结

不是“用了 Proxy 很高级”,而是职责拆得很清楚:

  • Proxy 只做入口分发
  • wrapper 只做平台适配

这样设计的好处是:

  • uni 对开发者始终保持统一
  • 平台差异被收口在框架内部
  • 后续要支持更多小程序平台时,只需要补适配逻辑,不需要改开发者写法

微信交流群

我有个 uni-app 微信交流群,大家有想进群的可以加我的微信 13460036576

昨天以前首页

uni-app 编译小程序原生组件时疑似丢属性,可以给官方提 PR 了

2026年4月2日 17:41

前言

大家好,我是 uni-app 的核心开发 笨笨狗吞噬者。不少开发者都遇到过某些小程序原生组件在原生项目能跑,但是写到 uni-app 中就用不了,今天来分析下背后的原因以及原理,以后再遇到类似的问题,你也可以给 uni-app 仓库提 pr。

问题

我们以微信小程序的 official-account-publish 组件为例,看下错误的产物和正确的产物都是什么样的,

测试代码

<template>
  <official-account-publish1
    topic="和coco一起做好事"
  ></official-account-publish1>
</template>

错误产物

<official-account-publish
  wx:if="{{d}}"
  u-i="1d1772a4-1"
  bind:__l="__l"
  u-p="{{d}}"
></official-account-publish>

正确产物

<official-account-publish topic="和coco一起做好事"></official-account-publish>

true.jpg

可以看到错误的产物上面多了不少属性,比如 u-iu-p,而正常产物是没有这些的

原理

其实这个差异的核心,不在于 official-account-publish 组件本身,而在于编译阶段它有没有命中小程序的 isCustomElement

uni-app 在编译小程序模板时,会先给当前平台准备一份原生组件名单。这个名单通常就在对应平台包的 compiler/options 里维护,比如微信小程序这一份就放在平台编译配置里 github.com/dcloudio/un…

wx.jpg

除此之外,项目侧还可以通过 manifest.json 里的 nativeTags 再补一份,最后一起参与 isCustomElement 判断

也就是说,编译器遇到一个标签时,并不是简单看它长得像不像组件,而是先看它在不在这份名单里

如果命中了,说明它是小程序原生组件,那就按原生标签处理,模板产物会尽量保留原样,像 topic 这类属性也会直接落到最终产物里,不会额外补组件运行时需要的标记

如果没有命中,编译器就会把它当成普通 vue 组件继续处理。这时候就会进入组件那套编译分支,给节点补上 u-iu-pbind:__l 这一类运行时字段,用来做组件实例关联、props 同步这些事情,所以最终产物看起来就“不像原生组件了”

所以这个问题本质上就是一句话:official-account-publish 应该走小程序 isCustomElement 分支,而不是走 vue 组件分支。一旦判断错了,生成结果就会立刻不一样

这类问题的修复思路通常也很直接,如果某个小程序原生组件没有被正确识别,除了补平台内置名单,更常见的做法就是在项目配置里把它加到 nativeTags,让编译器把它当成原生标签处理

比如:

// manifest.json
{
  "mp-weixin": {
    "nativeTags": ["official-account-publish"]
  }
}

加完之后,这个标签就会进入 isCustomElement 的判断范围,编译时不再走 vue 组件分支,生成结果也就会回到小程序原生组件该有的样子

交流群

我建了一个微信群(非官方),大家可以在群里和我沟通交流 uniapp 开发遇到的问题、uniapp 的源码等问题。

mmqrcode1775031270633.png

❌
❌