Javascript.RU

Грамотное javascript-дерево за 7 шагов

В этой статье описана DOM/CSS-структура дерева, которую я в свое время разработал для dojo toolkit.

Основные особенности:

  • Семантическая удобная CSS-разметка.
    Внешний вид дерева определяется исключительно CSS.
  • Скрытие/раскрытие узлов
  • Структура дерева обозначена линиями
  • Допускает многострочное HTML-содержимое в узлах
  • Оптимизация по количеству HTML-тагов
  • Легко дополняется новыми фишками

Например:

Root
  • Item 1
    • Item 1.1
      • Item 1.1.2
    • Item 1.2
  • Item 2
    title long yeah
    • Item 2.1
  • Item 3
    • Item 3.1

Основной строительный блок дерева - его узел.

Каждый узел имеет класс Node и состоит из иконки Expand, заголовка Content и контейнера для детей Container.

Визуальное представление узла:

Например, вот так выглядит разметка просто корневого узла Root, без детей:

  • 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.

  • Root
    • 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 1.1
    • 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-правилами.

  1. Узел Node поддерживает вертикальную линию к своему следующему соседу
    .Node {
        margin-left: 18px;
        zoom: 1;
        /* линия слева образуется повторяющимся фоновым рисунком */
        background-image : url(/forum/img/i.gif);
        background-position : top left;
        background-repeat : repeat-y;
    }
    
  2. Если соседа ниже нет, то линию вниз продолжать не надо:
    /* это правило - ниже .Node, поэтому имеет больший приоритет */
    .IsLast {
        /* добавить соединительную черточку наверх */
        background-image: url(/forum/img/i_half.gif);
        background-repeat : no-repeat;
    }
    

Получается, что все узлы на одном уровне соединены вертикальной чертой.

Размер рисунков для фоновых черточек сделан такой, чтобы вертикальная черта проходила строго посередине иконок Expand.

Поэтому получается, что эти иконки автоматически "нанизываются" на вертикальную линию.

Чтобы получить более целостную картину, можно обновить иконки Expand, добавив к ним соединительную черту для подключения заголовка к вертикальной линии.

Вот такие новые иконки для Expand*-классов.

Открытый узел ExpandOpen
Закрытый узел ExpandClosed
Лист ExpandLeaf
Горизонтальные коннекторы готовы.

Вертикальные линии образуют каркас, а новые иконки 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 1.1
      • Item 1.1.2
    • Item 1.2
  • Item 2
    title long yeah
    • Item 2.1
  • Item 3
    • Item 3.1
<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 делать ему осталось всего ничего:

  1. Определить, произошел ли клик на иконке Expand, используя event.target(или event.srcElement для IE)
  2. Получить узел Node для иконки
  3. Если узел - не лист, то поменять класс 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:

  • Item 1
    • Item 1.1
      • Item 1.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 1.1
      • Item 1.1.2
    • Item 1.2
  • Item 2
    title long yeah
    • Item 2.1
  • Item 3
    • Item 3.1
<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
#permalink

Замечательная и полезная статья!


Автор: ГостьMarium Marium (не зарегистрирован), дата: 2 мая, 2023 - 19:28
#permalink

I have found your material to be of great assistance. I've been able to use what I've learned to greatly enhance my life.

https://wick-john.blogspot.com/
https://wickjohn7.wordpress.com/
https://wickjohn.weebly.com
https://wickjohn1.wixsite.com/mysite


Автор: 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
#permalink

Update: Добавил 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 тыщ узлов ?


Автор: Илья Кантор, дата: 9 июля, 2008 - 16:38
#permalink

Да, в статье не разобран вопрос динамической подгрузки данных.

С другой стороны, её достаточно просто реализовать самому.

Структура дерева удобна для добавления и удаления узлов, а как получить AJAX'ом с сервера данные - это уже можно, например, раздел про AJAX посмотреть.

UPDATE: В разделе по AJAX появились статьи про интеграцию AJAX в интерфейс и статья про AJAX-дерево.


Автор: Гость (не зарегистрирован), дата: 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 - 15:13
#permalink

В статье сделан именно крестик, т.к на клик на имени часто вешается что-то другое, например, открытие страницы с этим именем.

Хотя, конечно, в вашем случае может быть целесообразно расширить дерево именно так


Автор: Гость (не зарегистрирован), дата: 24 ноября, 2008 - 12:44
#permalink

Что-то у меня такой вариант не работает, а очень нужно, не могу понять почему?


Автор: король ящериц (не зарегистрирован), дата: 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 вызывается функция, которая её меняет для данного тага?


Автор: Илья Кантор, дата: 26 октября, 2008 - 23:44
#permalink

В чем именно проблема? Это вопрос про AJAX или про анимацию или про то как они связаны ?


Автор: EugenyK, дата: 28 октября, 2008 - 02:43
#permalink

Именно про анимацию. Чем достигается появление результата ответа не сразу, а через "выкатывание" вниз?

У меня было предположение по поводу неё такое:
Получают результат запроса в div, далее дают ему более меньший zIndex и относительное позиционирование, смещают top на -offsetHeight и привязывают к DOM, а потом в вызывают функцию рекурсивно, которая увеличивает top до 0 и возвращают прежний zIndex.
Реализовывать это не пробовал (вижу в этом некую нерациональность)


Автор: Илья Кантор, дата: 28 октября, 2008 - 11:21
#permalink

Там изменение height + setTimeout.

То есть, появляется div, у него размер 0.
Дальше, скажем, каждые 5 ms размер увеличивается на 1px.
И так - до полного выкатывания, т.е пока height < div.scrollHeight


Автор: EugenyK, дата: 18 июня, 2009 - 14:47
#permalink

Если менять высоту div'а итерациями от 0 до scrollHeight через глобальную функцию, то на достаточно большом кол-ве содержащихся в этом div'е дочерних элементов время загрузки очень сильно возрастает (независимо от пересчёта интервала времени вызова на бОльшую высоту). Такого не наблюдается, если создать класс и нужное кол-во раз вызывать метод.

Но эффектнее это выглядит, если сделать анимацию высоты от 0 до 5-10px, а от 5-10px сразу до scrollHeigh. В обратном порядке реализовать скрытие.


Автор: yarich (не зарегистрирован), дата: 27 октября, 2008 - 14:05
#permalink

хорошо бы добавить функционал сохранения состояния дерева при обновлении страницы


Автор: kirilloid (не зарегистрирован), дата: 5 июня, 2009 - 20:10
#permalink

Проблем-то. Добавляем функции serialize/unserialize и прикручиваем метод хранения данных по желанию: cookie / DOM storage / flash LSO ...


Автор: 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. На мой взгляд, запоминать состояние всего дерева не практично, да и зачем это может быть нужно. Чтобы задача имела нормальное решение надо вводить разумные ограничения - иначе заколебешься.


Автор: Гость (не зарегистрирован), дата: 17 декабря, 2008 - 21:02
#permalink

а можете написать как это сделать?


Автор: 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


Автор: Гость (не зарегистрирован), дата: 19 июня, 2020 - 15:41
#permalink

+


Автор: Shock, дата: 16 января, 2009 - 05:14
#permalink
require_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,
);

если надо - заменяем нулл на еще один вложенный массив и так до бесконечности.


Автор: constantant, дата: 29 января, 2009 - 06:48
#permalink

Как-то всё очень сложно... я сделал на рекурсивной функции:
index.php

<html>
<head>
<link rel='StyleSheet' type='text/css' href='nogrid.css'/>
</head>
<body>
<?php
require_once("../config.php");
function tree($uid){
$sql = "SELECT id_folder,id_mother,id_type,pos,hide,name FROM folder WHERE id_mother=".$uid;
$a = mysql_query($sql);
for($i=0;$i<mysql_num_rows($a);$i++){
if(mysql_num_rows(mysql_query("SELECT id_folder FROM folder WHERE id_mother=".mysql_result($a,$i,0)))>0){
$class="Node ExpandClosed";
}
else{
$class="Node ExpandLeaf IsLast";
}
echo"<ul class='Container' id='".mysql_result($a,$i,0)."'><li class='".$class."'><div class='Expand'></div><input type='checkbox'/>";
echo "<div class='Input InputFolder Closed'></div><div class='Input InputNear NearOff'></div><div class='Content'>".mysql_result($a,$i,5)."</div>";
tree(mysql_result($a,$i,0));
}
echo"</li></ul>\n";
}
echo"<div id='0'><div id='root' onclick='tree_toggle(arguments[0])' onselectstart='return false' style='-moz-user-select:none;-khtml-user-select:none;user-select:none;'>";
tree(0,0);
echo"</div></div>";
?>
<script type="text/javascript" src="dnd_03.js"></script>
</body>
</html>

а ещё я извратился и сделал это дерево drag&drop-изменяемым))) писецнна! Но работает)))
dnd_03.js

var dragEl=[];
var InCont=[];
var NrCont=[];
var msDw=false;
var dragHelper=null;

if(window.HTMLElement){
HTMLElement.prototype.removeNode = function(removeChildren) {
  if (Boolean(removeChildren))
    return this.parentNode.removeChild(this);
  else {
    var r=document.createRange();
    r.selectNodeContents(this);
    return this.parentNode.replaceChild(r.extractContents(),this);
  }
}}
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)
}
Number.prototype.NaN0=function(){return isNaN(this)?0:this;}
function getPosition(e){
	var left = 0;
	var top  = 0;
	while (e.offsetParent){
		left += e.offsetLeft + (e.currentStyle?(parseInt(e.currentStyle.borderLeftWidth)).NaN0():0);
		top  += e.offsetTop  + (e.currentStyle?(parseInt(e.currentStyle.borderTopWidth)).NaN0():0);
		e     = e.offsetParent;
	}
	left += e.offsetLeft + (e.currentStyle?(parseInt(e.currentStyle.borderLeftWidth)).NaN0():0);
	top  += e.offsetTop  + (e.currentStyle?(parseInt(e.currentStyle.borderTopWidth)).NaN0():0);
	return {x:left, y:top};
}
function mouseCoords(ev){
	if(ev.pageX || ev.pageY){
		return {x:ev.pageX, y:ev.pageY};
	}
	return {
		x:ev.clientX + document.body.scrollLeft - document.body.clientLeft,
		y:ev.clientY + document.body.scrollTop  - document.body.clientTop
	};
}
function ChangeClass(type,obj){
switch(type){
case 'Open':
var re =  /(^|\s)(Open|Closed)(\s|$)/;
break;
case 'Closed':
var re =  /(^|\s)(Open|Closed)(\s|$)/;
break;
case 'NearOn':
var re =  /(^|\s)(NearOn|NearOff)(\s|$)/;
break;
case 'NearOff':
var re =  /(^|\s)(NearOn|NearOff)(\s|$)/;
break;
case 'ExpandOpen':
var re =  /(^|\s)(ExpandLeaf IsLast)(\s|$)/;
break;
case 'ExpandLeaf IsLast':
var re =  /(^|\s)(ExpandOpen|ExpandClosed)(\s|$)/;
break;
}
obj.className = obj.className.replace(re, '$1'+type+'$3')
}
function mouseMove(ev){
	ev         = ev || window.event;
	var target = ev.target || ev.srcElement;
if(dragEl[3]==true){
for(var i=0; i<dragHelper.childNodes.length; i++){
dragHelper.removeChild(dragHelper.childNodes[0]);
}
dragHelper.appendChild(dragEl[0].cloneNode(true));
dragHelper.style.display = 'block';
dragHelper.childNodes[0].style.background = '#a1a1a1';
dragHelper.childNodes[0].style.visibility='visible';
var mousePos = mouseCoords(ev);
			dragHelper.style.top  = mousePos.y + 30;
			dragHelper.style.left = mousePos.x;
dragEl[0].style.visibility = 'hidden';
}}

