| // Copyright (c) 2022, Compiler Explorer Authors |
| // All rights reserved. |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are met: |
| // |
| // * Redistributions of source code must retain the above copyright notice, |
| // this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above copyright |
| // notice, this list of conditions and the following disclaimer in the |
| // documentation and/or other materials provided with the distribution. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE |
| // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
| // POSSIBILITY OF SUCH DAMAGE. |
| |
| import $ from 'jquery'; |
| import _ from 'underscore'; |
| import * as monaco from 'monaco-editor'; |
| import {Container} from 'golden-layout'; |
| |
| import {MonacoPane} from './pane.js'; |
| import {MonacoPaneState} from './pane.interfaces.js'; |
| import {HaskellStgState} from './haskellstg-view.interfaces.js'; |
| |
| import {ga} from '../analytics.js'; |
| import {extendConfig} from '../monaco-config.js'; |
| import {Hub} from '../hub.js'; |
| |
| export class HaskellStg extends MonacoPane<monaco.editor.IStandaloneCodeEditor, HaskellStgState> { |
| constructor(hub: Hub, container: Container, state: HaskellStgState & MonacoPaneState) { |
| super(hub, container, state); |
| if (state.haskellStgOutput) { |
| this.showHaskellStgResults(state.haskellStgOutput); |
| } |
| } |
| |
| override getInitialHTML(): string { |
| return $('#haskellStg').html(); |
| } |
| |
| override createEditor(editorRoot: HTMLElement): monaco.editor.IStandaloneCodeEditor { |
| return monaco.editor.create( |
| editorRoot, |
| extendConfig({ |
| language: 'haskell', |
| readOnly: true, |
| glyphMargin: true, |
| lineNumbersMinChars: 3, |
| }), |
| ); |
| } |
| |
| override registerOpeningAnalyticsEvent(): void { |
| ga.proxy('send', { |
| hitType: 'event', |
| eventCategory: 'OpenViewPane', |
| eventAction: 'HaskellStg', |
| }); |
| } |
| |
| override getDefaultPaneName(): string { |
| return 'GHC STG Viewer'; |
| } |
| |
| override registerCallbacks(): void { |
| const throttleFunction = _.throttle(event => this.onDidChangeCursorSelection(event), 500); |
| this.editor.onDidChangeCursorSelection(event => throttleFunction(event)); |
| this.eventHub.emit('haskellStgViewOpened', this.compilerInfo.compilerId); |
| this.eventHub.emit('requestSettings'); |
| } |
| |
| override onCompileResult(compilerId: number, compiler: any, result: any): void { |
| if (this.compilerInfo.compilerId !== compilerId) return; |
| if (result.hasHaskellStgOutput) { |
| this.showHaskellStgResults(result.haskellStgOutput); |
| } else if (compiler.supportsHaskellStgView) { |
| this.showHaskellStgResults([{text: '<No output>'}]); |
| } |
| } |
| |
| override onCompiler(compilerId: number, compiler: any, options: any, editorId?: number, treeId?: number): void { |
| if (this.compilerInfo.compilerId === compilerId) { |
| this.compilerInfo.compilerName = compiler ? compiler.name : ''; |
| this.compilerInfo.editorId = editorId; |
| this.compilerInfo.treeId = treeId; |
| this.updateTitle(); |
| if (compiler && !compiler.supportsHaskellStgView) { |
| this.showHaskellStgResults([{text: '<GHC STG output is not supported for this compiler>'}]); |
| } |
| } |
| } |
| |
| showHaskellStgResults(result: Record<'text', string>[]): void { |
| this.editor.getModel()?.setValue(result.length ? _.pluck(result, 'text').join('\n') : '<No GHC STG generated>'); |
| |
| if (!this.isAwaitingInitialResults) { |
| if (this.selection) { |
| this.editor.setSelection(this.selection); |
| this.editor.revealLinesInCenter(this.selection.selectionStartLineNumber, this.selection.endLineNumber); |
| } |
| this.isAwaitingInitialResults = true; |
| } |
| } |
| |
| override close(): void { |
| this.eventHub.unsubscribe(); |
| this.eventHub.emit('haskellStgViewClosed', this.compilerInfo.compilerId); |
| this.editor.dispose(); |
| } |
| } |