This commit is contained in:
2020-06-18 00:00:19 +00:00
parent 1daa40a804
commit e5858bebd1
4 changed files with 327 additions and 314 deletions

View File

@@ -108,7 +108,10 @@ namespace rockfishCore.Controllers
newLicense.RegisteredTo = l.RegisteredTo; newLicense.RegisteredTo = l.RegisteredTo;
newLicense.DbId = l.DbId; newLicense.DbId = l.DbId;
newLicense.LicenseExpiration = DateUtil.EpochToDate(l.LicenseExpirationDate); if (l.LicenseExpires)
newLicense.LicenseExpiration = DateUtil.EpochToDate(l.LicenseExpirationDate);
else
newLicense.LicenseExpiration = DateUtil.EmptyDateValue;
newLicense.MaintenanceExpiration = DateUtil.EpochToDate(l.MaintenanceExpirationDate); newLicense.MaintenanceExpiration = DateUtil.EpochToDate(l.MaintenanceExpirationDate);
foreach (dtoLicenseFeature f in l.Features) foreach (dtoLicenseFeature f in l.Features)
{ {
@@ -125,6 +128,7 @@ namespace rockfishCore.Controllers
DBLicense.SiteId = l.SiteId; DBLicense.SiteId = l.SiteId;
DBLicense.Email = Customer.AdminEmail; DBLicense.Email = Customer.AdminEmail;
DBLicense.DbId = l.DbId; DBLicense.DbId = l.DbId;
DBLicense.Code = "na";
DBLicense.Key = Key; DBLicense.Key = Key;
DBLicense.RegTo = l.RegisteredTo; DBLicense.RegTo = l.RegisteredTo;
await ct.License.AddAsync(DBLicense); await ct.License.AddAsync(DBLicense);
@@ -142,7 +146,7 @@ throw new System.Exception("UNCOMMENT THIS FOR PRODUCTION");
#endif #endif
return NoContent(); return Ok("ok");
} }
catch (Exception ex) catch (Exception ex)
{ {

View File

@@ -64,18 +64,28 @@ app.ravLicense = (function () {
}); });
var submitData = app.utilB.objectifyFormDataArray(formData); var submitData = app.utilB.objectifyFormDataArray(formData);
//do features separately, above can't deal with multi select and we need to add schedtechs anyway
let features = [];
features.push({
Feature: "ServiceTechs",
Count: Number($("#techcount").val())
});
let options = $("#options").val();
//clean up the data before submit for (let i = 0; i < options.length; i++) {
debugger; features.push({ Feature: options[i], Count: 0 });
}
submitData["features"]=features;
//submit //submit
// alert("STUB submit"); // alert("STUB submit");
app.api.createRavLicense(submitData, function (res) { app.api.createRavLicense(submitData, function (res) {
if (res.error) { if (res.error) {
$.gevent.publish('app-show-error', res.msg); $.gevent.publish("app-show-error", res.msg);
} else { } else {
$('#key').val(res); $("#key").val(res);
return false; return false;
} }
}); });
return false; //prevent default return false; //prevent default

View File

