├── .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 | ![image20220606011534.png](screenshots/image20220606011534.png) 25 | 26 | In version 1.0.4, add the language in the top right, like this: 27 | ![screenshots/image-20220601202203.png](screenshots/image-20220601202203.png) 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 | --------------------------------------------------------------------------------