Грамотное 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; }
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 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 и т.п.
|
Автор: Dmitry A. Soshnikov, дата: 7 апреля, 2008 - 21:02
Замечательная и полезная статья!
Автор: tenshi, дата: 7 апреля, 2008 - 22:13
#permalinkнеплохо.
я бы предложил использовать классы с единым префиксом (нэймспэйсом) при создании подобных абстрактных реализаций, дабы избежать случайного конфликта с другими не менее абстрактными реализациями ^_^
например, для элементов ( e - element ):
tree-e-root
tree-e-branch
tree-e-leaf
для состояний ( s - state ):
tree-s-opened
tree-s-closed
tree-s-loading
.ня
Автор: Илья Кантор, дата: 8 апреля, 2008 - 01:39
#permalinkUpdate: Добавил downloads и замечание о префиксах/неймспейсах в конце статьи.
Небольшие редакторские правки для лучшего раскрытия некоторых моментов.
Автор: remitmaster, дата: 7 мая, 2008 - 21:33
#permalinkА вот такой вопрос. По умолчанию дерево полностью раскрыто, как сделать его закрытым?
Автор: remitmaster, дата: 7 мая, 2008 - 21:34
#permalinkИли чтобы запоминалось, ну это наверное в куки надо закидывать...
Автор: Илья Кантор, дата: 10 мая, 2008 - 08:45
#permalinkПосмотри последний пример или скачай исходники... Там дерево полностью закрыто должно быть.
Автор: Andrzej (не зарегистрирован), дата: 8 июля, 2008 - 15:38
#permalinkДа Твоё дерево не очень, так как сразу глотает всю структуру, а если у тебя будет 5 тыщ узлов
?
Автор: Гость (не зарегистрирован), дата: 23 июля, 2008 - 12:04
#permalinkдерево и в правду сразу открывается((
Автор: Илья Кантор, дата: 23 июля, 2008 - 23:47
#permalinkКонечно, дерево в примере раскрыто. Там ведь у каждого узла класс ExpandOpen стоит. Если Вам хочется закрытое дерево - замените его на ExpandClosed.
В этом примере стоит ExpandClosed, так что дерево закрыто.
Автор: Гость (не зарегистрирован), дата: 19 августа, 2008 - 23:51
#permalinkОтличная статья! Долго искал нечто подобное. Респект автору за грамотный подход к задаче!
Большинство подобных деревьев обычно делается через Хм... пень колоду. В данном случае, все четко соответствует спецификациям и обеспечивает широкую кроссбраузерность.
Несомненный плюс данного дерева в том, что оно не генерируется скриптом, а полностью выполнено в виде HTML кода. Что в данном случае сохраняет саму логическую разметку документа, плюс позволяет свободно индексировать содержимое поисковиками.
Ну и несомненную ценность имеет не только конечный результат, но и сама статья.
Спасибо огромное автору! Добавил в избранное!
Автор: Андрей Кумыков (не зарегистрирован), дата: 30 сентября, 2008 - 11:42
#permalinkЕсли расширить условие функции раскурывания/закрывания таким образом, поведение дерево станет более юзабельным: для раскрытия узла не нужно целиться в крестик, а можно нажать на его имя.
if (!hasClass(clickedElem,'Expand') && !hasClass(clickedElem,'Content')) {
return // клик не там
}
Автор: король ящериц (не зарегистрирован), дата: 24 октября, 2008 - 12:14
#permalinkВ Drupal интегрируется как пить дать, причем javascript просто дописывается к уже имеющемуся одному из файлов скриптов, а css соответственно к имеющейся таблице стилей, которые используются на этой странице и все. Нюанс: все классы, используемые деревом, желательно переименовать или хотя бы единичку дописать в имени каждого класса во избежание случайного пересечения описания классов, к примеру, класс node уже используется системой drupal, а вот node1 - нет
Автор: Вася (не зарегистрирован), дата: 26 октября, 2008 - 15:22
#permalinkСпасибо!
Автор: EugenyK, дата: 26 октября, 2008 - 22:13
#permalinkВ статье http://javascript.ru/ajax/tutorial/intro приведён пример ajax-бесконечного дерева. Его просто создать, когда подгрузка узлов осуществляется без анимации.
Хотелось бы знать, как оптимально сделать, чтобы список дочерних узлов "выезжал" вниз или вверх (при сворачивании)?
Это делается через некую глобальную переменную типа width и через setTimeout вызывается функция, которая её меняет для данного тага?
Автор: yarich (не зарегистрирован), дата: 27 октября, 2008 - 14:05
#permalinkхорошо бы добавить функционал сохранения состояния дерева при обновлении страницы
Автор: alex_css (не зарегистрирован), дата: 7 ноября, 2008 - 14:37
#permalinkОтличная статья!
Код - предельно прост, ясен, без хаков, а главное - КОМПАКТЕН и РАБОЧИЙ!
Вот бы ещё додумать как сохранять состояние узлов в куки, а потом разворачивать.
Есть некий пример с DTHMLGoodies.com (называется folder-tree-static), там на JS написаны мудреные функции как раз для таких целей.... Однако моя башка никак не допрет как его можно применить здесь :-(
Кстати насчёт индексации поисковиками - не думаю, что в областях "display: none;" они что-то будут индексировать. Всё-таки поисковики умнеют и защищаются от спама. ИМХО, мнение что гугль\яндекс не разбирает css и не парсит эти display и none миф...
Автор: Гость (не зарегистрирован), дата: 24 ноября, 2008 - 12:46
#permalinkХотел сделать так:
if (!hasClass(clickedElem,'Expand') && !hasClass(clickedElem,'Content')) {
return // клик не там
}
но что0-то не работает, нужно нажимать именно на крестих, а нужно ещё и чтобы при нажатии на имя нода раскрывался контейнер
Автор: Гость (не зарегистрирован), дата: 28 ноября, 2008 - 02:49
#permalinkКто-то хотел запоминать состояние дерева. Есть вариант для простоты запоминать в куках последний активный узел, если считать, что активный узел - это раскрытый узел. При обновлении страницы JS скрипт после загрузки берет значение из кук и активирует (раскрывает) требуемый узел. Соответственно надо раскрыть и всех его предков перебором узлов родителей с классом, где нет "Leaf" в пределах дерева. Это просто сделать циклом, используя свойство .parentNode и пару проверок. Если предок не является "листом", то поставить ему стиль ExpandOpen. На мой взгляд, запоминать состояние всего дерева не практично, да и зачем это может быть нужно. Чтобы задача имела нормальное решение надо вводить разумные ограничения - иначе заколебешься.
Автор: Shock (не зарегистрирован), дата: 16 января, 2009 - 04:28
#permalinkКстати, есть способ избежать использования класса 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;
}
Автор: Shock, дата: 16 января, 2009 - 05:04
#permalinkОчень понравился этот скрипт, потому незначительное улучшение от меня + класс на PHP для быстрого создания подобного списка.
Автор: Shock, дата: 16 января, 2009 - 05:09
#permalinkФайл JsTree.php
<?php class JsTree { private $final_html = null; private $finally_generated = false; private $title = null; private $tree = null; public function __construct($array) { if(!is_array($array)) return; $this->tree = $this->_parse_array_level($array); } public function set_title($title) { $this->title = $title; } private function _node_generate($content, $children=null, $islast=false) { if(!is_array($children)) $children = null; $islast = $islast ? " IsLast" : ""; $expand = $children ? " ExpandClosed" : " ExpandLeaf"; $children = $children ? $this->_parse_array_level($children) : ""; $node = "<ul class='Container'> <li class='Node$islast$expand'> <div class='Expand'></div> <div class='Content'>$content</div> $children</li> </ul>"; return $node; } private function _parse_array_level($array) { $result = ""; reset($array); for($i=1, $c=count($array); $i<=$c; $i++) { list($content, $children) = each($array); $islast = ($i==$c); $result.= $this->_node_generate($content, $children, $islast); } return $result; } private function _final_generate() { if($this->finally_generated) return; $header = $this->title ? "<div class='header'>" . $this->title . "</div>" : ""; $this->final_html = "<div class='Tree' onclick='tree_toggle(arguments[0])'>" . $header . $this->tree . "</div>"; $this->finally_generated = true; } public function render($output) { if(!$this->finally_generated) $this->_final_generate(); if($output) echo $this->final_html; else return $this->final_html; } } ?>Файл index.php
<!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Transitional//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd'> <html> <head> <title>Tree Example</title> <link rel='stylesheet' type='text/css' href='Tree.css' /> <script type='text/javascript' src='Tree.js'> </script> </head> <body> <?php require_once("JsTree.php"); $tree_array = array( 'elem.1' => null, 'elem.2' => array ( 'elem.2.1' => null, 'elem.2.2' => array ( 'elem.2.2.1' => null, 'elem.2.2.2' => array ( 'elem.2.2.2.1' => null, 'elem.2.2.2.2' => null, 'elem.2.2.2.3' => null, ), 'elem.2.2.3' => array ( 'elem.2.2.3.1' => null, ), ), 'elem.2.3' => array ( 'elem.2.3.1' => array ( 'elem.2.3.1.1' => null, 'elem.2.3.1.2' => null, 'elem.2.3.1.3' => null, ), 'elem.2.3.2' => null, 'elem.2.3.3' => null, ), ), 'elem.3' => null, ); $tree = new JsTree($tree_array); $tree->set_title('TreeTest'); $tree->render(true); ?> </body> </html>Файл Tree.css
.Tree .Container { padding: 0; margin: 0; } .Tree .Container li { list-style-type: none; } /* indent for all tree children excepts root */ .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; } /* left vertical line (grid) for all nodes */ .Tree .IsLast { background-image: url(img/i_half.gif); background-repeat : no-repeat; } .Tree .ExpandOpen .Expand { background-image: url(img/expand_minus.gif); } /* closed is higher priority than open */ .Tree .ExpandClosed .Expand { background-image: url(img/expand_plus.gif); } /* highest priority */ .Tree .ExpandLeaf .Expand { background-image: url(img/expand_leaf.gif); } .Tree .Content { min-height: 18px; margin-left:18px; } * html .Tree .Content { height: 18px; } .Tree .Expand { width: 18px; height: 18px; float: left; } .Tree .ExpandOpen .Container { display: block; } .Tree .ExpandClosed .Container { display: none; } .Tree .ExpandOpen .Expand, .ExpandClosed .Expand { cursor: pointer; } .Tree .ExpandLeaf .Expand { cursor: auto; }Файл Tree.js
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) }И картинки в папке imgs
Автор: Shock, дата: 16 января, 2009 - 05:14
#permalinkrequire_once("JsTree.php"); // Подключаем класс $tree_array = array();// Все элементы $tree = new JsTree($tree_array); // Создаем $tree->set_title('TreeTest'); //Необязательно - генерируем шапку $tree->render(true); // Выводим. Если необязательный аргумент указан - будет вызвано echo, иначе - return;как строить переменную $tree_array?
$tree_array = array( 'elem.1' => null, 'elem.2' => null, 'elem.3' => null, );если надо - заменяем нулл на еще один вложенный массив и так до бесконечности.
Автор: Кристина (не зарегистрирован), дата: 8 марта, 2009 - 19:51
#permalinkОгромное спасибо!!! Долго искала образец дерева, это самое удобное!
Автор: nim (не зарегистрирован), дата: 10 марта, 2009 - 02:26
#permalinkВ таком дереве есть недостаток. Нельзя добавить колонку(и) кнопок, которые были бы выровнены по правому краю. пришлось дерево сделать в таблице.
Автор: Ursus Drew (не зарегистрирован), дата: 20 марта, 2009 - 23:56
#permalinkСтоит заменить zoom: 1 на height: 1% - чтобы пройти валидацию.
Автор: Илья Кантор, дата: 30 марта, 2009 - 16:39
#permalinkВ разделе по AJAX появились статьи про интеграцию AJAX в интерфейс и статья про AJAX-дерево.
Автор: Yojik, дата: 31 марта, 2009 - 04:47
#permalinkКак раз делал дерево, только для jQuery UI, вот это, мой подход оказался очень похож на ваш, это клево.
Значит я на верном пути.
Автор: exxxtremist (не зарегистрирован), дата: 9 апреля, 2009 - 13:57
#permalinkОчень полезная статья, как в качестве примера работы с CSS, так и в качестве действительно полезного практического материала. Автору огромный респект. Аффтар, пеши исчо!!!
Автор: Ira (не зарегистрирован), дата: 20 мая, 2009 - 10:13
#permalinkЗдравствуйте! огромное спасибо за статью от всех новичков!
а не будет ли кто-нибудь так добр подсказать, как добавить к дереву кнопки "свернуть / развернуть все"?
заранее спасибо :-)
Автор: Mirok (не зарегистрирован), дата: 5 июня, 2009 - 12:44
#permalinkСПАСИБО!
Автор: Serg7 (не зарегистрирован), дата: 12 августа, 2009 - 00:00
#permalinkПрисоеденяюсь к вопросу Ira:
"Здравствуйте! огромное спасибо за статью от всех новичков!
а не будет ли кто-нибудь так добр подсказать, как добавить к дереву кнопки "свернуть / развернуть все"?
заранее спасибо :-)"
Спасибо за удобное решение!
Автор: Klimashkin (не зарегистрирован), дата: 18 августа, 2009 - 17:58
#permalinkВ IE8 отображается криво...
Автор: Гость (не зарегистрирован), дата: 19 августа, 2009 - 07:43
#permalinkНормально отображается, вообще все ок
Автор: Илья Кантор, дата: 19 августа, 2009 - 07:56
#permalinkОтвечаю на вопрос по "открыть все - свернуть все".
1) Без AJAX все достаточно просто. Функция рекурсивно пробегает по узлам дерева от корня к детям и все разворачивает/сворачивает. Придется использовать рекурсию или стек.
2) С AJAX отсутствующие узлы для операции "развернуть все" придется подгрузить. Но практика показывает, что функционал "развернуть все" к AJAX-узлам применяется редко.
Если вам действительно нужен этот код - я его писал и могу портировать в статью.
А так можно развернуть все загруженные узлы, используя простую рекурсию из п. 1.
Автор: Митяй (не зарегистрирован), дата: 19 августа, 2009 - 11:07
#permalinkОтлично все описано, есть пара идей по оптимизации
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')});
Вполне выразительно и короче, разве что не так наглядно.
Автор: alex39x, дата: 5 сентября, 2009 - 18:07
#permalinkПодскажите пожалуйста.
Если много элементов дерева - и раскрыть их - как сделать чтобы при закрытии одного из них страница не обновлялась?
Автор: Wolf11 (не зарегистрирован), дата: 7 октября, 2009 - 10:39
#permalinkХотелось выразить автору большую благодарность за статью. Даже я, совсем не зная JS, умудрился реализовать то, что мне надо. А точнее меню, в котором подменю появляется и исчезает по щелчку. Только оч. нужна ещё помощь: как сделать, что бы, скажем, открытый node1 первого уровня, становился закрытым при щелчке на node2 первого уровня?
Автор: Климашкин Павел (не зарегистрирован), дата: 9 октября, 2009 - 14:11
#permalinkВ класс .Node необходимо добавить свойство:
В противном случае при изменении масштаба страницы в firefox 3.5 и IE8 весь список уедет вправо из-за свойства float:left, так как они думают что все следующие списки должны обтекать предидущий!
Автор: 4vanger, дата: 13 октября, 2009 - 10:58
#permalinkдля hasClass вместо
new RegExp("(^|\\s)"+className+"(\\s|$)")лучше использовать "\b" для отделения слова от других. Делает то же, но выглядит чище. Плюс className надо экранировать (ну конечно если захочется использовать эту функцию чаще)
Автор: Гость (не зарегистрирован), дата: 2 ноября, 2009 - 22:51
#permalinkСпасибо большое!!! давно искала
Автор: Хыиуду (не зарегистрирован), дата: 3 ноября, 2009 - 17:07
#permalinkЕще одна реализация этого же дерева. CSS не менял, реализацию функций hasClass и tree_toggle тоже.
var offchange=" "; offset=offchange; level=1; function nodes_run(arr) { var cnt=0; for (var key in arr) cnt++; //А что еще делать, если яваскрипт такой убогий? for (var key in arr) { val=arr[key]; cnt--; if ((typeof(val)=='string')||(typeof(val)=='number')) // Это лист, финальный { result+=offset+'<li class="Node ExpandLeaf'; if (level==1) result+=' IsRoot '; if (cnt==0) result+=' IsLast'; result+='">\n'; offset=offset+offchange; result+=offset+'<div class="Expand"></div>\n'+ offset+'<div class="Content"><a href="'+val+'">'+key+'</a></div>\n'; offset=offset.slice(0, -offchange.length); result+=offset+'</li>\n'; } else if (typeof(val)=='object') //У этого рута есть еще дети { result+=offset+'<li class="Node ExpandClosed '; if (level==1) result+=' IsRoot'; if (cnt==0) result+=' IsLast'; result+='">\n'; offset=offset+offchange; result+=offset+'<div class="Expand"></div>\n'+ offset+'<div class="Content">'+key+'</div>\n'+ offset+'<ul class="Container">\n'; offset+=offchange; level+=1; nodes_run(val); level-=1; offset=offset.slice(0, -offchange.length); result+=offset+'</ul>\n'; offset=offset.slice(0, -offchange.length); result+=offset+'</li>\n'; } } return result; } function tree(arr, name) { result=""; var ret='<div onclick="tree_toggle(arguments[0])"> \n'+ '<div>'+name+'</div>\n'+ offset+'<ul class="Container">\n'; offset+=offchange; ret+=nodes_run(arr); offset=offset.slice(0, -offchange.length); ret+=offset+"</ul>\n</div>\n"; return ret; }Использование - скормить функции tree два параметра: ассоциативный массив с деревом и заголовок. Массив может выглядеть, к примеру, так:
var pump={ "111": { "foo": '1_1', "bar": '1_2' }, "foobar": { "qwerty": {"test1" : 'test2'}, "йцукен": '2_2' }, "temp": { "aaa": '3_1', "bbb": '3_2' }, "фываолдж": '4' };В каждой конечной паре ключ - это то, что будет выводиться, а значение - это ссылка, на которую поведет эта строка.
Автор: RadChand (не зарегистрирован), дата: 5 ноября, 2009 - 13:30
#permalinkМинималистический вариант
Функциональность таже что и у оригинала.
Удалены не нужные классы и правила css .Container, .Node, .Content, .Expand, .ExpandLeaf
.IsRoot тоже не нужен т.к. маргин заменен на паддинг, т.е. теперь, при построении отступов, не сам элемент дерева отодвигается в право относительно своего родителя, а отодвигает своих детей
.ExpandOpen, .ExpandClosed и .IsLast переименованы в .Expanded, .Contracted и .Last соответственно.
.Tree ul { padding: 0; margin: 0; } .Tree li { list-style-position: inside; list-style-image: url(/forum/img/expand_leaf.gif); background-image : url(/forum/img/i.gif); background-position : top left; background-repeat : repeat-y; padding-left: 18px; text-indent: -18px; cursor: pointer; } .Tree .Last { background-image: url(/forum/img/i_half.gif); background-repeat : no-repeat; } .Tree .Expanded { list-style-image: url(/forum/img/expand_minus.gif); } .Tree .Contracted { list-style-image: url(/forum/img/expand_plus.gif); } .Tree .Expanded ul { display: block; } .Tree .Contracted ul { display: none; }Также сокращен html. Удалены ненужные div-ы. Теперь выглядит так:
<div class="Tree" onclick="tree_toggle(arguments[0])"> <div>Tree</div> <ul> <li class="Last Expanded"> Root <ul> <li class="Contracted">Item 1 <ul> <li>Item 1.1<br />second line</li> <li class="Last">Item 1.2</li> </ul> </li> <li class="Last">Item 2</li> </ul> </li> </ul> </div>Ну и js тоже не избежал кастрации
function tree_toggle(event) { event = event || window.event; var node = event.target || event.srcElement; if(node.className.search("\\bExpanded\\b") != -1) node.className = node.className.replace('Expanded','Contracted'); else if(node.className.search("\\bContracted\\b") != -1) node.className = node.className.replace('Contracted','Expanded'); }Единственный минус по сравнении с оригиналом, курсор всегда pointer. Хотя для меня это плюс, т.к. листья у меня - ссылки, т.е. тоже кликабельные.
Автор: RadChand (не зарегистрирован), дата: 5 ноября, 2009 - 13:42
#permalinkPS Если дерево без линий, то и .Last не нужен.
Автор: Гость (не зарегистрирован), дата: 1 декабря, 2009 - 18:37
#permalinkВы пишите, что (цитата):
Обратите внимание - вся используемая разметка является исключительно семантической. В данном случае CSS-класс говорит не "каким образом следует выделить элемент", а "что элемент обозначает".
На самом деле у Вас разметка говорит не только что элемент обозначает, но и хранит его состояние - ExpandOpen, что не есть хорошо, т.к. расплывается понятие "узел дерева" до "узел дерева в состоянии" * на количество состояний. А если добавить чекбоксы, селектор, ещё что нибудь, то будет бо-бо
Состояние узла и применяемый в этом состоянии стиль следует хранить отдельно. Я использую для этого statemap (хэш-таблица, где ключ статус, значение - доп. стиль) и функцию switch(statename). При небольшом расширении это позволяет менять не только стиль, но любые свойства элемента DOM при переходе между состояниями, а так же при необходимости отслеживать валидность переходов и сохранять состояние страницы, чтобы оно восстанавливалось при следующем заходе пользователя.
Автор: Роман (не зарегистрирован), дата: 5 февраля, 2010 - 17:45
#permalink-- Ну начнем с того что пустой див ни как не может быть семантической разметкой - по определению.
-- Два - именование классов к семантике имеет слабое отношение, боюсь что даже никакое.
Автор: xxen (не зарегистрирован), дата: 4 марта, 2010 - 00:02
#permalinkА у меня выполнено в виде плагина к 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".
Автор: Гость (не зарегистрирован), дата: 25 мая, 2010 - 11:57
#permalinkСтатья превосходна в техническом плане. Но в плане практики у меня остались вопросы. Ведь мне например понадобится не только показать это дерево, но и произвести с ним какие-либо операции, например отметить какой то элемент дерева галкой и отправить данные из моей формы с учетом выбранной галки на сервер. Таким образом получаем новое требование: массив дерева должен содержать не только "Имя_элемента", но и "ID_элемента"!!! Без ID никак нельзя, сами понимаете, ведь в имени могут быть и русские буквы, к которым сервер может иметь неприязнь, так еще и имена из соседних веток дерева могут совпадать. Надеюсь доступно объяснил свою мысль. Вобщем нужно прикрутить массив, в котором не один ключ, а два и более: [id] и [name]
Автор: Гость (не зарегистрирован), дата: 25 мая, 2010 - 12:23
#permalinkPS.:
интересует реализация такого дерева на php + javascript
Автор: Гость (не зарегистрирован), дата: 22 июня, 2010 - 10:10
#permalinkУважаемые, построил дерево, используя указанный код (мой опыт JS близок к нулю).
Задача: передать в скрытую форму значение, содержащееся в классе 'Content' для дальнейшей обработки.
Проблема: не передает IE 6.0 и Опера 9.61, но с Фоксом и Сафари работает.
f
unction tree_toggle(event) { event = event || window.event var clickedElem = event.target || event.srcElement var node = clickedElem.parentNode if (hasClass(node, 'Content')) { document.getElementById('TTT').value = clickedElem.parentNode.textContent document.getElementById('myform').submit() return // клик на листе } … }Подскажите как правильно.
Автор: Гость (не зарегистрирован), дата: 28 июля, 2010 - 12:11
#permalinkя в java плохо разбераюсь. Подскажите как сделать сворачивание открытой ветки при при открытии другой ветки, что бы у дерева постоянно была открыта только одна ветка.
Спасибо.
Автор: ivanmfan (не зарегистрирован), дата: 12 августа, 2010 - 12:12
#permalinkПарни подскажите плиз как сделать автоперенос текста на новую строку? Ну ограничить ширину меню. Использую все стандартное...
Автор: Cibeboing (не зарегистрирован), дата: 4 февраля, 2011 - 10:11
#permalinkда, наверно так и есть
Автор: tratatun tratatun4k (не зарегистрирован), дата: 14 марта, 2011 - 05:09
#permalinkавтору огромное спасибо за статью!!
от себя бы дополнил, что класс IsLast можно смело заменить на следующее и не засорять ваш js код ненужной ф-ностью :
.IsLastul:last-child>li:last-child { background-image: url(../img/i_half.gif); background-repeat: no-repeat; }Автор: Jungle Dread (не зарегистрирован), дата: 16 марта, 2011 - 21:37
#permalinkОбьясните пожалуйста эту строку:
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 то ... Зачем создавать функцию?
}
Спасибо, только в гугл не посылайте! А то раньше на х... ,а теперь в гугл!)
Автор: 0931454574, дата: 16 марта, 2011 - 22:36
#permalinkfunction hasClass(elem, className) {
return new RegExp("(^|\\s)"+className+"(\\s|$)").test(elem.className) А вот тут не соасем ясно что это?"(^|\\s)" "(\\s|$)" .test(elem.className)
ОБЬЯСНИТЕ ПОЖАЛУЙСТА как и что значит в этой функции, плииз, нашел все в нете но в голове каша! я понимаю что меняется что т оместами но как задается условие немогу понять,СПАСИБО СПАСИБО
Автор: Дмитрий Синюков (не зарегистрирован), дата: 22 апреля, 2011 - 19:03
#permalinkВопрос к автору. Можно ли данный код использовать в коммерческом проекте? Есть ли лицензия?
Автор: Гость (не зарегистрирован), дата: 10 мая, 2011 - 14:35
#permalinkСпасибо за исходник! Очень выручает!
вот пример кода на JQuery для отметки всех дочерних checkbox'ов
window.onload = function(){ tree("tree", "index.php?s=Tree"); $(".myCheckbox").change(function(eventData, eventObject){ var status = $(this).attr("checked"); var id = $(this).parent().parent().attr("id"); $("#" + id + " :checkbox").each(function(k, v){ $(this).attr("checked", status); }); }); }Автор: Иваннн, дата: 31 августа, 2011 - 16:32
#permalinkКогда писал свое дерево не додумался использовать классы для изменения картинок и закрытия и открытия LI, но как убирается продолжение бекгроунда первого (коренного LI) элемента, если последний из его детей сам имеет детей и раскрыт. Ведь коренной LI охватывает все элементы, и соответсвенно его бекгроунд тоже продолжается до конца. хотя должен закончится на последнем ребенке?
Автор: deivan (не зарегистрирован), дата: 5 декабря, 2011 - 13:59
#permalinkЛюди добрые, а как сделать, чтобы сам корень Root не отображался?.. Т.е. дерево сразу показывало ветки, без корня?..
Автор: Юрий 123456789 (не зарегистрирован), дата: 23 декабря, 2011 - 14:23
#permalinkСделаю закладочку. Очень толково написано.
Автор: Doctor Death (не зарегистрирован), дата: 26 декабря, 2011 - 05:40
#permalinkСпасибо за хороший и простой и понятный скрипт
использовал версию с чекбоксами
не могли бы посоветовать как реализовать такю фунцию
нужно поставить чекбокс на верхние уровни папок с таким расчетом чтобы при его выбори выбирались все в ветке
Автор: aее (не зарегистрирован), дата: 2 февраля, 2012 - 09:19
#permalinkКак сделать все элементы в сетке, пробовал в container style=border:1px solid black
обводятся элементы, а дерева в сетке не получается
Автор: Гость (не зарегистрирован), дата: 22 апреля, 2012 - 21:17
#permalinkВ таком виде всё открывается в 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>
Автор: Зуек (не зарегистрирован), дата: 8 июля, 2012 - 13:01
#permalinkДобрый день!
Подскажите, как выделить цветом выделенный элемент, соответственно, сбросить цвет предыдущего выделенного, т.е. щелкаю Item1 - он подсвечивается.
Спасибо!
Автор: Довольный (не зарегистрирован), дата: 10 августа, 2012 - 11:33
#permalinkОгромное спасибо за статью, жаль что не могу ссылку повесить
Автор: Гость (не зарегистрирован), дата: 7 сентября, 2012 - 11:57
#permalinkАвтору - в Firefox 15 идет баг с checkbox. Надо убрать из стилей width и height для input. Тогда все нормально.
Автор: Гость (не зарегистрирован), дата: 7 сентября, 2012 - 12:04
#permalinkИли надо выставить width и height по 13px;
Автор: Гость (не зарегистрирован), дата: 31 октября, 2012 - 23:46
#permalinkИлья, скажите пожалуйста, что означает передача arguments[0] в функцию tree_toggle ?
Заранее спасибо
Автор: Гость (не зарегистрирован), дата: 19 ноября, 2012 - 09:17
#permalinkСтатья очень помогла бы, если бы вместе с кодом не копировались номера строк кода!!!
Автор: .* (не зарегистрирован), дата: 22 марта, 2013 - 12:39
#permalinkxml-овский минимализьм (-|+) пользую покамест, оный у разработчиков степенно взяв.) оказывается html + css + js + skin это xml.
Автор: mirt (не зарегистрирован), дата: 7 сентября, 2013 - 11:27
#permalinkПодскажите, под какой лицензией вы выпускаете этот код?
Автор: Гость (не зарегистрирован), дата: 12 ноября, 2013 - 17:09
#permalinkПодскажите, пожалуйста, как можно реализовать это дерево от дочерних ссылок к родительским. Спасибо
Автор: Виталя (не зарегистрирован), дата: 12 мая, 2014 - 11:09
#permalinkДа, это хорошо
Автор: Гость (не зарегистрирован), дата: 3 июля, 2014 - 14:37
#permalinkВсе сделал как было указано закачал на хостинг, по умолчанию при открытии страницы все списки раскрываются, хотя на примере не так. Подскажите что надо сделать чтобы при открытии страницы списки были свернуты при открытии страниц.
Автор: Гость (не зарегистрирован), дата: 2 июля, 2015 - 12:10
#permalinkИ все равно не понял (даже прочитав комментарии), как грузить подобный список из бд (ms sql) с помощью php =/
Автор: Ирина86 (не зарегистрирован), дата: 2 марта, 2016 - 13:04
#permalinkБольшое спасибо за статью! А можете подсказать, как это дерево убрать в выпадающий список, т.е. чтобы изначально кроме "- Выберите значение-" ничего не было?
Автор: Кос123456789 (не зарегистрирован), дата: 25 апреля, 2017 - 15:49
#permalinkСпасибо!
Автор: Кос123456789 (не зарегистрирован), дата: 25 апреля, 2017 - 16:49
#permalinkОптимизация кода:
.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;
Автор: Кос123456789 (не зарегистрирован), дата: 26 апреля, 2017 - 10:17
#permalinkДля 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;
}
}
Автор: Гость (не зарегистрирован), дата: 8 января, 2019 - 20:09
#permalinkспасибо чувак
Автор: Гость (не зарегистрирован), дата: 20 ноября, 2019 - 20:49
#permalinkПодскажите, пожалуйста, как приделать справа от имени узлов текстовые input-ты и прижать их к правому краю. Первый прижимается, а остальные выстраиваются в лесенку.
Дата рождения