BigW Consortium Gitlab
Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
G
gitlab-ce
Project
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Commits
Issue Boards
Open sidebar
Forest Godfrey
gitlab-ce
Commits
c06dad62
Commit
c06dad62
authored
Jul 05, 2017
by
Mike Greiling
Committed by
Clement Ho
Jul 05, 2017
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Remove IIFEs around several javascript classes
parent
11e03bf4
Hide whitespace changes
Inline
Side-by-side
Showing
15 changed files
with
763 additions
and
785 deletions
+763
-785
signin_tabs_memoizer.js
app/assets/javascripts/signin_tabs_memoizer.js
+40
-42
single_file_diff.js
app/assets/javascripts/single_file_diff.js
+78
-80
smart_interval.js
app/assets/javascripts/smart_interval.js
+126
-127
snippets_list.js
app/assets/javascripts/snippets_list.js
+6
-10
star.js
app/assets/javascripts/star.js
+24
-26
subscription.js
app/assets/javascripts/subscription.js
+36
-38
subscription_select.js
app/assets/javascripts/subscription_select.js
+30
-31
syntax_highlight.js
app/assets/javascripts/syntax_highlight.js
+13
-14
tree.js
app/assets/javascripts/tree.js
+58
-60
user.js
app/assets/javascripts/user.js
+29
-28
user_tabs.js
app/assets/javascripts/user_tabs.js
+94
-93
username_validator.js
app/assets/javascripts/username_validator.js
+107
-109
visibility_select.js
app/assets/javascripts/visibility_select.js
+19
-22
wikis.js
app/assets/javascripts/wikis.js
+47
-48
zen_mode.js
app/assets/javascripts/zen_mode.js
+56
-57
No files found.
app/assets/javascripts/signin_tabs_memoizer.js
View file @
c06dad62
...
...
@@ -2,56 +2,54 @@
/* eslint no-new: "off" */
import
AccessorUtilities
from
'./lib/utils/accessor'
;
((
global
)
=>
{
/**
* Memorize the last selected tab after reloading a page.
* Does that setting the current selected tab in the localStorage
*/
class
ActiveTabMemoizer
{
constructor
({
currentTabKey
=
'current_signin_tab'
,
tabSelector
=
'ul.nav-tabs'
}
=
{})
{
this
.
currentTabKey
=
currentTabKey
;
this
.
tabSelector
=
tabSelector
;
this
.
isLocalStorageAvailable
=
AccessorUtilities
.
isLocalStorageAccessSafe
();
this
.
bootstrap
();
}
bootstrap
()
{
const
tabs
=
document
.
querySelectorAll
(
this
.
tabSelector
);
if
(
tabs
.
length
>
0
)
{
tabs
[
0
].
addEventListener
(
'click'
,
(
e
)
=>
{
if
(
e
.
target
&&
e
.
target
.
nodeName
===
'A'
)
{
const
anchorName
=
e
.
target
.
getAttribute
(
'href'
);
this
.
saveData
(
anchorName
);
}
});
}
/**
* Memorize the last selected tab after reloading a page.
* Does that setting the current selected tab in the localStorage
*/
class
ActiveTabMemoizer
{
constructor
({
currentTabKey
=
'current_signin_tab'
,
tabSelector
=
'ul.nav-tabs'
}
=
{})
{
this
.
currentTabKey
=
currentTabKey
;
this
.
tabSelector
=
tabSelector
;
this
.
isLocalStorageAvailable
=
AccessorUtilities
.
isLocalStorageAccessSafe
();
this
.
bootstrap
();
}
this
.
showTab
();
bootstrap
()
{
const
tabs
=
document
.
querySelectorAll
(
this
.
tabSelector
);
if
(
tabs
.
length
>
0
)
{
tabs
[
0
].
addEventListener
(
'click'
,
(
e
)
=>
{
if
(
e
.
target
&&
e
.
target
.
nodeName
===
'A'
)
{
const
anchorName
=
e
.
target
.
getAttribute
(
'href'
);
this
.
saveData
(
anchorName
);
}
});
}
showTab
()
{
const
anchorName
=
this
.
readData
();
if
(
anchorName
)
{
const
tab
=
document
.
querySelector
(
`
${
this
.
tabSelector
}
a[href="
${
anchorName
}
"]`
);
if
(
tab
)
{
tab
.
click
();
}
this
.
showTab
();
}
showTab
()
{
const
anchorName
=
this
.
readData
();
if
(
anchorName
)
{
const
tab
=
document
.
querySelector
(
`
${
this
.
tabSelector
}
a[href="
${
anchorName
}
"]`
);
if
(
tab
)
{
tab
.
click
();
}
}
}
saveData
(
val
)
{
if
(
!
this
.
isLocalStorageAvailable
)
return
undefined
;
saveData
(
val
)
{
if
(
!
this
.
isLocalStorageAvailable
)
return
undefined
;
return
window
.
localStorage
.
setItem
(
this
.
currentTabKey
,
val
);
}
return
window
.
localStorage
.
setItem
(
this
.
currentTabKey
,
val
);
}
readData
()
{
if
(
!
this
.
isLocalStorageAvailable
)
return
null
;
readData
()
{
if
(
!
this
.
isLocalStorageAvailable
)
return
null
;
return
window
.
localStorage
.
getItem
(
this
.
currentTabKey
);
}
return
window
.
localStorage
.
getItem
(
this
.
currentTabKey
);
}
}
global
.
ActiveTabMemoizer
=
ActiveTabMemoizer
;
})(
window
);
window
.
ActiveTabMemoizer
=
ActiveTabMemoizer
;
app/assets/javascripts/single_file_diff.js
View file @
c06dad62
...
...
@@ -2,99 +2,97 @@
import
FilesCommentButton
from
'./files_comment_button'
;
(
function
()
{
window
.
SingleFileDiff
=
(
function
()
{
var
COLLAPSED_HTML
,
ERROR_HTML
,
LOADING_HTML
,
WRAPPER
;
window
.
SingleFileDiff
=
(
function
()
{
var
COLLAPSED_HTML
,
ERROR_HTML
,
LOADING_HTML
,
WRAPPER
;
WRAPPER
=
'<div class="diff-content"></div>'
;
WRAPPER
=
'<div class="diff-content"></div>'
;
LOADING_HTML
=
'<i class="fa fa-spinner fa-spin"></i>'
;
LOADING_HTML
=
'<i class="fa fa-spinner fa-spin"></i>'
;
ERROR_HTML
=
'<div class="nothing-here-block"><i class="fa fa-warning"></i> Could not load diff</div>'
;
ERROR_HTML
=
'<div class="nothing-here-block"><i class="fa fa-warning"></i> Could not load diff</div>'
;
COLLAPSED_HTML
=
'<div class="nothing-here-block diff-collapsed">This diff is collapsed. <a class="click-to-expand">Click to expand it.</a></div>'
;
COLLAPSED_HTML
=
'<div class="nothing-here-block diff-collapsed">This diff is collapsed. <a class="click-to-expand">Click to expand it.</a></div>'
;
function
SingleFileDiff
(
file
)
{
this
.
file
=
file
;
this
.
toggleDiff
=
this
.
toggleDiff
.
bind
(
this
);
this
.
content
=
$
(
'.diff-content'
,
this
.
file
);
this
.
$toggleIcon
=
$
(
'.diff-toggle-caret'
,
this
.
file
);
this
.
diffForPath
=
this
.
content
.
find
(
'[data-diff-for-path]'
).
data
(
'diff-for-path'
);
this
.
isOpen
=
!
this
.
diffForPath
;
if
(
this
.
diffForPath
)
{
this
.
collapsedContent
=
this
.
content
;
this
.
loadingContent
=
$
(
WRAPPER
).
addClass
(
'loading'
).
html
(
LOADING_HTML
).
hide
();
this
.
content
=
null
;
this
.
collapsedContent
.
after
(
this
.
loadingContent
);
this
.
$toggleIcon
.
addClass
(
'fa-caret-right'
);
}
else
{
this
.
collapsedContent
=
$
(
WRAPPER
).
html
(
COLLAPSED_HTML
).
hide
();
this
.
content
.
after
(
this
.
collapsedContent
);
this
.
$toggleIcon
.
addClass
(
'fa-caret-down'
);
}
function
SingleFileDiff
(
file
)
{
this
.
file
=
file
;
this
.
toggleDiff
=
this
.
toggleDiff
.
bind
(
this
);
this
.
content
=
$
(
'.diff-content'
,
this
.
file
);
this
.
$toggleIcon
=
$
(
'.diff-toggle-caret'
,
this
.
file
);
this
.
diffForPath
=
this
.
content
.
find
(
'[data-diff-for-path]'
).
data
(
'diff-for-path'
);
this
.
isOpen
=
!
this
.
diffForPath
;
if
(
this
.
diffForPath
)
{
this
.
collapsedContent
=
this
.
content
;
this
.
loadingContent
=
$
(
WRAPPER
).
addClass
(
'loading'
).
html
(
LOADING_HTML
).
hide
();
this
.
content
=
null
;
this
.
collapsedContent
.
after
(
this
.
loadingContent
);
this
.
$toggleIcon
.
addClass
(
'fa-caret-right'
);
}
else
{
this
.
collapsedContent
=
$
(
WRAPPER
).
html
(
COLLAPSED_HTML
).
hide
();
this
.
content
.
after
(
this
.
collapsedContent
);
this
.
$toggleIcon
.
addClass
(
'fa-caret-down'
);
}
$
(
'.js-file-title, .click-to-expand'
,
this
.
file
).
on
(
'click'
,
(
function
(
e
)
{
this
.
toggleDiff
(
$
(
e
.
target
));
}).
bind
(
this
));
}
$
(
'.js-file-title, .click-to-expand'
,
this
.
file
).
on
(
'click'
,
(
function
(
e
)
{
this
.
toggleDiff
(
$
(
e
.
target
));
}).
bind
(
this
));
SingleFileDiff
.
prototype
.
toggleDiff
=
function
(
$target
,
cb
)
{
if
(
!
$target
.
hasClass
(
'js-file-title'
)
&&
!
$target
.
hasClass
(
'click-to-expand'
)
&&
!
$target
.
hasClass
(
'diff-toggle-caret'
))
return
;
this
.
isOpen
=
!
this
.
isOpen
;
if
(
!
this
.
isOpen
&&
!
this
.
hasError
)
{
this
.
content
.
hide
();
this
.
$toggleIcon
.
addClass
(
'fa-caret-right'
).
removeClass
(
'fa-caret-down'
);
this
.
collapsedContent
.
show
();
if
(
typeof
gl
.
diffNotesCompileComponents
!==
'undefined'
)
{
gl
.
diffNotesCompileComponents
();
}
}
else
if
(
this
.
content
)
{
this
.
collapsedContent
.
hide
();
this
.
content
.
show
();
this
.
$toggleIcon
.
addClass
(
'fa-caret-down'
).
removeClass
(
'fa-caret-right'
);
if
(
typeof
gl
.
diffNotesCompileComponents
!==
'undefined'
)
{
gl
.
diffNotesCompileComponents
();
}
}
else
{
this
.
$toggleIcon
.
addClass
(
'fa-caret-down'
).
removeClass
(
'fa-caret-right'
);
return
this
.
getContentHTML
(
cb
);
}
};
SingleFileDiff
.
prototype
.
toggleDiff
=
function
(
$target
,
cb
)
{
if
(
!
$target
.
hasClass
(
'js-file-title'
)
&&
!
$target
.
hasClass
(
'click-to-expand'
)
&&
!
$target
.
hasClass
(
'diff-toggle-caret'
))
return
;
this
.
isOpen
=
!
this
.
isOpen
;
if
(
!
this
.
isOpen
&&
!
this
.
hasError
)
{
this
.
content
.
hide
();
this
.
$toggleIcon
.
addClass
(
'fa-caret-right'
).
removeClass
(
'fa-caret-down'
);
this
.
collapsedContent
.
show
();
if
(
typeof
gl
.
diffNotesCompileComponents
!==
'undefined'
)
{
gl
.
diffNotesCompileComponents
();
SingleFileDiff
.
prototype
.
getContentHTML
=
function
(
cb
)
{
this
.
collapsedContent
.
hide
();
this
.
loadingContent
.
show
();
$
.
get
(
this
.
diffForPath
,
(
function
(
_this
)
{
return
function
(
data
)
{
_this
.
loadingContent
.
hide
();
if
(
data
.
html
)
{
_this
.
content
=
$
(
data
.
html
);
_this
.
content
.
syntaxHighlight
();
}
else
{
_this
.
hasError
=
true
;
_this
.
content
=
$
(
ERROR_HTML
);
}
}
else
if
(
this
.
content
)
{
this
.
collapsedContent
.
hide
();
this
.
content
.
show
();
this
.
$toggleIcon
.
addClass
(
'fa-caret-down'
).
removeClass
(
'fa-caret-right'
);
_this
.
collapsedContent
.
after
(
_this
.
content
);
if
(
typeof
gl
.
diffNotesCompileComponents
!==
'undefined'
)
{
gl
.
diffNotesCompileComponents
();
}
}
else
{
this
.
$toggleIcon
.
addClass
(
'fa-caret-down'
).
removeClass
(
'fa-caret-right'
);
return
this
.
getContentHTML
(
cb
);
}
};
SingleFileDiff
.
prototype
.
getContentHTML
=
function
(
cb
)
{
this
.
collapsedContent
.
hide
();
this
.
loadingContent
.
show
();
$
.
get
(
this
.
diffForPath
,
(
function
(
_this
)
{
return
function
(
data
)
{
_this
.
loadingContent
.
hide
();
if
(
data
.
html
)
{
_this
.
content
=
$
(
data
.
html
);
_this
.
content
.
syntaxHighlight
();
}
else
{
_this
.
hasError
=
true
;
_this
.
content
=
$
(
ERROR_HTML
);
}
_this
.
collapsedContent
.
after
(
_this
.
content
);
if
(
typeof
gl
.
diffNotesCompileComponents
!==
'undefined'
)
{
gl
.
diffNotesCompileComponents
();
}
FilesCommentButton
.
init
(
$
(
_this
.
file
));
FilesCommentButton
.
init
(
$
(
_this
.
file
));
if
(
cb
)
cb
();
};
})(
this
));
};
if
(
cb
)
cb
();
};
})(
this
));
};
return
SingleFileDiff
;
})();
return
SingleFileDiff
;
})();
$
.
fn
.
singleFileDiff
=
function
()
{
return
this
.
each
(
function
()
{
if
(
!
$
.
data
(
this
,
'singleFileDiff'
))
{
return
$
.
data
(
this
,
'singleFileDiff'
,
new
window
.
SingleFileDiff
(
this
));
}
});
};
}).
call
(
window
);
$
.
fn
.
singleFileDiff
=
function
()
{
return
this
.
each
(
function
()
{
if
(
!
$
.
data
(
this
,
'singleFileDiff'
))
{
return
$
.
data
(
this
,
'singleFileDiff'
,
new
window
.
SingleFileDiff
(
this
));
}
});
};
app/assets/javascripts/smart_interval.js
View file @
c06dad62
/*
* Instances of SmartInterval extend the functionality of `setInterval`, make it configurable
* and controllable by a public API.
*
* */
(()
=>
{
class
SmartInterval
{
/**
* @param { function } opts.callback Function to be called on each iteration (required)
* @param { milliseconds } opts.startingInterval `currentInterval` is set to this initially
* @param { milliseconds } opts.maxInterval `currentInterval` will be incremented to this
* @param { milliseconds } opts.hiddenInterval `currentInterval` is set to this
* when the page is hidden
* @param { integer } opts.incrementByFactorOf `currentInterval` is incremented by this factor
* @param { boolean } opts.lazyStart Configure if timer is initialized on
* instantiation or lazily
* @param { boolean } opts.immediateExecution Configure if callback should
* be executed before the first interval.
*/
constructor
(
opts
=
{})
{
this
.
cfg
=
{
callback
:
opts
.
callback
,
startingInterval
:
opts
.
startingInterval
,
maxInterval
:
opts
.
maxInterval
,
hiddenInterval
:
opts
.
hiddenInterval
,
incrementByFactorOf
:
opts
.
incrementByFactorOf
,
lazyStart
:
opts
.
lazyStart
,
immediateExecution
:
opts
.
immediateExecution
,
};
this
.
state
=
{
intervalId
:
null
,
currentInterval
:
this
.
cfg
.
startingInterval
,
pageVisibility
:
'visible'
,
};
this
.
initInterval
();
}
/* public */
start
()
{
const
cfg
=
this
.
cfg
;
const
state
=
this
.
state
;
if
(
cfg
.
immediateExecution
)
{
cfg
.
immediateExecution
=
false
;
cfg
.
callback
();
}
/**
* Instances of SmartInterval extend the functionality of `setInterval`, make it configurable
* and controllable by a public API.
*/
class
SmartInterval
{
/**
* @param { function } opts.callback Function to be called on each iteration (required)
* @param { milliseconds } opts.startingInterval `currentInterval` is set to this initially
* @param { milliseconds } opts.maxInterval `currentInterval` will be incremented to this
* @param { milliseconds } opts.hiddenInterval `currentInterval` is set to this
* when the page is hidden
* @param { integer } opts.incrementByFactorOf `currentInterval` is incremented by this factor
* @param { boolean } opts.lazyStart Configure if timer is initialized on
* instantiation or lazily
* @param { boolean } opts.immediateExecution Configure if callback should
* be executed before the first interval.
*/
constructor
(
opts
=
{})
{
this
.
cfg
=
{
callback
:
opts
.
callback
,
startingInterval
:
opts
.
startingInterval
,
maxInterval
:
opts
.
maxInterval
,
hiddenInterval
:
opts
.
hiddenInterval
,
incrementByFactorOf
:
opts
.
incrementByFactorOf
,
lazyStart
:
opts
.
lazyStart
,
immediateExecution
:
opts
.
immediateExecution
,
};
this
.
state
=
{
intervalId
:
null
,
currentInterval
:
this
.
cfg
.
startingInterval
,
pageVisibility
:
'visible'
,
};
this
.
initInterval
();
}
state
.
intervalId
=
window
.
setInterval
(()
=>
{
cfg
.
callback
();
/* public */
if
(
this
.
getCurrentInterval
()
===
cfg
.
maxInterval
)
{
return
;
}
start
(
)
{
const
cfg
=
this
.
cfg
;
const
state
=
this
.
state
;
this
.
incrementInterval
();
this
.
resume
()
;
},
this
.
getCurrentInterval
()
);
if
(
cfg
.
immediateExecution
)
{
cfg
.
immediateExecution
=
false
;
cfg
.
callback
(
);
}
// cancel the existing timer, setting the currentInterval back to startingInterval
cancel
()
{
this
.
setCurrentInterval
(
this
.
cfg
.
startingInterval
);
this
.
stopTimer
();
}
state
.
intervalId
=
window
.
setInterval
(()
=>
{
cfg
.
callback
();
onVisibilityHidden
()
{
if
(
this
.
cfg
.
hiddenInterval
)
{
this
.
setCurrentInterval
(
this
.
cfg
.
hiddenInterval
);
this
.
resume
();
}
else
{
this
.
cancel
();
if
(
this
.
getCurrentInterval
()
===
cfg
.
maxInterval
)
{
return
;
}
}
// start a timer, using the existing interval
resume
()
{
this
.
stopTimer
();
// stop exsiting timer, in case timer was not previously stopped
this
.
start
();
}
this
.
incrementInterval
();
this
.
resume
();
},
this
.
getCurrentInterval
());
}
onVisibilityVisible
()
{
this
.
cancel
();
this
.
start
();
}
// cancel the existing timer, setting the currentInterval back to startingInterval
cancel
()
{
this
.
setCurrentInterval
(
this
.
cfg
.
startingInterval
);
this
.
stopTimer
();
}
destroy
()
{
onVisibilityHidden
()
{
if
(
this
.
cfg
.
hiddenInterval
)
{
this
.
setCurrentInterval
(
this
.
cfg
.
hiddenInterval
);
this
.
resume
();
}
else
{
this
.
cancel
();
document
.
removeEventListener
(
'visibilitychange'
,
this
.
handleVisibilityChange
);
$
(
document
).
off
(
'visibilitychange'
).
off
(
'beforeunload'
);
}
}
/* private */
// start a timer, using the existing interval
resume
()
{
this
.
stopTimer
();
// stop exsiting timer, in case timer was not previously stopped
this
.
start
();
}
initInterval
()
{
const
cfg
=
this
.
cfg
;
onVisibilityVisible
()
{
this
.
cancel
();
this
.
start
();
}
if
(
!
cfg
.
lazyStart
)
{
this
.
start
();
}
destroy
()
{
this
.
cancel
();
document
.
removeEventListener
(
'visibilitychange'
,
this
.
handleVisibilityChange
);
$
(
document
).
off
(
'visibilitychange'
).
off
(
'beforeunload'
);
}
this
.
initVisibilityChangeHandling
();
this
.
initPageUnloadHandling
();
}
/* private */
initVisibilityChangeHandling
()
{
// cancel interval when tab no longer shown (prevents cached pages from polling)
document
.
addEventListener
(
'visibilitychange'
,
this
.
handleVisibilityChange
.
bind
(
this
));
}
initInterval
()
{
const
cfg
=
this
.
cfg
;
initPageUnloadHandling
()
{
// TODO: Consider refactoring in light of turbolinks removal.
// prevent interval continuing after page change, when kept in cache by Turbolinks
$
(
document
).
on
(
'beforeunload'
,
()
=>
this
.
cancel
());
if
(
!
cfg
.
lazyStart
)
{
this
.
start
();
}
handleVisibilityChange
(
e
)
{
this
.
state
.
pageVisibility
=
e
.
target
.
visibilityState
;
const
intervalAction
=
this
.
isPageVisible
()
?
this
.
onVisibilityVisible
:
this
.
onVisibilityHidden
;
this
.
initVisibilityChangeHandling
();
this
.
initPageUnloadHandling
();
}
intervalAction
.
apply
(
this
);
}
initVisibilityChangeHandling
()
{
// cancel interval when tab no longer shown (prevents cached pages from polling)
document
.
addEventListener
(
'visibilitychange'
,
this
.
handleVisibilityChange
.
bind
(
this
));
}
getCurrentInterval
()
{
return
this
.
state
.
currentInterval
;
}
initPageUnloadHandling
()
{
// TODO: Consider refactoring in light of turbolinks removal.
// prevent interval continuing after page change, when kept in cache by Turbolinks
$
(
document
).
on
(
'beforeunload'
,
()
=>
this
.
cancel
());
}
setCurrentInterval
(
newInterval
)
{
this
.
state
.
currentInterval
=
newInterval
;
}
handleVisibilityChange
(
e
)
{
this
.
state
.
pageVisibility
=
e
.
target
.
visibilityState
;
const
intervalAction
=
this
.
isPageVisible
()
?
this
.
onVisibilityVisible
:
this
.
onVisibilityHidden
;
incrementInterval
()
{
const
cfg
=
this
.
cfg
;
const
currentInterval
=
this
.
getCurrentInterval
();
if
(
cfg
.
hiddenInterval
&&
!
this
.
isPageVisible
())
return
;
let
nextInterval
=
currentInterval
*
cfg
.
incrementByFactorOf
;
intervalAction
.
apply
(
this
);
}
if
(
nextInterval
>
cfg
.
maxInterval
)
{
nextInterval
=
cfg
.
maxInterval
;
}
getCurrentInterval
()
{
return
this
.
state
.
currentInterval
;
}
setCurrentInterval
(
newInterval
)
{
this
.
state
.
currentInterval
=
newInterval
;
}
incrementInterval
()
{
const
cfg
=
this
.
cfg
;
const
currentInterval
=
this
.
getCurrentInterval
();
if
(
cfg
.
hiddenInterval
&&
!
this
.
isPageVisible
())
return
;
let
nextInterval
=
currentInterval
*
cfg
.
incrementByFactorOf
;
this
.
setCurrentInterval
(
nextInterval
);
if
(
nextInterval
>
cfg
.
maxInterval
)
{
nextInterval
=
cfg
.
maxInterval
;
}
isPageVisible
()
{
return
this
.
state
.
pageVisibility
===
'visible'
;
}
this
.
setCurrentInterval
(
nextInterval
);
}
stopTimer
()
{
const
state
=
this
.
state
;
isPageVisible
()
{
return
this
.
state
.
pageVisibility
===
'visible'
;
}
state
.
intervalId
=
window
.
clearInterval
(
state
.
intervalId
);
}
stopTimer
()
{
const
state
=
this
.
state
;
state
.
intervalId
=
window
.
clearInterval
(
state
.
intervalId
);
}
gl
.
SmartInterval
=
SmartInterval
;
})(
window
.
gl
||
(
window
.
gl
=
{}));
}
window
.
gl
.
SmartInterval
=
SmartInterval
;
app/assets/javascripts/snippets_list.js
View file @
c06dad62
/* eslint-disable arrow-parens, no-param-reassign, space-before-function-paren, func-names, no-var, max-len */
(
global
=>
{
global
.
gl
=
global
.
gl
||
{}
;
window
.
gl
.
SnippetsList
=
function
()
{
var
$holder
=
$
(
'.snippets-list-holder'
)
;
gl
.
SnippetsList
=
function
()
{
var
$holder
=
$
(
'.snippets-list-holder'
);
$holder
.
find
(
'.pagination'
).
on
(
'ajax:success'
,
(
e
,
data
)
=>
{
$holder
.
replaceWith
(
data
.
html
);
});
};
})(
window
);
$holder
.
find
(
'.pagination'
).
on
(
'ajax:success'
,
(
e
,
data
)
=>
{
$holder
.
replaceWith
(
data
.
html
);
});
};
app/assets/javascripts/star.js
View file @
c06dad62
/* eslint-disable func-names, space-before-function-paren, wrap-iife, no-unused-vars, one-var, no-var, one-var-declaration-per-line, prefer-arrow-callback, no-new, max-len */
/* global Flash */
(
function
()
{
this
.
Star
=
(
function
()
{
function
Star
()
{
$
(
'.project-home-panel .toggle-star'
).
on
(
'ajax:success'
,
function
(
e
,
data
,
status
,
xhr
)
{
var
$starIcon
,
$starSpan
,
$this
,
toggleStar
;
$this
=
$
(
this
);
$starSpan
=
$this
.
find
(
'span'
);
$starIcon
=
$this
.
find
(
'i'
);
toggleStar
=
function
(
isStarred
)
{
$this
.
parent
().
find
(
'.star-count'
).
text
(
data
.
star_count
);
if
(
isStarred
)
{
$starSpan
.
removeClass
(
'starred'
).
text
(
'Star'
);
$starIcon
.
removeClass
(
'fa-star'
).
addClass
(
'fa-star-o'
);
}
else
{
$starSpan
.
addClass
(
'starred'
).
text
(
'Unstar'
);
$starIcon
.
removeClass
(
'fa-star-o'
).
addClass
(
'fa-star'
);
}
};
toggleStar
(
$starSpan
.
hasClass
(
'starred'
));
}).
on
(
'ajax:error'
,
function
(
e
,
xhr
,
status
,
error
)
{
new
Flash
(
'Star toggle failed. Try again later.'
,
'alert'
);
});
}
window
.
Star
=
(
function
()
{
function
Star
()
{
$
(
'.project-home-panel .toggle-star'
).
on
(
'ajax:success'
,
function
(
e
,
data
,
status
,
xhr
)
{
var
$starIcon
,
$starSpan
,
$this
,
toggleStar
;
$this
=
$
(
this
);
$starSpan
=
$this
.
find
(
'span'
);
$starIcon
=
$this
.
find
(
'i'
);
toggleStar
=
function
(
isStarred
)
{
$this
.
parent
().
find
(
'.star-count'
).
text
(
data
.
star_count
);
if
(
isStarred
)
{
$starSpan
.
removeClass
(
'starred'
).
text
(
'Star'
);
$starIcon
.
removeClass
(
'fa-star'
).
addClass
(
'fa-star-o'
);
}
else
{
$starSpan
.
addClass
(
'starred'
).
text
(
'Unstar'
);
$starIcon
.
removeClass
(
'fa-star-o'
).
addClass
(
'fa-star'
);
}
};
toggleStar
(
$starSpan
.
hasClass
(
'starred'
));
}).
on
(
'ajax:error'
,
function
(
e
,
xhr
,
status
,
error
)
{
new
Flash
(
'Star toggle failed. Try again later.'
,
'alert'
);
});
}
return
Star
;
})();
}).
call
(
window
);
return
Star
;
})();
app/assets/javascripts/subscription.js
View file @
c06dad62
(()
=>
{
class
Subscription
{
constructor
(
containerElm
)
{
this
.
containerElm
=
containerElm
;
class
Subscription
{
constructor
(
containerElm
)
{
this
.
containerElm
=
containerElm
;
const
subscribeButton
=
containerElm
.
querySelector
(
'.js-subscribe-button'
);
if
(
subscribeButton
)
{
// remove class so we don't bind twice
subscribeButton
.
classList
.
remove
(
'js-subscribe-button'
);
subscribeButton
.
addEventListener
(
'click'
,
this
.
toggleSubscription
.
bind
(
this
));
}
const
subscribeButton
=
containerElm
.
querySelector
(
'.js-subscribe-button'
);
if
(
subscribeButton
)
{
// remove class so we don't bind twice
subscribeButton
.
classList
.
remove
(
'js-subscribe-button'
);
subscribeButton
.
addEventListener
(
'click'
,
this
.
toggleSubscription
.
bind
(
this
));
}
}
toggleSubscription
(
event
)
{
const
button
=
event
.
currentTarget
;
const
buttonSpan
=
button
.
querySelector
(
'span'
);
if
(
!
buttonSpan
||
button
.
classList
.
contains
(
'disabled'
))
{
return
;
}
button
.
classList
.
add
(
'disabled'
);
toggleSubscription
(
event
)
{
const
button
=
event
.
currentTarget
;
const
buttonSpan
=
button
.
querySelector
(
'span'
);
if
(
!
buttonSpan
||
button
.
classList
.
contains
(
'disabled'
))
{
return
;
}
button
.
classList
.
add
(
'disabled'
);
const
isSubscribed
=
buttonSpan
.
innerHTML
.
trim
().
toLowerCase
()
!==
'subscribe'
;
const
toggleActionUrl
=
this
.
containerElm
.
dataset
.
url
;
const
isSubscribed
=
buttonSpan
.
innerHTML
.
trim
().
toLowerCase
()
!==
'subscribe'
;
const
toggleActionUrl
=
this
.
containerElm
.
dataset
.
url
;
$
.
post
(
toggleActionUrl
,
()
=>
{
button
.
classList
.
remove
(
'disabled'
);
$
.
post
(
toggleActionUrl
,
()
=>
{
button
.
classList
.
remove
(
'disabled'
);
// hack to allow this to work with the issue boards Vue object
if
(
document
.
querySelector
(
'html'
).
classList
.
contains
(
'issue-boards-page'
))
{
gl
.
issueBoards
.
boardStoreIssueSet
(
'subscribed'
,
!
gl
.
issueBoards
.
BoardsStore
.
detail
.
issue
.
subscribed
,
);
}
else
{
buttonSpan
.
innerHTML
=
isSubscribed
?
'Subscribe'
:
'Unsubscribe'
;
}
});
}
// hack to allow this to work with the issue boards Vue object
if
(
document
.
querySelector
(
'html'
).
classList
.
contains
(
'issue-boards-page'
))
{
gl
.
issueBoards
.
boardStoreIssueSet
(
'subscribed'
,
!
gl
.
issueBoards
.
BoardsStore
.
detail
.
issue
.
subscribed
,
);
}
else
{
buttonSpan
.
innerHTML
=
isSubscribed
?
'Subscribe'
:
'Unsubscribe'
;
}
});
}
static
bindAll
(
selector
)
{
[].
forEach
.
call
(
document
.
querySelectorAll
(
selector
),
elm
=>
new
Subscription
(
elm
));
}
static
bindAll
(
selector
)
{
[].
forEach
.
call
(
document
.
querySelectorAll
(
selector
),
elm
=>
new
Subscription
(
elm
));
}
}
window
.
gl
=
window
.
gl
||
{};
window
.
gl
.
Subscription
=
Subscription
;
})();
window
.
gl
=
window
.
gl
||
{};
window
.
gl
.
Subscription
=
Subscription
;
app/assets/javascripts/subscription_select.js
View file @
c06dad62
/* eslint-disable func-names, space-before-function-paren, wrap-iife, no-var, quotes, object-shorthand, no-unused-vars, no-shadow, one-var, one-var-declaration-per-line, comma-dangle, max-len */
(
function
()
{
this
.
SubscriptionSelect
=
(
function
()
{
function
SubscriptionSelect
()
{
$
(
'.js-subscription-event'
).
each
(
function
(
i
,
el
)
{
var
fieldName
;
fieldName
=
$
(
el
).
data
(
"field-name"
);
return
$
(
el
).
glDropdown
({
selectable
:
true
,
fieldName
:
fieldName
,
toggleLabel
:
(
function
(
_this
)
{
return
function
(
selected
,
el
,
instance
)
{
var
$item
,
label
;
label
=
'Subscription'
;
$item
=
instance
.
dropdown
.
find
(
'.is-active'
);
if
(
$item
.
length
)
{
label
=
$item
.
text
();
}
return
label
;
};
})(
this
),
clicked
:
function
(
options
)
{
return
options
.
e
.
preventDefault
();
},
id
:
function
(
obj
,
el
)
{
return
$
(
el
).
data
(
"id"
);
}
});
window
.
SubscriptionSelect
=
(
function
()
{
function
SubscriptionSelect
()
{
$
(
'.js-subscription-event'
).
each
(
function
(
i
,
el
)
{
var
fieldName
;
fieldName
=
$
(
el
).
data
(
"field-name"
);
return
$
(
el
).
glDropdown
({
selectable
:
true
,
fieldName
:
fieldName
,
toggleLabel
:
(
function
(
_this
)
{
return
function
(
selected
,
el
,
instance
)
{
var
$item
,
label
;
label
=
'Subscription'
;
$item
=
instance
.
dropdown
.
find
(
'.is-active'
);
if
(
$item
.
length
)
{
label
=
$item
.
text
();
}
return
label
;
};
})(
this
),
clicked
:
function
(
options
)
{
return
options
.
e
.
preventDefault
();
},
id
:
function
(
obj
,
el
)
{
return
$
(
el
).
data
(
"id"
);
}
});
}
});
}
return
SubscriptionSelect
;
})();
}).
call
(
window
);
return
SubscriptionSelect
;
})();
app/assets/javascripts/syntax_highlight.js
View file @
c06dad62
...
...
@@ -9,19 +9,18 @@
//
// <div class="js-syntax-highlight"></div>
//
(
function
()
{
$
.
fn
.
syntaxHighlight
=
function
()
{
var
$children
;
if
(
$
(
this
).
hasClass
(
'js-syntax-highlight'
))
{
// Given the element itself, apply highlighting
return
$
(
this
).
addClass
(
gon
.
user_color_scheme
);
}
else
{
// Given a parent element, recurse to any of its applicable children
$children
=
$
(
this
).
find
(
'.js-syntax-highlight'
);
if
(
$children
.
length
)
{
return
$children
.
syntaxHighlight
();
}
$
.
fn
.
syntaxHighlight
=
function
()
{
var
$children
;
if
(
$
(
this
).
hasClass
(
'js-syntax-highlight'
))
{
// Given the element itself, apply highlighting
return
$
(
this
).
addClass
(
gon
.
user_color_scheme
);
}
else
{
// Given a parent element, recurse to any of its applicable children
$children
=
$
(
this
).
find
(
'.js-syntax-highlight'
);
if
(
$children
.
length
)
{
return
$children
.
syntaxHighlight
();
}
}
;
}
).
call
(
window
)
;
}
};
app/assets/javascripts/tree.js
View file @
c06dad62
/* eslint-disable func-names, space-before-function-paren, wrap-iife, max-len, quotes, consistent-return, no-var, one-var, one-var-declaration-per-line, no-else-return, prefer-arrow-callback, max-len */
(
function
()
{
this
.
TreeView
=
(
function
()
{
function
TreeView
()
{
this
.
initKeyNav
();
// Code browser tree slider
// Make the entire tree-item row clickable, but not if clicking another link (like a commit message)
$
(
".tree-content-holder .tree-item"
).
on
(
'click'
,
function
(
e
)
{
var
$clickedEl
,
path
;
$clickedEl
=
$
(
e
.
target
);
path
=
$
(
'.tree-item-file-name a'
,
this
).
attr
(
'href'
);
if
(
!
$clickedEl
.
is
(
'a'
)
&&
!
$clickedEl
.
is
(
'.str-truncated'
))
{
if
(
e
.
metaKey
||
e
.
which
===
2
)
{
e
.
preventDefault
();
return
window
.
open
(
path
,
'_blank'
);
}
else
{
return
gl
.
utils
.
visitUrl
(
path
);
}
window
.
TreeView
=
(
function
()
{
function
TreeView
()
{
this
.
initKeyNav
();
// Code browser tree slider
// Make the entire tree-item row clickable, but not if clicking another link (like a commit message)
$
(
".tree-content-holder .tree-item"
).
on
(
'click'
,
function
(
e
)
{
var
$clickedEl
,
path
;
$clickedEl
=
$
(
e
.
target
);
path
=
$
(
'.tree-item-file-name a'
,
this
).
attr
(
'href'
);
if
(
!
$clickedEl
.
is
(
'a'
)
&&
!
$clickedEl
.
is
(
'.str-truncated'
))
{
if
(
e
.
metaKey
||
e
.
which
===
2
)
{
e
.
preventDefault
();
return
window
.
open
(
path
,
'_blank'
);
}
else
{
return
gl
.
utils
.
visitUrl
(
path
);
}
});
// Show the "Loading commit data" for only the first element
$
(
'span.log_loading:first'
).
removeClass
(
'hide'
);
}
}
});
// Show the "Loading commit data" for only the first element
$
(
'span.log_loading:first'
).
removeClass
(
'hide'
);
}
TreeView
.
prototype
.
initKeyNav
=
function
()
{
var
li
,
liSelected
;
li
=
$
(
"tr.tree-item"
);
liSelected
=
null
;
return
$
(
'body'
).
keydown
(
function
(
e
)
{
var
next
,
path
;
if
(
$
(
"input:focus"
).
length
>
0
&&
(
e
.
which
===
38
||
e
.
which
===
40
))
{
return
false
;
}
if
(
e
.
which
===
40
)
{
if
(
liSelected
)
{
next
=
liSelected
.
next
();
if
(
next
.
length
>
0
)
{
liSelected
.
removeClass
(
"selected"
);
liSelected
=
next
.
addClass
(
"selected"
);
}
}
else
{
liSelected
=
li
.
eq
(
0
).
addClass
(
"selected"
);
}
return
$
(
liSelected
).
focus
();
}
else
if
(
e
.
which
===
38
)
{
if
(
liSelected
)
{
next
=
liSelected
.
prev
();
if
(
next
.
length
>
0
)
{
liSelected
.
removeClass
(
"selected"
);
liSelected
=
next
.
addClass
(
"selected"
);
}
}
else
{
liSelected
=
li
.
last
().
addClass
(
"selected"
);
TreeView
.
prototype
.
initKeyNav
=
function
()
{
var
li
,
liSelected
;
li
=
$
(
"tr.tree-item"
);
liSelected
=
null
;
return
$
(
'body'
).
keydown
(
function
(
e
)
{
var
next
,
path
;
if
(
$
(
"input:focus"
).
length
>
0
&&
(
e
.
which
===
38
||
e
.
which
===
40
))
{
return
false
;
}
if
(
e
.
which
===
40
)
{
if
(
liSelected
)
{
next
=
liSelected
.
next
();
if
(
next
.
length
>
0
)
{
liSelected
.
removeClass
(
"selected"
);
liSelected
=
next
.
addClass
(
"selected"
);
}
return
$
(
liSelected
).
focus
();
}
else
if
(
e
.
which
===
13
)
{
path
=
$
(
'.tree-item.selected .tree-item-file-name a'
).
attr
(
'href'
);
if
(
path
)
{
return
gl
.
utils
.
visitUrl
(
path
);
}
else
{
liSelected
=
li
.
eq
(
0
).
addClass
(
"selected"
);
}
return
$
(
liSelected
).
focus
();
}
else
if
(
e
.
which
===
38
)
{
if
(
liSelected
)
{
next
=
liSelected
.
prev
();
if
(
next
.
length
>
0
)
{
liSelected
.
removeClass
(
"selected"
);
liSelected
=
next
.
addClass
(
"selected"
);
}
}
else
{
liSelected
=
li
.
last
().
addClass
(
"selected"
);
}
return
$
(
liSelected
).
focus
();
}
else
if
(
e
.
which
===
13
)
{
path
=
$
(
'.tree-item.selected .tree-item-file-name a'
).
attr
(
'href'
);
if
(
path
)
{
return
gl
.
utils
.
visitUrl
(
path
);
}
});
};
}
});
};
return
TreeView
;
})();
}).
call
(
window
);
return
TreeView
;
})();
app/assets/javascripts/user.js
View file @
c06dad62
...
...
@@ -2,34 +2,35 @@
import
Cookies
from
'js-cookie'
;
((
global
)
=>
{
global
.
User
=
class
{
constructor
({
action
})
{
this
.
action
=
action
;
this
.
placeProfileAvatarsToTop
();
this
.
initTabs
();
this
.
hideProjectLimitMessage
();
}
class
User
{
constructor
({
action
})
{
this
.
action
=
action
;
this
.
placeProfileAvatarsToTop
();
this
.
initTabs
();
this
.
hideProjectLimitMessage
();
}
placeProfileAvatarsToTop
()
{
$
(
'.profile-groups-avatars'
).
tooltip
({
placement
:
'top'
});
}
placeProfileAvatarsToTop
()
{
$
(
'.profile-groups-avatars'
).
tooltip
({
placement
:
'top'
});
}
initTabs
()
{
return
new
globa
l
.
UserTabs
({
parentEl
:
'.user-profile'
,
action
:
this
.
action
});
}
initTabs
()
{
return
new
window
.
g
l
.
UserTabs
({
parentEl
:
'.user-profile'
,
action
:
this
.
action
});
}
hideProjectLimitMessage
()
{
$
(
'.hide-project-limit-message'
).
on
(
'click'
,
e
=>
{
e
.
preventDefault
();
Cookies
.
set
(
'hide_project_limit_message'
,
'false'
);
$
(
this
).
parents
(
'.project-limit-message'
).
remove
();
});
}
};
})(
window
.
gl
||
(
window
.
gl
=
{}));
hideProjectLimitMessage
()
{
$
(
'.hide-project-limit-message'
).
on
(
'click'
,
e
=>
{
e
.
preventDefault
();
Cookies
.
set
(
'hide_project_limit_message'
,
'false'
);
$
(
this
).
parents
(
'.project-limit-message'
).
remove
();
});
}
}
window
.
gl
=
window
.
gl
||
{};
window
.
gl
.
User
=
User
;
app/assets/javascripts/user_tabs.js
View file @
c06dad62
...
...
@@ -59,117 +59,118 @@ content on the Users#show page.
</div>
</div>
*/
((
global
)
=>
{
class
UserTabs
{
constructor
({
defaultAction
,
action
,
parentEl
})
{
this
.
loaded
=
{};
this
.
defaultAction
=
defaultAction
||
'activity'
;
this
.
action
=
action
||
this
.
defaultAction
;
this
.
$parentEl
=
$
(
parentEl
)
||
$
(
document
);
this
.
_location
=
window
.
location
;
this
.
$parentEl
.
find
(
'.nav-links a'
)
.
each
((
i
,
navLink
)
=>
{
this
.
loaded
[
$
(
navLink
).
attr
(
'data-action'
)]
=
false
;
});
this
.
actions
=
Object
.
keys
(
this
.
loaded
);
this
.
bindEvents
();
if
(
this
.
action
===
'show'
)
{
this
.
action
=
this
.
defaultAction
;
}
this
.
activateTab
(
this
.
action
);
class
UserTabs
{
constructor
({
defaultAction
,
action
,
parentEl
})
{
this
.
loaded
=
{};
this
.
defaultAction
=
defaultAction
||
'activity'
;
this
.
action
=
action
||
this
.
defaultAction
;
this
.
$parentEl
=
$
(
parentEl
)
||
$
(
document
);
this
.
_location
=
window
.
location
;
this
.
$parentEl
.
find
(
'.nav-links a'
)
.
each
((
i
,
navLink
)
=>
{
this
.
loaded
[
$
(
navLink
).
attr
(
'data-action'
)]
=
false
;
});
this
.
actions
=
Object
.
keys
(
this
.
loaded
);
this
.
bindEvents
();
if
(
this
.
action
===
'show'
)
{
this
.
action
=
this
.
defaultAction
;
}
bindEvents
()
{
this
.
changeProjectsPageWrapper
=
this
.
changeProjectsPage
.
bind
(
this
);
this
.
activateTab
(
this
.
action
);
}
this
.
$parentEl
.
off
(
'shown.bs.tab'
,
'.nav-links a[data-toggle="tab"]'
)
.
on
(
'shown.bs.tab'
,
'.nav-links a[data-toggle="tab"]'
,
event
=>
this
.
tabShown
(
event
)
);
bindEvents
()
{
this
.
changeProjectsPageWrapper
=
this
.
changeProjectsPage
.
bind
(
this
);
this
.
$parentEl
.
on
(
'click'
,
'.gl-pagination a'
,
this
.
changeProjectsPageWrapper
);
}
this
.
$parentEl
.
off
(
'shown.bs.tab'
,
'.nav-links a[data-toggle="tab"]'
)
.
on
(
'shown.bs.tab'
,
'.nav-links a[data-toggle="tab"]'
,
event
=>
this
.
tabShown
(
event
));
changeProjectsPage
(
e
)
{
e
.
preventDefault
();
this
.
$parentEl
.
on
(
'click'
,
'.gl-pagination a'
,
this
.
changeProjectsPageWrapper
);
}
$
(
'.tab-pane.active'
).
empty
();
const
endpoint
=
$
(
e
.
target
).
attr
(
'href'
);
this
.
loadTab
(
this
.
getCurrentAction
(),
endpoint
);
}
changeProjectsPage
(
e
)
{
e
.
preventDefault
();
tabShown
(
event
)
{
const
$target
=
$
(
event
.
target
);
const
action
=
$target
.
data
(
'action'
);
const
source
=
$target
.
attr
(
'href'
);
const
endpoint
=
$target
.
data
(
'endpoint'
);
this
.
setTab
(
action
,
endpoint
);
return
this
.
setCurrentAction
(
source
);
}
$
(
'.tab-pane.active'
).
empty
();
const
endpoint
=
$
(
e
.
target
).
attr
(
'href'
);
this
.
loadTab
(
this
.
getCurrentAction
(),
endpoint
);
}
activateTab
(
action
)
{
return
this
.
$parentEl
.
find
(
`.nav-links .js-
${
action
}
-tab a`
)
.
tab
(
'show'
);
}
tabShown
(
event
)
{
const
$target
=
$
(
event
.
target
);
const
action
=
$target
.
data
(
'action'
);
const
source
=
$target
.
attr
(
'href'
);
const
endpoint
=
$target
.
data
(
'endpoint'
);
this
.
setTab
(
action
,
endpoint
);
return
this
.
setCurrentAction
(
source
);
}
setTab
(
action
,
endpoint
)
{
if
(
this
.
loaded
[
action
])
{
return
;
}
if
(
action
===
'activity'
)
{
this
.
loadActivities
();
}
activateTab
(
action
)
{
return
this
.
$parentEl
.
find
(
`.nav-links .js-
${
action
}
-tab a`
)
.
tab
(
'show'
);
}
const
loadableActions
=
[
'groups'
,
'contributed'
,
'projects'
,
'snippets'
];
if
(
loadableActions
.
indexOf
(
action
)
>
-
1
)
{
return
this
.
loadTab
(
action
,
endpoint
);
}
setTab
(
action
,
endpoint
)
{
if
(
this
.
loaded
[
action
])
{
return
;
}
if
(
action
===
'activity'
)
{
this
.
loadActivities
();
}
loadTab
(
action
,
endpoint
)
{
return
$
.
ajax
({
beforeSend
:
()
=>
this
.
toggleLoading
(
true
),
complete
:
()
=>
this
.
toggleLoading
(
false
),
dataType
:
'json'
,
type
:
'GET'
,
url
:
endpoint
,
success
:
(
data
)
=>
{
const
tabSelector
=
`div#
${
action
}
`
;
this
.
$parentEl
.
find
(
tabSelector
).
html
(
data
.
html
);
this
.
loaded
[
action
]
=
true
;
return
gl
.
utils
.
localTimeAgo
(
$
(
'.js-timeago'
,
tabSelector
));
}
});
const
loadableActions
=
[
'groups'
,
'contributed'
,
'projects'
,
'snippets'
];
if
(
loadableActions
.
indexOf
(
action
)
>
-
1
)
{
return
this
.
loadTab
(
action
,
endpoint
);
}
}
loadActivities
()
{
if
(
this
.
loaded
[
'activity'
])
{
return
;
loadTab
(
action
,
endpoint
)
{
return
$
.
ajax
({
beforeSend
:
()
=>
this
.
toggleLoading
(
true
),
complete
:
()
=>
this
.
toggleLoading
(
false
),
dataType
:
'json'
,
type
:
'GET'
,
url
:
endpoint
,
success
:
(
data
)
=>
{
const
tabSelector
=
`div#
${
action
}
`
;
this
.
$parentEl
.
find
(
tabSelector
).
html
(
data
.
html
);
this
.
loaded
[
action
]
=
true
;
return
gl
.
utils
.
localTimeAgo
(
$
(
'.js-timeago'
,
tabSelector
));
}
const
$calendarWrap
=
this
.
$parentEl
.
find
(
'.user-calendar'
);
$calendarWrap
.
load
(
$calendarWrap
.
data
(
'href'
));
new
gl
.
Activities
();
return
this
.
loaded
[
'activity'
]
=
true
;
}
});
}
toggleLoading
(
status
)
{
return
this
.
$parentEl
.
find
(
'.loading-status .loading'
)
.
toggle
(
status
)
;
loadActivities
(
)
{
if
(
this
.
loaded
[
'activity'
])
{
return
;
}
const
$calendarWrap
=
this
.
$parentEl
.
find
(
'.user-calendar'
);
$calendarWrap
.
load
(
$calendarWrap
.
data
(
'href'
));
new
gl
.
Activities
();
return
this
.
loaded
[
'activity'
]
=
true
;
}
setCurrentAction
(
source
)
{
let
new_state
=
source
;
new_state
=
new_state
.
replace
(
/
\/
+$/
,
''
);
new_state
+=
this
.
_location
.
search
+
this
.
_location
.
hash
;
history
.
replaceState
({
url
:
new_state
},
document
.
title
,
new_state
);
return
new_state
;
}
toggleLoading
(
status
)
{
return
this
.
$parentEl
.
find
(
'.loading-status .loading'
)
.
toggle
(
status
);
}
getCurrentAction
()
{
return
this
.
$parentEl
.
find
(
'.nav-links .active a'
).
data
(
'action'
);
}
setCurrentAction
(
source
)
{
let
new_state
=
source
;
new_state
=
new_state
.
replace
(
/
\/
+$/
,
''
);
new_state
+=
this
.
_location
.
search
+
this
.
_location
.
hash
;
history
.
replaceState
({
url
:
new_state
},
document
.
title
,
new_state
);
return
new_state
;
}
global
.
UserTabs
=
UserTabs
;
})(
window
.
gl
||
(
window
.
gl
=
{}));
getCurrentAction
()
{
return
this
.
$parentEl
.
find
(
'.nav-links .active a'
).
data
(
'action'
);
}
}
window
.
gl
=
window
.
gl
||
{};
window
.
gl
.
UserTabs
=
UserTabs
;
app/assets/javascripts/username_validator.js
View file @
c06dad62
/* eslint-disable comma-dangle, consistent-return, class-methods-use-this, arrow-parens, no-param-reassign, max-len */
((
global
)
=>
{
const
debounceTimeoutDuration
=
1000
;
const
invalidInputClass
=
'gl-field-error-outline'
;
const
successInputClass
=
'gl-field-success-outline'
;
const
unavailableMessageSelector
=
'.username .validation-error'
;
const
successMessageSelector
=
'.username .validation-success'
;
const
pendingMessageSelector
=
'.username .validation-pending'
;
const
invalidMessageSelector
=
'.username .gl-field-error'
;
class
UsernameValidator
{
constructor
()
{
this
.
inputElement
=
$
(
'#new_user_username'
);
this
.
inputDomElement
=
this
.
inputElement
.
get
(
0
);
this
.
state
=
{
available
:
false
,
valid
:
false
,
pending
:
false
,
empty
:
true
};
const
debounceTimeout
=
_
.
debounce
((
username
)
=>
{
this
.
validateUsername
(
username
);
},
debounceTimeoutDuration
);
this
.
inputElement
.
on
(
'keyup.username_check'
,
()
=>
{
const
username
=
this
.
inputElement
.
val
();
this
.
state
.
valid
=
this
.
inputDomElement
.
validity
.
valid
;
this
.
state
.
empty
=
!
username
.
length
;
if
(
this
.
state
.
valid
)
{
return
debounceTimeout
(
username
);
}
this
.
renderState
();
});
// Override generic field validation
this
.
inputElement
.
on
(
'invalid'
,
this
.
interceptInvalid
.
bind
(
this
));
}
const
debounceTimeoutDuration
=
1000
;
const
invalidInputClass
=
'gl-field-error-outline'
;
const
successInputClass
=
'gl-field-success-outline'
;
const
unavailableMessageSelector
=
'.username .validation-error'
;
const
successMessageSelector
=
'.username .validation-success'
;
const
pendingMessageSelector
=
'.username .validation-pending'
;
const
invalidMessageSelector
=
'.username .gl-field-error'
;
class
UsernameValidator
{
constructor
()
{
this
.
inputElement
=
$
(
'#new_user_username'
);
this
.
inputDomElement
=
this
.
inputElement
.
get
(
0
);
this
.
state
=
{
available
:
false
,
valid
:
false
,
pending
:
false
,
empty
:
true
};
const
debounceTimeout
=
_
.
debounce
((
username
)
=>
{
this
.
validateUsername
(
username
);
},
debounceTimeoutDuration
);
this
.
inputElement
.
on
(
'keyup.username_check'
,
()
=>
{
const
username
=
this
.
inputElement
.
val
();
this
.
state
.
valid
=
this
.
inputDomElement
.
validity
.
valid
;
this
.
state
.
empty
=
!
username
.
length
;
renderState
()
{
// Clear all state
this
.
clearFieldValidationState
();
if
(
this
.
state
.
valid
&&
this
.
state
.
available
)
{
return
this
.
setSuccessState
();
if
(
this
.
state
.
valid
)
{
return
debounceTimeout
(
username
);
}
if
(
this
.
state
.
empty
)
{
return
this
.
clearFieldValidationState
();
}
this
.
renderState
();
});
if
(
this
.
state
.
pending
)
{
return
this
.
setPendingState
(
);
}
// Override generic field validation
this
.
inputElement
.
on
(
'invalid'
,
this
.
interceptInvalid
.
bind
(
this
)
);
}
if
(
!
this
.
state
.
available
)
{
return
this
.
setUnavailableState
();
}
renderState
(
)
{
// Clear all state
this
.
clearFieldValidationState
();
if
(
!
this
.
state
.
valid
)
{
return
this
.
setInvalidState
();
}
if
(
this
.
state
.
valid
&&
this
.
state
.
available
)
{
return
this
.
setSuccessState
();
}
interceptInvalid
(
event
)
{
event
.
preventDefault
();
event
.
stopPropagation
();
if
(
this
.
state
.
empty
)
{
return
this
.
clearFieldValidationState
();
}
validateUsername
(
username
)
{
if
(
this
.
state
.
valid
)
{
this
.
state
.
pending
=
true
;
this
.
state
.
available
=
false
;
this
.
renderState
();
return
$
.
ajax
({
type
:
'GET'
,
url
:
`
${
gon
.
relative_url_root
}
/users/
${
username
}
/exists`
,
dataType
:
'json'
,
success
:
(
res
)
=>
this
.
setAvailabilityState
(
res
.
exists
)
});
}
if
(
this
.
state
.
pending
)
{
return
this
.
setPendingState
();
}
setAvailabilityState
(
usernameTaken
)
{
if
(
usernameTaken
)
{
this
.
state
.
valid
=
false
;
this
.
state
.
available
=
false
;
}
else
{
this
.
state
.
available
=
true
;
}
this
.
state
.
pending
=
false
;
this
.
renderState
();
if
(
!
this
.
state
.
available
)
{
return
this
.
setUnavailableState
();
}
clearFieldValidationState
()
{
this
.
inputElement
.
siblings
(
'p'
).
hide
();
this
.
inputElement
.
removeClass
(
invalidInputClass
)
.
removeClass
(
successInputClass
);
if
(
!
this
.
state
.
valid
)
{
return
this
.
setInvalidState
();
}
}
setUnavailableState
()
{
const
$usernameUnavailableMessage
=
this
.
inputElement
.
siblings
(
unavailableMessageSelector
);
this
.
inputElement
.
addClass
(
invalidInputClass
).
removeClass
(
successInputClass
);
$usernameUnavailableMessage
.
show
();
}
interceptInvalid
(
event
)
{
event
.
preventDefault
();
event
.
stopPropagation
();
}
setSuccessState
()
{
const
$usernameSuccessMessage
=
this
.
inputElement
.
siblings
(
successMessageSelector
);
this
.
inputElement
.
addClass
(
successInputClass
).
removeClass
(
invalidInputClass
);
$usernameSuccessMessage
.
show
();
validateUsername
(
username
)
{
if
(
this
.
state
.
valid
)
{
this
.
state
.
pending
=
true
;
this
.
state
.
available
=
false
;
this
.
renderState
();
return
$
.
ajax
({
type
:
'GET'
,
url
:
`
${
gon
.
relative_url_root
}
/users/
${
username
}
/exists`
,
dataType
:
'json'
,
success
:
(
res
)
=>
this
.
setAvailabilityState
(
res
.
exists
)
});
}
}
setPendingState
()
{
const
$usernamePendingMessage
=
$
(
pendingMessageSelector
);
if
(
this
.
state
.
pending
)
{
$usernamePendingMessage
.
show
();
}
else
{
$usernamePendingMessage
.
hide
();
}
setAvailabilityState
(
usernameTaken
)
{
if
(
usernameTaken
)
{
this
.
state
.
valid
=
false
;
this
.
state
.
available
=
false
;
}
else
{
this
.
state
.
available
=
true
;
}
this
.
state
.
pending
=
false
;
this
.
renderState
();
}
clearFieldValidationState
()
{
this
.
inputElement
.
siblings
(
'p'
).
hide
();
setInvalidState
()
{
const
$inputErrorMessage
=
$
(
invalidMessageSelector
);
this
.
inputElement
.
addClass
(
invalidInputClass
).
removeClass
(
successInputClass
);
$inputErrorMessage
.
show
();
this
.
inputElement
.
removeClass
(
invalidInputClass
)
.
removeClass
(
successInputClass
);
}
setUnavailableState
()
{
const
$usernameUnavailableMessage
=
this
.
inputElement
.
siblings
(
unavailableMessageSelector
);
this
.
inputElement
.
addClass
(
invalidInputClass
).
removeClass
(
successInputClass
);
$usernameUnavailableMessage
.
show
();
}
setSuccessState
()
{
const
$usernameSuccessMessage
=
this
.
inputElement
.
siblings
(
successMessageSelector
);
this
.
inputElement
.
addClass
(
successInputClass
).
removeClass
(
invalidInputClass
);
$usernameSuccessMessage
.
show
();
}
setPendingState
()
{
const
$usernamePendingMessage
=
$
(
pendingMessageSelector
);
if
(
this
.
state
.
pending
)
{
$usernamePendingMessage
.
show
();
}
else
{
$usernamePendingMessage
.
hide
();
}
}
global
.
UsernameValidator
=
UsernameValidator
;
})(
window
);
setInvalidState
()
{
const
$inputErrorMessage
=
$
(
invalidMessageSelector
);
this
.
inputElement
.
addClass
(
invalidInputClass
).
removeClass
(
successInputClass
);
$inputErrorMessage
.
show
();
}
}
window
.
UsernameValidator
=
UsernameValidator
;
app/assets/javascripts/visibility_select.js
View file @
c06dad62
(()
=>
{
const
gl
=
window
.
gl
||
(
window
.
gl
=
{});
class
VisibilitySelect
{
constructor
(
container
)
{
if
(
!
container
)
throw
new
Error
(
'VisibilitySelect requires a container element as argument 1'
);
this
.
container
=
container
;
this
.
helpBlock
=
this
.
container
.
querySelector
(
'.help-block'
);
this
.
select
=
this
.
container
.
querySelector
(
'select'
);
}
class
VisibilitySelect
{
constructor
(
container
)
{
if
(
!
container
)
throw
new
Error
(
'VisibilitySelect requires a container element as argument 1'
);
this
.
container
=
container
;
this
.
helpBlock
=
this
.
container
.
querySelector
(
'.help-block'
);
this
.
select
=
this
.
container
.
querySelector
(
'select'
);
}
init
()
{
if
(
this
.
select
)
{
this
.
updateHelpText
();
this
.
select
.
addEventListener
(
'change'
,
this
.
updateHelpText
.
bind
(
this
));
}
else
{
this
.
helpBlock
.
textContent
=
this
.
container
.
querySelector
(
'.js-locked'
).
dataset
.
helpBlock
;
}
init
()
{
if
(
this
.
select
)
{
this
.
updateHelpText
();
this
.
select
.
addEventListener
(
'change'
,
this
.
updateHelpText
.
bind
(
this
));
}
else
{
this
.
helpBlock
.
textContent
=
this
.
container
.
querySelector
(
'.js-locked'
).
dataset
.
helpBlock
;
}
}
updateHelpText
()
{
this
.
helpBlock
.
textContent
=
this
.
select
.
querySelector
(
'option:checked'
).
dataset
.
description
;
}
updateHelpText
()
{
this
.
helpBlock
.
textContent
=
this
.
select
.
querySelector
(
'option:checked'
).
dataset
.
description
;
}
}
gl
.
VisibilitySelect
=
VisibilitySelect
;
})()
;
window
.
gl
=
window
.
gl
||
{}
;
window
.
gl
.
VisibilitySelect
=
VisibilitySelect
;
app/assets/javascripts/wikis.js
View file @
c06dad62
...
...
@@ -4,66 +4,65 @@
import
'vendor/jquery.nicescroll'
;
import
'./breakpoints'
;
((
global
)
=>
{
class
Wikis
{
constructor
()
{
this
.
bp
=
Breakpoints
.
get
();
this
.
sidebarEl
=
document
.
querySelector
(
'.js-wiki-sidebar'
);
this
.
sidebarExpanded
=
false
;
$
(
this
.
sidebarEl
).
niceScroll
();
class
Wikis
{
constructor
()
{
this
.
bp
=
Breakpoints
.
get
();
this
.
sidebarEl
=
document
.
querySelector
(
'.js-wiki-sidebar'
);
this
.
sidebarExpanded
=
false
;
$
(
this
.
sidebarEl
).
niceScroll
();
const
sidebarToggles
=
document
.
querySelectorAll
(
'.js-sidebar-wiki-toggle'
);
for
(
let
i
=
0
;
i
<
sidebarToggles
.
length
;
i
+=
1
)
{
sidebarToggles
[
i
].
addEventListener
(
'click'
,
e
=>
this
.
handleToggleSidebar
(
e
));
}
this
.
newWikiForm
=
document
.
querySelector
(
'form.new-wiki-page'
);
if
(
this
.
newWikiForm
)
{
this
.
newWikiForm
.
addEventListener
(
'submit'
,
e
=>
this
.
handleNewWikiSubmit
(
e
));
}
const
sidebarToggles
=
document
.
querySelectorAll
(
'.js-sidebar-wiki-toggle'
);
for
(
let
i
=
0
;
i
<
sidebarToggles
.
length
;
i
+=
1
)
{
sidebarToggles
[
i
].
addEventListener
(
'click'
,
e
=>
this
.
handleToggleSidebar
(
e
));
}
window
.
addEventListener
(
'resize'
,
()
=>
this
.
renderSidebar
());
this
.
renderSidebar
();
this
.
newWikiForm
=
document
.
querySelector
(
'form.new-wiki-page'
);
if
(
this
.
newWikiForm
)
{
this
.
newWikiForm
.
addEventListener
(
'submit'
,
e
=>
this
.
handleNewWikiSubmit
(
e
));
}
handleNewWikiSubmit
(
e
)
{
if
(
!
this
.
newWikiForm
)
return
;
window
.
addEventListener
(
'resize'
,
()
=>
this
.
renderSidebar
());
this
.
renderSidebar
();
}
const
slugInput
=
this
.
newWikiForm
.
querySelector
(
'#new_wiki_path'
);
const
slug
=
gl
.
text
.
slugify
(
slugInput
.
value
)
;
handleNewWikiSubmit
(
e
)
{
if
(
!
this
.
newWikiForm
)
return
;
if
(
slug
.
length
>
0
)
{
const
wikisPath
=
slugInput
.
getAttribute
(
'data-wikis-path'
);
window
.
location
.
href
=
`
${
wikisPath
}
/
${
slug
}
`
;
e
.
preventDefault
();
}
}
const
slugInput
=
this
.
newWikiForm
.
querySelector
(
'#new_wiki_path'
);
const
slug
=
gl
.
text
.
slugify
(
slugInput
.
value
);
handleToggleSidebar
(
e
)
{
if
(
slug
.
length
>
0
)
{
const
wikisPath
=
slugInput
.
getAttribute
(
'data-wikis-path'
);
window
.
location
.
href
=
`
${
wikisPath
}
/
${
slug
}
`
;
e
.
preventDefault
();
this
.
sidebarExpanded
=
!
this
.
sidebarExpanded
;
this
.
renderSidebar
();
}
}
sidebarCanCollapse
()
{
const
bootstrapBreakpoint
=
this
.
bp
.
getBreakpointSize
();
return
bootstrapBreakpoint
===
'xs'
||
bootstrapBreakpoint
===
'sm'
;
}
handleToggleSidebar
(
e
)
{
e
.
preventDefault
();
this
.
sidebarExpanded
=
!
this
.
sidebarExpanded
;
this
.
renderSidebar
();
}
sidebarCanCollapse
()
{
const
bootstrapBreakpoint
=
this
.
bp
.
getBreakpointSize
();
return
bootstrapBreakpoint
===
'xs'
||
bootstrapBreakpoint
===
'sm'
;
}
renderSidebar
()
{
if
(
!
this
.
sidebarEl
)
return
;
const
{
classList
}
=
this
.
sidebarEl
;
if
(
this
.
sidebarExpanded
||
!
this
.
sidebarCanCollapse
())
{
if
(
!
classList
.
contains
(
'right-sidebar-expanded'
))
{
classList
.
remove
(
'right-sidebar-collapsed'
);
classList
.
add
(
'right-sidebar-expanded'
);
}
}
else
if
(
classList
.
contains
(
'right-sidebar-expanded'
))
{
classList
.
add
(
'right-sidebar-collapsed'
);
classList
.
remove
(
'right-sidebar-expanded'
);
renderSidebar
()
{
if
(
!
this
.
sidebarEl
)
return
;
const
{
classList
}
=
this
.
sidebarEl
;
if
(
this
.
sidebarExpanded
||
!
this
.
sidebarCanCollapse
())
{
if
(
!
classList
.
contains
(
'right-sidebar-expanded'
))
{
classList
.
remove
(
'right-sidebar-collapsed'
);
classList
.
add
(
'right-sidebar-expanded'
);
}
}
else
if
(
classList
.
contains
(
'right-sidebar-expanded'
))
{
classList
.
add
(
'right-sidebar-collapsed'
);
classList
.
remove
(
'right-sidebar-expanded'
);
}
}
}
global
.
Wikis
=
Wikis
;
})(
window
.
gl
||
(
window
.
gl
=
{}))
;
window
.
gl
=
window
.
gl
||
{}
;
window
.
gl
.
Wikis
=
Wikis
;
app/assets/javascripts/zen_mode.js
View file @
c06dad62
...
...
@@ -34,65 +34,64 @@ window.Dropzone = Dropzone;
// **Cancelable** No
// **Target** a.js-zen-leave
//
(
function
()
{
this
.
ZenMode
=
(
function
()
{
function
ZenMode
()
{
this
.
active_backdrop
=
null
;
this
.
active_textarea
=
null
;
$
(
document
).
on
(
'click'
,
'.js-zen-enter'
,
function
(
e
)
{
e
.
preventDefault
();
return
$
(
e
.
currentTarget
).
trigger
(
'zen_mode:enter'
);
});
$
(
document
).
on
(
'click'
,
'.js-zen-leave'
,
function
(
e
)
{
window
.
ZenMode
=
(
function
()
{
function
ZenMode
()
{
this
.
active_backdrop
=
null
;
this
.
active_textarea
=
null
;
$
(
document
).
on
(
'click'
,
'.js-zen-enter'
,
function
(
e
)
{
e
.
preventDefault
();
return
$
(
e
.
currentTarget
).
trigger
(
'zen_mode:enter'
);
});
$
(
document
).
on
(
'click'
,
'.js-zen-leave'
,
function
(
e
)
{
e
.
preventDefault
();
return
$
(
e
.
currentTarget
).
trigger
(
'zen_mode:leave'
);
});
$
(
document
).
on
(
'zen_mode:enter'
,
(
function
(
_this
)
{
return
function
(
e
)
{
return
_this
.
enter
(
$
(
e
.
target
).
closest
(
'.md-area'
).
find
(
'.zen-backdrop'
));
};
})(
this
));
$
(
document
).
on
(
'zen_mode:leave'
,
(
function
(
_this
)
{
return
function
(
e
)
{
return
_this
.
exit
();
};
})(
this
));
$
(
document
).
on
(
'keydown'
,
function
(
e
)
{
// Esc
if
(
e
.
keyCode
===
27
)
{
e
.
preventDefault
();
return
$
(
e
.
currentTarget
).
trigger
(
'zen_mode:leave'
);
});
$
(
document
).
on
(
'zen_mode:enter'
,
(
function
(
_this
)
{
return
function
(
e
)
{
return
_this
.
enter
(
$
(
e
.
target
).
closest
(
'.md-area'
).
find
(
'.zen-backdrop'
));
};
})(
this
));
$
(
document
).
on
(
'zen_mode:leave'
,
(
function
(
_this
)
{
return
function
(
e
)
{
return
_this
.
exit
();
};
})(
this
));
$
(
document
).
on
(
'keydown'
,
function
(
e
)
{
// Esc
if
(
e
.
keyCode
===
27
)
{
e
.
preventDefault
();
return
$
(
document
).
trigger
(
'zen_mode:leave'
);
}
});
}
return
$
(
document
).
trigger
(
'zen_mode:leave'
);
}
});
}
ZenMode
.
prototype
.
enter
=
function
(
backdrop
)
{
Mousetrap
.
pause
();
this
.
active_backdrop
=
$
(
backdrop
);
this
.
active_backdrop
.
addClass
(
'fullscreen'
);
this
.
active_textarea
=
this
.
active_backdrop
.
find
(
'textarea'
);
// Prevent a user-resized textarea from persisting to fullscreen
this
.
active_textarea
.
removeAttr
(
'style'
);
return
this
.
active_textarea
.
focus
();
};
ZenMode
.
prototype
.
enter
=
function
(
backdrop
)
{
Mousetrap
.
pause
();
this
.
active_backdrop
=
$
(
backdrop
);
this
.
active_backdrop
.
addClass
(
'fullscreen'
);
this
.
active_textarea
=
this
.
active_backdrop
.
find
(
'textarea'
);
// Prevent a user-resized textarea from persisting to fullscreen
this
.
active_textarea
.
removeAttr
(
'style'
);
return
this
.
active_textarea
.
focus
();
};
ZenMode
.
prototype
.
exit
=
function
()
{
if
(
this
.
active_textarea
)
{
Mousetrap
.
unpause
();
this
.
active_textarea
.
closest
(
'.zen-backdrop'
).
removeClass
(
'fullscreen'
);
this
.
scrollTo
(
this
.
active_textarea
);
this
.
active_textarea
=
null
;
this
.
active_backdrop
=
null
;
return
Dropzone
.
forElement
(
'.div-dropzone'
).
enable
();
}
};
ZenMode
.
prototype
.
exit
=
function
()
{
if
(
this
.
active_textarea
)
{
Mousetrap
.
unpause
();
this
.
active_textarea
.
closest
(
'.zen-backdrop'
).
removeClass
(
'fullscreen'
);
this
.
scrollTo
(
this
.
active_textarea
);
this
.
active_textarea
=
null
;
this
.
active_backdrop
=
null
;
return
Dropzone
.
forElement
(
'.div-dropzone'
).
enable
();
}
};
ZenMode
.
prototype
.
scrollTo
=
function
(
zen_area
)
{
return
$
.
scrollTo
(
zen_area
,
0
,
{
offset
:
-
150
});
};
ZenMode
.
prototype
.
scrollTo
=
function
(
zen_area
)
{
return
$
.
scrollTo
(
zen_area
,
0
,
{
offset
:
-
150
});
};
return
ZenMode
;
})();
}).
call
(
window
);
return
ZenMode
;
})();
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment