mirror of
https://github.com/2234839/web-font.git
synced 2025-04-06 05:25:44 +08:00
✨feat(fontmin): 完成基础的动态字体生成功能
This commit is contained in:
parent
88f21ef2f6
commit
fdde780528
5
.vscode/settings.json
vendored
5
.vscode/settings.json
vendored
@ -1,3 +1,6 @@
|
|||||||
{
|
{
|
||||||
"git.ignoreLimitWarning": true
|
"git.ignoreLimitWarning": true,
|
||||||
|
"cSpell.words": [
|
||||||
|
"Fontmin"
|
||||||
|
]
|
||||||
}
|
}
|
3
README.md
Normal file
3
README.md
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
# web font serverless 版
|
||||||
|
|
||||||
|
[在线尝试地址](http://webfontserverless.shenzilong.cn/)
|
@ -2,8 +2,9 @@ targets:
|
|||||||
- backend
|
- backend
|
||||||
|
|
||||||
malagu:
|
malagu:
|
||||||
# serve-static:
|
serve-static:
|
||||||
# root: .malagu/frontend/dist
|
# apiPath: /*
|
||||||
|
apiPath: /api/*
|
||||||
|
|
||||||
faas-adapter:
|
faas-adapter:
|
||||||
customDomain:
|
customDomain:
|
||||||
@ -11,3 +12,8 @@ malagu:
|
|||||||
service:
|
service:
|
||||||
name: web_font # 默认值是 malagu
|
name: web_font # 默认值是 malagu
|
||||||
|
|
||||||
|
includeModules: true
|
||||||
|
webpack:
|
||||||
|
config:
|
||||||
|
externals: fontmin
|
||||||
|
|
||||||
|
@ -12,7 +12,9 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@malagu/fc-adapter": "latest",
|
"@malagu/fc-adapter": "latest",
|
||||||
"@malagu/mvc": "latest",
|
"@malagu/mvc": "latest",
|
||||||
"@malagu/serve-static": "latest"
|
"@malagu/serve-static": "latest",
|
||||||
|
"fontmin": "^0.9.8",
|
||||||
|
"zip-a-folder": "^0.1.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@malagu/cli": "latest",
|
"@malagu/cli": "latest",
|
||||||
@ -27,4 +29,4 @@
|
|||||||
"deploy:pre": "malagu deploy -m pre",
|
"deploy:pre": "malagu deploy -m pre",
|
||||||
"deploy:prod": "malagu deploy -m prod"
|
"deploy:prod": "malagu deploy -m prod"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
91
apps/backend/src/fontmin.ts
Normal file
91
apps/backend/src/fontmin.ts
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
import { Controller, File, Get, Query } from "@malagu/mvc/lib/node";
|
||||||
|
import { Context } from "@malagu/web/lib/node";
|
||||||
|
import { createHash } from "crypto";
|
||||||
|
//@ts-ignore
|
||||||
|
import * as Fontmin from "fontmin";
|
||||||
|
import { promises as fs } from "fs";
|
||||||
|
import * as path from "path";
|
||||||
|
import { join } from "path";
|
||||||
|
//@ts-ignore
|
||||||
|
import { zip } from "zip-a-folder";
|
||||||
|
import { Stream } from "stream";
|
||||||
|
|
||||||
|
const font_src = path.join(__dirname, "../../frontend/font");
|
||||||
|
console.log("[__dirname]", __dirname);
|
||||||
|
console.log("[font_src]", font_src);
|
||||||
|
@Controller("api")
|
||||||
|
export class FontMinController {
|
||||||
|
@Get("generate_fonts_dynamically")
|
||||||
|
@File()
|
||||||
|
async generate_fonts_dynamically(
|
||||||
|
@Query("text") text: string,
|
||||||
|
@Query("font") font: string,
|
||||||
|
@Query("temp") temp: string = "true",
|
||||||
|
@Query("type") type: string = "ttf",
|
||||||
|
) {
|
||||||
|
console.log("字体请求", { text, font, temp, type });
|
||||||
|
|
||||||
|
text += "●";
|
||||||
|
const res = Context.getResponse();
|
||||||
|
|
||||||
|
const hash = createHash("md5");
|
||||||
|
hash.update(`${type}${font}${text}`);
|
||||||
|
const hash_str = hash.digest("hex");
|
||||||
|
|
||||||
|
const srcPath = path.join(font_src, `${font.replace(/.ttf$/, "")}.ttf`); // 字体源文件
|
||||||
|
const outPath = `asset/dynamically/${hash_str}`;
|
||||||
|
const destPath = `./${outPath}`; // 输出路径
|
||||||
|
|
||||||
|
const full_path = join(__dirname, "../../", destPath, `${font}.${type}`);
|
||||||
|
/** 需要持久化 */
|
||||||
|
if (temp !== "true") {
|
||||||
|
try {
|
||||||
|
return await fs.readFile(full_path);
|
||||||
|
} catch (error) {
|
||||||
|
console.log(`开始生成 ${full_path}`, error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 初始化
|
||||||
|
const fontmin = new Fontmin()
|
||||||
|
.src(srcPath) // 输入配置
|
||||||
|
.use(
|
||||||
|
Fontmin.glyph({
|
||||||
|
text, // 所需文字
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
if ("eot" === type) {
|
||||||
|
fontmin.use(Fontmin.ttf2eot()); // eot 转换插件
|
||||||
|
}
|
||||||
|
if ("woff" === type) {
|
||||||
|
fontmin.use(Fontmin.ttf2woff()); // eot 转换插件
|
||||||
|
}
|
||||||
|
if ("svg" === type) {
|
||||||
|
fontmin.use(Fontmin.ttf2svg()); // eot 转换插件
|
||||||
|
}
|
||||||
|
/** 缓存数据 */
|
||||||
|
if (temp !== "true") {
|
||||||
|
fontmin.dest(destPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 执行
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
fontmin.run(async function (err: Error, files: any[]) {
|
||||||
|
if (err) {
|
||||||
|
// 异常捕捉
|
||||||
|
reject(err);
|
||||||
|
} else {
|
||||||
|
const buffer = files.filter((f) =>
|
||||||
|
/** 筛选需要的类型 */
|
||||||
|
(f.history[f.history.length - 1] as string).endsWith(type),
|
||||||
|
)[0]._contents;
|
||||||
|
console.log("[buffer]", Buffer.isBuffer(buffer));
|
||||||
|
const bufferStream = new Stream.PassThrough();
|
||||||
|
bufferStream.end(buffer);
|
||||||
|
bufferStream.pipe(res);
|
||||||
|
// resolve(buffer); // 成功
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
@ -1,12 +0,0 @@
|
|||||||
import { Controller, Get, Text } from "@malagu/mvc/lib/node";
|
|
||||||
|
|
||||||
@Controller()
|
|
||||||
export class HomeController {
|
|
||||||
@Get("/12")
|
|
||||||
@Text()
|
|
||||||
home(): string {
|
|
||||||
console.log(222);
|
|
||||||
|
|
||||||
return "Welcome to Malagu";
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +1,4 @@
|
|||||||
import './home-controller';
|
import './fontmin';
|
||||||
import { autoBind } from '@malagu/core';
|
import { autoBind } from '@malagu/core';
|
||||||
console.log("app start");
|
|
||||||
|
|
||||||
export default autoBind();
|
export default autoBind();
|
||||||
|
@ -1,27 +1,80 @@
|
|||||||
<template>
|
<template>
|
||||||
<img alt="Vue logo" src="./assets/logo.png" />
|
<h1>
|
||||||
<HelloWorld msg="Hello Vue 3 + TypeScript + Vite" />
|
web font 字体裁剪工具 (<a href="https://github.com/cellbang/malagu"
|
||||||
|
>serverless版</a
|
||||||
|
>)
|
||||||
|
</h1>
|
||||||
|
<label style="display: flex; width: 100%">
|
||||||
|
<span style="width: 30%; flex-shrink: 0">使用该字体的文字:</span>
|
||||||
|
<input type="text" v-model="text" style="flex: 1" />
|
||||||
|
</label>
|
||||||
|
<label style="display: flex; width: 100%">
|
||||||
|
<span style="width: 30%; flex-shrink: 0">选择使用的字体:</span>
|
||||||
|
<input disabled type="text" v-model="font" style="flex: 1" />
|
||||||
|
</label>
|
||||||
|
<label style="display: flex; width: 100%">
|
||||||
|
<span style="width: 30%; flex-shrink: 0">选择字体类型:</span>
|
||||||
|
<input disabled type="text" v-model="type" style="flex: 1" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<span>你可以直接 copy 下面的 css 去使用</span>
|
||||||
|
<textarea
|
||||||
|
v-model="cssCode"
|
||||||
|
disabled
|
||||||
|
:cols="20"
|
||||||
|
:rows="16"
|
||||||
|
style="width: 95%; font-family: test"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<footer>
|
||||||
|
<a href="https://github.com/2234839/web-font/tree/serverless"
|
||||||
|
>github repo</a
|
||||||
|
>
|
||||||
|
</footer>
|
||||||
|
<div v-html="style" />
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from 'vue'
|
import { computed, defineComponent, ref } from "vue";
|
||||||
import HelloWorld from './components/HelloWorld.vue'
|
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'App',
|
name: "App",
|
||||||
components: {
|
setup() {
|
||||||
HelloWorld
|
const text = ref("千图小兔体webfont字体裁剪工具(serverless版)");
|
||||||
}
|
const font = ref("千图小兔体.ttf");
|
||||||
})
|
const temp = ref("true");
|
||||||
|
const type = ref("ttf");
|
||||||
|
const cssCode = computed(
|
||||||
|
() => `* { font-family: test}\n
|
||||||
|
@font-face {
|
||||||
|
font-family: "test";
|
||||||
|
src: url("http://webfontserverless.shenzilong.cn/api/generate_fonts_dynamically?text=${text.value.trim()}&font=${
|
||||||
|
font.value
|
||||||
|
}&temp=${temp.value}&type=${type.value}")
|
||||||
|
format("truetype");
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: normal;
|
||||||
|
}`,
|
||||||
|
);
|
||||||
|
const style = computed(() => `<style>${cssCode.value}</style>`);
|
||||||
|
return { cssCode, style, text, font, temp, type };
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const defaultCssCode = ``;
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
#app {
|
#app {
|
||||||
font-family: Avenir, Helvetica, Arial, sans-serif;
|
text-align: center;
|
||||||
-webkit-font-smoothing: antialiased;
|
color: #2c3e50;
|
||||||
-moz-osx-font-smoothing: grayscale;
|
display: flex;
|
||||||
text-align: center;
|
flex-direction: column;
|
||||||
color: #2c3e50;
|
align-items: center;
|
||||||
margin-top: 60px;
|
}
|
||||||
}
|
#app,
|
||||||
</style>
|
textarea,
|
||||||
|
input {
|
||||||
|
font-size: 1.3em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user