function mouseUp(){
if(dragEl[3]==true){
if((InCont[2]==true && InCont[1]!=dragEl[2]) || NrCont[3]==true){
if(InCont[2]==true){
dragEl[0].style.visibility='visible';
InCont[0].appendChild(dragEl[0]);
dragEl[3]=false;
dragHelper.style.display = 'none';
InCont[2]=false;
ChangeClass('Closed',InCont[3]);
ChangeClass('ExpandOpen',InCont[0]);
if(dragEl[4].getElementsByTagName('ul').length==0){
ChangeClass('ExpandLeaf IsLast',dragEl[4]);
}

}
if(NrCont[3]==true){
dragEl[0].style.visibility='visible';
NrCont[4].insertBefore(dragEl[0], NrCont[0]);
dragEl[3]=false;
dragHelper.style.display = 'none';
NrCont[3]=false;
ChangeClass('NearOff',NrCont[5]);
if(dragEl[4].getElementsByTagName('ul').length==0){
ChangeClass('ExpandLeaf IsLast',dragEl[4]);
}
}
}else{
if(dragEl[3]=true){
dragEl[0].style.visibility = 'visible';
dragHelper.style.display = 'none';
dragEl[3]=false;
if(InCont[2]==true){
ChangeClass('Closed',InCont[3]);
}}}}}
function PutInThis(){
if(dragEl[3]==true){
if(hasClass(this, 'InputFolder')){
ChangeClass('Open',this);
var InFolderCont=this.parentNode;
InCont[0]=InFolderCont;
InCont[1]=InFolderCont.parentNode.id;
InCont[2]=true;
InCont[3]=this;
}
if(hasClass(this, 'InputNear')){
ChangeClass('NearOn',this);
var NearCont=this.parentNode.parentNode;
NrCont[0]=NearCont;
NrCont[1]=NearCont.id;
NrCont[2]=NearCont.parentNode.parentNode.id;
NrCont[3]=true;
NrCont[4]=NearCont.parentNode;
NrCont[5]=this;
}}}
function NotThis(){
if(dragEl[3]==true){
if(hasClass(this, 'InputFolder')){
ChangeClass('Closed',this);
InCont[2]=false;
}
if(hasClass(this, 'InputNear')){
ChangeClass('NearOff',this);
NrCont[3]=false;
}}}
function GetThis(ev){
var dragElprs=this.parentNode.parentNode;
dragEl[0]=dragElprs;
dragEl[1]=dragElprs.id;
dragEl[2]=dragElprs.parentNode.parentNode.id;
dragEl[3]=true;
dragEl[4]=dragElprs.parentNode;
}

var container=document.getElementById('root');
for(var i=0; i<container.getElementsByTagName('div').length; i++){
var tElem=container.getElementsByTagName('div')[i];
if(hasClass(tElem, 'InputFolder')){
tElem.onmouseover=PutInThis;
tElem.onmouseout=NotThis;
}
if(hasClass(tElem, 'InputNear')){
tElem.onmousedown=GetThis;
tElem.onmouseover=PutInThis;
tElem.onmouseout=NotThis;
}}
if(dragHelper==null){
dragHelper = document.createElement('div');
dragHelper.style.cssText = 'position:absolute;display:none;';
document.body.appendChild(dragHelper);
}
document.onmouseup = mouseUp;
document.onmousemove = mouseMove;

просьба к тем, кто силём в яваскрипте, посоветовать чего и как тут можно оптимизировать))
nogrid.css

img{margin: 0px;padding: 0px;border: 0}

.Container {
	padding: 0;
	margin: 0;
}

.Container li {
	list-style-type: none;
}

/* indent for all tree children excepts root */
.Node {
    margin-left: 18px;
   /* zoom: 1;*/

}

.IsRoot {
    margin-left: 0;
}
 
