Leo

Leo

С нами с 11 декабря 2015; Место в рейтинге пользователей: #626
Александр Туниеков
04 марта 2020, 23:49
1
+1
Известная проблема :-)
В файле faq.mysql.schema.xml
<field key="question" dbtype="text" length="1023" phptype="text" null="false" default=""/>
phptype=«text» нет такого. Когда-то работало, а сейчас надо phptype=«string». Замените phptype=«text» на phptype=«string» везде в схеме и перегенируйте классы XPDO. Я с помощью migx это делаю. Создание таблиц через MIGX
Владимир
31 января 2019, 01:03
3
0
Спасибо, Евгений!
Может для шаблона кому пригодится(копи/паст):
{block 'title'}
    {'ms2_email_subject_new_manager' | lexicon : $order}
    <ul style="font-size: 15px;">
        <li>Покупатель: [[+address.receiver]]</li>
        <li>Телефон: [[+address.phone]]</li>
        <li>Почта: [[+user.email]]</li>
        <li>Адрес: [[+address.city]], [[+address.street]], [[+address.building]], [[+address.room]]</li>
        <li>Способ доставки: [[+delivery.name]]</li>
        <li>Тип оплаты: [[+payment.name]]</li>
        <li>Комментрарий: [[+address.comment]]</li>
    </ul>
    <hr>
{/block}
Плагин на событие OnFileManagerUpload
<?php
// проверяем нужное событие
if ($modx->event->name != 'OnFileManagerUpload') {return;}

// подключаем phpthumb
require_once MODX_CORE_PATH.'model/phpthumb/phpthumb.class.php';

// параметры загружаемого файла
$file = $modx->event->params['files']['file'];
$directory = $modx->event->params['directory'];

// получаем media source
$ms = $modx->event->params['source'];
if($ms == null){
	return;
}

// настройки media source
$msProperties = $ms->get('properties');
$directory = $msProperties['basePath']['value'].$directory;

// на всякий случай проверяем наличие // и заменяем на /
$directory = str_replace('//', '/', $directory);

// настройки плагина
$config = array(
        $directory => array(
		'src' => array('w' => 1280,'h' => 1024,'zc' => 0,'q' => 80, 'fltr' => 'wmi|/watermark/logo.png|BR|50|5'),
		),
);

// смотрим, что при загрузке не возникло ошибок
if ($file['error'] != 0) {
	//fwrite($handle,"Ошибка при загрузке файлов \n");
	return;
}

$name = $file['name'];
$extensions = explode(',', $modx->getOption('upload_images'));

// проверям, что наша категория задана в настройках плагина
if (array_key_exists($directory, $config)) {
	$config = $config[$directory];
} else {
	return;
}

// путь к файлу, имя файла, расширение
$filename = MODX_BASE_PATH.$directory.$name;
$def_fn = pathinfo($name, PATHINFO_FILENAME);

$ext = pathinfo($name, PATHINFO_EXTENSION);

// проверяем, что расширение файла задано в настройках MODX, как изображение
if (in_array($ext, $extensions)) {
	$sizes = getimagesize($filename);
	$format = substr($sizes['mime'],6);

	// бежим по всем полям массива с конфигом
	foreach($config as $imgKey =>$imgConfig){
		$options = '';
		if($imgKey == 'src'){
		  // для ключа src имя файла совпадает с исходным
		  $imgName = $filename;
		} else {
		// формируем имя файла
		  $imgName = MODX_BASE_PATH.$directory.$def_fn.$imgKey.'.'.$ext;
		}

		// создаем объект phpThumb..
		$phpThumb = new phpThumb();
		// ..и задаем параметры
		$phpThumb->setSourceFilename($filename);
		foreach ($imgConfig as $k => $v) {
			$phpThumb->setParameter($k, $v);
		}

		// генерируем файл
		if ($phpThumb->GenerateThumbnail()) {
			if ($phpThumb->RenderToFile($imgName)) {
			    //$source->renameObject($imgName, $newName);
				// устанавливаем права на файл, это опционально, зависит от сервера
				//chmod($imgName, 0666);
			}
		}
	}
} else { 
    return;
}
Этот плагин выкладывал Безумкин Василий, вот только не помню где.
Не забудьте поменять настройки плагина под ваши потребности. Там не только ватермарк накладывается но и ресайз изображения делается. Вы можете оставить только наложения ватермарка.
Сергей Кепкин
02 ноября 2017, 14:58
3
0
ну поискал бы тут на форуме!!! почему предлагаю это — я уже псыкал разжеванный ответ!!!
еще раз, ок))):
1. создаешь сниппет под названием yaSuperProger
<?php
//умолчальная сортировка по убыванию id
$DetectGetBy = 'id';
$DetectGetDir = 'desc';
//подсветка активного пункта
$DetectGetHL = 'default';
$DetectGetText = 'сначала новинки';

