formNews.php 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981
  1. <?php
  2. use app\assets\GalleryManagerAsset;
  3. use manager\models\News;
  4. use yii\helpers\Html;
  5. use yii\helpers\ArrayHelper;
  6. use yii\widgets\ActiveForm;
  7. use kartik\datetime\DateTimePicker;
  8. use \app\models\base\Tags;
  9. use \app\models\base\Story;
  10. use \app\models\base\Image;
  11. use \app\models\base\NewsTopic;
  12. use app\models\Authors;
  13. use app\widgets\CropperjsWidget;
  14. use zxbodya\yii2\galleryManager\GalleryManager;
  15. use \app\models\Tagsfilter;
  16. /**
  17. * @var $news News
  18. */
  19. GalleryManagerAsset::register($this);
  20. $cache = Yii::$app->cache;
  21. $tmodel = new Tags();
  22. $session = Yii::$app->session;
  23. $session->open();
  24. $tagsmenu = Yii::$app->cache->getOrSet("tagsmenu",function () use($tmodel){
  25. return $tmodel->find()->rightJoin(['m'=>Tagsfilter::find()], 'm.id = tags.id')->orderBy('sort')->All();
  26. });
  27. //echo $news->text;
  28. ?>
  29. <ul class="nav nav-tabs">
  30. <li class="nav-item">
  31. <a class="nav-link active" data-toggle="tab" href="#edit">Статья</a>
  32. </li>
  33. <li class="nav-item">
  34. <a class="nav-link" data-toggle="tab" href="#comments">Комментарии</a>
  35. </li>
  36. </ul>
  37. <div class="tab-content">
  38. <div class="tab-pane fade show active" id="edit">
  39. <?
  40. // редактор
  41. $form = ActiveForm::begin([
  42. 'id' => 'news-form',
  43. 'options' => ['class' => 'form-horizontal','enctype' => 'multipart/form-data'],]);
  44. //иничиализация
  45. DateTimePicker::widget([
  46. 'name' => 'dt_pub',
  47. 'options' => [],
  48. 'convertFormat' => true,
  49. 'pluginOptions' => [
  50. 'format' => 'd-M-Y H:i ',
  51. 'startDate' => '01-Mar-2000 12:00',
  52. 'todayHighlight' => true
  53. ]
  54. ]);
  55. $tcount = 85;
  56. if( mb_strlen( $news->title ) > 85 ){
  57. $tcount = mb_strlen( $news->title );
  58. }
  59. $model = new Authors();
  60. $authors = $model->findAll(['active' => 'Y']);
  61. //$author = $model->findOne(['id' => $news->author]);
  62. $author = $model->getAuthor($news->author);
  63. $ids = ArrayHelper::getColumn($authors, 'id');
  64. if( !in_array($author->id, $ids) ){
  65. $authors = array_merge($authors, [$author]);
  66. };
  67. $news->commerc = ($news->inscription == 2)?'Y':'N';
  68. $news->mcommerc = ($news->inscription == 1)?'Y':'N';
  69. $news->pcommerc = ($news->inscription == 3)?'Y':'N';
  70. $news->fdzen = ($news['export_rss'] & 1)?'Y':'N';
  71. $news->fmail = ($news['export_rss'] & 2)?'Y':'N';
  72. $nofirst = ($news->flags & 1);
  73. if( isset($uuid) )
  74. {
  75. $news->uid = $uuid;
  76. }else{
  77. $uuid = $news->uid;
  78. }
  79. if($news->isNewRecord)
  80. {
  81. ?>
  82. <div class="info-box">
  83. <span class="info-box-icon bg-info"><i class="far fa-bookmark"></i></span>
  84. <div class="info-box-content">
  85. <span class="info-box-text">Сгенерировал. UUID</span>
  86. <span class="info-box-number"><?=$uuid?></span>
  87. <div class="progress">
  88. <div class="progress-bar bg-info" style="width: 70%"></div>
  89. </div>
  90. <span class="progress-description">
  91. Информация для отладки
  92. </span>
  93. </div>
  94. </div>
  95. <?
  96. }else{
  97. ?>
  98. <div class="info-box">
  99. <span class="info-box-icon bg-info"><i class="far fa-bookmark"></i></span>
  100. <div class="info-box-content">
  101. <span class="info-box-text">ID Новости</span>
  102. <span class="info-box-number"><?=$news->id?></span>
  103. <span class="info-box-text">uid Новости</span>
  104. <span class="info-box-number"><?=$uuid?></span>
  105. <div class="progress">
  106. <div class="progress-bar bg-info" style="width: 70%"></div>
  107. </div>
  108. <span class="progress-description">
  109. Информация для отладки
  110. </span>
  111. </div>
  112. </div>
  113. <input type="hidden" name="News[id]" value="<?=$news->id?>">
  114. <input type="hidden" name="News[alias]" value="<?=$news->alias?>">
  115. <input type="hidden" name="News[photo_name]" id="photo_name" value="<?=$news->photo_name?>">
  116. <?
  117. if( $res = $cache->get("editNews_".$news->id) ){
  118. if( isset( unserialize($res)['author'] ) && unserialize($res)['author'] != Yii::$app->user->identity->profile->name){
  119. //Yii::$app->user->identity->profile->name //self
  120. /*
  121. * Предупреждение о пути истинного
  122. */
  123. ?>
  124. <div class="modal" tabindex="-1" data-show="true" id="editing">
  125. <div class="modal-dialog">
  126. <div class="modal-content">
  127. <div class="modal-header">
  128. <h5 class="modal-title">Заблокированная новость</h5>
  129. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  130. <span aria-hidden="true">&times;</span>
  131. </button>
  132. </div>
  133. <div class="modal-body">
  134. <p>Новость уже открыта в другом окне автором: <?=unserialize($res)['author']?></p>
  135. </div>
  136. <div class="modal-footer">
  137. <a href="/manager/news/list"><button type="button" class="btn btn-secondary">Покинуть новость</button></a>
  138. <button type="button" class="btn btn-primary" data-dismiss="modal">Всё равно войти в новость</button>
  139. </div>
  140. </div>
  141. </div>
  142. </div>
  143. <script>$(function () {$('#editing').modal('show')});</script>
  144. <?
  145. }
  146. }else{
  147. ?>
  148. <script>
  149. $(function () {
  150. $.get( "newsping", { id: <?=$news->id?> }, function(data){} );
  151. });
  152. </script>
  153. <?
  154. }
  155. }
  156. if( $news->errors ){
  157. ?>
  158. <div class="card card-danger">
  159. <div class="card-header">
  160. <h3 class="card-title">Обнаружены ошибки</h3>
  161. <div class="card-tools">
  162. <span class="badge badge-dark">Error</span>
  163. </div>
  164. <!-- /.card-tools -->
  165. </div>
  166. <!-- /.card-header -->
  167. <div class="card-body">
  168. <?=$form->errorSummary($news)?>
  169. </div>
  170. <!-- /.card-body -->
  171. </div>
  172. <!-- /.card -->
  173. <?
  174. }
  175. $_SESSION['ckfinder_url'] = '/images/upload/'.(string)$news->uid;
  176. $_SESSION['KCFINDER']['admin_news'] = true;
  177. ?>
  178. <input type="hidden" name="News[editors]" value="<?=$news->editors?>">
  179. <input type="hidden" name="News[uid]" value="<?=$news->uid?>">
  180. <div class="row"><div class="col">
  181. <div class="row justify-content-between">
  182. <div class="input-group mb-3 col-4">
  183. <div class="input-group-prepend">
  184. <span class="input-group-text" id="basic-addon1" onclick="jQuery('#dt_pub').datetimepicker('show')"><i class="fa fa-calendar"></i></span>
  185. </div>
  186. <input type="text" id="dt_pub" class="col-auto" placeholder="Дата публикации" aria-label="Дата публикации" aria-describedby="basic-addon1" value="<?=date("d-m-Y H:i", strtotime( $news->dt_pub ) )?>" name="News[dt_pub]">
  187. </div>
  188. <?
  189. if( $news->dt_cr ){
  190. ?>
  191. <div class="col-auto mr-md-auto">
  192. <small class="badge badge-secondary">Создан: <i class="far fa-clock"></i> <?=date("d-m-Y H:i", strtotime( $news->dt_cr ) )?></small> <small class="badge badge-secondary">Обновлён: <i class="far fa-clock"></i> <?=date("d-m-Y H:i", strtotime( $news->dt_upd ) )?></small>
  193. </div>
  194. <?
  195. }
  196. ?>
  197. <div class="col-4">
  198. <?/*
  199. 0 => 'НОВОСТЬ',
  200. 1 => 'ВОПРОС ДНЯ',
  201. 2 => 'СТАТЬЯ',
  202. 3 => 'ИНТЕРВЬЮ',
  203. 4 => 'ССЫЛКА',
  204. 5 => 'ПОДКАСТ',
  205. 6 => 'ЛОНГРИД',
  206. 7 => 'ВИДЕО',
  207. */?>
  208. <select class="form-control" name="News[type]">
  209. <option value="0" <?=($news->type == 0)?'selected':''?>>НОВОСТЬ</option>
  210. <option value="5" <?=($news->type == 5)?'selected':''?>>Подкаст</option>
  211. <option value="7" <?=($news->type == 7)?'selected':''?>>Видео</option>
  212. <option value="6" <?=($news->type == 6)?'selected':''?>>Лонгрид</option>
  213. <option value="2" <?=($news->type == 2)?'selected':''?>>Разворот</option>
  214. <option value="4" <?=($news->type == 4)?'selected':''?>>Ссылка</option>
  215. <option value="8" <?=($news->type == 8)?'selected':''?>>Проект из Tilda</option>
  216. </select>
  217. </div>
  218. </div>
  219. <?
  220. /*
  221. echo $form->field($news, 'title')->textInput([
  222. 'maxlength' => true,
  223. 'class' => 'form-control js-word-count-input',
  224. 'placeholder' => 'Заголовок новости'
  225. ])->label('Заголовок');
  226. _errors:yii\base\Model:private
  227. print_a($model->errors);
  228. */
  229. ?>
  230. <div class="form-group field-news-title required">
  231. <label class="control-label col-form-label" for="news-title">Заголовок</label>
  232. <div class="row">
  233. <input type="text" maxlength="<?=$tcount?>" id="news-title" class="form-control js-word-count-input col-sm-10 ml-2" name="News[title]" placeholder="Заголовок новости" aria-required="true" value='<?=str_replace("'", '&apos;', $news->title)?>'>
  234. <div class="help-block px-lg-2"><div class="news__input-word-count"><span class="badge badge-info" id="title_count"></span></div></div>
  235. <a class="btn btn-primary px-lg-2" data-toggle="collapse" href="#collapseSEOTitle" role="button" aria-expanded="false" aria-controls="collapseSEOTitle">SEO title</a>
  236. <div class="help-block"></div>
  237. </div>
  238. </div>
  239. <div class="collapse show" id="collapseSEOTitle">
  240. <div class="card card-body py-1">
  241. <input type="text" name="News[meta_title]" maxlength="<?=$tcount?>" value="<?=htmlentities($news->meta_title)?>" placeholder="SEO Title" class="form-control js-word-count-input col-sm-10 ml-1 py-0">
  242. <div class="help-block">Поле для альтернативного SEO Title заголовка, если не заполнять совпадёт с основным заголовком</div>
  243. </div>
  244. </div>
  245. <div class="form-group field-news-lid required">
  246. <label class="control-label" for="news-lid">Новость кратко</label> <span class="badge badge-info" id="lid_count"></span>
  247. <textarea id="news-lid" class="form-control js-word-count-input" name="News[lid]" placeholder="Лид" aria-required="true"><?=$news->lid?></textarea>
  248. <div class="help-block">Краткое содержание статьи</div>
  249. </div>
  250. <div class="form-group">
  251. <label class="control-label" for="news-link">Прямой переход на URL</label>
  252. <div class="row">
  253. <input type="text" class="form-control" name="News[link]" id="news-link" value="<?=$news->link?>">
  254. <div class="help-block"></div>
  255. </div>
  256. </div>
  257. <div class="form-group">
  258. <label class="control-label" for="news-embed">вставка медиа URL</label>
  259. <div class="row">
  260. <input type="text" class="form-control" name="News[embed_url]" id="news-embed" value="<?=$news->embed_url?>">
  261. <div class="help-block"></div>
  262. </div>
  263. </div>
  264. <div class="form-group">
  265. <div class="card-primary" id="galleriesContainer" style="border: 1px solid gray;">
  266. <div class="card-header">
  267. <h3 class="card-title">Галлереи</h3>
  268. </div>
  269. <div class="card-body">
  270. <?php if($news->isNewRecord):?>
  271. Чтобы добавить галлерею сохраните набросок новости
  272. <?php else:?>
  273. <div class="galleries-list">
  274. <?php foreach ($news->galleries as $gallery){ echo $this->render('@manager/views/news/form/galleryItem',["model"=>$gallery]);} ?>
  275. </div>
  276. <div class="form-group">
  277. <button type="button" class="btn btn-success" data-toggle="modal" data-target="#addGalleryModal"><i class="fas fa-plus"></i> Добавить галлерею</button>
  278. </div>
  279. <?php endif;?>
  280. </div>
  281. </div>
  282. </div>
  283. <div class="form-group field-js_news_content required">
  284. <label class="control-label" for="js_news_content">Контент</label>
  285. <textarea id="js_news_content" class="form-control" name="News[text]" rows="12" aria-required="true"><?=htmlentities($news->text)?></textarea>
  286. </div>
  287. <blockquote class="quote-info mt-0">
  288. <h5 id="tip">Новость подготовили</h5>
  289. <p><?=str_replace(',', ', ', $news->editors)?></p>
  290. </blockquote>
  291. <div class="input-group">
  292. <label class="control-label col-form-label mr-3" >Автор</label>
  293. <select class="custom-select custom-select-sm" name="News[author]" id="news-author">
  294. <option value="0" disabled selected hidden>Выберете основного автора</option>
  295. <?
  296. foreach( $authors as $item ){
  297. ?>
  298. <option value="<?=$item->id?>" <?=($item->id == $author->id)?'selected':''?>><?=$item->name?></option>
  299. <?
  300. }
  301. ?>
  302. </select>
  303. <label class="input-group-append btn btn-primary" title="показывать автора" for="news-author">
  304. <input type="checkbox" autocomplete="off" name="News[show_author]"<?=($news->show_author == 'Y')?' checked':''?> value="Y"><i class="fa fa-eye ml-2"></i>
  305. </label>
  306. </div>
  307. <div class="card card-outline card-primary">
  308. <div class="card-body">
  309. <div class="custom-control custom-checkbox custom-control-inline">
  310. <input type="checkbox" id="customRadioInline1" name="News[verifed]" class="custom-control-input"<?=($news->verifed == 'Y')?' checked':''?> value="Y">
  311. <label class="custom-control-label" for="customRadioInline1"><i class="fas fa-spell-check"></i>Корректор</label>
  312. </div>
  313. <div class="custom-control custom-checkbox custom-control-inline">
  314. <input type="checkbox" id="customRadioInline2" name="News[noindex]" class="custom-control-input"<?=($news->noindex == 'Y')?' checked':''?> value="Y">
  315. <label class="custom-control-label" for="customRadioInline2"><s><i class="fas fa-search"></i></s>Не отдавать в поиск</label>
  316. </div>
  317. <div class="custom-control custom-checkbox custom-control-inline">
  318. <input type="checkbox" id="customRadioInline4" name="News[fdzen]" class="custom-control-input"<?=($news->fdzen == 'Y')?' checked':''?> value="Y">
  319. <label class="custom-control-label" for="customRadioInline4"><i class="fas fa-yin-yang"></i>Не отдавать в <strong class="text-indigo">DZEN</strong> <span class="text-danger">новости</span></label>
  320. </div>
  321. <div class="custom-control custom-checkbox custom-control-inline">
  322. <input type="checkbox" id="customRadioInline4m" name="News[fmail]" class="custom-control-input"<?=($news->fmail == 'Y')?' checked':''?> value="Y">
  323. <label class="custom-control-label" for="customRadioInline4m"><i class="fas fa-yin-yang"></i>В <strong class="text-indigo">DZEN</strong> <span class="text-danger">канал</span> как черновик</label>
  324. </div>
  325. <div class="custom-control custom-checkbox custom-control-inline">
  326. <input type="checkbox" id="customRadioInline3" name="News[comments]" class="custom-control-input"<?=($news->comments == 'N')?' checked':''?> value="N">
  327. <label class="custom-control-label" for="customRadioInline3"><i class="fas fa-comment-slash"></i>Без коментариев</label>
  328. </div>
  329. <div class="custom-control custom-checkbox custom-control-inline">
  330. <input type="checkbox" id="nofirst" name="News[nofirst]" class="custom-control-input"<?=($nofirst)?' checked':''?> value="Y">
  331. <label class="custom-control-label" for="nofirst"><svg xmlns="http://www.w3.org/2000/svg" width="14px" height="14px" viewBox="0 0 512 512"><path d="M245.8 220.9c-14.5-17.6-21.8-39.2-21.8-60.8C224 80 320 0 416 0c53 0 96 43 96 96c0 96-80 192-160.2 192c-21.6 0-43.2-7.3-60.8-21.8L54.6 502.6c-12.5 12.5-32.8 12.5-45.3 0s-12.5-32.8 0-45.3L245.8 220.9z"/></svg>Спрятать на главной</label>
  332. </div>
  333. </div>
  334. </div>
  335. <div class="card card-outline card-teal">
  336. <div class="card-body">
  337. <div class="custom-control custom-checkbox custom-control-inline">
  338. <input type="checkbox" name="News[photo]" id="fcont1" class="custom-control-input"<?=($news->photo == 'Y')?' checked':''?> value="Y">
  339. <label class="custom-control-label" for="fcont1"><i class="fa fa-camera"></i> C фото материалом</label>
  340. </div>
  341. <div class="custom-control custom-checkbox custom-control-inline">
  342. <input type="checkbox" name="News[video]" id="fcont2" class="custom-control-input"<?=($news->video == 'Y')?' checked':''?> value="Y">
  343. <label class="custom-control-label" for="fcont2"><i class="fa fa-video"></i> C видео материалом</label>
  344. </div>
  345. <div class="custom-control custom-checkbox custom-control-inline">
  346. <input type="checkbox" name="News[audio]" id="fcont3" class="custom-control-input"<?=($news->audio == 'Y')?' checked':''?> value="Y">
  347. <label class="custom-control-label" for="fcont3"><i class="fa fa-headphones"></i> C аудио материалом</label>
  348. </div>
  349. </div>
  350. </div>
  351. <div class="card card-outline card-olive">
  352. <div class="card-body">
  353. <div class="custom-control custom-radio custom-control-inline">
  354. <input type="radio" name="News[commerc]" id="fcont44" class="custom-control-input"<?=($news->commerc == 'N' && $news->mcommerc == 'N')?' checked':''?> value="0">
  355. <label class="custom-control-label" for="fcont44"><i class="fas fa-comment-dollar"></i> Не коммерческая</label>
  356. </div>
  357. <div class="custom-control custom-radio custom-control-inline">
  358. <input type="radio" name="News[commerc]" id="fcont4" class="custom-control-input"<?=($news->commerc == 'Y')?' checked':''?> value="2">
  359. <label class="custom-control-label" for="fcont4"><i class="fas fa-comment-dollar"></i> Простая коммерческая</label>
  360. </div>
  361. <div class="custom-control custom-radio custom-control-inline">
  362. <input type="radio" name="News[commerc]" id="fcont5" class="custom-control-input"<?=($news->mcommerc == 'Y')?' checked':''?> value="1">
  363. <label class="custom-control-label" for="fcont5"><i class="fas fa-comment-dollar"></i> Медицинская коммерческая</label>
  364. </div>
  365. <div class="custom-control custom-radio custom-control-inline">
  366. <input type="radio" name="News[commerc]" id="fcont15" class="custom-control-input"<?=($news->pcommerc == 'Y')?' checked':''?> value="3">
  367. <label class="custom-control-label" for="fcont15"><i class="fas fa-comment-dollar"></i> Партнёр</label>
  368. </div>
  369. <div class="custom-control custom-control-inline p-0">
  370. <button type="button" class="btn btn-secondary btn-sm" onClick="javascript:$('#modalcsl').modal('show');" title="rocket boost"><i class="fas fa-space-shuttle"></i></button>
  371. </div>
  372. <div class="custom-control custom-control-inline pl-1">
  373. <label class="control-label my-0 mr-2" for="news-mark">Метка <small class="form-text text-muted my-0">(токен)</small></label>
  374. <input type="text" class="form-control" name="News[mark]" id="news-mark" value="<?=$news->mark?>">
  375. </div>
  376. </div>
  377. </div>
  378. <div class="card card-outline card-maroon">
  379. <div class="card-body">
  380. <div class="custom-control custom-checkbox custom-control-inline">
  381. <input type="checkbox" name="News[top]" id="fcont6" class="custom-control-input"<?=($news->top == 'Y')?' checked':''?> value="Y">
  382. <label class="custom-control-label" for="fcont6"><i class="fas fa-bomb"></i> В КД</label>
  383. </div>
  384. <div id="grpnh" class="custom-control-inline">
  385. <div class="custom-control custom-checkbox custom-control-inline">
  386. <input type="checkbox" name="News[NH]" id="fcont7" class="custom-control-input"<?=($news->NH == 'Y')?' checked':''?> value="Y">
  387. <label class="custom-control-label" for="fcont7"><i class="fas fa-rocket"></i> В НЧ</label>
  388. </div>
  389. <div class="custom-control custom-checkbox custom-control-inline">
  390. <input type="checkbox" name="News[NH]" id="fcont77" class="custom-control-input"<?=($news->NH == 'F')?' checked':''?> value="F">
  391. <label class="custom-control-label" for="fcont77"><i class="fas fa-thumbtack"></i> В НЧ (закрепить)</label>
  392. </div>
  393. </div>
  394. </div>
  395. </div>
  396. <?
  397. if(!$news->isNewRecord)
  398. {
  399. $Tagitems = $tmodel->getForNews($news->id);
  400. }else{
  401. $Tagitems = [];
  402. }
  403. ?>
  404. <div class="card card-outline card-warning">
  405. <div class="card-body">
  406. <label for="tags"><i class="fas fa-tags"></i>Тэги </label>
  407. <div class="tags-input" id="myTags">
  408. <span class="data">
  409. <?
  410. foreach($Tagitems as $item){
  411. ?>
  412. <span class="btn btn-sm btn-primary tag"><span class="text" _value="<?=$item->id?>"><?=$item->title?></span><span class="close">&times;</span></span>
  413. <?
  414. }
  415. ?>
  416. </span>
  417. <span class="autocomplete">
  418. <input type="text">
  419. <div class="autocomplete-items">
  420. </div>
  421. </span>
  422. </div>
  423. </div>
  424. </div>
  425. <div class="btn-group">
  426. <label class="btn btn-primary active">
  427. <input type="checkbox" name="News[active]" id="active" style="width:3vw;height:3vh;top: 3px;position: relative;" class="active"<?=($news->active == 'Y')?' checked':''?> value="Y"><i class="far fa-play-circle"></i> Опубликовать</label>
  428. </label>
  429. </div>
  430. <div class="custom-control"><br>
  431. <button type="submit" class="btn btn-success mx-1" name="send" onClick="window.onbeforeunload = null;">Сохранить и выйти</button>
  432. <button type="submit" class="btn btn-success mx-1" name="save" onClick="window.onbeforeunload = null;">Сохранить</button>
  433. <?
  434. if(!$news->isNewRecord){
  435. ?>
  436. <a href="/news/preview/<?=$uuid?>" class="btn btn-info" target="_blank"><i class="fas fa-cat"></i> Предпросмотр</a>
  437. <?
  438. }
  439. ?>
  440. <div class="card card-info my-3">
  441. <div class="card-header">
  442. <h3 class="card-title">Локальные копии автосохранение</h3>
  443. </div>
  444. <div class="card-body text-right">
  445. <button class="btn btn-primary" type="button" onClick="SaveLocal()">Сохранить</button>
  446. <button class="btn btn-primary" type="button" onClick="LoadLocal()">Востановить</button>
  447. </div>
  448. <div class="card-footer">
  449. Копии сохраняются каждые 60сек. Для лучшего эффекта заполните заголовок
  450. </div>
  451. </div>
  452. <?
  453. if($news->isNewRecord){
  454. ?>
  455. <div class="alert alert-success m-4 bg-olive" role="alert">
  456. <h4 class="alert-heading">Для не сохранённых статей!</h4>
  457. <p>Вы ещё ни разу не сохранили документ, по этому нет возможности сделать его предпросмотр.</p>
  458. <hr>
  459. <p class="mb-0">Когда вы сохраните документ, станут доступны дополнительные функсии, в том числе авто сохранение локальной копии.</p>
  460. </div>
  461. <?
  462. }
  463. ?>
  464. </div>
  465. </div>
  466. <!-- правая колонка --->
  467. <div class="col-3">
  468. <div class="card card-primary">
  469. <div class="card-header">
  470. <h3 class="card-title">Фото</h3>
  471. </div>
  472. <div class="card-body">
  473. <style>
  474. #image {
  475. width: 100%;
  476. height: 56%;
  477. background: URL(/img/image.svg) no-repeat 50% 50%;
  478. background-size: contain;
  479. background-color: #fff;
  480. cursor: pointer;
  481. padding: 0px 0px 8px 0px;
  482. }
  483. #myTags {
  484. box-shadow: 0 0 5px rgba(81, 203, 238, 1);
  485. padding: 3px 0px 3px 3px;
  486. margin: 5px 1px 3px 0px;
  487. border: 1px solid rgba(81, 203, 238, 1);
  488. }
  489. </style>
  490. <?
  491. if($news->id){
  492. $imageSrc = $news->getImage()->geturl(Image::SIZE_1040x586);
  493. /*
  494. $imageSrc = "/images/news/news/".$news->id."_sizehd.jpg";
  495. if( file_exists( \Yii::getAlias('@webroot').$imageSrc ) === false ){
  496. if( $news->photo_name != '' ){
  497. $imageSrc = $news->photo_name;
  498. }else{
  499. $imageSrc = '/img/e.gif';
  500. }
  501. }
  502. */
  503. $imageSrc .= "?r=".rand();
  504. }else{
  505. $imageSrc = '/img/e.gif';
  506. }
  507. ?>
  508. <img id="image" src="<?=$imageSrc?>" onClick="$('#modal').modal('show');document.getElementById('inputImage').click();" >
  509. <?
  510. echo CropperjsWidget::widget(['image' => 'image', 'file' => 'News[photo]', 'post' => 'cropping', 'aspectRatio' => '16/9']);
  511. ?><script>
  512. $('#container_image').attr('src', '<?=$imageSrc?>');
  513. </script>
  514. <label class="control-label" for="status-photo">Статусы для фото</label>
  515. <div class="btn-group" data-toggle="buttons" id="status-photo">
  516. <label class="btn btn-primary active btn-sm px-3" title="показывать фото">
  517. <input type="checkbox" autocomplete="off" name="News[photo_include]"<?=($news->photo_include == 'Y')?' checked':''?> value="Y"><i class="fa fa-eye ml-2"></i>
  518. </label>
  519. <label class="btn btn-info active btn-sm px-3" title="фото в пр. ленте">
  520. <input type="checkbox" autocomplete="off" name="News[photo_rcol]"<?=($news->photo_rcol == 'Y')?' checked':''?> value="Y"><i class="fa fa-list-alt ml-2"></i>
  521. </label>
  522. <label class="btn btn-info active btn-sm px-3" title="фото фильтр" onClick="javascript:$('#fphoto').modal('show');">
  523. <i class="fa fa-adjust"></i>
  524. </label>
  525. <label class="btn btn-dark active btn-sm px-3" title="удалить фото">
  526. <input type="checkbox" autocomplete="off" name="News[del_photo]" value="Y"><i class="fa fa-trash ml-2"></i>
  527. </label>
  528. </div>
  529. <div class="form-group field-title-photo required">
  530. <label class="control-label" for="title-photo">Подпись фото (автор и описание) от 33 знаков</label>
  531. <div class="help-block px-lg-2 d-inline"><span class="badge badge-info" id="phototitle_count"></span></div>
  532. <textarea id="title-photo" class="form-control" name="News[photo_title]" rows="1" aria-required="true" <?=($imageSrc == '/img/e.gif')?'':'minlength="33"'?>><?=$news->photo_title?></textarea>
  533. </div>
  534. <button type="button" class="btn btn-primary btn-block" onClick="javascript:$('#sphoto').modal('show');">Поискать</button>
  535. </div>
  536. </div>
  537. <!-- test -->
  538. <div class="card card-primary">
  539. <div class="ribbon-wrapper">
  540. <div class="ribbon bg-lightblue">
  541. SuperTag
  542. </div>
  543. </div>
  544. <div class="card-body bg-gradient-olive">
  545. <?
  546. $tagscheck = [];
  547. foreach( $Tagitems as $setmenuitem ){
  548. $tagscheck[$setmenuitem->id] = true;
  549. }
  550. foreach( $tagsmenu as $menuitem ){
  551. ?>
  552. <div class="custom-control custom-switch">
  553. <input class="custom-control-input" type="checkbox" value="<?=$menuitem->id?>" id="tflexCheck<?=$menuitem->id?>" name="ExtTag[<?=$menuitem->id?>]"<?=isset($tagscheck[$menuitem->id])?' checked':''?>>
  554. <label class="custom-control-label" for="tflexCheck<?=$menuitem->id?>"><?=$menuitem->title?></label>
  555. </div>
  556. <?
  557. }
  558. ?>
  559. </div>
  560. </div>
  561. <!-- /test -->
  562. <div class="card card-primary">
  563. <div class="ribbon-wrapper">
  564. <div class="ribbon bg-primary">
  565. Рубрики
  566. </div>
  567. </div>
  568. <div class="card-body">
  569. <?
  570. $atopic = array();
  571. if(!$news->isNewRecord)
  572. {
  573. $model = new NewsTopic();
  574. $Titems = $model->getForNews($news->id);
  575. foreach( $Titems as $item ){
  576. $atopic[$item->id] = $item->id;
  577. }
  578. }
  579. ?>
  580. <div class="accordion" id="TopicsSel">
  581. <div class="card">
  582. <div class="card-header" id="headingOne">
  583. <h2 class="mb-0">
  584. <button class="btn btn-link btn-block text-left" type="button" data-toggle="collapse" data-target="#collapse1" aria-expanded="true" aria-controls="collapse1">
  585. Общий рубрикатор
  586. </button>
  587. </h2>
  588. </div>
  589. <div id="collapse1" class="collapse show" aria-labelledby="heading1" data-parent="#TopicsSel">
  590. <div class="card-body">
  591. <?
  592. foreach( NewsTopic::findActive()->getModels() as $item ){
  593. if( in_array( $item->id, [127,128]) ) continue; // пропустить разделы видео/подкасты
  594. if( $item->url == '----' ){
  595. ?>
  596. <div class="checkbox_title_hr">
  597. <span><?=$item->title?></span>
  598. </div>
  599. <?
  600. }else{
  601. ?>
  602. <div class="custom-control <?=($item->show == 'Y')?'custom-checkbox':'custom-switch bg-purple'?>">
  603. <input class="custom-control-input" type="checkbox" value="<?=$item->id?>" id="flexCheck<?=$item->id?>" name="Topics[<?=$item->id?>]"<?=(in_array($item->id, $atopic, true))?' checked':''?>>
  604. <label class="custom-control-label" for="flexCheck<?=$item->id?>">
  605. <?=$item->title?>
  606. </label>
  607. </div>
  608. <?
  609. }
  610. }
  611. ?>
  612. </div>
  613. </div>
  614. </div>
  615. <div class="card">
  616. <div class="card-header" id="heading2">
  617. <h2 class="mb-0">
  618. <button class="btn btn-link btn-block text-left" type="button" data-toggle="collapse" data-target="#collapse2" aria-expanded="true" aria-controls="collapse2">
  619. Слушать
  620. </button>
  621. </h2>
  622. </div>
  623. <div id="collapse2" class="collapse" aria-labelledby="heading2" data-parent="#TopicsSel">
  624. <div class="card-body">
  625. <?
  626. foreach( NewsTopic::findActive(127)->getModels() as $item ){
  627. ?>
  628. <div class="custom-control <?=($item->show == 'Y')?'custom-checkbox':'custom-switch bg-purple'?>">
  629. <input class="custom-control-input" type="checkbox" value="<?=$item->id?>" id="flexCheck<?=$item->id?>" name="Topics[<?=$item->id?>]"<?=(in_array($item->id, $atopic, true))?' checked':''?>>
  630. <label class="custom-control-label" for="flexCheck<?=$item->id?>">
  631. <?=$item->title?>
  632. </label>
  633. </div>
  634. <?
  635. }
  636. ?>
  637. </div>
  638. </div>
  639. </div>
  640. <div class="card">
  641. <div class="card-header" id="heading3">
  642. <h2 class="mb-0">
  643. <button class="btn btn-link btn-block text-left" type="button" data-toggle="collapse" data-target="#collapse3" aria-expanded="true" aria-controls="collapse3">
  644. Смотреть
  645. </button>
  646. </h2>
  647. </div>
  648. <div id="collapse3" class="collapse" aria-labelledby="heading3" data-parent="#TopicsSel">
  649. <div class="card-body">
  650. <?
  651. foreach( NewsTopic::findActive(128)->getModels() as $item ){
  652. ?>
  653. <div class="custom-control <?=($item->show == 'Y')?'custom-checkbox':'custom-switch bg-purple'?>">
  654. <input class="custom-control-input" type="checkbox" value="<?=$item->id?>" id="flexCheck<?=$item->id?>" name="Topics[<?=$item->id?>]"<?=(in_array($item->id, $atopic, true))?' checked':''?>>
  655. <label class="custom-control-label" for="flexCheck<?=$item->id?>">
  656. <?=$item->title?>
  657. </label>
  658. </div>
  659. <?
  660. }
  661. ?>
  662. </div>
  663. </div>
  664. </div>
  665. </div>
  666. </div>
  667. </div>
  668. <div class="card card-primary">
  669. <div class="ribbon-wrapper">
  670. <div class="ribbon bg-primary">
  671. сюжеты
  672. </div>
  673. </div>
  674. <div class="card-body">
  675. <div class="input-group">
  676. <div class="input-group-prepend">
  677. <span class="btn btn-primary" id="validatedInputGroupPrepend"><i class="fas fa-search"></i></span>
  678. </div>
  679. <input type="text" class="form-control" aria-describedby="validatedInputGroupPrepend" id="story">
  680. </div>
  681. <?
  682. $model = new Story();
  683. if(!$news->isNewRecord)
  684. {
  685. $Sitems = $model->getForNews($news->id);
  686. }else{
  687. $Sitems = [];
  688. }
  689. ?>
  690. <div class="custom-control custom-checkbox">
  691. <?
  692. foreach($Sitems as $item){
  693. ?>
  694. <div class="custom-control custom-checkbox">
  695. <input class="custom-control-input" type="checkbox" value="<?=$item->id?>" id="Story<?=$item->id?>" name="Story[<?=$item->id?>]" checked>
  696. <label class="custom-control-label" for="Story<?=$item->id?>">
  697. <?=$item->title?>
  698. </label>
  699. </div>
  700. <?
  701. }
  702. ?>
  703. <div id="story_items"></div>
  704. </div>
  705. </div>
  706. </div>
  707. <!-- end col r-->
  708. </div></div>
  709. <?
  710. echo $this->render('modalPhotoFilter', ['imageSrc'=>$imageSrc, 'news'=>$news]);
  711. ActiveForm::end();
  712. if(!$news->isNewRecord){
  713. ?>
  714. <!-- Modal -->
  715. <div class="modal fade" id="addGalleryModal" tabindex="-1" role="dialog" aria-labelledby="exampleModalLabel" aria-hidden="true">
  716. <div class="modal-dialog" role="document">
  717. <div class="modal-content">
  718. <form action="" class="gallery-form" id="galleryForm">
  719. <div class="modal-header">
  720. <h5 class="modal-title" id="exampleModalLabel">Добавление галлереи</h5>
  721. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  722. <span aria-hidden="true">&times;</span>
  723. </button>
  724. </div>
  725. <div class="modal-body">
  726. <input type="hidden" value="<?=$news->id?>" name="post_id">
  727. <div class="form-group">
  728. <label for="name">Название</label>
  729. <input type="text" class="form-control" name="name">
  730. </div>
  731. <div class="form-group">
  732. <label for="description">Описание</label>
  733. <input type="text" class="form-control" name="description">
  734. </div>
  735. </div>
  736. <div class="modal-footer">
  737. <button type="button" class="btn btn-secondary" data-dismiss="modal">Отмена</button>
  738. <button type="submit" class="btn btn-primary save">Сохранить</button>
  739. </div>
  740. </form>
  741. </div>
  742. </div>
  743. </div>
  744. <script>
  745. $(document).ready(function () {
  746. $("#galleryForm").submit(function (e) {
  747. e.preventDefault()
  748. $.get(
  749. '<?=\yii\helpers\Url::to(["/manager/news/add-gallery"])?>',
  750. $(this).serialize(),
  751. function (data){
  752. $("#addGalleryModal").modal('hide')
  753. $("#galleryForm")[0].reset()
  754. $("#galleriesContainer .card-body .galleries-list").append(data)
  755. }
  756. )
  757. })
  758. })
  759. let galleryRemove = function (id)
  760. {
  761. $.get(
  762. '<?=\yii\helpers\Url::to(["/manager/news/remove-gallery"])?>',
  763. {id:id},
  764. function (data){
  765. $("#gallery-" + id).remove();
  766. }
  767. )
  768. }
  769. </script>
  770. <?
  771. }
  772. ?>
  773. <script src="/components/admin/js/ls.js"></script>
  774. <script>
  775. $(function() {
  776. var cache = {};
  777. $( "#story" ).autocomplete({
  778. source: function( request, response ) {
  779. var term = request.term;
  780. if ( term in cache ) {
  781. response( cache[ term ] );
  782. return;
  783. }
  784. $.getJSON( "ajaxstory", {'q':request.term}, function( data, status, xhr ) {
  785. if( data.status == 'ok' ){
  786. cache[ term ] = data.item;
  787. response( data.item );
  788. }
  789. });
  790. },
  791. minLength: 3,
  792. select: function( event, ui ) {
  793. if( $( "#story_items :input[value='"+ui.item.id+"']" ).val() != ui.item.id ){
  794. $( "#story_items" ).append('<div class="custom-control custom-checkbox"><input class="custom-control-input" type="checkbox" name="Story[]" value="'+ui.item.id+'" checked id="storyi'+ui.item.id+'"><label class="custom-control-label" for="storyi'+ui.item.id+'">'+ ui.item.label +"</label></div>");
  795. }
  796. },
  797. /* open: function() {
  798. $( this ).removeClass( "ui-corner-all" ).addClass( "ui-corner-top" );
  799. },
  800. close: function() {
  801. $( this ).removeClass( "ui-corner-top" ).addClass( "ui-corner-all" );
  802. }*/
  803. });
  804. });
  805. window.datetimepicker_conf = {"bootcssVer":4,"icontype":"glyphicon","fontAwesome":true,"icons":{"leftArrow":"glyphicon-arrow-left","rightArrow":"glyphicon-arrow-right"},"format":"dd-mm-yyyy hh:ii","type":3,"startDate":"11-Oct-2000 04:35","todayHighlight":true,"language":"ru","autoclose":true,"timezone":"UTC"};
  806. jQuery(function ($) {
  807. if (jQuery('#dt_pub').data('datetimepicker')) { jQuery('#dt_pub').datetimepicker('destroy'); }
  808. $('#dt_pub').datetimepicker(datetimepicker_conf);
  809. });
  810. $(function() {
  811. window.varsubmit = false;
  812. $("input[id='news-title']").keyup(function count(){
  813. number = $("input[id='news-title']").val().length;
  814. $("#title_count").html(number);
  815. });
  816. //phototitle_count
  817. $("textarea[id='title-photo']").keyup(function count(){
  818. number = $("textarea[id='title-photo']").val().length;
  819. $("#phototitle_count").html(number);
  820. });
  821. number = $("textarea[id='title-photo']").val().length;
  822. $("#phototitle_count").html(number);
  823. $("input[name='News[photo]']").on('change', function() {
  824. $("textarea[id='title-photo']").attr( "minlength", 33 );
  825. });
  826. $('#news-lid').keyup(function(){
  827. number = $(this).val().length;
  828. $("#lid_count").html(number);
  829. });
  830. $('#grpnh input:checkbox').click(function(){
  831. if ($(this).is(':checked')) {
  832. $('#grpnh input:checkbox').not(this).prop('checked', false);
  833. }
  834. });
  835. setInterval( function (){ sendping( '<?=$news->id?>' ) }, 10000 );
  836. function sendping( newsid ){
  837. $.get( "newsping", { id: newsid }, function(data){
  838. data = JSON.parse(data);
  839. if( data.status == 'warn' ) alert( 'В новости появился '+data['author'] );
  840. } );
  841. }
  842. $('#news-form').submit(function( event ) {
  843. console.log(window.varsubmit);
  844. val = $(document.activeElement).attr('name')
  845. if( (val == 'save' || val == 'send') && $('#news-lid').val().length < 1 ){
  846. if( !window.varsubmit ){
  847. alert("Не заполнен lid");
  848. window.varsubmit = true;
  849. }else{
  850. window.varsubmit = false;
  851. }
  852. return false;
  853. }
  854. if( (val == 'save' || val == 'send') && !window.varsubmit ){
  855. window.varsubmit = true; //запрет повторного вызова
  856. // отправка тэгов
  857. tags = $('#myTags').tagsValues();
  858. nform = $('#news-form');
  859. $.each(tags,function (key,value) {
  860. tpl = $('<input type="hidden" value="'+value+'" name="Tags[]">');
  861. nform.append(tpl);
  862. });
  863. }
  864. if( (val == 'save' || val == 'send') ) return true;
  865. });
  866. $("input[type=text]").keydown(function(event){
  867. if(event.keyCode == 13){
  868. event.preventDefault();
  869. return false;
  870. }
  871. });
  872. });
  873. function runSuggestions(element,query) {
  874. let sug_area=$(element).parents().eq(2).find('.autocomplete .autocomplete-items');
  875. if( query.length > 1 ){
  876. $.getJSON("/manager/tags/search", {'q':query}, function( data ) {
  877. _tag_input_suggestions_data = data;
  878. $.each(data,function (key,value) {
  879. let template = $("<div>"+value.name+"</div>").hide()
  880. sug_area.append(template)
  881. template.show()
  882. })
  883. });
  884. }
  885. }
  886. // CKEditor
  887. $(function () {
  888. if($('#js_news_content').length != 0){
  889. CKEDITOR.replace('js_news_content', {
  890. 'filebrowserBrowseUrl': '/components/admin/ckeditor/kcfinder/browse.php?type=files&uid=<?=$uuid?>',
  891. 'filebrowserImageBrowseUrl': '/components/admin/ckeditor/kcfinder/browse.php?type=images&uid=<?=$uuid?>',
  892. 'filebrowserFlashBrowseUrl': '/components/admin/ckeditor/kcfinder/browse.php?type=flash&uid=<?=$uuid?>',
  893. 'filebrowserUploadUrl': '/components/admin/ckeditor/kcfinder/upload.php?type=files&uid=<?=$uuid?>',
  894. 'filebrowserImageUploadUrl': '/components/admin/ckeditor/kcfinder/upload.php?type=images&uid=<?=$uuid?>',
  895. 'filebrowserFlashUploadUrl': '/components/admin/ckeditor/kcfinder/upload.php?type=flash&uid=<?=$uuid?>',
  896. 'filebrowserAudioBrowseUrl': '/components/admin/ckeditor/kcfinder/browse.php?type=files&uid=<?=$uuid?>',
  897. 'filebrowserlightboxUploadUrl': '/components/admin/ckeditor/kcfinder/browse.php?type=files&uid=<?=$uuid?>',
  898. 'height': 400
  899. });
  900. CKEDITOR.on('instanceReady', function(ev) {
  901. ev.editor.on('paste', function(evt) {
  902. evt.data.dataValue = evt.data.dataValue.replace(/[«»]/g,'"');
  903. console.log(evt.data.dataValue);
  904. }, null, null, 9);
  905. });
  906. }
  907. });
  908. window.onbeforeunload = function(e) {
  909. return "Вы уверены, что покидаете страницу?";
  910. };
  911. </script>
  912. <?
  913. // редактор end
  914. ?>
  915. </div>
  916. <div class="tab-pane fade" id="comments">
  917. <?
  918. // коменты
  919. echo $this->render('../comments/newsindex', [
  920. 'model' => $news,
  921. ]);
  922. ?>
  923. </div>
  924. </div>
  925. <?
  926. echo $this->render('formSP', [
  927. 'model' => $news,
  928. ]);
  929. echo $this->render('../top-slider/modalform', ['news'=>$news]);
  930. ?>