├── .npmrc
├── .eslintignore
├── versions.json
├── .gitattributes
├── screenshots
├── image-20220601202203.png
├── image20220606011534.png
└── image-20220402200431096.png
├── .editorconfig
├── manifest.json
├── .gitignore
├── tsconfig.json
├── version-bump.mjs
├── .eslintrc
├── package.json
├── LICENSE
├── esbuild.config.mjs
├── README.md
├── styles.css
├── main.ts
└── main.js
/.npmrc:
--------------------------------------------------------------------------------
1 | tag-version-prefix=""
--------------------------------------------------------------------------------
/.eslintignore:
--------------------------------------------------------------------------------
1 | npm node_modules
2 | build
--------------------------------------------------------------------------------
/versions.json:
--------------------------------------------------------------------------------
1 | {
2 | "1.0.0": "0.9.7",
3 | "1.0.1": "0.12.0"
4 | }
5 |
--------------------------------------------------------------------------------
/.gitattributes:
--------------------------------------------------------------------------------
1 | # Auto detect text files and perform LF normalization
2 | * text=auto
3 |
--------------------------------------------------------------------------------
/screenshots/image-20220601202203.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stargrey/obsidian-better-codeblock/HEAD/screenshots/image-20220601202203.png
--------------------------------------------------------------------------------
/screenshots/image20220606011534.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stargrey/obsidian-better-codeblock/HEAD/screenshots/image20220606011534.png
--------------------------------------------------------------------------------
/screenshots/image-20220402200431096.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stargrey/obsidian-better-codeblock/HEAD/screenshots/image-20220402200431096.png
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | # top-most EditorConfig file
2 | root = true
3 |
4 | [*]
5 | charset = utf-8
6 | insert_final_newline = true
7 | indent_style = tab
8 | indent_size = 4
9 | tab_width = 4
10 |
--------------------------------------------------------------------------------
/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "id": "obsidian-better-codeblock",
3 | "name": "Better CodeBlock",
4 | "version": "1.0.8",
5 | "minAppVersion": "0.12.0",
6 | "description": "Add title, line number to Obsidian code block",
7 | "author": "StarGrey",
8 | "authorUrl": "https://github.com/stargrey",
9 | "isDesktopOnly": false
10 | }
11 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # vscode
2 | .vscode
3 |
4 | # Intellij
5 | *.iml
6 | .idea
7 |
8 | # npm
9 | node_modules
10 |
11 | # Don't include the compiled main.js file in the repo.
12 | # They should be uploaded to GitHub releases instead.
13 | main.js
14 |
15 | # Exclude sourcemaps
16 | *.map
17 |
18 | # obsidian
19 | data.json
20 |
21 | # Exclude macOS Finder (System Explorer) View States
22 | .DS_Store
23 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "baseUrl": ".",
4 | "inlineSourceMap": true,
5 | "inlineSources": true,
6 | "module": "ESNext",
7 | "target": "ES6",
8 | "allowJs": true,
9 | "noImplicitAny": true,
10 | "moduleResolution": "node",
11 | "importHelpers": true,
12 | "isolatedModules": true,
13 | "lib": [
14 | "DOM",
15 | "ES5",
16 | "ES6",
17 | "ES7"
18 | ]
19 | },
20 | "include": [
21 | "**/*.ts"
22 | ]
23 | }
24 |
--------------------------------------------------------------------------------
/version-bump.mjs:
--------------------------------------------------------------------------------
1 | import { readFileSync, writeFileSync } from "fs";
2 |
3 | const targetVersion = process.env.npm_package_version;
4 |
5 | // read minAppVersion from manifest.json and bump version to target version
6 | let manifest = JSON.parse(readFileSync("manifest.json", "utf8"));
7 | const { minAppVersion } = manifest;
8 | manifest.version = targetVersion;
9 | writeFileSync("manifest.json", JSON.stringify(manifest, null, "\t"));
10 |
11 | // update versions.json with target version and minAppVersion from manifest.json
12 | let versions = JSON.parse(readFileSync("versions.json", "utf8"));
13 | versions[targetVersion] = minAppVersion;
14 | writeFileSync("versions.json", JSON.stringify(versions, null, "\t"));
15 |
--------------------------------------------------------------------------------
/.eslintrc:
--------------------------------------------------------------------------------
1 | {
2 | "root": true,
3 | "parser": "@typescript-eslint/parser",
4 | "env": { "node": true },
5 | "plugins": [
6 | "@typescript-eslint"
7 | ],
8 | "extends": [
9 | "eslint:recommended",
10 | "plugin:@typescript-eslint/eslint-recommended",
11 | "plugin:@typescript-eslint/recommended"
12 | ],
13 | "parserOptions": {
14 | "sourceType": "module"
15 | },
16 | "rules": {
17 | "no-unused-vars": "off",
18 | "@typescript-eslint/no-unused-vars": ["error", { "args": "none" }],
19 | "@typescript-eslint/ban-ts-comment": "off",
20 | "no-prototype-builtins": "off",
21 | "@typescript-eslint/no-empty-function": "off"
22 | }
23 | }
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "obsidian-sample-plugin",
3 | "version": "1.0.1",
4 | "description": "This is a sample plugin for Obsidian (https://obsidian.md)",
5 | "main": "main.js",
6 | "scripts": {
7 | "dev": "node esbuild.config.mjs",
8 | "build": "tsc -noEmit -skipLibCheck && node esbuild.config.mjs production",
9 | "version": "node version-bump.mjs && git add manifest.json versions.json"
10 | },
11 | "keywords": [],
12 | "author": "",
13 | "license": "MIT",
14 | "devDependencies": {
15 | "@types/node": "^16.11.6",
16 | "@typescript-eslint/eslint-plugin": "^5.2.0",
17 | "@typescript-eslint/parser": "^5.2.0",
18 | "builtin-modules": "^3.2.0",
19 | "esbuild": "0.13.12",
20 | "obsidian": "latest",
21 | "tslib": "2.3.1",
22 | "typescript": "4.4.4"
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2022 stargrey
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/esbuild.config.mjs:
--------------------------------------------------------------------------------
1 | import esbuild from "esbuild";
2 | import process from "process";
3 | import builtins from 'builtin-modules'
4 |
5 | const banner =
6 | `/*
7 | THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
8 | if you want to view the source, please visit the github repository of this plugin
9 | */
10 | `;
11 |
12 | const prod = (process.argv[2] === 'production');
13 |
14 | esbuild.build({
15 | banner: {
16 | js: banner,
17 | },
18 | entryPoints: ['main.ts'],
19 | bundle: true,
20 | external: [
21 | 'obsidian',
22 | 'electron',
23 | '@codemirror/autocomplete',
24 | '@codemirror/closebrackets',
25 | '@codemirror/collab',
26 | '@codemirror/commands',
27 | '@codemirror/comment',
28 | '@codemirror/fold',
29 | '@codemirror/gutter',
30 | '@codemirror/highlight',
31 | '@codemirror/history',
32 | '@codemirror/language',
33 | '@codemirror/lint',
34 | '@codemirror/matchbrackets',
35 | '@codemirror/panel',
36 | '@codemirror/rangeset',
37 | '@codemirror/rectangular-selection',
38 | '@codemirror/search',
39 | '@codemirror/state',
40 | '@codemirror/stream-parser',
41 | '@codemirror/text',
42 | '@codemirror/tooltip',
43 | '@codemirror/view',
44 | ...builtins],
45 | format: 'cjs',
46 | watch: !prod,
47 | target: 'es2016',
48 | logLevel: "info",
49 | sourcemap: prod ? false : 'inline',
50 | treeShaking: true,
51 | outfile: 'main.js',
52 | }).catch(() => process.exit(1));
53 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## Obsidian Better Code Block
2 |
3 | This is a plugin for Obsidian (https://obsidian.md).
4 |
5 | Most of the code in this plugin comes from the following two plugins (thanks to their contributions), and the icons are from Admonition.
6 |
7 | https://github.com/tadashi-aikawa/obsidian-embedded-code-title
8 |
9 | https://github.com/nyable/obsidian-code-block-enhancer
10 |
11 | I have merged the code in both plugins and modified some of their functionality.
12 |
13 | ### Features
14 | Enhancer the markdown code block in preview mode. Add title, line number, highlight to code blocks, you can click on the title to collapse or expand the block.
15 |
16 | In version 1.0.5, use the syntax in the diagram below to set the block title, highlight, fold
17 |
18 | - Use `TI:"your title"` to add title
19 | - Use `HL:"numbers"` to add highlight, such as `HL:"1,2,3"`, `HL:"1-3"`, separate by `,`
20 | - Use `"FOLD"` to set the default fold
21 |
22 | If you have a better idea, please submit an issue
23 |
24 | 
25 |
26 | In version 1.0.4, add the language in the top right, like this:
27 | 
28 | ### Known issues
29 | - Sometimes the auto linefeed error, can be solved by switching the preview mode once
30 | - The PDF export cannot be auto linefeed
31 | ### Manually installing the plugin
32 |
33 | - Copy over `main.js`, `styles.css`, `manifest.json` to your vault `VaultFolder/.obsidian/plugins/obsidian-better-codeblock/`.
34 |
--------------------------------------------------------------------------------
/styles.css:
--------------------------------------------------------------------------------
1 |
2 | .obsidian-embedded-code-title__code-block-title {
3 | position: absolute !important;
4 | top: 0;
5 | left: 0;
6 | width: 100%;
7 | /* font-size: 85%!important; */
8 | padding: 3px !important;
9 | padding-left: 15px !important;
10 | margin: 0 !important;
11 | border-radius: 0 !important;
12 | }
13 |
14 | .copy-code-button{
15 | margin-top: 42px !important; /* 为自带的按钮增加上边距 */
16 | }
17 |
18 |
19 | pre[class*=language-] {
20 | font-size: var(--editor-font-size);
21 | line-height: 1.5em;
22 | padding-bottom: 0px;
23 | }
24 | .obsidian-embedded-code-title__code-block-title + code[class*=language-]{
25 | padding: 0em 0em 0em 0em !important;
26 | /* padding-top: 0 !important; */
27 | font-size: var(--editor-font-size) !important;
28 | line-height: 1.5em !important;
29 | }
30 | /* pre[class*=language-] > code[class*=language-] {
31 | padding: 0em 0em 0em 0.5em !important;
32 | /* padding-top: 0 !important; */
33 | /* font-size: var(--editor-font-size) !important;
34 | line-height: 1.5em !important;
35 | } */
36 |
37 | pre[class*=language-].code-block-pre__has-linenum {
38 | padding-left: 3.5em;
39 | }
40 |
41 | .code-block-pre__has-linenum::before {
42 | padding-top: 6px;
43 | }
44 |
45 | /* 代码行号 */
46 | .code-block-linenum-wrap {
47 | position: absolute;
48 | /* top: 35px; */
49 | left: 0px;
50 | min-width: 3em;
51 | font-size: var(--editor-font-size);
52 | line-height: 1.5em;
53 | counter-reset: line-num;
54 | text-align: center;
55 | /* border-right: #999 2px solid; 行号与代码间分隔线 */
56 | user-select: none;
57 | pointer-events: none;
58 | background-color: transparent;
59 | /* background-color: inherit; */
60 | }
61 | .code-block-linenum-wrap .code-block-linenum {
62 | display: block;
63 | counter-increment: line-num;
64 | pointer-events: none;
65 | }
66 | .code-block-linenum-wrap .code-block-linenum::before {
67 | content: counter(line-num);
68 | }
69 |
70 | /* 代码高亮 */
71 | pre[class*=language-] .code-block-highlight-wrap {
72 | margin: 0;
73 | padding: 0;
74 | position: absolute;
75 | left: 0px;
76 | top: 35px;
77 | width: 100%;
78 | height: 100%;
79 | background-color: transparent;
80 | pointer-events: none;
81 | }
82 |
83 | pre[class*=language-] .code-block-highlight-wrap span {
84 | display: block;
85 | height: 1.5em;
86 | width: 100%;
87 | }
88 |
89 | /* 折叠代码块 */
90 |
91 | :root {
92 | --admonition-details-icon: url("data:image/svg+xml;charset=utf-8,");
93 | }
94 | .obsidian-embedded-code-title__code-block-title{
95 | line-height: 35px;
96 | height: 35px !important;
97 | color: currentColor !important;
98 | }
99 |
100 | .obsidian-embedded-code-title__code-block-title .langName {
101 | display: inline;
102 | float: right;
103 | line-height: 29px;
104 | margin-right: 35px;
105 | font-weight: bold;
106 | font-size: 14px;
107 | font-family: var(--font-default);
108 | }
109 |
110 | .obsidian-embedded-code-title__code-block-title .collapser {
111 | position: absolute;
112 | top: 50%;
113 | right: 8px;
114 | transform: translateY(-50%);
115 | content: "";
116 | }
117 | .obsidian-embedded-code-title__code-block-title .collapser .handle {
118 | transform: rotate(90deg);
119 | transition: transform 0.25s;
120 | background-color: currentColor;
121 | -webkit-mask-repeat: no-repeat;
122 | mask-repeat: no-repeat;
123 | -webkit-mask-size: contain;
124 | mask-size: contain;
125 | -webkit-mask-image: var(--admonition-details-icon);
126 | mask-image: var(--admonition-details-icon);
127 | width: 20px;
128 | height: 20px;
129 | }
130 | .obsidian-embedded-code-title__code-block-title[closed] .collapser .handle{
131 | transform: rotate(0deg);
132 | }
133 | .obsidian-embedded-code-title__code-block-title[closed] + code{
134 | height: 0;
135 | }
136 | .obsidian-embedded-code-title__code-block-title[closed] + code + span{
137 | height: 0;
138 | }
139 |
140 | .obsidian-embedded-code-title__code-block-title[closed] + code + span span{
141 | visibility: hidden;
142 | }
143 |
144 | .obsidian-embedded-code-title__code-block-title[closed] + code + span + span span{
145 | visibility: hidden;
146 | }
147 |
148 | .obsidian-embedded-code-title__code-block-title > .title {
149 | display: inline-block;
150 | position: relative;
151 | margin-left: 5px !important;
152 | margin: 0;
153 | padding: 0;
154 |
155 | top: 50%;
156 | transform: translateY(-50%);
157 | }
158 |
159 | /* .obsidian-embedded-code-title__code-block-title > .icon-wrap {
160 | display: inline-block;
161 | position: relative;
162 | width: 20px;
163 | height: 20px;
164 | background-position: center;
165 |
166 | top: 50%;
167 | transform: translateY(-50%);
168 | } */
169 |
170 | .code-block-wrap > pre > code[class*=language-]{
171 | padding: 0em 0em 0em 0em !important;
172 | /* padding-top: 0 !important; */
173 | font-size: var(--editor-font-size) !important;
174 | line-height: 1.5em !important;
175 | }
--------------------------------------------------------------------------------
/main.ts:
--------------------------------------------------------------------------------
1 | import { linkSync } from 'fs';
2 | import { App, Editor, MarkdownView, Modal, Notice, Plugin, PluginSettingTab, Setting, MarkdownPostProcessorContext, Menu, SettingTab, TAbstractFile, TFile, SectionCache, Vault } from 'obsidian';
3 | import { json } from 'stream/consumers';
4 |
5 | const DEFAULT_LANG_ATTR = 'language-text'
6 | const DEFAULT_LANG = ''
7 | const LANG_REG = /^language-/
8 | const LINE_SPLIT_MARK = '\n'
9 |
10 | const titleRegExp = /TI:"([^"]*)"/i
11 | const highLightLinesRegExp = /HL:"([^"]*)"/i
12 | const foldRegExp = /"FOLD"/i
13 |
14 | const CB_PADDING_TOP = "35px" // 代码块上边距
15 |
16 | interface Settings {
17 | substitutionTokenForSpace: string;
18 | titleBackgroundColor: string;
19 | titleFontColor: string;
20 | highLightColor: string;
21 |
22 | excludeLangs: string[]; // 需要排除的语言
23 |
24 | showLineNumber: boolean; // 显示行号
25 | showDividingLine: boolean;
26 | showLangNameInTopRight: boolean;
27 | }
28 |
29 | const DEFAULT_SETTINGS: Settings = {
30 | substitutionTokenForSpace: undefined,
31 | titleBackgroundColor: "#00000020",
32 | titleFontColor: undefined,
33 | highLightColor: "#2d82cc20",
34 |
35 | excludeLangs: [],
36 |
37 | showLineNumber: true,
38 | showDividingLine: false,
39 | showLangNameInTopRight: true
40 | };
41 |
42 | interface CodeBlockMeta {
43 | // Language name
44 | langName: string;
45 |
46 | // Code block total line size
47 | lineSize: number;
48 |
49 | // Code block 'pre' HTMLElement
50 | pre: HTMLElement;
51 |
52 | // Code block 'code' HTMLElement
53 | code: HTMLElement;
54 |
55 | title: string; // 代码块标题
56 | isCollapse:boolean; // 是否默认折叠
57 |
58 | // Code block wrap div
59 | div: HTMLElement;
60 | contentList: string[];
61 | highLightLines: number[];
62 | }
63 |
64 | // Refer https://developer.mozilla.org/ja/docs/Web/JavaScript/Guide/Regular_Expressions#escaping
65 | function escapeRegExp(str: string): string {
66 | return str.replace(/[.*+?^=!:${}()|[\]\/\\]/g, "\\$&"); // 为特殊符号加上转义符号"\"
67 | }
68 |
69 | export default class BetterCodeBlock extends Plugin {
70 | settings: Settings;
71 |
72 | async onload() {
73 | console.log("Loading Better Code Block Plugin");
74 | await this.loadSettings();
75 | this.addSettingTab(new BetterCodeBlockTab(this.app, this));
76 | this.registerMarkdownPostProcessor((el, ctx) => {
77 | BetterCodeBlocks(el, ctx, this)
78 | app.workspace.on('resize', () => {
79 | resizeNumWrapAndHLWrap(el, ctx)
80 | })
81 | })
82 |
83 | }
84 |
85 | onunload () {
86 | console.log('Unloading Better Code Block Plugin');
87 | }
88 |
89 | async loadSettings() {
90 | this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());
91 | }
92 |
93 | async saveSettings() {
94 | await this.saveData(this.settings);
95 | }
96 | }
97 |
98 | class BetterCodeBlockTab extends PluginSettingTab {
99 | plugin: BetterCodeBlock;
100 |
101 | constructor(app: App, plugin: BetterCodeBlock) {
102 | super(app, plugin);
103 | this.plugin = plugin;
104 | }
105 |
106 | display(): void {
107 | let { containerEl } = this;
108 |
109 | containerEl.empty();
110 |
111 | new Setting(containerEl)
112 | .setName("Exclude language list")
113 | .setDesc("Title and line numbers do not apply in these languages, separate by `,`")
114 | .addText(text => text.setPlaceholder('like todoist,other,...')
115 | .setValue(this.plugin.settings.excludeLangs.join(','))
116 | .onChange(async (value) => {
117 | this.plugin.settings.excludeLangs = value.split(',');
118 | await this.plugin.saveSettings();
119 | })
120 | )
121 |
122 | new Setting(containerEl).setName("Font color of title").addText((tc) =>
123 | tc
124 | .setPlaceholder("Enter a color")
125 | .setValue(this.plugin.settings.titleFontColor)
126 | .onChange(async (value) => {
127 | this.plugin.settings.titleFontColor = value;
128 | await this.plugin.saveSettings();
129 | })
130 | );
131 |
132 | new Setting(containerEl)
133 | .setName("Background color of title")
134 | .addText((tc) =>
135 | tc
136 | .setPlaceholder("#00000020")
137 | .setValue(this.plugin.settings.titleBackgroundColor)
138 | .onChange(async (value) => {
139 | this.plugin.settings.titleBackgroundColor = value;
140 | await this.plugin.saveSettings();
141 | })
142 | );
143 |
144 | new Setting(containerEl)
145 | .setName("HighLight Color")
146 | .addText((tc) =>
147 | tc
148 | .setPlaceholder("#2d82cc20")
149 | .setValue(this.plugin.settings.highLightColor)
150 | .onChange(async (value) => {
151 | this.plugin.settings.highLightColor = value;
152 | await this.plugin.saveSettings();
153 | })
154 | );
155 |
156 | new Setting(containerEl)
157 | .setName("Show line number")
158 | .addToggle((tc) =>
159 | tc.setValue(this.plugin.settings.showLineNumber)
160 | .onChange(async(value) => {
161 | this.plugin.settings.showLineNumber = value;
162 | await this.plugin.saveSettings();
163 | })
164 | )
165 |
166 | new Setting(containerEl)
167 | .setName("Show dividing line")
168 | .addToggle((tc) =>
169 | tc.setValue(this.plugin.settings.showDividingLine)
170 | .onChange(async(value) => {
171 | this.plugin.settings.showDividingLine = value;
172 | await this.plugin.saveSettings();
173 | })
174 | )
175 |
176 | new Setting(containerEl)
177 | .setName("Show language name in the top right")
178 | .addToggle((tc) =>
179 | tc.setValue(this.plugin.settings.showLangNameInTopRight)
180 | .onChange(async(value) => {
181 | this.plugin.settings.showLangNameInTopRight = value;
182 | await this.plugin.saveSettings();
183 | })
184 | )
185 | }
186 | }
187 |
188 |
189 | export async function BetterCodeBlocks(el: HTMLElement, context: MarkdownPostProcessorContext, plugin: BetterCodeBlock) {
190 | const settings = plugin.settings
191 | const codeElm: HTMLElement = el.querySelector('pre > code')
192 | // only change pre>code
193 | if (!codeElm) { return }
194 |
195 | let lang = DEFAULT_LANG
196 | // return when lang is in exclude list
197 | if (plugin.settings.excludeLangs.some(eLangName => codeElm.classList.contains(`language-${eLangName}`))) {
198 | return
199 | }
200 |
201 | codeElm.classList.forEach((value, key, parent) => {
202 | if (LANG_REG.test(value)) {
203 | lang = value.replace('language-', '')
204 | return
205 | }
206 | })
207 |
208 | // if the code block is not described, return
209 | if(lang == DEFAULT_LANG) {
210 | return
211 | }
212 |
213 | let codeBlock = context.getSectionInfo(codeElm)
214 | let codeBlockFirstLine = ""
215 |
216 | if(codeBlock) {
217 | let view = app.workspace.getActiveViewOfType(MarkdownView)
218 | codeBlockFirstLine = view.editor.getLine(codeBlock.lineStart)
219 | } else {
220 | let file = app.vault.getAbstractFileByPath(context.sourcePath)
221 | let cache = app.metadataCache.getCache(context.sourcePath)
222 | let fileContent = await app.vault.cachedRead( file)
223 | let fileContentLines = fileContent.split(/\n/g)
224 |
225 | let codeBlockFirstLines: string[] = []
226 | let codeBlockSections: SectionCache[] = []
227 |
228 | cache.sections?.forEach(async element => {
229 | if(element.type == "code") {
230 | let lineStart = element.position.start.line
231 | codeBlockFirstLine = fileContentLines[lineStart]
232 | codeBlockSections.push(element)
233 | codeBlockFirstLines.push(codeBlockFirstLine)
234 | }
235 | });
236 | exportPDF(el, plugin, codeBlockFirstLines, codeBlockSections)
237 | return
238 | }
239 |
240 | let title: string = ""
241 | let highLightLines: number[] = []
242 | if(codeBlockFirstLine.match(titleRegExp) != null) {
243 | title = codeBlockFirstLine.match(titleRegExp)[1]
244 | }
245 | if(codeBlockFirstLine.match(highLightLinesRegExp) != null) {
246 | let highLightLinesInfo = codeBlockFirstLine.match(highLightLinesRegExp)[1]
247 | highLightLines = analyseHighLightLines(highLightLinesInfo)
248 | }
249 |
250 | let isCollapse = false;
251 | if(foldRegExp.test(codeBlockFirstLine)) {
252 | isCollapse = true
253 | }
254 |
255 | const pre = codeElm.parentElement // code-block-pre__has-linenum
256 | const div = pre.parentElement // class code-block-wrap
257 |
258 | /* const { lineStart, lineEnd } = ctx.getSectionInfo(el)
259 | const lineSize = lineEnd - lineStart - 1 */
260 | const contentList: string[] = codeElm.textContent.split(LINE_SPLIT_MARK)
261 | // const lineSize = contentList.length - 1
262 | const lineSize = codeBlock.lineEnd - codeBlock.lineStart - 1
263 |
264 | const cbMeta = { langName: lang, lineSize, pre, code: codeElm, title, isCollapse, div, contentList, highLightLines}
265 |
266 | const {showLineNumber} = plugin.settings
267 |
268 | addCodeTitleWrapper(plugin, pre, cbMeta)
269 | //addIconToTitle(plugin, pre, cbMeta)
270 | addCodeTitle(plugin, pre, cbMeta);
271 |
272 | // add line number
273 | if (showLineNumber) {
274 | addLineNumber(plugin, cbMeta)
275 | }
276 |
277 | addLineHighLight(plugin, pre, cbMeta)
278 |
279 | resizeNumWrapAndHLWrap(el,context) // 调用一次以解决某些时候打开文件行高未被重设高度
280 | }
281 |
282 | function createElement (tagName: string, defaultClassName?: string) {
283 | const element = document.createElement(tagName)
284 | if (defaultClassName) {
285 | element.className = defaultClassName
286 | }
287 | return element
288 | }
289 |
290 | function addCodeTitleWrapper(plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {
291 | preElm.style.setProperty("position", "relative", "important");
292 | preElm.style.setProperty("padding-top", CB_PADDING_TOP, "important");
293 |
294 | let wrapper = document.createElement("pre")
295 | if(cbMeta.isCollapse) {
296 | wrapper.setAttribute("closed","")
297 | }
298 | wrapper.className = "obsidian-embedded-code-title__code-block-title"
299 |
300 | wrapper.style.backgroundColor = plugin.settings.titleBackgroundColor || "#00000020";
301 |
302 | let collapser = createElement("div","collapser")
303 | let handle = createElement("div", "handle")
304 | collapser.appendChild(handle)
305 | wrapper.appendChild(collapser)
306 |
307 | wrapper.addEventListener('click',function(this: any) {
308 | if(wrapper.hasAttribute("closed")){
309 | wrapper.removeAttribute("closed")
310 | } else {
311 | wrapper.setAttribute("closed",'')
312 | }
313 | })
314 |
315 | preElm.appendChild(wrapper)
316 | }
317 |
318 | function addCodeTitle (plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {
319 | let wrapper = preElm.querySelector(".obsidian-embedded-code-title__code-block-title")
320 |
321 | let titleElm = document.createElement("div")
322 | titleElm.className = "title"
323 |
324 | titleElm.appendText(cbMeta.title)
325 | wrapper.appendChild(titleElm)
326 |
327 | if(plugin.settings.titleFontColor) {
328 | titleElm.style.setProperty("color", plugin.settings.titleFontColor, "important")
329 | }
330 |
331 | if(plugin.settings.showLangNameInTopRight) {
332 | let langName = document.createElement("div"); // 在右侧添加代码类型
333 | let langNameString = cbMeta.langName
334 | langNameString = langNameString[0].toUpperCase() + langNameString.slice(1) // 首字母大写
335 | langName.appendText(langNameString);
336 | langName.className = "langName";
337 | wrapper.appendChild(langName);
338 | }
339 |
340 | preElm.prepend(wrapper);
341 |
342 | }
343 |
344 | function addLineNumber (plugin: BetterCodeBlock, cbMeta: CodeBlockMeta) {
345 | const { lineSize, pre, div } = cbMeta
346 | // let div position: relative;
347 | div.classList.add('code-block-wrap')
348 |
349 | // const { fontSize, lineHeight } = window.getComputedStyle(cbMeta.code)
350 | const lineNumber = createElement('span', 'code-block-linenum-wrap')
351 | lineNumber.style.top = CB_PADDING_TOP;
352 | Array.from({ length: lineSize }, (v, k) => k).forEach(i => {
353 | const singleLine = createElement('span', 'code-block-linenum')
354 | // singleLine.style.fontSize = fontSize
355 | // singleLine.style.lineHeight = lineHeight
356 | lineNumber.appendChild(singleLine)
357 | })
358 |
359 | if(plugin.settings.showDividingLine) {
360 | lineNumber.style.borderRight = "1px currentColor solid"
361 | }
362 |
363 | pre.appendChild(lineNumber)
364 | pre.classList.add('code-block-pre__has-linenum')
365 | }
366 |
367 | function addLineHighLight(plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {
368 | if(cbMeta.highLightLines.length == 0) return
369 |
370 | let highLightWrap = document.createElement("pre")
371 | highLightWrap.className = "code-block-highlight-wrap"
372 | for(let i = 0; i < cbMeta.lineSize; i++) {
373 | const singleLine = createElement("span", 'code-block-highlight')
374 | if(cbMeta.highLightLines.contains(i+1)) {
375 | singleLine.style.backgroundColor = plugin.settings.highLightColor || "#2d82cc20"
376 | }
377 | highLightWrap.appendChild(singleLine)
378 | }
379 |
380 | preElm.appendChild(highLightWrap)
381 | }
382 |
383 | function analyseHighLightLines(str: string): number[] {
384 | str = str.replace(/\s*/g, "") // 去除字符串中所有空格
385 | const result: number[] = []
386 |
387 | let strs = str.split(",")
388 | strs.forEach(it => {
389 | if(/\w+-\w+/.test(it)) { // 如果匹配 1-3 这样的格式,依次添加数字
390 | let left = Number(it.split('-')[0])
391 | let right = Number(it.split('-')[1])
392 | for(let i = left; i <= right; i++) {
393 | result.push(i)
394 | }
395 | } else {
396 | result.push(Number(it))
397 | }
398 | })
399 |
400 | return result
401 | }
402 |
403 | function addIconToTitle(plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {
404 | let title = preElm.querySelectorAll(".obsidian-embedded-code-title__code-block-title")
405 |
406 | title.forEach(it => {
407 | let iconWrap = createElement("div","icon-wrap")
408 | let icon = document.createElement("img")
409 | icon.src = ""
410 | iconWrap.appendChild(icon)
411 | it.appendChild(iconWrap)
412 | })
413 |
414 | }
415 |
416 | // 在自动换行时对数字和高亮行重新设置高度
417 | // These codes refer to the https://github.com/lijyze/obsidian-advanced-codeblock
418 | function resizeNumWrapAndHLWrap(el: HTMLElement, context: MarkdownPostProcessorContext) {
419 | setTimeout(async function(){ // 延时100毫秒以解决某些时候打开文件行高未被重设高度
420 | // console.log('on resize')
421 | let codeBlockEl : HTMLElement = el.querySelector('pre > code')
422 | if(!codeBlockEl) return
423 |
424 | let numWrap = el.querySelector('.code-block-linenum-wrap')
425 | let highWrap = el.querySelector('.code-block-highlight-wrap')
426 |
427 | let codeBlockInfo = context.getSectionInfo(codeBlockEl)
428 | // let view = app.workspace.getActiveViewOfType(MarkdownView)
429 | // let codeBlockLineNum = codeBlockInfo.lineEnd - codeBlockInfo.lineStart - 1 // 除去首尾两行
430 | let view
431 | let codeBlockLineNum
432 |
433 | let lineStart = 0
434 | let lineEnd = 0
435 | if(codeBlockInfo) {
436 | view = app.workspace.getActiveViewOfType(MarkdownView)
437 | codeBlockLineNum = codeBlockInfo.lineEnd - codeBlockInfo.lineStart - 1 // 除去首尾两行
438 | } else {
439 | return
440 | // let file = app.vault.getAbstractFileByPath(context.sourcePath)
441 | // let cache = app.metadataCache.getCache(context.sourcePath)
442 |
443 | // cache.sections?.forEach(async element => {
444 | // if(element.type == "code") {
445 | // lineStart = element.position.start.line
446 | // lineEnd = element.position.end.line
447 | // codeBlockLineNum = lineEnd - lineStart - 1
448 | // return
449 | // }
450 | // });
451 | // let file = app.vault.getAbstractFileByPath(context.sourcePath)
452 | // let cache = app.metadataCache.getCache(context.sourcePath)
453 | // let fileContent = await app.vault.cachedRead( file)
454 | // let fileContentLines = fileContent.split(/\n/g)
455 | }
456 |
457 | let span = createElement("span")
458 |
459 | for(let i = 0; i < codeBlockLineNum; i++) {
460 | let oneLineText
461 | if(view){
462 | oneLineText = view.editor.getLine(codeBlockInfo.lineStart + i + 1)
463 | } else {
464 | // oneLineText = fileContentLines[lineStart + 1 + i]
465 | // let file = app.vault.getAbstractFileByPath(context.sourcePath)
466 | // let cache = app.metadataCache.getCache(context.sourcePath)
467 | // let fileContent = await app.vault.cachedRead( file)
468 | // let fileContentLines = fileContent.split(/\n/g)
469 | // oneLineText = fileContentLines[cache.sections]
470 | }
471 | span.innerHTML = oneLineText || "0"
472 |
473 | codeBlockEl.appendChild(span)
474 | span.style.display = 'block'
475 |
476 | let lineHeight = span.getBoundingClientRect().height + 'px' // 测量本行文字的高度
477 |
478 | // console.log(lineHeight + ' ' + span.getBoundingClientRect().width);
479 |
480 | let numOneLine = numWrap? numWrap.childNodes[i] as HTMLElement : null
481 | let hlOneLine = highWrap? highWrap.childNodes[i] as HTMLElement : null
482 |
483 | if(numOneLine) numOneLine.style.height = lineHeight;
484 | if(hlOneLine) hlOneLine.style.height = lineHeight;
485 |
486 | span.remove() // 测量完后删掉
487 | }
488 | }, 100)
489 | }
490 |
491 | function exportPDF(el: HTMLElement, plugin: BetterCodeBlock, codeBlockFirstLines: string[], codeBlockSections: SectionCache[]) {
492 | let codeBlocks = el.querySelectorAll('pre > code')
493 | codeBlocks.forEach((codeElm, key) => {
494 | let langName = "", title = "", highLightLines: number[] = []
495 | codeElm.classList.forEach(value => {
496 | if(LANG_REG.test(value)) {
497 | langName = value.replace('language-', '')
498 | return
499 | }
500 | })
501 |
502 | if(codeBlockFirstLines[key].match(titleRegExp) != null) {
503 | title = codeBlockFirstLines[key].match(titleRegExp)[1]
504 | }
505 | if(codeBlockFirstLines[key].match(highLightLinesRegExp) != null) {
506 | let highLightLinesInfo = codeBlockFirstLines[key].match(highLightLinesRegExp)[1]
507 | highLightLines = analyseHighLightLines(highLightLinesInfo)
508 | }
509 |
510 | let lineSize = codeBlockSections[key].position.end.line - codeBlockSections[key].position.start.line - 1
511 |
512 | let cbMeta: CodeBlockMeta = {
513 | langName: langName,
514 | lineSize: lineSize,
515 | pre: codeElm.parentElement,
516 | code: codeElm as HTMLElement,
517 | title: title,
518 | isCollapse: false,
519 | div: codeElm.parentElement.parentElement,
520 | contentList: [],
521 | highLightLines: highLightLines
522 | }
523 | addCodeTitleWrapper(plugin, codeElm.parentElement, cbMeta) // 导出取消代码块折叠
524 | addCodeTitle(plugin, cbMeta.pre, cbMeta)
525 | if(plugin.settings.showLineNumber) {
526 | addLineNumber(plugin, cbMeta)
527 | }
528 | addLineHighLight(plugin, cbMeta.pre, cbMeta)
529 | })
530 | }
--------------------------------------------------------------------------------
/main.js:
--------------------------------------------------------------------------------
1 | /*
2 | THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
3 | if you want to view the source, please visit the github repository of this plugin
4 | */
5 |
6 | var __create = Object.create;
7 | var __defProp = Object.defineProperty;
8 | var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
9 | var __getOwnPropNames = Object.getOwnPropertyNames;
10 | var __getProtoOf = Object.getPrototypeOf;
11 | var __hasOwnProp = Object.prototype.hasOwnProperty;
12 | var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
13 | var __export = (target, all) => {
14 | __markAsModule(target);
15 | for (var name in all)
16 | __defProp(target, name, { get: all[name], enumerable: true });
17 | };
18 | var __reExport = (target, module2, desc) => {
19 | if (module2 && typeof module2 === "object" || typeof module2 === "function") {
20 | for (let key of __getOwnPropNames(module2))
21 | if (!__hasOwnProp.call(target, key) && key !== "default")
22 | __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
23 | }
24 | return target;
25 | };
26 | var __toModule = (module2) => {
27 | return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
28 | };
29 | var __async = (__this, __arguments, generator) => {
30 | return new Promise((resolve, reject) => {
31 | var fulfilled = (value) => {
32 | try {
33 | step(generator.next(value));
34 | } catch (e) {
35 | reject(e);
36 | }
37 | };
38 | var rejected = (value) => {
39 | try {
40 | step(generator.throw(value));
41 | } catch (e) {
42 | reject(e);
43 | }
44 | };
45 | var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected);
46 | step((generator = generator.apply(__this, __arguments)).next());
47 | });
48 | };
49 |
50 | // main.ts
51 | __export(exports, {
52 | BetterCodeBlocks: () => BetterCodeBlocks,
53 | default: () => BetterCodeBlock
54 | });
55 | var import_obsidian = __toModule(require("obsidian"));
56 | var DEFAULT_LANG = "";
57 | var LANG_REG = /^language-/;
58 | var LINE_SPLIT_MARK = "\n";
59 | var titleRegExp = /TI:"([^"]*)"/i;
60 | var highLightLinesRegExp = /HL:"([^"]*)"/i;
61 | var foldRegExp = /"FOLD"/i;
62 | var CB_PADDING_TOP = "35px";
63 | var DEFAULT_SETTINGS = {
64 | substitutionTokenForSpace: void 0,
65 | titleBackgroundColor: "#00000020",
66 | titleFontColor: void 0,
67 | highLightColor: "#2d82cc20",
68 | excludeLangs: [],
69 | showLineNumber: true,
70 | showDividingLine: false,
71 | showLangNameInTopRight: true
72 | };
73 | var BetterCodeBlock = class extends import_obsidian.Plugin {
74 | onload() {
75 | return __async(this, null, function* () {
76 | console.log("Loading Better Code Block Plugin");
77 | yield this.loadSettings();
78 | this.addSettingTab(new BetterCodeBlockTab(this.app, this));
79 | this.registerMarkdownPostProcessor((el, ctx) => {
80 | BetterCodeBlocks(el, ctx, this);
81 | app.workspace.on("resize", () => {
82 | resizeNumWrapAndHLWrap(el, ctx);
83 | });
84 | });
85 | });
86 | }
87 | onunload() {
88 | console.log("Unloading Better Code Block Plugin");
89 | }
90 | loadSettings() {
91 | return __async(this, null, function* () {
92 | this.settings = Object.assign({}, DEFAULT_SETTINGS, yield this.loadData());
93 | });
94 | }
95 | saveSettings() {
96 | return __async(this, null, function* () {
97 | yield this.saveData(this.settings);
98 | });
99 | }
100 | };
101 | var BetterCodeBlockTab = class extends import_obsidian.PluginSettingTab {
102 | constructor(app2, plugin) {
103 | super(app2, plugin);
104 | this.plugin = plugin;
105 | }
106 | display() {
107 | let { containerEl } = this;
108 | containerEl.empty();
109 | new import_obsidian.Setting(containerEl).setName("Exclude language list").setDesc("Title and line numbers do not apply in these languages, separate by `,`").addText((text) => text.setPlaceholder("like todoist,other,...").setValue(this.plugin.settings.excludeLangs.join(",")).onChange((value) => __async(this, null, function* () {
110 | this.plugin.settings.excludeLangs = value.split(",");
111 | yield this.plugin.saveSettings();
112 | })));
113 | new import_obsidian.Setting(containerEl).setName("Font color of title").addText((tc) => tc.setPlaceholder("Enter a color").setValue(this.plugin.settings.titleFontColor).onChange((value) => __async(this, null, function* () {
114 | this.plugin.settings.titleFontColor = value;
115 | yield this.plugin.saveSettings();
116 | })));
117 | new import_obsidian.Setting(containerEl).setName("Background color of title").addText((tc) => tc.setPlaceholder("#00000020").setValue(this.plugin.settings.titleBackgroundColor).onChange((value) => __async(this, null, function* () {
118 | this.plugin.settings.titleBackgroundColor = value;
119 | yield this.plugin.saveSettings();
120 | })));
121 | new import_obsidian.Setting(containerEl).setName("HighLight Color").addText((tc) => tc.setPlaceholder("#2d82cc20").setValue(this.plugin.settings.highLightColor).onChange((value) => __async(this, null, function* () {
122 | this.plugin.settings.highLightColor = value;
123 | yield this.plugin.saveSettings();
124 | })));
125 | new import_obsidian.Setting(containerEl).setName("Show line number").addToggle((tc) => tc.setValue(this.plugin.settings.showLineNumber).onChange((value) => __async(this, null, function* () {
126 | this.plugin.settings.showLineNumber = value;
127 | yield this.plugin.saveSettings();
128 | })));
129 | new import_obsidian.Setting(containerEl).setName("Show dividing line").addToggle((tc) => tc.setValue(this.plugin.settings.showDividingLine).onChange((value) => __async(this, null, function* () {
130 | this.plugin.settings.showDividingLine = value;
131 | yield this.plugin.saveSettings();
132 | })));
133 | new import_obsidian.Setting(containerEl).setName("Show language name in the top right").addToggle((tc) => tc.setValue(this.plugin.settings.showLangNameInTopRight).onChange((value) => __async(this, null, function* () {
134 | this.plugin.settings.showLangNameInTopRight = value;
135 | yield this.plugin.saveSettings();
136 | })));
137 | }
138 | };
139 | function BetterCodeBlocks(el, context, plugin) {
140 | return __async(this, null, function* () {
141 | var _a;
142 | const settings = plugin.settings;
143 | const codeElm = el.querySelector("pre > code");
144 | if (!codeElm) {
145 | return;
146 | }
147 | let lang = DEFAULT_LANG;
148 | if (plugin.settings.excludeLangs.some((eLangName) => codeElm.classList.contains(`language-${eLangName}`))) {
149 | return;
150 | }
151 | codeElm.classList.forEach((value, key, parent) => {
152 | if (LANG_REG.test(value)) {
153 | lang = value.replace("language-", "");
154 | return;
155 | }
156 | });
157 | if (lang == DEFAULT_LANG) {
158 | return;
159 | }
160 | let codeBlock = context.getSectionInfo(codeElm);
161 | let codeBlockFirstLine = "";
162 | if (codeBlock) {
163 | let view = app.workspace.getActiveViewOfType(import_obsidian.MarkdownView);
164 | codeBlockFirstLine = view.editor.getLine(codeBlock.lineStart);
165 | } else {
166 | let file = app.vault.getAbstractFileByPath(context.sourcePath);
167 | let cache = app.metadataCache.getCache(context.sourcePath);
168 | let fileContent = yield app.vault.cachedRead(file);
169 | let fileContentLines = fileContent.split(/\n/g);
170 | let codeBlockFirstLines = [];
171 | let codeBlockSections = [];
172 | (_a = cache.sections) == null ? void 0 : _a.forEach((element) => __async(this, null, function* () {
173 | if (element.type == "code") {
174 | let lineStart = element.position.start.line;
175 | codeBlockFirstLine = fileContentLines[lineStart];
176 | codeBlockSections.push(element);
177 | codeBlockFirstLines.push(codeBlockFirstLine);
178 | }
179 | }));
180 | exportPDF(el, plugin, codeBlockFirstLines, codeBlockSections);
181 | return;
182 | }
183 | let title = "";
184 | let highLightLines = [];
185 | if (codeBlockFirstLine.match(titleRegExp) != null) {
186 | title = codeBlockFirstLine.match(titleRegExp)[1];
187 | }
188 | if (codeBlockFirstLine.match(highLightLinesRegExp) != null) {
189 | let highLightLinesInfo = codeBlockFirstLine.match(highLightLinesRegExp)[1];
190 | highLightLines = analyseHighLightLines(highLightLinesInfo);
191 | }
192 | let isCollapse = false;
193 | if (foldRegExp.test(codeBlockFirstLine)) {
194 | isCollapse = true;
195 | }
196 | const pre = codeElm.parentElement;
197 | const div = pre.parentElement;
198 | const contentList = codeElm.textContent.split(LINE_SPLIT_MARK);
199 | const lineSize = codeBlock.lineEnd - codeBlock.lineStart - 1;
200 | const cbMeta = { langName: lang, lineSize, pre, code: codeElm, title, isCollapse, div, contentList, highLightLines };
201 | const { showLineNumber } = plugin.settings;
202 | addCodeTitleWrapper(plugin, pre, cbMeta);
203 | addCodeTitle(plugin, pre, cbMeta);
204 | if (showLineNumber) {
205 | addLineNumber(plugin, cbMeta);
206 | }
207 | addLineHighLight(plugin, pre, cbMeta);
208 | resizeNumWrapAndHLWrap(el, context);
209 | });
210 | }
211 | function createElement(tagName, defaultClassName) {
212 | const element = document.createElement(tagName);
213 | if (defaultClassName) {
214 | element.className = defaultClassName;
215 | }
216 | return element;
217 | }
218 | function addCodeTitleWrapper(plugin, preElm, cbMeta) {
219 | preElm.style.setProperty("position", "relative", "important");
220 | preElm.style.setProperty("padding-top", CB_PADDING_TOP, "important");
221 | let wrapper = document.createElement("pre");
222 | if (cbMeta.isCollapse) {
223 | wrapper.setAttribute("closed", "");
224 | }
225 | wrapper.className = "obsidian-embedded-code-title__code-block-title";
226 | wrapper.style.backgroundColor = plugin.settings.titleBackgroundColor || "#00000020";
227 | let collapser = createElement("div", "collapser");
228 | let handle = createElement("div", "handle");
229 | collapser.appendChild(handle);
230 | wrapper.appendChild(collapser);
231 | wrapper.addEventListener("click", function() {
232 | if (wrapper.hasAttribute("closed")) {
233 | wrapper.removeAttribute("closed");
234 | } else {
235 | wrapper.setAttribute("closed", "");
236 | }
237 | });
238 | preElm.appendChild(wrapper);
239 | }
240 | function addCodeTitle(plugin, preElm, cbMeta) {
241 | let wrapper = preElm.querySelector(".obsidian-embedded-code-title__code-block-title");
242 | let titleElm = document.createElement("div");
243 | titleElm.className = "title";
244 | titleElm.appendText(cbMeta.title);
245 | wrapper.appendChild(titleElm);
246 | if (plugin.settings.titleFontColor) {
247 | titleElm.style.setProperty("color", plugin.settings.titleFontColor, "important");
248 | }
249 | if (plugin.settings.showLangNameInTopRight) {
250 | let langName = document.createElement("div");
251 | let langNameString = cbMeta.langName;
252 | langNameString = langNameString[0].toUpperCase() + langNameString.slice(1);
253 | langName.appendText(langNameString);
254 | langName.className = "langName";
255 | wrapper.appendChild(langName);
256 | }
257 | preElm.prepend(wrapper);
258 | }
259 | function addLineNumber(plugin, cbMeta) {
260 | const { lineSize, pre, div } = cbMeta;
261 | div.classList.add("code-block-wrap");
262 | const lineNumber = createElement("span", "code-block-linenum-wrap");
263 | lineNumber.style.top = CB_PADDING_TOP;
264 | Array.from({ length: lineSize }, (v, k) => k).forEach((i) => {
265 | const singleLine = createElement("span", "code-block-linenum");
266 | lineNumber.appendChild(singleLine);
267 | });
268 | if (plugin.settings.showDividingLine) {
269 | lineNumber.style.borderRight = "1px currentColor solid";
270 | }
271 | pre.appendChild(lineNumber);
272 | pre.classList.add("code-block-pre__has-linenum");
273 | }
274 | function addLineHighLight(plugin, preElm, cbMeta) {
275 | if (cbMeta.highLightLines.length == 0)
276 | return;
277 | let highLightWrap = document.createElement("pre");
278 | highLightWrap.className = "code-block-highlight-wrap";
279 | for (let i = 0; i < cbMeta.lineSize; i++) {
280 | const singleLine = createElement("span", "code-block-highlight");
281 | if (cbMeta.highLightLines.contains(i + 1)) {
282 | singleLine.style.backgroundColor = plugin.settings.highLightColor || "#2d82cc20";
283 | }
284 | highLightWrap.appendChild(singleLine);
285 | }
286 | preElm.appendChild(highLightWrap);
287 | }
288 | function analyseHighLightLines(str) {
289 | str = str.replace(/\s*/g, "");
290 | const result = [];
291 | let strs = str.split(",");
292 | strs.forEach((it) => {
293 | if (/\w+-\w+/.test(it)) {
294 | let left = Number(it.split("-")[0]);
295 | let right = Number(it.split("-")[1]);
296 | for (let i = left; i <= right; i++) {
297 | result.push(i);
298 | }
299 | } else {
300 | result.push(Number(it));
301 | }
302 | });
303 | return result;
304 | }
305 | function resizeNumWrapAndHLWrap(el, context) {
306 | setTimeout(function() {
307 | return __async(this, null, function* () {
308 | let codeBlockEl = el.querySelector("pre > code");
309 | if (!codeBlockEl)
310 | return;
311 | let numWrap = el.querySelector(".code-block-linenum-wrap");
312 | let highWrap = el.querySelector(".code-block-highlight-wrap");
313 | let codeBlockInfo = context.getSectionInfo(codeBlockEl);
314 | let view;
315 | let codeBlockLineNum;
316 | let lineStart = 0;
317 | let lineEnd = 0;
318 | if (codeBlockInfo) {
319 | view = app.workspace.getActiveViewOfType(import_obsidian.MarkdownView);
320 | codeBlockLineNum = codeBlockInfo.lineEnd - codeBlockInfo.lineStart - 1;
321 | } else {
322 | return;
323 | }
324 | let span = createElement("span");
325 | for (let i = 0; i < codeBlockLineNum; i++) {
326 | let oneLineText;
327 | if (view) {
328 | oneLineText = view.editor.getLine(codeBlockInfo.lineStart + i + 1);
329 | } else {
330 | }
331 | span.innerHTML = oneLineText || "0";
332 | codeBlockEl.appendChild(span);
333 | span.style.display = "block";
334 | let lineHeight = span.getBoundingClientRect().height + "px";
335 | let numOneLine = numWrap ? numWrap.childNodes[i] : null;
336 | let hlOneLine = highWrap ? highWrap.childNodes[i] : null;
337 | if (numOneLine)
338 | numOneLine.style.height = lineHeight;
339 | if (hlOneLine)
340 | hlOneLine.style.height = lineHeight;
341 | span.remove();
342 | }
343 | });
344 | }, 100);
345 | }
346 | function exportPDF(el, plugin, codeBlockFirstLines, codeBlockSections) {
347 | let codeBlocks = el.querySelectorAll("pre > code");
348 | codeBlocks.forEach((codeElm, key) => {
349 | let langName = "", title = "", highLightLines = [];
350 | codeElm.classList.forEach((value) => {
351 | if (LANG_REG.test(value)) {
352 | langName = value.replace("language-", "");
353 | return;
354 | }
355 | });
356 | if (codeBlockFirstLines[key].match(titleRegExp) != null) {
357 | title = codeBlockFirstLines[key].match(titleRegExp)[1];
358 | }
359 | if (codeBlockFirstLines[key].match(highLightLinesRegExp) != null) {
360 | let highLightLinesInfo = codeBlockFirstLines[key].match(highLightLinesRegExp)[1];
361 | highLightLines = analyseHighLightLines(highLightLinesInfo);
362 | }
363 | let lineSize = codeBlockSections[key].position.end.line - codeBlockSections[key].position.start.line - 1;
364 | let cbMeta = {
365 | langName,
366 | lineSize,
367 | pre: codeElm.parentElement,
368 | code: codeElm,
369 | title,
370 | isCollapse: false,
371 | div: codeElm.parentElement.parentElement,
372 | contentList: [],
373 | highLightLines
374 | };
375 | addCodeTitleWrapper(plugin, codeElm.parentElement, cbMeta);
376 | addCodeTitle(plugin, cbMeta.pre, cbMeta);
377 | if (plugin.settings.showLineNumber) {
378 | addLineNumber(plugin, cbMeta);
379 | }
380 | addLineHighLight(plugin, cbMeta.pre, cbMeta);
381 | });
382 | }
383 | //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["main.ts"],
  "sourcesContent": ["import { linkSync } from 'fs';\nimport { App, Editor, MarkdownView, Modal, Notice, Plugin, PluginSettingTab, Setting, MarkdownPostProcessorContext, Menu, SettingTab, TAbstractFile, TFile, SectionCache, Vault } from 'obsidian';\nimport { json } from 'stream/consumers';\n\nconst DEFAULT_LANG_ATTR = 'language-text'\nconst DEFAULT_LANG = ''\nconst LANG_REG = /^language-/\nconst LINE_SPLIT_MARK = '\\n'\n\nconst titleRegExp = /TI:\"([^\"]*)\"/i\nconst highLightLinesRegExp = /HL:\"([^\"]*)\"/i\nconst foldRegExp = /\"FOLD\"/i\n\nconst CB_PADDING_TOP = \"35px\" // \u4EE3\u7801\u5757\u4E0A\u8FB9\u8DDD\n\ninterface Settings {\n\tsubstitutionTokenForSpace: string;\n\ttitleBackgroundColor: string;\n\ttitleFontColor: string;\n\thighLightColor: string;\n\n\texcludeLangs: string[]; // \u9700\u8981\u6392\u9664\u7684\u8BED\u8A00\n\n\tshowLineNumber: boolean; // \u663E\u793A\u884C\u53F7\n\tshowDividingLine: boolean;\n\tshowLangNameInTopRight: boolean;\n}\n\nconst DEFAULT_SETTINGS: Settings = {\n\tsubstitutionTokenForSpace: undefined,\n\ttitleBackgroundColor: \"#00000020\",\n\ttitleFontColor: undefined,\n\thighLightColor: \"#2d82cc20\",\n\n\texcludeLangs: [],\n\n\tshowLineNumber: true,\n\tshowDividingLine: false,\n\tshowLangNameInTopRight: true\n};\n\ninterface CodeBlockMeta {\n\t// Language name\n\tlangName: string;\n\n\t// Code block total line size\n\tlineSize: number;\n\n\t// Code block 'pre' HTMLElement\n\tpre: HTMLElement;\n\n\t// Code block 'code' HTMLElement\n\tcode: HTMLElement;\n\n\ttitle: string; // \u4EE3\u7801\u5757\u6807\u9898\n\tisCollapse:boolean; // \u662F\u5426\u9ED8\u8BA4\u6298\u53E0\n\n\t// Code block wrap div\n\tdiv: HTMLElement;\n\tcontentList: string[];\n\thighLightLines: number[];\n}\n\n// Refer https://developer.mozilla.org/ja/docs/Web/JavaScript/Guide/Regular_Expressions#escaping\nfunction escapeRegExp(str: string): string {\n\treturn str.replace(/[.*+?^=!:${}()|[\\]\\/\\\\]/g, \"\\\\$&\"); // \u4E3A\u7279\u6B8A\u7B26\u53F7\u52A0\u4E0A\u8F6C\u4E49\u7B26\u53F7\"\\\"\n}\n\nexport default class BetterCodeBlock extends Plugin {\n\tsettings: Settings;\n\n\tasync onload() {\n\t\tconsole.log(\"Loading Better Code Block Plugin\");\n\t\tawait this.loadSettings();\n\t\tthis.addSettingTab(new BetterCodeBlockTab(this.app, this));\n\t\tthis.registerMarkdownPostProcessor((el, ctx) => {\n\t\t\tBetterCodeBlocks(el, ctx, this)\n\t\t\tapp.workspace.on('resize', () => {\n\t\t\t\tresizeNumWrapAndHLWrap(el, ctx)\n\t\t\t})\n\t\t})\n\n\t}\n\n\tonunload () {\n\t\tconsole.log('Unloading Better Code Block Plugin');\n\t}\n\t\n\tasync loadSettings() {\n\t\tthis.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData());\n\t}\n\t\n\tasync saveSettings() {\n\t\tawait this.saveData(this.settings);\n\t}\n}\n\nclass BetterCodeBlockTab extends PluginSettingTab {\n\tplugin: BetterCodeBlock;\n  \n\tconstructor(app: App, plugin: BetterCodeBlock) {\n\t  super(app, plugin);\n\t  this.plugin = plugin;\n\t}\n  \n\tdisplay(): void {\n\t  let { containerEl } = this;\n  \n\t  containerEl.empty();\n\t\n\t  new Setting(containerEl)\n\t\t.setName(\"Exclude language list\")\n\t\t.setDesc(\"Title and line numbers do not apply in these languages, separate by `,`\")\n\t\t.addText(text => text.setPlaceholder('like todoist,other,...')\n\t\t.setValue(this.plugin.settings.excludeLangs.join(','))\n\t\t.onChange(async (value) => {\n\t\t\tthis.plugin.settings.excludeLangs = value.split(',');\n\t\t\tawait this.plugin.saveSettings();\n\t\t})\n\t\t)\n  \n\t  new Setting(containerEl).setName(\"Font color of title\").addText((tc) =>\n\t\ttc\n\t\t  .setPlaceholder(\"Enter a color\")\n\t\t  .setValue(this.plugin.settings.titleFontColor)\n\t\t  .onChange(async (value) => {\n\t\t\tthis.plugin.settings.titleFontColor = value;\n\t\t\tawait this.plugin.saveSettings();\n\t\t  })\n\t  );\n  \n\t  new Setting(containerEl)\n\t\t.setName(\"Background color of title\")\n\t\t.addText((tc) =>\n\t\t  tc\n\t\t\t.setPlaceholder(\"#00000020\")\n\t\t\t.setValue(this.plugin.settings.titleBackgroundColor)\n\t\t\t.onChange(async (value) => {\n\t\t\t  this.plugin.settings.titleBackgroundColor = value;\n\t\t\t  await this.plugin.saveSettings();\n\t\t\t})\n\t\t);\n\n\t\tnew Setting(containerEl)\n\t\t.setName(\"HighLight Color\")\n\t\t.addText((tc) =>\n\t\t  tc\n\t\t\t.setPlaceholder(\"#2d82cc20\")\n\t\t\t.setValue(this.plugin.settings.highLightColor)\n\t\t\t.onChange(async (value) => {\n\t\t\t  this.plugin.settings.highLightColor = value;\n\t\t\t  await this.plugin.saveSettings();\n\t\t\t})\n\t\t);\n\n\t\tnew Setting(containerEl)\n\t\t.setName(\"Show line number\")\n\t\t.addToggle((tc) => \n\t\ttc.setValue(this.plugin.settings.showLineNumber)\n\t\t.onChange(async(value) => {\n\t\t\tthis.plugin.settings.showLineNumber = value;\n\t\t\tawait this.plugin.saveSettings();\n\t\t})\n\t\t)\n\n\t\tnew Setting(containerEl)\n\t\t.setName(\"Show dividing line\")\n\t\t.addToggle((tc) =>\n\t\ttc.setValue(this.plugin.settings.showDividingLine)\n\t\t.onChange(async(value) => {\n\t\t\tthis.plugin.settings.showDividingLine = value;\n\t\t\tawait this.plugin.saveSettings();\n\t\t})\n\t\t)\n\n\t\tnew Setting(containerEl)\n\t\t.setName(\"Show language name in the top right\")\n\t\t.addToggle((tc) =>\n\t\ttc.setValue(this.plugin.settings.showLangNameInTopRight)\n\t\t.onChange(async(value) => {\n\t\t\tthis.plugin.settings.showLangNameInTopRight = value;\n\t\t\tawait this.plugin.saveSettings();\n\t\t})\n\t\t)\n\t}\n  }\n\n\nexport async function BetterCodeBlocks(el: HTMLElement, context: MarkdownPostProcessorContext, plugin: BetterCodeBlock) {\n\tconst settings = plugin.settings\n\tconst codeElm: HTMLElement = el.querySelector('pre > code')\n\t// only change pre>code\n\tif (!codeElm) { return }\n\n\tlet lang = DEFAULT_LANG\n\t// return when lang is in exclude list\n\tif (plugin.settings.excludeLangs.some(eLangName => codeElm.classList.contains(`language-${eLangName}`))) {\n\t  return\n\t}\n\t\n\tcodeElm.classList.forEach((value, key, parent) => {\n\t  if (LANG_REG.test(value)) {\n\t\tlang = value.replace('language-', '')\n\t\treturn\n\t  }\n\t})\n\n\t// if the code block is not described, return\n\tif(lang == DEFAULT_LANG) {\n\t\treturn\n\t}\n\n\tlet codeBlock = context.getSectionInfo(codeElm)\n\tlet codeBlockFirstLine = \"\"\n\n\tif(codeBlock) {\n\t\tlet view = app.workspace.getActiveViewOfType(MarkdownView)\n\t\tcodeBlockFirstLine = view.editor.getLine(codeBlock.lineStart)\n\t} else { \n\t\tlet file = app.vault.getAbstractFileByPath(context.sourcePath)\n\t\tlet cache = app.metadataCache.getCache(context.sourcePath)\n\t\tlet fileContent = await app.vault.cachedRead(<TFile> file)\n\t\tlet fileContentLines = fileContent.split(/\\n/g)\n\n\t\tlet codeBlockFirstLines: string[] = []\n\t\tlet codeBlockSections: SectionCache[] = []\n\n\t\tcache.sections?.forEach(async element => {\n\t\t\tif(element.type == \"code\") {\n\t\t\t\tlet lineStart = element.position.start.line\n\t\t\t\tcodeBlockFirstLine = fileContentLines[lineStart]\n\t\t\t\tcodeBlockSections.push(element)\n\t\t\t\tcodeBlockFirstLines.push(codeBlockFirstLine)\n\t\t\t}\n\t\t});\n\t\texportPDF(el, plugin, codeBlockFirstLines, codeBlockSections)\n\t\treturn\n\t}\n\n\tlet title: string = \"\"\n\tlet highLightLines: number[] = []\n\tif(codeBlockFirstLine.match(titleRegExp) != null) {\n\t\ttitle = codeBlockFirstLine.match(titleRegExp)[1]\n\t}\n\tif(codeBlockFirstLine.match(highLightLinesRegExp) != null) {\n\t\tlet highLightLinesInfo = codeBlockFirstLine.match(highLightLinesRegExp)[1]\n\t\thighLightLines = analyseHighLightLines(highLightLinesInfo)\n\t}\n\n\tlet isCollapse = false;\n\tif(foldRegExp.test(codeBlockFirstLine)) {\n\t\tisCollapse = true\n\t}\n\n\tconst pre = codeElm.parentElement // code-block-pre__has-linenum\n\tconst div = pre.parentElement // class code-block-wrap\n\n\t/* const { lineStart, lineEnd } = ctx.getSectionInfo(el)\n\tconst lineSize = lineEnd - lineStart - 1 */\n\tconst contentList: string[] = codeElm.textContent.split(LINE_SPLIT_MARK)\n\t// const lineSize = contentList.length - 1\n\tconst lineSize = codeBlock.lineEnd - codeBlock.lineStart - 1\n\n\tconst cbMeta = { langName: lang, lineSize, pre, code: codeElm, title, isCollapse, div, contentList, highLightLines}\n\n\tconst {showLineNumber} = plugin.settings\n\n\taddCodeTitleWrapper(plugin, pre, cbMeta)\n\t//addIconToTitle(plugin, pre, cbMeta)\n\taddCodeTitle(plugin, pre, cbMeta);\n\n\t// add line number\n\tif (showLineNumber) {\n\t\taddLineNumber(plugin, cbMeta)\n\t}\n\n\taddLineHighLight(plugin, pre, cbMeta)\n\n\tresizeNumWrapAndHLWrap(el,context) // \u8C03\u7528\u4E00\u6B21\u4EE5\u89E3\u51B3\u67D0\u4E9B\u65F6\u5019\u6253\u5F00\u6587\u4EF6\u884C\u9AD8\u672A\u88AB\u91CD\u8BBE\u9AD8\u5EA6\n}\n\nfunction createElement (tagName: string, defaultClassName?: string) {\n\tconst element = document.createElement(tagName)\n\tif (defaultClassName) {\n\t  element.className = defaultClassName\n\t}\n\treturn element\n}\n\nfunction addCodeTitleWrapper(plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {\n\tpreElm.style.setProperty(\"position\", \"relative\", \"important\");\n\tpreElm.style.setProperty(\"padding-top\", CB_PADDING_TOP, \"important\");\n\n\tlet wrapper = document.createElement(\"pre\")\n\tif(cbMeta.isCollapse) {\n\t\twrapper.setAttribute(\"closed\",\"\")\n\t}\n\twrapper.className = \"obsidian-embedded-code-title__code-block-title\"\n\n\twrapper.style.backgroundColor = plugin.settings.titleBackgroundColor || \"#00000020\";\n\n\tlet collapser = createElement(\"div\",\"collapser\")\n\tlet handle = createElement(\"div\", \"handle\")\n\tcollapser.appendChild(handle)\n\twrapper.appendChild(collapser)\n\n\twrapper.addEventListener('click',function(this: any) {\n\t\tif(wrapper.hasAttribute(\"closed\")){\n\t\t\twrapper.removeAttribute(\"closed\")\n\t\t} else {\n\t\t\twrapper.setAttribute(\"closed\",'')\n\t\t}\n\t})\n\n\tpreElm.appendChild(wrapper)\n}\n\nfunction addCodeTitle (plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {\n\tlet wrapper = preElm.querySelector(\".obsidian-embedded-code-title__code-block-title\")\n\n\tlet titleElm = document.createElement(\"div\")\n\ttitleElm.className = \"title\"\n\n\ttitleElm.appendText(cbMeta.title)\n\twrapper.appendChild(titleElm)\n\n\tif(plugin.settings.titleFontColor) {\n\t\ttitleElm.style.setProperty(\"color\", plugin.settings.titleFontColor, \"important\")\n\t}\n\t\n\tif(plugin.settings.showLangNameInTopRight) {\n\t\tlet langName = document.createElement(\"div\"); // \u5728\u53F3\u4FA7\u6DFB\u52A0\u4EE3\u7801\u7C7B\u578B\n\t\tlet langNameString = cbMeta.langName\n\t\tlangNameString = langNameString[0].toUpperCase() + langNameString.slice(1) // \u9996\u5B57\u6BCD\u5927\u5199\n\t\tlangName.appendText(langNameString);\n\t\tlangName.className = \"langName\";\n\t\twrapper.appendChild(langName);\n\t}\n\n\tpreElm.prepend(wrapper);\n\n}\n\nfunction addLineNumber (plugin: BetterCodeBlock, cbMeta: CodeBlockMeta) {\n\tconst { lineSize, pre, div } = cbMeta\n\t// let div position: relative;\n\tdiv.classList.add('code-block-wrap')\n\n\t// const { fontSize, lineHeight } = window.getComputedStyle(cbMeta.code)\n\tconst lineNumber = createElement('span', 'code-block-linenum-wrap')\n\tlineNumber.style.top = CB_PADDING_TOP;\n\tArray.from({ length: lineSize }, (v, k) => k).forEach(i => {\n\t  const singleLine = createElement('span', 'code-block-linenum')\n\t  // singleLine.style.fontSize = fontSize\n\t  // singleLine.style.lineHeight = lineHeight\n\t  lineNumber.appendChild(singleLine)\n\t})\n\t\n\tif(plugin.settings.showDividingLine) {\n\t\tlineNumber.style.borderRight = \"1px currentColor solid\"\n\t}\n\n\tpre.appendChild(lineNumber)\n\tpre.classList.add('code-block-pre__has-linenum')\n}\n\nfunction addLineHighLight(plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {\n\tif(cbMeta.highLightLines.length == 0) return\n\n\tlet highLightWrap = document.createElement(\"pre\")\n\thighLightWrap.className = \"code-block-highlight-wrap\"\n\tfor(let i = 0; i < cbMeta.lineSize; i++) {\n\t\tconst singleLine = createElement(\"span\", 'code-block-highlight')\n\t\tif(cbMeta.highLightLines.contains(i+1)) {\n\t\t\tsingleLine.style.backgroundColor = plugin.settings.highLightColor || \"#2d82cc20\"\n\t\t}\n\t\thighLightWrap.appendChild(singleLine)\n\t}\n\n\tpreElm.appendChild(highLightWrap)\n}\n\nfunction analyseHighLightLines(str: string): number[] {\n\tstr = str.replace(/\\s*/g, \"\") // \u53BB\u9664\u5B57\u7B26\u4E32\u4E2D\u6240\u6709\u7A7A\u683C\n\tconst result: number[] = []\n\n\tlet strs = str.split(\",\")\n\tstrs.forEach(it => {\n\t\tif(/\\w+-\\w+/.test(it)) { // \u5982\u679C\u5339\u914D 1-3 \u8FD9\u6837\u7684\u683C\u5F0F\uFF0C\u4F9D\u6B21\u6DFB\u52A0\u6570\u5B57\n\t\t\tlet left = Number(it.split('-')[0])\n\t\t\tlet right = Number(it.split('-')[1])\n\t\t\tfor(let i = left; i <= right; i++) {\n\t\t\t\tresult.push(i)\n\t\t\t}\n\t\t} else {\n\t\t\tresult.push(Number(it))\n\t\t}\n\t})\n\n\treturn result\n}\n\nfunction addIconToTitle(plugin: BetterCodeBlock, preElm: HTMLElement, cbMeta: CodeBlockMeta) {\n\tlet title = preElm.querySelectorAll(\".obsidian-embedded-code-title__code-block-title\")\n\n\ttitle.forEach(it => {\n\t\tlet iconWrap = createElement(\"div\",\"icon-wrap\")\n\t\tlet icon = document.createElement(\"img\")\n\t\ticon.src = \"\"\n\t\ticonWrap.appendChild(icon)\n\t\tit.appendChild(iconWrap)\n\t})\n\t\n}\n\n// \u5728\u81EA\u52A8\u6362\u884C\u65F6\u5BF9\u6570\u5B57\u548C\u9AD8\u4EAE\u884C\u91CD\u65B0\u8BBE\u7F6E\u9AD8\u5EA6\n// These codes refer to the https://github.com/lijyze/obsidian-advanced-codeblock\nfunction resizeNumWrapAndHLWrap(el: HTMLElement, context: MarkdownPostProcessorContext) {\n\tsetTimeout(async function(){ // \u5EF6\u65F6100\u6BEB\u79D2\u4EE5\u89E3\u51B3\u67D0\u4E9B\u65F6\u5019\u6253\u5F00\u6587\u4EF6\u884C\u9AD8\u672A\u88AB\u91CD\u8BBE\u9AD8\u5EA6\n\t\t// console.log('on resize')\n\t\tlet codeBlockEl : HTMLElement = el.querySelector('pre > code')\n\t\tif(!codeBlockEl) return\n\n\t\tlet numWrap = el.querySelector('.code-block-linenum-wrap')\n\t\tlet highWrap = el.querySelector('.code-block-highlight-wrap')\n\n\t\tlet codeBlockInfo = context.getSectionInfo(codeBlockEl)\n\t\t// let view = app.workspace.getActiveViewOfType(MarkdownView)\n\t\t// let codeBlockLineNum = codeBlockInfo.lineEnd - codeBlockInfo.lineStart - 1 // \u9664\u53BB\u9996\u5C3E\u4E24\u884C\n\t\tlet view\n\t\tlet codeBlockLineNum\n\n\t\tlet lineStart = 0\n\t\tlet lineEnd = 0\n\t\tif(codeBlockInfo) {\n\t\t\tview = app.workspace.getActiveViewOfType(MarkdownView)\n\t\t\tcodeBlockLineNum = codeBlockInfo.lineEnd - codeBlockInfo.lineStart - 1 // \u9664\u53BB\u9996\u5C3E\u4E24\u884C\n\t\t} else {\n\t\t\treturn\n\t\t\t// let file = app.vault.getAbstractFileByPath(context.sourcePath)\n\t\t\t// let cache = app.metadataCache.getCache(context.sourcePath)\n\t\n\t\t\t// cache.sections?.forEach(async element => {\n\t\t\t// \tif(element.type == \"code\") {\n\t\t\t// \t\tlineStart = element.position.start.line\n\t\t\t// \t\tlineEnd = element.position.end.line\n\t\t\t// \t\tcodeBlockLineNum = lineEnd - lineStart - 1\n\t\t\t// \t\treturn\n\t\t\t// \t}\n\t\t\t// });\n\t\t\t// let file = app.vault.getAbstractFileByPath(context.sourcePath)\n\t\t\t// let cache = app.metadataCache.getCache(context.sourcePath)\n\t\t\t// let fileContent = await app.vault.cachedRead(<TFile> file)\n\t\t\t// let fileContentLines = fileContent.split(/\\n/g)\n\t\t}\n\n\t\tlet span = createElement(\"span\")\n\n\t\tfor(let i = 0; i < codeBlockLineNum; i++) {\n\t\t\tlet oneLineText\n\t\t\tif(view){\n\t\t\t\toneLineText = view.editor.getLine(codeBlockInfo.lineStart + i + 1)\n\t\t\t} else {\n\t\t\t\t// oneLineText = fileContentLines[lineStart + 1 + i]\n\t\t\t\t// let file = app.vault.getAbstractFileByPath(context.sourcePath)\n\t\t\t\t// let cache = app.metadataCache.getCache(context.sourcePath)\n\t\t\t\t// let fileContent = await app.vault.cachedRead(<TFile> file)\n\t\t\t\t// let fileContentLines = fileContent.split(/\\n/g)\n\t\t\t\t// oneLineText = fileContentLines[cache.sections]\n\t\t\t}\n\t\t\tspan.innerHTML = oneLineText || \"0\"\n\n\t\t\tcodeBlockEl.appendChild(span)\n\t\t\tspan.style.display = 'block'\n\n\t\t\tlet lineHeight = span.getBoundingClientRect().height + 'px' // \u6D4B\u91CF\u672C\u884C\u6587\u5B57\u7684\u9AD8\u5EA6\n\n\t\t\t// console.log(lineHeight + '    ' + span.getBoundingClientRect().width);\n\t\t\t\n\t\t\tlet numOneLine = numWrap? numWrap.childNodes[i] as HTMLElement : null\n\t\t\tlet hlOneLine = highWrap? highWrap.childNodes[i] as HTMLElement : null\n\n\t\t\tif(numOneLine) numOneLine.style.height = lineHeight;\n\t\t\tif(hlOneLine) hlOneLine.style.height = lineHeight;\n\n\t\t\tspan.remove() // \u6D4B\u91CF\u5B8C\u540E\u5220\u6389\n\t\t}\n\t}, 100)\n}\n\nfunction exportPDF(el: HTMLElement, plugin: BetterCodeBlock, codeBlockFirstLines: string[], codeBlockSections: SectionCache[]) {\n\tlet codeBlocks = el.querySelectorAll('pre > code')\n\tcodeBlocks.forEach((codeElm, key) => {\n\t\tlet langName = \"\", title = \"\", highLightLines: number[] = []\n\t\tcodeElm.classList.forEach(value => {\n\t\t\tif(LANG_REG.test(value)) {\n\t\t\t\tlangName = value.replace('language-', '')\n\t\t\t\treturn\n\t\t\t}\n\t\t})\n\n\t\tif(codeBlockFirstLines[key].match(titleRegExp) != null) {\n\t\t\ttitle = codeBlockFirstLines[key].match(titleRegExp)[1]\n\t\t}\n\t\tif(codeBlockFirstLines[key].match(highLightLinesRegExp) != null) {\n\t\t\tlet highLightLinesInfo = codeBlockFirstLines[key].match(highLightLinesRegExp)[1]\n\t\t\thighLightLines = analyseHighLightLines(highLightLinesInfo)\n\t\t}\n\n\t\tlet lineSize = codeBlockSections[key].position.end.line - codeBlockSections[key].position.start.line - 1\n\n\t\tlet cbMeta: CodeBlockMeta = {\n\t\t\tlangName: langName,\n\t\t\tlineSize: lineSize,\n\t\t\tpre: codeElm.parentElement,\n\t\t\tcode: codeElm as HTMLElement,\n\t\t\ttitle: title,\n\t\t\tisCollapse: false,\n\t\t\tdiv: codeElm.parentElement.parentElement,\n\t\t\tcontentList: [],\n\t\t\thighLightLines: highLightLines\n\t\t}\n\t\taddCodeTitleWrapper(plugin, codeElm.parentElement, cbMeta) // \u5BFC\u51FA\u53D6\u6D88\u4EE3\u7801\u5757\u6298\u53E0\n\t\taddCodeTitle(plugin, cbMeta.pre, cbMeta)\n\t\tif(plugin.settings.showLineNumber) {\n\t\t\taddLineNumber(plugin, cbMeta)\n\t\t}\n\t\taddLineHighLight(plugin, cbMeta.pre, cbMeta)\n\t})\n}"],
  "mappings": "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",
  "names": []
}

384 |
--------------------------------------------------------------------------------