if (!empty($_GET['sort'])) {
	if ($_GET["sort"] == "price_up") { //сортировка по возрастанию цены
		$DetectGetBy = 'Data.price';
		$DetectGetDir = 'asc';
		$DetectGetHL = 'price_up';
		$DetectGetText = 'сначала подешевле';
	}
	if ($_GET["sort"] == "price_down") { //сортировка по убыванию цены
		$DetectGetBy = 'Data.price';
		$DetectGetDir = 'desc';
		$DetectGetHL = 'price_down';
		$DetectGetText = 'сначала подороже';
	}
}

$modx->setPlaceholder('DetectGetBy', $DetectGetBy);
$modx->setPlaceholder('DetectGetDir', $DetectGetDir);
$modx->setPlaceholder('DetectGetHL', $DetectGetHL);
$modx->setPlaceholder('DetectGetText', $DetectGetText);
2. в шаблон страницы категории вставляешь вызов этого сниппета и чанк с кнопками, или упрощенно без чанка так:
[[!yaSuperProger]]
<div class="row box-filter">
<div class="col-sm-12">
	<div class="btn-group">
		<button type="button" class="btn btn-default dropdown-toggle sort" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
							[[+DetectGetText]] <span class="caret"></span>
		</button>
		<ul class="dropdown-menu [[+DetectGetHL]]">
			<li class="first"><a class="sort_default" href="[[~[[*id]]]]">сначала новинки</a></li>
			<li><a  class="sort_price_up"href="[[~[[*id]]]]?sort=price_up">сначала подешевле</a></li>
			<li class="last"><a class="sort_price_down" href="[[~[[*id]]]]?sort=price_down">сначала подороже</a></li>
		</ul>
	</div>
</div>
</div>
[[!pdoPage?
	&element=`msProducts`
	&limit=`12`
	&depth=`0`
	&sortby=`[[+DetectGetBy]]` &sortdir=`[[+DetectGetDir]]`
	&ajaxMode=`default`
]]
Олег
19 апреля 2017, 16:33
1
+4
Я сделал такой компонент, но его нет в магазине.
Могу поделиться, если понравится.
Ответ осуществляется из админки.
Вот ссылка (пример работы): yaguar-m.ru/vopros-i-otvet
Вот так редактируется в админке:
joxi.ru/p27Wz9DHovRa72
Leo
Leo
16 декабря 2016, 13:38
3
0
рабочий вариант, проверено

<?php
define('MODX_API_MODE', true);
require 'index.php';		// Если файл лежит не в корне - здесь нужно указать верный путь
$modx->getService('error','error.modError');
$modx->setLogLevel(modX::LOG_LEVEL_INFO);
$modx->setLogTarget(XPDO_CLI_MODE ? 'ECHO' : 'HTML');

if (!isset($parents)) {$parents = 0;}		// Можно указать список категорий для поиска товаров
if (!isset($resources)) {$resources = '';}	// Можно указать конкретный список товаров

$pdo = $modx->getService('pdoFetch');
$condition = array('parents' => $parents);
$files = $pdo->getCollection('msProduct',
	array(
		'msProduct.class_key' => 'msProduct',
	),
	array(
		'class' => 'msProduct',
		'parents' => $parents,
		'resources' => $resources,
		'innerJoin' => array(
			'msProductFile' => array(
				'alias' => 'msProductFile',
				'on' => array(
					'msProduct.id = msProductFile.product_id',
					'msProductFile.parent' => 0,
					'msProductFile.type' => 'image',
				)
			)
		),
		'select' => array(
			'msProductFile' => 'all'
		),
		'sortby' => 'msProduct.id'
	)
);

echo '<pre>';
echo $pdo->getTime();
foreach ($files as $row) {
	$file = $modx->newObject('msProductFile');	
	$file->fromArray($row, '', true, true);
	
	$children = $file->getMany('Children');
	foreach ($children as $child) {
		$child->remove();
	}
	$file->generateThumbnails();
	
	// Обновляем thumb и image товара
	if ($product = $file->getOne('Product')) {
		$product->updateProductImage();
	}
}

echo microtime(true) - $modx->startTime;
Илья
08 сентября 2015, 22:16
1
0
Для nginx:
if ($host = "www.example.com") {
  rewrite ^ $scheme://example.com$uri permanent;
}
location / {
  try_files $uri $uri/ @modx-rewrite;
}
Наумов Алексей
17 октября 2014, 15:46
4
+2
Пример с MIGx db, в общих чертах, код взят с одного из сайтов кусками, и для вашего сайта его надо будет переработать.

1. Настраиваем MIGx

