From 8d2c18dece57694a6808a89505e7265a2c049ed3 Mon Sep 17 00:00:00 2001 From: Pavel Komiagin Date: Fri, 24 Aug 2018 11:13:55 +0500 Subject: [PATCH 1/6] Now can resize sidebar width and toggle full width of PR view --- src/js/components/tree.jsx | 86 +++++++++++++++++++++++++++++++++++--- src/js/style.css | 67 ++++++++++++++++++++++++++--- 2 files changed, 142 insertions(+), 11 deletions(-) diff --git a/src/js/components/tree.jsx b/src/js/components/tree.jsx index 6525c08..7d18763 100644 --- a/src/js/components/tree.jsx +++ b/src/js/components/tree.jsx @@ -2,12 +2,34 @@ import React from 'react' import Branch from './branch.jsx' import { isElementVisible } from '../lib' +// Minimum and maximum resizable area width +const MIN_WIDTH = 55 +const MAX_WIDTH = 700 + +const widthLocalStorageKey = '__better_github_pr_tree_width' + class Tree extends React.Component { constructor (props) { super(props) this.onClose = this.onClose.bind(this) this.onScroll = this.onScroll.bind(this) + this.onResizerMouseDown = this.onResizerMouseDown.bind(this) + this.onMouseMove = this.onMouseMove.bind(this) + this.onMouseUp = this.onMouseUp.bind(this) + this.toggleDocumentFullWidth = this.toggleDocumentFullWidth.bind(this) + + this.isResizing = false + this.resizeDelta = 0 + + this.treeContainer = document.querySelector('.__better_github_pr') + this.reviewContainers = document.querySelectorAll('.enable_better_github_pr .diff-view, .enable_better_github_pr .commit.full-commit.prh-commit') + + // Set initial width from value saved to localStorrage + const savedWitdh = window.localStorage.getItem(widthLocalStorageKey) + if (savedWitdh) { + this.setWidth(parseInt(savedWitdh, 10)) + } this.state = { show: true, @@ -20,6 +42,10 @@ class Tree extends React.Component { window.addEventListener('load', this.onScroll, false) window.addEventListener('scroll', this.onScroll, false) window.addEventListener('resize', this.onScroll, false) + + this.resizer.addEventListener('mousedown', this.onResizerMouseDown, false) + document.addEventListener('mousemove', this.onMouseMove, false) + document.addEventListener('mouseup', this.onMouseUp, false) } componentWillUnmount () { @@ -27,6 +53,33 @@ class Tree extends React.Component { window.removeEventListener('load', this.onScroll, false) window.removeEventListener('scroll', this.onScroll, false) window.removeEventListener('resize', this.onScroll, false) + + this.resizer.removeEventListener('mousedown', this.onResizerMouseDown, false) + document.removeEventListener('mousemove', this.onMouseMove, false) + document.removeEventListener('mouseup', this.onMouseUp, false) + } + + onResizerMouseDown () { + this.isResizing = true + this.treeContainer.classList.add('__better_github_pr_noselect') + this.prevWidth = this.treeContainer.offsetWidth + this.startResizeX = this.resizer.getBoundingClientRect().x + } + + onMouseMove (e) { + if (!this.isResizing) return + + this.resizeDelta = e.clientX - this.startResizeX + let newWidth = this.prevWidth + this.resizeDelta + setTimeout(() => this.setWidth(newWidth), 0) + } + + onMouseUp () { + if (!this.isResizing) return + + this.isResizing = false + this.treeContainer.classList.remove('__better_github_pr_noselect') + window.localStorage.setItem(widthLocalStorageKey, this.treeContainer.offsetWidth) } onScroll () { @@ -45,6 +98,23 @@ class Tree extends React.Component { const show = false this.setState({ show }) document.body.classList.toggle('enable_better_github_pr', show) + this.setWidth(0, false) + } + + setWidth (width, withConstraints = true) { + if (withConstraints) { + if (width <= MIN_WIDTH) width = MIN_WIDTH + if (width >= MAX_WIDTH) width = MAX_WIDTH + } + + this.treeContainer.style.width = `${width}px` + this.reviewContainers.forEach((element, index) => { + element.style['margin-left'] = `${width + 10}px` + }) + } + + toggleDocumentFullWidth () { + document.querySelector('body').classList.toggle('__better_github_pr_wide') } render () { @@ -56,13 +126,17 @@ class Tree extends React.Component { } return ( -
+
+ - {root.list.map(node => ( - - - - ))} +
+
{ this.resizer = node }} /> + {root.list.map(node => ( + + + + ))} +
) } diff --git a/src/js/style.css b/src/js/style.css index f790752..d2821eb 100644 --- a/src/js/style.css +++ b/src/js/style.css @@ -7,13 +7,29 @@ display: none; } +._better_github_pr_resizer { + position: absolute; + width: 5px; + height: 100%; + background: lightgray; + right: 0; + top: 0; + cursor: col-resize; + opacity: .4; + transition: opacity .2s ease; + z-index: 1; +} + +._better_github_pr_resizer:hover { + opacity: 1; +} + .enable_better_github_pr .__better_github_pr { display: block; position: relative; margin-top: 20px; - padding: 10px; - width: 240px; - overflow: auto; + padding: 30px 10px 0; + width: 330px; height: calc(100vh - 61px); background-color: #fafbfc; border: 1px solid #e1e4e8; @@ -22,7 +38,13 @@ } .__better_github_pr > div { - display: inline-block; + overflow: auto; + height: 100%; + padding-bottom: 10px; +} + +.__better_github_pr > div > div { + width: fit-content; } .__better_github_pr .close_button { @@ -34,11 +56,46 @@ padding: 5px 2px; } +.__better_github_pr_full_width { + width: 16px; + height: 16px; + background-size: cover; + position: absolute; + top: 6px; + right: 30px; + cursor: pointer; + background-image: url(''); +} + +.__better_github_pr_noselect { + -webkit-touch-callout: none; /* iOS Safari */ + -webkit-user-select: none; /* Safari */ + -khtml-user-select: none; /* Konqueror HTML */ + -moz-user-select: none; /* Firefox */ + -ms-user-select: none; /* Internet Explorer/Edge */ + user-select: none; /* Non-prefixed version, currently + supported by Chrome and Opera */ +} + +.__better_github_pr_wide .container { + min-width: 980px; + width: 96% !important; + transition: width .2s ease; +} + +.__better_github_pr_wide .repository-content .discussion-timeline { + width: calc(100% - 220px) !important; +} + +.__better_github_pr_wide .repository-content .discussion-timeline .timeline-new-comment { + max-width: 100% !important; +} + /* GitHub page changes */ .enable_better_github_pr .diff-view, .enable_better_github_pr .commit.full-commit.prh-commit { - margin-left: 250px; + margin-left: 340px; } .enable_better_github_pr .diff-view .file-header { From ad5d8bd64687a6a552eba17966316e49edc85b57 Mon Sep 17 00:00:00 2001 From: Pavel Komiagin Date: Fri, 24 Aug 2018 11:43:26 +0500 Subject: [PATCH 2/6] Fix toggle full width button style --- src/js/style.css | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/js/style.css b/src/js/style.css index d2821eb..c0c17bc 100644 --- a/src/js/style.css +++ b/src/js/style.css @@ -65,6 +65,8 @@ right: 30px; cursor: pointer; background-image: url(''); + background-color: transparent; + border: none; } .__better_github_pr_noselect { From 43a67a0fb01fd69fa1d5ce101708386bb40d7c47 Mon Sep 17 00:00:00 2001 From: jake Rivett Date: Tue, 2 Oct 2018 20:17:22 +0100 Subject: [PATCH 3/6] Added the concating of the folder structure together --- src/js/folderconcat/folder-concat.jsx | 19 +++++++++++++++++++ src/js/index.jsx | 4 +++- 2 files changed, 22 insertions(+), 1 deletion(-) create mode 100644 src/js/folderconcat/folder-concat.jsx diff --git a/src/js/folderconcat/folder-concat.jsx b/src/js/folderconcat/folder-concat.jsx new file mode 100644 index 0000000..9d03961 --- /dev/null +++ b/src/js/folderconcat/folder-concat.jsx @@ -0,0 +1,19 @@ +export const folderConcat = (node) => { + // file or empty + if (node.list === undefined || node.list.length === 0 || (node.href !== null && node.href !== undefined)){ + return node; + } + + if (node.list.length === 1) { + let collapsed = folderConcat(node.list[0]); + // if the last collapsed thing is a folder merge into one. + if (collapsed.href === null || collapsed.href === undefined) { + node.nodeLabel = node.nodeLabel + "/" + collapsed.nodeLabel; + node.hasComments = collapsed.hasComments || node.hasComments; + node.list = collapsed.list; + } + return node; + } + node.list.map(x => folderConcat(x) ); + return node; +}; diff --git a/src/js/index.jsx b/src/js/index.jsx index 709bd1f..d2e1ddc 100644 --- a/src/js/index.jsx +++ b/src/js/index.jsx @@ -2,6 +2,7 @@ import React from 'react' import { render } from 'react-dom' import Tree from './components/tree' import { createFileTree, createRootElement } from './lib' +import { folderConcat } from './folderconcat/folder-concat' import './style.css' const { document, MutationObserver, parseInt } = window @@ -23,7 +24,8 @@ const renderTree = () => { return } - const { tree, count } = createFileTree() + let { tree, count } = createFileTree(); + tree = folderConcat(tree) render(, rootElement) if (fileCount !== count) { setTimeout(renderTree.bind(this, rootElement), 100) From c748b57b4ea588b697fdddbd13be19628005e907 Mon Sep 17 00:00:00 2001 From: jake Rivett Date: Wed, 3 Oct 2018 09:36:57 +0100 Subject: [PATCH 4/6] Changed indentation --- src/js/folderconcat/folder-concat.jsx | 28 +++++++++++++-------------- src/js/index.jsx | 2 +- 2 files changed, 15 insertions(+), 15 deletions(-) diff --git a/src/js/folderconcat/folder-concat.jsx b/src/js/folderconcat/folder-concat.jsx index 9d03961..b0caa35 100644 --- a/src/js/folderconcat/folder-concat.jsx +++ b/src/js/folderconcat/folder-concat.jsx @@ -1,19 +1,19 @@ export const folderConcat = (node) => { // file or empty - if (node.list === undefined || node.list.length === 0 || (node.href !== null && node.href !== undefined)){ - return node; - } + if (node.list === undefined || node.list.length === 0 || (node.href !== null && node.href !== undefined)) { + return node + } - if (node.list.length === 1) { - let collapsed = folderConcat(node.list[0]); + if (node.list.length === 1) { + let collapsed = folderConcat(node.list[0]) // if the last collapsed thing is a folder merge into one. - if (collapsed.href === null || collapsed.href === undefined) { - node.nodeLabel = node.nodeLabel + "/" + collapsed.nodeLabel; - node.hasComments = collapsed.hasComments || node.hasComments; - node.list = collapsed.list; - } - return node; + if (collapsed.href === null || collapsed.href === undefined) { + node.nodeLabel = node.nodeLabel + '/' + collapsed.nodeLabel + node.hasComments = collapsed.hasComments || node.hasComments + node.list = collapsed.list } - node.list.map(x => folderConcat(x) ); - return node; -}; + return node + } + node.list.map(x => folderConcat(x)) + return node +} diff --git a/src/js/index.jsx b/src/js/index.jsx index d2e1ddc..b165d51 100644 --- a/src/js/index.jsx +++ b/src/js/index.jsx @@ -24,7 +24,7 @@ const renderTree = () => { return } - let { tree, count } = createFileTree(); + let { tree, count } = createFileTree() tree = folderConcat(tree) render(, rootElement) if (fileCount !== count) { From 7b09cff058500c268d9c0d6e9e2348ccec999b77 Mon Sep 17 00:00:00 2001 From: jake Rivett Date: Thu, 4 Oct 2018 11:33:20 +0100 Subject: [PATCH 5/6] Removed the top-level folder being part of the concat --- src/js/folderconcat/folder-concat.jsx | 2 +- src/js/style.css | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/js/folderconcat/folder-concat.jsx b/src/js/folderconcat/folder-concat.jsx index b0caa35..6a1b625 100644 --- a/src/js/folderconcat/folder-concat.jsx +++ b/src/js/folderconcat/folder-concat.jsx @@ -7,7 +7,7 @@ export const folderConcat = (node) => { if (node.list.length === 1) { let collapsed = folderConcat(node.list[0]) // if the last collapsed thing is a folder merge into one. - if (collapsed.href === null || collapsed.href === undefined) { + if (node.nodeLabel !== "/" && (collapsed.href === null || collapsed.href === undefined)) { node.nodeLabel = node.nodeLabel + '/' + collapsed.nodeLabel node.hasComments = collapsed.hasComments || node.hasComments node.list = collapsed.list diff --git a/src/js/style.css b/src/js/style.css index c0c17bc..39f129f 100644 --- a/src/js/style.css +++ b/src/js/style.css @@ -51,8 +51,8 @@ background: none; border: none; position: absolute; - top: 0; - right: 0; + top: 1px; + right: 10px; padding: 5px 2px; } @@ -62,7 +62,7 @@ background-size: cover; position: absolute; top: 6px; - right: 30px; + right: 40px; cursor: pointer; background-image: url(''); background-color: transparent; From b5cf32a7363272c71fed1c7f676fb7a7342748cb Mon Sep 17 00:00:00 2001 From: jake Rivett Date: Thu, 4 Oct 2018 11:33:26 +0100 Subject: [PATCH 6/6] Removed the top-level folder being part of the concat --- src/js/folderconcat/folder-concat.jsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/js/folderconcat/folder-concat.jsx b/src/js/folderconcat/folder-concat.jsx index 6a1b625..3d98ce8 100644 --- a/src/js/folderconcat/folder-concat.jsx +++ b/src/js/folderconcat/folder-concat.jsx @@ -7,7 +7,7 @@ export const folderConcat = (node) => { if (node.list.length === 1) { let collapsed = folderConcat(node.list[0]) // if the last collapsed thing is a folder merge into one. - if (node.nodeLabel !== "/" && (collapsed.href === null || collapsed.href === undefined)) { + if (node.nodeLabel !== '/' && (collapsed.href === null || collapsed.href === undefined)) { node.nodeLabel = node.nodeLabel + '/' + collapsed.nodeLabel node.hasComments = collapsed.hasComments || node.hasComments node.list = collapsed.list