-
Notifications
You must be signed in to change notification settings - Fork 4.3k
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
[WIP] Add footnotes support to Gutenberg #6549
Conversation
8c22644
to
ee08de9
Compare
Thanks for workin on this, @Aljullu! Few high-level questions: What are the tradeoffs of this being implemented per-block (paragraph like it's now) or per-component (e.g. for all What's the expected structure of the What other alternatives were considered for representing the list of footnotes? I can see several drawbacks in using a block for that (many operations need to be disabled, might not be future-proof, makes it complicated for other block-level editors to implement the feature), but I am not sure there is a better option. Do we need to touch the parser? Can't we do our job at a higher-level of abstraction, so that we don't pollute the parser code with block-specific logic? |
This looks pretty neat! I think the footnotes of a document should be stored in the meta of post, and not directly in a block. The block would merely provide the visual display of the footnotes, as well as provide the place where you would edit them. (Though a panel in the Document sidebar or perhaps a separate sidebar using the plugin sidebar API to view/edit the footnotes could also be added.) Also, if this is implemented, it should definitely be implemented for at least everything using the RichText component; I can not think of any cases where you would not want the ability to use footnotes in a block using the RichText component. |
Thank you for your comments, I will try to reply all of them:
I'm not sure to understand you here. You mean the parsing should be implemented in RichText or that RichText must be the responsible of updating the list of footnotes in the state?
The texts attribute is used to store the texts attached to each footnote. So every footnote uid is a key and the text associated is the value.
The main reason I created a "footnotes block" was to be able to reorder it and add other blocks below it. I can see several use cases when it might be interesting to add content after the footnotes (for example, Wikipedia articles usually show a "See also" and "External links" section after the "References", something similar might be desirable in Gutenberg, I think). However, it's true implementing it as a block can be problematic. Also, making the block auto-appear and disappear depending on whether the other blocks contain footnotes or not is not straight-forward and means we have to be aware of every block change that might modify the footnotes, like split or remove...
Sure! The function added to parser.js can be easily moved somewhere else. Depending on the way we end up implementing it, it might go to a utils.js file or inside the RichText component if needed. I initially placed it in parser.js because semantically it "parses" the content, but there is no other reason it must be there. @SuperGeniusZeb
Storing them in the meta of the post sounds good to me too, but at the same time I see some disadvantages. Storing them in the block attributes, the footnotes are logically linked to the block, so we get some things for free: when removing a block, for example, the footnotes are removed with it. Also we get the footnotes ordered for free. Storing them in the meta of the post means we have to update the list when some actions are performad like remove, reorder and split blocks.
In that case, I think we should still use RichText for writing the footnotes texts. I imagine people using footnotes might want to use links, italics and possibly break lines. I don't know whether that's possible in the current sidebar text fields and whether it would be easy to implement it.
Yes, it was my goal to implement it in every block that has a RichText. But maybe there is an easier way implementing it directly in RichText so we don't have to update all other blocks? |
I think you answered this in the end of your comment saying: “Yes, it was my goal to implement it in every block that has a RichText. But maybe there is an easier way implementing it directly in RichText so we don't have to update all other blocks?” Ideally we should be able to add footnotes in all RichText components. It’s not a must-have, but it would be cool to have a way for way for developers to add footnotes support for other blocks/components. For example if I am writing a contact form plugin, I may want to allow users to add footnotes to form fields. |
In my case (#6549 (comment)) I saw a lot more than text in the value. Were the React Element objects also supposed to be part of |
I totally agree with your assessment and I wonder what else could we explore here. Some random (and probably bad ideas):
|
core-blocks/paragraph/index.js
Outdated
@@ -258,7 +258,7 @@ class ParagraphBlock extends Component { | |||
setAttributes( { | |||
content: before, | |||
blockFootnotes: parseFootnotesFromContent( before ), | |||
} ); | |||
}, false ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reading this code, it's hard for me to understand the purpose of the second argument, it’s just a lonely false
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also, why wouldn't we want to update the blocks visibility in this specific case?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
100% agree on this, I don't like passing booleans as parameters either. I refactored a bit this code to optimize it and make it easier to understand in 089dd9d.
Adds a button to the toolbar to add footnotes. When clicked, a SUP tag is added the paragraph content. For now, the button is only activated in paragraphs blocks.
Everytime the block is modified, it parses its code and stores the footnotes in the block object.
…attributes update On editor setup create the footnotes block if it doesn't exist and when any block attributes are updated, update the footnotes block with the new parsed footnotes. This commit depends on a followup that creates the footnotes block.
Create Footnotes block which renders the list of footnotes texts when saved and allows editing them in the post editor.
Make footnotes be numbers instead of asterisks and link to the relevant footnote.
In the case a block is inserted or removed from the list of blocks, footnotes might change. Because of that, we must update the list of footnotes in those cases.
Until now, the footnotes block was always rendering OL tags even if there were no footnotes.
…ith the other functions
…rty object That allows us to parse the block texts when updating its attributes intead of doing it when the new attributes are handled by the reducer. This will make it easier to expand the footnotes feature to blocks other than the paragraph block.
…hen footnotes change This allows us to clean a lot of code from the reducer.
… required This allows us to create the block only if footnotes are present, before we were attaching the footnotes block on editor setup.
Otherwise, getFootnotes wasn't returning the footnotes of inner blocks.
Only check if footnotes block must be added or removed when the footnotes of a block change. Until now, the check was made everytime a block changed its attributes or was split.
…ntly added tests A better solution will be needed in order to ignore the key during tests so adding more tests doesn't break the previous ones.
Code continues working as usual but it has been optimized, made simpler and the dependency to getFlattenedBlocks has been removed, so it no longer has to be stored in the utils.
@@ -15,7 +15,7 @@ | |||
"Paragraph ", | |||
{ | |||
"type": "strong", | |||
"key": "_domReact71", | |||
"key": "_domReact73", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure why, but my changes made this test to fail even though they don't seem to be related. I changed the key for now so the tests pass again and will investigate it further in the following days.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think you can drop this information from the tests, along with the ref etc. Have a look at some of the others tests, some of which should not include these properties.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I created an independent PR to deal with this: #7448.
I added some more commits which address some of the topics we discussed above. They also improve performance, fix some bugs and are a general clean up of the code. @nb feel free to take another look when you have a moment! |
55ad109
to
ee0f5ef
Compare
There was a bug that when reordering blocks with footnotes, those were not reordered when saving the post.
core-blocks/footnotes/editor.js
Outdated
}; | ||
} | ||
|
||
componentWillReceiveProps( nextProps ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This lifecycle method will be deprecated soon.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right! Will replace it.
core-blocks/footnotes/editor.js
Outdated
const { footnotesOrder } = this.props; | ||
const nextFootnotesOrder = nextProps.footnotesOrder; | ||
|
||
if ( ! isEqual( footnotesOrder, nextFootnotesOrder ) ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you think we can avoid the deep comparison here?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can't think of any way to avoid this comparison for now.
What we could easily do is to store these two arrays as an array of strings instead of an array of objects (from [ { id: 'abcd' }, ... ]
to [ 'abcd', ... ]
). That will make the comparison a little bit faster.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I refactored the footnotes block so this comparison is no longer necessary (11794cc).
core-blocks/footnotes/editor.js
Outdated
} | ||
|
||
export default withSelect( ( select ) => ( { | ||
footnotesOrder: select( 'core/editor' ).getFootnotes(), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we can improve this name a bit – it doesn't only include the order, but also the footnotes. How about orderedFootnotes
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sounds good.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I ended up renaming it to oderedFootnoteUids
because it only includes the footnote uids (not the footnotes text), so I wanted to distinguish it from footnotes
.
This way we don't have to update the footnotes block attributes everytime the order changes.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice work.
UX feedback:
- It would be nice if the footnote list could link to the block contain the footnote, both in the editor and front-end.
- I can unlink the footnotes and edit the link.
- I cannot add footnotes to e.g. headings.
- The footnote should link to the footnote block.
- Would love too see it in the inserter after Add Inline Images and Inline Blocks API #6959.
- Does the ID have to be so long? Would be nice to have e.g.
#footnote-1
etc.
core-blocks/footnotes/editor.js
Outdated
@@ -0,0 +1,76 @@ | |||
/** |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should this file be edit.js
like the other blocks?
core-blocks/footnotes/style.scss
Outdated
@@ -0,0 +1,9 @@ | |||
.post, | |||
.edit-post-visual-editor { | |||
counter-reset: footnotes; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are spaces here instead of tabs.
@@ -140,6 +144,7 @@ class ParagraphBlock extends Component { | |||
render() { | |||
const { | |||
attributes, | |||
updateFootnotes, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ideally the paragraph block is not aware of the footnotes. This should move to RichText
instead. You should be able to add footnotes in any RichText
area. Also there it would be good to add filters so this could be done by a plugin too (not just footnotes, but adding anything in RichText
areas).
@@ -15,7 +15,7 @@ | |||
"Paragraph ", | |||
{ | |||
"type": "strong", | |||
"key": "_domReact71", | |||
"key": "_domReact73", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think you can drop this information from the tests, along with the ref etc. Have a look at some of the others tests, some of which should not include these properties.
@@ -60,6 +61,7 @@ export class BlockListBlock extends Component { | |||
|
|||
this.setBlockListRef = this.setBlockListRef.bind( this ); | |||
this.bindBlockNode = this.bindBlockNode.bind( this ); | |||
this.updateFootnotes = this.updateFootnotes.bind( this ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also here, ideally there should be no footnote specific logic in this component.
@@ -47,6 +47,11 @@ const FORMATTING_CONTROLS = [ | |||
shortcut: displayShortcut.primary( 'k' ), | |||
format: 'link', | |||
}, | |||
{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like we'll need something like #6642 so this can move to the block registration.
editor/components/rich-text/index.js
Outdated
return; | ||
} | ||
const uid = uuid(); | ||
this.editor.insertContent( `<sup data-wp-footnote-id="${ uid }"><a href="#${ uid }" class="wp-footnote"><span class="screen-reader-text">${ __( 'See footnote' ) }</span></a></sup> ` ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should this be contenteditable=false
? Looks like complex markup that should be protected.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also here we need proper extension points in place. Maybe this could also make use of #6959 when merged.
Hi @iseulde. Thank you for the input! I already fixed a couple of small things now and will take a deeper look to the other comments in the following days. |
Adding contenteditable="false" the user can't edit inside the SUP tag, which makes interacting with footnote numbers much easier because the cursor can't get caught inside its tags.
This pull request appears to have languished and will not be easily reconciled with master. Please feel free to reopen and rebase against the current master, or open a new pull request. |
Why is this tagged as done by automation? The code has never been merged... I'd reopen it, if possible |
Description
This is a WIP Pull Request with an implementation of footnotes for Gutenberg. Issue #1890.
Screenshots
Types of changes
It is a new feature which adds footnotes support to the Gutenberg editor.
User visible changes
It adds a button to the toolbar. When clicked, it inserts a footnote (a superscript number) in the content and the footnotes block displays a
RichText
input area to enter the text for that footnote.(the icon is not definitive)
Internal changes
blockFootnotes
. For now, only the Paragraph block has footnotes implemented.How has this been tested?
The relevant parts of the code are covered by tests.
In order to test it manually: create or edit a post and focus on a paragraph, a new button should appear in the toolbar allowing you to add footnotes. Once clicked, a footnotes block should appear with a 1. and an empty space to write the text for the footnote.
Checklist
data-footnote-id
todata-footnote-wp-id
.Save the footnotes texts as plain HTML: given that we are no longer saving the footnotes texts attribute in the JSON comments, there is no need to convert it to plain HTML. Indeed, all other blocks using RichText save text as an array of nodes.Open bugs
<block-uid>-<footnote-uid>
. Inside the block, they are identified by the footnote id, but in the footnotes block and when saving the post, they are identified with the combination of block uid + footnote id.Next steps
This is a list of 'big' tasks that will require some time to implement/fix:
sup
>a
>span
).That happens because the
footnotesOrder
prop is an array, so when performing the shallow equal comparison oncomponentWillReceiveProps
(https://github.com/WordPress/gutenberg/blob/master/data/index.js#L293) it thinks it's a different prop and setsshouldComponentUpdate
to true even if the array contains the same footnotes. There are several possible solutions to this problem. I will be investigating which is the best one in the following days.