1.1. Создаем таблицы в БД и объекты
— Ставим MIGx, как написано в документации к нему!
Я назвал этот «полукомпонент» xqa, все имена/префиксы делал с этим названием…
— Открываем MIGx в админке, и на первой вкладке создаем package xqa
— берем эту xml
<?xml version="1.0" encoding="UTF-8"?>
<model package="xqa" baseClass="xPDOObject" platform="mysql" defaultEngine="MyISAM" version="1.1">
	<object class="xQa" table="xqa" extends="xPDOSimpleObject" >
	<field key="resource_id" dbtype="int" precision="11" phptype="integer" null="false" default="0" />
	<field key="author" dbtype="varchar" precision="255" phptype="string" null="false" default="" />
	<field key="author_email" dbtype="varchar" precision="255" phptype="string" null="false" default="" />
	<field key="question" dbtype="text" phptype="string" />
	<field key="answer" dbtype="text" phptype="string" />
	<field key="notify" dbtype="tinyint" precision="1" attributes="unsigned" phptype="integer" null="false" default="0" />
	<field key="published" dbtype="tinyint" precision="1" attributes="unsigned" phptype="integer" null="false" default="0" />
	<field key="createdby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<field key="createdon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="editedby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<field key="editedon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="deleted" dbtype="tinyint" precision="1" attributes="unsigned" phptype="integer" null="false" default="0" />
	<field key="deletedon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="deletedby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<field key="publishedon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="publishedby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<index alias="PRIMARY" name="PRIMARY" primary="true" unique="true">
		<column key="id" collation="A" null="false" />
	</index>
	<aggregate alias="Resource" class="modResource" local="resource_id" foreign="id" cardinality="one" owner="foreign" />
	</object>
</model>
и на вкладке xml schema ждем save, сохраняя ее.

— теперь создаем таблицы в БД на вкладке create Tables и создаем классы на вкладке parse Schema.

Таким образом мы получаем таблицу в БД и xpdo классы для работы с этой таблицей.

Структура моей таблицы:
resource_id - документ, к которому "привязан" вопрос
author - имя автора
author_email - его email
question - текст вопроса
answer - текст ответа
notify - галочка, символизирующая что нужно отправить уведомление автору о том, что не его вопрос появился ответ

---- с колонками ниже MIGx работает САМ, т.е. мы в коде ничего с ними не делаем, они сами заполняются
published
createdby
createdon
editedby
editedon
deleted
deletedon
deletedby
publishedon
publishedby

1.2 Создаем MIGx настройку.
Идем в MIGx — вкладка MIGx (вторая вкладка с настройками)

Делаем новую xqa и по ней правой кнопкой — Экспорт/Импорт, вставляем следующий код:
{
  "formtabs":[
    {
      "MIGX_id":1,
      "caption":"\u0412\u043e\u043f\u0440\u043e\u0441-\u043e\u0442\u0432\u0435\u0442",
      "print_before_tabs":"0",
      "fields":[
        {
          "MIGX_id":1,
          "field":"author",
          "caption":"\u0410\u0432\u0442\u043e\u0440",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":4,
          "field":"author_email",
          "caption":"E-mail",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":2,
          "field":"question",
          "caption":"\u0412\u043e\u043f\u0440\u043e\u0441",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"textarea",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":3,
          "field":"answer",
          "caption":"\u041e\u0442\u0432\u0435\u0442",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"textarea",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":6,
          "field":"notify",
          "caption":"\u0423\u0432\u0435\u0434\u043e\u043c\u0438\u0442\u044c \u0430\u0432\u0442\u043e\u0440\u0430 \u043e\u0431 \u043e\u0442\u0432\u0435\u0442\u0435?",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"listbox",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"\u041d\u0435\u0442==||\u0414\u0430==1",
          "default":""
        },
        {
          "MIGX_id":5,
          "field":"published",
          "caption":"\u041e\u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u043d",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"listbox",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"\u041d\u0435\u0442==||\u0414\u0430==1",
          "default":""
        }
      ]
    }
  ],
  "contextmenus":"update||publish||unpublish||recall_remove_delete",
  "actionbuttons":"addItem||toggletrash",
  "columnbuttons":"",
  "filters":"[]",
  "extended":{
    "migx_add":"\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432\u043e\u043f\u0440\u043e\u0441",
    "formcaption":"",
    "update_win_title":"\u0420\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0432\u043e\u043f\u0440\u043e\u0441",
    "win_id":"xqa",
    "maxRecords":"",
    "addNewItemAt":"top",
    "multiple_formtabs":"",
    "extrahandlers":"",
    "packageName":"xqa",
    "classname":"xQa",
    "task":"",
    "getlistsort":"createdon",
    "getlistsortdir":"DESC",
    "use_custom_prefix":"0",
    "prefix":"",
    "grid":"",
    "gridload_mode":2,
    "check_resid":1,
    "check_resid_TV":"",
    "join_alias":"",
    "has_jointable":"yes",
    "getlistwhere":"",
    "joins":"",
    "cmpmaincaption":"",
    "cmptabcaption":"",
    "cmptabdescription":"",
    "cmptabcontroller":"",
    "winbuttons":"",
    "onsubmitsuccess":"",
    "submitparams":""
  },
  "columns":[
    {
      "MIGX_id":7,
      "header":"ID",
      "dataIndex":"id",
      "width":"",
      "sortable":"false",
      "show_in_grid":"0",
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":6,
      "header":"\u0414\u0430\u0442\u0430 \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u044f",
      "dataIndex":"createdon",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"this.renderDate",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":1,
      "header":"\u0410\u0432\u0442\u043e\u0440",
      "dataIndex":"author",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":8,
      "header":"E-mail",
      "dataIndex":"author_email",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":2,
      "header":"\u0412\u043e\u043f\u0440\u043e\u0441",
      "dataIndex":"question",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":3,
      "header":"\u041e\u0442\u0432\u0435\u0442",
      "dataIndex":"answer",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":9,
      "header":"\u0423\u0432\u0435\u0434\u043e\u043c\u0438\u0442\u044c?",
      "dataIndex":"notify",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"this.renderCrossTick",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":4,
      "header":"\u041e\u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u043d",
      "dataIndex":"published",
      "width":"",
      "sortable":true,
      "show_in_grid":1,
      "renderer":"this.renderCrossTick",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":5,
      "header":"\u0423\u0434\u0430\u043b\u0435\u043d",
      "dataIndex":"deleted",
      "width":"",
      "sortable":"false",
      "show_in_grid":"0",
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    }
  ]
}
Сохраняем и ради интереса жмем редактировать, смотрим какие поля и колонки у нас есть.

