mirror of
https://github.com/moparisthebest/mailcatcher
synced 2024-11-12 04:05:01 -05:00
32ae22e4c7
this should avoid ugly '#' in the location bar
209 lines
8.7 KiB
CoffeeScript
209 lines
8.7 KiB
CoffeeScript
class MailCatcher
|
|
constructor: ->
|
|
$('#messages tr').live 'click', (e) =>
|
|
e.preventDefault()
|
|
@loadMessage $(e.currentTarget).attr 'data-message-id'
|
|
|
|
$('#message .views .format.tab a').live 'click', (e) =>
|
|
e.preventDefault()
|
|
@loadMessageBody @selectedMessage(), $($(e.currentTarget).parent('li')).data 'message-format'
|
|
|
|
$('#message .views .analysis.tab a').live 'click', (e) =>
|
|
e.preventDefault()
|
|
@loadMessageAnalysis @selectedMessage()
|
|
|
|
$('#resizer').live
|
|
mousedown: (e) ->
|
|
e.preventDefault()
|
|
$(window).bind events =
|
|
mouseup: (e) ->
|
|
e.preventDefault()
|
|
$(window).unbind events
|
|
mousemove: (e) ->
|
|
e.preventDefault()
|
|
$('#messages').css
|
|
height: e.clientY - $('#messages').offset().top
|
|
|
|
$('nav.app .clear a').live 'click', (e) =>
|
|
e.preventDefault()
|
|
if confirm "You will lose all your received messages.\n\nAre you sure you want to clear all messages?"
|
|
$.ajax
|
|
url: '/messages'
|
|
type: 'DELETE'
|
|
success: ->
|
|
$('#messages tbody, #message .metadata dd').empty()
|
|
$('#message .metadata .attachments').hide()
|
|
$('#message iframe').attr 'src', 'about:blank'
|
|
error: ->
|
|
alert 'Error while quitting.'
|
|
|
|
$('nav.app .quit a').live 'click', (e) =>
|
|
e.preventDefault()
|
|
if confirm "You will lose all your received messages.\n\nAre you sure you want to quit?"
|
|
$.ajax
|
|
type: 'DELETE'
|
|
success: ->
|
|
location.replace $('body > header h1 a').attr('href')
|
|
error: ->
|
|
alert 'Error while quitting.'
|
|
|
|
@refresh()
|
|
@subscribe()
|
|
|
|
# Only here because Safari's Date parsing *sucks*
|
|
# We throw away the timezone, but you could use it for something...
|
|
parseDateRegexp: /^(\d{4})[-\/\\](\d{2})[-\/\\](\d{2})(?:\s+|T)(\d{2})[:-](\d{2})[:-](\d{2})(?:([ +-]\d{2}:\d{2}|\s*\S+|Z?))?$/
|
|
parseDate: (date) ->
|
|
if match = @parseDateRegexp.exec(date)
|
|
new Date match[1], match[2] - 1, match[3], match[4], match[5], match[6], 0
|
|
|
|
formatDate: (date) ->
|
|
date &&= @parseDate(date) if typeof(date) == "string"
|
|
date &&= date.toString("dddd, d MMM yyyy h:mm:ss tt")
|
|
|
|
haveMessage: (message) ->
|
|
message = message.id if message.id?
|
|
$("#messages tbody tr[data-message-id=\"#{message}\"]").length > 0
|
|
|
|
selectedMessage: ->
|
|
$('#messages tr.selected').data 'message-id'
|
|
|
|
addMessage: (message) ->
|
|
$('#messages tbody').append \
|
|
$('<tr />').attr('data-message-id', message.id.toString())
|
|
.append($('<td/>').text(message.sender or "No sender").toggleClass("blank", !message.sender))
|
|
.append($('<td/>').text((message.recipients || []).join(', ') or "No receipients").toggleClass("blank", !message.recipients.length))
|
|
.append($('<td/>').text(message.subject or "No subject").toggleClass("blank", !message.subject))
|
|
.append($('<td/>').text @formatDate message.created_at)
|
|
|
|
loadMessage: (id) ->
|
|
id = id.id if id?.id?
|
|
id ||= $('#messages tr.selected').attr 'data-message-id'
|
|
|
|
if id?
|
|
$('#messages tbody tr:not([data-message-id="'+id+'"])').removeClass 'selected'
|
|
$('#messages tbody tr[data-message-id="'+id+'"]').addClass 'selected'
|
|
|
|
$.getJSON '/messages/' + id + '.json', (message) =>
|
|
$('#message .metadata dd.created_at').text @formatDate message.created_at
|
|
$('#message .metadata dd.from').text message.sender
|
|
$('#message .metadata dd.to').text (message.recipients || []).join(', ')
|
|
$('#message .metadata dd.subject').text message.subject
|
|
$('#message .views .tab.format').each (i, el) ->
|
|
$el = $(el)
|
|
format = $el.attr 'data-message-format'
|
|
if $.inArray(format, message.formats) >= 0
|
|
$el.find('a').attr('href', '/messages/' + id + '.' + format)
|
|
$el.show()
|
|
else
|
|
$el.hide()
|
|
|
|
if $("#message .views .tab.selected:not(:visible)").length
|
|
$("#message .views .tab.selected").removeClass "selected"
|
|
$("#message .views .tab:visible:first").addClass "selected"
|
|
|
|
if message.attachments.length
|
|
$ul = $('<ul/>').appendTo($('#message .metadata dd.attachments').empty())
|
|
|
|
$.each message.attachments, (i, attachment) ->
|
|
$ul.append($('<li>').append($('<a>').attr('href', attachment['href']).addClass(attachment['type'].split('/', 1)[0]).addClass(attachment['type'].replace('/', '-')).text(attachment['filename'])));
|
|
$('#message .metadata .attachments').show()
|
|
else
|
|
$('#message .metadata .attachments').hide()
|
|
|
|
$('#message .views .download a').attr 'href', "/messages/#{id}.eml"
|
|
|
|
if $('#message .views .tab.analysis.selected').length
|
|
@loadMessageAnalysis()
|
|
else
|
|
@loadMessageBody()
|
|
|
|
# XXX: These should probably cache their iframes for the current message now we're using a remote service:
|
|
|
|
loadMessageBody: (id, format) ->
|
|
id ||= @selectedMessage()
|
|
format ||= $('#message .views .tab.format.selected').attr 'data-message-format'
|
|
format ||= 'html'
|
|
|
|
$("#message .views .tab[data-message-format=\"#{format}\"]:not(.selected)").addClass 'selected'
|
|
$("#message .views .tab:not([data-message-format=\"#{format}\"]).selected").removeClass 'selected'
|
|
|
|
if id?
|
|
$('#message iframe').attr "src", "/messages/#{id}.#{format}"
|
|
|
|
loadMessageAnalysis: (id) ->
|
|
id ||= @selectedMessage()
|
|
|
|
$("#message .views .analysis.tab:not(.selected)").addClass 'selected'
|
|
$("#message .views :not(.analysis).tab.selected").removeClass 'selected'
|
|
|
|
if id?
|
|
# Makes a new iframe which will have a foreign origin eventually, and populate it with a quick intro and a form to send to Fractal.
|
|
$iframe = $('#message iframe').contents().children().html("""
|
|
<html class="mailcatcher"><head>#{$('link[rel="stylesheet"]')[0].outerHTML}</head><body><iframe></iframe></body></html>
|
|
""")
|
|
.find("head").append($('link[rel="stylesheet"]').clone()).end()
|
|
.find('iframe').contents().children().html("""
|
|
<html>
|
|
<head>
|
|
<title>Analysis</title>
|
|
#{$('link[rel="stylesheet"]')[0].outerHTML}
|
|
</head>
|
|
<body class="iframe">
|
|
<h1>Analyse your email with Fractal</h1>
|
|
<p><a href="http://getfractal.com/" target="_blank">Fractal</a> is a really neat service that applies common email design and development knowledge from <a href="http://www.email-standards.org/" target="_blank">Email Standards Project</a> to your HTML email and tells you what you've done wrong or what you should do instead.</p>
|
|
<p>Please note that this <strong>sends your email to the Fractal service</strong> for analysis. Read their <a href="http://getfractal.com/terms" target="_blank">terms of service</a> if you're paranoid.</p>
|
|
<form action="http://getfractal.com/validate" method="POST">
|
|
<input type="hidden" name="html" />
|
|
<input type="submit" value="Analyse" disabled="disabled" /><span class="loading" style="color: #999">Loading your email...</span>
|
|
</form>
|
|
</body>
|
|
</html>
|
|
""")
|
|
# This should be cached if already accessed, so it's actually quite quick
|
|
$.get "/messages/#{id}.html", (html) ->
|
|
$iframe
|
|
.find('input[name="html"]').attr('value', html).end()
|
|
.find('.loading').hide().end()
|
|
.find('input[type="submit"]').attr('disabled', null).end()
|
|
.find('form').submit ->
|
|
$(this)
|
|
.find('input[type="submit"]').attr('disabled', 'disabled').end()
|
|
.find('.loading').text('Analysing...').show()
|
|
|
|
# FIXME: Fractal need to allow GET requests to their JSONP endpoint, then we can use the API:
|
|
# $.ajax
|
|
# url: 'http://getfractal.com/api/v1/validate/format/jsonp'
|
|
# data:
|
|
# api_key: '59372c4f65426f78282c5c657d'
|
|
# html: html
|
|
# dataType: 'jsonp'
|
|
# success: (json) ->
|
|
# console.log json
|
|
# $iframe.children().html json
|
|
|
|
refresh: ->
|
|
$.getJSON '/messages', (messages) =>
|
|
$.each messages, (i, message) =>
|
|
unless @haveMessage message
|
|
@addMessage message
|
|
|
|
subscribe: ->
|
|
if WebSocket?
|
|
@subscribeWebSocket()
|
|
else
|
|
@subscribePoll()
|
|
|
|
subscribeWebSocket: ->
|
|
secure = window.location.scheme == 'https'
|
|
@websocket = new WebSocket("#{if secure then 'wss' else 'ws'}://#{window.location.host}/messages");
|
|
@websocket.onmessage = (event) =>
|
|
@addMessage $.parseJSON event.data
|
|
|
|
subscribePoll: ->
|
|
unless @refreshInterval?
|
|
@refreshInterval = setInterval (=> @refresh()), 1000
|
|
|
|
|
|
$ -> window.MailCatcher = new MailCatcher
|