Skip to content

Commit 037300e

Browse files
committedJun 16, 2023
chore: update
1 parent e10d69b commit 037300e

13 files changed

+351
-25
lines changed
 

‎package.json

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@webfansplz/vuedoc-parser",
3-
"version": "0.0.2",
3+
"version": "0.0.3",
44
"description": "Generate a JSON documentation for a Vue file",
55
"type": "module",
66
"exports": {
@@ -57,7 +57,6 @@
5757
],
5858
"license": "MIT",
5959
"dependencies": {
60-
"@b613/utils": "^1.0.0-beta21",
6160
"@babel/parser": "^7.18.11",
6261
"@babel/traverse": "^7.18.11",
6362
"resolve": "^1.22.1",

‎src/entity/EventEntry.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import { AbstractCategorizeEntry } from './AbstractCategorizeEntry.js';
22
import { Type } from '../lib/Enum.js';
33
import { Value } from './Value.js';
4-
import { toKebabCase } from '@b613/utils/lib/string.js';
4+
import { toKebabCase } from '../utils/string.js';
55
import { Parser } from '../../types/Parser.js';
66
import { Entry } from '../../types/Entry.js';
77

‎src/entity/ModelEntry.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { toKebabCase } from '@b613/utils/lib/string.js';
1+
import { toKebabCase } from '../utils/string.js';
22
import { Entry } from '../../types/Entry.js';
33
import { Visibility } from '../lib/Enum.js';
44
import { AbstractDecorativeEntry } from './AbstractDecorativeEntry.js';

‎src/entity/PropEntry.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { Type } from '../lib/Enum.js';
22
import { AbstractCategorizeEntry } from './AbstractCategorizeEntry.js';
3-
import { toKebabCase } from '@b613/utils/lib/string.js';
3+
import { toKebabCase } from '../utils/string.js';
44
import { Parser } from '../../types/Parser.js';
55
import { Entry } from '../../types/Entry.js';
66

‎src/lib/Composition.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { get } from '@b613/utils/lib/object.js';
1+
import { get } from '../utils/object.js';
22
import { Parser } from '../../types/Parser.js';
33
import { Value } from '../entity/Value.js';
44
import { Syntax } from './Enum.js';

‎src/lib/FS.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import { readFileSync } from 'node:fs';
55
import { FileSystem, Options, File } from '../../types/FileSystem.js';
66
import { extname, join } from 'node:path';
77
import { parse, ParserPlugin } from '@babel/parser';
8-
import { merge } from '@b613/utils/lib/object.js';
8+
import { merge } from '../utils/object.js';
99

1010
import { ImportResolver } from '../../types/ImportResolver.js';
1111
import { Loader, LoaderOptions } from './Loader.js';
@@ -162,7 +162,7 @@ function parsePlugins(lang: string, options: Pick<FSOptions, 'jsx'> & { path?: s
162162
return plugins;
163163
}
164164

165-
export class ParsingError extends Error {}
165+
export class ParsingError extends Error { }
166166

167167
export class FS implements FileSystem {
168168
private options: Options;

‎src/parsers/AbstractParser.ts

+12-13
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,16 @@
11
import { RestValue } from '../entity/RestValue.js';
22
import { Value, generateUndefineValue, generateNullGenerator, generateObjectGenerator, generateArrayGenerator } from '../entity/Value.js';
33
import { Syntax, Type, TypeList, CompositionTypes, Feature, CompositionFeature } from '../lib/Enum.js';
4-
import { clear, get } from '@b613/utils/lib/object.js';
4+
import { clear, get } from '../utils/object.js';
55
import { Entry } from '../../types/Entry.js';
66
import { Parser } from '../../types/Parser.js';
77
import { FileSystem } from '../../types/FileSystem.js';
88
import { Composition } from '../lib/Composition.js';
9-
import { PropType } from '@b613/utils/typings.js';
109
import { DTS } from '../lib/DTS.js';
11-
1210
import * as Babel from '@babel/types';
1311

12+
type PropType<TObj, TProp extends keyof TObj> = TObj[TProp];
13+
1414
export type CompositionValueOptions = Pick<Babel.VariableDeclarator, 'init'> & {
1515
key?: string;
1616
local?: string;
@@ -249,37 +249,37 @@ export class AbstractParser<Source extends Parser.Source, Root> {
249249
/* istanbul ignore next */
250250
/* eslint-disable no-unused-vars */
251251
/* eslint-disable class-methods-use-this */
252-
parseFunctionDeclaration(_node: Babel.FunctionDeclaration) {}
252+
parseFunctionDeclaration(_node: Babel.FunctionDeclaration) { }
253253

254254
/* istanbul ignore next */
255255
/* eslint-disable no-unused-vars */
256256
/* eslint-disable class-methods-use-this */
257-
parseEntryComment(_entry: Entry.Type, _node) {}
257+
parseEntryComment(_entry: Entry.Type, _node) { }
258258

259259
/* istanbul ignore next */
260260
/* eslint-disable no-unused-vars */
261261
/* eslint-disable class-methods-use-this */
262-
parseArrayExpression(_node: Babel.ArrayExpression) {}
262+
parseArrayExpression(_node: Babel.ArrayExpression) { }
263263

264264
/* istanbul ignore next */
265265
/* eslint-disable no-unused-vars */
266266
/* eslint-disable class-methods-use-this */
267-
parseCallExpression(_node: Babel.CallExpression) {}
267+
parseCallExpression(_node: Babel.CallExpression) { }
268268

269269
/* istanbul ignore next */
270270
/* eslint-disable no-unused-vars */
271271
/* eslint-disable class-methods-use-this */
272-
parseFunctionExpression(_node: Babel.FunctionExpression) {}
272+
parseFunctionExpression(_node: Babel.FunctionExpression) { }
273273

274274
/* istanbul ignore next */
275275
/* eslint-disable no-unused-vars */
276276
/* eslint-disable class-methods-use-this */
277-
parseExpressionStatement(_node: Babel.ExpressionStatement) {}
277+
parseExpressionStatement(_node: Babel.ExpressionStatement) { }
278278

279279
/* istanbul ignore next */
280280
/* eslint-disable no-unused-vars */
281281
/* eslint-disable class-methods-use-this */
282-
parseObjectExpression(_node: Babel.ObjectExpression) {}
282+
parseObjectExpression(_node: Babel.ObjectExpression) { }
283283

284284
parseObjectProperty(node: Babel.ObjectProperty) {
285285
this.parse(node.value);
@@ -1506,9 +1506,8 @@ export class AbstractParser<Source extends Parser.Source, Root> {
15061506
return new Value(element.type, value);
15071507
});
15081508

1509-
output.raw = `[${
1510-
output.rawObject.map((ref) => ref.raw).join(',')
1511-
}]`;
1509+
output.raw = `[${output.rawObject.map((ref) => ref.raw).join(',')
1510+
}]`;
15121511

15131512
return output;
15141513
}

‎src/parsers/ScriptRegisterParser.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { Options } from './ScriptParser.js';
22
import { CompositionParser } from './CompositionParser.js';
3-
import { clear } from '@b613/utils/lib/object.js';
3+
import { clear } from '../utils/object.js';
44
import { Syntax } from '../lib/Enum.js';
55
import { Parser } from '../../types/Parser.js';
66
import { File } from '../../types/FileSystem.js';

‎src/parsers/VuedocParser.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import { NameEntry } from '../entity/NameEntry.js';
77
import { Composition } from '../lib/Composition.js';
88
import { Feature, Features, DEFAULT_IGNORED_VISIBILITIES, TypedocTag, JSDocTag, FeatureEvent } from '../lib/Enum.js';
99
import { Loader as LoaderNS } from '../../types/Loader.js';
10-
import { clear, merge } from '@b613/utils/lib/object.js';
10+
import { clear, merge } from '../utils/object.js';
1111
import { RegisterFactory, ScriptRegisterParser } from './ScriptRegisterParser.js';
1212

1313
import { FS, ParsingError } from '../lib/FS.js';

‎src/test/utils.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { parseComponent } from '../index.js';
22
import { beforeAll, describe, expect, it } from 'vitest';
3-
import { merge } from '@b613/utils/lib/object.js';
3+
import { merge } from '../utils/object.js';
44

55
export const ComponentTestCase = ({ name, description, only = false, expected, options }) => {
66
const title = description ? `${name}: ${description}` : name;

‎src/utils/object.ts

+242
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,242 @@
1+
/* eslint-disable radar/cognitive-complexity */
2+
3+
export class ToStringEscape {
4+
readonly value: string;
5+
6+
constructor(value: string) {
7+
this.value = value;
8+
}
9+
}
10+
11+
export function isObject(object: Record<string, any>, strict = true): boolean {
12+
return typeof object === 'object' && object !== null && (!strict || !Array.isArray(object));
13+
}
14+
15+
export function isEmpty(object: Record<string, any>): boolean {
16+
// eslint-disable-next-line no-unreachable-loop
17+
for (const key in object) {
18+
return false;
19+
}
20+
21+
return true;
22+
}
23+
24+
export function isEqual(object1: Record<string, any>, object2: Record<string, any>): boolean {
25+
if (object1 === object2) {
26+
return true;
27+
}
28+
29+
if (isObject(object1, false) && isObject(object2, false)) {
30+
if (Object.keys(object1).length !== Object.keys(object2).length) {
31+
return false;
32+
}
33+
34+
for (const prop in object1) {
35+
if (object2.hasOwnProperty(prop)) {
36+
if (!isEqual(object1[prop], object2[prop])) {
37+
return false;
38+
}
39+
} else {
40+
return false;
41+
}
42+
}
43+
44+
return true;
45+
}
46+
47+
return false;
48+
}
49+
50+
/**
51+
* Delete all object's properties
52+
*/
53+
export function clear(object: Record<string, any>) {
54+
for (const key in object) {
55+
delete object[key];
56+
}
57+
}
58+
59+
export function copy<T extends Record<string, any> = Record<string, any>>(object: T): T {
60+
if (typeof object !== 'object' || object === null) {
61+
return object;
62+
}
63+
64+
if (object instanceof Array) {
65+
return object.map(copy) as any;
66+
}
67+
68+
const copyObject: Record<string, any> = {};
69+
70+
for (const key in object) {
71+
const value: any = object[key];
72+
73+
if (typeof value === 'object' && value !== null) {
74+
if (value instanceof Array) {
75+
copyObject[key] = value.map((item) => copy(item));
76+
} else {
77+
copyObject[key] = copy(value);
78+
}
79+
} else {
80+
copyObject[key] = value;
81+
}
82+
}
83+
84+
return copyObject as T;
85+
}
86+
87+
/**
88+
* Deep assign implementation (merge)
89+
*/
90+
export function merge(dest: Record<string, any>, src: Record<string, any>, strategy = 'append') {
91+
if (isObject(dest) && isObject(src)) {
92+
for (const key in src) {
93+
if (isObject(src[key])) {
94+
dest[key] = key in dest && isObject(dest[key])
95+
? merge(dest[key], src[key], strategy)
96+
: copy(src[key]);
97+
} else if (dest[key] instanceof Array) {
98+
if (strategy === 'replace') {
99+
dest[key].splice(0);
100+
}
101+
102+
dest[key].push(...copy(src[key]));
103+
} else {
104+
Object.assign(dest, { [key]: src[key] });
105+
}
106+
}
107+
} else if (src instanceof Array && dest instanceof Array) {
108+
dest.push(...copy(src));
109+
}
110+
111+
return dest;
112+
}
113+
114+
export function get<T = unknown>(object: Record<string, any>, path: string): T | undefined {
115+
const paths = parsePath(path);
116+
117+
return paths.length
118+
? foundPaths(object, paths, null, (result, o, key, val) => (result ? o[key] : undefined))
119+
: object as T;
120+
}
121+
122+
export function set(object: Record<string, any>, path: string, value: unknown): boolean {
123+
return applyPaths<boolean>(object, path, value, (result, o, key, val) => {
124+
if (result) {
125+
o[key] = val;
126+
}
127+
128+
return result;
129+
}, true);
130+
}
131+
132+
export function del(object: Record<string, any>, path: string): boolean {
133+
return applyPaths<boolean>(object, path, null, (result, o, key, val) => {
134+
if (result) {
135+
delete o[key];
136+
}
137+
138+
return result;
139+
});
140+
}
141+
142+
/**
143+
* Converts a JavaScript object or value to a JSON string with support of function value
144+
*
145+
* @param object The value to convert to a JSON string
146+
* @param space The space argument may be used to control spacing in the final string.
147+
* - If it is a number, successive levels in the stringification will each
148+
* be indented by this many space characters (up to 10).
149+
* - If it is a string, successive levels will be indented by this string
150+
* (or the first ten characters of it).
151+
* @returns A JSON string representing the given value with stringified functions, or undefined.
152+
*/
153+
export function toString(object: object, space?: string | number): string {
154+
let index = 0;
155+
const cache: Record<string, string> = {};
156+
let objectString = JSON.stringify(object, (key, value) => {
157+
if (value instanceof Function) {
158+
const cacheKey = `__${key}${index++}__`;
159+
const body = value.toString();
160+
161+
cache[cacheKey] = body.startsWith(`${key}(`)
162+
? 'function ' + body.substring(key.length)
163+
: body;
164+
165+
return cacheKey;
166+
}
167+
168+
if (value instanceof ToStringEscape) {
169+
const cacheKey = `__${key}${index++}__`;
170+
171+
cache[cacheKey] = value.value;
172+
173+
return cacheKey;
174+
}
175+
176+
return value;
177+
}, space);
178+
179+
Object.entries(cache).forEach(([cacheKey, cacheValue]) => {
180+
objectString = objectString.replace(`"${cacheKey}"`, cacheValue);
181+
});
182+
183+
return objectString;
184+
}
185+
186+
function parsePath(path: string) {
187+
return path.split(/\./).filter((item) => item);
188+
}
189+
190+
function applyPaths<T>(
191+
object: Record<string, any>,
192+
path: string,
193+
value: unknown,
194+
callback: (result: boolean, o: Record<string, any>, key: string, value?: unknown) => unknown,
195+
initMode = false
196+
): T {
197+
const paths = parsePath(path);
198+
const results = foundPaths<boolean | boolean[]>(object, paths, value, callback, initMode);
199+
200+
return results instanceof Array
201+
? results.every((result) => result)
202+
: results as any;
203+
}
204+
205+
function foundPaths<T>(
206+
object: Record<string, any>,
207+
[key, ...nextKeys]: string[],
208+
value: unknown,
209+
callback: (result: boolean, o: Record<string, any>, key: string, value?: unknown) => unknown,
210+
initMode = false
211+
): T {
212+
if (typeof object === 'object' && object !== null) {
213+
const nextKey = nextKeys[0];
214+
215+
if (initMode && !(key in object)) {
216+
object[key] = {};
217+
}
218+
219+
if (nextKeys.length === 1 && nextKey) {
220+
const entry = object[key];
221+
const entryIsArray = entry instanceof Array;
222+
const entries = entryIsArray ? entry as Record<string, any>[] : [entry];
223+
const results = value instanceof Array
224+
? entries.filter((item) => item).map((item, index) => callback(true, item, nextKey, value[index])) as T[]
225+
: entries.filter((item) => item).map((item) => callback(true, item, nextKey, value));
226+
227+
return entryIsArray ? results as any : results[0];
228+
}
229+
230+
if (key in object) {
231+
return nextKey
232+
? foundPaths(object[key], nextKeys, value, callback, initMode)
233+
: callback(true, object, key, value) as T;
234+
}
235+
236+
if (object instanceof Array) {
237+
return object.map((item) => foundPaths(item, nextKeys, value, callback, initMode)) as any;
238+
}
239+
}
240+
241+
return callback(false, object, key) as T;
242+
}

‎src/utils/string.ts

+85
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
2+
const RE_TRIMLINES = /(.+)(\n|\r\n)?/g;
3+
const CHAR_CODE_0 = '0'.charCodeAt(0);
4+
const CHAR_CODE_9 = '9'.charCodeAt(0);
5+
// eslint-disable-next-line camelcase
6+
const CHAR_CODE_a = 'a'.charCodeAt(0);
7+
// eslint-disable-next-line camelcase
8+
const CHAR_CODE_z = 'z'.charCodeAt(0);
9+
const CHAR_CODE_A = 'A'.charCodeAt(0);
10+
const CHAR_CODE_Z = 'Z'.charCodeAt(0);
11+
12+
13+
export function trimlines(str: string, separator = ''): string {
14+
let index = 0;
15+
const output: string[] = [];
16+
17+
while (index < str.length && index !== -1) {
18+
const matches = RE_TRIMLINES.exec(str);
19+
20+
if (!matches) {
21+
break;
22+
}
23+
24+
index = matches.index;
25+
output.push(matches[0].trim());
26+
}
27+
28+
return output.join(separator);
29+
}
30+
31+
function isLowerCode(code: number) {
32+
// eslint-disable-next-line camelcase
33+
return code >= CHAR_CODE_a && code <= CHAR_CODE_z;
34+
}
35+
36+
function isNumberCode(code: number) {
37+
// eslint-disable-next-line camelcase
38+
return code >= CHAR_CODE_0 && code <= CHAR_CODE_9;
39+
}
40+
41+
export type ParseCaseOptions = {
42+
escapedChars?: string[];
43+
separator: string;
44+
};
45+
46+
// eslint-disable-next-line radar/cognitive-complexity
47+
export function parseCase(str: string, { escapedChars = [], separator }: ParseCaseOptions): string {
48+
let output = '';
49+
50+
for (let index = 0; index < str.length; index++) {
51+
const char = str.charAt(index);
52+
53+
if (escapedChars.includes(char)) {
54+
output += char;
55+
} else {
56+
const code = str.charCodeAt(index);
57+
58+
if (code >= CHAR_CODE_A && code <= CHAR_CODE_Z) {
59+
if (index) {
60+
const previousToken = output.at(-1);
61+
62+
if (previousToken !== separator && !escapedChars.includes(previousToken)) {
63+
const previousCode = str.charCodeAt(index - 1);
64+
65+
if (isLowerCode(previousCode) || isNumberCode(previousCode)) {
66+
output += separator;
67+
}
68+
}
69+
}
70+
71+
output += String.fromCharCode(code + 32);
72+
} else if (isLowerCode(code) || isNumberCode(code)) {
73+
output += char;
74+
} else {
75+
output += separator;
76+
}
77+
}
78+
}
79+
80+
return output;
81+
}
82+
83+
export function toKebabCase(str: string, escapedChars: string[] = []): string {
84+
return parseCase(str, { escapedChars, separator: '-' });
85+
}

‎types/Parser.d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
import { PropType } from '@b613/utils/typings';
21
import { ParseResult } from '@babel/parser';
32
import { ImportResolver } from './ImportResolver';
43
import { File as FileSystemFile } from './FileSystem';
@@ -7,6 +6,8 @@ import { Entry } from './Entry';
76

87
import * as Babel from '@babel/types';
98

9+
type PropType<TObj, TProp extends keyof TObj> = TObj[TProp];
10+
1011
export namespace Parser {
1112
type FilenameOptions = {
1213
/**

0 commit comments

Comments
 (0)
Please sign in to comment.