diff --git a/Gruntfile.js b/Gruntfile.js index f0b29624..534cafc0 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -12,6 +12,8 @@ module.exports = function (grunt) { // Time how long tasks take. Can help when optimizing build times require('time-grunt')(grunt); + var serveStatic = require('serve-static'); + // Automatically load required Grunt tasks require('jit-grunt')(grunt, { useminPrepare: 'grunt-usemin', @@ -82,16 +84,16 @@ module.exports = function (grunt) { open: true, middleware: function (connect) { return [ - connect.static('.tmp'), + serveStatic('.tmp'), connect().use( '/bower_components', - connect.static('./bower_components') + serveStatic('./bower_components') ), connect().use( '/app/styles', - connect.static('./app/styles') + serveStatic('./app/styles') ), - connect.static(appConfig.app) + serveStatic(appConfig.app) ]; } } @@ -101,13 +103,13 @@ module.exports = function (grunt) { port: 9001, middleware: function (connect) { return [ - connect.static('.tmp'), - connect.static('test'), + serveStatic('.tmp'), + serveStatic('test'), connect().use( '/bower_components', - connect.static('./bower_components') + serveStatic('./bower_components') ), - connect.static(appConfig.app) + serveStatic(appConfig.app) ]; } } diff --git a/README.md b/README.md index 69d54c9d..a9d3a9a4 100644 --- a/README.md +++ b/README.md @@ -5,8 +5,8 @@ A scalable and highly flexible HTML5/JS platform to build and run in-browser app * Web site: http://socr.umich.edu * Issue-tracking and project management: https://socredu.atlassian.net/browse/SOCRFW -[![Build Status](https://travis-ci.org/SOCRedu/SOCR-framework.svg?branch=master)](https://travis-ci.org/SOCRedu/SOCR-framework) -[![Dependency Status](https://gemnasium.com/SOCRedu/SOCR-framework.png?branch=master)](https://gemnasium.com/SOCRedu/SOCR-framework) +[![Build Status](https://travis-ci.org/SOCR/SOCRAT.svg?branch=master)](https://travis-ci.org/SOCR/SOCRAT) +[![Dependency Status](https://gemnasium.com/SOCR/SOCRAT.png?branch=master)](https://gemnasium.com/SOCR/SOCRAT) Installation ------------ diff --git a/app/partials/analysis/getData/main.jade b/app/partials/analysis/getData/main.jade index 0606413b..a08ff693 100644 --- a/app/partials/analysis/getData/main.jade +++ b/app/partials/analysis/getData/main.jade @@ -44,4 +44,7 @@ div(ng-controller='getDataMainCtrl') button.btn.btn-primary(ng-click="getJsonByUrl()") Parse br - handsontable(purpose="json") \ No newline at end of file + div.lead.bg-danger(ng-hide="dataType != DATA_TYPES.NESTED") + | Visual representation of hierarchical data currently is not available. + div(ng-class="{'vis-hidden': dataType != DATA_TYPES.FLAT}") + handsontable(purpose="json") diff --git a/app/partials/analysis/tools/machineLearning/kMeans/main.jade b/app/partials/analysis/tools/machineLearning/kMeans/main.jade index 6b27e7d6..d8a592cc 100644 --- a/app/partials/analysis/tools/machineLearning/kMeans/main.jade +++ b/app/partials/analysis/tools/machineLearning/kMeans/main.jade @@ -1,16 +1,20 @@ div(ng-controller="kMeansMainCtrl") h2 2D k-means Clustering - p. - k-means clustering aims to partition n observations into k clusters in which each observation belongs - to the cluster with the nearest mean, serving as a prototype of the cluster. - div(ng-show="showresults").kmeans-results - p.lead Average accuracy: {{avgAccuracy}} - div.table-responsive - table.table.table-bordered.table-condensed - tr - th Label - th Accuracy - tr(ng-repeat="(label, acc) in accs") - td {{label}} - td {{acc.toFixed(2)}} - app-kmeans + div.lead.bg-danger(ng-hide="dataType == DATA_TYPES.FLAT") + | 2D k-means Clustering doesn't support current dataset. + | Only "flat" data tables are supported. + div(ng-hide="dataType != DATA_TYPES.FLAT") + p. + k-means clustering aims to partition n observations into k clusters in which each observation belongs + to the cluster with the nearest mean, serving as a prototype of the cluster. + div(ng-show="showresults").kmeans-results + p.lead Average accuracy: {{avgAccuracy}} + div.table-responsive + table.table.table-bordered.table-condensed + tr + th Label + th Accuracy + tr(ng-repeat="(label, acc) in accs") + td {{label}} + td {{acc.toFixed(2)}} + app-kmeans diff --git a/app/partials/analysis/tools/psychometrics/instrPerfEval/main.jade b/app/partials/analysis/tools/psychometrics/instrPerfEval/main.jade index d2812819..065059c0 100644 --- a/app/partials/analysis/tools/psychometrics/instrPerfEval/main.jade +++ b/app/partials/analysis/tools/psychometrics/instrPerfEval/main.jade @@ -1,73 +1,77 @@ div(ng-controller="instrPerfEvalMainCtrl") h2 Instrument Performance Evaluation - p. - Cronbach’s Alpha (α) is a measure of internal consistency or reliability of a psychometric instrument and measures - how well a set of items measure a single, one-dimensional latent aspect of individuals. - p.lead Cronbach's α: {{cronAlpha}} - div.table-responsive - table.table.table-bordered.table-striped - thead - tr - th Cronbach's alpha - th Internal consistency - tbody - tr.success - td α ≥ 0.9 - td Excellent (High-Stakes testing) - tr.success - td 0.7 ≤ α < 0.9 - td Good (Low-Stakes testing) - tr.info - td 0.6 ≤ α < 0.7 - td Acceptable - tr.warning - td 0.5 ≤ α < 0.6 - td Poor - tr.danger - td α < 0.5 - td Unacceptable - p. - Cronbach's α coefficient is a point estimate of the reliability. - Its standard error is important to construct an interval estimation of its true value - and to obtain statistical inference about its significance. - There are parametric and non-parametric methods to estimate the variance of Cronbach's α, - and compute confidence intervals. - p Cronbach’s Alpha confidence intervals - p.bg-info ID confidence interval: {{cronAlphaIdInterval}} - p.bg-info Koning and Franses confidence interval: {{cronAlphaKfInterval}} - p.bg-info Bootstrap confidence interval: {{cronAlphaBootstrapInterval}} - p.bg-info Logit confidence interval: {{cronAlphaLogitInterval}} - p.bg-info Asymptotically distribution-free (ADF) interval: {{cronAlphaAdfInterval}} - br - p Other metrics of reliability: - p. - The Intra-class correlation coefficient (ICC) assesses the consistency, or reproducibility, - of quantitative measurements made by different observers measuring the same quantity. - Broadly speaking, the ICC is defined as the ratio of between-cluster variance to total variance. - p.bg-info Intraclass correlation coefficient (ICC): {{icc}} - p. - In Split-Half Reliability assessment, the test is split in half (e.g., odd / even) creating "equivalent forms". - The two "forms" are correlated with each other and the correlation coefficient is adjusted - to reflect the entire test length, using the Spearman-Brown Prophecy formula. - p.bg-info Split-Half Reliability coefficient: {{splitHalfCoef}} - p. - The Kuder–Richardson Formula 20 (KR-20) is a very reliable internal reliability estimate which simulates - calculating split-half reliability for every possible combination of items. - The Cronbach's α and KR-20 are similar ― KR-20 is a derivative of the Cronbach's α with the advantage that - it can handle both dichotomous and continuous variables, however, KR-20 can't be used - when multiple-choice questions involve partial credit and require systematic item-based analysis. - p.bg-info Kuder–Richardson Formula 20 (KR-20): {{kr20}} - br - p - small - p References: - p 1. - a(href='http://wiki.socr.umich.edu/index.php/SMHS_Cronbachs'). - Scientific Methods for Health Sciences - Instrument Performance Evaluation: Cronbach's α - p. - 2. TSAGRIS, MICHAIL, CONSTANTINOS C. FRANGOS, and CHRISTOS C. FRANGOS. - "Confidence intervals for Cronbach’s reliability coefficient." - p. - 3. Maydeu-Olivares, Alberto, Donna L. Coffman, and Wolfgang M. Hartmann. - "Asymptotically distribution-free (ADF) interval estimation of coefficient alpha." - Psychological methods 12.2 (2007): 157. + div.lead.bg-danger(ng-hide="dataType == DATA_TYPES.FLAT") + | Instrument Performance Evaluation doesn't support current dataset. + | Only "flat" data tables are supported. + div(ng-hide="dataType != DATA_TYPES.FLAT") + p. + Cronbach’s Alpha (α) is a measure of internal consistency or reliability of a psychometric instrument and measures + how well a set of items measure a single, one-dimensional latent aspect of individuals. + p.lead Cronbach's α: {{cronAlpha}} + div.table-responsive + table.table.table-bordered.table-striped + thead + tr + th Cronbach's alpha + th Internal consistency + tbody + tr.success + td α ≥ 0.9 + td Excellent (High-Stakes testing) + tr.success + td 0.7 ≤ α < 0.9 + td Good (Low-Stakes testing) + tr.info + td 0.6 ≤ α < 0.7 + td Acceptable + tr.warning + td 0.5 ≤ α < 0.6 + td Poor + tr.danger + td α < 0.5 + td Unacceptable + p. + Cronbach's α coefficient is a point estimate of the reliability. + Its standard error is important to construct an interval estimation of its true value + and to obtain statistical inference about its significance. + There are parametric and non-parametric methods to estimate the variance of Cronbach's α, + and compute confidence intervals. + p Cronbach’s Alpha confidence intervals + p.bg-info ID confidence interval: {{cronAlphaIdInterval}} + p.bg-info Koning and Franses confidence interval: {{cronAlphaKfInterval}} + p.bg-info Bootstrap confidence interval: {{cronAlphaBootstrapInterval}} + p.bg-info Logit confidence interval: {{cronAlphaLogitInterval}} + p.bg-info Asymptotically distribution-free (ADF) interval: {{cronAlphaAdfInterval}} + br + p Other metrics of reliability: + p. + The Intra-class correlation coefficient (ICC) assesses the consistency, or reproducibility, + of quantitative measurements made by different observers measuring the same quantity. + Broadly speaking, the ICC is defined as the ratio of between-cluster variance to total variance. + p.bg-info Intraclass correlation coefficient (ICC): {{icc}} + p. + In Split-Half Reliability assessment, the test is split in half (e.g., odd / even) creating "equivalent forms". + The two "forms" are correlated with each other and the correlation coefficient is adjusted + to reflect the entire test length, using the Spearman-Brown Prophecy formula. + p.bg-info Split-Half Reliability coefficient: {{splitHalfCoef}} + p. + The Kuder–Richardson Formula 20 (KR-20) is a very reliable internal reliability estimate which simulates + calculating split-half reliability for every possible combination of items. + The Cronbach's α and KR-20 are similar ― KR-20 is a derivative of the Cronbach's α with the advantage that + it can handle both dichotomous and continuous variables, however, KR-20 can't be used + when multiple-choice questions involve partial credit and require systematic item-based analysis. + p.bg-info Kuder–Richardson Formula 20 (KR-20): {{kr20}} + br + p + small + p References: + p 1. + a(href='http://wiki.socr.umich.edu/index.php/SMHS_Cronbachs'). + Scientific Methods for Health Sciences - Instrument Performance Evaluation: Cronbach's α + p. + 2. TSAGRIS, MICHAIL, CONSTANTINOS C. FRANGOS, and CHRISTOS C. FRANGOS. + "Confidence intervals for Cronbach’s reliability coefficient." + p. + 3. Maydeu-Olivares, Alberto, Donna L. Coffman, and Wolfgang M. Hartmann. + "Asymptotically distribution-free (ADF) interval estimation of coefficient alpha." + Psychological methods 12.2 (2007): 157. diff --git a/app/partials/analysis/tools/psychometrics/instrPerfEval/sidebar.jade b/app/partials/analysis/tools/psychometrics/instrPerfEval/sidebar.jade index a2fe336b..6cff0775 100644 --- a/app/partials/analysis/tools/psychometrics/instrPerfEval/sidebar.jade +++ b/app/partials/analysis/tools/psychometrics/instrPerfEval/sidebar.jade @@ -1,6 +1,6 @@ div(ng-controller="instrPerfEvalSidebarCtrl") form - fieldset + fieldset(ng-disabled="!perfeval") legend Data parameters label Number of columns input(ng-model="nCols").input-small diff --git a/app/partials/analysis/wrangleData/wrangler.jade b/app/partials/analysis/wrangleData/wrangler.jade index ef7ba308..6b726e80 100644 --- a/app/partials/analysis/wrangleData/wrangler.jade +++ b/app/partials/analysis/wrangleData/wrangler.jade @@ -1,10 +1,14 @@ -#dt_example(style='height: 400px') - .ui-layout-north#wranglerNorthPanel - #wranglerDashboard - .ui-layout-west#profilerWestPanel - #transformEditor.transformEditor - .ui-layout-center#profilerCenterPanel - #table - #preview - .spacer - .ui-layout-south \ No newline at end of file +div + div.lead.bg-danger(ng-hide="dataType == DATA_TYPES.FLAT") + | Data Wrangler doesn't support current dataset. + | Only "flat" data tables are supported. + #dt_example(style='height: 400px', ng-hide="dataType != DATA_TYPES.FLAT", ng-cloak) + .ui-layout-north#wranglerNorthPanel + #wranglerDashboard + .ui-layout-west#profilerWestPanel + #transformEditor.transformEditor + .ui-layout-center#profilerCenterPanel + #table + #preview + .spacer + .ui-layout-south diff --git a/app/scripts/analysis/getData/getData.coffee b/app/scripts/analysis/getData/getData.coffee index 5de01be0..fd71ede9 100644 --- a/app/scripts/analysis/getData/getData.coffee +++ b/app/scripts/analysis/getData/getData.coffee @@ -54,9 +54,16 @@ getData = angular.module('app_analysis_getData', [ _getMsgList = () -> _msgList + _getSupportedDataTypes = () -> + if _sb + _sb.getSupportedDataTypes() + else + false + getSb: _getSb setSb: _setSb getMsgList: _getMsgList + getSupportedDataTypes: _getSupportedDataTypes ]) # ### @@ -74,6 +81,7 @@ getData = angular.module('app_analysis_getData', [ '$timeout' (manager, $q, $stateParams, $rootScope, $timeout) -> + DATA_TYPES = manager.getSupportedDataTypes() sb = manager.getSb() _data = {} _timer = null @@ -83,12 +91,15 @@ getData = angular.module('app_analysis_getData', [ _data _saveDataToDb = (data, deferred) -> + + msgEnding = if data.dataType is DATA_TYPES.FLAT then ' as 2D data table' else ' as hierarchical object' + $rootScope.$broadcast 'app:push notification', initial: msg: 'Data is being saved in the database...' type: 'alert-info' success: - msg: 'Successfully loaded data into database' + msg: 'Successfully loaded data into database' + msgEnding type: 'alert-success' failure: msg: 'Error in Database' @@ -236,14 +247,25 @@ getData = angular.module('app_analysis_getData', [ 'app_analysis_getData_jsonParser' '$stateParams' 'app_analysis_getData_inputCache' - ($q, $scope, getDataEventMngr, jsonParser, $stateParams, inputCache) -> - # get the sandbox made for this module - # sb = getDataSb.getSb() - # console.log 'sandbox created' + ($q, $scope, eventManager, jsonParser, $stateParams, inputCache) -> $scope.jsonUrl = '' flag = true $scope.selected = null + DATA_TYPES = eventManager.getSupportedDataTypes() + + passReceivedData = (data) -> + if data.dataType is DATA_TYPES.NESTED + inputCache.set data + else + # default data type is 2d 'flat' table + data.dataType = DATA_TYPES.FLAT + # pass a message to update the handsontable div + # data is the formatted data which plugs into the + # handontable. + # TODO: getData module shouldn't know about controllers listening for handsontable update + $scope.$emit 'update handsontable', data + # showGrid $scope.show = (val) -> switch val @@ -255,7 +277,7 @@ getData = angular.module('app_analysis_getData', [ data = default: true purpose: 'json' - $scope.$emit 'update handsontable', data + passReceivedData data $scope.$emit 'change in showStates', 'grid' when 'socrData' @@ -276,7 +298,6 @@ getData = angular.module('app_analysis_getData', [ # getJson $scope.getJson = -> - console.log 123 console.log $scope.jsonUrl if $scope.jsonUrl is '' @@ -290,7 +311,7 @@ getData = angular.module('app_analysis_getData', [ # Pass a message to update the handsontable div. # data is the formatted data which plugs into the # handontable. - $scope.$emit 'update handsontable', data + passReceivedData data $scope.$emit 'get Data from handsontable', inputCache , (msg) -> @@ -308,10 +329,31 @@ getData = angular.module('app_analysis_getData', [ '$scope' 'showState' 'app_analysis_getData_jsonParser' + 'app_analysis_getData_dataAdaptor' + 'app_analysis_getData_inputCache' '$state' - (getDataEventMngr, $scope, showState, jsonParser, state) -> + (eventManager, $scope, showState, jsonParser, dataAdaptor, inputCache, state) -> console.log 'getDataMainCtrl executed' + DATA_TYPES = eventManager.getSupportedDataTypes() + $scope.DATA_TYPES = DATA_TYPES + $scope.dataType = '' + + passReceivedData = (data) -> + if data.dataType is DATA_TYPES.NESTED + $scope.dataType = DATA_TYPES.NESTED + inputCache.set data + else + # default data type is 2d 'flat' table + data.dataType = DATA_TYPES.FLAT + $scope.dataType = DATA_TYPES.FLAT + # pass a message to update the handsontable div + # data is the formatted data which plugs into the + # handontable. + # TODO: getData module shouldn't know about controllers listening for handsontable update + $scope.$emit 'update handsontable', data + + # available SOCR Datasets $scope.socrDatasets = [ id: 'IRIS' @@ -341,14 +383,7 @@ getData = angular.module('app_analysis_getData', [ .then( (data) -> console.log 'resolved' - # Pass a message to update the handsontable div. - # data is the formatted data which plugs into the - # handontable. - - # TODO: getData module shouldn't know about controllers listening for handsontable update - $scope.$emit 'update handsontable', data - # Switch the accordion from getJson to grid. - #$scope.$emit("change in showStates","grid") + passReceivedData data , (msg) -> console.log 'rejected:' + msg @@ -374,34 +409,32 @@ getData = angular.module('app_analysis_getData', [ # handsontable directive to update. purpose: 'json' console.log 'resolved' - # pass a message to update the handsontable div - # data is the formatted data which plugs into the - # handontable. - $scope.$emit 'update handsontable', _data - # switch the accordion from getJson to grid - # $scope.$emit("change in showStates","grid") + passReceivedData _data else console.log 'rejected:' + msg - $scope.getJsonByUrl = -> + $scope.getJsonByUrl = (type) -> d3.json $scope.jsonUrl, (dataResults) -> - if dataResults?.length > 0 - _data = - columnHeader: dataResults.shift() - data: [null, dataResults] - # purpose is helps in pin pointing which - # handsontable directive to update. - purpose: 'json' - console.log 'resolved' - # pass a message to update the handsontable div - # data is the formatted data which plugs into the - # handontable. - $scope.$emit 'update handsontable', _data - # switch the accordion from getJson to grid - # $scope.$emit("change in showStates","grid") + # check that data object is not empty + if dataResults? and Object.keys(dataResults)?.length > 0 + res = dataAdaptor.jsonToFlatTable dataResults + # check if JSON contains "flat data" - 2d array + if res + _data = + columnHeader: if res.length > 1 then res.shift() else [] + data: [null, res] + # purpose is helps in pin pointing which + # handsontable directive to update. + purpose: 'json' + dataType: DATA_TYPES.FLAT + else + _data = + data: dataResults + dataType: DATA_TYPES.NESTED + passReceivedData _data else - console.log 'rejected:' + msg + console.log 'GETDATA: request failed' try _showState = new showState(['grid', 'socrData', 'worldBank', 'generate', 'jsonParse'], $scope) @@ -445,7 +478,24 @@ getData = angular.module('app_analysis_getData', [ # @description: Reformats data from input table format to the universal dataFrame object. # ### .factory('app_analysis_getData_dataAdaptor', [ - () -> + 'app_analysis_getData_manager' + (eventManager) -> + + DATA_TYPES = eventManager.getSupportedDataTypes() + + # https://coffeescript-cookbook.github.io/chapters/arrays/check-type-is-array + typeIsArray = Array.isArray || ( value ) -> return {}.toString.call(value) is '[object Array]' + + haveSameKeys = (obj1, obj2) -> + if Object.keys(obj1).length is Object.keys(obj2).length + res = (k of obj2 for k of obj1) + res.every (e) -> e is true + else + false + + isNumStringArray = (arr) -> + console.log arr + arr.every (el) -> typeof el in ['number', 'string'] # accepts handsontable table as input and returns dataFrame _toDataFrame = (tableData, nSpareCols, nSpareRows) -> @@ -467,26 +517,117 @@ getData = angular.module('app_analysis_getData', [ header: tableData.header nRows: tableData.nRows - nSpareRows nCols: tableData.nCols - nSpareCols + dataType: DATA_TYPES.FLAT _toHandsontable = () -> # TODO: implement for poping up data when coming back from analysis tabs + # tries to convert JSON to 2d flat data table, + # assumes JSON object is not empty - has values, + # returns coverted data or false if not possible + _jsonToFlatTable = (data) -> + # check if JSON contains "flat data" - 2d array + if data? and typeof data is 'object' + if typeIsArray data + # non-empty array + if not (data.every (el) -> typeof el is 'object') + # 1d array of strings or numbers + if (data.every (el) -> typeof el in ['number', 'string']) + data + else + # array of arrays or objects + if (data.every (el) -> typeIsArray el) + # array of arrays + if (data.every (col) -> col.every (el) -> typeof el in ['number', 'string']) + # array of arrays of (numbers or strings) + data + else + # non-string values + false + else + # array of arbitrary objects + # http://stackoverflow.com/a/21266395/1237809 + if (not not data.reduce((prev, next) -> + # check if objects have same keys + if haveSameKeys prev, next + prevValues = Object.keys(prev).map (k) -> prev[k] + nextValues = Object.keys(prev).map (k) -> next[k] + # check that values are numeric/string + if ((prevValues.length is nextValues.length) and + (isNumStringArray prevValues) and + (isNumStringArray nextValues) + ) + next + else NaN + else NaN + )) + # array of objects with the same keys - make them columns + cols = Object.keys data[0] + # reorder values according to keys order + data = (cols.map((col) -> row[col]) for row in data) + # insert keys as a header + data.splice 0, 0, cols + data + else + false + else + # arbitrary object + ks = Object.keys(data) + vals = ks.map (k) -> data[k] + if (vals.every (el) -> typeof el in ['number', 'string']) + # 1d object + data = [ks, vals] + else if (vals.every (el) -> typeof el is 'object') + # object of arrays or objects + if (vals.every (row) -> typeIsArray row) and + (vals.every (row) -> row.every (el) -> typeof el in ['number', 'string']) + # object of arrays + vals = (t[i] for t in vals for i of vals) # transpose + vals.splice 0, 0, ks # add header + vals + else + # object of arbitrary objects + if (not not vals.reduce((prev, next) -> + # check if objects have same keys + if haveSameKeys prev, next + prevValues = Object.keys(prev).map (k) -> prev[k] + nextValues = Object.keys(prev).map (k) -> next[k] + # check that values are + if ((prevValues.length is nextValues.length) and + (isNumStringArray prevValues) and + (isNumStringArray nextValues) + ) + next + else NaN + else NaN + )) + subKs = Object.keys vals[0] + data = ([sk].concat(vals.map((val)-> val[sk])) for sk in subKs) + # insert keys as a header + data.splice 0, 0, [""].concat ks + data + else false + toDataFrame: _toDataFrame toHandsontable: _toHandsontable + jsonToFlatTable: _jsonToFlatTable ]) .directive 'handsontable', [ + 'app_analysis_getData_manager' 'app_analysis_getData_inputCache' 'app_analysis_getData_dataAdaptor' '$exceptionHandler' - (inputCache, dataAdaptor, $exceptionHandler) -> + '$timeout' + (eventManager, inputCache, dataAdaptor, $exceptionHandler, $timeout) -> + restrict: 'E' transclude: true # to the name attribute on the directive element. # the template for the directive. - template: "
" + template: "" #the controller for the directive controller: ($scope) -> @@ -498,84 +639,93 @@ getData = angular.module('app_analysis_getData', [ # It is run before the controller link: (scope, elem, attr) -> - N_SPARE_COLS = 1 - N_SPARE_ROWS = 1 - DEFAULT_ROW_HEIGHT = 24 - - # useful to identify which handsontable instance to update - scope.purpose = attr.purpose - - # retrieves data from handsontable object - _format = (obj) -> - - data = obj.getData() - header = obj.getColHeader() - nCols = obj.countCols() - nRows = obj.countRows() - - table = - data: data - header: header - nCols: nCols - nRows: nRows - - scope.update = (evt, arg) -> - console.log 'handsontable: update called' - - currHeight = elem.height() - - #check if data is in the right format -# if arg? and typeof arg.data is 'object' and typeof arg.columns is 'object' - if arg? and typeof arg.data is 'object' - obj = - data: arg.data[1] -# startRows: Object.keys(arg.data[1]).length -# startCols: arg.columns.length - colHeaders: arg.columnHeader -# columns: arg.columns - minSpareRows: N_SPARE_ROWS - minSpareCols: N_SPARE_COLS - allowInsertRow: true - allowInsertColumn: true - else if arg.default is true - obj = - data: [ - ['Copy', 'paste', 'your', 'data', 'here'] - ] - colHeaders: true - minSpareRows: N_SPARE_ROWS - minSpareCols: N_SPARE_COLS - allowInsertRow: true - allowInsertColumn: true - rowHeaders: false - else - $exceptionHandler - message: 'handsontable configuration is missing' - - obj['change'] = true - obj['afterChange'] = (change, source) -> - # saving data to be globally accessible. - # only place from where data is saved before DB: inputCache. - # onSave, data is picked up from inputCache. - if source is 'loadData' or 'paste' - ht = $(this)[0] - tableData = _format ht - dataFrame = dataAdaptor.toDataFrame tableData, N_SPARE_COLS, N_SPARE_ROWS - inputCache.set dataFrame - ht.updateSettings - height: Math.max currHeight, ht.countRows() * DEFAULT_ROW_HEIGHT + $timeout -> + N_SPARE_COLS = 1 + N_SPARE_ROWS = 1 + # from handsontable defaults + # https://docs.handsontable.com/0.24.1/demo-stretching.html + DEFAULT_ROW_HEIGHT = 23 + DEFAULT_COL_WIDTH = 47 + + # useful to identify which handsontable instance to update + scope.purpose = attr.purpose + + # retrieves data from handsontable object + _format = (obj) -> + data = obj.getData() + header = obj.getColHeader() + nCols = obj.countCols() + nRows = obj.countRows() + + table = + data: data + header: header + nCols: nCols + nRows: nRows + + scope.update = (evt, arg) -> + console.log 'handsontable: update called' + + DATA_TYPES = eventManager.getSupportedDataTypes() + + currHeight = elem.height() + currWidth = elem.width() + + #check if data is in the right format + # if arg? and typeof arg.data is 'object' and typeof arg.columns is 'object' + if arg? and typeof arg.data is 'object' and arg.dataType is DATA_TYPES.FLAT + # TODO: not to pass nested data to ht, but save in db + obj = + data: arg.data[1] + # startRows: Object.keys(arg.data[1]).length + # startCols: arg.columns.length + colHeaders: arg.columnHeader + # columns: arg.columns + minSpareRows: N_SPARE_ROWS + minSpareCols: N_SPARE_COLS + allowInsertRow: true + allowInsertColumn: true + stretchH: "all" + else if arg.default is true + obj = + data: [ + ['Copy', 'paste', 'your', 'data', 'here'] + ] + colHeaders: true + minSpareRows: N_SPARE_ROWS + minSpareCols: N_SPARE_COLS + allowInsertRow: true + allowInsertColumn: true + rowHeaders: false else - inputCache.set source - - try - # hook for pushing data changes to handsontable - # TODO: get rid of tight coupling :-/ - ht = elem.handsontable obj - window['inputCache'] = inputCache.ht = $(ht[0]).data('handsontable') - catch e - $exceptionHandler e - - # subscribing to handsontable update. - scope.$on attr.purpose + ':load data to handsontable', scope.update - console.log 'handsontable directive linked' + $exceptionHandler + message: 'handsontable configuration is missing' + + obj['change'] = true + obj['afterChange'] = (change, source) -> + # saving data to be globally accessible. + # only place from where data is saved before DB: inputCache. + # onSave, data is picked up from inputCache. + if source is 'loadData' or 'paste' + ht = $(this)[0] + tableData = _format ht + dataFrame = dataAdaptor.toDataFrame tableData, N_SPARE_COLS, N_SPARE_ROWS + inputCache.set dataFrame + ht.updateSettings + height: Math.max currHeight, ht.countRows() * DEFAULT_ROW_HEIGHT + width: Math.max currWidth, ht.countCols() * DEFAULT_COL_WIDTH + else + inputCache.set source + + try + # hook for pushing data changes to handsontable + # TODO: get rid of tight coupling :-/ + ht = elem.handsontable obj + window['inputCache'] = inputCache.ht = $(ht[0]).data('handsontable') + catch e + $exceptionHandler e + + # subscribing to handsontable update + scope.$on attr.purpose + ':load data to handsontable', scope.update + console.log 'handsontable directive linked' ] diff --git a/app/scripts/analysis/tools/machineLearning/kMeans/kmeans.coffee b/app/scripts/analysis/tools/machineLearning/kMeans/kmeans.coffee index c840ab5b..ad8f5b05 100644 --- a/app/scripts/analysis/tools/machineLearning/kMeans/kmeans.coffee +++ b/app/scripts/analysis/tools/machineLearning/kMeans/kmeans.coffee @@ -37,6 +37,12 @@ kMeans = angular.module('app_analysis_kMeans', []) _getMsgList = () -> _msgList + _getSupportedDataTypes = () -> + if _sb + _sb.getSupportedDataTypes() + else + false + # wrapper function for controller communications _broadcast = (msg, data) -> $rootScope.$broadcast msg, data @@ -45,6 +51,7 @@ kMeans = angular.module('app_analysis_kMeans', []) setSb: _setSb getMsgList: _getMsgList broadcast: _broadcast + getSupportedDataTypes: _getSupportedDataTypes ]) .controller('kMeansMainCtrl', [ @@ -61,6 +68,8 @@ kMeans = angular.module('app_analysis_kMeans', []) $scope.showresults = off $scope.avgAccuracy = '' $scope.accs = {} + $scope.dataType = '' + $scope.DATA_TYPES = msgManager.getSupportedDataTypes() prettifyArrayOutput = (arr) -> if arr? @@ -90,6 +99,9 @@ kMeans = angular.module('app_analysis_kMeans', []) $scope.$on 'kmeans:updateDataPoints', (event, dataPoints) -> _update dataPoints + $scope.$on 'kmeans:updateDataType', (event, dataType) -> + $scope.dataType = dataType + _finish = (results=null) -> msgManager.broadcast 'kmeans:done', results showResults results @@ -112,6 +124,8 @@ kMeans = angular.module('app_analysis_kMeans', []) (msgManager, kmeans, $scope, $stateParams, $q, $timeout) -> console.log 'kMeansSidebarCtrl executed' + DATA_TYPES = msgManager.getSupportedDataTypes() + DEFAULT_CONTROL_VALUES = k: 2 distance: 'Euclidean' @@ -224,20 +238,26 @@ kMeans = angular.module('app_analysis_kMeans', []) $scope.running = 'hidden' kmeans.run data, $scope.k, $scope.dist, $scope.initMethod + parseData = (data) -> + updateSidebarControls(data) + updateDataPoints(data) + $scope.detectKValue = -> + detectedK = detectKValue data + setDetectedKValue detectedK + $scope.run = -> + _data = parseDataForKMeans data + callKMeans _data + # subscribe for incoming message with data subscribeForData = -> token = sb.subscribe msg: 'take data' msgScope: ['kMeans'] listener: (msg, data) -> - updateSidebarControls(data) - updateDataPoints(data) - $scope.detectKValue = -> - detectedK = detectKValue data - setDetectedKValue detectedK - $scope.run = -> - _data = parseDataForKMeans data - callKMeans _data + if data.dataType? and data.dataType is DATA_TYPES.FLAT + $timeout -> + msgManager.broadcast 'kmeans:updateDataType', data.dataType + parseData data # ask core for data sendDataRequest = (deferred, token) -> diff --git a/app/scripts/analysis/tools/psychometrics/instrPerfEval/instrPerfEval.coffee b/app/scripts/analysis/tools/psychometrics/instrPerfEval/instrPerfEval.coffee index 94c75e28..06eb6e98 100644 --- a/app/scripts/analysis/tools/psychometrics/instrPerfEval/instrPerfEval.coffee +++ b/app/scripts/analysis/tools/psychometrics/instrPerfEval/instrPerfEval.coffee @@ -3,59 +3,77 @@ instrPerfEval = angular.module('app_analysis_instrPerfEval', []) .factory('app_analysis_instrPerfEval_constructor', [ - 'app_analysis_instrPerfEval_manager' - (manager) -> - (sb) -> + 'app_analysis_instrPerfEval_manager' + (manager) -> + (sb) -> - manager.setSb sb unless !sb? - _msgList = manager.getMsgList() + manager.setSb sb unless !sb? + _msgList = manager.getMsgList() - init: (opt) -> - console.log 'instrPerfEval init invoked' + init: (opt) -> + console.log 'instrPerfEval init invoked' - destroy: () -> + destroy: () -> - msgList: _msgList + msgList: _msgList ]) .factory('app_analysis_instrPerfEval_manager', [ - () -> - _sb = null + '$rootScope' + ($rootScope) -> + _sb = null - _msgList = - outgoing: ['get table'] - incoming: ['take table'] - scope: ['instrPerfEval'] + _msgList = + outgoing: ['get table'] + incoming: ['take table'] + scope: ['instrPerfEval'] - _setSb = (sb) -> - _sb = sb + _setSb = (sb) -> + _sb = sb - _getSb = () -> - _sb + _getSb = () -> + _sb - _getMsgList = () -> - _msgList + _getMsgList = () -> + _msgList - getSb: _getSb - setSb: _setSb - getMsgList: _getMsgList - ]) + _getSupportedDataTypes = () -> + if _sb + _sb.getSupportedDataTypes() + else + false + + # wrapper function for controller communications + _broadcast = (msg, data) -> + $rootScope.$broadcast msg, data + + getSb: _getSb + setSb: _setSb + getMsgList: _getMsgList + broadcast: _broadcast + getSupportedDataTypes: _getSupportedDataTypes +]) .controller('instrPerfEvalMainCtrl', [ - 'app_analysis_instrPerfEval_manager' - 'app_analysis_instrPerfEval_alphaCalculator' - '$scope' - (ctrlMngr, alphaCalculator, $scope) -> - console.log 'instrPerfEvalViewMainCtrl executed' + 'app_analysis_instrPerfEval_manager' + 'app_analysis_instrPerfEval_alphaCalculator' + '$scope' + (ctrlMngr, alphaCalculator, $scope) -> + console.log 'instrPerfEvalViewMainCtrl executed' - prettifyArrayOutput = (arr) -> - if arr? - arr = arr.map (x) -> x.toFixed 3 - '[' + arr.toString().split(',').join('; ') + ']' + $scope.DATA_TYPES = ctrlMngr.getSupportedDataTypes() + $scope.dataType = '' - data = alphaCalculator.getAlpha() + prettifyArrayOutput = (arr) -> + if arr? + arr = arr.map (x) -> x.toFixed 3 + '[' + arr.toString().split(',').join('; ') + ']' + + calculateMetrics = () -> + data = alphaCalculator.getAlpha() cAlpha = Number data.cronAlpha + if not isNaN(cAlpha) $scope.cronAlpha = cAlpha.toFixed(3) $scope.cronAlphaIdInterval = prettifyArrayOutput(data.idInterval) @@ -68,44 +86,56 @@ instrPerfEval = angular.module('app_analysis_instrPerfEval', []) $scope.kr20 = if data.kr20 is 'Not a binary data' then data.kr20 else Number(data.kr20).toFixed(3) $scope.splitHalfCoef = Number(data.adjRCorrCoef).toFixed(3) + + $scope.$on 'instrPerfEval:updateDataType', (event, dataType) -> + $scope.dataType = dataType +# if $scope.dataType is 'flat' + + calculateMetrics() ]) .controller('instrPerfEvalSidebarCtrl', [ - 'app_analysis_instrPerfEval_manager' - 'app_analysis_instrPerfEval_alphaCalculator' - '$scope' - '$stateParams' - '$q' - (ctrlMngr, alphaCalculator, $scope, $stateParams, $q) -> - console.log 'instrPerfEvalViewSidebarCtrl executed' - - sb = ctrlMngr.getSb() - - $scope.nCols = '5' - $scope.nRows = '5' - - deferred = $q.defer() - - $scope.confLevel = 0.95 - - # subscribe for incoming message with data - token = sb.subscribe - msg: 'take table' - msgScope: ['instrPerfEval'] - listener: (msg, data) -> - _data = data - $scope.nRows = _data.data?.length - $scope.nCols = _data.data[0]?.length - console.log data - alphaCalculator.calculate data, $scope.confLevel - - sb.publish - msg: 'get table' - msgScope: ['instrPerfEval'] - callback: -> sb.unsubscribe token - data: - tableName: $stateParams.projectId + ':' + $stateParams.forkId - promise: deferred + 'app_analysis_instrPerfEval_manager' + 'app_analysis_instrPerfEval_alphaCalculator' + '$scope' + '$stateParams' + '$q' + '$timeout' + (msgMngr, alphaCalculator, $scope, $stateParams, $q, $timeout) -> + console.log 'instrPerfEvalViewSidebarCtrl executed' + + DATA_TYPES = msgMngr.getSupportedDataTypes() + sb = msgMngr.getSb() + deferred = $q.defer() + + $scope.nCols = '5' + $scope.nRows = '5' + $scope.confLevel = 0.95 + $scope.perfeval = off + + parseData = (obj) -> + $scope.nRows = obj.data?.length + $scope.nCols = obj.data[0]?.length + $scope.perfeval = on + alphaCalculator.calculate obj, $scope.confLevel + + # subscribe for incoming message with data + token = sb.subscribe + msg: 'take table' + msgScope: ['instrPerfEval'] + listener: (msg, data) -> + if data.dataType? and data.dataType is DATA_TYPES.FLAT + $timeout -> + msgMngr.broadcast 'instrPerfEval:updateDataType', data.dataType + parseData data + + sb.publish + msg: 'get table' + msgScope: ['instrPerfEval'] + callback: -> sb.unsubscribe token + data: + tableName: $stateParams.projectId + ':' + $stateParams.forkId + promise: deferred ]) .factory('app_analysis_instrPerfEval_alphaCalculator', [ @@ -326,4 +356,4 @@ instrPerfEval = angular.module('app_analysis_instrPerfEval', []) calculate: _calculate getAlpha: _getAlpha -]) \ No newline at end of file +]) diff --git a/app/scripts/analysis/wrangleData/wrangleData.coffee b/app/scripts/analysis/wrangleData/wrangleData.coffee index 885bfabc..f6b26b3c 100644 --- a/app/scripts/analysis/wrangleData/wrangleData.coffee +++ b/app/scripts/analysis/wrangleData/wrangleData.coffee @@ -42,6 +42,12 @@ wrangleData = angular.module('app_analysis_wrangleData', []) _getMsgList = () -> _msgList + _getSupportedDataTypes = () -> + if _sb + _sb.getSupportedDataTypes() + else + false + # wrapper function for controller communications _broadcast = (msg, data) -> $rootScope.$broadcast msg, data @@ -50,6 +56,7 @@ wrangleData = angular.module('app_analysis_wrangleData', []) setSb: _setSb getMsgList: _getMsgList broadcast: _broadcast + getSupportedDataTypes: _getSupportedDataTypes ]) .factory('app_analysis_wrangleData_dataRetriever', [ @@ -87,7 +94,10 @@ wrangleData = angular.module('app_analysis_wrangleData', []) ]) .factory('app_analysis_wrangleData_dataAdaptor', [ - () -> + 'app_analysis_wrangleData_manager' + (eventManager) -> + + DATA_TYPES = eventManager.getSupportedDataTypes() _toCsvString = (dataFrame) -> @@ -133,6 +143,7 @@ wrangleData = angular.module('app_analysis_wrangleData', []) types: _types nRows: _nRows nCols: _nCols + dataType: DATA_TYPES.FLAT toDvTable: _toDvTable toDataFrame: _toDataFrame @@ -151,17 +162,26 @@ wrangleData = angular.module('app_analysis_wrangleData', []) _initial_transforms = [] _table = [] + _csvData = [] - _start = (viewContainers) -> + DATA_TYPES = manager.getSupportedDataTypes() + + _init = () -> data = dataRetriever.getData() - csvData = dataAdaptor.toCsvString data - _table = _wrangle csvData, viewContainers + if data.dataType is DATA_TYPES.FLAT + _csvData = dataAdaptor.toCsvString data + true + else + false - _wrangle = (csvData, viewContainers) -> + _start = (viewContainers) -> + _table = _wrangle(viewContainers) + + _wrangle = (viewContainers) -> # TODO: abstract from using dv directly #SOCRFW-143 - table = dv.table csvData + table = dv.table _csvData - _initial_transforms = dw.raw_inference(csvData).transforms + _initial_transforms = dw.raw_inference(_csvData).transforms dw.wrangler table: table @@ -193,12 +213,14 @@ wrangleData = angular.module('app_analysis_wrangleData', []) _timer = $timeout ( -> + msgEnding = if dataFrame.dataType is DATA_TYPES.FLAT then ' as 2D data table' else ' as hierarchical object' + $rootScope.$broadcast 'app:push notification', initial: msg: 'Data is being saved in the database...' type: 'alert-info' success: - msg: 'Successfully loaded data into database' + msg: 'Successfully loaded data into database' + msgEnding type: 'alert-success' failure: msg: 'Error in Database' @@ -208,6 +230,7 @@ wrangleData = angular.module('app_analysis_wrangleData', []) ), 1000 true + init: _init start: _start saveData: _saveDataToDb ]) @@ -232,71 +255,83 @@ wrangleData = angular.module('app_analysis_wrangleData', []) 'app_analysis_wrangleData_manager' ($scope, $rootScope, wrangler, msgManager) -> - # TODO: isolate dw from global scope - w = dw.wrangle() + DATA_TYPES = msgManager.getSupportedDataTypes() + $scope.DATA_TYPES = DATA_TYPES + $scope.dataType = '' + + data = wrangler.init() + if data + $scope.dataType = DATA_TYPES.FLAT + + # TODO: isolate dw from global scope + w = dw.wrangle() # listen to state change and save data when exiting Wrangle Data stateListener = $rootScope.$on '$stateChangeStart', (event, toState, toParams, fromState, fromParams) -> if fromState.name? and fromState.name is 'wrangleData' - # save data to db on exit from wrangler - wrangler.saveData() + if $scope.dataType is DATA_TYPES.FLAT + # save data to db on exit from wrangler + wrangler.saveData() # signal to show sidebar msgManager.broadcast 'wrangler:done' # unsubscribe stateListener() - console.log 'wrangleDataMainCtrl executed' + + console.log 'wrangleDataMainCtrl executed' ]) .directive 'datawrangler', [ '$exceptionHandler' 'app_analysis_wrangleData_wrangler' - ($exceptionHandler, wrangler) -> + 'app_analysis_wrangleData_manager' + ($exceptionHandler, wrangler, msgManager) -> restrict: 'E' transclude: true templateUrl: '../partials/analysis/wrangleData/wrangler.html' + replace: true # replace the directive element with the output of the template # the controller for the directive controller: ($scope) -> - myLayout = $('#dt_example').layout - north: - spacing_open: 0 - resizable: false - slidable: false - fxName: 'none' - south: - spacing_open: 0 - resizable: false - slidable: false - fxName: 'none' - west: - minSize: 310 - - container = $('#table') - previewContainer = $('#preview') - transformContainer = $('#transformEditor') - dashboardContainer = $("#wranglerDashboard") - - wrangler.start - tableContainer: container - transformContainer: transformContainer - previewContainer: previewContainer - dashboardContainer: dashboardContainer - - # TODO: find correct programmatic way to invoke header propagation - # assuming there always is a header in data, propagate it in Wrangler - $('#table .odd .rowHeader').first().mouseup().mousedown() - d3.select('div.menu_option.Promote')[0][0].__onmousedown() - $('div.suggestion.selected').click() - - replace: true # replace the directive element with the output of the template - # The link method does the work of setting the directive # up, things like bindings, jquery calls, etc are done in here - # It is run before the controller link: (scope, elem, attr) -> - # useful to identify which handsontable instance to update scope.purpose = attr.purpose + + DATA_TYPES = msgManager.getSupportedDataTypes() + + # check if received dataset is flat + if scope.dataType? and scope.dataType is DATA_TYPES.FLAT + myLayout = $('#dt_example').layout + north: + spacing_open: 0 + resizable: false + slidable: false + fxName: 'none' + south: + spacing_open: 0 + resizable: false + slidable: false + fxName: 'none' + west: + minSize: 310 + + container = $('#table') + previewContainer = $('#preview') + transformContainer = $('#transformEditor') + dashboardContainer = $("#wranglerDashboard") + + wrangler.start + tableContainer: container + transformContainer: transformContainer + previewContainer: previewContainer + dashboardContainer: dashboardContainer + + # TODO: find correct programmatic way to invoke header propagation + # assuming there always is a header in data, propagate it in Wrangler + $('#table .odd .rowHeader').first().mouseup().mousedown() + d3.select('div.menu_option.Promote')[0][0].__onmousedown() + $('div.suggestion.selected').click() ] diff --git a/app/scripts/core/eventMngr.coffee b/app/scripts/core/eventMngr.coffee index 38684dcf..0bc40dda 100644 --- a/app/scripts/core/eventMngr.coffee +++ b/app/scripts/core/eventMngr.coffee @@ -8,18 +8,13 @@ eventMngr = angular.module('app_eventMngr', ['app_mediator', 'app_utils']) (pubSub, utils) -> incomeCallbacks = {} -# _defaultEventManager = (msg, data) -> -# try -# #_data = msgList.income[msg].method.apply null,data -# _data = incomeCallbacks[msg] data -# #last item in data is a promise. -# data[data.length - 1].resolve _data if _data isnt false -# catch e -# console.log e.message -# -# pubSub.publish -# msg: msg -# data: _data + # supported data types + DATA_TYPES = + 'FLAT': 'FLAT' + 'NESTED': 'NESTED' + + _getSupportedDataTypes = () -> + DATA_TYPES # Serialized subscription for a list of events _subscribeForEvents = (events, listener) -> @@ -33,7 +28,8 @@ eventMngr = angular.module('app_eventMngr', ['app_mediator', 'app_utils']) context: events.context subscribeForEvents: _subscribeForEvents + getSupportedDataTypes: _getSupportedDataTypes publish: pubSub.publish subscribe: pubSub.subscribe unsubscribe: pubSub.unsubscribe -]) \ No newline at end of file +]) diff --git a/app/scripts/db/db.coffee b/app/scripts/db/db.coffee index 963004d2..7b05ef73 100644 --- a/app/scripts/db/db.coffee +++ b/app/scripts/db/db.coffee @@ -33,8 +33,7 @@ db.factory 'app_database_constructor', [ ] db.factory 'app_database_manager', [ - 'app_database_handler' - (database) -> + () -> _sb = null #_msgList = # incoming:['create table','get table','delete table'], @@ -48,7 +47,7 @@ db.factory 'app_database_manager', [ _setSb = (sb) -> _sb = sb - database.setSb sb +# database.setSb sb _getSb = -> _sb @@ -56,9 +55,16 @@ db.factory 'app_database_manager', [ _getMsgList = -> _msgList + _getDataTypes = () -> + if _sb + _sb.getSupportedDataTypes() + else + false + getSb: _getSb setSb: _setSb getMsgList: _getMsgList + getSupportedDataTypes: _getDataTypes ] # ### @@ -67,7 +73,10 @@ db.factory 'app_database_manager', [ # @description: Reformats data from the universal dataFrame object to datavore format # ### db.factory 'app_database_dataAdaptor', [ - () -> + 'app_database_manager' + (eventManager) -> + + DATA_TYPES = eventManager.getSupportedDataTypes() _toDvTable = (dataFrame) -> @@ -101,11 +110,26 @@ db.factory 'app_database_dataAdaptor', [ types: _types nRows: _nRows nCols: _nCols + dataType: DATA_TYPES.FLAT toDvTable: _toDvTable toDataFrame: _toDataFrame ] +db.factory 'app_database_nested', [ + () -> + _nestedObj = null + + _save = (obj) -> + _nestedObj = obj + + _get = () -> + _nestedObj + + save: _save + get: _get +] + db.service 'app_database_dv', -> # contains references to all the tables created. @@ -261,61 +285,106 @@ db.service 'app_database_dv', -> db.factory 'app_database_handler', [ '$q' + '$timeout' 'app_database_dv' + 'app_database_nested' 'app_database_dataAdaptor' - ($q, _db, dataAdaptor) -> - - # set all the callbacks here. - _setSb = ((_db) -> - window.db = _db - (sb) -> - - #registering database callbacks for all possible incoming messages. - # TODO: add wrapper layer on top of _db methods? - _methods = [ - {incoming: 'save table', outgoing: 'table saved', event: _db.create} - {incoming: 'get table', outgoing: 'take table', event: _db.get} - {incoming: 'add listener', outgoing: 'listener added', event: _db.addListener} - ] - - _status = _methods.map (method) -> - sb.subscribe - msg: method['incoming'] - msgScope: ['database'] - listener: (msg, data) -> - console.log "%cDATABASE: listener called for"+msg , "color:green" - - # convert from the universal dataFrame object to datavore table - dvTableData = if msg is 'save table' then dataAdaptor.toDvTable data.dataFrame else data - - # arrange arguments for a callback - # @todo need to find a better way for this. - _data = switch - when msg is 'save table' then [ dvTableData, data.tableName ] - when msg is 'get table' then [ data.tableName ] - else data - - # invoke callback - _data = method.event.apply null, _data - - # convert data to DataFrame if returning it - _data = dataAdaptor.toDataFrame _data if msg is 'get table' - - # all publish calls should pass a promise in the data object - # if promise is not defined, create one and pass it along - deferred = data.promise - if typeof deferred isnt 'undefined' - if _data isnt false then deferred.resolve() else deferred.reject() - else - _data.promise = $q.defer() - - console.log '%cDATABASE: listener response: ' + _data, 'color:green' - - sb.publish - msg: method.outgoing - data: _data - msgScope: ['database'] - )(_db) + 'app_database_manager' + ($q, $timeout, _db, nestedDb, dataAdaptor, eventManager) -> + + sb = null + DATA_TYPES = null + _lastDataType = '' + + _getLastDataType = () -> + _lastDataType + + _saveData = (obj) -> + if obj.dataFrame? + dataFrame = obj.dataFrame + # convert from the universal dataFrame object to datavore table or keep as is + if dataFrame.dataType? + _lastDataType = dataFrame.dataType + switch dataFrame.dataType + when DATA_TYPES.FLAT + dvData = dataAdaptor.toDvTable dataFrame + res = _db.create dvData, obj.tableName + res + when DATA_TYPES.NESTED + nestedDb.save obj.data + true + else console.log '%cDATABASE: data type is unknown' , 'color:green' + else console.log '%cDATABASE: data type is unknown' , 'color:green' + else console.log '%cDATABASE: nothing to save' , 'color:green' + + _getData = (data) -> + switch _lastDataType + when DATA_TYPES.FLAT + _data = _db.get data.tableName + # convert data to DataFrame if returning it + _data = dataAdaptor.toDataFrame _data + _data.dataType = DATA_TYPES.FLAT + _data + when DATA_TYPES.NESTED + _data = nestedDb.get() + _data = + data: _data + dataType: DATA_TYPES.NESTED + else console.log '%cDATABASE: data type is unknown' , 'color:green' + + _setDbListeners = () -> + # registering database callbacks for all possible incoming messages + # TODO: add wrapper layer on top of _db methods? + _methods = [ + incoming: 'save table' + outgoing: 'table saved' + event: _saveData + , + incoming: 'get table' + outgoing: 'take table' + event: _getData + , + incoming: 'add listener' + outgoing: 'listener added' + event: _db.addListener + ] + + _status = _methods.map (method) -> + sb.subscribe + msg: method['incoming'] + msgScope: ['database'] + listener: (msg, obj) -> + console.log "%cDATABASE: listener called for" + msg , "color:green" + # invoke callback + _data = method.event.apply null, [obj] + + # all publish calls should pass a promise in the data object + # if promise is not defined, create one and pass it along + deferred = obj.promise + if typeof deferred isnt 'undefined' + if _data isnt false then deferred.resolve() else deferred.reject() + else + _data.promise = $q.defer() + + console.log '%cDATABASE: listener response: ' + _data, 'color:green' + + sb.publish + msg: method['outgoing'] + data: _data + msgScope: ['database'] + + _initDb = () -> + $timeout -> + window.db = _db + sb = eventManager.getSb() + DATA_TYPES = eventManager.getSupportedDataTypes() + _setDbListeners() + + initDb: _initDb +] - setSb: _setSb - ] +db.run [ + 'app_database_handler' + (handler) -> + handler.initDb() +] diff --git a/app/styles/custom.less b/app/styles/custom.less index ad9284b4..024f3706 100644 --- a/app/styles/custom.less +++ b/app/styles/custom.less @@ -21,6 +21,10 @@ filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=3); } +.vis-hidden{ + visibility: hidden; +} + #welcome-screen { height: 100%; width: 100%; diff --git a/bower.json b/bower.json index 73270917..17257b9b 100644 --- a/bower.json +++ b/bower.json @@ -1,7 +1,7 @@ { "author": "Alexandr Kalinin