2014-04-06 21:46:33 +04:00
|
|
|
React = require 'react'
|
2014-04-06 23:56:38 +04:00
|
|
|
{extend} = require 'underscore-plus'
|
2014-03-28 05:02:24 +04:00
|
|
|
EditorComponent = require '../src/editor-component'
|
|
|
|
|
|
|
|
describe "EditorComponent", ->
|
2014-04-02 20:59:57 +04:00
|
|
|
[editor, component, node, lineHeightInPixels, charWidth] = []
|
2014-03-28 05:02:24 +04:00
|
|
|
|
|
|
|
beforeEach ->
|
2014-04-02 20:59:57 +04:00
|
|
|
waitsForPromise ->
|
|
|
|
atom.packages.activatePackage('language-javascript')
|
2014-04-02 03:06:59 +04:00
|
|
|
|
2014-04-02 20:59:57 +04:00
|
|
|
runs ->
|
|
|
|
spyOn(window, 'requestAnimationFrame').andCallFake (fn) -> fn()
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-02 20:59:57 +04:00
|
|
|
editor = atom.project.openSync('sample.js')
|
|
|
|
container = document.querySelector('#jasmine-content')
|
|
|
|
component = React.renderComponent(EditorComponent({editor}), container)
|
|
|
|
component.setLineHeight(1.3)
|
|
|
|
component.setFontSize(20)
|
|
|
|
{lineHeightInPixels, charWidth} = component.measureLineDimensions()
|
|
|
|
node = component.getDOMNode()
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-04 22:58:12 +04:00
|
|
|
describe "scrolling", ->
|
|
|
|
it "renders only the currently-visible lines", ->
|
|
|
|
node.style.height = 4.5 * lineHeightInPixels + 'px'
|
|
|
|
component.updateAllDimensions()
|
|
|
|
|
|
|
|
lines = node.querySelectorAll('.line')
|
|
|
|
expect(lines.length).toBe 5
|
|
|
|
expect(lines[0].textContent).toBe editor.lineForScreenRow(0).text
|
|
|
|
expect(lines[4].textContent).toBe editor.lineForScreenRow(4).text
|
|
|
|
|
|
|
|
node.querySelector('.vertical-scrollbar').scrollTop = 2.5 * lineHeightInPixels
|
|
|
|
component.onVerticalScroll()
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-04 22:58:12 +04:00
|
|
|
expect(node.querySelector('.scrollable-content').style['-webkit-transform']).toBe "translateY(#{-2.5 * lineHeightInPixels}px)"
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-04 22:58:12 +04:00
|
|
|
lines = node.querySelectorAll('.line')
|
|
|
|
expect(lines.length).toBe 5
|
|
|
|
expect(lines[0].textContent).toBe editor.lineForScreenRow(2).text
|
|
|
|
expect(lines[4].textContent).toBe editor.lineForScreenRow(6).text
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-04 22:58:12 +04:00
|
|
|
spacers = node.querySelectorAll('.spacer')
|
|
|
|
expect(spacers[0].offsetHeight).toBe 2 * lineHeightInPixels
|
|
|
|
expect(spacers[1].offsetHeight).toBe (editor.getScreenLineCount() - 7) * lineHeightInPixels
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-04 22:58:12 +04:00
|
|
|
it "updates the scroll bar when the scrollTop is changed in the model", ->
|
|
|
|
node.style.height = 4.5 * lineHeightInPixels + 'px'
|
|
|
|
component.updateAllDimensions()
|
|
|
|
|
|
|
|
scrollbarNode = node.querySelector('.vertical-scrollbar')
|
|
|
|
expect(scrollbarNode.scrollTop).toBe 0
|
2014-03-28 05:02:24 +04:00
|
|
|
|
2014-04-04 22:58:12 +04:00
|
|
|
editor.setScrollTop(10)
|
|
|
|
expect(scrollbarNode.scrollTop).toBe 10
|
2014-04-02 01:06:38 +04:00
|
|
|
|
2014-04-04 03:04:19 +04:00
|
|
|
describe "cursor rendering", ->
|
|
|
|
it "renders the currently visible cursors", ->
|
|
|
|
cursor1 = editor.getCursor()
|
|
|
|
cursor1.setScreenPosition([0, 5])
|
|
|
|
|
|
|
|
node.style.height = 4.5 * lineHeightInPixels + 'px'
|
|
|
|
component.updateAllDimensions()
|
|
|
|
|
|
|
|
cursorNodes = node.querySelectorAll('.cursor')
|
|
|
|
expect(cursorNodes.length).toBe 1
|
|
|
|
expect(cursorNodes[0].offsetHeight).toBe lineHeightInPixels
|
|
|
|
expect(cursorNodes[0].offsetWidth).toBe charWidth
|
|
|
|
expect(cursorNodes[0].offsetTop).toBe 0
|
|
|
|
expect(cursorNodes[0].offsetLeft).toBe 5 * charWidth
|
|
|
|
|
|
|
|
cursor2 = editor.addCursorAtScreenPosition([6, 11])
|
|
|
|
cursor3 = editor.addCursorAtScreenPosition([4, 10])
|
|
|
|
|
|
|
|
cursorNodes = node.querySelectorAll('.cursor')
|
|
|
|
expect(cursorNodes.length).toBe 2
|
|
|
|
expect(cursorNodes[0].offsetTop).toBe 0
|
|
|
|
expect(cursorNodes[0].offsetLeft).toBe 5 * charWidth
|
|
|
|
expect(cursorNodes[1].offsetTop).toBe 4 * lineHeightInPixels
|
|
|
|
expect(cursorNodes[1].offsetLeft).toBe 10 * charWidth
|
|
|
|
|
|
|
|
node.querySelector('.vertical-scrollbar').scrollTop = 2.5 * lineHeightInPixels
|
|
|
|
component.onVerticalScroll()
|
|
|
|
|
|
|
|
cursorNodes = node.querySelectorAll('.cursor')
|
|
|
|
expect(cursorNodes.length).toBe 2
|
|
|
|
expect(cursorNodes[0].offsetTop).toBe 6 * lineHeightInPixels
|
|
|
|
expect(cursorNodes[0].offsetLeft).toBe 11 * charWidth
|
|
|
|
expect(cursorNodes[1].offsetTop).toBe 4 * lineHeightInPixels
|
|
|
|
expect(cursorNodes[1].offsetLeft).toBe 10 * charWidth
|
|
|
|
|
|
|
|
cursor3.destroy()
|
|
|
|
cursorNodes = node.querySelectorAll('.cursor')
|
|
|
|
expect(cursorNodes.length).toBe 1
|
|
|
|
expect(cursorNodes[0].offsetTop).toBe 6 * lineHeightInPixels
|
|
|
|
expect(cursorNodes[0].offsetLeft).toBe 11 * charWidth
|
|
|
|
|
|
|
|
it "accounts for character widths when positioning cursors", ->
|
|
|
|
atom.config.set('editor.fontFamily', 'sans-serif')
|
|
|
|
editor.setCursorScreenPosition([0, 16])
|
|
|
|
|
|
|
|
cursor = node.querySelector('.cursor')
|
|
|
|
cursorRect = cursor.getBoundingClientRect()
|
|
|
|
|
|
|
|
cursorLocationTextNode = node.querySelector('.storage.type.function.js').firstChild.firstChild
|
|
|
|
range = document.createRange()
|
|
|
|
range.setStart(cursorLocationTextNode, 0)
|
|
|
|
range.setEnd(cursorLocationTextNode, 1)
|
|
|
|
rangeRect = range.getBoundingClientRect()
|
|
|
|
|
|
|
|
expect(cursorRect.left).toBe rangeRect.left
|
|
|
|
expect(cursorRect.width).toBe rangeRect.width
|
|
|
|
|
|
|
|
describe "selection rendering", ->
|
|
|
|
it "renders 1 region for 1-line selections", ->
|
|
|
|
# 1-line selection
|
|
|
|
editor.setSelectedScreenRange([[1, 6], [1, 10]])
|
|
|
|
regions = node.querySelectorAll('.selection .region')
|
|
|
|
expect(regions.length).toBe 1
|
|
|
|
regionRect = regions[0].getBoundingClientRect()
|
|
|
|
expect(regionRect.top).toBe 1 * lineHeightInPixels
|
|
|
|
expect(regionRect.height).toBe 1 * lineHeightInPixels
|
|
|
|
expect(regionRect.left).toBe 6 * charWidth
|
|
|
|
expect(regionRect.width).toBe 4 * charWidth
|
|
|
|
|
|
|
|
it "renders 2 regions for 2-line selections", ->
|
|
|
|
editor.setSelectedScreenRange([[1, 6], [2, 10]])
|
|
|
|
regions = node.querySelectorAll('.selection .region')
|
|
|
|
expect(regions.length).toBe 2
|
|
|
|
|
|
|
|
region1Rect = regions[0].getBoundingClientRect()
|
|
|
|
expect(region1Rect.top).toBe 1 * lineHeightInPixels
|
|
|
|
expect(region1Rect.height).toBe 1 * lineHeightInPixels
|
|
|
|
expect(region1Rect.left).toBe 6 * charWidth
|
|
|
|
expect(region1Rect.right).toBe node.clientWidth
|
|
|
|
|
|
|
|
region2Rect = regions[1].getBoundingClientRect()
|
|
|
|
expect(region2Rect.top).toBe 2 * lineHeightInPixels
|
|
|
|
expect(region2Rect.height).toBe 1 * lineHeightInPixels
|
|
|
|
expect(region2Rect.left).toBe 0
|
|
|
|
expect(region2Rect.width).toBe 10 * charWidth
|
|
|
|
|
|
|
|
it "renders 3 regions for selections with more than 2 lines", ->
|
|
|
|
editor.setSelectedScreenRange([[1, 6], [5, 10]])
|
|
|
|
regions = node.querySelectorAll('.selection .region')
|
|
|
|
expect(regions.length).toBe 3
|
|
|
|
|
|
|
|
region1Rect = regions[0].getBoundingClientRect()
|
|
|
|
expect(region1Rect.top).toBe 1 * lineHeightInPixels
|
|
|
|
expect(region1Rect.height).toBe 1 * lineHeightInPixels
|
|
|
|
expect(region1Rect.left).toBe 6 * charWidth
|
|
|
|
expect(region1Rect.right).toBe node.clientWidth
|
|
|
|
|
|
|
|
region2Rect = regions[1].getBoundingClientRect()
|
|
|
|
expect(region2Rect.top).toBe 2 * lineHeightInPixels
|
|
|
|
expect(region2Rect.height).toBe 3 * lineHeightInPixels
|
|
|
|
expect(region2Rect.left).toBe 0
|
|
|
|
expect(region2Rect.right).toBe node.clientWidth
|
|
|
|
|
|
|
|
region3Rect = regions[2].getBoundingClientRect()
|
|
|
|
expect(region3Rect.top).toBe 5 * lineHeightInPixels
|
|
|
|
expect(region3Rect.height).toBe 1 * lineHeightInPixels
|
|
|
|
expect(region3Rect.left).toBe 0
|
|
|
|
expect(region3Rect.width).toBe 10 * charWidth
|
2014-04-02 01:06:38 +04:00
|
|
|
|
2014-04-06 23:34:50 +04:00
|
|
|
describe "mouse interactions", ->
|
2014-04-06 23:56:38 +04:00
|
|
|
describe "when a non-folded line is single-clicked", ->
|
|
|
|
describe "when no modifier keys are held down", ->
|
|
|
|
it "moves the cursor to the nearest row and column", ->
|
|
|
|
node.style.height = 4.5 * lineHeightInPixels + 'px'
|
|
|
|
component.updateAllDimensions()
|
|
|
|
editor.setScrollTop(3.5 * lineHeightInPixels)
|
|
|
|
|
|
|
|
component.onMouseDown(clientCoordinatesForScreenPosition([4, 8]))
|
|
|
|
expect(editor.getCursorScreenPosition()).toEqual [4, 8]
|
|
|
|
|
|
|
|
describe "when the shift key is held down", ->
|
|
|
|
it "selects to the nearest row and column", ->
|
|
|
|
editor.setCursorScreenPosition([3, 4])
|
|
|
|
event = extend(clientCoordinatesForScreenPosition([5, 6]), shiftKey: true)
|
|
|
|
component.onMouseDown(event)
|
|
|
|
expect(editor.getSelectedScreenRange()).toEqual [[3, 4], [5, 6]]
|
|
|
|
|
2014-04-06 23:34:50 +04:00
|
|
|
clientCoordinatesForScreenPosition = (screenPosition) ->
|
|
|
|
positionOffset = editor.pixelPositionForScreenPosition(screenPosition)
|
|
|
|
editorClientRect = node.getBoundingClientRect()
|
|
|
|
clientX = editorClientRect.left + positionOffset.left
|
|
|
|
clientY = editorClientRect.top + positionOffset.top - editor.getScrollTop()
|
|
|
|
{clientX, clientY}
|
|
|
|
|
2014-04-02 03:22:24 +04:00
|
|
|
it "transfers focus to the hidden input", ->
|
|
|
|
expect(document.activeElement).toBe document.body
|
|
|
|
node.focus()
|
|
|
|
expect(document.activeElement).toBe node.querySelector('.hidden-input')
|