2021-05-23 16:14:03 +02:00
/ * *
* Attaches ` input ` handlers to markdown rendered tasklist checkboxes in comments .
*
* When a checkbox value changes , the corresponding [ ] or [ x ] in the markdown string
* is set accordingly and sent to the server . On success it updates the raw - content on
* error it resets the checkbox to its original value .
* /
const preventListener = ( e ) => e . preventDefault ( ) ;
export function initMarkupTasklist ( ) {
for ( const el of document . querySelectorAll ( ` .markup[data-can-edit=true] ` ) || [ ] ) {
const container = el . parentNode ;
const checkboxes = el . querySelectorAll ( ` .task-list-item input[type=checkbox] ` ) ;
for ( const checkbox of checkboxes ) {
2021-11-22 09:19:01 +01:00
if ( checkbox . hasAttribute ( 'data-editable' ) ) {
2021-06-14 22:42:58 +02:00
return ;
}
2021-11-22 09:19:01 +01:00
checkbox . setAttribute ( 'data-editable' , 'true' ) ;
2021-05-23 16:14:03 +02:00
checkbox . addEventListener ( 'input' , async ( ) => {
const checkboxCharacter = checkbox . checked ? 'x' : ' ' ;
2021-11-22 09:19:01 +01:00
const position = parseInt ( checkbox . getAttribute ( 'data-source-position' ) ) + 1 ;
2021-05-23 16:14:03 +02:00
const rawContent = container . querySelector ( '.raw-content' ) ;
const oldContent = rawContent . textContent ;
2021-06-14 22:42:58 +02:00
const encoder = new TextEncoder ( ) ;
const buffer = encoder . encode ( oldContent ) ;
buffer . set ( encoder . encode ( checkboxCharacter ) , position ) ;
const newContent = new TextDecoder ( ) . decode ( buffer ) ;
if ( newContent === oldContent ) {
return ;
}
2021-05-23 16:14:03 +02:00
// Prevent further inputs until the request is done. This does not use the
// `disabled` attribute because it causes the border to flash on click.
for ( const checkbox of checkboxes ) {
checkbox . addEventListener ( 'click' , preventListener ) ;
}
try {
const editContentZone = container . querySelector ( '.edit-content-zone' ) ;
2021-11-22 09:19:01 +01:00
const updateUrl = editContentZone . getAttribute ( 'data-update-url' ) ;
const context = editContentZone . getAttribute ( 'data-context' ) ;
2021-05-23 16:14:03 +02:00
await $ . post ( updateUrl , {
2021-08-20 21:26:19 +02:00
ignore _attachments : true ,
2021-10-21 15:37:43 +08:00
_csrf : window . config . csrfToken ,
2021-05-23 16:14:03 +02:00
content : newContent ,
2021-08-20 21:26:19 +02:00
context
2021-05-23 16:14:03 +02:00
} ) ;
rawContent . textContent = newContent ;
} catch ( err ) {
checkbox . checked = ! checkbox . checked ;
console . error ( err ) ;
}
// Enable input on checkboxes again
for ( const checkbox of checkboxes ) {
checkbox . removeEventListener ( 'click' , preventListener ) ;
}
} ) ;
}
// Enable the checkboxes as they are initially disabled by the markdown renderer
for ( const checkbox of checkboxes ) {
checkbox . disabled = false ;
}
}
}