mirror of
https://github.com/2234839/web-font.git
synced 2025-04-06 05:25:44 +08:00
🔥 允许 post 方式请求 fontmin 接口
This commit is contained in:
parent
1f574912b4
commit
f21034d5ef
3
.gitignore
vendored
3
.gitignore
vendored
@ -35,4 +35,5 @@ lerna-debug.log*
|
|||||||
|
|
||||||
.cache
|
.cache
|
||||||
dist
|
dist
|
||||||
asset
|
asset/font
|
||||||
|
asset/dynamically
|
11
README.md
11
README.md
@ -41,7 +41,7 @@ ui 需要展现一些特定的字体,但直接引入字体包又过大,于
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
3.将 ttf 的字体包放置在 ./src/font/ 目录下自然可以检测到新的可用字体,无需重启服务
|
3.将 ttf 的字体包放置在 ./asset/font_src/ 目录下自然可以检测到新的可用字体,无需重启服务
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@ -61,6 +61,11 @@ ui 需要展现一些特定的字体,但直接引入字体包又过大,于
|
|||||||
|
|
||||||
如图可见每个返回的字体资源,访问即可下载。另外在访问该目录下的 asset.zip 可以直接下载全部的文件,生成的资源目录结构见下图
|
如图可见每个返回的字体资源,访问即可下载。另外在访问该目录下的 asset.zip 可以直接下载全部的文件,生成的资源目录结构见下图
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
注意,此接口是还支持 post 方式访问的,这样可以一次请求多个类型的字体文件,而且不会如同 get 方法那样有长度限制
|
||||||
|
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 动态生成字体
|
### 动态生成字体
|
||||||
@ -73,7 +78,9 @@ ui 需要展现一些特定的字体,但直接引入字体包又过大,于
|
|||||||
|
|
||||||
## 写项目时遇到的问题
|
## 写项目时遇到的问题
|
||||||
|
|
||||||
使用 svelte https://github.com/DeMoorJasper/parcel-plugin-svelte 通过这个插件使用 parcel 然后报 new 的错 需要限制 编译的版本,在package.json browserslist 字段限制一下版本就好
|
1. 使用 svelte https://github.com/DeMoorJasper/parcel-plugin-svelte 通过这个插件使用 parcel 然后报 new 的错 需要限制 编译的版本,在package.json browserslist 字段限制一下版本就好
|
||||||
|
|
||||||
|
2. parcel 对 post purgecss 支持好像有问题,需要修改 postcss.config.js 文件他才能正确的删除样式
|
||||||
|
|
||||||
## 启动
|
## 启动
|
||||||
|
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 66 KiB After Width: | Height: | Size: 82 KiB |
BIN
doc_img/api/fontmin_post.jpg
Normal file
BIN
doc_img/api/fontmin_post.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 101 KiB |
@ -9,7 +9,7 @@ const purgecss = require('@fullhuman/postcss-purgecss')({
|
|||||||
'./static/*.svelte',
|
'./static/*.svelte',
|
||||||
'./static/*.js',
|
'./static/*.js',
|
||||||
'./static/*.ts',
|
'./static/*.ts',
|
||||||
// etc.
|
`111 `
|
||||||
],
|
],
|
||||||
|
|
||||||
// Include any special characters you're using in this regular expression
|
// Include any special characters you're using in this regular expression
|
||||||
|
@ -6,27 +6,53 @@ import {
|
|||||||
Response,
|
Response,
|
||||||
Res,
|
Res,
|
||||||
Req,
|
Req,
|
||||||
|
Post,
|
||||||
|
Body,
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
import { AppService } from './app.service';
|
import { AppService } from './app.service';
|
||||||
import { join } from 'path';
|
import { join } from 'path';
|
||||||
import { promises as fs } from 'fs';
|
import { promises as fs } from 'fs';
|
||||||
import { Request } from 'express';
|
import { Request } from 'express';
|
||||||
import { Stream } from 'stream';
|
import { Stream } from 'stream';
|
||||||
|
import { req_par } from './req.decorator';
|
||||||
@Controller()
|
@Controller()
|
||||||
export class AppController {
|
export class AppController {
|
||||||
constructor(private readonly appService: AppService) {}
|
constructor(private readonly appService: AppService) {}
|
||||||
|
|
||||||
/** 压缩字体 */
|
/** 压缩字体 */
|
||||||
@Get('fontmin')
|
@Get('fontmin')
|
||||||
font_min(@Query('text') text, @Query('font') font) {
|
font_min(@Query('text') text, @Query('font') font,@req_par('host_url') host_url:string) {
|
||||||
return this.appService.font_min(text, font);
|
|
||||||
|
return this.appService.font_min(text, font,host_url);
|
||||||
|
}
|
||||||
|
@Post('fontmin')
|
||||||
|
font_min_post(@Body() body: { text: string; font: string }[],@req_par('host_url') host_url:string) {
|
||||||
|
const res = body.map(par => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
this.appService
|
||||||
|
.font_min(par.text, par.font,host_url)
|
||||||
|
.then(r => {
|
||||||
|
resolve({
|
||||||
|
font: par.font,
|
||||||
|
css:r,
|
||||||
|
status: 'success',
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.catch(e => {
|
||||||
|
resolve({
|
||||||
|
font: par.font,
|
||||||
|
status: 'failure',
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return Promise.all(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 返回字体列表 */
|
/** 返回字体列表 */
|
||||||
@Get('font_list')
|
@Get('font_list')
|
||||||
font_list() {
|
font_list() {
|
||||||
const font_dir = join(__dirname, '../../src/font');
|
return this.appService.font_list();
|
||||||
return fs.readdir(font_dir);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 压缩字体 */
|
/** 压缩字体 */
|
||||||
@ -42,8 +68,7 @@ export class AppController {
|
|||||||
res.set({
|
res.set({
|
||||||
'Content-Type': `font/${type}`,
|
'Content-Type': `font/${type}`,
|
||||||
});
|
});
|
||||||
if(!text)
|
if (!text) return ' ';
|
||||||
return ' '
|
|
||||||
const file = await this.appService.generate_fonts_dynamically(
|
const file = await this.appService.generate_fonts_dynamically(
|
||||||
text,
|
text,
|
||||||
font,
|
font,
|
||||||
@ -56,3 +81,5 @@ export class AppController {
|
|||||||
bufferStream.pipe(res);
|
bufferStream.pipe(res);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function promise_execute_all<T>(params: Promise<T>[]) {}
|
||||||
|
@ -5,10 +5,18 @@ import { join } from 'path';
|
|||||||
import crypto from 'crypto';
|
import crypto from 'crypto';
|
||||||
import { config } from './config';
|
import { config } from './config';
|
||||||
import { promises as fs } from 'fs';
|
import { promises as fs } from 'fs';
|
||||||
|
import { req_par } from './req.decorator';
|
||||||
|
|
||||||
|
const font_src="./asset/font_src/"
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class AppService {
|
export class AppService {
|
||||||
font_min(text: string, font: string) {
|
font_list() {
|
||||||
const srcPath = `./src/font/${font}.ttf`; // 字体源文件
|
const font_dir = join(__dirname, `../../${font_src}`);
|
||||||
|
return fs.readdir(font_dir);
|
||||||
|
}
|
||||||
|
font_min(text: string, font: string,server_url:string) {
|
||||||
|
const srcPath = `${font_src}${font}.ttf`; // 字体源文件
|
||||||
const outPath = `asset/font/${Date.now()}/`;
|
const outPath = `asset/font/${Date.now()}/`;
|
||||||
const destPath = `./${outPath}`; // 输出路径
|
const destPath = `./${outPath}`; // 输出路径
|
||||||
// 初始化
|
// 初始化
|
||||||
@ -22,7 +30,7 @@ export class AppService {
|
|||||||
.use(Fontmin.ttf2eot()) // eot 转换插件
|
.use(Fontmin.ttf2eot()) // eot 转换插件
|
||||||
.use(Fontmin.ttf2woff()) // woff 转换插件
|
.use(Fontmin.ttf2woff()) // woff 转换插件
|
||||||
.use(Fontmin.ttf2svg()) // svg 转换插件
|
.use(Fontmin.ttf2svg()) // svg 转换插件
|
||||||
.use(Fontmin.css({ fontPath: `${config.web_font_path}${outPath}` })) // css 生成插件
|
.use(Fontmin.css({ fontPath: `${server_url}${outPath}` })) // css 生成插件
|
||||||
.dest(destPath); // 输出配置
|
.dest(destPath); // 输出配置
|
||||||
|
|
||||||
// 执行
|
// 执行
|
||||||
@ -36,7 +44,7 @@ export class AppService {
|
|||||||
.filter(f =>
|
.filter(f =>
|
||||||
(f.history[f.history.length - 1] as string).endsWith('.css'),
|
(f.history[f.history.length - 1] as string).endsWith('.css'),
|
||||||
)
|
)
|
||||||
.map(f => f._contents.toString());
|
.map(f => f._contents.toString())[0];
|
||||||
zip(
|
zip(
|
||||||
join(__dirname, '../../', destPath),
|
join(__dirname, '../../', destPath),
|
||||||
join(__dirname, '../../', destPath, 'asset.zip'),
|
join(__dirname, '../../', destPath, 'asset.zip'),
|
||||||
@ -58,7 +66,7 @@ export class AppService {
|
|||||||
const hash = crypto.createHash('md5');
|
const hash = crypto.createHash('md5');
|
||||||
hash.update(`${type}${font}${text}`);
|
hash.update(`${type}${font}${text}`);
|
||||||
const hash_str = hash.digest('hex');
|
const hash_str = hash.digest('hex');
|
||||||
const srcPath = `./src/font/${font}.ttf`; // 字体源文件
|
const srcPath = `${font_src}${font}.ttf`; // 字体源文件
|
||||||
const outPath = `asset/dynamically/${hash_str}`;
|
const outPath = `asset/dynamically/${hash_str}`;
|
||||||
const destPath = `./${outPath}`; // 输出路径
|
const destPath = `./${outPath}`; // 输出路径
|
||||||
|
|
||||||
|
9
src/req.decorator.ts
Normal file
9
src/req.decorator.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import { createParamDecorator } from '@nestjs/common';
|
||||||
|
import { Request } from 'express';
|
||||||
|
|
||||||
|
export const req_par = createParamDecorator((data: string, req:Request) => {
|
||||||
|
if('host_url'===data){
|
||||||
|
return `//${req.headers.host}/`
|
||||||
|
}
|
||||||
|
return req
|
||||||
|
});
|
@ -1,19 +1,21 @@
|
|||||||
<script>
|
<script>
|
||||||
import { writable } from 'svelte/store';
|
import { writable } from 'svelte/store';
|
||||||
import { get_font, get_font_list,server } from './req';
|
import { get_font, get_font_list,server,post_fontmin } from './req';
|
||||||
/** 可用的字体列表 {id:number,name:string:selected:undefined | boolen,css:undefined|string}*/
|
/** 可用的字体列表 {id:number,name:string:selected:undefined | boolen,css:undefined|string}*/
|
||||||
$: font_list = [];
|
$: font_list = [];
|
||||||
get_font_list().then(r => {
|
get_font_list().then(r => {
|
||||||
font_list = r.map(ttf => ({ name: ttf.replace(/\.ttf$/, '') }));
|
font_list = r.map(ttf => ({ name: ttf.replace(/\.ttf$/, '') }));
|
||||||
});
|
});
|
||||||
/** 选择的文字 */
|
/** 选择的文字 */
|
||||||
let text = '';
|
let text = '在此输入需要提取的文字\n在右侧选择字体\n然后点击下方的生成字体按钮';
|
||||||
|
/** 请求方式 */
|
||||||
|
let request_method="post"
|
||||||
/** 用于测试动态生成接口 */
|
/** 用于测试动态生成接口 */
|
||||||
let generate_fonts_dynamically=`<style>
|
let generate_fonts_dynamically=`<style>
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "test";
|
font-family: "test";
|
||||||
src:
|
src:
|
||||||
url("${location.pathname}generate_fonts_dynamically.ttf?temp=true&font=优设标题黑&text=优设标题黑(直接改这里和前面的字体名看效果)") format("truetype");
|
url("${server}generate_fonts_dynamically.ttf?temp=true&font=优设标题黑&text=优设标题黑(直接改这里和前面的字体名看效果)") format("truetype");
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
}
|
}
|
||||||
@ -21,9 +23,30 @@
|
|||||||
</style>`
|
</style>`
|
||||||
$: selected_font = font_list.filter(font => font.selected);
|
$: selected_font = font_list.filter(font => font.selected);
|
||||||
function generate_font() {
|
function generate_font() {
|
||||||
|
if('post'===request_method){
|
||||||
|
/** 使用 post 请求,单请求方式 */
|
||||||
|
post_fontmin(
|
||||||
|
selected_font.map(f=>({
|
||||||
|
font:f.name, text
|
||||||
|
}))
|
||||||
|
).then(res=>{
|
||||||
|
selected_font.forEach(font=>{
|
||||||
|
let r=res.find(o=>o.font===font.name).css
|
||||||
|
font_processing(font,r)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if('get'===request_method){
|
||||||
|
/** 使用 get 请求,多请求方式 */
|
||||||
selected_font.forEach(font => {
|
selected_font.forEach(font => {
|
||||||
get_font(font.name, text)
|
get_font(font.name, text)
|
||||||
.then(r => {
|
.then(r => {
|
||||||
|
font_processing(font,r)
|
||||||
|
})
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function font_processing(font,r) {
|
||||||
r=r.replace(/\/\/.*?\//g,server)
|
r=r.replace(/\/\/.*?\//g,server)
|
||||||
|
|
||||||
const family = r.match(/font-family: "(.*)"/)[1];
|
const family = r.match(/font-family: "(.*)"/)[1];
|
||||||
@ -32,11 +55,7 @@
|
|||||||
font.zip=server+r.match(/(asset\/font\/\d+\/)/)[0]+'asset.zip'
|
font.zip=server+r.match(/(asset\/font\/\d+\/)/)[0]+'asset.zip'
|
||||||
/** 因为要触发其他更新则必须对这个变量重新赋值 */
|
/** 因为要触发其他更新则必须对这个变量重新赋值 */
|
||||||
font_list = font_list;
|
font_list = font_list;
|
||||||
})
|
}
|
||||||
.catch(e => {
|
|
||||||
console.log(e);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
function copy(str) {
|
function copy(str) {
|
||||||
var input = document.getElementById("copy_box");
|
var input = document.getElementById("copy_box");
|
||||||
@ -57,7 +76,7 @@
|
|||||||
<textarea
|
<textarea
|
||||||
bind:value={text}
|
bind:value={text}
|
||||||
class="border flex-1 m-1"
|
class="border flex-1 m-1"
|
||||||
placeholder="在此输入需要提取的文字"
|
placeholder="在此输入需要提取的文字 在右侧选择字体 然后点击下方的生成字体按钮"
|
||||||
cols="40"
|
cols="40"
|
||||||
rows="3" />
|
rows="3" />
|
||||||
<div class="flex-1 m-1 flex flex-wrap">
|
<div class="flex-1 m-1 flex flex-wrap">
|
||||||
@ -73,9 +92,25 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex">
|
<div class="flex">
|
||||||
<div on:click={generate_font} class="bg-red-200 text-red-600 rounded-md px-1 hover:bg-red-400 hover:text-white duration-75">
|
<div on:click={generate_font} class="bg-red-200 text-red-600 rounded-md px-2 hover:bg-red-400 hover:text-white duration-75 flex items-center shadow-md">
|
||||||
生成字体
|
生成字体
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="flex border ml-2 items-end">
|
||||||
|
<div
|
||||||
|
on:click={e => request_method="post"}
|
||||||
|
class="c-label {request_method==="post" ? 'c-label-selected' : ''}">
|
||||||
|
使用 post 请求
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
on:click={e => request_method="get"}
|
||||||
|
class="c-label {request_method==="get" ? 'c-label-selected' : ''}">
|
||||||
|
使用 get 请求
|
||||||
|
</div>
|
||||||
|
<div class="text-sm">* 具体区别请打开控制台查看请求</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#each selected_font as font, i}
|
{#each selected_font as font, i}
|
||||||
@ -94,7 +129,7 @@
|
|||||||
|
|
||||||
|
|
||||||
<h2 class="text-lg text-center my-3 font-bold"> 动态生成字体(generate_fonts_dynamically 接口)</h2>
|
<h2 class="text-lg text-center my-3 font-bold"> 动态生成字体(generate_fonts_dynamically 接口)</h2>
|
||||||
<p>使用如下的方式引入,则可以直接使用</p>
|
<p class="ml-1">使用如下的方式引入,则可以直接使用</p>
|
||||||
<textarea
|
<textarea
|
||||||
bind:value={generate_fonts_dynamically}
|
bind:value={generate_fonts_dynamically}
|
||||||
class="border flex-1 m-1 w-full text-lg"
|
class="border flex-1 m-1 w-full text-lg"
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
export const server='//'+location.host+location.pathname
|
export const server = '//' + location.host + location.pathname;
|
||||||
|
/** get 方式压缩字体 */
|
||||||
export function get_font(font: string, text: string) {
|
export function get_font(font: string, text: string) {
|
||||||
return new Promise((rs, re) => {
|
return new Promise((rs, re) => {
|
||||||
var xhr = new XMLHttpRequest();
|
var xhr = new XMLHttpRequest();
|
||||||
xhr.addEventListener('readystatechange', function() {
|
xhr.addEventListener('readystatechange', function() {
|
||||||
if (this.readyState === 4) {
|
if (this.readyState === 4) {
|
||||||
rs(JSON.parse(this.responseText)[0]);
|
rs(this.responseText);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
xhr.open(
|
xhr.open(
|
||||||
@ -13,9 +14,25 @@ export function get_font(font:string, text:string) {
|
|||||||
font,
|
font,
|
||||||
)}&text=${encodeURIComponent(text)}`,
|
)}&text=${encodeURIComponent(text)}`,
|
||||||
);
|
);
|
||||||
xhr.onerror=re
|
xhr.onerror = re;
|
||||||
xhr.send();
|
xhr.send();
|
||||||
})
|
});
|
||||||
|
}
|
||||||
|
/** post 方式压缩字体 */
|
||||||
|
export function post_fontmin(par:{font:string,text:string}[]) {
|
||||||
|
return new Promise((rs, re) => {
|
||||||
|
var data = JSON.stringify(par);
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.addEventListener('readystatechange', function() {
|
||||||
|
if (this.readyState === 4) {
|
||||||
|
rs(JSON.parse(this.responseText) );
|
||||||
|
}
|
||||||
|
});
|
||||||
|
xhr.open('POST', `${server}fontmin`);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/json');
|
||||||
|
xhr.onerror = re;
|
||||||
|
xhr.send(data);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
export function get_font_list(font: string, text: string) {
|
export function get_font_list(font: string, text: string) {
|
||||||
return new Promise((rs, re) => {
|
return new Promise((rs, re) => {
|
||||||
@ -25,11 +42,8 @@ export function get_font_list(font:string, text:string) {
|
|||||||
rs(JSON.parse(this.responseText));
|
rs(JSON.parse(this.responseText));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
xhr.open(
|
xhr.open('GET', `${server}font_list`);
|
||||||
'GET',
|
xhr.onerror = re;
|
||||||
`${server}font_list`,
|
|
||||||
);
|
|
||||||
xhr.onerror=re
|
|
||||||
xhr.send();
|
xhr.send();
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user