feat(drag-threads): Move threads to folders/labels from thread list

Summary:
Drag threads to the folders / labels in the sidebar
WIP

Drag and drop is styled!

Test Plan: Tests WIP

Reviewers: evan

Reviewed By: evan

Differential Revision: https://phab.nylas.com/D1785
This commit is contained in:
Ben Gotow 2015-07-23 11:10:51 -07:00
parent 4be05ff754
commit a559e3f89f
14 changed files with 236 additions and 80 deletions

View file

@ -199,8 +199,6 @@ module.exports = (grunt) ->
sidebarPath = path.join(docsOutputDir, '_sidebar.json')
grunt.file.write(sidebarPath, sidebarJson)
# Prepare to render by loading handlebars partials
templatesPath = path.resolve(__dirname, '..', '..', 'docs-templates')
@ -214,40 +212,6 @@ module.exports = (grunt) ->
for classname, val of apiJSON.classes
knownClassnames[classname.toLowerCase()] = val
expandTypeReferences = (val) ->
refRegex = /{([\w.]*)}/g
while (match = refRegex.exec(val)) isnt null
term = match[1].toLowerCase()
label = match[1]
url = false
if term in standardClasses
url = standardClassURLRoot+term
else if thirdPartyClasses[term]
url = thirdPartyClasses[term]
else if knownClassnames[term]
url = relativePathForClass(term)
else
console.warn("Cannot find class named #{term}")
if url
val = val.replace(match[0], "<a href='#{url}'>#{label}</a>")
val
expandFuncReferences = (val) ->
refRegex = /{([\w]*)?::([\w]*)}/g
while (match = refRegex.exec(val)) isnt null
[text, a, b] = match
url = false
if a and b
url = "#{relativePathForClass(a)}##{b}"
label = "#{a}::#{b}"
else
url = "##{b}"
label = "#{b}"
if url
val = val.replace(text, "<a href='#{url}'>#{label}</a>")
val
# Render Class Pages
classTemplatePath = path.join(templatesPath, 'class.html')
@ -256,8 +220,9 @@ module.exports = (grunt) ->
for {name, documentation, section} in classes
# Recursively process `description` and `type` fields to process markdown,
# expand references to types, functions and other files.
processFields(documentation, ['description'], [marked.noMeta, expandTypeReferences, expandFuncReferences])
processFields(documentation, ['type'], [expandTypeReferences])
processFields(documentation, ['description'], [marked.noMeta])
processFields(documentation, ['type'], [])
result = classTemplate({name, documentation, section})
console.log(outputPathFor(relativePathForClass(name)))
grunt.file.write(outputPathFor(relativePathForClass(name)), result)

View file

@ -7,6 +7,7 @@
module.exports =
# Models
Menu: require '../src/components/menu'
DropZone: require '../src/components/drop-zone'
Spinner: require '../src/components/spinner'
Popover: require '../src/components/popover'
Flexbox: require '../src/components/flexbox'

View file

@ -45,6 +45,7 @@ Exports =
# Utils
Utils: Utils
DOMUtils: require '../src/dom-utils'
CanvasUtils: require '../src/canvas-utils'
RegExpUtils: require '../src/regexp-utils'
MessageUtils: require '../src/flux/models/message-utils'

View file

