This repository has been archived on 2025-06-21. You can view files and clone it, but you cannot make any changes to it's state, such as pushing and creating new issues, pull requests or comments.
suitedesk/modules/views/js/dependent.js

195 lines
6.5 KiB
JavaScript

/**
* @file dependent.js
*
* Written by dmitrig01 (Dmitri Gaskin) for Views; this provides dependent
* visibility for form items in Views' ajax forms.
*
* To your $form item definition add:
* - '#process' => array('views_process_dependency'),
* - Add '#dependency' => array('id-of-form-item' => array(list, of, values, that,
make, this, item, show),
*
* Special considerations:
* - radios are harder. Because Drupal doesn't give radio groups individual ids,
* use 'radio:name-of-radio'
*
* - Checkboxes don't have their own id, so you need to add one in a div
* around the checkboxes via #prefix and #suffix. You actually need to add TWO
* divs because it's the parent that gets hidden. Also be sure to retain the
* 'expand_checkboxes' in the #process array, because the views process will
* override it.
*/
Drupal.Views = Drupal.Views || {};
Drupal.Views.dependent = { bindings: {}, activeBindings: {}, activeTriggers: [] };
Drupal.Views.dependent.inArray = function(array, search_term) {
var i = array.length;
if (i > 0) {
do {
if (array[i] == search_term) {
return true;
}
} while (i--);
}
return false;
}
Drupal.Views.dependent.autoAttach = function() {
// Clear active bindings and triggers.
for (i in Drupal.Views.dependent.activeTriggers) {
jQuery(Drupal.Views.dependent.activeTriggers[i]).unbind('change');
}
Drupal.Views.dependent.activeTriggers = [];
Drupal.Views.dependent.activeBindings = {};
Drupal.Views.dependent.bindings = {};
if (!Drupal.settings.viewsAjax) {
return;
}
// Iterate through all relationships
for (id in Drupal.settings.viewsAjax.formRelationships) {
// Drupal.Views.dependent.activeBindings[id] is a boolean,
// whether the binding is active or not. Defaults to no.
Drupal.Views.dependent.activeBindings[id] = 0;
// Iterate through all possible values
for(bind_id in Drupal.settings.viewsAjax.formRelationships[id].values) {
// This creates a backward relationship. The bind_id is the ID
// of the element which needs to change in order for the id to hide or become shown.
// The id is the ID of the item which will be conditionally hidden or shown.
// Here we're setting the bindings for the bind
// id to be an empty array if it doesn't already have bindings to it
if (!Drupal.Views.dependent.bindings[bind_id]) {
Drupal.Views.dependent.bindings[bind_id] = [];
}
// Add this ID
Drupal.Views.dependent.bindings[bind_id].push(id);
// Big long if statement.
// Drupal.settings.viewsAjax.formRelationships[id].values[bind_id] holds the possible values
if (bind_id.substring(0, 6) == 'radio:') {
var trigger_id = "input[name='" + bind_id.substring(6) + "']";
}
else {
var trigger_id = '#' + bind_id;
}
Drupal.Views.dependent.activeTriggers.push(trigger_id);
if (jQuery(trigger_id).attr('type') == 'checkbox') {
$(trigger_id).parent().addClass('hidden-options');
}
var getValue = function(item, trigger) {
if (item.substring(0, 6) == 'radio:') {
var val = jQuery(trigger + ':checked').val();
}
else {
switch (jQuery(trigger).attr('type')) {
case 'checkbox':
var val = jQuery(trigger).attr('checked') || 0;
if (val) {
$(trigger).parent().removeClass('hidden-options').addClass('expanded-options');
}
else {
$(trigger).parent().removeClass('expanded-options').addClass('hidden-options');
}
break;
default:
var val = jQuery(trigger).val();
}
}
return val;
}
var setChangeTrigger = function(trigger_id, bind_id) {
// Triggered when change() is clicked.
var changeTrigger = function() {
var val = getValue(bind_id, trigger_id);
for (i in Drupal.Views.dependent.bindings[bind_id]) {
var id = Drupal.Views.dependent.bindings[bind_id][i];
// Fix numerous errors
if (typeof id != 'string') {
continue;
}
// This bit had to be rewritten a bit because two properties on the
// same set caused the counter to go up and up and up.
if (!Drupal.Views.dependent.activeBindings[id]) {
Drupal.Views.dependent.activeBindings[id] = {};
}
if (Drupal.Views.dependent.inArray(Drupal.settings.viewsAjax.formRelationships[id].values[bind_id], val)) {
Drupal.Views.dependent.activeBindings[id][bind_id] = 'bind';
}
else {
delete Drupal.Views.dependent.activeBindings[id][bind_id];
}
var len = 0;
for (i in Drupal.Views.dependent.activeBindings[id]) {
len++;
}
var object = jQuery('#' + id + '-wrapper');
if (!object.size()) {
object = jQuery('#' + id).parent();
}
var rel_num = Drupal.settings.viewsAjax.formRelationships[id].num;
if (typeof rel_num === 'object') {
rel_num = Drupal.settings.viewsAjax.formRelationships[id].num[0];
}
if (rel_num <= len) {
// Show if the element if criteria is matched
object.show(0);
object.addClass('dependent-options');
}
else {
// Otherwise hide
object.hide(0);
}
}
}
jQuery(trigger_id).change(function() {
// Trigger the internal change function
// the attr('id') is used because closures are more confusing
changeTrigger(trigger_id, bind_id);
});
// Trigger initial reaction
changeTrigger(trigger_id, bind_id);
}
setChangeTrigger(trigger_id, bind_id);
}
}
}
Drupal.behaviors.viewsDependent = function (context) {
Drupal.Views.dependent.autoAttach();
// Really large sets of fields are too slow with the above method, so this
// is a sort of hacked one that's faster but much less flexible.
$("select.views-master-dependent:not(.views-processed)")
.addClass('views-processed')
.change(function() {
var val = $(this).val();
if (val == 'all') {
$('.views-dependent-all').show(0);
}
else {
$('.views-dependent-all').hide(0);
$('.views-dependent-' + val).show(0);
}
})
.trigger('change');
}