@@ -13,322 +13,320 @@
/*global $, app, getComputedStyle */ /*global $, app, getComputedStyle */
app.utilB = (function () { app.utilB = (function () {
'use strict'; "use strict";
//---------------- BEGIN MODULE SCOPE VARIABLES -------------- //---------------- BEGIN MODULE SCOPE VARIABLES --------------
var var configMap = {
configMap = { regex_encode_html: /[&"'><]/g,
regex_encode_html: /[&"'><]/g, regex_encode_noamp: /["'><]/g,
regex_encode_noamp: /["'><]/g, html_encode_map: {
html_encode_map: { "&": "&#38;",
'&': '&#38;', '"': "&#34;",
'"': '&#34;', "'": "&#39;",
"'": '&#39;', ">": "&#62;",
'>': '&#62;', "<": "&#60;"
'<': '&#60;' }
} },
}, decodeHtml,
encodeHtml,
getEmSize,
getApiUrl,
getUrlParams,
formData,
objectifyFormDataArray,
getMediaSize,
prepareObjectForClient,
fixDatesToStrings,
epochToShortDate,
epochToLocalShortDate,
epochToLocalShortDateTime,
getCurrentDateTimeAsEpoch,
genListColumn,
genListColumnNoLink,
autoComplete,
prepareObjectDatesForServer,
fixStringToServerDate;
decodeHtml, encodeHtml, getEmSize, getApiUrl, getUrlParams, formData, objectifyFormDataArray, configMap.encode_noamp_map = $.extend({}, configMap.html_encode_map);
getMediaSize, prepareObjectForClient, fixDatesToStrings, delete configMap.encode_noamp_map["&"];
epochToShortDate, epochToLocalShortDate, epochToLocalShortDateTime, getCurrentDateTimeAsEpoch, //----------------- END MODULE SCOPE VARIABLES ---------------
genListColumn,
genListColumnNoLink, autoComplete, prepareObjectDatesForServer,
fixStringToServerDate;
configMap.encode_noamp_map = $.extend({}, configMap.html_encode_map); //------------------- BEGIN UTILITY METHODS ------------------
delete configMap.encode_noamp_map['&'];
//----------------- END MODULE SCOPE VARIABLES ---------------
//------------------- BEGIN UTILITY METHODS ------------------ // Begin decodeHtml
// Decodes HTML entities in a browser-friendly way
// See http://stackoverflow.com/questions/1912501/\
// unescape-html-entities-in-javascript
//
decodeHtml = function (str) {
return $("<div/>")
.html(str || "")
.text();
};
// End decodeHtml
// Begin encodeHtml
// This is single pass encoder for html entities and handles
// an arbitrary number of characters
//
encodeHtml = function (input_arg_str, exclude_amp) {
var input_str = String(input_arg_str),
regex,
lookup_map;
// Begin decodeHtml if (exclude_amp) {
// Decodes HTML entities in a browser-friendly way lookup_map = configMap.encode_noamp_map;
// See http://stackoverflow.com/questions/1912501/\ regex = configMap.regex_encode_noamp;
// unescape-html-entities-in-javascript } else {
// lookup_map = configMap.html_encode_map;
decodeHtml = function (str) { regex = configMap.regex_encode_html;
return $('<div/>').html(str || '').text(); }
}; return input_str.replace(regex, function (match, name) {
// End decodeHtml return lookup_map[match] || "";
});
};
// End encodeHtml
// Begin getEmSize
// returns size of ems in pixels
//
getEmSize = function (elem) {
return Number(getComputedStyle(elem, "").fontSize.match(/\d*\.?\d*/)[0]);
};
// End getEmSize
// Begin encodeHtml //Begin getApiUrl
// This is single pass encoder for html entities and handles //returns url for api methods by parsing current window location url
// an arbitrary number of characters //
// getApiUrl = function () {
encodeHtml = function (input_arg_str, exclude_amp) { var u =
var window.location.href
input_str = String(input_arg_str), .replace(window.location.hash, "")
regex, lookup_map; .replace("default.htm", "") + "api/";
//is it a dev local url?
if (u.indexOf("localhost:8080") != -1) {
u = u.replace("8080", "8081");
}
// End getApiUrl
if (exclude_amp) { //fix for random recurrence of extraneous ? on iPhone when using api
lookup_map = configMap.encode_noamp_map; //(Cannot post to http://gl-gztw.rhcloud.com/?api/list (404))
regex = configMap.regex_encode_noamp; u = u.replace("?", "");
return u;
};
// Begin formData
// Get or set all form fields
//
formData = function (data) {
//fix dates into strings
prepareObjectForClient(data);
var inps = $(":input").get();
if (typeof data != "object") {
// return all data
data = {};
$.each(inps, function () {
if (
this.name &&
(this.checked ||
/select|textarea/i.test(this.nodeName) ||
/text|hidden|password/i.test(this.type))
) {
data[this.name] = $(this).val();
}
});
return data;
} else {
$.each(inps, function () {
if (this.name && data[this.name]) {
if (this.type == "checkbox" || this.type == "radio") {
$(this).prop("checked", data[this.name]);
} else {
$(this).val(data[this.name]);
}
} else if (this.type == "checkbox") {
$(this).prop("checked", false);
}
});
return $(this);
}
};
// End formdata
// Begin getMediaSize
// Retrieves results of css media query in hidden pseudo element
// :after body element
//objectifyFormDataArray
getMediaSize = function () {
return window
.getComputedStyle(document.querySelector("body"), ":after")
.getPropertyValue("content")
.replace(/\"/g, "");
};
// End getMediaSize
//Begin objectifyFormDataArray
//takes name value form input pairs in array and turns into a keyed object
//suitable for sending as json object
//
objectifyFormDataArray = function (arr) {
var rv = {};
for (var i = 0; i < arr.length; ++i)
if (arr[i] !== undefined) {
rv[arr[i].name] = arr[i].value.trim(); //case 3205 added trim
}
return prepareObjectDatesForServer(rv);
};
//Prepare an object for server needed format before submission
//called by objectifyFormDataArray
prepareObjectDatesForServer = function (obj) {
if (Array.isArray(obj)) {
for (var i = 0; i < obj.length; i++) {
fixStringToServerDate(obj[i]);
}
} else {
fixStringToServerDate(obj);
}
return obj;
};
//
//Turn string date fields from client into server compatible format (Unix epoch like this: 1498262400 1499904000)
fixStringToServerDate = function (obj) {
var keys = Object.keys(obj);
keys.forEach(function (key) {
if (key.endsWith("Date") || key.startsWith("dt")) {
var value = obj[key];
if (value == null) {
obj[key] = "";
} else { } else {
lookup_map = configMap.html_encode_map; //this is the sample format we will see: 2017-07-13
regex = configMap.regex_encode_html; //TODO: is this assuming UTC?
obj[key] = moment.utc(value, "YYYY-MM-DD").unix();
} }
return input_str.replace(regex, }
function (match, name) { });
return lookup_map[match] || ''; };
}
);
};
// End encodeHtml
// Begin getEmSize //This function exists to change the properties of the passed in object
// returns size of ems in pixels //to values compatible with jquery form filling functions (mostly dates to strings for now)
// //
getEmSize = function (elem) { prepareObjectForClient = function (obj) {
return Number( if (Array.isArray(obj)) {
getComputedStyle(elem, '').fontSize.match(/\d*\.?\d*/)[0] for (var i = 0; i < obj.length; i++) {
); fixDatesToStrings(obj[i]);
}; }
// End getEmSize } else {
fixDatesToStrings(obj);
//Begin getApiUrl
//returns url for api methods by parsing current window location url
//
getApiUrl = function () {
var u = window.location.href.replace(window.location.hash, "").replace('default.htm', '') + "api/";
//is it a dev local url?
if (u.indexOf("localhost:8080") != -1) {
u = u.replace("8080", "8081");
}
// End getApiUrl
//fix for random recurrence of extraneous ? on iPhone when using api
//(Cannot post to http://gl-gztw.rhcloud.com/?api/list (404))
u = u.replace("?", "");
return u;
} }
return obj;
};
//
//Turn date fields of object coming from db into stringified values for consumption by client
//turn null dates into empty strings and iso date values into strings in iso format
fixDatesToStrings = function (obj) {
var keys = Object.keys(obj);
// Begin formData keys.forEach(function (key) {
// Get or set all form fields if (key.endsWith("Date") || key.startsWith("dt")) {
// var value = obj[key];
formData = function (data) { if (value == null) {
obj[key] = "";
//fix dates into strings
prepareObjectForClient(data);
var inps = $(":input").get();
if (typeof data != "object") {
// return all data
data = {};
$.each(inps, function () {
if (this.name && (this.checked || /select|textarea/i.test(this.nodeName) || /text|hidden|password/i.test(this.type))) {
data[this.name] = $(this).val();
}
});
return data;
} else { } else {
$.each(inps, function () { //Now with sqlite they come and go as unix epoch seconds
if (this.name && data[this.name]) { //needs to be yyyy-MM-dd
if (this.type == "checkbox" || this.type == "radio") { obj[key] = moment.utc(new Date(value * 1000)).format("YYYY-MM-DD");
$(this).prop("checked", (data[this.name]));
} else {
$(this).val(data[this.name]);
}
} else if (this.type == "checkbox") {
$(this).prop("checked", false);
}
});
return $(this);
} }
}; }
// End formdata });
};
//
// Begin getMediaSize //Turn date values coming in from server into displayable short date format
// Retrieves results of css media query in hidden pseudo element //used to display dates in various lists (where the source epoch is already localized)
// :after body element epochToShortDate = function (epoch) {
//objectifyFormDataArray if (epoch == null || epoch == 0) return "";
getMediaSize = function () { return moment.utc(new Date(epoch * 1000)).format("YYYY-MM-DD");
return window.getComputedStyle(document.querySelector('body'), ':after').getPropertyValue('content').replace(/\"/g, ''); };
}; //
// End getMediaSize
//LOCAL VERSION: Turn date values coming in from server into displayable short date format
//used to display dates in various lists where the source epoch is in UTC
epochToLocalShortDate = function (epoch) {
if (epoch == null || epoch == 0) return "";
var utdate = moment.utc(new Date(epoch * 1000));
var localdate = moment(utdate).local();
return localdate.format("YYYY-MM-DD");
};
//LOCAL VERSION: Turn date values coming in from server into displayable short date AND TIME format
//used to display dates in various lists where the source epoch is in UTC
epochToLocalShortDateTime = function (epoch) {
if (epoch == null || epoch == 0) return "";
var utdate = moment.utc(new Date(epoch * 1000));
var localdate = moment(utdate).local();
return localdate.format("YYYY-MM-DD LT");
};
//////////////////////////
//Get current date and time as a utc unix epoch
getCurrentDateTimeAsEpoch = function () {
return moment().utc().unix();
};
// Begin genListColumn
// This function is used to demarcate 'columns' of fields in basic list forms by wrapping each column field in html
//
genListColumn = function (val) {
return '<span class="rf-list-column">' + val + "</span>";
};
// End genListColumn
//Begin objectifyFormDataArray // Begin genListColumnNoLink
//takes name value form input pairs in array and turns into a keyed object // This function is used to demarcate and style 'columns' of fields in basic list forms by wrapping each column field in html
//suitable for sending as json object // that are not link columns
// genListColumnNoLink = function (val) {
objectifyFormDataArray = function (arr) { return '<span class="rf-list-column-nolink">' + val + "</span>";
var rv = {}; };
for (var i = 0; i < arr.length; ++i) // End genListColumn
if (arr[i] !== undefined) {
rv[arr[i].name] = arr[i].value.trim();//case 3205 added trim
}
return prepareObjectDatesForServer(rv);
}
// Begin autoComplete
// This function is used to attach an autocomplete method to an input
//
autoComplete = function (controlId, acGetToken) {
$("#" + controlId).autocomplete({
serviceUrl: app.shell.stateMap.apiUrl + "autocomplete",
params: {
acget: acGetToken
},
ajaxSettings: {
headers: app.api.getAuthHeaderObject()
}
});
};
// End autoComplete
// export methods
return {
decodeHtml: decodeHtml,
//Prepare an object for server needed format before submission encodeHtml: encodeHtml,
//called by objectifyFormDataArray getEmSize: getEmSize,
prepareObjectDatesForServer = function (obj) { getApiUrl: getApiUrl,
if (Array.isArray(obj)) { getUrlParams: getUrlParams,
for (var i = 0; i < obj.length; i++) { formData: formData,
fixStringToServerDate(obj[i]); getMediaSize: getMediaSize,
} objectifyFormDataArray: objectifyFormDataArray,
} else { epochToShortDate: epochToShortDate,
fixStringToServerDate(obj); epochToLocalShortDate: epochToLocalShortDate,
} epochToLocalShortDateTime: epochToLocalShortDateTime,
return obj; getCurrentDateTimeAsEpoch: getCurrentDateTimeAsEpoch,
} genListColumn: genListColumn,
// genListColumnNoLink: genListColumnNoLink,
autoComplete: autoComplete
//Turn string date fields from client into server compatible format (Unix epoch like this: 1498262400 1499904000) };
fixStringToServerDate = function (obj) { //------------------- END PUBLIC METHODS ---------------------
var keys = Object.keys(obj); })();
keys.forEach(function (key) {
if (key.endsWith('Date') || key.startsWith('dt')) {
var value = obj[key];
if (value == null) {
obj[key] = '';
} else {
//this is the sample format we will see: 2017-07-13
//TODO: is this assuming UTC?
obj[key] = moment.utc(value, "YYYY-MM-DD").unix();
}
}
});
}
//This function exists to change the properties of the passed in object
//to values compatible with jquery form filling functions (mostly dates to strings for now)
//
prepareObjectForClient = function (obj) {
if (Array.isArray(obj)) {
for (var i = 0; i < obj.length; i++) {
fixDatesToStrings(obj[i]);
}
} else {
fixDatesToStrings(obj);
}
return obj;
}
//
//Turn date fields of object coming from db into stringified values for consumption by client
//turn null dates into empty strings and iso date values into strings in iso format
fixDatesToStrings = function (obj) {
var keys = Object.keys(obj);
keys.forEach(function (key) {
if (key.endsWith('Date') || key.startsWith('dt')) {
var value = obj[key];
if (value == null) {
obj[key] = '';
} else {
//Now with sqlite they come and go as unix epoch seconds
//needs to be yyyy-MM-dd
obj[key] = moment.utc(new Date(value * 1000)).format("YYYY-MM-DD");
}
}
});
}
//
//Turn date values coming in from server into displayable short date format
//used to display dates in various lists (where the source epoch is already localized)
epochToShortDate = function (epoch) {
if (epoch == null || epoch == 0) return '';
return moment.utc(new Date(epoch * 1000)).format("YYYY-MM-DD");
}
//
//LOCAL VERSION: Turn date values coming in from server into displayable short date format
//used to display dates in various lists where the source epoch is in UTC
epochToLocalShortDate = function (epoch) {
if (epoch == null || epoch == 0) return '';
var utdate = moment.utc(new Date(epoch * 1000));
var localdate = moment(utdate).local();
return localdate.format("YYYY-MM-DD");
}
//LOCAL VERSION: Turn date values coming in from server into displayable short date AND TIME format
//used to display dates in various lists where the source epoch is in UTC
epochToLocalShortDateTime = function (epoch) {
if (epoch == null || epoch == 0) return '';
var utdate = moment.utc(new Date(epoch * 1000));
var localdate = moment(utdate).local();
return localdate.format("YYYY-MM-DD LT");
}
//////////////////////////
//Get current date and time as a utc unix epoch
getCurrentDateTimeAsEpoch = function () {
return moment().utc().unix();
}
// Begin genListColumn
// This function is used to demarcate 'columns' of fields in basic list forms by wrapping each column field in html
//
genListColumn = function (val) {
return '<span class="rf-list-column">' + val + '</span>'
};
// End genListColumn
// Begin genListColumnNoLink
// This function is used to demarcate and style 'columns' of fields in basic list forms by wrapping each column field in html
// that are not link columns
genListColumnNoLink = function (val) {
return '<span class="rf-list-column-nolink">' + val + '</span>'
};
// End genListColumn
// Begin autoComplete
// This function is used to attach an autocomplete method to an input
//
autoComplete = function (controlId, acGetToken) {
$('#' + controlId).autocomplete({
serviceUrl: app.shell.stateMap.apiUrl + 'autocomplete',
params: {
acget: acGetToken
},
ajaxSettings: {
headers: app.api.getAuthHeaderObject()
}
});
};
// End autoComplete
// export methods
return {
decodeHtml: decodeHtml,
encodeHtml: encodeHtml,
getEmSize: getEmSize,
getApiUrl: getApiUrl,
getUrlParams: getUrlParams,
formData: formData,
getMediaSize: getMediaSize,
objectifyFormDataArray: objectifyFormDataArray,
epochToShortDate: epochToShortDate,
epochToLocalShortDate: epochToLocalShortDate,
epochToLocalShortDateTime: epochToLocalShortDateTime,
getCurrentDateTimeAsEpoch: getCurrentDateTimeAsEpoch,
genListColumn: genListColumn,
genListColumnNoLink: genListColumnNoLink,
autoComplete: autoComplete
};
//------------------- END PUBLIC METHODS ---------------------
}());

View File

@@ -21,8 +21,8 @@
<div class="col-sm-6"> <div class="col-sm-6">
<div class="form-group"> <div class="form-group">
<label for="maintenanceExpirationDate">Support expires</label> <label for="maintenanceExpirationDate">Support expires</label>
<input class="form-control" type="date" id="maintenanceExpirationDate" name="maintenanceExpirationDate" <input class="form-control" type="date" id="maintenanceExpirationDate"
value=""> name="maintenanceExpirationDate" value="">
</div> </div>
</div> </div>
@@ -32,7 +32,8 @@
<input class="form-check-input" type="checkbox" name="licenseExpires" id="licenseExpires"> <input class="form-check-input" type="checkbox" name="licenseExpires" id="licenseExpires">
Temporary key / expires Temporary key / expires
</label> </label>
<input class="form-control" type="date" id="licenseExpirationDate" name="licenseExpirationDate" value=""> <input class="form-control" type="date" id="licenseExpirationDate" name="licenseExpirationDate"
value="">
</div> </div>
</div> </div>
@@ -45,10 +46,10 @@
<div class="col-sm-6"> <div class="col-sm-6">
<div class="form-group"> <div class="form-group">
<label for="options">Additional features</label> <label for="options">Additional features</label>
<select class="form-control" name="options" multiple> <select class="form-control" name="options" id="options" multiple>
<option value="ServiceMode">Rental Key</option> <option value="ServiceMode">Rental Key</option>
<option value="Accounting">Accounting</option> <option value="Accounting">Accounting</option>
</select> </select>
</div> </div>
</div> </div>