@ -1,7 +1,15 @@
React = require 'react'
classNames = require 'classnames'
{Actions, Utils, UnreadCountStore, WorkspaceStore} = require 'nylas-exports'
{RetinaImg} = require 'nylas-component-kit'
{Actions,
Utils,
UnreadCountStore,
WorkspaceStore,
NamespaceStore,
FocusedCategoryStore,
ChangeLabelsTask,
ChangeFolderTask,
CategoryStore} = require 'nylas-exports'
{RetinaImg, DropZone} = require 'nylas-component-kit'
class AccountSidebarCategoryItem extends React.Component
@displayName: 'AccountSidebarCategoryItem'
@ -19,9 +27,8 @@ class AccountSidebarCategoryItem extends React.Component
_onUnreadCountChange: =>
@setState unreadCount: UnreadCountStore.count()
shouldComponentUpdate: (nextProps) =>
@props?.item.name isnt nextProps.item.name or
@props?.select isnt nextProps.select
shouldComponentUpdate: (nextProps, nextState) =>
!Utils.isEqualReact(@props, nextProps) or !Utils.isEqualReact(@state, nextState)
render: =>
unread = []
@ -31,12 +38,47 @@ class AccountSidebarCategoryItem extends React.Component
containerClass = classNames
'item': true
'selected': @props.select
'dropping': @state.isDropping
<div className={containerClass} onClick={@_onClick} id={@props.item.id}>
<DropZone className={containerClass}
onClick={@_onClick}
id={@props.item.id}
shouldAcceptDrop={@_shouldAcceptDrop}
onDragStateChange={ ({isDropping}) => @setState({isDropping}) }
onDrop={@_onDrop}>
{unread}
<RetinaImg name={"#{@props.item.name}.png"} fallback={'folder.png'} mode={RetinaImg.Mode.ContentIsMask} />
<span className="name"> {@props.item.displayName}</span>
</div>
</DropZone>
_shouldAcceptDrop: (e) =>
return false if @props.item.name in CategoryStore.LockedCategoryNames
return false if @props.item.name is FocusedCategoryStore.categoryName()
'nylas-thread-ids' in e.dataTransfer.types
_onDrop: (e) =>
jsonString = e.dataTransfer.getData('nylas-thread-ids')
try
ids = JSON.parse(jsonString)
catch err
console.error("AccountSidebarCategoryItem onDrop: JSON parse #{err}")
return unless ids
if NamespaceStore.current().usesLabels()
currentLabel = FocusedCategoryStore.category()
if currentLabel and not (currentLabel in CategoryStore.LockedCategoryNames)
labelsToRemove = [currentLabel]
task = new ChangeLabelsTask
threadIds: ids,
labelsToAdd: [@props.item],
labelsToRemove: labelsToRemove
else
task = new ChangeFolderTask
folderOrId: @props.item,
threadIds: ids
Actions.queueTask(task)
_onClick: (event) =>
event.preventDefault()

View file

