Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: convert internal heading links when task is rendered in different file #3357

Merged
merged 10 commits into from
Mar 9, 2025

Conversation

byronluk
Copy link
Contributor

@byronluk byronluk commented Feb 28, 2025

Types of changes

Changes visible to users:

  • New feature (prefix: feat - non-breaking change which adds functionality)
    • Issue/discussion:
    • WARNING: If the link is absent, the PR may be closed without review, due to the workload that such reviews usually require.
  • Breaking change (prefix: feat!! or fix!! - fix or feature that would cause existing functionality to not work as expected)
    • Issue/discussion:
  • Bug fix (prefix: fix - non-breaking change which fixes an issue)
  • Translation (prefix: i18n - additions or improvements to the translations - see Support a new language)
  • Documentation (prefix: docs - improvements to any documentation content for users)
  • Sample vault (prefix: vault - improvements to the Tasks-Demo sample vault)
  • Contributing Guidelines (prefix: contrib - any improvements to documentation content for contributors - see Contributing to Tasks)

Internal changes:

  • Refactor (prefix: refactor - non-breaking change which only improves the design or structure of existing code, and making no changes to its external behaviour)
  • Tests (prefix: test - additions and improvements to unit tests and the smoke tests)
  • Infrastructure (prefix: chore - examples include GitHub Actions, issue templates)

Description

