Drupalで開発したWebサイトがあります。 collapsiblock(基本的にはJQueryプラグイン)と呼ばれるモジュールを使用して、アコーディオンのような効果を実現します。 (ベータ版ではありますが)うまく機能しています。ただし、ユーザーがアコーディオンの1つのアイテムをクリックすると、他のアイテムが折りたたまれるように変更します。
現在の統計では、ユーザーが1つのアイテムをクリックすると、そのアイテムが既に折りたたまれているか展開されているかを確認し、そのアイテムを反対にするように機能しています。つまり、ユーザーが1つのアイテムをクリックすると展開され、別のアイテムをクリックすると展開されますが、以前にクリックしたアイテムは折りたたまれません。
以下のコードを見ることができます。折りたたみにコードを追加する場所と、折りたたみと展開の方法を知っています。私の質問は、ユーザーがクリックしたものを除く、クラス「.collapsiblock」を持つすべてのアイテムを選択するにはどうすればよいですか?
注:クラス「.collapsiblockCollapsed」を持つアイテムは折りたたまれ、このクラスがアイテムから削除されると展開されます。
// $Id: collapsiblock.js,v 1.6 2010/08/18 19:17:37 gagarine Exp $
Drupal.Collapsiblock = Drupal.Collapsiblock || {};
Drupal.behaviors.collapsiblock = function (context) {
var cookieData = Drupal.Collapsiblock.getCookieData();
var slidetype = Drupal.settings.collapsiblock.slide_type;
var defaultState = Drupal.settings.collapsiblock.default_state;
var slidespeed = parseInt(Drupal.settings.collapsiblock.slide_speed);
$('div.block:not(.collapsiblock-processed)', context).addClass('collapsiblock-processed').each(function () {
var id = this.id;
var titleElt = $(':header:first', this).not($('.content :header',this));
if (titleElt.size()) {
titleElt = titleElt[0];
// Status values: 1 = not collapsible, 2 = collapsible and expanded, 3 = collapsible and collapsed, 4 = always collapsed
var stat = Drupal.settings.collapsiblock.blocks[this.id] ? Drupal.settings.collapsiblock.blocks[this.id] : defaultState;
if (stat == 1) {
return;
}
titleElt.target = $(this).find('div.content');
$(titleElt)
.addClass('collapsiblock')
.click(function () {
var st = Drupal.Collapsiblock.getCookieData();
if ($(this).is('.collapsiblockCollapsed')) {
$(this).removeClass('collapsiblockCollapsed');
if (slidetype == 1) {
$(this.target).slideDown(slidespeed);
}
else {
$(this.target).animate({height:'show', opacity:'show'}, slidespeed);
}
// Don't save cookie data if the block is always collapsed.
if (stat != 4) {
st[id] = 1;
}
}
else {
$(this).addClass('collapsiblockCollapsed');
if (slidetype == 1) {
$(this.target).slideUp(slidespeed);
}
else {
$(this.target).animate({height:'hide', opacity:'hide'}, slidespeed);
}
// Don't save cookie data if the block is always collapsed.
if (stat != 4) {
st[id] = 0;
}
}
// Stringify the object in JSON format for saving in the cookie.
var cookieString = '{ ';
var cookieParts = [];
$.each(st, function (id, setting) {
cookieParts[cookieParts.length] = ' "' + id + '": ' + setting;
});
cookieString += cookieParts.join(', ') + ' }';
$.cookie('collapsiblock', cookieString, {path: Drupal.settings.basePath});
});
// Leave active blocks uncollapsed. If the block is expanded, do nothing.
if (stat == 4 || (cookieData[id] == 0 || (stat == 3 && cookieData[id] == undefined)) && !$(this).find('a.active').size()) {
$(titleElt).addClass('collapsiblockCollapsed');
$(titleElt.target).hide();
}
}
});
};
Drupal.Collapsiblock.getCookieData = function () {
var cookieString = $.cookie('collapsiblock');
return cookieString ? Drupal.parseJson(cookieString) : {};
};
次のコードを追加することで問題が解決しました。
$('.collapsiblock').not(this).each(function(){
$(this).addClass('collapsiblockCollapsed');
$(this.target).animate({height:'hide', opacity:'hide'}, slidespeed);
});
次の行のすぐ上:
$(this).removeClass('collapsiblockCollapsed');
not
セレクターを使用します。
例:
$('.collapsiblock').click(function(){
$('.collapsiblock').not(this).each(function(){
$(this).slideUp();
});
$(this).slideDown();
})
これを試して、
例:
$('.collapsiblock').click(function(){
$('.collapsiblock').not(this).slideUp();
$(this).slideDown();
});
これは、各関数を使用してロードすると将来的には数千のdivがあり、SlideUpまたはSlideDownに時間がかかるため、より良い方法です。
独自のjqueryクリックハンドラとjqueryのdata(...)関数を使用して、どの要素が既にクリックされたかを追跡できます。次に、jqueryのフィルター(...)関数を使用して.collapsiblockアイテムを繰り返しフィルタリングし、必要なアイテムを含めます。