@ -49,6 +49,11 @@
color: @source-list-active-color;
img.content-mask { background-color: @source-list-active-color; }
}
&.dropping {
background-color: lighten(@source-list-bg, 20%);
color: @source-list-active-color;
img.content-mask { background-color: @source-list-active-color; }
}
&:hover {
cursor: default;

View file

@ -14,6 +14,7 @@ _ = require 'underscore'
InjectedComponentSet,
InjectedComponent,
ScrollRegion,
DropZone,
RetinaImg} = require 'nylas-component-kit'
FileUpload = require './file-upload'
@ -159,7 +160,10 @@ class ComposerView extends React.Component
"composer-outer-wrap #{@props.className ? ""}"
_renderComposer: =>
<div className="composer-inner-wrap" onDragEnter={@_onDragEnter} onDragLeave={@_onDragLeave} onDrop={@_onDrop}>
<DropZone className="composer-inner-wrap"
shouldAcceptDrop={@_shouldAcceptDrop}
onDragStateChange={ ({isDropping}) => @setState({isDropping}) }
onDrop={@_onDrop}>
<div className="composer-cover" style={display: if @state.isSending then 'block' else 'none'}></div>
<div className="composer-drop-cover" style={display: if @state.isDropping then 'block' else 'none'}>
<div className="centered">
@ -207,7 +211,7 @@ class ComposerView extends React.Component
<div className="composer-action-bar-wrap">
{@_renderActionsRegion()}
</div>
</div>
</DropZone>
_renderFields: =>
# Note: We need to physically add and remove these elements, not just hide them.
@ -470,7 +474,7 @@ class ComposerView extends React.Component
else if @isForwardedMessage() then return true
else return false
_shouldAcceptDrop: (event) ->
_shouldAcceptDrop: (event) =>
# Ensure that you can't pick up a file and drop it on the same draft
existingFilePaths = @state.files.map (f) ->
FileUploadStore.linkedUpload(f)?.filePath
@ -484,7 +488,7 @@ class ComposerView extends React.Component
return hasNativeFile or hasNonNativeFilePath
_nonNativeFilePathForDrop: (event) ->
_nonNativeFilePathForDrop: (event) =>
if "text/nylas-file-url" in event.dataTransfer.types
downloadURL = event.dataTransfer.getData("text/nylas-file-url")
downloadFilePath = downloadURL.split('file://')[1]
@ -500,30 +504,7 @@ class ComposerView extends React.Component
return null
# We maintain a "dragCounter" because dragEnter and dragLeave events *stack*
# when the user moves the item in and out of DOM elements inside of our container.
# It's really awful and everyone hates it.
#
# Alternative solution *maybe* is to set pointer-events:none; during drag.
_onDragEnter: (e) =>
return unless @_shouldAcceptDrop(e)
@_dragCounter ?= 0
@_dragCounter += 1
if @_dragCounter is 1
@setState(isDropping: true)
_onDragLeave: (e) =>
return unless @_shouldAcceptDrop(e)
@_dragCounter -= 1
if @_dragCounter is 0
@setState(isDropping: false)
_onDrop: (e) =>
return unless @_shouldAcceptDrop(e)
@setState(isDropping: false)
@_dragCounter = 0
# Accept drops of real files from other applications
for file in e.dataTransfer.files
Actions.attachFilePath({path: file.path, messageLocalId: @props.localId})
@ -532,8 +513,6 @@ class ComposerView extends React.Component
if (uri = @_nonNativeFilePathForDrop(e))
Actions.attachFilePath({path: uri, messageLocalId: @props.localId})
return
_onFilePaste: (path) =>
Actions.attachFilePath({path: path, messageLocalId: @props.localId})

View file

