mirror of
https://github.com/notion-enhancer/notion-enhancer.git
synced 2025-04-04 04:39:03 +00:00
feat(line-numbers): border & bg decoration styles
This commit is contained in:
parent
75a864db8c
commit
f2128fbf10
@ -1,63 +0,0 @@
|
|||||||
/**
|
|
||||||
* notion-enhancer: line numbers
|
|
||||||
* (c) 2020 CloudHill <rl.cloudhill@gmail.com> (https://github.com/CloudHill)
|
|
||||||
* (c) 2024 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
|
|
||||||
* (https://notion-enhancer.github.io/) under the MIT license
|
|
||||||
*/
|
|
||||||
|
|
||||||
.notion-code-block.line-numbers {
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
.code_line_numbers--plain:not(:empty) + div,
|
|
||||||
.code_line_numbers--background:not(:empty) + div,
|
|
||||||
.code_line_numbers--border:not(:empty) + div {
|
|
||||||
padding-left: 64px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.code_line_numbers--plain,
|
|
||||||
.code_line_numbers--background,
|
|
||||||
.code_line_numbers--border {
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
right: calc(100% - 64px);
|
|
||||||
top: 34px;
|
|
||||||
bottom: 32px;
|
|
||||||
padding-right: 27px;
|
|
||||||
|
|
||||||
font-size: 85%;
|
|
||||||
font-family: var(--theme--font_code);
|
|
||||||
text-align: right;
|
|
||||||
line-height: 1.5;
|
|
||||||
opacity: 0.8;
|
|
||||||
color: var(--theme--text_secondary);
|
|
||||||
|
|
||||||
overflow: hidden;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
.code_line_numbers--plain:empty,
|
|
||||||
.code_line_numbers--background:empty,
|
|
||||||
.code_line_numbers--border:empty {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
.code_line_numbers--background::before {
|
|
||||||
content: '';
|
|
||||||
position: absolute;
|
|
||||||
top: 0;
|
|
||||||
left: 7.25px;
|
|
||||||
width: calc(100% - 27px);
|
|
||||||
height: 100%;
|
|
||||||
display: block;
|
|
||||||
background-color: var(--theme--bg);
|
|
||||||
border-radius: 4px;
|
|
||||||
z-index: -1;
|
|
||||||
}
|
|
||||||
.code_line_numbers--border::before {
|
|
||||||
content: '';
|
|
||||||
position: absolute;
|
|
||||||
top: 0;
|
|
||||||
right: calc(100% - 52px);
|
|
||||||
width: 2px;
|
|
||||||
height: 100%;
|
|
||||||
display: block;
|
|
||||||
background-color: var(--theme--ui_divider);
|
|
||||||
}
|
|
75
src/extensions/line-numbers/client.mjs
Normal file → Executable file
75
src/extensions/line-numbers/client.mjs
Normal file → Executable file
@ -5,21 +5,27 @@
|
|||||||
* (https://notion-enhancer.github.io/) under the MIT license
|
* (https://notion-enhancer.github.io/) under the MIT license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function LineNumbers() {
|
function LineNumbers({ decorationStyle = "None" }) {
|
||||||
const { html } = globalThis.__enhancerApi;
|
const { html } = globalThis.__enhancerApi,
|
||||||
|
decorations = {
|
||||||
|
Border: `pr-[16px] border-r-([2px]
|
||||||
|
[color:var(--theme--bg-hover)])`,
|
||||||
|
Background: `pr-[4px] before:(absolute block
|
||||||
|
h-full w-[calc(100%-24px)] rounded-[4px] right-0
|
||||||
|
content-empty bg-[var(--theme--bg-hover)] z-[-1])`,
|
||||||
|
};
|
||||||
return html`<div
|
return html`<div
|
||||||
class="notion-enhancer--line-numbers flex-grow
|
class="notion-enhancer--line-numbers mt-[34px]
|
||||||
text-([85%] [var(--theme--fg-secondary)] right)
|
text-([85%] [var(--theme--fg-secondary)] right)
|
||||||
font-[var(--font--code)] pt-[34px] pb-[32px]
|
font-[var(--font--code)] overflow-hidden select-none
|
||||||
overflow-hidden"
|
relative flex-grow ${decorations[decorationStyle] || ""}"
|
||||||
></div>`;
|
></div>`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default async (api, db) => {
|
export default async (api, db) => {
|
||||||
const { html, addMutationListener } = api,
|
const { html, addMutationListener } = api,
|
||||||
|
decorationStyle = await db.get("decorationStyle"),
|
||||||
numberSingleLines = await db.get("numberSingleLines"),
|
numberSingleLines = await db.get("numberSingleLines"),
|
||||||
lineNumberDecoration = await db.get("lineNumberDecoration"),
|
|
||||||
lineNumbersClass = "notion-enhancer--line-numbers",
|
|
||||||
codeBlockSelector = ".notion-code-block.line-numbers > .notranslate";
|
codeBlockSelector = ".notion-code-block.line-numbers > .notranslate";
|
||||||
|
|
||||||
// get character width in pixels
|
// get character width in pixels
|
||||||
@ -50,45 +56,48 @@ export default async (api, db) => {
|
|||||||
for (const $code of document.querySelectorAll(codeBlockSelector)) {
|
for (const $code of document.querySelectorAll(codeBlockSelector)) {
|
||||||
const wrap = $code.style.wordBreak === "break-all",
|
const wrap = $code.style.wordBreak === "break-all",
|
||||||
lines = $code.innerText.split("\n"),
|
lines = $code.innerText.split("\n"),
|
||||||
lineCount = Math.max(lines.length - 1, 1),
|
numLines = Math.max(lines.length - 1, 1),
|
||||||
lineNumDigits = (Math.log(lineCount) * Math.LOG10E + 1) | 0;
|
numChars = lines.map((line) => line.length).join(","),
|
||||||
|
numDigits = (Math.log(numLines) * Math.LOG10E + 1) | 0;
|
||||||
|
|
||||||
const doUpdate =
|
if ($code.dataset.lines === wrap + "," + numChars) continue;
|
||||||
$code.dataset.lines !== String(lineCount) ||
|
$code.dataset.lines = wrap + "," + numChars;
|
||||||
$code.dataset.wrap !== String(wrap);
|
|
||||||
if (!doUpdate) continue;
|
|
||||||
$code.dataset.lines = lineCount;
|
|
||||||
$code.dataset.wrap = wrap;
|
|
||||||
|
|
||||||
|
// do not add to single-line blocks if disabled
|
||||||
|
const visible = numberSingleLines || numLines > 1,
|
||||||
|
width = visible
|
||||||
|
? decorationStyle === "Border"
|
||||||
|
? `calc(100% - 50px - ${numDigits}ch)`
|
||||||
|
: `calc(100% - 32px - ${numDigits}ch)`
|
||||||
|
: "",
|
||||||
|
paddingLeft = visible && decorationStyle === "Border" ? "16px" : "32px";
|
||||||
// shrink block to allow space for numbers
|
// shrink block to allow space for numbers
|
||||||
const width = `calc(100% - 32px - ${lineNumDigits}ch)`;
|
|
||||||
applyStyles($code.parentElement, { justifyContent: "flex-end" });
|
applyStyles($code.parentElement, { justifyContent: "flex-end" });
|
||||||
applyStyles($code, { minWidth: width, maxWidth: width });
|
applyStyles($code, { minWidth: width, maxWidth: width, paddingLeft });
|
||||||
|
|
||||||
// work out height of wrapped lines
|
// calculate heights of wrapped lines and render line nums
|
||||||
|
let totalHeight = 0;
|
||||||
const lineHeight = getLineHeight($code),
|
const lineHeight = getLineHeight($code),
|
||||||
charsPerLine = Math.floor(getLineWidth($code) / getCharWidth($code));
|
charsPerLine = Math.floor(getLineWidth($code) / getCharWidth($code));
|
||||||
|
$code._$lineNumbers ||= html`<${LineNumbers}...${{ decorationStyle }} />`;
|
||||||
// update line numbers in dom
|
for (let i = 1; i <= numLines; i++) {
|
||||||
let totalHeight = 34;
|
const $n = $code._$lineNumbers.children[i - 1] || html`<p>${i}</p>`;
|
||||||
$code._$lineNumbers ||= LineNumbers();
|
if (!$code._$lineNumbers.contains($n)) $code._$lineNumbers.append($n);
|
||||||
for (let i = 1; i <= lineCount; i++) {
|
const wrappedHeight =
|
||||||
let $n = $code._$lineNumbers.children[i - 1];
|
|
||||||
if (!$n) {
|
|
||||||
$n = html`<span class="block text-right">${i}</span>`;
|
|
||||||
$code._$lineNumbers.append($n);
|
|
||||||
}
|
|
||||||
const height =
|
|
||||||
wrap && lines[i - 1].length > charsPerLine
|
wrap && lines[i - 1].length > charsPerLine
|
||||||
? Math.ceil(lines[i - 1].length / charsPerLine) * lineHeight
|
? Math.ceil(lines[i - 1].length / charsPerLine) * lineHeight
|
||||||
: lineHeight;
|
: lineHeight;
|
||||||
applyStyles($n, { height: `${height}px` });
|
applyStyles($n, { height: `${wrappedHeight}px` });
|
||||||
totalHeight += height;
|
totalHeight += wrappedHeight;
|
||||||
}
|
}
|
||||||
applyStyles($code._$lineNumbers, { height: `${totalHeight}px` });
|
applyStyles($code._$lineNumbers, {
|
||||||
|
display: visible ? "" : "none",
|
||||||
|
height: `${totalHeight}px`,
|
||||||
|
});
|
||||||
|
|
||||||
if (!document.contains($code._$lineNumbers))
|
if (visible && !document.contains($code._$lineNumbers)) {
|
||||||
$code.before($code._$lineNumbers);
|
$code.before($code._$lineNumbers);
|
||||||
|
} else if (!visible) $code._$lineNumbers.style.display = "none";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 5.8 KiB |
4
src/extensions/line-numbers/mod.json
Normal file → Executable file
4
src/extensions/line-numbers/mod.json
Normal file → Executable file
@ -25,8 +25,8 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "select",
|
"type": "select",
|
||||||
"key": "lineNumberDecoration",
|
"key": "decorationStyle",
|
||||||
"description": "Select a decoration style to distinguish line numbers from code block content.",
|
"description": "Decorate line numbers with additional styling to distinguish them from code block content.",
|
||||||
"values": ["Border", "Background", "None"]
|
"values": ["Border", "Background", "None"]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
Loading…
Reference in New Issue
Block a user