2016-06-29 04:41:50 +00:00
|
|
|
/* jslint node: true */
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
// ENiGMA½
|
|
|
|
const MenuModule = require('../core/menu_module.js').MenuModule;
|
|
|
|
const getModDatabasePath = require('../core/database.js').getModDatabasePath;
|
|
|
|
const ViewController = require('../core/view_controller.js').ViewController;
|
|
|
|
const theme = require('../core/theme.js');
|
|
|
|
const ansi = require('../core/ansi_term.js');
|
2016-08-27 03:30:51 +00:00
|
|
|
const stringFormat = require('../core/string_format.js');
|
2016-06-29 04:41:50 +00:00
|
|
|
|
|
|
|
// deps
|
|
|
|
const sqlite3 = require('sqlite3');
|
|
|
|
const async = require('async');
|
|
|
|
const _ = require('lodash');
|
2016-06-29 05:53:30 +00:00
|
|
|
const moment = require('moment');
|
2016-06-29 04:41:50 +00:00
|
|
|
|
2016-07-04 18:56:54 +00:00
|
|
|
/*
|
|
|
|
Module :TODO:
|
|
|
|
* Add pipe code support
|
|
|
|
- override max length & monitor *display* len as user types in order to allow for actual display len with color
|
|
|
|
* Add preview control: Shows preview with pipe codes resolved
|
|
|
|
* Add ability to at least alternate formatStrings -- every other
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2016-06-29 04:41:50 +00:00
|
|
|
exports.moduleInfo = {
|
|
|
|
name : 'Onelinerz',
|
|
|
|
desc : 'Standard local onelinerz',
|
|
|
|
author : 'NuSkooler',
|
|
|
|
packageName : 'codes.l33t.enigma.onelinerz',
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getModule = OnelinerzModule;
|
|
|
|
|
|
|
|
const MciCodeIds = {
|
|
|
|
ViewForm : {
|
|
|
|
Entries : 1,
|
|
|
|
AddPrompt : 2,
|
|
|
|
},
|
|
|
|
AddForm : {
|
2016-06-29 05:53:30 +00:00
|
|
|
NewEntry : 1,
|
|
|
|
EntryPreview : 2,
|
|
|
|
AddPrompt : 3,
|
2016-06-29 04:41:50 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const FormIds = {
|
|
|
|
View : 0,
|
|
|
|
Add : 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
function OnelinerzModule(options) {
|
|
|
|
MenuModule.call(this, options);
|
|
|
|
|
|
|
|
const self = this;
|
|
|
|
const config = this.menuConfig.config;
|
|
|
|
|
|
|
|
this.initSequence = function() {
|
|
|
|
async.series(
|
|
|
|
[
|
|
|
|
function beforeDisplayArt(callback) {
|
|
|
|
self.beforeArt(callback);
|
|
|
|
},
|
|
|
|
function display(callback) {
|
|
|
|
self.displayViewScreen(false, callback);
|
|
|
|
}
|
|
|
|
],
|
|
|
|
err => {
|
|
|
|
if(err) {
|
|
|
|
// :TODO: Handle me -- initSequence() should really take a completion callback
|
|
|
|
}
|
|
|
|
self.finishedLoading();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
this.displayViewScreen = function(clearScreen, cb) {
|
|
|
|
async.waterfall(
|
|
|
|
[
|
|
|
|
function clearAndDisplayArt(callback) {
|
|
|
|
if(self.viewControllers.add) {
|
|
|
|
self.viewControllers.add.setFocus(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(clearScreen) {
|
|
|
|
self.client.term.rawWrite(ansi.resetScreen());
|
|
|
|
}
|
|
|
|
|
|
|
|
theme.displayThemedAsset(
|
|
|
|
config.art.entries,
|
|
|
|
self.client,
|
|
|
|
{ font : self.menuConfig.font, trailingLF : false },
|
|
|
|
(err, artData) => {
|
|
|
|
return callback(err, artData);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
},
|
|
|
|
function initOrRedrawViewController(artData, callback) {
|
|
|
|
if(_.isUndefined(self.viewControllers.add)) {
|
|
|
|
const vc = self.addViewController(
|
|
|
|
'view',
|
|
|
|
new ViewController( { client : self.client, formId : FormIds.View } )
|
|
|
|
);
|
|
|
|
|
|
|
|
const loadOpts = {
|
|
|
|
callingMenu : self,
|
|
|
|
mciMap : artData.mciMap,
|
|
|
|
formId : FormIds.View,
|
|
|
|
};
|
|
|
|
|
|
|
|
return vc.loadFromMenuConfig(loadOpts, callback);
|
|
|
|
} else {
|
|
|
|
self.viewControllers.view.setFocus(true);
|
|
|
|
self.viewControllers.view.getView(MciCodeIds.ViewForm.AddPrompt).redraw();
|
|
|
|
return callback(null);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function fetchEntries(callback) {
|
|
|
|
const entriesView = self.viewControllers.view.getView(MciCodeIds.ViewForm.Entries);
|
|
|
|
const limit = entriesView.dimens.height;
|
|
|
|
let entries = [];
|
|
|
|
|
|
|
|
self.db.each(
|
2016-06-30 05:23:06 +00:00
|
|
|
`SELECT *
|
|
|
|
FROM (
|
|
|
|
SELECT *
|
|
|
|
FROM onelinerz
|
|
|
|
ORDER BY timestamp DESC
|
|
|
|
LIMIT ${limit}
|
|
|
|
)
|
|
|
|
ORDER BY timestamp ASC;`,
|
2016-06-29 04:41:50 +00:00
|
|
|
(err, row) => {
|
|
|
|
if(!err) {
|
2016-06-29 05:53:30 +00:00
|
|
|
row.timestamp = moment(row.timestamp); // convert -> moment
|
2016-06-29 04:41:50 +00:00
|
|
|
entries.push(row);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
err => {
|
|
|
|
return callback(err, entriesView, entries);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
},
|
|
|
|
function populateEntries(entriesView, entries, callback) {
|
2016-06-30 05:23:06 +00:00
|
|
|
const listFormat = config.listFormat || '{username}@{ts}: {oneliner}';// :TODO: should be userName to be consistent
|
|
|
|
const tsFormat = config.timestampFormat || 'ddd h:mma';
|
2016-06-29 04:41:50 +00:00
|
|
|
|
|
|
|
entriesView.setItems(entries.map( e => {
|
2016-08-27 03:30:51 +00:00
|
|
|
return stringFormat(listFormat, {
|
2016-06-29 04:41:50 +00:00
|
|
|
userId : e.user_id,
|
|
|
|
username : e.user_name,
|
|
|
|
oneliner : e.oneliner,
|
2016-06-30 05:23:06 +00:00
|
|
|
ts : e.timestamp.format(tsFormat),
|
2016-06-29 04:41:50 +00:00
|
|
|
} );
|
|
|
|
}));
|
|
|
|
|
|
|
|
entriesView.redraw();
|
|
|
|
|
2016-07-04 18:56:54 +00:00
|
|
|
return callback(null);
|
|
|
|
},
|
|
|
|
function finalPrep(callback) {
|
|
|
|
const promptView = self.viewControllers.view.getView(MciCodeIds.ViewForm.AddPrompt);
|
|
|
|
promptView.setFocusItemIndex(1); // default to NO
|
2016-06-29 04:41:50 +00:00
|
|
|
return callback(null);
|
|
|
|
}
|
|
|
|
],
|
2016-06-29 05:53:30 +00:00
|
|
|
err => {
|
|
|
|
if(cb) {
|
|
|
|
return cb(err);
|
|
|
|
}
|
|
|
|
}
|
2016-06-29 04:41:50 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
this.displayAddScreen = function(cb) {
|
|
|
|
async.waterfall(
|
|
|
|
[
|
|
|
|
function clearAndDisplayArt(callback) {
|
|
|
|
self.viewControllers.view.setFocus(false);
|
|
|
|
self.client.term.rawWrite(ansi.resetScreen());
|
|
|
|
|
|
|
|
theme.displayThemedAsset(
|
|
|
|
config.art.add,
|
|
|
|
self.client,
|
|
|
|
{ font : self.menuConfig.font },
|
|
|
|
(err, artData) => {
|
|
|
|
return callback(err, artData);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
},
|
|
|
|
function initOrRedrawViewController(artData, callback) {
|
|
|
|
if(_.isUndefined(self.viewControllers.add)) {
|
|
|
|
const vc = self.addViewController(
|
|
|
|
'add',
|
|
|
|
new ViewController( { client : self.client, formId : FormIds.Add } )
|
|
|
|
);
|
|
|
|
|
|
|
|
const loadOpts = {
|
|
|
|
callingMenu : self,
|
|
|
|
mciMap : artData.mciMap,
|
|
|
|
formId : FormIds.Add,
|
|
|
|
};
|
|
|
|
|
|
|
|
return vc.loadFromMenuConfig(loadOpts, callback);
|
|
|
|
} else {
|
2016-06-29 05:53:30 +00:00
|
|
|
self.viewControllers.add.setFocus(true);
|
2016-06-29 04:41:50 +00:00
|
|
|
self.viewControllers.add.redrawAll();
|
2016-06-29 05:53:30 +00:00
|
|
|
self.viewControllers.add.switchFocus(MciCodeIds.AddForm.NewEntry);
|
2016-06-29 04:41:50 +00:00
|
|
|
return callback(null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
2016-06-29 05:53:30 +00:00
|
|
|
err => {
|
|
|
|
if(cb) {
|
|
|
|
return cb(err);
|
|
|
|
}
|
|
|
|
}
|
2016-06-29 04:41:50 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2016-06-29 05:53:30 +00:00
|
|
|
this.clearAddForm = function() {
|
|
|
|
const newEntryView = self.viewControllers.add.getView(MciCodeIds.AddForm.NewEntry);
|
|
|
|
const previewView = self.viewControllers.add.getView(MciCodeIds.AddForm.EntryPreview);
|
|
|
|
|
|
|
|
newEntryView.setText('');
|
2016-07-01 04:30:46 +00:00
|
|
|
|
|
|
|
// preview is optional
|
|
|
|
if(previewView) {
|
|
|
|
previewView.setText('');
|
|
|
|
}
|
2016-06-29 05:53:30 +00:00
|
|
|
};
|
|
|
|
|
2016-06-29 04:41:50 +00:00
|
|
|
this.menuMethods = {
|
2016-07-25 20:35:58 +00:00
|
|
|
viewAddScreen : function(formData, extraArgs, cb) {
|
|
|
|
return self.displayAddScreen(cb);
|
2016-06-29 05:53:30 +00:00
|
|
|
},
|
|
|
|
|
2016-07-25 20:35:58 +00:00
|
|
|
addEntry : function(formData, extraArgs, cb) {
|
2016-06-29 05:53:30 +00:00
|
|
|
if(_.isString(formData.value.oneliner) && formData.value.oneliner.length > 0) {
|
|
|
|
const oneliner = formData.value.oneliner.trim(); // remove any trailing ws
|
|
|
|
|
|
|
|
self.storeNewOneliner(oneliner, err => {
|
2016-06-30 05:23:06 +00:00
|
|
|
if(err) {
|
|
|
|
self.client.log.warn( { error : err.message }, 'Failed saving oneliner');
|
|
|
|
}
|
|
|
|
|
2016-06-29 05:53:30 +00:00
|
|
|
self.clearAddForm();
|
2016-07-25 20:35:58 +00:00
|
|
|
return self.displayViewScreen(true, cb); // true=cls
|
2016-06-29 05:53:30 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// empty message - treat as if cancel was hit
|
2016-07-25 20:35:58 +00:00
|
|
|
return self.displayViewScreen(true, cb); // true=cls
|
2016-06-29 05:53:30 +00:00
|
|
|
}
|
|
|
|
},
|
2016-06-29 04:41:50 +00:00
|
|
|
|
2016-07-25 20:35:58 +00:00
|
|
|
cancelAdd : function(formData, extraArgs, cb) {
|
2016-06-29 05:53:30 +00:00
|
|
|
self.clearAddForm();
|
2016-07-25 20:35:58 +00:00
|
|
|
return self.displayViewScreen(true, cb); // true=cls
|
2016-06-29 04:41:50 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.initDatabase = function(cb) {
|
|
|
|
async.series(
|
|
|
|
[
|
|
|
|
function openDatabase(callback) {
|
|
|
|
self.db = new sqlite3.Database(
|
|
|
|
getModDatabasePath(exports.moduleInfo),
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
},
|
|
|
|
function createTables(callback) {
|
|
|
|
self.db.serialize( () => {
|
|
|
|
self.db.run(
|
|
|
|
`CREATE TABLE IF NOT EXISTS onelinerz (
|
|
|
|
id INTEGER PRIMARY KEY,
|
|
|
|
user_id INTEGER_NOT NULL,
|
|
|
|
user_name VARCHAR NOT NULL,
|
|
|
|
oneliner VARCHAR NOT NULL,
|
|
|
|
timestamp DATETIME NOT NULL
|
|
|
|
)`
|
|
|
|
);
|
|
|
|
});
|
|
|
|
callback(null);
|
|
|
|
}
|
|
|
|
],
|
|
|
|
cb
|
|
|
|
);
|
|
|
|
};
|
2016-06-29 05:53:30 +00:00
|
|
|
|
|
|
|
this.storeNewOneliner = function(oneliner, cb) {
|
|
|
|
const ts = moment().format('YYYY-MM-DDTHH:mm:ss.SSSZ');
|
|
|
|
|
2016-06-30 05:23:06 +00:00
|
|
|
async.series(
|
|
|
|
[
|
|
|
|
function addRec(callback) {
|
|
|
|
self.db.run(
|
|
|
|
`INSERT INTO onelinerz (user_id, user_name, oneliner, timestamp)
|
|
|
|
VALUES (?, ?, ?, ?);`,
|
|
|
|
[ self.client.user.userId, self.client.user.username, oneliner, ts ],
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
},
|
|
|
|
function removeOld(callback) {
|
|
|
|
// keep 25 max most recent items - remove the older ones
|
|
|
|
self.db.run(
|
|
|
|
`DELETE FROM onelinerz
|
|
|
|
WHERE id IN (
|
|
|
|
SELECT id
|
|
|
|
FROM onelinerz
|
|
|
|
ORDER BY id DESC
|
|
|
|
LIMIT -1 OFFSET 25
|
|
|
|
);`,
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
}
|
|
|
|
],
|
|
|
|
cb
|
|
|
|
);
|
2016-06-29 05:53:30 +00:00
|
|
|
};
|
2016-06-29 04:41:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
require('util').inherits(OnelinerzModule, MenuModule);
|
|
|
|
|
|
|
|
OnelinerzModule.prototype.beforeArt = function(cb) {
|
|
|
|
OnelinerzModule.super_.prototype.beforeArt.call(this, err => {
|
|
|
|
return err ? cb(err) : this.initDatabase(cb);
|
|
|
|
});
|
|
|
|
};
|