@ -5,6 +5,7 @@ classNames = require 'classnames'
{timestamp, subject} = require './formatting-utils'
{Actions,
Utils,
CanvasUtils,
Thread,
WorkspaceStore,
NamespaceStore,
@ -152,6 +153,7 @@ class ThreadList extends React.Component
@itemPropsProvider = (item) ->
className: classNames
'unread': item.unread
'data-thread-id': item.id
componentDidMount: =>
window.addEventListener('resize', @_onResize, true)
@ -171,6 +173,9 @@ class ThreadList extends React.Component
className="thread-list"
scrollTooltipComponent={ThreadListScrollTooltip}
emptyComponent={EmptyState}
onDragStart={@_onDragStart}
onDragEnd={@_onDragEnd}
draggable="true"
collection="thread" />
else if @state.style is 'narrow'
<MultiselectList
@ -186,6 +191,32 @@ class ThreadList extends React.Component
else
<div></div>
_threadIdAtPoint: (x, y) ->
item = document.elementFromPoint(event.clientX, event.clientY).closest('.list-item')
return null unless item
return item.dataset.threadId
_onDragStart: (event) =>
itemThreadId = @_threadIdAtPoint(event.clientX, event.clientY)
unless itemThreadId
event.preventDefault()
return
if itemThreadId in ThreadListStore.view().selection.ids()
dragThreadIds = ThreadListStore.view().selection.ids()
else
dragThreadIds = [itemThreadId]
event.dataTransfer.effectAllowed = "move"
event.dataTransfer.dragEffect = "move"
canvas = CanvasUtils.canvasWithThreadDragImage(dragThreadIds.length)
event.dataTransfer.setDragImage(canvas, 10, 10)
event.dataTransfer.setData('nylas-thread-ids', JSON.stringify(dragThreadIds))
return
_onDragEnd: (event) =>
_onResize: (event) =>
current = @state.style
desired = if React.findDOMNode(@).offsetWidth < 540 then 'narrow' else 'wide'

73
src/canvas-utils.coffee Normal file
View file

@ -0,0 +1,73 @@
ThreadDragImage = document.createElement("img")
ThreadDragImage.src = """"""
DragCanvas = document.createElement("canvas")
document.body.appendChild(DragCanvas)
CanvasUtils =
roundRect: (ctx, x, y, width, height, radius = 5, fill, stroke = true) ->
ctx.beginPath()
ctx.moveTo(x + radius, y)
ctx.lineTo(x + width - radius, y)
ctx.quadraticCurveTo(x + width, y, x + width, y + radius)
ctx.lineTo(x + width, y + height - radius)
ctx.quadraticCurveTo(x + width, y + height, x + width - radius, y + height)
ctx.lineTo(x + radius, y + height)
ctx.quadraticCurveTo(x, y + height, x, y + height - radius)
ctx.lineTo(x, y + radius)
ctx.quadraticCurveTo(x, y, x + radius, y)
ctx.closePath()
ctx.stroke() if stroke
ctx.fill() if fill
canvasWithThreadDragImage: (count) ->
canvas = DragCanvas
# Make sure the canvas has a 2x pixel density on retina displays
scale = window.devicePixelRatio
canvas.width = 58 * scale
canvas.height = 55 * scale
canvas.style.position = "absolute"
canvas.style.width = "58px"
canvas.style.height = "55px"
# necessary for setDragImage to work
ctx = canvas.getContext('2d')
# mail background image
if count > 1
ctx.rotate(-20*Math.PI/180)
ctx.drawImage(ThreadDragImage, -10*scale, 2*scale, 48*scale, 48*scale)
ctx.rotate(20*Math.PI/180)
ctx.drawImage(ThreadDragImage, 0, 0, 48*scale, 48*scale)
# count bubble
dotGradient = ctx.createLinearGradient(0, 0, 0, 15 * scale)
dotGradient.addColorStop(0, "rgb(116, 124, 143)")
dotGradient.addColorStop(1, "rgb(67, 77, 104)")
ctx.strokeStyle = "rgba(39, 48, 68, 0.6)"
ctx.lineWidth = 1
ctx.fillStyle = dotGradient
textX = 49
text = "#{count}"
if (count < 10)
CanvasUtils.roundRect(ctx, 41 * scale, 1 * scale, 16 * scale, 14 * scale, 7 * scale, true, true)
else if (count < 100)
CanvasUtils.roundRect(ctx, 37 * scale, 1 * scale, 20 * scale, 14 * scale, 7 * scale, true, true)
textX = 46
else
CanvasUtils.roundRect(ctx, 33 * scale, 1 * scale, 25 * scale, 14 * scale, 7 * scale, true, true)
text = "99+"
textX = 46
# count text
ctx.fillStyle = "rgba(255,255,255,0.9)"
ctx.font = "#{11 * scale}px sans-serif"
ctx.textAlign = "center"
ctx.fillText(text, textX * scale, 12 * scale, 30 * scale)
return DragCanvas
module.exports = CanvasUtils

View file

@ -1,9 +1,10 @@
React = require 'react'
###
# Images are supposed to by default show a ghost image when dragging and
# dropping. Unfortunatley this does not work in Electron. Since we're a
# desktop app we don't want all images draggable, but we do want some (like attachments) to be able to be dragged away with a preview image.
Public: Images are supposed to by default show a ghost image when dragging and
dropping. Unfortunately this does not work in Electron. Since we're a
desktop app we don't want all images draggable, but we do want some (like
attachments) to be able to be dragged away with a preview image.
###
class DraggableImg extends React.Component
@displayName: 'DraggableImg'

View file

