Skip to content

Commit b3b45cc

Browse files
authored
feat: support ignore code (#71)
1 parent cb67ea7 commit b3b45cc

9 files changed

+145
-40
lines changed

README.md

+12
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,18 @@ Converts between `px` and `rem` units in VSCode, And support WXSS.
2727
+ Move the cursor to `14px`, press `Alt + z` to convert `rem`
2828
+ CLI: Press `F1`, enter `cssrem`
2929

30+
## Ignoring code
31+
32+
You can ignore:
33+
- Entire files by setting ignores in `.cssrem` or `.vscode/settings.json`
34+
- The next line by using `// cssrem-disable-next-line`
35+
36+
```less
37+
// cssrem-disable-next-line
38+
@media (min-width: 768px) {}
39+
```
40+
> The above code will ignore processing of `768px`
41+
3042
# Support Language
3143

3244
html vue css less scss sass stylus tpl(php smarty3) tsx jsx

README.zh-CN.md

+12
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,18 @@
2525
+ 光标至 `14px` 上,按下 `Alt + z` 对应转化 `rem`
2626
+ 通过 `F1` 面板查找:`cssrem`
2727

28+
## 忽略代码
29+
30+
你可以忽略:
31+
- 通过 `.cssrem``.vscode/settings.json` 设置 `ignores` 忽略整个文件
32+
- 通过使用 `// cssrem-disable-next-line` 表示忽略下一行的处理
33+
34+
```less
35+
// cssrem-disable-next-line
36+
@media (min-width: 768px) {}
37+
```
38+
> 以上代码将忽略 `768px` 的处理
39+
2840
# 支持语言
2941

3042
html vue css less scss sass stylus tpl(php smarty3) tsx jsx

examples/.cssrem

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
{
2+
"$schema": "https://raw.githubusercontent.com/cipchk/vscode-cssrem/master/schema.json",
3+
"rootFontSize": 24,
4+
"fixedDigits": 3,
5+
"vw": true,
6+
"currentLine": "show",
7+
"ignores": [
8+
// "**/*.less",
9+
// "*.tsx"
10+
],
11+
"ignoresViaCommand": []
12+
}

examples/demo.less

+11-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,14 @@
11
body {
2-
font-size: 1rem;
3-
padding: 10px 12px 1rem 1.2rem;
2+
font-size: 48px;
3+
padding: 10px 12px 24px 28.8px;
44
margin: 7.5px 1vw;
55
}
6+
7+
// cssrem-disable-next-line
8+
@media (min-width: 768px) {
9+
body {
10+
font-size: 28.8px;
11+
padding: 10px 12px 24px 28.8px;
12+
margin: 7.5px 1vw;
13+
}
14+
}

src/completion.ts

+2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { CompletionItem, CompletionItemKind, CompletionItemProvider, MarkdownString, Position, Range, TextDocument } from 'vscode';
22
import { cog, isIngore } from './config';
33
import { CssRemProcess } from './process';
4+
import { isDisabledNextLine } from './ignore-comment';
45

56
export default class implements CompletionItemProvider {
67
constructor(private _: string, private process: CssRemProcess) {}
@@ -9,6 +10,7 @@ export default class implements CompletionItemProvider {
910
if (isIngore(document.uri)) return Promise.resolve([]);
1011

1112
return new Promise(resolve => {
13+
if (isDisabledNextLine(document, position.line)) return null;
1214
const lineText = document.getText(new Range(position.with(undefined, 0), position));
1315
const res = this.process.convert(lineText);
1416
if (res == null || res.length === 0) {

src/hover.ts

+28-12
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,26 @@
1-
import { Hover, HoverProvider, MarkdownString, Position, ProviderResult, TextDocument } from 'vscode';
2-
import { cog, isIngore } from './config';
3-
import { RULES } from './rules';
1+
import {
2+
Hover,
3+
HoverProvider,
4+
MarkdownString,
5+
Position,
6+
ProviderResult,
7+
TextDocument,
8+
} from "vscode";
9+
import { cog, isIngore } from "./config";
10+
import { RULES } from "./rules";
11+
import { isDisabledNextLine } from "./ignore-comment";
412

513
export default class implements HoverProvider {
614
private getText(line: string, pos: Position): string {
715
const point = pos.character;
8-
let text = '';
16+
let text = "";
917
line.replace(/[.0-9]+(px|rem|rpx|vw)/g, (a, _, idx) => {
1018
const start = idx + 1;
1119
const end = idx + a.length + 1;
1220
if (!text && point >= start && point <= end) {
1321
text = a;
1422
}
15-
return '';
23+
return "";
1624
});
1725
return text;
1826
}
@@ -24,15 +32,23 @@ export default class implements HoverProvider {
2432
if (!text) {
2533
return null;
2634
}
27-
let results = RULES.filter(w => w.hover && w.hover.test(text) && w.hoverFn != null)
28-
.map(rule => rule.hoverFn!(text))
29-
.filter(h => h != null && h.documentation);
30-
if (cog.hover === 'onlyMark') {
31-
results = results.filter(w => !line.includes(`/* ${w.type} */`));
35+
if (isDisabledNextLine(doc, pos.line)) {
36+
return null;
37+
}
38+
let results = RULES.filter(
39+
(w) => w.hover && w.hover.test(text) && w.hoverFn != null
40+
)
41+
.map((rule) => rule.hoverFn!(text))
42+
.filter((h) => h != null && h.documentation);
43+
if (cog.hover === "onlyMark") {
44+
results = results.filter((w) => !line.includes(`/* ${w.type} */`));
3245
}
3346
if (results.length === 0) return null;
34-
if (results.length === 1) return new Hover(new MarkdownString(results[0].documentation));
47+
if (results.length === 1)
48+
return new Hover(new MarkdownString(results[0].documentation));
3549

36-
return new Hover(new MarkdownString(results.map(h => `- ${h.documentation}`).join('\n')));
50+
return new Hover(
51+
new MarkdownString(results.map((h) => `- ${h.documentation}`).join("\n"))
52+
);
3753
}
3854
}

src/ignore-comment.ts

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import { TextDocument } from 'vscode';
2+
3+
export const DISABLE_NEXT_LINE_COMMAND = 'cssrem-disable-next-line';
4+
5+
export function isDisabledNextLineViaText(line: string): boolean {
6+
return line.includes(DISABLE_NEXT_LINE_COMMAND);
7+
}
8+
9+
export function isDisabledNextLine(doc: TextDocument, lineNumber: number): boolean {
10+
if (lineNumber - 1 < 0) return false;
11+
const line = doc.lineAt(lineNumber - 1).text.trim();
12+
return isDisabledNextLineViaText(line);
13+
}

src/line-annotation.ts

+2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import {
1313
import { cog, isIngore } from './config';
1414
import { HoverResult } from './interface';
1515
import { RULES } from './rules';
16+
import { isDisabledNextLine } from './ignore-comment';
1617

1718
const annotationDecoration = window.createTextEditorDecorationType({
1819
after: {
@@ -93,6 +94,7 @@ export class LineAnnotation implements Disposable {
9394
private genMessage(doc: TextDocument, lineNumber: number): string | null {
9495
const text = doc.lineAt(lineNumber).text.trim();
9596
if (text.length <= 0) return null;
97+
if (isDisabledNextLine(doc, lineNumber)) return null;
9698
const values = text.match(/([.0-9]+(px|rem|rpx|vw))/g);
9799
if (values == null) return null;
98100

src/process.ts

+53-26
Original file line numberDiff line numberDiff line change
@@ -1,41 +1,55 @@
1-
import { ConvertResult, Rule, RuleOPType, Type } from './interface';
2-
import { RULES } from './rules';
3-
import { Position, Range, Selection, TextEditor } from 'vscode';
4-
import { isIngore } from './config';
1+
import { ConvertResult, Rule, RuleOPType, Type } from "./interface";
2+
import { RULES } from "./rules";
3+
import { Position, Range, Selection, TextEditor } from "vscode";
4+
import { isIngore } from "./config";
5+
import { isDisabledNextLineViaText } from "./ignore-comment";
56

67
export class CssRemProcess {
78
convert(text: string): ConvertResult[] | null {
8-
const res = this.getRule('single', text);
9+
const res = this.getRule("single", text);
910
if (res.length === 0) {
1011
return null;
1112
}
1213

13-
return res.map(i => i.rule.fn(i.text)!);
14+
return res.map((i) => i.rule.fn(i.text)!);
1415
}
1516

16-
convertAll(code: string, ignores: string[], type: Type): string {
17+
private convertAll(code: string, ignores: string[], type: Type): string {
1718
if (!code) {
1819
return code;
1920
}
2021

21-
const rule = RULES.find(w => w.type === type);
22+
const rule = RULES.find((w) => w.type === type);
2223
if (!rule) {
2324
return code;
2425
}
2526

26-
return code.replace(rule.all, (word: string) => {
27-
if (ignores.includes(word)) {
28-
return word;
29-
}
30-
const res = rule.fn(word);
31-
if (res) {
32-
return res.value;
33-
}
34-
return word;
35-
});
27+
const lines = code.split("\n");
28+
29+
const result = lines
30+
.map((line, lineIndex) => {
31+
return line.replace(rule.all, (word) => {
32+
if (ignores.includes(word)) {
33+
return word;
34+
}
35+
if (lineIndex > 0 && isDisabledNextLineViaText(lines[lineIndex - 1]))
36+
return word;
37+
38+
const res = rule.fn(word);
39+
if (res) {
40+
return res.value;
41+
}
42+
return word;
43+
});
44+
})
45+
.join("\n");
46+
return result;
3647
}
3748

38-
private getRule(type: RuleOPType, text: string): { rule: Rule; text: string }[] {
49+
private getRule(
50+
type: RuleOPType,
51+
text: string
52+
): { rule: Rule; text: string }[] {
3953
const result: { rule: Rule; text: string }[] = [];
4054
for (const rule of RULES) {
4155
const match = text.match((rule as any)[type]);
@@ -47,36 +61,49 @@ export class CssRemProcess {
4761
}
4862

4963
private getWordRange(textEditor: TextEditor, type: Type): Range | null {
50-
const position = new Position(textEditor.selection.start.line, textEditor.selection.start.character);
64+
const position = new Position(
65+
textEditor.selection.start.line,
66+
textEditor.selection.start.character
67+
);
5168
const range = textEditor.document.getWordRangeAtPosition(position);
5269
if (!range) return null;
5370
const word = textEditor.document.getText(range);
5471
if (!word) return null;
55-
const rule = RULES.find(w => w.type === type);
72+
const rule = RULES.find((w) => w.type === type);
5673
return rule && rule.all.test(word) ? range : null;
5774
}
5875

59-
modifyDocument(textEditor: TextEditor, ignoresViaCommand: string[], type: Type): void {
76+
modifyDocument(
77+
textEditor: TextEditor,
78+
ignoresViaCommand: string[],
79+
type: Type
80+
): void {
6081
const doc = textEditor.document;
6182
if (isIngore(doc.uri)) return;
6283

6384
let selection: Selection | Range = textEditor.selection;
6485
// When the cursor is in the valid range in switch mode
65-
if (selection.isEmpty && type.toLowerCase().includes('switch')) {
86+
if (selection.isEmpty && type.toLowerCase().includes("switch")) {
6687
const wordRange = this.getWordRange(textEditor, type);
6788
if (wordRange) {
6889
selection = wordRange;
6990
}
7091
}
7192
if (selection.isEmpty) {
7293
const start = new Position(0, 0);
73-
const end = new Position(doc.lineCount - 1, doc.lineAt(doc.lineCount - 1).text.length);
94+
const end = new Position(
95+
doc.lineCount - 1,
96+
doc.lineAt(doc.lineCount - 1).text.length
97+
);
7498
selection = new Range(start, end);
7599
}
76100

77101
const text = doc.getText(selection);
78-
textEditor.edit(builder => {
79-
builder.replace(selection, this.convertAll(text, ignoresViaCommand, type));
102+
textEditor.edit((builder) => {
103+
builder.replace(
104+
selection,
105+
this.convertAll(text, ignoresViaCommand, type)
106+
);
80107
});
81108
}
82109
}

0 commit comments

Comments
 (0)