angular.module('NodeForm.templates', []).run(['$templateCache', function($templateCache) { "use strict"; $templateCache.put("modules/core/views/header.client.view.html", "
"); $templateCache.put("modules/core/views/home.client.view.html", "

TellForm

Craft beautiful forms in seconds.

Craft beautiful forms.

TellForm is an opensource alternative to TypeForm that can create stunning forms from PDFs or from scratch

TellForm is an opensource alternative to TypeForm that can create stunning forms from PDFs or from scratch

Create your next ______.

Tell a story with a form.

"); $templateCache.put("modules/forms/admin/views/admin-form.client.view.html", "

"); $templateCache.put("modules/forms/admin/views/list-forms.client.view.html", "

Create a new form
Name
Language

Created on
"); $templateCache.put("modules/forms/base/views/submit-form.client.view.html", "
"); $templateCache.put("modules/forms/admin/views/adminTabs/analyze.html", ""); $templateCache.put("modules/forms/admin/views/adminTabs/configure.html", ""); $templateCache.put("modules/forms/admin/views/adminTabs/create.html", ""); $templateCache.put("modules/forms/admin/views/adminTabs/design.html", "

Change how your Form Looks

Change how your Form Looks

Background Color
Question Text Color
Answer Text Color
Button Background Color
Button Text Color
"); $templateCache.put("modules/forms/admin/views/directiveViews/cgBusy/update-form-message-TypeA.html", "
{{$message}}
"); $templateCache.put("modules/forms/admin/views/directiveViews/cgBusy/update-form-message-TypeB.html", "
{{$message}}
"); $templateCache.put("modules/forms/admin/views/directiveViews/form/configure-form.client.view.html", "

PDF Generation/EMR

PDF Generation/EMR

Save Submissions as PDFs?
Upload Your PDF Template
{{myform.pdf.name}}
Upload your PDF
Autogenerate Form?
Use Oscarhost API?
Oscarhost API Username
Oscarhost API Password
Oscarhost API URL
Oscarhost API Update Type


Advanced Settings

Advanced Settings

Form Name
Form Status
Google Analytics Tracking Code
Language
* required
Display Form Footer?
Display Start Page?
"); $templateCache.put("modules/forms/admin/views/directiveViews/form/edit-form.client.view.html", "

Click to Add New Field

Add New Field

Add Field

Start Page

Preview Start Page

    {{myform.startPage.introTitle}}

    {{myform.startPage.introParagraph}}

Edit Start Page


Intro Title:
Intro Paragraph:
\n" + "
\n" + "\n" + "

\n" + "
\n" + "
Options:
\n" + "
\n" + "
\n" + " \n" + "\n" + " \n" + " \n" + " \n" + "
\n" + "
\n" + " \n" + "
\n" + "
\n" + "
\n" + "\n" + "\n" + "

\n" + "
\n" + "
Number of Steps:
\n" + "
\n" + " \n" + "
\n" + "
\n" + "
Shape:
\n" + "
\n" + " \n" + "
\n" + "
\n" + "\n" + "

\n" + "\n" + "
\n" + "
Required:
\n" + "
\n" + " \n" + "\n" + " \n" + "
\n" + "
\n" + "\n" + "
\n" + "
Disabled:
\n" + "
\n" + " \n" + "\n" + " \n" + "
\n" + "
\n" + "\n" + "
\n" + " \n" + "\n" + "
\n" + "
\n" + "

\n" + " Click on Fields to add them here\n" + "

\n" + "
\n" + "
\n" + "\n" + "
\n" + " \n" + "
\n" + "\n" + "
\n" + "
\n" + "
\n" + "
\n" + " \n" + " \n" + " \n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + " \n" + " \n" + " \n" + "
\n" + "
\n" + "
\n" + "
\n" + "\n" + "
\n" + "
\n" + "
\n" + " \n" + " \n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "\n" + ""); $templateCache.put("modules/forms/admin/views/directiveViews/form/edit-submissions-form.client.view.html", "
Total Views: {{myform.analytics.views}}
Submissions: {{myform.analytics.submissions}}
Conversion Rate: {{myform.analytics.conversionRate}}%

Field Title
Field Views
User dropoff rate at this field
{{fieldStats.field.title}}
{{fieldStats.totalViews}}
{{fieldStats.dropoffRate}}%

#{{value.title}}OscarEMR User ProfilePercentage CompleteTime ElapsedDeviceLocationIP AddressDate Submitted (UTC)Generated PDF
{{$index+1}}{{field.fieldValue}}User Profile #{{row.oscarDemoNum}}{{row.percentageComplete}}%{{row.timeElapsed}}{{row.device.name}}, {{row.device.type}}{{row.geoLocation.city}}, {{row.geoLocation.country}}{{row.ipAddr}}{{row.created | date:'yyyy-MM-dd HH:mm:ss'}}Generated PDF
"); $templateCache.put("modules/forms/base/views/directiveViews/entryPage/startPage.html", "

{{pageData.introTitle}}

{{pageData.introParagraph}}

"); $templateCache.put("modules/forms/base/views/directiveViews/field/date.html", "

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{field.description}}

"); $templateCache.put("modules/forms/base/views/directiveViews/field/dropdown.html", "
0\">

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{field.description}}


"); $templateCache.put("modules/forms/base/views/directiveViews/field/file.html", "

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{field.file.originalname}}
{{ UPLOAD_FILE | translate }}
"); $templateCache.put("modules/forms/base/views/directiveViews/field/hidden.html", ""); $templateCache.put("modules/forms/base/views/directiveViews/field/legal.html", "

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}


{{field.description}}


"); $templateCache.put("modules/forms/base/views/directiveViews/field/radio.html", "
0\">

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{field.description}}


"); $templateCache.put("modules/forms/base/views/directiveViews/field/rating.html", "

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{field.description}}

"); $templateCache.put("modules/forms/base/views/directiveViews/field/statement.html", "

{{field.title}}

{{field.description}}

{{field.description}}


"); $templateCache.put("modules/forms/base/views/directiveViews/field/textarea.html", "

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{ 'NEWLINE' | translate }}

{{field.description}}

Press SHIFT+ENTER to add a newline
{{ 'ENTER' | translate }}
"); $templateCache.put("modules/forms/base/views/directiveViews/field/textfield.html", "

{{index+1}} {{field.title}} ({{ 'OPTIONAL' | translate }})

{{field.description}}

{{ 'ENTER' | translate }}
"); $templateCache.put("modules/forms/base/views/directiveViews/field/yes_no.html", "

{{index+1}} {{field.title}} {{ 'OPTIONAL' | translate }}

{{field.description}}


"); $templateCache.put("modules/forms/base/views/directiveViews/form/submit-form.client.view.html", "
{{ 'COMPLETING_NEEDED' | translate:translateAdvancementData }}
{{ 'ENTER' | translate }}

{{ 'ADVANCEMENT' | translate:translateAdvancementData }}

"); $templateCache.put("modules/users/views/authentication/access-denied.client.view.html", "

You need to be logged in to access this page

Login
"); $templateCache.put("modules/users/views/authentication/signin.client.view.html", "

Sign into your account

Error:
  or  Sign up
"); $templateCache.put("modules/users/views/authentication/signup-success.client.view.html", "

Signup Successful

You've successfully registered an account at TellForm.

But your account is not activated yet



Before you continue, make sure to check your email for our verification. If you don't receive it within 24h drop us a line at polydaic@gmail.com

"); $templateCache.put("modules/users/views/authentication/signup.client.view.html", "

Signup with your email

Couldn't submit form due to errors:

"); $templateCache.put("modules/users/views/password/forgot-password.client.view.html", "

Restore your password

Enter your account email.

{{error}}
{{success}}
"); $templateCache.put("modules/users/views/password/reset-password-invalid.client.view.html", "

Password reset is invalid

Ask for a new password reset
"); $templateCache.put("modules/users/views/password/reset-password-success.client.view.html", "

Password successfully reset

Continue to home page
"); $templateCache.put("modules/users/views/password/reset-password.client.view.html", "

Reset your password

{{error}}
{{success}}
"); $templateCache.put("modules/users/views/settings/change-password.client.view.html", "

Change your password


Password Changed Successfully
"); $templateCache.put("modules/users/views/settings/edit-profile.client.view.html", "

Edit your profile

Profile Saved Successfully
Couldn't Save Your Profile.
Error:
First Name
Last Name

Language
Email (also your username)
"); $templateCache.put("modules/users/views/settings/social-accounts.client.view.html", "

Connected social accounts:

Connect other social accounts:

"); $templateCache.put("modules/users/views/verify/resend-verify-email.client.view.html", "

Resend your account verification email

Enter your account email.

{{error}}

Verification Email has been Sent

A verification email has been sent to {{username}}.
But your account is still not activated yet

Check your email and click on the activation link to activate your account. If you have any questions drop us a line at polydaic@gmail.com

"); $templateCache.put("modules/users/views/verify/verify-account.client.view.html", "

Account successfuly activated

Continue to login page

Verification link is invalid or has expired

Resend your verification email Signin to your account
"); }]); 'use strict'; // Init the application configuration module for AngularJS application var ApplicationConfiguration = (function() { // Init module configuration options var applicationModuleName = 'NodeForm'; var applicationModuleVendorDependencies = ['duScroll', 'ui.select', 'cgBusy', 'ngSanitize', 'vButton', 'ngResource', 'NodeForm.templates', 'ui.router', 'ui.bootstrap', 'ui.utils']; // Add a new vertical module var registerModule = function(moduleName, dependencies) { // Create angular module angular.module(moduleName, dependencies || []); // Add the module to the AngularJS configuration file angular.module(applicationModuleName).requires.push(moduleName); }; return { applicationModuleName: applicationModuleName, applicationModuleVendorDependencies: applicationModuleVendorDependencies, registerModule: registerModule }; })(); 'use strict'; //Start by defining the main module and adding the module dependencies angular.module(ApplicationConfiguration.applicationModuleName, ApplicationConfiguration.applicationModuleVendorDependencies); // Setting HTML5 Location Mode angular.module(ApplicationConfiguration.applicationModuleName).config(['$locationProvider', function($locationProvider) { $locationProvider.hashPrefix('!'); } ]); //Permission Constants angular.module(ApplicationConfiguration.applicationModuleName).constant('APP_PERMISSIONS', { viewAdminSettings: 'viewAdminSettings', editAdminSettings: 'editAdminSettings', editForm: 'editForm', viewPrivateForm: 'viewPrivateForm' }); //User Role constants angular.module(ApplicationConfiguration.applicationModuleName).constant('USER_ROLES', { admin: 'admin', normal: 'user', superuser: 'superuser' }); //form url angular.module(ApplicationConfiguration.applicationModuleName).constant('FORM_URL', '/forms/:formId'); angular.module(ApplicationConfiguration.applicationModuleName).run(['$rootScope', 'Auth', '$state', '$stateParams', function($rootScope, Auth, $state, $stateParams) { $rootScope.$state = $state; $rootScope.$stateParams = $stateParams; // add previous state property $rootScope.$on('$stateChangeSuccess', function (event, toState, toParams, fromState) { $state.previous = fromState; //console.log('toState: '+toState.name); var statesToIgnore = ['home', 'signin', 'resendVerifyEmail', 'verify', 'signup', 'signup-success', 'forgot', 'reset-invalid', 'reset', 'reset-success']; //Redirect to listForms if user is authenticated if(statesToIgnore.indexOf(toState.name) > 0){ if(Auth.isAuthenticated()){ event.preventDefault(); // stop current execution //console.log('go to forms'); $state.go('listForms'); // go to listForms page } } //Redirect to 'signup' route if user is not authenticated else if(toState.name !== 'access_denied' && !Auth.isAuthenticated() && toState.name !== 'submitForm'){ console.log('go to signup'); event.preventDefault(); // stop current execution $state.go('listForms'); // go to listForms page } }); } ]); //Page access/authorization logic angular.module(ApplicationConfiguration.applicationModuleName).run(['$rootScope', 'Auth', 'User', 'Authorizer', '$state', '$stateParams', function($rootScope, Auth, User, Authorizer, $state, $stateParams) { $rootScope.$on('$stateChangeStart', function(event, next) { var authenticator, permissions, user; permissions = next && next.data && next.data.permissions ? next.data.permissions : null; Auth.ensureHasCurrentUser(User); user = Auth.currentUser; if(user){ authenticator = new Authorizer(user); //console.log('access denied: '+!authenticator.canAccess(permissions)); //console.log(permissions); if( (permissions != null) ){ if( !authenticator.canAccess(permissions) ){ event.preventDefault(); //console.log('access denied'); $state.go('access_denied'); } } } }); }]); //Then define the init function for starting up the application angular.element(document).ready(function() { //Fixing facebook bug with redirect if (window.location.hash === '#_=_') window.location.hash = '#!'; //Then init the app angular.bootstrap(document, [ApplicationConfiguration.applicationModuleName]); }); angular.module('NodeForm.templates', []).run(['$templateCache', function($templateCache) { "use strict"; $templateCache.put("../public/modules/core/views/header.client.view.html", "
"); $templateCache.put("../public/modules/core/views/home.client.view.html", "

TellForm

Craft beautiful forms in seconds.

Craft beautiful forms.

TellForm is an opensource alternative to TypeForm that can create stunning forms from PDFs or from scratch

TellForm is an opensource alternative to TypeForm that can create stunning forms from PDFs or from scratch

Create your next ______.

Tell a story with a form.

"); $templateCache.put("../public/modules/forms/views/admin-form.client.view.html", "
"); $templateCache.put("../public/modules/forms/views/list-forms.client.view.html", "

Create a new form
Name
Language

Created on
"); $templateCache.put("../public/modules/forms/views/submit-form.client.view.html", "
"); $templateCache.put("../public/modules/forms/views/adminTabs/analyze.html", ""); $templateCache.put("../public/modules/forms/views/adminTabs/configure.html", ""); $templateCache.put("../public/modules/forms/views/adminTabs/create.html", ""); $templateCache.put("../public/modules/forms/views/adminTabs/design.html", "

Change how your Form Looks

Change how your Form Looks

Background Color
Question Text Color
Answer Text Color
Button Background Color
Button Text Color
"); $templateCache.put("../public/modules/forms/views/directiveViews/cgBusy/update-form-message-TypeA.html", "
{{$message}}
"); $templateCache.put("../public/modules/forms/views/directiveViews/cgBusy/update-form-message-TypeB.html", "
{{$message}}
"); $templateCache.put("../public/modules/forms/views/directiveViews/entryPage/startPage.html", "

{{pageData.introTitle}}

{{pageData.introParagraph}}

"); $templateCache.put("../public/modules/forms/views/directiveViews/field/date.html", "

{{field.title}} *(required)

"); $templateCache.put("../public/modules/forms/views/directiveViews/field/dropdown.html", "
0\">

