Closed houlang closed 5 months ago
确实有这个问题。 因为目前没有好办法获取到“窗口中间是哪一行”这个信息,所以用的策略是把从上到下第8行作为近似。obsidian里面没有提供这样的api,所以不太好解决。不管是第8行还是再往上一点,本质都是一样的,一行的高度变化是很大的,如果插入一个图片,高度非常大,但是仍然会被编辑器看成一行。
而按照光标位置来确定在哪一行,就非常简单了,所以大纲里面高亮位置也很准确。
@guopenghui Hi,开发者,你好:
关于上面提到的问题(点击大纲激活不准确的问题)
根据我的需求(1、定位准确;2、同时内容上下滚动大纲跟随;3、光标定位同时也支持)以及开发者你提供的信息,当前我通过修改源码中的src\Outline.vue
文件,已经实现了上面的三个功能,问题也解决了。
主要通过添加了一个 let flagScrollTimes: number = 0;
实现,由于我平时用react较多,Vue用的少,不太熟,只做了功能测试,其他测试没有做。希望开发者你在经过测试验证没问题过后,可以在后面的版本中更新发布。。
同时,我也建议将定位到光标处这个选项移除掉,默认就直接开启,不用提供关闭功能,代码中我也没用到plugin.settings.locate_by_cursor
修改后效果如下:
下面是我修改后的 Outline.vue(v0.3.20为基础) 文件,
<template>
<div id="container">
<NConfigProvider :theme="theme" :theme-overrides="theme === null ? lightThemeConfig : darkThemeConfig">
<div class="function-bar" v-if="store.searchSupport">
<NButton size="small" circle @click="toBottom" aria-label="To Bottom">
<template #icon>
<Icon>
<ArrowCircleDownRound :style="iconColor"/>
</Icon>
</template>
</NButton>
<NButton size="small" circle @click="reset" aria-label="Reset">
<template #icon>
<Icon>
<SettingsBackupRestoreRound :style="iconColor"/>
</Icon>
</template>
</NButton>
<NInput v-model:value="pattern" placeholder="Input to search" size="small" clearable/>
</div>
<NSlider v-if="store.levelSwitch" v-model:value="level" :marks="marks" step="mark" :min="0" :max="5"
style="margin:4px 0;" :format-tooltip="formatTooltip"/>
<code v-if="pattern">{{ matchCount }} result(s): </code>
<NTree block-line :pattern="pattern" :data="data2" :on-update:selected-keys="jump"
:render-label="renderMethod" :node-props="setAttrs" :expanded-keys="expanded"
:on-update:expanded-keys="expand" :key="update_tree" :filter="filter"
:show-irrelevant-nodes="!store.hideUnsearched" :class="{ 'ellipsis': store.ellipsis }"
:draggable="store.dragModify" @drop="onDrop"/>
</NConfigProvider>
</div>
</template>
<script setup lang="ts">
import {
computed,
getCurrentInstance,
h,
HTMLAttributes,
nextTick,
onMounted,
onUnmounted,
reactive,
ref,
toRaw,
watch,
watchEffect
} from 'vue';
import {HeadingCache, MarkdownView, sanitizeHTMLToDom} from 'obsidian';
import {
darkTheme,
GlobalThemeOverrides,
NButton,
NConfigProvider,
NInput,
NSlider,
NTree,
TreeDropInfo,
TreeOption
} from 'naive-ui';
import {Icon} from '@vicons/utils';
import {ArrowCircleDownRound, SettingsBackupRestoreRound} from '@vicons/material';
import {marked} from 'marked';
import {formula, highlight, internal_link, nolist, remove_href, remove_ref, renderer, tag} from './parser';
import {store} from './store';
import {QuietOutline} from "./plugin";
let flagScrollTimes: number = 0;
const lightThemeConfig = reactive<GlobalThemeOverrides>({
common: {
primaryColor: "",
primaryColorHover: "",
},
Slider: {
handleSize: "10px",
fillColor: "",
fillColorHover: "",
dotBorderActive: ""
},
Tree: {
nodeTextColor: "var(--nav-item-color)",
},
});
const darkThemeConfig = reactive<GlobalThemeOverrides>({
common: {
primaryColor: "",
primaryColorHover: "",
},
Slider: {
handleSize: "10px",
fillColor: "",
fillColorHover: "",
dotBorderActive: ""
},
Tree: {
nodeTextColor: "var(--nav-item-color)",
}
});
// toggle light/dark theme
let theme: any = computed(() => {
if (store.dark) {
return darkTheme;
}
return null;
});
let iconColor = computed(() => {
if (store.dark) {
return { color: "var(--icon-color)" };
}
return { color: "var(--icon-color)" };
});
function getDefaultColor() {
let button = document.body.createEl("button", { cls: "mod-cta", attr: { style: "width: 0px; height: 0px;" } });
let color = getComputedStyle(button, null).getPropertyValue("background-color");
button.remove();
return color;
}
let locatedColor = ref(getDefaultColor());
watchEffect(() => {
if (store.patchColor) {
lightThemeConfig.common.primaryColor
= lightThemeConfig.common.primaryColorHover
= lightThemeConfig.Slider.fillColor
= lightThemeConfig.Slider.fillColorHover
= store.primaryColorLight;
lightThemeConfig.Slider.dotBorderActive = `2px solid ${store.primaryColorLight}`;
darkThemeConfig.common.primaryColor
= darkThemeConfig.common.primaryColorHover
= darkThemeConfig.Slider.fillColor
= darkThemeConfig.Slider.fillColorHover
= store.primaryColorDark;
darkThemeConfig.Slider.dotBorderActive = `2px solid ${store.primaryColorDark}`;
return;
}
// when css changed, recompute
if (store.cssChange === store.cssChange) {
let color = getDefaultColor();
lightThemeConfig.common.primaryColor
= lightThemeConfig.common.primaryColorHover
= lightThemeConfig.Slider.fillColor
= lightThemeConfig.Slider.fillColorHover
= darkThemeConfig.common.primaryColor
= darkThemeConfig.common.primaryColorHover
= darkThemeConfig.Slider.fillColor
= darkThemeConfig.Slider.fillColorHover
= color;
lightThemeConfig.Slider.dotBorderActive
= darkThemeConfig.Slider.dotBorderActive
= `2px solid ${color}`;
locatedColor.value = color;
}
});
let rainbowColor1 = ref("");
let rainbowColor2 = ref("");
let rainbowColor3 = ref("");
let rainbowColor4 = ref("");
let rainbowColor5 = ref("");
function hexToRGB(hex: string) {
return `${parseInt(hex.slice(1, 3), 16)},`
+ `${parseInt(hex.slice(3, 5), 16)},`
+ `${parseInt(hex.slice(5, 7), 16)}`;
}
watchEffect(() => {
if (store.rainbowLine) {
rainbowColor1.value = `rgba(${hexToRGB(store.rainbowColor1)}, 0.6)`;
rainbowColor2.value = `rgba(${hexToRGB(store.rainbowColor2)}, 0.6)`;
rainbowColor3.value = `rgba(${hexToRGB(store.rainbowColor3)}, 0.6)`;
rainbowColor4.value = `rgba(${hexToRGB(store.rainbowColor4)}, 0.6)`;
rainbowColor5.value = `rgba(${hexToRGB(store.rainbowColor5)}, 0.6)`;
return;
}
if (store.cssChange === store.cssChange) {
rainbowColor1.value
= rainbowColor2.value
= rainbowColor3.value
= rainbowColor4.value
= rainbowColor5.value
= "var(--nav-indentation-guide-color)";
}
});
onMounted(() => {
addEventListener("quiet-outline-reset", reset);
});
onUnmounted(() => {
removeEventListener("quiet-outline-reset", reset);
});
let compomentSelf = getCurrentInstance();
let plugin = compomentSelf.appContext.config.globalProperties.plugin as QuietOutline;
let container = compomentSelf.appContext.config.globalProperties.container as HTMLElement;
// register scroll event
onMounted(() => {
document.addEventListener("scroll", handleScroll, true);
});
onUnmounted(() => {
document.removeEventListener("scroll", handleScroll, true);
});
let toKey = (h: HeadingCache, i: number) => "item-" + h.level + "-" + i;
let fromKey = (key: string) => parseInt((key as string).split('-')[2]);
let handleScroll = debounce(_handleScroll, 100);
function _handleScroll(evt: Event) {
let target = evt.target as HTMLElement;
if (!target.classList.contains("markdown-preview-view") &&
!target.classList.contains("cm-scroller") &&
// fix conflict with outliner
// https://github.com/guopenghui/obsidian-quiet-outline/issues/133
!target.classList.contains("outliner-plugin-list-lines-scroller")) {
return;
}
flagScrollTimes = flagScrollTimes + 1;
// if (plugin.settings.locate_by_cursorflagScrollTimes<=3) {
// if (flagScrollTimes <= 3) {
// return;
// }
onPosChange();
}
function onPosChange() {
let current = currentLine();
let index = nearestHeading(current);
if (index === undefined) return;
autoExpand(index);
resetLocated(index);
}
onMounted(() => {
document.addEventListener("quiet-outline-cursorchange", handleCursorChange);
});
onUnmounted(() => {
document.removeEventListener("quiet-outline-cursorchange", handleCursorChange);
});
function handleCursorChange() {
flagScrollTimes = 0;
onPosChange();
}
function currentLine() {
// const view = plugin.app.workspace.getActiveViewOfType(MarkdownView);
const view = plugin.current_note;
if (!view || view.getViewType() !== "markdown") return;
if (flagScrollTimes <= 3) {
return (view as MarkdownView).editor.getCursor("from").line;
} else {
// @ts-ignore
return view.currentMode.getScroll() + 8;
}
}
function nearestHeading(line: number): undefined | number {
let current_heading = null;
let i = store.headers.length;
while (--i >= 0) {
if (store.headers[i].position.start.line <= line) {
current_heading = store.headers[i];
break;
}
}
if (!current_heading) {
return;
}
return i;
}
function autoExpand(index: number) {
if (plugin.settings.auto_expand) {
let current_heading = store.headers[index];
let should_expand = index < store.headers.length - 1 && store.headers[index].level < store.headers[index + 1].level
? [toKey(current_heading, index)]
: [];
let level = current_heading.level;
let i = index;
while (i-- > 0) {
if (store.headers[i].level < level) {
should_expand.push(toKey(store.headers[i], i));
level = store.headers[i].level;
}
if (level === 1) {
break;
}
}
expanded.value = should_expand;
}
}
function resetLocated(idx: number) {
let path = getPath(idx)
let index = path.find((v) => !expanded.value.contains(toKey(store.headers[v], v)));
index = index === undefined ? path[path.length - 1] : index;
let prevLocation = container.querySelector(".n-tree-node.located");
if (prevLocation) {
prevLocation.removeClass("located");
}
let curLocation = container.querySelector(`#no-${index}`);
if (curLocation) {
curLocation.addClass("located");
curLocation.scrollIntoView({ block: "center", behavior: "smooth" });
} else {
setTimeout(() => {
let curLocation = container.querySelector(`#no-${index}`);
if (curLocation) {
curLocation.addClass("located");
curLocation.scrollIntoView({ block: "center", behavior: "smooth" });
}
}, 100);
}
}
// add html attributes to nodes
interface HTMLAttr extends HTMLAttributes {
"aria-label-position": "top" | "bottom" | "left" | "right";
"raw": string;
}
const setAttrs = computed(() => {
return (info: { option: TreeOption; }): HTMLAttr => {
let lev = parseInt((info.option.key as string).split('-')[1]);
let no = parseInt((info.option.key as string).split('-')[2]);
let raw = info.option.label;
return {
class: `level-${lev}`,
id: `no-${no}`,
"aria-label": store.ellipsis ? info.option.label : "",
"aria-label-position": store.labelDirection,
raw,
};
};
});
// on Mouseover, show popover
let triggerNode: HTMLElement = undefined;
let mouseEvent: MouseEvent = undefined;
let prevShowed = ""
function onMouseEnter(event: MouseEvent) {
let target = event.target as HTMLElement;
let node = target.closest(".n-tree-node") as HTMLElement;
if (!node) {
return;
}
triggerNode = node;
mouseEvent = event;
addEventListener("keydown", openPopover)
}
function onMouseLeave(event: MouseEvent) {
removeEventListener("keydown", openPopover)
}
const funcKeyPressed = (event: KeyboardEvent): boolean => {
return plugin.settings.show_popover_key === "ctrlKey" && event.ctrlKey
|| plugin.settings.show_popover_key === "altKey" && event.altKey
|| plugin.settings.show_popover_key === "metaKey" && event.metaKey
}
function _openPopover(e: KeyboardEvent) {
if (funcKeyPressed(e)) {
plugin.app.workspace.trigger("hover-link", {
event: mouseEvent,
source: "preview",
targetEl: triggerNode,
hoverParent: { hoverPopover: null },
linktext: "#" + triggerNode.getAttribute("raw"),
sourcePath: plugin.current_note.file?.path,
});
}
}
const openPopover = debounce(_openPopover, 100);
// excute on first time, and wait until next fresh
// ... or take a break when node pointed changes
function debounce(func: (_: any) => void, delay: number) {
let fresh = true;
let timeoutId: any;
return function (...args: any) {
const context = this;
let currentLink = triggerNode?.getAttribute("raw");
if (currentLink !== prevShowed || fresh) {
func.apply(context, args);
fresh = false;
prevShowed = currentLink;
return;
}
// 如果已经设置了定时器,清除之前的定时器
if (timeoutId) {
clearTimeout(timeoutId);
}
// 设置新的定时器
timeoutId = setTimeout(() => {
fresh = true;
}, delay);
};
}
onMounted(() => {
container.addEventListener("mouseover", onMouseEnter);
container.addEventListener("mouseout", onMouseLeave);
});
onUnmounted(() => {
container.removeEventListener("mouseover", onMouseEnter);
container.removeEventListener("mouseout", onMouseLeave);
removeEventListener("keydown", openPopover);
});
// switch heading expand levels
let level = ref(parseInt(plugin.settings.expand_level));
let expanded = ref<string[]>([]);
switchLevel(level.value);
function expand(keys: string[], option: TreeOption[]) {
expanded.value = keys;
}
function switchLevel(lev: number) {
expanded.value = store.headers
.map((h, i) => {
return "item-" + h.level + "-" + i;
})
.filter((key, i, arr) => {
const get_level = (k: string): number => parseInt(k.split('-')[1]);
if (i === arr.length - 1) return false;
if (get_level(arr[i]) >= get_level(arr[i + 1])) return false;
return get_level(key) <= lev;
});
}
watch(
level,
(cur, prev) => {
switchLevel(cur);
}
);
// force remake tree
let update_tree = ref(0);
watch(
() => store.leafChange,
() => {
const old_level = level.value;
const old_pattern = pattern.value;
pattern.value = "";
level.value = parseInt(plugin.settings.expand_level);
if (old_level === level.value) {
switchLevel(level.value);
}
nextTick(() => {
pattern.value = old_pattern;
});
}
);
const marks = {
0: "",
1: "",
2: "",
3: "",
4: "",
5: "",
};
function formatTooltip(value: number): string {
let num = store.headers.filter((h) => h.level === value).length;
if (value > 0) {
return `H${value}: ${num}`;
}
return "No expand";
}
// load settings
let renderMethod = computed(() => {
if (store.markdown) {
return renderLabel;
}
return null;
});
// search
let pattern = ref("");
function regexFilter(pattern: string, option: TreeOption): boolean {
let rule = /.*/;
try {
rule = RegExp(pattern, "i");
} catch (e) {
} finally {
return rule.test(option.label);
}
}
function simpleFilter(pattern: string, option: TreeOption): boolean {
return option.label.toLowerCase().contains(pattern.toLowerCase());
}
let filter = computed(() => {
return store.regexSearch ? regexFilter : simpleFilter;
});
let matchCount = computed(() => {
return store.headers.filter((h) => {
let node = { label: h.heading } as TreeOption;
return filter.value(pattern.value, node);
}).length;
});
// click and jump
async function jump(_selected: any, nodes: TreeOption[]): Promise<number> {
if (nodes[0] === undefined) {
return;
}
// flagScrollTimes = 0;
const key_value = (nodes[0].key as string).split("-");
const key = parseInt(key_value[2]);
// const view = store.plugin.app.workspace.getActiveViewOfType(MarkdownView)
// let line: number = store.headers[key].position.start.line;
// const view = plugin.current_note;
// if (view) {
// view.setEphemeralState({ line });
// setTimeout(() => { view.setEphemeralState({ line }); }, 100);
// }
store.jumpBy(plugin, key);
}
// prepare data for tree component
let data2 = computed(() => {
return makeTree(store.headers);
});
function makeTree(headers: HeadingCache[]): TreeOption[] {
let tree: TreeOption[] = arrToTree(headers);
return tree;
}
function arrToTree(headers: HeadingCache[]): TreeOption[] {
const root: TreeOption = { children: [] };
const stack = [{ node: root, level: -1 }];
headers.forEach((h, i) => {
let node: TreeOption = {
label: h.heading,
key: "item-" + h.level + "-" + i,
line: h.position.start.line,
};
while (h.level <= stack.last().level) {
stack.pop();
}
let parent = stack.last().node;
if (parent.children === undefined) {
parent.children = [];
}
parent.children.push(node);
stack.push({ node, level: h.level });
});
return root.children;
}
function getPath(index: number) {
let res: number[] = [];
function pushLastGreatEq(nodes: TreeOption[]) {
if (!nodes || nodes.length === 0) {
return;
}
let idx = 0;
for (let i = nodes.length - 1; i >= 0; i--) {
let pos = fromKey(nodes[i].key as string);
if (pos <= index) {
res.push(pos);
idx = i;
break;
}
}
pushLastGreatEq(nodes[idx].children);
}
pushLastGreatEq(data2.value);
return res;
}
// render markdown
marked.use({ extensions: [formula, internal_link, highlight, tag, remove_ref, nolist] });
marked.use({ walkTokens: remove_href });
marked.use({ renderer });
function renderLabel({ option }: { option: TreeOption; }) {
let result = marked.parse(option.label).trim();
// save mjx elements
let i = 0;
let mjxes = result.match(/<mjx-container.*?>.*?<\/mjx-container>/g);
result = result.replace(/<mjx-container.*?>.*?<\/mjx-container>/g, () => {
return `<math></math>`;
});
result = sanitizeHTMLToDom(`<div>${result}</div>`).children[0].innerHTML;
// restore mjx elements
result = result.replace(/<math.*?>.*?<\/math>/g, () => {
return mjxes[i++];
});
return h("div", { innerHTML: result });
}
// to-bottom button
async function toBottom() {
const file = plugin.app.workspace.getActiveFile();
let lines = (await plugin.app.vault.read(file)).split("\n");
const view = plugin.current_note;
const scroll = () => {
// For some reason, scrolling to last 4 lines gets an error.
view.setEphemeralState({ line: lines.length - 5 });
}
scroll();
setTimeout(scroll, 100);
}
// reset button
function reset() {
pattern.value = "";
level.value = parseInt(plugin.settings.expand_level);
switchLevel(level.value);
}
// drag and drop
async function onDrop({ node, dragNode, dropPosition }: TreeDropInfo) {
// return;
const file = plugin.app.workspace.getActiveFile();
let lines = (await plugin.app.vault.read(file)).split("\n");
let rawExpand = toRaw(expanded.value);
const dragStart = getNo(dragNode);
const dragEnd = dragStart + countTree(dragNode) - 1;
let moveStart = 0, moveEnd = 0;
switch (dropPosition) {
case "inside": {
node = node.children.last();
}
case "after": {
if (dragStart > getNo(node) + countTree(node)) {
moveStart = getNo(node) + countTree(node);
moveEnd = dragStart - 1;
} else {
moveStart = dragEnd + 1;
moveEnd = getNo(node) + countTree(node) - 1;
}
break;
}
case "before": {
if (dragStart > getNo(node)) {
moveStart = getNo(node);
moveEnd = dragStart - 1;
} else {
moveStart = dragStart + countTree(dragNode);
moveEnd = getNo(node) - 1;
}
break;
}
}
const levDelta = getLevel(node) - getLevel(dragNode);
changeExpandKey(rawExpand, dragStart, dragEnd, moveStart, moveEnd, levDelta);
moveSection(
lines,
getLine(dragStart)[0],
getLine(dragEnd)[1] || lines.length - 1,
getLine(moveStart)[0],
getLine(moveEnd)[1] || lines.length - 1,
levDelta
);
plugin.app.vault.modify(file, lines.join("\n"));
}
function getLine(headNo: number) {
return [
store.headers[headNo].position.start.line,
store.headers[headNo + 1]?.position.start.line - 1
];
}
// dls: drag lines start mle: move lines end
function moveSection(lines: string[], dls: number, dle: number, mls: number, mle: number, delta: number) {
let newPos = 0;
if (dls < mls) {
let moved = lines.splice(mls, mle - mls + 1);
lines.splice(dls, 0, ...moved);
newPos = dls + (mle - mls) + 1;
} else {
let moved = lines.splice(dls, dle - dls + 1);
lines.splice(mls, 0, ...moved);
newPos = mls;
}
for (let i = newPos; i <= newPos + (dle - dls); ++i) {
if (lines[i].match(/^#+ /)) {
delta > 0
? lines[i] = Array(delta).fill("#").join("") + lines[i]
: lines[i] = lines[i].slice(-delta);
}
}
}
function changeExpandKey(expanded: string[], ds: number, de: number, ms: number, me: number, delta: number) {
let dNewPos = 0, mNewPos = 0;
if (ds < ms) {
mNewPos = ds;
dNewPos = ds + (me - ms) + 1;
} else {
dNewPos = ms;
mNewPos = ms + (de - ds) + 1;
}
expanded.forEach((key, i) => {
const no = getNo(key);
if (ds <= no && no <= de) {
expanded[i] = `item-${getLevel(key) + delta}-${dNewPos + (no - ds)}`;
}
if (ms <= no && no <= me) {
expanded[i] = `item-${getLevel(key)}-${mNewPos + (no - ms)}`;
}
});
}
function getNo(node: TreeOption | string): number {
if (typeof node !== "string") {
node = node.key as string;
}
return parseInt(node.split("-")[2]);
}
function getLevel(node: TreeOption | string): number {
if (typeof node !== "string") {
node = node.key as string;
}
return parseInt(node.split("-")[1]);
}
function countTree(node: TreeOption): number {
if (!node.children) return 1;
return node.children.reduce((sum, n) => {
return sum + countTree(n);
}, 1);
}
</script>
<style>
/* ============ */
/* 彩虹大纲线 */
/* rainbow line */
/* ============ */
.quiet-outline .n-tree .n-tree-node-indent {
content: "";
height: unset;
align-self: stretch;
}
.quiet-outline .level-2 .n-tree-node-indent,
.quiet-outline .level-3 .n-tree-node-indent:first-child,
.quiet-outline .level-4 .n-tree-node-indent:first-child,
.quiet-outline .level-5 .n-tree-node-indent:first-child,
.quiet-outline .level-6 .n-tree-node-indent:first-child {
border-right: var(--nav-indentation-guide-width) solid v-bind(rainbowColor1);
/* border-right: 2px solid rgb(253, 139, 31, 0.6); */
}
.quiet-outline .level-3 .n-tree-node-indent,
.quiet-outline .level-4 .n-tree-node-indent:nth-child(2),
.quiet-outline .level-5 .n-tree-node-indent:nth-child(2),
.quiet-outline .level-6 .n-tree-node-indent:nth-child(2) {
border-right: var(--nav-indentation-guide-width) solid v-bind(rainbowColor2);
/* border-right: 2px solid rgb(255, 223, 0, 0.6); */
}
.quiet-outline .level-4 .n-tree-node-indent,
.quiet-outline .level-5 .n-tree-node-indent:nth-child(3),
.quiet-outline .level-6 .n-tree-node-indent:nth-child(3) {
border-right: var(--nav-indentation-guide-width) solid v-bind(rainbowColor3);
/* border-right: 2px solid rgb(7, 235, 35, 0.6); */
}
.quiet-outline .level-5 .n-tree-node-indent,
.quiet-outline .level-6 .n-tree-node-indent:nth-child(4) {
border-right: var(--nav-indentation-guide-width) solid v-bind(rainbowColor4);
/* border-right: 2px solid rgb(45, 143, 240, 0.6); */
}
.quiet-outline .level-6 .n-tree-node-indent {
border-right: var(--nav-indentation-guide-width) solid v-bind(rainbowColor5);
/* border-right: 2px solid rgb(188, 1, 226, 0.6); */
}
/* located heading*/
.n-tree-node.located p {
color: v-bind('locatedColor');
}
</style>
@houlang 直接提交一个 PR 吧
@guopenghui 我有一点想法,就是同时使用光标与滚动定位.当光标移动时以光标定位,而页面滚动并且光标不在屏幕上时以页面定位.这样应该可以滚动跟随并且点击标题与移动光标也能正确展开
顺便可以设置判断页面中心的行数.或者更智能一点,往上滚动时以第一行作为定位,往下滚动时以最后一行定位.因为往上滚时肯定是想看上面的内容,当想要的内容刚刚出现在屏幕上的时候就以它们作为定位应该是好的,下滚时也一样
还可以这样,同时展开光标与视窗显示的内容.就是在现在的光标定位的基础上,用一个虚框把当前视窗内容单独框起来并展开,大概就像这样
@houlang 您好,方便提交个PR吗😘
不用了,我已经实现,下次发布时更新。 默认只响应滚动,启用光标设置后,同时响应滚动和光标位置变化,两种操作都能改变激活位置。
但是仍然没有好的办法来获取视口中行数范围
0.3.30
里修改了标题定位的方法,会按照编辑器的中间位置来判断
0.3.30
里修改了标题定位的方法,会按照编辑器的中间位置来判断
已测试,问题已解决,很好,感谢,感谢!,建议大家开启以下两个设置。
版本:v0.3.20 场景:大纲标题可以跟随内容上下自动滚动。 问题:点击标题后大纲激活的标题和点击的不一致!
打开定位到光标选项后,点击大纲标题可以正确显示大纲上激活的标题,但是大纲滚动跟随却又没有了
诉求:在不打开定位到光标功能时,能准确激活被点击的大纲标题不,鱼和熊掌可以兼得不?
这里有个gif动画演示,加载的比较慢..
感觉和内容页标题定位中间有关,如果可以再往上一点,可能就可以解决这个问题。