├── promo.png ├── sketches ├── icon.sketch ├── folder.sketch └── exports │ ├── icon16.png │ ├── icon32.png │ ├── icon48.png │ ├── favicon.png │ ├── icon32@4x.png │ ├── folder-dark.png │ ├── folder-dark@2x.png │ ├── folder-light.png │ └── folder-light@2x.png ├── unpacked-extension ├── icons │ ├── icon16.png │ ├── icon32.png │ ├── icon48.png │ └── icon128.png ├── resources │ ├── folder@2x.png │ └── folder-dark@2x.png ├── newtab.html ├── manifest.json ├── newtab.css └── newtab.js └── README.md /promo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/promo.png -------------------------------------------------------------------------------- /sketches/icon.sketch: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/icon.sketch -------------------------------------------------------------------------------- /sketches/folder.sketch: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/folder.sketch -------------------------------------------------------------------------------- /sketches/exports/icon16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/icon16.png -------------------------------------------------------------------------------- /sketches/exports/icon32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/icon32.png -------------------------------------------------------------------------------- /sketches/exports/icon48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/icon48.png -------------------------------------------------------------------------------- /sketches/exports/favicon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/favicon.png -------------------------------------------------------------------------------- /sketches/exports/icon32@4x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/icon32@4x.png -------------------------------------------------------------------------------- /sketches/exports/folder-dark.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/folder-dark.png -------------------------------------------------------------------------------- /sketches/exports/folder-dark@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/folder-dark@2x.png -------------------------------------------------------------------------------- /sketches/exports/folder-light.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/folder-light.png -------------------------------------------------------------------------------- /unpacked-extension/icons/icon16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/unpacked-extension/icons/icon16.png -------------------------------------------------------------------------------- /unpacked-extension/icons/icon32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/unpacked-extension/icons/icon32.png -------------------------------------------------------------------------------- /unpacked-extension/icons/icon48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/unpacked-extension/icons/icon48.png -------------------------------------------------------------------------------- /sketches/exports/folder-light@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/sketches/exports/folder-light@2x.png -------------------------------------------------------------------------------- /unpacked-extension/icons/icon128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/unpacked-extension/icons/icon128.png -------------------------------------------------------------------------------- /unpacked-extension/resources/folder@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/unpacked-extension/resources/folder@2x.png -------------------------------------------------------------------------------- /unpacked-extension/resources/folder-dark@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/adamschwartz/chrome-new-tab/HEAD/unpacked-extension/resources/folder-dark@2x.png -------------------------------------------------------------------------------- /unpacked-extension/newtab.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | New Tab 4 | 5 |
6 |
7 |
8 |
9 | 10 | -------------------------------------------------------------------------------- /unpacked-extension/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "author": "Adam Schwartz", 3 | "name": "New Tab", 4 | "version": "15.0", 5 | "manifest_version": 3, 6 | "minimum_chrome_version": "110", 7 | "description": "A minimalist replacement for Chrome's New Tab page.", 8 | "icons": { 9 | "128": "icons/icon128.png", 10 | "48": "icons/icon48.png", 11 | "32": "icons/icon32.png", 12 | "16": "icons/icon16.png" 13 | }, 14 | "permissions": [ 15 | "favicon", 16 | "bookmarks" 17 | ], 18 | "chrome_url_overrides": { 19 | "newtab": "newtab.html" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ## Chrome New Tab 2 | 3 | ### On the Chrome Web Store 4 | 5 | Visit the [New Tab on the Chrome Web Store](https://chrome.google.com/webstore/detail/new-tab/adcpijkmbecohfalcbafjgadfnpchhlg) and choose "Add to Chrome". 6 | 7 | [Buy me a beer →](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=D6XM3J8GW548W) 8 | 9 | ### Local Installation 10 | `git clone` this repo. Go to Chrome Settings > Extensions ([chrome://extensions](chrome://extensions)). Choose "Load unpacked extension" and choose the `unpacked-extension` folder in this repo. Disable the extension to switch back to the original Chrome New Tab page. 11 | -------------------------------------------------------------------------------- /unpacked-extension/newtab.css: -------------------------------------------------------------------------------- 1 | * { 2 | font: inherit; 3 | box-sizing: inherit; 4 | } 5 | 6 | :root { 7 | --menu-box-shadow: 0 0 0 0.5px rgba(0, 0, 0, .25), 0 2px 11px rgba(0, 0, 0, .3); 8 | } 9 | 10 | body { 11 | margin: 0; 12 | padding: 0; 13 | font-family: -apple-system, BlinkMacSystemFont, sans-serif; 14 | font-size: 12px; 15 | line-height: 1.5; 16 | color: #3c4043; 17 | background: #fff; 18 | box-sizing: border-box; 19 | -webkit-user-select: none; 20 | min-height: 100vh; 21 | } 22 | 23 | @media (prefers-color-scheme: dark) { 24 | body { 25 | color: #fff; 26 | background: #1f2020; 27 | } 28 | } 29 | 30 | @media (prefers-color-scheme: dark) { 31 | .bookmarks { 32 | background: #3c3c3c; 33 | border-bottom: 1px solid #454746; 34 | } 35 | } 36 | 37 | ul { 38 | list-style: none; 39 | } 40 | 41 | .bookmarks { 42 | display: flex; 43 | } 44 | 45 | #bookmarks-bar { 46 | margin-right: auto; 47 | } 48 | 49 | #other-bookmarks > div > ul > li:not(:last-child) { 50 | display: none; 51 | } 52 | 53 | .bookmarks ul { 54 | display: flex; 55 | flex-wrap: wrap; 56 | margin: 0; 57 | padding: 4px 6px; 58 | } 59 | 60 | .bookmarks ul li { 61 | position: relative; 62 | display: inline-flex; 63 | align-items: center; 64 | justify-content: center; 65 | } 66 | 67 | .bookmarks .wrap.wrap-is-close { 68 | display: none; 69 | } 70 | 71 | .bookmarks li a { 72 | display: inline-block; 73 | overflow: hidden; 74 | text-overflow: ellipsis; 75 | max-width: 142px; 76 | letter-spacing: .004em; 77 | white-space: nowrap; 78 | text-decoration: none; 79 | color: inherit; 80 | padding: 5px 6px; 81 | margin: 0 2px; 82 | border-radius: 99em; 83 | cursor: pointer; 84 | transition: background .3s ease; 85 | } 86 | 87 | .bookmarks li a:focus { 88 | outline: none; 89 | } 90 | 91 | .bookmarks li a:hover, .bookmarks li a:focus { 92 | background: #edeeee; 93 | } 94 | 95 | .bookmarks li a:hover:active, .bookmarks li a.open { 96 | background: #e1e2e2; 97 | transition: none; 98 | } 99 | 100 | @media (prefers-color-scheme: dark) { 101 | .bookmarks li a:hover, .bookmarks li a:focus { 102 | background: #35363a; 103 | } 104 | 105 | .bookmarks li a:hover:active, .bookmarks li a.open { 106 | background: #505154; 107 | } 108 | } 109 | 110 | .bookmarks li .icon { 111 | display: inline-block; 112 | vertical-align: middle; 113 | width: 16px; 114 | height: 16px; 115 | background-size: 100% 100%; 116 | margin-right: 8px; 117 | position: relative; 118 | top: -1px; 119 | } 120 | 121 | .folder.open + div { 122 | position: absolute; 123 | background: #fff; 124 | padding: 10px 0; 125 | z-index: 100; 126 | border-radius: 10px; 127 | box-shadow: var(--menu-box-shadow); 128 | top: calc(100% + 4px); 129 | left: 2px; 130 | } 131 | 132 | #other-bookmarks .folder.open + div { 133 | left: auto; 134 | right: 2px; 135 | } 136 | 137 | @media (prefers-color-scheme: dark) { 138 | .folder.open + div { 139 | background: #292a2d; 140 | } 141 | } 142 | 143 | .folder.open + div .folder.open + div { 144 | left: 50px; 145 | top: calc(100% + 4px); 146 | } 147 | 148 | #other-bookmarks .folder.open + div .folder.open + div { 149 | left: auto; 150 | right: 50px; 151 | } 152 | 153 | .folder.open + div ul { 154 | display: block; 155 | padding: 0; 156 | min-width: 240px; 157 | max-width: min(calc(100vw - 32px), 400px); 158 | } 159 | 160 | .folder.open + div ul li { 161 | display: block; 162 | } 163 | 164 | .folder.open + div li + li { 165 | margin-left: 0; 166 | } 167 | 168 | .folder.open + div li a { 169 | width: 100%; 170 | display: block; 171 | padding: 4px 24px; 172 | border-radius: 0; 173 | max-width: 100%; 174 | transition: none; 175 | font-size: 13px; 176 | margin: 0; 177 | } 178 | 179 | .folder.open + div li .icon { 180 | margin-right: 12px; 181 | } 182 | 183 | .folder.open + div li a:hover { 184 | background: #e8e8e9; 185 | } 186 | 187 | @media (prefers-color-scheme: dark) { 188 | .folder.open + div li a:hover { 189 | background: #3f4042; 190 | } 191 | } 192 | 193 | .folder.open + div li { 194 | position: relative; 195 | } 196 | 197 | a.empty { 198 | opacity: .4; 199 | pointer-events: none; 200 | } 201 | 202 | a.empty .icon { 203 | display: none; 204 | } 205 | 206 | .folder .icon { 207 | background-image: -webkit-image-set( 208 | url() 1x, 209 | url() 2x 210 | ); 211 | } 212 | 213 | @media (prefers-color-scheme: dark) { 214 | .folder .icon { 215 | background-image: -webkit-image-set( 216 | url() 1x, 217 | url() 2x 218 | ); 219 | } 220 | } 221 | 222 | .menu { 223 | margin: 0; 224 | padding: 0; 225 | list-style: none; 226 | } 227 | 228 | .menu { 229 | position: absolute; 230 | white-space: nowrap; 231 | z-index: 200; 232 | background: #fff; 233 | padding: 10px 0; 234 | font-size: 13px; 235 | border-radius: 10px; 236 | box-shadow: var(--menu-box-shadow); 237 | } 238 | 239 | @media (prefers-color-scheme: dark) { 240 | .menu { 241 | background: #292a2d; 242 | } 243 | } 244 | 245 | .menu a { 246 | display: block; 247 | color: inherit; 248 | width: 100%; 249 | padding: 4px 24px; 250 | text-decoration: none; 251 | } 252 | 253 | .menu .icon { 254 | margin-right: 12px; 255 | } 256 | 257 | .menu a:hover { 258 | background: #e8e8e9; 259 | } 260 | 261 | @media (prefers-color-scheme: dark) { 262 | .menu a:hover { 263 | background: #3f4042; 264 | } 265 | } 266 | -------------------------------------------------------------------------------- /unpacked-extension/newtab.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const foldersOpen = {} 4 | 5 | const bookmarksBarEl = document.getElementById('bookmarks-bar') 6 | const otherBookmarksEl = document.getElementById('other-bookmarks') 7 | 8 | function renderAll(nodes, target, toplevel) { 9 | var ul = document.createElement('ul') 10 | 11 | for (var i = 0; i < nodes.length; i++) { 12 | var node = nodes[i] 13 | render(node, ul) 14 | } 15 | 16 | if (ul.childNodes.length === 0) { 17 | render({ id: 'empty', className: 'empty', title: '(empty)' }, ul) 18 | } 19 | 20 | if (toplevel) { 21 | target.appendChild(ul) 22 | 23 | } else { 24 | var wrap = document.createElement('div') 25 | wrap.appendChild(ul) 26 | target.appendChild(wrap) 27 | } 28 | 29 | return ul 30 | } 31 | 32 | function render(node, target) { 33 | var li = document.createElement('li') 34 | var a = document.createElement('a') 35 | 36 | var url = node.url || node.appLaunchUrl 37 | if (url) a.href = url 38 | 39 | a.innerText = node.title || node.name || '' 40 | if (node.tooltip) a.title = node.tooltip 41 | 42 | setClass(a, node) 43 | 44 | a.insertBefore(getIcon(node), a.firstChild) 45 | 46 | if (url) { 47 | var items = getMenuItems(node) 48 | 49 | a.oncontextmenu = event => { 50 | renderMenu(items, event.pageX, event.pageY) 51 | return false 52 | } 53 | 54 | var urlStart = url.substring(0, 6) 55 | if (urlStart === 'chrome' || urlStart === 'file:/') { 56 | a.onclick = function(event) { 57 | openLink(node, event.metaKey) 58 | return false 59 | } 60 | } 61 | 62 | } else if (!node.children && !node.type) { 63 | a.style.pointerEvents = 'none' 64 | } 65 | 66 | li.appendChild(a) 67 | 68 | // Folders 69 | if (node.children) { 70 | a.onclick = function() { 71 | toggle(node, a, getChildrenFunction(node)) 72 | return false 73 | } 74 | 75 | var items = getMenuItems(node) 76 | 77 | a.oncontextmenu = event => { 78 | renderMenu(items, event.pageX, event.pageY) 79 | return false 80 | } 81 | } 82 | 83 | target.appendChild(li) 84 | return li 85 | } 86 | 87 | function getMenuItems(node) { 88 | const items = [] 89 | 90 | if (node.children) { 91 | items.push({ 92 | label: 'Open all links in folder', 93 | action: () => { 94 | openLinks(node) 95 | } 96 | }) 97 | 98 | } else { 99 | items.push({ 100 | label: 'Open link in tab', 101 | action: () => { 102 | openLink(node, 1) 103 | } 104 | }) 105 | } 106 | 107 | items.push({ 108 | label: 'Edit bookmarks', 109 | action: () => { 110 | openLink({ url: 'chrome://bookmarks' }, 1) 111 | } 112 | }) 113 | 114 | return items 115 | } 116 | 117 | function onMenuClick(item) { 118 | return function() { 119 | item.action() 120 | return false 121 | } 122 | } 123 | 124 | function renderMenu(items, x, y) { 125 | closeMenu() 126 | 127 | var ul = document.createElement('ul') 128 | ul.className = 'menu' 129 | 130 | for (var i = 0; i < items.length; i++) { 131 | var li = document.createElement('li') 132 | var a = document.createElement('a') 133 | a.setAttribute('tabindex', 0) 134 | a.innerText = items[i].label 135 | a.onclick = onMenuClick(items[i]) 136 | li.appendChild(a) 137 | ul.appendChild(li) 138 | } 139 | 140 | document.body.appendChild(ul) 141 | 142 | ul.style.left = Math.max(Math.min(x, window.innerWidth + window.scrollX - ul.clientWidth), 0) + 'px' 143 | ul.style.top = Math.max(Math.min(y, window.innerHeight + window.scrollY - ul.clientHeight), 0) + 'px' 144 | 145 | ul.onmousedown = function(event) { 146 | event.stopPropagation() 147 | return true 148 | } 149 | 150 | return ul 151 | } 152 | 153 | document.onmousedown = function(event) { 154 | closeMenu() 155 | } 156 | 157 | document.onkeydown = function() { 158 | if (event.keyCode == 27) { 159 | closeMenu() 160 | closeOpenFolders() 161 | } 162 | } 163 | 164 | function closeMenu(ul) { 165 | const menu = document.querySelector('body > .menu') 166 | if (menu) menu.remove() 167 | } 168 | 169 | const closeOpenFolders = (scopeElement = document.body) => { 170 | const folders = scopeElement.querySelectorAll('.folder.open') 171 | Array.from(folders).reverse().forEach(folder => folder.click()) 172 | } 173 | 174 | document.body.addEventListener('mousedown', event => { 175 | if (event.target.tagName === 'A' && event.target.href) { 176 | return 177 | } 178 | 179 | const closestFolder = event.target.closest('.folder') 180 | 181 | if (!closestFolder) { 182 | closeOpenFolders() 183 | } else { 184 | closeOpenFolders(closestFolder) 185 | } 186 | }) 187 | 188 | function getChildrenFunction(node) { 189 | switch(node.id) { 190 | default: 191 | if (node.children) 192 | return function(callback) { 193 | callback(node.children) 194 | } 195 | else 196 | return function(callback) { 197 | chrome.bookmarks.getSubTree(node.id, function(result) { 198 | if (!result) return 199 | callback(result[0].children) 200 | }) 201 | } 202 | } 203 | } 204 | 205 | function setClass(target, node, isOpen) { 206 | if (node.className) { 207 | target.classList.add(node.className) 208 | } 209 | 210 | if (node.children) { 211 | target.classList.add('folder') 212 | target.setAttribute('tabindex', 0) 213 | } 214 | 215 | if (isOpen) { 216 | target.classList.add('open') 217 | } else { 218 | target.classList.remove('open') 219 | } 220 | } 221 | 222 | function getIcon(node) { 223 | var url, url2x 224 | if (node.icons) { 225 | var size 226 | for (var i in node.icons) { 227 | var iconInfo = node.icons[i] 228 | if (iconInfo.url && (!size || (iconInfo.size < size && iconInfo.size > 15))) { 229 | url = iconInfo.url 230 | if (iconInfo.size > 31) url2x = iconInfo.url 231 | size = iconInfo.size 232 | } 233 | } 234 | 235 | } else if (node.icon) { 236 | url = node.icon 237 | 238 | } else if (node.url || node.appLaunchUrl) { 239 | let pageURL = node.url || node.appLaunchUrl 240 | pageURL = pageURL.replace(/ /g, '%20') 241 | 242 | if (!pageURL || pageURL.substr(0, 11) === 'javascript:' || pageURL.substr(0, 5) === 'data:') { 243 | pageURL = 'https://example.com' 244 | } 245 | 246 | // See https://bugs.chromium.org/p/chromium/issues/detail?id=104102 247 | url = `chrome-extension://${chrome.runtime.id}/_favicon/?pageUrl=${encodeURIComponent(pageURL)}&size=16` 248 | url2x = `chrome-extension://${chrome.runtime.id}/_favicon/?pageUrl=${encodeURIComponent(pageURL)}&size=32` 249 | } 250 | 251 | var icon = document.createElement(url ? 'img' : 'div') 252 | icon.className = 'icon' 253 | icon.src = url 254 | if (url && url2x) icon.srcset = url2x + ' 2x' 255 | icon.alt = '' 256 | return icon 257 | } 258 | 259 | // toggle folder open state 260 | function toggle(node, a) { 261 | var isOpen = foldersOpen[node.id] 262 | setClass(a, node, !isOpen) 263 | a.open = !isOpen 264 | 265 | if (isOpen) { 266 | delete foldersOpen[node.id] 267 | if (a.nextSibling) { 268 | var children = (a.nextSibling.tagName == 'DIV' ? a.nextSibling.firstChild : a.nextSibling).children 269 | for (var i = 0; i < children.length; i++) { 270 | var child = children[i].firstChild 271 | if (child.open) 272 | child.onclick() 273 | } 274 | 275 | toggleOpenClose(node, a, isOpen) 276 | } 277 | 278 | } else { 279 | foldersOpen[node.id] = true 280 | 281 | var siblings = a.parentNode.parentNode.children 282 | for (var i = 0; i < siblings.length; i++) { 283 | var sibling = siblings[i].firstChild 284 | if (sibling != a && sibling.open) 285 | sibling.onclick() 286 | } 287 | 288 | if (a.nextSibling) 289 | toggleOpenClose(node, a, isOpen) 290 | 291 | else 292 | getChildrenFunction(node)(function(result) { 293 | if (!a.nextSibling && foldersOpen[node.id]) { 294 | renderAll(result, a.parentNode) 295 | toggleOpenClose(node, a, isOpen) 296 | } 297 | }) 298 | } 299 | } 300 | 301 | function toggleOpenClose(node, a, isOpen) { 302 | var wrap = a.nextSibling 303 | wrap.className = 'wrap ' + (isOpen ? 'wrap-is-close' : 'wrap-is-open') 304 | } 305 | 306 | // opens immediate children of given node in new tabs 307 | function openLinks(node) { 308 | chrome.tabs.getCurrent(function(tab) { 309 | getChildrenFunction(node)(function(result) { 310 | for (var i = 0; i < result.length; i++) 311 | openLink(result[i], 2) 312 | }) 313 | }) 314 | } 315 | 316 | // opens given node 317 | function openLink(node, newtab) { 318 | var url = node.url || node.appLaunchUrl 319 | if (url) { 320 | chrome.tabs.getCurrent(function(tab) { 321 | if (newtab) 322 | chrome.tabs.create({url: url, active: (newtab == 1), openerTabId: tab.id}) 323 | else 324 | chrome.tabs.update(tab.id, {url: url}) 325 | }) 326 | } 327 | } 328 | 329 | chrome.bookmarks.getTree(function(result) { 330 | var nodes = result[0].children 331 | 332 | const bookmarksBar = nodes[0] 333 | const otherBookmarks = nodes[1] 334 | 335 | if (bookmarksBar.children.length) { 336 | renderAll(bookmarksBar.children, bookmarksBarEl) 337 | } else { 338 | bookmarksBarEl.remove() 339 | } 340 | 341 | if (otherBookmarks.children.length) { 342 | renderAll([otherBookmarks], otherBookmarksEl) 343 | } else { 344 | otherBookmarksEl.remove() 345 | } 346 | }) 347 | --------------------------------------------------------------------------------