mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-12 19:23:13 +08:00
44a7fa7117
Summary: This commit adds a couple of missing assets, including the icon for the plugin list and other misc icons. It also switches to the new UI where we use the thread timestamps to display the reminder date in the Reminders perspective instead of using mail labels. It also adds a header to the threads inside the reminders perspective to indicate that a reminder will be triggered if no one replies to their email. It also adds a header to indicate when a message has been brought back to the inbox due to a reminder based on sdw's designs. Finally it restores some code that magically disappeared when landing reminders + other misc cleanup Test Plan: Manual Reviewers: bengotow, halla Reviewed By: bengotow, halla Differential Revision: https://phab.nylas.com/D3388
200 lines
6.2 KiB
CoffeeScript
200 lines
6.2 KiB
CoffeeScript
_ = require 'underscore'
|
|
React = require 'react'
|
|
classNames = require 'classnames'
|
|
moment = require 'moment'
|
|
|
|
{ListTabular,
|
|
RetinaImg,
|
|
MailLabelSet,
|
|
MailImportantIcon,
|
|
InjectedComponent,
|
|
InjectedComponentSet} = require 'nylas-component-kit'
|
|
|
|
{Thread, FocusedPerspectiveStore, Utils, DateUtils} = require 'nylas-exports'
|
|
|
|
{ThreadArchiveQuickAction,
|
|
ThreadTrashQuickAction} = require './thread-list-quick-actions'
|
|
|
|
ThreadListParticipants = require './thread-list-participants'
|
|
ThreadListStore = require './thread-list-store'
|
|
ThreadListIcon = require './thread-list-icon'
|
|
|
|
# Get and format either last sent or last received timestamp depending on thread-list being viewed
|
|
ThreadListTimestamp = ({thread}) ->
|
|
if FocusedPerspectiveStore.current().isSent()
|
|
rawTimestamp = thread.lastMessageSentTimestamp
|
|
else
|
|
rawTimestamp = thread.lastMessageReceivedTimestamp
|
|
timestamp = DateUtils.shortTimeString(rawTimestamp)
|
|
return <span className="timestamp">{timestamp}</span>
|
|
ThreadListTimestamp.containerRequired = false
|
|
|
|
subject = (subj) ->
|
|
if (subj ? "").trim().length is 0
|
|
return <span className="no-subject">(No Subject)</span>
|
|
else if subj.split(/([\uD800-\uDBFF][\uDC00-\uDFFF])/g).length > 1
|
|
subjComponents = []
|
|
subjParts = subj.split /([\uD800-\uDBFF][\uDC00-\uDFFF])/g
|
|
for part, idx in subjParts
|
|
if part.match /([\uD800-\uDBFF][\uDC00-\uDFFF])/g
|
|
subjComponents.push <span className="emoji" key={idx}>{part}</span>
|
|
else
|
|
subjComponents.push <span key={idx}>{part}</span>
|
|
return subjComponents
|
|
else
|
|
return subj
|
|
|
|
getSnippet = (thread) ->
|
|
messages = thread.__messages || []
|
|
if (messages.length is 0)
|
|
return thread.snippet
|
|
|
|
return messages[messages.length - 1].snippet
|
|
|
|
|
|
c1 = new ListTabular.Column
|
|
name: "★"
|
|
resolver: (thread) =>
|
|
[
|
|
<ThreadListIcon key="thread-list-icon" thread={thread} />
|
|
<MailImportantIcon
|
|
key="mail-important-icon"
|
|
thread={thread}
|
|
showIfAvailableForAnyAccount={true}
|
|
/>
|
|
<InjectedComponentSet
|
|
key="injected-component-set"
|
|
inline={true}
|
|
containersRequired={false}
|
|
matching={role: "ThreadListIcon"}
|
|
className="thread-injected-icons"
|
|
exposedProps={thread: thread}
|
|
/>
|
|
]
|
|
|
|
c2 = new ListTabular.Column
|
|
name: "Participants"
|
|
width: 200
|
|
resolver: (thread) =>
|
|
hasDraft = (thread.__messages || []).find((m) => m.draft)
|
|
if hasDraft
|
|
<div style={display: 'flex'}>
|
|
<ThreadListParticipants thread={thread} />
|
|
<RetinaImg name="icon-draft-pencil.png"
|
|
className="draft-icon"
|
|
mode={RetinaImg.Mode.ContentPreserve} />
|
|
</div>
|
|
else
|
|
<ThreadListParticipants thread={thread} />
|
|
|
|
c3 = new ListTabular.Column
|
|
name: "Message"
|
|
flex: 4
|
|
resolver: (thread) =>
|
|
attachment = false
|
|
messages = thread.__messages || []
|
|
|
|
hasAttachments = thread.hasAttachments and messages.find (m) -> Utils.showIconForAttachments(m.files)
|
|
if hasAttachments
|
|
attachment = <div className="thread-icon thread-icon-attachment"></div>
|
|
|
|
<span className="details">
|
|
<MailLabelSet thread={thread} />
|
|
<span className="subject">{subject(thread.subject)}</span>
|
|
<span className="snippet">{getSnippet(thread)}</span>
|
|
{attachment}
|
|
</span>
|
|
|
|
c4 = new ListTabular.Column
|
|
name: "Date"
|
|
resolver: (thread) =>
|
|
return (
|
|
<InjectedComponent
|
|
className="thread-injected-timestamp"
|
|
fallback={ThreadListTimestamp}
|
|
exposedProps={thread: thread}
|
|
matching={role: "ThreadListTimestamp"}
|
|
/>
|
|
)
|
|
|
|
c5 = new ListTabular.Column
|
|
name: "HoverActions"
|
|
resolver: (thread) =>
|
|
<div className="inner">
|
|
<InjectedComponentSet
|
|
key="injected-component-set"
|
|
inline={true}
|
|
containersRequired={false}
|
|
children=
|
|
{[
|
|
<ThreadTrashQuickAction key="thread-trash-quick-action" thread={thread} />
|
|
<ThreadArchiveQuickAction key="thread-archive-quick-action" thread={thread} />
|
|
]}
|
|
matching={role: "ThreadListQuickAction"}
|
|
className="thread-injected-quick-actions"
|
|
exposedProps={thread: thread}
|
|
/>
|
|
</div>
|
|
|
|
cNarrow = new ListTabular.Column
|
|
name: "Item"
|
|
flex: 1
|
|
resolver: (thread) =>
|
|
pencil = false
|
|
attachment = false
|
|
messages = thread.__messages || []
|
|
|
|
hasAttachments = thread.hasAttachments and messages.find (m) -> Utils.showIconForAttachments(m.files)
|
|
if hasAttachments
|
|
attachment = <div className="thread-icon thread-icon-attachment"></div>
|
|
|
|
hasDraft = messages.find((m) => m.draft)
|
|
if hasDraft
|
|
pencil = <RetinaImg name="icon-draft-pencil.png" className="draft-icon" mode={RetinaImg.Mode.ContentPreserve} />
|
|
|
|
# TODO We are limiting the amount on injected icons in narrow mode to 1
|
|
# until we revisit the UI to accommodate more icons
|
|
<div style={display: 'flex', alignItems: 'flex-start'}>
|
|
<div className="icons-column">
|
|
<ThreadListIcon thread={thread} />
|
|
<InjectedComponentSet
|
|
inline={true}
|
|
matchLimit={1}
|
|
direction="column"
|
|
containersRequired={false}
|
|
key="injected-component-set"
|
|
exposedProps={thread: thread}
|
|
matching={role: "ThreadListIcon"}
|
|
className="thread-injected-icons"
|
|
/>
|
|
<MailImportantIcon
|
|
thread={thread}
|
|
showIfAvailableForAnyAccount={true}
|
|
/>
|
|
</div>
|
|
<div className="thread-info-column">
|
|
<div className="participants-wrapper">
|
|
<ThreadListParticipants thread={thread} />
|
|
{pencil}
|
|
<span style={flex:1}></span>
|
|
{attachment}
|
|
<InjectedComponent
|
|
key="thread-injected-timestamp"
|
|
className="thread-injected-timestamp"
|
|
fallback={ThreadListTimestamp}
|
|
exposedProps={thread: thread}
|
|
matching={role: "ThreadListTimestamp"}
|
|
/>
|
|
</div>
|
|
<div className="subject">{subject(thread.subject)}</div>
|
|
<div className="snippet-and-labels">
|
|
<div className="snippet">{getSnippet(thread)} </div>
|
|
<div style={flex: 1, flexShrink: 1}></div>
|
|
<MailLabelSet thread={thread} />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
module.exports =
|
|
Narrow: [cNarrow]
|
|
Wide: [c1, c2, c3, c4, c5]
|