{{field.title}} *(required)

{{$select.selected.option_value}}

"); $templateCache.put("../public/modules/forms/views/directiveViews/field/file.html", "
{{field.title}} (* required)
{{field.file.originalname}}
Upload your File
"); $templateCache.put("../public/modules/forms/views/directiveViews/field/hidden.html", ""); $templateCache.put("../public/modules/forms/views/directiveViews/field/legal.html", "

{{field.title}} *(required)


{{field.description}}


"); $templateCache.put("../public/modules/forms/views/directiveViews/field/natural.html", "

{{field.title}} *(required)


"); $templateCache.put("../public/modules/forms/views/directiveViews/field/password.html", "

{{field.title}} *(required)

"); $templateCache.put("../public/modules/forms/views/directiveViews/field/radio.html", "
0\">

{{field.title}} *(required)


"); $templateCache.put("../public/modules/forms/views/directiveViews/field/rating.html", "

{{field.title}} *(required)

"); $templateCache.put("../public/modules/forms/views/directiveViews/field/statement.html", "

{{field.title}}

{{field.description}}


"); $templateCache.put("../public/modules/forms/views/directiveViews/field/textarea.html", "

{{field.title}} *(required)

press ENTER
"); $templateCache.put("../public/modules/forms/views/directiveViews/field/textfield.html", "

{{field.title}} *(required)

press ENTER
"); $templateCache.put("../public/modules/forms/views/directiveViews/field/yes_no.html", "

{{field.title}} *(required)

{{field.description}}


"); $templateCache.put("../public/modules/forms/views/directiveViews/form/configure-form.client.view.html", "

PDF Generation/EMR

PDF Generation/EMR

Save Submissions as PDFs?
Upload Your PDF Template
{{myform.pdf.originalname}}
Upload your PDF
Autogenerate Form?
Use Oscarhost API?
Oscarhost API Username
Oscarhost API Password
Oscarhost API URL
Oscarhost API Update Type


Advanced Settings

Advanced Settings

Form Name
Form Status
Language
* required
Display Form Footer?
Display Start Page?
"); $templateCache.put("../public/modules/forms/views/directiveViews/form/edit-form.client.view.html", "

Click to Add New Field

Add New Field

Add Field

Start Page

Preview Start Page

    {{myform.startPage.introTitle}}

    {{myform.startPage.introParagraph}}

Edit Start Page


Intro Title:
Intro Paragraph:
\n" + "
\n" + "\n" + "

\n" + "
\n" + "
Options:
\n" + "
\n" + "
\n" + " \n" + "\n" + " \n" + " \n" + " \n" + "
\n" + "
\n" + " \n" + "
\n" + "
\n" + "
\n" + "\n" + "

\n" + "\n" + "
\n" + "
Required:
\n" + "
\n" + " \n" + "\n" + " \n" + "
\n" + "
\n" + "\n" + "
\n" + "
Disabled:
\n" + "
\n" + " \n" + "\n" + " \n" + "
\n" + "
\n" + "\n" + "
\n" + " \n" + "\n" + "
\n" + "
\n" + "

\n" + " Click on Fields to add them here\n" + "

\n" + "
\n" + "
\n" + "\n" + "
\n" + " \n" + "
\n" + "\n" + "
\n" + "
\n" + "
\n" + "
\n" + " \n" + " \n" + " \n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + " \n" + " \n" + " \n" + "
\n" + "
\n" + "
\n" + "
\n" + "\n" + "
\n" + "
\n" + "
\n" + " \n" + " \n" + "
\n" + "
\n" + "
\n" + "
\n" + "
\n" + "\n" + ""); $templateCache.put("../public/modules/forms/views/directiveViews/form/edit-submissions-form.client.view.html", "
#{{value.title}}OscarEMR User ProfilePercentage CompleteTime ElapsedDeviceLocationIP AddressDate Submitted (UTC)Generated PDF
{{$index+1}}{{field.fieldValue}}User Profile #{{row.oscarDemoNum}}{{row.percentageComplete}}%{{row.timeElapsed}}{{row.device.name}}, {{row.device.type}}{{row.geoLocation.city}}, {{row.geoLocation.country}}{{row.ipAddr}}{{row.created | date:'yyyy-MM-dd HH:mm:ss'}}Generated PDF
"); $templateCache.put("../public/modules/forms/views/directiveViews/form/submit-form.client.view.html", "
press ENTER

{{myform | formValidity}} out of {{form_fields_count}} answered

"); $templateCache.put("../public/modules/users/views/authentication/access-denied.client.view.html", "

You need to be logged in to access this page

Login
"); $templateCache.put("../public/modules/users/views/authentication/signin.client.view.html", "

Sign into your account

Error:
  or  Sign up
"); $templateCache.put("../public/modules/users/views/authentication/signup-success.client.view.html", "

Signup Successful

You've successfully registered an account at TellForm.

But your account is not activated yet



Before you continue, make sure to check your email for our verification. If you don't receive it within 24h drop us a line at hi@TellForm.com

"); $templateCache.put("../public/modules/users/views/authentication/signup.client.view.html", "

Signup with your email

Couldn't submit form due to errors:

"); $templateCache.put("../public/modules/users/views/password/forgot-password.client.view.html", "

Restore your password

Enter your account email.

{{error}}
{{success}}
"); $templateCache.put("../public/modules/users/views/password/reset-password-invalid.client.view.html", "

Password reset is invalid

Ask for a new password reset
"); $templateCache.put("../public/modules/users/views/password/reset-password-success.client.view.html", "

Password successfully reset

Continue to home page
"); $templateCache.put("../public/modules/users/views/password/reset-password.client.view.html", "

Reset your password

{{error}}
{{success}}
"); $templateCache.put("../public/modules/users/views/settings/change-password.client.view.html", "

Change your password


Password Changed Successfully
"); $templateCache.put("../public/modules/users/views/settings/edit-profile.client.view.html", "

Edit your profile

Profile Saved Successfully
Couldn't Save Your Profile.
Error:
First Name
Last Name

Language
Email (also your username)
"); $templateCache.put("../public/modules/users/views/settings/social-accounts.client.view.html", "

Connected social accounts:

Connect other social accounts:

"); $templateCache.put("../public/modules/users/views/verify/resend-verify-email.client.view.html", "

Resend your account verification email

Enter your account email.

{{error}}

Verification Email has been Sent

A verification email has been sent to {{username}}.
But your account is still not activated yet

Check your email and click on the activation link to activate your account. If you have any questions drop us a line at hi@TellForm.com

"); $templateCache.put("../public/modules/users/views/verify/verify-account.client.view.html", "

Account successfuly activated

Continue to login page

Verification link is invalid or has expired