When a task containing an internal heading link (e.g. [[#Header]]) is rendered in a different file than where it was created, the link is now automatically converted to include the source file path (e.g. [[original.md#Header]]).

Changes:

  • Added queryPath parameter to TaskLineRenderer to track source context
  • Updated link conversion logic in renderComponentText
  • Added comprehensive test suite for internal heading link handling
  • Updated InlineRenderer to pass current file path

Motivation and Context

Fixes #3297.

I use inline links frequently with tasks and also run queries on tasks, which results in the broken link.

How has this been tested?

Screenshots (if appropriate)

image

image

Checklist

Terms

…t file

When a task containing an internal heading link (e.g. [[#Header]]) is rendered
in a different file than where it was created, the link is now automatically
converted to include the source file path (e.g. [[original.md#Header]]).

Changes:
- Added queryPath parameter to TaskLineRenderer to track source context
- Updated link conversion logic in renderComponentText
- Added comprehensive test suite for internal heading link handling
- Updated InlineRenderer to pass current file path

Fixes obsidian-tasks-group#3297
@claremacrae claremacrae added the scope: rendering of tasks How the plugin displays tasks, including search results (except CSS issues) label Feb 28, 2025
@claremacrae
Copy link
Collaborator

claremacrae commented Feb 28, 2025

Hi @byronluk, wow, thank you for looking at this - I appreciate it greatly.

And thank you for taking care with the test cases.

Before looking at the code, I did some exploratory testing - I spent about 45 minutes trying to come up with cases to fool it.

Summary of thoughts after exploratory testing

  1. In moderately deeply nested vaults, I think users may grumble at the new rendering of the whole file path in links.
  2. The new code doesn't fix links that have have display/alias text after | inside the link
  3. I don't think using a regular expression is going to be workable, realistically
    • in testing, it is too easy to trick it, and make it change text which looks like a link but is not
    • so it will break display results for users in real-world cases

I don't know how experienced you are with the Obsidian API?
If not familiar, and if you are interested in refining this PR to use Obsidian's parsing of links, I think we should probably have a chat before you start on it... My email address is in my profile here, if you would like to set up a virtual meeting some time soon....

Cases not tested

  • I haven't yet test []() style links - only [[]]

How I tested

These are the files I used:

3297 - Search.md

# 3297 - links in descriptions - Search

```tasks
folder includes {{query.file.folder}}

sort by function task.lineNumber
group by heading

hide backlinks
```

3297 - Tasks.md

# 3297 - links in descriptions - Tasks

## 1 Works before and after the PR

- [ ] #task ✅ Simple case - single link to current file name and heading <br>[[3297 - Tasks#1 Works before and after the PR]]

## 2 Only works after the PR

... but should it display the whole path?

I believe that the Tasks plugin only puts the full path in to its "backlinks" if the filename isn't unique in the vault.

Should this code do the same?

- [ ] #task ✅ Simple case - single relative link to heading in current file <br>[[#2 Only works after the PR]]
- [ ] #task ✅ Simple case - two relative link to heading in current file - both are updated <br>[[#2 Only works after the PR]]<br>[[#3297 - links in descriptions - Tasks]]

## 3 What I think the new behaviour should possibly look like

- [ ] #task ✅ Simple case - single  link <br>[[3297 - Tasks#3 What I think the new behaviour should possibly look like|#3 What I think the new behaviour should possibly look like]]

## 4 Does not work before or after PR

It appears that the regular expression does not allow for text like `|I am an alias`  to set display text of a link.

- [ ] #task ❌ Simple case - single relative link to heading in current file - with alias <br>[[#4 Does not work before or after PR|I am an alias]]

## 5 Should not be modified by PR

The regular expression is modifying that Obsidian does not render as a link in Reading mode, in Restricted mode.

- [ ] #task ❌ Formatted text looks like a link but should not be modified `[[#5 Should not be modified by PR]]`
- [ ] #task ❌ Escaped `[` should not be modified \[[#5 Should not be modified by PR]]

We don't want to be writing Markdown-parsing code in Regexes.

Instead we should be using the [LinkCache](https://docs.obsidian.md/Reference/TypeScript+API/LinkCache).

There are some examples of the LinkCache context in Tasks test data: [link_in_task_wikilink.json](https://github.com/obsidian-tasks-group/obsidian-tasks/blob/c6cb8ddc5491d36e48b183b1c38786be09504103/tests/Obsidian/__test_data__/link_in_task_wikilink.json#L21-L56).

Here is what the query looks like, running the code in the PR:

image

@claremacrae
Copy link
Collaborator

It would also be worth testing rendering of nested list items that contain links to headings in their own file - and when their parent task line is shown in search results that contain show tree.

@claremacrae
Copy link
Collaborator

It would also be worth testing rendering of nested list items that contain links to headings in their own file - and when their parent task line is shown in search results that contain show tree.

At the moment list item and task descriptions are drawn by different code - @ilandikov and I are soon going to start unifying those two bodies of code.

So it would be fine for this PR to focus just on fixing task lines - and then we can later make sure that any fix is also applied to list items.

@ilandikov
Copy link
Collaborator

@byronluk many thanks for looking at this and spending time for this PR. I left a couple of suggestions on code level.

@claremacrae it would be nice to refactor this is in a way to make this change easyER (I don't find it not_easy). Maybe we can discuss this on our next session before refactoring the description rendering you mentioned before. In any case I agree with you that this could be merged without the latter.

@claremacrae
Copy link
Collaborator

@byronluk many thanks for looking at this and spending time for this PR. I left a couple of suggestions on code level.

Thanks...

@claremacrae it would be nice to refactor this is in a way to make this change easyER (I don't find it not_easy). Maybe we can discuss this on our next session before refactoring the description rendering you mentioned before. In any case I agree with you that this could be merged without the latter.

Where I haven't already paired with contributors, I don't request the 'make the change easy/make the easy change' approach, as the technique is just not sufficiently well known, and most people don't use refactoring IDEs either... It's a skill that takes quite some time to learn...

So I focus on small PRs, and ensuring good exploratory testing and agreeing the desired behaviour before giving feedback on code.

@byronluk
Copy link
Contributor Author

byronluk commented Mar 1, 2025

Thanks for the detailed feedback @claremacrae, @ilandikov! I agreed, I'm not a huge fan of the extra parameter I introduced either 😅. I'm not very familiar with the codebase and wanted to timebox, so figured it'd be best to put a PR out where we could anchor our discussion around.

I don't know how experienced you are with the Obsidian API?
If not familiar, and if you are interested in refining this PR to use Obsidian's parsing of links, I think we should probably have a chat before you start on it... My email address is in my profile here, if you would like to set up a virtual meeting some time soon....

This sounds like a great idea! I'll poke around and see if we can resolve this async, as I'm quite busy with work and my 1 year old haha..

I'll take a look at:

  1. Finding a better solution that doesn't involve using regex (will look into using Obsidian's parsing of links like @claremacrae mentioned)
  2. Avoidng the extra parameter introduced into the constructor,
  3. The thorough test cases provided.

Will keep you all updated on my progress here! Probably should be able to get around to this within the next week or so.

@claremacrae
Copy link
Collaborator

claremacrae commented Mar 1, 2025

Super - thanks very much @byronluk.

Given your time constraints, I suggest you don’t worry about the parameter, though. We can always refactor the code easily after the PR is merged.

@claremacrae
Copy link
Collaborator

PS @byronluk ...

I like your mentioning of time-boxing...

Feel free to time-box your next investigations, and then send me an email to have a chat.

Because of my familiarity with the code, I can see the next few steps on implementation and testing and it will save you hours, whilst still leaving you the actual problem to fix. :-)

@claremacrae
Copy link
Collaborator

Plan B: If you really, really don't have time for a chat, let me know if you are happy for me to push some demo code to your PR branch instead - and I'll explain things in comments instead.

It would be lower band-width communication, but might fit in better with your schedule...

And could be followed by a later conversation, too, if need be...

byronluk added 2 commits March 5, 2025 09:24
…erer

This commit:

1. Adds a new processTaskLinks method to QueryResultsRenderer that correctly processes internal heading links
2. Removes the link handling from TaskLineRenderer (removing queryPath parameter and related code)
3. Ensures links are rendered with proper display text format using the '|' separator
4. Adds comprehensive test cases for various heading link scenarios
5. Adds test data for internal heading links
6. Properly handles different link formats including links with spaces, dashes, underscores, and aliases

This improves how internal heading links display in query results by keeping the proper link text formatting while ensuring correct file references.
@byronluk
Copy link
Contributor Author

byronluk commented Mar 5, 2025

@claremacrae, @ilandikov I took another pass at the change using the linkCache. Would appreciate your feedback!

@byronluk
Copy link
Contributor Author

byronluk commented Mar 5, 2025

I went through the test cases you provided @claremacrae and this change should cover them all correctly

Copy link
Collaborator

@claremacrae claremacrae left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wow - I'm super impressed at how much you have figured out about our testing mechanism.

This is just a quick initial comment...

@byronluk byronluk force-pushed the byronluk/inline-links branch from ee1133a to 604d054 Compare March 6, 2025 13:15
Copy link
Collaborator

@claremacrae claremacrae left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wow, this is a huge improvement. 🎉 ❤️

I have approved this, as if you wanted to put it down now, I think with just fixing probably wrong test cases, it is good enough to merge and release - perhaps logging the problem cases I pointed out as bugs, to possible be addressed in future...

@byronluk
Copy link
Contributor Author

byronluk commented Mar 7, 2025

I'll take another pass through when I have some time tomorrow! Will fix up the simple mistakes (sorry about the messy test cases!) and think about the positioning problem.

Thank you for the thorough review!

@claremacrae
Copy link
Collaborator

I'll take another pass through when I have some time tomorrow! Will fix up the simple mistakes (sorry about the messy test cases!) and think about the positioning problem.

Thank you for the thorough review!

Thank you for this reply, but....

tl;dr I have been thinking about it quite a lot, and I actively do not want you to adjust this code to address the positioning problem! 😄

Rather than having the rendering code try to find links, I feel the the right-or-sustainable answer is to make links first-class objects in the Tasks code, and to use the stored link positions in the rendering.

So, please do not spend your time making the current PR use Obsidian's link-location data.

As well as making the current code a lot more complicated, for what would be an interim measure, the writing of tests for it would be nightmarishly complicated. Here are some of the scenarios to consider:

  • No Global Filter:
  • Global filter set:
    • and hidden in search results
    • and shown in search results
    • in the middle of the line
    • at the end of the line
  • Extra whitespace
    • at beginning of line
    • end of line
  • Position of links
    • After signifiers (invalid task line, but still useful test)
  • etc etc

Let's instead focus on getting the current implementation ready to merge (fix the misleading comment, improve the tests)...

- Enhance link processing to only transform internal heading links (starting with #)

- Implement position-based link replacement to correctly handle multiple occurrences

- Add test cases for mixed formatting scenarios (code blocks, HTML tags)

- Reorganize test data structure and simplify test approach

- Use snapshot testing for more precise verification
@byronluk
Copy link
Contributor Author

byronluk commented Mar 8, 2025

Rather than having the rendering code try to find links, I feel the the right-or-sustainable answer is to make links first-class objects in the Tasks code, and to use the stored link positions in the rendering.

I agree with this :) I did fix up my solution before seeing your comment and I do think it is a fairly reasonable approach with minimal added complexity- I'll push it up and see what you think and happy to revert back to the current approach we have here if you'd like.

Copy link

sonarqubecloud bot commented Mar 9, 2025

Copy link
Collaborator

@claremacrae claremacrae left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Awesome work - huge thanks @byronluk. 🎉 😄

@claremacrae claremacrae merged commit 041da5d into obsidian-tasks-group:main Mar 9, 2025
3 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
scope: rendering of tasks How the plugin displays tasks, including search results (except CSS issues)
Projects
None yet
Development

Successfully merging this pull request may close these issues.

File-internal Link to Heading assigned file containing query instead of file containing task
3 participants