'yii\filters\HttpCache',
'only' => ['index'],
'enabled' => true,
'cacheControlHeader' => 'public, max-age=120',
'lastModified' => function ($action, $params) {
return time();
}
],
];
}
public function actionIndex($topic=NULL)
{
$url = \Yii::$app->request->pathInfo;
if( $url == 'news/' ){
$queryString = \Yii::$app->request->queryString?'?'.\Yii::$app->request->queryString:'';
$this->redirect('/news'.$queryString, 301);
return '';
}
$gets = \Yii::$app->request->get();
$valid_get = array_filter($gets, function($k) {
$valid = ['page'=>true,'daterange'=>true,'topic'=>true];
return isset( $valid[$k] );
}, ARRAY_FILTER_USE_KEY);
if( count($gets) != count($valid_get) ){
$url = Url::toRoute(['news/index']+$valid_get);
$this->redirect($url, 301);
return '';
}
if(\Yii::$app->request->get("page",0)>500) {
\Yii::$app->request->setQueryParams(["page"=>500]);
}
$url = $topic;
$topic = NewsTopic::findOne(['url'=>$topic,'active'=>1]);
if(!$topic instanceof NewsTopic){
$topic = new \stdClass();
$topic->id = NULL;
}
if( $url && !$topic->id ) throw new NotFoundHttpException("Раздел не найден");
$pages = null;
if( $topic->id ){
// $news_query = \app\models\front\News::find()->leftJoin('news_topic_relation t', 't.news_id = id')->andWhere(['t.topic_id'=>$topic->id]);
$news_query = News::find()
->from(new Expression("news FORCE INDEX (top,calendar) , news_topic_relation t USE index(news_id)"))
->andWhere(new Expression('t.news_id=id'))
->andWhere(['t.topic_id'=>$topic->id]);
$dateRange = \Yii::$app->request->get('daterange',NULL);
$dateRanges = $dateRange;
if(!is_null($dateRange) && $dateRange!=""){
$dateRange = explode(" — ",$dateRange);
$dateRange[0] = date("Y-m-d 00:00:01",strtotime($dateRange[0]));
$dateRange[1] = isset($dateRange[1])?date("Y-m-d 23:59:59",strtotime($dateRange[1])):date("Y-m-d 23:59:59",strtotime($dateRange[0]));
$news_query->andWhere(["BETWEEN","dt_pub",$dateRange[0],$dateRange[1]]);
}
$cnt = \Yii::$app->cache->getOrSet('topic_cnt_'.$topic->id."_$dateRanges", function () use ($news_query){
return $news_query->count();
},7200);
/*
* Оптимизировано но не понятно как убрать автоматический offset при пагинации во внешнем селекте 4с->0.7c
$sub_query = News::find()->select('news.id')->from(new Expression("news as news USE INDEX (calendar)"))
->join('inner join', 'news_topic_relation', 'news_topic_relation.news_id=news.id AND news_topic_relation.topic_id='.$topic->id)->offset($pages->offset)->limit($pages->limit);
$news_query = News::find()
->from( 'news' )
->join( 'join',['nl'=>$sub_query], 'news.id=nl.id' );
*/
}else{
$news_query = \app\models\front\News::findFilter();
$keyadd = \app\models\front\News::keyFilter();
$dateRange = \Yii::$app->request->get('daterange',NULL);
$dateRanges = crc32($dateRange);
if(!is_null($dateRange) && $dateRange!=""){
$dateRange = explode(" — ",$dateRange);
$dateRange[0] = date("Y-m-d 00:00:01",strtotime($dateRange[0]));
$dateRange[1] = isset($dateRange[1])?date("Y-m-d 23:59:59",strtotime($dateRange[1])):date("Y-m-d 23:59:59",strtotime($dateRange[0]));
$news_query->andWhere(["BETWEEN","dt_pub",$dateRange[0],$dateRange[1]]);
}
$cnt = \Yii::$app->cache->getOrSet('topic_cnt'.$keyadd."_$dateRanges", function () use ($news_query){
return $news_query->count();
},7200);
}
$cnt = ($cnt/20 > 500)?500*20:$cnt;
$pages = new \yii\data\Pagination(['totalCount' => $cnt, 'pageSize'=>20]);
if($topic instanceof NewsTopic){
/*
Приведение к каноническому маршруту в пагинации
*/
unset($valid_get['topic']);
$pages->params = $valid_get;
$pages->route = 'news/'.$topic->url;
//
// Нормализация к каноническому URL
//
$queryString = \Yii::$app->request->queryString?'?'.\Yii::$app->request->queryString:'';
if( \Yii::$app->request->url != '/'.$pages->route.$queryString ){
$url = Url::toRoute(['news/'.$topic->url]+$valid_get);
/* echo \Yii::$app->request->url.'
';
echo $pages->route.$queryString.'
';
echo $url; */
$this->redirect($url, 301);
return '';
}
}
//$pages->urlManager = Url::toRoute(['news/index']+$valid_get);
// $headers = Yii::$app->response->headers;
// $headers->set('Expires', gmdate('D, d M Y H:i:s \G\M\T', time() + (60 * 2)));
return $this->render("index",["topic"=>$topic,"daterange"=>$dateRange,"news_query"=>$news_query,"pages"=>$pages,"keycahe"=>isset($keyadd)?$keyadd:'']);
}
public function actionStory($topic=NULL)
{
$topic = Story::findOne(['url'=>$topic]);
if(!$topic instanceof Story){
$topic = new \stdClass();
$topic->id = NULL;
}
if( $topic->id ){
$news_query = \app\models\front\News::find()->leftJoin('story_relation t', 't.news_id = id')->andWhere(['t.topic_id'=>$topic->id]); //joinWith('topics t')->andFilterWhere(['t.id'=>$topic->id]);
// print_r($news_query->createCommand()->rawSQL);exit;
}else{
$news_query = \app\models\front\News::find(); // optimized
}
$cnt = \Yii::$app->cache->getOrSet('story_cnt_'.$topic->id, function () use ($news_query){
return $news_query->count();
},7200);
$pages = new \yii\data\Pagination(['totalCount' => $cnt, 'pageSize'=>20]);
$dateRange = \Yii::$app->request->get('daterange',NULL);
if(!is_null($dateRange) && $dateRange!=""){
$dateRange = explode(" — ",$dateRange);
$dateRange[0] = date("Y-m-d 00:00:01",strtotime($dateRange[0]));
$dateRange[1] = isset($dateRange[1])?date("Y-m-d 23:59:59",strtotime($dateRange[1])):date("Y-m-d 23:59:59",strtotime($dateRange[0]));
$news_query->andWhere(["BETWEEN","dt_pub",$dateRange[0],$dateRange[1]]);
}
return $this->render("index",["topic"=>$topic,"daterange"=>$dateRange,"news_query"=>$news_query,"pages"=>$pages]);
}
/**
* @param int $id
*/
public function actionView(int $id) {
$model = $this->findModel($id);
if(!$model instanceof News) throw new NotFoundHttpException("Новость не найдена");
if( $link = trim( $model->link ) ){
$link = str_replace( 'http://www.amic.ru', '', $link);
$this->redirect($link, 301);
return '';
}
$queryString = \Yii::$app->request->queryString?'?'.\Yii::$app->request->queryString:'';
if( \Yii::$app->request->url != $model->getUrl().$queryString ){
$this->redirect($model->getUrl(true).$queryString, 301);
return '';
}
if(\Yii::$app->request->isPost){
$capcha_result = \Yii::$app->session->getFlash('capcha_a')+\Yii::$app->session->getFlash('capcha_b');
if(
\Yii::$app->request->post('capcha')=="" ||
\Yii::$app->request->post('capcha')!= $capcha_result
){
\Yii::$app->session->setFlash('danger','Ошибка размещения комментария, пример решен неверно. Правильный ответ был: '.$capcha_result);
} else {
if(\Yii::$app->request->post('message')!=""){
$comment = new Comments();
if( $comment::$conf->isActive() ){
$comment->fakename = str_replace('"',"'", (\Yii::$app->request->post('fakename')));
$comment->message = strip_tags(\Yii::$app->request->post('message'),'');
$comment->news_id = $model->id;
$comment->visible = ( $comment::$conf->isPreModerate() )?'N':'Y'; //если пользователь зареген поставить true
$comment->ip_address = \Yii::$app->request->getRemoteIP().",".\Yii::$app->request->userIP;
$comment->user_agent = \Yii::$app->request->userAgent;
$comment->created_at = date("Y-m-d H:i:s");
$comment->parent_id = \Yii::$app->request->post('parent_id');
$res = $comment->save();
if($res){
\Yii::$app->session->setFlash('success','Комментарий успешно отправлен на модерацию');
$this->redirect($model->getUrl()."#comments");
\Yii::$app->end();
} else {
\Yii::$app->session->setFlash('danger','Ошибка размещения комментария');
}
}
}
}
}
return $this->render('view',['model'=>$model]);
}
/**
* @param string $uid
* @return string
*/
public function actionUidView(string $uid):string {
if( Uuid::isvalid($uid) ){
$model = \app\models\News::find()->andWhere(['uid'=>$uid])->one();
if(!$model instanceof \app\models\News) throw new NotFoundHttpException("Новость не найдена");
if( $model->active == 'N' || strtotime($model->dt_pub) > time() ){
return $this->render('view',['model'=>$model]);
}else{
$this->redirect($model->geturl());
return '';
}
}else{
throw new NotFoundHttpException("Новость не найдена");
}
}
public function actionStoryslugView($storyslug):string
{
$model = News::find()->andWhere(['alias'=>$storyslug])->one();
if(!$model instanceof News) throw new NotFoundHttpException("Новость не найдена");
return $this->render('view',['model'=>$model]);
}
public function actionInfinityNewsfeed($post_id,$last_post_id=NULL,$amount=0){
if($last_post_id=="null"){
$last_post = News::find()->limit(1)->one();
} else {
$last_post = News::findOne(["id"=>$last_post_id]);
}
$feed = News::find()
->andWhere(['!=','id',$post_id])
//->andFilterWhere(['<','id',$last_post_id])
->andFilterWhere(['<','dt_pub',$last_post->dt_pub])
->limit(1)
->one();
if(!is_null($feed)){
return $this->asJson([
"id"=>$feed->id,
"html"=>$this->renderPartial('/news/view/infinity_newsfeed/_item',["model"=>$feed,"index"=>$amount])
]);
}
return NULL;
}
public function actionClike(){
if(\Yii::$app->request->isPost){
if(\Yii::$app->request->post('id') && \Yii::$app->request->post('grd') ){
$id = (int) \Yii::$app->request->post('id'); //комент
$nid = (int) \Yii::$app->request->post('nid'); //новость
$grd = (int) \Yii::$app->request->post('grd'); //градиент
if( Comments::isUserLike($id) ) return $this->asJson(['err']);
if( $id > 0 && ( $grd == 1 || $grd == -1 ) ){
$comment = Comments::findOne($id);
$comment->likeed += $grd;
if( $comment->save() ){
Comments::setUserLike($nid, $id);
}
}
}
}
return $this->asJson(['ok']);
}
public function actionSsearch()
{
if (Yii::$app->request->isGet && $get = Yii::$app->request->get('query')){
$model = new Story();
$gets = mb_strlen( $get ) < 6?$get.'*':$get;
$items = $model->search($gets, 10);
$res = array();
$json = '{}';
$resj = json_decode($json);
$resj->query = $get;
if( $items && is_array( $items ) ){
foreach( $items as $item ){
$res[] = array( 'data'=>$item['id']*1, 'value'=>$item['title'] );
}
$resj->suggestions = $res;
return json_encode( $resj );
}
}
$json = '{}';
$resj = json_decode($json);
$resj->query = $get;
$resj->suggestions = [];
return json_encode( $resj );
}
private function findModel($id)
{
return News::find()->andWhere(['id'=>$id])->limit(1)->one();
}
}