forked from clonemeagain/attachment_preview
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscript.js
294 lines (273 loc) · 10.6 KB
/
script.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
"use strict";
// Attachment Preview Script, with HTML Sanitizer.
// Setup default config options
var AttachmentPreview = {
open_attachments: 'normal',
text_hide: 'Hide Attachment',
text_show: 'Show Attachment',
hide_seen: 0,
hide_age: 14,
limit: 'No limit'
};
// Leave the next line intact, as the plugin will replace it with settings overriding the defaults above.
/* REPLACED_BY_PLUGIN */
AttachmentPreview.oldest_timestamp = new Date(Date.now() - 1000 * 60 * 60 * 24 * this.hide_age);
/**
* https://developer.mozilla.org/en-US/docs/Web/API/Web_Storage_API/Using_the_Web_Storage_API
* @param type
* @returns
*/
AttachmentPreview.storageAvailable = function(type) {
try {
var storage = window[type],
x = '__storage_test__';
storage.setItem(x, x);
storage.removeItem(x);
return true;
} catch (e) {
return e instanceof DOMException && (
// everything except Firefox
e.code === 22 ||
// Firefox
e.code === 1014 ||
// test name field too, because code might not be present
// everything except Firefox
e.name === 'QuotaExceededError' ||
// Firefox
e.name === 'NS_ERROR_DOM_QUOTA_REACHED') &&
// acknowledge QuotaExceededError only if there's something already stored
storage.length !== 0;
}
};
AttachmentPreview.getAvailableStorage = function() {
if (typeof this.storage == 'undefined') {
if (this.storageAvailable('localStorage')) {
this.storage = window.localStorage;
} else if (this.storageAvailable('sessionStorage')) {
this.storage = window.sessionStorage;
}
}
return this.storage;
};
// Maintain a list of all the attachments the user has seen.. likely needs some form of cache/expiry type system.
AttachmentPreview.seen_ids = {};
AttachmentPreview.cache_key = 'attachment_preview_ids';
AttachmentPreview.already_seen = function(tid, id) {
if (typeof this.seen_ids == 'undefined' || !this.seen_ids.length) {
this.seen_ids = JSON.parse(this.getAvailableStorage().getItem(this.cache_key)) || {};
// Check if this ticket it has expired
if (typeof this.seen_ids[tid] !== 'undefined') {
if (this.seen_ids[tid].last_seen > this.max_age) {
delete this.seen_ids[tid];
};
}
}
if (typeof this.seen_ids[tid] !== 'undefined') {
var ticket = this.seen_ids[tid];
if ($.inArray(id, ticket.items) !== -1) {
// we have already seen that attachment on that ticket
return true;
} else {
// we have seen that ticket, but not that attachment.
ticket.items.push(id);
ticket.last_seen = Date.now();
}
} else {
// this ticket hasn't been seen before.
this.seen_ids[tid] = {
last_seen: Date.now(),
items: [id]
};
}
// bit crap, setting this over and over if there are many tickets in the page.. hmm.
this.getAvailableStorage().setItem(this.cache_key, JSON.stringify(this.seen_ids));
return false;
};
AttachmentPreview.init = function() {
// Don't we need pjax:success any more?
var attachments = $(".ap_embedded:not(.hidden)");
if (attachments.length) {
console.log("Fetching " + attachments.length + " non-hidden attachment[s].");
if (AttachmentPreview.hide_seen) {
var ticket_id = $('input[name="lockCode"]').closest('form').data('lock-object-id').substring(7);
attachments.each(function(idx) {
var $a = $(this),
id = $a.attr('id'),
attachment_number = id.substring(8);
if (!AttachmentPreview.already_seen(ticket_id, attachment_number)) {
$a.trigger('ap:fetch');
} else {
// User has already seen this attachment, show the button to toggle if they want to fetch/view it again:
var js_id = "'" + id.trim() + "'";
$a.hide().addClass('hidden');
$a.parent().find('small.filesize').after('<a class="button" onclick="ap_toggle(this,' + js_id + ');" target="_blank">' + AttachmentPreview.text_show + '</a>');
}
});
} else {
attachments.trigger('ap:fetch');
}
}
if (AttachmentPreview.open_attachments === 'new-tab') {
$('a.file').prop('target', '_blank');
}
console.log("AP: Plugin running, initial fetch limit configured to " + AttachmentPreview.limit + ".");
};
/**
* Slightly more convoluted that the other types Load the PDF into an Object
* Blob and shoves it into the <object> :-)
*
* @param id id of the element to inject the pdf
* @param url of the file to convert into a Blob and inject
*/
AttachmentPreview.fetch_pdf = function(id, url) {
var pdf = document.getElementById(id),
URL = window.URL || window.webkitURL;
if ( /* @cc_on!@ */ false || !!document.documentMode) {
// IE still cant display a PDF inside an <object>
console.log("Why Microsoft?");
// Fetch the "you suck IE" element inside the <object> and replace
// the object with it:
var b = $(pdf).contents();
$(pdf).replaceWith(b);
return;
}
var req = new XMLHttpRequest();
req.open("GET", url, true);
req.responseType = "blob"; // don't need an arraybuffer conversion, can just make a Blob directly!
req.onload = function() {
var ab = req.response;
var blob = new Blob([ab], {
type: "application/pdf"
});
//console.log("Loaded " + blob.type + " of size: " + blob.size);
// Convert the binary blob of PDF data into an Object URL
var object_url = URL.createObjectURL(blob);
if (!object_url) {
console.log("Failed to construct usable ObjectURL for the attachment. Bailing");
return;
}
var newpdf = pdf.cloneNode();
newpdf.setAttribute('type', 'application/pdf');
newpdf.setAttribute('data', object_url);
newpdf.setAttribute('style', 'display:inline-block');
newpdf.onload = function() {
// I think we are supposed to remove the ObjectURL's at some point:
URL.revokeObjectURL(object_url);
};
// Replace the node with our new one which displays it:
pdf.parentNode.replaceChild(newpdf, pdf);
// prevent repeated fetch events from re-fetching
newpdf.setAttribute('data-url', '');
};
req.send();
};
AttachmentPreview.handleFetchEvent = function(e) {
var elem = $(e.target).find('[data-type]').first(),
type = elem.data('type'),
url = elem.data('url');
if (type && url) {
switch (type) {
case 'image':
{
// We just have to set the src url, let the browser fetch
// the file as normal.
elem.attr('src', url);
break;
}
case 'pdf':
{
// Call our Wunderbar Blobinator function
var id = elem.attr('id');
this.fetch_pdf(id, url);
break;
}
case 'text':
// Replace the <pre> element's text with the Attachment:
$.get(url, function(data) {
elem.text(data);
});
break;
case 'html':
// Replace the html with the attachment, after passing
// through the sanitizer:
$.get(url, function(data) {
elem.html($("<div>" + $.trim(sanitizer.sanitize(data)) +
"</div>"));
});
}
// prevent repeated fetch events from re-fetching
elem.data('type', '');
}
};
// The HTML inject methods will probably fail at some point, but for now.. I'll just leave this here.
// src: https://gist.github.com/ufologist/5a0da51b2b9ef1b861c30254172ac3c9
var sanitizer = {};
(function($) {
var safe = '<a><b><blockquote><dd><div><dl><dt><em><h1><h2><h3><h4><i><img><li><ol><p><pre><s><sup><sub><strong><strike><ul><br><hr><table><th><tr><td><tbody><tfoot>';
function trimAttributes(node) {
$.each(node.attributes, function() {
var attrName = this.name;
var attrValue = this.value;
// we could filter the "bad" attributes, or just purge them all..
$(node).removeAttr(attrName);
});
}
sanitizer.sanitize = function(html) {
html = strip_tags(html, safe);
var output = $($.parseHTML('<div>' + $.trim(html) + '</div>', null,
false));
output.find('*').each(function() {
trimAttributes(this);
});
return output.html();
}
// http://locutus.io/php/strings/strip_tags/ filter the html to only those
// acceptable tags defined above as safe
function strip_tags(input, allowed) {
allowed = (((allowed || '') + '').toLowerCase().match(
/<[a-z][a-z0-9]*>/g) || []).join('')
var tags = /<\/?([a-z][a-z0-9]*)\b[^>]*>/gi
var commentsAndPhpTags = /<!--[\s\S]*?-->|<\?(?:php)?[\s\S]*?\?>/gi
return input
.replace(commentsAndPhpTags, '')
.replace(
tags,
function($0, $1) {
return allowed
.indexOf('<' + $1.toLowerCase() + '>') > -1 ? $0 :
''
})
}
})(jQuery);
/**
* Setup handler to receive Attachments Preview Fetch events, and act on them.
* Events are triggered by the buttons inserted for hidden attachment previews via ap_toggle
*/
$(document).on('ap:fetch', function(e) {
AttachmentPreview.handleFetchEvent(e);
});
$(document).ready(AttachmentPreview.init());
/**
* Toggle function for buttons, shows the attachment's wrapper element and
* triggers the fetch if the attachment isn't already there.
* Needs to be global.
*
* @param item
* (the button that was clicked)
* @param key
* (the id of the element we want to expand/load)
* @return false to prevent bubbling of event.
*/
function ap_toggle(item, key) {
var i = $(item),
elem = $('#' + key);
elem.slideToggle();
if (i.text() === AttachmentPreview.text_hide) {
i.text(AttachmentPreview.text_show);
} else {
elem.trigger('ap:fetch');
i.text(AttachmentPreview.text_hide);
}
return false;
}
console.log("AP: Plugin loaded");