From c7034ac1e3a7cf616868c9df224a24b5625b7fe4 Mon Sep 17 00:00:00 2001 From: Anatoliy Bazko Date: Tue, 14 Aug 2018 10:42:11 +0300 Subject: [PATCH] Restyling debug panel Signed-off-by: Anatoliy Bazko --- packages/core/src/browser/shell/tab-bars.ts | 2 +- packages/debug/src/browser/style/index.css | 100 +++++++----------- .../browser/view/debug-breakpoints-widget.ts | 26 +++-- .../browser/view/debug-stack-frames-widget.ts | 36 ++++--- .../src/browser/view/debug-threads-widget.ts | 29 +++-- .../browser/view/debug-variables-widget.tsx | 22 ++-- .../browser/view/debug-view-contribution.ts | 10 +- 7 files changed, 118 insertions(+), 107 deletions(-) diff --git a/packages/core/src/browser/shell/tab-bars.ts b/packages/core/src/browser/shell/tab-bars.ts index 6c144d5d63ea8..51994bec6a1ec 100644 --- a/packages/core/src/browser/shell/tab-bars.ts +++ b/packages/core/src/browser/shell/tab-bars.ts @@ -104,7 +104,7 @@ export class TabBarRenderer extends TabBar.Renderer { const iconSize = data.iconSize; let height: string | undefined; if (labelSize || iconSize) { - const labelHeight = labelSize ? labelSize.width : 0; + const labelHeight = labelSize ? (this.tabBar && this.tabBar.orientation === 'horizontal' ? labelSize.height : labelSize.width) : 0; const iconHeight = iconSize ? iconSize.height : 0; let paddingTop = data.paddingTop || 0; if (labelHeight > 0 && iconHeight > 0) { diff --git a/packages/debug/src/browser/style/index.css b/packages/debug/src/browser/style/index.css index 8498ba4ba1a07..dd7cccff97742 100644 --- a/packages/debug/src/browser/style/index.css +++ b/packages/debug/src/browser/style/index.css @@ -14,9 +14,7 @@ * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 ********************************************************************************/ -.theia-debug-panel { - display: flex; -} +.theia-debug-panel { } .theia-debug-panel:focus { outline: 0; @@ -24,95 +22,67 @@ } .theia-debug-target { - display: flex; -} - -.theia-debug-threads-container { - overflow: auto; + display: inline-table; color: var(--theia-ui-font-color1); background: var(--theia-layout-color0); font-size: var(--theia-ui-font-size1); - max-height: calc(100% - var(--theia-border-width)); - position: relative; - width: 300px; + width: 100%; } -.theia-debug-threads-container:focus .theia-mod-selected { +.theia-debug-entry { } + +.theia-debug-entry:focus .theia-mod-selected { background: var(--theia-accent-color3); } -.theia-debug-threads-container:not(:focus) .theia-mod-selected { +.theia-debug-entry:not(:focus) .theia-mod-selected { background: var(--theia-accent-color4); } -.theia-debug-thread { +.theia-debug-threads { + overflow: auto; + height: 180px; +} + +.theia-debug-thread-item { line-height: var(--theia-private-horizontal-tab-height); align-items: baseline; padding-left: 5px; - display: flex; } -.theia-debug-thread:hover { +.theia-debug-thread-item:hover { background: var(--theia-accent-color4); cursor: pointer; } -.theia-debug-frames-container { +.theia-debug-frames { overflow: auto; - color: var(--theia-ui-font-color1); - background: var(--theia-layout-color0); - font-size: var(--theia-ui-font-size1); - max-height: calc(100% - var(--theia-border-width)); - position: relative; - width: 400px; -} - -.theia-debug-frames-container:focus .theia-mod-selected { - background: var(--theia-accent-color3); -} - -.theia-debug-frames-container:not(:focus) .theia-mod-selected { - background: var(--theia-accent-color4); + height: 200px; } -.theia-debug-frame { +.theia-debug-frame-item { line-height: var(--theia-private-horizontal-tab-height); align-items: baseline; padding-left: 5px; - display: flex; } -.theia-debug-frame:hover { +.theia-debug-frame-item:hover { background: var(--theia-accent-color4); cursor: pointer; } -.theia-debug-breakpoints-container { +.theia-debug-breakpoints { overflow: auto; - color: var(--theia-ui-font-color1); - background: var(--theia-layout-color0); - font-size: var(--theia-ui-font-size1); - max-height: calc(100% - var(--theia-border-width)); - position: relative; - width: 400px; -} - -.theia-debug-breakpoints-container:focus .theia-mod-selected { - background: var(--theia-accent-color3); -} - -.theia-debug-breakpoints-container:not(:focus) .theia-mod-selected { - background: var(--theia-accent-color4); + height: 150px; } -.theia-debug-breakpoint { +.theia-debug-breakpoint-item { line-height: var(--theia-private-horizontal-tab-height); align-items: baseline; padding-left: 5px; - display: flex; } -.theia-debug-breakpoint:hover { +.theia-debug-breakpoint-item:hover { background: var(--theia-accent-color4); cursor: pointer; } @@ -121,24 +91,36 @@ height: 400px; } -.theia-debug-variables-container { - overflow: auto; - color: var(--theia-ui-font-color1); +.theia-debug-breakpoint-dialog .theia-debug-breakpoints { background: var(--theia-layout-color0); - font-size: var(--theia-ui-font-size1); - max-height: calc(100% - var(--theia-border-width)); - position: relative; - width: 400px; + height: 380px +} + +.theia-debug-variables { + overflow: hidden; + height: 200px; + white-space: nowrap; } .theia-debug-active-line { background-color: rgba(255, 120, 100, 0.5); } +.theia-debug-header { + background: var(--theia-layout-color3); + text-transform: uppercase; + font-size: var(--theia-ui-font-size0); + font-weight: 800; + border: 4px solid; + border-color: var(--theia-layout-color3); +} + .theia-debug-inactive-breakpoint { + transform: scale(0.5); background-image: url(); } .theia-debug-active-breakpoint { + transform: scale(0.5); background-image: url(); } diff --git a/packages/debug/src/browser/view/debug-breakpoints-widget.ts b/packages/debug/src/browser/view/debug-breakpoints-widget.ts index 8a90896be6a3e..3ab0f1005b854 100644 --- a/packages/debug/src/browser/view/debug-breakpoints-widget.ts +++ b/packages/debug/src/browser/view/debug-breakpoints-widget.ts @@ -25,6 +25,7 @@ import { injectable, inject, postConstruct } from "inversify"; import { BreakpointsManager } from "../breakpoint/breakpoint-manager"; import { ExtDebugProtocol } from "../../common/debug-common"; import { DebugUtils } from "../debug-utils"; +import { Disposable } from "@theia/core"; /** * Is it used to display breakpoints. @@ -41,16 +42,19 @@ export class DebugBreakpointsWidget extends VirtualWidget { super(); this.id = 'debug-breakpoints' + (debugSession ? `-${debugSession.sessionId}` : ''); - this.addClass(Styles.BREAKPOINTS_CONTAINER); + this.addClass('theia-debug-entry'); this.node.setAttribute("tabIndex", "0"); } @postConstruct() protected init() { - this.breakpointManager.onDidChangeBreakpoints(() => this.refreshBreakpoints()); + this.toDisposeOnDetach.push(this.breakpointManager.onDidChangeBreakpoints(() => this.refreshBreakpoints())); if (this.debugSession) { - this.debugSession.on('configurationDone', () => this.refreshBreakpoints()); + const configurationDoneListener = () => this.refreshBreakpoints(); + + this.debugSession.on('configurationDone', configurationDoneListener); + this.toDisposeOnDetach.push(Disposable.create(() => this.debugSession!.removeListener('configurationDone', configurationDoneListener))); } } @@ -68,24 +72,26 @@ export class DebugBreakpointsWidget extends VirtualWidget { } else { this.breakpoints = await this.breakpointManager.getAll(); } + this.breakpoints.sort(); + super.update(); } protected render(): h.Child { - const header = h.div({ className: "theia-header" }, "Breakpoints"); + const header = h.div({ className: "theia-debug-header" }, "Breakpoints"); const items: h.Child = []; for (const breakpoint of this.breakpoints) { const item = h.div({ id: DebugUtils.makeBreakpointId(breakpoint), - className: Styles.BREAKPOINT, + className: Styles.BREAKPOINT_ITEM, onclick: event => { const selected = this.node.getElementsByClassName(SELECTED_CLASS)[0]; if (selected) { - selected.className = Styles.BREAKPOINT; + selected.className = Styles.BREAKPOINT_ITEM; } - (event.target as HTMLDivElement).className = `${Styles.BREAKPOINT} ${SELECTED_CLASS}`; + (event.target as HTMLDivElement).className = `${Styles.BREAKPOINT_ITEM} ${SELECTED_CLASS}`; this.onDidClickBreakpointEmitter.fire(breakpoint); }, @@ -94,7 +100,7 @@ export class DebugBreakpointsWidget extends VirtualWidget { items.push(item); } - return [header, h.div(items)]; + return [header, h.div({ className: Styles.BREAKPOINTS }, items)]; } private toDisplayName(breakpoint: ExtDebugProtocol.AggregatedBreakpoint): string { @@ -165,7 +171,7 @@ export class BreakpointsDialog extends AbstractDialog { } namespace Styles { - export const BREAKPOINTS_CONTAINER = 'theia-debug-breakpoints-container'; - export const BREAKPOINT = 'theia-debug-breakpoint'; + export const BREAKPOINTS = 'theia-debug-breakpoints'; + export const BREAKPOINT_ITEM = 'theia-debug-breakpoint-item'; export const BREAKPOINT_DIALOG = 'theia-debug-breakpoint-dialog'; } diff --git a/packages/debug/src/browser/view/debug-stack-frames-widget.ts b/packages/debug/src/browser/view/debug-stack-frames-widget.ts index 4ae2fef35a594..eacbdf899a234 100644 --- a/packages/debug/src/browser/view/debug-stack-frames-widget.ts +++ b/packages/debug/src/browser/view/debug-stack-frames-widget.ts @@ -21,9 +21,10 @@ import { import { DebugSession } from "../debug-model"; import { h } from '@phosphor/virtualdom'; import { DebugProtocol } from 'vscode-debugprotocol'; -import { injectable, inject } from "inversify"; +import { injectable, inject, postConstruct } from "inversify"; import { DebugSelection } from "./debug-selection-service"; import { SourceOpener, DebugUtils } from "../debug-utils"; +import { Disposable } from "@theia/core"; /** * Is it used to display call stack. @@ -39,11 +40,22 @@ export class DebugStackFramesWidget extends VirtualWidget { super(); this.id = this.createId(); - this.addClass(Styles.FRAMES_CONTAINER); + this.addClass('theia-debug-entry'); this.node.setAttribute("tabIndex", "0"); - this.debugSession.on('stopped', event => this.onStoppedEvent(event)); - this.debugSession.on('continued', event => this.onContinuedEvent(event)); - this.debugSelection.onDidSelectThread(thread => this.onThreadSelected(thread)); + } + + @postConstruct() + protected init() { + this.toDisposeOnDetach.push(this.debugSelection.onDidSelectThread(thread => this.onThreadSelected(thread))); + + const stoppedEventListener = (event: DebugProtocol.StoppedEvent) => this.onStoppedEvent(event); + const continuedEventListener = (event: DebugProtocol.ContinuedEvent) => this.onContinuedEvent(event); + + this.debugSession.on('stopped', stoppedEventListener); + this.debugSession.on('continued', continuedEventListener); + + this.toDisposeOnDetach.push(Disposable.create(() => this.debugSession.removeListener('stopped', stoppedEventListener))); + this.toDisposeOnDetach.push(Disposable.create(() => this.debugSession.removeListener('continued', continuedEventListener))); } get frames(): DebugProtocol.StackFrame[] { @@ -56,12 +68,12 @@ export class DebugStackFramesWidget extends VirtualWidget { } protected render(): h.Child { - const header = h.div({ className: "theia-header" }, "Call stack"); + const header = h.div({ className: "theia-debug-header" }, "Call stack"); const items: h.Child = []; const selectedFrame = this.debugSelection.frame; for (const frame of this._frames) { - const className = Styles.FRAME + (DebugUtils.isEqual(selectedFrame, frame) ? ` ${SELECTED_CLASS}` : ''); + const className = Styles.FRAME_ITEM + (DebugUtils.isEqual(selectedFrame, frame) ? ` ${SELECTED_CLASS}` : ''); const id = this.createId(frame); const item = @@ -76,7 +88,7 @@ export class DebugStackFramesWidget extends VirtualWidget { items.push(item); } - return [header, h.div(items)]; + return [header, h.div({ className: Styles.FRAMES }, items)]; } protected onThreadSelected(thread: DebugProtocol.Thread | undefined) { @@ -93,14 +105,14 @@ export class DebugStackFramesWidget extends VirtualWidget { if (currentFrame) { const element = document.getElementById(this.createId(currentFrame)); if (element) { - element.className = Styles.FRAME; + element.className = Styles.FRAME_ITEM; } } if (newFrame) { const element = document.getElementById(this.createId(newFrame)); if (element) { - element.className = `${Styles.FRAME} ${SELECTED_CLASS}`; + element.className = `${Styles.FRAME_ITEM} ${SELECTED_CLASS}`; } } @@ -151,6 +163,6 @@ export class DebugStackFramesWidget extends VirtualWidget { } namespace Styles { - export const FRAMES_CONTAINER = 'theia-debug-frames-container'; - export const FRAME = 'theia-debug-frame'; + export const FRAMES = 'theia-debug-frames'; + export const FRAME_ITEM = 'theia-debug-frame-item'; } diff --git a/packages/debug/src/browser/view/debug-threads-widget.ts b/packages/debug/src/browser/view/debug-threads-widget.ts index 19739caae29e6..48a2252f5fe1c 100644 --- a/packages/debug/src/browser/view/debug-threads-widget.ts +++ b/packages/debug/src/browser/view/debug-threads-widget.ts @@ -22,6 +22,7 @@ import { injectable, inject, postConstruct } from "inversify"; import { DEBUG_SESSION_THREAD_CONTEXT_MENU } from "../debug-command"; import { DebugSelection } from "./debug-selection-service"; import { DebugUtils } from "../debug-utils"; +import { Disposable } from "@theia/core"; /** * Is it used to display list of threads. @@ -37,15 +38,21 @@ export class DebugThreadsWidget extends VirtualWidget { super(); this.id = this.createId(); - this.addClass(Styles.THREADS_CONTAINER); + this.addClass('theia-debug-entry'); this.node.setAttribute("tabIndex", "0"); - - this.debugSession.on('thread', event => this.onThreadEvent(event)); - this.debugSession.on('connected', () => this.updateThreads()); } @postConstruct() protected init() { + const threadEventListener = (event: DebugProtocol.ThreadEvent) => this.onThreadEvent(event); + const connectedEventListener = () => this.updateThreads(); + + this.debugSession.on('thread', threadEventListener); + this.debugSession.on('connected', connectedEventListener); + + this.toDisposeOnDetach.push(Disposable.create(() => this.debugSession.removeListener('thread', threadEventListener))); + this.toDisposeOnDetach.push(Disposable.create(() => this.debugSession.removeListener('connected', connectedEventListener))); + if (this.debugSession.state.isConnected) { this.updateThreads(); } @@ -61,11 +68,11 @@ export class DebugThreadsWidget extends VirtualWidget { } protected render(): h.Child { - const header = h.div({ className: "theia-header" }, "Threads"); + const header = h.div({ className: "theia-debug-header" }, "Threads"); const items: h.Child = []; for (const thread of this.threads) { - const className = Styles.THREAD + (DebugUtils.isEqual(this.debugSelection.thread, thread) ? ` ${SELECTED_CLASS}` : ''); + const className = Styles.THREAD_ITEM + (DebugUtils.isEqual(this.debugSelection.thread, thread) ? ` ${SELECTED_CLASS}` : ''); const id = this.createId(thread); const item = @@ -82,7 +89,7 @@ export class DebugThreadsWidget extends VirtualWidget { items.push(item); } - return [header, h.div(items)]; + return [header, h.div({ className: Styles.THREADS }, items)]; } protected selectThread(newThread: DebugProtocol.Thread | undefined) { @@ -95,14 +102,14 @@ export class DebugThreadsWidget extends VirtualWidget { if (currentThread) { const element = document.getElementById(this.createId(currentThread)); if (element) { - element.className = Styles.THREAD; + element.className = Styles.THREAD_ITEM; } } if (newThread) { const element = document.getElementById(this.createId(newThread)); if (element) { - element.className = `${Styles.THREAD} ${SELECTED_CLASS}`; + element.className = `${Styles.THREAD_ITEM} ${SELECTED_CLASS}`; } } @@ -133,6 +140,6 @@ export class DebugThreadsWidget extends VirtualWidget { } namespace Styles { - export const THREADS_CONTAINER = 'theia-debug-threads-container'; - export const THREAD = 'theia-debug-thread'; + export const THREADS = 'theia-debug-threads'; + export const THREAD_ITEM = 'theia-debug-thread-item'; } diff --git a/packages/debug/src/browser/view/debug-variables-widget.tsx b/packages/debug/src/browser/view/debug-variables-widget.tsx index 23aa099918a99..31412c14d2434 100644 --- a/packages/debug/src/browser/view/debug-variables-widget.tsx +++ b/packages/debug/src/browser/view/debug-variables-widget.tsx @@ -35,6 +35,7 @@ import { DebugSession } from "../debug-model"; import { DebugSelection } from "../view/debug-selection-service"; import { ExtDebugProtocol } from "../../common/debug-common"; import * as React from "react"; +import { Disposable } from "@theia/core"; /** * Is it used to display variables. @@ -53,9 +54,18 @@ export class DebugVariablesWidget extends TreeWidget { this.id = `debug-variables-${debugSession.sessionId}`; this.title.label = 'Variables'; - this.addClass(Styles.VARIABLES_CONTAINER); - this.debugSession.on('variableUpdated', (event: ExtDebugProtocol.VariableUpdatedEvent) => this.onVariableUpdated(event)); - this.debugSelection.onDidSelectFrame(frame => this.onFrameSelected(frame)); + this.addClass('theia-debug-entry'); + } + + @postConstruct() + protected init() { + super.init(); + + this.toDisposeOnDetach.push(Disposable.create(() => this.debugSession.removeListener('variableUpdated', variableUpdateListener))); + + const variableUpdateListener = (event: ExtDebugProtocol.VariableUpdatedEvent) => this.onVariableUpdated(event); + this.debugSession.on('variableUpdated', variableUpdateListener); + this.toDisposeOnDetach.push(this.debugSelection.onDidSelectFrame(frame => this.onFrameSelected(frame))); } protected onFrameSelected(frame: DebugProtocol.StackFrame | undefined) { @@ -73,7 +83,7 @@ export class DebugVariablesWidget extends TreeWidget { } protected renderTree(model: TreeModel): React.ReactNode { - return
Variables
{super.renderTree(model)}
; + return
Variables
{super.renderTree(model)}
; } protected renderCaption(node: TreeNode, props: NodeProps): React.ReactNode { @@ -249,7 +259,3 @@ namespace FrameNode { function createId(sessionId: string, itemId: string | number, parentId?: string | number): string { return `debug-variables-${sessionId}` + (parentId && `-${parentId}`) + `-${itemId}`; } - -namespace Styles { - export const VARIABLES_CONTAINER = 'theia-debug-variables-container'; -} diff --git a/packages/debug/src/browser/view/debug-view-contribution.ts b/packages/debug/src/browser/view/debug-view-contribution.ts index 5521a76e0bca5..8b286d5dea259 100644 --- a/packages/debug/src/browser/view/debug-view-contribution.ts +++ b/packages/debug/src/browser/view/debug-view-contribution.ts @@ -21,7 +21,6 @@ import { TabBarRenderer, TabBarRendererFactory, SideTabBar, - LEFT_RIGHT_AREA_CLASS, Widget, Message, VirtualWidget, @@ -149,7 +148,7 @@ export class DebugWidget extends Panel { private createTabBar(): SideTabBar { const renderer = this.tabBarRendererFactory(); const tabBar = new SideTabBar({ - orientation: 'vertical', + orientation: 'horizontal', insertBehavior: 'none', removeBehavior: 'select-previous-tab', allowDeselect: false, @@ -162,8 +161,7 @@ export class DebugWidget extends Panel { }); renderer.tabBar = tabBar; renderer.contextMenuPath = DEBUG_SESSION_CONTEXT_MENU; - tabBar.addClass('theia-app-left'); - tabBar.addClass(LEFT_RIGHT_AREA_CLASS); + tabBar.addClass('theia-app-centers'); tabBar.currentChanged.connect(this.onCurrentTabChanged, this); tabBar.tabCloseRequested.connect(this.onTabCloseRequested, this); return tabBar; @@ -192,7 +190,7 @@ export class DebugTargetWidget extends VirtualWidget { this.title.closable = true; this.addClass(Styles.DEBUG_TARGET); this.sessionId = debugSession.sessionId; - this.widgets = [this.breakpoints, this.threads, this.frames, this.variables]; + this.widgets = [this.variables, this.threads, this.frames, this.breakpoints]; } protected onUpdateRequest(msg: Message): void { @@ -223,7 +221,7 @@ export class DebugViewContribution extends AbstractViewContribution widgetId: DEBUG_FACTORY_ID, widgetName: 'Debug', defaultWidgetOptions: { - area: 'bottom', + area: 'left', rank: 500 }, toggleCommandId: 'debug.view.toggle',