@ -30,6 +30,7 @@ import { Commands, ContextKeys, CoreGitCommands } from '../../../constants';
import type { Container } from '../../../container' ;
import type { Container } from '../../../container' ;
import { getContext , onDidChangeContext , setContext } from '../../../context' ;
import { getContext , onDidChangeContext , setContext } from '../../../context' ;
import { PlusFeatures } from '../../../features' ;
import { PlusFeatures } from '../../../features' ;
import { GitSearchError } from '../../../git/errors' ;
import type { GitCommit } from '../../../git/models/commit' ;
import type { GitCommit } from '../../../git/models/commit' ;
import { GitGraphRowType } from '../../../git/models/graph' ;
import { GitGraphRowType } from '../../../git/models/graph' ;
import type { GitGraph } from '../../../git/models/graph' ;
import type { GitGraph } from '../../../git/models/graph' ;
@ -40,9 +41,12 @@ import type {
GitTagReference ,
GitTagReference ,
} from '../../../git/models/reference' ;
} from '../../../git/models/reference' ;
import { GitReference , GitRevision } from '../../../git/models/reference' ;
import { GitReference , GitRevision } from '../../../git/models/reference' ;
import type { Repository , RepositoryChangeEvent , RepositoryFileSystemChangeEvent } from '../../../git/models/repository' ;
import type {
Repository ,
RepositoryChangeEvent ,
RepositoryFileSystemChangeEvent ,
} from '../../../git/models/repository' ;
import { RepositoryChange , RepositoryChangeComparisonMode } from '../../../git/models/repository' ;
import { RepositoryChange , RepositoryChangeComparisonMode } from '../../../git/models/repository' ;
import type { GitStatus } from '../../../git/models/status' ;
import type { GitSearch } from '../../../git/search' ;
import type { GitSearch } from '../../../git/search' ;
import { getSearchQueryComparisonKey } from '../../../git/search' ;
import { getSearchQueryComparisonKey } from '../../../git/search' ;
import { executeActionCommand , executeCommand , executeCoreGitCommand , registerCommand } from '../../../system/command' ;
import { executeActionCommand , executeCommand , executeCoreGitCommand , registerCommand } from '../../../system/command' ;
@ -50,8 +54,9 @@ import { gate } from '../../../system/decorators/gate';
import { debug } from '../../../system/decorators/log' ;
import { debug } from '../../../system/decorators/log' ;
import type { Deferrable } from '../../../system/function' ;
import type { Deferrable } from '../../../system/function' ;
import { debounce , once } from '../../../system/function' ;
import { debounce , once } from '../../../system/function' ;
import { first , last } from '../../../system/iterable' ;
import { last } from '../../../system/iterable' ;
import { updateRecordValue } from '../../../system/object' ;
import { updateRecordValue } from '../../../system/object' ;
import { getSettledValue } from '../../../system/promise' ;
import { isDarkTheme , isLightTheme } from '../../../system/utils' ;
import { isDarkTheme , isLightTheme } from '../../../system/utils' ;
import type { WebviewItemContext } from '../../../system/webview' ;
import type { WebviewItemContext } from '../../../system/webview' ;
import { isWebviewItemContext , serializeWebviewItemContext } from '../../../system/webview' ;
import { isWebviewItemContext , serializeWebviewItemContext } from '../../../system/webview' ;
@ -68,17 +73,18 @@ import type { SubscriptionChangeEvent } from '../../subscription/subscriptionSer
import { arePlusFeaturesEnabled , ensurePlusFeaturesEnabled } from '../../subscription/utils' ;
import { arePlusFeaturesEnabled , ensurePlusFeaturesEnabled } from '../../subscription/utils' ;
import type {
import type {
DismissBannerParams ,
DismissBannerParams ,
EnsureCommit Params ,
EnsureRow Params ,
GetMissingAvatarsParams ,
GetMissingAvatarsParams ,
GetMoreCommit sParams ,
GetMoreRow sParams ,
GraphColumnConfig ,
GraphColumnConfig ,
GraphColumnName ,
GraphColumnName ,
GraphColumnsSettings ,
GraphColumnsSettings ,
GraphComponentConfig ,
GraphComponentConfig ,
GraphRepository ,
GraphRepository ,
GraphWorkDirStat s ,
SearchCommitsParam s,
GraphSelectedRow s ,
GraphWorkingTreeStat s,
SearchOpenInViewParams ,
SearchOpenInViewParams ,
SearchParams ,
State ,
State ,
UpdateColumnParams ,
UpdateColumnParams ,
UpdateSelectedRepositoryParams ,
UpdateSelectedRepositoryParams ,
@ -87,19 +93,19 @@ import type {
import {
import {
DidChangeAvatarsNotificationType ,
DidChangeAvatarsNotificationType ,
DidChangeColumnsNotificationType ,
DidChangeColumnsNotificationType ,
DidChangeCommitsNotificationType ,
DidChangeGraphConfigurationNotificationType ,
DidChangeGraphConfigurationNotificationType ,
DidChangeNotificationType ,
DidChangeNotificationType ,
DidChangeRowsNotificationType ,
DidChangeSelectionNotificationType ,
DidChangeSelectionNotificationType ,
DidChangeSubscriptionNotificationType ,
DidChangeSubscriptionNotificationType ,
DidChangeWorkDirStats NotificationType ,
DidEnsureCommit NotificationType ,
DidSearchCommits NotificationType ,
DidChangeWorkingTree NotificationType ,
DidEnsureRow NotificationType ,
DidSearchNotificationType ,
DismissBannerCommandType ,
DismissBannerCommandType ,
EnsureCommit CommandType ,
EnsureRow CommandType ,
GetMissingAvatarsCommandType ,
GetMissingAvatarsCommandType ,
GetMoreCommit sCommandType ,
SearchCommitsComm andType ,
GetMoreRow sCommandType ,
SearchCommandType ,
SearchOpenInViewCommandType ,
SearchOpenInViewCommandType ,
UpdateColumnCommandType ,
UpdateColumnCommandType ,
UpdateSelectedRepositoryCommandType ,
UpdateSelectedRepositoryCommandType ,
@ -150,7 +156,9 @@ export class GraphWebview extends WebviewBase {
) ;
) ;
}
}
this . updateState ( ) ;
if ( this . isReady ) {
this . updateState ( ) ;
}
}
}
private _selection : readonly GitCommit [ ] | undefined ;
private _selection : readonly GitCommit [ ] | undefined ;
@ -164,8 +172,8 @@ export class GraphWebview extends WebviewBase {
private _pendingIpcNotifications = new Map < IpcNotificationType , IpcMessage | ( ( ) = > Promise < boolean > ) > ( ) ;
private _pendingIpcNotifications = new Map < IpcNotificationType , IpcMessage | ( ( ) = > Promise < boolean > ) > ( ) ;
private _search : GitSearch | undefined ;
private _search : GitSearch | undefined ;
private _searchCancellation : CancellationTokenSource | undefined ;
private _searchCancellation : CancellationTokenSource | undefined ;
private _selectedSha ? : string ;
private _selectedRows : { [ sha : string ] : true } = { } ;
private _selectedId ? : string ;
private _selectedRows : GraphSelectedRows | undefined ;
private _repositoryEventsDisposable : Disposable | undefined ;
private _repositoryEventsDisposable : Disposable | undefined ;
private _statusBarItem : StatusBarItem | undefined ;
private _statusBarItem : StatusBarItem | undefined ;
@ -198,26 +206,26 @@ export class GraphWebview extends WebviewBase {
args : ShowInCommitGraphCommandArgs | BranchNode | CommitNode | CommitFileNode | StashNode | TagNode ,
args : ShowInCommitGraphCommandArgs | BranchNode | CommitNode | CommitFileNode | StashNode | TagNode ,
) = > {
) = > {
this . repository = this . container . git . getRepository ( args . ref . repoPath ) ;
this . repository = this . container . git . getRepository ( args . ref . repoPath ) ;
let sha = args . ref . ref ;
if ( ! GitRevision . isSha ( sha ) ) {
sha = await this . container . git . resolveReference ( args . ref . repoPath , sha , undefined , {
let id = args . ref . ref ;
if ( ! GitRevision . isSha ( id ) ) {
id = await this . container . git . resolveReference ( args . ref . repoPath , id , undefined , {
force : true ,
force : true ,
} ) ;
} ) ;
}
}
this . setSelectedRows ( sha ) ;
this . setSelectedRows ( id ) ;
const preserveFocus = 'preserveFocus' in args ? args . preserveFocus ? ? false : false ;
const preserveFocus = 'preserveFocus' in args ? args . preserveFocus ? ? false : false ;
if ( this . _panel == null ) {
if ( this . _panel == null ) {
void this . show ( { preserveFocus : preserveFocus } ) ;
void this . show ( { preserveFocus : preserveFocus } ) ;
} else {
} else {
this . _panel . reveal ( this . _panel . viewColumn ? ? ViewColumn . Active , preserveFocus ? ? false ) ;
this . _panel . reveal ( this . _panel . viewColumn ? ? ViewColumn . Active , preserveFocus ? ? false ) ;
if ( this . _graph ? . ids . has ( sha ) ) {
if ( this . _graph ? . ids . has ( id ) ) {
void this . notifyDidChangeSelection ( ) ;
void this . notifyDidChangeSelection ( ) ;
return ;
return ;
}
}
this . setSelectedRows ( sha ) ;
void this . onGetMoreCommits ( { sha : sha } ) ;
this . setSelectedRows ( id ) ;
void this . onGetMoreRows ( { id : id } , true ) ;
}
}
} ,
} ,
) ,
) ,
@ -259,6 +267,9 @@ export class GraphWebview extends WebviewBase {
protected override refresh ( force? : boolean ) : Promise < void > {
protected override refresh ( force? : boolean ) : Promise < void > {
this . resetRepositoryState ( ) ;
this . resetRepositoryState ( ) ;
if ( force ) {
this . _pendingIpcNotifications . clear ( ) ;
}
return super . refresh ( force ) ;
return super . refresh ( force ) ;
}
}
@ -332,17 +343,17 @@ export class GraphWebview extends WebviewBase {
case DismissBannerCommandType . method :
case DismissBannerCommandType . method :
onIpc ( DismissBannerCommandType , e , params = > this . dismissBanner ( params ) ) ;
onIpc ( DismissBannerCommandType , e , params = > this . dismissBanner ( params ) ) ;
break ;
break ;
case EnsureCommit CommandType . method :
onIpc ( EnsureCommit CommandType , e , params = > this . onEnsureCommit ( params , e . completionId ) ) ;
case EnsureRow CommandType . method :
onIpc ( EnsureRow CommandType , e , params = > this . onEnsureRow ( params , e . completionId ) ) ;
break ;
break ;
case GetMissingAvatarsCommandType . method :
case GetMissingAvatarsCommandType . method :
onIpc ( GetMissingAvatarsCommandType , e , params = > this . onGetMissingAvatars ( params ) ) ;
onIpc ( GetMissingAvatarsCommandType , e , params = > this . onGetMissingAvatars ( params ) ) ;
break ;
break ;
case GetMoreCommit sCommandType . method :
onIpc ( GetMoreCommit sCommandType , e , params = > this . onGetMoreCommit s ( params ) ) ;
case GetMoreRow sCommandType . method :
onIpc ( GetMoreRow sCommandType , e , params = > this . onGetMoreRow s ( params ) ) ;
break ;
break ;
case SearchCommitsComm andType . method :
onIpc ( SearchCommitsComm andType , e , params = > this . onSearchCommits ( params , e . completionId ) ) ;
case SearchCommandType . method :
onIpc ( SearchCommandType , e , params = > this . onSearch ( params , e . completionId ) ) ;
break ;
break ;
case SearchOpenInViewCommandType . method :
case SearchOpenInViewCommandType . method :
onIpc ( SearchOpenInViewCommandType , e , params = > this . onSearchOpenInView ( params ) ) ;
onIpc ( SearchOpenInViewCommandType , e , params = > this . onSearchOpenInView ( params ) ) ;
@ -354,7 +365,7 @@ export class GraphWebview extends WebviewBase {
onIpc ( UpdateSelectedRepositoryCommandType , e , params = > this . onRepositorySelectionChanged ( params ) ) ;
onIpc ( UpdateSelectedRepositoryCommandType , e , params = > this . onRepositorySelectionChanged ( params ) ) ;
break ;
break ;
case UpdateSelectionCommandType . method :
case UpdateSelectionCommandType . method :
onIpc ( UpdateSelectionCommandType , e , debounce ( this . onSelectionChanged . bind ( this ) , 100 ) ) ;
onIpc ( UpdateSelectionCommandType , e , this . onSelectionChanged . bind ( this ) ) ;
break ;
break ;
}
}
}
}
@ -380,7 +391,9 @@ export class GraphWebview extends WebviewBase {
return ;
return ;
}
}
this . sendPendingIpcNotifications ( ) ;
if ( this . isReady && visible ) {
this . sendPendingIpcNotifications ( ) ;
}
}
}
private onConfigurationChanged ( e : ConfigurationChangeEvent ) {
private onConfigurationChanged ( e : ConfigurationChangeEvent ) {
@ -433,6 +446,11 @@ export class GraphWebview extends WebviewBase {
this . updateState ( ) ;
this . updateState ( ) ;
}
}
private onRepositoryFileSystemChanged ( e : RepositoryFileSystemChangeEvent ) {
if ( e . repository ? . path !== this . repository ? . path ) return ;
void this . notifyDidChangeWorkingTree ( ) ;
}
private onSubscriptionChanged ( e : SubscriptionChangeEvent ) {
private onSubscriptionChanged ( e : SubscriptionChangeEvent ) {
if ( e . etag === this . _etagSubscription ) return ;
if ( e . etag === this . _etagSubscription ) return ;
@ -472,30 +490,30 @@ export class GraphWebview extends WebviewBase {
}
}
@debug ( )
@debug ( )
private async onEnsureCommit ( e : EnsureCommit Params , completionId? : string ) {
private async onEnsureRow ( e : EnsureRow Params , completionId? : string ) {
if ( this . _graph ? . more == null || this . _repository ? . etag !== this . _etagRepository ) {
if ( this . _graph ? . more == null || this . _repository ? . etag !== this . _etagRepository ) {
this . updateState ( true ) ;
this . updateState ( true ) ;
if ( completionId != null ) {
if ( completionId != null ) {
void this . notify ( DidEnsureCommit NotificationType , { } , completionId ) ;
void this . notify ( DidEnsureRow NotificationType , { } , completionId ) ;
}
}
return ;
return ;
}
}
let selected : boolean | undefined ;
if ( ! this . _graph . ids . has ( e . id ) ) {
await this . updateGraphWithMoreCommits ( this . _graph , e . id ) ;
if ( e . select && this . _graph . ids . has ( e . id ) ) {
selected = true ;
this . setSelectedRows ( e . id ) ;
let id : string | undefined ;
if ( ! this . _graph . skippedIds ? . has ( e . id ) ) {
if ( this . _graph . ids . has ( e . id ) ) {
id = e . id ;
} else {
await this . updateGraphWithMoreRows ( this . _graph , e . id , this . _search ) ;
void this . notifyDidChangeRows ( ) ;
if ( this . _graph . ids . has ( e . id ) ) {
id = e . id ;
}
}
}
void this . notifyDidChangeCommits ( ) ;
} else if ( e . select ) {
selected = true ;
this . setSelectedRows ( e . id ) ;
}
}
void this . notify ( DidEnsureCommit NotificationType , { id : e. id , selected : selecte d } , completionId ) ;
void this . notify ( DidEnsureRowNotificationType , { id : id } , completionId ) ;
}
}
private async onGetMissingAvatars ( e : GetMissingAvatarsParams ) {
private async onGetMissingAvatars ( e : GetMissingAvatarsParams ) {
@ -503,17 +521,17 @@ export class GraphWebview extends WebviewBase {
const repoPath = this . _graph . repoPath ;
const repoPath = this . _graph . repoPath ;
async function getAvatar ( this : GraphWebview , email : string , sha : string ) {
const uri = await getAvatarUri ( email , { ref : sha , repoPath : repoPath } ) ;
async function getAvatar ( this : GraphWebview , email : string , id : string ) {
const uri = await getAvatarUri ( email , { ref : id , repoPath : repoPath } ) ;
this . _graph ! . avatars . set ( email , uri . toString ( true ) ) ;
this . _graph ! . avatars . set ( email , uri . toString ( true ) ) ;
}
}
const promises : Promise < void > [ ] = [ ] ;
const promises : Promise < void > [ ] = [ ] ;
for ( const [ email , sha ] of Object . entries ( e . emails ) ) {
for ( const [ email , id ] of Object . entries ( e . emails ) ) {
if ( this . _graph . avatars . has ( email ) ) continue ;
if ( this . _graph . avatars . has ( email ) ) continue ;
promises . push ( getAvatar . call ( this , email , sha ) ) ;
promises . push ( getAvatar . call ( this , email , id ) ) ;
}
}
if ( promises . length ) {
if ( promises . length ) {
@ -524,19 +542,19 @@ export class GraphWebview extends WebviewBase {
@gate ( )
@gate ( )
@debug ( )
@debug ( )
private async onGetMoreCommits ( e : GetMoreCommitsParams ) {
private async onGetMoreRows ( e : GetMoreRowsParams , sendSelectedRows : boolean = false ) {
if ( this . _graph ? . more == null || this . _repository ? . etag !== this . _etagRepository ) {
if ( this . _graph ? . more == null || this . _repository ? . etag !== this . _etagRepository ) {
this . updateState ( true ) ;
this . updateState ( true ) ;
return ;
return ;
}
}
await this . updateGraphWithMoreCommit s ( this . _graph , e . sha ) ;
void this . notifyDidChangeCommits ( ) ;
await this . updateGraphWithMoreRow s ( this . _graph , e . id span>, t hisan>. _search ) ;
void this . notifyDidChangeRows ( sendSelectedRows ) ;
}
}
@debug ( )
@debug ( )
private async onSearchCommits ( e : SearchCommits Params , completionId? : string ) {
private async onSearch ( e : SearchParams , completionId? : string ) {
if ( e . search == null ) {
if ( e . search == null ) {
this . resetSearchState ( ) ;
this . resetSearchState ( ) ;
@ -553,15 +571,19 @@ export class GraphWebview extends WebviewBase {
search = await search . more ( e . limit ? ? configuration . get ( 'graph.searchItemLimit' ) ? ? 100 ) ;
search = await search . more ( e . limit ? ? configuration . get ( 'graph.searchItemLimit' ) ? ? 100 ) ;
if ( search != null ) {
if ( search != null ) {
this . _search = search ;
this . _search = search ;
void ( await this . ensureSearchStartsInRange ( this . _graph ! , search ) ) ;
void this . notify (
void this . notify (
DidSearchCommits NotificationType ,
DidSearchNotificationType ,
{
{
results : {
ids : Object.fromEntries ( search . results ) ,
paging : { hasMore : search.paging?.hasMore ? ? false } ,
} ,
selectedRows : this._selectedRows ,
results :
search . results . size > 0
? {
ids : Object.fromEntries ( search . results ) ,
count : search.results.size ,
paging : { hasMore : search.paging?.hasMore ? ? false } ,
}
: undefined ,
} ,
} ,
completionId ,
completionId ,
) ;
) ;
@ -585,15 +607,30 @@ export class GraphWebview extends WebviewBase {
const cancellation = new CancellationTokenSource ( ) ;
const cancellation = new CancellationTokenSource ( ) ;
this . _searchCancellation = cancellation ;
this . _searchCancellation = cancellation ;
search = await this . _repository . searchCommits ( e . search , {
limit : configuration.get ( 'graph.searchItemLimit' ) ? ? 100 ,
ordering : configuration.get ( 'graph.commitOrdering' ) ,
cancellation : cancellation.token ,
} ) ;
try {
search = await this . _repository . searchCommits ( e . search , {
limit : configuration.get ( 'graph.searchItemLimit' ) ? ? 100 ,
ordering : configuration.get ( 'graph.commitOrdering' ) ,
cancellation : cancellation.token ,
} ) ;
} catch ( ex ) {
this . _search = undefined ;
void this . notify (
DidSearchNotificationType ,
{
results : {
error : ex instanceof GitSearchError ? 'Invalid search pattern' : 'Unexpected error' ,
} ,
} ,
completionId ,
) ;
return ;
}
if ( cancellation . token . isCancellationRequested ) {
if ( cancellation . token . isCancellationRequested ) {
if ( completionId != null ) {
if ( completionId != null ) {
void this . notify ( DidSearchCommitsNotificationType , { results : undefined } , completionId ) ;
void this . notify ( DidSearchNotificationType , { results : undefined } , completionId ) ;
}
}
return ;
return ;
}
}
@ -603,18 +640,26 @@ export class GraphWebview extends WebviewBase {
search = this . _search ! ;
search = this . _search ! ;
}
}
if ( search . results . size > 0 ) {
this . setSelectedRows ( first ( search . results ) ! [ 0 ] ) ;
const firstResult = await this . ensureSearchStartsInRange ( this . _graph ! , search ) ;
let sendSelectedRows = false ;
if ( firstResult != null ) {
sendSelectedRows = true ;
this . setSelectedRows ( firstResult ) ;
}
}
void this . notify (
void this . notify (
DidSearchCommitsNotificationType ,
DidSearchNotificationType ,
{
{
results : {
ids : Object.fromEntries ( search . results ) ,
paging : { hasMore : search.paging?.hasMore ? ? false } ,
} ,
selectedRows : this._selectedRows ,
results :
search . results . size === 0
? { count : 0 }
: {
ids : Object.fromEntries ( search . results ) ,
count : search.results.size ,
paging : { hasMore : search.paging?.hasMore ? ? false } ,
} ,
selectedRows : sendSelectedRows ? this . _selectedRows : undefined ,
} ,
} ,
completionId ,
completionId ,
) ;
) ;
@ -633,29 +678,34 @@ export class GraphWebview extends WebviewBase {
} ) ;
} ) ;
}
}
private onRepositoryFileSystemChanged ( e : RepositoryFileSystemChangeEvent ) {
if ( ! ( e . repository ? . path === this . repository ? . path ) ) return ;
this . updateWorkDirStats ( ) ;
}
private onRepositorySelectionChanged ( e : UpdateSelectedRepositoryParams ) {
private onRepositorySelectionChanged ( e : UpdateSelectedRepositoryParams ) {
this . repository = this . container . git . getRepository ( e . path ) ;
this . repository = this . container . git . getRepository ( e . path ) ;
}
}
private async onSelectionChanged ( e : UpdateSelectionParams ) {
private _fireSelectionChangedDebounced : Deferrable < GraphWebview [ ' fireSelectionChanged ' ] > | undefined = undefined ;
private onSelectionChanged ( e : UpdateSelectionParams ) {
const item = e . selection [ 0 ] ;
const item = e . selection [ 0 ] ;
this . setSelectedRows ( item ? . id ) ;
this . setSelectedRows ( item ? . id ) ;
if ( this . _fireSelectionChangedDebounced == null ) {
this . _fireSelectionChangedDebounced = debounce ( this . fireSelectionChanged . bind ( this ) , 250 ) ;
}
void this . _fireSelectionChangedDebounced ( item ? . id , item ? . type ) ;
}
private async fireSelectionChanged ( id : string | undefined , type : GitGraphRowType | undefined ) {
let commits : GitCommit [ ] | undefined ;
let commits : GitCommit [ ] | undefined ;
if ( item ? . id != null ) {
if ( id != null ) {
let commit ;
let commit ;
if ( item . type === GitGraphRowType . Stash ) {
if ( type === GitGraphRowType . Stash ) {
const stash = await this . repository ? . getStash ( ) ;
const stash = await this . repository ? . getStash ( ) ;
commit = stash ? . commits . get ( item . id ) ;
} else if ( item . type === GitGraphRowType . Working ) {
commit = await this . repository ? . getCommit ( '0000000000000000000000000000000000000000' ) ;
commit = stash ? . commits . get ( id ) ;
} else if ( type === GitGraphRowType . Working ) {
commit = await this . repository ? . getCommit ( GitRevision . uncommitted ) ;
} else {
} else {
commit = await this . repository ? . getCommit ( item ? . i d ) ;
commit = await this . repository ? . getCommit ( id ) ;
}
}
if ( commit != null ) {
if ( commit != null ) {
commits = [ commit ] ;
commits = [ commit ] ;
@ -670,7 +720,7 @@ export class GraphWebview extends WebviewBase {
void GitActions . Commit . showDetailsView ( commits [ 0 ] , { pin : true , preserveFocus : true } ) ;
void GitActions . Commit . showDetailsView ( commits [ 0 ] , { pin : true , preserveFocus : true } ) ;
}
}
private _notifyDidChangeStateDebounced : Deferrable < ( ) = > void > | undefined = undefined ;
private _notifyDidChangeStateDebounced : Deferrable < GraphWebview [ ' notifyDidChangeState ' ] > | undefined = undefined ;
@debug ( )
@debug ( )
private updateState ( immediate : boolean = false ) {
private updateState ( immediate : boolean = false ) {
@ -685,10 +735,11 @@ export class GraphWebview extends WebviewBase {
this . _notifyDidChangeStateDebounced = debounce ( this . notifyDidChangeState . bind ( this ) , 500 ) ;
this . _notifyDidChangeStateDebounced = debounce ( this . notifyDidChangeState . bind ( this ) , 500 ) ;
}
}
this . _notifyDidChangeStateDebounced ( ) ;
void this. _notifyDidChangeStateDebounced ( ) ;
}
}
private _notifyDidChangeAvatarsDebounced : Deferrable < ( ) = > void > | undefined = undefined ;
private _notifyDidChangeAvatarsDebounced : Deferrable < GraphWebview [ ' notifyDidChangeAvatars ' ] > | undefined =
undefined ;
@debug ( )
@debug ( )
private updateAvatars ( immediate : boolean = false ) {
private updateAvatars ( immediate : boolean = false ) {
@ -701,25 +752,7 @@ export class GraphWebview extends WebviewBase {
this . _notifyDidChangeAvatarsDebounced = debounce ( this . notifyDidChangeAvatars . bind ( this ) , 100 ) ;
this . _notifyDidChangeAvatarsDebounced = debounce ( this . notifyDidChangeAvatars . bind ( this ) , 100 ) ;
}
}
this . _notifyDidChangeAvatarsDebounced ( ) ;
}
private _notifyDidChangeWorkDirStatsDebounced : Deferrable < ( ) = > void > | undefined = undefined ;
@debug ( )
private updateWorkDirStats ( immediate : boolean = false ) {
if ( ! this . isReady || ! this . visible ) return ;
if ( immediate ) {
void this . notifyDidChangeWorkDirStats ( ) ;
return ;
}
if ( this . _notifyDidChangeWorkDirStatsDebounced == null ) {
this . _notifyDidChangeWorkDirStatsDebounced = debounce ( this . notifyDidChangeWorkDirStats . bind ( this ) , 500 ) ;
}
this . _notifyDidChangeWorkDirStatsDebounced ( ) ;
void this . _notifyDidChangeAvatarsDebounced ( ) ;
}
}
@debug ( )
@debug ( )
@ -759,16 +792,16 @@ export class GraphWebview extends WebviewBase {
}
}
@debug ( )
@debug ( )
private async notifyDidChangeCommits ( completionId? : string ) {
private async notifyDidChangeRows ( sendSelectedRows : boolean = false , completionId? : string ) {
if ( this . _graph == null ) return ;
if ( this . _graph == null ) return ;
const data = this . _graph ;
const data = this . _graph ;
return this . notify (
return this . notify (
DidChangeCommit sNotificationType ,
DidChangeRow sNotificationType ,
{
{
rows : data.rows ,
rows : data.rows ,
avatars : Object.fromEntries ( data . avatars ) ,
avatars : Object.fromEntries ( data . avatars ) ,
selectedRows : this._selectedRows ,
selectedRows : sendSelectedRows ? this . _selectedRows : undefined ,
paging : {
paging : {
startingCursor : data.paging?.startingCursor ,
startingCursor : data.paging?.startingCursor ,
hasMore : data.paging?.hasMore ? ? false ,
hasMore : data.paging?.hasMore ? ? false ,
@ -779,11 +812,14 @@ export class GraphWebview extends WebviewBase {
}
}
@debug ( )
@debug ( )
private async notifyDidChangeWorkDirStats() {
if ( ! this . isReady || ! this . visible ) return false ;
private async notifyDidChangeWorkingTree() {
if ( ! this . isReady || ! this . visible ) {
this . addPendingIpcNotification ( DidChangeWorkingTreeNotificationType ) ;
return false ;
}
return this . notify ( DidChangeWorkDirStatsNotificationType , {
workDirStats : await this . getWorkDirStats ( ) ? ? { added : 0 , deleted : 0 , modified : 0 } ,
return this . notify ( DidChangeWorkingTree NotificationType , {
stats : ( await this . getWorkingT ree Stats ( ) ) ? ? { added : 0 , deleted : 0 , modified : 0 } ,
} ) ;
} ) ;
}
}
@ -795,7 +831,7 @@ export class GraphWebview extends WebviewBase {
}
}
return this . notify ( DidChangeSelectionNotificationType , {
return this . notify ( DidChangeSelectionNotificationType , {
selection : this._selectedRows ,
selection : this._selectedRows ? ? { } ,
} ) ;
} ) ;
}
}
@ -806,7 +842,7 @@ export class GraphWebview extends WebviewBase {
return false ;
return false ;
}
}
const access = await this . getGraphAccess ( ) ;
const [ access ] = await this . getGraphAccess ( ) ;
return this . notify ( DidChangeSubscriptionNotificationType , {
return this . notify ( DidChangeSubscriptionNotificationType , {
subscription : access.subscription.current ,
subscription : access.subscription.current ,
allowed : access.allowed !== false ,
allowed : access.allowed !== false ,
@ -849,6 +885,7 @@ export class GraphWebview extends WebviewBase {
[ DidChangeNotificationType , this . notifyDidChangeState ] ,
[ DidChangeNotificationType , this . notifyDidChangeState ] ,
[ DidChangeSelectionNotificationType , this . notifyDidChangeSelection ] ,
[ DidChangeSelectionNotificationType , this . notifyDidChangeSelection ] ,
[ DidChangeSubscriptionNotificationType , this . notifyDidChangeSubscription ] ,
[ DidChangeSubscriptionNotificationType , this . notifyDidChangeSubscription ] ,
[ DidChangeWorkingTreeNotificationType , this . notifyDidChangeWorkingTree ] ,
] ) ;
] ) ;
private addPendingIpcNotification ( type : IpcNotificationType < any > , msg? : IpcMessage ) {
private addPendingIpcNotification ( type : IpcNotificationType < any > , msg? : IpcMessage ) {
@ -885,6 +922,26 @@ export class GraphWebview extends WebviewBase {
}
}
}
}
private async ensureSearchStartsInRange ( graph : GitGraph , search : GitSearch ) {
if ( search . results . size === 0 ) return undefined ;
let firstResult : string | undefined ;
for ( const id of search . results . keys ( ) ) {
if ( graph . ids . has ( id ) ) return id ;
if ( graph . skippedIds ? . has ( id ) ) continue ;
firstResult = id ;
break ;
}
if ( firstResult == null ) return undefined ;
await this . updateGraphWithMoreRows ( graph , firstResult ) ;
void this . notifyDidChangeRows ( ) ;
return graph . ids . has ( firstResult ) ? firstResult : undefined ;
}
private getColumns ( ) : Record < GraphColumnName , GraphColumnConfig > | undefined {
private getColumns ( ) : Record < GraphColumnName , GraphColumnConfig > | undefined {
return this . container . storage . getWorkspace ( 'graph:columns' ) ;
return this . container . storage . getWorkspace ( 'graph:columns' ) ;
}
}
@ -933,12 +990,30 @@ export class GraphWebview extends WebviewBase {
enableMultiSelection : false ,
enableMultiSelection : false ,
highlightRowsOnRefHover : configuration.get ( 'graph.highlightRowsOnRefHover' ) ,
highlightRowsOnRefHover : configuration.get ( 'graph.highlightRowsOnRefHover' ) ,
showGhostRefsOnRowHover : configuration.get ( 'graph.showGhostRefsOnRowHover' ) ,
showGhostRefsOnRowHover : configuration.get ( 'graph.showGhostRefsOnRowHover' ) ,
sha Length : configuration.get ( 'advanced.abbreviatedShaLength' ) ,
id Length : configuration.get ( 'advanced.abbreviatedShaLength' ) ,
} ;
} ;
return config ;
return config ;
}
}
private async getWorkDirStats ( ) : Promise < GraphWorkDirStats | undefined > {
private async getGraphAccess() {
let access = await this . container . git . access ( PlusFeatures . Graph , this . repository ? . path ) ;
this . _etagSubscription = this . container . subscription . etag ;
// If we don't have access to GitLens+, but the preview trial hasn't been started, auto-start it
if ( access . allowed === false && access . subscription . current . previewTrial == null ) {
await this . container . subscription . startPreviewTrial ( true ) ;
access = await this . container . git . access ( PlusFeatures . Graph , this . repository ? . path ) ;
}
let visibility = access ? . visibility ;
if ( visibility == null && this . repository != null ) {
visibility = await this . container . git . visibility ( this . repository ? . path ) ;
}
return [ access , visibility ] as const ;
}
private async getWorkingTreeStats ( ) : Promise < GraphWorkingTreeStats | undefined > {
if ( this . container . git . repositoryCount === 0 ) return undefined ;
if ( this . container . git . repositoryCount === 0 ) return undefined ;
if ( this . repository == null ) {
if ( this . repository == null ) {
@ -946,7 +1021,7 @@ export class GraphWebview extends WebviewBase {
if ( this . repository == null ) return undefined ;
if ( this . repository == null ) return undefined ;
}
}
const status : GitStatus | undefined = await this . container . git . getStatusForRepo ( this . repository . path ) ;
const status = await this . container . git . getStatusForRepo ( this . repository . path ) ;
const workingTreeStatus = status ? . getDiffStatus ( ) ;
const workingTreeStatus = status ? . getDiffStatus ( ) ;
return {
return {
added : workingTreeStatus?.added ? ? 0 ,
added : workingTreeStatus?.added ? ? 0 ,
@ -955,18 +1030,6 @@ export class GraphWebview extends WebviewBase {
} ;
} ;
}
}
private async getGraphAccess() {
let access = await this . container . git . access ( PlusFeatures . Graph , this . repository ? . path ) ;
this . _etagSubscription = this . container . subscription . etag ;
// If we don't have access to GitLens+, but the preview trial hasn't been started, auto-start it
if ( access . allowed === false && access . subscription . current . previewTrial == null ) {
await this . container . subscription . startPreviewTrial ( true ) ;
access = await this . container . git . access ( PlusFeatures . Graph , this . repository ? . path ) ;
}
return access ;
}
private async getState ( deferRows? : boolean ) : Promise < State > {
private async getState ( deferRows? : boolean ) : Promise < State > {
if ( this . container . git . repositoryCount === 0 ) return { allowed : true , repositories : [ ] } ;
if ( this . container . git . repositoryCount === 0 ) return { allowed : true , repositories : [ ] } ;
@ -993,30 +1056,32 @@ export class GraphWebview extends WebviewBase {
// If we have a set of data refresh to the same set
// If we have a set of data refresh to the same set
const limit = Math . max ( defaultItemLimit , this . _graph ? . ids . size ? ? defaultItemLimit ) ;
const limit = Math . max ( defaultItemLimit , this . _graph ? . ids . size ? ? defaultItemLimit ) ;
// Check for GitLens+ access
const access = await this . getGraphAccess ( ) ;
const visibility = access . visibility ? ? ( await this . container . git . visibility ( this . repository . path ) ) ;
const dataPromise = this . container . git . getCommitsForGraph (
const dataPromise = this . container . git . getCommitsForGraph (
this . repository . path ,
this . repository . path ,
this . _panel ! . webview . asWebviewUri . bind ( this . _panel ! . webview ) ,
this . _panel ! . webview . asWebviewUri . bind ( this . _panel ! . webview ) ,
{ limit : limit , ref : this._selectedSha ? ? 'HEAD' } ,
{ limit : limit , ref : this._selectedId ? ? 'HEAD' } ,
) ;
) ;
let data ;
// Check for GitLens+ access and working tree stats
const [ accessResult , workingStatsResult ] = await Promise . allSettled ( [
this . getGraphAccess ( ) ,
this . getWorkingTreeStats ( ) ,
] ) ;
const [ access , visibility ] = getSettledValue ( accessResult ) ? ? [ ] ;
let data ;
if ( deferRows ) {
if ( deferRows ) {
queueMicrotask ( async ( ) = > {
queueMicrotask ( async ( ) = > {
const data = await dataPromise ;
const data = await dataPromise ;
this . setGraph ( data ) ;
this . setGraph ( data ) ;
this . setSelectedRows ( data . sha ) ;
this . setSelectedRows ( data . id ) ;
void this . notifyDidChangeCommits ( ) ;
void this . notifyDidChangeRows ( true ) ;
} ) ;
} ) ;
} else {
} else {
data = await dataPromise ;
data = await dataPromise ;
this . setGraph ( data ) ;
this . setGraph ( data ) ;
this . setSelectedRows ( data . sha ) ;
this . setSelectedRows ( data . id ) ;
}
}
const columns = this . getColumns ( ) ;
const columns = this . getColumns ( ) ;
@ -1028,8 +1093,8 @@ export class GraphWebview extends WebviewBase {
selectedRepository : this.repository.path ,
selectedRepository : this.repository.path ,
selectedRepositoryVisibility : visibility ,
selectedRepositoryVisibility : visibility ,
selectedRows : this._selectedRows ,
selectedRows : this._selectedRows ,
subscription : access.subscription.current ,
allowed : access.allowed !== false ,
subscription : access? .subscription.current ,
allowed : ( access ? . allowed ? ? false ) !== false ,
avatars : data != null ? Object . fromEntries ( data . avatars ) : undefined ,
avatars : data != null ? Object . fromEntries ( data . avatars ) : undefined ,
loading : deferRows ,
loading : deferRows ,
rows : data?.rows ,
rows : data?.rows ,
@ -1046,7 +1111,7 @@ export class GraphWebview extends WebviewBase {
header : this.getColumnHeaderContext ( columns ) ,
header : this.getColumnHeaderContext ( columns ) ,
} ,
} ,
nonce : this.cspNonce ,
nonce : this.cspNonce ,
dirStats : await this . getWorkDirStats ( ) ? ? { added : 0 , deleted : 0 , modified : 0 } ,
workingTreeStats : getSettledValue ( workingStatsResult ) ? ? { added : 0 , deleted : 0 , modified : 0 } ,
} ;
} ;
}
}
@ -1068,11 +1133,11 @@ export class GraphWebview extends WebviewBase {
this . _searchCancellation = undefined ;
this . _searchCancellation = undefined ;
}
}
private setSelectedRows ( sha : string | undefined ) {
if ( this . _selectedSha === sha ) return ;
private setSelectedRows ( id : string | undefined ) {
if ( this . _selectedId === id ) return ;
this . _selectedSha = sha ;
this . _selectedRows = sha != null ? { [ sha ] : true } : { } ;
this . _selectedId = id ;
this . _selectedRows = id != null ? { [ id ] : true } : undefined ;
}
}
private setGraph ( graph : GitGraph | undefined ) {
private setGraph ( graph : GitGraph | undefined ) {
@ -1082,17 +1147,16 @@ export class GraphWebview extends WebviewBase {
}
}
}
}
private async updateGraphWithMoreCommit s ( graph : GitGraph , sha? : string ) {
private async updateGraphWithMoreRow s ( graph : GitGraph , id : string | undefined , search? : GitSearch ) {
const { defaultItemLimit , pageItemLimit } = configuration . get ( 'graph' ) ;
const { defaultItemLimit , pageItemLimit } = configuration . get ( 'graph' ) ;
const updatedGraph = await graph . more ? . ( pageItemLimit ? ? defaultItemLimit , sha ) ;
const updatedGraph = await graph . more ? . ( pageItemLimit ? ? defaultItemLimit , id ) ;
if ( updatedGraph != null ) {
if ( updatedGraph != null ) {
this . setGraph ( updatedGraph ) ;
this . setGraph ( updatedGraph ) ;
if ( this . _search != null ) {
const search = this . _search ;
if ( search ? . paging ? . hasMore ) {
const lastId = last ( search . results ) ? . [ 0 ] ;
const lastId = last ( search . results ) ? . [ 0 ] ;
if ( lastId != null && updatedGraph . ids . has ( lastId ) ) {
queueMicrotask ( ( ) = > void this . onSearchCommits ( { search : search.query , more : true } ) ) ;
if ( lastId != null && ( updatedGraph . ids . has ( lastId ) || updatedGraph . skippedIds ? . has ( lastId ) ) ) {
queueMicrotask ( ( ) = > void this . onSearch ( { search : search.query , more : true } ) ) ;
}
}
}
}
} else {
} else {