@ -0,0 +1,47 @@
React = require 'react'
_ = require 'underscore'
class DropZone extends React.Component
@propTypes:
shouldAcceptDrop: React.PropTypes.func.isRequired
onDrop: React.PropTypes.func.isRequired
onDragStateChange: React.PropTypes.func
constructor: ->
render: ->
otherProps = _.omit(@props, _.keys(@constructor.propTypes))
<div {...otherProps} onDragEnter={@_onDragEnter} onDragLeave={@_onDragLeave} onDrop={@_onDrop}>
{@props.children}
</div>
# We maintain a "dragCounter" because dragEnter and dragLeave events *stack*
# when the user moves the item in and out of DOM elements inside of our container.
# It's really awful and everyone hates it.
#
# Alternative solution *maybe* is to set pointer-events:none; during drag.
_onDragEnter: (e) =>
return unless @props.shouldAcceptDrop(e)
@_dragCounter ?= 0
@_dragCounter += 1
if @_dragCounter is 1 and @props.onDragStateChange
@props.onDragStateChange(isDropping: true)
return
_onDragLeave: (e) =>
return unless @props.shouldAcceptDrop(e)
@_dragCounter -= 1
if @_dragCounter is 0 and @props.onDragStateChange
@props.onDragStateChange(isDropping: false)
return
_onDrop: (e) =>
return unless @props.shouldAcceptDrop(e)
if @props.onDragStateChange
@props.onDragStateChange(isDropping: false)
@_dragCounter = 0
@props.onDrop(e)
return
module.exports = DropZone

View file

@ -98,6 +98,8 @@ class MultiselectList extends React.Component
className = @props.className
className += " ready" if @state.ready
otherProps = _.omit(@props, _.keys(@constructor.propTypes))
if @state.dataView and @state.handler
className += " " + @state.handler.cssClass()
@ -116,7 +118,7 @@ class MultiselectList extends React.Component
visible={@state.ready && @state.dataView.count() is 0}
dataView={@state.dataView} />
<div className={className}>
<div className={className} {...otherProps}>
<ListTabular
ref="list"
columns={@state.computedColumns}
@ -130,7 +132,7 @@ class MultiselectList extends React.Component
{emptyElement}
</div>
else
<div className={className}>
<div className={className} {...otherProps}>
<Spinner visible={@state.ready is false} />
</div>

View file

@ -27,6 +27,10 @@ class CategoryStore extends NylasStore
"trash"
]
LockedCategoryNames: [
"sent"
]
AllMailName: "all"
byId: (id) -> @_categoryCache[id]

View file

@ -88,6 +88,7 @@ class Toolbar extends React.Component
@state = @_getStateFromStores()
componentDidMount: =>
@mounted = true
@unlisteners = []
@unlisteners.push WorkspaceStore.listen (event) =>
@setState(@_getStateFromStores())
@ -97,6 +98,7 @@ class Toolbar extends React.Component
window.requestAnimationFrame => @recomputeLayout()
componentWillUnmount: =>
@mounted = false
window.removeEventListener("resize", @_onWindowResize)
unlistener() for unlistener in @unlisteners
@ -105,7 +107,6 @@ class Toolbar extends React.Component
componentDidUpdate: =>
# Wait for other components that are dirty (the actual columns in the sheet)
# to update as well.
window.requestAnimationFrame => @recomputeLayout()
shouldComponentUpdate: (nextProps, nextState) =>
@ -148,6 +149,9 @@ class Toolbar extends React.Component
</TimeoutTransitionGroup>
recomputeLayout: =>
# Yes this really happens - do not remove!
return unless @mounted
# Find our item containers that are tied to specific columns
columnToolbarEls = React.findDOMNode(@).querySelectorAll('[data-column]')

View file

@ -44,6 +44,7 @@
display: inline-block;
cursor:default;
line-height: 22px;
-webkit-user-select: none;
}
.mail-label.removable {
padding-left:12px;