Migrate to Webpack; refactor.

This commit is contained in:
Maddison Hellstrom 2020-12-30 02:45:19 -08:00
parent 10537a48bb
commit c9a0281606
18 changed files with 2412 additions and 4784 deletions

View File

@ -22,9 +22,10 @@ module.exports = {
node: true, node: true,
}, },
rules: { rules: {
semi: ["error", "never"], semi: ["error", "never"],
"comma-dangle": ["warn", "always-multiline"], "comma-dangle": ["warn", "always-multiline"],
quotes: ["warn", "double"], quotes: ["warn", "double"],
"no-cond-assign": ["error", "except-parens"],
"newline-per-chained-call": "off", "newline-per-chained-call": "off",
"no-restricted-syntax": "off", "no-restricted-syntax": "off",

View File

@ -12,8 +12,8 @@ Maddy's SurfingKeys Configuration
This is my configuration for the [SurfingKeys](https://github.com/brookhong/Surfingkeys) browser extension. This is my configuration for the [SurfingKeys](https://github.com/brookhong/Surfingkeys) browser extension.
Its primary features include Its primary features include
- 129 [Site-Specific Key Mappings](#site-specific-key-mappings) which **automate common actions** on many popular websites - 139 [Site-Specific Key Mappings](#site-specific-key-mappings) which **automate common actions** on many popular websites
- 53 [Search Suggestion Engines](#search-suggestion-engines) providing **interactive search suggestions** from dozens of knowledge sources - 55 [Search Suggestion Engines](#search-suggestion-engines) providing **interactive search suggestions** from dozens of knowledge sources
Table of Contents Table of Contents
----------------- -----------------
@ -30,7 +30,7 @@ Features
### Site-Specific Key Mappings ### Site-Specific Key Mappings
129 key mappings for 21 unique 139 key mappings for 22 unique
sites have been included which can help you perform some common actions: sites have been included which can help you perform some common actions:
<table> <table>
@ -42,6 +42,7 @@ sites have been included which can help you perform some common actions:
<tr><td><code>s</code></td><td>Scroll down</td></tr> <tr><td><code>s</code></td><td>Scroll down</td></tr>
<tr><td><code>K</code></td><td>Scroll half page up</td></tr> <tr><td><code>K</code></td><td>Scroll half page up</td></tr>
<tr><td><code>J</code></td><td>Scroll half page down</td></tr> <tr><td><code>J</code></td><td>Scroll half page down</td></tr>
<tr><td><code>gh</code></td><td>Scroll to element targeted by URL hash</td></tr>
<tr><td><code>gi</code></td><td>Edit current URL with vim editor</td></tr> <tr><td><code>gi</code></td><td>Edit current URL with vim editor</td></tr>
<tr><td><code>gi</code></td><td>Edit current URL with vim editor</td></tr> <tr><td><code>gi</code></td><td>Edit current URL with vim editor</td></tr>
<tr><td><code>gI</code></td><td>View image in new tab</td></tr> <tr><td><code>gI</code></td><td>View image in new tab</td></tr>
@ -63,6 +64,7 @@ sites have been included which can help you perform some common actions:
<tr><td><code>;pd</code></td><td>Toggle PDF viewer from SurfingKeys</td></tr> <tr><td><code>;pd</code></td><td>Toggle PDF viewer from SurfingKeys</td></tr>
<tr><td><code>gxE</code></td><td>Close tab to left</td></tr> <tr><td><code>gxE</code></td><td>Close tab to left</td></tr>
<tr><td><code>gxR</code></td><td>Close tab to right</td></tr> <tr><td><code>gxR</code></td><td>Close tab to right</td></tr>
<tr><td><code>\cgh</code></td><td>Open clipboard string as GitHub path (e.g. 'torvalds/linux')</td></tr>
<tr><th colspan="2"><a href="//algolia.com"><img src="./assets/favicons/algolia.com.ico" width="16px"> algolia.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//algolia.com"><img src="./assets/favicons/algolia.com.ico" width="16px"> algolia.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;a</code></td><td>Open search result</td></tr> <tr><td><code>&lt;Space&gt;a</code></td><td>Open search result</td></tr>
<tr><th colspan="2"><a href="//amazon.com"><img src="./assets/favicons/amazon.com.ico" width="16px"> amazon.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//amazon.com"><img src="./assets/favicons/amazon.com.ico" width="16px"> amazon.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
@ -122,13 +124,23 @@ sites have been included which can help you perform some common actions:
<tr><td><code>&lt;Space&gt;Y</code></td><td>Copy Project Path (including domain)</td></tr> <tr><td><code>&lt;Space&gt;Y</code></td><td>Copy Project Path (including domain)</td></tr>
<tr><td><code>&lt;Space&gt;l</code></td><td>Toggle repo language stats</td></tr> <tr><td><code>&lt;Space&gt;l</code></td><td>Toggle repo language stats</td></tr>
<tr><td><code>&lt;Space&gt;D</code></td><td>View GoDoc for Project</td></tr> <tr><td><code>&lt;Space&gt;D</code></td><td>View GoDoc for Project</td></tr>
<tr><td><code>&lt;Space&gt;H</code></td><td>View RepoHealth Report for Project</td></tr> <tr><td><code>&lt;Space&gt;G</code></td><td>View on SourceGraph</td></tr>
<tr><td><code>&lt;Space&gt;ra</code></td><td>View live raw version of file</td></tr> <tr><td><code>&lt;Space&gt;ra</code></td><td>View live raw version of file</td></tr>
<tr><td><code>&lt;Space&gt;gcp</code></td><td>Open clipboard string as file path in repo</td></tr>
<tr><th colspan="2"><a href="//gitlab.com"><img src="./assets/favicons/gitlab.com.ico" width="16px"> gitlab.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//gitlab.com"><img src="./assets/favicons/gitlab.com.ico" width="16px"> gitlab.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;s</code></td><td>Toggle Star</td></tr> <tr><td><code>&lt;Space&gt;s</code></td><td>Toggle Star</td></tr>
<tr><td><code>&lt;Space&gt;y</code></td><td>Copy Project Path</td></tr> <tr><td><code>&lt;Space&gt;y</code></td><td>Copy Project Path</td></tr>
<tr><td><code>&lt;Space&gt;Y</code></td><td>Copy Project Path (including domain)</td></tr> <tr><td><code>&lt;Space&gt;Y</code></td><td>Copy Project Path (including domain)</td></tr>
<tr><td><code>&lt;Space&gt;D</code></td><td>View GoDoc for Project</td></tr> <tr><td><code>&lt;Space&gt;D</code></td><td>View GoDoc for Project</td></tr>
<tr><th colspan="2"><a href="//home.nest.com"><img src="./assets/favicons/home.nest.com.ico" width="16px"> home.nest.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&#x3D;</code></td><td>Increment temperature</td></tr>
<tr><td><code>-</code></td><td>Decrement temperature</td></tr>
<tr><td><code>&lt;Space&gt;h</code></td><td>Switch mode to Heat</td></tr>
<tr><td><code>&lt;Space&gt;c</code></td><td>Switch mode to Cool</td></tr>
<tr><td><code>&lt;Space&gt;r</code></td><td>Switch mode to Heat/Cool</td></tr>
<tr><td><code>&lt;Space&gt;o</code></td><td>Switch mode to Off</td></tr>
<tr><td><code>&lt;Space&gt;f</code></td><td>Switch fan On</td></tr>
<tr><td><code>&lt;Space&gt;F</code></td><td>Switch fan Off</td></tr>
<tr><th colspan="2"><a href="//news.ycombinator.com"><img src="./assets/favicons/news.ycombinator.com.ico" width="16px"> news.ycombinator.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//news.ycombinator.com"><img src="./assets/favicons/news.ycombinator.com.ico" width="16px"> news.ycombinator.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;x</code></td><td>Collapse comment</td></tr> <tr><td><code>&lt;Space&gt;x</code></td><td>Collapse comment</td></tr>
<tr><td><code>&lt;Space&gt;X</code></td><td>Collapse next comment</td></tr> <tr><td><code>&lt;Space&gt;X</code></td><td>Collapse next comment</td></tr>
@ -163,19 +175,19 @@ sites have been included which can help you perform some common actions:
<tr><td><code>&lt;Space&gt;s</code></td><td>Like tweet</td></tr> <tr><td><code>&lt;Space&gt;s</code></td><td>Like tweet</td></tr>
<tr><td><code>&lt;Space&gt;R</code></td><td>Retweet</td></tr> <tr><td><code>&lt;Space&gt;R</code></td><td>Retweet</td></tr>
<tr><td><code>&lt;Space&gt;c</code></td><td>Comment/Reply</td></tr> <tr><td><code>&lt;Space&gt;c</code></td><td>Comment/Reply</td></tr>
<tr><td><code>&lt;Space&gt;t</code></td><td>New tweet</td></tr> <tr><td><code>&lt;Space&gt;T</code></td><td>New tweet</td></tr>
<tr><td><code>&lt;Space&gt;T</code></td><td>Tweet to</td></tr> <tr><td><code>&lt;Space&gt;u</code></td><td>Goto user</td></tr>
<tr><td><code>&lt;Space&gt;r</code></td><td>Load new tweets</td></tr> <tr><td><code>&lt;Space&gt;t</code></td><td>Goto tweet</td></tr>
<tr><td><code>&lt;Space&gt;g</code></td><td>Goto user</td></tr>
<tr><th colspan="2"><a href="//vimeo.com"><img src="./assets/favicons/vimeo.com.ico" width="16px"> vimeo.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//vimeo.com"><img src="./assets/favicons/vimeo.com.ico" width="16px"> vimeo.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;F</code></td><td>Toggle fullscreen</td></tr> <tr><td><code>&lt;Space&gt;F</code></td><td>Toggle fullscreen</td></tr>
<tr><th colspan="2"><a href="//wikipedia.org"><img src="./assets/favicons/wikipedia.org.ico" width="16px"> wikipedia.org</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//wikipedia.org"><img src="./assets/favicons/wikipedia.org.ico" width="16px"> wikipedia.org</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;s</code></td><td>Toggle simple version of current article</td></tr> <tr><td><code>&lt;Space&gt;s</code></td><td>Toggle simple version of current article</td></tr>
<tr><td><code>&lt;Space&gt;a</code></td><td>View page</td></tr> <tr><td><code>&lt;Space&gt;a</code></td><td>View page</td></tr>
<tr><td><code>&lt;Space&gt;e</code></td><td>View external link</td></tr> <tr><td><code>&lt;Space&gt;e</code></td><td>View external link</td></tr>
<tr><td><code>&lt;Space&gt;R</code></td><td>View WikiRank for current article</td></tr>
<tr><th colspan="2"><a href="//www.google.com"><img src="./assets/favicons/www.google.com.ico" width="16px"> www.google.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//www.google.com"><img src="./assets/favicons/www.google.com.ico" width="16px"> www.google.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;a</code></td><td>Open search result</td></tr> <tr><td><code>&lt;Space&gt;a</code></td><td>Open search result</td></tr>
<tr><td><code>&lt;Space&gt;A</code></td><td>Open search result (non-active new tab)</td></tr> <tr><td><code>&lt;Space&gt;A</code></td><td>Open search result (new tab)</td></tr>
<tr><td><code>&lt;Space&gt;d</code></td><td>Open search in DuckDuckGo</td></tr> <tr><td><code>&lt;Space&gt;d</code></td><td>Open search in DuckDuckGo</td></tr>
<tr><th colspan="2"><a href="//yelp.com"><img src="./assets/favicons/yelp.com.ico" width="16px"> yelp.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr> <tr><th colspan="2"><a href="//yelp.com"><img src="./assets/favicons/yelp.com.ico" width="16px"> yelp.com</a></th></tr><tr><td><strong>Mapping</strong></td><td><strong>Description</strong></td></tr>
<tr><td><code>&lt;Space&gt;fs</code></td><td>Fakespot</td></tr> <tr><td><code>&lt;Space&gt;fs</code></td><td>Fakespot</td></tr>
@ -184,14 +196,13 @@ sites have been included which can help you perform some common actions:
<tr><td><code>C</code></td><td>Open channel</td></tr> <tr><td><code>C</code></td><td>Open channel</td></tr>
<tr><td><code>gH</code></td><td>Goto homepage</td></tr> <tr><td><code>gH</code></td><td>Goto homepage</td></tr>
<tr><td><code>F</code></td><td>Toggle fullscreen</td></tr> <tr><td><code>F</code></td><td>Toggle fullscreen</td></tr>
<tr><td><code>&lt;Space&gt;</code></td><td>Play/pause</td></tr>
</tbody> </tbody>
</table> </table>
### Search Suggestion Engines ### Search Suggestion Engines
There are currently 53 search suggestion engines. There are currently 55 search suggestion engines.
You can access a search suggestion prompt by pressing the search leader key, which is `a` by default, followed by the search engine alias. You can access a search suggestion prompt by pressing the search leader key, which is `a` by default, followed by the search engine alias.
@ -381,6 +392,13 @@ For example, to open the Wikipedia suggestion engine, you would type `awp` while
<td><a href="https://addons.mozilla.org">addons.mozilla.org</a></td> <td><a href="https://addons.mozilla.org">addons.mozilla.org</a></td>
<td></td> <td></td>
</tr> </tr>
<tr>
<td><a href="https://www.google.com"><img src="./assets/favicons/www.google.com.ico" width="16px"> </a></td>
<td><code>gI</code></td>
<td>google-reverse-image</td>
<td><a href="https://www.google.com">www.google.com</a></td>
<td></td>
</tr>
<tr> <tr>
<td><a href="https://godoc.org"><img src="./assets/favicons/godoc.org.ico" width="16px"> </a></td> <td><a href="https://godoc.org"><img src="./assets/favicons/godoc.org.ico" width="16px"> </a></td>
<td><code>gd</code></td> <td><code>gd</code></td>
@ -416,6 +434,13 @@ For example, to open the Wikipedia suggestion engine, you would type `awp` while
<td><a href="https://www.google.com">www.google.com</a></td> <td><a href="https://www.google.com">www.google.com</a></td>
<td></td> <td></td>
</tr> </tr>
<tr>
<td><a href="https://scholar.google.com"><img src="./assets/favicons/scholar.google.com.ico" width="16px"> </a></td>
<td><code>gs</code></td>
<td>google-scholar</td>
<td><a href="https://scholar.google.com">scholar.google.com</a></td>
<td></td>
</tr>
<tr> <tr>
<td><a href="https://gowalker.org"><img src="./assets/favicons/gowalker.org.ico" width="16px"> </a></td> <td><a href="https://gowalker.org"><img src="./assets/favicons/gowalker.org.ico" width="16px"> </a></td>
<td><code>gw</code></td> <td><code>gw</code></td>

View File

@ -26,13 +26,13 @@ actions.getURLPath = ({ count = 0, domain = false } = {}) => {
} }
actions.copyURLPath = ({ count, domain } = {}) => () => actions.copyURLPath = ({ count, domain } = {}) => () =>
Clipboard.write(actions.getURLPath({ count, domain })) Clipboard.write(actions.getURLPath({ count, domain })) // TODO: use navigator.clipboard
actions.copyOrgLink = () => actions.copyOrgLink = () =>
Clipboard.write(`[[${util.getCurrentLocation("href")}][${document.title}]]`) Clipboard.write(`[[${util.getCurrentLocation("href")}][${document.title}]]`) // TODO: use navigator.clipboard
actions.copyMarkdownLink = () => actions.copyMarkdownLink = () =>
Clipboard.write(`[${document.title}](${util.getCurrentLocation("href")})`) Clipboard.write(`[${document.title}](${util.getCurrentLocation("href")})`) // TODO: use navigator.clipboard
actions.duplicateTab = () => actions.duplicateTab = () =>
actions.openLink(util.getCurrentLocation("href"), { newTab: true, active: false })() actions.openLink(util.getCurrentLocation("href"), { newTab: true, active: false })()
@ -104,7 +104,7 @@ actions.scrollToHash = (hash = null) => {
// Surfingkeys-specific actions // Surfingkeys-specific actions
// ---------------------------- // ----------------------------
actions.createHint = (selector, action) => () => { actions.createHints = (selector, action) => () => {
if (typeof action === "undefined") { if (typeof action === "undefined") {
// Use manual reassignment rather than a default arg so that we can lint/bundle without access // Use manual reassignment rather than a default arg so that we can lint/bundle without access
// to the Hints object // to the Hints object
@ -137,7 +137,7 @@ actions.togglePdfViewer = () => chrome.storage.local.get("noPdfViewer", (resp) =
} }
}) })
actions.previewLink = actions.createHint("a[href]", (a) => actions.previewLink = actions.createHints("a[href]", (a) =>
Front.showEditor(a.href, (url) => actions.openLink(url)(), "url")) Front.showEditor(a.href, (url) => actions.openLink(url)(), "url"))
// FakeSpot // FakeSpot
@ -298,6 +298,7 @@ actions.dg.goog = () => {
// GitHub // GitHub
// ------ // ------
// TODO: This is a mess
actions.gh = {} actions.gh = {}
actions.gh.star = ({ toggle = false } = {}) => async () => { actions.gh.star = ({ toggle = false } = {}) => async () => {
const hasDisplayNoneParent = (e) => const hasDisplayNoneParent = (e) =>
@ -363,6 +364,8 @@ actions.gh.parseRepo = (url = util.getCurrentLocation(), rootOnly = false) => {
return cond return cond
? { ? {
type: "repo", type: "repo",
user,
repo,
owner: user, owner: user,
name: repo, name: repo,
href: url, href: url,
@ -390,6 +393,7 @@ actions.gh.parseUser = (url = util.getCurrentLocation(), rootOnly = false) => {
? { ? {
type: "user", type: "user",
name: user, name: user,
user,
href: url, href: url,
url: u, url: u,
userRoot: isRoot, userRoot: isRoot,
@ -400,16 +404,17 @@ actions.gh.parseUser = (url = util.getCurrentLocation(), rootOnly = false) => {
actions.gh.parseFile = (url = util.getCurrentLocation()) => { actions.gh.parseFile = (url = util.getCurrentLocation()) => {
const u = url instanceof URL ? url : new URL(url) const u = url instanceof URL ? url : new URL(url)
const [user, repo, maybeBlob, ...rest] = u.pathname.split("/").filter((s) => s !== "") const [user, repo, pathType, commitHash, ...rest] = u.pathname.split("/").filter((s) => s !== "")
const cond = ( const cond = (
u.origin === util.getCurrentLocation("origin") u.origin === util.getCurrentLocation("origin")
&& typeof user === "string" && typeof user === "string"
&& user.length > 0 && user.length > 0
&& typeof repo === "string" && typeof repo === "string"
&& repo.length > 0 && repo.length > 0
&& typeof maybeBlob === "string" && typeof pathType === "string"
&& (maybeBlob === "blob" || maybeBlob === "tree") && (pathType === "blob" || pathType === "tree")
&& rest.length > 0 && typeof commitHash === "string"
&& commitHash.length > 0
&& /^([a-zA-Z0-9]+-?)+$/.test(user) && /^([a-zA-Z0-9]+-?)+$/.test(user)
&& !ghReservedNames.check(user) && !ghReservedNames.check(user)
) )
@ -418,16 +423,19 @@ actions.gh.parseFile = (url = util.getCurrentLocation()) => {
type: "file", type: "file",
user, user,
repo, repo,
pathType,
commitHash,
href: url, href: url,
url: u, url: u,
filePath: rest, filePath: rest,
repoBase: `/${user}/${repo}`,
} }
: null : null
} }
actions.gh.parseCommit = (url = util.getCurrentLocation()) => { actions.gh.parseCommit = (url = util.getCurrentLocation()) => {
const u = url instanceof URL ? url : new URL(url) const u = url instanceof URL ? url : new URL(url)
const [user, repo, commit, hash] = u.pathname.split("/").filter((s) => s !== "") const [user, repo, commit, commitHash] = u.pathname.split("/").filter((s) => s !== "")
const cond = ( const cond = (
u.origin === util.getCurrentLocation("origin") u.origin === util.getCurrentLocation("origin")
&& typeof user === "string" && typeof user === "string"
@ -436,17 +444,19 @@ actions.gh.parseCommit = (url = util.getCurrentLocation()) => {
&& repo.length > 0 && repo.length > 0
&& typeof commit === "string" && typeof commit === "string"
&& commit === "commit" && commit === "commit"
&& typeof hash === "string" && typeof commitHash === "string"
&& hash.length > 0 && commitHash.length > 0
&& /^([a-zA-Z0-9]+-?)+$/.test(user) && /^([a-zA-Z0-9]+-?)+$/.test(user)
&& !ghReservedNames.check(user) && !ghReservedNames.check(user)
) )
return cond return cond
? { ? {
type: "commit", type: "commit",
href: url, user,
url: u, repo,
commitHash: hash, commitHash,
href: url,
url: u,
} }
: null : null
} }
@ -551,12 +561,12 @@ actions.gh.goParent = () => {
const segments = util.getCurrentLocation("pathname") const segments = util.getCurrentLocation("pathname")
.split("/").filter((s) => s !== "") .split("/").filter((s) => s !== "")
const newPath = (() => { const newPath = (() => {
const [user, repo, maybeBlob] = segments const [user, repo, pathType] = segments
switch (segments.length) { switch (segments.length) {
case 0: case 0:
return false return false
case 4: case 4:
switch (maybeBlob) { switch (pathType) {
case "blob": case "blob":
case "tree": case "tree":
return [user, repo] return [user, repo]
@ -567,7 +577,7 @@ actions.gh.goParent = () => {
} }
break break
case 5: case 5:
if (maybeBlob === "blob") { if (pathType === "blob") {
return [user, repo] return [user, repo]
} }
break break
@ -582,10 +592,33 @@ actions.gh.goParent = () => {
} }
} }
actions.gh.viewRepoHealth = () => { actions.gh.viewSourceGraph = () => {
const repo = actions.gh.parseRepo() const url = new URL("https://sourcegraph.com/github.com")
if (repo === null) return let page = null
actions.openLink(`https://repohealth.info/report${repo.repoBase}`, { newTab: true })() // The following conditional expressions are indeed intended to be
// assignments, this is not a bug.
if ((page = actions.gh.parseFile(window.location.href)) !== null) {
const filePath = page.filePath.join("/")
url.pathname += `/${page.user}/${page.repo}@${page.commitHash}/-/${page.pathType}/${filePath}`
if (window.location.hash !== "") {
url.hash = window.location.hash
} else if (!util.isElementInViewport(document.querySelector("#L1"))) {
for (const e of document.querySelectorAll(".js-line-number")) {
if (util.isElementInViewport(e)) {
url.hash = e.id
break
}
}
}
} else if ((page = actions.gh.parseCommit(window.location.href)) !== null) {
url.pathname += `/${page.user}/${page.repo}@${page.commitHash}`
} else if ((page = actions.gh.parseRepo(window.location.href)) !== null) {
url.pathname += `/${page.user}/${page.repo}`
} else {
url.pathname = ""
}
actions.openLink(url.href, { newTab: true })()
} }
actions.gh.viewRaw = () => { actions.gh.viewRaw = () => {
@ -594,6 +627,49 @@ actions.gh.viewRaw = () => {
actions.openLink(`https://raw.githack.com/${file.user}/${file.repo}/${file.filePath.join("/")}`, { newTab: true })() actions.openLink(`https://raw.githack.com/${file.user}/${file.repo}/${file.filePath.join("/")}`, { newTab: true })()
} }
actions.gh.openRepoFromClipboard = async ({ newTab = true } = {}) =>
actions.openLink(`https://github.com/${await navigator.clipboard.readText()}`, { newTab })()
actions.gh.openFileFromClipboard = async ({ newTab = true } = {}) => {
const clip = await navigator.clipboard.readText()
if (typeof clip !== "string" || clip.length === 0) {
return
}
const loc = util.getCurrentLocation()
const dest = {
user: null,
repo: null,
commitHash: "master",
}
const file = actions.gh.parseFile(loc)
if (file !== null) {
dest.user = file.user
dest.repo = file.repo
dest.commitHash = file.commitHash
} else {
const commit = actions.gh.parseCommit(loc)
if (commit !== null) {
dest.user = commit.user
dest.repo = commit.repo
dest.commitHash = commit.commitHash
} else {
const repository = actions.gh.parseRepo(loc)
if (repository !== null) {
return
}
dest.user = repository.user
dest.repo = repository.repo
}
}
actions.openLink(
`https://github.com/${dest.user}/${dest.repo}/tree/${dest.commitHash}/${clip}`,
{ newTab },
)()
}
// GitLab // GitLab
// ------ // ------
actions.gl = {} actions.gl = {}
@ -609,6 +685,17 @@ actions.gl.star = () => {
Front.showBanner(`${star} Repository ${repo} ${action}`) Front.showBanner(`${star} Repository ${repo} ${action}`)
} }
// Twitter
// ------
actions.tw = {}
actions.tw.openUser = () =>
actions.createHints([].concat(
[...document.querySelectorAll("a[role='link'] img[src^='https://pbs.twimg.com/profile_images']")]
.map((e) => e.closest("a")),
[...document.querySelectorAll("a[role='link']")]
.filter((e) => e.text.match(/^@/)),
))()
// Reddit // Reddit
// ------ // ------
actions.re = {} actions.re = {}
@ -696,7 +783,7 @@ actions.ph.openExternal = () => {
// Dribbble // Dribbble
// -------- // --------
actions.dr = {} actions.dr = {}
actions.dr.attachment = (cb = (a) => actions.openLink(a, { newTab: true })()) => actions.createHint(".attachments .thumb", (a) => cb(a.src.replace("/thumbnail/", "/"))) actions.dr.attachment = (cb = (a) => actions.openLink(a, { newTab: true })()) => actions.createHints(".attachments .thumb", (a) => cb(a.src.replace("/thumbnail/", "/")))
// Wikipedia // Wikipedia
// --------- // ---------

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

After

Width:  |  Height:  |  Size: 1.2 KiB

View File

@ -1 +1 @@
{"www.archlinux.org":"www.archlinux.org.ico","aur.archlinux.org":"aur.archlinux.org.ico","wiki.archlinux.org":"wiki.archlinux.org.ico","bbs.archlinux.org":"bbs.archlinux.org.ico","alternativeto.net":"alternativeto.net.ico","chrome.google.com":"chrome.google.com.ico","addons.mozilla.org":"addons.mozilla.org.ico","www.owasp.org":"www.owasp.org.ico","stackoverflow.com":"stackoverflow.com.ico","stackexchange.com":"stackexchange.com.ico","hub.docker.com":"hub.docker.com.ico","github.com":"github.com.ico","domainr.com":"domainr.com.ico","vim.fandom.com":"vim.fandom.com.ico","smile.amazon.com":"smile.amazon.com.ico","www.craigslist.org":"www.craigslist.org.ico","www.ebay.com":"www.ebay.com.ico","www.yelp.com":"www.yelp.com.ico","onelook.com":"onelook.com.ico","www.onelook.com":"www.onelook.com.ico","en.wikipedia.org":"en.wikipedia.org.ico","simple.wikipedia.org":"simple.wikipedia.org.ico","en.wiktionary.org":"en.wiktionary.org.ico","www.wolframalpha.com":"www.wolframalpha.com.ico","www.crunchbase.com":"www.crunchbase.com.ico","duckduckgo.com":"duckduckgo.com.ico","www.google.com":"www.google.com.ico","hex.pm":"hex.pm.ico","golang.org":"golang.org.ico","godoc.org":"godoc.org.ico","gowalker.org":"gowalker.org.ico","hackage.haskell.org":"hackage.haskell.org.ico","www.haskell.org":"www.haskell.org.ico","wiki.haskell.org":"wiki.haskell.org.ico","jquery.com":"jquery.com.ico","nodejs.org":"nodejs.org.ico","developer.mozilla.org":"developer.mozilla.org.ico","www.npmjs.com":"www.npmjs.com.ico","news.ycombinator.com":"news.ycombinator.com.ico","www.reddit.com":"www.reddit.com.ico","www.youtube.com":"www.youtube.com.ico","amazon.com":"amazon.com.ico","algolia.com":"algolia.com.ico","yelp.com":"yelp.com.ico","youtube.com":"youtube.com.ico","vimeo.com":"vimeo.com.ico","gitlab.com":"gitlab.com.ico","twitter.com":"twitter.com.ico","reddit.com":"reddit.com.ico","producthunt.com":"producthunt.com.ico","dribbble.com":"dribbble.com.ico","behance.net":"behance.net.ico","fonts.adobe.com":"fonts.adobe.com.ico","wikipedia.org":"wikipedia.org.ico","craigslist.org":"craigslist.org.ico"} {"www.archlinux.org":"www.archlinux.org.ico","aur.archlinux.org":"aur.archlinux.org.ico","wiki.archlinux.org":"wiki.archlinux.org.ico","bbs.archlinux.org":"bbs.archlinux.org.ico","alternativeto.net":"alternativeto.net.ico","chrome.google.com":"chrome.google.com.ico","addons.mozilla.org":"addons.mozilla.org.ico","www.owasp.org":"www.owasp.org.ico","stackoverflow.com":"stackoverflow.com.ico","stackexchange.com":"stackexchange.com.ico","hub.docker.com":"hub.docker.com.ico","github.com":"github.com.ico","domainr.com":"domainr.com.ico","vim.fandom.com":"vim.fandom.com.ico","smile.amazon.com":"smile.amazon.com.ico","www.craigslist.org":"www.craigslist.org.ico","www.ebay.com":"www.ebay.com.ico","www.yelp.com":"www.yelp.com.ico","onelook.com":"onelook.com.ico","www.onelook.com":"www.onelook.com.ico","en.wikipedia.org":"en.wikipedia.org.ico","simple.wikipedia.org":"simple.wikipedia.org.ico","en.wiktionary.org":"en.wiktionary.org.ico","www.wolframalpha.com":"www.wolframalpha.com.ico","www.crunchbase.com":"www.crunchbase.com.ico","duckduckgo.com":"duckduckgo.com.ico","www.google.com":"www.google.com.ico","scholar.google.com":"scholar.google.com.ico","hex.pm":"hex.pm.ico","golang.org":"golang.org.ico","godoc.org":"godoc.org.ico","gowalker.org":"gowalker.org.ico","hackage.haskell.org":"hackage.haskell.org.ico","www.haskell.org":"www.haskell.org.ico","wiki.haskell.org":"wiki.haskell.org.ico","jquery.com":"jquery.com.ico","nodejs.org":"nodejs.org.ico","developer.mozilla.org":"developer.mozilla.org.ico","www.npmjs.com":"www.npmjs.com.ico","news.ycombinator.com":"news.ycombinator.com.ico","www.reddit.com":"www.reddit.com.ico","www.youtube.com":"www.youtube.com.ico","amazon.com":"amazon.com.ico","algolia.com":"algolia.com.ico","yelp.com":"yelp.com.ico","youtube.com":"youtube.com.ico","vimeo.com":"vimeo.com.ico","gitlab.com":"gitlab.com.ico","twitter.com":"twitter.com.ico","reddit.com":"reddit.com.ico","producthunt.com":"producthunt.com.ico","dribbble.com":"dribbble.com.ico","behance.net":"behance.net.ico","fonts.adobe.com":"fonts.adobe.com.ico","wikipedia.org":"wikipedia.org.ico","craigslist.org":"craigslist.org.ico","home.nest.com":"home.nest.com.ico"}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.3 KiB

After

Width:  |  Height:  |  Size: 5.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.1 KiB

After

Width:  |  Height:  |  Size: 498 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 11 KiB

After

Width:  |  Height:  |  Size: 8.5 KiB

View File

@ -5,6 +5,8 @@ const wpDefaultIcon = "" const cbDefaultIcon = ""
const locale = typeof navigator !== "undefined" ? navigator.language : ""
const completions = {} const completions = {}
// Helper functions for Google Custom Search Engine completions // Helper functions for Google Custom Search Engine completions
@ -167,7 +169,7 @@ const parseFirefoxAddonsRes = (response) => JSON.parse(response.text).results.ma
completions.fa = { completions.fa = {
alias: "fa", alias: "fa",
name: "firefox-addons", name: "firefox-addons",
search: "https://addons.mozilla.org/en-US/firefox/search/?q=", search: `https://addons.mozilla.org/${locale}/firefox/search/?q=`,
compl: "https://addons.mozilla.org/api/v4/addons/autocomplete/?q=", compl: "https://addons.mozilla.org/api/v4/addons/autocomplete/?q=",
callback: parseFirefoxAddonsRes, callback: parseFirefoxAddonsRes,
} }
@ -176,7 +178,7 @@ completions.fa = {
completions.ft = { completions.ft = {
alias: "ft", alias: "ft",
name: "firefox-themes", name: "firefox-themes",
search: "https://addons.mozilla.org/en-US/firefox/search/?type=statictheme&q=", search: `https://addons.mozilla.org/${locale}/firefox/search/?type=statictheme&q=`,
compl: "https://addons.mozilla.org/api/v4/addons/autocomplete/?type=statictheme&q=", compl: "https://addons.mozilla.org/api/v4/addons/autocomplete/?type=statictheme&q=",
callback: parseFirefoxAddonsRes, callback: parseFirefoxAddonsRes,
} }
@ -185,7 +187,7 @@ completions.ft = {
completions.fe = { completions.fe = {
alias: "fe", alias: "fe",
name: "firefox-extensions", name: "firefox-extensions",
search: "https://addons.mozilla.org/en-US/firefox/search/?type=extension&q=", search: `https://addons.mozilla.org/${locale}/firefox/search/?type=extension&q=`,
compl: "https://addons.mozilla.org/api/v4/addons/autocomplete/?type=extension&q=", compl: "https://addons.mozilla.org/api/v4/addons/autocomplete/?type=extension&q=",
callback: parseFirefoxAddonsRes, callback: parseFirefoxAddonsRes,
} }
@ -983,8 +985,8 @@ completions.no = {
completions.md = { completions.md = {
alias: "md", alias: "md",
name: "mdn", name: "mdn",
search: "https://developer.mozilla.org/en-US/search?q=", search: `https://developer.mozilla.org/${locale}/search?q=`,
compl: "https://developer.mozilla.org/en-US/search.json?q=", compl: `https://developer.mozilla.org/api/v1/search/${locale}?q=`,
} }
completions.md.callback = (response) => { completions.md.callback = (response) => {
@ -1005,7 +1007,7 @@ completions.md.callback = (response) => {
<div style="font-size:0.8em"><em>${slug}</em></div> <div style="font-size:0.8em"><em>${slug}</em></div>
<div>${excerpt}</div> <div>${excerpt}</div>
</div> </div>
`, { url: `https://developer.mozilla.org/${s.slug}` }) `, { url: `https://developer.mozilla.org/${s.locale}/docs/${s.slug}` })
}) })
} }

View File

@ -8,6 +8,10 @@ util.addSettings({
omnibarSuggestionTimeout: 500, omnibarSuggestionTimeout: 500,
richHintsForKeystroke: 1, richHintsForKeystroke: 1,
theme: ` theme: `
body {
font-family: "DejaVu Sans", DejaVu, Arial, sans-serif;
}
/* Disable RichHints CSS animation */ /* Disable RichHints CSS animation */
.expandRichHints { .expandRichHints {
animation: 0s ease-in-out 1 forwards expandRichHints; animation: 0s ease-in-out 1 forwards expandRichHints;

View File

@ -3,7 +3,9 @@ const gulp = require("gulp")
const { task, src, dest } = gulp const { task, src, dest } = gulp
const { parallel, series } = gulp const { parallel, series } = gulp
const parcel = require("gulp-parcel") const webpack = require("webpack")
const webpackStream = require("webpack-stream")
const replace = require("gulp-replace") const replace = require("gulp-replace")
const rename = require("gulp-rename") const rename = require("gulp-rename")
const eslint = require("gulp-eslint") const eslint = require("gulp-eslint")
@ -15,11 +17,14 @@ const fs = require("fs").promises
const fetch = require("node-fetch") const fetch = require("node-fetch")
const express = require("express") const express = require("express")
const gulpIf = require("gulp-if") const gulpIf = require("gulp-if")
const gulpNotify = require("gulp-notify")
const PluginError = require("plugin-error")
const { COPYFILE_EXCL } = require("fs").constants const { COPYFILE_EXCL } = require("fs").constants
const { URL } = require("url") const { URL } = require("url")
const pkg = require("./package.json") const pkg = require("./package.json")
const webpackConfig = require("./webpack.config.js")
const copyrightYearOne = 2017 const copyrightYearOne = 2017
@ -34,6 +39,7 @@ const paths = {
faviconsManifest: "favicons.json", faviconsManifest: "favicons.json",
readmeOut: "README.md", readmeOut: "README.md",
scriptOut: "surfingkeys.js", scriptOut: "surfingkeys.js",
buildDir: "build/",
installDir: platforms.getConfigHome(), installDir: platforms.getConfigHome(),
} }
@ -82,13 +88,48 @@ The source file is ${paths.readme}
-->` -->`
task("clean", () => del(["build", ".cache", ".tmp-gulp-compile-*"])) const notify = Object.assign((opts, ...args) => gulpNotify({
icon: null,
onLast: true,
timeout: 2000,
...opts,
}, ...args), gulpNotify)
notify.onError = (opts, ...args) => gulpNotify.onError({
onLast: true,
timeout: 7500,
...opts,
}, ...args)
task("clean", () => del(["build", ".cache", ".tmp-gulp-compile-*"]))
task("clean-favicons", () => del([paths.favicons])) task("clean-favicons", () => del([paths.favicons]))
const lint = (globs, opts = {}) => gulp.src(globs) const lint = (globs, opts = {}) => gulp.src(globs)
.pipe(eslint(opts)) .pipe(eslint(opts))
.pipe(eslint.format()) .pipe(eslint.format())
.pipe(eslint.results((res) => {
if (res.warningCount + res.errorCount > 0) {
const msg = []
if (res.warningCount > 0) {
msg.push(`${res.warningCount} warning${res.warningCount === 1 ? "" : "s"}`)
}
if (res.errorCount > 0) {
msg.push(`${res.errorCount} error${res.errorCount === 1 ? "" : "s"}`)
}
const numBadFiles = res.reduce((acc, e) =>
(e.errorCount + e.warningCount > 0 ? acc + 1 : acc),
0)
const err = new PluginError("gulp-eslint", {
name: "ESLintError",
message: `${msg.join(" and ")} in ${numBadFiles} file${numBadFiles === 1 ? "" : "s"}`,
})
notify.onError({
title: `Lint failure [${pkg.name}]`,
message: err.message,
})(err)
throw err
}
}))
task("lint", () => lint([...paths.scripts, paths.gulpfile])) task("lint", () => lint([...paths.scripts, paths.gulpfile]))
@ -295,24 +336,41 @@ task("favicons", series("clean-favicons", async () => {
task("docs-full", series("favicons", "docs")) task("docs-full", series("favicons", "docs"))
const build = () =>
src(paths.entry)
.pipe(webpackStream(webpackConfig, webpack))
.on("error", (err) => {
notify.onError({
title: `Build failure [${pkg.name}]`,
message: `${err.message.split("\n").slice(0, 1)}`,
})(err)
throw err
})
.pipe(rename(paths.scriptOut))
.pipe(dest(paths.buildDir))
.pipe(notify({
title: `Build success [${pkg.name}]`,
message: "No issues",
}))
task("build", task("build",
parallel(
"lint",
build,
))
task("build-full",
series( series(
parallel( parallel(
"check-priv", "check-priv",
"clean", "clean",
), ),
parallel( "build",
"lint",
() => src(paths.entry, { read: false })
.pipe(parcel())
.pipe(rename(paths.scriptOut))
.pipe(dest("build")),
),
)) ))
task("dist", parallel("docs-full", "build")) task("dist", parallel("docs-full", "build-full"))
task("install", series("build", () => src(path.join("build", paths.scriptOut)) task("install", series("build", () => src(path.join(paths.buildDir, paths.scriptOut))
.pipe(dest(paths.installDir)))) .pipe(dest(paths.installDir))))
const watch = (g, t) => () => const watch = (g, t) => () =>
@ -334,7 +392,7 @@ const serve = (done) => {
const handler = (allowedOrigin) => async (req, res) => { const handler = (allowedOrigin) => async (req, res) => {
console.log(`${new Date().toISOString()} ${req.method} ${req.url}`) // eslint-disable-line no-console console.log(`${new Date().toISOString()} ${req.method} ${req.url}`) // eslint-disable-line no-console
try { try {
res.sendFile(path.resolve(__dirname, "build", paths.scriptOut), { res.sendFile(path.resolve(__dirname, paths.buildDir, paths.scriptOut), {
headers: { headers: {
"Content-Type": "text/javascript; charset=UTF-8", "Content-Type": "text/javascript; charset=UTF-8",
"Access-Control-Allow-Origin": allowedOrigin, "Access-Control-Allow-Origin": allowedOrigin,
@ -366,3 +424,5 @@ task("serve-build", parallel("watch-build", "serve-simple"))
task("serve", series("serve-build")) task("serve", series("serve-build"))
task("watch", series("watch-install")) task("watch", series("watch-install"))
task("default", series("build")) task("default", series("build"))
// debugger

1592
keys.js

File diff suppressed because it is too large Load Diff

View File

@ -20,30 +20,25 @@
"contributors": [], "contributors": [],
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"github-reserved-names": "^1.1.9" "github-reserved-names": "^2.0.4"
}, },
"devDependencies": { "devDependencies": {
"babel-core": "^6.26.3", "del": "^6.0.0",
"babel-preset-env": "^1.7.0", "eslint": "7.16.0",
"del": "^5.1.0", "eslint-config-airbnb-base": "14.2.1",
"eslint": "7.3.1", "eslint-plugin-import": "^2.22.1",
"eslint-config-airbnb-base": "14.2.0",
"eslint-plugin-import": "^2.22.0",
"express": "^4.17.1", "express": "^4.17.1",
"gulp": "^4.0.2", "gulp": "^4.0.2",
"gulp-eslint": "^6.0.0", "gulp-eslint": "^6.0.0",
"gulp-file": "^0.4.0", "gulp-file": "^0.4.0",
"gulp-if": "^3.0.0", "gulp-if": "^3.0.0",
"gulp-parcel": "^0.3.6", "gulp-notify": "^3.2.0",
"gulp-rename": "^2.0.0", "gulp-rename": "^2.0.0",
"gulp-replace": "^1.0.0", "gulp-replace": "^1.0.0",
"json5": "^2.1.1", "node-fetch": "^2.6.1",
"node-fetch": "^2.6.0", "platform-folders": "^0.5.2",
"platform-folders": "^0.5.1" "plugin-error": "^1.0.1",
}, "webpack": "^5.11.1",
"config": { "webpack-stream": "^6.1.1"
"commitizen": {
"path": "./node_modules/cz-conventional-changelog"
}
} }
} }

View File

@ -149,7 +149,7 @@ util.processCompletions = (completions, searchleader) => Object.values(completio
addSearchAliasX(s.alias, s.name, s.search, searchleader, s.compl, s.callback) addSearchAliasX(s.alias, s.name, s.search, searchleader, s.compl, s.callback)
mapkey(`${searchleader}${s.alias}`, `#8Search ${s.name}`, () => Front.openOmnibar({ type: "SearchEngine", extra: s.alias })) mapkey(`${searchleader}${s.alias}`, `#8Search ${s.name}`, () => Front.openOmnibar({ type: "SearchEngine", extra: s.alias }))
mapkey(`c${searchleader}${s.alias}`, `#8Search ${s.name} with clipboard contents`, () => { mapkey(`c${searchleader}${s.alias}`, `#8Search ${s.name} with clipboard contents`, () => {
Clipboard.read((c) => { Clipboard.read((c) => { // TODO: use navigator.clipboard
Front.openOmnibar({ type: "SearchEngine", pref: c.data, extra: s.alias }) Front.openOmnibar({ type: "SearchEngine", pref: c.data, extra: s.alias })
}) })
}) })

3
webpack.config.js Normal file
View File

@ -0,0 +1,3 @@
module.exports = {
mode: "development",
}

5279
yarn.lock

File diff suppressed because it is too large Load Diff