.ExpandOpen .Expand {
    background-image: url(http://javascript.ru/forum/img/minus.gif);
}
 
/* closed is higher priority than open */
.ExpandClosed .Expand {
    background-image: url(http://javascript.ru/forum/img/plus.gif);
}
 
/* highest priority */
.ExpandLeaf .Expand {
    background-image: url(http://javascript.ru/forum/img/leaf.gif);
}


* html  .Content {
	float: left;
   /* height: 18px;*/
}

.Expand {
    width: 18px;
    height: 18px;
    float: left;
}


.ExpandOpen .Container {
        display: block;
}

.ExpandClosed .Container {
        display: none;
}

.ExpandOpen .Expand, .ExpandClosed .Expand {
        cursor: pointer;
}
.ExpandLeaf .Expand {
        cursor: auto;
}
.Closed {
    background-image: url(http://javascript.ru/forum/img/closed.gif);
}
.Open {
    background-image: url(http://javascript.ru/forum/img/open.gif);
}
.NearOff {
    background-image: url(http://javascript.ru/forum/img/closed.gif);
}
.NearOn {
    background-image: url(http://javascript.ru/forum/img/open.gif);
}
.Input {
    width: 18px;
    height: 18px;
    float: left;
    cursor: pointer;
}
.Node {
        margin-left: 36px;
}
.Content {
        /*margin-left: 36px;*/
}
.IsRoot { margin-left: 0; }

.Container input {
    width: 14px;
    height: 14px;
    float: left;
    margin: 2px;
}

Автор: Кристина (не зарегистрирован), дата: 8 марта, 2009 - 19:51
#permalink

Огромное спасибо!!! Долго искала образец дерева, это самое удобное!


Автор: nim (не зарегистрирован), дата: 10 марта, 2009 - 02:26
#permalink

В таком дереве есть недостаток. Нельзя добавить колонку(и) кнопок, которые были бы выровнены по правому краю. пришлось дерево сделать в таблице.


Автор: Илья Кантор, дата: 11 марта, 2009 - 17:47
#permalink

Да, и еще оно на баяне не играет.


Автор: Гость (не зарегистрирован), дата: 23 марта, 2009 - 14:31
#permalink

+1


Автор: Гость (не зарегистрирован), дата: 24 апреля, 2009 - 17:42
#permalink

Взаимно, +1 ))


Автор: Илья Кантор, дата: 25 апреля, 2009 - 22:38
#permalink

Впрочем, на самом деле добавить колонку таких кнопок можно, CSS-структура это замечательно позволяет.

Сейчас узел построен по принципу двухколоночной верстки: [fixed width] + [left margin]. Добавить правую колонку - это всего лишь перейти на трехколоночную верстку дивами: [fixed width] + [left/right margin] + [fixed width].


Автор: 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')});

Вполне выразительно и короче, разве что не так наглядно.


Автор: Митяй (не зарегистрирован), дата: 19 августа, 2009 - 13:11
#permalink

Последний листинг читать как
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 необходимо добавить свойство:

clear:left;

В противном случае при изменении масштаба страницы в 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. Хотя для меня это плюс, т.к. листья у меня - ссылки, т.е. тоже кликабельные.


Автор: Geb (не зарегистрирован), дата: 7 сентября, 2010 - 16:52
#permalink

Вот только не надо про функционал заливать. Он конкретно урезан. Автор молодец предусмотрел все варианты. Мне например надо кнопочки на каждый нод и еще подсветка бзкграундом, что в вашем случае недостижимо.

Я с JS на вы, кто-то может выложить готовый код для Toggle All Nodes. Должно быть просто изменение Close <=> Open с перебором всех узлов. Респект автору!


Автор: Гость (не зарегистрирован), дата: 19 июня, 2020 - 15:42
#permalink

+


Автор: RadChand (не зарегистрирован), дата: 5 ноября, 2009 - 13:42
#permalink

PS Если дерево без линий, то и .Last не нужен.


Автор: Sera (не зарегистрирован), дата: 6 ноября, 2009 - 14:10
#permalink

попробовал твой сокращённый вариант, работает отлично... на IE, а вот на firefox дерево получается кривое, в чём может быть проблемма?


Автор: RadChand (не зарегистрирован), дата: 9 ноября, 2009 - 13:37
#permalink

Да, действительно, FF делает отступ между маркером списка и текстом. Хотя, я бы не сказал, что дерево сильно кривое, просто лишний пробел перед текстом.
Но, тем не менее, пофиксить можно так:

Придется завернуть текст в спаны.

<li><span>Item 1.1</span></li>

И в CSS добавить хак для FF.

/*FF hack*/
@-moz-document url-prefix() {
  .Tree span {margin-left:-8px;}
}

Автор: Sera (не зарегистрирован), дата: 10 ноября, 2009 - 09:46
#permalink

Да, это помогло, спасибо!


Автор: Гость (не зарегистрирован), дата: 1 декабря, 2009 - 18:37
#permalink

Вы пишите, что (цитата):

Обратите внимание - вся используемая разметка является исключительно семантической. В данном случае CSS-класс говорит не "каким образом следует выделить элемент", а "что элемент обозначает".

На самом деле у Вас разметка говорит не только что элемент обозначает, но и хранит его состояние - ExpandOpen, что не есть хорошо, т.к. расплывается понятие "узел дерева" до "узел дерева в состоянии" * на количество состояний. А если добавить чекбоксы, селектор, ещё что нибудь, то будет бо-бо

Состояние узла и применяемый в этом состоянии стиль следует хранить отдельно. Я использую для этого statemap (хэш-таблица, где ключ статус, значение - доп. стиль) и функцию switch(statename). При небольшом расширении это позволяет менять не только стиль, но любые свойства элемента DOM при переходе между состояниями, а так же при необходимости отслеживать валидность переходов и сохранять состояние страницы, чтобы оно восстанавливалось при следующем заходе пользователя.


Автор: Роман (не зарегистрирован), дата: 5 февраля, 2010 - 17:45
#permalink

Обратите внимание - вся используемая разметка является исключительно семантической. В данном случае CSS-класс говорит не "каким образом следует выделить элемент", а "что элемент обозначает".

-- Ну начнем с того что пустой див ни как не может быть семантической разметкой - по определению.

-- Два - именование классов к семантике имеет слабое отношение, боюсь что даже никакое.


Автор: 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
#permalink

PS.:

интересует реализация такого дерева на php + javascript


Автор: Гость (не зарегистрирован), дата: 22 июня, 2010 - 10:10
#permalink

Уважаемые, построил дерево, используя указанный код (мой опыт JS близок к нулю).
Задача: передать в скрытую форму значение, содержащееся в классе 'Content' для дальнейшей обработки.
Проблема: не передает IE 6.0 и Опера 9.61, но с Фоксом и Сафари работает.

<?php …

$classCont="<ul class='Container'><li class='Node ExpandLeaf IsLast'><div class='Expand'></div><div class='Content'>";
…
$Cont="<a href='#'>".$classCont.$N. "</a></div></li></ul>";
…?>

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 код ненужной ф-ностью :

.IsLast

ul: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 - 21:54
#permalink

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? Что значит ? и двоеточие?


Автор: 0931454574, дата: 16 марта, 2011 - 22:36
#permalink

function 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 ?
Заранее спасибо


Автор: Гость (не зарегистрирован), дата: 31 октября, 2012 - 23:59
#permalink

Наверное, сам понял:
передаётся только объект, по которому кликнули, игнорируя всплывание клика


Автор: Гость (не зарегистрирован), дата: 19 ноября, 2012 - 09:17
#permalink

Статья очень помогла бы, если бы вместе с кодом не копировались номера строк кода!!!


Автор: Гость (не зарегистрирован), дата: 3 декабря, 2012 - 09:16
#permalink

Там кнопки есть, нажи и откроется новое окно с кодом без номеров строк.


Автор: .* (не зарегистрирован), дата: 22 марта, 2013 - 12:39
#permalink

xml-овский минимализьм (-|+) пользую покамест, оный у разработчиков степенно взяв.) оказывается 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

спасибо чувак


Автор: Гость (не зарегистрирован), дата: 30 сентября, 2019 - 13:11
#permalink

The knowledge you share really changes me in my life, I sincerely thank you for what you have done, surely your blog will help more people.\
vex 3


Автор: Patsy J. Moore (не зарегистрирован), дата: 10 октября, 2019 - 12:23
#permalink

I agree with this! I say nice job for those folks. bottle flip


Автор: Гость (не зарегистрирован), дата: 20 ноября, 2019 - 20:49
#permalink

Подскажите, пожалуйста, как приделать справа от имени узлов текстовые input-ты и прижать их к правому краю. Первый прижимается, а остальные выстраиваются в лесенку.

Дата рождения

Автор: baokolai (не зарегистрирован), дата: 21 января, 2020 - 05:49
#permalink

I do not know what to say really what you share very well and useful to the community, I feel that it makes our community much more developed, thanks happy wheels


Автор: Гость (не зарегистрирован), дата: 31 марта, 2020 - 19:20
#permalink

very good post.


Автор: Гость (не зарегистрирован), дата: 31 марта, 2020 - 19:21
#permalink

great post.


Автор: Гость (не зарегистрирован), дата: 31 марта, 2020 - 19:23
#permalink

very good post.


Автор: nobita1990 (не зарегистрирован), дата: 9 июня, 2020 - 06:51
#permalink

A very interesting topic that I have considered, I think this is one of the most important information for me. And I'm glad to read your article. Thanks for sharing! paper io 2


Автор: jsdg225 (не зарегистрирован), дата: 23 июня, 2020 - 20:43
#permalink

his is great article shared. Love to read this.https://jsdg225.tumblr.com/


Автор: jsdg225 (не зарегистрирован), дата: 23 июня, 2020 - 20:45
#permalink

Автор: 8 ball (не зарегистрирован), дата: 14 июля, 2020 - 11:17
#permalink

It is wonderful to be here with everyone, I have a lot of knowledge from what you share, to say thank you, the information and knowledge here helps me a lot basketball legends


Автор: 먹튀검증업체 (не зарегистрирован), дата: 22 ноября, 2020 - 05:58
#permalink

I really like and appreciate your post.Thanks Again. Awesome. 먹튀검증업체


Автор: 사업자대출 (не зарегистрирован), дата: 22 ноября, 2020 - 05:59
#permalink

Hello to every one, it’s genuinely a nice for me to pay a visit this
website, it contains important Information. 사업자대출


Автор: Гость, дата: 25 ноября, 2020 - 13:40
#permalink

Some fairytale bliss
Just something I can turn to the impossible quiz


Автор: sex chat manchester (не зарегистрирован), дата: 2 апреля, 2021 - 10:03
#permalink

sex chat manchester is the best platform for casual chat with local girls in Manchester


Автор: casual sex in edinburgh (не зарегистрирован), дата: 15 апреля, 2021 - 13:32
#permalink

If you are ready for hot chat with local ladies in UK check this platform casual sex in edinburgh


Автор: julia2131 (не зарегистрирован), дата: 18 мая, 2021 - 20:39
#permalink

I am happy that I was granted an artisanal analysis by the company. As the handmadewriting reviews stated: the project is excellent quality, the price for everyone are affordable, the service is good.


Автор: Allan Poshel (не зарегистрирован), дата: 20 мая, 2021 - 14:07
#permalink

Hello everyone! The site mail order wives I want to promote today. Because I believe it has become a lot more challenging to find real life in your city and there are many mail order wives with high demands or simply commercial interests. But a decent, caring and real girl may be found on this site. You will certainly discover a female you like because on this site there are millions of accounts of females throughout the world.


Автор: Minecraft Classic (не зарегистрирован), дата: 26 мая, 2021 - 12:15
#permalink

Minecraft Classic certainly will not disappoint you about the challenges that the game brings. There is nothing better than playing fun and useful games with children, right? Minecraft Classic


Автор: essex sex chat (не зарегистрирован), дата: 7 июня, 2021 - 16:15
#permalink

Visit the most popular web platform for free casual sex contacts with hot ladies in UK - essex sex chat


Автор: Casual Sex Ads (не зарегистрирован), дата: 16 июня, 2021 - 07:46
#permalink

Casual Sex Ads is the best web place for hot chat with young girls in United Kingdom, so check out right now


Автор: horny mature women (не зарегистрирован), дата: 16 июня, 2021 - 09:03
#permalink

You can find fine British ladies ready for casual sexy chat with you at our web platform horny mature women


Автор: jained (не зарегистрирован), дата: 22 июня, 2021 - 18:15
#permalink

You'll effectively prepare for all your examinations on the https://myexamcoach.com website. You have a lot of topics to assist prepare for your examinations. I propose you go to their website since I think they have skilled specialists. I wish all of them successfully pass their examinations.


Автор: Sex in East England (не зарегистрирован), дата: 29 июня, 2021 - 08:17
#permalink

If you are looking for sexy chat contacts with hot ladies in UK you must to visit Sex in East England


Автор: leeds casual encounters (не зарегистрирован), дата: 17 июля, 2021 - 18:04
#permalink

leeds casual encounters is the best web place in the United Kingdom for finding hot women for sexy chat


Автор: sepehrsoule (не зарегистрирован), дата: 28 июля, 2021 - 12:16
#permalink

it was great, I really liked it. thanks


Автор: sepehrsoule (не зарегистрирован), дата: 28 июля, 2021 - 12:18
#permalink

شرکت سپهر سوله صفر تا صد ساخت سوله های مرغداری گوشتی را توسط مهندسین مجرب و متخصص انجام می دهد و هزینه های اضافی را حذف و قیمتی منصفانه جهت ساخت و اجرای سوله ارزان به شما پیشنهاد می­دهد. برای کسب اطلاعات بیشتر و هرگونه مشاوره در زمینه ساخت سوله مرغداری می توانید با مجموعه ما تماس حاصل نمایید.

قیمت سوله

سوله مرغداری

پوشش سقف سوله

قیمت استراکچر فلزی


Автор: tak grass (не зарегистрирован), дата: 28 июля, 2021 - 12:23
#permalink

یکی دیگر از هزینه های مربوط به چمن مصنوعی هزینه های مربوط به نصب چمن مصنوعی می باشد. قیمت نصب چمن مصنوعی با توجه به نوع چمن، فضا، متراژ و نوع کاربری که دارد متغییر است. برای نصب چمن مصنوعی تزئینی متریالی چون چسب و نوار اتصال نیاز می باشد، همچنین برای نصب چمن مصنوعی فوتبالی و چمن مصنوعی فوتسال به پرکننده هایی نظیر سیلیس و گرانول احتیاج می باشد. بنابراین علاوه بر قیمت خود محصول می بایست قیمت متریالی نظیر چسب دو جزئی، سیلیس، گرانول، نوار اتصال و همچنین هزینه های باربری آن را در نظر داشت.
قیمت چمن مصنوعی
چمن مصنوعی بالکن
چمن مصنوعی حیاط
خرید چمن مصنوعی


Автор: pamchalpaint (не зарегистрирован), дата: 28 июля, 2021 - 12:24
#permalink

رنگ های دریایی پوشش هایی هستند که بر روی بدنه کشتی ها و سازه های دریایی اعمال می شود تا علاوه بر ایجاد زیبایی ظاهری از آن ها در برابر عوامل مختلف محافظت کند. در ادامه با ما همراه باشید تا با رنگ های دریایی، اهمیت رنگ های دریایی، مزیای رنگ های دریایی، انواع رنگ های دریایی و لزوم لایه های رنگ های دریایی آشنا شوید.
رنگ اپوکسی
رنگ پامچال
رنگ صنعتی
رنگ دریایی
قیمت رنگ صنعتی


Автор: atlaschaman (не зарегистрирован), дата: 28 июля, 2021 - 12:26
#permalink

اولین گام برای خرید چمن مصنوعی تنظیم بودجه برای اینکار است. قیمت چمن مصنوعی و بودجه شما عامل تعیین کننده در نوع چمن مصنوعی هایی است که به آنها نگاه می‌کنید. شما حتما نمی‌خواهید عاشق یک نوع چمن مصنوعی شوید که خرید آن برایتان هزینه سنگینی داشته باشد. بیشتر چمن مصنوعی ها بر اساس متر مربع فروخته می‌شوند. بنابراین با یک دو دوتا چهار تا کردن مبلغی که میتوانید برای خرید هر متر مربع چمن مصنوعی پرداخت کنید را بدست می‌آورید و این مبلغ به در انتخاب نوع چمن مصنوعی در محدوده قیمتی مناسب برای شما، به شما کمک می‌کند. با مقایسه قیمت چمن مصنوعی ها برای نصب چمن مصنوعی مورد نظرتان با نصاب چمن مصنوعی به اندازه بودجه خودتان صحبت کنید.
چمن مصنوعی بالکن
خرید چمن مصنوعی
قیمت چمن مصنوعی


Автор: chapta (не зарегистрирован), дата: 9 сентября, 2021 - 07:01
#permalink

I do not know what to say really what you share very well and useful to the community, I feel that it makes our community much more developed, thanks moto x3m


Автор: live cam sex (не зарегистрирован), дата: 13 сентября, 2021 - 20:26
#permalink

Visit our web platform for the best livecam contacts in Europe live cam sex


Автор: sex dates (не зарегистрирован), дата: 27 сентября, 2021 - 17:01
#permalink

sex dates is the best web place in EU for finding hot girls ready for casual contacts


Автор: sex treffen hamburg (не зарегистрирован), дата: 28 сентября, 2021 - 16:19
#permalink

Your work is very good. Thank you for sharing great information to us. sex treffen hamburg


Автор: Гость (не зарегистрирован), дата: 3 ноября, 2021 - 10:29
#permalink

cool site


Автор: Гость (не зарегистрирован), дата: 3 ноября, 2021 - 10:29
#permalink

cool site


Автор: sextreff bayern (не зарегистрирован), дата: 19 ноября, 2021 - 17:25
#permalink

Check out young hot girls at sextreff bayern for casual contacts in EU


Автор: sextreffen schweiz (не зарегистрирован), дата: 20 ноября, 2021 - 09:26
#permalink

sextreffen schweiz is the best web place for casual contacts with fine ladies in EU


Автор: Гость (не зарегистрирован), дата: 27 декабря, 2021 - 06:06
#permalink

Thanks for sharing. check site


Автор: mary lanada (не зарегистрирован), дата: 5 января, 2022 - 04:25
#permalink

Nice article, Thanks for describing that how to convert video to multiple format.
Check this out


Автор: 안전놀이터추천 (не зарегистрирован), дата: 12 января, 2022 - 12:18
#permalink

Your post is very helpful and information is reliable. I am satisfied with your post. Thank you so much for sharing this wonderful post. If you have any assignment requirement then you are at the right place. 안전놀이터추천


Автор: 안전놀이터추천 (не зарегистрирован), дата: 12 января, 2022 - 12:37
#permalink

Your post is very helpful and information is reliable. I am satisfied with your post. Thank you so much for sharing this wonderful post. If you have any assignment requirement then you are at the right place. 안전놀이터추천


Автор: mochika (не зарегистрирован), дата: 24 января, 2022 - 09:31
#permalink

This website was a lot of fun for me. You are providing a fantastic resource, and you are giving it away for free. temple run


Автор: Гость (не зарегистрирован), дата: 12 апреля, 2022 - 15:29
#permalink

Автор: Andree Michaux (не зарегистрирован), дата: 12 апреля, 2022 - 20:37
#permalink

Автор: Гость (не зарегистрирован), дата: 13 апреля, 2022 - 08:56
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 00:15
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 00:18
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 00:35
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 00:49
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 02:16
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 12:25
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 13:28
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 13:30
#permalink

Автор: Гость (не зарегистрирован), дата: 16 апреля, 2022 - 14:01
#permalink

Автор: Sex in Hörde (не зарегистрирован), дата: 4 мая, 2022 - 19:11
#permalink

Sex in Hörde is web platform created for your best sexy chat experience in EU so check out right now


Автор: Online Sex Chat (не зарегистрирован), дата: 11 мая, 2022 - 21:22
#permalink

Visit our web platform Online Sex Chat and meet your needs by chatting with the best girls in UK


Автор: frenchies (не зарегистрирован), дата: 17 мая, 2022 - 13:35
#permalink

Автор: nova farms (не зарегистрирован), дата: 17 мая, 2022 - 13:36
#permalink

Автор: nova farms (не зарегистрирован), дата: 17 мая, 2022 - 13:36
#permalink

Автор: Tadeja Lesjak (не зарегистрирован), дата: 12 июля, 2022 - 20:37
#permalink

Urgunca je že nekaj časa na voljo in ima veliko naročnikov. Člani so vseh starosti (18-60+) in vse zanima SEX


Автор: Judith Heuts (не зарегистрирован), дата: 15 июля, 2022 - 20:39
#permalink

Rijpe oma is de beste plaats om lokale vrouwen te ontmoeten. Wij maken het voor u moeiteloos gemakkelijk om door onze site te bladeren, hete lokale sexpartners te vinden en dat eerste bericht te sturen. Dan kunt u uw gesprekken van overal voortzetten en de bal aan het rollen houden.


Автор: Гость (не зарегистрирован), дата: 17 июля, 2022 - 14:44
#permalink

Автор: Гость (не зарегистрирован), дата: 17 июля, 2022 - 21:14
#permalink

Автор: Гость (не зарегистрирован), дата: 19 июля, 2022 - 12:26
#permalink

Автор: Гость (не зарегистрирован), дата: 12 августа, 2022 - 13:17
#permalink

Автор: Гость (не зарегистрирован), дата: 12 августа, 2022 - 13:18
#permalink

Автор: Sexe Chat Hauts de France (не зарегистрирован), дата: 16 августа, 2022 - 21:06
#permalink

Sexe Chat Hauts de France is the web platform where you can find hot young ladies for casual contacts in France


Автор: Гость (не зарегистрирован), дата: 23 августа, 2022 - 17:06
#permalink

Автор: Гость (не зарегистрирован), дата: 23 августа, 2022 - 17:06
#permalink

Автор: Гость (не зарегистрирован), дата: 23 августа, 2022 - 17:07
#permalink

Автор: شركة تنظيف في دبي (не зарегистрирован), дата: 30 августа, 2022 - 21:57
#permalink

Автор: gadgets (не зарегистрирован), дата: 5 сентября, 2022 - 20:26
#permalink

380 AUTO 100 GR RNFP REMAN 500 ROUNDS


41 MAG 210 GR FP NEW 50 RDS
44 MAG DESERT EAGLE 240 GR RNFP 100RDS

44 SPECIAL 240 GR RNFP 150 RDS
50 AE 325GR RNFP BIG GRAINS NEW
590M SHOCKWAVE
AMERICAN EAGLE 223
AMERICAN EAGLE 5.7X28MM 40GR FMJ
AMERICAN TACTICAL

AMERICAN TACTICAL IMPORTS
BARNESBARNES 7MMBARNES AMMUNITION 6.5 MM CREEDMOORBARNES VOR TXBARNES VOR-TX 300 AAC BLACKOUT 110 GRCUSTOM SPRINGFIELD HELLCAT/a>
FIOCCHI
FIOCCHI 10 MM 180 GR. FMJ-TC 1250 FPS 50 ROUNDSFIOCCHI 25 AUTO 35 GR XTP JHP
FIOCCHI AMMUNITIONHORNADY 30HORNADY 410 TRIPLE DEFENSE
HORNADY AMMO
MOSSBERG 590A1
PMC BRONZE
PRVI PARTIZAN 308 WIN 180

RECON TACTICAL- WILSON COMBATREMINGTON 700
SIERRA 300<

Автор: gadgets (не зарегистрирован), дата: 5 сентября, 2022 - 20:27
#permalink

(комментарий удалён)


Автор: oscar (не зарегистрирован), дата: 19 сентября, 2022 - 05:05
#permalink

Автор: oscar (не зарегистрирован), дата: 20 сентября, 2022 - 00:53
#permalink

Автор: 카지노사이트 (не зарегистрирован), дата: 20 сентября, 2022 - 05:19
#permalink

There must have been many difficulties in providing this information. 카지노사이트 Nevertheless, thank you for providing such high-quality information.


Автор: Oxycodon kopen zonder recept (не зарегистрирован), дата: 21 сентября, 2022 - 22:09
#permalink

Автор: Oxycontin kopen zonder recept (не зарегистрирован), дата: 21 сентября, 2022 - 22:10
#permalink

Автор: Kaufen Sie Oxycodon ohne Rezept (не зарегистрирован), дата: 21 сентября, 2022 - 22:11
#permalink

Автор: Buy Oxycodone USA (не зарегистрирован), дата: 21 сентября, 2022 - 22:12
#permalink

Автор: Гостьfef (не зарегистрирован), дата: 12 октября, 2022 - 17:42
#permalink

Автор: Гостьfef (не зарегистрирован), дата: 16 октября, 2022 - 00:15
#permalink

Автор: shemale sex (не зарегистрирован), дата: 25 октября, 2022 - 22:43
#permalink

shemale sex for your best casual chat experience with fine ladies in France


Автор: Гостьfef (не зарегистрирован), дата: 29 октября, 2022 - 21:22
#permalink

Автор: ibogaine for sale (не зарегистрирован), дата: 9 ноября, 2022 - 00:46
#permalink

Автор: gamefowl (не зарегистрирован), дата: 13 ноября, 2022 - 01:42
#permalink

Автор: baby jeeter prerolls (не зарегистрирован), дата: 13 ноября, 2022 - 01:42
#permalink

Автор: gghfh (не зарегистрирован), дата: 13 ноября, 2022 - 01:43
#permalink

Автор: Henry Rifles (не зарегистрирован), дата: 13 ноября, 2022 - 01:44
#permalink

Henry Classic Lever Action Rimfire Rifle

Henry Classic Large Loop Lever Action Rimfire Rifle

Henry Big Boy Straight Grip Lever Action Centerfire Rifle

Henry Big Boy Steel Lever Action Centerfire Rifle

 
Henry Big Boy Silver Lever Action Deluxe Engraved Lever Action Centerfire Rifle

Henry Big Boy Silver Lever Action Centerfire Rifle

Henry Big Boy Silver Large Loop Lever Action Centerfire Rifle 45 Colt (Long Colt) 20″ Barrel Blued and American Walnut

Henry Big Boy Side Gate Lever Action Centerfire Rifle

Henry Big Boy Lever Action Centerfire Rifle

Henry Big Boy Deluxe 3rd Edition Lever Action Centerfire Rifle

Henry Big Boy Side Gate Lever Action Centerfire Rifle

Henry Big Boy Lever Action Centerfire Rifle

Henry Big Boy Deluxe 3rd Edition Lever Action Centerfire Rifle

Henry Big Boy Carbine Lever Action Centerfire Rifle

Henry Big Boy All-Weather Side Gate Lever Action Centerfire Rifle

Henry AR-7 Semi-Automatic Rimfire Rifle 22 Long Rifle 16.125″ Barrel True Timber Viper Western Camo

Henry American Beauty Lever Action Rimfire Rifle 22 Long Rifle 20″ Barrel Blued and Walnut

Henry All-Weather Side Gate Lever Action Centerfire Rifle

Henry X Model 410 Bore Lever Action Shotgun 19.8″ Barrel Blued and Black

Henry SS Slug 12 Gauge Single Shot Shotgun 24″ Barrel Blued and Walnut

Henry Single Shot Shotgun Walnut

Henry Single Barrel Youth 20 Gauge Single Shot Shotgun 26″ Barrel Blued and Walnut

Henry Side Gate 410 Bore Lever Action Shotgun 20″ Barrel Blued and Walnut Straight Grip

Henry Load Gate Lever Action Shotgun

Henry Axe Lever Action Shotgun

Henry 12 Gauge Single-Shot Shotgun with Mossy Oak Obsession Finish

Henry 12 Gauge Single-Shot Shotgun with Mossy Oak Obsession Finish
Henry SS Slug 12 Gauge Single Shot Shotgun 24″ Barrel Blued and Walnut
Henry Side Gate 410 Bore Lever Action Shotgun 20″ Barrel Blued and Walnut Straight Grip
Henry Single Shot Shotgun Walnut
Henry Axe Lever Action Shotgun
Henry X Model 410 Bore Lever Action Shotgun 19.8″ Barrel Blued and Black
Henry Single Barrel Youth 20 Gauge Single Shot Shotgun 26″ Barrel Blued and Walnut
Henry Varmint Express Lever Action Rimfire Rifle
Henry Frontier Suppressor Ready Lever Action Rimfire Rifle
Henry Big Boy Silver Large Loop Lever Action Centerfire Rifle 45 Colt (Long Colt) 20″ Barrel Blued and American Walnut
Henry Single Barrel Single Shot Centerfire Rifle 357 Magnum 22″ Barrel Blued and Walnut
Henry Golden Boy Youth Lever Action Youth Rimfire Rifle 22 Long Rifle 16.25″ Barrel Blued and Walnut Straight Grip
Henry Long Ranger Antelope Lever Action Centerfire Rifle 243 Winchester 20″ Barrel Blued and Walnut
Henry Lever Action Youth Rimfire Rifle 22 Long Rifle 16.125″ Barrel Blued and Walnut Straight Grip
Henry Big Boy Steel Lever Action Centerfire Rifle
Henry Big Boy Lever Action Centerfire Rifle
Henry Evil Roy Lever Action Rimfire Rifle
Henry Magnum Express Lever Action Rimfire Rifle 22 Winchester Magnum Rimfire (WMR) 19.25″ Barrel Blued and American Walnut

Henry Big Boy Deluxe 3rd Edition Lever Action Centerfire Rifle


Автор: bqseo (не зарегистрирован), дата: 13 ноября, 2022 - 12:54
#permalink

Автор: bqseo (не зарегистрирован), дата: 13 ноября, 2022 - 12:55
#permalink

Автор: gadgets (не зарегистрирован), дата: 17 ноября, 2022 - 20:11
#permalink

IMR 4895 Smokeless Gun Powder

Hodgdon Hi-Skor 700-X Smokeless Gun PowderAlliant Reloder 22 Smokeless Gun PowderAlliant Steel Smokeless Gun PowderCCI Large Pistol Magnum Primers #350 Box of 1000 (10 Trays of 100)Springfield Armory 1911 Garrison Semi-Automatic PistolSig Sauer P229R Legion Semi-Automatic PistolSmith and Wesson M&P 2.0 9mm 4.25 Inch 15Rds CO CompliantRuger American Pistol 9mm 4.2 Inch 17RdFN 502 Tactical 22LR PistolFN 509 Tactical 9mmCCI 450 primersAccurate 4350 Smokeless Gun PowderAccurate No. 5 Smokeless Gun PowderAccurate 2200 Smokeless Gun PowderAccurate 4064 Smokeless Gun PowderAccurate 1680 Smokeless Gun PowderAccurate 4100Accurate 2015Accurate LT-32 Smokeless Gun PowderAccurate LT-30 Smokeless Gun PowderAlliant Reloder 19Alliant Reloder 33Alliant American Select Smokeless Gun PowderAlliant Reloder 15<

Автор: gadgets (не зарегистрирован), дата: 17 ноября, 2022 - 20:12
#permalink

Accurate Powder XMR-5744 1LB

Accurate Powder - TCM 1lbNo. 7 1lb - Accurate PowderBARNES AMMUNITION 6.5 MM CREEDMOORweatherby vanguard 270sig sauer P322d166 bulletsLapua Brass 7.62x54mm Rimmed Russian (7.62x53mm Rimmed) Box of 100Hornady Outfitter Ammunition 308 Winchester 165 Grain CX Polymer Tip Lead Free Box of 20Remington Core-Lokt Tipped Ammunition 300 Winchester Short Magnum (WSM) 150 Grain Polymer Tip Box of 20Sig Sauer Elite Hunting Copper Ammunition 30-06 Springfield 150 Grain Solid Hollow Point Lead Free Box of 20Winchester Drylok High Velocity Plated Ammunition 10 Gauge 3-1/2" 1-3/8 oz Non-Toxic Steel ShotRemington UMC Ammunition 303 British 174 Grain Full Metal JacketBrowning 1911-380 Black Label Medallion Pro Semi-Automatic PistolSpringfield Armory Garrison 1911 Semi-Automatic Pistolkeltec p15sig p320 xtenbrg9 elitesig sauer p320 spectre compsig p229 prosig sauer p365sig sauer p320 xcompactsig sauer p226 legionGlock 45 Gen 5 9mm 4.02-inch 10Rdssmith and wesson m&p 2.0 9mmglock 19 gen5smith and wesson mp9glock 17 gen 5Sig Sauer P210Ruger American Pistol 9mm 4.2 Inch 17Rd Glock G19xGlock 36 Gen 3sig sauer p938GLOCK G43GLOCK G19XFN AMERICAGLOCK G43XTAURUS G2CCustom Springfield Hellcatsig sauer P322700-X 4lb. - Hodgdon PowderHodgdon #800X 8 Lb. Keghttps://gunsandgadgetscom.com/best-primers-for-handguns/cci 200 large rifle primersWinchester USA Ready Large Rifle Match Primers Box of 1000 (10 Trays of 100)Remington Large Pistol Primers #2-1/2 Box of 1000 (10 Trays of 100)Remington 5 1/2 Primers Small PistolWinchester Small Rifle Primers | 5,000 CountWinchester Small Rifle Primers | 1,000 Countgirsan 1911EAA Girsan MC P35 pistol 4.625″ 15+1 9MMAmerican Tactical FXH-45 MOXIE 45 ACP 1911 Hybrid Pistol308 win magFederal Small Pistol Match PrimersHodgdon Powder Pyrodex RS 1lbHodgdon Titegroup Smokeless Gun PowderHodgdon Varget Smokeless Gun Powder

Автор: Гость (не зарегистрирован), дата: 17 ноября, 2022 - 20:12
#permalink

Remington Premier 209 STS Primers 5000Pcs

Federal LG Magazine Pistol Match PrimersFIOCCHI 209 SHOTSHELL PRIMERS 5000pcsWinchester Large Rifle Magnum Primers #8-1/2M Box of 1000Winchester #209 Shotshell PrimersCCI Percussion Caps #10 Box of 1000 (10 Cans of 100)CCI 11 Magnum Percussion Caps Box of 1000 (10 Cans of 100)cci 209 muzzleloader primersCCI Large Pistol Primers 0012Remington SM Pistol PrimerSWinchester Small Pistol Match PrimersCCI 400 Small Rifle PrimersWinchester WLR Large Rifle Primers 1000CountsRemington Centerfire Primers 6-1/2 Small Rifle Primers Box Of 1000CtsCCI 500 Small Pistol PrimersVihtavuori N32C TIN STAR Smokeless Gun PowderVihtavuori N568 Smokeless Gun Powdervihtavuori powderHodgdon Pyrodex Black Powder Substitute 50 Caliber 50 Grain PelletsHodgdon Pyrodex P Black Powder Substitute 1 lbBlackhorn 209 5lbHodgdon titegroup powder 1lbHodgdon H4895Hodgdon Powders Varget - 8lbsHodgdon H322-8lbHodgdon H4198 8lbsHodgdon Hi-Skor 700X PowderAlliant Unique Smokeless Gun PowderAlliant Reloder 23 Smokeless Gun PowderAlliant Reloder 26 Smokeless Gun PowderHodgdon HS6 Smokeless Gun PowderIMR 4895 Smokeless Gun PowderHodgdon Powder Co Inc Rifle PowderHodgdon Powder - H-4350 8lbWinchester Powder AutoComp 8lbWinchester Powder - 296 4 Lb.Hodgdon Powder - BLC-2 8lbHodgdon Powder - H-4350 1LbHodgdon Powder - H1000 8lbHodgdon Powder - 800-X 8lbHodgdon Powder Pyrodex RS 1lbH4831SC 8lbs - Hodgdon PowderCFE-223 8lbs - Hodgdon PowderH380 8lbs - Hodgdon PowderH4831 1lb - Hodgdon PowderGreen Dot 8lbs - Alliant PowderHerco 8lbs - Alliant PowderRe-16 8lbs - Alliant Powder700-X 4lb. - Hodgdon PowderAlliant Powder - AR-Comp 8lbAlliant Powder - Clay Dot 8lbAlliant Powder - Clay Dot 8lbAlliant Powder - Bullseye 4lbsAlliant Powder - Blue Dot 1 lbAlliant Powder - Black Dot MZ 1lballiant varmintAlliant Powder - Ex.-Lite 1 LbAlliant Powder - Herco 1 LBAlliant Powder - Re-22 1lb.Alliant Powder - Re-16 1lb.Alliant Powder - Re-33 8lb.power pistol powderalliant unique powder>Alliant Powder - Sport PistolBE-86 8 lbs - Alliant PowderRed Dot 8 lbs - Alliant PowderAccurate 2700 Powder - 1LbAccurate 2520 Powder Magpro1lb - Accurate Powder

Автор: Гость (не зарегистрирован), дата: 17 ноября, 2022 - 20:13
#permalink

(комментарий удалён)


Автор: Гость (не зарегистрирован), дата: 17 ноября, 2022 - 20:14
#permalink

380 AUTO 100 GR RNFP REMAN 500 ROUNDS


41 MAG 210 GR FP NEW 50 RDS
44 MAG DESERT EAGLE 240 GR RNFP 100RDS

44 SPECIAL 240 GR RNFP 150 RDS
50 AE 325GR RNFP BIG GRAINS NEW
590M SHOCKWAVE
AMERICAN EAGLE 223
AMERICAN EAGLE 5.7X28MM 40GR FMJ
AMERICAN TACTICAL

AMERICAN TACTICAL IMPORTS
BARNESBARNES 7MMBARNES AMMUNITION 6.5 MM CREEDMOORBARNES VOR TXBARNES VOR-TX 300 AAC BLACKOUT 110 GRCUSTOM SPRINGFIELD HELLCAT/a>
FIOCCHI
FIOCCHI 10 MM 180 GR. FMJ-TC 1250 FPS 50 ROUNDSFIOCCHI 25 AUTO 35 GR XTP JHP
FIOCCHI AMMUNITIONHORNADY 30HORNADY 410 TRIPLE DEFENSE
HORNADY AMMO
MOSSBERG 590A1
PMC BRONZE
PRVI PARTIZAN 308 WIN 180

RECON TACTICAL- WILSON COMBATREMINGTON 700
SIERRA 300
SIERRA 300 WIN MAG 180 GR TIPPED GM
SPEER GOLD DOT 38 SPL
SUPER-X AMMUNITION 32 WIN SPECIAL 170 GRAINTAURUS 4510 JUDGE
TAURUS G2CWINCHESTER 6.5 CREEDMOOR 125GRCOPPEREXTREMEPOINT16? 350 LEGEND PARKERIZED COMPLETE UPPER223 5.56×45 AMMO 55GR FMJ FEDERAL AMERICAN EAGLE (AE223BKX) 1000 ROUND BULK CASE380 ACPREMINGTON UMC AMMUNITION 380 ACP 95 GRAIN FULL METAL JACKET 250 ROUNDS380 AUTO 100 GR RNFP REMAN 500 ROUNDS44 MAG DESERT EAGLE 240 GR RNFP 100RDS44 SPECIAL 240 GR RNFP 150 RDS9MM 9×19 AMMO 124GR FMJ FEDERAL AMERICAN EAGLE(AE9AP)1000ROUNDCASEACCURATE 2230 8LBSALLIANT POWDER – CLAY DOT 8LBAMERICAN EAGLE 223

Автор: STIIIZY PREMIUM INDOOR PRE-ROLLS (не зарегистрирован), дата: 24 ноября, 2022 - 03:32
#permalink

Автор: baby jeeter prerolls (не зарегистрирован), дата: 24 ноября, 2022 - 03:33
#permalink

Автор: baby jeeter prerolls (не зарегистрирован), дата: 24 ноября, 2022 - 03:34
#permalink

Автор: baby jeeter prerolls (не зарегистрирован), дата: 24 ноября, 2022 - 03:34
#permalink

Автор: baby jeeter prerolls (не зарегистрирован), дата: 24 ноября, 2022 - 03:34
#permalink

Автор: Chiara Gotthardt (не зарегистрирован), дата: 1 декабря, 2022 - 21:03
#permalink

Sex Ulm to aplikacja randkowa, którą polecam. Jeśli wiesz bardzo mało o danej osobie, nawiązanie pierwszego kontaktu może być dużym wyzwaniem. Musisz przekopać się przez morze profili, co sprawia, że łatwo jest pominąć osoby, którym być może dałbyś szansę w innych okolicznościach.


Автор: Гость (не зарегистрирован), дата: 2 декабря, 2022 - 16:19
#permalink

Автор: Гость (не зарегистрирован), дата: 2 декабря, 2022 - 16:20
#permalink

Автор: gadgets (не зарегистрирован), дата: 19 декабря, 2022 - 00:21
#permalink

IMR 4895 Smokeless Gun Powder

Hodgdon Hi-Skor 700-X Smokeless Gun PowderAlliant Reloder 22 Smokeless Gun PowderAlliant Steel Smokeless Gun PowderCCI Large Pistol Magnum Primers #350 Box of 1000 (10 Trays of 100)Springfield Armory 1911 Garrison Semi-Automatic PistolSig Sauer P229R Legion Semi-Automatic PistolSmith and Wesson M&P 2.0 9mm 4.25 Inch 15Rds CO CompliantRuger American Pistol 9mm 4.2 Inch 17RdFN 502 Tactical 22LR PistolFN 509 Tactical 9mmCCI 450 primersAccurate 4350 Smokeless Gun PowderAccurate No. 5 Smokeless Gun PowderAccurate 2200 Smokeless Gun PowderAccurate 4064 Smokeless Gun PowderAccurate 1680 Smokeless Gun PowderAccurate 4100Accurate 2015Accurate LT-32 Smokeless Gun PowderAccurate LT-30 Smokeless Gun PowderAlliant Reloder 19Alliant Reloder 33Alliant American Select Smokeless Gun PowderAlliant Reloder 15<>Henry Lever Action Centerfire Rifle

Автор: gadgets (не зарегистрирован), дата: 19 декабря, 2022 - 00:22
#permalink

(комментарий удалён)


Автор: gadgets (не зарегистрирован), дата: 19 декабря, 2022 - 00:23
#permalink

Vihtavuori N150VIHTAVUORI N550vihtavuori n110Vihtavuori N540cci large rifle magnum primersfederal 210m primer209 primersfederal primersWinchester primersremington primersfederal 210 primerswinchester large pistol primerscci 209 primersfederal 215m primerswinchester 209 primerswinchester 209 primerscci 11 percussion capsremington 9 1/2primersremington small rifle primerremington 7 1/2 primerswinchester usa readyremington 300remington ammunitionremington kleanborecci large pistol primersfiocchi primers209 primers209 muzzleloader primersfederal fusion209 primersfederal primerscci 41 primersfederal gold medal matchremington primersfederal 215 primersfederal 205m primerscci 400 primerscci 500 primers7mm 08federal22lr6.5mm308winmagamericantacticalfxh4moxis:sandgadgetscom.com/product/450-bushmaster/" rel="dofollow">450bushmastereaagirsanmcp35girsan1911cz9mmamericantacticalalpha15p322CCI 300 LARGE PISTOL PRIMERS 5000CTSspringfield sa-35sig sauer p365xl spectre compCCI 41 Small Military Rifle Primers 1000CCI Primers 209 Muzzleloading Box of 100Remington Kleanbore Primers #209 Muzzleloading Box of 100Remington Ultimate Muzzleloader Ignition System Box of 24Federal Large Pistol Primers #150 Box of 1000 (10 Trays of 100)Remington Small Rifle Primers #6-1/2 Box of 5000 (50 Trays of 100)Winchester Small Rifle Primers For 5.56mm (NATO-Spec Military Primers #41)Remington 5 1/2 Primers Small PistolRemington Small Pistol Primers #1-1/2 Box of 1000 (10 Trays of 100)

Remington Large Pistol Primers #2-1/2 Box of 1000 (10 Trays of 100)Winchester USA Ready Large Rifle Match Primers Box of 1000 (10 Trays of 100)Federal Premium Gold Medal Small Pistol Magnum Match Primers #200M Box of 1000 (10 Trays of 100)>Remington STS Primers #209 Shotshell Premier 2000PcsRemington EtronX Electronic Primers Box of 1000 (10 Trays of 100)Winchester #209 Large Pistol Primers 5000ctsRemington Premier 209 STS Primers 5000PcsFederal LG Magazine Pistol Match PrimersFIOCCHI 209 SHOTSHELL PRIMERS 5000pcsWinchester Large Rifle Magnum Primers #8-1/2M Box of 1000Winchester #209 Shotshell PrimersCCI Percussion Caps #10 Box of 1000 (10 Cans of 100)CCI 11 Magnum Percussion Caps Box of 1000 (10 Cans of 100)cci 209 muzzleloader primersCCI Large Pistol Primers 0012

Автор: gadgets (не зарегистрирован), дата: 19 декабря, 2022 - 00:25
#permalink

Vihtavuori N150VIHTAVUORI N550vihtavuori n110Vihtavuori N540cci large rifle magnum primersfederal 210m primer209 primersfederal primersWinchester primersremington primersfederal 210 primerswinchester large pistol primerscci 209 primersfederal 215m primerswinchester 209 primerswinchester 209 primerscci 11 percussion capsremington 9 1/2primersremington small rifle primerremington 7 1/2 primerswinchester usa readyremington 300remington ammunitionremington kleanborecci large pistol primersfiocchi primers209 primers209 muzzleloader primersfederal fusion209 primersfederal primerscci 41 primersfederal gold medal matchremington primersfederal 215 primersfederal 205m primerscci 400 primerscci 500 primers7mm 08federal22lr6.5mm308winmagamericantacticalfxh4moxis:sandgadgetscom.com/product/450-bushmaster/" rel="dofollow">450bushmastereaagirsanmcp35girsan1911cz9mmamericantacticalalpha15p322CCI 300 LARGE PISTOL PRIMERS 5000CTSspringfield sa-35sig sauer p365xl spectre compCCI 41 Small Military Rifle Primers 1000CCI Primers 209 Muzzleloading Box of 100Remington Kleanbore Primers #209 Muzzleloading Box of 100Remington Ultimate Muzzleloader Ignition System Box of 24Federal Large Pistol Primers #150 Box of 1000 (10 Trays of 100)Remington Small Rifle Primers #6-1/2 Box of 5000 (50 Trays of 100)Winchester Small Rifle Primers For 5.56mm (NATO-Spec Military Primers #41)Remington 5 1/2 Primers Small PistolRemington Small Pistol Primers #1-1/2 Box of 1000 (10 Trays of 100)

Remington Large Pistol Primers #2-1/2 Box of 1000 (10 Trays of 100)Winchester USA Ready Large Rifle Match Primers Box of 1000 (10 Trays of 100)Federal Premium Gold Medal Small Pistol Magnum Match Primers #200M Box of 1000 (10 Trays of 100)>Remington STS Primers #209 Shotshell Premier 2000PcsRemington EtronX Electronic Primers Box of 1000 (10 Trays of 100)Winchester #209 Large Pistol Primers 5000ctsRemington Premier 209 STS Primers 5000PcsFederal LG Magazine Pistol Match PrimersFIOCCHI 209 SHOTSHELL PRIMERS 5000pcsWinchester Large Rifle Magnum Primers #8-1/2M Box of 1000Winchester #209 Shotshell PrimersCCI Percussion Caps #10 Box of 1000 (10 Cans of 100)CCI 11 Magnum Percussion Caps Box of 1000 (10 Cans of 100)cci 209 muzzleloader primersCCI Large Pistol Primers 0012

Автор: Гость (не зарегистрирован), дата: 21 декабря, 2022 - 17:05
#permalink

(комментарий удалён)


Автор: Гость (не зарегистрирован), дата: 21 декабря, 2022 - 17:07
#permalink

Автор: Гость (не зарегистрирован), дата: 21 декабря, 2022 - 17:08
#permalink

WEATHERBY SA-459 TURKEY SHOTGUN
Weatherby Mark V Accumark Bolt Action Centerfire Rifle 6.5 Creedmoor 24″ Fluted Barrel Black
Weatherby Vanguard High Country Bolt Action Centerfire Rifle 6.5 Creedmoor 24″ Fluted Barrel Flat Dark Earth Cerakote and Camo Monte Carlo
Weatherby Element Turkey 12 Gauge Semi-Automatic Shotgun 22″ Barrel Mossy Oak Bottomland
Weatherby SA-08 Youth 20 Gauge Semi-Automatic Shotgun 24″ Barrel Black
Weatherby 18i 12 Gauge Semi-Automatic Shotgun 28″ Barrel Blued and Walnut
WEATHERBY MARK V BACKCOUNTRY MCMILLAN TAN BOLT ACTION RIFLE – 6.5 CREEDMOOR
WEATHERBY MARK V BACKCOUNTRY TI GRAPHITE BLACK BOLT ACTION RIFLE – 6.5 CREEDMOOR
WEATHERBY VANGUARD MEATEATER EDITION TUNGSTEN CERAKOTE BOLT ACTION RIFLE – 7MM REMINGTON MAGNUM
WEATHERBY MARK V BACKCOUNTRY LEFT HAND MCMILLAN TAN BOLT ACTION RIFLE – 257 WEATHERBY MAGNUM – 26IN
WEATHERBY MARK V BACKCOUNTRY TI LEFT HAND GRAPHITE BLACK BOLT ACTION RIFLE – 300 WEATHERBY MAGNUM – 26IN
https://weatherbyusashop.com/index.php/product-category/shotguns/
WEATHERBY MARK V BACKCOUNTRY TI GRAPHITE BLACK BOLT ACTION RIFLE – 6.5-300 WEATHERBY MAGNUM
WEATHERBY VANGUARD WEATHERGUARD BLACK/BRONZE BOLT ACTION RIFLE – 300 WINCHESTER MAGNUM – 26IN
Weatherby Element Turkey 12 Gauge Semi-Automatic Shotgun 22″ Barrel Mossy Oak Bottomland
WEATHERBY MARK V DELUXE GLOSS WALNUT BOLT ACTION RIFLE – 30-378 WEATHERBY MAGNUM
WEATHERBY MARK V DANGEROUS GAME BROWN/BLACK BOLT ACTION RIFLE – 460 WEATHERBY MAGNUM – 24IN

WEATHERBY MARK V ACCUMARK LIMITED GRAPHITE BLACK BOLT ACTION RIFLE – 257 WEATHERBY MAGNUM
WEATHERBY MARK V WEATHERMARK LIMITED CERAKOTE BLACK BOLT ACTION RIFLE – 257 WEATHERBY MAGNUM – 26IN
Weatherby Vanguard Bolt Action Centerfire Rifle
Weatherby 18i SuperMag Shotgun 12 Gauge 28″ Barrel 3.5″ Chamber, 4-Round Synthetic Stock
WEATHERBY MARK V MEATEATER LEFT HAND TUNGSTEN CERAKOTE BOLT ACTION RIFLE – 6.5-300 WEATHERBY MAGNUM – 26IN
WEATHERBY MARK V ACCUMARK PRO LEFT HAND TUNGSTEN GRAY BOLT ACTION RIFLE – 257 WEATHERBY MAGNUM – 26IN
WEATHERBY MARK V WEATHERMARK LT LEFT HAND FDE/GREEN BOLT ACTION RIFLE – 6.5-300 WEATHERBY MAGNUM – 26IN
WEATHERBY MARK V ACCUMARK ELITE LEFT HAND COYOTE TAN BOLT ACTION RIFLE – 300 WEATHERBY MAGNUM – 26IN
WEATHERBY MARK V ACCUMARK GRAPHITE BLACK CERAKOTE BOLT ACTION RIFLE – 257 WEATHERBY MAGNUM
WEATHERBY MARK V ACCUMARK GRAPHITE BLACK CERAKOTEBOLT ACTION RIFLE – 338-378 WEATHERBY MAGNUM
WEATHERBY MARK V ACCUMARK GRAPHITE BLACK CERAKOTE BOLT ACTION RIFLE – 6.5 WEATHERBY RPM
WEATHERBY ORION SPORTING SHOTGUN
WEATHERBY ORION I OVER AND UNDER SHOTGUN
WEATHERBY MARK V DELUXE GLOSS WALNUT BOLT ACTION RIFLE – 338-378 WEATHERBY MAGNUM
WEATHERBY VANGUARD CAMILLA BLUED/WALNUT BOLT ACTION RIFLE – 7MM-08 REMINGTON


Автор: Гость (не зарегистрирован), дата: 21 декабря, 2022 - 17:08
#permalink

Автор: lopez (не зарегистрирован), дата: 22 декабря, 2022 - 04:10
#permalink

(комментарий удалён)


Автор: Udin (не зарегистрирован), дата: 27 декабря, 2022 - 12:44
#permalink

Hodgdon Hi-Skor 700-X Smokeless Gun PowderAlliant Reloder 22 Smokeless Gun PowderAlliant Steel Smokeless Gun PowderCCI Large Pistol Magnum Primers #350 Box of 1000 (10 Trays of 100)Springfield Armory 1911 Garrison Semi-Automatic PistolSig Sauer P229R Legion Semi-Automatic PistolSmith and Wesson M&P 2.0 9mm 4.25 Inch 15Rds CO CompliantRuger American Pistol 9mm 4.2 Inch 17RdFN 502 Tactical 22LR PistolFN 509 Tactical 9mmCCI 450 primersAccurate 4350 Smokeless Gun PowderAccurate No. 5 Smokeless Gun PowderAccurate 2200 Smokeless Gun PowderAccurate 4064 Smokeless Gun PowderAccurate 1680 Smokeless Gun PowderAccurate 4100Accurate 2015Accurate LT-32 Smokeless Gun PowderAccurate LT-30 Smokeless Gun PowderAlliant Reloder 19Alliant Reloder 33Alliant American Select Smokeless Gun PowderAlliant Reloder 15<>Henry Lever Action Centerfire Rifle

Jasa Convert Tukar Transfer Rate Tinggi Terpercaya Link Ucapan Online Surprise Gift Ulang Tahun Perintah untuk mengganti baris pada html adalah
Shortcut untuk menyimpan workbook pada excel adalah


Автор: NKHU (не зарегистрирован), дата: 30 декабря, 2022 - 22:39
#permalink

Автор: pdf xchange editor crack (не зарегистрирован), дата: 1 января, 2023 - 10:32
#permalink

There is a lot of software that we have to buy but we want to get that software for free. Now you don't have to go to any website and buy the software. Visit our website activation keys now and download Paid software for free. Our website provides you with free software activation keys. You can download paid software for free and also can enjoy the software's premium features. AOMEI Partition Assistant Crack


Автор: pdf xchange editor crack (не зарегистрирован), дата: 1 января, 2023 - 10:32
#permalink

There is a lot of software that we have to buy but we want to get that software for free. Now you don't have to go to any website and buy the software. Visit our website activation keys now and download Paid software for free. Our website provides you with free software activation keys. You can download paid software for free and also can enjoy the software's premium features. AOMEI Partition Assistant Crack


Автор: Buy Counterfeit USD 20 Bills (не зарегистрирован), дата: 5 января, 2023 - 11:45
#permalink

I think you have mentioned some very interesting points, thankyou for the post.

Buy fake USD,AUD,CAD,GBP banknotes online

Buy Counterfeit Money Online


Автор: STIIIZY PODS (не зарегистрирован), дата: 9 января, 2023 - 03:15
#permalink

Автор: jet fuel strain (не зарегистрирован), дата: 9 января, 2023 - 03:16
#permalink

Автор: 카지노사이트추천 (не зарегистрирован), дата: 9 января, 2023 - 06:13
#permalink

What an interesting story! I'm glad I finally found what I was looking for 카지노사이트추천.


Автор: Гость (не зарегистрирован), дата: 19 января, 2023 - 20:10
#permalink

You are searching Factory For Rent in Sahibabad Industrial area site 4? Then you are on the right platform Om Property in Sahibabad industrial area has the list of properties for rent in the Sahibabad Industrial Area Site 4, Detail of the property listed below


Автор: buy ibogaine online (не зарегистрирован), дата: 22 января, 2023 - 18:38
#permalink

Автор: legit online dispensary shipping worldwide (не зарегистрирован), дата: 22 января, 2023 - 18:39
#permalink

Автор: cavapoo puppies for sale under $500 (не зарегистрирован), дата: 22 января, 2023 - 18:40
#permalink

Автор: buy pain killers online (не зарегистрирован), дата: 22 января, 2023 - 18:41
#permalink

Автор: LINUS (не зарегистрирован), дата: 25 января, 2023 - 02:53
#permalink

(комментарий удалён)


Автор: Trans Palermo (не зарегистрирован), дата: 25 января, 2023 - 22:56
#permalink

Trans Palermo is one of the most popular web platforms created for your own sexy contacts in Italy


Автор: qwfdafsddfs (не зарегистрирован), дата: 25 января, 2023 - 23:44
#permalink

Автор: Lisa Angela (не зарегистрирован), дата: 27 января, 2023 - 18:15
#permalink

(комментарий удалён)


Автор: Teds Budz (не зарегистрирован), дата: 27 января, 2023 - 18:49
#permalink

Here at Teds Budz Shop we pride ourselves on supplying the market with legitimate genetics from grassroots cultivators We are a small, family run, black owned business that distributes only small-batch, high quality flower. People trust our credibility and we will never distribute product that we would not personally enjoy ourselves. Enter Shop

Teds Budz
Teds Budz Distro
Teds Budz Distribution Co
Teds Budz Goods New Drops
Teds Budz Products
teds budz seeds
teds budz logo
teds budz uni strain
teds budz owner
teds budz tuna belly
teds budz los angeles
official gooniez
teds budz review
teds budz instagram
teds budz zuckaz

teds budz moo shu
teds budz reddit
ted budz strain
teds budz weedmaps
teds budz animal land
teds budz dahlia


Автор: Teds Budz (не зарегистрирован), дата: 27 января, 2023 - 18:49
#permalink

Here at Teds Budz Shop we pride ourselves on supplying the market with legitimate genetics from grassroots cultivators We are a small, family run, black owned business that distributes only small-batch, high quality flower. People trust our credibility and we will never distribute product that we would not personally enjoy ourselves. Enter Shop

Teds Budz
Teds Budz Distro
Teds Budz Distribution Co
Teds Budz Goods New Drops
Teds Budz Products
teds budz seeds
teds budz logo
teds budz uni strain
teds budz owner
teds budz tuna belly
teds budz los angeles
official gooniez
teds budz review
teds budz instagram
teds budz zuckaz

teds budz moo shu
teds budz reddit
ted budz strain
teds budz weedmaps
teds budz animal land
teds budz dahlia


Автор: Teds Budz (не зарегистрирован), дата: 27 января, 2023 - 18:49
#permalink

Here at Teds Budz Shop we pride ourselves on supplying the market with legitimate genetics from grassroots cultivators We are a small, family run, black owned business that distributes only small-batch, high quality flower. People trust our credibility and we will never distribute product that we would not personally enjoy ourselves. Enter Shop

Teds Budz
Teds Budz Distro
Teds Budz Distribution Co
Teds Budz Goods New Drops
Teds Budz Products
teds budz seeds
teds budz logo
teds budz uni strain
teds budz owner
teds budz tuna belly
teds budz los angeles
official gooniez
teds budz review
teds budz instagram
teds budz zuckaz

teds budz moo shu
teds budz reddit
ted budz strain
teds budz weedmaps
teds budz animal land
teds budz dahlia


Автор: Teds Budz (не зарегистрирован), дата: 27 января, 2023 - 18:49
#permalink

Here at Teds Budz Shop we pride ourselves on supplying the market with legitimate genetics from grassroots cultivators We are a small, family run, black owned business that distributes only small-batch, high quality flower. People trust our credibility and we will never distribute product that we would not personally enjoy ourselves. Enter Shop

Teds Budz
Teds Budz Distro
Teds Budz Distribution Co
Teds Budz Goods New Drops
Teds Budz Products
teds budz seeds
teds budz logo
teds budz uni strain
teds budz owner
teds budz tuna belly
teds budz los angeles
official gooniez
teds budz review
teds budz instagram
teds budz zuckaz

teds budz moo shu
teds budz reddit
ted budz strain
teds budz weedmaps
teds budz animal land
teds budz dahlia


Автор: STIIIZY POD SHOP (не зарегистрирован), дата: 27 января, 2023 - 18:53
#permalink

STIIIZY POD SHOP. is an award-winning cannabis company directly from Cali and it’s best known for its premium cannabis oil vaporizer called STIIIZY, which uses its own pod system. STIIIZY makes and grows its own over 20 different strains and flavors of cannabis concentrate.SHOP WITH US NOW

Stiiizy Pods For Sale Online
Buy Stiiizy Pods Online
Best Stiiizy Pods - Stiiizy Carts - Order Stiiizy Products from Stiiizypod Shop
Stiiizy vape pod flavors
Stiiizy Pod Sativa Flavors
What Are Stiiizy Pods?
Stiiizy Pen
Stiiizy Pod Hybrid and CBD Blends
Stiiizy Battery
Stiiizy Starter Kit
Stiiizy
stiiizypop
stiiizy pods
stiiizy near me
stiiizy disposable
stiiizy battery
stiiizy california
stiiizy pods flavors
stiiizy pods new jersey
stiiizy pod battery
stiiizy pen battery
stiiizy pen case
stiiizy pen battery near me
Is Stizzy a dab pen?
How much is a 1g Stiiizy pod?
Is Stiiizy harmful?
Is Stiiizy different from vaping?
stiiizy pen for sale
stiiizy delivery
stiiizy instagram
stiiizy review
How much does a STIIIZY set cost?
What are STIIIZY pods called?
What strains does STIIIZY have?
Which Stiiizy pod is best for sleep?
What strain gives you giggles?
Is there nicotine in Stiiizy?
How long do Stiiizy disposables last?
What is better than Stiiizy?
stiiizy 1oomg gummy edibles
stiiizy vape store
stiiizy menu
stiiizy instagram
BUY OFFICIAL STIIIZY BIIIG BATTERY
BUY OFFICIAL STIIIZY PORTABLE POWER CASE
BUY LIIIL DISPOSABLES
Are STIIIZY batteries good?
What battery fits STIIIZY pods?
Proper care of your STIIIZY battery
How long do STIIIZY batteries last?
What battery fits STIIIZY pods?
Is it OK to leave a STIIIZY pod in my battery?
Are STIIIZY batteries good?
What kind of batteries does STIIIZY use?
Stiiizy vape pod flavors
Is Stizzy a dab pen?
Stiiizy Pod Sativa Flavors
Stiiizy Pod Indica Flavors
What Are Stiiizy Pods?
Stiiizy Pen
Stiiizy Pod Hybrid and CBD Blends
Does STIIIZY have good flower?
What is STIIIZY Liiit flower?
What is STIIIZY premium?
What is premium flower?
What are Stiiizy extracts?
Is Stiiizy live resin?
What are Stiiizy concentrates?
Does Stiiizy make Rosin pods?
Curated Live Resin
Live Rosin Badder
Live Rosin Jam
Live Resin Pods
Are Stiiizy edibles strong?
Does Stizzy have edibles?
Is Stiiizy only in California?
stiiizy edibles review
stiiizy edibles 100mg
stiiizy edibles reddit
stiiizy edibles ingredients
stiiizy gummies price
stiiizy gummy triangles review
stiiizy edibles nano enhanced review
stiiizy pods
STIIIZY RETAIL PODs SHOP
stiiizy
stiiizy pods
stiiizy pods
stiiizy near me
Buy stiiizy pods
buy stiiizy pods online
stiiizy pods near me
stiiizy pods for sale
stiiizy near me
stiiizy pods for sale online
stiiizy pods flavors
stiiizy pods retailers
stiiizy store
stiiizy near me
stiiizy pods battery
stiiizy pod flavors
stiiizy pod price
how much are stiiizy pods
stiiizy solventless pod
how long do stiiizy pods last
how much do stiiizy pods last
full gram stiiizy pod cost
where to buy stiiizy pods near
stiiizy half gram pod
sativa stiiizy pods
hybrid stiiizy pods
buy stiiizy pods
where can i buy stiiizy pods near me
where to buy stiiizy pods near


Автор: fidels hash hole (не зарегистрирован), дата: 27 января, 2023 - 18:58
#permalink

Автор: Psilocybin Chocolate (не зарегистрирован), дата: 27 января, 2023 - 19:05
#permalink

Psilocybin Chocolate Bars Shop is dedicated to dismantling the stigma surrounding medical magic mushrooms. Our aim is to help educate those who are health curious while at the same time fostering a seamless microdose experience using psilocybin chocolate for our customers from start to finish. PSILOCYBIN CHOCOLATE- ELEVATE YOUR MICRODOSE

Psilocybin Chocolate Bars
Buy Psilocybin Chocolate Bars Online
Psilocybin mushrooms
Stem Chocolate
STEM chocolate bars
STEM Shroom Chocolate Canada
Buy Psilo – Psilocybin Mushroom Gummies Online
Psilocybin Edibles | Order Mushroom Online Today
Buy Psilocybin Edibles Online
magic mushrooms for sale
Buy Psychedelics s Online
Order Magic Mushrooms Online
buy dried magic mushrooms online
Psychedelic Mushroom Chocolate Bar
One up mushroom chocolate bars
Mushroom Chocolate Bars
Psychedelic Mushroom Chocolate bars for sale
Shrooms chocolate for sale
One Up Mushroom Chocolate Bar Where To Buy
Choco shroom available
Buy Psilocybin Chocolate Bars Online – STEM Shroom Chocolate
psilo mushroom
Psilocybin Chocolate Bars
Psilocybin mushrooms
Stem Chocolate
STEM chocolate bars
STEM Shroom Chocolate Canada
STEM Shroom Chocolate Canada
Buy One up bar milk chocolate online
Buy Polka dot Mushroom chocolate bar
Polka Dot Mushroom Chocolate Bar
POLKA DOT PSILOCYBIN MUSHROOM CHOCOLATE BAR REVIEW
POLKA DOT MUSHROOM CHOCOLATE BAR WHERE TO BUY?
BENEFITS OF POLKA DOT MUSHROOM CHOCOLATE BARS
BUY POLKA DOT MUSHROOM CHOCOLATE BAR FOR SALE
Buy Trippy Flip Mushroom Chocolate Bar Online
Trippy Flip Mushroom Cannabis Infused Chocolate Bar
Microdosing Mushroom Chocolates
Order Trippy Flip Milk Mushroom Cookies Online
Where To Buy Trippy Flip Milk Mushroom Cookies Online
Buy Shroom Bros Magic Mushroom Chocolates Online
Buy Wonkabar Milk Chocolate 500mg Online
Buy Bulk Wonka Bars Near Me
Buy Matrix Genetix Chocolate Bars Online
Mushroom Chocolate Smores MATRIX GENETIX
Matrix Bar Magic Mushroom


Автор: Гость (не зарегистрирован), дата: 28 января, 2023 - 02:58
#permalink

Автор: 온라인카지노 (не зарегистрирован), дата: 29 января, 2023 - 10:47
#permalink

I finally found what I was looking for! I'm so happy. 온라인카지노 Your article is what I've been looking for for a long time. I'm happy to find you like this. Could you visit my website if you have time? I'm sure you'll find a post of interest that you'll find interesting.
HJK


Автор: SexyPG168 (не зарегистрирован), дата: 29 января, 2023 - 12:52
#permalink

SexyPG168 MGM99MAGNUM


Автор: dashkenn (не зарегистрирован), дата: 31 января, 2023 - 16:25
#permalink

I love this blog, such a important information, i like your writing skills.
https://discreetarmsdealer.com
https://marijuanabudshop.com
https://buypalletsnearme.com


Автор: 온라인카지노 (не зарегистрирован), дата: 1 февраля, 2023 - 05:27
#permalink

I've been looking for photos and articles on this topic over the past few days due to a school assignment, 온라인카지노 and I'm really happy to find a post with the material I was looking for! I bookmark and will come often! Thanks


Автор: Гость (не зарегистрирован), дата: 1 февраля, 2023 - 18:07
#permalink

(комментарий удалён)


Автор: jesino (не зарегистрирован), дата: 3 февраля, 2023 - 12:04
#permalink

(комментарий удалён)


Отправить комментарий

Приветствуются комментарии:
  • Полезные.
  • Дополняющие прочитанное.
  • Вопросы по прочитанному. Именно по прочитанному, чтобы ответ на него помог другим разобраться в предмете статьи. Другие вопросы могут быть удалены.
    Для остальных вопросов и обсуждений есть форум.
P.S. Лучшее "спасибо" - не комментарий, как все здорово, а рекомендация или ссылка на статью.
Содержание этого поля является приватным и не предназначено к показу.
  • Адреса страниц и электронной почты автоматически преобразуются в ссылки.
  • Разрешены HTML-таги: <strike> <a> <em> <strong> <cite> <code> <ul> <ol> <li> <dl> <dt> <dd> <u> <i> <b> <pre> <img> <abbr> <blockquote> <h1> <h2> <h3> <h4> <h5> <p> <div> <span> <sub> <sup>
  • Строки и параграфы переносятся автоматически.
  • Текстовые смайлы будут заменены на графические.

Подробнее о форматировании

CAPTCHA
Антиспам
12 + 2 =
Введите результат. Например, для 1+3, введите 4.
 
Текущий раздел
Поиск по сайту
Содержание

Учебник javascript

Основные элементы языка

Сундучок с инструментами

Интерфейсы

Все об AJAX

Оптимизация

Разное

Дерево всех статей

Последние комментарии
Последние темы на форуме
Forum