2023-04-07 08:11:02 +08:00
import $ from 'jquery' ;
import { updateIssuesMeta } from './repo-issue.js' ;
2024-03-16 16:08:10 +01:00
import { toggleElem , hideElem , isElemHidden } from '../utils/dom.js' ;
2023-04-07 08:11:02 +08:00
import { htmlEscape } from 'escape-goat' ;
2023-06-19 15:46:50 +08:00
import { confirmModal } from './comp/ConfirmModal.js' ;
2023-06-27 04:45:24 +02:00
import { showErrorToast } from '../modules/toast.js' ;
2023-07-17 20:06:37 +02:00
import { createSortable } from '../modules/sortable.js' ;
2023-09-19 02:50:30 +02:00
import { DELETE , POST } from '../modules/fetch.js' ;
2024-03-31 01:14:57 +03:00
import { parseDom } from '../utils.js' ;
2023-04-07 08:11:02 +08:00
function initRepoIssueListCheckboxes ( ) {
2024-03-16 16:08:10 +01:00
const issueSelectAll = document . querySelector ( '.issue-checkbox-all' ) ;
2024-03-17 13:50:32 +01:00
if ( ! issueSelectAll ) return ; // logged out state
2024-03-16 16:08:10 +01:00
const issueCheckboxes = document . querySelectorAll ( '.issue-checkbox' ) ;
2023-04-07 08:11:02 +08:00
const syncIssueSelectionState = ( ) => {
2024-03-16 16:08:10 +01:00
const checkedCheckboxes = Array . from ( issueCheckboxes ) . filter ( ( el ) => el . checked ) ;
const anyChecked = Boolean ( checkedCheckboxes . length ) ;
const allChecked = anyChecked && checkedCheckboxes . length === issueCheckboxes . length ;
2023-04-07 08:11:02 +08:00
if ( allChecked ) {
2024-03-16 16:08:10 +01:00
issueSelectAll . checked = true ;
issueSelectAll . indeterminate = false ;
2023-04-07 08:11:02 +08:00
} else if ( anyChecked ) {
2024-03-16 16:08:10 +01:00
issueSelectAll . checked = false ;
issueSelectAll . indeterminate = true ;
2023-04-07 08:11:02 +08:00
} else {
2024-03-16 16:08:10 +01:00
issueSelectAll . checked = false ;
issueSelectAll . indeterminate = false ;
2023-04-07 08:11:02 +08:00
}
// if any issue is selected, show the action panel, otherwise show the filter panel
toggleElem ( $ ( '#issue-filters' ) , ! anyChecked ) ;
toggleElem ( $ ( '#issue-actions' ) , anyChecked ) ;
// there are two panels but only one select-all checkbox, so move the checkbox to the visible panel
2024-03-16 16:08:10 +01:00
const panels = document . querySelectorAll ( '#issue-filters, #issue-actions' ) ;
const visiblePanel = Array . from ( panels ) . find ( ( el ) => ! isElemHidden ( el ) ) ;
const toolbarLeft = visiblePanel . querySelector ( '.issue-list-toolbar-left' ) ;
toolbarLeft . prepend ( issueSelectAll ) ;
2023-04-07 08:11:02 +08:00
} ;
2024-03-16 16:08:10 +01:00
for ( const el of issueCheckboxes ) {
el . addEventListener ( 'change' , syncIssueSelectionState ) ;
}
2023-04-07 08:11:02 +08:00
2024-03-16 16:08:10 +01:00
issueSelectAll . addEventListener ( 'change' , ( ) => {
for ( const el of issueCheckboxes ) {
el . checked = issueSelectAll . checked ;
}
2023-04-07 08:11:02 +08:00
syncIssueSelectionState ( ) ;
} ) ;
$ ( '.issue-action' ) . on ( 'click' , async function ( e ) {
e . preventDefault ( ) ;
2023-06-19 15:46:50 +08:00
const url = this . getAttribute ( 'data-url' ) ;
2023-04-07 08:11:02 +08:00
let action = this . getAttribute ( 'data-action' ) ;
let elementId = this . getAttribute ( 'data-element-id' ) ;
2023-06-19 15:46:50 +08:00
let issueIDs = [ ] ;
for ( const el of document . querySelectorAll ( '.issue-checkbox:checked' ) ) {
issueIDs . push ( el . getAttribute ( 'data-issue-id' ) ) ;
}
issueIDs = issueIDs . join ( ',' ) ;
if ( ! issueIDs ) return ;
// for assignee
if ( elementId === '0' && url . endsWith ( '/assignee' ) ) {
2023-04-07 08:11:02 +08:00
elementId = '' ;
action = 'clear' ;
}
2023-06-19 15:46:50 +08:00
// for toggle
2023-04-07 08:11:02 +08:00
if ( action === 'toggle' && e . altKey ) {
action = 'toggle-alt' ;
}
2023-06-19 15:46:50 +08:00
// for delete
if ( action === 'delete' ) {
const confirmText = e . target . getAttribute ( 'data-action-delete-confirm' ) ;
2024-06-07 21:51:54 +08:00
if ( ! await confirmModal ( confirmText , { confirmButtonColor : 'red' } ) ) {
2023-06-19 15:46:50 +08:00
return ;
}
}
2024-02-16 22:41:23 +01:00
try {
await updateIssuesMeta ( url , action , issueIDs , elementId ) ;
2023-04-07 08:11:02 +08:00
window . location . reload ( ) ;
2024-02-16 22:41:23 +01:00
} catch ( err ) {
showErrorToast ( err . responseJSON ? . error ? ? err . message ) ;
}
2023-04-07 08:11:02 +08:00
} ) ;
}
function initRepoIssueListAuthorDropdown ( ) {
const $searchDropdown = $ ( '.user-remote-search' ) ;
if ( ! $searchDropdown . length ) return ;
2024-03-20 01:39:36 +02:00
let searchUrl = $searchDropdown [ 0 ] . getAttribute ( 'data-search-url' ) ;
const actionJumpUrl = $searchDropdown [ 0 ] . getAttribute ( 'data-action-jump-url' ) ;
const selectedUserId = $searchDropdown [ 0 ] . getAttribute ( 'data-selected-user-id' ) ;
2023-04-07 08:11:02 +08:00
if ( ! searchUrl . includes ( '?' ) ) searchUrl += '?' ;
$searchDropdown . dropdown ( 'setting' , {
fullTextSearch : true ,
selectOnKeydown : false ,
apiSettings : {
cache : false ,
url : ` ${ searchUrl } &q={query} ` ,
onResponse ( resp ) {
// the content is provided by backend IssuePosters handler
const processedResults = [ ] ; // to be used by dropdown to generate menu items
for ( const item of resp . results ) {
2024-03-22 14:45:10 +01:00
let html = ` <img class="ui avatar tw-align-middle" src=" ${ htmlEscape ( item . avatar _link ) } " aria-hidden="true" alt="" width="20" height="20"><span class="gt-ellipsis"> ${ htmlEscape ( item . username ) } </span> ` ;
Migrate margin and padding helpers to tailwind (#30043)
This will conclude the refactor of 1:1 class replacements to tailwind,
except `gt-hidden`. Commands ran:
```bash
perl -p -i -e 's#gt-(p|m)([lrtbxy])?-0#tw-$1$2-0#g' {web_src/js,templates,routers,services}/**/*
perl -p -i -e 's#gt-(p|m)([lrtbxy])?-1#tw-$1$2-0.5#g' {web_src/js,templates,routers,services}/**/*
perl -p -i -e 's#gt-(p|m)([lrtbxy])?-2#tw-$1$2-1#g' {web_src/js,templates,routers,services}/**/*
perl -p -i -e 's#gt-(p|m)([lrtbxy])?-3#tw-$1$2-2#g' {web_src/js,templates,routers,services}/**/*
perl -p -i -e 's#gt-(p|m)([lrtbxy])?-4#tw-$1$2-4#g' {web_src/js,templates,routers,services}/**/*
perl -p -i -e 's#gt-(p|m)([lrtbxy])?-5#tw-$1$2-8#g' {web_src/js,templates,routers,services}/**/*
```
2024-03-24 17:42:49 +01:00
if ( item . full _name ) html += ` <span class="search-fullname tw-ml-2"> ${ htmlEscape ( item . full _name ) } </span> ` ;
2023-04-07 08:11:02 +08:00
processedResults . push ( { value : item . user _id , name : html } ) ;
}
resp . results = processedResults ;
return resp ;
} ,
} ,
action : ( _text , value ) => {
window . location . href = actionJumpUrl . replace ( '{user_id}' , encodeURIComponent ( value ) ) ;
} ,
onShow : ( ) => {
$searchDropdown . dropdown ( 'filter' , ' ' ) ; // trigger a search on first show
} ,
} ) ;
// we want to generate the dropdown menu items by ourselves, replace its internal setup functions
const dropdownSetup = { ... $searchDropdown . dropdown ( 'internal' , 'setup' ) } ;
const dropdownTemplates = $searchDropdown . dropdown ( 'setting' , 'templates' ) ;
$searchDropdown . dropdown ( 'internal' , 'setup' , dropdownSetup ) ;
dropdownSetup . menu = function ( values ) {
2024-03-31 01:14:57 +03:00
const menu = $searchDropdown . find ( '> .menu' ) [ 0 ] ;
// remove old dynamic items
for ( const el of menu . querySelectorAll ( ':scope > .dynamic-item' ) ) {
el . remove ( ) ;
}
2023-04-07 08:11:02 +08:00
const newMenuHtml = dropdownTemplates . menu ( values , $searchDropdown . dropdown ( 'setting' , 'fields' ) , true /* html */ , $searchDropdown . dropdown ( 'setting' , 'className' ) ) ;
if ( newMenuHtml ) {
2024-03-31 01:14:57 +03:00
const newMenuItems = parseDom ( newMenuHtml , 'text/html' ) . querySelectorAll ( 'body > div' ) ;
for ( const newMenuItem of newMenuItems ) {
newMenuItem . classList . add ( 'dynamic-item' ) ;
}
2024-03-16 15:25:27 +02:00
const div = document . createElement ( 'div' ) ;
div . classList . add ( 'divider' , 'dynamic-item' ) ;
2024-03-31 01:14:57 +03:00
menu . append ( div , ... newMenuItems ) ;
2023-04-07 08:11:02 +08:00
}
$searchDropdown . dropdown ( 'refresh' ) ;
// defer our selection to the next tick, because dropdown will set the selection item after this `menu` function
setTimeout ( ( ) => {
2024-03-31 14:27:39 +03:00
for ( const el of menu . querySelectorAll ( '.item.active, .item.selected' ) ) {
el . classList . remove ( 'active' , 'selected' ) ;
}
2024-03-31 01:14:57 +03:00
menu . querySelector ( ` .item[data-value=" ${ selectedUserId } "] ` ) ? . classList . add ( 'selected' ) ;
2023-04-07 08:11:02 +08:00
} , 0 ) ;
} ;
}
2023-05-25 15:17:19 +02:00
function initPinRemoveButton ( ) {
2023-08-12 12:30:28 +02:00
for ( const button of document . getElementsByClassName ( 'issue-card-unpin' ) ) {
2023-05-25 15:17:19 +02:00
button . addEventListener ( 'click' , async ( event ) => {
const el = event . currentTarget ;
const id = Number ( el . getAttribute ( 'data-issue-id' ) ) ;
// Send the unpin request
2023-09-19 02:50:30 +02:00
const response = await DELETE ( el . getAttribute ( 'data-unpin-url' ) ) ;
2023-05-25 15:17:19 +02:00
if ( response . ok ) {
// Delete the tooltip
el . _tippy . destroy ( ) ;
// Remove the Card
2023-08-12 12:30:28 +02:00
el . closest ( ` div.issue-card[data-issue-id=" ${ id } "] ` ) . remove ( ) ;
2023-05-25 15:17:19 +02:00
}
} ) ;
}
}
async function pinMoveEnd ( e ) {
const url = e . item . getAttribute ( 'data-move-url' ) ;
const id = Number ( e . item . getAttribute ( 'data-issue-id' ) ) ;
2023-09-19 02:50:30 +02:00
await POST ( url , { data : { id , position : e . newIndex + 1 } } ) ;
2023-05-25 15:17:19 +02:00
}
2023-07-17 20:06:37 +02:00
async function initIssuePinSort ( ) {
2023-05-25 15:17:19 +02:00
const pinDiv = document . getElementById ( 'issue-pins' ) ;
if ( pinDiv === null ) return ;
// If the User is not a Repo Admin, we don't need to proceed
if ( ! pinDiv . hasAttribute ( 'data-is-repo-admin' ) ) return ;
initPinRemoveButton ( ) ;
// If only one issue pinned, we don't need to make this Sortable
if ( pinDiv . children . length < 2 ) return ;
2023-07-17 20:06:37 +02:00
createSortable ( pinDiv , {
2023-05-25 15:17:19 +02:00
group : 'shared' ,
onEnd : pinMoveEnd ,
} ) ;
}
2023-10-01 18:34:39 +05:30
function initArchivedLabelFilter ( ) {
const archivedLabelEl = document . querySelector ( '#archived-filter-checkbox' ) ;
if ( ! archivedLabelEl ) {
return ;
}
const url = new URL ( window . location . href ) ;
const archivedLabels = document . querySelectorAll ( '[data-is-archived]' ) ;
2023-10-17 19:40:45 +05:30
if ( ! archivedLabels . length ) {
hideElem ( '.archived-label-filter' ) ;
return ;
}
2023-10-01 18:34:39 +05:30
const selectedLabels = ( url . searchParams . get ( 'labels' ) || '' )
. split ( ',' )
. map ( ( id ) => id < 0 ? ` ${ ~ id + 1 } ` : id ) ; // selectedLabels contains -ve ids, which are excluded so convert any -ve value id to +ve
const archivedElToggle = ( ) => {
for ( const label of archivedLabels ) {
const id = label . getAttribute ( 'data-label-id' ) ;
toggleElem ( label , archivedLabelEl . checked || selectedLabels . includes ( id ) ) ;
}
} ;
archivedElToggle ( ) ;
archivedLabelEl . addEventListener ( 'change' , ( ) => {
archivedElToggle ( ) ;
if ( archivedLabelEl . checked ) {
url . searchParams . set ( 'archived' , 'true' ) ;
} else {
url . searchParams . delete ( 'archived' ) ;
}
window . location . href = url . href ;
} ) ;
}
2023-04-07 08:11:02 +08:00
export function initRepoIssueList ( ) {
if ( ! document . querySelectorAll ( '.page-content.repository.issue-list, .page-content.repository.milestone-issue-list' ) . length ) return ;
initRepoIssueListCheckboxes ( ) ;
initRepoIssueListAuthorDropdown ( ) ;
2023-05-25 15:17:19 +02:00
initIssuePinSort ( ) ;
2023-10-01 18:34:39 +05:30
initArchivedLabelFilter ( ) ;
2023-04-07 08:11:02 +08:00
}