2022-01-29 00:00:11 +03:00
import $ from 'jquery' ;
2021-10-16 20:28:04 +03:00
import { initCompReactionSelector } from './comp/ReactionSelector.js' ;
2021-12-06 06:57:51 +03:00
import { initRepoIssueContentHistory } from './repo-issue-content.js' ;
2023-04-09 16:11:02 +03:00
import { initDiffFileTree } from './repo-diff-filetree.js' ;
2023-07-28 22:18:12 +03:00
import { initDiffCommitSelect } from './repo-diff-commitselect.js' ;
2023-04-03 13:06:57 +03:00
import { validateTextareaNonEmpty } from './comp/ComboMarkdownEditor.js' ;
2023-04-09 16:11:02 +03:00
import { initViewedCheckboxListenerFor , countAndUpdateViewedFiles , initExpandAndCollapseFilesButton } from './pull-view-file.js' ;
2023-07-04 11:36:27 +03:00
import { initImageDiff } from './imagediff.js' ;
2023-11-21 05:12:31 +03:00
import { showErrorToast } from '../modules/toast.js' ;
2023-12-15 02:26:36 +03:00
import { submitEventSubmitter } from '../utils/dom.js' ;
2022-01-29 00:00:11 +03:00
2023-11-21 05:12:31 +03:00
const { csrfToken , pageData , i18n } = window . config ;
2021-10-16 20:28:04 +03:00
2023-04-09 16:11:02 +03:00
function initRepoDiffReviewButton ( ) {
2022-05-07 08:35:12 +03:00
const $reviewBox = $ ( '#review-box' ) ;
const $counter = $reviewBox . find ( '.review-comments-counter' ) ;
2023-03-04 10:13:37 +03:00
$ ( document ) . on ( 'click' , 'button[name="pending_review"]' , ( e ) => {
2022-05-07 08:35:12 +03:00
const $form = $ ( e . target ) . closest ( 'form' ) ;
// Watch for the form's submit event.
$form . on ( 'submit' , ( ) => {
const num = parseInt ( $counter . attr ( 'data-pending-comment-number' ) ) + 1 || 1 ;
$counter . attr ( 'data-pending-comment-number' , num ) ;
$counter . text ( num ) ;
// 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-16 20:28:04 +03:00
} ) ;
}
2023-04-09 16:11:02 +03:00
function initRepoDiffFileViewToggle ( ) {
2021-10-16 20:28:04 +03:00
$ ( '.file-view-toggle' ) . on ( 'click' , function ( ) {
const $this = $ ( this ) ;
$this . parent ( ) . children ( ) . removeClass ( 'active' ) ;
$this . addClass ( 'active' ) ;
2022-01-16 14:19:26 +03:00
const $target = $ ( $this . data ( 'toggle-selector' ) ) ;
2023-02-19 07:06:14 +03:00
$target . parent ( ) . children ( ) . addClass ( 'gt-hidden' ) ;
$target . removeClass ( 'gt-hidden' ) ;
2021-10-16 20:28:04 +03:00
} ) ;
}
2023-04-09 16:11:02 +03:00
function initRepoDiffConversationForm ( ) {
2021-10-16 23:30:31 +03:00
$ ( document ) . on ( 'submit' , '.conversation-holder form' , async ( e ) => {
2021-10-16 20:28:04 +03:00
e . preventDefault ( ) ;
2022-01-03 01:31:03 +03:00
2022-08-17 00:05:40 +03:00
const $form = $ ( e . target ) ;
2024-02-20 13:37:37 +03:00
const textArea = e . target . querySelector ( 'textarea' ) ;
if ( ! validateTextareaNonEmpty ( textArea ) ) {
2022-01-03 01:31:03 +03:00
return ;
}
2023-11-21 05:12:31 +03:00
if ( $form . hasClass ( 'is-loading' ) ) return ;
try {
$form . addClass ( 'is-loading' ) ;
const formData = new FormData ( $form [ 0 ] ) ;
2023-03-04 10:13:37 +03:00
2023-11-21 05:12:31 +03:00
// if the form is submitted by a button, append the button's name and value to the form data
2024-02-17 23:48:10 +03:00
const submitter = submitEventSubmitter ( e ) ;
2023-11-21 05:12:31 +03:00
const isSubmittedByButton = ( submitter ? . nodeName === 'BUTTON' ) || ( submitter ? . nodeName === 'INPUT' && submitter . type === 'submit' ) ;
if ( isSubmittedByButton && submitter . name ) {
formData . append ( submitter . name , submitter . value ) ;
}
const formDataString = String ( new URLSearchParams ( formData ) ) ;
const $newConversationHolder = $ ( await $ . post ( $form . attr ( 'action' ) , formDataString ) ) ;
const { path , side , idx } = $newConversationHolder . data ( ) ;
2021-10-16 20:28:04 +03:00
2023-11-21 05:12:31 +03:00
$form . closest ( '.conversation-holder' ) . replaceWith ( $newConversationHolder ) ;
if ( $form . closest ( 'tr' ) . data ( 'line-type' ) === 'same' ) {
$ ( ` [data-path=" ${ path } "] .add-code-comment[data-idx=" ${ idx } "] ` ) . addClass ( 'gt-invisible' ) ;
} else {
$ ( ` [data-path=" ${ path } "] .add-code-comment[data-side=" ${ side } "][data-idx=" ${ idx } "] ` ) . addClass ( 'gt-invisible' ) ;
}
$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
showErrorToast ( i18n . network _error ) ;
} finally {
$form . removeClass ( 'is-loading' ) ;
2021-10-16 20:28:04 +03:00
}
} ) ;
2022-01-18 20:28:38 +03:00
$ ( document ) . on ( 'click' , '.resolve-conversation' , async function ( e ) {
2021-10-16 20:28:04 +03: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' ) ;
2021-10-21 10:37:43 +03:00
const data = await $ . post ( url , { _csrf : csrfToken , origin , action , comment _id } ) ;
2021-10-16 20:28:04 +03:00
if ( $ ( this ) . closest ( '.conversation-holder' ) . length ) {
const conversation = $ ( data ) ;
$ ( this ) . closest ( '.conversation-holder' ) . replaceWith ( conversation ) ;
conversation . find ( '.dropdown' ) . dropdown ( ) ;
initCompReactionSelector ( conversation ) ;
} else {
window . location . reload ( ) ;
}
} ) ;
}
export function initRepoDiffConversationNav ( ) {
// Previous/Next code review conversation
$ ( document ) . on ( 'click' , '.previous-conversation' , ( e ) => {
const $conversation = $ ( e . currentTarget ) . closest ( '.comment-code-cloud' ) ;
2023-02-19 07:06:14 +03:00
const $conversations = $ ( '.comment-code-cloud:not(.gt-hidden)' ) ;
2021-10-16 20:28:04 +03:00
const index = $conversations . index ( $conversation ) ;
const previousIndex = index > 0 ? index - 1 : $conversations . length - 1 ;
const $previousConversation = $conversations . eq ( previousIndex ) ;
const anchor = $previousConversation . find ( '.comment' ) . first ( ) . attr ( 'id' ) ;
window . location . href = ` # ${ anchor } ` ;
} ) ;
$ ( document ) . on ( 'click' , '.next-conversation' , ( e ) => {
const $conversation = $ ( e . currentTarget ) . closest ( '.comment-code-cloud' ) ;
2023-02-19 07:06:14 +03:00
const $conversations = $ ( '.comment-code-cloud:not(.gt-hidden)' ) ;
2021-10-16 20:28:04 +03:00
const index = $conversations . index ( $conversation ) ;
const nextIndex = index < $conversations . length - 1 ? index + 1 : 0 ;
const $nextConversation = $conversations . eq ( nextIndex ) ;
const anchor = $nextConversation . find ( '.comment' ) . first ( ) . attr ( 'id' ) ;
window . location . href = ` # ${ anchor } ` ;
} ) ;
}
2021-11-09 12:27:25 +03:00
2022-05-07 21:28:10 +03:00
// Will be called when the show more (files) button has been pressed
function onShowMoreFiles ( ) {
initRepoIssueContentHistory ( ) ;
initViewedCheckboxListenerFor ( ) ;
countAndUpdateViewedFiles ( ) ;
2023-07-04 11:36:27 +03:00
initImageDiff ( ) ;
2022-05-07 21:28:10 +03:00
}
2023-05-30 13:53:15 +03:00
export function loadMoreFiles ( url ) {
2023-01-13 08:50:32 +03:00
const $target = $ ( 'a#diff-show-more-files' ) ;
2023-05-30 13:53:15 +03:00
if ( $target . hasClass ( 'disabled' ) || pageData . diffFileInfo . isLoadingNewData ) {
2023-01-13 08:50:32 +03:00
return ;
}
2023-05-30 13:53:15 +03:00
pageData . diffFileInfo . isLoadingNewData = true ;
2023-01-13 08:50:32 +03:00
$target . addClass ( 'disabled' ) ;
2022-09-27 08:22:19 +03:00
$ . ajax ( {
type : 'GET' ,
url ,
} ) . done ( ( resp ) => {
2023-05-30 13:53:15 +03: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 08:50:32 +03:00
onShowMoreFiles ( ) ;
2023-05-30 13:53:15 +03:00
} ) . always ( ( ) => {
2023-01-13 08:50:32 +03:00
$target . removeClass ( 'disabled' ) ;
2023-05-30 13:53:15 +03:00
pageData . diffFileInfo . isLoadingNewData = false ;
2021-11-21 19:51:08 +03:00
} ) ;
2022-09-27 08:22:19 +03:00
}
2023-04-09 16:11:02 +03:00
function initRepoDiffShowMore ( ) {
2023-01-13 08:50:32 +03:00
$ ( document ) . on ( 'click' , 'a#diff-show-more-files' , ( e ) => {
e . preventDefault ( ) ;
const $target = $ ( e . target ) ;
2023-05-30 13:53:15 +03:00
const linkLoadMore = $target . attr ( 'data-href' ) ;
loadMoreFiles ( linkLoadMore ) ;
2023-01-13 08:50:32 +03:00
} ) ;
$ ( document ) . on ( 'click' , 'a.diff-load-button' , ( e ) => {
2021-11-21 19:51:08 +03:00
e . preventDefault ( ) ;
const $target = $ ( e . target ) ;
if ( $target . hasClass ( 'disabled' ) ) {
return ;
}
$target . addClass ( 'disabled' ) ;
const url = $target . data ( 'href' ) ;
$ . ajax ( {
type : 'GET' ,
url ,
} ) . done ( ( resp ) => {
if ( ! resp ) {
$target . removeClass ( 'disabled' ) ;
return ;
}
$target . parent ( ) . replaceWith ( $ ( resp ) . find ( '#diff-file-boxes .diff-file-body .file-body' ) . children ( ) ) ;
2022-05-07 21:28:10 +03:00
onShowMoreFiles ( ) ;
2021-11-21 19:51:08 +03:00
} ) . fail ( ( ) => {
$target . removeClass ( 'disabled' ) ;
2021-11-09 12:27:25 +03:00
} ) ;
} ) ;
}
2023-04-09 16:11:02 +03:00
export function initRepoDiffView ( ) {
2023-04-19 20:05:25 +03:00
initRepoDiffConversationForm ( ) ;
2023-04-09 16:11:02 +03:00
const diffFileList = $ ( '#diff-file-list' ) ;
if ( diffFileList . length === 0 ) return ;
initDiffFileTree ( ) ;
2023-07-28 22:18:12 +03:00
initDiffCommitSelect ( ) ;
2023-04-09 16:11:02 +03:00
initRepoDiffShowMore ( ) ;
initRepoDiffReviewButton ( ) ;
initRepoDiffFileViewToggle ( ) ;
initViewedCheckboxListenerFor ( ) ;
initExpandAndCollapseFilesButton ( ) ;
}