Resend your verification email Signin to your account
"); }]); 'use strict'; // Use Application configuration module to register a new module ApplicationConfiguration.registerModule('core', ['users']); 'use strict'; // Use Application configuration module to register a new module ApplicationConfiguration.registerModule('forms', [ 'ngFileUpload', 'ui.router.tabs', 'ui.date', 'ui.sortable', 'angular-input-stars', 'users', 'pascalprecht.translate' ]);//, 'colorpicker.module' @TODO reactivate this module 'use strict'; // Use Application configuration module to register a new module ApplicationConfiguration.registerModule('users'); 'use strict'; // Setting up route angular.module('core').config(['$stateProvider', '$urlRouterProvider', function($stateProvider, $urlRouterProvider, Authorization) { // Redirect to home view when route not found $urlRouterProvider.otherwise('/forms'); } ]); 'use strict'; angular.module('core').controller('HeaderController', ['$rootScope', '$scope', 'Menus', '$state', 'Auth', 'User', '$window', function ($rootScope, $scope, Menus, $state, Auth, User, $window) { $rootScope.signupDisabled = $window.signupDisabled; $scope.user = $rootScope.user = Auth.ensureHasCurrentUser(User); $scope.authentication = $rootScope.authentication = Auth; $rootScope.languages = $scope.languages = ['english', 'french', 'spanish']; $scope.isCollapsed = false; $rootScope.hideNav = false; $scope.menu = Menus.getMenu('topbar'); $scope.signout = function() { var promise = User.logout(); promise.then(function() { Auth.logout(); Auth.ensureHasCurrentUser(User); $scope.user = $rootScope.user = null; $state.go('listForms'); }, function(reason) { console.log('Logout Failed: ' + reason); }); }; $scope.toggleCollapsibleMenu = function() { $scope.isCollapsed = !$scope.isCollapsed; }; // Collapsing the menu after navigation $scope.$on('$stateChangeSuccess', function(event, toState, toParams, fromState, fromParams) { $scope.isCollapsed = false; $rootScope.hideNav = false; if ( angular.isDefined( toState.data ) ) { if ( angular.isDefined( toState.data.hideNav ) ) { $rootScope.hideNav = toState.data.hideNav; } } }); } ]); 'use strict'; angular.module('core').controller('HomeController', ['$rootScope', '$scope', 'User', '$state', function($rootScope, $scope, User, $state) { $scope = $rootScope; } ]); 'use strict'; //Menu service used for managing menus angular.module('core').service('Menus', [ function() { // Define a set of default roles this.defaultRoles = ['*']; // Define the menus object this.menus = {}; // A private function for rendering decision var shouldRender = function(user) { if (user) { if (~this.roles.indexOf('*')) { return true; } else { for (var userRoleIndex in user.roles) { for (var roleIndex in this.roles) { console.log(this.roles[roleIndex]); console.log( this.roles[roleIndex] === user.roles[userRoleIndex]); if (this.roles[roleIndex] === user.roles[userRoleIndex]) { return true; } } } } } else { return this.isPublic; } return false; }; // Validate menu existance this.validateMenuExistance = function(menuId) { if (menuId && menuId.length) { if (this.menus[menuId]) { return true; } else { throw new Error('Menu does not exists'); } } else { throw new Error('MenuId was not provided'); } return false; }; // Get the menu object by menu id this.getMenu = function(menuId) { // Validate that the menu exists this.validateMenuExistance(menuId); // Return the menu object return this.menus[menuId]; }; // Add new menu object by menu id this.addMenu = function(menuId, isPublic, roles) { // Create the new menu this.menus[menuId] = { isPublic: isPublic || false, roles: roles || this.defaultRoles, items: [], shouldRender: shouldRender }; // Return the menu object return this.menus[menuId]; }; // Remove existing menu object by menu id this.removeMenu = function(menuId) { // Validate that the menu exists this.validateMenuExistance(menuId); // Return the menu object delete this.menus[menuId]; }; // Add menu item object this.addMenuItem = function(menuId, menuItemTitle, menuItemURL, menuItemType, menuItemUIRoute, isPublic, roles, position) { // Validate that the menu exists this.validateMenuExistance(menuId); // Push new menu item this.menus[menuId].items.push({ title: menuItemTitle, link: menuItemURL, menuItemType: menuItemType || 'item', menuItemClass: menuItemType, uiRoute: menuItemUIRoute || ('/' + menuItemURL), isPublic: ((isPublic === null || typeof isPublic === 'undefined') ? this.menus[menuId].isPublic : isPublic), roles: ((roles === null || typeof roles === 'undefined') ? this.menus[menuId].roles : roles), position: position || 0, items: [], shouldRender: shouldRender }); // Return the menu object return this.menus[menuId]; }; // Add submenu item object this.addSubMenuItem = function(menuId, rootMenuItemURL, menuItemTitle, menuItemURL, menuItemUIRoute, isPublic, roles, position) { // Validate that the menu exists this.validateMenuExistance(menuId); // Search for menu item for (var itemIndex in this.menus[menuId].items) { if (this.menus[menuId].items[itemIndex].link === rootMenuItemURL) { // Push new submenu item this.menus[menuId].items[itemIndex].items.push({ title: menuItemTitle, link: menuItemURL, uiRoute: menuItemUIRoute || ('/' + menuItemURL), isPublic: ((isPublic === null || typeof isPublic === 'undefined') ? this.menus[menuId].items[itemIndex].isPublic : isPublic), roles: ((roles === null || typeof roles === 'undefined') ? this.menus[menuId].items[itemIndex].roles : roles), position: position || 0, shouldRender: shouldRender }); } } // Return the menu object return this.menus[menuId]; }; // Remove existing menu object by menu id this.removeMenuItem = function(menuId, menuItemURL) { // Validate that the menu exists this.validateMenuExistance(menuId); // Search for menu item to remove for (var itemIndex in this.menus[menuId].items) { if (this.menus[menuId].items[itemIndex].link === menuItemURL) { this.menus[menuId].items.splice(itemIndex, 1); } } // Return the menu object return this.menus[menuId]; }; // Remove existing menu object by menu id this.removeSubMenuItem = function(menuId, submenuItemURL) { // Validate that the menu exists this.validateMenuExistance(menuId); // Search for menu item to remove for (var itemIndex in this.menus[menuId].items) { for (var subitemIndex in this.menus[menuId].items[itemIndex].items) { if (this.menus[menuId].items[itemIndex].items[subitemIndex].link === submenuItemURL) { this.menus[menuId].items[itemIndex].items.splice(subitemIndex, 1); } } } // Return the menu object return this.menus[menuId]; }; //Adding the topbar menu this.addMenu('topbar', false, ['*']); //Adding the bottombar menu for the Form-Footer view this.addMenu('bottombar', false, ['*']); } ]); (function () { 'use strict'; // Create the Socket.io wrapper service angular .module('core') .factory('Socket', Socket); Socket.$inject = ['$timeout', '$window']; function Socket($timeout, $window) { var service = { connect: connect, emit: emit, on: on, removeListener: removeListener, socket: null }; connect(window.location.protocol+'//'+window.location.hostname+':'+$window.socketPort); return service; // Connect to Socket.io server function connect(url) { service.socket = io(url, {'transports': ['websocket', 'polling']}); } // Wrap the Socket.io 'emit' method function emit(eventName, data) { if (service.socket) { service.socket.emit(eventName, data); } } // Wrap the Socket.io 'on' method function on(eventName, callback) { if (service.socket) { service.socket.on(eventName, function (data) { $timeout(function () { callback(data); }); }); } } // Wrap the Socket.io 'removeListener' method function removeListener(eventName) { if (service.socket) { service.socket.removeListener(eventName); } } } }()); 'use strict'; module.exports = function(grunt) { require('jit-grunt')(grunt); // Project Configuration grunt.initConfig({ ngAnnotate: { production: { files: { 'dist/form.js': [ 'config/**/*.js', 'controllers/**/*.js', 'directives/**/*.js', 'services/**/*.js', 'dist/template.js' ] } } }, html2js: { options: { base: '', module: 'NodeForm.templates', singleModule: true, rename: function (moduleName) { return 'modules/forms/base/' + moduleName; } }, form: { src: ['views/**/*.html'], dest: 'dist/template.js' } }, cssmin: { combine: { files: { 'dist/form.css': 'css/**/*.css' } } }, }); // Making grunt default to force in order not to break the project. grunt.option('force', true); // Default task(s). grunt.registerTask('default', ['html2js:form', 'ngAnnotate', 'cssmin']); }; 'use strict'; // Configuring the Forms drop-down menus angular.module('forms').run(['Menus', function(Menus) { // Set top bar menu items Menus.addMenuItem('topbar', 'My Forms', 'forms', '', '/forms', false); } ]).filter('formValidity', function(){ return function(formObj){ if(formObj && formObj.form_fields && formObj.visible_form_fields){ //get keys var formKeys = Object.keys(formObj); //we only care about things that don't start with $ var fieldKeys = formKeys.filter(function(key){ return key[0] !== '$'; }); var fields = formObj.form_fields; var valid_count = fields.filter(function(field){ if(typeof field === 'object' && field.fieldType !== 'statement' && field.fieldType !== 'rating'){ return !!(field.fieldValue); } }).length; return valid_count - (formObj.form_fields.length - formObj.visible_form_fields.length); } return 0; }; }).config(['$provide', function ($provide){ $provide.decorator('accordionDirective', ["$delegate", function($delegate) { var directive = $delegate[0]; directive.replace = true; return $delegate; }]); }]); 'use strict'; // Setting up route angular.module('forms').config(['$stateProvider', function($stateProvider) { // Forms state routing $stateProvider. state('listForms', { url: '/forms', templateUrl: 'modules/forms/admin/views/list-forms.client.view.html' }). state('submitForm', { url: '/forms/:formId', templateUrl: 'modules/forms/base/views/submit-form.client.view.html', data: { hideNav: true }, resolve: { Forms: 'Forms', myForm: ["Forms", "$stateParams", function (Forms, $stateParams) { return Forms.get({formId: $stateParams.formId}).$promise; }] }, controller: 'SubmitFormController', controllerAs: 'ctrl' }).state('viewForm', { url: '/forms/:formId/admin', templateUrl: 'modules/forms/admin/views/admin-form.client.view.html', data: { permissions: [ 'editForm' ] }, resolve: { Forms: 'Forms', myForm: ["Forms", "$stateParams", function (Forms, $stateParams) { return Forms.get({formId: $stateParams.formId}).$promise; }] }, controller: 'AdminFormController' }).state('viewForm.configure', { url: '/configure', templateUrl: 'modules/forms/admin/views/adminTabs/configure.html' }).state('viewForm.design', { url: '/design', templateUrl: 'modules/forms/admin/views/adminTabs/design.html' }).state('viewForm.analyze', { url: '/analyze', templateUrl: 'modules/forms/admin/views/adminTabs/analyze.html', }).state('viewForm.create', { url: '/create', templateUrl: 'modules/forms/admin/views/adminTabs/create.html' }); } ]); (function () { 'use strict'; // Create the SendVisitorData service angular .module('forms') .factory('SendVisitorData', SendVisitorData); SendVisitorData.$inject = ['Socket', '$state']; function SendVisitorData(Socket, $state) { // Create a controller method for sending visitor data function send(form, lastActiveIndex, timeElapsed) { // Create a new message object var visitorData = { referrer: document.referrer, isSubmitted: form.submitted, formId: form._id, lastActiveField: form.form_fields[lastActiveIndex]._id, timeElapsed: timeElapsed }; Socket.emit('form-visitor-data', visitorData); } function init(){ // Make sure the Socket is connected if (!Socket.socket) { Socket.connect(); } } var service = { send: send }; init(); return service; } }()); 'use strict'; angular.module('forms').directive('keyToOption', function(){ return { restrict: 'A', scope: { field: '=' }, link: function($scope, $element, $attrs, $select) { $element.bind('keydown keypress', function(event) { var keyCode = event.which || event.keyCode; var index = parseInt(String.fromCharCode(keyCode))-1; //console.log($scope.field); if (index < $scope.field.fieldOptions.length) { event.preventDefault(); $scope.$apply(function () { $scope.field.fieldValue = $scope.field.fieldOptions[index].option_value; }); } }); } }; }); 'use strict'; angular.module('forms').directive('keyToTruthy', ['$rootScope', function($rootScope){ return { restrict: 'A', scope: { field: '=' }, link: function($scope, $element, $attrs) { $element.bind('keydown keypress', function(event) { var keyCode = event.which || event.keyCode; var truthyKeyCode = $attrs.keyCharTruthy.charCodeAt(0) - 32; var falseyKeyCode = $attrs.keyCharFalsey.charCodeAt(0) - 32; if(keyCode === truthyKeyCode ) { event.preventDefault(); $scope.$apply(function() { $scope.field.fieldValue = 'true'; }); }else if(keyCode === falseyKeyCode){ event.preventDefault(); $scope.$apply(function() { $scope.field.fieldValue = 'false'; }); } }); } }; }]); 'use strict'; // Config HTTP Error Handling angular.module('users').config(['$httpProvider', function($httpProvider) { $httpProvider.interceptors.push(["$q", "$location", function($q, $location) { return { responseError: function(response) { if( $location.path() !== '/users/me' && response.config){ if(response.config.url !== '/users/me'){ console.log('intercepted rejection of ', response.config.url, response.status); if (response.status === 401) { console.log($location.path()); // save the current location so that login can redirect back $location.nextAfterLogin = $location.path(); $location.path('/signin'); }else if(response.status === 403){ $location.path('/access_denied'); } } } return $q.reject(response); } }; }]); }]); 'use strict'; // Setting up route angular.module('users').config(['$stateProvider', function($stateProvider) { var checkLoggedin = function($q, $timeout, $state, User, Auth) { var deferred = $q.defer(); //console.log(Auth.ensureHasCurrentUser(User)); if (Auth.currentUser && Auth.currentUser.email) { $timeout(deferred.resolve); } else { Auth.currentUser = User.getCurrent( function() { Auth.login(); $timeout(deferred.resolve()); }, function() { Auth.logout(); $timeout(deferred.reject()); $state.go('signin', {reload: true}); }); } return deferred.promise; }; checkLoggedin.$inject = ["$q", "$timeout", "$state", "User", "Auth"]; var checkSignupDisabled = function($window, $timeout, $q) { var deferred = $q.defer(); if($window.signupDisabled) { $timeout(deferred.reject()); } else { $timeout(deferred.resolve()); } return deferred.promise; }; checkSignupDisabled.$inject = ["$window", "$timeout", "$q"]; // Users state routing $stateProvider. state('profile', { resolve: { loggedin: checkLoggedin }, url: '/settings/profile', templateUrl: 'modules/users/views/settings/edit-profile.client.view.html' }). state('password', { resolve: { loggedin: checkLoggedin }, url: '/settings/password', templateUrl: 'modules/users/views/settings/change-password.client.view.html' }). state('accounts', { resolve: { loggedin: checkLoggedin }, url: '/settings/accounts', templateUrl: 'modules/users/views/settings/social-accounts.client.view.html' }). state('signup', { resolve: { isDisabled: checkSignupDisabled }, url: '/signup', templateUrl: 'modules/users/views/authentication/signup.client.view.html' }). state('signup-success', { resolve: { isDisabled: checkSignupDisabled }, url: '/signup-success', templateUrl: 'modules/users/views/authentication/signup-success.client.view.html' }). state('signin', { url: '/signin', templateUrl: 'modules/users/views/authentication/signin.client.view.html' }). state('access_denied', { url: '/access_denied', templateUrl: 'modules/users/views/authentication/access-denied.client.view.html' }). state('verify', { resolve: { isDisabled: checkSignupDisabled }, url: '/verify/:token', templateUrl: 'modules/users/views/verify/verify-account.client.view.html' }). state('resendVerifyEmail', { resolve: { isDisabled: checkSignupDisabled }, url: '/verify', templateUrl: 'modules/users/views/verify/resend-verify-email.client.view.html' }). state('forgot', { url: '/password/forgot', templateUrl: 'modules/users/views/password/forgot-password.client.view.html' }). state('reset-invalid', { url: '/password/reset/invalid', templateUrl: 'modules/users/views/password/reset-password-invalid.client.view.html' }). state('reset-success', { url: '/password/reset/success', templateUrl: 'modules/users/views/password/reset-password-success.client.view.html' }). state('reset', { url: '/password/reset/:token', templateUrl: 'modules/users/views/password/reset-password.client.view.html' }); } ]); 'use strict'; angular.module('users').controller('AuthenticationController', ['$scope', '$location', '$state', '$rootScope', 'User', 'Auth', function($scope, $location, $state, $rootScope, User, Auth) { $scope = $rootScope; $scope.credentials = {}; $scope.error = ''; $scope.signin = function() { $scope.credentials.email = $scope.credentials.username; User.login($scope.credentials).then( function(response) { Auth.login(response); $scope.user = $rootScope.user = Auth.ensureHasCurrentUser(User); if($state.previous.name !== 'home' && $state.previous.name !== 'verify' && $state.previous.name !== ''){ $state.go($state.previous.name); }else{ $state.go('listForms'); } }, function(error) { $rootScope.user = Auth.ensureHasCurrentUser(User); $scope.user = $rootScope.user; $scope.error = error; console.log('loginError: '+error); } ); }; $scope.signup = function() { console.log($scope.credentials); User.signup($scope.credentials).then( function(response) { console.log('signup-success'); $state.go('signup-success'); }, function(error) { console.log('Error: '); console.log(error); if(error) { $scope.error = error; console.log(error); }else { console.log('No response received'); } } ); }; } ]); 'use strict'; angular.module('users').controller('PasswordController', ['$scope', '$stateParams', '$state', 'User', function($scope, $stateParams, $state, User) { $scope.error = ''; // Submit forgotten password account id $scope.askForPasswordReset = function() { User.askForPasswordReset($scope.credentials).then( function(response){ $scope.success = response.message; $scope.credentials = null; }, function(error){ $scope.error = error; $scope.credentials = null; } ); }; // Change user password $scope.resetUserPassword = function() { $scope.success = $scope.error = null; User.resetPassword($scope.passwordDetails, $stateParams.token).then( function(response){ // If successful show success message and clear form $scope.success = response.message; $scope.passwordDetails = null; // And redirect to the index page $state.go('reset-success'); }, function(error){ $scope.error = error.message || error; $scope.passwordDetails = null; } ); }; } ]); 'use strict'; angular.module('users').controller('SettingsController', ['$scope', '$rootScope', '$http', '$state', 'Users', function($scope, $rootScope, $http, $state, Users) { $scope.user = $rootScope.user; // Check if there are additional accounts $scope.hasConnectedAdditionalSocialAccounts = function(provider) { for (var i in $scope.user.additionalProvidersData) { return true; } return false; }; // Check if provider is already in use with current user $scope.isConnectedSocialAccount = function(provider) { return $scope.user.provider === provider || ($scope.user.additionalProvidersData && $scope.user.additionalProvidersData[provider]); }; // Remove a user social account $scope.removeUserSocialAccount = function(provider) { $scope.success = $scope.error = null; $http.delete('/users/accounts', { params: { provider: provider } }).success(function(response) { // If successful show success message and clear form $scope.success = true; $scope.user = response; }).error(function(response) { $scope.error = response.message; }); }; // Update a user profile $scope.updateUserProfile = function(isValid) { if (isValid) { $scope.success = $scope.error = null; var user = new Users($scope.user); user.$update(function(response) { $scope.success = true; $scope.user = response; }, function(response) { $scope.error = response.data.message; }); } else { $scope.submitted = true; } }; // Change user password $scope.changeUserPassword = function() { $scope.success = $scope.error = null; $http.post('/users/password', $scope.passwordDetails).success(function(response) { // If successful show success message and clear form $scope.success = true; $scope.passwordDetails = null; }).error(function(response) { $scope.error = response.message; }); }; } ]); 'use strict'; angular.module('users').controller('VerifyController', ['$scope', '$state', '$rootScope', 'User', 'Auth', '$stateParams', function($scope, $state, $rootScope, User, Auth, $stateParams) { $scope.isResetSent = false; $scope.credentials = {}; $scope.error = ''; // Submit forgotten password account id $scope.resendVerifyEmail = function() { // console.log($scope.credentials); // console.log($scope.credentials.email); User.resendVerifyEmail($scope.credentials.email).then( function(response){ console.log(response); $scope.success = response.message; $scope.credentials = null; $scope.isResetSent = true; }, function(error){ $scope.error = error; $scope.credentials.email = null; $scope.isResetSent = false; } ); }; //Validate Verification Token $scope.validateVerifyToken = function() { if($stateParams.token){ console.log($stateParams.token); User.validateVerifyToken($stateParams.token).then( function(response){ console.log('Success: '+response.message); $scope.success = response.message; $scope.isResetSent = true; $scope.credentials.email = null; }, function(error){ console.log('Error: '+error.message); $scope.isResetSent = false; $scope.error = error; $scope.credentials.email = null; } ); } }; } ]); 'use strict'; angular.module('users').factory('Auth', ['$window', function($window) { var userState = { isLoggedIn: false }; var service = { _currentUser: null, get currentUser(){ return this._currentUser; }, // Note: we can't make the User a dependency of Auth // because that would create a circular dependency // Auth <- $http <- $resource <- LoopBackResource <- User <- Auth ensureHasCurrentUser: function(User) { if (service._currentUser && service._currentUser.username) { //console.log('Using local current user.'); //console.log(service._currentUser); return service._currentUser; } else if ($window.user){ //console.log('Using cached current user.'); //console.log($window.user); service._currentUser = $window.user; return service._currentUser; } else{ //console.log('Fetching current user from the server.'); User.getCurrent().then(function(user) { // success service._currentUser = user; userState.isLoggedIn = true; $window.user = service._currentUser; return service._currentUser; }, function(response) { userState.isLoggedIn = false; service._currentUser = null; $window.user = null; console.log('User.getCurrent() err', response); return null; }); } }, isAuthenticated: function() { return !!service._currentUser; }, getUserState: function() { return userState; }, login: function(new_user) { userState.isLoggedIn = true; service._currentUser = new_user; }, logout: function() { $window.user = null; userState.isLoggedIn = false; service._currentUser = null; } }; return service; } ]); 'use strict'; angular.module('users').service('Authorizer', ["APP_PERMISSIONS", "USER_ROLES", function(APP_PERMISSIONS, USER_ROLES) { return function(user) { return { canAccess: function(permissions) { var i, len, permission; if (!angular.isArray(permissions)) { permissions = [permissions]; } for (i = 0, len = permissions.length; i < len; i++) { permission = permissions[i]; if (APP_PERMISSIONS[permission] === null) { throw 'Bad permission value'; } if (user && user.roles) { switch (permission) { case APP_PERMISSIONS.viewAdminSettings: case APP_PERMISSIONS.editAdminSettings: return user.roles.indexOf(USER_ROLES.admin) > -1; case APP_PERMISSIONS.viewPrivateForm: case APP_PERMISSIONS.editForm: return user.roles.indexOf(USER_ROLES.admin) > -1 || user.roles.indexOf(USER_ROLES.normal) > -1; } } else { return false; } } return false; } }; }; }]); 'use strict'; angular.module('users').factory('User', ['$window', '$q', '$timeout', '$http', '$state', function($window, $q, $timeout, $http, $state) { var userService = { getCurrent: function() { var deferred = $q.defer(); $http.get('/users/me') .success(function(response) { deferred.resolve(response); }) .error(function() { deferred.reject('User\'s session has expired'); }); return deferred.promise; }, login: function(credentials) { var deferred = $q.defer(); $http.post('/auth/signin', credentials).success(function(response) { deferred.resolve(response); }).error(function(error) { deferred.reject(error.message || error); }); return deferred.promise; }, logout: function() { var deferred = $q.defer(); $http.get('/auth/signout').success(function(response) { deferred.resolve(null); }).error(function(error) { deferred.reject(error.message || error); }); return deferred.promise; }, signup: function(credentials) { var deferred = $q.defer(); $http.post('/auth/signup', credentials).success(function(response) { // If successful we assign the response to the global user model deferred.resolve(response); }).error(function(error) { deferred.reject(error.message || error); }); return deferred.promise; }, resendVerifyEmail: function(_email) { var deferred = $q.defer(); $http.post('/auth/verify', {email: _email}).success(function(response) { deferred.resolve(response); }).error(function(error) { deferred.reject(error.message || error); }); return deferred.promise; }, validateVerifyToken: function(token) { //DAVID: TODO: The valid length of a token should somehow be linked to server config values //DAVID: TODO: SEMI-URGENT: Should we even be doing this? var validTokenRe = /^([A-Za-z0-9]{48})$/g; if( !validTokenRe.test(token) ) throw new Error('Error token: '+token+' is not a valid verification token'); var deferred = $q.defer(); $http.get('/auth/verify/'+token).success(function(response) { deferred.resolve(response); }).error(function(error) { deferred.reject(error); }); return deferred.promise; }, resetPassword: function(passwordDetails, token) { var deferred = $q.defer(); $http.get('/auth/password/'+token, passwordDetails).success(function(response) { deferred.resolve(); }).error(function(error) { deferred.reject(error.message || error); }); return deferred.promise; }, // Submit forgotten password account id askForPasswordReset: function(credentials) { var deferred = $q.defer(); $http.post('/auth/forgot', credentials).success(function(response) { // Show user success message and clear form deferred.resolve(response); }).error(function(error) { // Show user error message deferred.reject(error.message || error); }); return deferred.promise; } }; return userService; } ]); 'use strict'; // Users service used for communicating with the users REST endpoint angular.module('users').factory('Users', ['$resource', function($resource) { return $resource('users', {}, { update: { method: 'PUT' } }); } ]); 'use strict'; (function() { describe('HeaderController', function() { //Initialize global variables var scope, HeaderController; // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(inject(function($controller, $rootScope) { scope = $rootScope.$new(); HeaderController = $controller('HeaderController', { $scope: scope }); })); it('should expose the authentication service', function() { expect(scope.authentication).toBeTruthy(); }); }); })(); 'use strict'; (function() { describe('HomeController', function() { //Initialize global variables var scope, HomeController; // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(inject(function($controller, $rootScope) { scope = $rootScope.$new(); HomeController = $controller('HomeController', { $scope: scope }); })); }); })(); 'use strict'; // Forms controller angular.module('forms').controller('AdminFormController', ['$rootScope', '$scope', '$stateParams', '$state', 'Forms', 'CurrentForm', '$http', '$uibModal', 'myForm', function($rootScope, $scope, $stateParams, $state, Forms, CurrentForm, $http, $uibModal, myForm) { $scope = $rootScope; $scope.animationsEnabled = true; $scope.myform = myForm; $rootScope.saveInProgress = false; CurrentForm.setForm($scope.myform); $scope.tabData = [ { heading: 'Create', route: 'viewForm.create' }, { heading: 'Design', route: 'viewForm.design' }, { heading: 'Configure', route: 'viewForm.configure' }, { heading: 'Analyze', route: 'viewForm.analyze' } ]; $scope.setForm = function(form){ $scope.myform = form; }; $rootScope.resetForm = function(){ $scope.myform = Forms.get({ formId: $stateParams.formId }); }; /* ** DeleteModal Functions */ $scope.openDeleteModal = function(){ $scope.deleteModal = $uibModal.open({ animation: $scope.animationsEnabled, templateUrl: 'myModalContent.html', controller: 'AdminFormController', resolve: { myForm: function(){ return $scope.myform; } } }); $scope.deleteModal.result.then(function (selectedItem) { $scope.selected = selectedItem; }, function () { console.log('Modal dismissed at: ' + new Date()); }); }; $scope.cancelDeleteModal = function(){ if($scope.deleteModal){ $scope.deleteModal.dismiss('cancel'); } }; // Remove existing Form $scope.removeCurrentForm = function() { if($scope.deleteModal && $scope.deleteModal.opened){ $scope.deleteModal.close(); var form_id = $scope.myform._id; if(!form_id) throw new Error('Error - removeCurrentForm(): $scope.myform._id does not exist'); $http.delete('/forms/'+form_id) .success(function(data, status, headers){ console.log('form deleted successfully'); $state.go('listForms', {}, {reload: true}); }).error(function(error){ console.log('ERROR: Form could not be deleted.'); console.error(error); }); } }; // Update existing Form $scope.update = $rootScope.update = function(updateImmediately, cb){ var continueUpdate = true; if(!updateImmediately){ continueUpdate = !$rootScope.saveInProgress; } //Update form **if we are not currently updating** or if **shouldUpdateNow flag is set** if(continueUpdate){ var err = null; if(!updateImmediately){ $rootScope.saveInProgress = true; } $scope.updatePromise = $http.put('/forms/'+$scope.myform._id, {form: $scope.myform}) .then(function(response){ $rootScope.myform = $scope.myform = response.data; // console.log(response.data); }).catch(function(response){ console.log('Error occured during form UPDATE.\n'); // console.log(response.data); err = response.data; }).finally(function() { // console.log('finished updating'); if(!updateImmediately){$rootScope.saveInProgress = false; } if( (typeof cb) === 'function'){ return cb(err); } }); } }; } ]); 'use strict'; // Forms controller angular.module('forms').controller('ListFormsController', ['$rootScope', '$scope', '$stateParams', '$state', 'Forms', 'CurrentForm', '$http', function($rootScope, $scope, $stateParams, $state, Forms, CurrentForm, $http) { $scope = $rootScope; $scope.forms = {}; $scope.showCreateModal = false; // Return all user's Forms $scope.findAll = function() { Forms.query(function(_forms){ $scope.myforms = _forms; }); }; //Modal functions $scope.openCreateModal = function(){ if(!$scope.showCreateModal){ $scope.showCreateModal = true; } }; $scope.closeCreateModal = function(){ if($scope.showCreateModal){ $scope.showCreateModal = false; } }; $scope.setForm = function (form) { $scope.myform = form; }; $scope.goToWithId = function(route, id) { $state.go(route, {'formId': id}, {reload: true}); }; $scope.duplicateForm = function(form_index){ var form = _.cloneDeep($scope.myforms[form_index]); delete form._id; $http.post('/forms', {form: form}) .success(function(data, status, headers){ $scope.myforms.splice(form_index+1, 0, data); }).error(function(errorResponse){ console.error(errorResponse); if(errorResponse === null){ $scope.error = errorResponse.data.message; } }); }; // Create new Form $scope.createNewForm = function(){ // console.log($scope.forms.createForm); var form = {}; form.title = $scope.forms.createForm.title.$modelValue; form.language = $scope.forms.createForm.language.$modelValue; if($scope.forms.createForm.$valid && $scope.forms.createForm.$dirty){ $http.post('/forms', {form: form}) .success(function(data, status, headers){ console.log('new form created'); // Redirect after save $scope.goToWithId('viewForm.create', data._id+''); }).error(function(errorResponse){ console.error(errorResponse); $scope.error = errorResponse.data.message; }); } }; $scope.removeForm = function(form_index) { if(form_index >= $scope.myforms.length || form_index < 0){ throw new Error('Error: form_index in removeForm() must be between 0 and '+$scope.myforms.length-1); } $http.delete('/forms/'+$scope.myforms[form_index]._id) .success(function(data, status, headers){ console.log('form deleted successfully'); $scope.myforms.splice(form_index, 1); }).error(function(error){ console.log('ERROR: Form could not be deleted.'); console.error(error); }); }; } ]); 'use strict'; function removeDateFieldsFunc(o) { var clone = _.clone(o); function eachObject(v,k){ if(k === 'lastModified' || k === 'created'){ delete clone[k]; } } for(var i=0; i FormFields $scope.oscarFieldsLeft = function(field_id){ if($scope.myform && $scope.myform.plugins.oscarhost.settings.validFields.length > 0){ if(!$scope.myform.plugins.oscarhost.settings.fieldMap) $scope.myform.plugins.oscarhost.settings.fieldMap = {}; var oscarhostFields = $scope.myform.plugins.oscarhost.settings.validFields; var currentFields = _($scope.myform.plugins.oscarhost.settings.fieldMap).invert().keys().value(); if( $scope.myform.plugins.oscarhost.settings.fieldMap.hasOwnProperty(field_id) ){ currentFields = _(currentFields).difference($scope.myform.plugins.oscarhost.settings.fieldMap[field_id]); } //Get all oscarhostFields that haven't been mapped to a formfield return _(oscarhostFields).difference(currentFields).value(); } return []; }; /* ** FormFields (ui-sortable) drag-and-drop configuration */ $scope.dropzone = { handle: '.handle', containment: '.dropzoneContainer', cursor: 'grabbing' }; /* ** Field CRUD Methods */ // Add a new field $scope.addNewField = function(modifyForm, fieldType){ // increment lastAddedID counter $scope.addField.lastAddedID++; var fieldTitle; for(var i = 0; i < $scope.addField.types.length; i++){ if($scope.addField.types[i].name === fieldType){ $scope.addField.types[i].lastAddedID++; fieldTitle = $scope.addField.types[i].value+$scope.addField.types[i].lastAddedID; break; } } var newField = { title: fieldTitle, fieldType: fieldType, fieldValue: '', required: true, disabled: false, deletePreserved: false }; if($scope.showAddOptions(newField)){ newField.fieldOptions = []; newField.fieldOptions.push({ 'option_id' : Math.floor(100000*Math.random()), //Generate pseudo-random option id 'option_title' : 'Option 0', 'option_value' : 'Option 0' }); } if(modifyForm){ //Add newField to form_fields array $scope.myform.form_fields.push(newField); } return newField; }; // Delete particular field on button click $scope.deleteField = function (field_index){ //Delete field from field map var currFieldId = $scope.myform.form_fields[field_index]._id; if($scope.myform.hasOwnProperty('plugins.oscarhost.baseUrl')) delete $scope.myform.plugins.oscarhost.settings.fieldMap[currFieldId]; //Delete field $scope.myform.form_fields.splice(field_index, 1); }; $scope.duplicateField = function (field_index){ var currField = _.cloneDeep($scope.myform.form_fields[field_index]); currField._id = 'cloned'+_.uniqueId(); currField.title += ' copy'; //Insert field at selected index $scope.myform.form_fields.splice(field_index+1, 0, currField); }; /* ** startPage Button Methods */ // add new Button to the startPage $scope.addButton = function(){ var newButton = {}; newButton.bgColor = '#ddd'; newButton.color = '#ffffff'; newButton.text = 'Button'; newButton._id = Math.floor(100000*Math.random()); $scope.myform.startPage.buttons.push(newButton); }; // delete particular Button from startPage $scope.deleteButton = function(button){ var currID; for(var i = 0; i < $scope.myform.startPage.buttons.length; i++){ currID = $scope.myform.startPage.buttons[i]._id; console.log(currID); if(currID === button._id){ $scope.myform.startPage.buttons.splice(i, 1); break; } } }; /* ** Field Option Methods */ // add new option to the field $scope.addOption = function(field_index){ var currField = $scope.myform.form_fields[field_index]; //console.log(field_index); //console.log(currField); if(currField.fieldType === 'checkbox' || currField.fieldType === 'dropdown' || currField.fieldType === 'radio'){ if(!currField.fieldOptions){ $scope.myform.form_fields[field_index].fieldOptions = []; } var lastOptionID = $scope.myform.form_fields[field_index].fieldOptions.length+1; // new option's id var newOption = { 'option_id' : Math.floor(100000*Math.random()), 'option_title' : 'Option '+lastOptionID, 'option_value' : 'Option ' +lastOptionID }; // put new option into fieldOptions array $scope.myform.form_fields[field_index].fieldOptions.push(newOption); } }; // delete particular option $scope.deleteOption = function (field_index, option){ var currField = $scope.myform.form_fields[field_index]; if(currField.fieldType === 'checkbox' || currField.fieldType === 'dropdown' || currField.fieldType === 'radio'){ for(var i = 0; i < currField.fieldOptions.length; i++){ if(currField.fieldOptions[i].option_id === option.option_id){ $scope.myform.form_fields[field_index].fieldOptions.splice(i, 1); break; } } } }; // decides whether field options block will be shown (true for dropdown and radio fields) $scope.showAddOptions = function (field){ if(field.fieldType === 'dropdown' || field.fieldType === 'checkbox' || field.fieldType === 'radio'){ return true; } else { return false; } }; // decides whether field options block will be shown (true for dropdown and radio fields) $scope.showRatingOptions = function (field){ if(field.fieldType === 'rating'){ return true; } else { return false; } }; }] }; } ]); 'use strict'; angular.module('forms').directive('editSubmissionsFormDirective', ['$rootScope', '$http', function ($rootScope, $http) { return { templateUrl: 'modules/forms/admin/views/directiveViews/form/edit-submissions-form.client.view.html', restrict: 'E', scope: { myform:'=', user:'=' }, controller: ["$scope", function($scope){ $scope.table = { masterChecker: false, rows: [] }; /* ** Table Functions */ $scope.isAtLeastOneChecked = function(){ for(var i=0; i<$scope.table.rows.length; i++){ if($scope.table.rows[i].selected) return true; } return false; }; $scope.toggleAllCheckers = function(){ for(var i=0; i<$scope.table.rows.length; i++){ $scope.table.rows[i].selected = $scope.table.masterChecker; } }; $scope.toggleObjSelection = function($event, description) { $event.stopPropagation(); }; $scope.rowClicked = function(row_index) { $scope.table.rows[row_index].selected = !$scope.table.rows[row_index].selected; }; /* * Form Submission Methods */ //Fetch and display submissions of Form $scope.initFormSubmissions = function(){ $http.get('/forms/'+$scope.myform._id+'/submissions') .success(function(data, status, headers){ var _tmpSubFormFields, defaultFormFields = _.cloneDeep($scope.myform.form_fields); // console.log('before textField2: '+data[0].form_fields[1].fieldValue); //Iterate through form's submissions for(var i=0; i', restrict: 'E', scope: { typeName: '@' }, controller: ["$scope", function($scope){ var iconTypeMap = { 'textfield': 'fa fa-pencil-square-o', 'dropdown': 'fa fa-th-list', 'date': 'fa fa-calendar', 'checkbox': 'fa fa-check-square-o', 'radio': 'fa fa-dot-circle-o', 'email': 'fa fa-envelope-o', 'textarea': 'fa fa-pencil-square', 'legal': 'fa fa-legal', 'file': 'fa fa-cloud-upload', 'rating': 'fa fa-star-half-o', 'link': 'fa fa-link', 'scale': 'fa fa-sliders', 'stripe': 'fa fa-credit-card', 'statement': 'fa fa-quote-left', 'yes_no': 'fa fa-toggle-on', 'number': 'fa fa-slack' }; $scope.typeIcon = iconTypeMap[$scope.typeName]; }] }; }); 'use strict'; // coffeescript's for in loop var __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }; angular.module('forms').directive('fieldDirective', ['$http', '$compile', '$rootScope', '$templateCache', 'supportedFields', function($http, $compile, $rootScope, $templateCache, supportedFields) { var getTemplateUrl = function(fieldType) { var type = fieldType; var supported_fields = [ 'textfield', 'textarea', 'date', 'dropdown', 'hidden', 'password', 'radio', 'legal', 'statement', 'rating', 'yes_no', 'number', 'natural' ]; if (__indexOf.call(supported_fields, type) >= 0) { var templateUrl = 'modules/forms/views/directiveViews/field/'; templateUrl = templateUrl+type+'.html'; } return $templateCache.get(templateUrl); }; return { template: '
{{field.title}}
', restrict: 'E', scope: { field: '=', required: '&', design: '=', index: '=', forms: '=' }, link: function(scope, element) { $rootScope.chooseDefaultOption = scope.chooseDefaultOption = function(type) { if(type === 'yes_no'){ scope.field.fieldValue = 'true'; }else if(type === 'rating'){ scope.field.fieldValue = 0; }else if(scope.field.fieldType === 'radio'){ console.log(scope.field); scope.field.fieldValue = scope.field.fieldOptions[0].option_value; console.log(scope.field.fieldValue); }else if(type === 'legal'){ scope.field.fieldValue = 'true'; $rootScope.nextField(); } }; scope.setActiveField = $rootScope.setActiveField; //Set format only if field is a date if(scope.field.fieldType === 'date'){ scope.dateOptions = { changeYear: true, changeMonth: true, altFormat: 'mm/dd/yyyy', yearRange: '1900:-0', defaultDate: 0 }; } var fieldType = scope.field.fieldType; if(scope.field.fieldType === 'number' || scope.field.fieldType === 'textfield' || scope.field.fieldType === 'email' || scope.field.fieldType === 'link'){ switch(scope.field.fieldType){ case 'textfield': scope.field.input_type = 'text'; break; case 'email': scope.field.input_type = 'email'; scope.field.placeholder = 'joesmith@example.com'; break; case 'number': scope.field.input_type = 'text'; scope.field.validateRegex = /^-?\d+$/; break; default: scope.field.input_type = 'url'; scope.field.placeholder = 'http://example.com'; break; } fieldType = 'textfield'; } var template = getTemplateUrl(fieldType); element.html(template).show(); var output = $compile(element.contents())(scope); } }; }]); 'use strict'; //TODO: DAVID: Need to refactor this angular.module('forms').directive('onEnterKey', ['$rootScope', function($rootScope){ return { restrict: 'A', link: function($scope, $element, $attrs) { $element.bind('keydown keypress', function(event) { var keyCode = event.which || event.keyCode; var onEnterKeyDisabled = false; if($attrs.onEnterKeyDisabled !== null) onEnterKeyDisabled = $attrs.onEnterKeyDisabled; if(keyCode === 13 && !event.shiftKey && !onEnterKeyDisabled) { event.preventDefault(); $rootScope.$apply(function() { $rootScope.$eval($attrs.onEnterKey); }); } }); } }; }]).directive('onTabKey', ['$rootScope', function($rootScope){ return { restrict: 'A', link: function($scope, $element, $attrs) { $element.bind('keydown keypress', function(event) { var keyCode = event.which || event.keyCode; if(keyCode === 9 && !event.shiftKey) { event.preventDefault(); $rootScope.$apply(function() { $rootScope.$eval($attrs.onTabKey); }); } }); } }; }]).directive('onEnterOrTabKey', ['$rootScope', function($rootScope){ return { restrict: 'A', link: function($scope, $element, $attrs) { $element.bind('keydown keypress', function(event) { var keyCode = event.which || event.keyCode; if((keyCode === 13 || keyCode === 9) && !event.shiftKey) { event.preventDefault(); $rootScope.$apply(function() { $rootScope.$eval($attrs.onEnterOrTabKey); }); } }); } }; }]).directive('onTabAndShiftKey', ['$rootScope', function($rootScope){ return { restrict: 'A', link: function($scope, $element, $attrs) { $element.bind('keydown keypress', function(event) { var keyCode = event.which || event.keyCode; if(keyCode === 9 && event.shiftKey) { event.preventDefault(); $rootScope.$apply(function() { $rootScope.$eval($attrs.onTabAndShiftKey); }); } }); } }; }]); 'use strict'; angular.module('forms').directive('onFinishRender', ["$rootScope", "$timeout", function ($rootScope, $timeout) { return { restrict: 'A', link: function (scope, element, attrs) { //Don't do anything if we don't have a ng-repeat on the current element if(!element.attr('ng-repeat') && !element.attr('data-ng-repeat')){ return; } var broadcastMessage = attrs.onFinishRender || 'ngRepeat'; if(scope.$first && !scope.$last) { scope.$evalAsync(function () { $rootScope.$broadcast(broadcastMessage+' Started'); }); }else if(scope.$last) { scope.$evalAsync(function () { // console.log(broadcastMessage+'Finished'); $rootScope.$broadcast(broadcastMessage+' Finished'); }); } } }; }]); 'use strict'; angular.module('forms').directive('submitFormDirective', ['$http', 'TimeCounter', '$filter', '$rootScope', 'Auth', 'SendVisitorData', function ($http, TimeCounter, $filter, $rootScope, Auth, SendVisitorData) { return { templateUrl: 'modules/forms/base/views/directiveViews/form/submit-form.client.view.html', restrict: 'E', scope: { myform:'=' }, controller: ["$document", "$window", "$scope", function($document, $window, $scope){ $scope.authentication = $rootScope.authentication; $scope.noscroll = false; $scope.forms = {}; var form_fields_count = $scope.myform.visible_form_fields.filter(function(field){ if(field.fieldType === 'statement' || field.fieldType === 'rating'){ return false; } return true; }).length; var nb_valid = $filter('formValidity')($scope.myform); $scope.translateAdvancementData = { done: nb_valid, total: form_fields_count, answers_not_completed: form_fields_count - nb_valid }; $scope.reloadForm = function(){ //Reset Form $scope.myform.submitted = false; $scope.myform.form_fields = _.chain($scope.myform.visible_form_fields).map(function(field){ field.fieldValue = ''; return field; }).value(); $scope.loading = false; $scope.error = ''; $scope.selected = { _id: '', index: 0 }; $scope.setActiveField($scope.myform.visible_form_fields[0]._id, 0, false); //console.log($scope.selected); //Reset Timer TimeCounter.restartClock(); }; //Fire event when window is scrolled $window.onscroll = function(){ $scope.scrollPos = document.body.scrollTop || document.documentElement.scrollTop || 0; var elemBox = document.getElementsByClassName('activeField')[0].getBoundingClientRect(); $scope.fieldTop = elemBox.top; $scope.fieldBottom = elemBox.bottom; //console.log($scope.forms.myForm); var field_id; var field_index; if(!$scope.noscroll){ //Focus on submit button if( $scope.selected.index === $scope.myform.visible_form_fields.length-1 && $scope.fieldBottom < 200){ field_index = $scope.selected.index+1; field_id = 'submit_field'; $scope.setActiveField(field_id, field_index, false); } //Focus on field above submit button else if($scope.selected.index === $scope.myform.visible_form_fields.length){ if($scope.fieldTop > 200){ field_index = $scope.selected.index-1; field_id = $scope.myform.visible_form_fields[field_index]._id; $scope.setActiveField(field_id, field_index, false); } }else if( $scope.fieldBottom < 0){ field_index = $scope.selected.index+1; field_id = $scope.myform.visible_form_fields[field_index]._id; $scope.setActiveField(field_id, field_index, false); }else if ( $scope.selected.index !== 0 && $scope.fieldTop > 0) { field_index = $scope.selected.index-1; field_id = $scope.myform.visible_form_fields[field_index]._id; $scope.setActiveField(field_id, field_index, false); } //console.log('$scope.selected.index: '+$scope.selected.index); //console.log('scroll pos: '+$scope.scrollPos+' fieldTop: '+$scope.fieldTop+' fieldBottom: '+$scope.fieldBottom); $scope.$apply(); } }; /* ** Field Controls */ var getActiveField = function(){ if($scope.selected === null){ console.error('current active field is null'); throw new Error('current active field is null'); } if($scope.selected._id === 'submit_field') { return $scope.myform.form_fields.length - 1; } else { return $scope.selected.index; } }; $scope.setActiveField = $rootScope.setActiveField = function(field_id, field_index, animateScroll) { if($scope.selected === null || $scope.selected._id === field_id){ //console.log('not scrolling'); //console.log($scope.selected); return; } //console.log('field_id: '+field_id); //console.log('field_index: '+field_index); //console.log($scope.selected); $scope.selected._id = field_id; $scope.selected.index = field_index; var nb_valid = $filter('formValidity')($scope.myform); $scope.translateAdvancementData = { done: nb_valid, total: form_fields_count, answers_not_completed: form_fields_count - nb_valid }; if(animateScroll){ $scope.noscroll=true; setTimeout(function() { $document.scrollToElement(angular.element('.activeField'), -10, 200).then(function() { $scope.noscroll = false; setTimeout(function() { if (document.querySelectorAll('.activeField .focusOn').length) { //Handle default case document.querySelectorAll('.activeField .focusOn')[0].focus(); } else if(document.querySelectorAll('.activeField input').length) { //Handle case for rating input document.querySelectorAll('.activeField input')[0].focus(); } else { //Handle case for dropdown input document.querySelectorAll('.activeField .selectize-input')[0].focus(); } }); }); }); }else { setTimeout(function() { if (document.querySelectorAll('.activeField .focusOn')[0]) { //FIXME: DAVID: Figure out how to set focus without scroll movement in HTML Dom document.querySelectorAll('.activeField .focusOn')[0].focus(); } else { document.querySelectorAll('.activeField input')[0].focus(); } }); } SendVisitorData.send($scope.myform, getActiveField(), TimeCounter.getTimeElapsed()); }; $rootScope.nextField = $scope.nextField = function(){ //console.log('nextfield'); //console.log($scope.selected.index); //console.log($scope.myform.visible_form_fields.length-1); var selected_index, selected_id; if($scope.selected.index < $scope.myform.visible_form_fields.length-1){ selected_index = $scope.selected.index+1; selected_id = $scope.myform.visible_form_fields[selected_index]._id; $rootScope.setActiveField(selected_id, selected_index, true); } else if($scope.selected.index === $scope.myform.visible_form_fields.length-1) { //console.log('Second last element'); selected_index = $scope.selected.index+1; selected_id = 'submit_field'; $rootScope.setActiveField(selected_id, selected_index, true); } }; $rootScope.prevField = $scope.prevField = function(){ if($scope.selected.index > 0){ var selected_index = $scope.selected.index - 1; var selected_id = $scope.myform.visible_form_fields[selected_index]._id; $scope.setActiveField(selected_id, selected_index, true); } }; /* ** Form Display Functions */ $scope.exitStartPage = function(){ $scope.myform.startPage.showStart = false; if($scope.myform.visible_form_fields.length > 0){ $scope.selected._id = $scope.myform.visible_form_fields[0]._id; } }; $rootScope.goToInvalid = $scope.goToInvalid = function() { document.querySelectorAll('.ng-invalid.focusOn')[0].focus(); }; $rootScope.submitForm = $scope.submitForm = function() { var _timeElapsed = TimeCounter.stopClock(); $scope.loading = true; var form = _.cloneDeep($scope.myform); form.timeElapsed = _timeElapsed; form.percentageComplete = $filter('formValidity')($scope.myform) / $scope.myform.visible_form_fields.length * 100; delete form.visible_form_fields; for(var i=0; i < $scope.myform.form_fields.length; i++){ if($scope.myform.form_fields[i].fieldType === 'dropdown' && !$scope.myform.form_fields[i].deletePreserved){ $scope.myform.form_fields[i].fieldValue = $scope.myform.form_fields[i].fieldValue.option_value; } } setTimeout(function () { $scope.submitPromise = $http.post('/forms/' + $scope.myform._id, form) .success(function (data, status, headers) { console.log($scope.myform.form_fields[0]); $scope.myform.submitted = true; $scope.loading = false; SendVisitorData.send($scope.myform, getActiveField(), _timeElapsed); }) .error(function (error) { $scope.loading = false; console.error(error); $scope.error = error.message; }); }, 500); }; //Reload our form $scope.reloadForm(); }] }; } ]); 'use strict'; //Forms service used for communicating with the forms REST endpoints angular.module('forms').service('CurrentForm', function(){ //Private variables var _form = {}; //Public Methods this.getForm = function() { return _form; }; this.setForm = function(form) { _form = form; }; } ); 'use strict'; //Forms service used for communicating with the forms REST endpoints angular.module('forms').factory('Forms', ['$resource', 'FORM_URL', function($resource, FORM_URL) { return $resource(FORM_URL, { formId: '@_id' }, { 'query' : { method: 'GET', isArray: true //DAVID: TODO: Do we really need to get visible_form_fields for a Query? // transformResponse: function(data, header) { // var forms = angular.fromJson(data); // angular.forEach(forms, function(form, idx) { // form.visible_form_fields = _.filter(form.form_fields, function(field){ // return (field.deletePreserved === false); // }); // }); // return forms; // } }, 'get' : { method: 'GET', transformResponse: function(data, header) { var form = angular.fromJson(data); form.visible_form_fields = _.filter(form.form_fields, function(field){ return (field.deletePreserved === false); }); return form; } }, 'update': { method: 'PUT' }, 'save': { method: 'POST' } }); } ]); 'use strict'; angular.module('forms').service('TimeCounter', [ function(){ var _startTime, _endTime = null, that=this; this.timeSpent = 0; this.restartClock = function(){ _startTime = Date.now(); _endTime = null; // console.log('Clock Started'); }; this.getTimeElapsed = function(){ if(_startTime) { return Math.abs(Date.now().valueOf() - _startTime.valueOf()) / 1000; } }; this.stopClock = function(){ if(_startTime && _endTime === null){ _endTime = Date.now(); this.timeSpent = Math.abs(_endTime.valueOf() - _startTime.valueOf())/1000; this._startTime = this._endTime = null; return this.timeSpent; }else{ return new Error('Clock has not been started'); } }; this.clockStarted = function(){ return !!this._startTime; }; } ]); 'use strict'; (function() { // Forms Controller Spec describe('AdminForm Controller Tests', function() { // Initialize global variables var AdminFormController, createAdminFormController, scope, $httpBackend, $stateParams, $location, $state; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9' }; var sampleForm = { title: 'Form Title', admin: 'ed873933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false, _id:'56340745f59a6fc9e22028e9'}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false, _id:'5c9e22028e907634f45f59a6'}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false, _id:'56e90745f5934fc9e22028a6'} ], _id: '525a8422f6d0f87f0e407a33' }; var expectedForm = { title: 'Form Title', admin: 'ed873933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false, _id:'56340745f59a6fc9e22028e9'}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false, _id:'5c9e22028e907634f45f59a6'}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false, _id:'56e90745f5934fc9e22028a6'} ], visible_form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false, _id:'56340745f59a6fc9e22028e9'}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false, _id:'5c9e22028e907634f45f59a6'}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false, _id:'56e90745f5934fc9e22028a6'} ], _id: '525a8422f6d0f87f0e407a33' }; var newFakeModal = function(){ var result = { opened: true, result: { then: function(confirmCallback, cancelCallback) { //Store the callbacks for later when the user clicks on the OK or Cancel button of the dialog this.confirmCallBack = confirmCallback; this.cancelCallback = cancelCallback; } }, close: function( item ) { //The user clicked OK on the modal dialog, call the stored confirm callback with the selected item this.opened = false; this.result.confirmCallBack( item ); }, dismiss: function( type ) { //The user clicked cancel on the modal dialog, call the stored cancel callback this.opened = false; this.result.cancelCallback( type ); } }; return result; }; //Mock Users Service beforeEach(module(function($provide) { $provide.service('myForm', function($q) { return sampleForm; }); })); // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(module('stateMock')); //Mock Users Service beforeEach(module(function($provide) { $provide.service('User', function($q) { return { getCurrent: function() { var deferred = $q.defer(); deferred.resolve( JSON.stringify(sampleUser) ); return deferred.promise; }, login: function(credentials) { var deferred = $q.defer(); if( credentials.password === sampleUser.password && credentials.username === sampleUser.username){ deferred.resolve( JSON.stringify(sampleUser) ); }else { deferred.resolve('Error: User could not be loggedin'); } return deferred.promise; }, logout: function() { var deferred = $q.defer(); deferred.resolve(null); return deferred.promise; }, signup: function(credentials) { var deferred = $q.defer(); if( credentials.password === sampleUser.password && credentials.username === sampleUser.username){ deferred.resolve( JSON.stringify(sampleUser) ); }else { deferred.resolve('Error: User could not be signed up'); } return deferred.promise; } }; }); })); //Mock Authentication Service beforeEach(module(function($provide) { $provide.service('Auth', function() { return { ensureHasCurrentUser: function() { return sampleUser; }, isAuthenticated: function() { return true; }, getUserState: function() { return true; } }; }); })); //Mock $uibModal beforeEach(inject(function($uibModal) { var modal = newFakeModal(); spyOn($uibModal, 'open').and.returnValue(modal); //spyOn($uibModal, 'close').and.callFake(modal.close()); })); // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // This allows us to inject a service but then attach it to a variable // with the same name as the service. beforeEach(inject(function($controller, $rootScope, _$state_, _$location_, _$stateParams_, _$httpBackend_, CurrentForm, Forms) { // Set a new global scope scope = $rootScope.$new(); //Set CurrentForm CurrentForm.setForm(sampleForm); // Point global variables to injected services $stateParams = _$stateParams_; $httpBackend = _$httpBackend_; $location = _$location_; $state = _$state_; $httpBackend.whenGET(/\.html$/).respond(''); $httpBackend.whenGET('/users/me/').respond(''); // Initialize the Forms controller. createAdminFormController = function(){ return $controller('AdminFormController', { $scope: scope }); }; })); it('AdminFormController should fetch current Form when instantiated', function() { // Run controller functionality var controller = createAdminFormController(); // Test scope value expect(scope.myform).toEqualData(sampleForm); }); it('$scope.removeCurrentForm() with valid form data should send a DELETE request with the id of form', function() { var controller = createAdminFormController(); //Set $state transition $state.expectTransitionTo('listForms'); // Set DELETE response $httpBackend.expect('DELETE', /^(\/forms\/)([0-9a-fA-F]{24})$/).respond(200, sampleForm); //Run controller functionality scope.openDeleteModal(); scope.removeCurrentForm(); $httpBackend.flush(); $state.ensureAllTransitionsHappened(); }); it('$scope.update() should send a PUT request with the id of form', function() { var controller = createAdminFormController(); //Set PUT response $httpBackend.expect('PUT', /^(\/forms\/)([0-9a-fA-F]{24})$/).respond(200, sampleForm); //Run controller functionality scope.update(false, null); $httpBackend.flush(); }); it('$scope.openDeleteModal() should open scope.deleteModal', function() { var controller = createAdminFormController(); //Run controller functionality scope.openDeleteModal(); console.log(scope.deleteModal); expect(scope.deleteModal.opened).toEqual(true); }); it('$scope.cancelDeleteModal() should close $scope.deleteModal', inject(function($uibModal) { var controller = createAdminFormController(); //Run controller functionality scope.openDeleteModal(); //Run controller functionality scope.cancelDeleteModal(); expect( scope.deleteModal.opened ).toEqual(false); })); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('ListForms Controller Tests', function() { // Initialize global variables var ListFormsController, createListFormsController, scope, $httpBackend, $stateParams, $location, $state; var sampleForm = { title: 'Form Title', admin: 'ed873933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} ], _id: '525a8422f6d0f87f0e407a33' }; var sampleFormList = [{ title: 'Form Title1', admin: 'ed873933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} ], _id: '525a8422f6d0f87f0e407a33' },{ title: 'Form Title2', admin: '39223933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} ], _id: '52f6d0f87f5a407a384220e3' },{ title: 'Form Title3', admin: '2fab9ed873937f0e1dea0ce1', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} ], _id: '922f6d0f87fed8730e4e1233' } ]; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(module('stateMock')); //Mock Users Service beforeEach(module(function($provide) { $provide.service('myForm', function($q) { return sampleForm; }); })); // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // This allows us to inject a service but then attach it to a variable // with the same name as the service. beforeEach(inject(function($controller, $rootScope, _$state_, _$location_, _$stateParams_, _$httpBackend_, CurrentForm, Forms) { // Set a new global scope scope = $rootScope.$new(); //Set CurrentForm CurrentForm.setForm(sampleForm); // Point global variables to injected services $stateParams = _$stateParams_; $httpBackend = _$httpBackend_; $location = _$location_; $state = _$state_; $httpBackend.whenGET(/\.html$/).respond(''); $httpBackend.whenGET('/users/me/').respond(''); // Initialize the Forms controller. createListFormsController = function(){ return $controller('ListFormsController', { $scope: scope }); }; })); it('$scope.findAll() should query all User\'s Forms', inject(function(Forms) { var controller = createListFormsController(); // Set GET response $httpBackend.expectGET(/^(\/forms)$/).respond(200, sampleFormList); // Run controller functionality scope.findAll(); $httpBackend.flush(); // Test scope value expect( scope.myforms ).toEqualData(sampleFormList); })); it('$scope.duplicateForm() should duplicate a Form', inject(function(Forms) { var dupSampleForm = sampleFormList[2], dupSampleForm_index = 3, newSampleFormList = _.clone(sampleFormList); dupSampleForm._id = 'a02df75b44c1d26b6a5e05b8'; newSampleFormList.splice(3, 0, dupSampleForm); var controller = createListFormsController(); // Set GET response $httpBackend.expectGET(/^(\/forms)$/).respond(200, sampleFormList); // Run controller functionality scope.findAll(); $httpBackend.flush(); // Set GET response $httpBackend.expect('POST', '/forms').respond(200, dupSampleForm); // Run controller functionality scope.duplicateForm(2); $httpBackend.flush(); // Test scope value expect( scope.myforms.length ).toEqual(newSampleFormList.length); for(var i=0; i'); $compile(el)(tmp_scope); $rootScope.$digest(); // Point global variables to injected services $httpBackend = _$httpBackend_; // $httpBackend.whenGET(/.+\.html$/).respond(''); $httpBackend.whenGET('/users/me/').respond(''); //Grab controller instance controller = el.controller(); //Grab scope. Depends on type of scope. //See angular.element documentation. scope = el.isolateScope() || el.scope(); })); it('$scope.uploadPDF() should upload a pdf file', function() { // expect(scope.isInitialized).toBeDefined() // expect(scope.log).toEqual(''); expect(scope.pdfLoading).toBe(false); //Set POST response $httpBackend.when('POST', '/upload/pdf').respond(pdfObj); var files = [{}]; scope.uploadPDF(files); $httpBackend.flush(); expect(scope.myform.pdf).toEqualData(pdfObj); }); it('$scope.removePDF() should removed uploaded pdf file', function() { // expect(scope.isInitialized).toBeDefined() // expect(scope.log).toEqual(''); scope.myform.pdf = pdfObj; scope.myform.isGenerated = true; scope.myform.autofillPDFs = true; scope.removePDF(); expect(scope.myform.pdf).toEqual(null); expect(scope.myform.isGenerated).toBe(false); expect(scope.myform.autofillPDFs).toBe(false); }); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('EditSubmissions Directive-Controller Tests', function() { // Initialize global variables var el, scope, controller, $httpBackend; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9' }; var pdfObj = { fieldname:'file', originalname:'test.pdf', name:'1440112660375.pdf', encoding:'7bit', mimetype:'application/pdf', path:'uploads/tmp/test@test.com/1440112660375.pdf', extension:'pdf', size:56223, truncated:false, buffer:null }; var sampleForm = { title: 'Form Title', admin: 'ed873933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed873933b0ce121f1deafab9'}, {fieldType:'checkbox', title:'nascar', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed83b0ce121f17393deafab9'}, {fieldType:'checkbox', title:'hockey', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed8317393deab0ce121ffab9'} ], pdf: {}, pdfFieldMap: {}, startPage: { showStart: false }, hideFooter: false, isGenerated: false, isLive: false, autofillPDFs: false, _id: '525a8422f6d0f87f0e407a33' }; var sampleSubmission = { form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'John Smith', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: 1, deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: 0, deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 17.55 }; var sampleSubmissions = [{ form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'The Terminator', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: 0, deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: 1, deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 10.33 }, { form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'John Smith', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: 1, deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: 0, deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 2.33 }, { form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'Jane Doe', deletePreserved: false}, {fieldType:'checkbox', title:'nascar', fieldValue: 1, deletePreserved: false}, {fieldType:'checkbox', title:'hockey', fieldValue: 1, deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 11.11 }]; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(module('module-templates')); beforeEach(module('stateMock')); beforeEach(inject(function($compile, $controller, $rootScope, _$httpBackend_) { // Point global variables to injected services $httpBackend = _$httpBackend_; $httpBackend.whenGET('/users/me/').respond(''); $httpBackend.whenGET(/^(\/forms\/)([0-9a-fA-F]{24})(\/submissions)$/).respond(200, sampleSubmissions); //Instantiate directive. var tmp_scope = $rootScope.$new(); tmp_scope.myform = sampleForm; tmp_scope.user = sampleUser; //gotacha: Controller and link functions will execute. el = angular.element(''); $compile(el)(tmp_scope); $rootScope.$digest(); //Grab controller instance controller = el.controller(); //Grab scope. Depends on type of scope. //See angular.element documentation. scope = el.isolateScope() || el.scope(); })); it('$scope.initFormSubmissions() should fetch all relevant form submissions', function() { $httpBackend.expectGET(/^(\/forms\/)([0-9a-fA-F]{24})(\/submissions)$/).respond(200, sampleSubmissions); scope.initFormSubmissions(); $httpBackend.flush(); scope.$digest(); }); describe('Form Table Methods', function(){ it('$scope.toggleAllCheckers should toggle all checkboxes in table', function(){ scope.initFormSubmissions(); $httpBackend.flush(); //Run Controller Logic to Test scope.table.masterChecker = true; scope.toggleAllCheckers(); for(var i=0; i'); $compile(el)(tmp_scope); $rootScope.$digest(); // Point global variables to injected services $httpBackend = _$httpBackend_; //$httpBackend.whenGET(/.+\.html$/).respond(''); $httpBackend.whenGET('/users/me/').respond(''); //Grab controller instance controller = el.controller(); //Grab scope. Depends on type of scope. //See angular.element documentation. scope = el.isolateScope() || el.scope(); })); describe('> Form Field >',function(){ beforeEach(function(){ scope.myform = _.cloneDeep(sampleForm); }); it('$scope.addNewField() should ADD a new field to $scope.myform.form_fields', function() { //Run controller methods scope.addNewField(true, 'textfield'); var expectedFormField = { title:'Short Text2', fieldType:'textfield', fieldValue: '', required: true, disabled: false, deletePreserved: false }; var actualFormField = _.cloneDeep(_.last(scope.myform.form_fields)); delete actualFormField._id; expect(scope.myform.form_fields.length).toEqual(sampleForm.form_fields.length+1); expect(actualFormField).toEqualData(expectedFormField); }); it('$scope.deleteField() should DELETE a field to $scope.myform.form_fields', function() { //Run controller methods scope.deleteField(0); expect(scope.myform.form_fields.length).toEqual(sampleForm.form_fields.length-1); expect(_.first(scope.myform.form_fields)).toEqualData(sampleForm.form_fields[1]); }); it('$scope.duplicateField() should DUPLICATE a field and update $scope.myform.form_fields', function() { //Run controller methods scope.duplicateField(0); var originalField = _.cloneDeep(scope.myform.form_fields[0]); originalField.title += ' copy'; delete originalField._id; var copyField = _.cloneDeep(scope.myform.form_fields[1]); delete copyField._id; expect(scope.myform.form_fields.length).toEqual(sampleForm.form_fields.length+1); expect(originalField).toEqualData(copyField); }); }); describe('> Form Field Button >',function(){ it('$scope.addButton() should ADD a button to $scope.myform.startPage.buttons', function() { var expectedStartPageBtn = { bgColor:'#ddd', color:'#ffffff', text: 'Button' }; //Run controller methods scope.addButton(); var actualStartPageBtn = _.cloneDeep(_.last(scope.myform.startPage.buttons)); delete actualStartPageBtn._id; expect(scope.myform.startPage.buttons.length).toEqual(sampleForm.startPage.buttons.length+1); expect(actualStartPageBtn).toEqualData(expectedStartPageBtn); }); it('$scope.deleteButton() should DELETE a button from $scope.myform.startPage.buttons', function() { //Run controller methods scope.deleteButton(scope.myform.startPage.buttons[0]); expect(scope.myform.startPage.buttons.length).toEqual(0); }); }); describe('> Form Field Option >',function(){ it('$scope.addOption() should ADD a new option to a field.fieldOptions', function() { var originalOptionLen = scope.myform.form_fields[1].fieldOptions.length; //Run controller methods scope.addOption(1); expect(originalOptionLen+1).toEqual(scope.myform.form_fields[1].fieldOptions.length); expect(scope.myform.form_fields[1].fieldOptions[0].option_title).toEqualData('Option 0'); expect(scope.myform.form_fields[1].fieldOptions[0].option_value).toEqualData('Option 0'); }); it('$scope.deleteOption() should DELETE remove option from field.fieldOptions', function() { //Run controller methods scope.deleteOption(1, scope.myform.form_fields[1].fieldOptions[0]); expect(scope.myform.form_fields[0].fieldOptions.length).toEqual(0); expect(scope.myform.form_fields[0].fieldOptions[0]).not.toBeDefined(); }); }); }); }()); // 'use strict'; // (function() { // // Forms Controller Spec // describe('entryPage Directive Tests', function() { // // Initialize global variables // var scope, // $templateCache, // $httpBackend, // $compile; // var sampleStartPage = { // showStart: true, // introTitle: 'Welcome to Form', // introParagraph: 'Sample intro paragraph', // buttons:[ // { // url: 'http://google.com', // action: '', // text: 'Google', // bgColor: '#ffffff', // color: '#000000', // }, // { // url: 'http://facebook.com', // action: '', // text: 'Facebook', // bgColor: '#0000ff', // color: '#000000', // } // ] // }; // // The $resource service augments the response object with methods for updating and deleting the resource. // // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // // When the toEqualData matcher compares two objects, it takes only object properties into // // account and ignores methods. // beforeEach(function() { // jasmine.addMatchers({ // toEqualData: function(util, customEqualityTesters) { // return { // compare: function(actual, expected) { // return { // pass: angular.equals(actual, expected) // }; // } // }; // } // }); // }); // // Load the main application module // beforeEach(module(ApplicationConfiguration.applicationModuleName)); // beforeEach(inject(function($rootScope, _$compile_, _$httpBackend_) { // scope = $rootScope.$new(); // $compile = _$compile_; // // Point global variables to injected services // $httpBackend = _$httpBackend_; // })); // it('should be able to render entryPage in html', function() { // scope.myStartPage = _.cloneDeep(sampleStartPage); // console.log(scope.myStartPage); // var element = angular.element(''); // $compile(element)(scope); // scope.$digest(); // // console.log(element.html()); // expect(element.html()).not.toEqual('
Start Page
'); // }); // // it('exitStartPage should work for "startPage" type of entryPage', inject(function($rootScope) { // // scope.myPage = _.cloneDeep(sampleStartPage); // // var el = angular.element(''); // // $compile(el)(scope); // // scope.$digest(); // // $httpBackend.whenGET(/.+\.html$/).respond(''); // // $httpBackend.whenGET('/users/me/').respond(''); // // scope = el.isolateScope() || el.scope(); // // scope.exitStartPage(); // // // expect(scope.myStartPage.showStart).toBe(false); // // expect(el.html()).not.toEqual('
Start Page
'); // // })); // }); // }()); 'use strict'; (function() { // Forms Controller Spec describe('FieldIcon Directive Tests', function() { // Initialize global variables var scope, FormFields, faClasses = { 'textfield': 'fa fa-pencil-square-o', 'dropdown': 'fa fa-th-list', 'date': 'fa fa-calendar', 'checkbox': 'fa fa-check-square-o', 'radio': 'fa fa-dot-circle-o', 'email': 'fa fa-envelope-o', 'textarea': 'fa fa-pencil-square', 'legal': 'fa fa-legal', 'file': 'fa fa-cloud-upload', 'rating': 'fa fa-star-half-o', 'link': 'fa fa-link', 'scale': 'fa fa-sliders', 'stripe': 'fa fa-credit-card', 'statement': 'fa fa-quote-left', 'yes_no': 'fa fa-toggle-on', 'number': 'fa fa-slack' }; // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(inject(function ($rootScope, _FormFields_) { scope = $rootScope.$new(); FormFields = _FormFields_; })); it('should be able render all field-icon types', inject(function($compile) { var currType, currClass; for(var i=0; i')(scope); scope.$digest(); expect(currClass).toBeDefined(); expect(element.find('i')).not.toBe(null); expect(element.find('i').hasClass(currClass)).toBe(true); } })); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('Field Directive Tests', function() { // Initialize global variables var scope, FormFields, $templateCache, $httpBackend, $compile; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9', }; var sampleFields = [ {fieldType:'textfield', title:'First Name', fieldValue: 'AoeuName', deletePreserved: false, required: true, disabled: false}, {fieldType:'email', title:'Email', fieldValue: 'aoeu@aoeu.com', deletePreserved: false, required: true, disabled: false}, {fieldType:'yes_no', title:'Do you Play Hockey?', fieldValue: 'true', deletePreserved: false, required: true, disabled: false}, {fieldType:'url', title:'Github Account', fieldValue: 'http://github.com/aoeu', deletePreserved: false, required: true, disabled: false}, {fieldType:'textarea', title:'Bio', fieldValue: 'This is my bio.', deletePreserved: false, required: true, disabled: false}, {fieldType:'number', title:'Phone #', fieldValue: 5325325325, deletePreserved: false, required: true, disabled: false}, {fieldType:'legal', title:'You agree to terms and conditions', description:'By selecting \'I agree\' you are agreeing under Canadian law that you have read and accept terms and conditions outlayed below', fieldValue: '', deletePreserved: false, required: true, disabled: false}, {fieldType:'dropdown', title:'Your Sex', fieldValue: '', fieldOptions:[ { 'option_id': 0, 'option_title': 'M', 'option_value': 'male' }, { 'option_id': 1, 'option_title': 'F', 'option_value': 'female' }], deletePreserved: false, required: true, disabled: false}, {fieldType:'radio', title:'Your Sexual Orientation', fieldValue: '', fieldOptions:[ { 'option_id': 0, 'option_title': 'Heterosexual', 'option_value': 'hetero' }, { 'option_id': 1, 'option_title': 'Homosexual', 'option_value': 'homo' }, { 'option_id': 2, 'option_title': 'Bisexual', 'option_value': 'bi' }, { 'option_id': 3, 'option_title': 'Asexual', 'option_value': 'asex' }], deletePreserved: false, required: true, disabled: false}, {fieldType:'rating', title:'Your Current Happiness', fieldValue: '0', deletePreserved: false, required: true, disabled: false}, ]; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); beforeEach(module(function ($sceProvider) { $sceProvider.enabled(false); })); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(module('stateMock')); beforeEach(module('module-templates')); beforeEach(module('ngSanitize', 'ui.select')); beforeEach(inject(function($rootScope, _FormFields_, _$compile_) { scope = $rootScope.$new(); FormFields = _FormFields_; $compile = _$compile_; })); it('should be able to render all field types in html', inject(function($rootScope) { scope.fields = sampleFields; for(var i=0; i'); $compile(element)(scope); scope.$digest(); console.log('Actual: '); console.log(element.html()); console.log('\nExpected: '); console.log('
'+field.title+'
'); } })); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('onFinishRender Directive Tests', function() { // Initialize global variables var scope, FormFields; // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(inject(function ($rootScope, _FormFields_) { scope = $rootScope.$new(); FormFields = _FormFields_; spyOn($rootScope, '$broadcast'); })); it('should emit Custom "Finished" and "Started" events on ng-repeat', inject(function($compile, $rootScope) { scope.myfields = FormFields.types; var e = $compile('
{{item.name}}
')(scope); scope.$digest(); //run code to test expect($rootScope.$broadcast).toHaveBeenCalledWith('editFormFields Started'); expect(scope.$broadcast).toHaveBeenCalledWith('editFormFields Finished'); })); it('should emit "ngRepeat Finished" and "ngRepeat Started" events on ng-repeat when attr is not set to string', inject(function($compile, $rootScope) { // console.log(FormFields.types); scope.myfields = FormFields.types; var e = $compile('
{{item.name}}
')(scope); scope.$digest(); //run code to test expect($rootScope.$broadcast).toHaveBeenCalledWith('ngRepeat Started'); expect(scope.$broadcast).toHaveBeenCalledWith('ngRepeat Finished'); })); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('SubmitForm Directive-Controller Tests', function() { // Initialize global variables var scope, controller, $httpBackend; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9' }; var pdfObj = { fieldname:'file', originalname:'test.pdf', name:'1440112660375.pdf', encoding:'7bit', mimetype:'application/pdf', path:'uploads/tmp/test@test.com/1440112660375.pdf', extension:'pdf', size:56223, truncated:false, buffer:null }; var sampleForm = { title: 'Form Title', admin: 'ed873933b1f1dea0ce12fab9', language: 'english', form_fields: [ {fieldType:'textfield', title:'First Name', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed873933b0ce121f1deafab9'}, {fieldType:'checkbox', title:'nascar', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed83b0ce121f17393deafab9'}, {fieldType:'checkbox', title:'hockey', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed8317393deab0ce121ffab9'} ], visible_form_fields: [ {fieldType:'textfield', title:'First Name', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed873933b0ce121f1deafab9'}, {fieldType:'checkbox', title:'nascar', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed83b0ce121f17393deafab9'}, {fieldType:'checkbox', title:'hockey', fieldOptions: [], fieldValue: '', required: true, disabled: false, deletePreserved: false, _id: 'ed8317393deab0ce121ffab9'} ], pdf: {}, pdfFieldMap: {}, startPage: { showStart: false }, hideFooter: false, isGenerated: false, isLive: false, autofillPDFs: false, _id: '525a8422f6d0f87f0e407a33' }; var sampleSubmission = { form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'John Smith', deletePreserved: false, _id: 'ed873933b0ce121f1deafab9'}, {fieldType:'yes_no', title:'Do you like nascar', fieldValue: true, deletePreserved: false, _id: 'ed83b0ce121f17393deafab9'}, {fieldType:'yes_no', title:'Do you like hockey', fieldValue: false, deletePreserved: false, _id: 'ed8317393deab0ce121ffab9'} ], admin: sampleUser, form: sampleForm, timeElapsed: 17.55 }; var sampleSubmissions = [{ form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'The Terminator', deletePreserved: false}, {fieldType:'yes_no', title:'Do you like nascar', fieldValue: 'true', deletePreserved: false}, {fieldType:'yes_no', title:'Do you like hockey', fieldValue: 'false', deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 10.33 }, { form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'John Smith', deletePreserved: false}, {fieldType:'yes_no', title:'Do you like nascar', fieldValue: 'true', deletePreserved: false}, {fieldType:'yes_no', title:'Do you like hockey', fieldValue: 'true', deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 2.33 }, { form_fields: [ {fieldType:'textfield', title:'First Name', fieldValue: 'Jane Doe', deletePreserved: false}, {fieldType:'yes_no', title:'Do you like nascar', fieldValue: 'false', deletePreserved: false}, {fieldType:'yes_no', title:'Do you like hockey', fieldValue: 'false', deletePreserved: false} ], admin: sampleUser, form: sampleForm, timeElapsed: 11.11 }]; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(module('module-templates')); beforeEach(module('stateMock')); beforeEach(inject(function($compile, $controller, $rootScope, _$httpBackend_) { // Point global variables to injected services $httpBackend = _$httpBackend_; $httpBackend.whenGET('/users/me/').respond(''); //Instantiate directive. var tmp_scope = $rootScope.$new(); tmp_scope.myform = sampleForm; //gotacha: Controller and link functions will execute. var el = angular.element(''); $compile(el)(tmp_scope); tmp_scope.$digest(); $rootScope.$digest(); //Grab controller instance controller = el.controller(); //Grab scope. Depends on type of scope. //See angular.element documentation. scope = el.isolateScope() || el.scope(); console.log(scope); })); var Validator = (function() { return { hasMinimumFields: function(entry) { return !_.isEmpty(entry._id) && !_.isEmpty(entry.title); }, isNewForm: function(entry) { return this.hasMinimumFields(entry); } }; })(); it('$scope.submitForm() should submit valid form', function(){ //Initialize variables scope.myform.form_fields = sampleSubmissions[0].form_fields; var expectedForm = _.cloneDeep(sampleForm); expectedForm.form_fields = sampleSubmissions[0].form_fields; delete expectedForm.visible_form_fields; var data = function(data) { var form = angular.fromJson(data); var compareForm = _.cloneDeep(form); delete compareForm.timeElapsed; delete compareForm.percentageComplete; return Validator.isNewForm(form) && _.isEqual(compareForm, expectedForm); }; //Set expected HTTP requests $httpBackend.expect('POST',/^(\/forms\/)([0-9a-fA-F]{24})$/, data).respond(200); //Run Controller Logic to Test scope.submitForm(); $httpBackend.flush(); setTimeout(function(){ expect(scope.myform.submitted).toBe(true); expect(scope.error).toEqual(''); }, 25); }); it('$scope.reloadForm() should reset and reload form', function(){ scope.submitForm(); scope.reloadForm(); expect(scope.myform.submitted).toBe(false); for(var i=0; i 0){ var expectedState = this.expectedTransitions.shift(); if(expectedState !== stateName){ throw Error('Expected transition to state: ' + expectedState + ' but transitioned to ' + stateName ); } }else{ throw Error('No more transitions were expected! Tried to transition to '+ stateName ); } console.log('Mock transition to: ' + stateName); var deferred = $q.defer(); var promise = deferred.promise; deferred.resolve(); return promise; }; this.go = this.transitionTo; this.expectTransitionTo = function(stateName){ this.expectedTransitions.push(stateName); }; this.ensureAllTransitionsHappened = function(){ if(this.expectedTransitions.length > 0){ throw Error('Not all transitions happened!'); } }; }]); // 'use strict'; // (function() { // // Principal controller Spec for E2E Tests // describe('AuthenticationController E2E Tests', function() { // describe('/signup should work for a unique username', function() { // beforeEach(function() { // var ptor = protractor.getInstance(); // ptor.get('http://localhost:3000/#!/signup'); // }); // it('should show the signup panel on page load', function() { // expect($('section > section.row.auth > .col-md-12.text-center')).toEqual('Signup with your email'); // }); // //Jasmine it statement : What "it" will do. // it('Verify that the user is logged in', function() { // //Delete all cookies // browser.driver.manage().deleteAllCookies(); // //Enter UserName // element.all(by.model('username')).get(0).sendKeys('abc@wingify.com'); // //Enter Password // element(by.model('password')).sendKeys('test'); // //Click Submit button // element(by.css('.login-form button[type="submit"]')).click(); // //Wait for the current URL to change to welcome // browser.driver.wait(function() { // return browser.driver.getCurrentUrl().then(function(url) { // return (/welcome/).test(url); // }); // }); // var firstname = element(by.model('credentials.firstname')), // lastname = element(by.model('credentials.lastname')), // email = element(by.model('credentials.email')), // password = element(by.model('credentials.password')); // email.sendKeys('admin@app.com'); // firstname.sendKeys('admin_first'); // lastname.sendKeys('admin_last'); // password.sendKeys('1234'); // //Click signup button // element(by.css('.btn.btn-large.btn-primary')).click().then(function () { // expect(browser.getCurrentUrl()).toEqual('http://localhost:3000/#!/signup-success'); // }); // }); // }); // }); // // Principal controller Spec // describe('AuthenticationController Unit Tests', function() { // // Initialize global variables // var AuthenticationController, // scope, // $httpBackend, // $stateParams, // $location; // beforeEach(function() { // jasmine.addMatchers({ // toEqualData: function(util, customEqualityTesters) { // return { // compare: function(actual, expected) { // return { // pass: angular.equals(actual, expected) // }; // } // }; // } // }); // }); // // Load the main application module // beforeEach(module(ApplicationConfiguration.applicationModuleName)); // // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // // This allows us to inject a service but then attach it to a variable // // with the same name as the service. // beforeEach(inject(function($controller, $rootScope, _$location_, _$stateParams_, _$httpBackend_) { // // Set a new global scope // scope = $rootScope.$new(); // // Point global variables to injected services // $stateParams = _$stateParams_; // $httpBackend = _$httpBackend_; // $location = _$location_; // // Initialize the Principal controller // AuthenticationController = $controller('AuthenticationController', { // $scope: scope // }); // })); // it('$scope.signin() should login with a correct user and password', function() { // // Test expected GET request // $httpBackend.when('POST', '/auth/signin').respond(200, 'Fred'); // scope.signin(); // $httpBackend.flush(); // // Test scope value // expect(scope.authentication.user).toEqual('Fred'); // expect($location.url()).toEqual('/'); // }); // it('$scope.signin() should fail to log in with nothing', function() { // // Test expected POST request // $httpBackend.expectPOST('/auth/signin').respond(400, { // 'message': 'Missing credentials' // }); // scope.signin(); // $httpBackend.flush(); // // Test scope value // expect(scope.error).toEqual('Missing credentials'); // }); // it('$scope.signin() should fail to log in with wrong credentials', function() { // // Foo/Bar combo assumed to not exist // scope.authentication.user = 'Foo'; // scope.credentials = 'Bar'; // // Test expected POST request // $httpBackend.expectPOST('/auth/signin').respond(400, { // 'message': 'Unknown user' // }); // scope.signin(); // $httpBackend.flush(); // // Test scope value // expect(scope.error).toEqual('Unknown user'); // }); // it('$scope.signup() should register with correct data', function() { // // Test expected GET request // scope.authentication.user = 'Fred'; // $httpBackend.when('POST', '/auth/signup').respond(200, 'Fred'); // scope.signup(); // $httpBackend.flush(); // // test scope value // expect(scope.authentication.user).toBe('Fred'); // expect(scope.error).toEqual(undefined); // expect($location.url()).toBe('/'); // }); // it('$scope.signup() should fail to register with duplicate Username', function() { // // Test expected POST request // $httpBackend.when('POST', '/auth/signup').respond(400, { // 'message': 'Username already exists' // }); // scope.signup(); // $httpBackend.flush(); // // Test scope value // expect(scope.error).toBe('Username already exists'); // }); // }); // }()); // 'use strict'; // (function() { // // Forms Controller Spec // describe('Authentication Controller Tests', function() { // // Initialize global variables // var AuthenticationController, // scope, // $httpBackend, // $stateParams, // $location, // $state; // var sampleUser = { // firstName: 'Full', // lastName: 'Name', // email: 'test@test.com', // username: 'test@test.com', // password: 'password', // provider: 'local', // roles: ['user'], // _id: 'ed873933b1f1dea0ce12fab9' // }; // var sampleForm = { // title: 'Form Title', // admin: 'ed873933b1f1dea0ce12fab9', // language: 'english', // form_fields: [ // {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, // {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, // {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} // ], // _id: '525a8422f6d0f87f0e407a33' // }; // var expectedForm = { // title: 'Form Title', // admin: 'ed873933b1f1dea0ce12fab9', // language: 'english', // form_fields: [ // {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, // {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, // {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} // ], // visible_form_fields: [ // {fieldType:'textfield', title:'First Name', fieldValue: '', deletePreserved: false}, // {fieldType:'checkbox', title:'nascar', fieldValue: '', deletePreserved: false}, // {fieldType:'checkbox', title:'hockey', fieldValue: '', deletePreserved: false} // ], // _id: '525a8422f6d0f87f0e407a33' // }; // var sampleCredentials = { // username: sampleUser.username, // password: sampleUser.password, // }; // // The $resource service augments the response object with methods for updating and deleting the resource. // // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // // When the toEqualData matcher compares two objects, it takes only object properties into // // account and ignores methods. // beforeEach(function() { // jasmine.addMatchers({ // toEqualData: function(util, customEqualityTesters) { // return { // compare: function(actual, expected) { // return { // pass: angular.equals(actual, expected) // }; // } // }; // } // }); // }); // // Load the main application module // beforeEach(module(ApplicationConfiguration.applicationModuleName)); // beforeEach(module('stateMock')); // // Mock Users Service // beforeEach(module(function($provide) { // $provide.service('User', function($q) { // return { // getCurrent: function() { // var deferred = $q.defer(); // deferred.resolve( JSON.stringify(sampleUser) ); // return deferred.promise; // }, // login: function(credentials) { // var deferred = $q.defer(); // if( credentials.password === sampleUser.password && credentials.username === sampleUser.username){ // deferred.resolve( JSON.stringify(sampleUser) ); // }else { // deferred.resolve('Error: User could not be loggedin'); // } // return deferred.promise; // }, // logout: function() { // var deferred = $q.defer(); // deferred.resolve(null); // return deferred.promise; // }, // signup: function(credentials) { // var deferred = $q.defer(); // if( credentials.password === sampleUser.password && credentials.username === sampleUser.username){ // deferred.resolve( JSON.stringify(sampleUser) ); // }else { // deferred.resolve('Error: User could not be signed up'); // } // return deferred.promise; // } // }; // }); // })); // // Mock Authentication Service // beforeEach(module(function($provide) { // $provide.service('Auth', function() { // return { // ensureHasCurrentUser: function() { // return sampleUser; // }, // isAuthenticated: function() { // return true; // }, // getUserState: function() { // return true; // } // }; // }); // })); // // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // // This allows us to inject a service but then attach it to a variable // // with the same name as the service. // beforeEach(inject(function($controller, $rootScope, _$state_, _$location_, _$stateParams_, _$httpBackend_, CurrentForm, Forms) { // // Set a new global scope // scope = $rootScope.$new(); // scope.abc = 'hello'; // // Point global variables to injected services // $stateParams = _$stateParams_; // $httpBackend = _$httpBackend_; // $location = _$location_; // $state = _$state_; // // $httpBackend.whenGET(/\.html$/).respond(''); // $httpBackend.whenGET('/users/me/').respond(''); // // Initialize the Forms controller. // AuthenticationController = $controller('AuthenticationController', { $scope: scope }); // })); // it('$scope.signin should sigin in user with valid credentials', inject(function(Auth) { // //Set $state transition // // $state.expectTransitionTo('listForms'); // //Set POST response // // $httpBackend.expect('POST', '/auth/signin', sampleCredentials).respond(200, sampleUser); // scope.abc = 'sampleCredentials'; // //Run Controller Logic to Test // scope.signin(); // // $httpBackend.flush(); // // Test scope value // // expect(Auth.ensureHasCurrentUser()).toEqualData(sampleUser); // })); // }); // }()); 'use strict'; (function() { // Forms Controller Spec describe('Auth Service Tests', function() { // Initialize global variables var Auth; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9' }; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // This allows us to inject a service but then attach it to a variable // with the same name as the service. beforeEach(inject(function(_Auth_) { Auth = _Auth_; })); it('Auth.login() should save user in Auth.currentUser', function() { //Run Service Logic to Test Auth.login(sampleUser); expect(Auth.currentUser).toEqualData(sampleUser); }); it('Auth.logout() should remove saved user', inject(function($window) { //Run Service Logic to Test Auth.logout(); expect($window.user).toEqual(null); expect(Auth.currentUser).toEqual(null); expect(Auth.isAuthenticated()).toBe(false); expect(Auth.getUserState().isLoggedIn).toBe(false); })); it('Auth.getUserState() should fetch current user state', function() { //Run Service Logic to Test Auth.login(sampleUser); var currUserState = Auth.getUserState(); expect(currUserState.isLoggedIn).toBe(true); //Run Service Logic to Test Auth.logout(); currUserState = Auth.getUserState(); expect(currUserState.isLoggedIn).toBe(false); }); it('Auth.ensureHasCurrentUser() should fetch most current user if it exists in $window, currentUser or fetch it from /users/me', function() { Auth.login(sampleUser); //Run Service Logic to Test var currUser = Auth.ensureHasCurrentUser(sampleUser); expect(currUser).not.toEqual(null); expect(currUser).toEqualData(sampleUser); }); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('Authorizer Service Tests', function() { // Initialize global variables var Authorizer; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9' }; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // This allows us to inject a service but then attach it to a variable // with the same name as the service. beforeEach(inject(function(_Authorizer_) { Authorizer = _Authorizer_; })); it('Authorizer.canAccess() should return expected values for \'admin\' and \'user\' accounts', function() { var sampleAdminUser = _.cloneDeep(sampleUser); sampleAdminUser.roles.push('admin'); //Run Service Logic to Test var authenticatorUser = new Authorizer(sampleUser); var authenticatorAdmin = new Authorizer(sampleAdminUser); expect(authenticatorUser.canAccess('editForm')).toBe(true); expect(authenticatorUser.canAccess('editAdminSettings')).toBe(false); expect(authenticatorUser.canAccess('viewAdminSettings')).toBe(false); expect(authenticatorAdmin.canAccess('editForm')).toBe(true); expect(authenticatorAdmin.canAccess('editAdminSettings')).toBe(true); expect(authenticatorAdmin.canAccess('viewAdminSettings')).toBe(true); }); }); }()); 'use strict'; (function() { // Forms Controller Spec describe('User Service Tests', function() { // Initialize global variables var User, $httpBackend; var sampleUser = { firstName: 'Full', lastName: 'Name', email: 'test@test.com', username: 'test@test.com', password: 'password', provider: 'local', roles: ['user'], _id: 'ed873933b1f1dea0ce12fab9' }; var sampleVerifyToken = 'WyuAIchArQnstkq5erx0kiTcTbBbgixYeBGtThFmRpcAJNQ2'; var sampleForgotToken = 'c2e8f74455cdccc454dfef941ff315fa4f7b1f0a'; var sampleCredentials = { username: sampleUser.username, password: sampleUser.password, }; var samplePasswordDetails = { newPassword: sampleUser.password, verifyPassword: sampleUser.password, }; // The $resource service augments the response object with methods for updating and deleting the resource. // If we were to use the standard toEqual matcher, our tests would fail because the test values would not match // the responses exactly. To solve the problem, we define a new toEqualData Jasmine matcher. // When the toEqualData matcher compares two objects, it takes only object properties into // account and ignores methods. beforeEach(function() { jasmine.addMatchers({ toEqualData: function(util, customEqualityTesters) { return { compare: function(actual, expected) { return { pass: angular.equals(actual, expected) }; } }; } }); }); // Load the main application module beforeEach(module(ApplicationConfiguration.applicationModuleName)); beforeEach(module('stateMock')); // The injector ignores leading and trailing underscores here (i.e. _$httpBackend_). // This allows us to inject a service but then attach it to a variable // with the same name as the service. beforeEach(inject(function(_$httpBackend_, _User_) { // Point global variables to injected services $httpBackend = _$httpBackend_; User = _User_; })); it('User.login() should send a POST request to /auth/signin', function() { //Set POST response $httpBackend.expect('POST', '/auth/signin', sampleCredentials).respond(200, sampleUser); //Run Service Logic to Test User.login(sampleCredentials); $httpBackend.flush(); }); it('User.logout() should logout user with /auth/signout', function() { //Set POST response $httpBackend.expect('GET', '/auth/signout').respond(200); //Run Service Logic to Test User.logout(); $httpBackend.flush(); }); it('User.getCurrent() should fetch user from /users/me', function() { //Set POST response $httpBackend.expect('GET', '/users/me').respond(200, sampleUser); //Run Service Logic to Test User.getCurrent(); $httpBackend.flush(); }); it('User.signup() should signup user with /auth/signup', function() { //Set POST response $httpBackend.expect('POST', '/auth/signup', sampleCredentials).respond(200); //Run Service Logic to Test User.signup(sampleCredentials); $httpBackend.flush(); }); it('User.resendVerifyEmail() should send POST request to /auth/verify', function() { //Set POST response $httpBackend.expect('POST', '/auth/verify', {email: sampleUser.email}).respond(200); //Run Service Logic to Test User.resendVerifyEmail(sampleUser.email); $httpBackend.flush(); }); it('User.validateVerifyToken() should send GET request to /auth/verify/:token', function() { //Set POST response $httpBackend.expect('GET', '/auth/verify/'+sampleVerifyToken).respond(200); //Run Service Logic to Test expect(function(){ User.validateVerifyToken(sampleVerifyToken); }).not.toThrow(); $httpBackend.flush(); }); it('User.resetPassword() should send GET request to /auth/forgot/:token', function() { //Set POST response $httpBackend.expect('GET', '/auth/password/'+sampleForgotToken).respond(200); //Run Service Logic to Test User.resetPassword(samplePasswordDetails, sampleForgotToken); $httpBackend.flush(); }); it('User.askForPasswordReset() should send POST request to /auth/forgot', function() { //Set POST response $httpBackend.expect('POST', '/auth/forgot', sampleCredentials).respond(200, sampleUser); //Run Service Logic to Test User.askForPasswordReset(sampleCredentials); $httpBackend.flush(); }); }); }());