mirror of https://github.com/freeCodeCamp/devdocs
parent
6cc430ffc4
commit
e4fbca722b
@ -1,283 +1,352 @@
|
|||||||
@app =
|
/*
|
||||||
_$: $
|
* decaffeinate suggestions:
|
||||||
_$$: $$
|
* DS101: Remove unnecessary use of Array.from
|
||||||
_page: page
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
collections: {}
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
models: {}
|
* DS207: Consider shorter variations of null checks
|
||||||
templates: {}
|
* DS208: Avoid top-level this
|
||||||
views: {}
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
init: ->
|
this.app = {
|
||||||
try @initErrorTracking() catch
|
_$: $,
|
||||||
return unless @browserCheck()
|
_$$: $$,
|
||||||
|
_page: page,
|
||||||
@el = $('._app')
|
collections: {},
|
||||||
@localStorage = new LocalStorageStore
|
models: {},
|
||||||
@serviceWorker = new app.ServiceWorker if app.ServiceWorker.isEnabled()
|
templates: {},
|
||||||
@settings = new app.Settings
|
views: {},
|
||||||
@db = new app.DB()
|
|
||||||
|
init() {
|
||||||
@settings.initLayout()
|
try { this.initErrorTracking(); } catch (error) {}
|
||||||
|
if (!this.browserCheck()) { return; }
|
||||||
@docs = new app.collections.Docs
|
|
||||||
@disabledDocs = new app.collections.Docs
|
this.el = $('._app');
|
||||||
@entries = new app.collections.Entries
|
this.localStorage = new LocalStorageStore;
|
||||||
|
if (app.ServiceWorker.isEnabled()) { this.serviceWorker = new app.ServiceWorker; }
|
||||||
@router = new app.Router
|
this.settings = new app.Settings;
|
||||||
@shortcuts = new app.Shortcuts
|
this.db = new app.DB();
|
||||||
@document = new app.views.Document
|
|
||||||
@mobile = new app.views.Mobile if @isMobile()
|
this.settings.initLayout();
|
||||||
|
|
||||||
if document.body.hasAttribute('data-doc')
|
this.docs = new app.collections.Docs;
|
||||||
@DOC = JSON.parse(document.body.getAttribute('data-doc'))
|
this.disabledDocs = new app.collections.Docs;
|
||||||
@bootOne()
|
this.entries = new app.collections.Entries;
|
||||||
else if @DOCS
|
|
||||||
@bootAll()
|
this.router = new app.Router;
|
||||||
else
|
this.shortcuts = new app.Shortcuts;
|
||||||
@onBootError()
|
this.document = new app.views.Document;
|
||||||
return
|
if (this.isMobile()) { this.mobile = new app.views.Mobile; }
|
||||||
|
|
||||||
browserCheck: ->
|
if (document.body.hasAttribute('data-doc')) {
|
||||||
return true if @isSupportedBrowser()
|
this.DOC = JSON.parse(document.body.getAttribute('data-doc'));
|
||||||
document.body.innerHTML = app.templates.unsupportedBrowser
|
this.bootOne();
|
||||||
@hideLoadingScreen()
|
} else if (this.DOCS) {
|
||||||
false
|
this.bootAll();
|
||||||
|
} else {
|
||||||
initErrorTracking: ->
|
this.onBootError();
|
||||||
# Show a warning message and don't track errors when the app is loaded
|
}
|
||||||
# from a domain other than our own, because things are likely to break.
|
},
|
||||||
# (e.g. cross-domain requests)
|
|
||||||
if @isInvalidLocation()
|
browserCheck() {
|
||||||
new app.views.Notif 'InvalidLocation'
|
if (this.isSupportedBrowser()) { return true; }
|
||||||
else
|
document.body.innerHTML = app.templates.unsupportedBrowser;
|
||||||
if @config.sentry_dsn
|
this.hideLoadingScreen();
|
||||||
Raven.config @config.sentry_dsn,
|
return false;
|
||||||
release: @config.release
|
},
|
||||||
whitelistUrls: [/devdocs/]
|
|
||||||
includePaths: [/devdocs/]
|
initErrorTracking() {
|
||||||
ignoreErrors: [/NPObject/, /NS_ERROR/, /^null$/, /EvalError/]
|
// Show a warning message and don't track errors when the app is loaded
|
||||||
tags:
|
// from a domain other than our own, because things are likely to break.
|
||||||
mode: if @isSingleDoc() then 'single' else 'full'
|
// (e.g. cross-domain requests)
|
||||||
iframe: (window.top isnt window).toString()
|
if (this.isInvalidLocation()) {
|
||||||
electron: (!!window.process?.versions?.electron).toString()
|
new app.views.Notif('InvalidLocation');
|
||||||
shouldSendCallback: =>
|
} else {
|
||||||
try
|
if (this.config.sentry_dsn) {
|
||||||
if @isInjectionError()
|
Raven.config(this.config.sentry_dsn, {
|
||||||
@onInjectionError()
|
release: this.config.release,
|
||||||
return false
|
whitelistUrls: [/devdocs/],
|
||||||
if @isAndroidWebview()
|
includePaths: [/devdocs/],
|
||||||
return false
|
ignoreErrors: [/NPObject/, /NS_ERROR/, /^null$/, /EvalError/],
|
||||||
true
|
tags: {
|
||||||
dataCallback: (data) ->
|
mode: this.isSingleDoc() ? 'single' : 'full',
|
||||||
try
|
iframe: (window.top !== window).toString(),
|
||||||
$.extend(data.user ||= {}, app.settings.dump())
|
electron: (!!__guard__(window.process != null ? window.process.versions : undefined, x => x.electron)).toString()
|
||||||
data.user.docs = data.user.docs.split('/') if data.user.docs
|
},
|
||||||
data.user.lastIDBTransaction = app.lastIDBTransaction if app.lastIDBTransaction
|
shouldSendCallback: () => {
|
||||||
data.tags.scriptCount = document.scripts.length
|
try {
|
||||||
data
|
if (this.isInjectionError()) {
|
||||||
.install()
|
this.onInjectionError();
|
||||||
@previousErrorHandler = onerror
|
return false;
|
||||||
window.onerror = @onWindowError.bind(@)
|
}
|
||||||
CookiesStore.onBlocked = @onCookieBlocked
|
if (this.isAndroidWebview()) {
|
||||||
return
|
return false;
|
||||||
|
}
|
||||||
bootOne: ->
|
} catch (error) {}
|
||||||
@doc = new app.models.Doc @DOC
|
return true;
|
||||||
@docs.reset [@doc]
|
},
|
||||||
@doc.load @start.bind(@), @onBootError.bind(@), readCache: true
|
dataCallback(data) {
|
||||||
new app.views.Notice 'singleDoc', @doc
|
try {
|
||||||
delete @DOC
|
$.extend(data.user || (data.user = {}), app.settings.dump());
|
||||||
return
|
if (data.user.docs) { data.user.docs = data.user.docs.split('/'); }
|
||||||
|
if (app.lastIDBTransaction) { data.user.lastIDBTransaction = app.lastIDBTransaction; }
|
||||||
bootAll: ->
|
data.tags.scriptCount = document.scripts.length;
|
||||||
docs = @settings.getDocs()
|
} catch (error) {}
|
||||||
for doc in @DOCS
|
return data;
|
||||||
(if docs.indexOf(doc.slug) >= 0 then @docs else @disabledDocs).add(doc)
|
}
|
||||||
@migrateDocs()
|
}).install();
|
||||||
@docs.load @start.bind(@), @onBootError.bind(@), readCache: true, writeCache: true
|
}
|
||||||
delete @DOCS
|
this.previousErrorHandler = onerror;
|
||||||
return
|
window.onerror = this.onWindowError.bind(this);
|
||||||
|
CookiesStore.onBlocked = this.onCookieBlocked;
|
||||||
start: ->
|
}
|
||||||
@entries.add doc.toEntry() for doc in @docs.all()
|
},
|
||||||
@entries.add doc.toEntry() for doc in @disabledDocs.all()
|
|
||||||
@initDoc(doc) for doc in @docs.all()
|
bootOne() {
|
||||||
@trigger 'ready'
|
this.doc = new app.models.Doc(this.DOC);
|
||||||
@router.start()
|
this.docs.reset([this.doc]);
|
||||||
@hideLoadingScreen()
|
this.doc.load(this.start.bind(this), this.onBootError.bind(this), {readCache: true});
|
||||||
setTimeout =>
|
new app.views.Notice('singleDoc', this.doc);
|
||||||
@welcomeBack() unless @doc
|
delete this.DOC;
|
||||||
@removeEvent 'ready bootError'
|
},
|
||||||
, 50
|
|
||||||
return
|
bootAll() {
|
||||||
|
const docs = this.settings.getDocs();
|
||||||
initDoc: (doc) ->
|
for (var doc of Array.from(this.DOCS)) {
|
||||||
doc.entries.add type.toEntry() for type in doc.types.all()
|
(docs.indexOf(doc.slug) >= 0 ? this.docs : this.disabledDocs).add(doc);
|
||||||
@entries.add doc.entries.all()
|
}
|
||||||
return
|
this.migrateDocs();
|
||||||
|
this.docs.load(this.start.bind(this), this.onBootError.bind(this), {readCache: true, writeCache: true});
|
||||||
migrateDocs: ->
|
delete this.DOCS;
|
||||||
for slug in @settings.getDocs() when not @docs.findBy('slug', slug)
|
},
|
||||||
needsSaving = true
|
|
||||||
doc = @disabledDocs.findBy('slug', 'webpack') if slug == 'webpack~2'
|
start() {
|
||||||
doc = @disabledDocs.findBy('slug', 'angular') if slug == 'angular~4_typescript'
|
let doc;
|
||||||
doc = @disabledDocs.findBy('slug', 'angular~2') if slug == 'angular~2_typescript'
|
for (doc of Array.from(this.docs.all())) { this.entries.add(doc.toEntry()); }
|
||||||
doc ||= @disabledDocs.findBy('slug_without_version', slug)
|
for (doc of Array.from(this.disabledDocs.all())) { this.entries.add(doc.toEntry()); }
|
||||||
if doc
|
for (doc of Array.from(this.docs.all())) { this.initDoc(doc); }
|
||||||
@disabledDocs.remove(doc)
|
this.trigger('ready');
|
||||||
@docs.add(doc)
|
this.router.start();
|
||||||
|
this.hideLoadingScreen();
|
||||||
@saveDocs() if needsSaving
|
setTimeout(() => {
|
||||||
return
|
if (!this.doc) { this.welcomeBack(); }
|
||||||
|
return this.removeEvent('ready bootError');
|
||||||
enableDoc: (doc, _onSuccess, onError) ->
|
}
|
||||||
return if @docs.contains(doc)
|
, 50);
|
||||||
|
},
|
||||||
onSuccess = =>
|
|
||||||
return if @docs.contains(doc)
|
initDoc(doc) {
|
||||||
@disabledDocs.remove(doc)
|
for (var type of Array.from(doc.types.all())) { doc.entries.add(type.toEntry()); }
|
||||||
@docs.add(doc)
|
this.entries.add(doc.entries.all());
|
||||||
@docs.sort()
|
},
|
||||||
@initDoc(doc)
|
|
||||||
@saveDocs()
|
migrateDocs() {
|
||||||
if app.settings.get('autoInstall')
|
let needsSaving;
|
||||||
doc.install(_onSuccess, onError)
|
for (var slug of Array.from(this.settings.getDocs())) {
|
||||||
else
|
if (!this.docs.findBy('slug', slug)) {var doc;
|
||||||
_onSuccess()
|
|
||||||
return
|
needsSaving = true;
|
||||||
|
if (slug === 'webpack~2') { doc = this.disabledDocs.findBy('slug', 'webpack'); }
|
||||||
doc.load onSuccess, onError, writeCache: true
|
if (slug === 'angular~4_typescript') { doc = this.disabledDocs.findBy('slug', 'angular'); }
|
||||||
return
|
if (slug === 'angular~2_typescript') { doc = this.disabledDocs.findBy('slug', 'angular~2'); }
|
||||||
|
if (!doc) { doc = this.disabledDocs.findBy('slug_without_version', slug); }
|
||||||
saveDocs: ->
|
if (doc) {
|
||||||
@settings.setDocs(doc.slug for doc in @docs.all())
|
this.disabledDocs.remove(doc);
|
||||||
@db.migrate()
|
this.docs.add(doc);
|
||||||
@serviceWorker?.updateInBackground()
|
}
|
||||||
|
}
|
||||||
welcomeBack: ->
|
}
|
||||||
visitCount = @settings.get('count')
|
|
||||||
@settings.set 'count', ++visitCount
|
if (needsSaving) { this.saveDocs(); }
|
||||||
new app.views.Notif 'Share', autoHide: null if visitCount is 5
|
},
|
||||||
new app.views.News()
|
|
||||||
new app.views.Updates()
|
enableDoc(doc, _onSuccess, onError) {
|
||||||
@updateChecker = new app.UpdateChecker()
|
if (this.docs.contains(doc)) { return; }
|
||||||
|
|
||||||
reboot: ->
|
const onSuccess = () => {
|
||||||
if location.pathname isnt '/' and location.pathname isnt '/settings'
|
if (this.docs.contains(doc)) { return; }
|
||||||
window.location = "/##{location.pathname}"
|
this.disabledDocs.remove(doc);
|
||||||
else
|
this.docs.add(doc);
|
||||||
window.location = '/'
|
this.docs.sort();
|
||||||
return
|
this.initDoc(doc);
|
||||||
|
this.saveDocs();
|
||||||
reload: ->
|
if (app.settings.get('autoInstall')) {
|
||||||
@docs.clearCache()
|
doc.install(_onSuccess, onError);
|
||||||
@disabledDocs.clearCache()
|
} else {
|
||||||
if @serviceWorker then @serviceWorker.reload() else @reboot()
|
_onSuccess();
|
||||||
return
|
}
|
||||||
|
};
|
||||||
reset: ->
|
|
||||||
@localStorage.reset()
|
doc.load(onSuccess, onError, {writeCache: true});
|
||||||
@settings.reset()
|
},
|
||||||
@db?.reset()
|
|
||||||
@serviceWorker?.update()
|
saveDocs() {
|
||||||
window.location = '/'
|
this.settings.setDocs(Array.from(this.docs.all()).map((doc) => doc.slug));
|
||||||
return
|
this.db.migrate();
|
||||||
|
return (this.serviceWorker != null ? this.serviceWorker.updateInBackground() : undefined);
|
||||||
showTip: (tip) ->
|
},
|
||||||
return if @isSingleDoc()
|
|
||||||
tips = @settings.getTips()
|
welcomeBack() {
|
||||||
if tips.indexOf(tip) is -1
|
let visitCount = this.settings.get('count');
|
||||||
tips.push(tip)
|
this.settings.set('count', ++visitCount);
|
||||||
@settings.setTips(tips)
|
if (visitCount === 5) { new app.views.Notif('Share', {autoHide: null}); }
|
||||||
new app.views.Tip(tip)
|
new app.views.News();
|
||||||
return
|
new app.views.Updates();
|
||||||
|
return this.updateChecker = new app.UpdateChecker();
|
||||||
hideLoadingScreen: ->
|
},
|
||||||
document.body.classList.add '_overlay-scrollbars' if $.overlayScrollbarsEnabled()
|
|
||||||
document.documentElement.classList.remove '_booting'
|
reboot() {
|
||||||
return
|
if ((location.pathname !== '/') && (location.pathname !== '/settings')) {
|
||||||
|
window.location = `/#${location.pathname}`;
|
||||||
indexHost: ->
|
} else {
|
||||||
# Can't load the index files from the host/CDN when service worker is
|
window.location = '/';
|
||||||
# enabled because it doesn't support caching URLs that use CORS.
|
}
|
||||||
@config[if @serviceWorker and @settings.hasDocs() then 'index_path' else 'docs_origin']
|
},
|
||||||
|
|
||||||
onBootError: (args...) ->
|
reload() {
|
||||||
@trigger 'bootError'
|
this.docs.clearCache();
|
||||||
@hideLoadingScreen()
|
this.disabledDocs.clearCache();
|
||||||
return
|
if (this.serviceWorker) { this.serviceWorker.reload(); } else { this.reboot(); }
|
||||||
|
},
|
||||||
onQuotaExceeded: ->
|
|
||||||
return if @quotaExceeded
|
reset() {
|
||||||
@quotaExceeded = true
|
this.localStorage.reset();
|
||||||
new app.views.Notif 'QuotaExceeded', autoHide: null
|
this.settings.reset();
|
||||||
return
|
if (this.db != null) {
|
||||||
|
this.db.reset();
|
||||||
onCookieBlocked: (key, value, actual) ->
|
}
|
||||||
return if @cookieBlocked
|
if (this.serviceWorker != null) {
|
||||||
@cookieBlocked = true
|
this.serviceWorker.update();
|
||||||
new app.views.Notif 'CookieBlocked', autoHide: null
|
}
|
||||||
Raven.captureMessage "CookieBlocked/#{key}", level: 'warning', extra: {value, actual}
|
window.location = '/';
|
||||||
return
|
},
|
||||||
|
|
||||||
onWindowError: (args...) ->
|
showTip(tip) {
|
||||||
return if @cookieBlocked
|
if (this.isSingleDoc()) { return; }
|
||||||
if @isInjectionError args...
|
const tips = this.settings.getTips();
|
||||||
@onInjectionError()
|
if (tips.indexOf(tip) === -1) {
|
||||||
else if @isAppError args...
|
tips.push(tip);
|
||||||
@previousErrorHandler? args...
|
this.settings.setTips(tips);
|
||||||
@hideLoadingScreen()
|
new app.views.Tip(tip);
|
||||||
@errorNotif or= new app.views.Notif 'Error'
|
}
|
||||||
@errorNotif.show()
|
},
|
||||||
return
|
|
||||||
|
hideLoadingScreen() {
|
||||||
onInjectionError: ->
|
if ($.overlayScrollbarsEnabled()) { document.body.classList.add('_overlay-scrollbars'); }
|
||||||
unless @injectionError
|
document.documentElement.classList.remove('_booting');
|
||||||
@injectionError = true
|
},
|
||||||
alert """
|
|
||||||
JavaScript code has been injected in the page which prevents DevDocs from running correctly.
|
indexHost() {
|
||||||
Please check your browser extensions/addons. """
|
// Can't load the index files from the host/CDN when service worker is
|
||||||
Raven.captureMessage 'injection error', level: 'info'
|
// enabled because it doesn't support caching URLs that use CORS.
|
||||||
return
|
return this.config[this.serviceWorker && this.settings.hasDocs() ? 'index_path' : 'docs_origin'];
|
||||||
|
},
|
||||||
isInjectionError: ->
|
|
||||||
# Some browser extensions expect the entire web to use jQuery.
|
onBootError(...args) {
|
||||||
# I gave up trying to fight back.
|
this.trigger('bootError');
|
||||||
window.$ isnt app._$ or window.$$ isnt app._$$ or window.page isnt app._page or typeof $.empty isnt 'function' or typeof page.show isnt 'function'
|
this.hideLoadingScreen();
|
||||||
|
},
|
||||||
isAppError: (error, file) ->
|
|
||||||
# Ignore errors from external scripts.
|
onQuotaExceeded() {
|
||||||
file and file.indexOf('devdocs') isnt -1 and file.indexOf('.js') is file.length - 3
|
if (this.quotaExceeded) { return; }
|
||||||
|
this.quotaExceeded = true;
|
||||||
isSupportedBrowser: ->
|
new app.views.Notif('QuotaExceeded', {autoHide: null});
|
||||||
try
|
},
|
||||||
features =
|
|
||||||
bind: !!Function::bind
|
onCookieBlocked(key, value, actual) {
|
||||||
pushState: !!history.pushState
|
if (this.cookieBlocked) { return; }
|
||||||
matchMedia: !!window.matchMedia
|
this.cookieBlocked = true;
|
||||||
insertAdjacentHTML: !!document.body.insertAdjacentHTML
|
new app.views.Notif('CookieBlocked', {autoHide: null});
|
||||||
defaultPrevented: document.createEvent('CustomEvent').defaultPrevented is false
|
Raven.captureMessage(`CookieBlocked/${key}`, {level: 'warning', extra: {value, actual}});
|
||||||
cssVariables: !!CSS?.supports?('(--t: 0)')
|
},
|
||||||
|
|
||||||
for key, value of features when not value
|
onWindowError(...args) {
|
||||||
Raven.captureMessage "unsupported/#{key}", level: 'info'
|
if (this.cookieBlocked) { return; }
|
||||||
return false
|
if (this.isInjectionError(...Array.from(args || []))) {
|
||||||
|
this.onInjectionError();
|
||||||
true
|
} else if (this.isAppError(...Array.from(args || []))) {
|
||||||
catch error
|
if (typeof this.previousErrorHandler === 'function') {
|
||||||
Raven.captureMessage 'unsupported/exception', level: 'info', extra: { error: error }
|
this.previousErrorHandler(...Array.from(args || []));
|
||||||
false
|
}
|
||||||
|
this.hideLoadingScreen();
|
||||||
isSingleDoc: ->
|
if (!this.errorNotif) { this.errorNotif = new app.views.Notif('Error'); }
|
||||||
document.body.hasAttribute('data-doc')
|
this.errorNotif.show();
|
||||||
|
}
|
||||||
isMobile: ->
|
},
|
||||||
@_isMobile ?= app.views.Mobile.detect()
|
|
||||||
|
onInjectionError() {
|
||||||
isAndroidWebview: ->
|
if (!this.injectionError) {
|
||||||
@_isAndroidWebview ?= app.views.Mobile.detectAndroidWebview()
|
this.injectionError = true;
|
||||||
|
alert(`\
|
||||||
isInvalidLocation: ->
|
JavaScript code has been injected in the page which prevents DevDocs from running correctly.
|
||||||
@config.env is 'production' and location.host.indexOf(app.config.production_host) isnt 0
|
Please check your browser extensions/addons. `
|
||||||
|
);
|
||||||
$.extend app, Events
|
Raven.captureMessage('injection error', {level: 'info'});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
isInjectionError() {
|
||||||
|
// Some browser extensions expect the entire web to use jQuery.
|
||||||
|
// I gave up trying to fight back.
|
||||||
|
return (window.$ !== app._$) || (window.$$ !== app._$$) || (window.page !== app._page) || (typeof $.empty !== 'function') || (typeof page.show !== 'function');
|
||||||
|
},
|
||||||
|
|
||||||
|
isAppError(error, file) {
|
||||||
|
// Ignore errors from external scripts.
|
||||||
|
return file && (file.indexOf('devdocs') !== -1) && (file.indexOf('.js') === (file.length - 3));
|
||||||
|
},
|
||||||
|
|
||||||
|
isSupportedBrowser() {
|
||||||
|
try {
|
||||||
|
const features = {
|
||||||
|
bind: !!Function.prototype.bind,
|
||||||
|
pushState: !!history.pushState,
|
||||||
|
matchMedia: !!window.matchMedia,
|
||||||
|
insertAdjacentHTML: !!document.body.insertAdjacentHTML,
|
||||||
|
defaultPrevented: document.createEvent('CustomEvent').defaultPrevented === false,
|
||||||
|
cssVariables: !!__guardMethod__(CSS, 'supports', o => o.supports('(--t: 0)'))
|
||||||
|
};
|
||||||
|
|
||||||
|
for (var key in features) {
|
||||||
|
var value = features[key];
|
||||||
|
if (!value) {
|
||||||
|
Raven.captureMessage(`unsupported/${key}`, {level: 'info'});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
Raven.captureMessage('unsupported/exception', {level: 'info', extra: { error }});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
isSingleDoc() {
|
||||||
|
return document.body.hasAttribute('data-doc');
|
||||||
|
},
|
||||||
|
|
||||||
|
isMobile() {
|
||||||
|
return this._isMobile != null ? this._isMobile : (this._isMobile = app.views.Mobile.detect());
|
||||||
|
},
|
||||||
|
|
||||||
|
isAndroidWebview() {
|
||||||
|
return this._isAndroidWebview != null ? this._isAndroidWebview : (this._isAndroidWebview = app.views.Mobile.detectAndroidWebview());
|
||||||
|
},
|
||||||
|
|
||||||
|
isInvalidLocation() {
|
||||||
|
return (this.config.env === 'production') && (location.host.indexOf(app.config.production_host) !== 0);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$.extend(app, Events);
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
||||||
|
function __guardMethod__(obj, methodName, transform) {
|
||||||
|
if (typeof obj !== 'undefined' && obj !== null && typeof obj[methodName] === 'function') {
|
||||||
|
return transform(obj, methodName);
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
@ -1,382 +1,486 @@
|
|||||||
class app.DB
|
/*
|
||||||
NAME = 'docs'
|
* decaffeinate suggestions:
|
||||||
VERSION = 15
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
constructor: ->
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
@versionMultipler = if $.isIE() then 1e5 else 1e9
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@useIndexedDB = @useIndexedDB()
|
* DS207: Consider shorter variations of null checks
|
||||||
@callbacks = []
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
db: (fn) ->
|
(function() {
|
||||||
return fn() unless @useIndexedDB
|
let NAME = undefined;
|
||||||
@callbacks.push(fn) if fn
|
let VERSION = undefined;
|
||||||
return if @open
|
const Cls = (app.DB = class DB {
|
||||||
|
static initClass() {
|
||||||
try
|
NAME = 'docs';
|
||||||
@open = true
|
VERSION = 15;
|
||||||
req = indexedDB.open(NAME, VERSION * @versionMultipler + @userVersion())
|
}
|
||||||
req.onsuccess = @onOpenSuccess
|
|
||||||
req.onerror = @onOpenError
|
constructor() {
|
||||||
req.onupgradeneeded = @onUpgradeNeeded
|
this.onOpenSuccess = this.onOpenSuccess.bind(this);
|
||||||
catch error
|
this.onOpenError = this.onOpenError.bind(this);
|
||||||
@fail 'exception', error
|
this.checkForCorruptedDocs = this.checkForCorruptedDocs.bind(this);
|
||||||
return
|
this.deleteCorruptedDocs = this.deleteCorruptedDocs.bind(this);
|
||||||
|
this.versionMultipler = $.isIE() ? 1e5 : 1e9;
|
||||||
onOpenSuccess: (event) =>
|
this.useIndexedDB = this.useIndexedDB();
|
||||||
db = event.target.result
|
this.callbacks = [];
|
||||||
|
}
|
||||||
if db.objectStoreNames.length is 0
|
|
||||||
try db.close()
|
db(fn) {
|
||||||
@open = false
|
if (!this.useIndexedDB) { return fn(); }
|
||||||
@fail 'empty'
|
if (fn) { this.callbacks.push(fn); }
|
||||||
else if error = @buggyIDB(db)
|
if (this.open) { return; }
|
||||||
try db.close()
|
|
||||||
@open = false
|
try {
|
||||||
@fail 'buggy', error
|
this.open = true;
|
||||||
else
|
const req = indexedDB.open(NAME, (VERSION * this.versionMultipler) + this.userVersion());
|
||||||
@runCallbacks(db)
|
req.onsuccess = this.onOpenSuccess;
|
||||||
@open = false
|
req.onerror = this.onOpenError;
|
||||||
db.close()
|
req.onupgradeneeded = this.onUpgradeNeeded;
|
||||||
return
|
} catch (error) {
|
||||||
|
this.fail('exception', error);
|
||||||
onOpenError: (event) =>
|
}
|
||||||
event.preventDefault()
|
}
|
||||||
@open = false
|
|
||||||
error = event.target.error
|
onOpenSuccess(event) {
|
||||||
|
let error;
|
||||||
switch error.name
|
const db = event.target.result;
|
||||||
when 'QuotaExceededError'
|
|
||||||
@onQuotaExceededError()
|
if (db.objectStoreNames.length === 0) {
|
||||||
when 'VersionError'
|
try { db.close(); } catch (error1) {}
|
||||||
@onVersionError()
|
this.open = false;
|
||||||
when 'InvalidStateError'
|
this.fail('empty');
|
||||||
@fail 'private_mode'
|
} else if (error = this.buggyIDB(db)) {
|
||||||
else
|
try { db.close(); } catch (error2) {}
|
||||||
@fail 'cant_open', error
|
this.open = false;
|
||||||
return
|
this.fail('buggy', error);
|
||||||
|
} else {
|
||||||
fail: (reason, error) ->
|
this.runCallbacks(db);
|
||||||
@cachedDocs = null
|
this.open = false;
|
||||||
@useIndexedDB = false
|
db.close();
|
||||||
@reason or= reason
|
}
|
||||||
@error or= error
|
}
|
||||||
console.error? 'IDB error', error if error
|
|
||||||
@runCallbacks()
|
onOpenError(event) {
|
||||||
if error and reason is 'cant_open'
|
event.preventDefault();
|
||||||
Raven.captureMessage "#{error.name}: #{error.message}", level: 'warning', fingerprint: [error.name]
|
this.open = false;
|
||||||
return
|
const {
|
||||||
|
error
|
||||||
onQuotaExceededError: ->
|
} = event.target;
|
||||||
@reset()
|
|
||||||
@db()
|
switch (error.name) {
|
||||||
app.onQuotaExceeded()
|
case 'QuotaExceededError':
|
||||||
Raven.captureMessage 'QuotaExceededError', level: 'warning'
|
this.onQuotaExceededError();
|
||||||
return
|
break;
|
||||||
|
case 'VersionError':
|
||||||
onVersionError: ->
|
this.onVersionError();
|
||||||
req = indexedDB.open(NAME)
|
break;
|
||||||
req.onsuccess = (event) =>
|
case 'InvalidStateError':
|
||||||
@handleVersionMismatch event.target.result.version
|
this.fail('private_mode');
|
||||||
req.onerror = (event) ->
|
break;
|
||||||
event.preventDefault()
|
default:
|
||||||
@fail 'cant_open', error
|
this.fail('cant_open', error);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
handleVersionMismatch: (actualVersion) ->
|
|
||||||
if Math.floor(actualVersion / @versionMultipler) isnt VERSION
|
fail(reason, error) {
|
||||||
@fail 'version'
|
this.cachedDocs = null;
|
||||||
else
|
this.useIndexedDB = false;
|
||||||
@setUserVersion actualVersion - VERSION * @versionMultipler
|
if (!this.reason) { this.reason = reason; }
|
||||||
@db()
|
if (!this.error) { this.error = error; }
|
||||||
return
|
if (error) { if (typeof console.error === 'function') {
|
||||||
|
console.error('IDB error', error);
|
||||||
buggyIDB: (db) ->
|
} }
|
||||||
return if @checkedBuggyIDB
|
this.runCallbacks();
|
||||||
@checkedBuggyIDB = true
|
if (error && (reason === 'cant_open')) {
|
||||||
try
|
Raven.captureMessage(`${error.name}: ${error.message}`, {level: 'warning', fingerprint: [error.name]});
|
||||||
@idbTransaction(db, stores: $.makeArray(db.objectStoreNames)[0..1], mode: 'readwrite').abort() # https://bugs.webkit.org/show_bug.cgi?id=136937
|
}
|
||||||
return
|
}
|
||||||
catch error
|
|
||||||
return error
|
onQuotaExceededError() {
|
||||||
|
this.reset();
|
||||||
runCallbacks: (db) ->
|
this.db();
|
||||||
fn(db) while fn = @callbacks.shift()
|
app.onQuotaExceeded();
|
||||||
return
|
Raven.captureMessage('QuotaExceededError', {level: 'warning'});
|
||||||
|
}
|
||||||
onUpgradeNeeded: (event) ->
|
|
||||||
return unless db = event.target.result
|
onVersionError() {
|
||||||
|
const req = indexedDB.open(NAME);
|
||||||
objectStoreNames = $.makeArray(db.objectStoreNames)
|
req.onsuccess = event => {
|
||||||
|
return this.handleVersionMismatch(event.target.result.version);
|
||||||
unless $.arrayDelete(objectStoreNames, 'docs')
|
};
|
||||||
try db.createObjectStore('docs')
|
req.onerror = function(event) {
|
||||||
|
event.preventDefault();
|
||||||
for doc in app.docs.all() when not $.arrayDelete(objectStoreNames, doc.slug)
|
return this.fail('cant_open', error);
|
||||||
try db.createObjectStore(doc.slug)
|
};
|
||||||
|
}
|
||||||
for name in objectStoreNames
|
|
||||||
try db.deleteObjectStore(name)
|
handleVersionMismatch(actualVersion) {
|
||||||
return
|
if (Math.floor(actualVersion / this.versionMultipler) !== VERSION) {
|
||||||
|
this.fail('version');
|
||||||
store: (doc, data, onSuccess, onError, _retry = true) ->
|
} else {
|
||||||
@db (db) =>
|
this.setUserVersion(actualVersion - (VERSION * this.versionMultipler));
|
||||||
unless db
|
this.db();
|
||||||
onError()
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
txn = @idbTransaction db, stores: ['docs', doc.slug], mode: 'readwrite', ignoreError: false
|
buggyIDB(db) {
|
||||||
txn.oncomplete = =>
|
if (this.checkedBuggyIDB) { return; }
|
||||||
@cachedDocs?[doc.slug] = doc.mtime
|
this.checkedBuggyIDB = true;
|
||||||
onSuccess()
|
try {
|
||||||
return
|
this.idbTransaction(db, {stores: $.makeArray(db.objectStoreNames).slice(0, 2), mode: 'readwrite'}).abort(); // https://bugs.webkit.org/show_bug.cgi?id=136937
|
||||||
txn.onerror = (event) =>
|
return;
|
||||||
event.preventDefault()
|
} catch (error) {
|
||||||
if txn.error?.name is 'NotFoundError' and _retry
|
return error;
|
||||||
@migrate()
|
}
|
||||||
setTimeout =>
|
}
|
||||||
@store(doc, data, onSuccess, onError, false)
|
|
||||||
, 0
|
runCallbacks(db) {
|
||||||
else
|
let fn;
|
||||||
onError(event)
|
while ((fn = this.callbacks.shift())) { fn(db); }
|
||||||
return
|
}
|
||||||
|
|
||||||
store = txn.objectStore(doc.slug)
|
onUpgradeNeeded(event) {
|
||||||
store.clear()
|
let db;
|
||||||
store.add(content, path) for path, content of data
|
if (!(db = event.target.result)) { return; }
|
||||||
|
|
||||||
store = txn.objectStore('docs')
|
const objectStoreNames = $.makeArray(db.objectStoreNames);
|
||||||
store.put(doc.mtime, doc.slug)
|
|
||||||
return
|
if (!$.arrayDelete(objectStoreNames, 'docs')) {
|
||||||
return
|
try { db.createObjectStore('docs'); } catch (error) {}
|
||||||
|
}
|
||||||
unstore: (doc, onSuccess, onError, _retry = true) ->
|
|
||||||
@db (db) =>
|
for (var doc of Array.from(app.docs.all())) {
|
||||||
unless db
|
if (!$.arrayDelete(objectStoreNames, doc.slug)) {
|
||||||
onError()
|
try { db.createObjectStore(doc.slug); } catch (error1) {}
|
||||||
return
|
}
|
||||||
|
}
|
||||||
txn = @idbTransaction db, stores: ['docs', doc.slug], mode: 'readwrite', ignoreError: false
|
|
||||||
txn.oncomplete = =>
|
for (var name of Array.from(objectStoreNames)) {
|
||||||
delete @cachedDocs?[doc.slug]
|
try { db.deleteObjectStore(name); } catch (error2) {}
|
||||||
onSuccess()
|
}
|
||||||
return
|
}
|
||||||
txn.onerror = (event) ->
|
|
||||||
event.preventDefault()
|
store(doc, data, onSuccess, onError, _retry) {
|
||||||
if txn.error?.name is 'NotFoundError' and _retry
|
if (_retry == null) { _retry = true; }
|
||||||
@migrate()
|
this.db(db => {
|
||||||
setTimeout =>
|
if (!db) {
|
||||||
@unstore(doc, onSuccess, onError, false)
|
onError();
|
||||||
, 0
|
return;
|
||||||
else
|
}
|
||||||
onError(event)
|
|
||||||
return
|
const txn = this.idbTransaction(db, {stores: ['docs', doc.slug], mode: 'readwrite', ignoreError: false});
|
||||||
|
txn.oncomplete = () => {
|
||||||
store = txn.objectStore('docs')
|
if (this.cachedDocs != null) {
|
||||||
store.delete(doc.slug)
|
this.cachedDocs[doc.slug] = doc.mtime;
|
||||||
|
}
|
||||||
store = txn.objectStore(doc.slug)
|
onSuccess();
|
||||||
store.clear()
|
};
|
||||||
return
|
txn.onerror = event => {
|
||||||
return
|
event.preventDefault();
|
||||||
|
if (((txn.error != null ? txn.error.name : undefined) === 'NotFoundError') && _retry) {
|
||||||
version: (doc, fn) ->
|
this.migrate();
|
||||||
if (version = @cachedVersion(doc))?
|
setTimeout(() => {
|
||||||
fn(version)
|
return this.store(doc, data, onSuccess, onError, false);
|
||||||
return
|
}
|
||||||
|
, 0);
|
||||||
@db (db) =>
|
} else {
|
||||||
unless db
|
onError(event);
|
||||||
fn(false)
|
}
|
||||||
return
|
};
|
||||||
|
|
||||||
txn = @idbTransaction db, stores: ['docs'], mode: 'readonly'
|
let store = txn.objectStore(doc.slug);
|
||||||
store = txn.objectStore('docs')
|
store.clear();
|
||||||
|
for (var path in data) { var content = data[path]; store.add(content, path); }
|
||||||
req = store.get(doc.slug)
|
|
||||||
req.onsuccess = ->
|
store = txn.objectStore('docs');
|
||||||
fn(req.result)
|
store.put(doc.mtime, doc.slug);
|
||||||
return
|
});
|
||||||
req.onerror = (event) ->
|
}
|
||||||
event.preventDefault()
|
|
||||||
fn(false)
|
unstore(doc, onSuccess, onError, _retry) {
|
||||||
return
|
if (_retry == null) { _retry = true; }
|
||||||
return
|
this.db(db => {
|
||||||
return
|
if (!db) {
|
||||||
|
onError();
|
||||||
cachedVersion: (doc) ->
|
return;
|
||||||
return unless @cachedDocs
|
}
|
||||||
@cachedDocs[doc.slug] or false
|
|
||||||
|
const txn = this.idbTransaction(db, {stores: ['docs', doc.slug], mode: 'readwrite', ignoreError: false});
|
||||||
versions: (docs, fn) ->
|
txn.oncomplete = () => {
|
||||||
if versions = @cachedVersions(docs)
|
if (this.cachedDocs != null) {
|
||||||
fn(versions)
|
delete this.cachedDocs[doc.slug];
|
||||||
return
|
}
|
||||||
|
onSuccess();
|
||||||
@db (db) =>
|
};
|
||||||
unless db
|
txn.onerror = function(event) {
|
||||||
fn(false)
|
event.preventDefault();
|
||||||
return
|
if (((txn.error != null ? txn.error.name : undefined) === 'NotFoundError') && _retry) {
|
||||||
|
this.migrate();
|
||||||
txn = @idbTransaction db, stores: ['docs'], mode: 'readonly'
|
setTimeout(() => {
|
||||||
txn.oncomplete = ->
|
return this.unstore(doc, onSuccess, onError, false);
|
||||||
fn(result)
|
}
|
||||||
return
|
, 0);
|
||||||
store = txn.objectStore('docs')
|
} else {
|
||||||
result = {}
|
onError(event);
|
||||||
|
}
|
||||||
docs.forEach (doc) ->
|
};
|
||||||
req = store.get(doc.slug)
|
|
||||||
req.onsuccess = ->
|
let store = txn.objectStore('docs');
|
||||||
result[doc.slug] = req.result
|
store.delete(doc.slug);
|
||||||
return
|
|
||||||
req.onerror = (event) ->
|
store = txn.objectStore(doc.slug);
|
||||||
event.preventDefault()
|
store.clear();
|
||||||
result[doc.slug] = false
|
});
|
||||||
return
|
}
|
||||||
return
|
|
||||||
return
|
version(doc, fn) {
|
||||||
|
let version;
|
||||||
cachedVersions: (docs) ->
|
if ((version = this.cachedVersion(doc)) != null) {
|
||||||
return unless @cachedDocs
|
fn(version);
|
||||||
result = {}
|
return;
|
||||||
result[doc.slug] = @cachedVersion(doc) for doc in docs
|
}
|
||||||
result
|
|
||||||
|
this.db(db => {
|
||||||
load: (entry, onSuccess, onError) ->
|
if (!db) {
|
||||||
if @shouldLoadWithIDB(entry)
|
fn(false);
|
||||||
onError = @loadWithXHR.bind(@, entry, onSuccess, onError)
|
return;
|
||||||
@loadWithIDB entry, onSuccess, onError
|
}
|
||||||
else
|
|
||||||
@loadWithXHR entry, onSuccess, onError
|
const txn = this.idbTransaction(db, {stores: ['docs'], mode: 'readonly'});
|
||||||
|
const store = txn.objectStore('docs');
|
||||||
loadWithXHR: (entry, onSuccess, onError) ->
|
|
||||||
ajax
|
const req = store.get(doc.slug);
|
||||||
url: entry.fileUrl()
|
req.onsuccess = function() {
|
||||||
dataType: 'html'
|
fn(req.result);
|
||||||
success: onSuccess
|
};
|
||||||
error: onError
|
req.onerror = function(event) {
|
||||||
|
event.preventDefault();
|
||||||
loadWithIDB: (entry, onSuccess, onError) ->
|
fn(false);
|
||||||
@db (db) =>
|
};
|
||||||
unless db
|
});
|
||||||
onError()
|
}
|
||||||
return
|
|
||||||
|
cachedVersion(doc) {
|
||||||
unless db.objectStoreNames.contains(entry.doc.slug)
|
if (!this.cachedDocs) { return; }
|
||||||
onError()
|
return this.cachedDocs[doc.slug] || false;
|
||||||
@loadDocsCache(db)
|
}
|
||||||
return
|
|
||||||
|
versions(docs, fn) {
|
||||||
txn = @idbTransaction db, stores: [entry.doc.slug], mode: 'readonly'
|
let versions;
|
||||||
store = txn.objectStore(entry.doc.slug)
|
if (versions = this.cachedVersions(docs)) {
|
||||||
|
fn(versions);
|
||||||
req = store.get(entry.dbPath())
|
return;
|
||||||
req.onsuccess = ->
|
}
|
||||||
if req.result then onSuccess(req.result) else onError()
|
|
||||||
return
|
return this.db(db => {
|
||||||
req.onerror = (event) ->
|
if (!db) {
|
||||||
event.preventDefault()
|
fn(false);
|
||||||
onError()
|
return;
|
||||||
return
|
}
|
||||||
@loadDocsCache(db)
|
|
||||||
return
|
const txn = this.idbTransaction(db, {stores: ['docs'], mode: 'readonly'});
|
||||||
|
txn.oncomplete = function() {
|
||||||
loadDocsCache: (db) ->
|
fn(result);
|
||||||
return if @cachedDocs
|
};
|
||||||
@cachedDocs = {}
|
const store = txn.objectStore('docs');
|
||||||
|
var result = {};
|
||||||
txn = @idbTransaction db, stores: ['docs'], mode: 'readonly'
|
|
||||||
txn.oncomplete = =>
|
docs.forEach(function(doc) {
|
||||||
setTimeout(@checkForCorruptedDocs, 50)
|
const req = store.get(doc.slug);
|
||||||
return
|
req.onsuccess = function() {
|
||||||
|
result[doc.slug] = req.result;
|
||||||
req = txn.objectStore('docs').openCursor()
|
};
|
||||||
req.onsuccess = (event) =>
|
req.onerror = function(event) {
|
||||||
return unless cursor = event.target.result
|
event.preventDefault();
|
||||||
@cachedDocs[cursor.key] = cursor.value
|
result[doc.slug] = false;
|
||||||
cursor.continue()
|
};
|
||||||
return
|
});
|
||||||
req.onerror = (event) ->
|
});
|
||||||
event.preventDefault()
|
}
|
||||||
return
|
|
||||||
return
|
cachedVersions(docs) {
|
||||||
|
if (!this.cachedDocs) { return; }
|
||||||
checkForCorruptedDocs: =>
|
const result = {};
|
||||||
@db (db) =>
|
for (var doc of Array.from(docs)) { result[doc.slug] = this.cachedVersion(doc); }
|
||||||
@corruptedDocs = []
|
return result;
|
||||||
docs = (key for key, value of @cachedDocs when value)
|
}
|
||||||
return if docs.length is 0
|
|
||||||
|
load(entry, onSuccess, onError) {
|
||||||
for slug in docs when not app.docs.findBy('slug', slug)
|
if (this.shouldLoadWithIDB(entry)) {
|
||||||
@corruptedDocs.push(slug)
|
onError = this.loadWithXHR.bind(this, entry, onSuccess, onError);
|
||||||
|
return this.loadWithIDB(entry, onSuccess, onError);
|
||||||
for slug in @corruptedDocs
|
} else {
|
||||||
$.arrayDelete(docs, slug)
|
return this.loadWithXHR(entry, onSuccess, onError);
|
||||||
|
}
|
||||||
if docs.length is 0
|
}
|
||||||
setTimeout(@deleteCorruptedDocs, 0)
|
|
||||||
return
|
loadWithXHR(entry, onSuccess, onError) {
|
||||||
|
return ajax({
|
||||||
txn = @idbTransaction(db, stores: docs, mode: 'readonly', ignoreError: false)
|
url: entry.fileUrl(),
|
||||||
txn.oncomplete = =>
|
dataType: 'html',
|
||||||
setTimeout(@deleteCorruptedDocs, 0) if @corruptedDocs.length > 0
|
success: onSuccess,
|
||||||
return
|
error: onError
|
||||||
|
});
|
||||||
for doc in docs
|
}
|
||||||
txn.objectStore(doc).get('index').onsuccess = (event) =>
|
|
||||||
@corruptedDocs.push(event.target.source.name) unless event.target.result
|
loadWithIDB(entry, onSuccess, onError) {
|
||||||
return
|
return this.db(db => {
|
||||||
return
|
if (!db) {
|
||||||
return
|
onError();
|
||||||
|
return;
|
||||||
deleteCorruptedDocs: =>
|
}
|
||||||
@db (db) =>
|
|
||||||
txn = @idbTransaction(db, stores: ['docs'], mode: 'readwrite', ignoreError: false)
|
if (!db.objectStoreNames.contains(entry.doc.slug)) {
|
||||||
store = txn.objectStore('docs')
|
onError();
|
||||||
while doc = @corruptedDocs.pop()
|
this.loadDocsCache(db);
|
||||||
@cachedDocs[doc] = false
|
return;
|
||||||
store.delete(doc)
|
}
|
||||||
return
|
|
||||||
Raven.captureMessage 'corruptedDocs', level: 'info', extra: { docs: @corruptedDocs.join(',') }
|
const txn = this.idbTransaction(db, {stores: [entry.doc.slug], mode: 'readonly'});
|
||||||
return
|
const store = txn.objectStore(entry.doc.slug);
|
||||||
|
|
||||||
shouldLoadWithIDB: (entry) ->
|
const req = store.get(entry.dbPath());
|
||||||
@useIndexedDB and (not @cachedDocs or @cachedDocs[entry.doc.slug])
|
req.onsuccess = function() {
|
||||||
|
if (req.result) { onSuccess(req.result); } else { onError(); }
|
||||||
idbTransaction: (db, options) ->
|
};
|
||||||
app.lastIDBTransaction = [options.stores, options.mode]
|
req.onerror = function(event) {
|
||||||
txn = db.transaction(options.stores, options.mode)
|
event.preventDefault();
|
||||||
unless options.ignoreError is false
|
onError();
|
||||||
txn.onerror = (event) ->
|
};
|
||||||
event.preventDefault()
|
this.loadDocsCache(db);
|
||||||
return
|
});
|
||||||
unless options.ignoreAbort is false
|
}
|
||||||
txn.onabort = (event) ->
|
|
||||||
event.preventDefault()
|
loadDocsCache(db) {
|
||||||
return
|
if (this.cachedDocs) { return; }
|
||||||
txn
|
this.cachedDocs = {};
|
||||||
|
|
||||||
reset: ->
|
const txn = this.idbTransaction(db, {stores: ['docs'], mode: 'readonly'});
|
||||||
try indexedDB?.deleteDatabase(NAME) catch
|
txn.oncomplete = () => {
|
||||||
return
|
setTimeout(this.checkForCorruptedDocs, 50);
|
||||||
|
};
|
||||||
useIndexedDB: ->
|
|
||||||
try
|
const req = txn.objectStore('docs').openCursor();
|
||||||
if !app.isSingleDoc() and window.indexedDB
|
req.onsuccess = event => {
|
||||||
true
|
let cursor;
|
||||||
else
|
if (!(cursor = event.target.result)) { return; }
|
||||||
@reason = 'not_supported'
|
this.cachedDocs[cursor.key] = cursor.value;
|
||||||
false
|
cursor.continue();
|
||||||
catch
|
};
|
||||||
false
|
req.onerror = function(event) {
|
||||||
|
event.preventDefault();
|
||||||
migrate: ->
|
};
|
||||||
app.settings.set('schema', @userVersion() + 1)
|
}
|
||||||
return
|
|
||||||
|
checkForCorruptedDocs() {
|
||||||
setUserVersion: (version) ->
|
this.db(db => {
|
||||||
app.settings.set('schema', version)
|
let slug;
|
||||||
return
|
this.corruptedDocs = [];
|
||||||
|
const docs = ((() => {
|
||||||
userVersion: ->
|
const result = [];
|
||||||
app.settings.get('schema')
|
for (var key in this.cachedDocs) {
|
||||||
|
var value = this.cachedDocs[key];
|
||||||
|
if (value) {
|
||||||
|
result.push(key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
})());
|
||||||
|
if (docs.length === 0) { return; }
|
||||||
|
|
||||||
|
for (slug of Array.from(docs)) {
|
||||||
|
if (!app.docs.findBy('slug', slug)) {
|
||||||
|
this.corruptedDocs.push(slug);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (slug of Array.from(this.corruptedDocs)) {
|
||||||
|
$.arrayDelete(docs, slug);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (docs.length === 0) {
|
||||||
|
setTimeout(this.deleteCorruptedDocs, 0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const txn = this.idbTransaction(db, {stores: docs, mode: 'readonly', ignoreError: false});
|
||||||
|
txn.oncomplete = () => {
|
||||||
|
if (this.corruptedDocs.length > 0) { setTimeout(this.deleteCorruptedDocs, 0); }
|
||||||
|
};
|
||||||
|
|
||||||
|
for (var doc of Array.from(docs)) {
|
||||||
|
txn.objectStore(doc).get('index').onsuccess = event => {
|
||||||
|
if (!event.target.result) { this.corruptedDocs.push(event.target.source.name); }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteCorruptedDocs() {
|
||||||
|
this.db(db => {
|
||||||
|
let doc;
|
||||||
|
const txn = this.idbTransaction(db, {stores: ['docs'], mode: 'readwrite', ignoreError: false});
|
||||||
|
const store = txn.objectStore('docs');
|
||||||
|
while ((doc = this.corruptedDocs.pop())) {
|
||||||
|
this.cachedDocs[doc] = false;
|
||||||
|
store.delete(doc);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Raven.captureMessage('corruptedDocs', {level: 'info', extra: { docs: this.corruptedDocs.join(',') }});
|
||||||
|
}
|
||||||
|
|
||||||
|
shouldLoadWithIDB(entry) {
|
||||||
|
return this.useIndexedDB && (!this.cachedDocs || this.cachedDocs[entry.doc.slug]);
|
||||||
|
}
|
||||||
|
|
||||||
|
idbTransaction(db, options) {
|
||||||
|
app.lastIDBTransaction = [options.stores, options.mode];
|
||||||
|
const txn = db.transaction(options.stores, options.mode);
|
||||||
|
if (options.ignoreError !== false) {
|
||||||
|
txn.onerror = function(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (options.ignoreAbort !== false) {
|
||||||
|
txn.onabort = function(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return txn;
|
||||||
|
}
|
||||||
|
|
||||||
|
reset() {
|
||||||
|
try { if (typeof indexedDB !== 'undefined' && indexedDB !== null) {
|
||||||
|
indexedDB.deleteDatabase(NAME);
|
||||||
|
} } catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
useIndexedDB() {
|
||||||
|
try {
|
||||||
|
if (!app.isSingleDoc() && window.indexedDB) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
this.reason = 'not_supported';
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
migrate() {
|
||||||
|
app.settings.set('schema', this.userVersion() + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
setUserVersion(version) {
|
||||||
|
app.settings.set('schema', version);
|
||||||
|
}
|
||||||
|
|
||||||
|
userVersion() {
|
||||||
|
return app.settings.get('schema');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,154 +1,199 @@
|
|||||||
class app.Router
|
/*
|
||||||
$.extend @prototype, Events
|
* decaffeinate suggestions:
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
@routes: [
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
['*', 'before' ]
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
['/', 'root' ]
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
['/settings', 'settings' ]
|
* DS207: Consider shorter variations of null checks
|
||||||
['/offline', 'offline' ]
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
['/about', 'about' ]
|
*/
|
||||||
['/news', 'news' ]
|
const Cls = (app.Router = class Router {
|
||||||
['/help', 'help' ]
|
static initClass() {
|
||||||
['/:doc-:type/', 'type' ]
|
$.extend(this.prototype, Events);
|
||||||
['/:doc/', 'doc' ]
|
|
||||||
['/:doc/:path(*)', 'entry' ]
|
this.routes = [
|
||||||
['*', 'notFound' ]
|
['*', 'before' ],
|
||||||
]
|
['/', 'root' ],
|
||||||
|
['/settings', 'settings' ],
|
||||||
constructor: ->
|
['/offline', 'offline' ],
|
||||||
for [path, method] in @constructor.routes
|
['/about', 'about' ],
|
||||||
page path, @[method].bind(@)
|
['/news', 'news' ],
|
||||||
@setInitialPath()
|
['/help', 'help' ],
|
||||||
|
['/:doc-:type/', 'type' ],
|
||||||
start: ->
|
['/:doc/', 'doc' ],
|
||||||
page.start()
|
['/:doc/:path(*)', 'entry' ],
|
||||||
return
|
['*', 'notFound' ]
|
||||||
|
];
|
||||||
show: (path) ->
|
}
|
||||||
page.show(path)
|
|
||||||
return
|
constructor() {
|
||||||
|
for (var [path, method] of Array.from(this.constructor.routes)) {
|
||||||
triggerRoute: (name) ->
|
page(path, this[method].bind(this));
|
||||||
@trigger name, @context
|
}
|
||||||
@trigger 'after', name, @context
|
this.setInitialPath();
|
||||||
return
|
}
|
||||||
|
|
||||||
before: (context, next) ->
|
start() {
|
||||||
previousContext = @context
|
page.start();
|
||||||
@context = context
|
}
|
||||||
@trigger 'before', context
|
|
||||||
|
show(path) {
|
||||||
if res = next()
|
page.show(path);
|
||||||
@context = previousContext
|
}
|
||||||
return res
|
|
||||||
else
|
triggerRoute(name) {
|
||||||
return
|
this.trigger(name, this.context);
|
||||||
|
this.trigger('after', name, this.context);
|
||||||
doc: (context, next) ->
|
}
|
||||||
if doc = app.docs.findBySlug(context.params.doc) or app.disabledDocs.findBySlug(context.params.doc)
|
|
||||||
context.doc = doc
|
before(context, next) {
|
||||||
context.entry = doc.toEntry()
|
let res;
|
||||||
@triggerRoute 'entry'
|
const previousContext = this.context;
|
||||||
return
|
this.context = context;
|
||||||
else
|
this.trigger('before', context);
|
||||||
return next()
|
|
||||||
|
if (res = next()) {
|
||||||
type: (context, next) ->
|
this.context = previousContext;
|
||||||
doc = app.docs.findBySlug(context.params.doc)
|
return res;
|
||||||
|
} else {
|
||||||
if type = doc?.types.findBy 'slug', context.params.type
|
return;
|
||||||
context.doc = doc
|
}
|
||||||
context.type = type
|
}
|
||||||
@triggerRoute 'type'
|
|
||||||
return
|
doc(context, next) {
|
||||||
else
|
let doc;
|
||||||
return next()
|
if (doc = app.docs.findBySlug(context.params.doc) || app.disabledDocs.findBySlug(context.params.doc)) {
|
||||||
|
context.doc = doc;
|
||||||
entry: (context, next) ->
|
context.entry = doc.toEntry();
|
||||||
doc = app.docs.findBySlug(context.params.doc)
|
this.triggerRoute('entry');
|
||||||
return next() unless doc
|
return;
|
||||||
path = context.params.path
|
} else {
|
||||||
hash = context.hash
|
return next();
|
||||||
|
}
|
||||||
if entry = doc.findEntryByPathAndHash(path, hash)
|
}
|
||||||
context.doc = doc
|
|
||||||
context.entry = entry
|
type(context, next) {
|
||||||
@triggerRoute 'entry'
|
let type;
|
||||||
return
|
const doc = app.docs.findBySlug(context.params.doc);
|
||||||
else if path.slice(-6) is '/index'
|
|
||||||
path = path.substr(0, path.length - 6)
|
if (type = doc != null ? doc.types.findBy('slug', context.params.type) : undefined) {
|
||||||
return entry.fullPath() if entry = doc.findEntryByPathAndHash(path, hash)
|
context.doc = doc;
|
||||||
else
|
context.type = type;
|
||||||
path = "#{path}/index"
|
this.triggerRoute('type');
|
||||||
return entry.fullPath() if entry = doc.findEntryByPathAndHash(path, hash)
|
return;
|
||||||
|
} else {
|
||||||
return next()
|
return next();
|
||||||
|
}
|
||||||
root: ->
|
}
|
||||||
return '/' if app.isSingleDoc()
|
|
||||||
@triggerRoute 'root'
|
entry(context, next) {
|
||||||
return
|
let entry;
|
||||||
|
const doc = app.docs.findBySlug(context.params.doc);
|
||||||
settings: (context) ->
|
if (!doc) { return next(); }
|
||||||
return "/#/#{context.path}" if app.isSingleDoc()
|
let {
|
||||||
@triggerRoute 'settings'
|
|
||||||
return
|
|
||||||
|
|
||||||
offline: (context)->
|
|
||||||
return "/#/#{context.path}" if app.isSingleDoc()
|
|
||||||
@triggerRoute 'offline'
|
|
||||||
return
|
|
||||||
|
|
||||||
about: (context) ->
|
|
||||||
return "/#/#{context.path}" if app.isSingleDoc()
|
|
||||||
context.page = 'about'
|
|
||||||
@triggerRoute 'page'
|
|
||||||
return
|
|
||||||
|
|
||||||
news: (context) ->
|
|
||||||
return "/#/#{context.path}" if app.isSingleDoc()
|
|
||||||
context.page = 'news'
|
|
||||||
@triggerRoute 'page'
|
|
||||||
return
|
|
||||||
|
|
||||||
help: (context) ->
|
|
||||||
return "/#/#{context.path}" if app.isSingleDoc()
|
|
||||||
context.page = 'help'
|
|
||||||
@triggerRoute 'page'
|
|
||||||
return
|
|
||||||
|
|
||||||
notFound: (context) ->
|
|
||||||
@triggerRoute 'notFound'
|
|
||||||
return
|
|
||||||
|
|
||||||
isIndex: ->
|
|
||||||
@context?.path is '/' or (app.isSingleDoc() and @context?.entry?.isIndex())
|
|
||||||
|
|
||||||
isSettings: ->
|
|
||||||
@context?.path is '/settings'
|
|
||||||
|
|
||||||
setInitialPath: ->
|
|
||||||
# Remove superfluous forward slashes at the beginning of the path
|
|
||||||
if (path = location.pathname.replace /^\/{2,}/g, '/') isnt location.pathname
|
|
||||||
page.replace path + location.search + location.hash, null, true
|
|
||||||
|
|
||||||
if location.pathname is '/'
|
|
||||||
if path = @getInitialPathFromHash()
|
|
||||||
page.replace path + location.search, null, true
|
|
||||||
else if path = @getInitialPathFromCookie()
|
|
||||||
page.replace path + location.search + location.hash, null, true
|
|
||||||
return
|
|
||||||
|
|
||||||
getInitialPathFromHash: ->
|
|
||||||
try
|
|
||||||
(new RegExp "#/(.+)").exec(decodeURIComponent location.hash)?[1]
|
|
||||||
catch
|
|
||||||
|
|
||||||
getInitialPathFromCookie: ->
|
|
||||||
if path = Cookies.get('initial_path')
|
|
||||||
Cookies.expire('initial_path')
|
|
||||||
path
|
path
|
||||||
|
} = context.params;
|
||||||
replaceHash: (hash) ->
|
const {
|
||||||
page.replace location.pathname + location.search + (hash or ''), null, true
|
hash
|
||||||
return
|
} = context;
|
||||||
|
|
||||||
|
if (entry = doc.findEntryByPathAndHash(path, hash)) {
|
||||||
|
context.doc = doc;
|
||||||
|
context.entry = entry;
|
||||||
|
this.triggerRoute('entry');
|
||||||
|
return;
|
||||||
|
} else if (path.slice(-6) === '/index') {
|
||||||
|
path = path.substr(0, path.length - 6);
|
||||||
|
if (entry = doc.findEntryByPathAndHash(path, hash)) { return entry.fullPath(); }
|
||||||
|
} else {
|
||||||
|
path = `${path}/index`;
|
||||||
|
if (entry = doc.findEntryByPathAndHash(path, hash)) { return entry.fullPath(); }
|
||||||
|
}
|
||||||
|
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
|
||||||
|
root() {
|
||||||
|
if (app.isSingleDoc()) { return '/'; }
|
||||||
|
this.triggerRoute('root');
|
||||||
|
}
|
||||||
|
|
||||||
|
settings(context) {
|
||||||
|
if (app.isSingleDoc()) { return `/#/${context.path}`; }
|
||||||
|
this.triggerRoute('settings');
|
||||||
|
}
|
||||||
|
|
||||||
|
offline(context){
|
||||||
|
if (app.isSingleDoc()) { return `/#/${context.path}`; }
|
||||||
|
this.triggerRoute('offline');
|
||||||
|
}
|
||||||
|
|
||||||
|
about(context) {
|
||||||
|
if (app.isSingleDoc()) { return `/#/${context.path}`; }
|
||||||
|
context.page = 'about';
|
||||||
|
this.triggerRoute('page');
|
||||||
|
}
|
||||||
|
|
||||||
|
news(context) {
|
||||||
|
if (app.isSingleDoc()) { return `/#/${context.path}`; }
|
||||||
|
context.page = 'news';
|
||||||
|
this.triggerRoute('page');
|
||||||
|
}
|
||||||
|
|
||||||
|
help(context) {
|
||||||
|
if (app.isSingleDoc()) { return `/#/${context.path}`; }
|
||||||
|
context.page = 'help';
|
||||||
|
this.triggerRoute('page');
|
||||||
|
}
|
||||||
|
|
||||||
|
notFound(context) {
|
||||||
|
this.triggerRoute('notFound');
|
||||||
|
}
|
||||||
|
|
||||||
|
isIndex() {
|
||||||
|
return ((this.context != null ? this.context.path : undefined) === '/') || (app.isSingleDoc() && __guard__(this.context != null ? this.context.entry : undefined, x => x.isIndex()));
|
||||||
|
}
|
||||||
|
|
||||||
|
isSettings() {
|
||||||
|
return (this.context != null ? this.context.path : undefined) === '/settings';
|
||||||
|
}
|
||||||
|
|
||||||
|
setInitialPath() {
|
||||||
|
// Remove superfluous forward slashes at the beginning of the path
|
||||||
|
let path;
|
||||||
|
if ((path = location.pathname.replace(/^\/{2,}/g, '/')) !== location.pathname) {
|
||||||
|
page.replace(path + location.search + location.hash, null, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (location.pathname === '/') {
|
||||||
|
if (path = this.getInitialPathFromHash()) {
|
||||||
|
page.replace(path + location.search, null, true);
|
||||||
|
} else if (path = this.getInitialPathFromCookie()) {
|
||||||
|
page.replace(path + location.search + location.hash, null, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getInitialPathFromHash() {
|
||||||
|
try {
|
||||||
|
return __guard__((new RegExp("#/(.+)")).exec(decodeURIComponent(location.hash)), x => x[1]);
|
||||||
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
getInitialPathFromCookie() {
|
||||||
|
let path;
|
||||||
|
if (path = Cookies.get('initial_path')) {
|
||||||
|
Cookies.expire('initial_path');
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
replaceHash(hash) {
|
||||||
|
page.replace(location.pathname + location.search + (hash || ''), null, true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,292 +1,373 @@
|
|||||||
#
|
/*
|
||||||
# Match functions
|
* decaffeinate suggestions:
|
||||||
#
|
* DS002: Fix invalid constructor
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
SEPARATOR = '.'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS104: Avoid inline assignments
|
||||||
query =
|
* DS202: Simplify dynamic range loops
|
||||||
queryLength =
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
value =
|
* DS207: Consider shorter variations of null checks
|
||||||
valueLength =
|
* DS209: Avoid top-level return
|
||||||
matcher = # current match function
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
fuzzyRegexp = # query fuzzy regexp
|
*/
|
||||||
index = # position of the query in the string being matched
|
//
|
||||||
lastIndex = # last position of the query in the string being matched
|
// Match functions
|
||||||
match = # regexp match data
|
//
|
||||||
matchIndex =
|
|
||||||
matchLength =
|
let fuzzyRegexp, i, index, lastIndex, match, matcher, matchIndex, matchLength, queryLength, score, separators, value, valueLength;
|
||||||
score = # score for the current match
|
const SEPARATOR = '.';
|
||||||
separators = # counter
|
|
||||||
i = null # cursor
|
let query =
|
||||||
|
(queryLength =
|
||||||
`function exactMatch() {`
|
(value =
|
||||||
index = value.indexOf(query)
|
(valueLength =
|
||||||
return unless index >= 0
|
(matcher = // current match function
|
||||||
|
(fuzzyRegexp = // query fuzzy regexp
|
||||||
lastIndex = value.lastIndexOf(query)
|
(index = // position of the query in the string being matched
|
||||||
|
(lastIndex = // last position of the query in the string being matched
|
||||||
if index isnt lastIndex
|
(match = // regexp match data
|
||||||
return Math.max(scoreExactMatch(), ((index = lastIndex) and scoreExactMatch()) or 0)
|
(matchIndex =
|
||||||
else
|
(matchLength =
|
||||||
return scoreExactMatch()
|
(score = // score for the current match
|
||||||
`}`
|
(separators = // counter
|
||||||
|
(i = null))))))))))))); // cursor
|
||||||
`function scoreExactMatch() {`
|
|
||||||
# Remove one point for each unmatched character.
|
function exactMatch() {
|
||||||
score = 100 - (valueLength - queryLength)
|
index = value.indexOf(query);
|
||||||
|
if (!(index >= 0)) { return; }
|
||||||
if index > 0
|
|
||||||
# If the character preceding the query is a dot, assign the same score
|
lastIndex = value.lastIndexOf(query);
|
||||||
# as if the query was found at the beginning of the string, minus one.
|
|
||||||
if value.charAt(index - 1) is SEPARATOR
|
if (index !== lastIndex) {
|
||||||
score += index - 1
|
return Math.max(scoreExactMatch(), ((index = lastIndex) && scoreExactMatch()) || 0);
|
||||||
# Don't match a single-character query unless it's found at the beginning
|
} else {
|
||||||
# of the string or is preceded by a dot.
|
return scoreExactMatch();
|
||||||
else if queryLength is 1
|
}
|
||||||
return
|
}
|
||||||
# (1) Remove one point for each unmatched character up to the nearest
|
|
||||||
# preceding dot or the beginning of the string.
|
function scoreExactMatch() {
|
||||||
# (2) Remove one point for each unmatched character following the query.
|
// Remove one point for each unmatched character.
|
||||||
else
|
score = 100 - (valueLength - queryLength);
|
||||||
i = index - 2
|
|
||||||
i-- while i >= 0 and value.charAt(i) isnt SEPARATOR
|
if (index > 0) {
|
||||||
score -= (index - i) + # (1)
|
// If the character preceding the query is a dot, assign the same score
|
||||||
(valueLength - queryLength - index) # (2)
|
// as if the query was found at the beginning of the string, minus one.
|
||||||
|
if (value.charAt(index - 1) === SEPARATOR) {
|
||||||
# Remove one point for each dot preceding the query, except for the one
|
score += index - 1;
|
||||||
# immediately before the query.
|
// Don't match a single-character query unless it's found at the beginning
|
||||||
separators = 0
|
// of the string or is preceded by a dot.
|
||||||
i = index - 2
|
} else if (queryLength === 1) {
|
||||||
while i >= 0
|
return;
|
||||||
separators++ if value.charAt(i) is SEPARATOR
|
// (1) Remove one point for each unmatched character up to the nearest
|
||||||
i--
|
// preceding dot or the beginning of the string.
|
||||||
score -= separators
|
// (2) Remove one point for each unmatched character following the query.
|
||||||
|
} else {
|
||||||
# Remove five points for each dot following the query.
|
i = index - 2;
|
||||||
separators = 0
|
while ((i >= 0) && (value.charAt(i) !== SEPARATOR)) { i--; }
|
||||||
i = valueLength - queryLength - index - 1
|
score -= (index - i) + // (1)
|
||||||
while i >= 0
|
(valueLength - queryLength - index); // (2)
|
||||||
separators++ if value.charAt(index + queryLength + i) is SEPARATOR
|
}
|
||||||
i--
|
|
||||||
score -= separators * 5
|
// Remove one point for each dot preceding the query, except for the one
|
||||||
|
// immediately before the query.
|
||||||
return Math.max 1, score
|
separators = 0;
|
||||||
`}`
|
i = index - 2;
|
||||||
|
while (i >= 0) {
|
||||||
`function fuzzyMatch() {`
|
if (value.charAt(i) === SEPARATOR) { separators++; }
|
||||||
return if valueLength <= queryLength or value.indexOf(query) >= 0
|
i--;
|
||||||
return unless match = fuzzyRegexp.exec(value)
|
}
|
||||||
matchIndex = match.index
|
score -= separators;
|
||||||
matchLength = match[0].length
|
}
|
||||||
score = scoreFuzzyMatch()
|
|
||||||
if match = fuzzyRegexp.exec(value.slice(i = value.lastIndexOf(SEPARATOR) + 1))
|
// Remove five points for each dot following the query.
|
||||||
matchIndex = i + match.index
|
separators = 0;
|
||||||
matchLength = match[0].length
|
i = valueLength - queryLength - index - 1;
|
||||||
return Math.max(score, scoreFuzzyMatch())
|
while (i >= 0) {
|
||||||
else
|
if (value.charAt(index + queryLength + i) === SEPARATOR) { separators++; }
|
||||||
return score
|
i--;
|
||||||
`}`
|
}
|
||||||
|
score -= separators * 5;
|
||||||
`function scoreFuzzyMatch() {`
|
|
||||||
# When the match is at the beginning of the string or preceded by a dot.
|
return Math.max(1, score);
|
||||||
if matchIndex is 0 or value.charAt(matchIndex - 1) is SEPARATOR
|
}
|
||||||
return Math.max 66, 100 - matchLength
|
|
||||||
# When the match is at the end of the string.
|
function fuzzyMatch() {
|
||||||
else if matchIndex + matchLength is valueLength
|
if ((valueLength <= queryLength) || (value.indexOf(query) >= 0)) { return; }
|
||||||
return Math.max 33, 67 - matchLength
|
if (!(match = fuzzyRegexp.exec(value))) { return; }
|
||||||
# When the match is in the middle of the string.
|
matchIndex = match.index;
|
||||||
else
|
matchLength = match[0].length;
|
||||||
return Math.max 1, 34 - matchLength
|
score = scoreFuzzyMatch();
|
||||||
`}`
|
if (match = fuzzyRegexp.exec(value.slice(i = value.lastIndexOf(SEPARATOR) + 1))) {
|
||||||
|
matchIndex = i + match.index;
|
||||||
#
|
matchLength = match[0].length;
|
||||||
# Searchers
|
return Math.max(score, scoreFuzzyMatch());
|
||||||
#
|
} else {
|
||||||
|
return score;
|
||||||
class app.Searcher
|
}
|
||||||
$.extend @prototype, Events
|
}
|
||||||
|
|
||||||
CHUNK_SIZE = 20000
|
function scoreFuzzyMatch() {
|
||||||
|
// When the match is at the beginning of the string or preceded by a dot.
|
||||||
DEFAULTS =
|
if ((matchIndex === 0) || (value.charAt(matchIndex - 1) === SEPARATOR)) {
|
||||||
max_results: app.config.max_results
|
return Math.max(66, 100 - matchLength);
|
||||||
fuzzy_min_length: 3
|
// When the match is at the end of the string.
|
||||||
|
} else if ((matchIndex + matchLength) === valueLength) {
|
||||||
SEPARATORS_REGEXP = /#|::|:-|->|\$(?=\w)|\-(?=\w)|\:(?=\w)|\ [\/\-&]\ |:\ |\ /g
|
return Math.max(33, 67 - matchLength);
|
||||||
EOS_SEPARATORS_REGEXP = /(\w)[\-:]$/
|
// When the match is in the middle of the string.
|
||||||
INFO_PARANTHESES_REGEXP = /\ \(\w+?\)$/
|
} else {
|
||||||
EMPTY_PARANTHESES_REGEXP = /\(\)/
|
return Math.max(1, 34 - matchLength);
|
||||||
EVENT_REGEXP = /\ event$/
|
}
|
||||||
DOT_REGEXP = /\.+/g
|
}
|
||||||
WHITESPACE_REGEXP = /\s/g
|
|
||||||
|
//
|
||||||
EMPTY_STRING = ''
|
// Searchers
|
||||||
ELLIPSIS = '...'
|
//
|
||||||
STRING = 'string'
|
|
||||||
|
(function() {
|
||||||
@normalizeString: (string) ->
|
let CHUNK_SIZE = undefined;
|
||||||
string
|
let DEFAULTS = undefined;
|
||||||
.toLowerCase()
|
let SEPARATORS_REGEXP = undefined;
|
||||||
.replace ELLIPSIS, EMPTY_STRING
|
let EOS_SEPARATORS_REGEXP = undefined;
|
||||||
.replace EVENT_REGEXP, EMPTY_STRING
|
let INFO_PARANTHESES_REGEXP = undefined;
|
||||||
.replace INFO_PARANTHESES_REGEXP, EMPTY_STRING
|
let EMPTY_PARANTHESES_REGEXP = undefined;
|
||||||
.replace SEPARATORS_REGEXP, SEPARATOR
|
let EVENT_REGEXP = undefined;
|
||||||
.replace DOT_REGEXP, SEPARATOR
|
let DOT_REGEXP = undefined;
|
||||||
.replace EMPTY_PARANTHESES_REGEXP, EMPTY_STRING
|
let WHITESPACE_REGEXP = undefined;
|
||||||
.replace WHITESPACE_REGEXP, EMPTY_STRING
|
let EMPTY_STRING = undefined;
|
||||||
|
let ELLIPSIS = undefined;
|
||||||
@normalizeQuery: (string) ->
|
let STRING = undefined;
|
||||||
string = @normalizeString(string)
|
const Cls = (app.Searcher = class Searcher {
|
||||||
string.replace EOS_SEPARATORS_REGEXP, '$1.'
|
static initClass() {
|
||||||
|
$.extend(this.prototype, Events);
|
||||||
constructor: (options = {}) ->
|
|
||||||
@options = $.extend {}, DEFAULTS, options
|
CHUNK_SIZE = 20000;
|
||||||
|
|
||||||
find: (data, attr, q) ->
|
DEFAULTS = {
|
||||||
@kill()
|
max_results: app.config.max_results,
|
||||||
|
fuzzy_min_length: 3
|
||||||
@data = data
|
};
|
||||||
@attr = attr
|
|
||||||
@query = q
|
SEPARATORS_REGEXP = /#|::|:-|->|\$(?=\w)|\-(?=\w)|\:(?=\w)|\ [\/\-&]\ |:\ |\ /g;
|
||||||
@setup()
|
EOS_SEPARATORS_REGEXP = /(\w)[\-:]$/;
|
||||||
|
INFO_PARANTHESES_REGEXP = /\ \(\w+?\)$/;
|
||||||
if @isValid() then @match() else @end()
|
EMPTY_PARANTHESES_REGEXP = /\(\)/;
|
||||||
return
|
EVENT_REGEXP = /\ event$/;
|
||||||
|
DOT_REGEXP = /\.+/g;
|
||||||
setup: ->
|
WHITESPACE_REGEXP = /\s/g;
|
||||||
query = @query = @constructor.normalizeQuery(@query)
|
|
||||||
queryLength = query.length
|
EMPTY_STRING = '';
|
||||||
@dataLength = @data.length
|
ELLIPSIS = '...';
|
||||||
@matchers = [exactMatch]
|
STRING = 'string';
|
||||||
@totalResults = 0
|
}
|
||||||
@setupFuzzy()
|
|
||||||
return
|
static normalizeString(string) {
|
||||||
|
return string
|
||||||
setupFuzzy: ->
|
.toLowerCase()
|
||||||
if queryLength >= @options.fuzzy_min_length
|
.replace(ELLIPSIS, EMPTY_STRING)
|
||||||
fuzzyRegexp = @queryToFuzzyRegexp(query)
|
.replace(EVENT_REGEXP, EMPTY_STRING)
|
||||||
@matchers.push(fuzzyMatch)
|
.replace(INFO_PARANTHESES_REGEXP, EMPTY_STRING)
|
||||||
else
|
.replace(SEPARATORS_REGEXP, SEPARATOR)
|
||||||
fuzzyRegexp = null
|
.replace(DOT_REGEXP, SEPARATOR)
|
||||||
return
|
.replace(EMPTY_PARANTHESES_REGEXP, EMPTY_STRING)
|
||||||
|
.replace(WHITESPACE_REGEXP, EMPTY_STRING);
|
||||||
isValid: ->
|
}
|
||||||
queryLength > 0 and query isnt SEPARATOR
|
|
||||||
|
static normalizeQuery(string) {
|
||||||
end: ->
|
string = this.normalizeString(string);
|
||||||
@triggerResults [] unless @totalResults
|
return string.replace(EOS_SEPARATORS_REGEXP, '$1.');
|
||||||
@trigger 'end'
|
}
|
||||||
@free()
|
|
||||||
return
|
constructor(options) {
|
||||||
|
this.match = this.match.bind(this);
|
||||||
kill: ->
|
this.matchChunks = this.matchChunks.bind(this);
|
||||||
if @timeout
|
if (options == null) { options = {}; }
|
||||||
clearTimeout @timeout
|
this.options = $.extend({}, DEFAULTS, options);
|
||||||
@free()
|
}
|
||||||
return
|
|
||||||
|
find(data, attr, q) {
|
||||||
free: ->
|
this.kill();
|
||||||
@data = @attr = @dataLength = @matchers = @matcher = @query =
|
|
||||||
@totalResults = @scoreMap = @cursor = @timeout = null
|
this.data = data;
|
||||||
return
|
this.attr = attr;
|
||||||
|
this.query = q;
|
||||||
match: =>
|
this.setup();
|
||||||
if not @foundEnough() and @matcher = @matchers.shift()
|
|
||||||
@setupMatcher()
|
if (this.isValid()) { this.match(); } else { this.end(); }
|
||||||
@matchChunks()
|
}
|
||||||
else
|
|
||||||
@end()
|
setup() {
|
||||||
return
|
query = (this.query = this.constructor.normalizeQuery(this.query));
|
||||||
|
queryLength = query.length;
|
||||||
setupMatcher: ->
|
this.dataLength = this.data.length;
|
||||||
@cursor = 0
|
this.matchers = [exactMatch];
|
||||||
@scoreMap = new Array(101)
|
this.totalResults = 0;
|
||||||
return
|
this.setupFuzzy();
|
||||||
|
}
|
||||||
matchChunks: =>
|
|
||||||
@matchChunk()
|
setupFuzzy() {
|
||||||
|
if (queryLength >= this.options.fuzzy_min_length) {
|
||||||
if @cursor is @dataLength or @scoredEnough()
|
fuzzyRegexp = this.queryToFuzzyRegexp(query);
|
||||||
@delay @match
|
this.matchers.push(fuzzyMatch);
|
||||||
@sendResults()
|
} else {
|
||||||
else
|
fuzzyRegexp = null;
|
||||||
@delay @matchChunks
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
matchChunk: ->
|
isValid() {
|
||||||
matcher = @matcher
|
return (queryLength > 0) && (query !== SEPARATOR);
|
||||||
for [0...@chunkSize()]
|
}
|
||||||
value = @data[@cursor][@attr]
|
|
||||||
if value.split # string
|
end() {
|
||||||
valueLength = value.length
|
if (!this.totalResults) { this.triggerResults([]); }
|
||||||
@addResult(@data[@cursor], score) if score = matcher()
|
this.trigger('end');
|
||||||
else # array
|
this.free();
|
||||||
score = 0
|
}
|
||||||
for value in @data[@cursor][@attr]
|
|
||||||
valueLength = value.length
|
kill() {
|
||||||
score = Math.max(score, matcher() || 0)
|
if (this.timeout) {
|
||||||
@addResult(@data[@cursor], score) if score > 0
|
clearTimeout(this.timeout);
|
||||||
@cursor++
|
this.free();
|
||||||
return
|
}
|
||||||
|
}
|
||||||
chunkSize: ->
|
|
||||||
if @cursor + CHUNK_SIZE > @dataLength
|
free() {
|
||||||
@dataLength % CHUNK_SIZE
|
this.data = (this.attr = (this.dataLength = (this.matchers = (this.matcher = (this.query =
|
||||||
else
|
(this.totalResults = (this.scoreMap = (this.cursor = (this.timeout = null)))))))));
|
||||||
CHUNK_SIZE
|
}
|
||||||
|
|
||||||
scoredEnough: ->
|
match() {
|
||||||
@scoreMap[100]?.length >= @options.max_results
|
if (!this.foundEnough() && (this.matcher = this.matchers.shift())) {
|
||||||
|
this.setupMatcher();
|
||||||
foundEnough: ->
|
this.matchChunks();
|
||||||
@totalResults >= @options.max_results
|
} else {
|
||||||
|
this.end();
|
||||||
addResult: (object, score) ->
|
}
|
||||||
(@scoreMap[Math.round(score)] or= []).push(object)
|
}
|
||||||
@totalResults++
|
|
||||||
return
|
setupMatcher() {
|
||||||
|
this.cursor = 0;
|
||||||
getResults: ->
|
this.scoreMap = new Array(101);
|
||||||
results = []
|
}
|
||||||
for objects in @scoreMap by -1 when objects
|
|
||||||
results.push.apply results, objects
|
matchChunks() {
|
||||||
results[0...@options.max_results]
|
this.matchChunk();
|
||||||
|
|
||||||
sendResults: ->
|
if ((this.cursor === this.dataLength) || this.scoredEnough()) {
|
||||||
results = @getResults()
|
this.delay(this.match);
|
||||||
@triggerResults results if results.length
|
this.sendResults();
|
||||||
return
|
} else {
|
||||||
|
this.delay(this.matchChunks);
|
||||||
triggerResults: (results) ->
|
}
|
||||||
@trigger 'results', results
|
}
|
||||||
return
|
|
||||||
|
matchChunk() {
|
||||||
delay: (fn) ->
|
({
|
||||||
@timeout = setTimeout(fn, 1)
|
matcher
|
||||||
|
} = this);
|
||||||
queryToFuzzyRegexp: (string) ->
|
for (let j = 0, end = this.chunkSize(), asc = 0 <= end; asc ? j < end : j > end; asc ? j++ : j--) {
|
||||||
chars = string.split ''
|
value = this.data[this.cursor][this.attr];
|
||||||
chars[i] = $.escapeRegexp(char) for char, i in chars
|
if (value.split) { // string
|
||||||
new RegExp chars.join('.*?') # abc -> /a.*?b.*?c.*?/
|
valueLength = value.length;
|
||||||
|
if (score = matcher()) { this.addResult(this.data[this.cursor], score); }
|
||||||
class app.SynchronousSearcher extends app.Searcher
|
} else { // array
|
||||||
match: =>
|
score = 0;
|
||||||
if @matcher
|
for (value of Array.from(this.data[this.cursor][this.attr])) {
|
||||||
@allResults or= []
|
valueLength = value.length;
|
||||||
@allResults.push.apply @allResults, @getResults()
|
score = Math.max(score, matcher() || 0);
|
||||||
super
|
}
|
||||||
|
if (score > 0) { this.addResult(this.data[this.cursor], score); }
|
||||||
free: ->
|
}
|
||||||
@allResults = null
|
this.cursor++;
|
||||||
super
|
}
|
||||||
|
}
|
||||||
end: ->
|
|
||||||
@sendResults true
|
chunkSize() {
|
||||||
super
|
if ((this.cursor + CHUNK_SIZE) > this.dataLength) {
|
||||||
|
return this.dataLength % CHUNK_SIZE;
|
||||||
sendResults: (end) ->
|
} else {
|
||||||
if end and @allResults?.length
|
return CHUNK_SIZE;
|
||||||
@triggerResults @allResults
|
}
|
||||||
|
}
|
||||||
delay: (fn) ->
|
|
||||||
fn()
|
scoredEnough() {
|
||||||
|
return (this.scoreMap[100] != null ? this.scoreMap[100].length : undefined) >= this.options.max_results;
|
||||||
|
}
|
||||||
|
|
||||||
|
foundEnough() {
|
||||||
|
return this.totalResults >= this.options.max_results;
|
||||||
|
}
|
||||||
|
|
||||||
|
addResult(object, score) {
|
||||||
|
let name;
|
||||||
|
(this.scoreMap[name = Math.round(score)] || (this.scoreMap[name] = [])).push(object);
|
||||||
|
this.totalResults++;
|
||||||
|
}
|
||||||
|
|
||||||
|
getResults() {
|
||||||
|
const results = [];
|
||||||
|
for (let j = this.scoreMap.length - 1; j >= 0; j--) {
|
||||||
|
var objects = this.scoreMap[j];
|
||||||
|
if (objects) {
|
||||||
|
results.push.apply(results, objects);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return results.slice(0, this.options.max_results);
|
||||||
|
}
|
||||||
|
|
||||||
|
sendResults() {
|
||||||
|
const results = this.getResults();
|
||||||
|
if (results.length) { this.triggerResults(results); }
|
||||||
|
}
|
||||||
|
|
||||||
|
triggerResults(results) {
|
||||||
|
this.trigger('results', results);
|
||||||
|
}
|
||||||
|
|
||||||
|
delay(fn) {
|
||||||
|
return this.timeout = setTimeout(fn, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
queryToFuzzyRegexp(string) {
|
||||||
|
const chars = string.split('');
|
||||||
|
for (i = 0; i < chars.length; i++) { var char = chars[i]; chars[i] = $.escapeRegexp(char); }
|
||||||
|
return new RegExp(chars.join('.*?'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls; // abc -> /a.*?b.*?c.*?/
|
||||||
|
})();
|
||||||
|
|
||||||
|
app.SynchronousSearcher = class SynchronousSearcher extends app.Searcher {
|
||||||
|
constructor(...args) {
|
||||||
|
this.match = this.match.bind(this);
|
||||||
|
super(...args);
|
||||||
|
}
|
||||||
|
|
||||||
|
match() {
|
||||||
|
if (this.matcher) {
|
||||||
|
if (!this.allResults) { this.allResults = []; }
|
||||||
|
this.allResults.push.apply(this.allResults, this.getResults());
|
||||||
|
}
|
||||||
|
return super.match(...arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
free() {
|
||||||
|
this.allResults = null;
|
||||||
|
return super.free(...arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
end() {
|
||||||
|
this.sendResults(true);
|
||||||
|
return super.end(...arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
sendResults(end) {
|
||||||
|
if (end && (this.allResults != null ? this.allResults.length : undefined)) {
|
||||||
|
return this.triggerResults(this.allResults);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
delay(fn) {
|
||||||
|
return fn();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,49 +1,66 @@
|
|||||||
class app.ServiceWorker
|
/*
|
||||||
$.extend @prototype, Events
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.ServiceWorker = class ServiceWorker {
|
||||||
|
static initClass() {
|
||||||
|
$.extend(this.prototype, Events);
|
||||||
|
}
|
||||||
|
|
||||||
@isEnabled: ->
|
static isEnabled() {
|
||||||
!!navigator.serviceWorker and app.config.service_worker_enabled
|
return !!navigator.serviceWorker && app.config.service_worker_enabled;
|
||||||
|
}
|
||||||
|
|
||||||
constructor: ->
|
constructor() {
|
||||||
@registration = null
|
this.onUpdateFound = this.onUpdateFound.bind(this);
|
||||||
@notifyUpdate = true
|
this.onStateChange = this.onStateChange.bind(this);
|
||||||
|
this.registration = null;
|
||||||
|
this.notifyUpdate = true;
|
||||||
|
|
||||||
navigator.serviceWorker.register(app.config.service_worker_path, {scope: '/'})
|
navigator.serviceWorker.register(app.config.service_worker_path, {scope: '/'})
|
||||||
.then(
|
.then(
|
||||||
(registration) => @updateRegistration(registration),
|
registration => this.updateRegistration(registration),
|
||||||
(error) -> console.error('Could not register service worker:', error)
|
error => console.error('Could not register service worker:', error));
|
||||||
)
|
}
|
||||||
|
|
||||||
update: ->
|
update() {
|
||||||
return unless @registration
|
if (!this.registration) { return; }
|
||||||
@notifyUpdate = true
|
this.notifyUpdate = true;
|
||||||
return @registration.update().catch(->)
|
return this.registration.update().catch(function() {});
|
||||||
|
}
|
||||||
updateInBackground: ->
|
|
||||||
return unless @registration
|
updateInBackground() {
|
||||||
@notifyUpdate = false
|
if (!this.registration) { return; }
|
||||||
return @registration.update().catch(->)
|
this.notifyUpdate = false;
|
||||||
|
return this.registration.update().catch(function() {});
|
||||||
reload: ->
|
}
|
||||||
return @updateInBackground().then(() -> app.reboot())
|
|
||||||
|
reload() {
|
||||||
updateRegistration: (registration) ->
|
return this.updateInBackground().then(() => app.reboot());
|
||||||
@registration = registration
|
}
|
||||||
$.on @registration, 'updatefound', @onUpdateFound
|
|
||||||
return
|
updateRegistration(registration) {
|
||||||
|
this.registration = registration;
|
||||||
onUpdateFound: =>
|
$.on(this.registration, 'updatefound', this.onUpdateFound);
|
||||||
$.off @installingRegistration, 'statechange', @onStateChange() if @installingRegistration
|
}
|
||||||
@installingRegistration = @registration.installing
|
|
||||||
$.on @installingRegistration, 'statechange', @onStateChange
|
onUpdateFound() {
|
||||||
return
|
if (this.installingRegistration) { $.off(this.installingRegistration, 'statechange', this.onStateChange()); }
|
||||||
|
this.installingRegistration = this.registration.installing;
|
||||||
onStateChange: =>
|
$.on(this.installingRegistration, 'statechange', this.onStateChange);
|
||||||
if @installingRegistration and @installingRegistration.state == 'installed' and navigator.serviceWorker.controller
|
}
|
||||||
@installingRegistration = null
|
|
||||||
@onUpdateReady()
|
onStateChange() {
|
||||||
return
|
if (this.installingRegistration && (this.installingRegistration.state === 'installed') && navigator.serviceWorker.controller) {
|
||||||
|
this.installingRegistration = null;
|
||||||
onUpdateReady: ->
|
this.onUpdateReady();
|
||||||
@trigger 'updateready' if @notifyUpdate
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
|
onUpdateReady() {
|
||||||
|
if (this.notifyUpdate) { this.trigger('updateready'); }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,170 +1,219 @@
|
|||||||
class app.Settings
|
/*
|
||||||
PREFERENCE_KEYS = [
|
* decaffeinate suggestions:
|
||||||
'hideDisabled'
|
* DS101: Remove unnecessary use of Array.from
|
||||||
'hideIntro'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
'manualUpdate'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
'fastScroll'
|
* DS104: Avoid inline assignments
|
||||||
'arrowScroll'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
'analyticsConsent'
|
* DS207: Consider shorter variations of null checks
|
||||||
'docs'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
'dark' # legacy
|
*/
|
||||||
'theme'
|
(function() {
|
||||||
'layout'
|
let PREFERENCE_KEYS = undefined;
|
||||||
'size'
|
let INTERNAL_KEYS = undefined;
|
||||||
'tips'
|
const Cls = (app.Settings = class Settings {
|
||||||
'noAutofocus'
|
static initClass() {
|
||||||
'autoInstall'
|
PREFERENCE_KEYS = [
|
||||||
'spaceScroll'
|
'hideDisabled',
|
||||||
'spaceTimeout'
|
'hideIntro',
|
||||||
]
|
'manualUpdate',
|
||||||
|
'fastScroll',
|
||||||
INTERNAL_KEYS = [
|
'arrowScroll',
|
||||||
'count'
|
'analyticsConsent',
|
||||||
'schema'
|
'docs',
|
||||||
'version'
|
'dark', // legacy
|
||||||
'news'
|
'theme',
|
||||||
]
|
'layout',
|
||||||
|
'size',
|
||||||
LAYOUTS: [
|
'tips',
|
||||||
'_max-width'
|
'noAutofocus',
|
||||||
'_sidebar-hidden'
|
'autoInstall',
|
||||||
'_native-scrollbars'
|
'spaceScroll',
|
||||||
'_text-justify-hyphenate'
|
'spaceTimeout'
|
||||||
]
|
];
|
||||||
|
|
||||||
@defaults:
|
INTERNAL_KEYS = [
|
||||||
count: 0
|
'count',
|
||||||
hideDisabled: false
|
'schema',
|
||||||
hideIntro: false
|
'version',
|
||||||
news: 0
|
'news'
|
||||||
manualUpdate: false
|
];
|
||||||
schema: 1
|
|
||||||
analyticsConsent: false
|
this.prototype.LAYOUTS = [
|
||||||
theme: 'auto'
|
'_max-width',
|
||||||
spaceScroll: 1
|
'_sidebar-hidden',
|
||||||
spaceTimeout: 0.5
|
'_native-scrollbars',
|
||||||
|
'_text-justify-hyphenate'
|
||||||
constructor: ->
|
];
|
||||||
@store = new CookiesStore
|
|
||||||
@cache = {}
|
this.defaults = {
|
||||||
@autoSupported = window.matchMedia('(prefers-color-scheme)').media != 'not all'
|
count: 0,
|
||||||
if @autoSupported
|
hideDisabled: false,
|
||||||
@darkModeQuery = window.matchMedia('(prefers-color-scheme: dark)')
|
hideIntro: false,
|
||||||
@darkModeQuery.addListener => @setTheme(@get('theme'))
|
news: 0,
|
||||||
|
manualUpdate: false,
|
||||||
|
schema: 1,
|
||||||
get: (key) ->
|
analyticsConsent: false,
|
||||||
return @cache[key] if @cache.hasOwnProperty(key)
|
theme: 'auto',
|
||||||
@cache[key] = @store.get(key) ? @constructor.defaults[key]
|
spaceScroll: 1,
|
||||||
if key == 'theme' and @cache[key] == 'auto' and !@darkModeQuery
|
spaceTimeout: 0.5
|
||||||
@cache[key] = 'default'
|
};
|
||||||
else
|
}
|
||||||
@cache[key]
|
|
||||||
|
constructor() {
|
||||||
set: (key, value) ->
|
this.store = new CookiesStore;
|
||||||
@store.set(key, value)
|
this.cache = {};
|
||||||
delete @cache[key]
|
this.autoSupported = window.matchMedia('(prefers-color-scheme)').media !== 'not all';
|
||||||
@setTheme(value) if key == 'theme'
|
if (this.autoSupported) {
|
||||||
return
|
this.darkModeQuery = window.matchMedia('(prefers-color-scheme: dark)');
|
||||||
|
this.darkModeQuery.addListener(() => this.setTheme(this.get('theme')));
|
||||||
del: (key) ->
|
}
|
||||||
@store.del(key)
|
}
|
||||||
delete @cache[key]
|
|
||||||
return
|
|
||||||
|
get(key) {
|
||||||
hasDocs: ->
|
let left;
|
||||||
try !!@store.get('docs')
|
if (this.cache.hasOwnProperty(key)) { return this.cache[key]; }
|
||||||
|
this.cache[key] = (left = this.store.get(key)) != null ? left : this.constructor.defaults[key];
|
||||||
getDocs: ->
|
if ((key === 'theme') && (this.cache[key] === 'auto') && !this.darkModeQuery) {
|
||||||
@store.get('docs')?.split('/') or app.config.default_docs
|
return this.cache[key] = 'default';
|
||||||
|
} else {
|
||||||
setDocs: (docs) ->
|
return this.cache[key];
|
||||||
@set 'docs', docs.join('/')
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
getTips: ->
|
set(key, value) {
|
||||||
@store.get('tips')?.split('/') or []
|
this.store.set(key, value);
|
||||||
|
delete this.cache[key];
|
||||||
setTips: (tips) ->
|
if (key === 'theme') { this.setTheme(value); }
|
||||||
@set 'tips', tips.join('/')
|
}
|
||||||
return
|
|
||||||
|
del(key) {
|
||||||
setLayout: (name, enable) ->
|
this.store.del(key);
|
||||||
@toggleLayout(name, enable)
|
delete this.cache[key];
|
||||||
|
}
|
||||||
layout = (@store.get('layout') || '').split(' ')
|
|
||||||
$.arrayDelete(layout, '')
|
hasDocs() {
|
||||||
|
try { return !!this.store.get('docs'); } catch (error) {}
|
||||||
if enable
|
}
|
||||||
layout.push(name) if layout.indexOf(name) is -1
|
|
||||||
else
|
getDocs() {
|
||||||
$.arrayDelete(layout, name)
|
return __guard__(this.store.get('docs'), x => x.split('/')) || app.config.default_docs;
|
||||||
|
}
|
||||||
if layout.length > 0
|
|
||||||
@set 'layout', layout.join(' ')
|
setDocs(docs) {
|
||||||
else
|
this.set('docs', docs.join('/'));
|
||||||
@del 'layout'
|
}
|
||||||
return
|
|
||||||
|
getTips() {
|
||||||
hasLayout: (name) ->
|
return __guard__(this.store.get('tips'), x => x.split('/')) || [];
|
||||||
layout = (@store.get('layout') || '').split(' ')
|
}
|
||||||
layout.indexOf(name) isnt -1
|
|
||||||
|
setTips(tips) {
|
||||||
setSize: (value) ->
|
this.set('tips', tips.join('/'));
|
||||||
@set 'size', value
|
}
|
||||||
return
|
|
||||||
|
setLayout(name, enable) {
|
||||||
dump: ->
|
this.toggleLayout(name, enable);
|
||||||
@store.dump()
|
|
||||||
|
const layout = (this.store.get('layout') || '').split(' ');
|
||||||
export: ->
|
$.arrayDelete(layout, '');
|
||||||
data = @dump()
|
|
||||||
delete data[key] for key in INTERNAL_KEYS
|
if (enable) {
|
||||||
data
|
if (layout.indexOf(name) === -1) { layout.push(name); }
|
||||||
|
} else {
|
||||||
import: (data) ->
|
$.arrayDelete(layout, name);
|
||||||
for key, value of @export()
|
}
|
||||||
@del key unless data.hasOwnProperty(key)
|
|
||||||
for key, value of data
|
if (layout.length > 0) {
|
||||||
@set key, value if PREFERENCE_KEYS.indexOf(key) isnt -1
|
this.set('layout', layout.join(' '));
|
||||||
return
|
} else {
|
||||||
|
this.del('layout');
|
||||||
reset: ->
|
}
|
||||||
@store.reset()
|
}
|
||||||
@cache = {}
|
|
||||||
return
|
hasLayout(name) {
|
||||||
|
const layout = (this.store.get('layout') || '').split(' ');
|
||||||
initLayout: ->
|
return layout.indexOf(name) !== -1;
|
||||||
if @get('dark') is 1
|
}
|
||||||
@set('theme', 'dark')
|
|
||||||
@del 'dark'
|
setSize(value) {
|
||||||
@setTheme(@get('theme'))
|
this.set('size', value);
|
||||||
@toggleLayout(layout, @hasLayout(layout)) for layout in @LAYOUTS
|
}
|
||||||
@initSidebarWidth()
|
|
||||||
return
|
dump() {
|
||||||
|
return this.store.dump();
|
||||||
setTheme: (theme) ->
|
}
|
||||||
if theme is 'auto'
|
|
||||||
theme = if @darkModeQuery.matches then 'dark' else 'default'
|
export() {
|
||||||
classList = document.documentElement.classList
|
const data = this.dump();
|
||||||
classList.remove('_theme-default', '_theme-dark')
|
for (var key of Array.from(INTERNAL_KEYS)) { delete data[key]; }
|
||||||
classList.add('_theme-' + theme)
|
return data;
|
||||||
@updateColorMeta()
|
}
|
||||||
return
|
|
||||||
|
import(data) {
|
||||||
updateColorMeta: ->
|
let key, value;
|
||||||
color = getComputedStyle(document.documentElement).getPropertyValue('--headerBackground').trim()
|
const object = this.export();
|
||||||
$('meta[name=theme-color]').setAttribute('content', color)
|
for (key in object) {
|
||||||
return
|
value = object[key];
|
||||||
|
if (!data.hasOwnProperty(key)) { this.del(key); }
|
||||||
toggleLayout: (layout, enable) ->
|
}
|
||||||
classList = document.body.classList
|
for (key in data) {
|
||||||
# sidebar is always shown for settings; its state is updated in app.views.Settings
|
value = data[key];
|
||||||
classList.toggle(layout, enable) unless layout is '_sidebar-hidden' and app.router?.isSettings
|
if (PREFERENCE_KEYS.indexOf(key) !== -1) { this.set(key, value); }
|
||||||
classList.toggle('_overlay-scrollbars', $.overlayScrollbarsEnabled())
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
initSidebarWidth: ->
|
reset() {
|
||||||
size = @get('size')
|
this.store.reset();
|
||||||
document.documentElement.style.setProperty('--sidebarWidth', size + 'px') if size
|
this.cache = {};
|
||||||
return
|
}
|
||||||
|
|
||||||
|
initLayout() {
|
||||||
|
if (this.get('dark') === 1) {
|
||||||
|
this.set('theme', 'dark');
|
||||||
|
this.del('dark');
|
||||||
|
}
|
||||||
|
this.setTheme(this.get('theme'));
|
||||||
|
for (var layout of Array.from(this.LAYOUTS)) { this.toggleLayout(layout, this.hasLayout(layout)); }
|
||||||
|
this.initSidebarWidth();
|
||||||
|
}
|
||||||
|
|
||||||
|
setTheme(theme) {
|
||||||
|
if (theme === 'auto') {
|
||||||
|
theme = this.darkModeQuery.matches ? 'dark' : 'default';
|
||||||
|
}
|
||||||
|
const {
|
||||||
|
classList
|
||||||
|
} = document.documentElement;
|
||||||
|
classList.remove('_theme-default', '_theme-dark');
|
||||||
|
classList.add('_theme-' + theme);
|
||||||
|
this.updateColorMeta();
|
||||||
|
}
|
||||||
|
|
||||||
|
updateColorMeta() {
|
||||||
|
const color = getComputedStyle(document.documentElement).getPropertyValue('--headerBackground').trim();
|
||||||
|
$('meta[name=theme-color]').setAttribute('content', color);
|
||||||
|
}
|
||||||
|
|
||||||
|
toggleLayout(layout, enable) {
|
||||||
|
const {
|
||||||
|
classList
|
||||||
|
} = document.body;
|
||||||
|
// sidebar is always shown for settings; its state is updated in app.views.Settings
|
||||||
|
if ((layout !== '_sidebar-hidden') || !(app.router != null ? app.router.isSettings : undefined)) { classList.toggle(layout, enable); }
|
||||||
|
classList.toggle('_overlay-scrollbars', $.overlayScrollbarsEnabled());
|
||||||
|
}
|
||||||
|
|
||||||
|
initSidebarWidth() {
|
||||||
|
const size = this.get('size');
|
||||||
|
if (size) { document.documentElement.style.setProperty('--sidebarWidth', size + 'px'); }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,193 +1,259 @@
|
|||||||
class app.Shortcuts
|
/*
|
||||||
$.extend @prototype, Events
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
constructor: ->
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
@isMac = $.isMac()
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
@start()
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
start: ->
|
*/
|
||||||
$.on document, 'keydown', @onKeydown
|
const Cls = (app.Shortcuts = class Shortcuts {
|
||||||
$.on document, 'keypress', @onKeypress
|
static initClass() {
|
||||||
return
|
$.extend(this.prototype, Events);
|
||||||
|
}
|
||||||
stop: ->
|
|
||||||
$.off document, 'keydown', @onKeydown
|
constructor() {
|
||||||
$.off document, 'keypress', @onKeypress
|
this.onKeydown = this.onKeydown.bind(this);
|
||||||
return
|
this.onKeypress = this.onKeypress.bind(this);
|
||||||
|
this.isMac = $.isMac();
|
||||||
swapArrowKeysBehavior: ->
|
this.start();
|
||||||
app.settings.get('arrowScroll')
|
}
|
||||||
|
|
||||||
spaceScroll: ->
|
start() {
|
||||||
app.settings.get('spaceScroll')
|
$.on(document, 'keydown', this.onKeydown);
|
||||||
|
$.on(document, 'keypress', this.onKeypress);
|
||||||
showTip: ->
|
}
|
||||||
app.showTip('KeyNav')
|
|
||||||
@showTip = null
|
stop() {
|
||||||
|
$.off(document, 'keydown', this.onKeydown);
|
||||||
spaceTimeout: ->
|
$.off(document, 'keypress', this.onKeypress);
|
||||||
app.settings.get('spaceTimeout')
|
}
|
||||||
|
|
||||||
onKeydown: (event) =>
|
swapArrowKeysBehavior() {
|
||||||
return if @buggyEvent(event)
|
return app.settings.get('arrowScroll');
|
||||||
result = if event.ctrlKey or event.metaKey
|
}
|
||||||
@handleKeydownSuperEvent event unless event.altKey or event.shiftKey
|
|
||||||
else if event.shiftKey
|
spaceScroll() {
|
||||||
@handleKeydownShiftEvent event unless event.altKey
|
return app.settings.get('spaceScroll');
|
||||||
else if event.altKey
|
}
|
||||||
@handleKeydownAltEvent event
|
|
||||||
else
|
showTip() {
|
||||||
@handleKeydownEvent event
|
app.showTip('KeyNav');
|
||||||
|
return this.showTip = null;
|
||||||
event.preventDefault() if result is false
|
}
|
||||||
return
|
|
||||||
|
spaceTimeout() {
|
||||||
onKeypress: (event) =>
|
return app.settings.get('spaceTimeout');
|
||||||
return if @buggyEvent(event) or (event.charCode == 63 and document.activeElement.tagName == 'INPUT')
|
}
|
||||||
unless event.ctrlKey or event.metaKey
|
|
||||||
result = @handleKeypressEvent event
|
onKeydown(event) {
|
||||||
event.preventDefault() if result is false
|
if (this.buggyEvent(event)) { return; }
|
||||||
return
|
const result = (() => {
|
||||||
|
if (event.ctrlKey || event.metaKey) {
|
||||||
handleKeydownEvent: (event, _force) ->
|
if (!event.altKey && !event.shiftKey) { return this.handleKeydownSuperEvent(event); }
|
||||||
return @handleKeydownAltEvent(event, true) if not _force and event.which in [37, 38, 39, 40] and @swapArrowKeysBehavior()
|
} else if (event.shiftKey) {
|
||||||
|
if (!event.altKey) { return this.handleKeydownShiftEvent(event); }
|
||||||
if not event.target.form and (48 <= event.which <= 57 or 65 <= event.which <= 90)
|
} else if (event.altKey) {
|
||||||
@trigger 'typing'
|
return this.handleKeydownAltEvent(event);
|
||||||
return
|
} else {
|
||||||
|
return this.handleKeydownEvent(event);
|
||||||
switch event.which
|
}
|
||||||
when 8
|
})();
|
||||||
@trigger 'typing' unless event.target.form
|
|
||||||
when 13
|
if (result === false) { event.preventDefault(); }
|
||||||
@trigger 'enter'
|
}
|
||||||
when 27
|
|
||||||
@trigger 'escape'
|
onKeypress(event) {
|
||||||
false
|
if (this.buggyEvent(event) || ((event.charCode === 63) && (document.activeElement.tagName === 'INPUT'))) { return; }
|
||||||
when 32
|
if (!event.ctrlKey && !event.metaKey) {
|
||||||
if event.target.type is 'search' and @spaceScroll() and (not @lastKeypress or @lastKeypress < Date.now() - (@spaceTimeout() * 1000))
|
const result = this.handleKeypressEvent(event);
|
||||||
@trigger 'pageDown'
|
if (result === false) { event.preventDefault(); }
|
||||||
false
|
}
|
||||||
when 33
|
}
|
||||||
@trigger 'pageUp'
|
|
||||||
when 34
|
handleKeydownEvent(event, _force) {
|
||||||
@trigger 'pageDown'
|
if (!_force && [37, 38, 39, 40].includes(event.which) && this.swapArrowKeysBehavior()) { return this.handleKeydownAltEvent(event, true); }
|
||||||
when 35
|
|
||||||
@trigger 'pageBottom' unless event.target.form
|
if (!event.target.form && ((48 <= event.which && event.which <= 57) || (65 <= event.which && event.which <= 90))) {
|
||||||
when 36
|
this.trigger('typing');
|
||||||
@trigger 'pageTop' unless event.target.form
|
return;
|
||||||
when 37
|
}
|
||||||
@trigger 'left' unless event.target.value
|
|
||||||
when 38
|
switch (event.which) {
|
||||||
@trigger 'up'
|
case 8:
|
||||||
@showTip?()
|
if (!event.target.form) { return this.trigger('typing'); }
|
||||||
false
|
break;
|
||||||
when 39
|
case 13:
|
||||||
@trigger 'right' unless event.target.value
|
return this.trigger('enter');
|
||||||
when 40
|
case 27:
|
||||||
@trigger 'down'
|
this.trigger('escape');
|
||||||
@showTip?()
|
return false;
|
||||||
false
|
case 32:
|
||||||
when 191
|
if ((event.target.type === 'search') && this.spaceScroll() && (!this.lastKeypress || (this.lastKeypress < (Date.now() - (this.spaceTimeout() * 1000))))) {
|
||||||
unless event.target.form
|
this.trigger('pageDown');
|
||||||
@trigger 'typing'
|
return false;
|
||||||
false
|
}
|
||||||
|
break;
|
||||||
handleKeydownSuperEvent: (event) ->
|
case 33:
|
||||||
switch event.which
|
return this.trigger('pageUp');
|
||||||
when 13
|
case 34:
|
||||||
@trigger 'superEnter'
|
return this.trigger('pageDown');
|
||||||
when 37
|
case 35:
|
||||||
if @isMac
|
if (!event.target.form) { return this.trigger('pageBottom'); }
|
||||||
@trigger 'superLeft'
|
break;
|
||||||
false
|
case 36:
|
||||||
when 38
|
if (!event.target.form) { return this.trigger('pageTop'); }
|
||||||
@trigger 'pageTop'
|
break;
|
||||||
false
|
case 37:
|
||||||
when 39
|
if (!event.target.value) { return this.trigger('left'); }
|
||||||
if @isMac
|
break;
|
||||||
@trigger 'superRight'
|
case 38:
|
||||||
false
|
this.trigger('up');
|
||||||
when 40
|
if (typeof this.showTip === 'function') {
|
||||||
@trigger 'pageBottom'
|
this.showTip();
|
||||||
false
|
}
|
||||||
when 188
|
return false;
|
||||||
@trigger 'preferences'
|
case 39:
|
||||||
false
|
if (!event.target.value) { return this.trigger('right'); }
|
||||||
|
break;
|
||||||
handleKeydownShiftEvent: (event, _force) ->
|
case 40:
|
||||||
return @handleKeydownEvent(event, true) if not _force and event.which in [37, 38, 39, 40] and @swapArrowKeysBehavior()
|
this.trigger('down');
|
||||||
|
if (typeof this.showTip === 'function') {
|
||||||
if not event.target.form and 65 <= event.which <= 90
|
this.showTip();
|
||||||
@trigger 'typing'
|
}
|
||||||
return
|
return false;
|
||||||
|
case 191:
|
||||||
switch event.which
|
if (!event.target.form) {
|
||||||
when 32
|
this.trigger('typing');
|
||||||
@trigger 'pageUp'
|
return false;
|
||||||
false
|
}
|
||||||
when 38
|
break;
|
||||||
unless getSelection()?.toString()
|
}
|
||||||
@trigger 'altUp'
|
}
|
||||||
false
|
|
||||||
when 40
|
handleKeydownSuperEvent(event) {
|
||||||
unless getSelection()?.toString()
|
switch (event.which) {
|
||||||
@trigger 'altDown'
|
case 13:
|
||||||
false
|
return this.trigger('superEnter');
|
||||||
|
case 37:
|
||||||
handleKeydownAltEvent: (event, _force) ->
|
if (this.isMac) {
|
||||||
return @handleKeydownEvent(event, true) if not _force and event.which in [37, 38, 39, 40] and @swapArrowKeysBehavior()
|
this.trigger('superLeft');
|
||||||
|
return false;
|
||||||
switch event.which
|
}
|
||||||
when 9
|
break;
|
||||||
@trigger 'altRight', event
|
case 38:
|
||||||
when 37
|
this.trigger('pageTop');
|
||||||
unless @isMac
|
return false;
|
||||||
@trigger 'superLeft'
|
case 39:
|
||||||
false
|
if (this.isMac) {
|
||||||
when 38
|
this.trigger('superRight');
|
||||||
@trigger 'altUp'
|
return false;
|
||||||
false
|
}
|
||||||
when 39
|
break;
|
||||||
unless @isMac
|
case 40:
|
||||||
@trigger 'superRight'
|
this.trigger('pageBottom');
|
||||||
false
|
return false;
|
||||||
when 40
|
case 188:
|
||||||
@trigger 'altDown'
|
this.trigger('preferences');
|
||||||
false
|
return false;
|
||||||
when 67
|
}
|
||||||
@trigger 'altC'
|
}
|
||||||
false
|
|
||||||
when 68
|
handleKeydownShiftEvent(event, _force) {
|
||||||
@trigger 'altD'
|
if (!_force && [37, 38, 39, 40].includes(event.which) && this.swapArrowKeysBehavior()) { return this.handleKeydownEvent(event, true); }
|
||||||
false
|
|
||||||
when 70
|
if (!event.target.form && (65 <= event.which && event.which <= 90)) {
|
||||||
@trigger 'altF', event
|
this.trigger('typing');
|
||||||
when 71
|
return;
|
||||||
@trigger 'altG'
|
}
|
||||||
false
|
|
||||||
when 79
|
switch (event.which) {
|
||||||
@trigger 'altO'
|
case 32:
|
||||||
false
|
this.trigger('pageUp');
|
||||||
when 82
|
return false;
|
||||||
@trigger 'altR'
|
case 38:
|
||||||
false
|
if (!__guard__(getSelection(), x => x.toString())) {
|
||||||
when 83
|
this.trigger('altUp');
|
||||||
@trigger 'altS'
|
return false;
|
||||||
false
|
}
|
||||||
|
break;
|
||||||
handleKeypressEvent: (event) ->
|
case 40:
|
||||||
if event.which is 63 and not event.target.value
|
if (!__guard__(getSelection(), x1 => x1.toString())) {
|
||||||
@trigger 'help'
|
this.trigger('altDown');
|
||||||
false
|
return false;
|
||||||
else
|
}
|
||||||
@lastKeypress = Date.now()
|
break;
|
||||||
|
}
|
||||||
buggyEvent: (event) ->
|
}
|
||||||
try
|
|
||||||
event.target
|
handleKeydownAltEvent(event, _force) {
|
||||||
event.ctrlKey
|
if (!_force && [37, 38, 39, 40].includes(event.which) && this.swapArrowKeysBehavior()) { return this.handleKeydownEvent(event, true); }
|
||||||
event.which
|
|
||||||
return false
|
switch (event.which) {
|
||||||
catch
|
case 9:
|
||||||
return true
|
return this.trigger('altRight', event);
|
||||||
|
case 37:
|
||||||
|
if (!this.isMac) {
|
||||||
|
this.trigger('superLeft');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 38:
|
||||||
|
this.trigger('altUp');
|
||||||
|
return false;
|
||||||
|
case 39:
|
||||||
|
if (!this.isMac) {
|
||||||
|
this.trigger('superRight');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 40:
|
||||||
|
this.trigger('altDown');
|
||||||
|
return false;
|
||||||
|
case 67:
|
||||||
|
this.trigger('altC');
|
||||||
|
return false;
|
||||||
|
case 68:
|
||||||
|
this.trigger('altD');
|
||||||
|
return false;
|
||||||
|
case 70:
|
||||||
|
return this.trigger('altF', event);
|
||||||
|
case 71:
|
||||||
|
this.trigger('altG');
|
||||||
|
return false;
|
||||||
|
case 79:
|
||||||
|
this.trigger('altO');
|
||||||
|
return false;
|
||||||
|
case 82:
|
||||||
|
this.trigger('altR');
|
||||||
|
return false;
|
||||||
|
case 83:
|
||||||
|
this.trigger('altS');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
handleKeypressEvent(event) {
|
||||||
|
if ((event.which === 63) && !event.target.value) {
|
||||||
|
this.trigger('help');
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
return this.lastKeypress = Date.now();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
buggyEvent(event) {
|
||||||
|
try {
|
||||||
|
event.target;
|
||||||
|
event.ctrlKey;
|
||||||
|
event.which;
|
||||||
|
return false;
|
||||||
|
} catch (error) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,39 +1,54 @@
|
|||||||
class app.UpdateChecker
|
/*
|
||||||
constructor: ->
|
* decaffeinate suggestions:
|
||||||
@lastCheck = Date.now()
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.UpdateChecker = class UpdateChecker {
|
||||||
|
constructor() {
|
||||||
|
this.checkDocs = this.checkDocs.bind(this);
|
||||||
|
this.onFocus = this.onFocus.bind(this);
|
||||||
|
this.lastCheck = Date.now();
|
||||||
|
|
||||||
$.on window, 'focus', @onFocus
|
$.on(window, 'focus', this.onFocus);
|
||||||
app.serviceWorker?.on 'updateready', @onUpdateReady
|
if (app.serviceWorker != null) {
|
||||||
|
app.serviceWorker.on('updateready', this.onUpdateReady);
|
||||||
|
}
|
||||||
|
|
||||||
setTimeout @checkDocs, 0
|
setTimeout(this.checkDocs, 0);
|
||||||
|
}
|
||||||
|
|
||||||
check: ->
|
check() {
|
||||||
if app.serviceWorker
|
if (app.serviceWorker) {
|
||||||
app.serviceWorker.update()
|
app.serviceWorker.update();
|
||||||
else
|
} else {
|
||||||
ajax
|
ajax({
|
||||||
url: $('script[src*="application"]').getAttribute('src')
|
url: $('script[src*="application"]').getAttribute('src'),
|
||||||
dataType: 'application/javascript'
|
dataType: 'application/javascript',
|
||||||
error: (_, xhr) => @onUpdateReady() if xhr.status is 404
|
error: (_, xhr) => { if (xhr.status === 404) { return this.onUpdateReady(); } }
|
||||||
return
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onUpdateReady: ->
|
onUpdateReady() {
|
||||||
new app.views.Notif 'UpdateReady', autoHide: null
|
new app.views.Notif('UpdateReady', {autoHide: null});
|
||||||
return
|
}
|
||||||
|
|
||||||
checkDocs: =>
|
checkDocs() {
|
||||||
unless app.settings.get('manualUpdate')
|
if (!app.settings.get('manualUpdate')) {
|
||||||
app.docs.updateInBackground()
|
app.docs.updateInBackground();
|
||||||
else
|
} else {
|
||||||
app.docs.checkForUpdates (i) => @onDocsUpdateReady() if i > 0
|
app.docs.checkForUpdates(i => { if (i > 0) { return this.onDocsUpdateReady(); } });
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onDocsUpdateReady: ->
|
onDocsUpdateReady() {
|
||||||
new app.views.Notif 'UpdateDocs', autoHide: null
|
new app.views.Notif('UpdateDocs', {autoHide: null});
|
||||||
return
|
}
|
||||||
|
|
||||||
onFocus: =>
|
onFocus() {
|
||||||
if Date.now() - @lastCheck > 21600e3
|
if ((Date.now() - this.lastCheck) > 21600e3) {
|
||||||
@lastCheck = Date.now()
|
this.lastCheck = Date.now();
|
||||||
@check()
|
this.check();
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,31 +1,38 @@
|
|||||||
#= require_tree ./vendor
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require_tree ./vendor
|
||||||
|
|
||||||
#= require lib/license
|
//= require lib/license
|
||||||
#= require_tree ./lib
|
//= require_tree ./lib
|
||||||
|
|
||||||
#= require app/app
|
//= require app/app
|
||||||
#= require app/config
|
//= require app/config
|
||||||
#= require_tree ./app
|
//= require_tree ./app
|
||||||
|
|
||||||
#= require collections/collection
|
//= require collections/collection
|
||||||
#= require_tree ./collections
|
//= require_tree ./collections
|
||||||
|
|
||||||
#= require models/model
|
//= require models/model
|
||||||
#= require_tree ./models
|
//= require_tree ./models
|
||||||
|
|
||||||
#= require views/view
|
//= require views/view
|
||||||
#= require_tree ./views
|
//= require_tree ./views
|
||||||
|
|
||||||
#= require_tree ./templates
|
//= require_tree ./templates
|
||||||
|
|
||||||
#= require tracking
|
//= require tracking
|
||||||
|
|
||||||
init = ->
|
var init = function() {
|
||||||
document.removeEventListener 'DOMContentLoaded', init, false
|
document.removeEventListener('DOMContentLoaded', init, false);
|
||||||
|
|
||||||
if document.body
|
if (document.body) {
|
||||||
app.init()
|
return app.init();
|
||||||
else
|
} else {
|
||||||
setTimeout(init, 42)
|
return setTimeout(init, 42);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
document.addEventListener 'DOMContentLoaded', init, false
|
document.addEventListener('DOMContentLoaded', init, false);
|
||||||
|
@ -1,55 +1,75 @@
|
|||||||
class app.Collection
|
/*
|
||||||
constructor: (objects = []) ->
|
* decaffeinate suggestions:
|
||||||
@reset objects
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
model: ->
|
* DS207: Consider shorter variations of null checks
|
||||||
app.models[@constructor.model]
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
reset: (objects = []) ->
|
app.Collection = class Collection {
|
||||||
@models = []
|
constructor(objects) {
|
||||||
@add object for object in objects
|
if (objects == null) { objects = []; }
|
||||||
return
|
this.reset(objects);
|
||||||
|
}
|
||||||
add: (object) ->
|
|
||||||
if object instanceof app.Model
|
model() {
|
||||||
@models.push object
|
return app.models[this.constructor.model];
|
||||||
else if object instanceof Array
|
}
|
||||||
@add obj for obj in object
|
|
||||||
else if object instanceof app.Collection
|
reset(objects) {
|
||||||
@models.push object.all()...
|
if (objects == null) { objects = []; }
|
||||||
else
|
this.models = [];
|
||||||
@models.push new (@model())(object)
|
for (var object of Array.from(objects)) { this.add(object); }
|
||||||
return
|
}
|
||||||
|
|
||||||
remove: (model) ->
|
add(object) {
|
||||||
@models.splice @models.indexOf(model), 1
|
if (object instanceof app.Model) {
|
||||||
return
|
this.models.push(object);
|
||||||
|
} else if (object instanceof Array) {
|
||||||
size: ->
|
for (var obj of Array.from(object)) { this.add(obj); }
|
||||||
@models.length
|
} else if (object instanceof app.Collection) {
|
||||||
|
this.models.push(...Array.from(object.all() || []));
|
||||||
isEmpty: ->
|
} else {
|
||||||
@models.length is 0
|
this.models.push(new (this.model())(object));
|
||||||
|
}
|
||||||
each: (fn) ->
|
}
|
||||||
fn(model) for model in @models
|
|
||||||
return
|
remove(model) {
|
||||||
|
this.models.splice(this.models.indexOf(model), 1);
|
||||||
all: ->
|
}
|
||||||
@models
|
|
||||||
|
size() {
|
||||||
contains: (model) ->
|
return this.models.length;
|
||||||
@models.indexOf(model) >= 0
|
}
|
||||||
|
|
||||||
findBy: (attr, value) ->
|
isEmpty() {
|
||||||
for model in @models
|
return this.models.length === 0;
|
||||||
return model if model[attr] is value
|
}
|
||||||
return
|
|
||||||
|
each(fn) {
|
||||||
findAllBy: (attr, value) ->
|
for (var model of Array.from(this.models)) { fn(model); }
|
||||||
model for model in @models when model[attr] is value
|
}
|
||||||
|
|
||||||
countAllBy: (attr, value) ->
|
all() {
|
||||||
i = 0
|
return this.models;
|
||||||
i += 1 for model in @models when model[attr] is value
|
}
|
||||||
i
|
|
||||||
|
contains(model) {
|
||||||
|
return this.models.indexOf(model) >= 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
findBy(attr, value) {
|
||||||
|
for (var model of Array.from(this.models)) {
|
||||||
|
if (model[attr] === value) { return model; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
findAllBy(attr, value) {
|
||||||
|
return Array.from(this.models).filter((model) => model[attr] === value);
|
||||||
|
}
|
||||||
|
|
||||||
|
countAllBy(attr, value) {
|
||||||
|
let i = 0;
|
||||||
|
for (var model of Array.from(this.models)) { if (model[attr] === value) { i += 1; } }
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,85 +1,117 @@
|
|||||||
class app.collections.Docs extends app.Collection
|
/*
|
||||||
@model: 'Doc'
|
* decaffeinate suggestions:
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS202: Simplify dynamic range loops
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
(function() {
|
||||||
|
let NORMALIZE_VERSION_RGX = undefined;
|
||||||
|
let NORMALIZE_VERSION_SUB = undefined;
|
||||||
|
let CONCURRENCY = undefined;
|
||||||
|
const Cls = (app.collections.Docs = class Docs extends app.Collection {
|
||||||
|
static initClass() {
|
||||||
|
this.model = 'Doc';
|
||||||
|
|
||||||
findBySlug: (slug) ->
|
NORMALIZE_VERSION_RGX = /\.(\d)$/;
|
||||||
@findBy('slug', slug) or @findBy('slug_without_version', slug)
|
NORMALIZE_VERSION_SUB = '.0$1';
|
||||||
|
|
||||||
NORMALIZE_VERSION_RGX = /\.(\d)$/
|
// Load models concurrently.
|
||||||
NORMALIZE_VERSION_SUB = '.0$1'
|
// It's not pretty but I didn't want to import a promise library only for this.
|
||||||
sort: ->
|
CONCURRENCY = 3;
|
||||||
@models.sort (a, b) ->
|
}
|
||||||
if a.name is b.name
|
|
||||||
if not a.version or a.version.replace(NORMALIZE_VERSION_RGX, NORMALIZE_VERSION_SUB) > b.version.replace(NORMALIZE_VERSION_RGX, NORMALIZE_VERSION_SUB)
|
|
||||||
-1
|
|
||||||
else
|
|
||||||
1
|
|
||||||
else if a.name.toLowerCase() > b.name.toLowerCase()
|
|
||||||
1
|
|
||||||
else
|
|
||||||
-1
|
|
||||||
|
|
||||||
# Load models concurrently.
|
findBySlug(slug) {
|
||||||
# It's not pretty but I didn't want to import a promise library only for this.
|
return this.findBy('slug', slug) || this.findBy('slug_without_version', slug);
|
||||||
CONCURRENCY = 3
|
}
|
||||||
load: (onComplete, onError, options) ->
|
sort() {
|
||||||
i = 0
|
return this.models.sort(function(a, b) {
|
||||||
|
if (a.name === b.name) {
|
||||||
|
if (!a.version || (a.version.replace(NORMALIZE_VERSION_RGX, NORMALIZE_VERSION_SUB) > b.version.replace(NORMALIZE_VERSION_RGX, NORMALIZE_VERSION_SUB))) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else if (a.name.toLowerCase() > b.name.toLowerCase()) {
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
load(onComplete, onError, options) {
|
||||||
|
let i = 0;
|
||||||
|
|
||||||
next = =>
|
var next = () => {
|
||||||
if i < @models.length
|
if (i < this.models.length) {
|
||||||
@models[i].load(next, fail, options)
|
this.models[i].load(next, fail, options);
|
||||||
else if i is @models.length + CONCURRENCY - 1
|
} else if (i === ((this.models.length + CONCURRENCY) - 1)) {
|
||||||
onComplete()
|
onComplete();
|
||||||
i++
|
}
|
||||||
return
|
i++;
|
||||||
|
};
|
||||||
|
|
||||||
fail = (args...) ->
|
var fail = function(...args) {
|
||||||
if onError
|
if (onError) {
|
||||||
onError(args...)
|
onError(...Array.from(args || []));
|
||||||
onError = null
|
onError = null;
|
||||||
next()
|
}
|
||||||
return
|
next();
|
||||||
|
};
|
||||||
|
|
||||||
next() for [0...CONCURRENCY]
|
for (let j = 0, end = CONCURRENCY, asc = 0 <= end; asc ? j < end : j > end; asc ? j++ : j--) { next(); }
|
||||||
return
|
}
|
||||||
|
|
||||||
clearCache: ->
|
clearCache() {
|
||||||
doc.clearCache() for doc in @models
|
for (var doc of Array.from(this.models)) { doc.clearCache(); }
|
||||||
return
|
}
|
||||||
|
|
||||||
uninstall: (callback) ->
|
uninstall(callback) {
|
||||||
i = 0
|
let i = 0;
|
||||||
next = =>
|
var next = () => {
|
||||||
if i < @models.length
|
if (i < this.models.length) {
|
||||||
@models[i++].uninstall(next, next)
|
this.models[i++].uninstall(next, next);
|
||||||
else
|
} else {
|
||||||
callback()
|
callback();
|
||||||
return
|
}
|
||||||
next()
|
};
|
||||||
return
|
next();
|
||||||
|
}
|
||||||
|
|
||||||
getInstallStatuses: (callback) ->
|
getInstallStatuses(callback) {
|
||||||
app.db.versions @models, (statuses) ->
|
app.db.versions(this.models, function(statuses) {
|
||||||
if statuses
|
if (statuses) {
|
||||||
for key, value of statuses
|
for (var key in statuses) {
|
||||||
statuses[key] = installed: !!value, mtime: value
|
var value = statuses[key];
|
||||||
callback(statuses)
|
statuses[key] = {installed: !!value, mtime: value};
|
||||||
return
|
}
|
||||||
return
|
}
|
||||||
|
callback(statuses);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
checkForUpdates: (callback) ->
|
checkForUpdates(callback) {
|
||||||
@getInstallStatuses (statuses) =>
|
this.getInstallStatuses(statuses => {
|
||||||
i = 0
|
let i = 0;
|
||||||
if statuses
|
if (statuses) {
|
||||||
i += 1 for slug, status of statuses when @findBy('slug', slug).isOutdated(status)
|
for (var slug in statuses) { var status = statuses[slug]; if (this.findBy('slug', slug).isOutdated(status)) { i += 1; } }
|
||||||
callback(i)
|
}
|
||||||
return
|
callback(i);
|
||||||
return
|
});
|
||||||
|
}
|
||||||
|
|
||||||
updateInBackground: ->
|
updateInBackground() {
|
||||||
@getInstallStatuses (statuses) =>
|
this.getInstallStatuses(statuses => {
|
||||||
return unless statuses
|
if (!statuses) { return; }
|
||||||
for slug, status of statuses
|
for (var slug in statuses) {
|
||||||
doc = @findBy 'slug', slug
|
var status = statuses[slug];
|
||||||
doc.install($.noop, $.noop) if doc.isOutdated(status)
|
var doc = this.findBy('slug', slug);
|
||||||
return
|
if (doc.isOutdated(status)) { doc.install($.noop, $.noop); }
|
||||||
return
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,2 +1,11 @@
|
|||||||
class app.collections.Entries extends app.Collection
|
/*
|
||||||
@model: 'Entry'
|
* decaffeinate suggestions:
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.collections.Entries = class Entries extends app.Collection {
|
||||||
|
static initClass() {
|
||||||
|
this.model = 'Entry';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,19 +1,41 @@
|
|||||||
class app.collections.Types extends app.Collection
|
/*
|
||||||
@model: 'Type'
|
* decaffeinate suggestions:
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS104: Avoid inline assignments
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
(function() {
|
||||||
|
let GUIDES_RGX = undefined;
|
||||||
|
let APPENDIX_RGX = undefined;
|
||||||
|
const Cls = (app.collections.Types = class Types extends app.Collection {
|
||||||
|
static initClass() {
|
||||||
|
this.model = 'Type';
|
||||||
|
|
||||||
groups: ->
|
GUIDES_RGX = /(^|\()(guides?|tutorials?|reference|book|getting\ started|manual|examples)($|[\):])/i;
|
||||||
result = []
|
APPENDIX_RGX = /appendix/i;
|
||||||
for type in @models
|
}
|
||||||
(result[@_groupFor(type)] ||= []).push(type)
|
|
||||||
result.filter (e) -> e.length > 0
|
|
||||||
|
|
||||||
GUIDES_RGX = /(^|\()(guides?|tutorials?|reference|book|getting\ started|manual|examples)($|[\):])/i
|
groups() {
|
||||||
APPENDIX_RGX = /appendix/i
|
const result = [];
|
||||||
|
for (var type of Array.from(this.models)) {
|
||||||
|
var name;
|
||||||
|
(result[name = this._groupFor(type)] || (result[name] = [])).push(type);
|
||||||
|
}
|
||||||
|
return result.filter(e => e.length > 0);
|
||||||
|
}
|
||||||
|
|
||||||
_groupFor: (type) ->
|
_groupFor(type) {
|
||||||
if GUIDES_RGX.test(type.name)
|
if (GUIDES_RGX.test(type.name)) {
|
||||||
0
|
return 0;
|
||||||
else if APPENDIX_RGX.test(type.name)
|
} else if (APPENDIX_RGX.test(type.name)) {
|
||||||
2
|
return 2;
|
||||||
else
|
} else {
|
||||||
1
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,85 +1,110 @@
|
|||||||
return unless console?.time and console.groupCollapsed
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
#
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
# App
|
* DS203: Remove `|| {}` from converted for-own loops
|
||||||
#
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* DS208: Avoid top-level this
|
||||||
_init = app.init
|
* DS209: Avoid top-level return
|
||||||
app.init = ->
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
console.time 'Init'
|
*/
|
||||||
_init.call(app)
|
if (!(typeof console !== 'undefined' && console !== null ? console.time : undefined) || !console.groupCollapsed) { return; }
|
||||||
console.timeEnd 'Init'
|
|
||||||
console.time 'Load'
|
//
|
||||||
|
// App
|
||||||
_start = app.start
|
//
|
||||||
app.start = ->
|
|
||||||
console.timeEnd 'Load'
|
const _init = app.init;
|
||||||
console.time 'Start'
|
app.init = function() {
|
||||||
_start.call(app, arguments...)
|
console.time('Init');
|
||||||
console.timeEnd 'Start'
|
_init.call(app);
|
||||||
|
console.timeEnd('Init');
|
||||||
#
|
return console.time('Load');
|
||||||
# Searcher
|
};
|
||||||
#
|
|
||||||
|
const _start = app.start;
|
||||||
_super = app.Searcher
|
app.start = function() {
|
||||||
_proto = app.Searcher.prototype
|
console.timeEnd('Load');
|
||||||
|
console.time('Start');
|
||||||
app.Searcher = ->
|
_start.call(app, ...arguments);
|
||||||
_super.apply @, arguments
|
return console.timeEnd('Start');
|
||||||
|
};
|
||||||
_setup = @setup.bind(@)
|
|
||||||
@setup = ->
|
//
|
||||||
console.groupCollapsed "Search: #{@query}"
|
// Searcher
|
||||||
console.time 'Total'
|
//
|
||||||
_setup()
|
|
||||||
|
const _super = app.Searcher;
|
||||||
_match = @match.bind(@)
|
const _proto = app.Searcher.prototype;
|
||||||
@match = =>
|
|
||||||
console.timeEnd @matcher.name if @matcher
|
app.Searcher = function() {
|
||||||
_match()
|
_super.apply(this, arguments);
|
||||||
|
|
||||||
_setupMatcher = @setupMatcher.bind(@)
|
const _setup = this.setup.bind(this);
|
||||||
@setupMatcher = ->
|
this.setup = function() {
|
||||||
console.time @matcher.name
|
console.groupCollapsed(`Search: ${this.query}`);
|
||||||
_setupMatcher()
|
console.time('Total');
|
||||||
|
return _setup();
|
||||||
_end = @end.bind(@)
|
};
|
||||||
@end = ->
|
|
||||||
console.log "Results: #{@totalResults}"
|
const _match = this.match.bind(this);
|
||||||
console.timeEnd 'Total'
|
this.match = () => {
|
||||||
console.groupEnd()
|
if (this.matcher) { console.timeEnd(this.matcher.name); }
|
||||||
_end()
|
return _match();
|
||||||
|
};
|
||||||
_kill = @kill.bind(@)
|
|
||||||
@kill = ->
|
const _setupMatcher = this.setupMatcher.bind(this);
|
||||||
if @timeout
|
this.setupMatcher = function() {
|
||||||
console.timeEnd @matcher.name if @matcher
|
console.time(this.matcher.name);
|
||||||
console.groupEnd()
|
return _setupMatcher();
|
||||||
console.timeEnd 'Total'
|
};
|
||||||
console.warn 'Killed'
|
|
||||||
_kill()
|
const _end = this.end.bind(this);
|
||||||
|
this.end = function() {
|
||||||
return
|
console.log(`Results: ${this.totalResults}`);
|
||||||
|
console.timeEnd('Total');
|
||||||
$.extend(app.Searcher, _super)
|
console.groupEnd();
|
||||||
_proto.constructor = app.Searcher
|
return _end();
|
||||||
app.Searcher.prototype = _proto
|
};
|
||||||
|
|
||||||
#
|
const _kill = this.kill.bind(this);
|
||||||
# View tree
|
this.kill = function() {
|
||||||
#
|
if (this.timeout) {
|
||||||
|
if (this.matcher) { console.timeEnd(this.matcher.name); }
|
||||||
@viewTree = (view = app.document, level = 0, visited = []) ->
|
console.groupEnd();
|
||||||
return if visited.indexOf(view) >= 0
|
console.timeEnd('Total');
|
||||||
visited.push(view)
|
console.warn('Killed');
|
||||||
|
}
|
||||||
console.log "%c #{Array(level + 1).join(' ')}#{view.constructor.name}: #{!!view.activated}",
|
return _kill();
|
||||||
'color:' + (view.activated and 'green' or 'red')
|
};
|
||||||
|
|
||||||
for own key, value of view when key isnt 'view' and value
|
};
|
||||||
if typeof value is 'object' and value.setupElement
|
|
||||||
@viewTree(value, level + 1, visited)
|
$.extend(app.Searcher, _super);
|
||||||
else if value.constructor.toString().match(/Object\(\)/)
|
_proto.constructor = app.Searcher;
|
||||||
@viewTree(v, level + 1, visited) for own k, v of value when v and typeof v is 'object' and v.setupElement
|
app.Searcher.prototype = _proto;
|
||||||
return
|
|
||||||
|
//
|
||||||
|
// View tree
|
||||||
|
//
|
||||||
|
|
||||||
|
this.viewTree = function(view, level, visited) {
|
||||||
|
if (view == null) { view = app.document; }
|
||||||
|
if (level == null) { level = 0; }
|
||||||
|
if (visited == null) { visited = []; }
|
||||||
|
if (visited.indexOf(view) >= 0) { return; }
|
||||||
|
visited.push(view);
|
||||||
|
|
||||||
|
console.log(`%c ${Array(level + 1).join(' ')}${view.constructor.name}: ${!!view.activated}`,
|
||||||
|
'color:' + ((view.activated && 'green') || 'red'));
|
||||||
|
|
||||||
|
for (var key of Object.keys(view || {})) {
|
||||||
|
var value = view[key];
|
||||||
|
if ((key !== 'view') && value) {
|
||||||
|
if ((typeof value === 'object') && value.setupElement) {
|
||||||
|
this.viewTree(value, level + 1, visited);
|
||||||
|
} else if (value.constructor.toString().match(/Object\(\)/)) {
|
||||||
|
for (var k of Object.keys(value || {})) { var v = value[k]; if (v && (typeof v === 'object') && v.setupElement) { this.viewTree(v, level + 1, visited); } }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,118 +1,154 @@
|
|||||||
MIME_TYPES =
|
/*
|
||||||
json: 'application/json'
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* DS208: Avoid top-level this
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const MIME_TYPES = {
|
||||||
|
json: 'application/json',
|
||||||
html: 'text/html'
|
html: 'text/html'
|
||||||
|
};
|
||||||
|
|
||||||
@ajax = (options) ->
|
this.ajax = function(options) {
|
||||||
applyDefaults(options)
|
applyDefaults(options);
|
||||||
serializeData(options)
|
serializeData(options);
|
||||||
|
|
||||||
xhr = new XMLHttpRequest()
|
const xhr = new XMLHttpRequest();
|
||||||
xhr.open(options.type, options.url, options.async)
|
xhr.open(options.type, options.url, options.async);
|
||||||
|
|
||||||
applyCallbacks(xhr, options)
|
applyCallbacks(xhr, options);
|
||||||
applyHeaders(xhr, options)
|
applyHeaders(xhr, options);
|
||||||
|
|
||||||
xhr.send(options.data)
|
xhr.send(options.data);
|
||||||
|
|
||||||
if options.async
|
if (options.async) {
|
||||||
abort: abort.bind(undefined, xhr)
|
return {abort: abort.bind(undefined, xhr)};
|
||||||
else
|
} else {
|
||||||
parseResponse(xhr, options)
|
return parseResponse(xhr, options);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
ajax.defaults =
|
ajax.defaults = {
|
||||||
async: true
|
async: true,
|
||||||
dataType: 'json'
|
dataType: 'json',
|
||||||
timeout: 30
|
timeout: 30,
|
||||||
type: 'GET'
|
type: 'GET'
|
||||||
# contentType
|
};
|
||||||
# context
|
// contentType
|
||||||
# data
|
// context
|
||||||
# error
|
// data
|
||||||
# headers
|
// error
|
||||||
# progress
|
// headers
|
||||||
# success
|
// progress
|
||||||
# url
|
// success
|
||||||
|
// url
|
||||||
applyDefaults = (options) ->
|
|
||||||
for key of ajax.defaults
|
var applyDefaults = function(options) {
|
||||||
options[key] ?= ajax.defaults[key]
|
for (var key in ajax.defaults) {
|
||||||
return
|
if (options[key] == null) { options[key] = ajax.defaults[key]; }
|
||||||
|
}
|
||||||
serializeData = (options) ->
|
};
|
||||||
return unless options.data
|
|
||||||
|
var serializeData = function(options) {
|
||||||
if options.type is 'GET'
|
if (!options.data) { return; }
|
||||||
options.url += '?' + serializeParams(options.data)
|
|
||||||
options.data = null
|
if (options.type === 'GET') {
|
||||||
else
|
options.url += '?' + serializeParams(options.data);
|
||||||
options.data = serializeParams(options.data)
|
options.data = null;
|
||||||
return
|
} else {
|
||||||
|
options.data = serializeParams(options.data);
|
||||||
serializeParams = (params) ->
|
}
|
||||||
("#{encodeURIComponent key}=#{encodeURIComponent value}" for key, value of params).join '&'
|
};
|
||||||
|
|
||||||
applyCallbacks = (xhr, options) ->
|
var serializeParams = params => ((() => {
|
||||||
return unless options.async
|
const result = [];
|
||||||
|
for (var key in params) {
|
||||||
xhr.timer = setTimeout onTimeout.bind(undefined, xhr, options), options.timeout * 1000
|
var value = params[key];
|
||||||
xhr.onprogress = options.progress if options.progress
|
result.push(`${encodeURIComponent(key)}=${encodeURIComponent(value)}`);
|
||||||
xhr.onreadystatechange = ->
|
}
|
||||||
if xhr.readyState is 4
|
return result;
|
||||||
clearTimeout(xhr.timer)
|
})()).join('&');
|
||||||
onComplete(xhr, options)
|
|
||||||
return
|
var applyCallbacks = function(xhr, options) {
|
||||||
return
|
if (!options.async) { return; }
|
||||||
|
|
||||||
applyHeaders = (xhr, options) ->
|
xhr.timer = setTimeout(onTimeout.bind(undefined, xhr, options), options.timeout * 1000);
|
||||||
options.headers or= {}
|
if (options.progress) { xhr.onprogress = options.progress; }
|
||||||
|
xhr.onreadystatechange = function() {
|
||||||
if options.contentType
|
if (xhr.readyState === 4) {
|
||||||
options.headers['Content-Type'] = options.contentType
|
clearTimeout(xhr.timer);
|
||||||
|
onComplete(xhr, options);
|
||||||
if not options.headers['Content-Type'] and options.data and options.type isnt 'GET'
|
}
|
||||||
options.headers['Content-Type'] = 'application/x-www-form-urlencoded'
|
};
|
||||||
|
};
|
||||||
if options.dataType
|
|
||||||
options.headers['Accept'] = MIME_TYPES[options.dataType] or options.dataType
|
var applyHeaders = function(xhr, options) {
|
||||||
|
if (!options.headers) { options.headers = {}; }
|
||||||
for key, value of options.headers
|
|
||||||
xhr.setRequestHeader(key, value)
|
if (options.contentType) {
|
||||||
return
|
options.headers['Content-Type'] = options.contentType;
|
||||||
|
}
|
||||||
onComplete = (xhr, options) ->
|
|
||||||
if 200 <= xhr.status < 300
|
if (!options.headers['Content-Type'] && options.data && (options.type !== 'GET')) {
|
||||||
if (response = parseResponse(xhr, options))?
|
options.headers['Content-Type'] = 'application/x-www-form-urlencoded';
|
||||||
onSuccess response, xhr, options
|
}
|
||||||
else
|
|
||||||
onError 'invalid', xhr, options
|
if (options.dataType) {
|
||||||
else
|
options.headers['Accept'] = MIME_TYPES[options.dataType] || options.dataType;
|
||||||
onError 'error', xhr, options
|
}
|
||||||
return
|
|
||||||
|
for (var key in options.headers) {
|
||||||
onSuccess = (response, xhr, options) ->
|
var value = options.headers[key];
|
||||||
options.success?.call options.context, response, xhr, options
|
xhr.setRequestHeader(key, value);
|
||||||
return
|
}
|
||||||
|
};
|
||||||
onError = (type, xhr, options) ->
|
|
||||||
options.error?.call options.context, type, xhr, options
|
var onComplete = function(xhr, options) {
|
||||||
return
|
if (200 <= xhr.status && xhr.status < 300) {
|
||||||
|
let response;
|
||||||
onTimeout = (xhr, options) ->
|
if ((response = parseResponse(xhr, options)) != null) {
|
||||||
xhr.abort()
|
onSuccess(response, xhr, options);
|
||||||
onError 'timeout', xhr, options
|
} else {
|
||||||
return
|
onError('invalid', xhr, options);
|
||||||
|
}
|
||||||
abort = (xhr) ->
|
} else {
|
||||||
clearTimeout(xhr.timer)
|
onError('error', xhr, options);
|
||||||
xhr.onreadystatechange = null
|
}
|
||||||
xhr.abort()
|
};
|
||||||
return
|
|
||||||
|
var onSuccess = function(response, xhr, options) {
|
||||||
parseResponse = (xhr, options) ->
|
if (options.success != null) {
|
||||||
if options.dataType is 'json'
|
options.success.call(options.context, response, xhr, options);
|
||||||
parseJSON(xhr.responseText)
|
}
|
||||||
else
|
};
|
||||||
xhr.responseText
|
|
||||||
|
var onError = function(type, xhr, options) {
|
||||||
parseJSON = (json) ->
|
if (options.error != null) {
|
||||||
try JSON.parse(json) catch
|
options.error.call(options.context, type, xhr, options);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var onTimeout = function(xhr, options) {
|
||||||
|
xhr.abort();
|
||||||
|
onError('timeout', xhr, options);
|
||||||
|
};
|
||||||
|
|
||||||
|
var abort = function(xhr) {
|
||||||
|
clearTimeout(xhr.timer);
|
||||||
|
xhr.onreadystatechange = null;
|
||||||
|
xhr.abort();
|
||||||
|
};
|
||||||
|
|
||||||
|
var parseResponse = function(xhr, options) {
|
||||||
|
if (options.dataType === 'json') {
|
||||||
|
return parseJSON(xhr.responseText);
|
||||||
|
} else {
|
||||||
|
return xhr.responseText;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var parseJSON = function(json) {
|
||||||
|
try { return JSON.parse(json); } catch (error) {}
|
||||||
|
};
|
||||||
|
@ -1,42 +1,66 @@
|
|||||||
class @CookiesStore
|
/*
|
||||||
# Intentionally called CookiesStore instead of CookieStore
|
* decaffeinate suggestions:
|
||||||
# Calling it CookieStore causes issues when the Experimental Web Platform features flag is enabled in Chrome
|
* DS101: Remove unnecessary use of Array.from
|
||||||
# Related issue: https://github.com/freeCodeCamp/devdocs/issues/932
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
INT = /^\d+$/
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@onBlocked: ->
|
*/
|
||||||
|
(function() {
|
||||||
get: (key) ->
|
let INT = undefined;
|
||||||
value = Cookies.get(key)
|
const Cls = (this.CookiesStore = class CookiesStore {
|
||||||
value = parseInt(value, 10) if value? and INT.test(value)
|
static initClass() {
|
||||||
value
|
// Intentionally called CookiesStore instead of CookieStore
|
||||||
|
// Calling it CookieStore causes issues when the Experimental Web Platform features flag is enabled in Chrome
|
||||||
set: (key, value) ->
|
// Related issue: https://github.com/freeCodeCamp/devdocs/issues/932
|
||||||
if value == false
|
|
||||||
@del(key)
|
INT = /^\d+$/;
|
||||||
return
|
}
|
||||||
|
|
||||||
value = 1 if value == true
|
static onBlocked() {}
|
||||||
value = parseInt(value, 10) if value and INT.test?(value)
|
|
||||||
Cookies.set(key, '' + value, path: '/', expires: 1e8)
|
get(key) {
|
||||||
@constructor.onBlocked(key, value, @get(key)) if @get(key) != value
|
let value = Cookies.get(key);
|
||||||
return
|
if ((value != null) && INT.test(value)) { value = parseInt(value, 10); }
|
||||||
|
return value;
|
||||||
del: (key) ->
|
}
|
||||||
Cookies.expire(key)
|
|
||||||
return
|
set(key, value) {
|
||||||
|
if (value === false) {
|
||||||
reset: ->
|
this.del(key);
|
||||||
try
|
return;
|
||||||
for cookie in document.cookie.split(/;\s?/)
|
}
|
||||||
Cookies.expire(cookie.split('=')[0])
|
|
||||||
return
|
if (value === true) { value = 1; }
|
||||||
catch
|
if (value && (typeof INT.test === 'function' ? INT.test(value) : undefined)) { value = parseInt(value, 10); }
|
||||||
|
Cookies.set(key, '' + value, {path: '/', expires: 1e8});
|
||||||
dump: ->
|
if (this.get(key) !== value) { this.constructor.onBlocked(key, value, this.get(key)); }
|
||||||
result = {}
|
}
|
||||||
for cookie in document.cookie.split(/;\s?/) when cookie[0] isnt '_'
|
|
||||||
cookie = cookie.split('=')
|
del(key) {
|
||||||
result[cookie[0]] = cookie[1]
|
Cookies.expire(key);
|
||||||
result
|
}
|
||||||
|
|
||||||
|
reset() {
|
||||||
|
try {
|
||||||
|
for (var cookie of Array.from(document.cookie.split(/;\s?/))) {
|
||||||
|
Cookies.expire(cookie.split('=')[0]);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
dump() {
|
||||||
|
const result = {};
|
||||||
|
for (var cookie of Array.from(document.cookie.split(/;\s?/))) {
|
||||||
|
if (cookie[0] !== '_') {
|
||||||
|
cookie = cookie.split('=');
|
||||||
|
result[cookie[0]] = cookie[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,28 +1,51 @@
|
|||||||
@Events =
|
/*
|
||||||
on: (event, callback) ->
|
* decaffeinate suggestions:
|
||||||
if event.indexOf(' ') >= 0
|
* DS101: Remove unnecessary use of Array.from
|
||||||
@on name, callback for name in event.split(' ')
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
else
|
* DS104: Avoid inline assignments
|
||||||
((@_callbacks ?= {})[event] ?= []).push callback
|
* DS207: Consider shorter variations of null checks
|
||||||
@
|
* DS208: Avoid top-level this
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
this.Events = {
|
||||||
|
on(event, callback) {
|
||||||
|
if (event.indexOf(' ') >= 0) {
|
||||||
|
for (var name of Array.from(event.split(' '))) { this.on(name, callback); }
|
||||||
|
} else {
|
||||||
|
let base;
|
||||||
|
(((base = this._callbacks != null ? this._callbacks : (this._callbacks = {})))[event] != null ? base[event] : (base[event] = [])).push(callback);
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
|
||||||
off: (event, callback) ->
|
off(event, callback) {
|
||||||
if event.indexOf(' ') >= 0
|
let callbacks, index;
|
||||||
@off name, callback for name in event.split(' ')
|
if (event.indexOf(' ') >= 0) {
|
||||||
else if (callbacks = @_callbacks?[event]) and (index = callbacks.indexOf callback) >= 0
|
for (var name of Array.from(event.split(' '))) { this.off(name, callback); }
|
||||||
callbacks.splice index, 1
|
} else if ((callbacks = this._callbacks != null ? this._callbacks[event] : undefined) && ((index = callbacks.indexOf(callback)) >= 0)) {
|
||||||
delete @_callbacks[event] unless callbacks.length
|
callbacks.splice(index, 1);
|
||||||
@
|
if (!callbacks.length) { delete this._callbacks[event]; }
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
|
||||||
trigger: (event, args...) ->
|
trigger(event, ...args) {
|
||||||
@eventInProgress = { name: event, args: args }
|
let callbacks;
|
||||||
if callbacks = @_callbacks?[event]
|
this.eventInProgress = { name: event, args };
|
||||||
callback? args... for callback in callbacks.slice(0)
|
if (callbacks = this._callbacks != null ? this._callbacks[event] : undefined) {
|
||||||
@eventInProgress = null
|
for (var callback of Array.from(callbacks.slice(0))) { if (typeof callback === 'function') {
|
||||||
@trigger 'all', event, args... unless event is 'all'
|
callback(...Array.from(args || []));
|
||||||
@
|
} }
|
||||||
|
}
|
||||||
|
this.eventInProgress = null;
|
||||||
|
if (event !== 'all') { this.trigger('all', event, ...Array.from(args)); }
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
|
||||||
removeEvent: (event) ->
|
removeEvent(event) {
|
||||||
if @_callbacks?
|
if (this._callbacks != null) {
|
||||||
delete @_callbacks[name] for name in event.split(' ')
|
for (var name of Array.from(event.split(' '))) { delete this._callbacks[name]; }
|
||||||
@
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,76 +1,89 @@
|
|||||||
defaultUrl = null
|
/*
|
||||||
currentSlug = null
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
imageCache = {}
|
* DS208: Avoid top-level this
|
||||||
urlCache = {}
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
withImage = (url, action) ->
|
let defaultUrl = null;
|
||||||
if imageCache[url]
|
let currentSlug = null;
|
||||||
action(imageCache[url])
|
|
||||||
else
|
const imageCache = {};
|
||||||
img = new Image()
|
const urlCache = {};
|
||||||
img.crossOrigin = 'anonymous'
|
|
||||||
img.src = url
|
const withImage = function(url, action) {
|
||||||
img.onload = () =>
|
if (imageCache[url]) {
|
||||||
imageCache[url] = img
|
return action(imageCache[url]);
|
||||||
action(img)
|
} else {
|
||||||
|
const img = new Image();
|
||||||
@setFaviconForDoc = (doc) ->
|
img.crossOrigin = 'anonymous';
|
||||||
return if currentSlug == doc.slug
|
img.src = url;
|
||||||
|
return img.onload = () => {
|
||||||
favicon = $('link[rel="icon"]')
|
imageCache[url] = img;
|
||||||
|
return action(img);
|
||||||
if defaultUrl == null
|
};
|
||||||
defaultUrl = favicon.href
|
}
|
||||||
|
};
|
||||||
if urlCache[doc.slug]
|
|
||||||
favicon.href = urlCache[doc.slug]
|
this.setFaviconForDoc = function(doc) {
|
||||||
currentSlug = doc.slug
|
if (currentSlug === doc.slug) { return; }
|
||||||
return
|
|
||||||
|
const favicon = $('link[rel="icon"]');
|
||||||
iconEl = $("._icon-#{doc.slug.split('~')[0]}")
|
|
||||||
return if iconEl == null
|
if (defaultUrl === null) {
|
||||||
|
defaultUrl = favicon.href;
|
||||||
styles = window.getComputedStyle(iconEl, ':before')
|
}
|
||||||
|
|
||||||
backgroundPositionX = styles['background-position-x']
|
if (urlCache[doc.slug]) {
|
||||||
backgroundPositionY = styles['background-position-y']
|
favicon.href = urlCache[doc.slug];
|
||||||
return if backgroundPositionX == undefined || backgroundPositionY == undefined
|
currentSlug = doc.slug;
|
||||||
|
return;
|
||||||
bgUrl = app.config.favicon_spritesheet
|
}
|
||||||
sourceSize = 16
|
|
||||||
sourceX = Math.abs(parseInt(backgroundPositionX.slice(0, -2)))
|
const iconEl = $(`._icon-${doc.slug.split('~')[0]}`);
|
||||||
sourceY = Math.abs(parseInt(backgroundPositionY.slice(0, -2)))
|
if (iconEl === null) { return; }
|
||||||
|
|
||||||
withImage(bgUrl, (docImg) ->
|
const styles = window.getComputedStyle(iconEl, ':before');
|
||||||
withImage(defaultUrl, (defaultImg) ->
|
|
||||||
size = defaultImg.width
|
const backgroundPositionX = styles['background-position-x'];
|
||||||
|
const backgroundPositionY = styles['background-position-y'];
|
||||||
canvas = document.createElement('canvas')
|
if ((backgroundPositionX === undefined) || (backgroundPositionY === undefined)) { return; }
|
||||||
ctx = canvas.getContext('2d')
|
|
||||||
|
const bgUrl = app.config.favicon_spritesheet;
|
||||||
canvas.width = size
|
const sourceSize = 16;
|
||||||
canvas.height = size
|
const sourceX = Math.abs(parseInt(backgroundPositionX.slice(0, -2)));
|
||||||
ctx.drawImage(defaultImg, 0, 0)
|
const sourceY = Math.abs(parseInt(backgroundPositionY.slice(0, -2)));
|
||||||
|
|
||||||
docIconPercentage = 65
|
return withImage(bgUrl, docImg => withImage(defaultUrl, function(defaultImg) {
|
||||||
destinationCoords = size / 100 * (100 - docIconPercentage)
|
const size = defaultImg.width;
|
||||||
destinationSize = size / 100 * docIconPercentage
|
|
||||||
|
const canvas = document.createElement('canvas');
|
||||||
ctx.drawImage(docImg, sourceX, sourceY, sourceSize, sourceSize, destinationCoords, destinationCoords, destinationSize, destinationSize)
|
const ctx = canvas.getContext('2d');
|
||||||
|
|
||||||
try
|
canvas.width = size;
|
||||||
urlCache[doc.slug] = canvas.toDataURL()
|
canvas.height = size;
|
||||||
favicon.href = urlCache[doc.slug]
|
ctx.drawImage(defaultImg, 0, 0);
|
||||||
|
|
||||||
currentSlug = doc.slug
|
const docIconPercentage = 65;
|
||||||
catch error
|
const destinationCoords = (size / 100) * (100 - docIconPercentage);
|
||||||
Raven.captureException error, { level: 'info' }
|
const destinationSize = (size / 100) * docIconPercentage;
|
||||||
@resetFavicon()
|
|
||||||
)
|
ctx.drawImage(docImg, sourceX, sourceY, sourceSize, sourceSize, destinationCoords, destinationCoords, destinationSize, destinationSize);
|
||||||
)
|
|
||||||
|
try {
|
||||||
@resetFavicon = () ->
|
urlCache[doc.slug] = canvas.toDataURL();
|
||||||
if defaultUrl != null and currentSlug != null
|
favicon.href = urlCache[doc.slug];
|
||||||
$('link[rel="icon"]').href = defaultUrl
|
|
||||||
currentSlug = null
|
return currentSlug = doc.slug;
|
||||||
|
} catch (error) {
|
||||||
|
Raven.captureException(error, { level: 'info' });
|
||||||
|
return this.resetFavicon();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
};
|
||||||
|
|
||||||
|
this.resetFavicon = function() {
|
||||||
|
if ((defaultUrl !== null) && (currentSlug !== null)) {
|
||||||
|
$('link[rel="icon"]').href = defaultUrl;
|
||||||
|
return currentSlug = null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
###
|
/*
|
||||||
* Copyright 2013-2023 Thibaut Courouble and other contributors
|
* Copyright 2013-2023 Thibaut Courouble and other contributors
|
||||||
*
|
*
|
||||||
* This source code is licensed under the terms of the Mozilla
|
* This source code is licensed under the terms of the Mozilla
|
||||||
* Public License, v. 2.0, a copy of which may be obtained at:
|
* Public License, v. 2.0, a copy of which may be obtained at:
|
||||||
* http://mozilla.org/MPL/2.0/
|
* http://mozilla.org/MPL/2.0/
|
||||||
###
|
*/
|
||||||
|
@ -1,23 +1,33 @@
|
|||||||
class @LocalStorageStore
|
/*
|
||||||
get: (key) ->
|
* decaffeinate suggestions:
|
||||||
try
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
JSON.parse localStorage.getItem(key)
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
catch
|
*/
|
||||||
|
this.LocalStorageStore = class LocalStorageStore {
|
||||||
|
get(key) {
|
||||||
|
try {
|
||||||
|
return JSON.parse(localStorage.getItem(key));
|
||||||
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
set: (key, value) ->
|
set(key, value) {
|
||||||
try
|
try {
|
||||||
localStorage.setItem(key, JSON.stringify(value))
|
localStorage.setItem(key, JSON.stringify(value));
|
||||||
true
|
return true;
|
||||||
catch
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
del: (key) ->
|
del(key) {
|
||||||
try
|
try {
|
||||||
localStorage.removeItem(key)
|
localStorage.removeItem(key);
|
||||||
true
|
return true;
|
||||||
catch
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
reset: ->
|
reset() {
|
||||||
try
|
try {
|
||||||
localStorage.clear()
|
localStorage.clear();
|
||||||
true
|
return true;
|
||||||
catch
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,223 +1,280 @@
|
|||||||
###
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* DS208: Avoid top-level this
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
/*
|
||||||
* Based on github.com/visionmedia/page.js
|
* Based on github.com/visionmedia/page.js
|
||||||
* Licensed under the MIT license
|
* Licensed under the MIT license
|
||||||
* Copyright 2012 TJ Holowaychuk <tj@vision-media.ca>
|
* Copyright 2012 TJ Holowaychuk <tj@vision-media.ca>
|
||||||
###
|
*/
|
||||||
|
|
||||||
running = false
|
let running = false;
|
||||||
currentState = null
|
let currentState = null;
|
||||||
callbacks = []
|
const callbacks = [];
|
||||||
|
|
||||||
@page = (value, fn) ->
|
this.page = function(value, fn) {
|
||||||
if typeof value is 'function'
|
if (typeof value === 'function') {
|
||||||
page '*', value
|
page('*', value);
|
||||||
else if typeof fn is 'function'
|
} else if (typeof fn === 'function') {
|
||||||
route = new Route(value)
|
const route = new Route(value);
|
||||||
callbacks.push route.middleware(fn)
|
callbacks.push(route.middleware(fn));
|
||||||
else if typeof value is 'string'
|
} else if (typeof value === 'string') {
|
||||||
page.show(value, fn)
|
page.show(value, fn);
|
||||||
else
|
} else {
|
||||||
page.start(value)
|
page.start(value);
|
||||||
return
|
}
|
||||||
|
};
|
||||||
page.start = (options = {}) ->
|
|
||||||
unless running
|
page.start = function(options) {
|
||||||
running = true
|
if (options == null) { options = {}; }
|
||||||
addEventListener 'popstate', onpopstate
|
if (!running) {
|
||||||
addEventListener 'click', onclick
|
running = true;
|
||||||
page.replace currentPath(), null, null, true
|
addEventListener('popstate', onpopstate);
|
||||||
return
|
addEventListener('click', onclick);
|
||||||
|
page.replace(currentPath(), null, null, true);
|
||||||
page.stop = ->
|
}
|
||||||
if running
|
};
|
||||||
running = false
|
|
||||||
removeEventListener 'click', onclick
|
page.stop = function() {
|
||||||
removeEventListener 'popstate', onpopstate
|
if (running) {
|
||||||
return
|
running = false;
|
||||||
|
removeEventListener('click', onclick);
|
||||||
page.show = (path, state) ->
|
removeEventListener('popstate', onpopstate);
|
||||||
return if path is currentState?.path
|
}
|
||||||
context = new Context(path, state)
|
};
|
||||||
previousState = currentState
|
|
||||||
currentState = context.state
|
page.show = function(path, state) {
|
||||||
if res = page.dispatch(context)
|
let res;
|
||||||
currentState = previousState
|
if (path === (currentState != null ? currentState.path : undefined)) { return; }
|
||||||
location.assign(res)
|
const context = new Context(path, state);
|
||||||
else
|
const previousState = currentState;
|
||||||
context.pushState()
|
currentState = context.state;
|
||||||
updateCanonicalLink()
|
if (res = page.dispatch(context)) {
|
||||||
track()
|
currentState = previousState;
|
||||||
context
|
location.assign(res);
|
||||||
|
} else {
|
||||||
page.replace = (path, state, skipDispatch, init) ->
|
context.pushState();
|
||||||
context = new Context(path, state or currentState)
|
updateCanonicalLink();
|
||||||
context.init = init
|
track();
|
||||||
currentState = context.state
|
}
|
||||||
result = page.dispatch(context) unless skipDispatch
|
return context;
|
||||||
if result
|
};
|
||||||
context = new Context(result)
|
|
||||||
context.init = init
|
page.replace = function(path, state, skipDispatch, init) {
|
||||||
currentState = context.state
|
let result;
|
||||||
page.dispatch(context)
|
let context = new Context(path, state || currentState);
|
||||||
context.replaceState()
|
context.init = init;
|
||||||
updateCanonicalLink()
|
currentState = context.state;
|
||||||
track() unless skipDispatch
|
if (!skipDispatch) { result = page.dispatch(context); }
|
||||||
context
|
if (result) {
|
||||||
|
context = new Context(result);
|
||||||
page.dispatch = (context) ->
|
context.init = init;
|
||||||
i = 0
|
currentState = context.state;
|
||||||
next = ->
|
page.dispatch(context);
|
||||||
res = fn(context, next) if fn = callbacks[i++]
|
}
|
||||||
return res
|
context.replaceState();
|
||||||
return next()
|
updateCanonicalLink();
|
||||||
|
if (!skipDispatch) { track(); }
|
||||||
page.canGoBack = ->
|
return context;
|
||||||
not Context.isIntialState(currentState)
|
};
|
||||||
|
|
||||||
page.canGoForward = ->
|
page.dispatch = function(context) {
|
||||||
not Context.isLastState(currentState)
|
let i = 0;
|
||||||
|
var next = function() {
|
||||||
currentPath = ->
|
let fn, res;
|
||||||
location.pathname + location.search + location.hash
|
if (fn = callbacks[i++]) { res = fn(context, next); }
|
||||||
|
return res;
|
||||||
class Context
|
};
|
||||||
@initialPath: currentPath()
|
return next();
|
||||||
@sessionId: Date.now()
|
};
|
||||||
@stateId: 0
|
|
||||||
|
page.canGoBack = () => !Context.isIntialState(currentState);
|
||||||
@isIntialState: (state) ->
|
|
||||||
state.id == 0
|
page.canGoForward = () => !Context.isLastState(currentState);
|
||||||
|
|
||||||
@isLastState: (state) ->
|
var currentPath = () => location.pathname + location.search + location.hash;
|
||||||
state.id == @stateId - 1
|
|
||||||
|
class Context {
|
||||||
@isInitialPopState: (state) ->
|
static initClass() {
|
||||||
state.path is @initialPath and @stateId is 1
|
this.initialPath = currentPath();
|
||||||
|
this.sessionId = Date.now();
|
||||||
@isSameSession: (state) ->
|
this.stateId = 0;
|
||||||
state.sessionId is @sessionId
|
}
|
||||||
|
|
||||||
constructor: (@path = '/', @state = {}) ->
|
static isIntialState(state) {
|
||||||
@pathname = @path.replace /(?:\?([^#]*))?(?:#(.*))?$/, (_, query, hash) =>
|
return state.id === 0;
|
||||||
@query = query
|
}
|
||||||
@hash = hash
|
|
||||||
''
|
static isLastState(state) {
|
||||||
|
return state.id === (this.stateId - 1);
|
||||||
@state.id ?= @constructor.stateId++
|
}
|
||||||
@state.sessionId ?= @constructor.sessionId
|
|
||||||
@state.path = @path
|
static isInitialPopState(state) {
|
||||||
|
return (state.path === this.initialPath) && (this.stateId === 1);
|
||||||
pushState: ->
|
}
|
||||||
history.pushState @state, '', @path
|
|
||||||
return
|
static isSameSession(state) {
|
||||||
|
return state.sessionId === this.sessionId;
|
||||||
replaceState: ->
|
}
|
||||||
try history.replaceState @state, '', @path # NS_ERROR_FAILURE in Firefox
|
|
||||||
return
|
constructor(path, state) {
|
||||||
|
if (path == null) { path = '/'; }
|
||||||
class Route
|
this.path = path;
|
||||||
constructor: (@path, options = {}) ->
|
if (state == null) { state = {}; }
|
||||||
@keys = []
|
this.state = state;
|
||||||
@regexp = pathtoRegexp @path, @keys
|
this.pathname = this.path.replace(/(?:\?([^#]*))?(?:#(.*))?$/, (_, query, hash) => {
|
||||||
|
this.query = query;
|
||||||
middleware: (fn) ->
|
this.hash = hash;
|
||||||
(context, next) =>
|
return '';
|
||||||
if @match context.pathname, params = []
|
});
|
||||||
context.params = params
|
|
||||||
return fn(context, next)
|
if (this.state.id == null) { this.state.id = this.constructor.stateId++; }
|
||||||
else
|
if (this.state.sessionId == null) { this.state.sessionId = this.constructor.sessionId; }
|
||||||
return next()
|
this.state.path = this.path;
|
||||||
|
}
|
||||||
match: (path, params) ->
|
|
||||||
return unless matchData = @regexp.exec(path)
|
pushState() {
|
||||||
|
history.pushState(this.state, '', this.path);
|
||||||
for value, i in matchData[1..]
|
}
|
||||||
value = decodeURIComponent value if typeof value is 'string'
|
|
||||||
if key = @keys[i]
|
replaceState() {
|
||||||
params[key.name] = value
|
try { history.replaceState(this.state, '', this.path); } catch (error) {} // NS_ERROR_FAILURE in Firefox
|
||||||
else
|
}
|
||||||
params.push value
|
}
|
||||||
true
|
Context.initClass();
|
||||||
|
|
||||||
pathtoRegexp = (path, keys) ->
|
class Route {
|
||||||
return path if path instanceof RegExp
|
constructor(path, options) {
|
||||||
|
this.path = path;
|
||||||
path = "(#{path.join '|'})" if path instanceof Array
|
if (options == null) { options = {}; }
|
||||||
|
this.keys = [];
|
||||||
|
this.regexp = pathtoRegexp(this.path, this.keys);
|
||||||
|
}
|
||||||
|
|
||||||
|
middleware(fn) {
|
||||||
|
return (context, next) => {
|
||||||
|
let params;
|
||||||
|
if (this.match(context.pathname, (params = []))) {
|
||||||
|
context.params = params;
|
||||||
|
return fn(context, next);
|
||||||
|
} else {
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
match(path, params) {
|
||||||
|
let matchData;
|
||||||
|
if (!(matchData = this.regexp.exec(path))) { return; }
|
||||||
|
|
||||||
|
const iterable = matchData.slice(1);
|
||||||
|
for (let i = 0; i < iterable.length; i++) {
|
||||||
|
var key;
|
||||||
|
var value = iterable[i];
|
||||||
|
if (typeof value === 'string') { value = decodeURIComponent(value); }
|
||||||
|
if ((key = this.keys[i])) {
|
||||||
|
params[key.name] = value;
|
||||||
|
} else {
|
||||||
|
params.push(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var pathtoRegexp = function(path, keys) {
|
||||||
|
if (path instanceof RegExp) { return path; }
|
||||||
|
|
||||||
|
if (path instanceof Array) { path = `(${path.join('|')})`; }
|
||||||
path = path
|
path = path
|
||||||
.replace /\/\(/g, '(?:/'
|
.replace(/\/\(/g, '(?:/')
|
||||||
.replace /(\/)?(\.)?:(\w+)(?:(\(.*?\)))?(\?)?/g, (_, slash = '', format = '', key, capture, optional) ->
|
.replace(/(\/)?(\.)?:(\w+)(?:(\(.*?\)))?(\?)?/g, function(_, slash, format, key, capture, optional) {
|
||||||
keys.push name: key, optional: !!optional
|
if (slash == null) { slash = ''; }
|
||||||
str = if optional then '' else slash
|
if (format == null) { format = ''; }
|
||||||
str += '(?:'
|
keys.push({name: key, optional: !!optional});
|
||||||
str += slash if optional
|
let str = optional ? '' : slash;
|
||||||
str += format
|
str += '(?:';
|
||||||
str += capture or if format then '([^/.]+?)' else '([^/]+?)'
|
if (optional) { str += slash; }
|
||||||
str += ')'
|
str += format;
|
||||||
str += optional if optional
|
str += capture || (format ? '([^/.]+?)' : '([^/]+?)');
|
||||||
str
|
str += ')';
|
||||||
.replace /([\/.])/g, '\\$1'
|
if (optional) { str += optional; }
|
||||||
.replace /\*/g, '(.*)'
|
return str;
|
||||||
|
}).replace(/([\/.])/g, '\\$1')
|
||||||
new RegExp "^#{path}$"
|
.replace(/\*/g, '(.*)');
|
||||||
|
|
||||||
onpopstate = (event) ->
|
return new RegExp(`^${path}$`);
|
||||||
return if not event.state or Context.isInitialPopState(event.state)
|
};
|
||||||
|
|
||||||
if Context.isSameSession(event.state)
|
var onpopstate = function(event) {
|
||||||
page.replace(event.state.path, event.state)
|
if (!event.state || Context.isInitialPopState(event.state)) { return; }
|
||||||
else
|
|
||||||
location.reload()
|
if (Context.isSameSession(event.state)) {
|
||||||
return
|
page.replace(event.state.path, event.state);
|
||||||
|
} else {
|
||||||
onclick = (event) ->
|
location.reload();
|
||||||
try
|
}
|
||||||
return if event.which isnt 1 or event.metaKey or event.ctrlKey or event.shiftKey or event.defaultPrevented
|
};
|
||||||
catch
|
|
||||||
return
|
var onclick = function(event) {
|
||||||
|
try {
|
||||||
link = $.eventTarget(event)
|
if ((event.which !== 1) || event.metaKey || event.ctrlKey || event.shiftKey || event.defaultPrevented) { return; }
|
||||||
link = link.parentNode while link and link.tagName isnt 'A'
|
} catch (error) {
|
||||||
|
return;
|
||||||
if link and not link.target and isSameOrigin(link.href)
|
}
|
||||||
event.preventDefault()
|
|
||||||
path = link.pathname + link.search + link.hash
|
let link = $.eventTarget(event);
|
||||||
path = path.replace /^\/\/+/, '/' # IE11 bug
|
while (link && (link.tagName !== 'A')) { link = link.parentNode; }
|
||||||
page.show(path)
|
|
||||||
return
|
if (link && !link.target && isSameOrigin(link.href)) {
|
||||||
|
event.preventDefault();
|
||||||
isSameOrigin = (url) ->
|
let path = link.pathname + link.search + link.hash;
|
||||||
url.indexOf("#{location.protocol}//#{location.hostname}") is 0
|
path = path.replace(/^\/\/+/, '/'); // IE11 bug
|
||||||
|
page.show(path);
|
||||||
updateCanonicalLink = ->
|
}
|
||||||
@canonicalLink ||= document.head.querySelector('link[rel="canonical"]')
|
};
|
||||||
@canonicalLink.setAttribute('href', "https://#{location.host}#{location.pathname}")
|
|
||||||
|
var isSameOrigin = url => url.indexOf(`${location.protocol}//${location.hostname}`) === 0;
|
||||||
trackers = []
|
|
||||||
|
var updateCanonicalLink = function() {
|
||||||
page.track = (fn) ->
|
if (!this.canonicalLink) { this.canonicalLink = document.head.querySelector('link[rel="canonical"]'); }
|
||||||
trackers.push(fn)
|
return this.canonicalLink.setAttribute('href', `https://${location.host}${location.pathname}`);
|
||||||
return
|
};
|
||||||
|
|
||||||
track = ->
|
const trackers = [];
|
||||||
return unless app.config.env == 'production'
|
|
||||||
return if navigator.doNotTrack == '1'
|
page.track = function(fn) {
|
||||||
return if navigator.globalPrivacyControl
|
trackers.push(fn);
|
||||||
|
};
|
||||||
consentGiven = Cookies.get('analyticsConsent')
|
|
||||||
consentAsked = Cookies.get('analyticsConsentAsked')
|
var track = function() {
|
||||||
|
if (app.config.env !== 'production') { return; }
|
||||||
if consentGiven == '1'
|
if (navigator.doNotTrack === '1') { return; }
|
||||||
tracker.call() for tracker in trackers
|
if (navigator.globalPrivacyControl) { return; }
|
||||||
else if consentGiven == undefined and consentAsked == undefined
|
|
||||||
# Only ask for consent once per browser session
|
const consentGiven = Cookies.get('analyticsConsent');
|
||||||
Cookies.set('analyticsConsentAsked', '1')
|
const consentAsked = Cookies.get('analyticsConsentAsked');
|
||||||
|
|
||||||
new app.views.Notif 'AnalyticsConsent', autoHide: null
|
if (consentGiven === '1') {
|
||||||
return
|
for (var tracker of Array.from(trackers)) { tracker.call(); }
|
||||||
|
} else if ((consentGiven === undefined) && (consentAsked === undefined)) {
|
||||||
@resetAnalytics = ->
|
// Only ask for consent once per browser session
|
||||||
for cookie in document.cookie.split(/;\s?/)
|
Cookies.set('analyticsConsentAsked', '1');
|
||||||
name = cookie.split('=')[0]
|
|
||||||
if name[0] == '_' && name[1] != '_'
|
new app.views.Notif('AnalyticsConsent', {autoHide: null});
|
||||||
Cookies.expire(name)
|
}
|
||||||
return
|
};
|
||||||
|
|
||||||
|
this.resetAnalytics = function() {
|
||||||
|
for (var cookie of Array.from(document.cookie.split(/;\s?/))) {
|
||||||
|
var name = cookie.split('=')[0];
|
||||||
|
if ((name[0] === '_') && (name[1] !== '_')) {
|
||||||
|
Cookies.expire(name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,399 +1,491 @@
|
|||||||
#
|
/*
|
||||||
# Traversing
|
* decaffeinate suggestions:
|
||||||
#
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
@$ = (selector, el = document) ->
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
try el.querySelector(selector) catch
|
* DS104: Avoid inline assignments
|
||||||
|
* DS204: Change includes calls to have a more natural evaluation order
|
||||||
@$$ = (selector, el = document) ->
|
* DS207: Consider shorter variations of null checks
|
||||||
try el.querySelectorAll(selector) catch
|
* DS208: Avoid top-level this
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
$.id = (id) ->
|
*/
|
||||||
document.getElementById(id)
|
//
|
||||||
|
// Traversing
|
||||||
$.hasChild = (parent, el) ->
|
//
|
||||||
return unless parent
|
|
||||||
while el
|
let smoothDistance, smoothDuration, smoothEnd, smoothStart;
|
||||||
return true if el is parent
|
this.$ = function(selector, el) {
|
||||||
return if el is document.body
|
if (el == null) { el = document; }
|
||||||
el = el.parentNode
|
try { return el.querySelector(selector); } catch (error) {}
|
||||||
|
};
|
||||||
$.closestLink = (el, parent = document.body) ->
|
|
||||||
while el
|
this.$$ = function(selector, el) {
|
||||||
return el if el.tagName is 'A'
|
if (el == null) { el = document; }
|
||||||
return if el is parent
|
try { return el.querySelectorAll(selector); } catch (error) {}
|
||||||
el = el.parentNode
|
};
|
||||||
|
|
||||||
#
|
$.id = id => document.getElementById(id);
|
||||||
# Events
|
|
||||||
#
|
$.hasChild = function(parent, el) {
|
||||||
|
if (!parent) { return; }
|
||||||
$.on = (el, event, callback, useCapture = false) ->
|
while (el) {
|
||||||
if event.indexOf(' ') >= 0
|
if (el === parent) { return true; }
|
||||||
$.on el, name, callback for name in event.split(' ')
|
if (el === document.body) { return; }
|
||||||
else
|
el = el.parentNode;
|
||||||
el.addEventListener(event, callback, useCapture)
|
}
|
||||||
return
|
};
|
||||||
|
|
||||||
$.off = (el, event, callback, useCapture = false) ->
|
$.closestLink = function(el, parent) {
|
||||||
if event.indexOf(' ') >= 0
|
if (parent == null) { parent = document.body; }
|
||||||
$.off el, name, callback for name in event.split(' ')
|
while (el) {
|
||||||
else
|
if (el.tagName === 'A') { return el; }
|
||||||
el.removeEventListener(event, callback, useCapture)
|
if (el === parent) { return; }
|
||||||
return
|
el = el.parentNode;
|
||||||
|
}
|
||||||
$.trigger = (el, type, canBubble = true, cancelable = true) ->
|
};
|
||||||
event = document.createEvent 'Event'
|
|
||||||
event.initEvent(type, canBubble, cancelable)
|
//
|
||||||
el.dispatchEvent(event)
|
// Events
|
||||||
return
|
//
|
||||||
|
|
||||||
$.click = (el) ->
|
$.on = function(el, event, callback, useCapture) {
|
||||||
event = document.createEvent 'MouseEvent'
|
if (useCapture == null) { useCapture = false; }
|
||||||
event.initMouseEvent 'click', true, true, window, null, 0, 0, 0, 0, false, false, false, false, 0, null
|
if (event.indexOf(' ') >= 0) {
|
||||||
el.dispatchEvent(event)
|
for (var name of Array.from(event.split(' '))) { $.on(el, name, callback); }
|
||||||
return
|
} else {
|
||||||
|
el.addEventListener(event, callback, useCapture);
|
||||||
$.stopEvent = (event) ->
|
}
|
||||||
event.preventDefault()
|
};
|
||||||
event.stopPropagation()
|
|
||||||
event.stopImmediatePropagation()
|
$.off = function(el, event, callback, useCapture) {
|
||||||
return
|
if (useCapture == null) { useCapture = false; }
|
||||||
|
if (event.indexOf(' ') >= 0) {
|
||||||
$.eventTarget = (event) ->
|
for (var name of Array.from(event.split(' '))) { $.off(el, name, callback); }
|
||||||
event.target.correspondingUseElement || event.target
|
} else {
|
||||||
|
el.removeEventListener(event, callback, useCapture);
|
||||||
#
|
}
|
||||||
# Manipulation
|
};
|
||||||
#
|
|
||||||
|
$.trigger = function(el, type, canBubble, cancelable) {
|
||||||
buildFragment = (value) ->
|
if (canBubble == null) { canBubble = true; }
|
||||||
fragment = document.createDocumentFragment()
|
if (cancelable == null) { cancelable = true; }
|
||||||
|
const event = document.createEvent('Event');
|
||||||
if $.isCollection(value)
|
event.initEvent(type, canBubble, cancelable);
|
||||||
fragment.appendChild(child) for child in $.makeArray(value)
|
el.dispatchEvent(event);
|
||||||
else
|
};
|
||||||
fragment.innerHTML = value
|
|
||||||
|
$.click = function(el) {
|
||||||
fragment
|
const event = document.createEvent('MouseEvent');
|
||||||
|
event.initMouseEvent('click', true, true, window, null, 0, 0, 0, 0, false, false, false, false, 0, null);
|
||||||
$.append = (el, value) ->
|
el.dispatchEvent(event);
|
||||||
if typeof value is 'string'
|
};
|
||||||
el.insertAdjacentHTML 'beforeend', value
|
|
||||||
else
|
$.stopEvent = function(event) {
|
||||||
value = buildFragment(value) if $.isCollection(value)
|
event.preventDefault();
|
||||||
el.appendChild(value)
|
event.stopPropagation();
|
||||||
return
|
event.stopImmediatePropagation();
|
||||||
|
};
|
||||||
$.prepend = (el, value) ->
|
|
||||||
if not el.firstChild
|
$.eventTarget = event => event.target.correspondingUseElement || event.target;
|
||||||
$.append(value)
|
|
||||||
else if typeof value is 'string'
|
//
|
||||||
el.insertAdjacentHTML 'afterbegin', value
|
// Manipulation
|
||||||
else
|
//
|
||||||
value = buildFragment(value) if $.isCollection(value)
|
|
||||||
el.insertBefore(value, el.firstChild)
|
const buildFragment = function(value) {
|
||||||
return
|
const fragment = document.createDocumentFragment();
|
||||||
|
|
||||||
$.before = (el, value) ->
|
if ($.isCollection(value)) {
|
||||||
if typeof value is 'string' or $.isCollection(value)
|
for (var child of Array.from($.makeArray(value))) { fragment.appendChild(child); }
|
||||||
value = buildFragment(value)
|
} else {
|
||||||
|
fragment.innerHTML = value;
|
||||||
el.parentNode.insertBefore(value, el)
|
}
|
||||||
return
|
|
||||||
|
return fragment;
|
||||||
$.after = (el, value) ->
|
};
|
||||||
if typeof value is 'string' or $.isCollection(value)
|
|
||||||
value = buildFragment(value)
|
$.append = function(el, value) {
|
||||||
|
if (typeof value === 'string') {
|
||||||
if el.nextSibling
|
el.insertAdjacentHTML('beforeend', value);
|
||||||
el.parentNode.insertBefore(value, el.nextSibling)
|
} else {
|
||||||
else
|
if ($.isCollection(value)) { value = buildFragment(value); }
|
||||||
el.parentNode.appendChild(value)
|
el.appendChild(value);
|
||||||
return
|
}
|
||||||
|
};
|
||||||
$.remove = (value) ->
|
|
||||||
if $.isCollection(value)
|
$.prepend = function(el, value) {
|
||||||
el.parentNode?.removeChild(el) for el in $.makeArray(value)
|
if (!el.firstChild) {
|
||||||
else
|
$.append(value);
|
||||||
value.parentNode?.removeChild(value)
|
} else if (typeof value === 'string') {
|
||||||
return
|
el.insertAdjacentHTML('afterbegin', value);
|
||||||
|
} else {
|
||||||
$.empty = (el) ->
|
if ($.isCollection(value)) { value = buildFragment(value); }
|
||||||
el.removeChild(el.firstChild) while el.firstChild
|
el.insertBefore(value, el.firstChild);
|
||||||
return
|
}
|
||||||
|
};
|
||||||
# Calls the function while the element is off the DOM to avoid triggering
|
|
||||||
# unnecessary reflows and repaints.
|
$.before = function(el, value) {
|
||||||
$.batchUpdate = (el, fn) ->
|
if ((typeof value === 'string') || $.isCollection(value)) {
|
||||||
parent = el.parentNode
|
value = buildFragment(value);
|
||||||
sibling = el.nextSibling
|
}
|
||||||
parent.removeChild(el)
|
|
||||||
|
el.parentNode.insertBefore(value, el);
|
||||||
fn(el)
|
};
|
||||||
|
|
||||||
if (sibling)
|
$.after = function(el, value) {
|
||||||
parent.insertBefore(el, sibling)
|
if ((typeof value === 'string') || $.isCollection(value)) {
|
||||||
else
|
value = buildFragment(value);
|
||||||
parent.appendChild(el)
|
}
|
||||||
return
|
|
||||||
|
if (el.nextSibling) {
|
||||||
#
|
el.parentNode.insertBefore(value, el.nextSibling);
|
||||||
# Offset
|
} else {
|
||||||
#
|
el.parentNode.appendChild(value);
|
||||||
|
}
|
||||||
$.rect = (el) ->
|
};
|
||||||
el.getBoundingClientRect()
|
|
||||||
|
$.remove = function(value) {
|
||||||
$.offset = (el, container = document.body) ->
|
if ($.isCollection(value)) {
|
||||||
top = 0
|
for (var el of Array.from($.makeArray(value))) { if (el.parentNode != null) {
|
||||||
left = 0
|
el.parentNode.removeChild(el);
|
||||||
|
} }
|
||||||
while el and el isnt container
|
} else {
|
||||||
top += el.offsetTop
|
if (value.parentNode != null) {
|
||||||
left += el.offsetLeft
|
value.parentNode.removeChild(value);
|
||||||
el = el.offsetParent
|
}
|
||||||
|
}
|
||||||
top: top
|
};
|
||||||
left: left
|
|
||||||
|
$.empty = function(el) {
|
||||||
$.scrollParent = (el) ->
|
while (el.firstChild) { el.removeChild(el.firstChild); }
|
||||||
while (el = el.parentNode) and el.nodeType is 1
|
};
|
||||||
break if el.scrollTop > 0
|
|
||||||
break if getComputedStyle(el)?.overflowY in ['auto', 'scroll']
|
// Calls the function while the element is off the DOM to avoid triggering
|
||||||
el
|
// unnecessary reflows and repaints.
|
||||||
|
$.batchUpdate = function(el, fn) {
|
||||||
$.scrollTo = (el, parent, position = 'center', options = {}) ->
|
const parent = el.parentNode;
|
||||||
return unless el
|
const sibling = el.nextSibling;
|
||||||
|
parent.removeChild(el);
|
||||||
parent ?= $.scrollParent(el)
|
|
||||||
return unless parent
|
fn(el);
|
||||||
|
|
||||||
parentHeight = parent.clientHeight
|
if (sibling) {
|
||||||
parentScrollHeight = parent.scrollHeight
|
parent.insertBefore(el, sibling);
|
||||||
return unless parentScrollHeight > parentHeight
|
} else {
|
||||||
|
parent.appendChild(el);
|
||||||
top = $.offset(el, parent).top
|
}
|
||||||
offsetTop = parent.firstElementChild.offsetTop
|
};
|
||||||
|
|
||||||
switch position
|
//
|
||||||
when 'top'
|
// Offset
|
||||||
parent.scrollTop = top - offsetTop - (if options.margin? then options.margin else 0)
|
//
|
||||||
when 'center'
|
|
||||||
parent.scrollTop = top - Math.round(parentHeight / 2 - el.offsetHeight / 2)
|
$.rect = el => el.getBoundingClientRect();
|
||||||
when 'continuous'
|
|
||||||
scrollTop = parent.scrollTop
|
$.offset = function(el, container) {
|
||||||
height = el.offsetHeight
|
if (container == null) { container = document.body; }
|
||||||
|
let top = 0;
|
||||||
lastElementOffset = parent.lastElementChild.offsetTop + parent.lastElementChild.offsetHeight
|
let left = 0;
|
||||||
offsetBottom = if lastElementOffset > 0 then parentScrollHeight - lastElementOffset else 0
|
|
||||||
|
while (el && (el !== container)) {
|
||||||
# If the target element is above the visible portion of its scrollable
|
top += el.offsetTop;
|
||||||
# ancestor, move it near the top with a gap = options.topGap * target's height.
|
left += el.offsetLeft;
|
||||||
if top - offsetTop <= scrollTop + height * (options.topGap or 1)
|
el = el.offsetParent;
|
||||||
parent.scrollTop = top - offsetTop - height * (options.topGap or 1)
|
}
|
||||||
# If the target element is below the visible portion of its scrollable
|
|
||||||
# ancestor, move it near the bottom with a gap = options.bottomGap * target's height.
|
return {
|
||||||
else if top + offsetBottom >= scrollTop + parentHeight - height * ((options.bottomGap or 1) + 1)
|
top,
|
||||||
parent.scrollTop = top + offsetBottom - parentHeight + height * ((options.bottomGap or 1) + 1)
|
left
|
||||||
return
|
};
|
||||||
|
};
|
||||||
$.scrollToWithImageLock = (el, parent, args...) ->
|
|
||||||
parent ?= $.scrollParent(el)
|
$.scrollParent = function(el) {
|
||||||
return unless parent
|
while ((el = el.parentNode) && (el.nodeType === 1)) {
|
||||||
|
var needle;
|
||||||
$.scrollTo el, parent, args...
|
if (el.scrollTop > 0) { break; }
|
||||||
|
if ((needle = __guard__(getComputedStyle(el), x => x.overflowY), ['auto', 'scroll'].includes(needle))) { break; }
|
||||||
# Lock the scroll position on the target element for up to 3 seconds while
|
}
|
||||||
# nearby images are loaded and rendered.
|
return el;
|
||||||
for image in parent.getElementsByTagName('img') when not image.complete
|
};
|
||||||
do ->
|
|
||||||
onLoad = (event) ->
|
$.scrollTo = function(el, parent, position, options) {
|
||||||
clearTimeout(timeout)
|
if (position == null) { position = 'center'; }
|
||||||
unbind(event.target)
|
if (options == null) { options = {}; }
|
||||||
$.scrollTo el, parent, args...
|
if (!el) { return; }
|
||||||
|
|
||||||
unbind = (target) ->
|
if (parent == null) { parent = $.scrollParent(el); }
|
||||||
$.off target, 'load', onLoad
|
if (!parent) { return; }
|
||||||
|
|
||||||
$.on image, 'load', onLoad
|
const parentHeight = parent.clientHeight;
|
||||||
timeout = setTimeout unbind.bind(null, image), 3000
|
const parentScrollHeight = parent.scrollHeight;
|
||||||
return
|
if (!(parentScrollHeight > parentHeight)) { return; }
|
||||||
|
|
||||||
# Calls the function while locking the element's position relative to the window.
|
const {
|
||||||
$.lockScroll = (el, fn) ->
|
top
|
||||||
if parent = $.scrollParent(el)
|
} = $.offset(el, parent);
|
||||||
top = $.rect(el).top
|
const {
|
||||||
top -= $.rect(parent).top unless parent in [document.body, document.documentElement]
|
offsetTop
|
||||||
fn()
|
} = parent.firstElementChild;
|
||||||
parent.scrollTop = $.offset(el, parent).top - top
|
|
||||||
else
|
switch (position) {
|
||||||
fn()
|
case 'top':
|
||||||
return
|
parent.scrollTop = top - offsetTop - ((options.margin != null) ? options.margin : 0);
|
||||||
|
break;
|
||||||
smoothScroll = smoothStart = smoothEnd = smoothDistance = smoothDuration = null
|
case 'center':
|
||||||
|
parent.scrollTop = top - Math.round((parentHeight / 2) - (el.offsetHeight / 2));
|
||||||
$.smoothScroll = (el, end) ->
|
break;
|
||||||
unless window.requestAnimationFrame
|
case 'continuous':
|
||||||
el.scrollTop = end
|
var {
|
||||||
return
|
scrollTop
|
||||||
|
} = parent;
|
||||||
smoothEnd = end
|
var height = el.offsetHeight;
|
||||||
|
|
||||||
if smoothScroll
|
var lastElementOffset = parent.lastElementChild.offsetTop + parent.lastElementChild.offsetHeight;
|
||||||
newDistance = smoothEnd - smoothStart
|
var offsetBottom = lastElementOffset > 0 ? parentScrollHeight - lastElementOffset : 0;
|
||||||
smoothDuration += Math.min 300, Math.abs(smoothDistance - newDistance)
|
|
||||||
smoothDistance = newDistance
|
// If the target element is above the visible portion of its scrollable
|
||||||
return
|
// ancestor, move it near the top with a gap = options.topGap * target's height.
|
||||||
|
if ((top - offsetTop) <= (scrollTop + (height * (options.topGap || 1)))) {
|
||||||
smoothStart = el.scrollTop
|
parent.scrollTop = top - offsetTop - (height * (options.topGap || 1));
|
||||||
smoothDistance = smoothEnd - smoothStart
|
// If the target element is below the visible portion of its scrollable
|
||||||
smoothDuration = Math.min 300, Math.abs(smoothDistance)
|
// ancestor, move it near the bottom with a gap = options.bottomGap * target's height.
|
||||||
startTime = Date.now()
|
} else if ((top + offsetBottom) >= ((scrollTop + parentHeight) - (height * ((options.bottomGap || 1) + 1)))) {
|
||||||
|
parent.scrollTop = ((top + offsetBottom) - parentHeight) + (height * ((options.bottomGap || 1) + 1));
|
||||||
smoothScroll = ->
|
}
|
||||||
p = Math.min 1, (Date.now() - startTime) / smoothDuration
|
break;
|
||||||
y = Math.max 0, Math.floor(smoothStart + smoothDistance * (if p < 0.5 then 2 * p * p else p * (4 - p * 2) - 1))
|
}
|
||||||
el.scrollTop = y
|
};
|
||||||
if p is 1
|
|
||||||
smoothScroll = null
|
$.scrollToWithImageLock = function(el, parent, ...args) {
|
||||||
else
|
if (parent == null) { parent = $.scrollParent(el); }
|
||||||
requestAnimationFrame(smoothScroll)
|
if (!parent) { return; }
|
||||||
requestAnimationFrame(smoothScroll)
|
|
||||||
|
$.scrollTo(el, parent, ...Array.from(args));
|
||||||
#
|
|
||||||
# Utilities
|
// Lock the scroll position on the target element for up to 3 seconds while
|
||||||
#
|
// nearby images are loaded and rendered.
|
||||||
|
for (var image of Array.from(parent.getElementsByTagName('img'))) {
|
||||||
$.extend = (target, objects...) ->
|
if (!image.complete) {
|
||||||
for object in objects when object
|
(function() {
|
||||||
for key, value of object
|
let timeout;
|
||||||
target[key] = value
|
const onLoad = function(event) {
|
||||||
target
|
clearTimeout(timeout);
|
||||||
|
unbind(event.target);
|
||||||
$.makeArray = (object) ->
|
return $.scrollTo(el, parent, ...Array.from(args));
|
||||||
if Array.isArray(object)
|
};
|
||||||
object
|
|
||||||
else
|
var unbind = target => $.off(target, 'load', onLoad);
|
||||||
Array::slice.apply(object)
|
|
||||||
|
$.on(image, 'load', onLoad);
|
||||||
$.arrayDelete = (array, object) ->
|
return timeout = setTimeout(unbind.bind(null, image), 3000);
|
||||||
index = array.indexOf(object)
|
})();
|
||||||
if index >= 0
|
}
|
||||||
array.splice(index, 1)
|
}
|
||||||
true
|
};
|
||||||
else
|
|
||||||
false
|
// Calls the function while locking the element's position relative to the window.
|
||||||
|
$.lockScroll = function(el, fn) {
|
||||||
# Returns true if the object is an array or a collection of DOM elements.
|
let parent;
|
||||||
$.isCollection = (object) ->
|
if (parent = $.scrollParent(el)) {
|
||||||
Array.isArray(object) or typeof object?.item is 'function'
|
let {
|
||||||
|
top
|
||||||
ESCAPE_HTML_MAP =
|
} = $.rect(el);
|
||||||
'&': '&'
|
if (![document.body, document.documentElement].includes(parent)) { top -= $.rect(parent).top; }
|
||||||
'<': '<'
|
fn();
|
||||||
'>': '>'
|
parent.scrollTop = $.offset(el, parent).top - top;
|
||||||
'"': '"'
|
} else {
|
||||||
"'": '''
|
fn();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let smoothScroll = (smoothStart = (smoothEnd = (smoothDistance = (smoothDuration = null))));
|
||||||
|
|
||||||
|
$.smoothScroll = function(el, end) {
|
||||||
|
if (!window.requestAnimationFrame) {
|
||||||
|
el.scrollTop = end;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
smoothEnd = end;
|
||||||
|
|
||||||
|
if (smoothScroll) {
|
||||||
|
const newDistance = smoothEnd - smoothStart;
|
||||||
|
smoothDuration += Math.min(300, Math.abs(smoothDistance - newDistance));
|
||||||
|
smoothDistance = newDistance;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
smoothStart = el.scrollTop;
|
||||||
|
smoothDistance = smoothEnd - smoothStart;
|
||||||
|
smoothDuration = Math.min(300, Math.abs(smoothDistance));
|
||||||
|
const startTime = Date.now();
|
||||||
|
|
||||||
|
smoothScroll = function() {
|
||||||
|
const p = Math.min(1, (Date.now() - startTime) / smoothDuration);
|
||||||
|
const y = Math.max(0, Math.floor(smoothStart + (smoothDistance * (p < 0.5 ? 2 * p * p : (p * (4 - (p * 2))) - 1))));
|
||||||
|
el.scrollTop = y;
|
||||||
|
if (p === 1) {
|
||||||
|
return smoothScroll = null;
|
||||||
|
} else {
|
||||||
|
return requestAnimationFrame(smoothScroll);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return requestAnimationFrame(smoothScroll);
|
||||||
|
};
|
||||||
|
|
||||||
|
//
|
||||||
|
// Utilities
|
||||||
|
//
|
||||||
|
|
||||||
|
$.extend = function(target, ...objects) {
|
||||||
|
for (var object of Array.from(objects)) {
|
||||||
|
if (object) {
|
||||||
|
for (var key in object) {
|
||||||
|
var value = object[key];
|
||||||
|
target[key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return target;
|
||||||
|
};
|
||||||
|
|
||||||
|
$.makeArray = function(object) {
|
||||||
|
if (Array.isArray(object)) {
|
||||||
|
return object;
|
||||||
|
} else {
|
||||||
|
return Array.prototype.slice.apply(object);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$.arrayDelete = function(array, object) {
|
||||||
|
const index = array.indexOf(object);
|
||||||
|
if (index >= 0) {
|
||||||
|
array.splice(index, 1);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Returns true if the object is an array or a collection of DOM elements.
|
||||||
|
$.isCollection = object => Array.isArray(object) || (typeof (object != null ? object.item : undefined) === 'function');
|
||||||
|
|
||||||
|
const ESCAPE_HTML_MAP = {
|
||||||
|
'&': '&',
|
||||||
|
'<': '<',
|
||||||
|
'>': '>',
|
||||||
|
'"': '"',
|
||||||
|
"'": ''',
|
||||||
'/': '/'
|
'/': '/'
|
||||||
|
};
|
||||||
ESCAPE_HTML_REGEXP = /[&<>"'\/]/g
|
|
||||||
|
const ESCAPE_HTML_REGEXP = /[&<>"'\/]/g;
|
||||||
$.escape = (string) ->
|
|
||||||
string.replace ESCAPE_HTML_REGEXP, (match) -> ESCAPE_HTML_MAP[match]
|
$.escape = string => string.replace(ESCAPE_HTML_REGEXP, match => ESCAPE_HTML_MAP[match]);
|
||||||
|
|
||||||
ESCAPE_REGEXP = /([.*+?^=!:${}()|\[\]\/\\])/g
|
const ESCAPE_REGEXP = /([.*+?^=!:${}()|\[\]\/\\])/g;
|
||||||
|
|
||||||
$.escapeRegexp = (string) ->
|
$.escapeRegexp = string => string.replace(ESCAPE_REGEXP, "\\$1");
|
||||||
string.replace ESCAPE_REGEXP, "\\$1"
|
|
||||||
|
$.urlDecode = string => decodeURIComponent(string.replace(/\+/g, '%20'));
|
||||||
$.urlDecode = (string) ->
|
|
||||||
decodeURIComponent string.replace(/\+/g, '%20')
|
$.classify = function(string) {
|
||||||
|
string = string.split('_');
|
||||||
$.classify = (string) ->
|
for (let i = 0; i < string.length; i++) {
|
||||||
string = string.split('_')
|
var substr = string[i];
|
||||||
for substr, i in string
|
string[i] = substr[0].toUpperCase() + substr.slice(1);
|
||||||
string[i] = substr[0].toUpperCase() + substr[1..]
|
}
|
||||||
string.join('')
|
return string.join('');
|
||||||
|
};
|
||||||
$.framify = (fn, obj) ->
|
|
||||||
if window.requestAnimationFrame
|
$.framify = function(fn, obj) {
|
||||||
(args...) -> requestAnimationFrame(fn.bind(obj, args...))
|
if (window.requestAnimationFrame) {
|
||||||
else
|
return (...args) => requestAnimationFrame(fn.bind(obj, ...Array.from(args)));
|
||||||
fn
|
} else {
|
||||||
|
return fn;
|
||||||
$.requestAnimationFrame = (fn) ->
|
}
|
||||||
if window.requestAnimationFrame
|
};
|
||||||
requestAnimationFrame(fn)
|
|
||||||
else
|
$.requestAnimationFrame = function(fn) {
|
||||||
setTimeout(fn, 0)
|
if (window.requestAnimationFrame) {
|
||||||
return
|
requestAnimationFrame(fn);
|
||||||
|
} else {
|
||||||
#
|
setTimeout(fn, 0);
|
||||||
# Miscellaneous
|
}
|
||||||
#
|
};
|
||||||
|
|
||||||
$.noop = ->
|
//
|
||||||
|
// Miscellaneous
|
||||||
$.popup = (value) ->
|
//
|
||||||
try
|
|
||||||
win = window.open()
|
$.noop = function() {};
|
||||||
win.opener = null if win.opener
|
|
||||||
win.location = value.href or value
|
$.popup = function(value) {
|
||||||
catch
|
try {
|
||||||
window.open value.href or value, '_blank'
|
const win = window.open();
|
||||||
return
|
if (win.opener) { win.opener = null; }
|
||||||
|
win.location = value.href || value;
|
||||||
isMac = null
|
} catch (error) {
|
||||||
$.isMac = ->
|
window.open(value.href || value, '_blank');
|
||||||
isMac ?= navigator.userAgent?.indexOf('Mac') >= 0
|
}
|
||||||
|
};
|
||||||
isIE = null
|
|
||||||
$.isIE = ->
|
let isMac = null;
|
||||||
isIE ?= navigator.userAgent?.indexOf('MSIE') >= 0 || navigator.userAgent?.indexOf('rv:11.0') >= 0
|
$.isMac = () => isMac != null ? isMac : (isMac = (navigator.userAgent != null ? navigator.userAgent.indexOf('Mac') : undefined) >= 0);
|
||||||
|
|
||||||
isChromeForAndroid = null
|
let isIE = null;
|
||||||
$.isChromeForAndroid = ->
|
$.isIE = () => isIE != null ? isIE : (isIE = ((navigator.userAgent != null ? navigator.userAgent.indexOf('MSIE') : undefined) >= 0) || ((navigator.userAgent != null ? navigator.userAgent.indexOf('rv:11.0') : undefined) >= 0));
|
||||||
isChromeForAndroid ?= navigator.userAgent?.indexOf('Android') >= 0 && /Chrome\/([.0-9])+ Mobile/.test(navigator.userAgent)
|
|
||||||
|
let isChromeForAndroid = null;
|
||||||
isAndroid = null
|
$.isChromeForAndroid = () => isChromeForAndroid != null ? isChromeForAndroid : (isChromeForAndroid = ((navigator.userAgent != null ? navigator.userAgent.indexOf('Android') : undefined) >= 0) && /Chrome\/([.0-9])+ Mobile/.test(navigator.userAgent));
|
||||||
$.isAndroid = ->
|
|
||||||
isAndroid ?= navigator.userAgent?.indexOf('Android') >= 0
|
let isAndroid = null;
|
||||||
|
$.isAndroid = () => isAndroid != null ? isAndroid : (isAndroid = (navigator.userAgent != null ? navigator.userAgent.indexOf('Android') : undefined) >= 0);
|
||||||
isIOS = null
|
|
||||||
$.isIOS = ->
|
let isIOS = null;
|
||||||
isIOS ?= navigator.userAgent?.indexOf('iPhone') >= 0 || navigator.userAgent?.indexOf('iPad') >= 0
|
$.isIOS = () => isIOS != null ? isIOS : (isIOS = ((navigator.userAgent != null ? navigator.userAgent.indexOf('iPhone') : undefined) >= 0) || ((navigator.userAgent != null ? navigator.userAgent.indexOf('iPad') : undefined) >= 0));
|
||||||
|
|
||||||
$.overlayScrollbarsEnabled = ->
|
$.overlayScrollbarsEnabled = function() {
|
||||||
return false unless $.isMac()
|
if (!$.isMac()) { return false; }
|
||||||
div = document.createElement('div')
|
const div = document.createElement('div');
|
||||||
div.setAttribute('style', 'width: 100px; height: 100px; overflow: scroll; position: absolute')
|
div.setAttribute('style', 'width: 100px; height: 100px; overflow: scroll; position: absolute');
|
||||||
document.body.appendChild(div)
|
document.body.appendChild(div);
|
||||||
result = div.offsetWidth is div.clientWidth
|
const result = div.offsetWidth === div.clientWidth;
|
||||||
document.body.removeChild(div)
|
document.body.removeChild(div);
|
||||||
result
|
return result;
|
||||||
|
};
|
||||||
HIGHLIGHT_DEFAULTS =
|
|
||||||
className: 'highlight'
|
const HIGHLIGHT_DEFAULTS = {
|
||||||
|
className: 'highlight',
|
||||||
delay: 1000
|
delay: 1000
|
||||||
|
};
|
||||||
$.highlight = (el, options = {}) ->
|
|
||||||
options = $.extend {}, HIGHLIGHT_DEFAULTS, options
|
$.highlight = function(el, options) {
|
||||||
el.classList.add(options.className)
|
if (options == null) { options = {}; }
|
||||||
setTimeout (-> el.classList.remove(options.className)), options.delay
|
options = $.extend({}, HIGHLIGHT_DEFAULTS, options);
|
||||||
return
|
el.classList.add(options.className);
|
||||||
|
setTimeout((() => el.classList.remove(options.className)), options.delay);
|
||||||
$.copyToClipboard = (string) ->
|
};
|
||||||
textarea = document.createElement('textarea')
|
|
||||||
textarea.style.position = 'fixed'
|
$.copyToClipboard = function(string) {
|
||||||
textarea.style.opacity = 0
|
let result;
|
||||||
textarea.value = string
|
const textarea = document.createElement('textarea');
|
||||||
document.body.appendChild(textarea)
|
textarea.style.position = 'fixed';
|
||||||
try
|
textarea.style.opacity = 0;
|
||||||
textarea.select()
|
textarea.value = string;
|
||||||
result = !!document.execCommand('copy')
|
document.body.appendChild(textarea);
|
||||||
catch
|
try {
|
||||||
result = false
|
textarea.select();
|
||||||
finally
|
result = !!document.execCommand('copy');
|
||||||
document.body.removeChild(textarea)
|
} catch (error) {
|
||||||
result
|
result = false;
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
document.body.removeChild(textarea);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,147 +1,174 @@
|
|||||||
class app.models.Doc extends app.Model
|
/*
|
||||||
# Attributes: name, slug, type, version, release, db_size, mtime, links
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
constructor: ->
|
* DS207: Consider shorter variations of null checks
|
||||||
super
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@reset @
|
*/
|
||||||
@slug_without_version = @slug.split('~')[0]
|
app.models.Doc = class Doc extends app.Model {
|
||||||
@fullName = "#{@name}" + if @version then " #{@version}" else ''
|
// Attributes: name, slug, type, version, release, db_size, mtime, links
|
||||||
@icon = @slug_without_version
|
|
||||||
@short_version = @version.split(' ')[0] if @version
|
constructor() {
|
||||||
@text = @toEntry().text
|
super(...arguments);
|
||||||
|
this.reset(this);
|
||||||
reset: (data) ->
|
this.slug_without_version = this.slug.split('~')[0];
|
||||||
@resetEntries data.entries
|
this.fullName = `${this.name}` + (this.version ? ` ${this.version}` : '');
|
||||||
@resetTypes data.types
|
this.icon = this.slug_without_version;
|
||||||
return
|
if (this.version) { this.short_version = this.version.split(' ')[0]; }
|
||||||
|
this.text = this.toEntry().text;
|
||||||
resetEntries: (entries) ->
|
}
|
||||||
@entries = new app.collections.Entries(entries)
|
|
||||||
@entries.each (entry) => entry.doc = @
|
reset(data) {
|
||||||
return
|
this.resetEntries(data.entries);
|
||||||
|
this.resetTypes(data.types);
|
||||||
resetTypes: (types) ->
|
}
|
||||||
@types = new app.collections.Types(types)
|
|
||||||
@types.each (type) => type.doc = @
|
resetEntries(entries) {
|
||||||
return
|
this.entries = new app.collections.Entries(entries);
|
||||||
|
this.entries.each(entry => { return entry.doc = this; });
|
||||||
fullPath: (path = '') ->
|
}
|
||||||
path = "/#{path}" unless path[0] is '/'
|
|
||||||
"/#{@slug}#{path}"
|
resetTypes(types) {
|
||||||
|
this.types = new app.collections.Types(types);
|
||||||
fileUrl: (path) ->
|
this.types.each(type => { return type.doc = this; });
|
||||||
"#{app.config.docs_origin}#{@fullPath(path)}?#{@mtime}"
|
}
|
||||||
|
|
||||||
dbUrl: ->
|
fullPath(path) {
|
||||||
"#{app.config.docs_origin}/#{@slug}/#{app.config.db_filename}?#{@mtime}"
|
if (path == null) { path = ''; }
|
||||||
|
if (path[0] !== '/') { path = `/${path}`; }
|
||||||
indexUrl: ->
|
return `/${this.slug}${path}`;
|
||||||
"#{app.indexHost()}/#{@slug}/#{app.config.index_filename}?#{@mtime}"
|
}
|
||||||
|
|
||||||
toEntry: ->
|
fileUrl(path) {
|
||||||
return @entry if @entry
|
return `${app.config.docs_origin}${this.fullPath(path)}?${this.mtime}`;
|
||||||
@entry = new app.models.Entry
|
}
|
||||||
doc: @
|
|
||||||
name: @fullName
|
dbUrl() {
|
||||||
|
return `${app.config.docs_origin}/${this.slug}/${app.config.db_filename}?${this.mtime}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
indexUrl() {
|
||||||
|
return `${app.indexHost()}/${this.slug}/${app.config.index_filename}?${this.mtime}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
toEntry() {
|
||||||
|
if (this.entry) { return this.entry; }
|
||||||
|
this.entry = new app.models.Entry({
|
||||||
|
doc: this,
|
||||||
|
name: this.fullName,
|
||||||
path: 'index'
|
path: 'index'
|
||||||
@entry.addAlias(@name) if @version
|
});
|
||||||
@entry
|
if (this.version) { this.entry.addAlias(this.name); }
|
||||||
|
return this.entry;
|
||||||
findEntryByPathAndHash: (path, hash) ->
|
}
|
||||||
if hash and entry = @entries.findBy 'path', "#{path}##{hash}"
|
|
||||||
entry
|
findEntryByPathAndHash(path, hash) {
|
||||||
else if path is 'index'
|
let entry;
|
||||||
@toEntry()
|
if (hash && (entry = this.entries.findBy('path', `${path}#${hash}`))) {
|
||||||
else
|
return entry;
|
||||||
@entries.findBy 'path', path
|
} else if (path === 'index') {
|
||||||
|
return this.toEntry();
|
||||||
load: (onSuccess, onError, options = {}) ->
|
} else {
|
||||||
return if options.readCache and @_loadFromCache(onSuccess)
|
return this.entries.findBy('path', path);
|
||||||
|
}
|
||||||
callback = (data) =>
|
}
|
||||||
@reset data
|
|
||||||
onSuccess()
|
load(onSuccess, onError, options) {
|
||||||
@_setCache data if options.writeCache
|
if (options == null) { options = {}; }
|
||||||
return
|
if (options.readCache && this._loadFromCache(onSuccess)) { return; }
|
||||||
|
|
||||||
ajax
|
const callback = data => {
|
||||||
url: @indexUrl()
|
this.reset(data);
|
||||||
success: callback
|
onSuccess();
|
||||||
|
if (options.writeCache) { this._setCache(data); }
|
||||||
|
};
|
||||||
|
|
||||||
|
return ajax({
|
||||||
|
url: this.indexUrl(),
|
||||||
|
success: callback,
|
||||||
error: onError
|
error: onError
|
||||||
|
});
|
||||||
clearCache: ->
|
}
|
||||||
app.localStorage.del @slug
|
|
||||||
return
|
clearCache() {
|
||||||
|
app.localStorage.del(this.slug);
|
||||||
_loadFromCache: (onSuccess) ->
|
}
|
||||||
return unless data = @_getCache()
|
|
||||||
|
_loadFromCache(onSuccess) {
|
||||||
callback = =>
|
let data;
|
||||||
@reset data
|
if (!(data = this._getCache())) { return; }
|
||||||
onSuccess()
|
|
||||||
return
|
const callback = () => {
|
||||||
|
this.reset(data);
|
||||||
setTimeout callback, 0
|
onSuccess();
|
||||||
true
|
};
|
||||||
|
|
||||||
_getCache: ->
|
setTimeout(callback, 0);
|
||||||
return unless data = app.localStorage.get @slug
|
return true;
|
||||||
|
}
|
||||||
if data[0] is @mtime
|
|
||||||
return data[1]
|
_getCache() {
|
||||||
else
|
let data;
|
||||||
@clearCache()
|
if (!(data = app.localStorage.get(this.slug))) { return; }
|
||||||
return
|
|
||||||
|
if (data[0] === this.mtime) {
|
||||||
_setCache: (data) ->
|
return data[1];
|
||||||
app.localStorage.set @slug, [@mtime, data]
|
} else {
|
||||||
return
|
this.clearCache();
|
||||||
|
return;
|
||||||
install: (onSuccess, onError, onProgress) ->
|
}
|
||||||
return if @installing
|
}
|
||||||
@installing = true
|
|
||||||
|
_setCache(data) {
|
||||||
error = =>
|
app.localStorage.set(this.slug, [this.mtime, data]);
|
||||||
@installing = null
|
}
|
||||||
onError()
|
|
||||||
return
|
install(onSuccess, onError, onProgress) {
|
||||||
|
if (this.installing) { return; }
|
||||||
success = (data) =>
|
this.installing = true;
|
||||||
@installing = null
|
|
||||||
app.db.store @, data, onSuccess, error
|
const error = () => {
|
||||||
return
|
this.installing = null;
|
||||||
|
onError();
|
||||||
ajax
|
};
|
||||||
url: @dbUrl()
|
|
||||||
success: success
|
const success = data => {
|
||||||
error: error
|
this.installing = null;
|
||||||
progress: onProgress
|
app.db.store(this, data, onSuccess, error);
|
||||||
|
};
|
||||||
|
|
||||||
|
ajax({
|
||||||
|
url: this.dbUrl(),
|
||||||
|
success,
|
||||||
|
error,
|
||||||
|
progress: onProgress,
|
||||||
timeout: 3600
|
timeout: 3600
|
||||||
return
|
});
|
||||||
|
}
|
||||||
uninstall: (onSuccess, onError) ->
|
|
||||||
return if @installing
|
uninstall(onSuccess, onError) {
|
||||||
@installing = true
|
if (this.installing) { return; }
|
||||||
|
this.installing = true;
|
||||||
success = =>
|
|
||||||
@installing = null
|
const success = () => {
|
||||||
onSuccess()
|
this.installing = null;
|
||||||
return
|
onSuccess();
|
||||||
|
};
|
||||||
error = =>
|
|
||||||
@installing = null
|
const error = () => {
|
||||||
onError()
|
this.installing = null;
|
||||||
return
|
onError();
|
||||||
|
};
|
||||||
app.db.unstore @, success, error
|
|
||||||
return
|
app.db.unstore(this, success, error);
|
||||||
|
}
|
||||||
getInstallStatus: (callback) ->
|
|
||||||
app.db.version @, (value) ->
|
getInstallStatus(callback) {
|
||||||
callback installed: !!value, mtime: value
|
app.db.version(this, value => callback({installed: !!value, mtime: value}));
|
||||||
return
|
}
|
||||||
|
|
||||||
isOutdated: (status) ->
|
isOutdated(status) {
|
||||||
return false if not status
|
if (!status) { return false; }
|
||||||
isInstalled = status.installed or app.settings.get('autoInstall')
|
const isInstalled = status.installed || app.settings.get('autoInstall');
|
||||||
isInstalled and @mtime isnt status.mtime
|
return isInstalled && (this.mtime !== status.mtime);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,85 +1,116 @@
|
|||||||
#= require app/searcher
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require app/searcher
|
||||||
|
|
||||||
class app.models.Entry extends app.Model
|
(function() {
|
||||||
# Attributes: name, type, path
|
let applyAliases = undefined;
|
||||||
|
const Cls = (app.models.Entry = class Entry extends app.Model {
|
||||||
|
static initClass() {
|
||||||
|
|
||||||
constructor: ->
|
let ALIASES;
|
||||||
super
|
applyAliases = function(string) {
|
||||||
@text = applyAliases(app.Searcher.normalizeString(@name))
|
if (ALIASES.hasOwnProperty(string)) {
|
||||||
|
return [string, ALIASES[string]];
|
||||||
|
} else {
|
||||||
|
const words = string.split('.');
|
||||||
|
for (let i = 0; i < words.length; i++) {
|
||||||
|
var word = words[i];
|
||||||
|
if (ALIASES.hasOwnProperty(word)) {
|
||||||
|
words[i] = ALIASES[word];
|
||||||
|
return [string, words.join('.')];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return string;
|
||||||
|
};
|
||||||
|
|
||||||
addAlias: (name) ->
|
this.ALIASES = (ALIASES = {
|
||||||
text = applyAliases(app.Searcher.normalizeString(name))
|
'angular': 'ng',
|
||||||
@text = [@text] unless Array.isArray(@text)
|
'angular.js': 'ng',
|
||||||
@text.push(if Array.isArray(text) then text[1] else text)
|
'backbone.js': 'bb',
|
||||||
return
|
'c++': 'cpp',
|
||||||
|
'coffeescript': 'cs',
|
||||||
|
'crystal': 'cr',
|
||||||
|
'elixir': 'ex',
|
||||||
|
'javascript': 'js',
|
||||||
|
'julia': 'jl',
|
||||||
|
'jquery': '$',
|
||||||
|
'knockout.js': 'ko',
|
||||||
|
'kubernetes': 'k8s',
|
||||||
|
'less': 'ls',
|
||||||
|
'lodash': '_',
|
||||||
|
'löve': 'love',
|
||||||
|
'marionette': 'mn',
|
||||||
|
'markdown': 'md',
|
||||||
|
'matplotlib': 'mpl',
|
||||||
|
'modernizr': 'mdr',
|
||||||
|
'moment.js': 'mt',
|
||||||
|
'openjdk': 'java',
|
||||||
|
'nginx': 'ngx',
|
||||||
|
'numpy': 'np',
|
||||||
|
'pandas': 'pd',
|
||||||
|
'postgresql': 'pg',
|
||||||
|
'python': 'py',
|
||||||
|
'ruby.on.rails': 'ror',
|
||||||
|
'ruby': 'rb',
|
||||||
|
'rust': 'rs',
|
||||||
|
'sass': 'scss',
|
||||||
|
'tensorflow': 'tf',
|
||||||
|
'typescript': 'ts',
|
||||||
|
'underscore.js': '_'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// Attributes: name, type, path
|
||||||
|
|
||||||
fullPath: ->
|
constructor() {
|
||||||
@doc.fullPath if @isIndex() then '' else @path
|
super(...arguments);
|
||||||
|
this.text = applyAliases(app.Searcher.normalizeString(this.name));
|
||||||
|
}
|
||||||
|
|
||||||
dbPath: ->
|
addAlias(name) {
|
||||||
@path.replace /#.*/, ''
|
const text = applyAliases(app.Searcher.normalizeString(name));
|
||||||
|
if (!Array.isArray(this.text)) { this.text = [this.text]; }
|
||||||
|
this.text.push(Array.isArray(text) ? text[1] : text);
|
||||||
|
}
|
||||||
|
|
||||||
filePath: ->
|
fullPath() {
|
||||||
@doc.fullPath @_filePath()
|
return this.doc.fullPath(this.isIndex() ? '' : this.path);
|
||||||
|
}
|
||||||
|
|
||||||
fileUrl: ->
|
dbPath() {
|
||||||
@doc.fileUrl @_filePath()
|
return this.path.replace(/#.*/, '');
|
||||||
|
}
|
||||||
|
|
||||||
_filePath: ->
|
filePath() {
|
||||||
result = @path.replace /#.*/, ''
|
return this.doc.fullPath(this._filePath());
|
||||||
result += '.html' unless result[-5..-1] is '.html'
|
}
|
||||||
result
|
|
||||||
|
|
||||||
isIndex: ->
|
fileUrl() {
|
||||||
@path is 'index'
|
return this.doc.fileUrl(this._filePath());
|
||||||
|
}
|
||||||
|
|
||||||
getType: ->
|
_filePath() {
|
||||||
@doc.types.findBy 'name', @type
|
let result = this.path.replace(/#.*/, '');
|
||||||
|
if (result.slice(-5) !== '.html') { result += '.html'; }
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
loadFile: (onSuccess, onError) ->
|
isIndex() {
|
||||||
app.db.load(@, onSuccess, onError)
|
return this.path === 'index';
|
||||||
|
}
|
||||||
|
|
||||||
applyAliases = (string) ->
|
getType() {
|
||||||
if ALIASES.hasOwnProperty(string)
|
return this.doc.types.findBy('name', this.type);
|
||||||
return [string, ALIASES[string]]
|
}
|
||||||
else
|
|
||||||
words = string.split('.')
|
|
||||||
for word, i in words when ALIASES.hasOwnProperty(word)
|
|
||||||
words[i] = ALIASES[word]
|
|
||||||
return [string, words.join('.')]
|
|
||||||
return string
|
|
||||||
|
|
||||||
@ALIASES = ALIASES =
|
loadFile(onSuccess, onError) {
|
||||||
'angular': 'ng'
|
return app.db.load(this, onSuccess, onError);
|
||||||
'angular.js': 'ng'
|
}
|
||||||
'backbone.js': 'bb'
|
});
|
||||||
'c++': 'cpp'
|
Cls.initClass();
|
||||||
'coffeescript': 'cs'
|
return Cls;
|
||||||
'crystal': 'cr'
|
})();
|
||||||
'elixir': 'ex'
|
|
||||||
'javascript': 'js'
|
|
||||||
'julia': 'jl'
|
|
||||||
'jquery': '$'
|
|
||||||
'knockout.js': 'ko'
|
|
||||||
'kubernetes': 'k8s'
|
|
||||||
'less': 'ls'
|
|
||||||
'lodash': '_'
|
|
||||||
'löve': 'love'
|
|
||||||
'marionette': 'mn'
|
|
||||||
'markdown': 'md'
|
|
||||||
'matplotlib': 'mpl'
|
|
||||||
'modernizr': 'mdr'
|
|
||||||
'moment.js': 'mt'
|
|
||||||
'openjdk': 'java'
|
|
||||||
'nginx': 'ngx'
|
|
||||||
'numpy': 'np'
|
|
||||||
'pandas': 'pd'
|
|
||||||
'postgresql': 'pg'
|
|
||||||
'python': 'py'
|
|
||||||
'ruby.on.rails': 'ror'
|
|
||||||
'ruby': 'rb'
|
|
||||||
'rust': 'rs'
|
|
||||||
'sass': 'scss'
|
|
||||||
'tensorflow': 'tf'
|
|
||||||
'typescript': 'ts'
|
|
||||||
'underscore.js': '_'
|
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
class app.Model
|
app.Model = class Model {
|
||||||
constructor: (attributes) ->
|
constructor(attributes) {
|
||||||
@[key] = value for key, value of attributes
|
for (var key in attributes) { var value = attributes[key]; this[key] = value; }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,14 +1,24 @@
|
|||||||
class app.models.Type extends app.Model
|
/*
|
||||||
# Attributes: name, slug, count
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.models.Type = class Type extends app.Model {
|
||||||
|
// Attributes: name, slug, count
|
||||||
|
|
||||||
fullPath: ->
|
fullPath() {
|
||||||
"/#{@doc.slug}-#{@slug}/"
|
return `/${this.doc.slug}-${this.slug}/`;
|
||||||
|
}
|
||||||
|
|
||||||
entries: ->
|
entries() {
|
||||||
@doc.entries.findAllBy 'type', @name
|
return this.doc.entries.findAllBy('type', this.name);
|
||||||
|
}
|
||||||
|
|
||||||
toEntry: ->
|
toEntry() {
|
||||||
new app.models.Entry
|
return new app.models.Entry({
|
||||||
doc: @doc
|
doc: this.doc,
|
||||||
name: "#{@doc.name} / #{@name}"
|
name: `${this.doc.name} / ${this.name}`,
|
||||||
path: '..' + @fullPath()
|
path: '..' + this.fullPath()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,11 +1,19 @@
|
|||||||
app.templates.render = (name, value, args...) ->
|
/*
|
||||||
template = app.templates[name]
|
* decaffeinate suggestions:
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.templates.render = function(name, value, ...args) {
|
||||||
|
const template = app.templates[name];
|
||||||
|
|
||||||
if Array.isArray(value)
|
if (Array.isArray(value)) {
|
||||||
result = ''
|
let result = '';
|
||||||
result += template(val, args...) for val in value
|
for (var val of Array.from(value)) { result += template(val, ...Array.from(args)); }
|
||||||
result
|
return result;
|
||||||
else if typeof template is 'function'
|
} else if (typeof template === 'function') {
|
||||||
template(value, args...)
|
return template(value, ...Array.from(args));
|
||||||
else
|
} else {
|
||||||
template
|
return template;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -1,73 +1,85 @@
|
|||||||
error = (title, text = '', links = '') ->
|
/*
|
||||||
text = """<p class="_error-text">#{text}</p>""" if text
|
* decaffeinate suggestions:
|
||||||
links = """<p class="_error-links">#{links}</p>""" if links
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
"""<div class="_error"><h1 class="_error-title">#{title}</h1>#{text}#{links}</div>"""
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const error = function(title, text, links) {
|
||||||
|
if (text == null) { text = ''; }
|
||||||
|
if (links == null) { links = ''; }
|
||||||
|
if (text) { text = `<p class="_error-text">${text}</p>`; }
|
||||||
|
if (links) { links = `<p class="_error-links">${links}</p>`; }
|
||||||
|
return `<div class="_error"><h1 class="_error-title">${title}</h1>${text}${links}</div>`;
|
||||||
|
};
|
||||||
|
|
||||||
back = '<a href="#" data-behavior="back" class="_error-link">Go back</a>'
|
const back = '<a href="#" data-behavior="back" class="_error-link">Go back</a>';
|
||||||
|
|
||||||
app.templates.notFoundPage = ->
|
app.templates.notFoundPage = () => error(" Page not found. ",
|
||||||
error """ Page not found. """,
|
" It may be missing from the source documentation or this could be a bug. ",
|
||||||
""" It may be missing from the source documentation or this could be a bug. """,
|
back);
|
||||||
back
|
|
||||||
|
|
||||||
app.templates.pageLoadError = ->
|
app.templates.pageLoadError = () => error(" The page failed to load. ",
|
||||||
error """ The page failed to load. """,
|
` It may be missing from the server (try reloading the app) or you could be offline (try <a href="/offline">installing the documentation for offline usage</a> when online again).<br>
|
||||||
""" It may be missing from the server (try reloading the app) or you could be offline (try <a href="/offline">installing the documentation for offline usage</a> when online again).<br>
|
If you're online and you keep seeing this, you're likely behind a proxy or firewall that blocks cross-domain requests. `,
|
||||||
If you're online and you keep seeing this, you're likely behind a proxy or firewall that blocks cross-domain requests. """,
|
` ${back} · <a href="/#${location.pathname}" target="_top" class="_error-link">Reload</a>
|
||||||
""" #{back} · <a href="/##{location.pathname}" target="_top" class="_error-link">Reload</a>
|
· <a href="#" class="_error-link" data-retry>Retry</a> `
|
||||||
· <a href="#" class="_error-link" data-retry>Retry</a> """
|
);
|
||||||
|
|
||||||
app.templates.bootError = ->
|
app.templates.bootError = () => error(" The app failed to load. ",
|
||||||
error """ The app failed to load. """,
|
` Check your Internet connection and try <a href="#" data-behavior="reload">reloading</a>.<br>
|
||||||
""" Check your Internet connection and try <a href="#" data-behavior="reload">reloading</a>.<br>
|
If you keep seeing this, you're likely behind a proxy or firewall that blocks cross-domain requests. `
|
||||||
If you keep seeing this, you're likely behind a proxy or firewall that blocks cross-domain requests. """
|
);
|
||||||
|
|
||||||
app.templates.offlineError = (reason, exception) ->
|
app.templates.offlineError = function(reason, exception) {
|
||||||
if reason is 'cookie_blocked'
|
if (reason === 'cookie_blocked') {
|
||||||
return error """ Cookies must be enabled to use offline mode. """
|
return error(" Cookies must be enabled to use offline mode. ");
|
||||||
|
}
|
||||||
|
|
||||||
reason = switch reason
|
reason = (() => { switch (reason) {
|
||||||
when 'not_supported'
|
case 'not_supported':
|
||||||
""" DevDocs requires IndexedDB to cache documentations for offline access.<br>
|
return ` DevDocs requires IndexedDB to cache documentations for offline access.<br>
|
||||||
Unfortunately your browser either doesn't support IndexedDB or doesn't make it available. """
|
Unfortunately your browser either doesn't support IndexedDB or doesn't make it available. `;
|
||||||
when 'buggy'
|
case 'buggy':
|
||||||
""" DevDocs requires IndexedDB to cache documentations for offline access.<br>
|
return ` DevDocs requires IndexedDB to cache documentations for offline access.<br>
|
||||||
Unfortunately your browser's implementation of IndexedDB contains bugs that prevent DevDocs from using it. """
|
Unfortunately your browser's implementation of IndexedDB contains bugs that prevent DevDocs from using it. `;
|
||||||
when 'private_mode'
|
case 'private_mode':
|
||||||
""" Your browser appears to be running in private mode.<br>
|
return ` Your browser appears to be running in private mode.<br>
|
||||||
This prevents DevDocs from caching documentations for offline access."""
|
This prevents DevDocs from caching documentations for offline access.`;
|
||||||
when 'exception'
|
case 'exception':
|
||||||
""" An error occurred when trying to open the IndexedDB database:<br>
|
return ` An error occurred when trying to open the IndexedDB database:<br>
|
||||||
<code class="_label">#{exception.name}: #{exception.message}</code> """
|
<code class="_label">${exception.name}: ${exception.message}</code> `;
|
||||||
when 'cant_open'
|
case 'cant_open':
|
||||||
""" An error occurred when trying to open the IndexedDB database:<br>
|
return ` An error occurred when trying to open the IndexedDB database:<br>
|
||||||
<code class="_label">#{exception.name}: #{exception.message}</code><br>
|
<code class="_label">${exception.name}: ${exception.message}</code><br>
|
||||||
This could be because you're browsing in private mode or have disallowed offline storage on the domain. """
|
This could be because you're browsing in private mode or have disallowed offline storage on the domain. `;
|
||||||
when 'version'
|
case 'version':
|
||||||
""" The IndexedDB database was modified with a newer version of the app.<br>
|
return ` The IndexedDB database was modified with a newer version of the app.<br>
|
||||||
<a href="#" data-behavior="reload">Reload the page</a> to use offline mode. """
|
<a href="#" data-behavior="reload">Reload the page</a> to use offline mode. `;
|
||||||
when 'empty'
|
case 'empty':
|
||||||
""" The IndexedDB database appears to be corrupted. Try <a href="#" data-behavior="reset">resetting the app</a>. """
|
return " The IndexedDB database appears to be corrupted. Try <a href=\"#\" data-behavior=\"reset\">resetting the app</a>. ";
|
||||||
|
} })();
|
||||||
|
|
||||||
error 'Offline mode is unavailable.', reason
|
return error('Offline mode is unavailable.', reason);
|
||||||
|
};
|
||||||
|
|
||||||
app.templates.unsupportedBrowser = """
|
app.templates.unsupportedBrowser = `\
|
||||||
<div class="_fail">
|
<div class="_fail">
|
||||||
<h1 class="_fail-title">Your browser is unsupported, sorry.</h1>
|
<h1 class="_fail-title">Your browser is unsupported, sorry.</h1>
|
||||||
<p class="_fail-text">DevDocs is an API documentation browser which supports the following browsers:
|
<p class="_fail-text">DevDocs is an API documentation browser which supports the following browsers:
|
||||||
<ul class="_fail-list">
|
<ul class="_fail-list">
|
||||||
<li>Recent versions of Firefox, Chrome, or Opera
|
<li>Recent versions of Firefox, Chrome, or Opera
|
||||||
<li>Safari 11.1+
|
<li>Safari 11.1+
|
||||||
<li>Edge 17+
|
<li>Edge 17+
|
||||||
<li>iOS 11.3+
|
<li>iOS 11.3+
|
||||||
</ul>
|
</ul>
|
||||||
<p class="_fail-text">
|
<p class="_fail-text">
|
||||||
If you're unable to upgrade, we apologize.
|
If you're unable to upgrade, we apologize.
|
||||||
We decided to prioritize speed and new features over support for older browsers.
|
We decided to prioritize speed and new features over support for older browsers.
|
||||||
<p class="_fail-text">
|
<p class="_fail-text">
|
||||||
Note: if you're already using one of the browsers above, check your settings and add-ons.
|
Note: if you're already using one of the browsers above, check your settings and add-ons.
|
||||||
The app uses feature detection, not user agent sniffing.
|
The app uses feature detection, not user agent sniffing.
|
||||||
<p class="_fail-text">
|
<p class="_fail-text">
|
||||||
— <a href="https://twitter.com/DevDocs">@DevDocs</a>
|
— <a href="https://twitter.com/DevDocs">@DevDocs</a>
|
||||||
</div>
|
</div>\
|
||||||
"""
|
`;
|
||||||
|
@ -1,9 +1,14 @@
|
|||||||
notice = (text) -> """<p class="_notice-text">#{text}</p>"""
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const notice = text => `<p class="_notice-text">${text}</p>`;
|
||||||
|
|
||||||
app.templates.singleDocNotice = (doc) ->
|
app.templates.singleDocNotice = doc => notice(` You're browsing the ${doc.fullName} documentation. To browse all docs, go to
|
||||||
notice """ You're browsing the #{doc.fullName} documentation. To browse all docs, go to
|
<a href="//${app.config.production_host}" target="_top">${app.config.production_host}</a> (or press <code>esc</code>). `
|
||||||
<a href="//#{app.config.production_host}" target="_top">#{app.config.production_host}</a> (or press <code>esc</code>). """
|
);
|
||||||
|
|
||||||
app.templates.disabledDocNotice = ->
|
app.templates.disabledDocNotice = () => notice(` <strong>This documentation is disabled.</strong>
|
||||||
notice """ <strong>This documentation is disabled.</strong>
|
To enable it, go to <a href="/settings" class="_notice-link">Preferences</a>. `
|
||||||
To enable it, go to <a href="/settings" class="_notice-link">Preferences</a>. """
|
);
|
||||||
|
@ -1,76 +1,81 @@
|
|||||||
notif = (title, html) ->
|
/*
|
||||||
html = html.replace /<a /g, '<a class="_notif-link" '
|
* decaffeinate suggestions:
|
||||||
""" <h5 class="_notif-title">#{title}</h5>
|
* DS101: Remove unnecessary use of Array.from
|
||||||
#{html}
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
<button type="button" class="_notif-close" title="Close"><svg><use xlink:href="#icon-close"/></svg>Close</a>
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
"""
|
*/
|
||||||
|
const notif = function(title, html) {
|
||||||
|
html = html.replace(/<a /g, '<a class="_notif-link" ');
|
||||||
|
return ` <h5 class="_notif-title">${title}</h5>
|
||||||
|
${html}
|
||||||
|
<button type="button" class="_notif-close" title="Close"><svg><use xlink:href="#icon-close"/></svg>Close</a>\
|
||||||
|
`;
|
||||||
|
};
|
||||||
|
|
||||||
textNotif = (title, message) ->
|
const textNotif = (title, message) => notif(title, `<p class="_notif-text">${message}`);
|
||||||
notif title, """<p class="_notif-text">#{message}"""
|
|
||||||
|
|
||||||
app.templates.notifUpdateReady = ->
|
app.templates.notifUpdateReady = () => textNotif("<span data-behavior=\"reboot\">DevDocs has been updated.</span>",
|
||||||
textNotif """<span data-behavior="reboot">DevDocs has been updated.</span>""",
|
"<span data-behavior=\"reboot\"><a href=\"#\" data-behavior=\"reboot\">Reload the page</a> to use the new version.</span>");
|
||||||
"""<span data-behavior="reboot"><a href="#" data-behavior="reboot">Reload the page</a> to use the new version.</span>"""
|
|
||||||
|
|
||||||
app.templates.notifError = ->
|
app.templates.notifError = () => textNotif(" Oops, an error occurred. ",
|
||||||
textNotif """ Oops, an error occurred. """,
|
` Try <a href="#" data-behavior="hard-reload">reloading</a>, and if the problem persists,
|
||||||
""" Try <a href="#" data-behavior="hard-reload">reloading</a>, and if the problem persists,
|
<a href="#" data-behavior="reset">resetting the app</a>.<br>
|
||||||
<a href="#" data-behavior="reset">resetting the app</a>.<br>
|
You can also report this issue on <a href="https://github.com/freeCodeCamp/devdocs/issues/new" target="_blank" rel="noopener">GitHub</a>. `
|
||||||
You can also report this issue on <a href="https://github.com/freeCodeCamp/devdocs/issues/new" target="_blank" rel="noopener">GitHub</a>. """
|
);
|
||||||
|
|
||||||
app.templates.notifQuotaExceeded = ->
|
app.templates.notifQuotaExceeded = () => textNotif(" The offline database has exceeded its size limitation. ",
|
||||||
textNotif """ The offline database has exceeded its size limitation. """,
|
" Unfortunately this quota can't be detected programmatically, and the database can't be opened while over the quota, so it had to be reset. ");
|
||||||
""" Unfortunately this quota can't be detected programmatically, and the database can't be opened while over the quota, so it had to be reset. """
|
|
||||||
|
|
||||||
app.templates.notifCookieBlocked = ->
|
app.templates.notifCookieBlocked = () => textNotif(" Please enable cookies. ",
|
||||||
textNotif """ Please enable cookies. """,
|
" DevDocs will not work properly if cookies are disabled. ");
|
||||||
""" DevDocs will not work properly if cookies are disabled. """
|
|
||||||
|
|
||||||
app.templates.notifInvalidLocation = ->
|
app.templates.notifInvalidLocation = () => textNotif(` DevDocs must be loaded from ${app.config.production_host} `,
|
||||||
textNotif """ DevDocs must be loaded from #{app.config.production_host} """,
|
" Otherwise things are likely to break. ");
|
||||||
""" Otherwise things are likely to break. """
|
|
||||||
|
|
||||||
app.templates.notifImportInvalid = ->
|
app.templates.notifImportInvalid = () => textNotif(" Oops, an error occurred. ",
|
||||||
textNotif """ Oops, an error occurred. """,
|
" The file you selected is invalid. ");
|
||||||
""" The file you selected is invalid. """
|
|
||||||
|
|
||||||
app.templates.notifNews = (news) ->
|
app.templates.notifNews = news => notif('Changelog', `<div class="_notif-content _notif-news">${app.templates.newsList(news, {years: false})}</div>`);
|
||||||
notif 'Changelog', """<div class="_notif-content _notif-news">#{app.templates.newsList(news, years: false)}</div>"""
|
|
||||||
|
|
||||||
app.templates.notifUpdates = (docs, disabledDocs) ->
|
app.templates.notifUpdates = function(docs, disabledDocs) {
|
||||||
html = '<div class="_notif-content _notif-news">'
|
let doc;
|
||||||
|
let html = '<div class="_notif-content _notif-news">';
|
||||||
|
|
||||||
if docs.length > 0
|
if (docs.length > 0) {
|
||||||
html += '<div class="_news-row">'
|
html += '<div class="_news-row">';
|
||||||
html += '<ul class="_notif-list">'
|
html += '<ul class="_notif-list">';
|
||||||
for doc in docs
|
for (doc of Array.from(docs)) {
|
||||||
html += "<li>#{doc.name}"
|
html += `<li>${doc.name}`;
|
||||||
html += " <code>→</code> #{doc.release}" if doc.release
|
if (doc.release) { html += ` <code>→</code> ${doc.release}`; }
|
||||||
html += '</ul></div>'
|
}
|
||||||
|
html += '</ul></div>';
|
||||||
|
}
|
||||||
|
|
||||||
if disabledDocs.length > 0
|
if (disabledDocs.length > 0) {
|
||||||
html += '<div class="_news-row"><p class="_news-title">Disabled:'
|
html += '<div class="_news-row"><p class="_news-title">Disabled:';
|
||||||
html += '<ul class="_notif-list">'
|
html += '<ul class="_notif-list">';
|
||||||
for doc in disabledDocs
|
for (doc of Array.from(disabledDocs)) {
|
||||||
html += "<li>#{doc.name}"
|
html += `<li>${doc.name}`;
|
||||||
html += " <code>→</code> #{doc.release}" if doc.release
|
if (doc.release) { html += ` <code>→</code> ${doc.release}`; }
|
||||||
html += """<span class="_notif-info"><a href="/settings">Enable</a></span>"""
|
html += "<span class=\"_notif-info\"><a href=\"/settings\">Enable</a></span>";
|
||||||
html += '</ul></div>'
|
}
|
||||||
|
html += '</ul></div>';
|
||||||
|
}
|
||||||
|
|
||||||
notif 'Updates', "#{html}</div>"
|
return notif('Updates', `${html}</div>`);
|
||||||
|
};
|
||||||
|
|
||||||
app.templates.notifShare = ->
|
app.templates.notifShare = () => textNotif(" Hi there! ",
|
||||||
textNotif """ Hi there! """,
|
` Like DevDocs? Help us reach more developers by sharing the link with your friends on
|
||||||
""" Like DevDocs? Help us reach more developers by sharing the link with your friends on
|
<a href="https://out.devdocs.io/s/tw" target="_blank" rel="noopener">Twitter</a>, <a href="https://out.devdocs.io/s/fb" target="_blank" rel="noopener">Facebook</a>,
|
||||||
<a href="https://out.devdocs.io/s/tw" target="_blank" rel="noopener">Twitter</a>, <a href="https://out.devdocs.io/s/fb" target="_blank" rel="noopener">Facebook</a>,
|
<a href="https://out.devdocs.io/s/re" target="_blank" rel="noopener">Reddit</a>, etc.<br>Thanks :) `
|
||||||
<a href="https://out.devdocs.io/s/re" target="_blank" rel="noopener">Reddit</a>, etc.<br>Thanks :) """
|
);
|
||||||
|
|
||||||
app.templates.notifUpdateDocs = ->
|
app.templates.notifUpdateDocs = () => textNotif(" Documentation updates available. ",
|
||||||
textNotif """ Documentation updates available. """,
|
" <a href=\"/offline\">Install them</a> as soon as possible to avoid broken pages. ");
|
||||||
""" <a href="/offline">Install them</a> as soon as possible to avoid broken pages. """
|
|
||||||
|
|
||||||
app.templates.notifAnalyticsConsent = ->
|
app.templates.notifAnalyticsConsent = () => textNotif(" Tracking cookies ",
|
||||||
textNotif """ Tracking cookies """,
|
` We would like to gather usage data about how DevDocs is used through Google Analytics and Gauges. We only collect anonymous traffic information.
|
||||||
""" We would like to gather usage data about how DevDocs is used through Google Analytics and Gauges. We only collect anonymous traffic information.
|
Please confirm if you accept our tracking cookies. You can always change your decision in the settings.
|
||||||
Please confirm if you accept our tracking cookies. You can always change your decision in the settings.
|
<br><span class="_notif-right"><a href="#" data-behavior="accept-analytics">Accept</a> or <a href="#" data-behavior="decline-analytics">Decline</a></span> `
|
||||||
<br><span class="_notif-right"><a href="#" data-behavior="accept-analytics">Accept</a> or <a href="#" data-behavior="decline-analytics">Decline</a></span> """
|
);
|
||||||
|
@ -1,91 +1,106 @@
|
|||||||
app.templates.aboutPage = ->
|
/*
|
||||||
all_docs = app.docs.all().concat(app.disabledDocs.all()...)
|
* decaffeinate suggestions:
|
||||||
# de-duplicate docs by doc.name
|
* DS101: Remove unnecessary use of Array.from
|
||||||
docs = []
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
docs.push doc for doc in all_docs when not (docs.find (d) -> d.name == doc.name)
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
"""
|
* DS207: Consider shorter variations of null checks
|
||||||
<nav class="_toc" role="directory">
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
<h3 class="_toc-title">Table of Contents</h3>
|
*/
|
||||||
<ul class="_toc-list">
|
app.templates.aboutPage = function() {
|
||||||
<li><a href="#copyright">Copyright</a>
|
let doc;
|
||||||
<li><a href="#plugins">Plugins</a>
|
const all_docs = app.docs.all().concat(...Array.from(app.disabledDocs.all() || []));
|
||||||
<li><a href="#faq">FAQ</a>
|
// de-duplicate docs by doc.name
|
||||||
<li><a href="#credits">Credits</a>
|
const docs = [];
|
||||||
<li><a href="#privacy">Privacy Policy</a>
|
for (doc of Array.from(all_docs)) { if (!(docs.find(d => d.name === doc.name))) { docs.push(doc); } }
|
||||||
</ul>
|
return `\
|
||||||
</nav>
|
<nav class="_toc" role="directory">
|
||||||
|
<h3 class="_toc-title">Table of Contents</h3>
|
||||||
<h1 class="_lined-heading">DevDocs: API Documentation Browser</h1>
|
<ul class="_toc-list">
|
||||||
<p>DevDocs combines multiple developer documentations in a clean and organized web UI with instant search, offline support, mobile version, dark theme, keyboard shortcuts, and more.
|
<li><a href="#copyright">Copyright</a>
|
||||||
<p>DevDocs is free and <a href="https://github.com/freeCodeCamp/devdocs">open source</a>. It was created by <a href="https://thibaut.me">Thibaut Courouble</a> and is operated by <a href="https://www.freecodecamp.org/">freeCodeCamp</a>.
|
<li><a href="#plugins">Plugins</a>
|
||||||
<p>To keep up-to-date with the latest news:
|
<li><a href="#faq">FAQ</a>
|
||||||
<ul>
|
<li><a href="#credits">Credits</a>
|
||||||
<li>Follow <a href="https://twitter.com/DevDocs">@DevDocs</a> on Twitter
|
<li><a href="#privacy">Privacy Policy</a>
|
||||||
<li>Watch the repository on <a href="https://github.com/freeCodeCamp/devdocs/subscription">GitHub</a> <iframe class="_github-btn" src="https://ghbtns.com/github-btn.html?user=freeCodeCamp&repo=devdocs&type=watch&count=true" allowtransparency="true" frameborder="0" scrolling="0" width="100" height="20" tabindex="-1"></iframe>
|
|
||||||
<li>Join the <a href="https://discord.gg/PRyKn3Vbay">Discord</a> chat room
|
|
||||||
</ul>
|
</ul>
|
||||||
|
</nav>
|
||||||
|
|
||||||
<h2 class="_block-heading" id="copyright">Copyright and License</h2>
|
<h1 class="_lined-heading">DevDocs: API Documentation Browser</h1>
|
||||||
<p class="_note">
|
<p>DevDocs combines multiple developer documentations in a clean and organized web UI with instant search, offline support, mobile version, dark theme, keyboard shortcuts, and more.
|
||||||
<strong>Copyright 2013–2023 Thibaut Courouble and <a href="https://github.com/freeCodeCamp/devdocs/graphs/contributors">other contributors</a></strong><br>
|
<p>DevDocs is free and <a href="https://github.com/freeCodeCamp/devdocs">open source</a>. It was created by <a href="https://thibaut.me">Thibaut Courouble</a> and is operated by <a href="https://www.freecodecamp.org/">freeCodeCamp</a>.
|
||||||
This software is licensed under the terms of the Mozilla Public License v2.0.<br>
|
<p>To keep up-to-date with the latest news:
|
||||||
You may obtain a copy of the source code at <a href="https://github.com/freeCodeCamp/devdocs">github.com/freeCodeCamp/devdocs</a>.<br>
|
<ul>
|
||||||
For more information, see the <a href="https://github.com/freeCodeCamp/devdocs/blob/main/COPYRIGHT">COPYRIGHT</a>
|
<li>Follow <a href="https://twitter.com/DevDocs">@DevDocs</a> on Twitter
|
||||||
and <a href="https://github.com/freeCodeCamp/devdocs/blob/main/LICENSE">LICENSE</a> files.
|
<li>Watch the repository on <a href="https://github.com/freeCodeCamp/devdocs/subscription">GitHub</a> <iframe class="_github-btn" src="https://ghbtns.com/github-btn.html?user=freeCodeCamp&repo=devdocs&type=watch&count=true" allowtransparency="true" frameborder="0" scrolling="0" width="100" height="20" tabindex="-1"></iframe>
|
||||||
|
<li>Join the <a href="https://discord.gg/PRyKn3Vbay">Discord</a> chat room
|
||||||
|
</ul>
|
||||||
|
|
||||||
<h2 class="_block-heading" id="plugins">Plugins and Extensions</h2>
|
<h2 class="_block-heading" id="copyright">Copyright and License</h2>
|
||||||
<ul>
|
<p class="_note">
|
||||||
<li><a href="https://sublime.wbond.net/packages/DevDocs">Sublime Text package</a>
|
<strong>Copyright 2013–2023 Thibaut Courouble and <a href="https://github.com/freeCodeCamp/devdocs/graphs/contributors">other contributors</a></strong><br>
|
||||||
<li><a href="https://atom.io/packages/devdocs">Atom package</a>
|
This software is licensed under the terms of the Mozilla Public License v2.0.<br>
|
||||||
<li><a href="https://marketplace.visualstudio.com/items?itemName=deibit.devdocs">Visual Studio Code extension</a>
|
You may obtain a copy of the source code at <a href="https://github.com/freeCodeCamp/devdocs">github.com/freeCodeCamp/devdocs</a>.<br>
|
||||||
<li><a href="https://github.com/yannickglt/alfred-devdocs">Alfred workflow</a>
|
For more information, see the <a href="https://github.com/freeCodeCamp/devdocs/blob/main/COPYRIGHT">COPYRIGHT</a>
|
||||||
<li><a href="https://github.com/search?q=topic%3Adevdocs&type=Repositories">More…</a>
|
and <a href="https://github.com/freeCodeCamp/devdocs/blob/main/LICENSE">LICENSE</a> files.
|
||||||
</ul>
|
|
||||||
|
|
||||||
<h2 class="_block-heading" id="faq">Questions & Answers</h2>
|
<h2 class="_block-heading" id="plugins">Plugins and Extensions</h2>
|
||||||
<dl>
|
<ul>
|
||||||
<dt>Where can I suggest new docs and features?
|
<li><a href="https://sublime.wbond.net/packages/DevDocs">Sublime Text package</a>
|
||||||
<dd>You can suggest and vote for new docs on the <a href="https://trello.com/b/6BmTulfx/devdocs-documentation">Trello board</a>.<br>
|
<li><a href="https://atom.io/packages/devdocs">Atom package</a>
|
||||||
If you have a specific feature request, add it to the <a href="https://github.com/freeCodeCamp/devdocs/issues">issue tracker</a>.<br>
|
<li><a href="https://marketplace.visualstudio.com/items?itemName=deibit.devdocs">Visual Studio Code extension</a>
|
||||||
Otherwise, come talk to us in the <a href="https://discord.gg/PRyKn3Vbay">Discord</a> chat room.
|
<li><a href="https://github.com/yannickglt/alfred-devdocs">Alfred workflow</a>
|
||||||
<dt>Where can I report bugs?
|
<li><a href="https://github.com/search?q=topic%3Adevdocs&type=Repositories">More…</a>
|
||||||
<dd>In the <a href="https://github.com/freeCodeCamp/devdocs/issues">issue tracker</a>. Thanks!
|
</ul>
|
||||||
</dl>
|
|
||||||
|
|
||||||
<h2 class="_block-heading" id="credits">Credits</h2>
|
<h2 class="_block-heading" id="faq">Questions & Answers</h2>
|
||||||
|
<dl>
|
||||||
|
<dt>Where can I suggest new docs and features?
|
||||||
|
<dd>You can suggest and vote for new docs on the <a href="https://trello.com/b/6BmTulfx/devdocs-documentation">Trello board</a>.<br>
|
||||||
|
If you have a specific feature request, add it to the <a href="https://github.com/freeCodeCamp/devdocs/issues">issue tracker</a>.<br>
|
||||||
|
Otherwise, come talk to us in the <a href="https://discord.gg/PRyKn3Vbay">Discord</a> chat room.
|
||||||
|
<dt>Where can I report bugs?
|
||||||
|
<dd>In the <a href="https://github.com/freeCodeCamp/devdocs/issues">issue tracker</a>. Thanks!
|
||||||
|
</dl>
|
||||||
|
|
||||||
<p><strong>Special thanks to:</strong>
|
<h2 class="_block-heading" id="credits">Credits</h2>
|
||||||
<ul>
|
|
||||||
<li><a href="https://sentry.io/">Sentry</a> and <a href="https://get.gaug.es/?utm_source=devdocs&utm_medium=referral&utm_campaign=sponsorships" title="Real Time Web Analytics">Gauges</a> for offering a free account to DevDocs
|
|
||||||
<li><a href="https://out.devdocs.io/s/maxcdn">MaxCDN</a>, <a href="https://out.devdocs.io/s/shopify">Shopify</a>, <a href="https://out.devdocs.io/s/jetbrains">JetBrains</a> and <a href="https://out.devdocs.io/s/code-school">Code School</a> for sponsoring DevDocs in the past
|
|
||||||
<li><a href="https://www.heroku.com">Heroku</a> and <a href="https://newrelic.com/">New Relic</a> for providing awesome free service
|
|
||||||
<li><a href="https://www.jeremykratz.com/">Jeremy Kratz</a> for the C/C++ logo
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<div class="_table">
|
<p><strong>Special thanks to:</strong>
|
||||||
<table class="_credits">
|
<ul>
|
||||||
<tr>
|
<li><a href="https://sentry.io/">Sentry</a> and <a href="https://get.gaug.es/?utm_source=devdocs&utm_medium=referral&utm_campaign=sponsorships" title="Real Time Web Analytics">Gauges</a> for offering a free account to DevDocs
|
||||||
<th>Documentation
|
<li><a href="https://out.devdocs.io/s/maxcdn">MaxCDN</a>, <a href="https://out.devdocs.io/s/shopify">Shopify</a>, <a href="https://out.devdocs.io/s/jetbrains">JetBrains</a> and <a href="https://out.devdocs.io/s/code-school">Code School</a> for sponsoring DevDocs in the past
|
||||||
<th>Copyright/License
|
<li><a href="https://www.heroku.com">Heroku</a> and <a href="https://newrelic.com/">New Relic</a> for providing awesome free service
|
||||||
<th>Source code
|
<li><a href="https://www.jeremykratz.com/">Jeremy Kratz</a> for the C/C++ logo
|
||||||
#{(
|
</ul>
|
||||||
"<tr>
|
|
||||||
<td><a href=\"#{doc.links?.home}\">#{doc.name}</a></td>
|
|
||||||
<td>#{doc.attribution}</td>
|
|
||||||
<td><a href=\"#{doc.links?.code}\">Source code</a></td>
|
|
||||||
</tr>" for doc in docs
|
|
||||||
).join('')}
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h2 class="_block-heading" id="privacy">Privacy Policy</h2>
|
<div class="_table">
|
||||||
<ul>
|
<table class="_credits">
|
||||||
<li><a href="https://devdocs.io">devdocs.io</a> ("App") is operated by <a href="https://www.freecodecamp.org/">freeCodeCamp</a> ("We").
|
<tr>
|
||||||
<li>We do not collect personal information through the app.
|
<th>Documentation
|
||||||
<li>We use Google Analytics and Gauges to collect anonymous traffic information if you have given consent to this. You can change your decision in the <a href="/settings">settings</a>.
|
<th>Copyright/License
|
||||||
<li>We use Sentry to collect crash data and improve the app.
|
<th>Source code
|
||||||
<li>The app uses cookies to store user preferences.
|
${((() => {
|
||||||
<li>By using the app, you signify your acceptance of this policy. If you do not agree to this policy, please do not use the app.
|
const result = [];
|
||||||
<li>If you have any questions regarding privacy, please email <a href="mailto:privacy@freecodecamp.org">privacy@freecodecamp.org</a>.
|
|
||||||
</ul>
|
for (doc of Array.from(docs)) { result.push(`<tr> \
|
||||||
"""
|
<td><a href=\"${(doc.links != null ? doc.links.home : undefined)}\">${doc.name}</a></td> \
|
||||||
|
<td>${doc.attribution}</td> \
|
||||||
|
<td><a href=\"${(doc.links != null ? doc.links.code : undefined)}\">Source code</a></td> \
|
||||||
|
</tr>`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
})()).join('')}
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h2 class="_block-heading" id="privacy">Privacy Policy</h2>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://devdocs.io">devdocs.io</a> ("App") is operated by <a href="https://www.freecodecamp.org/">freeCodeCamp</a> ("We").
|
||||||
|
<li>We do not collect personal information through the app.
|
||||||
|
<li>We use Google Analytics and Gauges to collect anonymous traffic information if you have given consent to this. You can change your decision in the <a href="/settings">settings</a>.
|
||||||
|
<li>We use Sentry to collect crash data and improve the app.
|
||||||
|
<li>The app uses cookies to store user preferences.
|
||||||
|
<li>By using the app, you signify your acceptance of this policy. If you do not agree to this policy, please do not use the app.
|
||||||
|
<li>If you have any questions regarding privacy, please email <a href="mailto:privacy@freecodecamp.org">privacy@freecodecamp.org</a>.
|
||||||
|
</ul>\
|
||||||
|
`;
|
||||||
|
};
|
||||||
|
@ -1,169 +1,193 @@
|
|||||||
app.templates.helpPage = ->
|
/*
|
||||||
ctrlKey = if $.isMac() then 'cmd' else 'ctrl'
|
* decaffeinate suggestions:
|
||||||
navKey = if $.isMac() then 'cmd' else 'alt'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
arrowScroll = app.settings.get('arrowScroll')
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.templates.helpPage = function() {
|
||||||
|
let key, value;
|
||||||
|
const ctrlKey = $.isMac() ? 'cmd' : 'ctrl';
|
||||||
|
const navKey = $.isMac() ? 'cmd' : 'alt';
|
||||||
|
const arrowScroll = app.settings.get('arrowScroll');
|
||||||
|
|
||||||
aliases_one = {}
|
const aliases_one = {};
|
||||||
aliases_two = {}
|
const aliases_two = {};
|
||||||
keys = Object.keys(app.models.Entry.ALIASES)
|
const keys = Object.keys(app.models.Entry.ALIASES);
|
||||||
middle = Math.ceil(keys.length / 2) - 1
|
const middle = Math.ceil(keys.length / 2) - 1;
|
||||||
for key, i in keys
|
for (let i = 0; i < keys.length; i++) {
|
||||||
(if i > middle then aliases_two else aliases_one)[key] = app.models.Entry.ALIASES[key]
|
key = keys[i];
|
||||||
|
(i > middle ? aliases_two : aliases_one)[key] = app.models.Entry.ALIASES[key];
|
||||||
|
}
|
||||||
|
|
||||||
"""
|
return `\
|
||||||
<nav class="_toc" role="directory">
|
<nav class="_toc" role="directory">
|
||||||
<h3 class="_toc-title">Table of Contents</h3>
|
<h3 class="_toc-title">Table of Contents</h3>
|
||||||
<ul class="_toc-list">
|
<ul class="_toc-list">
|
||||||
<li><a href="#managing-documentations">Managing Documentations</a>
|
<li><a href="#managing-documentations">Managing Documentations</a>
|
||||||
<li><a href="#search">Search</a>
|
<li><a href="#search">Search</a>
|
||||||
<li><a href="#shortcuts">Keyboard Shortcuts</a>
|
<li><a href="#shortcuts">Keyboard Shortcuts</a>
|
||||||
<li><a href="#aliases">Search Aliases</a>
|
<li><a href="#aliases">Search Aliases</a>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
<h1 class="_lined-heading">User Guide</h1>
|
<h1 class="_lined-heading">User Guide</h1>
|
||||||
|
|
||||||
<h2 class="_block-heading" id="managing-documentations">Managing Documentations</h2>
|
<h2 class="_block-heading" id="managing-documentations">Managing Documentations</h2>
|
||||||
<p>
|
<p>
|
||||||
Documentations can be enabled and disabled in the <a href="/settings">Preferences</a>.
|
Documentations can be enabled and disabled in the <a href="/settings">Preferences</a>.
|
||||||
Alternatively, you can enable a documentation by searching for it in the main search
|
Alternatively, you can enable a documentation by searching for it in the main search
|
||||||
and clicking the "Enable" link in the results.
|
and clicking the "Enable" link in the results.
|
||||||
For faster and better search, only enable the documentations you plan on actively using.
|
For faster and better search, only enable the documentations you plan on actively using.
|
||||||
<p>
|
<p>
|
||||||
Once a documentation is enabled, it becomes part of the search and its content can be downloaded for offline access — and faster page loads when online — in the <a href="/offline">Offline</a> area.
|
Once a documentation is enabled, it becomes part of the search and its content can be downloaded for offline access — and faster page loads when online — in the <a href="/offline">Offline</a> area.
|
||||||
|
|
||||||
<h2 class="_block-heading" id="search">Search</h2>
|
<h2 class="_block-heading" id="search">Search</h2>
|
||||||
<p>
|
<p>
|
||||||
The search is case-insensitive and ignores whitespace. It supports fuzzy matching
|
The search is case-insensitive and ignores whitespace. It supports fuzzy matching
|
||||||
(e.g. <code class="_label">bgcp</code> matches <code class="_label">background-clip</code>)
|
(e.g. <code class="_label">bgcp</code> matches <code class="_label">background-clip</code>)
|
||||||
as well as aliases (full list <a href="#aliases">below</a>).
|
as well as aliases (full list <a href="#aliases">below</a>).
|
||||||
<dl>
|
<dl>
|
||||||
<dt id="doc_search">Searching a single documentation
|
<dt id="doc_search">Searching a single documentation
|
||||||
<dd>
|
<dd>
|
||||||
The search can be scoped to a single documentation by typing its name (or an abbreviation)
|
The search can be scoped to a single documentation by typing its name (or an abbreviation)
|
||||||
and pressing <code class="_label">tab</code> (<code class="_label">space</code> on mobile).
|
and pressing <code class="_label">tab</code> (<code class="_label">space</code> on mobile).
|
||||||
For example, to search the JavaScript documentation, enter <code class="_label">javascript</code>
|
For example, to search the JavaScript documentation, enter <code class="_label">javascript</code>
|
||||||
or <code class="_label">js</code>, then <code class="_label">tab</code>.<br>
|
or <code class="_label">js</code>, then <code class="_label">tab</code>.<br>
|
||||||
To clear the current scope, empty the search field and hit <code class="_label">backspace</code> or
|
To clear the current scope, empty the search field and hit <code class="_label">backspace</code> or
|
||||||
<code class="_label">esc</code>.
|
<code class="_label">esc</code>.
|
||||||
<dt id="url_search">Prefilling the search field
|
<dt id="url_search">Prefilling the search field
|
||||||
<dd>
|
<dd>
|
||||||
The search can be prefilled from the URL by visiting <a href="/#q=keyword" target="_top">devdocs.io/#q=keyword</a>.
|
The search can be prefilled from the URL by visiting <a href="/#q=keyword" target="_top">devdocs.io/#q=keyword</a>.
|
||||||
Characters after <code class="_label">#q=</code> will be used as search query.<br>
|
Characters after <code class="_label">#q=</code> will be used as search query.<br>
|
||||||
To search a single documentation, add its name (or an abbreviation) and a space before the keyword:
|
To search a single documentation, add its name (or an abbreviation) and a space before the keyword:
|
||||||
<a href="/#q=js%20date" target="_top">devdocs.io/#q=js date</a>.
|
<a href="/#q=js%20date" target="_top">devdocs.io/#q=js date</a>.
|
||||||
<dt id="browser_search">Searching using the address bar
|
<dt id="browser_search">Searching using the address bar
|
||||||
<dd>
|
<dd>
|
||||||
DevDocs supports OpenSearch. It can easily be installed as a search engine on most web browsers:
|
DevDocs supports OpenSearch. It can easily be installed as a search engine on most web browsers:
|
||||||
<ul>
|
<ul>
|
||||||
<li>On Chrome, the setup is done automatically. Simply press <code class="_label">tab</code> when devdocs.io is autocompleted
|
<li>On Chrome, the setup is done automatically. Simply press <code class="_label">tab</code> when devdocs.io is autocompleted
|
||||||
in the omnibox (to set a custom keyword, click <em>Manage search engines\u2026</em> in Chrome's settings).
|
in the omnibox (to set a custom keyword, click <em>Manage search engines\u2026</em> in Chrome's settings).
|
||||||
<li>On Firefox, <a href="https://support.mozilla.org/en-US/kb/add-or-remove-search-engine-firefox#w_add-a-search-engine-from-the-address-bar">add the search from the address bar</a>:
|
<li>On Firefox, <a href="https://support.mozilla.org/en-US/kb/add-or-remove-search-engine-firefox#w_add-a-search-engine-from-the-address-bar">add the search from the address bar</a>:
|
||||||
Click ••• in the address bar, and select <em>Add Search Engine</em>. Then, you can add a keyword for this search engine in the preferences.
|
Click ••• in the address bar, and select <em>Add Search Engine</em>. Then, you can add a keyword for this search engine in the preferences.
|
||||||
</dl>
|
</dl>
|
||||||
<p>
|
<p>
|
||||||
<i>Note: the above search features only work for documentations that are enabled.</i>
|
<i>Note: the above search features only work for documentations that are enabled.</i>
|
||||||
|
|
||||||
<h2 class="_block-heading" id="shortcuts">Keyboard Shortcuts</h2>
|
<h2 class="_block-heading" id="shortcuts">Keyboard Shortcuts</h2>
|
||||||
<h3 class="_shortcuts-title">Sidebar</h3>
|
<h3 class="_shortcuts-title">Sidebar</h3>
|
||||||
<dl class="_shortcuts-dl">
|
<dl class="_shortcuts-dl">
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
#{if arrowScroll then '<code class="_shortcut-code">shift</code> + ' else ''}
|
${arrowScroll ? '<code class="_shortcut-code">shift</code> + ' : ''}
|
||||||
<code class="_shortcut-code">↓</code>
|
<code class="_shortcut-code">↓</code>
|
||||||
<code class="_shortcut-code">↑</code>
|
<code class="_shortcut-code">↑</code>
|
||||||
<dd class="_shortcuts-dd">Move selection
|
<dd class="_shortcuts-dd">Move selection
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
#{if arrowScroll then '<code class="_shortcut-code">shift</code> + ' else ''}
|
${arrowScroll ? '<code class="_shortcut-code">shift</code> + ' : ''}
|
||||||
<code class="_shortcut-code">→</code>
|
<code class="_shortcut-code">→</code>
|
||||||
<code class="_shortcut-code">←</code>
|
<code class="_shortcut-code">←</code>
|
||||||
<dd class="_shortcuts-dd">Show/hide sub-list
|
<dd class="_shortcuts-dd">Show/hide sub-list
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">enter</code>
|
<code class="_shortcut-code">enter</code>
|
||||||
<dd class="_shortcuts-dd">Open selection
|
<dd class="_shortcuts-dd">Open selection
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">#{ctrlKey} + enter</code>
|
<code class="_shortcut-code">${ctrlKey} + enter</code>
|
||||||
<dd class="_shortcuts-dd">Open selection in a new tab
|
<dd class="_shortcuts-dd">Open selection in a new tab
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + r</code>
|
<code class="_shortcut-code">alt + r</code>
|
||||||
<dd class="_shortcuts-dd">Reveal current page in sidebar
|
<dd class="_shortcuts-dd">Reveal current page in sidebar
|
||||||
</dl>
|
</dl>
|
||||||
<h3 class="_shortcuts-title">Browsing</h3>
|
<h3 class="_shortcuts-title">Browsing</h3>
|
||||||
<dl class="_shortcuts-dl">
|
<dl class="_shortcuts-dl">
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">#{navKey} + ←</code>
|
<code class="_shortcut-code">${navKey} + ←</code>
|
||||||
<code class="_shortcut-code">#{navKey} + →</code>
|
<code class="_shortcut-code">${navKey} + →</code>
|
||||||
<dd class="_shortcuts-dd">Go back/forward
|
<dd class="_shortcuts-dd">Go back/forward
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
#{if arrowScroll
|
${arrowScroll ?
|
||||||
'<code class="_shortcut-code">↓</code> ' +
|
'<code class="_shortcut-code">↓</code> ' +
|
||||||
'<code class="_shortcut-code">↑</code>'
|
'<code class="_shortcut-code">↑</code>'
|
||||||
else
|
:
|
||||||
'<code class="_shortcut-code">alt + ↓</code> ' +
|
'<code class="_shortcut-code">alt + ↓</code> ' +
|
||||||
'<code class="_shortcut-code">alt + ↑</code>' +
|
'<code class="_shortcut-code">alt + ↑</code>' +
|
||||||
'<br>' +
|
'<br>' +
|
||||||
'<code class="_shortcut-code">shift + ↓</code> ' +
|
'<code class="_shortcut-code">shift + ↓</code> ' +
|
||||||
'<code class="_shortcut-code">shift + ↑</code>'}
|
'<code class="_shortcut-code">shift + ↑</code>'}
|
||||||
<dd class="_shortcuts-dd">Scroll step by step<br><br>
|
<dd class="_shortcuts-dd">Scroll step by step<br><br>
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">space</code>
|
<code class="_shortcut-code">space</code>
|
||||||
<code class="_shortcut-code">shift + space</code>
|
<code class="_shortcut-code">shift + space</code>
|
||||||
<dd class="_shortcuts-dd">Scroll screen by screen
|
<dd class="_shortcuts-dd">Scroll screen by screen
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">#{ctrlKey} + ↑</code>
|
<code class="_shortcut-code">${ctrlKey} + ↑</code>
|
||||||
<code class="_shortcut-code">#{ctrlKey} + ↓</code>
|
<code class="_shortcut-code">${ctrlKey} + ↓</code>
|
||||||
<dd class="_shortcuts-dd">Scroll to the top/bottom
|
<dd class="_shortcuts-dd">Scroll to the top/bottom
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + f</code>
|
<code class="_shortcut-code">alt + f</code>
|
||||||
<dd class="_shortcuts-dd">Focus first link in the content area<br>(press tab to focus the other links)
|
<dd class="_shortcuts-dd">Focus first link in the content area<br>(press tab to focus the other links)
|
||||||
</dl>
|
</dl>
|
||||||
<h3 class="_shortcuts-title">App</h3>
|
<h3 class="_shortcuts-title">App</h3>
|
||||||
<dl class="_shortcuts-dl">
|
<dl class="_shortcuts-dl">
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">ctrl + ,</code>
|
<code class="_shortcut-code">ctrl + ,</code>
|
||||||
<dd class="_shortcuts-dd">Open preferences
|
<dd class="_shortcuts-dd">Open preferences
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">esc</code>
|
<code class="_shortcut-code">esc</code>
|
||||||
<dd class="_shortcuts-dd">Clear search field / reset UI
|
<dd class="_shortcuts-dd">Clear search field / reset UI
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">?</code>
|
<code class="_shortcut-code">?</code>
|
||||||
<dd class="_shortcuts-dd">Show this page
|
<dd class="_shortcuts-dd">Show this page
|
||||||
</dl>
|
</dl>
|
||||||
<h3 class="_shortcuts-title">Miscellaneous</h3>
|
<h3 class="_shortcuts-title">Miscellaneous</h3>
|
||||||
<dl class="_shortcuts-dl">
|
<dl class="_shortcuts-dl">
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + c</code>
|
<code class="_shortcut-code">alt + c</code>
|
||||||
<dd class="_shortcuts-dd">Copy URL of original page
|
<dd class="_shortcuts-dd">Copy URL of original page
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + o</code>
|
<code class="_shortcut-code">alt + o</code>
|
||||||
<dd class="_shortcuts-dd">Open original page
|
<dd class="_shortcuts-dd">Open original page
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + g</code>
|
<code class="_shortcut-code">alt + g</code>
|
||||||
<dd class="_shortcuts-dd">Search on Google
|
<dd class="_shortcuts-dd">Search on Google
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + s</code>
|
<code class="_shortcut-code">alt + s</code>
|
||||||
<dd class="_shortcuts-dd">Search on Stack Overflow
|
<dd class="_shortcuts-dd">Search on Stack Overflow
|
||||||
<dt class="_shortcuts-dt">
|
<dt class="_shortcuts-dt">
|
||||||
<code class="_shortcut-code">alt + d</code>
|
<code class="_shortcut-code">alt + d</code>
|
||||||
<dd class="_shortcuts-dd">Search on DuckDuckGo
|
<dd class="_shortcuts-dd">Search on DuckDuckGo
|
||||||
</dl>
|
</dl>
|
||||||
<p class="_note _note-green">
|
<p class="_note _note-green">
|
||||||
<strong>Tip:</strong> If the cursor is no longer in the search field, press <code class="_label">/</code> or
|
<strong>Tip:</strong> If the cursor is no longer in the search field, press <code class="_label">/</code> or
|
||||||
continue to type and it will refocus the search field and start showing new results.
|
continue to type and it will refocus the search field and start showing new results.
|
||||||
|
|
||||||
<h2 class="_block-heading" id="aliases">Search Aliases</h2>
|
<h2 class="_block-heading" id="aliases">Search Aliases</h2>
|
||||||
<div class="_aliases">
|
<div class="_aliases">
|
||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<th>Word
|
<th>Word
|
||||||
<th>Alias
|
<th>Alias
|
||||||
#{("<tr><td class=\"_code\">#{key}<td class=\"_code\">#{value}" for key, value of aliases_one).join('')}
|
${((() => {
|
||||||
</table>
|
const result = [];
|
||||||
<table>
|
for (key in aliases_one) {
|
||||||
<tr>
|
value = aliases_one[key];
|
||||||
<th>Word
|
result.push(`<tr><td class=\"_code\">${key}<td class=\"_code\">${value}`);
|
||||||
<th>Alias
|
}
|
||||||
#{("<tr><td class=\"_code\">#{key}<td class=\"_code\">#{value}" for key, value of aliases_two).join('')}
|
return result;
|
||||||
</table>
|
})()).join('')}
|
||||||
</div>
|
</table>
|
||||||
<p>Feel free to suggest new aliases on <a href="https://github.com/freeCodeCamp/devdocs/issues/new">GitHub</a>.
|
<table>
|
||||||
"""
|
<tr>
|
||||||
|
<th>Word
|
||||||
|
<th>Alias
|
||||||
|
${((() => {
|
||||||
|
const result1 = [];
|
||||||
|
for (key in aliases_two) {
|
||||||
|
value = aliases_two[key];
|
||||||
|
result1.push(`<tr><td class=\"_code\">${key}<td class=\"_code\">${value}`);
|
||||||
|
}
|
||||||
|
return result1;
|
||||||
|
})()).join('')}
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
<p>Feel free to suggest new aliases on <a href="https://github.com/freeCodeCamp/devdocs/issues/new">GitHub</a>.\
|
||||||
|
`;
|
||||||
|
};
|
||||||
|
@ -1,80 +1,89 @@
|
|||||||
app.templates.offlinePage = (docs) -> """
|
/*
|
||||||
<h1 class="_lined-heading">Offline Documentation</h1>
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.templates.offlinePage = docs => `\
|
||||||
|
<h1 class="_lined-heading">Offline Documentation</h1>
|
||||||
|
|
||||||
<div class="_docs-tools">
|
<div class="_docs-tools">
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" name="autoUpdate" value="1" #{if app.settings.get('manualUpdate') then '' else 'checked'}>Install updates automatically
|
<input type="checkbox" name="autoUpdate" value="1" ${app.settings.get('manualUpdate') ? '' : 'checked'}>Install updates automatically
|
||||||
</label>
|
</label>
|
||||||
<div class="_docs-links">
|
<div class="_docs-links">
|
||||||
<button type="button" class="_btn-link" data-action-all="install">Install all</button><button type="button" class="_btn-link" data-action-all="update"><strong>Update all</strong></button><button type="button" class="_btn-link" data-action-all="uninstall">Uninstall all</button>
|
<button type="button" class="_btn-link" data-action-all="install">Install all</button><button type="button" class="_btn-link" data-action-all="update"><strong>Update all</strong></button><button type="button" class="_btn-link" data-action-all="uninstall">Uninstall all</button>
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="_table">
|
<div class="_table">
|
||||||
<table class="_docs">
|
<table class="_docs">
|
||||||
<tr>
|
<tr>
|
||||||
<th>Documentation</th>
|
<th>Documentation</th>
|
||||||
<th class="_docs-size">Size</th>
|
<th class="_docs-size">Size</th>
|
||||||
<th>Status</th>
|
<th>Status</th>
|
||||||
<th>Action</th>
|
<th>Action</th>
|
||||||
</tr>
|
</tr>
|
||||||
#{docs}
|
${docs}
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
<p class="_note"><strong>Note:</strong> your browser may delete DevDocs's offline data if your computer is running low on disk space and you haven't used the app in a while. Load this page before going offline to make sure the data is still there.
|
<p class="_note"><strong>Note:</strong> your browser may delete DevDocs's offline data if your computer is running low on disk space and you haven't used the app in a while. Load this page before going offline to make sure the data is still there.
|
||||||
<h2 class="_block-heading">Questions & Answers</h2>
|
<h2 class="_block-heading">Questions & Answers</h2>
|
||||||
<dl>
|
<dl>
|
||||||
<dt>How does this work?
|
<dt>How does this work?
|
||||||
<dd>Each page is cached as a key-value pair in <a href="https://devdocs.io/dom/indexeddb_api">IndexedDB</a> (downloaded from a single file).<br>
|
<dd>Each page is cached as a key-value pair in <a href="https://devdocs.io/dom/indexeddb_api">IndexedDB</a> (downloaded from a single file).<br>
|
||||||
The app also uses <a href="https://devdocs.io/dom/service_worker_api/using_service_workers">Service Workers</a> and <a href="https://devdocs.io/dom/web_storage_api">localStorage</a> to cache the assets and index files.
|
The app also uses <a href="https://devdocs.io/dom/service_worker_api/using_service_workers">Service Workers</a> and <a href="https://devdocs.io/dom/web_storage_api">localStorage</a> to cache the assets and index files.
|
||||||
<dt>Can I close the tab/browser?
|
<dt>Can I close the tab/browser?
|
||||||
<dd>#{canICloseTheTab()}
|
<dd>${canICloseTheTab()}
|
||||||
<dt>What if I don't update a documentation?
|
<dt>What if I don't update a documentation?
|
||||||
<dd>You'll see outdated content and some pages will be missing or broken, because the rest of the app (including data for the search and sidebar) uses a different caching mechanism that's updated automatically.
|
<dd>You'll see outdated content and some pages will be missing or broken, because the rest of the app (including data for the search and sidebar) uses a different caching mechanism that's updated automatically.
|
||||||
<dt>I found a bug, where do I report it?
|
<dt>I found a bug, where do I report it?
|
||||||
<dd>In the <a href="https://github.com/freeCodeCamp/devdocs/issues">issue tracker</a>. Thanks!
|
<dd>In the <a href="https://github.com/freeCodeCamp/devdocs/issues">issue tracker</a>. Thanks!
|
||||||
<dt>How do I uninstall/reset the app?
|
<dt>How do I uninstall/reset the app?
|
||||||
<dd>Click <a href="#" data-behavior="reset">here</a>.
|
<dd>Click <a href="#" data-behavior="reset">here</a>.
|
||||||
<dt>Why aren't all documentations listed above?
|
<dt>Why aren't all documentations listed above?
|
||||||
<dd>You have to <a href="/settings">enable</a> them first.
|
<dd>You have to <a href="/settings">enable</a> them first.
|
||||||
</dl>
|
</dl>\
|
||||||
"""
|
`;
|
||||||
|
|
||||||
canICloseTheTab = ->
|
var canICloseTheTab = function() {
|
||||||
if app.ServiceWorker.isEnabled()
|
if (app.ServiceWorker.isEnabled()) {
|
||||||
""" Yes! Even offline, you can open a new tab, go to <a href="//devdocs.io">devdocs.io</a>, and everything will work as if you were online (provided you installed all the documentations you want to use beforehand). """
|
return " Yes! Even offline, you can open a new tab, go to <a href=\"//devdocs.io\">devdocs.io</a>, and everything will work as if you were online (provided you installed all the documentations you want to use beforehand). ";
|
||||||
else
|
} else {
|
||||||
reason = "aren't available in your browser (or are disabled)"
|
let reason = "aren't available in your browser (or are disabled)";
|
||||||
|
|
||||||
if app.config.env != 'production'
|
if (app.config.env !== 'production') {
|
||||||
reason = "are disabled in your development instance of DevDocs (enable them by setting the <code>ENABLE_SERVICE_WORKER</code> environment variable to <code>true</code>)"
|
reason = "are disabled in your development instance of DevDocs (enable them by setting the <code>ENABLE_SERVICE_WORKER</code> environment variable to <code>true</code>)";
|
||||||
|
}
|
||||||
|
|
||||||
""" No. Service Workers #{reason}, so loading <a href="//devdocs.io">devdocs.io</a> offline won't work.<br>
|
return ` No. Service Workers ${reason}, so loading <a href="//devdocs.io">devdocs.io</a> offline won't work.<br>
|
||||||
The current tab will continue to function even when you go offline (provided you installed all the documentations beforehand). """
|
The current tab will continue to function even when you go offline (provided you installed all the documentations beforehand). `;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
app.templates.offlineDoc = (doc, status) ->
|
app.templates.offlineDoc = function(doc, status) {
|
||||||
outdated = doc.isOutdated(status)
|
const outdated = doc.isOutdated(status);
|
||||||
|
|
||||||
html = """
|
let html = `\
|
||||||
<tr data-slug="#{doc.slug}"#{if outdated then ' class="_highlight"' else ''}>
|
<tr data-slug="${doc.slug}"${outdated ? ' class="_highlight"' : ''}>
|
||||||
<td class="_docs-name _icon-#{doc.icon}">#{doc.fullName}</td>
|
<td class="_docs-name _icon-${doc.icon}">${doc.fullName}</td>
|
||||||
<td class="_docs-size">#{Math.ceil(doc.db_size / 100000) / 10} <small>MB</small></td>
|
<td class="_docs-size">${Math.ceil(doc.db_size / 100000) / 10} <small>MB</small></td>\
|
||||||
"""
|
`;
|
||||||
|
|
||||||
html += if !(status and status.installed)
|
html += !(status && status.installed) ?
|
||||||
"""
|
`\
|
||||||
<td>-</td>
|
<td>-</td>
|
||||||
<td><button type="button" class="_btn-link" data-action="install">Install</button></td>
|
<td><button type="button" class="_btn-link" data-action="install">Install</button></td>\
|
||||||
"""
|
`
|
||||||
else if outdated
|
: outdated ?
|
||||||
"""
|
`\
|
||||||
<td><strong>Outdated</strong></td>
|
<td><strong>Outdated</strong></td>
|
||||||
<td><button type="button" class="_btn-link _bold" data-action="update">Update</button> - <button type="button" class="_btn-link" data-action="uninstall">Uninstall</button></td>
|
<td><button type="button" class="_btn-link _bold" data-action="update">Update</button> - <button type="button" class="_btn-link" data-action="uninstall">Uninstall</button></td>\
|
||||||
"""
|
`
|
||||||
else
|
:
|
||||||
"""
|
`\
|
||||||
<td>Up‑to‑date</td>
|
<td>Up‑to‑date</td>
|
||||||
<td><button type="button" class="_btn-link" data-action="uninstall">Uninstall</button></td>
|
<td><button type="button" class="_btn-link" data-action="uninstall">Uninstall</button></td>\
|
||||||
"""
|
`;
|
||||||
|
|
||||||
html + '</tr>'
|
return html + '</tr>';
|
||||||
|
};
|
||||||
|
@ -1,81 +1,86 @@
|
|||||||
themeOption = ({ label, value }, settings) -> """
|
/*
|
||||||
<label class="_settings-label _theme-label">
|
* decaffeinate suggestions:
|
||||||
<input type="radio" name="theme" value="#{value}"#{if settings.theme == value then ' checked' else ''}>
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
#{label}
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
</label>
|
*/
|
||||||
"""
|
const themeOption = ({ label, value }, settings) => `\
|
||||||
|
<label class="_settings-label _theme-label">
|
||||||
|
<input type="radio" name="theme" value="${value}"${settings.theme === value ? ' checked' : ''}>
|
||||||
|
${label}
|
||||||
|
</label>\
|
||||||
|
`;
|
||||||
|
|
||||||
app.templates.settingsPage = (settings) -> """
|
app.templates.settingsPage = settings => `\
|
||||||
<h1 class="_lined-heading">Preferences</h1>
|
<h1 class="_lined-heading">Preferences</h1>
|
||||||
|
|
||||||
<div class="_settings-fieldset">
|
<div class="_settings-fieldset">
|
||||||
<h2 class="_settings-legend">Theme:</h2>
|
<h2 class="_settings-legend">Theme:</h2>
|
||||||
<div class="_settings-inputs">
|
<div class="_settings-inputs">
|
||||||
#{if settings.autoSupported
|
${settings.autoSupported ?
|
||||||
themeOption label: "Automatic <small>Matches system setting</small>", value: "auto", settings
|
themeOption({label: "Automatic <small>Matches system setting</small>", value: "auto"}, settings)
|
||||||
else
|
:
|
||||||
""}
|
""}
|
||||||
#{themeOption label: "Light", value: "default", settings}
|
${themeOption({label: "Light", value: "default"}, settings)}
|
||||||
#{themeOption label: "Dark", value: "dark", settings}
|
${themeOption({label: "Dark", value: "dark"}, settings)}
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="_settings-fieldset">
|
<div class="_settings-fieldset">
|
||||||
<h2 class="_settings-legend">General:</h2>
|
<h2 class="_settings-legend">General:</h2>
|
||||||
|
|
||||||
<div class="_settings-inputs">
|
<div class="_settings-inputs">
|
||||||
<label class="_settings-label _setting-max-width">
|
<label class="_settings-label _setting-max-width">
|
||||||
<input type="checkbox" form="settings" name="layout" value="_max-width"#{if settings['_max-width'] then ' checked' else ''}>Enable fixed-width layout
|
<input type="checkbox" form="settings" name="layout" value="_max-width"${settings['_max-width'] ? ' checked' : ''}>Enable fixed-width layout
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label _setting-text-justify-hyphenate">
|
<label class="_settings-label _setting-text-justify-hyphenate">
|
||||||
<input type="checkbox" form="settings" name="layout" value="_text-justify-hyphenate"#{if settings['_text-justify-hyphenate'] then ' checked' else ''}>Enable justified layout and automatic hyphenation
|
<input type="checkbox" form="settings" name="layout" value="_text-justify-hyphenate"${settings['_text-justify-hyphenate'] ? ' checked' : ''}>Enable justified layout and automatic hyphenation
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label _hide-on-mobile">
|
<label class="_settings-label _hide-on-mobile">
|
||||||
<input type="checkbox" form="settings" name="layout" value="_sidebar-hidden"#{if settings['_sidebar-hidden'] then ' checked' else ''}>Automatically hide and show the sidebar
|
<input type="checkbox" form="settings" name="layout" value="_sidebar-hidden"${settings['_sidebar-hidden'] ? ' checked' : ''}>Automatically hide and show the sidebar
|
||||||
<small>Tip: drag the edge of the sidebar to resize it.</small>
|
<small>Tip: drag the edge of the sidebar to resize it.</small>
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label _hide-on-mobile">
|
<label class="_settings-label _hide-on-mobile">
|
||||||
<input type="checkbox" form="settings" name="noAutofocus" value="_no-autofocus"#{if settings.noAutofocus then ' checked' else ''}>Disable autofocus of search input
|
<input type="checkbox" form="settings" name="noAutofocus" value="_no-autofocus"${settings.noAutofocus ? ' checked' : ''}>Disable autofocus of search input
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label">
|
<label class="_settings-label">
|
||||||
<input type="checkbox" form="settings" name="autoInstall" value="_auto-install"#{if settings.autoInstall then ' checked' else ''}>Automatically download documentation for offline use
|
<input type="checkbox" form="settings" name="autoInstall" value="_auto-install"${settings.autoInstall ? ' checked' : ''}>Automatically download documentation for offline use
|
||||||
<small>Only enable this when bandwidth isn't a concern to you.</small>
|
<small>Only enable this when bandwidth isn't a concern to you.</small>
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label _hide-in-development">
|
<label class="_settings-label _hide-in-development">
|
||||||
<input type="checkbox" form="settings" name="analyticsConsent"#{if settings.analyticsConsent then ' checked' else ''}>Enable tracking cookies
|
<input type="checkbox" form="settings" name="analyticsConsent"${settings.analyticsConsent ? ' checked' : ''}>Enable tracking cookies
|
||||||
<small>With this checked, we enable Google Analytics and Gauges to collect anonymous traffic information.</small>
|
<small>With this checked, we enable Google Analytics and Gauges to collect anonymous traffic information.</small>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="_settings-fieldset _hide-on-mobile">
|
<div class="_settings-fieldset _hide-on-mobile">
|
||||||
<h2 class="_settings-legend">Scrolling:</h2>
|
<h2 class="_settings-legend">Scrolling:</h2>
|
||||||
|
|
||||||
<div class="_settings-inputs">
|
<div class="_settings-inputs">
|
||||||
<label class="_settings-label">
|
<label class="_settings-label">
|
||||||
<input type="checkbox" form="settings" name="smoothScroll" value="1"#{if settings.smoothScroll then ' checked' else ''}>Use smooth scrolling
|
<input type="checkbox" form="settings" name="smoothScroll" value="1"${settings.smoothScroll ? ' checked' : ''}>Use smooth scrolling
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label _setting-native-scrollbar">
|
<label class="_settings-label _setting-native-scrollbar">
|
||||||
<input type="checkbox" form="settings" name="layout" value="_native-scrollbars"#{if settings['_native-scrollbars'] then ' checked' else ''}>Use native scrollbars
|
<input type="checkbox" form="settings" name="layout" value="_native-scrollbars"${settings['_native-scrollbars'] ? ' checked' : ''}>Use native scrollbars
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label">
|
<label class="_settings-label">
|
||||||
<input type="checkbox" form="settings" name="arrowScroll" value="1"#{if settings.arrowScroll then ' checked' else ''}>Use arrow keys to scroll the main content area
|
<input type="checkbox" form="settings" name="arrowScroll" value="1"${settings.arrowScroll ? ' checked' : ''}>Use arrow keys to scroll the main content area
|
||||||
<small>With this checked, use <code class="_label">shift</code> + <code class="_label">↑</code><code class="_label">↓</code><code class="_label">←</code><code class="_label">→</code> to navigate the sidebar.</small>
|
<small>With this checked, use <code class="_label">shift</code> + <code class="_label">↑</code><code class="_label">↓</code><code class="_label">←</code><code class="_label">→</code> to navigate the sidebar.</small>
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label">
|
<label class="_settings-label">
|
||||||
<input type="checkbox" form="settings" name="spaceScroll" value="1"#{if settings.spaceScroll then ' checked' else ''}>Use spacebar to scroll during search
|
<input type="checkbox" form="settings" name="spaceScroll" value="1"${settings.spaceScroll ? ' checked' : ''}>Use spacebar to scroll during search
|
||||||
</label>
|
</label>
|
||||||
<label class="_settings-label">
|
<label class="_settings-label">
|
||||||
<input type="number" step="0.1" form="settings" name="spaceTimeout" min="0" max="5" value="#{settings.spaceTimeout}"> Delay until you can scroll by pressing space
|
<input type="number" step="0.1" form="settings" name="spaceTimeout" min="0" max="5" value="${settings.spaceTimeout}"> Delay until you can scroll by pressing space
|
||||||
<small>Time in seconds</small>
|
<small>Time in seconds</small>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<p class="_hide-on-mobile">
|
<p class="_hide-on-mobile">
|
||||||
<button type="button" class="_btn" data-action="export">Export</button>
|
<button type="button" class="_btn" data-action="export">Export</button>
|
||||||
<label class="_btn _file-btn"><input type="file" form="settings" name="import" accept=".json">Import</label>
|
<label class="_btn _file-btn"><input type="file" form="settings" name="import" accept=".json">Import</label>
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
<button type="button" class="_btn-link _reset-btn" data-behavior="reset">Reset all preferences and data</button>
|
<button type="button" class="_btn-link _reset-btn" data-behavior="reset">Reset all preferences and data</button>\
|
||||||
"""
|
`;
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
app.templates.typePage = (type) ->
|
/*
|
||||||
""" <h1>#{type.doc.fullName} / #{type.name}</h1>
|
* decaffeinate suggestions:
|
||||||
<ul class="_entry-list">#{app.templates.render 'typePageEntry', type.entries()}</ul> """
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.templates.typePage = type => ` <h1>${type.doc.fullName} / ${type.name}</h1>
|
||||||
|
<ul class="_entry-list">${app.templates.render('typePageEntry', type.entries())}</ul> `;
|
||||||
|
|
||||||
app.templates.typePageEntry = (entry) ->
|
app.templates.typePageEntry = entry => `<li><a href="${entry.fullPath()}">${$.escape(entry.name)}</a></li>`;
|
||||||
"""<li><a href="#{entry.fullPath()}">#{$.escape entry.name}</a></li>"""
|
|
||||||
|
@ -1,7 +1,13 @@
|
|||||||
arrow = """<svg class="_path-arrow"><use xlink:href="#icon-dir"/></svg>"""
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const arrow = "<svg class=\"_path-arrow\"><use xlink:href=\"#icon-dir\"/></svg>";
|
||||||
|
|
||||||
app.templates.path = (doc, type, entry) ->
|
app.templates.path = function(doc, type, entry) {
|
||||||
html = """<a href="#{doc.fullPath()}" class="_path-item _icon-#{doc.icon}">#{doc.fullName}</a>"""
|
let html = `<a href="${doc.fullPath()}" class="_path-item _icon-${doc.icon}">${doc.fullName}</a>`;
|
||||||
html += """#{arrow}<a href="#{type.fullPath()}" class="_path-item">#{type.name}</a>""" if type
|
if (type) { html += `${arrow}<a href="${type.fullPath()}" class="_path-item">${type.name}</a>`; }
|
||||||
html += """#{arrow}<span class="_path-item">#{$.escape entry.name}</span>""" if entry
|
if (entry) { html += `${arrow}<span class="_path-item">${$.escape(entry.name)}</span>`; }
|
||||||
html
|
return html;
|
||||||
|
};
|
||||||
|
@ -1,68 +1,79 @@
|
|||||||
templates = app.templates
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const {
|
||||||
|
templates
|
||||||
|
} = app;
|
||||||
|
|
||||||
arrow = """<svg class="_list-arrow"><use xlink:href="#icon-dir"/></svg>"""
|
const arrow = "<svg class=\"_list-arrow\"><use xlink:href=\"#icon-dir\"/></svg>";
|
||||||
|
|
||||||
templates.sidebarDoc = (doc, options = {}) ->
|
templates.sidebarDoc = function(doc, options) {
|
||||||
link = """<a href="#{doc.fullPath()}" class="_list-item _icon-#{doc.icon} """
|
if (options == null) { options = {}; }
|
||||||
link += if options.disabled then '_list-disabled' else '_list-dir'
|
let link = `<a href="${doc.fullPath()}" class="_list-item _icon-${doc.icon} `;
|
||||||
link += """" data-slug="#{doc.slug}" title="#{doc.fullName}" tabindex="-1">"""
|
link += options.disabled ? '_list-disabled' : '_list-dir';
|
||||||
if options.disabled
|
link += `" data-slug="${doc.slug}" title="${doc.fullName}" tabindex="-1">`;
|
||||||
link += """<span class="_list-enable" data-enable="#{doc.slug}">Enable</span>"""
|
if (options.disabled) {
|
||||||
else
|
link += `<span class="_list-enable" data-enable="${doc.slug}">Enable</span>`;
|
||||||
link += arrow
|
} else {
|
||||||
link += """<span class="_list-count">#{doc.release}</span>""" if doc.release
|
link += arrow;
|
||||||
link += """<span class="_list-text">#{doc.name}"""
|
}
|
||||||
link += " #{doc.version}" if options.fullName or options.disabled and doc.version
|
if (doc.release) { link += `<span class="_list-count">${doc.release}</span>`; }
|
||||||
link + "</span></a>"
|
link += `<span class="_list-text">${doc.name}`;
|
||||||
|
if (options.fullName || (options.disabled && doc.version)) { link += ` ${doc.version}`; }
|
||||||
|
return link + "</span></a>";
|
||||||
|
};
|
||||||
|
|
||||||
templates.sidebarType = (type) ->
|
templates.sidebarType = type => `<a href="${type.fullPath()}" class="_list-item _list-dir" data-slug="${type.slug}" tabindex="-1">${arrow}<span class="_list-count">${type.count}</span><span class="_list-text">${$.escape(type.name)}</span></a>`;
|
||||||
"""<a href="#{type.fullPath()}" class="_list-item _list-dir" data-slug="#{type.slug}" tabindex="-1">#{arrow}<span class="_list-count">#{type.count}</span><span class="_list-text">#{$.escape type.name}</span></a>"""
|
|
||||||
|
|
||||||
templates.sidebarEntry = (entry) ->
|
templates.sidebarEntry = entry => `<a href="${entry.fullPath()}" class="_list-item _list-hover" tabindex="-1">${$.escape(entry.name)}</a>`;
|
||||||
"""<a href="#{entry.fullPath()}" class="_list-item _list-hover" tabindex="-1">#{$.escape entry.name}</a>"""
|
|
||||||
|
|
||||||
templates.sidebarResult = (entry) ->
|
templates.sidebarResult = function(entry) {
|
||||||
addons = if entry.isIndex() and app.disabledDocs.contains(entry.doc)
|
let addons = entry.isIndex() && app.disabledDocs.contains(entry.doc) ?
|
||||||
"""<span class="_list-enable" data-enable="#{entry.doc.slug}">Enable</span>"""
|
`<span class="_list-enable" data-enable="${entry.doc.slug}">Enable</span>`
|
||||||
else
|
:
|
||||||
"""<span class="_list-reveal" data-reset-list title="Reveal in list"></span>"""
|
"<span class=\"_list-reveal\" data-reset-list title=\"Reveal in list\"></span>";
|
||||||
addons += """<span class="_list-count">#{entry.doc.short_version}</span>""" if entry.doc.version and not entry.isIndex()
|
if (entry.doc.version && !entry.isIndex()) { addons += `<span class="_list-count">${entry.doc.short_version}</span>`; }
|
||||||
"""<a href="#{entry.fullPath()}" class="_list-item _list-hover _list-result _icon-#{entry.doc.icon}" tabindex="-1">#{addons}<span class="_list-text">#{$.escape entry.name}</span></a>"""
|
return `<a href="${entry.fullPath()}" class="_list-item _list-hover _list-result _icon-${entry.doc.icon}" tabindex="-1">${addons}<span class="_list-text">${$.escape(entry.name)}</span></a>`;
|
||||||
|
};
|
||||||
|
|
||||||
templates.sidebarNoResults = ->
|
templates.sidebarNoResults = function() {
|
||||||
html = """ <div class="_list-note">No results.</div> """
|
let html = " <div class=\"_list-note\">No results.</div> ";
|
||||||
html += """
|
if (!app.isSingleDoc() && !app.disabledDocs.isEmpty()) { html += `\
|
||||||
<div class="_list-note">Note: documentations must be <a href="/settings" class="_list-note-link">enabled</a> to appear in the search.</div>
|
<div class="_list-note">Note: documentations must be <a href="/settings" class="_list-note-link">enabled</a> to appear in the search.</div>\
|
||||||
""" unless app.isSingleDoc() or app.disabledDocs.isEmpty()
|
`; }
|
||||||
html
|
return html;
|
||||||
|
};
|
||||||
|
|
||||||
templates.sidebarPageLink = (count) ->
|
templates.sidebarPageLink = count => `<span role="link" class="_list-item _list-pagelink">Show more\u2026 (${count})</span>`;
|
||||||
"""<span role="link" class="_list-item _list-pagelink">Show more\u2026 (#{count})</span>"""
|
|
||||||
|
|
||||||
templates.sidebarLabel = (doc, options = {}) ->
|
templates.sidebarLabel = function(doc, options) {
|
||||||
label = """<label class="_list-item"""
|
if (options == null) { options = {}; }
|
||||||
label += " _icon-#{doc.icon}" unless doc.version
|
let label = "<label class=\"_list-item";
|
||||||
label += """"><input type="checkbox" name="#{doc.slug}" class="_list-checkbox" """
|
if (!doc.version) { label += ` _icon-${doc.icon}`; }
|
||||||
label += "checked" if options.checked
|
label += `"><input type="checkbox" name="${doc.slug}" class="_list-checkbox" `;
|
||||||
label + """><span class="_list-text">#{doc.fullName}</span></label>"""
|
if (options.checked) { label += "checked"; }
|
||||||
|
return label + `><span class="_list-text">${doc.fullName}</span></label>`;
|
||||||
|
};
|
||||||
|
|
||||||
templates.sidebarVersionedDoc = (doc, versions, options = {}) ->
|
templates.sidebarVersionedDoc = function(doc, versions, options) {
|
||||||
html = """<div class="_list-item _list-dir _list-rdir _icon-#{doc.icon}"""
|
if (options == null) { options = {}; }
|
||||||
html += " open" if options.open
|
let html = `<div class="_list-item _list-dir _list-rdir _icon-${doc.icon}`;
|
||||||
html + """" tabindex="0">#{arrow}#{doc.name}</div><div class="_list _list-sub">#{versions}</div>"""
|
if (options.open) { html += " open"; }
|
||||||
|
return html + `" tabindex="0">${arrow}${doc.name}</div><div class="_list _list-sub">${versions}</div>`;
|
||||||
|
};
|
||||||
|
|
||||||
templates.sidebarDisabled = (options) ->
|
templates.sidebarDisabled = options => `<h6 class="_list-title">${arrow}Disabled (${options.count}) <a href="/settings" class="_list-title-link" tabindex="-1">Customize</a></h6>`;
|
||||||
"""<h6 class="_list-title">#{arrow}Disabled (#{options.count}) <a href="/settings" class="_list-title-link" tabindex="-1">Customize</a></h6>"""
|
|
||||||
|
|
||||||
templates.sidebarDisabledList = (html) ->
|
templates.sidebarDisabledList = html => `<div class="_disabled-list">${html}</div>`;
|
||||||
"""<div class="_disabled-list">#{html}</div>"""
|
|
||||||
|
|
||||||
templates.sidebarDisabledVersionedDoc = (doc, versions) ->
|
templates.sidebarDisabledVersionedDoc = (doc, versions) => `<a class="_list-item _list-dir _icon-${doc.icon} _list-disabled" data-slug="${doc.slug_without_version}" tabindex="-1">${arrow}${doc.name}</a><div class="_list _list-sub">${versions}</div>`;
|
||||||
"""<a class="_list-item _list-dir _icon-#{doc.icon} _list-disabled" data-slug="#{doc.slug_without_version}" tabindex="-1">#{arrow}#{doc.name}</a><div class="_list _list-sub">#{versions}</div>"""
|
|
||||||
|
|
||||||
templates.docPickerHeader = """<div class="_list-picker-head"><span>Documentation</span> <span>Enable</span></div>"""
|
templates.docPickerHeader = "<div class=\"_list-picker-head\"><span>Documentation</span> <span>Enable</span></div>";
|
||||||
|
|
||||||
templates.docPickerNote = """
|
templates.docPickerNote = `\
|
||||||
<div class="_list-note">Tip: for faster and better search results, select only the docs you need.</div>
|
<div class="_list-note">Tip: for faster and better search results, select only the docs you need.</div>
|
||||||
<a href="https://trello.com/b/6BmTulfx/devdocs-documentation" class="_list-link" target="_blank" rel="noopener">Vote for new documentation</a>
|
<a href="https://trello.com/b/6BmTulfx/devdocs-documentation" class="_list-link" target="_blank" rel="noopener">Vote for new documentation</a>\
|
||||||
"""
|
`;
|
||||||
|
@ -1,10 +1,15 @@
|
|||||||
app.templates.tipKeyNav = () -> """
|
/*
|
||||||
<p class="_notif-text">
|
* decaffeinate suggestions:
|
||||||
<strong>ProTip</strong>
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
<span class="_notif-info">(click to dismiss)</span>
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
<p class="_notif-text">
|
*/
|
||||||
Hit #{if app.settings.get('arrowScroll') then '<code class="_label">shift</code> +' else ''} <code class="_label">↓</code> <code class="_label">↑</code> <code class="_label">←</code> <code class="_label">→</code> to navigate the sidebar.<br>
|
app.templates.tipKeyNav = () => `\
|
||||||
Hit <code class="_label">space / shift space</code>#{if app.settings.get('arrowScroll') then ' or <code class="_label">↓/↑</code>' else ', <code class="_label">alt ↓/↑</code> or <code class="_label">shift ↓/↑</code>'} to scroll the page.
|
<p class="_notif-text">
|
||||||
<p class="_notif-text">
|
<strong>ProTip</strong>
|
||||||
<a href="/help#shortcuts" class="_notif-link">See all keyboard shortcuts</a>
|
<span class="_notif-info">(click to dismiss)</span>
|
||||||
"""
|
<p class="_notif-text">
|
||||||
|
Hit ${app.settings.get('arrowScroll') ? '<code class="_label">shift</code> +' : ''} <code class="_label">↓</code> <code class="_label">↑</code> <code class="_label">←</code> <code class="_label">→</code> to navigate the sidebar.<br>
|
||||||
|
Hit <code class="_label">space / shift space</code>${app.settings.get('arrowScroll') ? ' or <code class="_label">↓/↑</code>' : ', <code class="_label">alt ↓/↑</code> or <code class="_label">shift ↓/↑</code>'} to scroll the page.
|
||||||
|
<p class="_notif-text">
|
||||||
|
<a href="/help#shortcuts" class="_notif-link">See all keyboard shortcuts</a>\
|
||||||
|
`;
|
||||||
|
@ -1,195 +1,259 @@
|
|||||||
class app.views.Content extends app.View
|
/*
|
||||||
@el: '._content'
|
* decaffeinate suggestions:
|
||||||
@loadingClass: '_content-loading'
|
* DS002: Fix invalid constructor
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
@events:
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
click: 'onClick'
|
* DS104: Avoid inline assignments
|
||||||
|
* DS204: Change includes calls to have a more natural evaluation order
|
||||||
@shortcuts:
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
altUp: 'scrollStepUp'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
altDown: 'scrollStepDown'
|
* DS207: Consider shorter variations of null checks
|
||||||
pageUp: 'scrollPageUp'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
pageDown: 'scrollPageDown'
|
*/
|
||||||
pageTop: 'scrollToTop'
|
const Cls = (app.views.Content = class Content extends app.View {
|
||||||
pageBottom: 'scrollToBottom'
|
constructor(...args) {
|
||||||
altF: 'onAltF'
|
this.scrollToTop = this.scrollToTop.bind(this);
|
||||||
|
this.scrollToBottom = this.scrollToBottom.bind(this);
|
||||||
@routes:
|
this.scrollStepUp = this.scrollStepUp.bind(this);
|
||||||
before: 'beforeRoute'
|
this.scrollStepDown = this.scrollStepDown.bind(this);
|
||||||
after: 'afterRoute'
|
this.scrollPageUp = this.scrollPageUp.bind(this);
|
||||||
|
this.scrollPageDown = this.scrollPageDown.bind(this);
|
||||||
init: ->
|
this.onReady = this.onReady.bind(this);
|
||||||
@scrollEl = if app.isMobile()
|
this.onBootError = this.onBootError.bind(this);
|
||||||
|
this.onEntryLoading = this.onEntryLoading.bind(this);
|
||||||
|
this.onEntryLoaded = this.onEntryLoaded.bind(this);
|
||||||
|
this.beforeRoute = this.beforeRoute.bind(this);
|
||||||
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
|
this.onClick = this.onClick.bind(this);
|
||||||
|
this.onAltF = this.onAltF.bind(this);
|
||||||
|
super(...args);
|
||||||
|
}
|
||||||
|
|
||||||
|
static initClass() {
|
||||||
|
this.el = '._content';
|
||||||
|
this.loadingClass = '_content-loading';
|
||||||
|
|
||||||
|
this.events =
|
||||||
|
{click: 'onClick'};
|
||||||
|
|
||||||
|
this.shortcuts = {
|
||||||
|
altUp: 'scrollStepUp',
|
||||||
|
altDown: 'scrollStepDown',
|
||||||
|
pageUp: 'scrollPageUp',
|
||||||
|
pageDown: 'scrollPageDown',
|
||||||
|
pageTop: 'scrollToTop',
|
||||||
|
pageBottom: 'scrollToBottom',
|
||||||
|
altF: 'onAltF'
|
||||||
|
};
|
||||||
|
|
||||||
|
this.routes = {
|
||||||
|
before: 'beforeRoute',
|
||||||
|
after: 'afterRoute'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
this.scrollEl = app.isMobile() ?
|
||||||
(document.scrollingElement || document.body)
|
(document.scrollingElement || document.body)
|
||||||
else
|
:
|
||||||
@el
|
this.el;
|
||||||
@scrollMap = {}
|
this.scrollMap = {};
|
||||||
@scrollStack = []
|
this.scrollStack = [];
|
||||||
|
|
||||||
@rootPage = new app.views.RootPage
|
this.rootPage = new app.views.RootPage;
|
||||||
@staticPage = new app.views.StaticPage
|
this.staticPage = new app.views.StaticPage;
|
||||||
@settingsPage = new app.views.SettingsPage
|
this.settingsPage = new app.views.SettingsPage;
|
||||||
@offlinePage = new app.views.OfflinePage
|
this.offlinePage = new app.views.OfflinePage;
|
||||||
@typePage = new app.views.TypePage
|
this.typePage = new app.views.TypePage;
|
||||||
@entryPage = new app.views.EntryPage
|
this.entryPage = new app.views.EntryPage;
|
||||||
|
|
||||||
@entryPage
|
this.entryPage
|
||||||
.on 'loading', @onEntryLoading
|
.on('loading', this.onEntryLoading)
|
||||||
.on 'loaded', @onEntryLoaded
|
.on('loaded', this.onEntryLoaded);
|
||||||
|
|
||||||
app
|
app
|
||||||
.on 'ready', @onReady
|
.on('ready', this.onReady)
|
||||||
.on 'bootError', @onBootError
|
.on('bootError', this.onBootError);
|
||||||
|
|
||||||
return
|
}
|
||||||
|
|
||||||
show: (view) ->
|
show(view) {
|
||||||
@hideLoading()
|
this.hideLoading();
|
||||||
unless view is @view
|
if (view !== this.view) {
|
||||||
@view?.deactivate()
|
if (this.view != null) {
|
||||||
@html @view = view
|
this.view.deactivate();
|
||||||
@view.activate()
|
}
|
||||||
return
|
this.html(this.view = view);
|
||||||
|
this.view.activate();
|
||||||
showLoading: ->
|
}
|
||||||
@addClass @constructor.loadingClass
|
}
|
||||||
return
|
|
||||||
|
showLoading() {
|
||||||
isLoading: ->
|
this.addClass(this.constructor.loadingClass);
|
||||||
@el.classList.contains @constructor.loadingClass
|
}
|
||||||
|
|
||||||
hideLoading: ->
|
isLoading() {
|
||||||
@removeClass @constructor.loadingClass
|
return this.el.classList.contains(this.constructor.loadingClass);
|
||||||
return
|
}
|
||||||
|
|
||||||
scrollTo: (value) ->
|
hideLoading() {
|
||||||
@scrollEl.scrollTop = value or 0
|
this.removeClass(this.constructor.loadingClass);
|
||||||
return
|
}
|
||||||
|
|
||||||
smoothScrollTo: (value) ->
|
scrollTo(value) {
|
||||||
if app.settings.get('fastScroll')
|
this.scrollEl.scrollTop = value || 0;
|
||||||
@scrollTo value
|
}
|
||||||
else
|
|
||||||
$.smoothScroll @scrollEl, value or 0
|
smoothScrollTo(value) {
|
||||||
return
|
if (app.settings.get('fastScroll')) {
|
||||||
|
this.scrollTo(value);
|
||||||
scrollBy: (n) ->
|
} else {
|
||||||
@smoothScrollTo @scrollEl.scrollTop + n
|
$.smoothScroll(this.scrollEl, value || 0);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
scrollToTop: =>
|
|
||||||
@smoothScrollTo 0
|
scrollBy(n) {
|
||||||
return
|
this.smoothScrollTo(this.scrollEl.scrollTop + n);
|
||||||
|
}
|
||||||
scrollToBottom: =>
|
|
||||||
@smoothScrollTo @scrollEl.scrollHeight
|
scrollToTop() {
|
||||||
return
|
this.smoothScrollTo(0);
|
||||||
|
}
|
||||||
scrollStepUp: =>
|
|
||||||
@scrollBy -80
|
scrollToBottom() {
|
||||||
return
|
this.smoothScrollTo(this.scrollEl.scrollHeight);
|
||||||
|
}
|
||||||
scrollStepDown: =>
|
|
||||||
@scrollBy 80
|
scrollStepUp() {
|
||||||
return
|
this.scrollBy(-80);
|
||||||
|
}
|
||||||
scrollPageUp: =>
|
|
||||||
@scrollBy 40 - @scrollEl.clientHeight
|
scrollStepDown() {
|
||||||
return
|
this.scrollBy(80);
|
||||||
|
}
|
||||||
scrollPageDown: =>
|
|
||||||
@scrollBy @scrollEl.clientHeight - 40
|
scrollPageUp() {
|
||||||
return
|
this.scrollBy(40 - this.scrollEl.clientHeight);
|
||||||
|
}
|
||||||
scrollToTarget: ->
|
|
||||||
if @routeCtx.hash and el = @findTargetByHash @routeCtx.hash
|
scrollPageDown() {
|
||||||
$.scrollToWithImageLock el, @scrollEl, 'top',
|
this.scrollBy(this.scrollEl.clientHeight - 40);
|
||||||
margin: if @scrollEl is @el then 0 else $.offset(@el).top
|
}
|
||||||
$.highlight el, className: '_highlight'
|
|
||||||
else
|
scrollToTarget() {
|
||||||
@scrollTo @scrollMap[@routeCtx.state.id]
|
let el;
|
||||||
return
|
if (this.routeCtx.hash && (el = this.findTargetByHash(this.routeCtx.hash))) {
|
||||||
|
$.scrollToWithImageLock(el, this.scrollEl, 'top',
|
||||||
onReady: =>
|
{margin: this.scrollEl === this.el ? 0 : $.offset(this.el).top});
|
||||||
@hideLoading()
|
$.highlight(el, {className: '_highlight'});
|
||||||
return
|
} else {
|
||||||
|
this.scrollTo(this.scrollMap[this.routeCtx.state.id]);
|
||||||
onBootError: =>
|
}
|
||||||
@hideLoading()
|
}
|
||||||
@html @tmpl('bootError')
|
|
||||||
return
|
onReady() {
|
||||||
|
this.hideLoading();
|
||||||
onEntryLoading: =>
|
}
|
||||||
@showLoading()
|
|
||||||
if @scrollToTargetTimeout
|
onBootError() {
|
||||||
clearTimeout @scrollToTargetTimeout
|
this.hideLoading();
|
||||||
@scrollToTargetTimeout = null
|
this.html(this.tmpl('bootError'));
|
||||||
return
|
}
|
||||||
|
|
||||||
onEntryLoaded: =>
|
onEntryLoading() {
|
||||||
@hideLoading()
|
this.showLoading();
|
||||||
if @scrollToTargetTimeout
|
if (this.scrollToTargetTimeout) {
|
||||||
clearTimeout @scrollToTargetTimeout
|
clearTimeout(this.scrollToTargetTimeout);
|
||||||
@scrollToTargetTimeout = null
|
this.scrollToTargetTimeout = null;
|
||||||
@scrollToTarget()
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
beforeRoute: (context) =>
|
onEntryLoaded() {
|
||||||
@cacheScrollPosition()
|
this.hideLoading();
|
||||||
@routeCtx = context
|
if (this.scrollToTargetTimeout) {
|
||||||
@scrollToTargetTimeout = @delay @scrollToTarget
|
clearTimeout(this.scrollToTargetTimeout);
|
||||||
return
|
this.scrollToTargetTimeout = null;
|
||||||
|
}
|
||||||
cacheScrollPosition: ->
|
this.scrollToTarget();
|
||||||
return if not @routeCtx or @routeCtx.hash
|
}
|
||||||
return if @routeCtx.path is '/'
|
|
||||||
|
beforeRoute(context) {
|
||||||
unless @scrollMap[@routeCtx.state.id]?
|
this.cacheScrollPosition();
|
||||||
@scrollStack.push @routeCtx.state.id
|
this.routeCtx = context;
|
||||||
while @scrollStack.length > app.config.history_cache_size
|
this.scrollToTargetTimeout = this.delay(this.scrollToTarget);
|
||||||
delete @scrollMap[@scrollStack.shift()]
|
}
|
||||||
|
|
||||||
@scrollMap[@routeCtx.state.id] = @scrollEl.scrollTop
|
cacheScrollPosition() {
|
||||||
return
|
if (!this.routeCtx || this.routeCtx.hash) { return; }
|
||||||
|
if (this.routeCtx.path === '/') { return; }
|
||||||
afterRoute: (route, context) =>
|
|
||||||
if route != 'entry' and route != 'type'
|
if (this.scrollMap[this.routeCtx.state.id] == null) {
|
||||||
resetFavicon()
|
this.scrollStack.push(this.routeCtx.state.id);
|
||||||
|
while (this.scrollStack.length > app.config.history_cache_size) {
|
||||||
switch route
|
delete this.scrollMap[this.scrollStack.shift()];
|
||||||
when 'root'
|
}
|
||||||
@show @rootPage
|
}
|
||||||
when 'entry'
|
|
||||||
@show @entryPage
|
this.scrollMap[this.routeCtx.state.id] = this.scrollEl.scrollTop;
|
||||||
when 'type'
|
}
|
||||||
@show @typePage
|
|
||||||
when 'settings'
|
afterRoute(route, context) {
|
||||||
@show @settingsPage
|
if ((route !== 'entry') && (route !== 'type')) {
|
||||||
when 'offline'
|
resetFavicon();
|
||||||
@show @offlinePage
|
}
|
||||||
else
|
|
||||||
@show @staticPage
|
switch (route) {
|
||||||
|
case 'root':
|
||||||
@view.onRoute(context)
|
this.show(this.rootPage);
|
||||||
app.document.setTitle @view.getTitle?()
|
break;
|
||||||
return
|
case 'entry':
|
||||||
|
this.show(this.entryPage);
|
||||||
onClick: (event) =>
|
break;
|
||||||
link = $.closestLink $.eventTarget(event), @el
|
case 'type':
|
||||||
if link and @isExternalUrl link.getAttribute('href')
|
this.show(this.typePage);
|
||||||
$.stopEvent(event)
|
break;
|
||||||
$.popup(link)
|
case 'settings':
|
||||||
return
|
this.show(this.settingsPage);
|
||||||
|
break;
|
||||||
onAltF: (event) =>
|
case 'offline':
|
||||||
unless document.activeElement and $.hasChild @el, document.activeElement
|
this.show(this.offlinePage);
|
||||||
@find('a:not(:empty)')?.focus()
|
break;
|
||||||
$.stopEvent(event)
|
default:
|
||||||
|
this.show(this.staticPage);
|
||||||
findTargetByHash: (hash) ->
|
}
|
||||||
el = try $.id decodeURIComponent(hash) catch
|
|
||||||
el or= try $.id(hash) catch
|
this.view.onRoute(context);
|
||||||
el
|
app.document.setTitle(typeof this.view.getTitle === 'function' ? this.view.getTitle() : undefined);
|
||||||
|
}
|
||||||
isExternalUrl: (url) ->
|
|
||||||
url?[0..5] in ['http:/', 'https:']
|
onClick(event) {
|
||||||
|
const link = $.closestLink($.eventTarget(event), this.el);
|
||||||
|
if (link && this.isExternalUrl(link.getAttribute('href'))) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
$.popup(link);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onAltF(event) {
|
||||||
|
if (!document.activeElement || !$.hasChild(this.el, document.activeElement)) {
|
||||||
|
__guard__(this.find('a:not(:empty)'), x => x.focus());
|
||||||
|
return $.stopEvent(event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
findTargetByHash(hash) {
|
||||||
|
let el = (() => { try { return $.id(decodeURIComponent(hash)); } catch (error) {} })();
|
||||||
|
if (!el) { el = (() => { try { return $.id(hash); } catch (error1) {} })(); }
|
||||||
|
return el;
|
||||||
|
}
|
||||||
|
|
||||||
|
isExternalUrl(url) {
|
||||||
|
let needle;
|
||||||
|
return (needle = __guard__(url, x => x.slice(0, 6)), ['http:/', 'https:'].includes(needle));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,166 +1,225 @@
|
|||||||
class app.views.EntryPage extends app.View
|
/*
|
||||||
@className: '_page'
|
* decaffeinate suggestions:
|
||||||
@errorClass: '_page-error'
|
* DS002: Fix invalid constructor
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
click: 'onClick'
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@shortcuts:
|
* DS207: Consider shorter variations of null checks
|
||||||
altC: 'onAltC'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
altO: 'onAltO'
|
*/
|
||||||
|
(function() {
|
||||||
@routes:
|
let LINKS = undefined;
|
||||||
before: 'beforeRoute'
|
const Cls = (app.views.EntryPage = class EntryPage extends app.View {
|
||||||
|
constructor(...args) {
|
||||||
init: ->
|
this.beforeRoute = this.beforeRoute.bind(this);
|
||||||
@cacheMap = {}
|
this.onSuccess = this.onSuccess.bind(this);
|
||||||
@cacheStack = []
|
this.onError = this.onError.bind(this);
|
||||||
return
|
this.onClick = this.onClick.bind(this);
|
||||||
|
this.onAltC = this.onAltC.bind(this);
|
||||||
deactivate: ->
|
this.onAltO = this.onAltO.bind(this);
|
||||||
if super
|
super(...args);
|
||||||
@empty()
|
}
|
||||||
@entry = null
|
|
||||||
return
|
static initClass() {
|
||||||
|
this.className = '_page';
|
||||||
loading: ->
|
this.errorClass = '_page-error';
|
||||||
@empty()
|
|
||||||
@trigger 'loading'
|
this.events =
|
||||||
return
|
{click: 'onClick'};
|
||||||
|
|
||||||
render: (content = '', fromCache = false) ->
|
this.shortcuts = {
|
||||||
return unless @activated
|
altC: 'onAltC',
|
||||||
@empty()
|
altO: 'onAltO'
|
||||||
@subview = new (@subViewClass()) @el, @entry
|
};
|
||||||
|
|
||||||
$.batchUpdate @el, =>
|
this.routes =
|
||||||
@subview.render(content, fromCache)
|
{before: 'beforeRoute'};
|
||||||
@addCopyButtons() unless fromCache
|
|
||||||
return
|
LINKS = {
|
||||||
|
home: 'Homepage',
|
||||||
if app.disabledDocs.findBy 'slug', @entry.doc.slug
|
code: 'Source code'
|
||||||
@hiddenView = new app.views.HiddenPage @el, @entry
|
};
|
||||||
|
}
|
||||||
setFaviconForDoc(@entry.doc)
|
|
||||||
@delay @polyfillMathML
|
init() {
|
||||||
@trigger 'loaded'
|
this.cacheMap = {};
|
||||||
return
|
this.cacheStack = [];
|
||||||
|
}
|
||||||
addCopyButtons: ->
|
|
||||||
unless @copyButton
|
deactivate() {
|
||||||
@copyButton = document.createElement('button')
|
if (super.deactivate(...arguments)) {
|
||||||
@copyButton.innerHTML = '<svg><use xlink:href="#icon-copy"/></svg>'
|
this.empty();
|
||||||
@copyButton.type = 'button'
|
this.entry = null;
|
||||||
@copyButton.className = '_pre-clip'
|
}
|
||||||
@copyButton.title = 'Copy to clipboard'
|
}
|
||||||
@copyButton.setAttribute 'aria-label', 'Copy to clipboard'
|
|
||||||
el.appendChild @copyButton.cloneNode(true) for el in @findAllByTag('pre')
|
loading() {
|
||||||
return
|
this.empty();
|
||||||
|
this.trigger('loading');
|
||||||
polyfillMathML: ->
|
}
|
||||||
return unless window.supportsMathML is false and !@polyfilledMathML and @findByTag('math')
|
|
||||||
@polyfilledMathML = true
|
render(content, fromCache) {
|
||||||
$.append document.head, """<link rel="stylesheet" href="#{app.config.mathml_stylesheet}">"""
|
if (content == null) { content = ''; }
|
||||||
return
|
if (fromCache == null) { fromCache = false; }
|
||||||
|
if (!this.activated) { return; }
|
||||||
LINKS =
|
this.empty();
|
||||||
home: 'Homepage'
|
this.subview = new (this.subViewClass())(this.el, this.entry);
|
||||||
code: 'Source code'
|
|
||||||
|
$.batchUpdate(this.el, () => {
|
||||||
prepareContent: (content) ->
|
this.subview.render(content, fromCache);
|
||||||
return content unless @entry.isIndex() and @entry.doc.links
|
if (!fromCache) { this.addCopyButtons(); }
|
||||||
|
});
|
||||||
links = for link, url of @entry.doc.links
|
|
||||||
"""<a href="#{url}" class="_links-link">#{LINKS[link]}</a>"""
|
if (app.disabledDocs.findBy('slug', this.entry.doc.slug)) {
|
||||||
|
this.hiddenView = new app.views.HiddenPage(this.el, this.entry);
|
||||||
"""<p class="_links">#{links.join('')}</p>#{content}"""
|
}
|
||||||
|
|
||||||
empty: ->
|
setFaviconForDoc(this.entry.doc);
|
||||||
@subview?.deactivate()
|
this.delay(this.polyfillMathML);
|
||||||
@subview = null
|
this.trigger('loaded');
|
||||||
|
}
|
||||||
@hiddenView?.deactivate()
|
|
||||||
@hiddenView = null
|
addCopyButtons() {
|
||||||
|
if (!this.copyButton) {
|
||||||
@resetClass()
|
this.copyButton = document.createElement('button');
|
||||||
super
|
this.copyButton.innerHTML = '<svg><use xlink:href="#icon-copy"/></svg>';
|
||||||
return
|
this.copyButton.type = 'button';
|
||||||
|
this.copyButton.className = '_pre-clip';
|
||||||
subViewClass: ->
|
this.copyButton.title = 'Copy to clipboard';
|
||||||
app.views["#{$.classify(@entry.doc.type)}Page"] or app.views.BasePage
|
this.copyButton.setAttribute('aria-label', 'Copy to clipboard');
|
||||||
|
}
|
||||||
getTitle: ->
|
for (var el of Array.from(this.findAllByTag('pre'))) { el.appendChild(this.copyButton.cloneNode(true)); }
|
||||||
@entry.doc.fullName + if @entry.isIndex() then ' documentation' else " / #{@entry.name}"
|
}
|
||||||
|
|
||||||
beforeRoute: =>
|
polyfillMathML() {
|
||||||
@cache()
|
if ((window.supportsMathML !== false) || !!this.polyfilledMathML || !this.findByTag('math')) { return; }
|
||||||
@abort()
|
this.polyfilledMathML = true;
|
||||||
return
|
$.append(document.head, `<link rel="stylesheet" href="${app.config.mathml_stylesheet}">`);
|
||||||
|
}
|
||||||
onRoute: (context) ->
|
|
||||||
isSameFile = context.entry.filePath() is @entry?.filePath()
|
prepareContent(content) {
|
||||||
@entry = context.entry
|
if (!this.entry.isIndex() || !this.entry.doc.links) { return content; }
|
||||||
@restore() or @load() unless isSameFile
|
|
||||||
return
|
const links = (() => {
|
||||||
|
const result = [];
|
||||||
load: ->
|
for (var link in this.entry.doc.links) {
|
||||||
@loading()
|
var url = this.entry.doc.links[link];
|
||||||
@xhr = @entry.loadFile @onSuccess, @onError
|
result.push(`<a href="${url}" class="_links-link">${LINKS[link]}</a>`);
|
||||||
return
|
}
|
||||||
|
return result;
|
||||||
abort: ->
|
})();
|
||||||
if @xhr
|
|
||||||
@xhr.abort()
|
return `<p class="_links">${links.join('')}</p>${content}`;
|
||||||
@xhr = @entry = null
|
}
|
||||||
return
|
|
||||||
|
empty() {
|
||||||
onSuccess: (response) =>
|
if (this.subview != null) {
|
||||||
return unless @activated
|
this.subview.deactivate();
|
||||||
@xhr = null
|
}
|
||||||
@render @prepareContent(response)
|
this.subview = null;
|
||||||
return
|
|
||||||
|
if (this.hiddenView != null) {
|
||||||
onError: =>
|
this.hiddenView.deactivate();
|
||||||
@xhr = null
|
}
|
||||||
@render @tmpl('pageLoadError')
|
this.hiddenView = null;
|
||||||
@resetClass()
|
|
||||||
@addClass @constructor.errorClass
|
this.resetClass();
|
||||||
app.serviceWorker?.update()
|
super.empty(...arguments);
|
||||||
return
|
}
|
||||||
|
|
||||||
cache: ->
|
subViewClass() {
|
||||||
return if @xhr or not @entry or @cacheMap[path = @entry.filePath()]
|
return app.views[`${$.classify(this.entry.doc.type)}Page`] || app.views.BasePage;
|
||||||
|
}
|
||||||
@cacheMap[path] = @el.innerHTML
|
|
||||||
@cacheStack.push(path)
|
getTitle() {
|
||||||
|
return this.entry.doc.fullName + (this.entry.isIndex() ? ' documentation' : ` / ${this.entry.name}`);
|
||||||
while @cacheStack.length > app.config.history_cache_size
|
}
|
||||||
delete @cacheMap[@cacheStack.shift()]
|
|
||||||
return
|
beforeRoute() {
|
||||||
|
this.cache();
|
||||||
restore: ->
|
this.abort();
|
||||||
if @cacheMap[path = @entry.filePath()]
|
}
|
||||||
@render @cacheMap[path], true
|
|
||||||
true
|
onRoute(context) {
|
||||||
|
const isSameFile = context.entry.filePath() === (this.entry != null ? this.entry.filePath() : undefined);
|
||||||
onClick: (event) =>
|
this.entry = context.entry;
|
||||||
target = $.eventTarget(event)
|
if (!isSameFile) { this.restore() || this.load(); }
|
||||||
if target.hasAttribute 'data-retry'
|
}
|
||||||
$.stopEvent(event)
|
|
||||||
@load()
|
load() {
|
||||||
else if target.classList.contains '_pre-clip'
|
this.loading();
|
||||||
$.stopEvent(event)
|
this.xhr = this.entry.loadFile(this.onSuccess, this.onError);
|
||||||
target.classList.add if $.copyToClipboard(target.parentNode.textContent) then '_pre-clip-success' else '_pre-clip-error'
|
}
|
||||||
setTimeout (-> target.className = '_pre-clip'), 2000
|
|
||||||
return
|
abort() {
|
||||||
|
if (this.xhr) {
|
||||||
onAltC: =>
|
this.xhr.abort();
|
||||||
return unless link = @find('._attribution:last-child ._attribution-link')
|
this.xhr = (this.entry = null);
|
||||||
console.log(link.href + location.hash)
|
}
|
||||||
navigator.clipboard.writeText(link.href + location.hash)
|
}
|
||||||
return
|
|
||||||
|
onSuccess(response) {
|
||||||
onAltO: =>
|
if (!this.activated) { return; }
|
||||||
return unless link = @find('._attribution:last-child ._attribution-link')
|
this.xhr = null;
|
||||||
@delay -> $.popup(link.href + location.hash)
|
this.render(this.prepareContent(response));
|
||||||
return
|
}
|
||||||
|
|
||||||
|
onError() {
|
||||||
|
this.xhr = null;
|
||||||
|
this.render(this.tmpl('pageLoadError'));
|
||||||
|
this.resetClass();
|
||||||
|
this.addClass(this.constructor.errorClass);
|
||||||
|
if (app.serviceWorker != null) {
|
||||||
|
app.serviceWorker.update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cache() {
|
||||||
|
let path;
|
||||||
|
if (this.xhr || !this.entry || this.cacheMap[(path = this.entry.filePath())]) { return; }
|
||||||
|
|
||||||
|
this.cacheMap[path] = this.el.innerHTML;
|
||||||
|
this.cacheStack.push(path);
|
||||||
|
|
||||||
|
while (this.cacheStack.length > app.config.history_cache_size) {
|
||||||
|
delete this.cacheMap[this.cacheStack.shift()];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
restore() {
|
||||||
|
let path;
|
||||||
|
if (this.cacheMap[(path = this.entry.filePath())]) {
|
||||||
|
this.render(this.cacheMap[path], true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
if (target.hasAttribute('data-retry')) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
this.load();
|
||||||
|
} else if (target.classList.contains('_pre-clip')) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
target.classList.add($.copyToClipboard(target.parentNode.textContent) ? '_pre-clip-success' : '_pre-clip-error');
|
||||||
|
setTimeout((() => target.className = '_pre-clip'), 2000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onAltC() {
|
||||||
|
let link;
|
||||||
|
if (!(link = this.find('._attribution:last-child ._attribution-link'))) { return; }
|
||||||
|
console.log(link.href + location.hash);
|
||||||
|
navigator.clipboard.writeText(link.href + location.hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
onAltO() {
|
||||||
|
let link;
|
||||||
|
if (!(link = this.find('._attribution:last-child ._attribution-link'))) { return; }
|
||||||
|
this.delay(() => $.popup(link.href + location.hash));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,92 +1,128 @@
|
|||||||
class app.views.OfflinePage extends app.View
|
/*
|
||||||
@className: '_static'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS101: Remove unnecessary use of Array.from
|
||||||
click: 'onClick'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
change: 'onChange'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
deactivate: ->
|
*/
|
||||||
if super
|
const Cls = (app.views.OfflinePage = class OfflinePage extends app.View {
|
||||||
@empty()
|
constructor(...args) {
|
||||||
return
|
this.onClick = this.onClick.bind(this);
|
||||||
|
super(...args);
|
||||||
render: ->
|
}
|
||||||
if app.cookieBlocked
|
|
||||||
@html @tmpl('offlineError', 'cookie_blocked')
|
static initClass() {
|
||||||
return
|
this.className = '_static';
|
||||||
|
|
||||||
app.docs.getInstallStatuses (statuses) =>
|
this.events = {
|
||||||
return unless @activated
|
click: 'onClick',
|
||||||
if statuses is false
|
change: 'onChange'
|
||||||
@html @tmpl('offlineError', app.db.reason, app.db.error)
|
};
|
||||||
else
|
}
|
||||||
html = ''
|
|
||||||
html += @renderDoc(doc, statuses[doc.slug]) for doc in app.docs.all()
|
deactivate() {
|
||||||
@html @tmpl('offlinePage', html)
|
if (super.deactivate(...arguments)) {
|
||||||
@refreshLinks()
|
this.empty();
|
||||||
return
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
renderDoc: (doc, status) ->
|
render() {
|
||||||
app.templates.render('offlineDoc', doc, status)
|
if (app.cookieBlocked) {
|
||||||
|
this.html(this.tmpl('offlineError', 'cookie_blocked'));
|
||||||
getTitle: ->
|
return;
|
||||||
'Offline'
|
}
|
||||||
|
|
||||||
refreshLinks: ->
|
app.docs.getInstallStatuses(statuses => {
|
||||||
for action in ['install', 'update', 'uninstall']
|
if (!this.activated) { return; }
|
||||||
@find("[data-action-all='#{action}']").classList[if @find("[data-action='#{action}']") then 'add' else 'remove']('_show')
|
if (statuses === false) {
|
||||||
return
|
this.html(this.tmpl('offlineError', app.db.reason, app.db.error));
|
||||||
|
} else {
|
||||||
docByEl: (el) ->
|
let html = '';
|
||||||
el = el.parentNode until slug = el.getAttribute('data-slug')
|
for (var doc of Array.from(app.docs.all())) { html += this.renderDoc(doc, statuses[doc.slug]); }
|
||||||
app.docs.findBy('slug', slug)
|
this.html(this.tmpl('offlinePage', html));
|
||||||
|
this.refreshLinks();
|
||||||
docEl: (doc) ->
|
}
|
||||||
@find("[data-slug='#{doc.slug}']")
|
});
|
||||||
|
}
|
||||||
onRoute: (context) ->
|
|
||||||
@render()
|
renderDoc(doc, status) {
|
||||||
return
|
return app.templates.render('offlineDoc', doc, status);
|
||||||
|
}
|
||||||
onClick: (event) =>
|
|
||||||
el = $.eventTarget(event)
|
getTitle() {
|
||||||
if action = el.getAttribute('data-action')
|
return 'Offline';
|
||||||
doc = @docByEl(el)
|
}
|
||||||
action = 'install' if action is 'update'
|
|
||||||
doc[action](@onInstallSuccess.bind(@, doc), @onInstallError.bind(@, doc), @onInstallProgress.bind(@, doc))
|
refreshLinks() {
|
||||||
el.parentNode.innerHTML = "#{el.textContent.replace(/e$/, '')}ing…"
|
for (var action of ['install', 'update', 'uninstall']) {
|
||||||
else if action = el.getAttribute('data-action-all') || el.parentElement.getAttribute('data-action-all')
|
this.find(`[data-action-all='${action}']`).classList[this.find(`[data-action='${action}']`) ? 'add' : 'remove']('_show');
|
||||||
return unless action isnt 'uninstall' or window.confirm('Uninstall all docs?')
|
}
|
||||||
app.db.migrate()
|
}
|
||||||
$.click(el) for el in @findAll("[data-action='#{action}']")
|
|
||||||
return
|
docByEl(el) {
|
||||||
|
let slug;
|
||||||
onInstallSuccess: (doc) ->
|
while (!(slug = el.getAttribute('data-slug'))) { el = el.parentNode; }
|
||||||
return unless @activated
|
return app.docs.findBy('slug', slug);
|
||||||
doc.getInstallStatus (status) =>
|
}
|
||||||
return unless @activated
|
|
||||||
if el = @docEl(doc)
|
docEl(doc) {
|
||||||
el.outerHTML = @renderDoc(doc, status)
|
return this.find(`[data-slug='${doc.slug}']`);
|
||||||
$.highlight el, className: '_highlight'
|
}
|
||||||
@refreshLinks()
|
|
||||||
return
|
onRoute(context) {
|
||||||
return
|
this.render();
|
||||||
|
}
|
||||||
onInstallError: (doc) ->
|
|
||||||
return unless @activated
|
onClick(event) {
|
||||||
if el = @docEl(doc)
|
let action;
|
||||||
el.lastElementChild.textContent = 'Error'
|
let el = $.eventTarget(event);
|
||||||
return
|
if (action = el.getAttribute('data-action')) {
|
||||||
|
const doc = this.docByEl(el);
|
||||||
onInstallProgress: (doc, event) ->
|
if (action === 'update') { action = 'install'; }
|
||||||
return unless @activated and event.lengthComputable
|
doc[action](this.onInstallSuccess.bind(this, doc), this.onInstallError.bind(this, doc), this.onInstallProgress.bind(this, doc));
|
||||||
if el = @docEl(doc)
|
el.parentNode.innerHTML = `${el.textContent.replace(/e$/, '')}ing…`;
|
||||||
percentage = Math.round event.loaded * 100 / event.total
|
} else if (action = el.getAttribute('data-action-all') || el.parentElement.getAttribute('data-action-all')) {
|
||||||
el.lastElementChild.textContent = el.lastElementChild.textContent.replace(/(\s.+)?$/, " (#{percentage}%)")
|
if ((action === 'uninstall') && !window.confirm('Uninstall all docs?')) { return; }
|
||||||
return
|
app.db.migrate();
|
||||||
|
for (el of Array.from(this.findAll(`[data-action='${action}']`))) { $.click(el); }
|
||||||
onChange: (event) ->
|
}
|
||||||
if event.target.name is 'autoUpdate'
|
}
|
||||||
app.settings.set 'manualUpdate', !event.target.checked
|
|
||||||
return
|
onInstallSuccess(doc) {
|
||||||
|
if (!this.activated) { return; }
|
||||||
|
doc.getInstallStatus(status => {
|
||||||
|
let el;
|
||||||
|
if (!this.activated) { return; }
|
||||||
|
if (el = this.docEl(doc)) {
|
||||||
|
el.outerHTML = this.renderDoc(doc, status);
|
||||||
|
$.highlight(el, {className: '_highlight'});
|
||||||
|
this.refreshLinks();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onInstallError(doc) {
|
||||||
|
let el;
|
||||||
|
if (!this.activated) { return; }
|
||||||
|
if (el = this.docEl(doc)) {
|
||||||
|
el.lastElementChild.textContent = 'Error';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onInstallProgress(doc, event) {
|
||||||
|
let el;
|
||||||
|
if (!this.activated || !event.lengthComputable) { return; }
|
||||||
|
if (el = this.docEl(doc)) {
|
||||||
|
const percentage = Math.round((event.loaded * 100) / event.total);
|
||||||
|
el.lastElementChild.textContent = el.lastElementChild.textContent.replace(/(\s.+)?$/, ` (${percentage}%)`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onChange(event) {
|
||||||
|
if (event.target.name === 'autoUpdate') {
|
||||||
|
app.settings.set('manualUpdate', !event.target.checked);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,43 +1,61 @@
|
|||||||
class app.views.RootPage extends app.View
|
/*
|
||||||
@events:
|
* decaffeinate suggestions:
|
||||||
click: 'onClick'
|
* DS002: Fix invalid constructor
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
init: ->
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@setHidden false unless @isHidden() # reserve space in local storage
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@render()
|
*/
|
||||||
return
|
const Cls = (app.views.RootPage = class RootPage extends app.View {
|
||||||
|
constructor(...args) {
|
||||||
render: ->
|
this.onClick = this.onClick.bind(this);
|
||||||
@empty()
|
super(...args);
|
||||||
|
}
|
||||||
tmpl = if app.isAndroidWebview()
|
|
||||||
|
static initClass() {
|
||||||
|
this.events =
|
||||||
|
{click: 'onClick'};
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
if (!this.isHidden()) { this.setHidden(false); } // reserve space in local storage
|
||||||
|
this.render();
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
this.empty();
|
||||||
|
|
||||||
|
const tmpl = app.isAndroidWebview() ?
|
||||||
'androidWarning'
|
'androidWarning'
|
||||||
else if @isHidden()
|
: this.isHidden() ?
|
||||||
'splash'
|
'splash'
|
||||||
else if app.isMobile()
|
: app.isMobile() ?
|
||||||
'mobileIntro'
|
'mobileIntro'
|
||||||
else
|
:
|
||||||
'intro'
|
'intro';
|
||||||
|
|
||||||
@append @tmpl(tmpl)
|
this.append(this.tmpl(tmpl));
|
||||||
return
|
}
|
||||||
|
|
||||||
hideIntro: ->
|
hideIntro() {
|
||||||
@setHidden true
|
this.setHidden(true);
|
||||||
@render()
|
this.render();
|
||||||
return
|
}
|
||||||
|
|
||||||
setHidden: (value) ->
|
setHidden(value) {
|
||||||
app.settings.set 'hideIntro', value
|
app.settings.set('hideIntro', value);
|
||||||
return
|
}
|
||||||
|
|
||||||
isHidden: ->
|
isHidden() {
|
||||||
app.isSingleDoc() or app.settings.get 'hideIntro'
|
return app.isSingleDoc() || app.settings.get('hideIntro');
|
||||||
|
}
|
||||||
onRoute: ->
|
|
||||||
|
onRoute() {}
|
||||||
onClick: (event) =>
|
|
||||||
if $.eventTarget(event).hasAttribute 'data-hide-intro'
|
onClick(event) {
|
||||||
$.stopEvent(event)
|
if ($.eventTarget(event).hasAttribute('data-hide-intro')) {
|
||||||
@hideIntro()
|
$.stopEvent(event);
|
||||||
return
|
this.hideIntro();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,116 +1,151 @@
|
|||||||
class app.views.SettingsPage extends app.View
|
/*
|
||||||
@className: '_static'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS101: Remove unnecessary use of Array.from
|
||||||
click: 'onClick'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
change: 'onChange'
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
render: ->
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@html @tmpl('settingsPage', @currentSettings())
|
*/
|
||||||
return
|
const Cls = (app.views.SettingsPage = class SettingsPage extends app.View {
|
||||||
|
constructor(...args) {
|
||||||
currentSettings: ->
|
this.onChange = this.onChange.bind(this);
|
||||||
settings = {}
|
this.onClick = this.onClick.bind(this);
|
||||||
settings.theme = app.settings.get('theme')
|
super(...args);
|
||||||
settings.smoothScroll = !app.settings.get('fastScroll')
|
}
|
||||||
settings.arrowScroll = app.settings.get('arrowScroll')
|
|
||||||
settings.noAutofocus = app.settings.get('noAutofocus')
|
static initClass() {
|
||||||
settings.autoInstall = app.settings.get('autoInstall')
|
this.className = '_static';
|
||||||
settings.analyticsConsent = app.settings.get('analyticsConsent')
|
|
||||||
settings.spaceScroll = app.settings.get('spaceScroll')
|
this.events = {
|
||||||
settings.spaceTimeout = app.settings.get('spaceTimeout')
|
click: 'onClick',
|
||||||
settings.autoSupported = app.settings.autoSupported
|
change: 'onChange'
|
||||||
settings[layout] = app.settings.hasLayout(layout) for layout in app.settings.LAYOUTS
|
};
|
||||||
settings
|
}
|
||||||
|
|
||||||
getTitle: ->
|
render() {
|
||||||
'Preferences'
|
this.html(this.tmpl('settingsPage', this.currentSettings()));
|
||||||
|
}
|
||||||
setTheme: (value) ->
|
|
||||||
app.settings.set('theme', value)
|
currentSettings() {
|
||||||
return
|
const settings = {};
|
||||||
|
settings.theme = app.settings.get('theme');
|
||||||
toggleLayout: (layout, enable) ->
|
settings.smoothScroll = !app.settings.get('fastScroll');
|
||||||
app.settings.setLayout(layout, enable)
|
settings.arrowScroll = app.settings.get('arrowScroll');
|
||||||
return
|
settings.noAutofocus = app.settings.get('noAutofocus');
|
||||||
|
settings.autoInstall = app.settings.get('autoInstall');
|
||||||
toggleSmoothScroll: (enable) ->
|
settings.analyticsConsent = app.settings.get('analyticsConsent');
|
||||||
app.settings.set('fastScroll', !enable)
|
settings.spaceScroll = app.settings.get('spaceScroll');
|
||||||
return
|
settings.spaceTimeout = app.settings.get('spaceTimeout');
|
||||||
|
settings.autoSupported = app.settings.autoSupported;
|
||||||
toggleAnalyticsConsent: (enable) ->
|
for (var layout of Array.from(app.settings.LAYOUTS)) { settings[layout] = app.settings.hasLayout(layout); }
|
||||||
app.settings.set('analyticsConsent', if enable then '1' else '0')
|
return settings;
|
||||||
resetAnalytics() unless enable
|
}
|
||||||
return
|
|
||||||
|
getTitle() {
|
||||||
toggleSpaceScroll: (enable) ->
|
return 'Preferences';
|
||||||
app.settings.set('spaceScroll', if enable then 1 else 0)
|
}
|
||||||
return
|
|
||||||
|
setTheme(value) {
|
||||||
setScrollTimeout: (value) ->
|
app.settings.set('theme', value);
|
||||||
app.settings.set('spaceTimeout', value)
|
}
|
||||||
|
|
||||||
toggle: (name, enable) ->
|
toggleLayout(layout, enable) {
|
||||||
app.settings.set(name, enable)
|
app.settings.setLayout(layout, enable);
|
||||||
return
|
}
|
||||||
|
|
||||||
export: ->
|
toggleSmoothScroll(enable) {
|
||||||
data = new Blob([JSON.stringify(app.settings.export())], type: 'application/json')
|
app.settings.set('fastScroll', !enable);
|
||||||
link = document.createElement('a')
|
}
|
||||||
link.href = URL.createObjectURL(data)
|
|
||||||
link.download = 'devdocs.json'
|
toggleAnalyticsConsent(enable) {
|
||||||
link.style.display = 'none'
|
app.settings.set('analyticsConsent', enable ? '1' : '0');
|
||||||
document.body.appendChild(link)
|
if (!enable) { resetAnalytics(); }
|
||||||
link.click()
|
}
|
||||||
document.body.removeChild(link)
|
|
||||||
return
|
toggleSpaceScroll(enable) {
|
||||||
|
app.settings.set('spaceScroll', enable ? 1 : 0);
|
||||||
import: (file, input) ->
|
}
|
||||||
unless file and file.type is 'application/json'
|
|
||||||
new app.views.Notif 'ImportInvalid', autoHide: false
|
setScrollTimeout(value) {
|
||||||
return
|
return app.settings.set('spaceTimeout', value);
|
||||||
|
}
|
||||||
reader = new FileReader()
|
|
||||||
reader.onloadend = ->
|
toggle(name, enable) {
|
||||||
data = try JSON.parse(reader.result)
|
app.settings.set(name, enable);
|
||||||
unless data and data.constructor is Object
|
}
|
||||||
new app.views.Notif 'ImportInvalid', autoHide: false
|
|
||||||
return
|
export() {
|
||||||
app.settings.import(data)
|
const data = new Blob([JSON.stringify(app.settings.export())], {type: 'application/json'});
|
||||||
$.trigger input.form, 'import'
|
const link = document.createElement('a');
|
||||||
return
|
link.href = URL.createObjectURL(data);
|
||||||
reader.readAsText(file)
|
link.download = 'devdocs.json';
|
||||||
return
|
link.style.display = 'none';
|
||||||
|
document.body.appendChild(link);
|
||||||
onChange: (event) =>
|
link.click();
|
||||||
input = event.target
|
document.body.removeChild(link);
|
||||||
switch input.name
|
}
|
||||||
when 'theme'
|
|
||||||
@setTheme input.value
|
import(file, input) {
|
||||||
when 'layout'
|
if (!file || (file.type !== 'application/json')) {
|
||||||
@toggleLayout input.value, input.checked
|
new app.views.Notif('ImportInvalid', {autoHide: false});
|
||||||
when 'smoothScroll'
|
return;
|
||||||
@toggleSmoothScroll input.checked
|
}
|
||||||
when 'import'
|
|
||||||
@import input.files[0], input
|
const reader = new FileReader();
|
||||||
when 'analyticsConsent'
|
reader.onloadend = function() {
|
||||||
@toggleAnalyticsConsent input.checked
|
const data = (() => { try { return JSON.parse(reader.result); } catch (error) {} })();
|
||||||
when 'spaceScroll'
|
if (!data || (data.constructor !== Object)) {
|
||||||
@toggleSpaceScroll input.checked
|
new app.views.Notif('ImportInvalid', {autoHide: false});
|
||||||
when 'spaceTimeout'
|
return;
|
||||||
@setScrollTimeout input.value
|
}
|
||||||
else
|
app.settings.import(data);
|
||||||
@toggle input.name, input.checked
|
$.trigger(input.form, 'import');
|
||||||
return
|
};
|
||||||
|
reader.readAsText(file);
|
||||||
onClick: (event) =>
|
}
|
||||||
target = $.eventTarget(event)
|
|
||||||
switch target.getAttribute('data-action')
|
onChange(event) {
|
||||||
when 'export'
|
const input = event.target;
|
||||||
$.stopEvent(event)
|
switch (input.name) {
|
||||||
@export()
|
case 'theme':
|
||||||
return
|
this.setTheme(input.value);
|
||||||
|
break;
|
||||||
onRoute: (context) ->
|
case 'layout':
|
||||||
@render()
|
this.toggleLayout(input.value, input.checked);
|
||||||
return
|
break;
|
||||||
|
case 'smoothScroll':
|
||||||
|
this.toggleSmoothScroll(input.checked);
|
||||||
|
break;
|
||||||
|
case 'import':
|
||||||
|
this.import(input.files[0], input);
|
||||||
|
break;
|
||||||
|
case 'analyticsConsent':
|
||||||
|
this.toggleAnalyticsConsent(input.checked);
|
||||||
|
break;
|
||||||
|
case 'spaceScroll':
|
||||||
|
this.toggleSpaceScroll(input.checked);
|
||||||
|
break;
|
||||||
|
case 'spaceTimeout':
|
||||||
|
this.setScrollTimeout(input.value);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
this.toggle(input.name, input.checked);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
switch (target.getAttribute('data-action')) {
|
||||||
|
case 'export':
|
||||||
|
$.stopEvent(event);
|
||||||
|
this.export();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onRoute(context) {
|
||||||
|
this.render();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,26 +1,39 @@
|
|||||||
class app.views.StaticPage extends app.View
|
/*
|
||||||
@className: '_static'
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.StaticPage = class StaticPage extends app.View {
|
||||||
|
static initClass() {
|
||||||
|
this.className = '_static';
|
||||||
|
|
||||||
@titles:
|
this.titles = {
|
||||||
about: 'About'
|
about: 'About',
|
||||||
news: 'News'
|
news: 'News',
|
||||||
help: 'User Guide'
|
help: 'User Guide',
|
||||||
notFound: '404'
|
notFound: '404'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
deactivate: ->
|
deactivate() {
|
||||||
if super
|
if (super.deactivate(...arguments)) {
|
||||||
@empty()
|
this.empty();
|
||||||
@page = null
|
this.page = null;
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
render: (page) ->
|
render(page) {
|
||||||
@page = page
|
this.page = page;
|
||||||
@html @tmpl("#{@page}Page")
|
this.html(this.tmpl(`${this.page}Page`));
|
||||||
return
|
}
|
||||||
|
|
||||||
getTitle: ->
|
getTitle() {
|
||||||
@constructor.titles[@page]
|
return this.constructor.titles[this.page];
|
||||||
|
}
|
||||||
|
|
||||||
onRoute: (context) ->
|
onRoute(context) {
|
||||||
@render context.page or 'notFound'
|
this.render(context.page || 'notFound');
|
||||||
return
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,20 +1,33 @@
|
|||||||
class app.views.TypePage extends app.View
|
/*
|
||||||
@className: '_page'
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.TypePage = class TypePage extends app.View {
|
||||||
|
static initClass() {
|
||||||
|
this.className = '_page';
|
||||||
|
}
|
||||||
|
|
||||||
deactivate: ->
|
deactivate() {
|
||||||
if super
|
if (super.deactivate(...arguments)) {
|
||||||
@empty()
|
this.empty();
|
||||||
@type = null
|
this.type = null;
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
render: (@type) ->
|
render(type) {
|
||||||
@html @tmpl('typePage', @type)
|
this.type = type;
|
||||||
setFaviconForDoc(@type.doc)
|
this.html(this.tmpl('typePage', this.type));
|
||||||
return
|
setFaviconForDoc(this.type.doc);
|
||||||
|
}
|
||||||
|
|
||||||
getTitle: ->
|
getTitle() {
|
||||||
"#{@type.doc.fullName} / #{@type.name}"
|
return `${this.type.doc.fullName} / ${this.type.name}`;
|
||||||
|
}
|
||||||
|
|
||||||
onRoute: (context) ->
|
onRoute(context) {
|
||||||
@render context.type
|
this.render(context.type);
|
||||||
return
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,85 +1,111 @@
|
|||||||
class app.views.Document extends app.View
|
/*
|
||||||
@el: document
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
visibilitychange: 'onVisibilityChange'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
@shortcuts:
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
help: 'onHelp'
|
*/
|
||||||
preferences: 'onPreferences'
|
const Cls = (app.views.Document = class Document extends app.View {
|
||||||
escape: 'onEscape'
|
constructor(...args) {
|
||||||
superLeft: 'onBack'
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
superRight: 'onForward'
|
this.onVisibilityChange = this.onVisibilityChange.bind(this);
|
||||||
|
super(...args);
|
||||||
@routes:
|
}
|
||||||
after: 'afterRoute'
|
|
||||||
|
static initClass() {
|
||||||
init: ->
|
this.el = document;
|
||||||
@addSubview @menu = new app.views.Menu,
|
|
||||||
@addSubview @sidebar = new app.views.Sidebar
|
this.events =
|
||||||
@addSubview @resizer = new app.views.Resizer if app.views.Resizer.isSupported()
|
{visibilitychange: 'onVisibilityChange'};
|
||||||
@addSubview @content = new app.views.Content
|
|
||||||
@addSubview @path = new app.views.Path unless app.isSingleDoc() or app.isMobile()
|
this.shortcuts = {
|
||||||
@settings = new app.views.Settings unless app.isSingleDoc()
|
help: 'onHelp',
|
||||||
|
preferences: 'onPreferences',
|
||||||
$.on document.body, 'click', @onClick
|
escape: 'onEscape',
|
||||||
|
superLeft: 'onBack',
|
||||||
@activate()
|
superRight: 'onForward'
|
||||||
return
|
};
|
||||||
|
|
||||||
setTitle: (title) ->
|
this.routes =
|
||||||
@el.title = if title then "#{title} — DevDocs" else 'DevDocs API Documentation'
|
{after: 'afterRoute'};
|
||||||
|
}
|
||||||
afterRoute: (route) =>
|
|
||||||
if route is 'settings'
|
init() {
|
||||||
@settings?.activate()
|
this.addSubview((this.menu = new app.views.Menu),
|
||||||
else
|
this.addSubview(this.sidebar = new app.views.Sidebar));
|
||||||
@settings?.deactivate()
|
if (app.views.Resizer.isSupported()) { this.addSubview(this.resizer = new app.views.Resizer); }
|
||||||
return
|
this.addSubview(this.content = new app.views.Content);
|
||||||
|
if (!app.isSingleDoc() && !app.isMobile()) { this.addSubview(this.path = new app.views.Path); }
|
||||||
onVisibilityChange: =>
|
if (!app.isSingleDoc()) { this.settings = new app.views.Settings; }
|
||||||
return unless @el.visibilityState is 'visible'
|
|
||||||
@delay ->
|
$.on(document.body, 'click', this.onClick);
|
||||||
location.reload() if app.isMobile() isnt app.views.Mobile.detect()
|
|
||||||
return
|
this.activate();
|
||||||
, 300
|
}
|
||||||
return
|
|
||||||
|
setTitle(title) {
|
||||||
onHelp: ->
|
return this.el.title = title ? `${title} — DevDocs` : 'DevDocs API Documentation';
|
||||||
app.router.show '/help#shortcuts'
|
}
|
||||||
return
|
|
||||||
|
afterRoute(route) {
|
||||||
onPreferences: ->
|
if (route === 'settings') {
|
||||||
app.router.show '/settings'
|
if (this.settings != null) {
|
||||||
return
|
this.settings.activate();
|
||||||
|
}
|
||||||
onEscape: ->
|
} else {
|
||||||
path = if !app.isSingleDoc() or location.pathname is app.doc.fullPath()
|
if (this.settings != null) {
|
||||||
|
this.settings.deactivate();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onVisibilityChange() {
|
||||||
|
if (this.el.visibilityState !== 'visible') { return; }
|
||||||
|
this.delay(function() {
|
||||||
|
if (app.isMobile() !== app.views.Mobile.detect()) { location.reload(); }
|
||||||
|
}
|
||||||
|
, 300);
|
||||||
|
}
|
||||||
|
|
||||||
|
onHelp() {
|
||||||
|
app.router.show('/help#shortcuts');
|
||||||
|
}
|
||||||
|
|
||||||
|
onPreferences() {
|
||||||
|
app.router.show('/settings');
|
||||||
|
}
|
||||||
|
|
||||||
|
onEscape() {
|
||||||
|
const path = !app.isSingleDoc() || (location.pathname === app.doc.fullPath()) ?
|
||||||
'/'
|
'/'
|
||||||
else
|
:
|
||||||
app.doc.fullPath()
|
app.doc.fullPath();
|
||||||
|
|
||||||
app.router.show(path)
|
app.router.show(path);
|
||||||
return
|
}
|
||||||
|
|
||||||
onBack: ->
|
onBack() {
|
||||||
history.back()
|
history.back();
|
||||||
return
|
}
|
||||||
|
|
||||||
onForward: ->
|
onForward() {
|
||||||
history.forward()
|
history.forward();
|
||||||
return
|
}
|
||||||
|
|
||||||
onClick: (event) ->
|
onClick(event) {
|
||||||
target = $.eventTarget(event)
|
const target = $.eventTarget(event);
|
||||||
return unless target.hasAttribute('data-behavior')
|
if (!target.hasAttribute('data-behavior')) { return; }
|
||||||
$.stopEvent(event)
|
$.stopEvent(event);
|
||||||
switch target.getAttribute('data-behavior')
|
switch (target.getAttribute('data-behavior')) {
|
||||||
when 'back' then history.back()
|
case 'back': history.back(); break;
|
||||||
when 'reload' then window.location.reload()
|
case 'reload': window.location.reload(); break;
|
||||||
when 'reboot' then app.reboot()
|
case 'reboot': app.reboot(); break;
|
||||||
when 'hard-reload' then app.reload()
|
case 'hard-reload': app.reload(); break;
|
||||||
when 'reset' then app.reset() if confirm('Are you sure you want to reset DevDocs?')
|
case 'reset': if (confirm('Are you sure you want to reset DevDocs?')) { app.reset(); } break;
|
||||||
when 'accept-analytics' then Cookies.set('analyticsConsent', '1', expires: 1e8) && app.reboot()
|
case 'accept-analytics': Cookies.set('analyticsConsent', '1', {expires: 1e8}) && app.reboot(); break;
|
||||||
when 'decline-analytics' then Cookies.set('analyticsConsent', '0', expires: 1e8) && app.reboot()
|
case 'decline-analytics': Cookies.set('analyticsConsent', '0', {expires: 1e8}) && app.reboot(); break;
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,23 +1,39 @@
|
|||||||
class app.views.Menu extends app.View
|
/*
|
||||||
@el: '._menu'
|
* decaffeinate suggestions:
|
||||||
@activeClass: 'active'
|
* DS002: Fix invalid constructor
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.Menu = class Menu extends app.View {
|
||||||
|
constructor(...args) {
|
||||||
|
this.onGlobalClick = this.onGlobalClick.bind(this);
|
||||||
|
super(...args);
|
||||||
|
}
|
||||||
|
|
||||||
@events:
|
static initClass() {
|
||||||
click: 'onClick'
|
this.el = '._menu';
|
||||||
|
this.activeClass = 'active';
|
||||||
|
|
||||||
init: ->
|
this.events =
|
||||||
$.on document.body, 'click', @onGlobalClick
|
{click: 'onClick'};
|
||||||
return
|
}
|
||||||
|
|
||||||
onClick: (event) ->
|
init() {
|
||||||
target = $.eventTarget(event)
|
$.on(document.body, 'click', this.onGlobalClick);
|
||||||
target.blur() if target.tagName is 'A'
|
}
|
||||||
return
|
|
||||||
|
|
||||||
onGlobalClick: (event) =>
|
onClick(event) {
|
||||||
return if event.which isnt 1
|
const target = $.eventTarget(event);
|
||||||
if event.target.hasAttribute?('data-toggle-menu')
|
if (target.tagName === 'A') { target.blur(); }
|
||||||
@toggleClass @constructor.activeClass
|
}
|
||||||
else if @hasClass @constructor.activeClass
|
|
||||||
@removeClass @constructor.activeClass
|
onGlobalClick(event) {
|
||||||
return
|
if (event.which !== 1) { return; }
|
||||||
|
if (typeof event.target.hasAttribute === 'function' ? event.target.hasAttribute('data-toggle-menu') : undefined) {
|
||||||
|
this.toggleClass(this.constructor.activeClass);
|
||||||
|
} else if (this.hasClass(this.constructor.activeClass)) {
|
||||||
|
this.removeClass(this.constructor.activeClass);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,155 +1,195 @@
|
|||||||
class app.views.Mobile extends app.View
|
/*
|
||||||
@className: '_mobile'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@elements:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
body: 'body'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
content: '._container'
|
* DS207: Consider shorter variations of null checks
|
||||||
sidebar: '._sidebar'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
docPicker: '._settings ._sidebar'
|
*/
|
||||||
|
const Cls = (app.views.Mobile = class Mobile extends app.View {
|
||||||
@shortcuts:
|
static initClass() {
|
||||||
escape: 'onEscape'
|
this.className = '_mobile';
|
||||||
|
|
||||||
@routes:
|
this.elements = {
|
||||||
after: 'afterRoute'
|
body: 'body',
|
||||||
|
content: '._container',
|
||||||
@detect: ->
|
sidebar: '._sidebar',
|
||||||
if Cookies.get('override-mobile-detect')?
|
docPicker: '._settings ._sidebar'
|
||||||
return JSON.parse Cookies.get('override-mobile-detect')
|
};
|
||||||
try
|
|
||||||
(window.matchMedia('(max-width: 480px)').matches) or
|
this.shortcuts =
|
||||||
(window.matchMedia('(max-width: 767px)').matches) or
|
{escape: 'onEscape'};
|
||||||
(window.matchMedia('(max-height: 767px) and (max-width: 1024px)').matches) or
|
|
||||||
# Need to sniff the user agent because some Android and Windows Phone devices don't take
|
this.routes =
|
||||||
# resolution (dpi) into account when reporting device width/height.
|
{after: 'afterRoute'};
|
||||||
(navigator.userAgent.indexOf('Android') isnt -1 and navigator.userAgent.indexOf('Mobile') isnt -1) or
|
}
|
||||||
(navigator.userAgent.indexOf('IEMobile') isnt -1)
|
|
||||||
catch
|
static detect() {
|
||||||
false
|
if (Cookies.get('override-mobile-detect') != null) {
|
||||||
|
return JSON.parse(Cookies.get('override-mobile-detect'));
|
||||||
@detectAndroidWebview: ->
|
}
|
||||||
try
|
try {
|
||||||
/(Android).*( Version\/.\.. ).*(Chrome)/.test(navigator.userAgent)
|
return (window.matchMedia('(max-width: 480px)').matches) ||
|
||||||
catch
|
(window.matchMedia('(max-width: 767px)').matches) ||
|
||||||
false
|
(window.matchMedia('(max-height: 767px) and (max-width: 1024px)').matches) ||
|
||||||
|
// Need to sniff the user agent because some Android and Windows Phone devices don't take
|
||||||
constructor: ->
|
// resolution (dpi) into account when reporting device width/height.
|
||||||
@el = document.documentElement
|
((navigator.userAgent.indexOf('Android') !== -1) && (navigator.userAgent.indexOf('Mobile') !== -1)) ||
|
||||||
super
|
(navigator.userAgent.indexOf('IEMobile') !== -1);
|
||||||
|
} catch (error) {
|
||||||
init: ->
|
return false;
|
||||||
$.on $('._search'), 'touchend', @onTapSearch
|
}
|
||||||
|
}
|
||||||
@toggleSidebar = $('button[data-toggle-sidebar]')
|
|
||||||
@toggleSidebar.removeAttribute('hidden')
|
static detectAndroidWebview() {
|
||||||
$.on @toggleSidebar, 'click', @onClickToggleSidebar
|
try {
|
||||||
|
return /(Android).*( Version\/.\.. ).*(Chrome)/.test(navigator.userAgent);
|
||||||
@back = $('button[data-back]')
|
} catch (error) {
|
||||||
@back.removeAttribute('hidden')
|
return false;
|
||||||
$.on @back, 'click', @onClickBack
|
}
|
||||||
|
}
|
||||||
@forward = $('button[data-forward]')
|
|
||||||
@forward.removeAttribute('hidden')
|
constructor() {
|
||||||
$.on @forward, 'click', @onClickForward
|
this.showSidebar = this.showSidebar.bind(this);
|
||||||
|
this.hideSidebar = this.hideSidebar.bind(this);
|
||||||
@docPickerTab = $('button[data-tab="doc-picker"]')
|
this.onClickBack = this.onClickBack.bind(this);
|
||||||
@docPickerTab.removeAttribute('hidden')
|
this.onClickForward = this.onClickForward.bind(this);
|
||||||
$.on @docPickerTab, 'click', @onClickDocPickerTab
|
this.onClickToggleSidebar = this.onClickToggleSidebar.bind(this);
|
||||||
|
this.onClickDocPickerTab = this.onClickDocPickerTab.bind(this);
|
||||||
@settingsTab = $('button[data-tab="settings"]')
|
this.onClickSettingsTab = this.onClickSettingsTab.bind(this);
|
||||||
@settingsTab.removeAttribute('hidden')
|
this.onTapSearch = this.onTapSearch.bind(this);
|
||||||
$.on @settingsTab, 'click', @onClickSettingsTab
|
this.onEscape = this.onEscape.bind(this);
|
||||||
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
|
this.el = document.documentElement;
|
||||||
|
super(...arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
$.on($('._search'), 'touchend', this.onTapSearch);
|
||||||
|
|
||||||
|
this.toggleSidebar = $('button[data-toggle-sidebar]');
|
||||||
|
this.toggleSidebar.removeAttribute('hidden');
|
||||||
|
$.on(this.toggleSidebar, 'click', this.onClickToggleSidebar);
|
||||||
|
|
||||||
|
this.back = $('button[data-back]');
|
||||||
|
this.back.removeAttribute('hidden');
|
||||||
|
$.on(this.back, 'click', this.onClickBack);
|
||||||
|
|
||||||
|
this.forward = $('button[data-forward]');
|
||||||
|
this.forward.removeAttribute('hidden');
|
||||||
|
$.on(this.forward, 'click', this.onClickForward);
|
||||||
|
|
||||||
|
this.docPickerTab = $('button[data-tab="doc-picker"]');
|
||||||
|
this.docPickerTab.removeAttribute('hidden');
|
||||||
|
$.on(this.docPickerTab, 'click', this.onClickDocPickerTab);
|
||||||
|
|
||||||
|
this.settingsTab = $('button[data-tab="settings"]');
|
||||||
|
this.settingsTab.removeAttribute('hidden');
|
||||||
|
$.on(this.settingsTab, 'click', this.onClickSettingsTab);
|
||||||
|
|
||||||
app.document.sidebar.search
|
app.document.sidebar.search
|
||||||
.on 'searching', @showSidebar
|
.on('searching', this.showSidebar);
|
||||||
|
|
||||||
@activate()
|
this.activate();
|
||||||
return
|
}
|
||||||
|
|
||||||
showSidebar: =>
|
showSidebar() {
|
||||||
if @isSidebarShown()
|
let selection;
|
||||||
window.scrollTo 0, 0
|
if (this.isSidebarShown()) {
|
||||||
return
|
window.scrollTo(0, 0);
|
||||||
|
return;
|
||||||
@contentTop = window.scrollY
|
}
|
||||||
@content.style.display = 'none'
|
|
||||||
@sidebar.style.display = 'block'
|
this.contentTop = window.scrollY;
|
||||||
|
this.content.style.display = 'none';
|
||||||
if selection = @findByClass app.views.ListSelect.activeClass
|
this.sidebar.style.display = 'block';
|
||||||
scrollContainer = if window.scrollY is @body.scrollTop then @body else document.documentElement
|
|
||||||
$.scrollTo selection, scrollContainer, 'center'
|
if (selection = this.findByClass(app.views.ListSelect.activeClass)) {
|
||||||
else
|
const scrollContainer = window.scrollY === this.body.scrollTop ? this.body : document.documentElement;
|
||||||
window.scrollTo 0, @findByClass(app.views.ListFold.activeClass) and @sidebarTop or 0
|
$.scrollTo(selection, scrollContainer, 'center');
|
||||||
return
|
} else {
|
||||||
|
window.scrollTo(0, (this.findByClass(app.views.ListFold.activeClass) && this.sidebarTop) || 0);
|
||||||
hideSidebar: =>
|
}
|
||||||
return unless @isSidebarShown()
|
}
|
||||||
@sidebarTop = window.scrollY
|
|
||||||
@sidebar.style.display = 'none'
|
hideSidebar() {
|
||||||
@content.style.display = 'block'
|
if (!this.isSidebarShown()) { return; }
|
||||||
window.scrollTo 0, @contentTop or 0
|
this.sidebarTop = window.scrollY;
|
||||||
return
|
this.sidebar.style.display = 'none';
|
||||||
|
this.content.style.display = 'block';
|
||||||
isSidebarShown: ->
|
window.scrollTo(0, this.contentTop || 0);
|
||||||
@sidebar.style.display isnt 'none'
|
}
|
||||||
|
|
||||||
onClickBack: =>
|
isSidebarShown() {
|
||||||
history.back()
|
return this.sidebar.style.display !== 'none';
|
||||||
|
}
|
||||||
onClickForward: =>
|
|
||||||
history.forward()
|
onClickBack() {
|
||||||
|
return history.back();
|
||||||
onClickToggleSidebar: =>
|
}
|
||||||
if @isSidebarShown() then @hideSidebar() else @showSidebar()
|
|
||||||
return
|
onClickForward() {
|
||||||
|
return history.forward();
|
||||||
onClickDocPickerTab: (event) =>
|
}
|
||||||
$.stopEvent(event)
|
|
||||||
@showDocPicker()
|
onClickToggleSidebar() {
|
||||||
return
|
if (this.isSidebarShown()) { this.hideSidebar(); } else { this.showSidebar(); }
|
||||||
|
}
|
||||||
onClickSettingsTab: (event) =>
|
|
||||||
$.stopEvent(event)
|
onClickDocPickerTab(event) {
|
||||||
@showSettings()
|
$.stopEvent(event);
|
||||||
return
|
this.showDocPicker();
|
||||||
|
}
|
||||||
showDocPicker: ->
|
|
||||||
window.scrollTo 0, 0
|
onClickSettingsTab(event) {
|
||||||
@docPickerTab.classList.add 'active'
|
$.stopEvent(event);
|
||||||
@settingsTab.classList.remove 'active'
|
this.showSettings();
|
||||||
@docPicker.style.display = 'block'
|
}
|
||||||
@content.style.display = 'none'
|
|
||||||
return
|
showDocPicker() {
|
||||||
|
window.scrollTo(0, 0);
|
||||||
showSettings: ->
|
this.docPickerTab.classList.add('active');
|
||||||
window.scrollTo 0, 0
|
this.settingsTab.classList.remove('active');
|
||||||
@docPickerTab.classList.remove 'active'
|
this.docPicker.style.display = 'block';
|
||||||
@settingsTab.classList.add 'active'
|
this.content.style.display = 'none';
|
||||||
@docPicker.style.display = 'none'
|
}
|
||||||
@content.style.display = 'block'
|
|
||||||
return
|
showSettings() {
|
||||||
|
window.scrollTo(0, 0);
|
||||||
onTapSearch: =>
|
this.docPickerTab.classList.remove('active');
|
||||||
window.scrollTo 0, 0
|
this.settingsTab.classList.add('active');
|
||||||
|
this.docPicker.style.display = 'none';
|
||||||
onEscape: =>
|
this.content.style.display = 'block';
|
||||||
@hideSidebar()
|
}
|
||||||
|
|
||||||
afterRoute: (route) =>
|
onTapSearch() {
|
||||||
@hideSidebar()
|
return window.scrollTo(0, 0);
|
||||||
|
}
|
||||||
if route is 'settings'
|
|
||||||
@showDocPicker()
|
onEscape() {
|
||||||
else
|
return this.hideSidebar();
|
||||||
@content.style.display = 'block'
|
}
|
||||||
|
|
||||||
if page.canGoBack()
|
afterRoute(route) {
|
||||||
@back.removeAttribute('disabled')
|
this.hideSidebar();
|
||||||
else
|
|
||||||
@back.setAttribute('disabled', 'disabled')
|
if (route === 'settings') {
|
||||||
|
this.showDocPicker();
|
||||||
if page.canGoForward()
|
} else {
|
||||||
@forward.removeAttribute('disabled')
|
this.content.style.display = 'block';
|
||||||
else
|
}
|
||||||
@forward.setAttribute('disabled', 'disabled')
|
|
||||||
return
|
if (page.canGoBack()) {
|
||||||
|
this.back.removeAttribute('disabled');
|
||||||
|
} else {
|
||||||
|
this.back.setAttribute('disabled', 'disabled');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (page.canGoForward()) {
|
||||||
|
this.forward.removeAttribute('disabled');
|
||||||
|
} else {
|
||||||
|
this.forward.setAttribute('disabled', 'disabled');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,43 +1,64 @@
|
|||||||
class app.views.Path extends app.View
|
/*
|
||||||
@className: '_path'
|
* decaffeinate suggestions:
|
||||||
@attributes:
|
* DS002: Fix invalid constructor
|
||||||
role: 'complementary'
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@events:
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
click: 'onClick'
|
*/
|
||||||
|
const Cls = (app.views.Path = class Path extends app.View {
|
||||||
@routes:
|
constructor(...args) {
|
||||||
after: 'afterRoute'
|
this.onClick = this.onClick.bind(this);
|
||||||
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
render: (args...) ->
|
super(...args);
|
||||||
@html @tmpl 'path', args...
|
}
|
||||||
@show()
|
|
||||||
return
|
static initClass() {
|
||||||
|
this.className = '_path';
|
||||||
show: ->
|
this.attributes =
|
||||||
@prependTo app.el unless @el.parentNode
|
{role: 'complementary'};
|
||||||
return
|
|
||||||
|
this.events =
|
||||||
hide: ->
|
{click: 'onClick'};
|
||||||
$.remove @el if @el.parentNode
|
|
||||||
return
|
this.routes =
|
||||||
|
{after: 'afterRoute'};
|
||||||
onClick: (event) =>
|
}
|
||||||
@clicked = true if link = $.closestLink event.target, @el
|
|
||||||
return
|
render(...args) {
|
||||||
|
this.html(this.tmpl('path', ...Array.from(args)));
|
||||||
afterRoute: (route, context) =>
|
this.show();
|
||||||
if context.type
|
}
|
||||||
@render context.doc, context.type
|
|
||||||
else if context.entry
|
show() {
|
||||||
if context.entry.isIndex()
|
if (!this.el.parentNode) { this.prependTo(app.el); }
|
||||||
@render context.doc
|
}
|
||||||
else
|
|
||||||
@render context.doc, context.entry.getType(), context.entry
|
hide() {
|
||||||
else
|
if (this.el.parentNode) { $.remove(this.el); }
|
||||||
@hide()
|
}
|
||||||
|
|
||||||
if @clicked
|
onClick(event) {
|
||||||
@clicked = null
|
let link;
|
||||||
app.document.sidebar.reset()
|
if (link = $.closestLink(event.target, this.el)) { this.clicked = true; }
|
||||||
return
|
}
|
||||||
|
|
||||||
|
afterRoute(route, context) {
|
||||||
|
if (context.type) {
|
||||||
|
this.render(context.doc, context.type);
|
||||||
|
} else if (context.entry) {
|
||||||
|
if (context.entry.isIndex()) {
|
||||||
|
this.render(context.doc);
|
||||||
|
} else {
|
||||||
|
this.render(context.doc, context.entry.getType(), context.entry);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.clicked) {
|
||||||
|
this.clicked = null;
|
||||||
|
app.document.sidebar.reset();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,49 +1,75 @@
|
|||||||
class app.views.Resizer extends app.View
|
/*
|
||||||
@className: '_resizer'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
dragstart: 'onDragStart'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
dragend: 'onDragEnd'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
@isSupported: ->
|
(function() {
|
||||||
'ondragstart' of document.createElement('div') and !app.isMobile()
|
let MIN = undefined;
|
||||||
|
let MAX = undefined;
|
||||||
init: ->
|
const Cls = (app.views.Resizer = class Resizer extends app.View {
|
||||||
@el.setAttribute('draggable', 'true')
|
constructor(...args) {
|
||||||
@appendTo $('._app')
|
this.onDragStart = this.onDragStart.bind(this);
|
||||||
return
|
this.onDrag = this.onDrag.bind(this);
|
||||||
|
this.onDragEnd = this.onDragEnd.bind(this);
|
||||||
MIN = 260
|
super(...args);
|
||||||
MAX = 600
|
}
|
||||||
|
|
||||||
resize: (value, save) ->
|
static initClass() {
|
||||||
value -= app.el.offsetLeft
|
this.className = '_resizer';
|
||||||
return unless value > 0
|
|
||||||
value = Math.min(Math.max(Math.round(value), MIN), MAX)
|
this.events = {
|
||||||
newSize = "#{value}px"
|
dragstart: 'onDragStart',
|
||||||
document.documentElement.style.setProperty('--sidebarWidth', newSize)
|
dragend: 'onDragEnd'
|
||||||
app.settings.setSize(value) if save
|
};
|
||||||
return
|
|
||||||
|
MIN = 260;
|
||||||
onDragStart: (event) =>
|
MAX = 600;
|
||||||
event.dataTransfer.effectAllowed = 'link'
|
}
|
||||||
event.dataTransfer.setData('Text', '')
|
|
||||||
$.on(window, 'dragover', @onDrag)
|
static isSupported() {
|
||||||
return
|
return 'ondragstart' in document.createElement('div') && !app.isMobile();
|
||||||
|
}
|
||||||
onDrag: (event) =>
|
|
||||||
value = event.pageX
|
init() {
|
||||||
return unless value > 0
|
this.el.setAttribute('draggable', 'true');
|
||||||
@lastDragValue = value
|
this.appendTo($('._app'));
|
||||||
return if @lastDrag and @lastDrag > Date.now() - 50
|
}
|
||||||
@lastDrag = Date.now()
|
|
||||||
@resize(value, false)
|
resize(value, save) {
|
||||||
return
|
value -= app.el.offsetLeft;
|
||||||
|
if (!(value > 0)) { return; }
|
||||||
onDragEnd: (event) =>
|
value = Math.min(Math.max(Math.round(value), MIN), MAX);
|
||||||
$.off(window, 'dragover', @onDrag)
|
const newSize = `${value}px`;
|
||||||
value = event.pageX or (event.screenX - window.screenX)
|
document.documentElement.style.setProperty('--sidebarWidth', newSize);
|
||||||
if @lastDragValue and not (@lastDragValue - 5 < value < @lastDragValue + 5) # https://github.com/freeCodeCamp/devdocs/issues/265
|
if (save) { app.settings.setSize(value); }
|
||||||
value = @lastDragValue
|
}
|
||||||
@resize(value, true)
|
|
||||||
return
|
onDragStart(event) {
|
||||||
|
event.dataTransfer.effectAllowed = 'link';
|
||||||
|
event.dataTransfer.setData('Text', '');
|
||||||
|
$.on(window, 'dragover', this.onDrag);
|
||||||
|
}
|
||||||
|
|
||||||
|
onDrag(event) {
|
||||||
|
const value = event.pageX;
|
||||||
|
if (!(value > 0)) { return; }
|
||||||
|
this.lastDragValue = value;
|
||||||
|
if (this.lastDrag && (this.lastDrag > (Date.now() - 50))) { return; }
|
||||||
|
this.lastDrag = Date.now();
|
||||||
|
this.resize(value, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
onDragEnd(event) {
|
||||||
|
$.off(window, 'dragover', this.onDrag);
|
||||||
|
let value = event.pageX || (event.screenX - window.screenX);
|
||||||
|
if (this.lastDragValue && !(this.lastDragValue - 5 < value && value < this.lastDragValue + 5)) { // https://github.com/freeCodeCamp/devdocs/issues/265
|
||||||
|
value = this.lastDragValue;
|
||||||
|
}
|
||||||
|
this.resize(value, true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,83 +1,127 @@
|
|||||||
class app.views.Settings extends app.View
|
/*
|
||||||
SIDEBAR_HIDDEN_LAYOUT = '_sidebar-hidden'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@el: '._settings'
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
@elements:
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
sidebar: '._sidebar'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
saveBtn: 'button[type="submit"]'
|
* DS207: Consider shorter variations of null checks
|
||||||
backBtn: 'button[data-back]'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
@events:
|
(function() {
|
||||||
import: 'onImport'
|
let SIDEBAR_HIDDEN_LAYOUT = undefined;
|
||||||
change: 'onChange'
|
const Cls = (app.views.Settings = class Settings extends app.View {
|
||||||
submit: 'onSubmit'
|
constructor(...args) {
|
||||||
click: 'onClick'
|
this.onChange = this.onChange.bind(this);
|
||||||
|
this.onEnter = this.onEnter.bind(this);
|
||||||
@shortcuts:
|
this.onSubmit = this.onSubmit.bind(this);
|
||||||
enter: 'onEnter'
|
this.onImport = this.onImport.bind(this);
|
||||||
|
this.onClick = this.onClick.bind(this);
|
||||||
init: ->
|
super(...args);
|
||||||
@addSubview @docPicker = new app.views.DocPicker
|
}
|
||||||
return
|
|
||||||
|
static initClass() {
|
||||||
activate: ->
|
SIDEBAR_HIDDEN_LAYOUT = '_sidebar-hidden';
|
||||||
if super
|
|
||||||
@render()
|
this.el = '._settings';
|
||||||
document.body.classList.remove(SIDEBAR_HIDDEN_LAYOUT)
|
|
||||||
return
|
this.elements = {
|
||||||
|
sidebar: '._sidebar',
|
||||||
deactivate: ->
|
saveBtn: 'button[type="submit"]',
|
||||||
if super
|
backBtn: 'button[data-back]'
|
||||||
@resetClass()
|
};
|
||||||
@docPicker.detach()
|
|
||||||
document.body.classList.add(SIDEBAR_HIDDEN_LAYOUT) if app.settings.hasLayout(SIDEBAR_HIDDEN_LAYOUT)
|
this.events = {
|
||||||
return
|
import: 'onImport',
|
||||||
|
change: 'onChange',
|
||||||
render: ->
|
submit: 'onSubmit',
|
||||||
@docPicker.appendTo @sidebar
|
click: 'onClick'
|
||||||
@refreshElements()
|
};
|
||||||
@addClass '_in'
|
|
||||||
return
|
this.shortcuts =
|
||||||
|
{enter: 'onEnter'};
|
||||||
save: (options = {}) ->
|
}
|
||||||
unless @saving
|
|
||||||
@saving = true
|
init() {
|
||||||
|
this.addSubview(this.docPicker = new app.views.DocPicker);
|
||||||
if options.import
|
}
|
||||||
docs = app.settings.getDocs()
|
|
||||||
else
|
activate() {
|
||||||
docs = @docPicker.getSelectedDocs()
|
if (super.activate(...arguments)) {
|
||||||
app.settings.setDocs(docs)
|
this.render();
|
||||||
|
document.body.classList.remove(SIDEBAR_HIDDEN_LAYOUT);
|
||||||
@saveBtn.textContent = 'Saving\u2026'
|
}
|
||||||
disabledDocs = new app.collections.Docs(doc for doc in app.docs.all() when docs.indexOf(doc.slug) is -1)
|
}
|
||||||
disabledDocs.uninstall ->
|
|
||||||
app.db.migrate()
|
deactivate() {
|
||||||
app.reload()
|
if (super.deactivate(...arguments)) {
|
||||||
return
|
this.resetClass();
|
||||||
|
this.docPicker.detach();
|
||||||
onChange: =>
|
if (app.settings.hasLayout(SIDEBAR_HIDDEN_LAYOUT)) { document.body.classList.add(SIDEBAR_HIDDEN_LAYOUT); }
|
||||||
@addClass('_dirty')
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
onEnter: =>
|
render() {
|
||||||
@save()
|
this.docPicker.appendTo(this.sidebar);
|
||||||
return
|
this.refreshElements();
|
||||||
|
this.addClass('_in');
|
||||||
onSubmit: (event) =>
|
}
|
||||||
event.preventDefault()
|
|
||||||
@save()
|
save(options) {
|
||||||
return
|
if (options == null) { options = {}; }
|
||||||
|
if (!this.saving) {
|
||||||
onImport: =>
|
let docs;
|
||||||
@addClass('_dirty')
|
this.saving = true;
|
||||||
@save(import: true)
|
|
||||||
return
|
if (options.import) {
|
||||||
|
docs = app.settings.getDocs();
|
||||||
onClick: (event) =>
|
} else {
|
||||||
return if event.which isnt 1
|
docs = this.docPicker.getSelectedDocs();
|
||||||
if event.target is @backBtn
|
app.settings.setDocs(docs);
|
||||||
$.stopEvent(event)
|
}
|
||||||
app.router.show '/'
|
|
||||||
return
|
this.saveBtn.textContent = 'Saving\u2026';
|
||||||
|
const disabledDocs = new app.collections.Docs((() => {
|
||||||
|
const result = [];
|
||||||
|
for (var doc of Array.from(app.docs.all())) { if (docs.indexOf(doc.slug) === -1) {
|
||||||
|
result.push(doc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
})());
|
||||||
|
disabledDocs.uninstall(function() {
|
||||||
|
app.db.migrate();
|
||||||
|
return app.reload();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onChange() {
|
||||||
|
this.addClass('_dirty');
|
||||||
|
}
|
||||||
|
|
||||||
|
onEnter() {
|
||||||
|
this.save();
|
||||||
|
}
|
||||||
|
|
||||||
|
onSubmit(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
this.save();
|
||||||
|
}
|
||||||
|
|
||||||
|
onImport() {
|
||||||
|
this.addClass('_dirty');
|
||||||
|
this.save({import: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
if (event.which !== 1) { return; }
|
||||||
|
if (event.target === this.backBtn) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
app.router.show('/');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,124 +1,177 @@
|
|||||||
class app.views.ListFocus extends app.View
|
/*
|
||||||
@activeClass: 'focus'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
click: 'onClick'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
@shortcuts:
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
up: 'onUp'
|
*/
|
||||||
down: 'onDown'
|
const Cls = (app.views.ListFocus = class ListFocus extends app.View {
|
||||||
left: 'onLeft'
|
static initClass() {
|
||||||
enter: 'onEnter'
|
this.activeClass = 'focus';
|
||||||
superEnter: 'onSuperEnter'
|
|
||||||
escape: 'blur'
|
this.events =
|
||||||
|
{click: 'onClick'};
|
||||||
constructor: (@el) ->
|
|
||||||
super
|
this.shortcuts = {
|
||||||
@focusOnNextFrame = $.framify(@focus, @)
|
up: 'onUp',
|
||||||
|
down: 'onDown',
|
||||||
focus: (el, options = {}) ->
|
left: 'onLeft',
|
||||||
if el and not el.classList.contains @constructor.activeClass
|
enter: 'onEnter',
|
||||||
@blur()
|
superEnter: 'onSuperEnter',
|
||||||
el.classList.add @constructor.activeClass
|
escape: 'blur'
|
||||||
$.trigger el, 'focus' unless options.silent is true
|
};
|
||||||
return
|
}
|
||||||
|
|
||||||
blur: =>
|
constructor(el) {
|
||||||
if cursor = @getCursor()
|
this.blur = this.blur.bind(this);
|
||||||
cursor.classList.remove @constructor.activeClass
|
this.onDown = this.onDown.bind(this);
|
||||||
$.trigger cursor, 'blur'
|
this.onUp = this.onUp.bind(this);
|
||||||
return
|
this.onLeft = this.onLeft.bind(this);
|
||||||
|
this.onEnter = this.onEnter.bind(this);
|
||||||
getCursor: ->
|
this.onSuperEnter = this.onSuperEnter.bind(this);
|
||||||
@findByClass(@constructor.activeClass) or @findByClass(app.views.ListSelect.activeClass)
|
this.onClick = this.onClick.bind(this);
|
||||||
|
this.el = el;
|
||||||
findNext: (cursor) ->
|
super(...arguments);
|
||||||
if next = cursor.nextSibling
|
this.focusOnNextFrame = $.framify(this.focus, this);
|
||||||
if next.tagName is 'A'
|
}
|
||||||
next
|
|
||||||
else if next.tagName is 'SPAN' # pagination link
|
focus(el, options) {
|
||||||
$.click(next)
|
if (options == null) { options = {}; }
|
||||||
@findNext cursor
|
if (el && !el.classList.contains(this.constructor.activeClass)) {
|
||||||
else if next.tagName is 'DIV' # sub-list
|
this.blur();
|
||||||
if cursor.className.indexOf(' open') >= 0
|
el.classList.add(this.constructor.activeClass);
|
||||||
@findFirst(next) or @findNext(next)
|
if (options.silent !== true) { $.trigger(el, 'focus'); }
|
||||||
else
|
}
|
||||||
@findNext(next)
|
}
|
||||||
else if next.tagName is 'H6' # title
|
|
||||||
@findNext(next)
|
blur() {
|
||||||
else if cursor.parentNode isnt @el
|
let cursor;
|
||||||
@findNext cursor.parentNode
|
if (cursor = this.getCursor()) {
|
||||||
|
cursor.classList.remove(this.constructor.activeClass);
|
||||||
findFirst: (cursor) ->
|
$.trigger(cursor, 'blur');
|
||||||
return unless first = cursor.firstChild
|
}
|
||||||
|
}
|
||||||
if first.tagName is 'A'
|
|
||||||
first
|
getCursor() {
|
||||||
else if first.tagName is 'SPAN' # pagination link
|
return this.findByClass(this.constructor.activeClass) || this.findByClass(app.views.ListSelect.activeClass);
|
||||||
$.click(first)
|
}
|
||||||
@findFirst cursor
|
|
||||||
|
findNext(cursor) {
|
||||||
findPrev: (cursor) ->
|
let next;
|
||||||
if prev = cursor.previousSibling
|
if (next = cursor.nextSibling) {
|
||||||
if prev.tagName is 'A'
|
if (next.tagName === 'A') {
|
||||||
prev
|
return next;
|
||||||
else if prev.tagName is 'SPAN' # pagination link
|
} else if (next.tagName === 'SPAN') { // pagination link
|
||||||
$.click(prev)
|
$.click(next);
|
||||||
@findPrev cursor
|
return this.findNext(cursor);
|
||||||
else if prev.tagName is 'DIV' # sub-list
|
} else if (next.tagName === 'DIV') { // sub-list
|
||||||
if prev.previousSibling.className.indexOf('open') >= 0
|
if (cursor.className.indexOf(' open') >= 0) {
|
||||||
@findLast(prev) or @findPrev(prev)
|
return this.findFirst(next) || this.findNext(next);
|
||||||
else
|
} else {
|
||||||
@findPrev(prev)
|
return this.findNext(next);
|
||||||
else if prev.tagName is 'H6' # title
|
}
|
||||||
@findPrev(prev)
|
} else if (next.tagName === 'H6') { // title
|
||||||
else if cursor.parentNode isnt @el
|
return this.findNext(next);
|
||||||
@findPrev cursor.parentNode
|
}
|
||||||
|
} else if (cursor.parentNode !== this.el) {
|
||||||
findLast: (cursor) ->
|
return this.findNext(cursor.parentNode);
|
||||||
return unless last = cursor.lastChild
|
}
|
||||||
|
}
|
||||||
if last.tagName is 'A'
|
|
||||||
last
|
findFirst(cursor) {
|
||||||
else if last.tagName is 'SPAN' or last.tagName is 'H6' # pagination link or title
|
let first;
|
||||||
@findPrev last
|
if (!(first = cursor.firstChild)) { return; }
|
||||||
else if last.tagName is 'DIV' # sub-list
|
|
||||||
@findLast last
|
if (first.tagName === 'A') {
|
||||||
|
return first;
|
||||||
onDown: =>
|
} else if (first.tagName === 'SPAN') { // pagination link
|
||||||
if cursor = @getCursor()
|
$.click(first);
|
||||||
@focusOnNextFrame @findNext(cursor)
|
return this.findFirst(cursor);
|
||||||
else
|
}
|
||||||
@focusOnNextFrame @findByTag('a')
|
}
|
||||||
return
|
|
||||||
|
findPrev(cursor) {
|
||||||
onUp: =>
|
let prev;
|
||||||
if cursor = @getCursor()
|
if (prev = cursor.previousSibling) {
|
||||||
@focusOnNextFrame @findPrev(cursor)
|
if (prev.tagName === 'A') {
|
||||||
else
|
return prev;
|
||||||
@focusOnNextFrame @findLastByTag('a')
|
} else if (prev.tagName === 'SPAN') { // pagination link
|
||||||
return
|
$.click(prev);
|
||||||
|
return this.findPrev(cursor);
|
||||||
onLeft: =>
|
} else if (prev.tagName === 'DIV') { // sub-list
|
||||||
cursor = @getCursor()
|
if (prev.previousSibling.className.indexOf('open') >= 0) {
|
||||||
if cursor and not cursor.classList.contains(app.views.ListFold.activeClass) and cursor.parentNode isnt @el
|
return this.findLast(prev) || this.findPrev(prev);
|
||||||
prev = cursor.parentNode.previousSibling
|
} else {
|
||||||
@focusOnNextFrame cursor.parentNode.previousSibling if prev and prev.classList.contains(app.views.ListFold.targetClass)
|
return this.findPrev(prev);
|
||||||
return
|
}
|
||||||
|
} else if (prev.tagName === 'H6') { // title
|
||||||
onEnter: =>
|
return this.findPrev(prev);
|
||||||
if cursor = @getCursor()
|
}
|
||||||
$.click(cursor)
|
} else if (cursor.parentNode !== this.el) {
|
||||||
return
|
return this.findPrev(cursor.parentNode);
|
||||||
|
}
|
||||||
onSuperEnter: =>
|
}
|
||||||
if cursor = @getCursor()
|
|
||||||
$.popup(cursor)
|
findLast(cursor) {
|
||||||
return
|
let last;
|
||||||
|
if (!(last = cursor.lastChild)) { return; }
|
||||||
onClick: (event) =>
|
|
||||||
return if event.which isnt 1 or event.metaKey or event.ctrlKey
|
if (last.tagName === 'A') {
|
||||||
target = $.eventTarget(event)
|
return last;
|
||||||
if target.tagName is 'A'
|
} else if ((last.tagName === 'SPAN') || (last.tagName === 'H6')) { // pagination link or title
|
||||||
@focus target, silent: true
|
return this.findPrev(last);
|
||||||
return
|
} else if (last.tagName === 'DIV') { // sub-list
|
||||||
|
return this.findLast(last);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onDown() {
|
||||||
|
let cursor;
|
||||||
|
if ((cursor = this.getCursor())) {
|
||||||
|
this.focusOnNextFrame(this.findNext(cursor));
|
||||||
|
} else {
|
||||||
|
this.focusOnNextFrame(this.findByTag('a'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onUp() {
|
||||||
|
let cursor;
|
||||||
|
if ((cursor = this.getCursor())) {
|
||||||
|
this.focusOnNextFrame(this.findPrev(cursor));
|
||||||
|
} else {
|
||||||
|
this.focusOnNextFrame(this.findLastByTag('a'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onLeft() {
|
||||||
|
const cursor = this.getCursor();
|
||||||
|
if (cursor && !cursor.classList.contains(app.views.ListFold.activeClass) && (cursor.parentNode !== this.el)) {
|
||||||
|
const prev = cursor.parentNode.previousSibling;
|
||||||
|
if (prev && prev.classList.contains(app.views.ListFold.targetClass)) { this.focusOnNextFrame(cursor.parentNode.previousSibling); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onEnter() {
|
||||||
|
let cursor;
|
||||||
|
if (cursor = this.getCursor()) {
|
||||||
|
$.click(cursor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onSuperEnter() {
|
||||||
|
let cursor;
|
||||||
|
if (cursor = this.getCursor()) {
|
||||||
|
$.popup(cursor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
if ((event.which !== 1) || event.metaKey || event.ctrlKey) { return; }
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
if (target.tagName === 'A') {
|
||||||
|
this.focus(target, {silent: true});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,71 +1,95 @@
|
|||||||
class app.views.ListFold extends app.View
|
/*
|
||||||
@targetClass: '_list-dir'
|
* decaffeinate suggestions:
|
||||||
@handleClass: '_list-arrow'
|
* DS002: Fix invalid constructor
|
||||||
@activeClass: 'open'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.ListFold = class ListFold extends app.View {
|
||||||
|
static initClass() {
|
||||||
|
this.targetClass = '_list-dir';
|
||||||
|
this.handleClass = '_list-arrow';
|
||||||
|
this.activeClass = 'open';
|
||||||
|
|
||||||
@events:
|
this.events =
|
||||||
click: 'onClick'
|
{click: 'onClick'};
|
||||||
|
|
||||||
@shortcuts:
|
this.shortcuts = {
|
||||||
left: 'onLeft'
|
left: 'onLeft',
|
||||||
right: 'onRight'
|
right: 'onRight'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
constructor: (@el) -> super
|
constructor(el) { this.onLeft = this.onLeft.bind(this); this.onRight = this.onRight.bind(this); this.onClick = this.onClick.bind(this); this.el = el; super(...arguments); }
|
||||||
|
|
||||||
open: (el) ->
|
open(el) {
|
||||||
if el and not el.classList.contains @constructor.activeClass
|
if (el && !el.classList.contains(this.constructor.activeClass)) {
|
||||||
el.classList.add @constructor.activeClass
|
el.classList.add(this.constructor.activeClass);
|
||||||
$.trigger el, 'open'
|
$.trigger(el, 'open');
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
close: (el) ->
|
close(el) {
|
||||||
if el and el.classList.contains @constructor.activeClass
|
if (el && el.classList.contains(this.constructor.activeClass)) {
|
||||||
el.classList.remove @constructor.activeClass
|
el.classList.remove(this.constructor.activeClass);
|
||||||
$.trigger el, 'close'
|
$.trigger(el, 'close');
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
toggle: (el) ->
|
toggle(el) {
|
||||||
if el.classList.contains @constructor.activeClass
|
if (el.classList.contains(this.constructor.activeClass)) {
|
||||||
@close el
|
this.close(el);
|
||||||
else
|
} else {
|
||||||
@open el
|
this.open(el);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
reset: ->
|
reset() {
|
||||||
while el = @findByClass @constructor.activeClass
|
let el;
|
||||||
@close el
|
while ((el = this.findByClass(this.constructor.activeClass))) {
|
||||||
return
|
this.close(el);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
getCursor: ->
|
getCursor() {
|
||||||
@findByClass(app.views.ListFocus.activeClass) or @findByClass(app.views.ListSelect.activeClass)
|
return this.findByClass(app.views.ListFocus.activeClass) || this.findByClass(app.views.ListSelect.activeClass);
|
||||||
|
}
|
||||||
|
|
||||||
onLeft: =>
|
onLeft() {
|
||||||
cursor = @getCursor()
|
const cursor = this.getCursor();
|
||||||
if cursor?.classList.contains @constructor.activeClass
|
if (cursor != null ? cursor.classList.contains(this.constructor.activeClass) : undefined) {
|
||||||
@close cursor
|
this.close(cursor);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onRight: =>
|
onRight() {
|
||||||
cursor = @getCursor()
|
const cursor = this.getCursor();
|
||||||
if cursor?.classList.contains @constructor.targetClass
|
if (cursor != null ? cursor.classList.contains(this.constructor.targetClass) : undefined) {
|
||||||
@open cursor
|
this.open(cursor);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onClick: (event) =>
|
onClick(event) {
|
||||||
return if event.which isnt 1 or event.metaKey or event.ctrlKey
|
if ((event.which !== 1) || event.metaKey || event.ctrlKey) { return; }
|
||||||
return unless event.pageY # ignore fabricated clicks
|
if (!event.pageY) { return; } // ignore fabricated clicks
|
||||||
el = $.eventTarget(event)
|
let el = $.eventTarget(event);
|
||||||
el = el.parentNode if el.parentNode.tagName.toUpperCase() is 'SVG'
|
if (el.parentNode.tagName.toUpperCase() === 'SVG') { el = el.parentNode; }
|
||||||
|
|
||||||
if el.classList.contains @constructor.handleClass
|
if (el.classList.contains(this.constructor.handleClass)) {
|
||||||
$.stopEvent(event)
|
$.stopEvent(event);
|
||||||
@toggle el.parentNode
|
this.toggle(el.parentNode);
|
||||||
else if el.classList.contains @constructor.targetClass
|
} else if (el.classList.contains(this.constructor.targetClass)) {
|
||||||
if el.hasAttribute('href')
|
if (el.hasAttribute('href')) {
|
||||||
if el.classList.contains(@constructor.activeClass)
|
if (el.classList.contains(this.constructor.activeClass)) {
|
||||||
@close(el) if el.classList.contains(app.views.ListSelect.activeClass)
|
if (el.classList.contains(app.views.ListSelect.activeClass)) { this.close(el); }
|
||||||
else
|
} else {
|
||||||
@open(el)
|
this.open(el);
|
||||||
else
|
}
|
||||||
@toggle(el)
|
} else {
|
||||||
return
|
this.toggle(el);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,43 +1,65 @@
|
|||||||
class app.views.ListSelect extends app.View
|
/*
|
||||||
@activeClass: 'active'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
click: 'onClick'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
constructor: (@el) -> super
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
deactivate: ->
|
const Cls = (app.views.ListSelect = class ListSelect extends app.View {
|
||||||
@deselect() if super
|
static initClass() {
|
||||||
return
|
this.activeClass = 'active';
|
||||||
|
|
||||||
select: (el) ->
|
this.events =
|
||||||
@deselect()
|
{click: 'onClick'};
|
||||||
if el
|
}
|
||||||
el.classList.add @constructor.activeClass
|
|
||||||
$.trigger el, 'select'
|
constructor(el) { this.onClick = this.onClick.bind(this); this.el = el; super(...arguments); }
|
||||||
return
|
|
||||||
|
deactivate() {
|
||||||
deselect: ->
|
if (super.deactivate(...arguments)) { this.deselect(); }
|
||||||
if selection = @getSelection()
|
}
|
||||||
selection.classList.remove @constructor.activeClass
|
|
||||||
$.trigger selection, 'deselect'
|
select(el) {
|
||||||
return
|
this.deselect();
|
||||||
|
if (el) {
|
||||||
selectByHref: (href) ->
|
el.classList.add(this.constructor.activeClass);
|
||||||
unless @getSelection()?.getAttribute('href') is href
|
$.trigger(el, 'select');
|
||||||
@select @find("a[href='#{href}']")
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
selectCurrent: ->
|
deselect() {
|
||||||
@selectByHref location.pathname + location.hash
|
let selection;
|
||||||
return
|
if (selection = this.getSelection()) {
|
||||||
|
selection.classList.remove(this.constructor.activeClass);
|
||||||
getSelection: ->
|
$.trigger(selection, 'deselect');
|
||||||
@findByClass @constructor.activeClass
|
}
|
||||||
|
}
|
||||||
onClick: (event) =>
|
|
||||||
return if event.which isnt 1 or event.metaKey or event.ctrlKey
|
selectByHref(href) {
|
||||||
target = $.eventTarget(event)
|
if (__guard__(this.getSelection(), x => x.getAttribute('href')) !== href) {
|
||||||
if target.tagName is 'A'
|
this.select(this.find(`a[href='${href}']`));
|
||||||
@select target
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
|
selectCurrent() {
|
||||||
|
this.selectByHref(location.pathname + location.hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
getSelection() {
|
||||||
|
return this.findByClass(this.constructor.activeClass);
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
if ((event.which !== 1) || event.metaKey || event.ctrlKey) { return; }
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
if (target.tagName === 'A') {
|
||||||
|
this.select(target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,90 +1,121 @@
|
|||||||
class app.views.PaginatedList extends app.View
|
/*
|
||||||
PER_PAGE = app.config.max_results
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
constructor: (@data) ->
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
(@constructor.events or= {}).click ?= 'onClick'
|
* DS104: Avoid inline assignments
|
||||||
super
|
* DS202: Simplify dynamic range loops
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
renderPaginated: ->
|
* DS207: Consider shorter variations of null checks
|
||||||
@page = 0
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
if @totalPages() > 1
|
(function() {
|
||||||
@paginateNext()
|
let PER_PAGE = undefined;
|
||||||
else
|
const Cls = (app.views.PaginatedList = class PaginatedList extends app.View {
|
||||||
@html @renderAll()
|
static initClass() {
|
||||||
return
|
PER_PAGE = app.config.max_results;
|
||||||
|
}
|
||||||
# render: (dataSlice) -> implemented by subclass
|
|
||||||
|
constructor(data) {
|
||||||
renderAll: ->
|
let base;
|
||||||
@render @data
|
this.onClick = this.onClick.bind(this);
|
||||||
|
this.data = data;
|
||||||
renderPage: (page) ->
|
if (((base = this.constructor.events || (this.constructor.events = {}))).click == null) { base.click = 'onClick'; }
|
||||||
@render @data[((page - 1) * PER_PAGE)...(page * PER_PAGE)]
|
super(...arguments);
|
||||||
|
}
|
||||||
renderPageLink: (count) ->
|
|
||||||
@tmpl 'sidebarPageLink', count
|
renderPaginated() {
|
||||||
|
this.page = 0;
|
||||||
renderPrevLink: (page) ->
|
|
||||||
@renderPageLink (page - 1) * PER_PAGE
|
if (this.totalPages() > 1) {
|
||||||
|
this.paginateNext();
|
||||||
renderNextLink: (page) ->
|
} else {
|
||||||
@renderPageLink @data.length - page * PER_PAGE
|
this.html(this.renderAll());
|
||||||
|
}
|
||||||
totalPages: ->
|
}
|
||||||
Math.ceil @data.length / PER_PAGE
|
|
||||||
|
// render: (dataSlice) -> implemented by subclass
|
||||||
paginate: (link) ->
|
|
||||||
$.lockScroll link.nextSibling or link.previousSibling, =>
|
renderAll() {
|
||||||
$.batchUpdate @el, =>
|
return this.render(this.data);
|
||||||
if link.nextSibling then @paginatePrev link else @paginateNext link
|
}
|
||||||
return
|
|
||||||
return
|
renderPage(page) {
|
||||||
return
|
return this.render(this.data.slice(((page - 1) * PER_PAGE), (page * PER_PAGE)));
|
||||||
|
}
|
||||||
paginateNext: ->
|
|
||||||
@remove @el.lastChild if @el.lastChild # remove link
|
renderPageLink(count) {
|
||||||
@hideTopPage() if @page >= 2 # keep previous page into view
|
return this.tmpl('sidebarPageLink', count);
|
||||||
@page++
|
}
|
||||||
@append @renderPage(@page)
|
|
||||||
@append @renderNextLink(@page) if @page < @totalPages()
|
renderPrevLink(page) {
|
||||||
return
|
return this.renderPageLink((page - 1) * PER_PAGE);
|
||||||
|
}
|
||||||
paginatePrev: ->
|
|
||||||
@remove @el.firstChild # remove link
|
renderNextLink(page) {
|
||||||
@hideBottomPage()
|
return this.renderPageLink(this.data.length - (page * PER_PAGE));
|
||||||
@page--
|
}
|
||||||
@prepend @renderPage(@page - 1) # previous page is offset by one
|
|
||||||
@prepend @renderPrevLink(@page - 1) if @page >= 3
|
totalPages() {
|
||||||
return
|
return Math.ceil(this.data.length / PER_PAGE);
|
||||||
|
}
|
||||||
paginateTo: (object) ->
|
|
||||||
index = @data.indexOf(object)
|
paginate(link) {
|
||||||
if index >= PER_PAGE
|
$.lockScroll(link.nextSibling || link.previousSibling, () => {
|
||||||
@paginateNext() for [0...(index // PER_PAGE)]
|
$.batchUpdate(this.el, () => {
|
||||||
return
|
if (link.nextSibling) { this.paginatePrev(link); } else { this.paginateNext(link); }
|
||||||
|
});
|
||||||
hideTopPage: ->
|
});
|
||||||
n = if @page <= 2
|
}
|
||||||
PER_PAGE
|
|
||||||
else
|
paginateNext() {
|
||||||
PER_PAGE + 1 # remove link
|
if (this.el.lastChild) { this.remove(this.el.lastChild); } // remove link
|
||||||
@remove @el.firstChild for [0...n]
|
if (this.page >= 2) { this.hideTopPage(); } // keep previous page into view
|
||||||
@prepend @renderPrevLink(@page)
|
this.page++;
|
||||||
return
|
this.append(this.renderPage(this.page));
|
||||||
|
if (this.page < this.totalPages()) { this.append(this.renderNextLink(this.page)); }
|
||||||
hideBottomPage: ->
|
}
|
||||||
n = if @page is @totalPages()
|
|
||||||
@data.length % PER_PAGE or PER_PAGE
|
paginatePrev() {
|
||||||
else
|
this.remove(this.el.firstChild); // remove link
|
||||||
PER_PAGE + 1 # remove link
|
this.hideBottomPage();
|
||||||
@remove @el.lastChild for [0...n]
|
this.page--;
|
||||||
@append @renderNextLink(@page - 1)
|
this.prepend(this.renderPage(this.page - 1)); // previous page is offset by one
|
||||||
return
|
if (this.page >= 3) { this.prepend(this.renderPrevLink(this.page - 1)); }
|
||||||
|
}
|
||||||
onClick: (event) =>
|
|
||||||
target = $.eventTarget(event)
|
paginateTo(object) {
|
||||||
if target.tagName is 'SPAN' # link
|
const index = this.data.indexOf(object);
|
||||||
$.stopEvent(event)
|
if (index >= PER_PAGE) {
|
||||||
@paginate target
|
for (let i = 0, end = Math.floor(index / PER_PAGE), asc = 0 <= end; asc ? i < end : i > end; asc ? i++ : i--) { this.paginateNext(); }
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
hideTopPage() {
|
||||||
|
const n = this.page <= 2 ?
|
||||||
|
PER_PAGE
|
||||||
|
:
|
||||||
|
PER_PAGE + 1; // remove link
|
||||||
|
for (let i = 0, end = n, asc = 0 <= end; asc ? i < end : i > end; asc ? i++ : i--) { this.remove(this.el.firstChild); }
|
||||||
|
this.prepend(this.renderPrevLink(this.page));
|
||||||
|
}
|
||||||
|
|
||||||
|
hideBottomPage() {
|
||||||
|
const n = this.page === this.totalPages() ?
|
||||||
|
(this.data.length % PER_PAGE) || PER_PAGE
|
||||||
|
:
|
||||||
|
PER_PAGE + 1; // remove link
|
||||||
|
for (let i = 0, end = n, asc = 0 <= end; asc ? i < end : i > end; asc ? i++ : i--) { this.remove(this.el.lastChild); }
|
||||||
|
this.append(this.renderNextLink(this.page - 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
if (target.tagName === 'SPAN') { // link
|
||||||
|
$.stopEvent(event);
|
||||||
|
this.paginate(target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
@ -1,34 +1,55 @@
|
|||||||
#= require views/misc/notif
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
class app.views.News extends app.views.Notif
|
* DS101: Remove unnecessary use of Array.from
|
||||||
@className += ' _notif-news'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
@defautOptions:
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
autoHide: 30000
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
init: ->
|
//= require views/misc/notif
|
||||||
@unreadNews = @getUnreadNews()
|
|
||||||
@show() if @unreadNews.length
|
const Cls = (app.views.News = class News extends app.views.Notif {
|
||||||
@markAllAsRead()
|
static initClass() {
|
||||||
return
|
this.className += ' _notif-news';
|
||||||
|
|
||||||
render: ->
|
this.defautOptions =
|
||||||
@html app.templates.notifNews(@unreadNews)
|
{autoHide: 30000};
|
||||||
return
|
}
|
||||||
|
|
||||||
getUnreadNews: ->
|
init() {
|
||||||
return [] unless time = @getLastReadTime()
|
this.unreadNews = this.getUnreadNews();
|
||||||
|
if (this.unreadNews.length) { this.show(); }
|
||||||
for news in app.news
|
this.markAllAsRead();
|
||||||
break if new Date(news[0]).getTime() <= time
|
}
|
||||||
news
|
|
||||||
|
render() {
|
||||||
getLastNewsTime: ->
|
this.html(app.templates.notifNews(this.unreadNews));
|
||||||
new Date(app.news[0][0]).getTime()
|
}
|
||||||
|
|
||||||
getLastReadTime: ->
|
getUnreadNews() {
|
||||||
app.settings.get 'news'
|
let time;
|
||||||
|
if (!(time = this.getLastReadTime())) { return []; }
|
||||||
markAllAsRead: ->
|
|
||||||
app.settings.set 'news', @getLastNewsTime()
|
return (() => {
|
||||||
return
|
const result = [];
|
||||||
|
for (var news of Array.from(app.news)) {
|
||||||
|
if (new Date(news[0]).getTime() <= time) { break; }
|
||||||
|
result.push(news);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
|
||||||
|
getLastNewsTime() {
|
||||||
|
return new Date(app.news[0][0]).getTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
getLastReadTime() {
|
||||||
|
return app.settings.get('news');
|
||||||
|
}
|
||||||
|
|
||||||
|
markAllAsRead() {
|
||||||
|
app.settings.set('news', this.getLastNewsTime());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,27 +1,38 @@
|
|||||||
class app.views.Notice extends app.View
|
/*
|
||||||
@className: '_notice'
|
* decaffeinate suggestions:
|
||||||
@attributes:
|
* DS002: Fix invalid constructor
|
||||||
role: 'alert'
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.Notice = class Notice extends app.View {
|
||||||
|
static initClass() {
|
||||||
|
this.className = '_notice';
|
||||||
|
this.attributes =
|
||||||
|
{role: 'alert'};
|
||||||
|
}
|
||||||
|
|
||||||
constructor: (@type, @args...) -> super
|
constructor(type, ...rest) { this.type = type; [...this.args] = Array.from(rest); super(...arguments); }
|
||||||
|
|
||||||
init: ->
|
init() {
|
||||||
@activate()
|
this.activate();
|
||||||
return
|
}
|
||||||
|
|
||||||
activate: ->
|
activate() {
|
||||||
@show() if super
|
if (super.activate(...arguments)) { this.show(); }
|
||||||
return
|
}
|
||||||
|
|
||||||
deactivate: ->
|
deactivate() {
|
||||||
@hide() if super
|
if (super.deactivate(...arguments)) { this.hide(); }
|
||||||
return
|
}
|
||||||
|
|
||||||
show: ->
|
show() {
|
||||||
@html @tmpl("#{@type}Notice", @args...)
|
this.html(this.tmpl(`${this.type}Notice`, ...Array.from(this.args)));
|
||||||
@prependTo app.el
|
this.prependTo(app.el);
|
||||||
return
|
}
|
||||||
|
|
||||||
hide: ->
|
hide() {
|
||||||
$.remove @el
|
$.remove(this.el);
|
||||||
return
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,59 +1,78 @@
|
|||||||
class app.views.Notif extends app.View
|
/*
|
||||||
@className: '_notif'
|
* decaffeinate suggestions:
|
||||||
@activeClass: '_in'
|
* DS002: Fix invalid constructor
|
||||||
@attributes:
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
role: 'alert'
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@defautOptions:
|
*/
|
||||||
autoHide: 15000
|
const Cls = (app.views.Notif = class Notif extends app.View {
|
||||||
|
static initClass() {
|
||||||
@events:
|
this.className = '_notif';
|
||||||
click: 'onClick'
|
this.activeClass = '_in';
|
||||||
|
this.attributes =
|
||||||
constructor: (@type, @options = {}) ->
|
{role: 'alert'};
|
||||||
@options = $.extend {}, @constructor.defautOptions, @options
|
|
||||||
super
|
this.defautOptions =
|
||||||
|
{autoHide: 15000};
|
||||||
init: ->
|
|
||||||
@show()
|
this.events =
|
||||||
return
|
{click: 'onClick'};
|
||||||
|
}
|
||||||
show: ->
|
|
||||||
if @timeout
|
constructor(type, options) {
|
||||||
clearTimeout @timeout
|
this.onClick = this.onClick.bind(this);
|
||||||
@timeout = @delay @hide, @options.autoHide
|
this.type = type;
|
||||||
else
|
if (options == null) { options = {}; }
|
||||||
@render()
|
this.options = options;
|
||||||
@position()
|
this.options = $.extend({}, this.constructor.defautOptions, this.options);
|
||||||
@activate()
|
super(...arguments);
|
||||||
@appendTo document.body
|
}
|
||||||
@el.offsetWidth # force reflow
|
|
||||||
@addClass @constructor.activeClass
|
init() {
|
||||||
@timeout = @delay @hide, @options.autoHide if @options.autoHide
|
this.show();
|
||||||
return
|
}
|
||||||
|
|
||||||
hide: ->
|
show() {
|
||||||
clearTimeout @timeout
|
if (this.timeout) {
|
||||||
@timeout = null
|
clearTimeout(this.timeout);
|
||||||
@detach()
|
this.timeout = this.delay(this.hide, this.options.autoHide);
|
||||||
return
|
} else {
|
||||||
|
this.render();
|
||||||
render: ->
|
this.position();
|
||||||
@html @tmpl("notif#{@type}")
|
this.activate();
|
||||||
return
|
this.appendTo(document.body);
|
||||||
|
this.el.offsetWidth; // force reflow
|
||||||
position: ->
|
this.addClass(this.constructor.activeClass);
|
||||||
notifications = $$ ".#{app.views.Notif.className}"
|
if (this.options.autoHide) { this.timeout = this.delay(this.hide, this.options.autoHide); }
|
||||||
if notifications.length
|
}
|
||||||
lastNotif = notifications[notifications.length - 1]
|
}
|
||||||
@el.style.top = lastNotif.offsetTop + lastNotif.offsetHeight + 16 + 'px'
|
|
||||||
return
|
hide() {
|
||||||
|
clearTimeout(this.timeout);
|
||||||
onClick: (event) =>
|
this.timeout = null;
|
||||||
return if event.which isnt 1
|
this.detach();
|
||||||
target = $.eventTarget(event)
|
}
|
||||||
return if target.hasAttribute('data-behavior')
|
|
||||||
if target.tagName isnt 'A' or target.classList.contains('_notif-close')
|
render() {
|
||||||
$.stopEvent(event)
|
this.html(this.tmpl(`notif${this.type}`));
|
||||||
@hide()
|
}
|
||||||
return
|
|
||||||
|
position() {
|
||||||
|
const notifications = $$(`.${app.views.Notif.className}`);
|
||||||
|
if (notifications.length) {
|
||||||
|
const lastNotif = notifications[notifications.length - 1];
|
||||||
|
this.el.style.top = lastNotif.offsetTop + lastNotif.offsetHeight + 16 + 'px';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
if (event.which !== 1) { return; }
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
if (target.hasAttribute('data-behavior')) { return; }
|
||||||
|
if ((target.tagName !== 'A') || target.classList.contains('_notif-close')) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
this.hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,11 +1,20 @@
|
|||||||
#= require views/misc/notif
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require views/misc/notif
|
||||||
|
|
||||||
class app.views.Tip extends app.views.Notif
|
const Cls = (app.views.Tip = class Tip extends app.views.Notif {
|
||||||
@className: '_notif _notif-tip'
|
static initClass() {
|
||||||
|
this.className = '_notif _notif-tip';
|
||||||
|
|
||||||
@defautOptions:
|
this.defautOptions =
|
||||||
autoHide: false
|
{autoHide: false};
|
||||||
|
}
|
||||||
|
|
||||||
render: ->
|
render() {
|
||||||
@html @tmpl("tip#{@type}")
|
this.html(this.tmpl(`tip${this.type}`));
|
||||||
return
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,34 +1,56 @@
|
|||||||
#= require views/misc/notif
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
class app.views.Updates extends app.views.Notif
|
* DS101: Remove unnecessary use of Array.from
|
||||||
@className += ' _notif-news'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS205: Consider reworking code to avoid use of IIFEs
|
||||||
@defautOptions:
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
autoHide: 30000
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
init: ->
|
//= require views/misc/notif
|
||||||
@lastUpdateTime = @getLastUpdateTime()
|
|
||||||
@updatedDocs = @getUpdatedDocs()
|
const Cls = (app.views.Updates = class Updates extends app.views.Notif {
|
||||||
@updatedDisabledDocs = @getUpdatedDisabledDocs()
|
static initClass() {
|
||||||
@show() if @updatedDocs.length > 0 or @updatedDisabledDocs.length > 0
|
this.className += ' _notif-news';
|
||||||
@markAllAsRead()
|
|
||||||
return
|
this.defautOptions =
|
||||||
|
{autoHide: 30000};
|
||||||
render: ->
|
}
|
||||||
@html app.templates.notifUpdates(@updatedDocs, @updatedDisabledDocs)
|
|
||||||
return
|
init() {
|
||||||
|
this.lastUpdateTime = this.getLastUpdateTime();
|
||||||
getUpdatedDocs: ->
|
this.updatedDocs = this.getUpdatedDocs();
|
||||||
return [] unless @lastUpdateTime
|
this.updatedDisabledDocs = this.getUpdatedDisabledDocs();
|
||||||
doc for doc in app.docs.all() when doc.mtime > @lastUpdateTime
|
if ((this.updatedDocs.length > 0) || (this.updatedDisabledDocs.length > 0)) { this.show(); }
|
||||||
|
this.markAllAsRead();
|
||||||
getUpdatedDisabledDocs: ->
|
}
|
||||||
return [] unless @lastUpdateTime
|
|
||||||
doc for doc in app.disabledDocs.all() when doc.mtime > @lastUpdateTime and app.docs.findBy('slug_without_version', doc.slug_without_version)
|
render() {
|
||||||
|
this.html(app.templates.notifUpdates(this.updatedDocs, this.updatedDisabledDocs));
|
||||||
getLastUpdateTime: ->
|
}
|
||||||
app.settings.get 'version'
|
|
||||||
|
getUpdatedDocs() {
|
||||||
markAllAsRead: ->
|
if (!this.lastUpdateTime) { return []; }
|
||||||
app.settings.set 'version', if app.config.env is 'production' then app.config.version else Math.floor(Date.now() / 1000)
|
return Array.from(app.docs.all()).filter((doc) => doc.mtime > this.lastUpdateTime);
|
||||||
return
|
}
|
||||||
|
|
||||||
|
getUpdatedDisabledDocs() {
|
||||||
|
if (!this.lastUpdateTime) { return []; }
|
||||||
|
return (() => {
|
||||||
|
const result = [];
|
||||||
|
for (var doc of Array.from(app.disabledDocs.all())) { if ((doc.mtime > this.lastUpdateTime) && app.docs.findBy('slug_without_version', doc.slug_without_version)) {
|
||||||
|
result.push(doc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
|
||||||
|
getLastUpdateTime() {
|
||||||
|
return app.settings.get('version');
|
||||||
|
}
|
||||||
|
|
||||||
|
markAllAsRead() {
|
||||||
|
app.settings.set('version', app.config.env === 'production' ? app.config.version : Math.floor(Date.now() / 1000));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,43 +1,61 @@
|
|||||||
class app.views.BasePage extends app.View
|
/*
|
||||||
constructor: (@el, @entry) -> super
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
app.views.BasePage = class BasePage extends app.View {
|
||||||
|
constructor(el, entry) { this.paintCode = this.paintCode.bind(this); this.el = el; this.entry = entry; super(...arguments); }
|
||||||
|
|
||||||
deactivate: ->
|
deactivate() {
|
||||||
if super
|
if (super.deactivate(...arguments)) {
|
||||||
@highlightNodes = []
|
return this.highlightNodes = [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
render: (content, fromCache = false) ->
|
render(content, fromCache) {
|
||||||
@highlightNodes = []
|
if (fromCache == null) { fromCache = false; }
|
||||||
@previousTiming = null
|
this.highlightNodes = [];
|
||||||
@addClass "_#{@entry.doc.type}" unless @constructor.className
|
this.previousTiming = null;
|
||||||
@html content
|
if (!this.constructor.className) { this.addClass(`_${this.entry.doc.type}`); }
|
||||||
@highlightCode() unless fromCache
|
this.html(content);
|
||||||
@activate()
|
if (!fromCache) { this.highlightCode(); }
|
||||||
@delay @afterRender if @afterRender
|
this.activate();
|
||||||
if @highlightNodes.length > 0
|
if (this.afterRender) { this.delay(this.afterRender); }
|
||||||
$.requestAnimationFrame => $.requestAnimationFrame(@paintCode)
|
if (this.highlightNodes.length > 0) {
|
||||||
return
|
$.requestAnimationFrame(() => $.requestAnimationFrame(this.paintCode));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
highlightCode: ->
|
highlightCode() {
|
||||||
for el in @findAll('pre[data-language]')
|
for (var el of Array.from(this.findAll('pre[data-language]'))) {
|
||||||
language = el.getAttribute('data-language')
|
var language = el.getAttribute('data-language');
|
||||||
el.classList.add("language-#{language}")
|
el.classList.add(`language-${language}`);
|
||||||
@highlightNodes.push(el)
|
this.highlightNodes.push(el);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
paintCode: (timing) =>
|
paintCode(timing) {
|
||||||
if @previousTiming
|
if (this.previousTiming) {
|
||||||
if Math.round(1000 / (timing - @previousTiming)) > 50 # fps
|
if (Math.round(1000 / (timing - this.previousTiming)) > 50) { // fps
|
||||||
@nodesPerFrame = Math.round(Math.min(@nodesPerFrame * 1.25, 50))
|
this.nodesPerFrame = Math.round(Math.min(this.nodesPerFrame * 1.25, 50));
|
||||||
else
|
} else {
|
||||||
@nodesPerFrame = Math.round(Math.max(@nodesPerFrame * .8, 10))
|
this.nodesPerFrame = Math.round(Math.max(this.nodesPerFrame * .8, 10));
|
||||||
else
|
}
|
||||||
@nodesPerFrame = 10
|
} else {
|
||||||
|
this.nodesPerFrame = 10;
|
||||||
|
}
|
||||||
|
|
||||||
for el in @highlightNodes.splice(0, @nodesPerFrame)
|
for (var el of Array.from(this.highlightNodes.splice(0, this.nodesPerFrame))) {
|
||||||
$.remove(clipEl) if clipEl = el.lastElementChild
|
var clipEl;
|
||||||
Prism.highlightElement(el)
|
if (clipEl = el.lastElementChild) { $.remove(clipEl); }
|
||||||
$.append(el, clipEl) if clipEl
|
Prism.highlightElement(el);
|
||||||
|
if (clipEl) { $.append(el, clipEl); }
|
||||||
|
}
|
||||||
|
|
||||||
$.requestAnimationFrame(@paintCode) if @highlightNodes.length > 0
|
if (this.highlightNodes.length > 0) { $.requestAnimationFrame(this.paintCode); }
|
||||||
@previousTiming = timing
|
this.previousTiming = timing;
|
||||||
return
|
}
|
||||||
|
};
|
||||||
|
@ -1,16 +1,28 @@
|
|||||||
class app.views.HiddenPage extends app.View
|
/*
|
||||||
@events:
|
* decaffeinate suggestions:
|
||||||
click: 'onClick'
|
* DS002: Fix invalid constructor
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.HiddenPage = class HiddenPage extends app.View {
|
||||||
|
static initClass() {
|
||||||
|
this.events =
|
||||||
|
{click: 'onClick'};
|
||||||
|
}
|
||||||
|
|
||||||
constructor: (@el, @entry) -> super
|
constructor(el, entry) { this.onClick = this.onClick.bind(this); this.el = el; this.entry = entry; super(...arguments); }
|
||||||
|
|
||||||
init: ->
|
init() {
|
||||||
@addSubview @notice = new app.views.Notice 'disabledDoc'
|
this.addSubview(this.notice = new app.views.Notice('disabledDoc'));
|
||||||
@activate()
|
this.activate();
|
||||||
return
|
}
|
||||||
|
|
||||||
onClick: (event) =>
|
onClick(event) {
|
||||||
if link = $.closestLink(event.target, @el)
|
let link;
|
||||||
$.stopEvent(event)
|
if (link = $.closestLink(event.target, this.el)) {
|
||||||
$.popup(link)
|
$.stopEvent(event);
|
||||||
return
|
$.popup(link);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,57 +1,81 @@
|
|||||||
#= require views/pages/base
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
class app.views.JqueryPage extends app.views.BasePage
|
* DS002: Fix invalid constructor
|
||||||
@demoClassName: '_jquery-demo'
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
afterRender: ->
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
# Prevent jQuery Mobile's demo iframes from scrolling the page
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
for iframe in @findAllByTag 'iframe'
|
*/
|
||||||
iframe.style.display = 'none'
|
//= require views/pages/base
|
||||||
$.on iframe, 'load', @onIframeLoaded
|
|
||||||
|
const Cls = (app.views.JqueryPage = class JqueryPage extends app.views.BasePage {
|
||||||
@runExamples()
|
constructor(...args) {
|
||||||
|
this.onIframeLoaded = this.onIframeLoaded.bind(this);
|
||||||
onIframeLoaded: (event) =>
|
super(...args);
|
||||||
event.target.style.display = ''
|
}
|
||||||
$.off event.target, 'load', @onIframeLoaded
|
|
||||||
return
|
static initClass() {
|
||||||
|
this.demoClassName = '_jquery-demo';
|
||||||
runExamples: ->
|
}
|
||||||
for el in @findAllByClass 'entry-example'
|
|
||||||
try @runExample el catch
|
afterRender() {
|
||||||
return
|
// Prevent jQuery Mobile's demo iframes from scrolling the page
|
||||||
|
for (var iframe of Array.from(this.findAllByTag('iframe'))) {
|
||||||
runExample: (el) ->
|
iframe.style.display = 'none';
|
||||||
source = el.getElementsByClassName('syntaxhighlighter')[0]
|
$.on(iframe, 'load', this.onIframeLoaded);
|
||||||
return unless source and source.innerHTML.indexOf('!doctype') isnt -1
|
}
|
||||||
|
|
||||||
unless iframe = el.getElementsByClassName(@constructor.demoClassName)[0]
|
return this.runExamples();
|
||||||
iframe = document.createElement 'iframe'
|
}
|
||||||
iframe.className = @constructor.demoClassName
|
|
||||||
iframe.width = '100%'
|
onIframeLoaded(event) {
|
||||||
iframe.height = 200
|
event.target.style.display = '';
|
||||||
el.appendChild(iframe)
|
$.off(event.target, 'load', this.onIframeLoaded);
|
||||||
|
}
|
||||||
doc = iframe.contentDocument
|
|
||||||
doc.write @fixIframeSource(source.textContent)
|
runExamples() {
|
||||||
doc.close()
|
for (var el of Array.from(this.findAllByClass('entry-example'))) {
|
||||||
return
|
try { this.runExample(el); } catch (error) {}
|
||||||
|
}
|
||||||
fixIframeSource: (source) ->
|
}
|
||||||
source = source.replace '"/resources/', '"https://api.jquery.com/resources/' # attr(), keydown()
|
|
||||||
source = source.replace '</head>', """
|
runExample(el) {
|
||||||
<style>
|
let iframe;
|
||||||
html, body { border: 0; margin: 0; padding: 0; }
|
const source = el.getElementsByClassName('syntaxhighlighter')[0];
|
||||||
body { font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; }
|
if (!source || (source.innerHTML.indexOf('!doctype') === -1)) { return; }
|
||||||
</style>
|
|
||||||
<script>
|
if (!(iframe = el.getElementsByClassName(this.constructor.demoClassName)[0])) {
|
||||||
$.ajaxPrefilter(function(opt, opt2, xhr) {
|
iframe = document.createElement('iframe');
|
||||||
if (opt.url.indexOf('http') !== 0) {
|
iframe.className = this.constructor.demoClassName;
|
||||||
xhr.abort();
|
iframe.width = '100%';
|
||||||
document.body.innerHTML = "<p><strong>This demo cannot run inside DevDocs.</strong></p>";
|
iframe.height = 200;
|
||||||
}
|
el.appendChild(iframe);
|
||||||
});
|
}
|
||||||
</script>
|
|
||||||
</head>
|
const doc = iframe.contentDocument;
|
||||||
"""
|
doc.write(this.fixIframeSource(source.textContent));
|
||||||
source.replace /<script>/gi, '<script nonce="devdocs">'
|
doc.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
fixIframeSource(source) {
|
||||||
|
source = source.replace('"/resources/', '"https://api.jquery.com/resources/'); // attr(), keydown()
|
||||||
|
source = source.replace('</head>', `\
|
||||||
|
<style>
|
||||||
|
html, body { border: 0; margin: 0; padding: 0; }
|
||||||
|
body { font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; }
|
||||||
|
</style>
|
||||||
|
<script>
|
||||||
|
$.ajaxPrefilter(function(opt, opt2, xhr) {
|
||||||
|
if (opt.url.indexOf('http') !== 0) {
|
||||||
|
xhr.abort();
|
||||||
|
document.body.innerHTML = "<p><strong>This demo cannot run inside DevDocs.</strong></p>";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</head>\
|
||||||
|
`
|
||||||
|
);
|
||||||
|
return source.replace(/<script>/gi, '<script nonce="devdocs">');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,15 +1,26 @@
|
|||||||
#= require views/pages/base
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require views/pages/base
|
||||||
|
|
||||||
class app.views.RdocPage extends app.views.BasePage
|
const Cls = (app.views.RdocPage = class RdocPage extends app.views.BasePage {
|
||||||
@events:
|
static initClass() {
|
||||||
click: 'onClick'
|
this.events =
|
||||||
|
{click: 'onClick'};
|
||||||
|
}
|
||||||
|
|
||||||
onClick: (event) ->
|
onClick(event) {
|
||||||
return unless event.target.classList.contains 'method-click-advice'
|
if (!event.target.classList.contains('method-click-advice')) { return; }
|
||||||
$.stopEvent(event)
|
$.stopEvent(event);
|
||||||
|
|
||||||
source = $ '.method-source-code', event.target.closest('.method-detail')
|
const source = $('.method-source-code', event.target.closest('.method-detail'));
|
||||||
isShown = source.style.display is 'block'
|
const isShown = source.style.display === 'block';
|
||||||
|
|
||||||
source.style.display = if isShown then 'none' else 'block'
|
source.style.display = isShown ? 'none' : 'block';
|
||||||
event.target.textContent = if isShown then 'Show source' else 'Hide source'
|
return event.target.textContent = isShown ? 'Show source' : 'Hide source';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,17 +1,34 @@
|
|||||||
#= require views/pages/base
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require views/pages/base
|
||||||
|
|
||||||
class app.views.SqlitePage extends app.views.BasePage
|
const Cls = (app.views.SqlitePage = class SqlitePage extends app.views.BasePage {
|
||||||
@events:
|
constructor(...args) {
|
||||||
click: 'onClick'
|
this.onClick = this.onClick.bind(this);
|
||||||
|
super(...args);
|
||||||
|
}
|
||||||
|
|
||||||
onClick: (event) =>
|
static initClass() {
|
||||||
return unless id = event.target.getAttribute('data-toggle')
|
this.events =
|
||||||
return unless el = @find("##{id}")
|
{click: 'onClick'};
|
||||||
$.stopEvent(event)
|
}
|
||||||
if el.style.display == 'none'
|
|
||||||
el.style.display = 'block'
|
onClick(event) {
|
||||||
event.target.textContent = 'hide'
|
let el, id;
|
||||||
else
|
if (!(id = event.target.getAttribute('data-toggle'))) { return; }
|
||||||
el.style.display = 'none'
|
if (!(el = this.find(`#${id}`))) { return; }
|
||||||
event.target.textContent = 'show'
|
$.stopEvent(event);
|
||||||
return
|
if (el.style.display === 'none') {
|
||||||
|
el.style.display = 'block';
|
||||||
|
event.target.textContent = 'hide';
|
||||||
|
} else {
|
||||||
|
el.style.display = 'none';
|
||||||
|
event.target.textContent = 'show';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,14 +1,23 @@
|
|||||||
#= require views/pages/base
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require views/pages/base
|
||||||
|
|
||||||
class app.views.SupportTablesPage extends app.views.BasePage
|
const Cls = (app.views.SupportTablesPage = class SupportTablesPage extends app.views.BasePage {
|
||||||
@events:
|
static initClass() {
|
||||||
click: 'onClick'
|
this.events =
|
||||||
|
{click: 'onClick'};
|
||||||
|
}
|
||||||
|
|
||||||
onClick: (event) ->
|
onClick(event) {
|
||||||
return unless event.target.classList.contains 'show-all'
|
if (!event.target.classList.contains('show-all')) { return; }
|
||||||
$.stopEvent(event)
|
$.stopEvent(event);
|
||||||
|
|
||||||
el = event.target
|
let el = event.target;
|
||||||
el = el.parentNode until el.tagName is 'TABLE'
|
while (el.tagName !== 'TABLE') { el = el.parentNode; }
|
||||||
el.classList.add 'show-all'
|
el.classList.add('show-all');
|
||||||
return
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,168 +1,225 @@
|
|||||||
class app.views.Search extends app.View
|
/*
|
||||||
SEARCH_PARAM = app.config.search_param
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@el: '._search'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
@activeClass: '_search-active'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@elements:
|
* DS207: Consider shorter variations of null checks
|
||||||
input: '._search-input'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
resetLink: '._search-clear'
|
*/
|
||||||
|
(function() {
|
||||||
@events:
|
let SEARCH_PARAM = undefined;
|
||||||
input: 'onInput'
|
let HASH_RGX = undefined;
|
||||||
click: 'onClick'
|
const Cls = (app.views.Search = class Search extends app.View {
|
||||||
submit: 'onSubmit'
|
constructor(...args) {
|
||||||
|
this.focus = this.focus.bind(this);
|
||||||
@shortcuts:
|
this.autoFocus = this.autoFocus.bind(this);
|
||||||
typing: 'focus'
|
this.onWindowFocus = this.onWindowFocus.bind(this);
|
||||||
altG: 'google'
|
this.onReady = this.onReady.bind(this);
|
||||||
altS: 'stackoverflow'
|
this.onInput = this.onInput.bind(this);
|
||||||
altD: 'duckduckgo'
|
this.searchUrl = this.searchUrl.bind(this);
|
||||||
|
this.google = this.google.bind(this);
|
||||||
@routes:
|
this.stackoverflow = this.stackoverflow.bind(this);
|
||||||
after: 'afterRoute'
|
this.duckduckgo = this.duckduckgo.bind(this);
|
||||||
|
this.onResults = this.onResults.bind(this);
|
||||||
init: ->
|
this.onEnd = this.onEnd.bind(this);
|
||||||
@addSubview @scope = new app.views.SearchScope @el
|
this.onClick = this.onClick.bind(this);
|
||||||
|
this.onScopeChange = this.onScopeChange.bind(this);
|
||||||
@searcher = new app.Searcher
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
@searcher
|
super(...args);
|
||||||
.on 'results', @onResults
|
}
|
||||||
.on 'end', @onEnd
|
|
||||||
|
static initClass() {
|
||||||
@scope
|
SEARCH_PARAM = app.config.search_param;
|
||||||
.on 'change', @onScopeChange
|
|
||||||
|
this.el = '._search';
|
||||||
app.on 'ready', @onReady
|
this.activeClass = '_search-active';
|
||||||
$.on window, 'hashchange', @searchUrl
|
|
||||||
$.on window, 'focus', @onWindowFocus
|
this.elements = {
|
||||||
return
|
input: '._search-input',
|
||||||
|
resetLink: '._search-clear'
|
||||||
focus: =>
|
};
|
||||||
return if document.activeElement is @input
|
|
||||||
return if app.settings.get('noAutofocus')
|
this.events = {
|
||||||
@input.focus()
|
input: 'onInput',
|
||||||
return
|
click: 'onClick',
|
||||||
|
submit: 'onSubmit'
|
||||||
autoFocus: =>
|
};
|
||||||
return if app.isMobile() or $.isAndroid() or $.isIOS()
|
|
||||||
return if document.activeElement?.tagName is 'INPUT'
|
this.shortcuts = {
|
||||||
return if app.settings.get('noAutofocus')
|
typing: 'focus',
|
||||||
@input.focus()
|
altG: 'google',
|
||||||
return
|
altS: 'stackoverflow',
|
||||||
|
altD: 'duckduckgo'
|
||||||
onWindowFocus: (event) =>
|
};
|
||||||
@autoFocus() if event.target is window
|
|
||||||
|
this.routes =
|
||||||
getScopeDoc: ->
|
{after: 'afterRoute'};
|
||||||
@scope.getScope() if @scope.isActive()
|
|
||||||
|
HASH_RGX = new RegExp(`^#${SEARCH_PARAM}=(.*)`);
|
||||||
reset: (force) ->
|
}
|
||||||
@scope.reset() if force or not @input.value
|
|
||||||
@el.reset()
|
init() {
|
||||||
@onInput()
|
this.addSubview(this.scope = new app.views.SearchScope(this.el));
|
||||||
@autoFocus()
|
|
||||||
return
|
this.searcher = new app.Searcher;
|
||||||
|
this.searcher
|
||||||
onReady: =>
|
.on('results', this.onResults)
|
||||||
@value = ''
|
.on('end', this.onEnd);
|
||||||
@delay @onInput
|
|
||||||
return
|
this.scope
|
||||||
|
.on('change', this.onScopeChange);
|
||||||
onInput: =>
|
|
||||||
return if not @value? or # ignore events pre-"ready"
|
app.on('ready', this.onReady);
|
||||||
@value is @input.value
|
$.on(window, 'hashchange', this.searchUrl);
|
||||||
@value = @input.value
|
$.on(window, 'focus', this.onWindowFocus);
|
||||||
|
}
|
||||||
if @value.length
|
|
||||||
@search()
|
focus() {
|
||||||
else
|
if (document.activeElement === this.input) { return; }
|
||||||
@clear()
|
if (app.settings.get('noAutofocus')) { return; }
|
||||||
return
|
this.input.focus();
|
||||||
|
}
|
||||||
search: (url = false) ->
|
|
||||||
@addClass @constructor.activeClass
|
autoFocus() {
|
||||||
@trigger 'searching'
|
if (app.isMobile() || $.isAndroid() || $.isIOS()) { return; }
|
||||||
|
if ((document.activeElement != null ? document.activeElement.tagName : undefined) === 'INPUT') { return; }
|
||||||
@hasResults = null
|
if (app.settings.get('noAutofocus')) { return; }
|
||||||
@flags = urlSearch: url, initialResults: true
|
this.input.focus();
|
||||||
@searcher.find @scope.getScope().entries.all(), 'text', @value
|
}
|
||||||
return
|
|
||||||
|
onWindowFocus(event) {
|
||||||
searchUrl: =>
|
if (event.target === window) { return this.autoFocus(); }
|
||||||
if location.pathname is '/'
|
}
|
||||||
@scope.searchUrl()
|
|
||||||
else if not app.router.isIndex()
|
getScopeDoc() {
|
||||||
return
|
if (this.scope.isActive()) { return this.scope.getScope(); }
|
||||||
|
}
|
||||||
return unless value = @extractHashValue()
|
|
||||||
@input.value = @value = value
|
reset(force) {
|
||||||
@input.setSelectionRange(value.length, value.length)
|
if (force || !this.input.value) { this.scope.reset(); }
|
||||||
@search true
|
this.el.reset();
|
||||||
true
|
this.onInput();
|
||||||
|
this.autoFocus();
|
||||||
clear: ->
|
}
|
||||||
@removeClass @constructor.activeClass
|
|
||||||
@trigger 'clear'
|
onReady() {
|
||||||
return
|
this.value = '';
|
||||||
|
this.delay(this.onInput);
|
||||||
externalSearch: (url) ->
|
}
|
||||||
if value = @value
|
|
||||||
value = "#{@scope.name()} #{value}" if @scope.name()
|
onInput() {
|
||||||
$.popup "#{url}#{encodeURIComponent value}"
|
if ((this.value == null) || // ignore events pre-"ready"
|
||||||
@reset()
|
(this.value === this.input.value)) { return; }
|
||||||
return
|
this.value = this.input.value;
|
||||||
|
|
||||||
google: =>
|
if (this.value.length) {
|
||||||
@externalSearch "https://www.google.com/search?q="
|
this.search();
|
||||||
return
|
} else {
|
||||||
|
this.clear();
|
||||||
stackoverflow: =>
|
}
|
||||||
@externalSearch "https://stackoverflow.com/search?q="
|
}
|
||||||
return
|
|
||||||
|
search(url) {
|
||||||
duckduckgo: =>
|
if (url == null) { url = false; }
|
||||||
@externalSearch "https://duckduckgo.com/?t=devdocs&q="
|
this.addClass(this.constructor.activeClass);
|
||||||
return
|
this.trigger('searching');
|
||||||
|
|
||||||
onResults: (results) =>
|
this.hasResults = null;
|
||||||
@hasResults = true if results.length
|
this.flags = {urlSearch: url, initialResults: true};
|
||||||
@trigger 'results', results, @flags
|
this.searcher.find(this.scope.getScope().entries.all(), 'text', this.value);
|
||||||
@flags.initialResults = false
|
}
|
||||||
return
|
|
||||||
|
searchUrl() {
|
||||||
onEnd: =>
|
let value;
|
||||||
@trigger 'noresults' unless @hasResults
|
if (location.pathname === '/') {
|
||||||
return
|
this.scope.searchUrl();
|
||||||
|
} else if (!app.router.isIndex()) {
|
||||||
onClick: (event) =>
|
return;
|
||||||
if event.target is @resetLink
|
}
|
||||||
$.stopEvent(event)
|
|
||||||
@reset()
|
if (!(value = this.extractHashValue())) { return; }
|
||||||
return
|
this.input.value = (this.value = value);
|
||||||
|
this.input.setSelectionRange(value.length, value.length);
|
||||||
onSubmit: (event) ->
|
this.search(true);
|
||||||
$.stopEvent(event)
|
return true;
|
||||||
return
|
}
|
||||||
|
|
||||||
onScopeChange: =>
|
clear() {
|
||||||
@value = ''
|
this.removeClass(this.constructor.activeClass);
|
||||||
@onInput()
|
this.trigger('clear');
|
||||||
return
|
}
|
||||||
|
|
||||||
afterRoute: (name, context) =>
|
externalSearch(url) {
|
||||||
return if app.shortcuts.eventInProgress?.name is 'escape'
|
let value;
|
||||||
@reset(true) if not context.init and app.router.isIndex()
|
if (value = this.value) {
|
||||||
@delay @searchUrl if context.hash
|
if (this.scope.name()) { value = `${this.scope.name()} ${value}`; }
|
||||||
$.requestAnimationFrame @autoFocus
|
$.popup(`${url}${encodeURIComponent(value)}`);
|
||||||
return
|
this.reset();
|
||||||
|
}
|
||||||
extractHashValue: ->
|
}
|
||||||
if (value = @getHashValue())?
|
|
||||||
app.router.replaceHash()
|
google() {
|
||||||
value
|
this.externalSearch("https://www.google.com/search?q=");
|
||||||
|
}
|
||||||
HASH_RGX = new RegExp "^##{SEARCH_PARAM}=(.*)"
|
|
||||||
|
stackoverflow() {
|
||||||
getHashValue: ->
|
this.externalSearch("https://stackoverflow.com/search?q=");
|
||||||
try HASH_RGX.exec($.urlDecode location.hash)?[1] catch
|
}
|
||||||
|
|
||||||
|
duckduckgo() {
|
||||||
|
this.externalSearch("https://duckduckgo.com/?t=devdocs&q=");
|
||||||
|
}
|
||||||
|
|
||||||
|
onResults(results) {
|
||||||
|
if (results.length) { this.hasResults = true; }
|
||||||
|
this.trigger('results', results, this.flags);
|
||||||
|
this.flags.initialResults = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
onEnd() {
|
||||||
|
if (!this.hasResults) { this.trigger('noresults'); }
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
if (event.target === this.resetLink) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
this.reset();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onSubmit(event) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
onScopeChange() {
|
||||||
|
this.value = '';
|
||||||
|
this.onInput();
|
||||||
|
}
|
||||||
|
|
||||||
|
afterRoute(name, context) {
|
||||||
|
if ((app.shortcuts.eventInProgress != null ? app.shortcuts.eventInProgress.name : undefined) === 'escape') { return; }
|
||||||
|
if (!context.init && app.router.isIndex()) { this.reset(true); }
|
||||||
|
if (context.hash) { this.delay(this.searchUrl); }
|
||||||
|
$.requestAnimationFrame(this.autoFocus);
|
||||||
|
}
|
||||||
|
|
||||||
|
extractHashValue() {
|
||||||
|
let value;
|
||||||
|
if ((value = this.getHashValue()) != null) {
|
||||||
|
app.router.replaceHash();
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getHashValue() {
|
||||||
|
try { return __guard__(HASH_RGX.exec($.urlDecode(location.hash)), x => x[1]); } catch (error) {}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,135 +1,180 @@
|
|||||||
class app.views.SearchScope extends app.View
|
/*
|
||||||
SEARCH_PARAM = app.config.search_param
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@elements:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
input: '._search-input'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
tag: '._search-tag'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
@events:
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
click: 'onClick'
|
*/
|
||||||
keydown: 'onKeydown'
|
(function() {
|
||||||
textInput: 'onTextInput'
|
let SEARCH_PARAM = undefined;
|
||||||
|
let HASH_RGX = undefined;
|
||||||
@routes:
|
const Cls = (app.views.SearchScope = class SearchScope extends app.View {
|
||||||
after: 'afterRoute'
|
static initClass() {
|
||||||
|
SEARCH_PARAM = app.config.search_param;
|
||||||
constructor: (@el) -> super
|
|
||||||
|
this.elements = {
|
||||||
init: ->
|
input: '._search-input',
|
||||||
@placeholder = @input.getAttribute 'placeholder'
|
tag: '._search-tag'
|
||||||
|
};
|
||||||
@searcher = new app.SynchronousSearcher
|
|
||||||
fuzzy_min_length: 2
|
this.events = {
|
||||||
max_results: 1
|
click: 'onClick',
|
||||||
@searcher.on 'results', @onResults
|
keydown: 'onKeydown',
|
||||||
|
textInput: 'onTextInput'
|
||||||
return
|
};
|
||||||
|
|
||||||
getScope: ->
|
this.routes =
|
||||||
@doc or app
|
{after: 'afterRoute'};
|
||||||
|
|
||||||
isActive: ->
|
HASH_RGX = new RegExp(`^#${SEARCH_PARAM}=(.+?) .`);
|
||||||
!!@doc
|
}
|
||||||
|
|
||||||
name: ->
|
constructor(el) { this.onResults = this.onResults.bind(this); this.reset = this.reset.bind(this); this.doScopeSearch = this.doScopeSearch.bind(this); this.onClick = this.onClick.bind(this); this.onKeydown = this.onKeydown.bind(this); this.onTextInput = this.onTextInput.bind(this); this.afterRoute = this.afterRoute.bind(this); this.el = el; super(...arguments); }
|
||||||
@doc?.name
|
|
||||||
|
init() {
|
||||||
search: (value, searchDisabled = false) ->
|
this.placeholder = this.input.getAttribute('placeholder');
|
||||||
return if @doc
|
|
||||||
@searcher.find app.docs.all(), 'text', value
|
this.searcher = new app.SynchronousSearcher({
|
||||||
@searcher.find app.disabledDocs.all(), 'text', value if not @doc and searchDisabled
|
fuzzy_min_length: 2,
|
||||||
return
|
max_results: 1
|
||||||
|
});
|
||||||
searchUrl: ->
|
this.searcher.on('results', this.onResults);
|
||||||
if value = @extractHashValue()
|
|
||||||
@search value, true
|
}
|
||||||
return
|
|
||||||
|
getScope() {
|
||||||
onResults: (results) =>
|
return this.doc || app;
|
||||||
return unless doc = results[0]
|
}
|
||||||
if app.docs.contains(doc)
|
|
||||||
@selectDoc(doc)
|
isActive() {
|
||||||
else
|
return !!this.doc;
|
||||||
@redirectToDoc(doc)
|
}
|
||||||
return
|
|
||||||
|
name() {
|
||||||
selectDoc: (doc) ->
|
return (this.doc != null ? this.doc.name : undefined);
|
||||||
previousDoc = @doc
|
}
|
||||||
return if doc is previousDoc
|
|
||||||
@doc = doc
|
search(value, searchDisabled) {
|
||||||
|
if (searchDisabled == null) { searchDisabled = false; }
|
||||||
@tag.textContent = doc.fullName
|
if (this.doc) { return; }
|
||||||
@tag.style.display = 'block'
|
this.searcher.find(app.docs.all(), 'text', value);
|
||||||
|
if (!this.doc && searchDisabled) { this.searcher.find(app.disabledDocs.all(), 'text', value); }
|
||||||
@input.removeAttribute 'placeholder'
|
}
|
||||||
@input.value = @input.value[@input.selectionStart..]
|
|
||||||
@input.style.paddingLeft = @tag.offsetWidth + 10 + 'px'
|
searchUrl() {
|
||||||
|
let value;
|
||||||
$.trigger @input, 'input'
|
if (value = this.extractHashValue()) {
|
||||||
@trigger 'change', @doc, previousDoc
|
this.search(value, true);
|
||||||
return
|
}
|
||||||
|
}
|
||||||
redirectToDoc: (doc) ->
|
|
||||||
hash = location.hash
|
onResults(results) {
|
||||||
app.router.replaceHash('')
|
let doc;
|
||||||
location.assign doc.fullPath() + hash
|
if (!(doc = results[0])) { return; }
|
||||||
return
|
if (app.docs.contains(doc)) {
|
||||||
|
this.selectDoc(doc);
|
||||||
reset: =>
|
} else {
|
||||||
return unless @doc
|
this.redirectToDoc(doc);
|
||||||
previousDoc = @doc
|
}
|
||||||
@doc = null
|
}
|
||||||
|
|
||||||
@tag.textContent = ''
|
selectDoc(doc) {
|
||||||
@tag.style.display = 'none'
|
const previousDoc = this.doc;
|
||||||
|
if (doc === previousDoc) { return; }
|
||||||
@input.setAttribute 'placeholder', @placeholder
|
this.doc = doc;
|
||||||
@input.style.paddingLeft = ''
|
|
||||||
|
this.tag.textContent = doc.fullName;
|
||||||
@trigger 'change', null, previousDoc
|
this.tag.style.display = 'block';
|
||||||
return
|
|
||||||
|
this.input.removeAttribute('placeholder');
|
||||||
doScopeSearch: (event) =>
|
this.input.value = this.input.value.slice(this.input.selectionStart);
|
||||||
@search @input.value[0...@input.selectionStart]
|
this.input.style.paddingLeft = this.tag.offsetWidth + 10 + 'px';
|
||||||
$.stopEvent(event) if @doc
|
|
||||||
return
|
$.trigger(this.input, 'input');
|
||||||
|
this.trigger('change', this.doc, previousDoc);
|
||||||
onClick: (event) =>
|
}
|
||||||
if event.target is @tag
|
|
||||||
@reset()
|
redirectToDoc(doc) {
|
||||||
$.stopEvent(event)
|
const {
|
||||||
return
|
hash
|
||||||
|
} = location;
|
||||||
onKeydown: (event) =>
|
app.router.replaceHash('');
|
||||||
if event.which is 8 # backspace
|
location.assign(doc.fullPath() + hash);
|
||||||
if @doc and @input.selectionEnd is 0
|
}
|
||||||
@reset()
|
|
||||||
$.stopEvent(event)
|
reset() {
|
||||||
else if not @doc and @input.value and not $.isChromeForAndroid()
|
if (!this.doc) { return; }
|
||||||
return if event.ctrlKey or event.metaKey or event.altKey or event.shiftKey
|
const previousDoc = this.doc;
|
||||||
if event.which is 9 or # tab
|
this.doc = null;
|
||||||
(event.which is 32 and app.isMobile()) # space
|
|
||||||
@doScopeSearch(event)
|
this.tag.textContent = '';
|
||||||
return
|
this.tag.style.display = 'none';
|
||||||
|
|
||||||
onTextInput: (event) =>
|
this.input.setAttribute('placeholder', this.placeholder);
|
||||||
return unless $.isChromeForAndroid()
|
this.input.style.paddingLeft = '';
|
||||||
if not @doc and @input.value and event.data == ' '
|
|
||||||
@doScopeSearch(event)
|
this.trigger('change', null, previousDoc);
|
||||||
return
|
}
|
||||||
|
|
||||||
extractHashValue: ->
|
doScopeSearch(event) {
|
||||||
if value = @getHashValue()
|
this.search(this.input.value.slice(0, this.input.selectionStart));
|
||||||
newHash = $.urlDecode(location.hash).replace "##{SEARCH_PARAM}=#{value} ", "##{SEARCH_PARAM}="
|
if (this.doc) { $.stopEvent(event); }
|
||||||
app.router.replaceHash(newHash)
|
}
|
||||||
value
|
|
||||||
|
onClick(event) {
|
||||||
HASH_RGX = new RegExp "^##{SEARCH_PARAM}=(.+?) ."
|
if (event.target === this.tag) {
|
||||||
|
this.reset();
|
||||||
getHashValue: ->
|
$.stopEvent(event);
|
||||||
try HASH_RGX.exec($.urlDecode location.hash)?[1] catch
|
}
|
||||||
|
}
|
||||||
afterRoute: (name, context) =>
|
|
||||||
if !app.isSingleDoc() and context.init and context.doc
|
onKeydown(event) {
|
||||||
@selectDoc(context.doc)
|
if (event.which === 8) { // backspace
|
||||||
return
|
if (this.doc && (this.input.selectionEnd === 0)) {
|
||||||
|
this.reset();
|
||||||
|
$.stopEvent(event);
|
||||||
|
}
|
||||||
|
} else if (!this.doc && this.input.value && !$.isChromeForAndroid()) {
|
||||||
|
if (event.ctrlKey || event.metaKey || event.altKey || event.shiftKey) { return; }
|
||||||
|
if ((event.which === 9) || // tab
|
||||||
|
((event.which === 32) && app.isMobile())) { // space
|
||||||
|
this.doScopeSearch(event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onTextInput(event) {
|
||||||
|
if (!$.isChromeForAndroid()) { return; }
|
||||||
|
if (!this.doc && this.input.value && (event.data === ' ')) {
|
||||||
|
this.doScopeSearch(event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extractHashValue() {
|
||||||
|
let value;
|
||||||
|
if (value = this.getHashValue()) {
|
||||||
|
const newHash = $.urlDecode(location.hash).replace(`#${SEARCH_PARAM}=${value} `, `#${SEARCH_PARAM}=`);
|
||||||
|
app.router.replaceHash(newHash);
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getHashValue() {
|
||||||
|
try { return __guard__(HASH_RGX.exec($.urlDecode(location.hash)), x => x[1]); } catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
afterRoute(name, context) {
|
||||||
|
if (!app.isSingleDoc() && context.init && context.doc) {
|
||||||
|
this.selectDoc(context.doc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
return Cls;
|
||||||
|
})();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,187 +1,234 @@
|
|||||||
class app.views.DocList extends app.View
|
/*
|
||||||
@className: '_list'
|
* decaffeinate suggestions:
|
||||||
@attributes:
|
* DS002: Fix invalid constructor
|
||||||
role: 'navigation'
|
* DS101: Remove unnecessary use of Array.from
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@events:
|
* DS207: Consider shorter variations of null checks
|
||||||
open: 'onOpen'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
close: 'onClose'
|
*/
|
||||||
click: 'onClick'
|
const Cls = (app.views.DocList = class DocList extends app.View {
|
||||||
|
constructor(...args) {
|
||||||
@routes:
|
this.render = this.render.bind(this);
|
||||||
after: 'afterRoute'
|
this.onOpen = this.onOpen.bind(this);
|
||||||
|
this.onClose = this.onClose.bind(this);
|
||||||
@elements:
|
this.onClick = this.onClick.bind(this);
|
||||||
disabledTitle: '._list-title'
|
this.onEnabled = this.onEnabled.bind(this);
|
||||||
disabledList: '._disabled-list'
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
|
super(...args);
|
||||||
init: ->
|
}
|
||||||
@lists = {}
|
|
||||||
|
static initClass() {
|
||||||
@addSubview @listFocus = new app.views.ListFocus @el
|
this.className = '_list';
|
||||||
@addSubview @listFold = new app.views.ListFold @el
|
this.attributes =
|
||||||
@addSubview @listSelect = new app.views.ListSelect @el
|
{role: 'navigation'};
|
||||||
|
|
||||||
app.on 'ready', @render
|
this.events = {
|
||||||
return
|
open: 'onOpen',
|
||||||
|
close: 'onClose',
|
||||||
activate: ->
|
click: 'onClick'
|
||||||
if super
|
};
|
||||||
list.activate() for slug, list of @lists
|
|
||||||
@listSelect.selectCurrent()
|
this.routes =
|
||||||
return
|
{after: 'afterRoute'};
|
||||||
|
|
||||||
deactivate: ->
|
this.elements = {
|
||||||
if super
|
disabledTitle: '._list-title',
|
||||||
list.deactivate() for slug, list of @lists
|
disabledList: '._disabled-list'
|
||||||
return
|
};
|
||||||
|
}
|
||||||
render: =>
|
|
||||||
html = ''
|
init() {
|
||||||
for doc in app.docs.all()
|
this.lists = {};
|
||||||
html += @tmpl('sidebarDoc', doc, fullName: app.docs.countAllBy('name', doc.name) > 1)
|
|
||||||
@html html
|
this.addSubview(this.listFocus = new app.views.ListFocus(this.el));
|
||||||
@renderDisabled() unless app.isSingleDoc() or app.disabledDocs.size() is 0
|
this.addSubview(this.listFold = new app.views.ListFold(this.el));
|
||||||
return
|
this.addSubview(this.listSelect = new app.views.ListSelect(this.el));
|
||||||
|
|
||||||
renderDisabled: ->
|
app.on('ready', this.render);
|
||||||
@append @tmpl('sidebarDisabled', count: app.disabledDocs.size())
|
}
|
||||||
@refreshElements()
|
|
||||||
@renderDisabledList()
|
activate() {
|
||||||
return
|
if (super.activate(...arguments)) {
|
||||||
|
for (var slug in this.lists) { var list = this.lists[slug]; list.activate(); }
|
||||||
renderDisabledList: ->
|
this.listSelect.selectCurrent();
|
||||||
if app.settings.get('hideDisabled')
|
}
|
||||||
@removeDisabledList()
|
}
|
||||||
else
|
|
||||||
@appendDisabledList()
|
deactivate() {
|
||||||
return
|
if (super.deactivate(...arguments)) {
|
||||||
|
for (var slug in this.lists) { var list = this.lists[slug]; list.deactivate(); }
|
||||||
appendDisabledList: ->
|
}
|
||||||
html = ''
|
}
|
||||||
docs = [].concat(app.disabledDocs.all()...)
|
|
||||||
|
render() {
|
||||||
while doc = docs.shift()
|
let html = '';
|
||||||
if doc.version?
|
for (var doc of Array.from(app.docs.all())) {
|
||||||
versions = ''
|
html += this.tmpl('sidebarDoc', doc, {fullName: app.docs.countAllBy('name', doc.name) > 1});
|
||||||
loop
|
}
|
||||||
versions += @tmpl('sidebarDoc', doc, disabled: true)
|
this.html(html);
|
||||||
break if docs[0]?.name isnt doc.name
|
if (!app.isSingleDoc() && (app.disabledDocs.size() !== 0)) { this.renderDisabled(); }
|
||||||
doc = docs.shift()
|
}
|
||||||
html += @tmpl('sidebarDisabledVersionedDoc', doc, versions)
|
|
||||||
else
|
renderDisabled() {
|
||||||
html += @tmpl('sidebarDoc', doc, disabled: true)
|
this.append(this.tmpl('sidebarDisabled', {count: app.disabledDocs.size()}));
|
||||||
|
this.refreshElements();
|
||||||
@append @tmpl('sidebarDisabledList', html)
|
this.renderDisabledList();
|
||||||
@disabledTitle.classList.add('open-title')
|
}
|
||||||
@refreshElements()
|
|
||||||
return
|
renderDisabledList() {
|
||||||
|
if (app.settings.get('hideDisabled')) {
|
||||||
removeDisabledList: ->
|
this.removeDisabledList();
|
||||||
$.remove @disabledList if @disabledList
|
} else {
|
||||||
@disabledTitle.classList.remove('open-title')
|
this.appendDisabledList();
|
||||||
@refreshElements()
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
reset: (options = {}) ->
|
appendDisabledList() {
|
||||||
@listSelect.deselect()
|
let doc;
|
||||||
@listFocus?.blur()
|
let html = '';
|
||||||
@listFold.reset()
|
const docs = [].concat(...Array.from(app.disabledDocs.all() || []));
|
||||||
@revealCurrent() if options.revealCurrent || app.isSingleDoc()
|
|
||||||
return
|
while ((doc = docs.shift())) {
|
||||||
|
if (doc.version != null) {
|
||||||
onOpen: (event) =>
|
var versions = '';
|
||||||
$.stopEvent(event)
|
while (true) {
|
||||||
doc = app.docs.findBy 'slug', event.target.getAttribute('data-slug')
|
versions += this.tmpl('sidebarDoc', doc, {disabled: true});
|
||||||
|
if ((docs[0] != null ? docs[0].name : undefined) !== doc.name) { break; }
|
||||||
if doc and not @lists[doc.slug]
|
doc = docs.shift();
|
||||||
@lists[doc.slug] = if doc.types.isEmpty()
|
}
|
||||||
new app.views.EntryList doc.entries.all()
|
html += this.tmpl('sidebarDisabledVersionedDoc', doc, versions);
|
||||||
else
|
} else {
|
||||||
new app.views.TypeList doc
|
html += this.tmpl('sidebarDoc', doc, {disabled: true});
|
||||||
$.after event.target, @lists[doc.slug].el
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
onClose: (event) =>
|
this.append(this.tmpl('sidebarDisabledList', html));
|
||||||
$.stopEvent(event)
|
this.disabledTitle.classList.add('open-title');
|
||||||
doc = app.docs.findBy 'slug', event.target.getAttribute('data-slug')
|
this.refreshElements();
|
||||||
|
}
|
||||||
if doc and @lists[doc.slug]
|
|
||||||
@lists[doc.slug].detach()
|
removeDisabledList() {
|
||||||
delete @lists[doc.slug]
|
if (this.disabledList) { $.remove(this.disabledList); }
|
||||||
return
|
this.disabledTitle.classList.remove('open-title');
|
||||||
|
this.refreshElements();
|
||||||
select: (model) ->
|
}
|
||||||
@listSelect.selectByHref model?.fullPath()
|
|
||||||
return
|
reset(options) {
|
||||||
|
if (options == null) { options = {}; }
|
||||||
reveal: (model) ->
|
this.listSelect.deselect();
|
||||||
@openDoc model.doc
|
if (this.listFocus != null) {
|
||||||
@openType model.getType() if model.type
|
this.listFocus.blur();
|
||||||
@focus model
|
}
|
||||||
@paginateTo model
|
this.listFold.reset();
|
||||||
@scrollTo model
|
if (options.revealCurrent || app.isSingleDoc()) { this.revealCurrent(); }
|
||||||
return
|
}
|
||||||
|
|
||||||
focus: (model) ->
|
onOpen(event) {
|
||||||
@listFocus?.focus @find("a[href='#{model.fullPath()}']")
|
$.stopEvent(event);
|
||||||
return
|
const doc = app.docs.findBy('slug', event.target.getAttribute('data-slug'));
|
||||||
|
|
||||||
revealCurrent: ->
|
if (doc && !this.lists[doc.slug]) {
|
||||||
if model = app.router.context.type or app.router.context.entry
|
this.lists[doc.slug] = doc.types.isEmpty() ?
|
||||||
@reveal model
|
new app.views.EntryList(doc.entries.all())
|
||||||
@select model
|
:
|
||||||
return
|
new app.views.TypeList(doc);
|
||||||
|
$.after(event.target, this.lists[doc.slug].el);
|
||||||
openDoc: (doc) ->
|
}
|
||||||
@listFold.open @find("[data-slug='#{doc.slug_without_version}']") if app.disabledDocs.contains(doc) and doc.version
|
}
|
||||||
@listFold.open @find("[data-slug='#{doc.slug}']")
|
|
||||||
return
|
onClose(event) {
|
||||||
|
$.stopEvent(event);
|
||||||
closeDoc: (doc) ->
|
const doc = app.docs.findBy('slug', event.target.getAttribute('data-slug'));
|
||||||
@listFold.close @find("[data-slug='#{doc.slug}']")
|
|
||||||
return
|
if (doc && this.lists[doc.slug]) {
|
||||||
|
this.lists[doc.slug].detach();
|
||||||
openType: (type) ->
|
delete this.lists[doc.slug];
|
||||||
@listFold.open @lists[type.doc.slug].find("[data-slug='#{type.slug}']")
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
paginateTo: (model) ->
|
select(model) {
|
||||||
@lists[model.doc.slug]?.paginateTo(model)
|
this.listSelect.selectByHref(model != null ? model.fullPath() : undefined);
|
||||||
return
|
}
|
||||||
|
|
||||||
scrollTo: (model) ->
|
reveal(model) {
|
||||||
$.scrollTo @find("a[href='#{model.fullPath()}']"), null, 'top', margin: if app.isMobile() then 48 else 0
|
this.openDoc(model.doc);
|
||||||
return
|
if (model.type) { this.openType(model.getType()); }
|
||||||
|
this.focus(model);
|
||||||
toggleDisabled: ->
|
this.paginateTo(model);
|
||||||
if @disabledTitle.classList.contains('open-title')
|
this.scrollTo(model);
|
||||||
@removeDisabledList()
|
}
|
||||||
app.settings.set 'hideDisabled', true
|
|
||||||
else
|
focus(model) {
|
||||||
@appendDisabledList()
|
if (this.listFocus != null) {
|
||||||
app.settings.set 'hideDisabled', false
|
this.listFocus.focus(this.find(`a[href='${model.fullPath()}']`));
|
||||||
return
|
}
|
||||||
|
}
|
||||||
onClick: (event) =>
|
|
||||||
target = $.eventTarget(event)
|
revealCurrent() {
|
||||||
if @disabledTitle and $.hasChild(@disabledTitle, target) and target.tagName isnt 'A'
|
let model;
|
||||||
$.stopEvent(event)
|
if (model = app.router.context.type || app.router.context.entry) {
|
||||||
@toggleDisabled()
|
this.reveal(model);
|
||||||
else if slug = target.getAttribute('data-enable')
|
this.select(model);
|
||||||
$.stopEvent(event)
|
}
|
||||||
doc = app.disabledDocs.findBy('slug', slug)
|
}
|
||||||
app.enableDoc(doc, @onEnabled, @onEnabled) if doc
|
|
||||||
return
|
openDoc(doc) {
|
||||||
|
if (app.disabledDocs.contains(doc) && doc.version) { this.listFold.open(this.find(`[data-slug='${doc.slug_without_version}']`)); }
|
||||||
onEnabled: =>
|
this.listFold.open(this.find(`[data-slug='${doc.slug}']`));
|
||||||
@reset()
|
}
|
||||||
@render()
|
|
||||||
return
|
closeDoc(doc) {
|
||||||
|
this.listFold.close(this.find(`[data-slug='${doc.slug}']`));
|
||||||
afterRoute: (route, context) =>
|
}
|
||||||
if context.init
|
|
||||||
@reset revealCurrent: true if @activated
|
openType(type) {
|
||||||
else
|
this.listFold.open(this.lists[type.doc.slug].find(`[data-slug='${type.slug}']`));
|
||||||
@select context.type or context.entry
|
}
|
||||||
return
|
|
||||||
|
paginateTo(model) {
|
||||||
|
if (this.lists[model.doc.slug] != null) {
|
||||||
|
this.lists[model.doc.slug].paginateTo(model);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollTo(model) {
|
||||||
|
$.scrollTo(this.find(`a[href='${model.fullPath()}']`), null, 'top', {margin: app.isMobile() ? 48 : 0});
|
||||||
|
}
|
||||||
|
|
||||||
|
toggleDisabled() {
|
||||||
|
if (this.disabledTitle.classList.contains('open-title')) {
|
||||||
|
this.removeDisabledList();
|
||||||
|
app.settings.set('hideDisabled', true);
|
||||||
|
} else {
|
||||||
|
this.appendDisabledList();
|
||||||
|
app.settings.set('hideDisabled', false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
let slug;
|
||||||
|
const target = $.eventTarget(event);
|
||||||
|
if (this.disabledTitle && $.hasChild(this.disabledTitle, target) && (target.tagName !== 'A')) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
this.toggleDisabled();
|
||||||
|
} else if (slug = target.getAttribute('data-enable')) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
const doc = app.disabledDocs.findBy('slug', slug);
|
||||||
|
if (doc) { app.enableDoc(doc, this.onEnabled, this.onEnabled); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onEnabled() {
|
||||||
|
this.reset();
|
||||||
|
this.render();
|
||||||
|
}
|
||||||
|
|
||||||
|
afterRoute(route, context) {
|
||||||
|
if (context.init) {
|
||||||
|
if (this.activated) { this.reset({revealCurrent: true}); }
|
||||||
|
} else {
|
||||||
|
this.select(context.type || context.entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,88 +1,130 @@
|
|||||||
class app.views.DocPicker extends app.View
|
/*
|
||||||
@className: '_list _list-picker'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS101: Remove unnecessary use of Array.from
|
||||||
mousedown: 'onMouseDown'
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
mouseup: 'onMouseUp'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
init: ->
|
* DS207: Consider shorter variations of null checks
|
||||||
@addSubview @listFold = new app.views.ListFold(@el)
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
return
|
*/
|
||||||
|
const Cls = (app.views.DocPicker = class DocPicker extends app.View {
|
||||||
activate: ->
|
constructor(...args) {
|
||||||
if super
|
this.onMouseDown = this.onMouseDown.bind(this);
|
||||||
@render()
|
this.onMouseUp = this.onMouseUp.bind(this);
|
||||||
$.on @el, 'focus', @onDOMFocus, true
|
this.onDOMFocus = this.onDOMFocus.bind(this);
|
||||||
return
|
super(...args);
|
||||||
|
}
|
||||||
deactivate: ->
|
|
||||||
if super
|
static initClass() {
|
||||||
@empty()
|
this.className = '_list _list-picker';
|
||||||
$.off @el, 'focus', @onDOMFocus, true
|
|
||||||
@focusEl = null
|
this.events = {
|
||||||
return
|
mousedown: 'onMouseDown',
|
||||||
|
mouseup: 'onMouseUp'
|
||||||
render: ->
|
};
|
||||||
html = @tmpl('docPickerHeader')
|
}
|
||||||
docs = app.docs.all().concat(app.disabledDocs.all()...)
|
|
||||||
|
init() {
|
||||||
while doc = docs.shift()
|
this.addSubview(this.listFold = new app.views.ListFold(this.el));
|
||||||
if doc.version?
|
}
|
||||||
[docs, versions] = @extractVersions(docs, doc)
|
|
||||||
html += @tmpl('sidebarVersionedDoc', doc, @renderVersions(versions), open: app.docs.contains(doc))
|
activate() {
|
||||||
else
|
if (super.activate(...arguments)) {
|
||||||
html += @tmpl('sidebarLabel', doc, checked: app.docs.contains(doc))
|
this.render();
|
||||||
|
$.on(this.el, 'focus', this.onDOMFocus, true);
|
||||||
@html html + @tmpl('docPickerNote')
|
}
|
||||||
|
}
|
||||||
$.requestAnimationFrame => @findByTag('input')?.focus()
|
|
||||||
return
|
deactivate() {
|
||||||
|
if (super.deactivate(...arguments)) {
|
||||||
renderVersions: (docs) ->
|
this.empty();
|
||||||
html = ''
|
$.off(this.el, 'focus', this.onDOMFocus, true);
|
||||||
html += @tmpl('sidebarLabel', doc, checked: app.docs.contains(doc)) for doc in docs
|
this.focusEl = null;
|
||||||
html
|
}
|
||||||
|
}
|
||||||
extractVersions: (originalDocs, version) ->
|
|
||||||
docs = []
|
render() {
|
||||||
versions = [version]
|
let doc;
|
||||||
for doc in originalDocs
|
let html = this.tmpl('docPickerHeader');
|
||||||
(if doc.name is version.name then versions else docs).push(doc)
|
let docs = app.docs.all().concat(...Array.from(app.disabledDocs.all() || []));
|
||||||
[docs, versions]
|
|
||||||
|
while ((doc = docs.shift())) {
|
||||||
empty: ->
|
if (doc.version != null) {
|
||||||
@resetClass()
|
var versions;
|
||||||
super
|
[docs, versions] = Array.from(this.extractVersions(docs, doc));
|
||||||
return
|
html += this.tmpl('sidebarVersionedDoc', doc, this.renderVersions(versions), {open: app.docs.contains(doc)});
|
||||||
|
} else {
|
||||||
getSelectedDocs: ->
|
html += this.tmpl('sidebarLabel', doc, {checked: app.docs.contains(doc)});
|
||||||
for input in @findAllByTag 'input' when input?.checked
|
}
|
||||||
input.name
|
}
|
||||||
|
|
||||||
onMouseDown: =>
|
this.html(html + this.tmpl('docPickerNote'));
|
||||||
@mouseDown = Date.now()
|
|
||||||
return
|
$.requestAnimationFrame(() => __guard__(this.findByTag('input'), x => x.focus()));
|
||||||
|
}
|
||||||
onMouseUp: =>
|
|
||||||
@mouseUp = Date.now()
|
renderVersions(docs) {
|
||||||
return
|
let html = '';
|
||||||
|
for (var doc of Array.from(docs)) { html += this.tmpl('sidebarLabel', doc, {checked: app.docs.contains(doc)}); }
|
||||||
onDOMFocus: (event) =>
|
return html;
|
||||||
target = event.target
|
}
|
||||||
if target.tagName is 'INPUT'
|
|
||||||
unless (@mouseDown and Date.now() < @mouseDown + 100) or (@mouseUp and Date.now() < @mouseUp + 100)
|
extractVersions(originalDocs, version) {
|
||||||
$.scrollTo target.parentNode, null, 'continuous'
|
const docs = [];
|
||||||
else if target.classList.contains(app.views.ListFold.targetClass)
|
const versions = [version];
|
||||||
target.blur()
|
for (var doc of Array.from(originalDocs)) {
|
||||||
unless @mouseDown and Date.now() < @mouseDown + 100
|
(doc.name === version.name ? versions : docs).push(doc);
|
||||||
if @focusEl is $('input', target.nextElementSibling)
|
}
|
||||||
@listFold.close(target) if target.classList.contains(app.views.ListFold.activeClass)
|
return [docs, versions];
|
||||||
prev = target.previousElementSibling
|
}
|
||||||
prev = prev.previousElementSibling until prev.tagName is 'LABEL' or prev.classList.contains(app.views.ListFold.targetClass)
|
|
||||||
prev = $.makeArray($$('input', prev.nextElementSibling)).pop() if prev.classList.contains(app.views.ListFold.activeClass)
|
empty() {
|
||||||
@delay -> prev.focus()
|
this.resetClass();
|
||||||
else
|
super.empty(...arguments);
|
||||||
@listFold.open(target) unless target.classList.contains(app.views.ListFold.activeClass)
|
}
|
||||||
@delay -> $('input', target.nextElementSibling).focus()
|
|
||||||
@focusEl = target
|
getSelectedDocs() {
|
||||||
return
|
return Array.from(this.findAllByTag('input')).filter((input) => (input != null ? input.checked : undefined)).map((input) =>
|
||||||
|
input.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
onMouseDown() {
|
||||||
|
this.mouseDown = Date.now();
|
||||||
|
}
|
||||||
|
|
||||||
|
onMouseUp() {
|
||||||
|
this.mouseUp = Date.now();
|
||||||
|
}
|
||||||
|
|
||||||
|
onDOMFocus(event) {
|
||||||
|
const {
|
||||||
|
target
|
||||||
|
} = event;
|
||||||
|
if (target.tagName === 'INPUT') {
|
||||||
|
if ((!this.mouseDown || !(Date.now() < (this.mouseDown + 100))) && (!this.mouseUp || !(Date.now() < (this.mouseUp + 100)))) {
|
||||||
|
$.scrollTo(target.parentNode, null, 'continuous');
|
||||||
|
}
|
||||||
|
} else if (target.classList.contains(app.views.ListFold.targetClass)) {
|
||||||
|
target.blur();
|
||||||
|
if (!this.mouseDown || !(Date.now() < (this.mouseDown + 100))) {
|
||||||
|
if (this.focusEl === $('input', target.nextElementSibling)) {
|
||||||
|
if (target.classList.contains(app.views.ListFold.activeClass)) { this.listFold.close(target); }
|
||||||
|
let prev = target.previousElementSibling;
|
||||||
|
while ((prev.tagName !== 'LABEL') && !prev.classList.contains(app.views.ListFold.targetClass)) { prev = prev.previousElementSibling; }
|
||||||
|
if (prev.classList.contains(app.views.ListFold.activeClass)) { prev = $.makeArray($$('input', prev.nextElementSibling)).pop(); }
|
||||||
|
this.delay(() => prev.focus());
|
||||||
|
} else {
|
||||||
|
if (!target.classList.contains(app.views.ListFold.activeClass)) { this.listFold.open(target); }
|
||||||
|
this.delay(() => $('input', target.nextElementSibling).focus());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.focusEl = target;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,15 +1,27 @@
|
|||||||
#= require views/list/paginated_list
|
/*
|
||||||
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
//= require views/list/paginated_list
|
||||||
|
|
||||||
class app.views.EntryList extends app.views.PaginatedList
|
const Cls = (app.views.EntryList = class EntryList extends app.views.PaginatedList {
|
||||||
@tagName: 'div'
|
static initClass() {
|
||||||
@className: '_list _list-sub'
|
this.tagName = 'div';
|
||||||
|
this.className = '_list _list-sub';
|
||||||
|
}
|
||||||
|
|
||||||
constructor: (@entries) -> super
|
constructor(entries) { this.entries = entries; super(...arguments); }
|
||||||
|
|
||||||
init: ->
|
init() {
|
||||||
@renderPaginated()
|
this.renderPaginated();
|
||||||
@activate()
|
this.activate();
|
||||||
return
|
}
|
||||||
|
|
||||||
render: (entries) ->
|
render(entries) {
|
||||||
@tmpl 'sidebarEntry', entries
|
return this.tmpl('sidebarEntry', entries);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,68 +1,93 @@
|
|||||||
class app.views.Results extends app.View
|
/*
|
||||||
@className: '_list'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
|
*/
|
||||||
|
const Cls = (app.views.Results = class Results extends app.View {
|
||||||
|
static initClass() {
|
||||||
|
this.className = '_list';
|
||||||
|
|
||||||
@events:
|
this.events =
|
||||||
click: 'onClick'
|
{click: 'onClick'};
|
||||||
|
|
||||||
@routes:
|
this.routes =
|
||||||
after: 'afterRoute'
|
{after: 'afterRoute'};
|
||||||
|
}
|
||||||
|
|
||||||
constructor: (@sidebar, @search) -> super
|
constructor(sidebar, search) { this.onResults = this.onResults.bind(this); this.onNoResults = this.onNoResults.bind(this); this.onClear = this.onClear.bind(this); this.afterRoute = this.afterRoute.bind(this); this.onClick = this.onClick.bind(this); this.sidebar = sidebar; this.search = search; super(...arguments); }
|
||||||
|
|
||||||
deactivate: ->
|
deactivate() {
|
||||||
if super
|
if (super.deactivate(...arguments)) {
|
||||||
@empty()
|
this.empty();
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
init: ->
|
init() {
|
||||||
@addSubview @listFocus = new app.views.ListFocus @el
|
this.addSubview(this.listFocus = new app.views.ListFocus(this.el));
|
||||||
@addSubview @listSelect = new app.views.ListSelect @el
|
this.addSubview(this.listSelect = new app.views.ListSelect(this.el));
|
||||||
|
|
||||||
@search
|
this.search
|
||||||
.on 'results', @onResults
|
.on('results', this.onResults)
|
||||||
.on 'noresults', @onNoResults
|
.on('noresults', this.onNoResults)
|
||||||
.on 'clear', @onClear
|
.on('clear', this.onClear);
|
||||||
return
|
}
|
||||||
|
|
||||||
onResults: (entries, flags) =>
|
onResults(entries, flags) {
|
||||||
@listFocus?.blur() if flags.initialResults
|
if (flags.initialResults) { if (this.listFocus != null) {
|
||||||
@empty() if flags.initialResults
|
this.listFocus.blur();
|
||||||
@append @tmpl('sidebarResult', entries)
|
} }
|
||||||
|
if (flags.initialResults) { this.empty(); }
|
||||||
|
this.append(this.tmpl('sidebarResult', entries));
|
||||||
|
|
||||||
if flags.initialResults
|
if (flags.initialResults) {
|
||||||
if flags.urlSearch then @openFirst() else @focusFirst()
|
if (flags.urlSearch) { this.openFirst(); } else { this.focusFirst(); }
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onNoResults: =>
|
onNoResults() {
|
||||||
@html @tmpl('sidebarNoResults')
|
this.html(this.tmpl('sidebarNoResults'));
|
||||||
return
|
}
|
||||||
|
|
||||||
onClear: =>
|
onClear() {
|
||||||
@empty()
|
this.empty();
|
||||||
return
|
}
|
||||||
|
|
||||||
focusFirst: ->
|
focusFirst() {
|
||||||
@listFocus?.focusOnNextFrame @el.firstElementChild unless app.isMobile()
|
if (!app.isMobile()) { if (this.listFocus != null) {
|
||||||
return
|
this.listFocus.focusOnNextFrame(this.el.firstElementChild);
|
||||||
|
} }
|
||||||
|
}
|
||||||
|
|
||||||
openFirst: ->
|
openFirst() {
|
||||||
@el.firstElementChild?.click()
|
if (this.el.firstElementChild != null) {
|
||||||
return
|
this.el.firstElementChild.click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onDocEnabled: (doc) ->
|
onDocEnabled(doc) {
|
||||||
app.router.show(doc.fullPath())
|
app.router.show(doc.fullPath());
|
||||||
@sidebar.onDocEnabled()
|
return this.sidebar.onDocEnabled();
|
||||||
|
}
|
||||||
|
|
||||||
afterRoute: (route, context) =>
|
afterRoute(route, context) {
|
||||||
if route is 'entry'
|
if (route === 'entry') {
|
||||||
@listSelect.selectByHref context.entry.fullPath()
|
this.listSelect.selectByHref(context.entry.fullPath());
|
||||||
else
|
} else {
|
||||||
@listSelect.deselect()
|
this.listSelect.deselect();
|
||||||
return
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onClick: (event) =>
|
onClick(event) {
|
||||||
return if event.which isnt 1
|
let slug;
|
||||||
if slug = $.eventTarget(event).getAttribute('data-enable')
|
if (event.which !== 1) { return; }
|
||||||
$.stopEvent(event)
|
if (slug = $.eventTarget(event).getAttribute('data-enable')) {
|
||||||
doc = app.disabledDocs.findBy('slug', slug)
|
$.stopEvent(event);
|
||||||
app.enableDoc(doc, @onDocEnabled.bind(@, doc), $.noop) if doc
|
const doc = app.disabledDocs.findBy('slug', slug);
|
||||||
|
if (doc) { return app.enableDoc(doc, this.onDocEnabled.bind(this, doc), $.noop); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
@ -1,164 +1,217 @@
|
|||||||
class app.views.Sidebar extends app.View
|
/*
|
||||||
@el: '._sidebar'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
focus: 'onFocus'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
select: 'onSelect'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
click: 'onClick'
|
* DS207: Consider shorter variations of null checks
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@routes:
|
*/
|
||||||
after: 'afterRoute'
|
const Cls = (app.views.Sidebar = class Sidebar extends app.View {
|
||||||
|
constructor(...args) {
|
||||||
@shortcuts:
|
this.resetHoverOnMouseMove = this.resetHoverOnMouseMove.bind(this);
|
||||||
altR: 'onAltR'
|
this.resetHover = this.resetHover.bind(this);
|
||||||
escape: 'onEscape'
|
this.showResults = this.showResults.bind(this);
|
||||||
|
this.onReady = this.onReady.bind(this);
|
||||||
init: ->
|
this.onScopeChange = this.onScopeChange.bind(this);
|
||||||
@addSubview @hover = new app.views.SidebarHover @el unless app.isMobile()
|
this.onSearching = this.onSearching.bind(this);
|
||||||
@addSubview @search = new app.views.Search
|
this.onSearchClear = this.onSearchClear.bind(this);
|
||||||
|
this.onFocus = this.onFocus.bind(this);
|
||||||
@search
|
this.onSelect = this.onSelect.bind(this);
|
||||||
.on 'searching', @onSearching
|
this.onClick = this.onClick.bind(this);
|
||||||
.on 'clear', @onSearchClear
|
this.onAltR = this.onAltR.bind(this);
|
||||||
|
this.onEscape = this.onEscape.bind(this);
|
||||||
|
this.afterRoute = this.afterRoute.bind(this);
|
||||||
|
super(...args);
|
||||||
|
}
|
||||||
|
|
||||||
|
static initClass() {
|
||||||
|
this.el = '._sidebar';
|
||||||
|
|
||||||
|
this.events = {
|
||||||
|
focus: 'onFocus',
|
||||||
|
select: 'onSelect',
|
||||||
|
click: 'onClick'
|
||||||
|
};
|
||||||
|
|
||||||
|
this.routes =
|
||||||
|
{after: 'afterRoute'};
|
||||||
|
|
||||||
|
this.shortcuts = {
|
||||||
|
altR: 'onAltR',
|
||||||
|
escape: 'onEscape'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
if (!app.isMobile()) { this.addSubview(this.hover = new app.views.SidebarHover(this.el)); }
|
||||||
|
this.addSubview(this.search = new app.views.Search);
|
||||||
|
|
||||||
|
this.search
|
||||||
|
.on('searching', this.onSearching)
|
||||||
|
.on('clear', this.onSearchClear)
|
||||||
.scope
|
.scope
|
||||||
.on 'change', @onScopeChange
|
.on('change', this.onScopeChange);
|
||||||
|
|
||||||
@results = new app.views.Results @, @search
|
this.results = new app.views.Results(this, this.search);
|
||||||
@docList = new app.views.DocList
|
this.docList = new app.views.DocList;
|
||||||
|
|
||||||
app.on 'ready', @onReady
|
app.on('ready', this.onReady);
|
||||||
|
|
||||||
$.on document.documentElement, 'mouseleave', => @hide()
|
$.on(document.documentElement, 'mouseleave', () => this.hide());
|
||||||
$.on document.documentElement, 'mouseenter', => @resetDisplay(forceNoHover: false)
|
$.on(document.documentElement, 'mouseenter', () => this.resetDisplay({forceNoHover: false}));
|
||||||
return
|
}
|
||||||
|
|
||||||
hide: ->
|
hide() {
|
||||||
@removeClass 'show'
|
this.removeClass('show');
|
||||||
return
|
}
|
||||||
|
|
||||||
display: ->
|
display() {
|
||||||
@addClass 'show'
|
this.addClass('show');
|
||||||
return
|
}
|
||||||
|
|
||||||
resetDisplay: (options = {}) ->
|
resetDisplay(options) {
|
||||||
return unless @hasClass 'show'
|
if (options == null) { options = {}; }
|
||||||
@removeClass 'show'
|
if (!this.hasClass('show')) { return; }
|
||||||
|
this.removeClass('show');
|
||||||
unless options.forceNoHover is false or @hasClass 'no-hover'
|
|
||||||
@addClass 'no-hover'
|
if ((options.forceNoHover !== false) && !this.hasClass('no-hover')) {
|
||||||
$.on window, 'mousemove', @resetHoverOnMouseMove
|
this.addClass('no-hover');
|
||||||
return
|
$.on(window, 'mousemove', this.resetHoverOnMouseMove);
|
||||||
|
}
|
||||||
resetHoverOnMouseMove: =>
|
}
|
||||||
$.off window, 'mousemove', @resetHoverOnMouseMove
|
|
||||||
$.requestAnimationFrame @resetHover
|
resetHoverOnMouseMove() {
|
||||||
|
$.off(window, 'mousemove', this.resetHoverOnMouseMove);
|
||||||
resetHover: =>
|
return $.requestAnimationFrame(this.resetHover);
|
||||||
@removeClass 'no-hover'
|
}
|
||||||
|
|
||||||
showView: (view) ->
|
resetHover() {
|
||||||
unless @view is view
|
return this.removeClass('no-hover');
|
||||||
@hover?.hide()
|
}
|
||||||
@saveScrollPosition()
|
|
||||||
@view?.deactivate()
|
showView(view) {
|
||||||
@view = view
|
if (this.view !== view) {
|
||||||
@render()
|
if (this.hover != null) {
|
||||||
@view.activate()
|
this.hover.hide();
|
||||||
@restoreScrollPosition()
|
}
|
||||||
return
|
this.saveScrollPosition();
|
||||||
|
if (this.view != null) {
|
||||||
render: ->
|
this.view.deactivate();
|
||||||
@html @view
|
}
|
||||||
return
|
this.view = view;
|
||||||
|
this.render();
|
||||||
showDocList: ->
|
this.view.activate();
|
||||||
@showView @docList
|
this.restoreScrollPosition();
|
||||||
return
|
}
|
||||||
|
}
|
||||||
showResults: =>
|
|
||||||
@display()
|
render() {
|
||||||
@showView @results
|
this.html(this.view);
|
||||||
return
|
}
|
||||||
|
|
||||||
reset: ->
|
showDocList() {
|
||||||
@display()
|
this.showView(this.docList);
|
||||||
@showDocList()
|
}
|
||||||
@docList.reset()
|
|
||||||
@search.reset()
|
showResults() {
|
||||||
return
|
this.display();
|
||||||
|
this.showView(this.results);
|
||||||
onReady: =>
|
}
|
||||||
@view = @docList
|
|
||||||
@render()
|
reset() {
|
||||||
@view.activate()
|
this.display();
|
||||||
return
|
this.showDocList();
|
||||||
|
this.docList.reset();
|
||||||
onScopeChange: (newDoc, previousDoc) =>
|
this.search.reset();
|
||||||
@docList.closeDoc(previousDoc) if previousDoc
|
}
|
||||||
if newDoc then @docList.reveal(newDoc.toEntry()) else @scrollToTop()
|
|
||||||
return
|
onReady() {
|
||||||
|
this.view = this.docList;
|
||||||
saveScrollPosition: ->
|
this.render();
|
||||||
if @view is @docList
|
this.view.activate();
|
||||||
@scrollTop = @el.scrollTop
|
}
|
||||||
return
|
|
||||||
|
onScopeChange(newDoc, previousDoc) {
|
||||||
restoreScrollPosition: ->
|
if (previousDoc) { this.docList.closeDoc(previousDoc); }
|
||||||
if @view is @docList and @scrollTop
|
if (newDoc) { this.docList.reveal(newDoc.toEntry()); } else { this.scrollToTop(); }
|
||||||
@el.scrollTop = @scrollTop
|
}
|
||||||
@scrollTop = null
|
|
||||||
else
|
saveScrollPosition() {
|
||||||
@scrollToTop()
|
if (this.view === this.docList) {
|
||||||
return
|
this.scrollTop = this.el.scrollTop;
|
||||||
|
}
|
||||||
scrollToTop: ->
|
}
|
||||||
@el.scrollTop = 0
|
|
||||||
return
|
restoreScrollPosition() {
|
||||||
|
if ((this.view === this.docList) && this.scrollTop) {
|
||||||
onSearching: =>
|
this.el.scrollTop = this.scrollTop;
|
||||||
@showResults()
|
this.scrollTop = null;
|
||||||
return
|
} else {
|
||||||
|
this.scrollToTop();
|
||||||
onSearchClear: =>
|
}
|
||||||
@resetDisplay()
|
}
|
||||||
@showDocList()
|
|
||||||
return
|
scrollToTop() {
|
||||||
|
this.el.scrollTop = 0;
|
||||||
onFocus: (event) =>
|
}
|
||||||
@display()
|
|
||||||
$.scrollTo event.target, @el, 'continuous', bottomGap: 2 unless event.target is @el
|
onSearching() {
|
||||||
return
|
this.showResults();
|
||||||
|
}
|
||||||
onSelect: =>
|
|
||||||
@resetDisplay()
|
onSearchClear() {
|
||||||
return
|
this.resetDisplay();
|
||||||
|
this.showDocList();
|
||||||
onClick: (event) =>
|
}
|
||||||
return if event.which isnt 1
|
|
||||||
if $.eventTarget(event).hasAttribute? 'data-reset-list'
|
onFocus(event) {
|
||||||
$.stopEvent(event)
|
this.display();
|
||||||
@onAltR()
|
if (event.target !== this.el) { $.scrollTo(event.target, this.el, 'continuous', {bottomGap: 2}); }
|
||||||
return
|
}
|
||||||
|
|
||||||
onAltR: =>
|
onSelect() {
|
||||||
@reset()
|
this.resetDisplay();
|
||||||
@docList.reset(revealCurrent: true)
|
}
|
||||||
@display()
|
|
||||||
return
|
onClick(event) {
|
||||||
|
if (event.which !== 1) { return; }
|
||||||
onEscape: =>
|
if (__guardMethod__($.eventTarget(event), 'hasAttribute', o => o.hasAttribute('data-reset-list'))) {
|
||||||
@reset()
|
$.stopEvent(event);
|
||||||
@resetDisplay()
|
this.onAltR();
|
||||||
if doc = @search.getScopeDoc() then @docList.reveal(doc.toEntry()) else @scrollToTop()
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
onDocEnabled: ->
|
onAltR() {
|
||||||
@docList.onEnabled()
|
this.reset();
|
||||||
@reset()
|
this.docList.reset({revealCurrent: true});
|
||||||
return
|
this.display();
|
||||||
|
}
|
||||||
afterRoute: (name, context) =>
|
|
||||||
return if app.shortcuts.eventInProgress?.name is 'escape'
|
onEscape() {
|
||||||
@reset() if not context.init and app.router.isIndex()
|
let doc;
|
||||||
@resetDisplay()
|
this.reset();
|
||||||
return
|
this.resetDisplay();
|
||||||
|
if ((doc = this.search.getScopeDoc())) { this.docList.reveal(doc.toEntry()); } else { this.scrollToTop(); }
|
||||||
|
}
|
||||||
|
|
||||||
|
onDocEnabled() {
|
||||||
|
this.docList.onEnabled();
|
||||||
|
this.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
afterRoute(name, context) {
|
||||||
|
if ((app.shortcuts.eventInProgress != null ? app.shortcuts.eventInProgress.name : undefined) === 'escape') { return; }
|
||||||
|
if (!context.init && app.router.isIndex()) { this.reset(); }
|
||||||
|
this.resetDisplay();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guardMethod__(obj, methodName, transform) {
|
||||||
|
if (typeof obj !== 'undefined' && obj !== null && typeof obj[methodName] === 'function') {
|
||||||
|
return transform(obj, methodName);
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
@ -1,100 +1,143 @@
|
|||||||
class app.views.SidebarHover extends app.View
|
/*
|
||||||
@itemClass: '_list-hover'
|
* decaffeinate suggestions:
|
||||||
|
* DS002: Fix invalid constructor
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
focus: 'onFocus'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
blur: 'onBlur'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
mouseover: 'onMouseover'
|
* DS207: Consider shorter variations of null checks
|
||||||
mouseout: 'onMouseout'
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
scroll: 'onScroll'
|
*/
|
||||||
click: 'onClick'
|
const Cls = (app.views.SidebarHover = class SidebarHover extends app.View {
|
||||||
|
static initClass() {
|
||||||
@routes:
|
this.itemClass = '_list-hover';
|
||||||
after: 'onRoute'
|
|
||||||
|
this.events = {
|
||||||
constructor: (@el) ->
|
focus: 'onFocus',
|
||||||
unless isPointerEventsSupported()
|
blur: 'onBlur',
|
||||||
delete @constructor.events.mouseover
|
mouseover: 'onMouseover',
|
||||||
super
|
mouseout: 'onMouseout',
|
||||||
|
scroll: 'onScroll',
|
||||||
show: (el) ->
|
click: 'onClick'
|
||||||
unless el is @cursor
|
};
|
||||||
@hide()
|
|
||||||
if @isTarget(el) and @isTruncated(el.lastElementChild or el)
|
this.routes =
|
||||||
@cursor = el
|
{after: 'onRoute'};
|
||||||
@clone = @makeClone @cursor
|
}
|
||||||
$.append document.body, @clone
|
|
||||||
@offsetTop ?= @el.offsetTop
|
constructor(el) {
|
||||||
@position()
|
this.position = this.position.bind(this);
|
||||||
return
|
this.onFocus = this.onFocus.bind(this);
|
||||||
|
this.onBlur = this.onBlur.bind(this);
|
||||||
hide: ->
|
this.onMouseover = this.onMouseover.bind(this);
|
||||||
if @cursor
|
this.onMouseout = this.onMouseout.bind(this);
|
||||||
$.remove @clone
|
this.onScroll = this.onScroll.bind(this);
|
||||||
@cursor = @clone = null
|
this.onClick = this.onClick.bind(this);
|
||||||
return
|
this.onRoute = this.onRoute.bind(this);
|
||||||
|
this.el = el;
|
||||||
position: =>
|
if (!isPointerEventsSupported()) {
|
||||||
if @cursor
|
delete this.constructor.events.mouseover;
|
||||||
rect = $.rect(@cursor)
|
}
|
||||||
if rect.top >= @offsetTop
|
super(...arguments);
|
||||||
@clone.style.top = rect.top + 'px'
|
}
|
||||||
@clone.style.left = rect.left + 'px'
|
|
||||||
else
|
show(el) {
|
||||||
@hide()
|
if (el !== this.cursor) {
|
||||||
return
|
this.hide();
|
||||||
|
if (this.isTarget(el) && this.isTruncated(el.lastElementChild || el)) {
|
||||||
makeClone: (el) ->
|
this.cursor = el;
|
||||||
clone = el.cloneNode(true)
|
this.clone = this.makeClone(this.cursor);
|
||||||
clone.classList.add 'clone'
|
$.append(document.body, this.clone);
|
||||||
clone
|
if (this.offsetTop == null) { this.offsetTop = this.el.offsetTop; }
|
||||||
|
this.position();
|
||||||
isTarget: (el) ->
|
}
|
||||||
el?.classList?.contains @constructor.itemClass
|
}
|
||||||
|
}
|
||||||
isSelected: (el) ->
|
|
||||||
el.classList.contains 'active'
|
hide() {
|
||||||
|
if (this.cursor) {
|
||||||
isTruncated: (el) ->
|
$.remove(this.clone);
|
||||||
el.scrollWidth > el.offsetWidth
|
this.cursor = (this.clone = null);
|
||||||
|
}
|
||||||
onFocus: (event) =>
|
}
|
||||||
@focusTime = Date.now()
|
|
||||||
@show event.target
|
position() {
|
||||||
return
|
if (this.cursor) {
|
||||||
|
const rect = $.rect(this.cursor);
|
||||||
onBlur: =>
|
if (rect.top >= this.offsetTop) {
|
||||||
@hide()
|
this.clone.style.top = rect.top + 'px';
|
||||||
return
|
this.clone.style.left = rect.left + 'px';
|
||||||
|
} else {
|
||||||
onMouseover: (event) =>
|
this.hide();
|
||||||
if @isTarget(event.target) and not @isSelected(event.target) and @mouseActivated()
|
}
|
||||||
@show event.target
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
onMouseout: (event) =>
|
makeClone(el) {
|
||||||
if @isTarget(event.target) and @mouseActivated()
|
const clone = el.cloneNode(true);
|
||||||
@hide()
|
clone.classList.add('clone');
|
||||||
return
|
return clone;
|
||||||
|
}
|
||||||
mouseActivated: ->
|
|
||||||
# Skip mouse events caused by focus events scrolling the sidebar.
|
isTarget(el) {
|
||||||
not @focusTime or Date.now() - @focusTime > 500
|
return __guard__(el != null ? el.classList : undefined, x => x.contains(this.constructor.itemClass));
|
||||||
|
}
|
||||||
onScroll: =>
|
|
||||||
@position()
|
isSelected(el) {
|
||||||
return
|
return el.classList.contains('active');
|
||||||
|
}
|
||||||
onClick: (event) =>
|
|
||||||
if event.target is @clone
|
isTruncated(el) {
|
||||||
$.click @cursor
|
return el.scrollWidth > el.offsetWidth;
|
||||||
return
|
}
|
||||||
|
|
||||||
onRoute: =>
|
onFocus(event) {
|
||||||
@hide()
|
this.focusTime = Date.now();
|
||||||
return
|
this.show(event.target);
|
||||||
|
}
|
||||||
isPointerEventsSupported = ->
|
|
||||||
el = document.createElement 'div'
|
onBlur() {
|
||||||
el.style.cssText = 'pointer-events: auto'
|
this.hide();
|
||||||
el.style.pointerEvents is 'auto'
|
}
|
||||||
|
|
||||||
|
onMouseover(event) {
|
||||||
|
if (this.isTarget(event.target) && !this.isSelected(event.target) && this.mouseActivated()) {
|
||||||
|
this.show(event.target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMouseout(event) {
|
||||||
|
if (this.isTarget(event.target) && this.mouseActivated()) {
|
||||||
|
this.hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mouseActivated() {
|
||||||
|
// Skip mouse events caused by focus events scrolling the sidebar.
|
||||||
|
return !this.focusTime || ((Date.now() - this.focusTime) > 500);
|
||||||
|
}
|
||||||
|
|
||||||
|
onScroll() {
|
||||||
|
this.position();
|
||||||
|
}
|
||||||
|
|
||||||
|
onClick(event) {
|
||||||
|
if (event.target === this.clone) {
|
||||||
|
$.click(this.cursor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onRoute() {
|
||||||
|
this.hide();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
var isPointerEventsSupported = function() {
|
||||||
|
const el = document.createElement('div');
|
||||||
|
el.style.cssText = 'pointer-events: auto';
|
||||||
|
return el.style.pointerEvents === 'auto';
|
||||||
|
};
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,53 +1,77 @@
|
|||||||
class app.views.TypeList extends app.View
|
/*
|
||||||
@tagName: 'div'
|
* decaffeinate suggestions:
|
||||||
@className: '_list _list-sub'
|
* DS002: Fix invalid constructor
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
@events:
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
open: 'onOpen'
|
* DS103: Rewrite code to no longer use __guard__, or convert again using --optional-chaining
|
||||||
close: 'onClose'
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
constructor: (@doc) -> super
|
*/
|
||||||
|
const Cls = (app.views.TypeList = class TypeList extends app.View {
|
||||||
init: ->
|
static initClass() {
|
||||||
@lists = {}
|
this.tagName = 'div';
|
||||||
@render()
|
this.className = '_list _list-sub';
|
||||||
@activate()
|
|
||||||
return
|
this.events = {
|
||||||
|
open: 'onOpen',
|
||||||
activate: ->
|
close: 'onClose'
|
||||||
if super
|
};
|
||||||
list.activate() for slug, list of @lists
|
}
|
||||||
return
|
|
||||||
|
constructor(doc) { this.onOpen = this.onOpen.bind(this); this.onClose = this.onClose.bind(this); this.doc = doc; super(...arguments); }
|
||||||
deactivate: ->
|
|
||||||
if super
|
init() {
|
||||||
list.deactivate() for slug, list of @lists
|
this.lists = {};
|
||||||
return
|
this.render();
|
||||||
|
this.activate();
|
||||||
render: ->
|
}
|
||||||
html = ''
|
|
||||||
html += @tmpl('sidebarType', group) for group in @doc.types.groups()
|
activate() {
|
||||||
@html(html)
|
if (super.activate(...arguments)) {
|
||||||
|
for (var slug in this.lists) { var list = this.lists[slug]; list.activate(); }
|
||||||
onOpen: (event) =>
|
}
|
||||||
$.stopEvent(event)
|
}
|
||||||
type = @doc.types.findBy 'slug', event.target.getAttribute('data-slug')
|
|
||||||
|
deactivate() {
|
||||||
if type and not @lists[type.slug]
|
if (super.deactivate(...arguments)) {
|
||||||
@lists[type.slug] = new app.views.EntryList(type.entries())
|
for (var slug in this.lists) { var list = this.lists[slug]; list.deactivate(); }
|
||||||
$.after event.target, @lists[type.slug].el
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
onClose: (event) =>
|
render() {
|
||||||
$.stopEvent(event)
|
let html = '';
|
||||||
type = @doc.types.findBy 'slug', event.target.getAttribute('data-slug')
|
for (var group of Array.from(this.doc.types.groups())) { html += this.tmpl('sidebarType', group); }
|
||||||
|
return this.html(html);
|
||||||
if type and @lists[type.slug]
|
}
|
||||||
@lists[type.slug].detach()
|
|
||||||
delete @lists[type.slug]
|
onOpen(event) {
|
||||||
return
|
$.stopEvent(event);
|
||||||
|
const type = this.doc.types.findBy('slug', event.target.getAttribute('data-slug'));
|
||||||
paginateTo: (model) ->
|
|
||||||
if model.type
|
if (type && !this.lists[type.slug]) {
|
||||||
@lists[model.getType().slug]?.paginateTo(model)
|
this.lists[type.slug] = new app.views.EntryList(type.entries());
|
||||||
return
|
$.after(event.target, this.lists[type.slug].el);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClose(event) {
|
||||||
|
$.stopEvent(event);
|
||||||
|
const type = this.doc.types.findBy('slug', event.target.getAttribute('data-slug'));
|
||||||
|
|
||||||
|
if (type && this.lists[type.slug]) {
|
||||||
|
this.lists[type.slug].detach();
|
||||||
|
delete this.lists[type.slug];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
paginateTo(model) {
|
||||||
|
if (model.type) {
|
||||||
|
__guard__(this.lists[model.getType().slug], x => x.paginateTo(model));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
|
||||||
|
function __guard__(value, transform) {
|
||||||
|
return (typeof value !== 'undefined' && value !== null) ? transform(value) : undefined;
|
||||||
|
}
|
@ -1,172 +1,214 @@
|
|||||||
class app.View
|
/*
|
||||||
$.extend @prototype, Events
|
* decaffeinate suggestions:
|
||||||
|
* DS101: Remove unnecessary use of Array.from
|
||||||
constructor: ->
|
* DS102: Remove unnecessary code created because of implicit returns
|
||||||
@setupElement()
|
* DS206: Consider reworking classes to avoid initClass
|
||||||
@originalClassName = @el.className if @el.className
|
* DS207: Consider shorter variations of null checks
|
||||||
@resetClass() if @constructor.className
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/main/docs/suggestions.md
|
||||||
@refreshElements()
|
*/
|
||||||
@init?()
|
const Cls = (app.View = class View {
|
||||||
@refreshElements()
|
static initClass() {
|
||||||
|
$.extend(this.prototype, Events);
|
||||||
setupElement: ->
|
}
|
||||||
@el ?= if typeof @constructor.el is 'string'
|
|
||||||
$ @constructor.el
|
constructor() {
|
||||||
else if @constructor.el
|
this.setupElement();
|
||||||
@constructor.el
|
if (this.el.className) { this.originalClassName = this.el.className; }
|
||||||
else
|
if (this.constructor.className) { this.resetClass(); }
|
||||||
document.createElement @constructor.tagName or 'div'
|
this.refreshElements();
|
||||||
|
if (typeof this.init === 'function') {
|
||||||
if @constructor.attributes
|
this.init();
|
||||||
for key, value of @constructor.attributes
|
}
|
||||||
@el.setAttribute(key, value)
|
this.refreshElements();
|
||||||
return
|
}
|
||||||
|
|
||||||
refreshElements: ->
|
setupElement() {
|
||||||
if @constructor.elements
|
if (this.el == null) { this.el = typeof this.constructor.el === 'string' ?
|
||||||
@[name] = @find selector for name, selector of @constructor.elements
|
$(this.constructor.el)
|
||||||
return
|
: this.constructor.el ?
|
||||||
|
this.constructor.el
|
||||||
addClass: (name) ->
|
:
|
||||||
@el.classList.add(name)
|
document.createElement(this.constructor.tagName || 'div'); }
|
||||||
return
|
|
||||||
|
if (this.constructor.attributes) {
|
||||||
removeClass: (name) ->
|
for (var key in this.constructor.attributes) {
|
||||||
@el.classList.remove(name)
|
var value = this.constructor.attributes[key];
|
||||||
return
|
this.el.setAttribute(key, value);
|
||||||
|
}
|
||||||
toggleClass: (name) ->
|
}
|
||||||
@el.classList.toggle(name)
|
}
|
||||||
return
|
|
||||||
|
refreshElements() {
|
||||||
hasClass: (name) ->
|
if (this.constructor.elements) {
|
||||||
@el.classList.contains(name)
|
for (var name in this.constructor.elements) { var selector = this.constructor.elements[name]; this[name] = this.find(selector); }
|
||||||
|
}
|
||||||
resetClass: ->
|
}
|
||||||
@el.className = @originalClassName or ''
|
|
||||||
if @constructor.className
|
addClass(name) {
|
||||||
@addClass name for name in @constructor.className.split ' '
|
this.el.classList.add(name);
|
||||||
return
|
}
|
||||||
|
|
||||||
find: (selector) ->
|
removeClass(name) {
|
||||||
$ selector, @el
|
this.el.classList.remove(name);
|
||||||
|
}
|
||||||
findAll: (selector) ->
|
|
||||||
$$ selector, @el
|
toggleClass(name) {
|
||||||
|
this.el.classList.toggle(name);
|
||||||
findByClass: (name) ->
|
}
|
||||||
@findAllByClass(name)[0]
|
|
||||||
|
hasClass(name) {
|
||||||
findLastByClass: (name) ->
|
return this.el.classList.contains(name);
|
||||||
all = @findAllByClass(name)[0]
|
}
|
||||||
all[all.length - 1]
|
|
||||||
|
resetClass() {
|
||||||
findAllByClass: (name) ->
|
this.el.className = this.originalClassName || '';
|
||||||
@el.getElementsByClassName(name)
|
if (this.constructor.className) {
|
||||||
|
for (var name of Array.from(this.constructor.className.split(' '))) { this.addClass(name); }
|
||||||
findByTag: (tag) ->
|
}
|
||||||
@findAllByTag(tag)[0]
|
}
|
||||||
|
|
||||||
findLastByTag: (tag) ->
|
find(selector) {
|
||||||
all = @findAllByTag(tag)
|
return $(selector, this.el);
|
||||||
all[all.length - 1]
|
}
|
||||||
|
|
||||||
findAllByTag: (tag) ->
|
findAll(selector) {
|
||||||
@el.getElementsByTagName(tag)
|
return $$(selector, this.el);
|
||||||
|
}
|
||||||
append: (value) ->
|
|
||||||
$.append @el, value.el or value
|
findByClass(name) {
|
||||||
return
|
return this.findAllByClass(name)[0];
|
||||||
|
}
|
||||||
appendTo: (value) ->
|
|
||||||
$.append value.el or value, @el
|
findLastByClass(name) {
|
||||||
return
|
const all = this.findAllByClass(name)[0];
|
||||||
|
return all[all.length - 1];
|
||||||
prepend: (value) ->
|
}
|
||||||
$.prepend @el, value.el or value
|
|
||||||
return
|
findAllByClass(name) {
|
||||||
|
return this.el.getElementsByClassName(name);
|
||||||
prependTo: (value) ->
|
}
|
||||||
$.prepend value.el or value, @el
|
|
||||||
return
|
findByTag(tag) {
|
||||||
|
return this.findAllByTag(tag)[0];
|
||||||
before: (value) ->
|
}
|
||||||
$.before @el, value.el or value
|
|
||||||
return
|
findLastByTag(tag) {
|
||||||
|
const all = this.findAllByTag(tag);
|
||||||
after: (value) ->
|
return all[all.length - 1];
|
||||||
$.after @el, value.el or value
|
}
|
||||||
return
|
|
||||||
|
findAllByTag(tag) {
|
||||||
remove: (value) ->
|
return this.el.getElementsByTagName(tag);
|
||||||
$.remove value.el or value
|
}
|
||||||
return
|
|
||||||
|
append(value) {
|
||||||
empty: ->
|
$.append(this.el, value.el || value);
|
||||||
$.empty @el
|
}
|
||||||
@refreshElements()
|
|
||||||
return
|
appendTo(value) {
|
||||||
|
$.append(value.el || value, this.el);
|
||||||
html: (value) ->
|
}
|
||||||
@empty()
|
|
||||||
@append value
|
prepend(value) {
|
||||||
return
|
$.prepend(this.el, value.el || value);
|
||||||
|
}
|
||||||
tmpl: (args...) ->
|
|
||||||
app.templates.render(args...)
|
prependTo(value) {
|
||||||
|
$.prepend(value.el || value, this.el);
|
||||||
delay: (fn, args...) ->
|
}
|
||||||
delay = if typeof args[args.length - 1] is 'number' then args.pop() else 0
|
|
||||||
setTimeout fn.bind(@, args...), delay
|
before(value) {
|
||||||
|
$.before(this.el, value.el || value);
|
||||||
onDOM: (event, callback) ->
|
}
|
||||||
$.on @el, event, callback
|
|
||||||
return
|
after(value) {
|
||||||
|
$.after(this.el, value.el || value);
|
||||||
offDOM: (event, callback) ->
|
}
|
||||||
$.off @el, event, callback
|
|
||||||
return
|
remove(value) {
|
||||||
|
$.remove(value.el || value);
|
||||||
bindEvents: ->
|
}
|
||||||
if @constructor.events
|
|
||||||
@onDOM name, @[method] for name, method of @constructor.events
|
empty() {
|
||||||
|
$.empty(this.el);
|
||||||
if @constructor.routes
|
this.refreshElements();
|
||||||
app.router.on name, @[method] for name, method of @constructor.routes
|
}
|
||||||
|
|
||||||
if @constructor.shortcuts
|
html(value) {
|
||||||
app.shortcuts.on name, @[method] for name, method of @constructor.shortcuts
|
this.empty();
|
||||||
return
|
this.append(value);
|
||||||
|
}
|
||||||
unbindEvents: ->
|
|
||||||
if @constructor.events
|
tmpl(...args) {
|
||||||
@offDOM name, @[method] for name, method of @constructor.events
|
return app.templates.render(...Array.from(args || []));
|
||||||
|
}
|
||||||
if @constructor.routes
|
|
||||||
app.router.off name, @[method] for name, method of @constructor.routes
|
delay(fn, ...args) {
|
||||||
|
const delay = typeof args[args.length - 1] === 'number' ? args.pop() : 0;
|
||||||
if @constructor.shortcuts
|
return setTimeout(fn.bind(this, ...Array.from(args)), delay);
|
||||||
app.shortcuts.off name, @[method] for name, method of @constructor.shortcuts
|
}
|
||||||
return
|
|
||||||
|
onDOM(event, callback) {
|
||||||
addSubview: (view) ->
|
$.on(this.el, event, callback);
|
||||||
(@subviews or= []).push(view)
|
}
|
||||||
|
|
||||||
activate: ->
|
offDOM(event, callback) {
|
||||||
return if @activated
|
$.off(this.el, event, callback);
|
||||||
@bindEvents()
|
}
|
||||||
view.activate() for view in @subviews if @subviews
|
|
||||||
@activated = true
|
bindEvents() {
|
||||||
true
|
let method, name;
|
||||||
|
if (this.constructor.events) {
|
||||||
deactivate: ->
|
for (name in this.constructor.events) { method = this.constructor.events[name]; this.onDOM(name, this[method]); }
|
||||||
return unless @activated
|
}
|
||||||
@unbindEvents()
|
|
||||||
view.deactivate() for view in @subviews if @subviews
|
if (this.constructor.routes) {
|
||||||
@activated = false
|
for (name in this.constructor.routes) { method = this.constructor.routes[name]; app.router.on(name, this[method]); }
|
||||||
true
|
}
|
||||||
|
|
||||||
detach: ->
|
if (this.constructor.shortcuts) {
|
||||||
@deactivate()
|
for (name in this.constructor.shortcuts) { method = this.constructor.shortcuts[name]; app.shortcuts.on(name, this[method]); }
|
||||||
$.remove @el
|
}
|
||||||
return
|
}
|
||||||
|
|
||||||
|
unbindEvents() {
|
||||||
|
let method, name;
|
||||||
|
if (this.constructor.events) {
|
||||||
|
for (name in this.constructor.events) { method = this.constructor.events[name]; this.offDOM(name, this[method]); }
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.constructor.routes) {
|
||||||
|
for (name in this.constructor.routes) { method = this.constructor.routes[name]; app.router.off(name, this[method]); }
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.constructor.shortcuts) {
|
||||||
|
for (name in this.constructor.shortcuts) { method = this.constructor.shortcuts[name]; app.shortcuts.off(name, this[method]); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addSubview(view) {
|
||||||
|
return (this.subviews || (this.subviews = [])).push(view);
|
||||||
|
}
|
||||||
|
|
||||||
|
activate() {
|
||||||
|
if (this.activated) { return; }
|
||||||
|
this.bindEvents();
|
||||||
|
if (this.subviews) { for (var view of Array.from(this.subviews)) { view.activate(); } }
|
||||||
|
this.activated = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
deactivate() {
|
||||||
|
if (!this.activated) { return; }
|
||||||
|
this.unbindEvents();
|
||||||
|
if (this.subviews) { for (var view of Array.from(this.subviews)) { view.deactivate(); } }
|
||||||
|
this.activated = false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
detach() {
|
||||||
|
this.deactivate();
|
||||||
|
$.remove(this.el);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Cls.initClass();
|
||||||
|
Loading…
Reference in new issue