过度设计的锚链接
Overengineered Anchor Links

原始链接: https://thirty-five.com/overengineered-anchoring

锚链接在长页面底部可能因为视窗限制而无法准确跳转到标题,影响用户体验。解决方法多种多样,从简单到复杂都有。 在底部添加填充是一种基本的修复方法,但可能不美观。调整触发线也有缺点,可能会将标题置于视窗底部。虚拟平移触发点可以提供更多控制。比例平移会按比例调整每个标题,但在长页面上可能会造成不必要的移动。 更高级的解决方案涉及自定义映射函数以最大限度地减少与所需触发线的偏差。使用数值优化算法优化标题临近度和章节间距,会导致复杂的结果。最有效的解决方案是使用带有参数“a”的smoothstep函数,仅在页面末尾平滑地调整标题。虽然复杂,但这最大限度地减少了干扰,并改善了长页面的用户体验。

这篇 Hacker News 讨论帖围绕 thirty-five.com 上一篇关于“过度设计的锚链接”的文章展开。评论者们就该博客非常规的设计展开了辩论,包括其右对齐的起始位置和内联左侧弹出式激活。一些人喜欢这种交互式设计,欣赏其在呈现补充信息的同时,采用了一种新颖的方式来补充主文本。 批评意见包括:锚链接依赖 JavaScript,导致键盘无法访问,或者在禁用 JS 的情况下无法使用。用户建议使用标准的 HTML 锚标签以获得更好的可访问性和功能性。由于使用了平滑滚动库 (Lenis.js),人们还担心可能存在滚动劫持问题。此外,一位用户指出了“最终解决方案”(The final solution)这个标题用词不当,因为它带有不幸的历史含义。作者 matser 回应了反馈,考虑移除平滑滚动并解决 UX 问题,包括移动端体验以及一些人认为令人困惑的悬停触发操作。
相关文章

原文

Anchor links are deceptively simple at first glance: click a button, scroll to the heading, and done. But if you ever had to implement them, you might have encountered the . The issue being that, headings towards the bottom of the page can be too far down to scroll them to the desired position. The example component above shows how the ‘conclusion’ heading can never be reached. Surely this will be detrimental to the user experience. We need to come up with a solution. In this blog post, I’ll show some of the solutions I’ve come up with — from a hotfix all the way to unhinged. But before we do that, let’s create a more . Here, we see a viewport moving down the page, with the trigger line being set at 25vh from the top of the viewport. This is what we’ll use to visualize the different solutions.

The most simple solution is to add We calculate the height of the padding by taking the delta between the last heading and the lowest point the anchor trigger can reach. Perfect, right? Well, sometimes the design team is not so fond of random extra padding, so lets keep searching.

Practical: shift the trigger line

Maybe instead of adding extra padding, This is also quite simple to do, we just need to calculate how far from the bottom the last heading is, and put the trigger line there as well. But, this would mean that when the users clicks an anchor tag, the heading could be put all the way at the bottom of the viewport. This is of course not great, since most people put the text they read on the top half of the screen. We need to keep looking.

Good: translate the trigger points

Instead of shifting the trigger line, we could the headings upwards. Instead of using the actual location of the headings as the ones causing the triggers, we create virtual headings and translate them upwards. A virtual heading is not actually visible in the article, its just the position we use to dictate the active state. One might argue that this is pretty much the same as shifting the trigger line, and they’d be right conceptually. However, thinking about translating the trigger points gives us more mental flexibility, as it allows us to consider applying different adjustments based on each heading’s position, which will be crucial later.

The example visualizations now show the location of these ‘virtual headings’. So, while the heading is still at the same place in the article, we visualize where its trigger point is.

In the example, we see one problem arising: the first heading is now too far up. The nice part of this new approach is that we can fix this quite elegantly, since we can shift the individual virtual headings with ease. But what would be a good way to do this?

Great: translate trigger points fractionally

If we think about it, we don’t need to translate all the trigger points. There’s only a few conditions that need to be met:

  1. The headings need to be reachable.
  2. The headings need to stay in order.

We can meet these conditions by translating the trigger points Here, the first heading doesn’t move, and the last heading moves up by the full amount necessary to become reachable. The other headings move up by a proportional amount based on their position between the first and last heading. Now we are getting somewhere! This is a solid solution. You might want to stop here before your product manager starts giving you puzzled looks, wondering how “fixing anchor links” has suddenly turned into a three-week epic.

Awesome: create a custom mapping function

While the fractional solution works, in the sense that our conditions are met, it does have some flaws. We have chosen a trigger line that’s 25% down from the top of the viewport. It would be nice if we can actually minimize the deviation from this ideal line across all headings. The closer the triggers happen to this (mind you — semi-arbitrarily chosen) line, the better the user experience should be. Minimizing deviation feels like a good heuristic. This for sure will make the users happier and result in increased shareholder value.

Let’s minimize the (MSE) of the delta between the headings’ original positions and their virtual positions. We use MSE because it heavily penalizes large deviations, pushing the system towards a state where most virtual headings are close to their original spots, while still satisfying our reachability constraints. Of course, the constraint that headings must stay in order still applies. This results in all points that are reachable staying at their original position. Seems that we have an issue. headings are bunched up at the bottom. This makes sense, since minimization of the mean squared error only cares about proximity to the original position; it has no ‘force’ that opposes this bunching. We need to define something that encourages the virtual trigger points to maintain a certain distance from each other, ideally related to their original spacing. Considering the user experience, we might assume that it’s nice to have the scroll distance needed to activate the next section’s anchor be somewhat proportional to the actual content length of that section. This ‘sections wanting to preserve their relative scroll length’-force is what we’ll use.

Side quest: minimization functions

To explore this idea we need to bust out… Python. Here, we (read: Claude and I) implemented a solver, which is a type of numerical optimization algorithm designed for constrained problems like ours. The core of the optimization lies in a loss function with two competing terms:

  • Anchor penalty: How far a virtual heading is moved from its original location. Minimizing this keeps virtual headings close to their original position. (Lanchor=(yvirtualyoriginal)2L_{anchor} = \sum (y_{virtual} - y_{original})^2
  • Section penalty: How much the size of each virtual section (the space between two virtual headings) differs from the original section size. Minimizing this ensures that sections don’t become disproportionately short or long in terms of scroll distance. (Lsection=((yi+1,virtualyi,virtual)(yi+1,originalyi,original))2L_{section} = \sum ((y_{i+1, virtual} - y_{i, virtual}) - (y_{i+1, original} - y_{i, original}))^2

We combine these into a total loss L=wanchorLanchor+wsectionLsectionL = w_{anchor} L_{anchor} + w_{section} L_{section}

We define constraints to:

  • Keep virtual headings inside the page boundaries.
  • Ensure the first heading doesn’t float upward (its virtual position must be >= its original position).
  • Keep virtual headings in order (yi+1,virtualyi,virtualy_{i+1, virtual} \ge y_{i, virtual}

From this, we generate a plot showing how the virtual headings’ locations change as we vary the weights (specifically, as wsectionw_{section}

Running that code gives us . The circles on the left (at wsection=0w_{section} = 0

Realizations

Staring at that optimization graph sparked a thought. Okay, maybe two thoughts. First, that need to preserve section spacing really kicks in towards the end of the page, where headings get forcibly shoved upwards to stay reachable, squashing the final sections together. Second, let’s consider the behavior of the ‘fractional translation’ method on an edge case.

Imagine, if you will, taking the entire Bible, from the “In the beginning” of Genesis to the final “Amen” of Revelation, and rendering it as one continuous, scrollable webpage. (For the tech bros among us: you could alternatively imagine gluing all of Paul Graham’s essays back-to-back). Now, suppose the very last heading, maybe “Revelation Chapter 22”, is just 200 pixels too low to hit our trigger line when scrolled to.

Does our previous ‘fractional translation’ make sense here? It means taking those 200 pixels of required uplift and meticulously spreading that adjustment across every single heading all the way back to the start. The Ten Commandments get a tiny bump, the Psalms slightly more, all culminating in Revelation 22 getting the full 200px boost.

Actually, if you think about it, with a fractional translation, the error (the distance between the virtual and original headings) grows with the page length. So if the page tends to infinity, so does the error! This would of course be sloppy, and something users could immediately notice as feeling off. So how are we going to fix this?

The final version

This leads to our desired behavior for a smarter mapping function:

  • For headings near the end of the page, apply more adjustment (act like high wsectionw_{section}
  • For headings near the beginning of the page, apply less (or ideally no) adjustment (act like high wanchorw_{anchor}
  • The transition between these states should be smooth.

We need a function that maps a heading’s normalized position x[0,1]x \in [0, 1]

We need this mapping function y=f(x)y = f(x)

  1. It must start at zero: f(0)=0f(0) = 0
  2. It must end at one: f(1)=1f(1) = 1
  3. The transition should start gently: f(0)=0f'(0) = 0
  4. The transition should end gently: f(1)=0f'(1) = 0

It turns out that we can borrow a function from the field of computer graphics to solve this problem. The function is a cubic polynomial that smoothly transitions from 0 to 1 over the range x[0,1]x \in [0, 1]

S(x)=3x22x3S(x) = 3x^2 - 2x^3
联系我们 contact @ memedata.com