diff --git a/bin/drush-install.sh b/bin/drush-install.sh
index 356173273cb..e1079a09a7b 100755
--- a/bin/drush-install.sh
+++ b/bin/drush-install.sh
@@ -11,6 +11,7 @@ uk.co.compucorp.civicrm.hremails
## List of extensions defining basic entity types
ENTITY_EXTS=\
org.civicrm.hrbank,\
+org.civicrm.hrdemog,\
org.civicrm.hrjobcontract,\
com.civicrm.hrjobroles,\
org.civicrm.hrmed,\
diff --git a/bin/git-release.sh b/bin/git-release.sh
index 4cd9dfd2411..8e774a84937 100755
--- a/bin/git-release.sh
+++ b/bin/git-release.sh
@@ -5,6 +5,7 @@ fileName=""
ENTITY_EXTS=( hrbank \
hrcareer \
hrcase \
+hrdemog \
hrim \
hrmed \
hrprofile \
diff --git a/doc/INSTALL.md b/doc/INSTALL.md
index ee4b8e25247..f58606fcc40 100644
--- a/doc/INSTALL.md
+++ b/doc/INSTALL.md
@@ -16,32 +16,4 @@ The [civihr-installer](https://github.com/compucorp/civihr-installer) script has
$ civibuild create hr17
```
-Read the drush-install.sh for details.
-
-## Install (Option B: Manual)
-
-CiviHR includes over a dozen extensions. These can be activated piecemeal.
-The following extensions provide the major features and may be activated
-individually:
-
- * org.civicrm.hrbank: Bank Details
- * org.civicrm.hrcareer: Career History
- * org.civicrm.hremerg: Emergency Contacts
- * org.civicrm.hrabsence: Absences
- * org.civicrm.hrjobcontract: Job Contracts
- * org.civicrm.hrmed: Medical and Disability
- * org.civicrm.hrqual: Qualifications
- * org.civicrm.hrreport: Reporting
- * org.civicrm.hrstaffdir: Staff Directory
- * org.civicrm.hrcase: Case
- * org.civicrm.hrcaseutils: Case Utils
- * org.civicrm.hrim: Instant messanger link
- * org.civicrm.hrrecruitment: Recruitment
- * org.civicrm.hrprofile: Profile
-
-Finally, these two extensions build on top of the others:
-
- * uk.co.compucorp.civicrm.hrsampledata: Generate random example data
- * org.civicrm.hrui: Trim/revise CiviCRM UI for CiviHR users
-
Please check the [civibuild documentation](https://docs.civicrm.org/dev/en/latest/tools/civibuild/) for more information on all the available params.
diff --git a/hrdemog/CRM/HRDemog/Upgrader.php b/hrdemog/CRM/HRDemog/Upgrader.php
new file mode 100644
index 00000000000..e32e54c4e78
--- /dev/null
+++ b/hrdemog/CRM/HRDemog/Upgrader.php
@@ -0,0 +1,185 @@
+executeSqlFile('sql/myinstall.sql');
+ }
+
+ /**
+ * Example: Run an external SQL script when the module is uninstalled
+ *
+ public function uninstall() {
+ $this->executeSqlFile('sql/myuninstall.sql');
+ }
+
+ /**
+ * Example: Run a simple query when a module is enabled
+ *
+ public function enable() {
+ CRM_Core_DAO::executeQuery('UPDATE foo SET is_active = 1 WHERE bar = "whiz"');
+ }
+
+ /**
+ * Example: Run a simple query when a module is disabled
+ *
+ public function disable() {
+ CRM_Core_DAO::executeQuery('UPDATE foo SET is_active = 0 WHERE bar = "whiz"');
+ }
+
+ /**
+ * Example: Run a couple simple queries
+ *
+ * @return TRUE on success
+ * @throws Exception
+ *
+ public function upgrade_4200() {
+ $this->ctx->log->info('Applying update 4200');
+ CRM_Core_DAO::executeQuery('UPDATE foo SET bar = "whiz"');
+ CRM_Core_DAO::executeQuery('DELETE FROM bang WHERE willy = wonka(2)');
+ return TRUE;
+ } // */
+
+
+ /**
+ * Example: Run an external SQL script
+ *
+ * @return TRUE on success
+ * @throws Exception
+ public function upgrade_4201() {
+ $this->ctx->log->info('Applying update 4201');
+ // this path is relative to the extension base dir
+ $this->executeSqlFile('sql/upgrade_4201.sql');
+ return TRUE;
+ } // */
+
+
+ /**
+ * Example: Run a slow upgrade process by breaking it up into smaller chunk
+ *
+ * @return TRUE on success
+ * @throws Exception
+ public function upgrade_4202() {
+ $this->ctx->log->info('Planning update 4202'); // PEAR Log interface
+
+ $this->addTask(ts('Process first step'), 'processPart1', $arg1, $arg2);
+ $this->addTask(ts('Process second step'), 'processPart2', $arg3, $arg4);
+ $this->addTask(ts('Process second step'), 'processPart3', $arg5);
+ return TRUE;
+ }
+ public function processPart1($arg1, $arg2) { sleep(10); return TRUE; }
+ public function processPart2($arg3, $arg4) { sleep(10); return TRUE; }
+ public function processPart3($arg5) { sleep(10); return TRUE; }
+ // */
+
+
+ /**
+ * Example: Run an upgrade with a query that touches many (potentially
+ * millions) of records by breaking it up into smaller chunks.
+ *
+ * @return TRUE on success
+ * @throws Exception
+ public function upgrade_4203() {
+ $this->ctx->log->info('Planning update 4203'); // PEAR Log interface
+
+ $minId = CRM_Core_DAO::singleValueQuery('SELECT coalesce(min(id),0) FROM civicrm_contribution');
+ $maxId = CRM_Core_DAO::singleValueQuery('SELECT coalesce(max(id),0) FROM civicrm_contribution');
+ for ($startId = $minId; $startId <= $maxId; $startId += self::BATCH_SIZE) {
+ $endId = $startId + self::BATCH_SIZE - 1;
+ $title = ts('Upgrade Batch (%1 => %2)', array(
+ 1 => $startId,
+ 2 => $endId,
+ ));
+ $sql = '
+ UPDATE civicrm_contribution SET foobar = whiz(wonky()+wanker)
+ WHERE id BETWEEN %1 and %2
+ ';
+ $params = array(
+ 1 => array($startId, 'Integer'),
+ 2 => array($endId, 'Integer'),
+ );
+ $this->addTask($title, 'executeSql', $sql, $params);
+ }
+ return TRUE;
+ } // */
+
+ public function upgrade_1400() {
+ $this->ctx->log->info('Planning update 1400'); // PEAR Log interface
+ foreach (array('ethnicity_20130725123943', 'religion_20130725124132', 'sexual_orientation_20130725124348', 'marital_status_20130913084916') as $key => $value) {
+ $optParams = array(
+ 'option_group_id' => $value,
+ 'label' => 'Prefer Not to Say',
+ 'value' => 'Prefer Not to Say',
+ 'name' => 'Prefer_Not_to_Say',
+ );
+ civicrm_api3('OptionValue', 'create', $optParams);
+ $optParam = array(
+ 'option_group_id' => $value,
+ 'label' => 'Not Applicable',
+ 'value' => 'Not Applicable',
+ 'name' => 'Not_Applicable',
+ );
+ civicrm_api3('OptionValue', 'create', $optParam);
+ }
+
+ $sql = "UPDATE civicrm_custom_field JOIN civicrm_custom_group ON civicrm_custom_group.id = civicrm_custom_field.custom_group_id SET civicrm_custom_field.default_value = CASE WHEN civicrm_custom_field.name = 'Ethnicity' THEN ' Not Applicable ' ELSE 'Not Applicable' END WHERE civicrm_custom_field.name IN ('Ethnicity','Religion', 'Sexual_Orientation', 'Marital_Status') AND civicrm_custom_group.name = 'Extended_Demographics'";
+ CRM_Core_DAO::executeQuery($sql);
+ CRM_Core_DAO::executeQuery("UPDATE civicrm_custom_group SET is_reserved = 0, collapse_display = 1 where name = 'Extended_Demographics'");
+ return TRUE;
+ }
+
+ /**
+ * Upgrade CustomGroup, setting Extended_Demographics is_reserved value to YES
+ *
+ * @return bool
+ */
+ public function upgrade_1401() {
+ $result = civicrm_api3('CustomGroup', 'get', [
+ 'sequential' => 1,
+ 'return' => ['id'],
+ 'name' => 'Extended_Demographics',
+ ]);
+
+ civicrm_api3('CustomGroup', 'create', [
+ 'id' => $result['id'],
+ 'is_reserved' => 1,
+ ]);
+
+ return TRUE;
+ }
+
+}
diff --git a/hrdemog/CRM/HRDemog/Upgrader/Base.php b/hrdemog/CRM/HRDemog/Upgrader/Base.php
new file mode 100644
index 00000000000..6e2a3fb9267
--- /dev/null
+++ b/hrdemog/CRM/HRDemog/Upgrader/Base.php
@@ -0,0 +1,298 @@
+ctx = array_shift($args);
+ $instance->queue = $instance->ctx->queue;
+ $method = array_shift($args);
+ return call_user_func_array(array($instance, $method), $args);
+ }
+
+ public function __construct($extensionName, $extensionDir) {
+ $this->extensionName = $extensionName;
+ $this->extensionDir = $extensionDir;
+ }
+
+ // ******** Task helpers ********
+
+ /**
+ * Run a CustomData file
+ *
+ * @param string $relativePath the CustomData XML file path (relative to this extension's dir)
+ * @return bool
+ */
+ public function executeCustomDataFile($relativePath) {
+ $xml_file = $this->extensionDir . '/' . $relativePath;
+ return $this->executeCustomDataFileByAbsPath($xml_file);
+ }
+
+ /**
+ * Run a CustomData file
+ *
+ * @param string $xml_file the CustomData XML file path (absolute path)
+ * @return bool
+ */
+ protected static function executeCustomDataFileByAbsPath($xml_file) {
+ require_once 'CRM/Utils/Migrate/Import.php';
+ $import = new CRM_Utils_Migrate_Import();
+ $import->run($xml_file);
+ return TRUE;
+ }
+
+ /**
+ * Run a SQL file
+ *
+ * @param string $relativePath the SQL file path (relative to this extension's dir)
+ * @return bool
+ */
+ public function executeSqlFile($relativePath) {
+ CRM_Utils_File::sourceSQLFile(
+ CIVICRM_DSN,
+ $this->extensionDir . '/' . $relativePath
+ );
+ return TRUE;
+ }
+
+ /**
+ * Run one SQL query
+ *
+ * This is just a wrapper for CRM_Core_DAO::executeSql, but it
+ * provides syntatic sugar for queueing several tasks that
+ * run different queries
+ */
+ public function executeSql($query, $params = array()) {
+ // FIXME verify that we raise an exception on error
+ CRM_Core_DAO::executeSql($query, $params);
+ return TRUE;
+ }
+
+ /**
+ * Syntatic sugar for enqueuing a task which calls a function
+ * in this class. The task is weighted so that it is processed
+ * as part of the currently-pending revision.
+ *
+ * After passing the $funcName, you can also pass parameters that will go to
+ * the function. Note that all params must be serializable.
+ */
+ public function addTask($title) {
+ $args = func_get_args();
+ $title = array_shift($args);
+ $task = new CRM_Queue_Task(
+ array(get_class($this), '_queueAdapter'),
+ $args,
+ $title
+ );
+ return $this->queue->createItem($task, array('weight' => -1));
+ }
+
+ // ******** Revision-tracking helpers ********
+
+ /**
+ * Determine if there are any pending revisions
+ *
+ * @return bool
+ */
+ public function hasPendingRevisions() {
+ $revisions = $this->getRevisions();
+ $currentRevision = $this->getCurrentRevision();
+
+ if (empty($revisions)) {
+ return FALSE;
+ }
+ if (empty($currentRevision)) {
+ return TRUE;
+ }
+
+ return ($currentRevision < max($revisions));
+ }
+
+ /**
+ * Add any pending revisions to the queue
+ */
+ public function enqueuePendingRevisions(CRM_Queue_Queue $queue) {
+ $this->queue = $queue;
+
+ $currentRevision = $this->getCurrentRevision();
+ foreach ($this->getRevisions() as $revision) {
+ if ($revision > $currentRevision) {
+ $title = ts('Upgrade %1 to revision %2', array(
+ 1 => $this->extensionName,
+ 2 => $revision,
+ ));
+
+ // note: don't use addTask() because it sets weight=-1
+
+ $task = new CRM_Queue_Task(
+ array(get_class($this), '_queueAdapter'),
+ array('upgrade_' . $revision),
+ $title
+ );
+ $this->queue->createItem($task);
+
+ $task = new CRM_Queue_Task(
+ array(get_class($this), '_queueAdapter'),
+ array('setCurrentRevision', $revision),
+ $title
+ );
+ $this->queue->createItem($task);
+ }
+ }
+ }
+
+ /**
+ * Get a list of revisions
+ *
+ * @return array(revisionNumbers) sorted numerically
+ */
+ public function getRevisions() {
+ if (! is_array($this->revisions)) {
+ $this->revisions = array();
+
+ $clazz = new ReflectionClass(get_class($this));
+ $methods = $clazz->getMethods();
+ foreach ($methods as $method) {
+ if (preg_match('/^upgrade_(.*)/', $method->name, $matches)) {
+ $this->revisions[] = $matches[1];
+ }
+ }
+ sort($this->revisions, SORT_NUMERIC);
+ }
+
+ return $this->revisions;
+ }
+
+ public function getCurrentRevision() {
+ // return CRM_Core_BAO_Extension::getSchemaVersion($this->extensionName);
+ $key = $this->extensionName . ':version';
+ return CRM_Core_BAO_Setting::getItem('Extension', $key);
+ }
+
+ public function setCurrentRevision($revision) {
+ // We call this during hook_civicrm_install, but the underlying SQL
+ // UPDATE fails because the extension record hasn't been INSERTed yet.
+ // Instead, track revisions in our own namespace.
+ // CRM_Core_BAO_Extension::setSchemaVersion($this->extensionName, $revision);
+
+ $key = $this->extensionName . ':version';
+ CRM_Core_BAO_Setting::setItem($revision, 'Extension', $key);
+ return TRUE;
+ }
+
+ // ******** Hook delegates ********
+
+ public function onInstall() {
+ $files = glob($this->extensionDir . '/sql/*_install.sql');
+ if (is_array($files)) {
+ foreach ($files as $file) {
+ CRM_Utils_File::sourceSQLFile(CIVICRM_DSN, $file);
+ }
+ }
+ $files = glob($this->extensionDir . '/xml/*_install.xml');
+ if (is_array($files)) {
+ foreach ($files as $file) {
+ $this->executeCustomDataFileByAbsPath($file);
+ }
+ }
+ if (is_callable(array($this, 'install'))) {
+ $this->install();
+ }
+ $revisions = $this->getRevisions();
+ if (!empty($revisions)) {
+ $this->setCurrentRevision(max($revisions));
+ }
+ }
+
+ public function onUninstall() {
+ if (is_callable(array($this, 'uninstall'))) {
+ $this->uninstall();
+ }
+ $files = glob($this->extensionDir . '/sql/*_uninstall.sql');
+ if (is_array($files)) {
+ foreach ($files as $file) {
+ CRM_Utils_File::sourceSQLFile(CIVICRM_DSN, $file);
+ }
+ }
+ $this->setCurrentRevision(NULL);
+ }
+
+ public function onEnable() {
+ // stub for possible future use
+ if (is_callable(array($this, 'enable'))) {
+ $this->enable();
+ }
+ }
+
+ public function onDisable() {
+ // stub for possible future use
+ if (is_callable(array($this, 'disable'))) {
+ $this->disable();
+ }
+ }
+
+ public function onUpgrade($op, CRM_Queue_Queue $queue = NULL) {
+ switch($op) {
+ case 'check':
+ return array($this->hasPendingRevisions());
+ case 'enqueue':
+ return $this->enqueuePendingRevisions($queue);
+ default:
+ }
+ }
+}
diff --git a/hrdemog/hrdemog.civix.php b/hrdemog/hrdemog.civix.php
new file mode 100644
index 00000000000..6b646c8cd27
--- /dev/null
+++ b/hrdemog/hrdemog.civix.php
@@ -0,0 +1,215 @@
+template_dir ) ) {
+ array_unshift( $template->template_dir, $extDir );
+ } else {
+ $template->template_dir = array( $extDir, $template->template_dir );
+ }
+
+ $include_path = $extRoot . PATH_SEPARATOR . get_include_path( );
+ set_include_path( $include_path );
+}
+
+/**
+ * (Delegated) Implementation of hook_civicrm_xmlMenu
+ *
+ * @param $files array(string)
+ */
+function _hrdemog_civix_civicrm_xmlMenu(&$files) {
+ foreach (_hrdemog_civix_glob(__DIR__ . '/xml/Menu/*.xml') as $file) {
+ $files[] = $file;
+ }
+}
+
+/**
+ * Implementation of hook_civicrm_install
+ */
+function _hrdemog_civix_civicrm_install() {
+ _hrdemog_civix_civicrm_config();
+ if ($upgrader = _hrdemog_civix_upgrader()) {
+ return $upgrader->onInstall();
+ }
+}
+
+/**
+ * Implementation of hook_civicrm_uninstall
+ */
+function _hrdemog_civix_civicrm_uninstall() {
+ _hrdemog_civix_civicrm_config();
+ if ($upgrader = _hrdemog_civix_upgrader()) {
+ return $upgrader->onUninstall();
+ }
+}
+
+/**
+ * (Delegated) Implementation of hook_civicrm_enable
+ */
+function _hrdemog_civix_civicrm_enable() {
+ _hrdemog_civix_civicrm_config();
+ if ($upgrader = _hrdemog_civix_upgrader()) {
+ if (is_callable(array($upgrader, 'onEnable'))) {
+ return $upgrader->onEnable();
+ }
+ }
+}
+
+/**
+ * (Delegated) Implementation of hook_civicrm_disable
+ */
+function _hrdemog_civix_civicrm_disable() {
+ _hrdemog_civix_civicrm_config();
+ if ($upgrader = _hrdemog_civix_upgrader()) {
+ if (is_callable(array($upgrader, 'onDisable'))) {
+ return $upgrader->onDisable();
+ }
+ }
+}
+
+/**
+ * (Delegated) Implementation of hook_civicrm_upgrade
+ *
+ * @param $op string, the type of operation being performed; 'check' or 'enqueue'
+ * @param $queue CRM_Queue_Queue, (for 'enqueue') the modifiable list of pending up upgrade tasks
+ *
+ * @return mixed based on op. for 'check', returns array(boolean) (TRUE if upgrades are pending)
+ * for 'enqueue', returns void
+ */
+function _hrdemog_civix_civicrm_upgrade($op, CRM_Queue_Queue $queue = NULL) {
+ if ($upgrader = _hrdemog_civix_upgrader()) {
+ return $upgrader->onUpgrade($op, $queue);
+ }
+}
+
+function _hrdemog_civix_upgrader() {
+ if (!file_exists(__DIR__.'/CRM/HRDemog/Upgrader.php')) {
+ return NULL;
+ } else {
+ return CRM_HRDemog_Upgrader_Base::instance();
+ }
+}
+
+/**
+ * Search directory tree for files which match a glob pattern
+ *
+ * Note: Dot-directories (like "..", ".git", or ".svn") will be ignored.
+ * Note: In Civi 4.3+, delegate to CRM_Utils_File::findFiles()
+ *
+ * @param $dir string, base dir
+ * @param $pattern string, glob pattern, eg "*.txt"
+ * @return array(string)
+ */
+function _hrdemog_civix_find_files($dir, $pattern) {
+ if (is_callable(array('CRM_Utils_File', 'findFiles'))) {
+ return CRM_Utils_File::findFiles($dir, $pattern);
+ }
+
+ $todos = array($dir);
+ $result = array();
+ while (!empty($todos)) {
+ $subdir = array_shift($todos);
+ foreach (_hrdemog_civix_glob("$subdir/$pattern") as $match) {
+ if (!is_dir($match)) {
+ $result[] = $match;
+ }
+ }
+ if ($dh = opendir($subdir)) {
+ while (FALSE !== ($entry = readdir($dh))) {
+ $path = $subdir . DIRECTORY_SEPARATOR . $entry;
+ if ($entry{0} == '.') {
+ } elseif (is_dir($path)) {
+ $todos[] = $path;
+ }
+ }
+ closedir($dh);
+ }
+ }
+ return $result;
+}
+/**
+ * (Delegated) Implementation of hook_civicrm_managed
+ *
+ * Find any *.mgd.php files, merge their content, and return.
+ */
+function _hrdemog_civix_civicrm_managed(&$entities) {
+ $mgdFiles = _hrdemog_civix_find_files(__DIR__, '*.mgd.php');
+ foreach ($mgdFiles as $file) {
+ $es = include $file;
+ foreach ($es as $e) {
+ if (empty($e['module'])) {
+ $e['module'] = 'org.civicrm.hrdemog';
+ }
+ $entities[] = $e;
+ }
+ }
+}
+
+/**
+ * Glob wrapper which is guaranteed to return an array.
+ *
+ * The documentation for glob() says, "On some systems it is impossible to
+ * distinguish between empty match and an error." Anecdotally, the return
+ * result for an empty match is sometimes array() and sometimes FALSE.
+ * This wrapper provides consistency.
+ *
+ * @see http://php.net/glob
+ * @param string $pattern
+ * @return array, possibly empty
+ */
+function _hrdemog_civix_glob($pattern) {
+ $result = glob($pattern);
+ return is_array($result) ? $result : array();
+}
+
+/**
+ * Inserts a navigation menu item at a given place in the hierarchy
+ *
+ * $menu - menu hierarchy
+ * $path - path where insertion should happen (ie. Administer/System Settings)
+ * $item - menu you need to insert (parent/child attributes will be filled for you)
+ * $parentId - used internally to recurse in the menu structure
+ */
+function _hrdemog_civix_insert_navigation_menu(&$menu, $path, $item, $parentId = NULL) {
+ static $navId;
+
+ // If we are done going down the path, insert menu
+ if (empty($path)) {
+ if (!$navId) $navId = CRM_Core_DAO::singleValueQuery("SELECT max(id) FROM civicrm_navigation");
+ $navId ++;
+ $menu[$navId] = array (
+ 'attributes' => array_merge($item, array(
+ 'label' => CRM_Utils_Array::value('name', $item),
+ 'active' => 1,
+ 'parentID' => $parentId,
+ 'navID' => $navId,
+ ))
+ );
+ return true;
+ } else {
+ // Find an recurse into the next level down
+ $found = false;
+ $path = explode('/', $path);
+ $first = array_shift($path);
+ foreach ($menu as $key => &$entry) {
+ if ($entry['attributes']['name'] == $first) {
+ if (!$entry['child']) $entry['child'] = array();
+ $found = _hrdemog_civix_insert_navigation_menu($entry['child'], implode('/', $path), $item, $key);
+ }
+ }
+ return $found;
+ }
+}
diff --git a/hrdemog/hrdemog.php b/hrdemog/hrdemog.php
new file mode 100644
index 00000000000..fdff6261f3f
--- /dev/null
+++ b/hrdemog/hrdemog.php
@@ -0,0 +1,116 @@
+ "id",'name' => "Extended_Demographics",));
+ civicrm_api3('CustomGroup', 'delete', array('id' => $customGroup['id']));
+ //delete optionGroup
+ if ($visaGroupID = CRM_Core_DAO::getFieldValue('CRM_Core_DAO_OptionGroup', 'is_visa_required_20130702051150', 'id', 'name')) {
+ CRM_Core_BAO_OptionGroup::del($visaGroupID);
+ }
+ return _hrdemog_civix_civicrm_uninstall();
+}
+
+/**
+ * Implementation of hook_civicrm_enable
+ */
+function hrdemog_civicrm_enable() {
+ _hrdemog_setActiveFields(1);
+ return _hrdemog_civix_civicrm_enable();
+}
+
+/**
+ * Implementation of hook_civicrm_disable
+ */
+function hrdemog_civicrm_disable() {
+ _hrdemog_setActiveFields(0);
+ return _hrdemog_civix_civicrm_disable();
+}
+
+function _hrdemog_setActiveFields($setActive) {
+ //disable/enable customgroup and customvalue
+ $sql = "UPDATE civicrm_custom_field JOIN civicrm_custom_group on civicrm_custom_group.id = civicrm_custom_field.custom_group_id SET civicrm_custom_field.is_active = {$setActive} WHERE civicrm_custom_group.name = 'Extended_Demographics'";
+ CRM_Core_DAO::executeQuery($sql);
+ CRM_Core_DAO::executeQuery("UPDATE civicrm_custom_group SET is_active = {$setActive} WHERE name = 'Extended_Demographics'");
+
+ //disable/enable optionGroup and optionValue
+ $query = "UPDATE civicrm_option_value JOIN civicrm_option_group on civicrm_option_group.id = civicrm_option_value.option_group_id SET civicrm_option_value.is_active = {$setActive} WHERE civicrm_option_group.name IN ('ethnicity_20130725123943','religion_20130725124132','sexual_orientation_20130725124348','marital_status_20130913084916','is_visa_required_20130702051150')";
+ CRM_Core_DAO::executeQuery($query);
+ CRM_Core_DAO::executeQuery("UPDATE civicrm_option_group SET is_active = {$setActive} WHERE name IN ('ethnicity_20130725123943','religion_20130725124132','sexual_orientation_20130725124348','marital_status_20130913084916','is_visa_required_20130702051150')");
+}
+
+
+/**
+ * Implementation of hook_civicrm_upgrade
+ *
+ * @param $op string, the type of operation being performed; 'check' or 'enqueue'
+ * @param $queue CRM_Queue_Queue, (for 'enqueue') the modifiable list of pending up upgrade tasks
+ *
+ * @return mixed based on op. for 'check', returns array(boolean) (TRUE if upgrades are pending)
+ * for 'enqueue', returns void
+ */
+function hrdemog_civicrm_upgrade($op, CRM_Queue_Queue $queue = NULL) {
+ return _hrdemog_civix_civicrm_upgrade($op, $queue);
+}
+
+/**
+ * Implementation of hook_civicrm_managed
+ *
+ * Generate a list of entities to create/deactivate/delete when this module
+ * is installed, disabled, uninstalled.
+ */
+function hrdemog_civicrm_managed(&$entities) {
+ return _hrdemog_civix_civicrm_managed($entities);
+}
diff --git a/hrdemog/info.xml b/hrdemog/info.xml
new file mode 100644
index 00000000000..5df28806d93
--- /dev/null
+++ b/hrdemog/info.xml
@@ -0,0 +1,20 @@
+
+
+ hrdemog
+ Extended Demographics
+ Add extended demographic fields for the contact
+ AGPL-3.0
+
+ CiviCRM LLC
+ info@civicrm.org
+
+ 2018-06-05
+ 1.7.7
+ stable
+
+ 4.7
+
+
+ CRM/HRDemog
+
+
diff --git a/uk.co.compucorp.civicrm.hrcore/xml/CustomGroups/extended_demographics_install.xml b/hrdemog/xml/auto_install.xml
similarity index 100%
rename from uk.co.compucorp.civicrm.hrcore/xml/CustomGroups/extended_demographics_install.xml
rename to hrdemog/xml/auto_install.xml
diff --git a/hrui/js/dist/hrui.min.js b/hrui/js/dist/hrui.min.js
index 78788ce52f9..609b9715b63 100644
--- a/hrui/js/dist/hrui.min.js
+++ b/hrui/js/dist/hrui.min.js
@@ -1,5 +1,5 @@
!function(t,o){function n(o){o&&o.departments&&o.departments.length>0?t(".crm-contact-detail-departments").html("Department: "+o.departments.join(", ")):t(".crm-contact-detail-departments").html("")}t(document).on("crmLoad",function(o){t(".crm-inline-edit").one("DOMSubtreeModified",function(){var o=t(this).find("form");1===o.length&&o.find("label").each(function(){var o=t(this),n=o.attr("for");t("#"+n).attr("placeholder",o.text())})})}).on("updateContactHeader",function(o,c){var a;void 0!==c.contract&&((a=c.contract)?(t(".crm-summary-contactname-block").removeClass("crm-summary-contactname-block-without-contract"),a.position&&t(".crm-contact-detail-position").html("Position: "+a.position),a.location&&t(".crm-contact-detail-location").html("Normal place of work: "+a.location)):(t(".crm-summary-contactname-block").addClass("crm-summary-contactname-block-without-contract"),t(".crm-contact-detail-position").html(""),t(".crm-contact-detail-location").html(""),n(null))),void 0!==c.roles&&n(c.roles)})}(CRM.$,CRM._);
-!function(e,t){function a(){var t=e(this)[0].files[0];e("#js-uploaded-file").remove(),void 0!==t&&(e(this).after(''+t.name+' Remove'),e(".uploaded-file-icon-trash").on("click",r))}function o(e){e.find("label").attr("for",e.find(".crm-form-date").attr("id"))}function r(){var t=e("#js-uploaded-file").parent().find('input[type="file"]');e("#js-uploaded-file").remove(),t.val("")}function n(t){e(t).removeClass("crm-collapsible collapsed").addClass("crm-summary-block").appendTo(".contactTopBar .contactCardLeft").find(".collapsible-title").hide().end().find(".crm-summary-block").show()}e(document).on("crmLoad",function(t){var r,n,i;1===e(r="input[type='file']").length&&e(r).on("change",a),1===e(".CRM_HRRecruitment_Form_HRVacancy").length&&(o(e('label[for="start_date"]').parents("tr")),o(e('label[for="end_date"]').parents("tr")),e('[name="entryURL"]').val().indexOf(";template=1")>-1&&e(e(".CRM_HRRecruitment_Form_HRVacancy tbody").get(0)).addClass("CRM_HRRecruitment_Form_HRVacancy_Template")),n=t,"contactForm"!==CRM.formName&&"viewSummary"!==CRM.pageName||(e(".crm-contact-job_title",".crm-summary-contactinfo-block").length&&e(".crm-contact-tabs-list #tab_summary a",n.target).text("Personal Details"),n.target,e("#customFields").length<1&&e("#Inline_Custom_Data").detach().find(".label").each(function(){var t=e(this),a=t.next(),o=e("
| ");o.append(t.find("label")),o.append(e("
")),o.append(a.html()),e("#nick_name").parent().after(o)}),i=n.target,e("div.crm-contact-current_employer, div.crm-contact-job_title",".crm-summary-contactinfo-block").parent("div.crm-summary-row").hide(),e("form#ContactInfo input#employer_id, form#ContactInfo input#job_title",i).closest("div.crm-summary-row").hide(),e("input#employer_id, input#job_title","form#Contact").parent("td").hide(),e(".HRJobContract_Summary",i).insertBefore(e(".crm-summary-contactinfo-block")),e("div.email-signature, td#Email-Bulkmail-html","form#Contact").hide(),e("#Email-Primary","form#Contact").prev("td").prev("td").hide(),e("td#Email-Bulkmail-html, #Email-Primary","form#Contact").prev("td").hide(),e(".crm-demographics-accordion","form#Contact").insertAfter(e(".crm-contactDetails-accordion")),e("tr#Phone_Block_2","form#Contact").length<1&&e("#addPhone").click()),e("#activityCustomData").attr("colspan",3),e("#crm-activity-view-table .crm-case-activity-view-Client .label").html("Contact"),e("span.crm-frozen-field",".crm-profile-name-hrident_tab").closest("div").parent("div").hide(),e(".crm-accordion-header.crm-master-accordion-header").on("click",function(){window.setTimeout(function(){Array.prototype.forEach.call(document.querySelectorAll(".listing-box"),function(e){Ps.initialize(e)})},0)}),e("INPUT#contact_source").parent("td").children("a").click(function(){e("#crm-notification-container .crm-help .notify-content").remove(),e("#crm-notification-container .crm-help p").length&&e("#crm-notification-container .crm-help p").remove(),e("#crm-notification-container .crm-help").append("Source is a useful field where data has been migrated to CiviHR from one or a number of other legacy systems. The Source field will indicate which legacy system the contact has come from.
")})}).ready(function(){var t,a,o,r,n,i;e.ajax("/civicrm/hrcore/usermenu?snippet=4",{dataType:"html",success:function(t){var a,o,r;a="civihr-menu",o=e(t),(r=e("")).attr("id",a),r.append(e("#civicrm-menu")),r.append(o),r.insertAfter("#page")}}),t=e(".crm-link-home"),a=e(".crm-logo-sm",t).addClass("chr_logo chr_logo--default-color").removeClass("crm-logo-sm").wrap('
').parent(),r=a,o=e("li > a",t).first().text("Home").wrapInner('').prepend(r),n=o,t.off().find("#civicrm-home").before(n).remove(),e(".CRM_HRRecruitment_Form_Application").addClass("crm-form-block"),e(".CRM_HRRecruitment_Form_Application .crm-profile-name-application_profile").addClass("form-layout-compressed"),e("#root-menu-div .menu-item-arrow").each(function(t){var a=e(this);a.before(''),a.remove()}),i="activetarget",e(".menumain").not(".crm-Self_Service_Portal").hover(function(){e(this).addClass(i)},function(){e(this).removeClass(i)})}),e(".Inline_Custom_Data").length&&n(".Inline_Custom_Data"),n(".Extended_Demographics")}(CRM.$,CRM._);
+!function(e,t){function a(){var t=e(this)[0].files[0];e("#js-uploaded-file").remove(),void 0!==t&&(e(this).after(''+t.name+' Remove'),e(".uploaded-file-icon-trash").on("click",o))}function r(e){e.find("label").attr("for",e.find(".crm-form-date").attr("id"))}function o(){var t=e("#js-uploaded-file").parent().find('input[type="file"]');e("#js-uploaded-file").remove(),t.val("")}e(document).on("crmLoad",function(t){var o,i,n;1===e(o="input[type='file']").length&&e(o).on("change",a),1===e(".CRM_HRRecruitment_Form_HRVacancy").length&&(r(e('label[for="start_date"]').parents("tr")),r(e('label[for="end_date"]').parents("tr")),e('[name="entryURL"]').val().indexOf(";template=1")>-1&&e(e(".CRM_HRRecruitment_Form_HRVacancy tbody").get(0)).addClass("CRM_HRRecruitment_Form_HRVacancy_Template")),i=t,"contactForm"!==CRM.formName&&"viewSummary"!==CRM.pageName||(e(".crm-contact-job_title",".crm-summary-contactinfo-block").length&&e(".crm-contact-tabs-list #tab_summary a",i.target).text("Personal Details"),i.target,e(".Inline_Custom_Data").length&&e(".Inline_Custom_Data").removeClass("crm-collapsible collapsed").addClass("crm-summary-block").insertAfter(".crm-summary-contactinfo-block").find(".collapsible-title").hide().end().find(".crm-summary-block").show(),e("#customFields").length<1&&e("#Inline_Custom_Data").detach().find(".label").each(function(){var t=e(this),a=t.next(),r=e(" | ");r.append(t.find("label")),r.append(e("
")),r.append(a.html()),e("#nick_name").parent().after(r)}),n=i.target,e("div.crm-contact-current_employer, div.crm-contact-job_title",".crm-summary-contactinfo-block").parent("div.crm-summary-row").hide(),e("form#ContactInfo input#employer_id, form#ContactInfo input#job_title",n).closest("div.crm-summary-row").hide(),e("input#employer_id, input#job_title","form#Contact").parent("td").hide(),e(".HRJobContract_Summary",n).insertBefore(e(".crm-summary-contactinfo-block")),e("div.email-signature, td#Email-Bulkmail-html","form#Contact").hide(),e("#Email-Primary","form#Contact").prev("td").prev("td").hide(),e("td#Email-Bulkmail-html, #Email-Primary","form#Contact").prev("td").hide(),e(".crm-demographics-accordion","form#Contact").insertAfter(e(".crm-contactDetails-accordion")),e("tr#Phone_Block_2","form#Contact").length<1&&e("#addPhone").click()),e("#activityCustomData").attr("colspan",3),e("#crm-activity-view-table .crm-case-activity-view-Client .label").html("Contact"),e("span.crm-frozen-field",".crm-profile-name-hrident_tab").closest("div").parent("div").hide(),e(".crm-accordion-header.crm-master-accordion-header").on("click",function(){window.setTimeout(function(){Array.prototype.forEach.call(document.querySelectorAll(".listing-box"),function(e){Ps.initialize(e)})},0)}),e("INPUT#contact_source").parent("td").children("a").click(function(){e("#crm-notification-container .crm-help .notify-content").remove(),e("#crm-notification-container .crm-help p").length&&e("#crm-notification-container .crm-help p").remove(),e("#crm-notification-container .crm-help").append("
Source is a useful field where data has been migrated to CiviHR from one or a number of other legacy systems. The Source field will indicate which legacy system the contact has come from.
")})}).ready(function(){var t,a,r,o,i,n;e.ajax("/civicrm/hrcore/usermenu?snippet=4",{dataType:"html",success:function(t){var a,r,o;a="civihr-menu",r=e(t),(o=e("
")).attr("id",a),o.append(e("#civicrm-menu")),o.append(r),o.insertAfter("#page")}}),t=e(".crm-link-home"),a=e(".crm-logo-sm",t).addClass("chr_logo chr_logo--default-color").removeClass("crm-logo-sm").wrap('
').parent(),o=a,r=e("li > a",t).first().text("Home").wrapInner('').prepend(o),i=r,t.off().find("#civicrm-home").before(i).remove(),e(".CRM_HRRecruitment_Form_Application").addClass("crm-form-block"),e(".CRM_HRRecruitment_Form_Application .crm-profile-name-application_profile").addClass("form-layout-compressed"),e("#root-menu-div .menu-item-arrow").each(function(t){var a=e(this);a.before(''),a.remove()}),n="activetarget",e(".menumain").not(".crm-Self_Service_Portal").hover(function(){e(this).addClass(n)},function(){e(this).removeClass(n)})})}(CRM.$,CRM._);
!function(t){"use strict";var n=window.MutationObserver||window.WebKitMutationObserver||window.MozMutationObserver;t.fn.attrchange=function(t){if(n){var e={subtree:!1,attributes:!0},i=new n(function(n){n.forEach(function(n){t.call(n.target,n)})});return this.each(function(){i.observe(this,e)})}}}(CRM.$);
!function(t){"use strict";var i=t("body");i.on("click","#bootstrap-theme .btn-slide",function(){var n,a,e=t(this),o=e.children("ul.panel");function c(){t(".civihr-popup").remove(),i.removeClass("civihr-popup-open")}c(),n=o.clone(!0).appendTo(i).addClass("civihr-popup"),a=e.offset(),n.css({left:+a.left-(n.width()-e.outerWidth()),top:+a.top+e.outerHeight()}),n.show(),i.addClass("civihr-popup-open"),n.find("a").click(function(){var i=t(this).parent().index();o.find("li:nth("+i+") a").click()}),e.data("attrchange-is-on")||(e.data("attrchange-is-on",!0),e.attrchange(function(){e.hasClass("btn-slide-active")||c()}))})}(CRM.$);
-//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["contact.js","hrui.js","civihr-popup/attrchange.js","civihr-popup/civihr-popup.js"],"names":["$","_","updateContactHeaderRolesDetails","roles","departments","length","html","join","document","on","e","one","$form","this","find","each","$label","id","attr","text","data","contract","removeClass","position","location","addClass","CRM","insertFile","fileName","files","remove","undefined","after","name","removeFile","linkLabelToDatepickerInput","$line","$input","parent","val","repositionPersonalDetailsBlock","selector","appendTo","hide","end","show","target","parents","indexOf","get","formName","pageName","detach","$labelCell","$fieldCell","next","$newTd","append","closest","insertBefore","prev","insertAfter","click","window","setTimeout","Array","prototype","forEach","call","querySelectorAll","element","Ps","initialize","children","ready","$menuItem","$wrappedLogo","$customHomeLink","$appLogo","$homeLink","className","ajax","dataType","success","menuMarkup","wrapperId","$menuMarkup","$menuWrapper","wrap","first","wrapInner","prepend","off","before","$element","$arrow","not","hover","MutationObserver","WebKitMutationObserver","MozMutationObserver","fn","attrchange","callback","options","subtree","attributes","observer","mutations","observe","$body","$popupClone","buttonOffset","$button","$popup","closePopupPanels","clone","offset","css","left","width","outerWidth","top","outerHeight","actionIndex","index","hasClass"],"mappings":"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","file":"hrui.min.js","sourcesContent":["(function ($, _) {\n  $(document)\n    .on('crmLoad', function (e) {\n      $('.crm-inline-edit').one('DOMSubtreeModified', function () {\n        var $form = $(this).find('form');\n\n        if ($form.length === 1) {\n          $form.find('label').each(function () {\n            var $label = $(this);\n            var id = $label.attr('for');\n            $('#' + id).attr('placeholder', $label.text());\n          });\n        }\n      });\n    })\n    .on('updateContactHeader', function (e, data) {\n      if (typeof data.contract !== 'undefined') {\n        updateContactHeaderContractDetails(data.contract);\n      }\n\n      if (typeof data.roles !== 'undefined') {\n        updateContactHeaderRolesDetails(data.roles);\n      }\n    });\n\n  /**\n   * Updates the contact header with the given contract details\n   *\n   * @param  {object} contract\n   */\n  function updateContactHeaderContractDetails (contract) {\n    if (contract) {\n      $('.crm-summary-contactname-block').removeClass('crm-summary-contactname-block-without-contract');\n\n      if (contract.position) {\n        $('.crm-contact-detail-position').html('<strong>Position:</strong> ' + contract.position);\n      }\n\n      if (contract.location) {\n        $('.crm-contact-detail-location').html('<strong>Normal place of work:</strong> ' + contract.location);\n      }\n    } else {\n      $('.crm-summary-contactname-block').addClass('crm-summary-contactname-block-without-contract');\n      $('.crm-contact-detail-position').html('');\n      $('.crm-contact-detail-location').html('');\n\n      updateContactHeaderRolesDetails(null);\n    }\n  }\n\n  /**\n   * Updates the contact header with the given roles details\n   *\n   * @param  {object} contract\n   */\n  function updateContactHeaderRolesDetails (roles) {\n    if (roles && roles.departments && roles.departments.length > 0) {\n      $('.crm-contact-detail-departments').html('<strong>Department:</strong> ' + roles.departments.join(', '));\n    } else {\n      $('.crm-contact-detail-departments').html('');\n    }\n  }\n}(CRM.$, CRM._));\n","/* global Ps */\n\n// Copyright CiviCRM LLC 2013. See http://civicrm.org/licensing\n(function ($, _) {\n  $(document)\n    .on('crmLoad', function (e) {\n      addUploadFileListener(\"input[type='file']\");\n      amendVacancyForm();\n      amendContactPageAndForm(e);\n      applyMiscChanges();\n      changeContactSourceFieldHelpText();\n    })\n    .ready(function () {\n      addUserMenuToMainMenu();\n      amendAppLogoMenuItem();\n      amendApplicationForm();\n      useFontAwesomeArrowsInSubMenuItems();\n      toggleActiveClassOnHoverOnAnyMainMenuItem();\n    });\n\n  /**\n   * Customizes the app logo menu item, switching from the CiviCRM logo\n   * to the CiviHR logo, and making the item a direct link instead of a\n   * toggle for a sub menu dropdown\n   */\n  function amendAppLogoMenuItem () {\n    var $menuItem = $('.crm-link-home');\n    var $wrappedLogo = swapAndWrapAppLogo($menuItem);\n    var $customHomeLink = customizeHomeLinkInLogoMenuItem($menuItem, $wrappedLogo);\n\n    removeLogoSubMenuAndKeepOnlyHomeLink($menuItem, $customHomeLink);\n  }\n\n  /**\n   * Adds the user menu by fetching it from the hrcore extension\n   */\n  function addUserMenuToMainMenu () {\n    $.ajax('/civicrm/hrcore/usermenu?snippet=4', {\n      dataType: 'html',\n      success: function (menuMarkup) {\n        injectUserMenuInAMainMenuWrapper(menuMarkup, 'civihr-menu');\n      }\n    });\n  }\n\n  /**\n   * Add an event listener on input[type=\"file\"]\n   * @param {jQuery Object} selector [selector from input file]\n   */\n  function addUploadFileListener (selector) {\n    if ($(selector).length === 1) {\n      $(selector).on('change', insertFile);\n    }\n  }\n\n  /**\n   * Amends the application form\n   */\n  function amendApplicationForm () {\n    $('.CRM_HRRecruitment_Form_Application').addClass('crm-form-block');\n    $('.CRM_HRRecruitment_Form_Application .crm-profile-name-application_profile').addClass('form-layout-compressed');\n  }\n\n  /**\n   * Amends the contact page and the contact form\n   */\n  function amendContactPageAndForm (e) {\n    if (CRM.formName === 'contactForm' || CRM.pageName === 'viewSummary') {\n      // Rename \"Summary\" tab to \"Personal Details\"\n      // Hack to check contact type - This field only appears for individuals\n      if ($('.crm-contact-job_title', '.crm-summary-contactinfo-block').length) {\n        $('.crm-contact-tabs-list #tab_summary a', e.target).text('Personal Details');\n      }\n\n      manipulateDOMOfInlineCustomData(e.target);\n      miscContactPageChanges(e.target);\n    }\n  }\n\n  /**\n   * Amends the vacancy form\n   */\n  function amendVacancyForm () {\n    if ($('.CRM_HRRecruitment_Form_HRVacancy').length === 1) {\n      linkLabelToDatepickerInput($('label[for=\"start_date\"]').parents('tr'));\n      linkLabelToDatepickerInput($('label[for=\"end_date\"]').parents('tr'));\n\n      // Add a class to identify the form 'New Vacancy Template'\n      if ($('[name=\"entryURL\"]').val().indexOf(';template=1') > -1) {\n        $($('.CRM_HRRecruitment_Form_HRVacancy tbody').get(0)).addClass('CRM_HRRecruitment_Form_HRVacancy_Template');\n      }\n    }\n  }\n\n  /**\n   * Applies miscellaneous UI changes\n   */\n  function applyMiscChanges () {\n    $('#activityCustomData').attr('colspan', 3);\n    $('#crm-activity-view-table .crm-case-activity-view-Client .label').html('Contact');\n    $('span.crm-frozen-field', '.crm-profile-name-hrident_tab').closest('div').parent('div').hide();\n\n    $('.crm-accordion-header.crm-master-accordion-header').on('click', function () {\n      window.setTimeout(function () {\n        Array.prototype.forEach.call(document.querySelectorAll('.listing-box'), function (element) {\n          Ps.initialize(element);\n        });\n      }, 0);\n    });\n  }\n\n  /**\n   * Changes of sorce help text\n   */\n  function changeContactSourceFieldHelpText () {\n    $('INPUT#contact_source').parent('td').children('a').click(function () {\n      $('#crm-notification-container .crm-help .notify-content').remove();\n\n      if ($('#crm-notification-container .crm-help p').length) {\n        $('#crm-notification-container .crm-help p').remove();\n      }\n\n      $('#crm-notification-container .crm-help').append('<p>Source is a useful field where data has been migrated to CiviHR from one or a number of other legacy systems. The Source field will indicate which legacy system the contact has come from.</p>');\n    });\n  }\n\n  /**\n   * Finds the original link to the homepage, changes the text, wraps it in a\n   * `menumain-label` element and prepends internally the given app logo\n   *\n   * @param {object} $menuItem The context where to find the link\n   * @param {object} $appLogo\n   * @return the customized home link\n   */\n  function customizeHomeLinkInLogoMenuItem ($menuItem, $appLogo) {\n    var $homeLink = $('li > a', $menuItem).first();\n\n    return $homeLink\n      .text('Home')\n      .wrapInner('<span class=\"menumain-label\">')\n      .prepend($appLogo);\n  }\n\n  /**\n   * Injects the given markup in a menu wrapper with the given id\n   * created to contain both the original menu and the user one\n   *\n   * @param {string} menuMarkup\n   * @param {string} wrapperId\n   */\n  function injectUserMenuInAMainMenuWrapper (menuMarkup, wrapperId) {\n    var $menuMarkup = $(menuMarkup);\n    var $menuWrapper = $('<div>');\n\n    $menuWrapper.attr('id', wrapperId);\n    $menuWrapper.append($('#civicrm-menu'));\n    $menuWrapper.append($menuMarkup);\n    $menuWrapper.insertAfter('#page');\n  }\n\n  /**\n   * Insert a DOM node after input[type=\"file\"]\n   * with the filename\n   */\n  function insertFile () {\n    var fileName = $(this)[0].files[0];\n\n    $('#js-uploaded-file').remove();\n    if (fileName !== undefined) {\n      $(this).after('<span id=\"js-uploaded-file\" class=\"uploaded-file\">' + fileName.name + ' <span class=\"uploaded-file-icon-trash\"><i class=\"fa fa-trash-o\"></i> Remove</span>');\n\n      $('.uploaded-file-icon-trash').on('click', removeFile);\n    }\n  }\n\n  /**\n   * Update label 'for' attr to works with the datepicker\n   *\n   * @param  {jQuery object} $line [datepicker's line parent]\n   */\n  function linkLabelToDatepickerInput ($line) {\n    $line.find('label').attr('for', $line.find('.crm-form-date').attr('id'));\n  }\n\n  /**\n   * Manipulates, at the DOM level, the blocks/fields belonging to the\n   * Inline Custom Data custom fields set\n   */\n  if ($('.Inline_Custom_Data').length) {\n    repositionPersonalDetailsBlock('.Inline_Custom_Data');\n  }\n\n  repositionPersonalDetailsBlock('.Extended_Demographics');\n\n  function manipulateDOMOfInlineCustomData () {\n    if ($('#customFields').length < 1) {\n      repositionInlineCustomDataFieldsInEditContactForm();\n    }\n  }\n\n  /**\n   * Misc changes to the page (hiding elements, inserting new ones, etc)\n   */\n  function miscContactPageChanges (target) {\n    // Hide current employer and job title\n    // Contact summary screen:\n    $('div.crm-contact-current_employer, div.crm-contact-job_title', '.crm-summary-contactinfo-block').parent('div.crm-summary-row').hide();\n    // Inline edit form\n    $('form#ContactInfo input#employer_id, form#ContactInfo input#job_title', target).closest('div.crm-summary-row').hide();\n    // Contact edit screen\n    $('input#employer_id, input#job_title', 'form#Contact').parent('td').hide();\n\n    /* Changes on Add Individual pages and Personal details tab for HR-358 */\n    // Move Job summary to top\n    $('.HRJobContract_Summary', target).insertBefore($('.crm-summary-contactinfo-block'));\n    // changes of email block, remove bulkmail and onhold\n    $('div.email-signature, td#Email-Bulkmail-html', 'form#Contact').hide();\n    $('#Email-Primary', 'form#Contact').prev('td').prev('td').hide();\n    $('td#Email-Bulkmail-html, #Email-Primary', 'form#Contact').prev('td').hide();\n\n    // shift demographic above extended demographic\n    $('.crm-demographics-accordion', 'form#Contact').insertAfter($('.crm-contactDetails-accordion'));\n\n    if ($('tr#Phone_Block_2', 'form#Contact').length < 1) {\n      $('#addPhone').click();\n    }\n  }\n\n  /**\n   * Remove the #js-uploaded-file DIV and\n   * clean input[type=\"file\"] value\n   */\n  function removeFile () {\n    var $input = $('#js-uploaded-file').parent().find('input[type=\"file\"]');\n\n    $('#js-uploaded-file').remove();\n    $input.val('');\n  }\n\n  /**\n   * Moves the given home link right under the menu item and gets rid\n   * of the original sub menu\n   *\n   * @param {object} $menuItem The context where to find the link\n   * @param {object} $homeLink\n   */\n  function removeLogoSubMenuAndKeepOnlyHomeLink ($menuItem, $homeLink) {\n    $menuItem\n      .off() // removes any handler that the original item had\n      .find('#civicrm-home')\n      .before($homeLink)\n      .remove();\n  }\n\n  /**\n   * Moves the \"Inline Custom Data\" fields towards the top of the\n   * edit contact form\n   */\n  function repositionInlineCustomDataFieldsInEditContactForm () {\n    var $fields = $('#Inline_Custom_Data').detach();\n\n    $fields.find('.label').each(function () {\n      var $labelCell = $(this);\n      var $fieldCell = $labelCell.next();\n      var $newTd = $('<td/>');\n\n      $newTd.append($labelCell.find('label'));\n      $newTd.append($('<br/>'));\n      $newTd.append($fieldCell.html());\n\n      $('#nick_name').parent().after($newTd);\n    });\n  }\n\n  /**\n   * Moves the specified selector element to the personal details tab\n   *\n   * @param selector\n   */\n  function repositionPersonalDetailsBlock (selector) {\n    $(selector)\n      .removeClass('crm-collapsible collapsed')\n      .addClass('crm-summary-block')\n      .appendTo('.contactTopBar .contactCardLeft')\n      .find('.collapsible-title').hide().end()\n      .find('.crm-summary-block').show();\n  }\n\n  /**\n   * Swaps the CiviCRM logo with the CiviHR logo\n   * and wraps it in a `menumain-icon` element\n   *\n   * @param {object} $menuItem The context where to find the logo\n   * @return the wrapper of the logo\n   */\n  function swapAndWrapAppLogo ($menuItem) {\n    var $appLogo = $('.crm-logo-sm', $menuItem);\n\n    return $appLogo\n      .addClass('chr_logo chr_logo--default-color')\n      .removeClass('crm-logo-sm')\n      .wrap('<span class=\"menumain-icon\">')\n      .parent();\n  }\n\n  /**\n   * CiviCRM by default applies on hover the .activetarget class\n   * only to main menu items with a submenu\n   *\n   * This functions makes sure that any item gets the class applied,\n   * even those with just a direct link\n   */\n  function toggleActiveClassOnHoverOnAnyMainMenuItem () {\n    var className = 'activetarget';\n\n    $('.menumain').not('.crm-Self_Service_Portal').hover(function () {\n      $(this).addClass(className);\n    }, function () {\n      $(this).removeClass(className);\n    });\n  }\n\n  /**\n   * Remove the arrow for menu items with sub-items, and replaces it\n   * with a font awesome caret\n   */\n  function useFontAwesomeArrowsInSubMenuItems () {\n    $('#root-menu-div .menu-item-arrow').each(function ($element) {\n      var $arrow = $(this);\n\n      $arrow.before('<i class=\"fa fa-caret-right menu-item-arrow\"></i>');\n      $arrow.remove();\n    });\n  }\n}(CRM.$, CRM._));\n","(function ($) {\n  'use strict';\n  var MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver;\n\n  /*\n  This plugin creates MutationObserver which listens to all changes to DOM Node.\n  If change has been done to it's attributes callback will be called.\n\n  @param {function} callback - to be called when attributes change\n  @returns {Array} - list of all observed elements\n  */\n  $.fn.attrchange = function (callback) {\n    if (MutationObserver) {\n      var options = {\n        subtree: false,\n        attributes: true\n      };\n\n      var observer = new MutationObserver(function (mutations) {\n        mutations.forEach(function (e) {\n          callback.call(e.target, e);\n        });\n      });\n\n      return this.each(function () {\n        observer.observe(this, options);\n      });\n    }\n  }\n})(CRM.$);\n","/* globals CRM */\n\n/**\n * Helps to display popup menus inside #bootstrap-theme elements, since .panel's\n * and other Bootstrap elements clash with CiviCRM styles producing undesired\n * results.\n *\n * It also helps with event bindings, in case the original popup button was\n * removed and recreated, by forwarding the events to the original source.\n */\n(function ($) {\n  'use strict';\n  var $body = $('body');\n\n  $body.on('click', '#bootstrap-theme .btn-slide', function () {\n    var $popupClone;\n    var $button = $(this);\n    var $popup = $button.children('ul.panel');\n\n    (function init () {\n      closePopupPanels();\n      createPopupClone();\n      openPopupClone();\n      mapCloneClickEventsToOrigin();\n      listenToMouseOutEvent();\n    })();\n\n    /**\n     * Closes the any popup panel that is currently open and removes the\n     * `.civihr-popup-open` class from the body.\n     */\n    function closePopupPanels () {\n      $('.civihr-popup').remove();\n      $body.removeClass('civihr-popup-open');\n    }\n\n    /**\n     * Creates a clone of the popup element and appends it to the document body.\n     * This prevents the popup to be hidden by any `overflow: hidden;` rule.\n     */\n    function createPopupClone () {\n      $popupClone = $popup.clone(true)\n        .appendTo($body)\n        .addClass('civihr-popup');\n    }\n\n    /**\n     * Listens to mouse events happening outside the of the panel in order to\n     * close it. The way to detected mouse out is by listening to the\n     * .btn-slide-active class change, which is added and removed by CiviCRM.\n     * the data *attrchange-is-on* is set to true in order to avoid adding\n     * multiple listeners to the same element.\n     */\n    function listenToMouseOutEvent () {\n      // If AttrChange event is already listened, then skip:\n      if ($button.data('attrchange-is-on')) {\n        return;\n      }\n\n      $button.data('attrchange-is-on', true);\n      $button.attrchange(function () {\n        // If button is already open, then skip:\n        if ($button.hasClass('btn-slide-active')) {\n          return;\n        }\n\n        closePopupPanels();\n      });\n    }\n\n    /**\n     * Maps click events on the popup options back to their original source.\n     * This is done because popup actions are executed as delegated events and\n     * the listener is not the *body* element.\n     */\n    function mapCloneClickEventsToOrigin () {\n      $popupClone.find('a').click(function () {\n        var actionIndex = $(this).parent().index();\n\n        $popup.find('li:nth(' + actionIndex + ') a').click();\n      });\n    }\n\n    /**\n     * Opens the popup panel clone\n     * and adds the .civihr-popup-open class to the body\n     */\n    function openPopupClone () {\n      var buttonOffset = $button.offset();\n\n      $popupClone.css({\n        left: +buttonOffset.left - ($popupClone.width() - $button.outerWidth()),\n        top: +buttonOffset.top + $button.outerHeight()\n      });\n      $popupClone.show();\n      $body.addClass('civihr-popup-open');\n    }\n  });\n})(CRM.$);\n"]}
+//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["contact.js","hrui.js","civihr-popup/attrchange.js","civihr-popup/civihr-popup.js"],"names":["$","_","updateContactHeaderRolesDetails","roles","departments","length","html","join","document","on","e","one","$form","this","find","each","$label","id","attr","text","data","contract","removeClass","position","location","addClass","CRM","insertFile","fileName","files","remove","undefined","after","name","removeFile","linkLabelToDatepickerInput","$line","$input","parent","val","selector","target","parents","indexOf","get","formName","pageName","insertAfter","hide","end","show","detach","$labelCell","$fieldCell","next","$newTd","append","closest","insertBefore","prev","click","window","setTimeout","Array","prototype","forEach","call","querySelectorAll","element","Ps","initialize","children","ready","$menuItem","$wrappedLogo","$customHomeLink","$appLogo","$homeLink","className","ajax","dataType","success","menuMarkup","wrapperId","$menuMarkup","$menuWrapper","wrap","first","wrapInner","prepend","off","before","$element","$arrow","not","hover","MutationObserver","WebKitMutationObserver","MozMutationObserver","fn","attrchange","callback","options","subtree","attributes","observer","mutations","observe","$body","$popupClone","buttonOffset","$button","$popup","closePopupPanels","clone","appendTo","offset","css","left","width","outerWidth","top","outerHeight","actionIndex","index","hasClass"],"mappings":"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","file":"hrui.min.js","sourcesContent":["(function ($, _) {\n  $(document)\n    .on('crmLoad', function (e) {\n      $('.crm-inline-edit').one('DOMSubtreeModified', function () {\n        var $form = $(this).find('form');\n\n        if ($form.length === 1) {\n          $form.find('label').each(function () {\n            var $label = $(this);\n            var id = $label.attr('for');\n            $('#' + id).attr('placeholder', $label.text());\n          });\n        }\n      });\n    })\n    .on('updateContactHeader', function (e, data) {\n      if (typeof data.contract !== 'undefined') {\n        updateContactHeaderContractDetails(data.contract);\n      }\n\n      if (typeof data.roles !== 'undefined') {\n        updateContactHeaderRolesDetails(data.roles);\n      }\n    });\n\n  /**\n   * Updates the contact header with the given contract details\n   *\n   * @param  {object} contract\n   */\n  function updateContactHeaderContractDetails (contract) {\n    if (contract) {\n      $('.crm-summary-contactname-block').removeClass('crm-summary-contactname-block-without-contract');\n\n      if (contract.position) {\n        $('.crm-contact-detail-position').html('<strong>Position:</strong> ' + contract.position);\n      }\n\n      if (contract.location) {\n        $('.crm-contact-detail-location').html('<strong>Normal place of work:</strong> ' + contract.location);\n      }\n    } else {\n      $('.crm-summary-contactname-block').addClass('crm-summary-contactname-block-without-contract');\n      $('.crm-contact-detail-position').html('');\n      $('.crm-contact-detail-location').html('');\n\n      updateContactHeaderRolesDetails(null);\n    }\n  }\n\n  /**\n   * Updates the contact header with the given roles details\n   *\n   * @param  {object} contract\n   */\n  function updateContactHeaderRolesDetails (roles) {\n    if (roles && roles.departments && roles.departments.length > 0) {\n      $('.crm-contact-detail-departments').html('<strong>Department:</strong> ' + roles.departments.join(', '));\n    } else {\n      $('.crm-contact-detail-departments').html('');\n    }\n  }\n}(CRM.$, CRM._));\n","/* global Ps */\n\n// Copyright CiviCRM LLC 2013. See http://civicrm.org/licensing\n(function ($, _) {\n  $(document)\n    .on('crmLoad', function (e) {\n      addUploadFileListener(\"input[type='file']\");\n      amendVacancyForm();\n      amendContactPageAndForm(e);\n      applyMiscChanges();\n      changeContactSourceFieldHelpText();\n    })\n    .ready(function () {\n      addUserMenuToMainMenu();\n      amendAppLogoMenuItem();\n      amendApplicationForm();\n      useFontAwesomeArrowsInSubMenuItems();\n      toggleActiveClassOnHoverOnAnyMainMenuItem();\n    });\n\n  /**\n   * Customizes the app logo menu item, switching from the CiviCRM logo\n   * to the CiviHR logo, and making the item a direct link instead of a\n   * toggle for a sub menu dropdown\n   */\n  function amendAppLogoMenuItem () {\n    var $menuItem = $('.crm-link-home');\n    var $wrappedLogo = swapAndWrapAppLogo($menuItem);\n    var $customHomeLink = customizeHomeLinkInLogoMenuItem($menuItem, $wrappedLogo);\n\n    removeLogoSubMenuAndKeepOnlyHomeLink($menuItem, $customHomeLink);\n  }\n\n  /**\n   * Adds the user menu by fetching it from the hrcore extension\n   */\n  function addUserMenuToMainMenu () {\n    $.ajax('/civicrm/hrcore/usermenu?snippet=4', {\n      dataType: 'html',\n      success: function (menuMarkup) {\n        injectUserMenuInAMainMenuWrapper(menuMarkup, 'civihr-menu');\n      }\n    });\n  }\n\n  /**\n   * Add an event listener on input[type=\"file\"]\n   * @param {jQuery Object} selector [selector from input file]\n   */\n  function addUploadFileListener (selector) {\n    if ($(selector).length === 1) {\n      $(selector).on('change', insertFile);\n    }\n  }\n\n  /**\n   * Amends the application form\n   */\n  function amendApplicationForm () {\n    $('.CRM_HRRecruitment_Form_Application').addClass('crm-form-block');\n    $('.CRM_HRRecruitment_Form_Application .crm-profile-name-application_profile').addClass('form-layout-compressed');\n  }\n\n  /**\n   * Amends the contact page and the contact form\n   */\n  function amendContactPageAndForm (e) {\n    if (CRM.formName === 'contactForm' || CRM.pageName === 'viewSummary') {\n      // Rename \"Summary\" tab to \"Personal Details\"\n      // Hack to check contact type - This field only appears for individuals\n      if ($('.crm-contact-job_title', '.crm-summary-contactinfo-block').length) {\n        $('.crm-contact-tabs-list #tab_summary a', e.target).text('Personal Details');\n      }\n\n      manipulateDOMOfInlineCustomData(e.target);\n      miscContactPageChanges(e.target);\n    }\n  }\n\n  /**\n   * Amends the vacancy form\n   */\n  function amendVacancyForm () {\n    if ($('.CRM_HRRecruitment_Form_HRVacancy').length === 1) {\n      linkLabelToDatepickerInput($('label[for=\"start_date\"]').parents('tr'));\n      linkLabelToDatepickerInput($('label[for=\"end_date\"]').parents('tr'));\n\n      // Add a class to identify the form 'New Vacancy Template'\n      if ($('[name=\"entryURL\"]').val().indexOf(';template=1') > -1) {\n        $($('.CRM_HRRecruitment_Form_HRVacancy tbody').get(0)).addClass('CRM_HRRecruitment_Form_HRVacancy_Template');\n      }\n    }\n  }\n\n  /**\n   * Applies miscellaneous UI changes\n   */\n  function applyMiscChanges () {\n    $('#activityCustomData').attr('colspan', 3);\n    $('#crm-activity-view-table .crm-case-activity-view-Client .label').html('Contact');\n    $('span.crm-frozen-field', '.crm-profile-name-hrident_tab').closest('div').parent('div').hide();\n\n    $('.crm-accordion-header.crm-master-accordion-header').on('click', function () {\n      window.setTimeout(function () {\n        Array.prototype.forEach.call(document.querySelectorAll('.listing-box'), function (element) {\n          Ps.initialize(element);\n        });\n      }, 0);\n    });\n  }\n\n  /**\n   * Changes of sorce help text\n   */\n  function changeContactSourceFieldHelpText () {\n    $('INPUT#contact_source').parent('td').children('a').click(function () {\n      $('#crm-notification-container .crm-help .notify-content').remove();\n\n      if ($('#crm-notification-container .crm-help p').length) {\n        $('#crm-notification-container .crm-help p').remove();\n      }\n\n      $('#crm-notification-container .crm-help').append('<p>Source is a useful field where data has been migrated to CiviHR from one or a number of other legacy systems. The Source field will indicate which legacy system the contact has come from.</p>');\n    });\n  }\n\n  /**\n   * Finds the original link to the homepage, changes the text, wraps it in a\n   * `menumain-label` element and prepends internally the given app logo\n   *\n   * @param {object} $menuItem The context where to find the link\n   * @param {object} $appLogo\n   * @return the customized home link\n   */\n  function customizeHomeLinkInLogoMenuItem ($menuItem, $appLogo) {\n    var $homeLink = $('li > a', $menuItem).first();\n\n    return $homeLink\n      .text('Home')\n      .wrapInner('<span class=\"menumain-label\">')\n      .prepend($appLogo);\n  }\n\n  /**\n   * Injects the given markup in a menu wrapper with the given id\n   * created to contain both the original menu and the user one\n   *\n   * @param {string} menuMarkup\n   * @param {string} wrapperId\n   */\n  function injectUserMenuInAMainMenuWrapper (menuMarkup, wrapperId) {\n    var $menuMarkup = $(menuMarkup);\n    var $menuWrapper = $('<div>');\n\n    $menuWrapper.attr('id', wrapperId);\n    $menuWrapper.append($('#civicrm-menu'));\n    $menuWrapper.append($menuMarkup);\n    $menuWrapper.insertAfter('#page');\n  }\n\n  /**\n   * Insert a DOM node after input[type=\"file\"]\n   * with the filename\n   */\n  function insertFile () {\n    var fileName = $(this)[0].files[0];\n\n    $('#js-uploaded-file').remove();\n    if (fileName !== undefined) {\n      $(this).after('<span id=\"js-uploaded-file\" class=\"uploaded-file\">' + fileName.name + ' <span class=\"uploaded-file-icon-trash\"><i class=\"fa fa-trash-o\"></i> Remove</span>');\n\n      $('.uploaded-file-icon-trash').on('click', removeFile);\n    }\n  }\n\n  /**\n   * Update label 'for' attr to works with the datepicker\n   *\n   * @param  {jQuery object} $line [datepicker's line parent]\n   */\n  function linkLabelToDatepickerInput ($line) {\n    $line.find('label').attr('for', $line.find('.crm-form-date').attr('id'));\n  }\n\n  /**\n   * Manipulates, at the DOM level, the blocks/fields belonging to the\n   * Inline Custom Data custom fields set\n   */\n  function manipulateDOMOfInlineCustomData () {\n    if ($('.Inline_Custom_Data').length) {\n      repositionInlineCustomDataBlockInPersonalDetailsTab();\n    }\n\n    if ($('#customFields').length < 1) {\n      repositionInlineCustomDataFieldsInEditContactForm();\n    }\n  }\n\n  /**\n   * Misc changes to the page (hiding elements, inserting new ones, etc)\n   */\n  function miscContactPageChanges (target) {\n    // Hide current employer and job title\n    // Contact summary screen:\n    $('div.crm-contact-current_employer, div.crm-contact-job_title', '.crm-summary-contactinfo-block').parent('div.crm-summary-row').hide();\n    // Inline edit form\n    $('form#ContactInfo input#employer_id, form#ContactInfo input#job_title', target).closest('div.crm-summary-row').hide();\n    // Contact edit screen\n    $('input#employer_id, input#job_title', 'form#Contact').parent('td').hide();\n\n    /* Changes on Add Individual pages and Personal details tab for HR-358 */\n    // Move Job summary to top\n    $('.HRJobContract_Summary', target).insertBefore($('.crm-summary-contactinfo-block'));\n    // changes of email block, remove bulkmail and onhold\n    $('div.email-signature, td#Email-Bulkmail-html', 'form#Contact').hide();\n    $('#Email-Primary', 'form#Contact').prev('td').prev('td').hide();\n    $('td#Email-Bulkmail-html, #Email-Primary', 'form#Contact').prev('td').hide();\n\n    // shift demographic above extended demographic\n    $('.crm-demographics-accordion', 'form#Contact').insertAfter($('.crm-contactDetails-accordion'));\n\n    if ($('tr#Phone_Block_2', 'form#Contact').length < 1) {\n      $('#addPhone').click();\n    }\n  }\n\n  /**\n   * Remove the #js-uploaded-file DIV and\n   * clean input[type=\"file\"] value\n   */\n  function removeFile () {\n    var $input = $('#js-uploaded-file').parent().find('input[type=\"file\"]');\n\n    $('#js-uploaded-file').remove();\n    $input.val('');\n  }\n\n  /**\n   * Moves the given home link right under the menu item and gets rid\n   * of the original sub menu\n   *\n   * @param {object} $menuItem The context where to find the link\n   * @param {object} $homeLink\n   */\n  function removeLogoSubMenuAndKeepOnlyHomeLink ($menuItem, $homeLink) {\n    $menuItem\n      .off() // removes any handler that the original item had\n      .find('#civicrm-home')\n      .before($homeLink)\n      .remove();\n  }\n\n  /**\n   * Moves the \"Inline Custom Data\" fields towards the top of the\n   * edit contact form\n   */\n  function repositionInlineCustomDataFieldsInEditContactForm () {\n    var $fields = $('#Inline_Custom_Data').detach();\n\n    $fields.find('.label').each(function () {\n      var $labelCell = $(this);\n      var $fieldCell = $labelCell.next();\n      var $newTd = $('<td/>');\n\n      $newTd.append($labelCell.find('label'));\n      $newTd.append($('<br/>'));\n      $newTd.append($fieldCell.html());\n\n      $('#nick_name').parent().after($newTd);\n    });\n  }\n\n  /**\n   * Moves the \"Inline Custom Data\" block towards the top of the\n   * personal details tab\n   */\n  function repositionInlineCustomDataBlockInPersonalDetailsTab () {\n    $('.Inline_Custom_Data')\n      .removeClass('crm-collapsible collapsed')\n      .addClass('crm-summary-block')\n      .insertAfter('.crm-summary-contactinfo-block')\n      .find('.collapsible-title').hide().end()\n      .find('.crm-summary-block').show();\n  }\n\n  /**\n   * Swaps the CiviCRM logo with the CiviHR logo\n   * and wraps it in a `menumain-icon` element\n   *\n   * @param {object} $menuItem The context where to find the logo\n   * @return the wrapper of the logo\n   */\n  function swapAndWrapAppLogo ($menuItem) {\n    var $appLogo = $('.crm-logo-sm', $menuItem);\n\n    return $appLogo\n      .addClass('chr_logo chr_logo--default-color')\n      .removeClass('crm-logo-sm')\n      .wrap('<span class=\"menumain-icon\">')\n      .parent();\n  }\n\n  /**\n   * CiviCRM by default applies on hover the .activetarget class\n   * only to main menu items with a submenu\n   *\n   * This functions makes sure that any item gets the class applied,\n   * even those with just a direct link\n   */\n  function toggleActiveClassOnHoverOnAnyMainMenuItem () {\n    var className = 'activetarget';\n\n    $('.menumain').not('.crm-Self_Service_Portal').hover(function () {\n      $(this).addClass(className);\n    }, function () {\n      $(this).removeClass(className);\n    });\n  }\n\n  /**\n   * Remove the arrow for menu items with sub-items, and replaces it\n   * with a font awesome caret\n   */\n  function useFontAwesomeArrowsInSubMenuItems () {\n    $('#root-menu-div .menu-item-arrow').each(function ($element) {\n      var $arrow = $(this);\n\n      $arrow.before('<i class=\"fa fa-caret-right menu-item-arrow\"></i>');\n      $arrow.remove();\n    });\n  }\n}(CRM.$, CRM._));\n","(function ($) {\n  'use strict';\n  var MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver;\n\n  /*\n  This plugin creates MutationObserver which listens to all changes to DOM Node.\n  If change has been done to it's attributes callback will be called.\n\n  @param {function} callback - to be called when attributes change\n  @returns {Array} - list of all observed elements\n  */\n  $.fn.attrchange = function (callback) {\n    if (MutationObserver) {\n      var options = {\n        subtree: false,\n        attributes: true\n      };\n\n      var observer = new MutationObserver(function (mutations) {\n        mutations.forEach(function (e) {\n          callback.call(e.target, e);\n        });\n      });\n\n      return this.each(function () {\n        observer.observe(this, options);\n      });\n    }\n  }\n})(CRM.$);\n","/* globals CRM */\n\n/**\n * Helps to display popup menus inside #bootstrap-theme elements, since .panel's\n * and other Bootstrap elements clash with CiviCRM styles producing undesired\n * results.\n *\n * It also helps with event bindings, in case the original popup button was\n * removed and recreated, by forwarding the events to the original source.\n */\n(function ($) {\n  'use strict';\n  var $body = $('body');\n\n  $body.on('click', '#bootstrap-theme .btn-slide', function () {\n    var $popupClone;\n    var $button = $(this);\n    var $popup = $button.children('ul.panel');\n\n    (function init () {\n      closePopupPanels();\n      createPopupClone();\n      openPopupClone();\n      mapCloneClickEventsToOrigin();\n      listenToMouseOutEvent();\n    })();\n\n    /**\n     * Closes the any popup panel that is currently open and removes the\n     * `.civihr-popup-open` class from the body.\n     */\n    function closePopupPanels () {\n      $('.civihr-popup').remove();\n      $body.removeClass('civihr-popup-open');\n    }\n\n    /**\n     * Creates a clone of the popup element and appends it to the document body.\n     * This prevents the popup to be hidden by any `overflow: hidden;` rule.\n     */\n    function createPopupClone () {\n      $popupClone = $popup.clone(true)\n        .appendTo($body)\n        .addClass('civihr-popup');\n    }\n\n    /**\n     * Listens to mouse events happening outside the of the panel in order to\n     * close it. The way to detected mouse out is by listening to the\n     * .btn-slide-active class change, which is added and removed by CiviCRM.\n     * the data *attrchange-is-on* is set to true in order to avoid adding\n     * multiple listeners to the same element.\n     */\n    function listenToMouseOutEvent () {\n      // If AttrChange event is already listened, then skip:\n      if ($button.data('attrchange-is-on')) {\n        return;\n      }\n\n      $button.data('attrchange-is-on', true);\n      $button.attrchange(function () {\n        // If button is already open, then skip:\n        if ($button.hasClass('btn-slide-active')) {\n          return;\n        }\n\n        closePopupPanels();\n      });\n    }\n\n    /**\n     * Maps click events on the popup options back to their original source.\n     * This is done because popup actions are executed as delegated events and\n     * the listener is not the *body* element.\n     */\n    function mapCloneClickEventsToOrigin () {\n      $popupClone.find('a').click(function () {\n        var actionIndex = $(this).parent().index();\n\n        $popup.find('li:nth(' + actionIndex + ') a').click();\n      });\n    }\n\n    /**\n     * Opens the popup panel clone\n     * and adds the .civihr-popup-open class to the body\n     */\n    function openPopupClone () {\n      var buttonOffset = $button.offset();\n\n      $popupClone.css({\n        left: +buttonOffset.left - ($popupClone.width() - $button.outerWidth()),\n        top: +buttonOffset.top + $button.outerHeight()\n      });\n      $popupClone.show();\n      $body.addClass('civihr-popup-open');\n    }\n  });\n})(CRM.$);\n"]}
diff --git a/hrui/js/src/hrui.js b/hrui/js/src/hrui.js
index a8085de481e..2bd80d64aae 100755
--- a/hrui/js/src/hrui.js
+++ b/hrui/js/src/hrui.js
@@ -186,13 +186,11 @@
* Manipulates, at the DOM level, the blocks/fields belonging to the
* Inline Custom Data custom fields set
*/
- if ($('.Inline_Custom_Data').length) {
- repositionPersonalDetailsBlock('.Inline_Custom_Data');
- }
-
- repositionPersonalDetailsBlock('.Extended_Demographics');
-
function manipulateDOMOfInlineCustomData () {
+ if ($('.Inline_Custom_Data').length) {
+ repositionInlineCustomDataBlockInPersonalDetailsTab();
+ }
+
if ($('#customFields').length < 1) {
repositionInlineCustomDataFieldsInEditContactForm();
}
@@ -273,15 +271,14 @@
}
/**
- * Moves the specified selector element to the personal details tab
- *
- * @param selector
+ * Moves the "Inline Custom Data" block towards the top of the
+ * personal details tab
*/
- function repositionPersonalDetailsBlock (selector) {
- $(selector)
+ function repositionInlineCustomDataBlockInPersonalDetailsTab () {
+ $('.Inline_Custom_Data')
.removeClass('crm-collapsible collapsed')
.addClass('crm-summary-block')
- .appendTo('.contactTopBar .contactCardLeft')
+ .insertAfter('.crm-summary-contactinfo-block')
.find('.collapsible-title').hide().end()
.find('.crm-summary-block').show();
}
diff --git a/hrvisa/tests/phpunit/CRM/HRVisa/ActivityTest.php b/hrvisa/tests/phpunit/CRM/HRVisa/ActivityTest.php
index 49e40ddaa7a..d822db9ec16 100644
--- a/hrvisa/tests/phpunit/CRM/HRVisa/ActivityTest.php
+++ b/hrvisa/tests/phpunit/CRM/HRVisa/ActivityTest.php
@@ -16,6 +16,8 @@ class CRM_HRVisa_ActivityTest extends PHPUnit_Framework_TestCase implements Head
public function setUpHeadless() {
return \Civi\Test::headless()
->installMe(__DIR__)
+ // hrdemog is necessary because it creates the Immigration fields used by the tests
+ ->install('org.civicrm.hrdemog')
->install('uk.co.compucorp.civicrm.hrcore')
->apply();
}
diff --git a/uk.co.compucorp.civicrm.hrcontactactionsmenu/CRM/HRContactActionsMenu/Page/UserMailNotifier.php b/uk.co.compucorp.civicrm.hrcontactactionsmenu/CRM/HRContactActionsMenu/Page/UserMailNotifier.php
index f3907946bd1..aadc269032f 100644
--- a/uk.co.compucorp.civicrm.hrcontactactionsmenu/CRM/HRContactActionsMenu/Page/UserMailNotifier.php
+++ b/uk.co.compucorp.civicrm.hrcontactactionsmenu/CRM/HRContactActionsMenu/Page/UserMailNotifier.php
@@ -14,7 +14,7 @@ class CRM_HRContactActionsMenu_Page_UserMailNotifier {
public static function sendPasswordResetEmail() {
$contactID = CRM_Utils_Array::value('cid', $_GET);
$contactInfo = ContactHelper::getUserInformation($contactID);
- $cmsUserMailNotifier = Civi::container()->get('civihr.cms.notifier');
+ $cmsUserMailNotifier = Civi::container()->get('hrcore.cms_notifier');
$cmsUserMailNotifier->sendPasswordResetEmail($contactInfo);
CRM_Core_Session::setStatus(ts('Password Reset Email sent'), 'Success', 'success');
@@ -30,7 +30,7 @@ public static function sendPasswordResetEmail() {
public static function sendWelcomeEmail() {
$contactID = CRM_Utils_Array::value('cid', $_GET);
$contactInfo = ContactHelper::getUserInformation($contactID);
- $cmsUserMailNotifier = Civi::container()->get('civihr.cms.notifier');
+ $cmsUserMailNotifier = Civi::container()->get('hrcore.cms_notifier');
$cmsUserMailNotifier->sendWelcomeEmail($contactInfo);
CRM_Core_Session::setStatus(ts('Welcome Email sent'), 'Success', 'success');
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Form/AbstractDrupalInteractionTaskForm.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Form/AbstractDrupalInteractionTaskForm.php
index d328fcbf322..0997556e4af 100644
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Form/AbstractDrupalInteractionTaskForm.php
+++ b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Form/AbstractDrupalInteractionTaskForm.php
@@ -26,7 +26,7 @@ public function __construct(
$method = 'post',
$name = NULL
) {
- $this->drupalUserService = Civi::container()->get('civihr.cms.drupal_user_service');
+ $this->drupalUserService = Civi::container()->get('drupal_user_service');
parent::__construct($state, $action, $method, $name);
}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Disable/CustomGroupDisabler.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Disable/CustomGroupDisabler.php
deleted file mode 100644
index bc2369c0dc8..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Disable/CustomGroupDisabler.php
+++ /dev/null
@@ -1,17 +0,0 @@
-get('civihr.custom_group_status_switcher');
-
- foreach ($customGroups as $groupName) {
- $switcher->disable($groupName);
- }
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Enable/CustomGroupEnabler.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Enable/CustomGroupEnabler.php
deleted file mode 100644
index b6043723f4a..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Enable/CustomGroupEnabler.php
+++ /dev/null
@@ -1,21 +0,0 @@
-enable($groupName);
- }
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Uninstall/CustomGroupRemover.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Uninstall/CustomGroupRemover.php
deleted file mode 100644
index 302e5ee0664..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Hook/Uninstall/CustomGroupRemover.php
+++ /dev/null
@@ -1,25 +0,0 @@
- 'id', 'name' => $customGroupName];
- $result = civicrm_api3('CustomGroup', 'get', $params);
-
- if ($result['count'] != 1) {
- continue;
- }
- $customGroup = array_shift($result['values']);
-
- civicrm_api3('CustomGroup', 'delete', ['id' => $customGroup['id']]);
- }
- }
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Service/CustomGroupStatusSwitcher.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Service/CustomGroupStatusSwitcher.php
deleted file mode 100644
index f381b3be7d5..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Service/CustomGroupStatusSwitcher.php
+++ /dev/null
@@ -1,102 +0,0 @@
-changeStatus($customGroupName, TRUE);
- }
-
- /**
- * Disable a custom group and all its fields
- *
- * @param string $customGroupName
- */
- public function disable($customGroupName) {
- $this->changeStatus($customGroupName, FALSE);
- }
-
- /**
- * Switches the status for custom group and all its fields
- *
- * @param string $customGroupName
- * @param bool $status
- */
- private function changeStatus($customGroupName, $status) {
- $customGroup = $this->getCustomGroup($customGroupName);
-
- if (!$customGroup) {
- $err = sprintf('Could not find group with name "%s"', $customGroupName);
- throw new \Exception($err);
- }
-
- $customGroupId = (int) $customGroup['id'];
- $this->changeGroupStatus($customGroupId, $status);
- $this->changeAllGroupFieldsStatus($customGroupId, $status);
- }
-
- /**
- * Update the 'is_active' status for a custom group
- *
- * @param int $customGroupId
- * @param bool $status
- */
- private function changeGroupStatus($customGroupId, $status) {
- $params = ['id' => $customGroupId, 'is_active' => $status];
- civicrm_api3('CustomGroup', 'create', $params);
- }
-
- /**
- * Update the 'is_active' status for all custom fields for a group
- *
- * @param int $customGroupId
- * @param bool $status
- */
- private function changeAllGroupFieldsStatus($customGroupId, $status) {
- $fields = $this->getAllCustomFields($customGroupId);
-
- foreach ($fields as $field) {
- $params = ['id' => $field['id'], 'is_active' => $status];
- // Custom field endpoint freaks out if custom_group_id is missing
- $params['custom_group_id'] = $field['custom_group_id'];
- civicrm_api3('CustomField', 'create', $params);
- }
- }
-
- /**
- * Fetches all custom fields for a custom group
- *
- * @param int $customGroupId
- *
- * @return array
- */
- private function getAllCustomFields($customGroupId) {
- $params = ['custom_group_id' => $customGroupId];
- $result = civicrm_api3('CustomField', 'get', $params);
-
- return $result['values'];
- }
-
- /**
- * Gets a custom group based on name, returns null if group doesn't exist
- *
- * @param string $customGroupName
- *
- * @return mixed|null
- */
- private function getCustomGroup($customGroupName) {
- $params = ['name' => $customGroupName];
- $result = civicrm_api3('CustomGroup', 'get', $params);
-
- if ($result['count'] != 1) {
- return NULL;
- }
-
- return array_shift($result['values']);
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Test/Fabricator/CustomField.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Test/Fabricator/CustomField.php
deleted file mode 100644
index e2030ae18c1..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Test/Fabricator/CustomField.php
+++ /dev/null
@@ -1,23 +0,0 @@
- 'test_custom_field',
- 'html_type' => 'Text',
- 'data_type' => 'String',
- ];
-
- public static function fabricate($params = []) {
- $params = array_merge(self::$defaultParams, $params);
-
- if (!isset($params['label'])) {
- $params['label'] = $params['name'];
- }
-
- $result = civicrm_api3('CustomField', 'create', $params);
-
- return array_shift($result['values']);
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Test/Fabricator/CustomGroup.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Test/Fabricator/CustomGroup.php
deleted file mode 100644
index 5ac6eecbbdd..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Test/Fabricator/CustomGroup.php
+++ /dev/null
@@ -1,28 +0,0 @@
- 'test_custom_group_' . $count,
- 'title' => 'Test Custom Group ' . $count,
- 'extends' => 'Individual'
- ];
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader.php
index 208c2f772d5..2d69d9e62dc 100755
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader.php
+++ b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader.php
@@ -25,7 +25,6 @@ class CRM_HRCore_Upgrader extends CRM_HRCore_Upgrader_Base {
use CRM_HRCore_Upgrader_Steps_1015;
use CRM_HRCore_Upgrader_Steps_1016;
use CRM_HRCore_Upgrader_Steps_1017;
- use CRM_HRCore_Upgrader_Steps_1021;
use CRM_HRCore_Upgrader_Steps_1018;
use CRM_HRCore_Upgrader_Steps_1019;
use CRM_HRCore_Upgrader_Steps_1020;
@@ -59,18 +58,10 @@ class CRM_HRCore_Upgrader extends CRM_HRCore_Upgrader_Base {
'Validate Email Address from Mailings.'
];
- /**
- * A list of directories to be scanned for XML installation files
- *
- * @var array
- */
- private $xmlDirectories = ['CustomGroups'];
-
/**
* Callback called when the extension is installed
*/
public function install() {
- $this->processXMLInstallationFiles();
$this->setScheduledJobsDefaultStatus();
$this->deleteLocationTypes();
$this->createRequiredLocationTypes();
@@ -79,25 +70,6 @@ public function install() {
$this->runAllUpgraders();
}
- /**
- * Scans all the directories in $xmlDirectories for installation files
- * (xml files ending with _install.xml) and processes them.
- */
- private function processXMLInstallationFiles() {
- foreach($this->xmlDirectories as $directory) {
- $files = glob($this->extensionDir . "/xml/{$directory}/*_install.xml");
- if (is_array($files)) {
- foreach ($files as $file) {
- $this->executeCustomDataFileByAbsPath($file);
- }
- }
- }
- // Flush the cache so that all pseudoconstants can be re-read from db
- // This is to avoid issues when running upgraders during installation
- // whereby some pseudoconstants were not available.
- CRM_Core_PseudoConstant::flush();
- }
-
/**
* Callback method called when the extension is uninstalled.
*
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader/Steps/1021.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader/Steps/1021.php
deleted file mode 100644
index 0a59b627d43..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader/Steps/1021.php
+++ /dev/null
@@ -1,27 +0,0 @@
- $key]);
- civicrm_api3('Extension', 'uninstall', ['keys' => $key]);
-
- return TRUE;
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader/Steps/1025.php b/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader/Steps/1025.php
deleted file mode 100644
index 56cf2743be4..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/CRM/HRCore/Upgrader/Steps/1025.php
+++ /dev/null
@@ -1,54 +0,0 @@
-up1025_deleteExtendDemographicFields([
- 'Ethnicity',
- 'Religion',
- 'Sexual_Orientation',
- ]);
-
- return TRUE;
- }
-
- /**
- * Deletes Custom Demographic Fields only if they are not used
- *
- * @param array $fieldsToDelete
- */
- private function up1025_deleteExtendDemographicFields($fieldsToDelete) {
- $customGroup = civicrm_api3('CustomGroup', 'get', [
- 'name' => 'Extended_Demographics',
- ]);
- $customGroup = array_shift($customGroup['values']);
- $customFields = civicrm_api3('CustomField', 'get', [
- 'name' => ['IN' => $fieldsToDelete],
- ]);
- $tableName = $customGroup['table_name'];
- foreach ($customFields['values'] as $customField) {
- $column = $customField['column_name'];
- $queryFormat = 'SELECT COUNT(id) FROM %s'
- . ' WHERE %s NOT LIKE "%%Not Applicable%%"'
- . ' AND %s IS NOT NULL'
- . ' AND %s <> ""';
-
- $query = sprintf($queryFormat, $tableName, $column, $column, $column);
- $customFieldItems = CRM_Core_DAO::singleValueQuery($query);
- if ($customFieldItems > 0) {
- continue;
- }
-
- civicrm_api3('CustomField', 'delete', [
- 'id' => $customField['id'],
- ]);
- }
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrcore/api/v3/Job/CheckCivihrVersion.php b/uk.co.compucorp.civicrm.hrcore/api/v3/Job/CheckCivihrVersion.php
index 22f3b99120a..d1956223a66 100644
--- a/uk.co.compucorp.civicrm.hrcore/api/v3/Job/CheckCivihrVersion.php
+++ b/uk.co.compucorp.civicrm.hrcore/api/v3/Job/CheckCivihrVersion.php
@@ -7,8 +7,8 @@
*/
function civicrm_api3_job_check_civihr_version() {
$container = Civi::container();
- $currentStats = $container->get('civihr.stats_cache')->fetchCurrent();
- $container->get('civihr.stats_sender')->send($currentStats);
+ $currentStats = $container->get('civihr_stats_cache')->fetchCurrent();
+ $container->get('civihr_stats_sender')->send($currentStats);
return civicrm_api3_create_success();
}
diff --git a/uk.co.compucorp.civicrm.hrcore/config/container/container.xml b/uk.co.compucorp.civicrm.hrcore/config/container/container.xml
index 3055b64c30b..ffa8e04b42c 100644
--- a/uk.co.compucorp.civicrm.hrcore/config/container/container.xml
+++ b/uk.co.compucorp.civicrm.hrcore/config/container/container.xml
@@ -3,50 +3,48 @@
xsi:schemaLocation="http://symfony.com/schema/dic/services http://symfony.com/schema/dic/services/services-1.0.xsd">
- 10
+ 10
-
-
-
-
+
+
+
-
-
-
+
+
+
- %civihr.connection_timeout%
+ %connection_timeout%
-
+
-
-
+
+
-
diff --git a/uk.co.compucorp.civicrm.hrcore/hrcore.php b/uk.co.compucorp.civicrm.hrcore/hrcore.php
index afc7549338b..6c3eb765a2e 100644
--- a/uk.co.compucorp.civicrm.hrcore/hrcore.php
+++ b/uk.co.compucorp.civicrm.hrcore/hrcore.php
@@ -180,14 +180,6 @@ function hrcore_civicrm_postInstall() {
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_uninstall
*/
function hrcore_civicrm_uninstall() {
- $listeners = [
- new CRM_HRCore_Hook_Uninstall_CustomGroupRemover()
- ];
-
- foreach ($listeners as $listener) {
- $listener->handle();
- }
-
_hrcore_civix_civicrm_uninstall();
}
@@ -197,19 +189,6 @@ function hrcore_civicrm_uninstall() {
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_enable
*/
function hrcore_civicrm_enable() {
- // _hrcore_civix_civicrm_config() will add this extension to the include path.
- // We need to do this here because before extension is enabled the config
- // hook is not called for it
- _hrcore_civix_civicrm_config();
-
- $listeners = [
- new CRM_HRCore_Hook_Enable_CustomGroupEnabler()
- ];
-
- foreach ($listeners as $listener) {
- $listener->handle();
- }
-
_hrcore_civix_civicrm_enable();
}
@@ -219,14 +198,6 @@ function hrcore_civicrm_enable() {
* @link http://wiki.civicrm.org/confluence/display/CRMDOC/hook_civicrm_disable
*/
function hrcore_civicrm_disable() {
- $listeners = [
- new CRM_HRCore_Hook_Disable_CustomGroupDisabler()
- ];
-
- foreach ($listeners as $listener) {
- $listener->handle();
- }
-
_hrcore_civix_civicrm_disable();
}
diff --git a/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Helper/ExtensionHelperTest.php b/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Helper/ExtensionHelperTest.php
index e67d3ea052d..f7bde0da97f 100644
--- a/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Helper/ExtensionHelperTest.php
+++ b/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Helper/ExtensionHelperTest.php
@@ -10,15 +10,16 @@ class CRM_HRCore_Helper_ExtensionHelperTest extends CRM_HRCore_Test_BaseHeadless
/**
* @var string
*/
- private $extensionKey = 'uk.co.compucorp.civicrm.hremails';
+ private $hrCoreKey = 'uk.co.compucorp.civicrm.hrcore';
public function testCheckIsFalseAfterExtensionIsDisabled() {
- civicrm_api3('Extension', 'disable', ['keys' => $this->extensionKey]);
- $this->assertFalse(ExtensionHelper::isExtensionEnabled($this->extensionKey));
+ // hrcore is enabled by default in CRM_HRCore_Test_BaseHeadlessTest
+ civicrm_api3('Extension', 'disable', ['keys' => $this->hrCoreKey]);
+ $this->assertFalse(ExtensionHelper::isExtensionEnabled($this->hrCoreKey));
}
public function testCheckIsTrueAfterExtensionIsEnabled() {
- civicrm_api3('Extension', 'enable', ['keys' => $this->extensionKey]);
- $this->assertTrue(ExtensionHelper::isExtensionEnabled($this->extensionKey));
+ civicrm_api3('Extension', 'enable', ['keys' => $this->hrCoreKey]);
+ $this->assertTrue(ExtensionHelper::isExtensionEnabled($this->hrCoreKey));
}
}
diff --git a/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Service/CustomGroupStatusSwitcherTest.php b/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Service/CustomGroupStatusSwitcherTest.php
deleted file mode 100644
index a1d65aacdd3..00000000000
--- a/uk.co.compucorp.civicrm.hrcore/tests/phpunit/CRM/HRCore/Service/CustomGroupStatusSwitcherTest.php
+++ /dev/null
@@ -1,140 +0,0 @@
-disable(self::$customGroup['name']);
-
- // check that all custom fields were disabled
- foreach (self::$customFields as $customField) {
- $id = $customField['id'];
- $updatedField = civicrm_api3('CustomField', 'getsingle', ['id' => $id]);
- $this->assertEquals(0, $updatedField['is_active']);
- }
-
- // check custom group was disabled
- $updatedGroup = civicrm_api3('CustomGroup', 'getsingle', ['id' => $groupId]);
- $this->assertEquals(0, $updatedGroup['is_active']);
-
- }
-
- public function testEnablingWillEnableAllFields() {
- $groupId = self::$customGroup['id'];
- // disable the group
- civicrm_api3('CustomGroup', 'create', ['id' => $groupId, 'is_active' => 0]);
-
- // disable the fields
- foreach (self::$customFields as $customField) {
- civicrm_api3('CustomField', 'create', [
- 'id' => $customField['id'],
- 'is_active' => 0,
- ]);
- }
-
- $switcher = new CustomGroupStatusSwitcher();
- $switcher->enable(self::$customGroup['name']);
-
- // check that all custom fields were enabled
- foreach (self::$customFields as $customField) {
- $id = $customField['id'];
- $updatedField = civicrm_api3('CustomField', 'getsingle', ['id' => $id]);
- $this->assertEquals(1, $updatedField['is_active']);
- }
-
- // check custom group was enabled
- $updatedGroup = civicrm_api3('CustomGroup', 'getsingle', ['id' => $groupId]);
- $this->assertEquals(1, $updatedGroup['is_active']);
- }
-
- public function testNonExistingGroupNameWillThrowException() {
- $groupName = 'Lala';
- $expectedMessage = 'Could not find group with name "Lala"';
- $this->setExpectedException(\Exception::class, $expectedMessage);
- $switcher = new CustomGroupStatusSwitcher();
- $switcher->enable($groupName);
- }
-
- /**
- * CiviCRM test transactions are broken by custom group creation. If you
- * create and subsequently delete a custom group in a test it will drop the
- * table but leave the civicrm_custom_group entry. This means the next
- * time the test is run it will fail when it tries to drop a table that
- * doesn't exist.
- *
- * Similarly with custom fields, if you drop one inside a test the column
- * will be dropped outside of the transaction, but the field will only be
- * dropped inside the transaction, meaning the next time you try to delete
- * a custom group it will fail because it tries to drop columns that don't
- * exist.
- *
- * To avoid these problems all custom group / field creation and deletion is
- * done outside the test. The CiviTestListener manages transactions in
- * SetUp and TearDown, so we use the "BeforeClass" and "AfterClass" methods
- */
- public static function setUpBeforeClass() {
- // If the last test critically errored the group might still exist
- self::deleteCustomGroupIfExists('Foo');
-
- self::$customGroup = CustomGroupFabricator::fabricate(['name' => 'Foo']);
- self::createCustomFields(2);
- }
-
- /**
- * Clean up everything created in this test class
- */
- public static function tearDownAfterClass() {
- static::deleteCustomGroupIfExists(self::$customGroup['name']);
- }
-
- /**
- * Checks if a custom group exists (by name) and deletes it if it does
- *
- * @param string $groupName
- */
- private static function deleteCustomGroupIfExists($groupName) {
- $existing = civicrm_api3('CustomGroup', 'get', ['name' => $groupName]);
- if ($existing['count'] > 0) {
- $existing = array_shift($existing['values']);
- civicrm_api3('CustomGroup', 'delete', ['id' => $existing['id']]);
- }
- }
-
- /**
- * Create some custom fields for the test custom group
- *
- * @param int $count
- */
- private static function createCustomFields($count) {
- $customFields = [];
-
- for ($i = 0; $i < $count; $i++) {
- $params['custom_group_id'] = self::$customGroup['id'];
- $params['name'] = 'bar_' . $i;
- $customFields[] = CustomFieldFabricator::fabricate($params);
- }
-
- self::$customFields = $customFields;
- }
-
-}
diff --git a/uk.co.compucorp.civicrm.hrsampledata/resources/csv/civicrm_option_value.csv b/uk.co.compucorp.civicrm.hrsampledata/resources/csv/civicrm_option_value.csv
index ed6ff7da462..70beb41a1e1 100644
--- a/uk.co.compucorp.civicrm.hrsampledata/resources/csv/civicrm_option_value.csv
+++ b/uk.co.compucorp.civicrm.hrsampledata/resources/csv/civicrm_option_value.csv
@@ -85,6 +85,9 @@ name,phone_type,Mobile,Mobile,,0,0,0,0,,0
name,marital_status_20130913084916,Single,Single,,0,0,0,0,,0
name,marital_status_20130913084916,Married,Married,,0,0,0,0,,0
name,marital_status_20130913084916,Divorced,Divorced,,0,0,0,0,,0
+name,sexual_orientation_20130725124348,Not Applicable,Not Applicable,,0,0,0,0,,0
+name,religion_20130725124132,Not Applicable,Not Applicable,,0,0,0,0,,0
+name,ethnicity_20130725123943,Not Applicable,Not Applicable,,0,0,0,0,,0
name,hrjc_benefit_type,Fixed,Fixed,,0,0,0,0,,0
name,hrjc_benefit_name,Bike,Bike,,0,0,0,0,,0
name,postal_greeting,Dear {contact.first_name},Dear {contact.first_name},,1,1,0,0,,0
diff --git a/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/BaseCSVProcessorTest.php b/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/BaseCSVProcessorTest.php
index 04c4a161ff7..59bd263f19d 100644
--- a/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/BaseCSVProcessorTest.php
+++ b/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/BaseCSVProcessorTest.php
@@ -20,6 +20,7 @@ public function setUpHeadless() {
->install('com.civicrm.hrjobroles')
->install('org.civicrm.hrrecruitment')
->install('org.civicrm.hremergency')
+ ->install('org.civicrm.hrdemog')
->install('org.civicrm.hrbank')
->install('uk.co.compucorp.civicrm.tasksassignments')
->install('org.civicrm.hrcase')
diff --git a/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/Importer/ExtendedDemographicsTest.php b/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/Importer/ExtendedDemographicsTest.php
index a429706bc4b..60a7184a6eb 100644
--- a/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/Importer/ExtendedDemographicsTest.php
+++ b/uk.co.compucorp.civicrm.hrsampledata/tests/phpunit/CRM/HRSampleData/Importer/ExtendedDemographicsTest.php
@@ -22,6 +22,9 @@ public function testProcess() {
$this->rows[] = [
$this->testContact['id'],
1020,
+ 'Not Applicable',
+ 'Not Applicable',
+ 'Not Applicable',
'Single',
];
@@ -40,6 +43,9 @@ private function importHeadersFixture() {
return [
'entity_id',
'Nationality',
+ 'Ethnicity',
+ 'Religion',
+ 'Sexual_Orientation',
'Marital_Status',
];
}