2022-01-28 13:00:11 -08:00
import $ from 'jquery' ;
2021-10-17 01:28:04 +08:00
import { initCompReactionSelector } from './comp/ReactionSelector.js' ;
2021-12-06 04:57:51 +01:00
import { initRepoIssueContentHistory } from './repo-issue-content.js' ;
2023-04-09 21:11:02 +08:00
import { initDiffFileTree } from './repo-diff-filetree.js' ;
2023-07-28 21:18:12 +02:00
import { initDiffCommitSelect } from './repo-diff-commitselect.js' ;
2023-04-03 18:06:57 +08:00
import { validateTextareaNonEmpty } from './comp/ComboMarkdownEditor.js' ;
2023-04-09 21:11:02 +08:00
import { initViewedCheckboxListenerFor , countAndUpdateViewedFiles , initExpandAndCollapseFilesButton } from './pull-view-file.js' ;
2023-07-04 16:36:27 +08:00
import { initImageDiff } from './imagediff.js' ;
2023-11-21 10:12:31 +08:00
import { showErrorToast } from '../modules/toast.js' ;
2023-12-15 07:26:36 +08:00
import { submitEventSubmitter } from '../utils/dom.js' ;
2024-03-15 00:21:14 +02:00
import { POST , GET } from '../modules/fetch.js' ;
2022-01-28 13:00:11 -08:00
2024-03-15 00:21:14 +02:00
const { pageData , i18n } = window . config ;
2021-10-17 01:28:04 +08:00
2023-04-09 21:11:02 +08:00
function initRepoDiffReviewButton ( ) {
2024-03-23 14:37:18 +02:00
const reviewBox = document . getElementById ( 'review-box' ) ;
if ( ! reviewBox ) return ;
const $reviewBox = $ ( reviewBox ) ;
const counter = reviewBox . querySelector ( '.review-comments-counter' ) ;
if ( ! counter ) return ;
2022-05-07 05:35:12 +00:00
2023-03-04 15:13:37 +08:00
$ ( document ) . on ( 'click' , 'button[name="pending_review"]' , ( e ) => {
2022-05-07 05:35:12 +00:00
const $form = $ ( e . target ) . closest ( 'form' ) ;
// Watch for the form's submit event.
$form . on ( 'submit' , ( ) => {
2024-03-23 14:37:18 +02:00
const num = parseInt ( counter . getAttribute ( 'data-pending-comment-number' ) ) + 1 || 1 ;
counter . setAttribute ( 'data-pending-comment-number' , num ) ;
counter . textContent = num ;
2022-05-07 05:35:12 +00:00
// Force the browser to reflow the DOM. This is to ensure that the browser replay the animation
$reviewBox . removeClass ( 'pulse' ) ;
$reviewBox . width ( ) ;
$reviewBox . addClass ( 'pulse' ) ;
} ) ;
2021-10-17 01:28:04 +08:00
} ) ;
}
2023-04-09 21:11:02 +08:00
function initRepoDiffFileViewToggle ( ) {
2021-10-17 01:28:04 +08:00
$ ( '.file-view-toggle' ) . on ( 'click' , function ( ) {
const $this = $ ( this ) ;
$this . parent ( ) . children ( ) . removeClass ( 'active' ) ;
$this . addClass ( 'active' ) ;
2022-01-16 19:19:26 +08:00
const $target = $ ( $this . data ( 'toggle-selector' ) ) ;
2024-03-24 19:23:38 +01:00
$target . parent ( ) . children ( ) . addClass ( 'tw-hidden' ) ;
$target . removeClass ( 'tw-hidden' ) ;
2021-10-17 01:28:04 +08:00
} ) ;
}
2023-04-09 21:11:02 +08:00
function initRepoDiffConversationForm ( ) {
2021-10-17 04:30:31 +08:00
$ ( document ) . on ( 'submit' , '.conversation-holder form' , async ( e ) => {
2021-10-17 01:28:04 +08:00
e . preventDefault ( ) ;
2022-01-02 22:31:03 +00:00
2022-08-16 23:05:40 +02:00
const $form = $ ( e . target ) ;
2024-02-20 12:37:37 +02:00
const textArea = e . target . querySelector ( 'textarea' ) ;
if ( ! validateTextareaNonEmpty ( textArea ) ) {
2022-01-02 22:31:03 +00:00
return ;
}
2023-11-21 10:12:31 +08:00
if ( $form . hasClass ( 'is-loading' ) ) return ;
try {
$form . addClass ( 'is-loading' ) ;
const formData = new FormData ( $form [ 0 ] ) ;
2023-03-04 15:13:37 +08:00
2024-02-17 12:52:11 +01:00
// If the form is submitted by a button, append the button's name and value to the form data.
// originalEvent can be undefined, such as an event that's caused by Ctrl+Enter, in that case
// sent the event itself.
const submitter = submitEventSubmitter ( e . originalEvent ? ? e ) ;
2023-11-21 10:12:31 +08:00
const isSubmittedByButton = ( submitter ? . nodeName === 'BUTTON' ) || ( submitter ? . nodeName === 'INPUT' && submitter . type === 'submit' ) ;
if ( isSubmittedByButton && submitter . name ) {
formData . append ( submitter . name , submitter . value ) ;
}
2024-03-15 00:21:14 +02:00
2024-03-23 14:37:18 +02:00
const response = await POST ( e . target . getAttribute ( 'action' ) , { data : formData } ) ;
2024-03-15 00:21:14 +02:00
const $newConversationHolder = $ ( await response . text ( ) ) ;
2023-11-21 10:12:31 +08:00
const { path , side , idx } = $newConversationHolder . data ( ) ;
2021-10-17 01:28:04 +08:00
2023-11-21 10:12:31 +08:00
$form . closest ( '.conversation-holder' ) . replaceWith ( $newConversationHolder ) ;
if ( $form . closest ( 'tr' ) . data ( 'line-type' ) === 'same' ) {
2024-03-05 06:29:32 +01:00
$ ( ` [data-path=" ${ path } "] .add-code-comment[data-idx=" ${ idx } "] ` ) . addClass ( 'tw-invisible' ) ;
2023-11-21 10:12:31 +08:00
} else {
2024-03-05 06:29:32 +01:00
$ ( ` [data-path=" ${ path } "] .add-code-comment[data-side=" ${ side } "][data-idx=" ${ idx } "] ` ) . addClass ( 'tw-invisible' ) ;
2023-11-21 10:12:31 +08:00
}
$newConversationHolder . find ( '.dropdown' ) . dropdown ( ) ;
initCompReactionSelector ( $newConversationHolder ) ;
} catch { // here the caught error might be a jQuery AJAX error (thrown by await $.post), which is not good to use for error message handling
2024-02-17 12:52:11 +01:00
console . error ( 'error when submitting conversation' , e ) ;
2023-11-21 10:12:31 +08:00
showErrorToast ( i18n . network _error ) ;
} finally {
$form . removeClass ( 'is-loading' ) ;
2021-10-17 01:28:04 +08:00
}
} ) ;
2022-01-19 01:28:38 +08:00
$ ( document ) . on ( 'click' , '.resolve-conversation' , async function ( e ) {
2021-10-17 01:28:04 +08:00
e . preventDefault ( ) ;
const comment _id = $ ( this ) . data ( 'comment-id' ) ;
const origin = $ ( this ) . data ( 'origin' ) ;
const action = $ ( this ) . data ( 'action' ) ;
const url = $ ( this ) . data ( 'update-url' ) ;
2024-03-15 00:21:14 +02:00
try {
const response = await POST ( url , { data : new URLSearchParams ( { origin , action , comment _id } ) } ) ;
const data = await response . text ( ) ;
if ( $ ( this ) . closest ( '.conversation-holder' ) . length ) {
2024-03-16 14:22:16 +02:00
const $conversation = $ ( data ) ;
$ ( this ) . closest ( '.conversation-holder' ) . replaceWith ( $conversation ) ;
$conversation . find ( '.dropdown' ) . dropdown ( ) ;
initCompReactionSelector ( $conversation ) ;
2024-03-15 00:21:14 +02:00
} else {
window . location . reload ( ) ;
}
} catch ( error ) {
console . error ( 'Error:' , error ) ;
2021-10-17 01:28:04 +08:00
}
} ) ;
}
export function initRepoDiffConversationNav ( ) {
// Previous/Next code review conversation
$ ( document ) . on ( 'click' , '.previous-conversation' , ( e ) => {
const $conversation = $ ( e . currentTarget ) . closest ( '.comment-code-cloud' ) ;
2024-03-24 19:23:38 +01:00
const $conversations = $ ( '.comment-code-cloud:not(.tw-hidden)' ) ;
2021-10-17 01:28:04 +08:00
const index = $conversations . index ( $conversation ) ;
const previousIndex = index > 0 ? index - 1 : $conversations . length - 1 ;
const $previousConversation = $conversations . eq ( previousIndex ) ;
2024-03-23 14:37:18 +02:00
const anchor = $previousConversation . find ( '.comment' ) . first ( ) [ 0 ] . getAttribute ( 'id' ) ;
2021-10-17 01:28:04 +08:00
window . location . href = ` # ${ anchor } ` ;
} ) ;
$ ( document ) . on ( 'click' , '.next-conversation' , ( e ) => {
const $conversation = $ ( e . currentTarget ) . closest ( '.comment-code-cloud' ) ;
2024-03-24 19:23:38 +01:00
const $conversations = $ ( '.comment-code-cloud:not(.tw-hidden)' ) ;
2021-10-17 01:28:04 +08:00
const index = $conversations . index ( $conversation ) ;
const nextIndex = index < $conversations . length - 1 ? index + 1 : 0 ;
const $nextConversation = $conversations . eq ( nextIndex ) ;
2024-03-23 14:37:18 +02:00
const anchor = $nextConversation . find ( '.comment' ) . first ( ) [ 0 ] . getAttribute ( 'id' ) ;
2021-10-17 01:28:04 +08:00
window . location . href = ` # ${ anchor } ` ;
} ) ;
}
2021-11-09 17:27:25 +08:00
2022-05-07 20:28:10 +02:00
// Will be called when the show more (files) button has been pressed
function onShowMoreFiles ( ) {
initRepoIssueContentHistory ( ) ;
initViewedCheckboxListenerFor ( ) ;
countAndUpdateViewedFiles ( ) ;
2023-07-04 16:36:27 +08:00
initImageDiff ( ) ;
2022-05-07 20:28:10 +02:00
}
2024-03-15 00:21:14 +02:00
export async function loadMoreFiles ( url ) {
2023-01-13 05:50:32 +00:00
const $target = $ ( 'a#diff-show-more-files' ) ;
2023-05-30 18:53:15 +08:00
if ( $target . hasClass ( 'disabled' ) || pageData . diffFileInfo . isLoadingNewData ) {
2023-01-13 05:50:32 +00:00
return ;
}
2023-05-30 18:53:15 +08:00
pageData . diffFileInfo . isLoadingNewData = true ;
2023-01-13 05:50:32 +00:00
$target . addClass ( 'disabled' ) ;
2024-03-15 00:21:14 +02:00
try {
const response = await GET ( url ) ;
const resp = await response . text ( ) ;
2023-05-30 18:53:15 +08:00
const $resp = $ ( resp ) ;
// the response is a full HTML page, we need to extract the relevant contents:
// 1. append the newly loaded file list items to the existing list
$ ( '#diff-incomplete' ) . replaceWith ( $resp . find ( '#diff-file-boxes' ) . children ( ) ) ;
// 2. re-execute the script to append the newly loaded items to the JS variables to refresh the DiffFileTree
$ ( 'body' ) . append ( $resp . find ( 'script#diff-data-script' ) ) ;
2023-01-13 05:50:32 +00:00
onShowMoreFiles ( ) ;
2024-03-15 00:21:14 +02:00
} catch ( error ) {
console . error ( 'Error:' , error ) ;
showErrorToast ( 'An error occurred while loading more files.' ) ;
} finally {
2023-01-13 05:50:32 +00:00
$target . removeClass ( 'disabled' ) ;
2023-05-30 18:53:15 +08:00
pageData . diffFileInfo . isLoadingNewData = false ;
2024-03-15 00:21:14 +02:00
}
2022-09-27 07:22:19 +02:00
}
2023-04-09 21:11:02 +08:00
function initRepoDiffShowMore ( ) {
2023-01-13 05:50:32 +00:00
$ ( document ) . on ( 'click' , 'a#diff-show-more-files' , ( e ) => {
e . preventDefault ( ) ;
2024-03-23 14:37:18 +02:00
const linkLoadMore = e . target . getAttribute ( 'data-href' ) ;
2023-05-30 18:53:15 +08:00
loadMoreFiles ( linkLoadMore ) ;
2023-01-13 05:50:32 +00:00
} ) ;
2024-03-15 00:21:14 +02:00
$ ( document ) . on ( 'click' , 'a.diff-load-button' , async ( e ) => {
2021-11-21 16:51:08 +00:00
e . preventDefault ( ) ;
const $target = $ ( e . target ) ;
if ( $target . hasClass ( 'disabled' ) ) {
return ;
}
$target . addClass ( 'disabled' ) ;
const url = $target . data ( 'href' ) ;
2024-03-15 00:21:14 +02:00
try {
const response = await GET ( url ) ;
const resp = await response . text ( ) ;
2021-11-21 16:51:08 +00:00
if ( ! resp ) {
return ;
}
$target . parent ( ) . replaceWith ( $ ( resp ) . find ( '#diff-file-boxes .diff-file-body .file-body' ) . children ( ) ) ;
2022-05-07 20:28:10 +02:00
onShowMoreFiles ( ) ;
2024-03-15 00:21:14 +02:00
} catch ( error ) {
console . error ( 'Error:' , error ) ;
} finally {
2021-11-21 16:51:08 +00:00
$target . removeClass ( 'disabled' ) ;
2024-03-15 00:21:14 +02:00
}
2021-11-09 17:27:25 +08:00
} ) ;
}
2023-04-09 21:11:02 +08:00
export function initRepoDiffView ( ) {
2023-04-19 19:05:25 +02:00
initRepoDiffConversationForm ( ) ;
2024-03-25 19:37:55 +01:00
if ( ! $ ( '#diff-file-list' ) . length ) return ;
2023-04-09 21:11:02 +08:00
initDiffFileTree ( ) ;
2023-07-28 21:18:12 +02:00
initDiffCommitSelect ( ) ;
2023-04-09 21:11:02 +08:00
initRepoDiffShowMore ( ) ;
initRepoDiffReviewButton ( ) ;
initRepoDiffFileViewToggle ( ) ;
initViewedCheckboxListenerFor ( ) ;
initExpandAndCollapseFilesButton ( ) ;
}