2. Делаем стандартную форму на Formit, в которой должны быть поля author, author_email, question. Как вы ее реализуете — ваши заботы.

Что важно для нас, у Formit указываем хук xQaAdd:
&hooks=`email,xQaAdd`
Хук email я тоже использую, что бы пришло администратору сайта уведомление «Добавлен новый вопрос...».

Код хука примерно такой, т.е. мы просто создаем новый объект xQa (вопрос):
$resource_id = $modx->resource->get('id');

$modx->addPackage('xqa', MODX_CORE_PATH.'components/xqa/model/');

$qa = $modx->newObject('xQa');
$qa->set('author', strip_tags($hook->getValue('author')));
$qa->set('author_email', strip_tags($hook->getValue('author_email')));
$qa->set('question', strip_tags($hook->getValue('question')));
$qa->set('resource_id', $resource_id);
$qa->set('published', 0);
$qa->set('createdby', 1);
$qa->set('createdon', strtotime("now"));
$qa->save();

return true;
3. Делаем TV поле с типом ввода migxdb, указываем Конфигурацию xqa (мы ее выше создавали), назначаем TV нужному шаблону.

У ресурса должна появится таблица:


4. Уведомление пользователя об ответе:
Создаем плагин notifyVisitors на событие OnDocFormSave:

<?php
if($modx->event->name != 'OnDocFormSave'){
  return false;
}

$template = $resource->get('template');

switch($template){
  case 7:
  case 15:
    // Вопрос-ответ
    $modx->addPackage('xqa', MODX_CORE_PATH.'components/xqa/model/');
    $qas = $modx->getCollection('xQa', array('deleted' => 0, 'notify' => 1, 'resource_id' => $resource->get('id')));
    foreach($qas as $qa){
      $to = $qa->get('author_email');
      if(!empty($to)){
	// ЗДЕСЬ отправляем письмо, объект $qa у нас есть
      }
      $qa->set('notify', 0);
      $qa->save();
    }
    break;
  default:
    return false;
}
Что делает плагин: при сохранении ресурса получаем все xQa, у которых стоит галочка «Оповестить автора об ответе», далее, например, отправляем письмо (я НЕ ПРИВОЖУ этот код, напишите его сами), ставим notify в 0.
Evgeny Epifanov
23 сентября 2014, 22:05
8
+1
Чанк tpl.msEmail.new.manager
[[+address.receiver]] — Покупатель
[[+address.phone]] — Телефон
[[+user.email]] — Почта
[[+address.index]] — Индекс
[[+address.region]] — Область
[[+address.city]] — Город
[[+address.street]] — Улица
[[+address.building]] — Дом
[[+address.room]] — Квартира
[[+delivery.name]] — Способ доставки
[[+payment.name]] — Тип оплаты
[[+address.comment]] — Комментрарий

Это то, что я знаю. А вообще где-то это уже обсуждалось.