diff --git a/eslint.config.mjs b/eslint.config.mjs index 753ae8d..e99a177 100644 --- a/eslint.config.mjs +++ b/eslint.config.mjs @@ -5,7 +5,7 @@ export default antfu({ unocss: true, typescript: true, formatters: true, - ignores: [`.github`, `bin`, `md-cli`, `src/assets`], + ignores: [`.github`, `bin`, `md-cli`, `src/assets`, `example`], }, { rules: { 'semi': [`error`, `never`], diff --git a/example/README.md b/example/README.md new file mode 100644 index 0000000..e47e6d2 --- /dev/null +++ b/example/README.md @@ -0,0 +1,16 @@ +# Example + +## worker.js + +公众号openapi接口代理服务示例,该项目将请求转发至微信公众号api。 + +开发调试: + +``` +cd example +npx wrangler dev worker.js +``` + +部署: + +请将其部署到cloudflare workers。 diff --git a/example/worker.js b/example/worker.js new file mode 100644 index 0000000..1f64377 --- /dev/null +++ b/example/worker.js @@ -0,0 +1,36 @@ +/** + * @typedef {object} Env + * @property + */ + +export default { + /** + * @param {Request} request + * @param {Env} env + * @param {ExecutionContext} ctx + * @returns {Promise} + */ + async fetch(request, env, ctx) { + const url = new URL(request.url) + const targetUrl = `https://api.weixin.qq.com` + const proxyRequest = new Request(targetUrl + url.pathname + url.search, { + method: request.method, + headers: request.headers, + body: request.body, + }) + const response = await fetch(proxyRequest) + const proxyResponse = new Response(response.body, { + status: response.status, + statusText: response.statusText, + headers: response.headers, + }) + setCorsHeaders(proxyResponse.headers) + return proxyResponse + }, +} +// 设置 CORS 头部 +function setCorsHeaders(headers) { + headers.set(`Access-Control-Allow-Origin`, `*`) + headers.set(`Access-Control-Allow-Methods`, `GET, POST, PUT, DELETE`) + headers.set(`Access-Control-Allow-Headers`, `*`) +} diff --git a/src/components/CodemirrorEditor/UploadImgDialog.vue b/src/components/CodemirrorEditor/UploadImgDialog.vue index 0c832a0..1a588bb 100644 --- a/src/components/CodemirrorEditor/UploadImgDialog.vue +++ b/src/components/CodemirrorEditor/UploadImgDialog.vue @@ -63,10 +63,11 @@ const minioOSS = ref({ }) const formMp = ref({ + proxyOrigin: ``, appID: ``, appsecret: ``, }) -const mpDisabled = ref(window.location.href.startsWith(`http`)) +const isWebsite = ref(window.location.href.startsWith(`http`)) const formCustom = ref<{ code: string, editor: CodeMirror.EditorFromTextArea | null }>({ code: localStorage.getItem(`formCustomConfig`) @@ -117,7 +118,6 @@ const options = [ { value: `mp`, label: `公众号素材`, - disabled: mpDisabled.value, }, { value: `formCustom`, @@ -272,6 +272,10 @@ function saveMpConfiguration() { ElMessage.error(`公众号图床 参数配置不全`) return } + if (isWebsite.value && !formMp.value.proxyOrigin) { + ElMessage.error(`代理域名必须配置`) + return + } localStorage.setItem(`mpConfig`, JSON.stringify(formMp.value)) ElMessage.success(`保存成功`) } @@ -328,7 +332,6 @@ function uploadImage(params: { file: any }) { :key="item.value" :label="item.label" :value="item.value" - :disabled="item?.disabled" />
公众号 图床
@@ -656,8 +659,13 @@ function uploadImage(params: { file: any }) { :model="formMp" label-position="right" label-width="150px" - :disabled="mpDisabled" > + + + - + - MpMd编辑器 + 公众号文章编辑器 diff --git a/src/utils/file.ts b/src/utils/file.ts index 2e89423..26a4550 100644 --- a/src/utils/file.ts +++ b/src/utils/file.ts @@ -308,7 +308,7 @@ interface MpResponse { errcode: number errmsg: string } -async function getMpToken(appID: string, appsecret: string) { +async function getMpToken(appID: string, appsecret: string, proxyOrigin: string) { const data = localStorage.getItem(`mpToken:${appID}`) if (data) { const token = JSON.parse(data) @@ -324,7 +324,10 @@ async function getMpToken(appID: string, appsecret: string) { secret: appsecret, }, } - const url = `https://api.weixin.qq.com/cgi-bin/stable_token` + let url = `https://api.weixin.qq.com/cgi-bin/stable_token` + if (proxyOrigin) { + url = `${proxyOrigin}/cgi-bin/stable_token` + } const res = await fetch(url, requestOptions) if (res.access_token) { const tokenInfo = { @@ -337,13 +340,13 @@ async function getMpToken(appID: string, appsecret: string) { return `` } async function mpFileUpload(file: File) { - const { appID, appsecret } = JSON.parse( + const { appID, appsecret, proxyOrigin } = JSON.parse( localStorage.getItem(`mpConfig`)!, ) /* eslint-disable no-async-promise-executor */ return new Promise(async (resolve, reject) => { try { - const access_token = await getMpToken(appID, appsecret).catch(e => console.error(e)) + const access_token = await getMpToken(appID, appsecret, proxyOrigin).catch(e => console.error(e)) if (!access_token) { reject(new Error(`获取 access_token 失败,请检查console日志`)) return @@ -354,11 +357,18 @@ async function mpFileUpload(file: File) { method: `POST`, data: formdata, } - const url = `https://api.weixin.qq.com/cgi-bin/material/add_material?access_token=${access_token}&type=image` + let url = `https://api.weixin.qq.com/cgi-bin/material/add_material?access_token=${access_token}&type=image` + if (proxyOrigin) { + url = `${proxyOrigin}/cgi-bin/material/add_material?access_token=${access_token}&type=image` + } const res = await fetch(url, requestOptions) - resolve(res.url) + let imageUrl = res.url + if (proxyOrigin && window.location.href.startsWith(`http`)) { + imageUrl = `https://wsrv.nl?url=${encodeURIComponent(imageUrl)}` + } + resolve(imageUrl) } catch (e) { reject(e) diff --git a/wxt.config.ts b/wxt.config.ts index e2b9417..8cc08cf 100644 --- a/wxt.config.ts +++ b/wxt.config.ts @@ -7,7 +7,8 @@ export default defineConfig({ extensionApi: `chrome`, manifest: { name: `公众号文章编辑器`, - version: `0.0.6`, + description: `一款高度简洁的微信 Markdown 编辑器:支持 Markdown 语法、色盘取色、多图上传、一键下载文档、自定义 CSS 样式、一键重置等特性`, + version: `0.0.7`, icons: { 256: `/mpmd/icon-256.png`, },