2017-08-17 22:00:37 +05:30
|
|
|
import sqljs from 'sql.js';
|
|
|
|
import { template as _template } from 'underscore';
|
|
|
|
|
|
|
|
const PREVIEW_TEMPLATE = _template(`
|
2018-11-08 19:23:39 +05:30
|
|
|
<div class="card">
|
|
|
|
<div class="card-header"><%- name %></div>
|
|
|
|
<div class="card-body">
|
2017-08-17 22:00:37 +05:30
|
|
|
<img class="img-thumbnail" src="data:image/png;base64,<%- image %>"/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`);
|
|
|
|
|
|
|
|
class BalsamiqViewer {
|
|
|
|
constructor(viewer) {
|
|
|
|
this.viewer = viewer;
|
|
|
|
}
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
loadFile(endpoint) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
|
|
|
|
xhr.open('GET', endpoint, true);
|
|
|
|
xhr.responseType = 'arraybuffer';
|
|
|
|
xhr.onload = loadEvent => this.fileLoaded(loadEvent, resolve, reject);
|
|
|
|
xhr.onerror = reject;
|
|
|
|
|
|
|
|
xhr.send();
|
|
|
|
});
|
|
|
|
}
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
fileLoaded(loadEvent, resolve, reject) {
|
|
|
|
if (loadEvent.target.status !== 200) return reject();
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
this.renderFile(loadEvent);
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
return resolve();
|
2017-08-17 22:00:37 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
renderFile(loadEvent) {
|
|
|
|
const container = document.createElement('ul');
|
|
|
|
|
|
|
|
this.initDatabase(loadEvent.target.response);
|
|
|
|
|
|
|
|
const previews = this.getPreviews();
|
|
|
|
previews.forEach((preview) => {
|
|
|
|
const renderedPreview = this.renderPreview(preview);
|
|
|
|
|
|
|
|
container.appendChild(renderedPreview);
|
|
|
|
});
|
|
|
|
|
|
|
|
container.classList.add('list-inline');
|
|
|
|
container.classList.add('previews');
|
|
|
|
|
|
|
|
this.viewer.appendChild(container);
|
|
|
|
}
|
|
|
|
|
|
|
|
initDatabase(data) {
|
|
|
|
const previewBinary = new Uint8Array(data);
|
|
|
|
|
|
|
|
this.database = new sqljs.Database(previewBinary);
|
|
|
|
}
|
|
|
|
|
|
|
|
getPreviews() {
|
|
|
|
const thumbnails = this.database.exec('SELECT * FROM thumbnails');
|
|
|
|
|
|
|
|
return thumbnails[0].values.map(BalsamiqViewer.parsePreview);
|
|
|
|
}
|
|
|
|
|
|
|
|
getResource(resourceID) {
|
|
|
|
const resources = this.database.exec(`SELECT * FROM resources WHERE id = '${resourceID}'`);
|
|
|
|
|
|
|
|
return resources[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
renderPreview(preview) {
|
|
|
|
const previewElement = document.createElement('li');
|
|
|
|
|
|
|
|
previewElement.classList.add('preview');
|
|
|
|
previewElement.innerHTML = this.renderTemplate(preview);
|
|
|
|
|
|
|
|
return previewElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
renderTemplate(preview) {
|
|
|
|
const resource = this.getResource(preview.resourceID);
|
|
|
|
const name = BalsamiqViewer.parseTitle(resource);
|
2018-11-08 19:23:39 +05:30
|
|
|
const { image } = preview;
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
const template = PREVIEW_TEMPLATE({
|
|
|
|
name,
|
|
|
|
image,
|
|
|
|
});
|
|
|
|
|
|
|
|
return template;
|
|
|
|
}
|
|
|
|
|
|
|
|
static parsePreview(preview) {
|
|
|
|
return JSON.parse(preview[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* resource = {
|
|
|
|
* columns: ['ID', 'BRANCHID', 'ATTRIBUTES', 'DATA'],
|
|
|
|
* values: [['id', 'branchId', 'attributes', 'data']],
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* 'attributes' being a JSON string containing the `name` property.
|
|
|
|
*/
|
|
|
|
static parseTitle(resource) {
|
|
|
|
return JSON.parse(resource.values[0][2]).name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default BalsamiqViewer;
|