Грамотное javascript-дерево за 7 шагов
В этой статье описана DOM/CSS-структура дерева, которую я в свое время разработал для dojo toolkit.
Основные особенности:
- Семантическая удобная CSS-разметка.
Внешний вид дерева определяется исключительно CSS.
- Скрытие/раскрытие узлов
- Структура дерева обозначена линиями
- Допускает многострочное HTML-содержимое в узлах
- Оптимизация по количеству HTML-тагов
- Легко дополняется новыми фишками
Например:
Root
-
Item 1
-
Item 2 title long yeah
-
Item 3
Основной строительный блок дерева - его узел.
Каждый узел имеет класс Node и состоит из иконки Expand , заголовка Content и контейнера для детей Container .
Визуальное представление узла:
Например, вот так выглядит разметка просто корневого узла Root , без детей:
<ul class="Container">
<li class="Node IsRoot ExpandOpen">
<div class="Expand"></div>
<div class="Content">Root</div>
</li>
</ul>
- Класс
IsRoot
- говорит о том, что узел является корнем дерева
- Класс
ExpandOpen
- Обозначает, что узел раскрыт
Обратите внимание - вся используемая разметка является исключительно семантической. В данном случае CSS-класс говорит не "каким образом следует выделить элемент", а "что элемент обозначает".
А вот - внутри узла появился Container с (пока) одним потомком Item 1 .
<ul class="Container">
<li class="Node IsRoot ExpandOpen">
<div class="Expand"></div>
<div class="Content">Root</div>
<ul class="Container">
<li class="Node ExpandLeaf">
<div class="Expand"></div>
<div class="Content">Item 1</div>
</li>
</ul>
</li>
</ul>
Семантические элементы:
- Контейнер
Container
- В контейнере содержатся все дети, т.е 1 или больше
Node . Это удобно, ведь чтобы скрыть/показать потомков - достаточно обратиться к контейнеру.
Перебрать всех детей можно, используя Container.childNodes .
- Класс
ExpandLeaf
- Обозначает, что узел является листом дерева.
Узел-потомок уже не имеет класса IsRoot .
Для начала - немного почистим стили для UL и LI : обнулим по умолчанию заданные значения margin, padding и list-style-type .
/* контейнер просто содержит узлы.
Узел сам будет отвечать за свой отступ */
.Container {
padding: 0;
margin: 0;
}
.Container li {
list-style-type: none; /* убрать кружочки/точечки */
}
Самым базовым является стиль для собственно узла Node .
Он задает иерархическую структуру за счет свойства margin-left , которое отодвигает узел-потомок от левой стенки контейнера.
/* узел отодвинут от левой стенки контейнера на 18px
благодаря этим отступам вложенные узлы формируют иерархию
*/
.Node {
margin-left: 18px;
zoom: 1; /* спецсвойство против багов IE6,7. Ставит hasLayout */
}
/* Корневой узел от родительского контейнера не отодвинут.
Ему же не надо демонстрировать отступом, чей он сын.
Это правило идет после .Node, поэтому имеет более высокий приоритет
Так что class="Node IsRoot" дает margin-left:0
*/
.IsRoot {
margin-left: 0;
}
Для того, чтобы иконка Expand находилась слева от содержания - использован принцип двухколоночной верстки.
Левая колонка с фиксированной шириной - Expand , правая колонка - Content .
/* иконка скрытого/раскрытого поддерева или листа
сами иконки идут дальше, здесь общие свойства
*/
.Expand {
width: 18px;
height: 18px;
/* принцип двухколоночной верстки. */
/* float:left и width дива Expand + margin-left дива Content */
float: left;
}
/* содержание (заголовок) узла */
.Content {
/* чтобы не налезать на Expand */
margin-left:18px;
/* высота заголовка - как минимум равна Expand
Т.е правая колонка всегда выше или равна левой.
Иначе нижний float будет пытаться разместиться на получившейся ступеньке
*/
min-height: 18px;
}
/* все правила после * html выполняет только IE6 */
* html .Content {
height: 18px; /* аналог min-height для IE6 */
}
Получившаяся структура допускает любые данные внутри Content , включая многострочные и т.п.
/* открытое поддерево */
.ExpandOpen .Expand {
background-image: url(/forum/img/minus.gif);
}
/* закрытое поддерево */
.ExpandClosed .Expand {
background-image: url(/forum/img/plus.gif);
}
/* лист */
.ExpandLeaf .Expand {
background-image: url(/forum/img/leaf.gif);
}
Здесь очень важен порядок, в котором следуют определения.
Поддеревья вложены, из-за этого получается такая конструкция:
<li class="...Node ExpandOpen...">
...
<li class="...Node ExpandClosed...">
<div class="Expand"></div>
..
</li>
</li>
Внутренний див Expand подходит под оба CSS-правила: и под ExpandOpen .Expand и под .ExpandClosed .Expand .
Правило .ExpandClosed .Expand идет позже, поэтому имеет более высокий приоритет, и будет (правильно) показана иконка закрытого раздела.
Структурные линии обрисовывают дерево, делая иерархию более наглядной.
В некоторых javascript-деревьях они пунктирные и используют кучу лишних тагов из-за неудачно выбранной DOM/CSS-модели.
Метод построения линий, который будем использовать мы, позволяет сделать линии гладкие, растягивающиеся при изменении размера деревьев.
Впрочем, пунктир добавить тоже никто не помешает.
И все это без добавления дополнительных тагов, исключительно средствами CSS.
Наша цель - получить дерево, которое выглядит так:
Info
-
Root
-
Item 1 Multiline test
-
Item 2
<div>Info</div>
<ul class="Container">
<li class="Node IsRoot IsLast ExpandOpen">
<div class="Expand"></div>
<div class="Content">Root</div>
<ul class="Container">
<li class="Node ExpandOpen">
<div class="Expand"></div>
<div class="Content">Item 1<br/>Multiline test</div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<div class="Content">Item 1.1</div>
</li>
</ul>
</li>
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<div class="Content">Item 2</div>
</li>
</ul>
</li>
</ul>
Каркас из линий образуется дополнительными CSS-правилами.
- Узел
Node поддерживает вертикальную линию к своему следующему соседу
.Node {
margin-left: 18px;
zoom: 1;
/* линия слева образуется повторяющимся фоновым рисунком */
background-image : url(/forum/img/i.gif);
background-position : top left;
background-repeat : repeat-y;
}
- Если соседа ниже нет, то линию вниз продолжать не надо:
/* это правило - ниже .Node, поэтому имеет больший приоритет */
.IsLast {
/* добавить соединительную черточку наверх */
background-image: url(/forum/img/i_half.gif);
background-repeat : no-repeat;
}
Получается, что все узлы на одном уровне соединены вертикальной чертой.
Размер рисунков для фоновых черточек сделан такой, чтобы вертикальная черта проходила строго посередине иконок Expand .
Поэтому получается, что эти иконки автоматически "нанизываются" на вертикальную линию.
Чтобы получить более целостную картину, можно обновить иконки Expand , добавив к ним соединительную черту для подключения заголовка к вертикальной линии.
Вот такие новые иконки для Expand* -классов.
Закрытый узел ExpandClosed
Горизонтальные коннекторы готовы.
Вертикальные линии образуют каркас, а новые иконки Expand* присоединяют узлы к каркасу. Структурные линии построены .
Для скрытия-раскрытия добавим два CSS-правила.
.ExpandOpen .Container {
display: block;
}
.ExpandClosed .Container {
display: none;
}
Как всегда, важен порядок. ExpandClosed идет после ExpandOpen , поэтому имеет больший приоритет, и вложенные закрытые узлы отображаются закрытыми.
Для скрытия-раскрытия javascript-функция всего лишь меняет класс узла. Остальное делает CSS.
Чтобы в дереве поддерживалось скрытие-раскрытие - достаточно повесить обработчик на самый внешний div .
И для красоты - обязательно поправить курсор при наведении на иконки скрытия/раскрытия:
.ExpandOpen .Expand, .ExpandClosed .Expand {
cursor: pointer; /* иконки скрытия-раскрытия */
}
.ExpandLeaf .Expand {
cursor: auto; /* листовой узел */
}
Обязательно задать определение для листового узла тоже, иначе курсор на нем тоже станет pointer (почему? - из-за вложенности div 'ов).
Root
-
Item 1
-
Item 2 title long yeah
-
Item 3
<div onclick="tree_toggle(arguments[0])">
<div>Root</div>
<ul class="Container">
<li class="Node IsRoot ExpandClosed">
<div class="Expand"></div>
<div class="Content">Item 1</div>
<ul class="Container">
<li class="Node ExpandClosed">
<div class="Expand"></div>
<div class="Content">Item 1.1</div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<div class="Content">Item 1.1.2</div>
</li>
</ul>
</li>
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<div class="Content">Item 1.2</div>
</li>
</ul>
</li>
<li class="Node IsRoot ExpandClosed">
<div class="Expand"></div>
<div class="Content">Item 2<br/>title long yeah</div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<div class="Content">Item 2.1</div>
</li>
</ul>
</li>
<li class="Node ExpandOpen IsRoot IsLast">
<div class="Expand"></div>
<div class="Content">Item 3</div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<div class="Content">Item 3.1</div>
</li>
</ul>
</li>
</ul>
</div>
А вот и сам обработчик события onclick . После правил CSS делать ему осталось всего ничего:
- Определить, произошел ли клик на иконке
Expand , используя event.target(или event.srcElement для IE)
- Получить узел
Node для иконки
- Если узел - не лист, то поменять класс
ExpandOpen <-> ExpandClosed
function tree_toggle(event) {
event = event || window.event
var clickedElem = event.target || event.srcElement
if (!hasClass(clickedElem, 'Expand')) {
return // клик не там
}
// Node, на который кликнули
var node = clickedElem.parentNode
if (hasClass(node, 'ExpandLeaf')) {
return // клик на листе
}
// определить новый класс для узла
var newClass = hasClass(node, 'ExpandOpen') ? 'ExpandClosed' : 'ExpandOpen'
// заменить текущий класс на newClass
// регексп находит отдельно стоящий open|close и меняет на newClass
var re = /(^|\s)(ExpandOpen|ExpandClosed)(\s|$)/
node.className = node.className.replace(re, '$1'+newClass+'$3')
}
function hasClass(elem, className) {
return new RegExp("(^|\\s)"+className+"(\\s|$)").test(elem.className)
}
Пока что мы строили дерево исключительно из HTML-разметки.
Полностью аналогично дерево работает при создании разметки при помощи Javascript. Как загружать данные с сервера в формате JSON, и многое другое Вы можете прочитать в цикле статей AJAX.
Здесь мы посмотрим, как добавить в дерево индикаторы обработки узла: .
Индикатор обработки, вообще говоря, может обозначать любые асинхронные операции. Начиная от загрузки детей и заканчивая удалением всего этого узла с сервера.
Опишем его CSS-правилом:
.ExpandLoading {
width: 18px;
height: 18px;
float: left;
background-image: url(/forum/img/expand_loading.gif);
}
Класс ExpandLoading на время операции будет заменять обычный класс Expand .
Почему нельзя добавить класс ExpandLoading к ExpandOpen/Closed/.. ?
Индикатор может понадобиться в любом месте. Среди потомков "активного" узла могут быть "неактивные" узлы, и среди его родителей - тоже.
Если поставить класс ExpandLoading в один ряд с ExpandOpen/Closed/.. , то он будет либо более приоритетен - и тогда все узлы под ним получат часики, либо менее приоритетен - тогда вообще ничего не будет видно.
И тот и другой варианты - не подходят, когда индикация нужна на одном-единственном узле посередине, например, после редактирования названия узла.
Например, так может выглядеть участок дерева с активным узлом Item 1.1 :
<ul class="Container">
<li class="Node IsRoot IsLast ExpandOpen">
<div class="Expand"></div>
<div class="Content">Item 1</div>
<ul class="Container">
<li class="Node ExpandOpen IsLast">
<div class="ExpandLoading"></div>
<div class="Content">Item 1.1</div>
<ul class="Container">
<li class="Node ExpandOpen">
<div class="Expand"></div>
<div class="Content">Item 1.1.1</div>
</li>
</ul>
</li>
</ul>
</li>
</ul>
Вы можете пожелать добавить в дерево дополнительные элементы. Например, чекбоксы или иконки с типом узла.
Для добавления, например, чекбокса <input type="checkbox"> после иконки Expand , нужно для начала вставить его в структуру сразу после иконки открытия/закрытия.
Указываем размеры, отступ и float: left :
/* Общий размер 14+2+2 = 18 - такой же как Expand */
.Node input {
width: 14px;
height: 14px;
float: left;
margin: 2px;
}
Теперь, сохраняя двухколоночную верстку, нужно отодвинуть Content вправо уже не на 18, а на общую ширину двух float 'ов - 36px.
После того как сдвинулся заголовок Content - естественно сдвинуть и сам узел Node , чтобы структурная линия шла от заголовка.
Все это осуществляется добавлением пары правил:
/* подвинем за оба float'а Node, Content */
.Node, .Content {
margin-left: 36px;
}
/* заново переопределим .IsRoot */
.IsRoot { margin-left: 0; }
<div onclick="tree_toggle(arguments[0])">
<div>Root</div>
<ul class="Container">
<li class="Node IsRoot ExpandOpen">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 1</div>
<ul class="Container">
<li class="Node ExpandOpen">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 1.1 </div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 1.1.2</div>
</li>
</ul>
</li>
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 1.2</div>
</li>
</ul>
</li>
<li class="Node IsRoot ExpandOpen">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 2<br/>title long yeah</div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 2.1</div>
</li>
</ul>
</li>
<li class="Node ExpandOpen IsRoot IsLast">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 3</div>
<ul class="Container">
<li class="Node ExpandLeaf IsLast">
<div class="Expand"></div>
<input type="checkbox"/>
<div class="Content">Item 3.1</div>
</li>
</ul>
</li>
</ul>
</div>
В принципе, можно использовать и CSS/JS/картинки напрямую со страницы, но они содержат некоторое количество лишних классов.
Для удобства дерево все-в-одном с JS/CSS/HTML находится на отдельной странице. В этом примере полностью расписано дерево без AJAX-индикации и чекбоксов.
Кроме того, можно скачать материалы по статье:
При использовании в своем окружении Вы, наверное, захотите удлинить все классы. добавив какой-то префикс. Например, TreeContainer , TreeNode , и т.п.
Другой вариант, возможно, более удобный - ограничить классы внешним селектором. Например, .Tree .Container , .Tree .Node , * html .Tree .Content и т.п.
|
Замечательная и полезная статья!
неплохо.
я бы предложил использовать классы с единым префиксом (нэймспэйсом) при создании подобных абстрактных реализаций, дабы избежать случайного конфликта с другими не менее абстрактными реализациями ^_^
например, для элементов ( e - element ):
tree-e-root
tree-e-branch
tree-e-leaf
для состояний ( s - state ):
tree-s-opened
tree-s-closed
tree-s-loading
.ня
Update: Добавил downloads и замечание о префиксах/неймспейсах в конце статьи.
Небольшие редакторские правки для лучшего раскрытия некоторых моментов.
А вот такой вопрос. По умолчанию дерево полностью раскрыто, как сделать его закрытым?
Или чтобы запоминалось, ну это наверное в куки надо закидывать...
Посмотри последний пример или скачай исходники... Там дерево полностью закрыто должно быть.
Да Твоё дерево не очень, так как сразу глотает всю структуру, а если у тебя будет 5 тыщ узлов ?
Да, в статье не разобран вопрос динамической подгрузки данных.
С другой стороны, её достаточно просто реализовать самому.
Структура дерева удобна для добавления и удаления узлов, а как получить AJAX'ом с сервера данные - это уже можно, например, раздел про AJAX посмотреть.
UPDATE: В разделе по AJAX появились статьи про интеграцию AJAX в интерфейс и статья про AJAX-дерево.
дерево и в правду сразу открывается((
Конечно, дерево в примере раскрыто. Там ведь у каждого узла класс ExpandOpen стоит. Если Вам хочется закрытое дерево - замените его на ExpandClosed.
В этом примере стоит ExpandClosed, так что дерево закрыто.
Отличная статья! Долго искал нечто подобное. Респект автору за грамотный подход к задаче!
Большинство подобных деревьев обычно делается через Хм... пень колоду. В данном случае, все четко соответствует спецификациям и обеспечивает широкую кроссбраузерность.
Несомненный плюс данного дерева в том, что оно не генерируется скриптом, а полностью выполнено в виде HTML кода. Что в данном случае сохраняет саму логическую разметку документа, плюс позволяет свободно индексировать содержимое поисковиками.
Ну и несомненную ценность имеет не только конечный результат, но и сама статья.
Спасибо огромное автору! Добавил в избранное!
Если расширить условие функции раскурывания/закрывания таким образом, поведение дерево станет более юзабельным: для раскрытия узла не нужно целиться в крестик, а можно нажать на его имя.
if (!hasClass(clickedElem,'Expand') && !hasClass(clickedElem,'Content')) {
return // клик не там
}
В статье сделан именно крестик, т.к на клик на имени часто вешается что-то другое, например, открытие страницы с этим именем.
Хотя, конечно, в вашем случае может быть целесообразно расширить дерево именно так
Что-то у меня такой вариант не работает, а очень нужно, не могу понять почему?
В Drupal интегрируется как пить дать, причем javascript просто дописывается к уже имеющемуся одному из файлов скриптов, а css соответственно к имеющейся таблице стилей, которые используются на этой странице и все. Нюанс: все классы, используемые деревом, желательно переименовать или хотя бы единичку дописать в имени каждого класса во избежание случайного пересечения описания классов, к примеру, класс node уже используется системой drupal, а вот node1 - нет
Спасибо!
В статье http://javascript.ru/ajax/tutorial/intro приведён пример ajax-бесконечного дерева. Его просто создать, когда подгрузка узлов осуществляется без анимации.
Хотелось бы знать, как оптимально сделать, чтобы список дочерних узлов "выезжал" вниз или вверх (при сворачивании)?
Это делается через некую глобальную переменную типа width и через setTimeout вызывается функция, которая её меняет для данного тага?
В чем именно проблема? Это вопрос про AJAX или про анимацию или про то как они связаны ?
Именно про анимацию. Чем достигается появление результата ответа не сразу, а через "выкатывание" вниз?
У меня было предположение по поводу неё такое:
Получают результат запроса в div, далее дают ему более меньший zIndex и относительное позиционирование, смещают top на -offsetHeight и привязывают к DOM, а потом в вызывают функцию рекурсивно, которая увеличивает top до 0 и возвращают прежний zIndex.
Реализовывать это не пробовал (вижу в этом некую нерациональность)
Там изменение height + setTimeout.
То есть, появляется div, у него размер 0.
Дальше, скажем, каждые 5 ms размер увеличивается на 1px.
И так - до полного выкатывания, т.е пока height < div.scrollHeight
Если менять высоту div'а итерациями от 0 до scrollHeight через глобальную функцию, то на достаточно большом кол-ве содержащихся в этом div'е дочерних элементов время загрузки очень сильно возрастает (независимо от пересчёта интервала времени вызова на бОльшую высоту). Такого не наблюдается, если создать класс и нужное кол-во раз вызывать метод.
Но эффектнее это выглядит, если сделать анимацию высоты от 0 до 5-10px, а от 5-10px сразу до scrollHeigh. В обратном порядке реализовать скрытие.
хорошо бы добавить функционал сохранения состояния дерева при обновлении страницы
Проблем-то. Добавляем функции serialize/unserialize и прикручиваем метод хранения данных по желанию: cookie / DOM storage / flash LSO ...
Отличная статья!
Код - предельно прост, ясен, без хаков, а главное - КОМПАКТЕН и РАБОЧИЙ!
Вот бы ещё додумать как сохранять состояние узлов в куки, а потом разворачивать.
Есть некий пример с DTHMLGoodies.com (называется folder-tree-static), там на JS написаны мудреные функции как раз для таких целей.... Однако моя башка никак не допрет как его можно применить здесь :-(
Кстати насчёт индексации поисковиками - не думаю, что в областях "display: none;" они что-то будут индексировать. Всё-таки поисковики умнеют и защищаются от спама. ИМХО, мнение что гугль\яндекс не разбирает css и не парсит эти display и none миф...
Хотел сделать так:
if (!hasClass(clickedElem,'Expand') && !hasClass(clickedElem,'Content')) {
return // клик не там
}
но что0-то не работает, нужно нажимать именно на крестих, а нужно ещё и чтобы при нажатии на имя нода раскрывался контейнер
Кто-то хотел запоминать состояние дерева. Есть вариант для простоты запоминать в куках последний активный узел, если считать, что активный узел - это раскрытый узел. При обновлении страницы JS скрипт после загрузки берет значение из кук и активирует (раскрывает) требуемый узел. Соответственно надо раскрыть и всех его предков перебором узлов родителей с классом, где нет "Leaf" в пределах дерева. Это просто сделать циклом, используя свойство .parentNode и пару проверок. Если предок не является "листом", то поставить ему стиль ExpandOpen. На мой взгляд, запоминать состояние всего дерева не практично, да и зачем это может быть нужно. Чтобы задача имела нормальное решение надо вводить разумные ограничения - иначе заколебешься.
а можете написать как это сделать?
Кстати, есть способ избежать использования класса IsRoot:
.Tree .Node {
background-image : url(img/i.gif);
background-position : top left;
background-repeat : repeat-y;
margin-left: 0;
zoom: 1;
}
.Tree .Node .Node {
margin-left: 18px;
}
Очень понравился этот скрипт, потому незначительное улучшение от меня + класс на PHP для быстрого создания подобного списка.
Файл JsTree.php
Файл index.php
Файл Tree.css
Файл Tree.js
И картинки в папке imgs
как строить переменную $tree_array?
если надо - заменяем нулл на еще один вложенный массив и так до бесконечности.
Как-то всё очень сложно... я сделал на рекурсивной функции:
index.php
а ещё я извратился и сделал это дерево drag&drop-изменяемым))) писецнна! Но работает)))
dnd_03.js
просьба к тем, кто силём в яваскрипте, посоветовать чего и как тут можно оптимизировать))
nogrid.css
Огромное спасибо!!! Долго искала образец дерева, это самое удобное!
В таком дереве есть недостаток. Нельзя добавить колонку(и) кнопок, которые были бы выровнены по правому краю. пришлось дерево сделать в таблице.
Да, и еще оно на баяне не играет.
Взаимно, +1 ))
Впрочем, на самом деле добавить колонку таких кнопок можно, CSS-структура это замечательно позволяет.
Сейчас узел построен по принципу двухколоночной верстки: [fixed width] + [left margin]. Добавить правую колонку - это всего лишь перейти на трехколоночную верстку дивами: [fixed width] + [left/right margin] + [fixed width].
Стоит заменить zoom: 1 на height: 1% - чтобы пройти валидацию.
В разделе по AJAX появились статьи про интеграцию AJAX в интерфейс и статья про AJAX-дерево.
Как раз делал дерево, только для jQuery UI, вот это, мой подход оказался очень похож на ваш, это клево. Значит я на верном пути.
Очень полезная статья, как в качестве примера работы с CSS, так и в качестве действительно полезного практического материала. Автору огромный респект. Аффтар, пеши исчо!!!
Здравствуйте! огромное спасибо за статью от всех новичков!
а не будет ли кто-нибудь так добр подсказать, как добавить к дереву кнопки "свернуть / развернуть все"?
заранее спасибо :-)
СПАСИБО!
Присоеденяюсь к вопросу Ira:
"Здравствуйте! огромное спасибо за статью от всех новичков!
а не будет ли кто-нибудь так добр подсказать, как добавить к дереву кнопки "свернуть / развернуть все"?
заранее спасибо :-)"
Спасибо за удобное решение!
В IE8 отображается криво...
Нормально отображается, вообще все ок
Отвечаю на вопрос по "открыть все - свернуть все".
1) Без AJAX все достаточно просто. Функция рекурсивно пробегает по узлам дерева от корня к детям и все разворачивает/сворачивает. Придется использовать рекурсию или стек.
2) С AJAX отсутствующие узлы для операции "развернуть все" придется подгрузить. Но практика показывает, что функционал "развернуть все" к AJAX-узлам применяется редко.
Если вам действительно нужен этот код - я его писал и могу портировать в статью.
А так можно развернуть все загруженные узлы, используя простую рекурсию из п. 1.
Отлично все описано, есть пара идей по оптимизации
function hasClass(elem, className) {
return new RegExp("(^|\\s)"+className+"(\\s|$)").test(elem.className)
}
превращаем в
function hasClass(elem, className) {
return elem.className.search("\\b"+className+"\\b")+1
}
а вот это -
// определить новый класс для узла
var newClass = hasClass(node, 'ExpandOpen') ? 'ExpandClosed' : 'ExpandOpen'
// заменить текущий класс на newClass
// регексп находит отдельно стоящий open|close и меняет на newClass
var re = /(^|\s)(ExpandOpen|ExpandClosed)(\s|$)/
node.className = node.className.replace(re, '$1'+newClass+'$3')
превращем в
node.className = node.className.replace(/(\bExpand)([\w]+\b)/,
function (m,s1,s2) {return s1+(s2!='Open'?'Open':'Closed')});
Вполне выразительно и короче, разве что не так наглядно.
Последний листинг читать как
node.className = node.className.replace(/(\bExpand)(\w+\b)/,
function (m,s1,s2) {return s1+(s2!='Open'?'Open':'Closed')});
убрал обертку символьного класса за ее ненужностью.
Подскажите пожалуйста.
Если много элементов дерева - и раскрыть их - как сделать чтобы при закрытии одного из них страница не обновлялась?
Хотелось выразить автору большую благодарность за статью. Даже я, совсем не зная JS, умудрился реализовать то, что мне надо. А точнее меню, в котором подменю появляется и исчезает по щелчку. Только оч. нужна ещё помощь: как сделать, что бы, скажем, открытый node1 первого уровня, становился закрытым при щелчке на node2 первого уровня?
В класс .Node необходимо добавить свойство:
В противном случае при изменении масштаба страницы в firefox 3.5 и IE8 весь список уедет вправо из-за свойства float:left, так как они думают что все следующие списки должны обтекать предидущий!
для hasClass вместо
лучше использовать "\b" для отделения слова от других. Делает то же, но выглядит чище. Плюс className надо экранировать (ну конечно если захочется использовать эту функцию чаще)
Спасибо большое!!! давно искала
Еще одна реализация этого же дерева. CSS не менял, реализацию функций hasClass и tree_toggle тоже.
Использование - скормить функции tree два параметра: ассоциативный массив с деревом и заголовок. Массив может выглядеть, к примеру, так:
В каждой конечной паре ключ - это то, что будет выводиться, а значение - это ссылка, на которую поведет эта строка.
Минималистический вариант
Функциональность таже что и у оригинала.
Удалены не нужные классы и правила css .Container, .Node, .Content, .Expand, .ExpandLeaf
.IsRoot тоже не нужен т.к. маргин заменен на паддинг, т.е. теперь, при построении отступов, не сам элемент дерева отодвигается в право относительно своего родителя, а отодвигает своих детей
.ExpandOpen, .ExpandClosed и .IsLast переименованы в .Expanded, .Contracted и .Last соответственно.
Также сокращен html. Удалены ненужные div-ы. Теперь выглядит так:
Ну и js тоже не избежал кастрации
Единственный минус по сравнении с оригиналом, курсор всегда pointer. Хотя для меня это плюс, т.к. листья у меня - ссылки, т.е. тоже кликабельные.
Вот только не надо про функционал заливать. Он конкретно урезан. Автор молодец предусмотрел все варианты. Мне например надо кнопочки на каждый нод и еще подсветка бзкграундом, что в вашем случае недостижимо.
Я с JS на вы, кто-то может выложить готовый код для Toggle All Nodes. Должно быть просто изменение Close <=> Open с перебором всех узлов. Респект автору!
PS Если дерево без линий, то и .Last не нужен.
попробовал твой сокращённый вариант, работает отлично... на IE, а вот на firefox дерево получается кривое, в чём может быть проблемма?
Да, действительно, FF делает отступ между маркером списка и текстом. Хотя, я бы не сказал, что дерево сильно кривое, просто лишний пробел перед текстом.
Но, тем не менее, пофиксить можно так:
Придется завернуть текст в спаны.
И в CSS добавить хак для FF.
Да, это помогло, спасибо!
Вы пишите, что (цитата):
Обратите внимание - вся используемая разметка является исключительно семантической. В данном случае CSS-класс говорит не "каким образом следует выделить элемент", а "что элемент обозначает".
На самом деле у Вас разметка говорит не только что элемент обозначает, но и хранит его состояние - ExpandOpen, что не есть хорошо, т.к. расплывается понятие "узел дерева" до "узел дерева в состоянии" * на количество состояний. А если добавить чекбоксы, селектор, ещё что нибудь, то будет бо-бо
Состояние узла и применяемый в этом состоянии стиль следует хранить отдельно. Я использую для этого statemap (хэш-таблица, где ключ статус, значение - доп. стиль) и функцию switch(statename). При небольшом расширении это позволяет менять не только стиль, но любые свойства элемента DOM при переходе между состояниями, а так же при необходимости отслеживать валидность переходов и сохранять состояние страницы, чтобы оно восстанавливалось при следующем заходе пользователя.
-- Ну начнем с того что пустой див ни как не может быть семантической разметкой - по определению.
-- Два - именование классов к семантике имеет слабое отношение, боюсь что даже никакое.
А у меня выполнено в виде плагина к jQuery, который хавает кошерный DOM-элемент из списков и строит древовидный javascript объект jsTree (или сразу из json).
Потом скрывает исходное дерево и вставляет на это место свое, построенное на основе jsTree. Там уже и события привязаны, и иконки нужные вставлены, и состояние узлов восстановлено(открытый/закрытый).
//Кстати, тут я решил не париться с колоночными div просто таблицами обошелся.
у такого подхода есть один большой плюс - можно легко менять способ отображения или даже cделать что-то вроде MVC
На стандарты пришлось забить в одном месте - у узлов в исходном html-дереве есть атрибут "c", где хранится конфиг узла. Выглядит примерно так:
<li c="{isOpen:true, icon:['image.png', 1, 0]}"> ...
плевать что валидатор ругается. Везде работает.
про хранение состояния..
Первое, что приходит в голову - cookie.
Я придумал вычислять хеш-сумму из строки заголовка узла, соединенной с заголовками узлов-предков. То есть для каждого узла строка должна быть разная и хеш будет уникальным(не всегда конечно). При сохранении, хеши открытых узлов кладутся блоками(по три символа для узла) в строку cookie. Это сделано, чтоб минимизировать длину cookie
хеш-функция - простая самодельная=) и намного быстрее какого-нибудь CRC. На входе - любая строка произвольной длины, на выходе - строка из трех символов, по сути 32-значное число от "000" до "vvv".
Статья превосходна в техническом плане. Но в плане практики у меня остались вопросы. Ведь мне например понадобится не только показать это дерево, но и произвести с ним какие-либо операции, например отметить какой то элемент дерева галкой и отправить данные из моей формы с учетом выбранной галки на сервер. Таким образом получаем новое требование: массив дерева должен содержать не только "Имя_элемента", но и "ID_элемента"!!! Без ID никак нельзя, сами понимаете, ведь в имени могут быть и русские буквы, к которым сервер может иметь неприязнь, так еще и имена из соседних веток дерева могут совпадать. Надеюсь доступно объяснил свою мысль. Вобщем нужно прикрутить массив, в котором не один ключ, а два и более: [id] и [name]
PS.:
интересует реализация такого дерева на php + javascript
Уважаемые, построил дерево, используя указанный код (мой опыт JS близок к нулю).
Задача: передать в скрытую форму значение, содержащееся в классе 'Content' для дальнейшей обработки.
Проблема: не передает IE 6.0 и Опера 9.61, но с Фоксом и Сафари работает.
f
Подскажите как правильно.
я в java плохо разбераюсь. Подскажите как сделать сворачивание открытой ветки при при открытии другой ветки, что бы у дерева постоянно была открыта только одна ветка.
Спасибо.
Парни подскажите плиз как сделать автоперенос текста на новую строку? Ну ограничить ширину меню. Использую все стандартное...
да, наверно так и есть
автору огромное спасибо за статью!!
от себя бы дополнил, что класс IsLast можно смело заменить на следующее и не засорять ваш js код ненужной ф-ностью :
.IsLastОбьясните пожалуйста эту строку:
function tree_toggle(event) {
event = event || window.event // Здесь ясно создаем обьек собитие
var clickedElem = event.target || event.srcElement // Здесь получаем имя инициатора события
if (!hasClass(clickedElem, 'Expand')) { Смотрим ниже(в энде кода) и что !значит перед вызовом функции(возврат фэлсе что ли?)
return // клик не там
}
// Node, на который кликнули
var node = clickedElem.parentNode
if (hasClass(node, 'ExpandLeaf')) {
return // клик на листе
}
// определить новый класс для узла
var newClass = hasClass(node, 'ExpandOpen') ? 'ExpandClosed' : 'ExpandOpen'
// заменить текущий класс на newClass
// регексп находит отдельно стоящий open|close и меняет на newClass
var re = /(^|\s)(ExpandOpen|ExpandClosed)(\s|$)/
node.className = node.className.replace(re, '$1'+newClass+'$3')
}
function hasClass(elem, className) {
return new RegExp("(^|\\s)"+className+"(\\s|$)").test(elem.className) А вот тут не соасем ясно что это?"(^|\\s)" "(\\s|$)" .test(elem.className) что значит слово test? я так понял тестирует на совпадение классов? почему бы не написать если класс инициатора = классу Expand то ... Зачем создавать функцию?
}
Спасибо, только в гугл не посылайте! А то раньше на х... ,а теперь в гугл!)
return new RegExp("(^|\\s)"+className+"(\\s|$)").test(elem.className)
нашел стать по этим регекспам я так понимаю что это вариант поиска текста в строке имени класса или названии класса? но так у нас есть class="Expand" ExpandOpen ExpandLeaf ExpandClosed . А понял щелчок должен быть по голой картинке . Ок
Есть еще строка
var re = /(^|\s)(ExpandOpen|ExpandClosed)(\s|$)/
node.className = node.className.replace(re, '$1'+newClass+'$3') что значит символ доллара с 1 и 3
var newClass = hasClass(node, 'ExpandOpen') ? 'ExpandClosed' : 'ExpandOpen' Это я так понял если стаит класс а то меняется на б и наоборот что то вроде toggle в jquery? Что значит ? и двоеточие?
function hasClass(elem, className) {
return new RegExp("(^|\\s)"+className+"(\\s|$)").test(elem.className) А вот тут не соасем ясно что это?"(^|\\s)" "(\\s|$)" .test(elem.className)
ОБЬЯСНИТЕ ПОЖАЛУЙСТА как и что значит в этой функции, плииз, нашел все в нете но в голове каша! я понимаю что меняется что т оместами но как задается условие немогу понять,СПАСИБО СПАСИБО
Вопрос к автору. Можно ли данный код использовать в коммерческом проекте? Есть ли лицензия?
Спасибо за исходник! Очень выручает!
вот пример кода на JQuery для отметки всех дочерних checkbox'ов
Когда писал свое дерево не додумался использовать классы для изменения картинок и закрытия и открытия LI, но как убирается продолжение бекгроунда первого (коренного LI) элемента, если последний из его детей сам имеет детей и раскрыт. Ведь коренной LI охватывает все элементы, и соответсвенно его бекгроунд тоже продолжается до конца. хотя должен закончится на последнем ребенке?
Люди добрые, а как сделать, чтобы сам корень Root не отображался?.. Т.е. дерево сразу показывало ветки, без корня?..
Сделаю закладочку. Очень толково написано.
Спасибо за хороший и простой и понятный скрипт
использовал версию с чекбоксами
не могли бы посоветовать как реализовать такю фунцию
нужно поставить чекбокс на верхние уровни папок с таким расчетом чтобы при его выбори выбирались все в ветке
Как сделать все элементы в сетке, пробовал в container style=border:1px solid black
обводятся элементы, а дерева в сетке не получается
В таком виде всё открывается в Mozilla Firefox, а в Internet Explorer 2й список не открывается. Подскажите пожалуйста как это исправить?
< div onClick="tree_toggle (arguments[0])">
< ul class="Container">
< li class="ExpandClosed">
< div class="Expand">
< div class="Content Text">Внешний тюнинг
< ul class="Container">
< li class="Node ExpandLeaf">
< div class="Expand">
< div class="Content">< a href="AirOb2107.html" title="Список обвесов">Аэродинамические обвесы< /a>
< div class="Expand">
< div class="ExpandLeaf">< a href="Schetki2107.html" title="Список щёток стеклоочистителя">Щётки стеклоочистителя< /a>
< /li>
< /ul>
< /li>
< /ul>
< div onClick="tree_toggle (arguments[1])">
< ul class="Container">
< li class="ExpandClosed">
< div class="Expand">
< div class="Content Text">Выхлопная система
< ul class="Container">
< li class="Node ExpandLeaf">
< div class="Expand">
< div class="Content">< a href="Kollektor2107.html" title="Список коллекторов">Коллекторый< /a>
< li class="Node ExpandLeaf IsLast">
< div class="Expand">
< div class="ExpandLeaf">< a href="Prochee.html" title="Список дополнительных деталей">Прочее< /a>
< /li>
< /ul>
< /li>
< /ul>
Добрый день!
Подскажите, как выделить цветом выделенный элемент, соответственно, сбросить цвет предыдущего выделенного, т.е. щелкаю Item1 - он подсвечивается.
Спасибо!
Огромное спасибо за статью, жаль что не могу ссылку повесить
Автору - в Firefox 15 идет баг с checkbox. Надо убрать из стилей width и height для input. Тогда все нормально.
Или надо выставить width и height по 13px;
Илья, скажите пожалуйста, что означает передача arguments[0] в функцию tree_toggle ?
Заранее спасибо
Наверное, сам понял:
передаётся только объект, по которому кликнули, игнорируя всплывание клика
Статья очень помогла бы, если бы вместе с кодом не копировались номера строк кода!!!
Там кнопки есть, нажи и откроется новое окно с кодом без номеров строк.
xml-овский минимализьм (-|+) пользую покамест, оный у разработчиков степенно взяв.) оказывается html + css + js + skin это xml.
Подскажите, под какой лицензией вы выпускаете этот код?
Подскажите, пожалуйста, как можно реализовать это дерево от дочерних ссылок к родительским. Спасибо
Да, это хорошо
Все сделал как было указано закачал на хостинг, по умолчанию при открытии страницы все списки раскрываются, хотя на примере не так. Подскажите что надо сделать чтобы при открытии страницы списки были свернуты при открытии страниц.
И все равно не понял (даже прочитав комментарии), как грузить подобный список из бд (ms sql) с помощью php =/
Большое спасибо за статью! А можете подсказать, как это дерево убрать в выпадающий список, т.е. чтобы изначально кроме "- Выберите значение-" ничего не было?
Спасибо!
Оптимизация кода:
.Node {
background: url(../../images/tree/i.gif) repeat-y top left;
вместо
background-image : url(../../images/tree/i.gif);
background-position : top left;
background-repeat : repeat-y;
Для checkbox-в родительских (хотим поставить/снять в детях):
в узле родительском добавляем
и функция
function setChildItems(event) {
var list = event.srcElement.parentElement.lastElementChild.children;
for (var i = 0; i < list.length; i++) {
list[i].children[1].checked = event.srcElement.checked;
}
}
спасибо чувак
Подскажите, пожалуйста, как приделать справа от имени узлов текстовые input-ты и прижать их к правому краю. Первый прижимается, а остальные выстраиваются в лесенку.