Commit 8e13c18cdd790e1fe099b636b47cd1cc47b88607
1 parent
543f1653
Search
Showing
11 changed files
with
429 additions
and
26 deletions
Show diff stats
common/models/SearchForm.php
| 1 | 1 | <?php |
| 2 | + | |
| 2 | 3 | namespace common\models; |
| 3 | 4 | |
| 5 | + use artbox\catalog\models\Category; | |
| 6 | + use artbox\catalog\models\Product; | |
| 4 | 7 | use yii\base\Model; |
| 8 | + use yii\db\ActiveQuery; | |
| 5 | 9 | |
| 6 | 10 | /** |
| 7 | 11 | * Class SearchForm for performing search around site. |
| ... | ... | @@ -21,7 +25,15 @@ |
| 21 | 25 | return [ |
| 22 | 26 | [ |
| 23 | 27 | [ 'word' ], |
| 28 | + 'required', | |
| 29 | + 'message' => \Yii::t('app', 'Write text to search for.'), | |
| 30 | + ], | |
| 31 | + [ | |
| 32 | + [ 'word' ], | |
| 24 | 33 | 'string', |
| 34 | + 'min' => 3, | |
| 35 | + 'message' => \Yii::t('app', 'Write at least 3 symbols.'), | |
| 36 | + 'tooShort' => \Yii::t('app', 'Write at least 3 symbols.'), | |
| 25 | 37 | ], |
| 26 | 38 | ]; |
| 27 | 39 | } |
| ... | ... | @@ -35,4 +47,43 @@ |
| 35 | 47 | 'word' => \Yii::t('app', 'Поиск по сайту'), |
| 36 | 48 | ]; |
| 37 | 49 | } |
| 50 | + | |
| 51 | + public function search() | |
| 52 | + { | |
| 53 | + return Product::find() | |
| 54 | + ->joinWith('lang') | |
| 55 | + ->joinWith('variants.lang') | |
| 56 | + ->andWhere( | |
| 57 | + [ | |
| 58 | + 'product.status' => true, | |
| 59 | + 'variant.status' => true, | |
| 60 | + ] | |
| 61 | + ) | |
| 62 | + ->andWhere( | |
| 63 | + [ | |
| 64 | + 'like', | |
| 65 | + 'product_lang.title', | |
| 66 | + $this->word, | |
| 67 | + ] | |
| 68 | + ) | |
| 69 | + ->orWhere( | |
| 70 | + [ | |
| 71 | + 'like', | |
| 72 | + 'variant_lang.title', | |
| 73 | + $this->word, | |
| 74 | + ] | |
| 75 | + ) | |
| 76 | + ->orWhere([ 'variant.sku' => $this->word ]) | |
| 77 | + ->groupBy('product.id'); | |
| 78 | + } | |
| 79 | + | |
| 80 | + public function searchCategory(Category $category, ActiveQuery $query) | |
| 81 | + { | |
| 82 | + return $query->joinWith('productToCategories') | |
| 83 | + ->andWhere( | |
| 84 | + [ | |
| 85 | + 'product_to_category.category_id' => $category->id, | |
| 86 | + ] | |
| 87 | + ); | |
| 88 | + } | |
| 38 | 89 | } |
| 39 | 90 | \ No newline at end of file | ... | ... |
| 1 | +<?php | |
| 2 | + | |
| 3 | + namespace frontend\controllers; | |
| 4 | + | |
| 5 | + use artbox\catalog\models\Category; | |
| 6 | + use common\models\SearchForm; | |
| 7 | + use yii\data\ActiveDataProvider; | |
| 8 | + use yii\web\Controller; | |
| 9 | + use yii\web\NotFoundHttpException; | |
| 10 | + | |
| 11 | + class SearchController extends Controller | |
| 12 | + { | |
| 13 | + public function actionIndex($word = '') | |
| 14 | + { | |
| 15 | + $searchForm = new SearchForm( | |
| 16 | + [ | |
| 17 | + 'word' => $word, | |
| 18 | + ] | |
| 19 | + ); | |
| 20 | + $dataProvider = null; | |
| 21 | + $categories = []; | |
| 22 | + if ($searchForm->validate()) { | |
| 23 | + $dataProvider = new ActiveDataProvider( | |
| 24 | + [ | |
| 25 | + 'query' => $searchForm->search(), | |
| 26 | + ] | |
| 27 | + ); | |
| 28 | + /** | |
| 29 | + * @var \artbox\catalog\models\queries\ProductQuery $categoryQuery | |
| 30 | + */ | |
| 31 | + $categoryQuery = clone $dataProvider->query; | |
| 32 | + $categoryQuery->select([ 'product.id' ]); | |
| 33 | + $categories = Category::find() | |
| 34 | + ->active() | |
| 35 | + ->with('lang') | |
| 36 | + ->joinWith('productToCategories') | |
| 37 | + ->andWhere([ 'product_to_category.product_id' => $categoryQuery->column() ]) | |
| 38 | + ->all(); | |
| 39 | + } | |
| 40 | + return $this->render( | |
| 41 | + 'index', | |
| 42 | + [ | |
| 43 | + 'searchForm' => $searchForm, | |
| 44 | + 'dataProvider' => $dataProvider, | |
| 45 | + 'categories' => $categories, | |
| 46 | + ] | |
| 47 | + ); | |
| 48 | + } | |
| 49 | + | |
| 50 | + public function actionCategory($categoryId, $word = '') | |
| 51 | + { | |
| 52 | + /** | |
| 53 | + * @var Category $category | |
| 54 | + */ | |
| 55 | + $category = Category::find() | |
| 56 | + ->active() | |
| 57 | + ->andWhere([ 'id' => $categoryId ]) | |
| 58 | + ->with('lang') | |
| 59 | + ->one(); | |
| 60 | + if (!$category) { | |
| 61 | + throw new NotFoundHttpException(\Yii::t('app', 'Category not found!')); | |
| 62 | + } | |
| 63 | + $searchForm = new SearchForm( | |
| 64 | + [ | |
| 65 | + 'word' => $word, | |
| 66 | + ] | |
| 67 | + ); | |
| 68 | + $dataProvider = null; | |
| 69 | + $categories = []; | |
| 70 | + if ($searchForm->validate()) { | |
| 71 | + /** | |
| 72 | + * @var \artbox\catalog\models\queries\ProductQuery $query | |
| 73 | + */ | |
| 74 | + $query = $searchForm->search(); | |
| 75 | + /** | |
| 76 | + * @var \artbox\catalog\models\queries\ProductQuery $categoryQuery | |
| 77 | + */ | |
| 78 | + $categoryQuery = clone $query; | |
| 79 | + $dataProvider = new ActiveDataProvider( | |
| 80 | + [ | |
| 81 | + 'query' => $searchForm->searchCategory($category, $query), | |
| 82 | + ] | |
| 83 | + ); | |
| 84 | + $categoryQuery->select([ 'product.id' ]); | |
| 85 | + $categories = Category::find() | |
| 86 | + ->active() | |
| 87 | + ->joinWith('productToCategories') | |
| 88 | + ->andWhere([ 'product_to_category.product_id' => $categoryQuery->column() ]) | |
| 89 | + ->all(); | |
| 90 | + } | |
| 91 | + return $this->render( | |
| 92 | + 'category', | |
| 93 | + [ | |
| 94 | + 'searchForm' => $searchForm, | |
| 95 | + 'dataProvider' => $dataProvider, | |
| 96 | + 'categories' => $categories, | |
| 97 | + 'category' => $category, | |
| 98 | + ] | |
| 99 | + ); | |
| 100 | + } | |
| 101 | + } | |
| 0 | 102 | \ No newline at end of file | ... | ... |
frontend/controllers/SiteController.php
| ... | ... | @@ -77,17 +77,17 @@ |
| 77 | 77 | ->all(); |
| 78 | 78 | $topItems = Product::find() |
| 79 | 79 | ->with('lang', 'image', 'variants') |
| 80 | - ->where('mask & 1 != 0') | |
| 80 | + ->is('mask', 1) | |
| 81 | 81 | ->limit(20) |
| 82 | 82 | ->all(); |
| 83 | 83 | $newItems = Product::find() |
| 84 | 84 | ->with('lang', 'image', 'variants') |
| 85 | - ->where('mask & 2 != 0') | |
| 85 | + ->is('mask', 2) | |
| 86 | 86 | ->limit(20) |
| 87 | 87 | ->all(); |
| 88 | 88 | $saleItems = Product::find() |
| 89 | 89 | ->with('lang', 'image', 'variants') |
| 90 | - ->where('mask & 4 != 0') | |
| 90 | + ->is('mask', 4) | |
| 91 | 91 | ->limit(20) |
| 92 | 92 | ->all(); |
| 93 | 93 | $productCount = Product::find() | ... | ... |
frontend/controllers/SpecialController.php
| ... | ... | @@ -29,13 +29,13 @@ |
| 29 | 29 | ->innerJoinWith('products', false) |
| 30 | 30 | ->groupBy('category.id'); |
| 31 | 31 | if ($type === 'new') { |
| 32 | - $query->where('product.mask & 2 != 0'); | |
| 32 | + $query->is('product.mask', 2); | |
| 33 | 33 | $productQuery = Product::find() |
| 34 | - ->where('product.mask & 2 != 0'); | |
| 34 | + ->is('product.mask', 2); | |
| 35 | 35 | } else { |
| 36 | - $query->where('product.mask & 4 != 0'); | |
| 36 | + $query->is('product.mask', 4); | |
| 37 | 37 | $productQuery = Product::find() |
| 38 | - ->where('product.mask & 4 != 0'); | |
| 38 | + ->is('product.mask', 4); | |
| 39 | 39 | } |
| 40 | 40 | $categories = $query->all(); |
| 41 | 41 | $dataProvider = new ActiveDataProvider( |
| ... | ... | @@ -74,9 +74,9 @@ |
| 74 | 74 | ] |
| 75 | 75 | ); |
| 76 | 76 | if ($type === 'new') { |
| 77 | - $query->andWhere('product.mask & 2 != 0'); | |
| 77 | + $query->is('product.mask', 2); | |
| 78 | 78 | } elseif ($type === 'sale') { |
| 79 | - $query->andWhere('product.mask & 4 != 0'); | |
| 79 | + $query->is('product.mask', 4); | |
| 80 | 80 | } |
| 81 | 81 | $dataProvider = new ActiveDataProvider( |
| 82 | 82 | [ | ... | ... |
frontend/views/category/_product_item.php
frontend/views/layouts/main.php
frontend/views/product/view.php
| ... | ... | @@ -174,7 +174,7 @@ |
| 174 | 174 | <p class="price"> |
| 175 | 175 | <span class="price-title">Цена:</span><?php echo $variant->price ? : 0; ?> грн  |
| 176 | 176 | <?php |
| 177 | - if ($variant->stock && $variant->price) { | |
| 177 | + if ($variant->canBuy()) { | |
| 178 | 178 | echo Html::a( |
| 179 | 179 | Html::tag( |
| 180 | 180 | 'i', |
| ... | ... | @@ -211,22 +211,24 @@ |
| 211 | 211 | </p> |
| 212 | 212 | <hr> |
| 213 | 213 | <?php |
| 214 | - echo Html::a( | |
| 215 | - Html::icon( | |
| 216 | - 'phone', | |
| 214 | + if ($variant->canBuy()) { | |
| 215 | + echo Html::a( | |
| 216 | + Html::icon( | |
| 217 | + 'phone', | |
| 218 | + [ | |
| 219 | + 'prefix' => 'fa fa-', | |
| 220 | + ] | |
| 221 | + ) . \Yii::t('app', 'Купить в один клик'), | |
| 222 | + '#', | |
| 217 | 223 | [ |
| 218 | - 'prefix' => 'fa fa-', | |
| 224 | + 'data' => [ | |
| 225 | + 'toggle' => 'modal', | |
| 226 | + 'target' => '#oneclick-modal', | |
| 227 | + ], | |
| 228 | + 'class' => 'btn btn-template-main', | |
| 219 | 229 | ] |
| 220 | - ) . \Yii::t('app', 'Купить в один клик'), | |
| 221 | - '#', | |
| 222 | - [ | |
| 223 | - 'data' => [ | |
| 224 | - 'toggle' => 'modal', | |
| 225 | - 'target' => '#oneclick-modal', | |
| 226 | - ], | |
| 227 | - 'class' => 'btn btn-template-main', | |
| 228 | - ] | |
| 229 | - ) | |
| 230 | + ); | |
| 231 | + } | |
| 230 | 232 | ?> |
| 231 | 233 | <?php |
| 232 | 234 | /* | ... | ... |
| 1 | +<?php | |
| 2 | + /** | |
| 3 | + * @var \yii\web\View $this | |
| 4 | + * @var \common\models\SearchForm $searchForm | |
| 5 | + * @var \yii\data\ActiveDataProvider $dataProvider | |
| 6 | + * @var \artbox\catalog\models\Category[] $categories | |
| 7 | + * @var \artbox\catalog\models\Category $category | |
| 8 | + */ | |
| 9 | + use artbox\catalog\models\Product; | |
| 10 | + use yii\bootstrap\ActiveForm; | |
| 11 | + use yii\bootstrap\Html; | |
| 12 | + use yii\widgets\LinkPager; | |
| 13 | + use yii\widgets\ListView; | |
| 14 | + | |
| 15 | + $view = $this; | |
| 16 | + | |
| 17 | + $this->params[ 'breadcrumbs' ][] = [ | |
| 18 | + 'label' => \Yii::t('app', 'Поиск'), | |
| 19 | + 'url' => [ | |
| 20 | + 'index', | |
| 21 | + 'word' => $searchForm->word, | |
| 22 | + ], | |
| 23 | + ]; | |
| 24 | + $this->params[ 'breadcrumbs' ][] = \Yii::t( | |
| 25 | + 'app', | |
| 26 | + 'Поиск по категории {category_name}', | |
| 27 | + [ | |
| 28 | + 'category_name' => $category->lang->title, | |
| 29 | + ] | |
| 30 | + ); | |
| 31 | +?> | |
| 32 | +<div id="content"> | |
| 33 | + <div class="container"> | |
| 34 | + <div class="row"> | |
| 35 | + <div class="col-sm-offset-3 col-sm-6"> | |
| 36 | + <?php | |
| 37 | + $form = ActiveForm::begin( | |
| 38 | + [ | |
| 39 | + 'action' => [ | |
| 40 | + 'category', | |
| 41 | + 'categoryId' => $category->id, | |
| 42 | + ], | |
| 43 | + 'method' => 'get', | |
| 44 | + ] | |
| 45 | + ); | |
| 46 | + echo $form->field($searchForm, 'word') | |
| 47 | + ->textInput( | |
| 48 | + [ | |
| 49 | + 'placeholder' => $searchForm->getAttributeLabel('word'), | |
| 50 | + 'name' => 'word', | |
| 51 | + ] | |
| 52 | + ) | |
| 53 | + ->label(false); | |
| 54 | + $form::end(); | |
| 55 | + ?> | |
| 56 | + </div> | |
| 57 | + </div> | |
| 58 | + <div class="row"> | |
| 59 | + <div class="col-xs-3"> | |
| 60 | + <div class="panel panel-default sidebar-menu"> | |
| 61 | + <div class="panel-body"> | |
| 62 | + <ul class="nav nav-pills nav-stacked category-menu"> | |
| 63 | + <?php | |
| 64 | + foreach ($categories as $item) { | |
| 65 | + echo Html::tag( | |
| 66 | + 'li', | |
| 67 | + Html::a( | |
| 68 | + $item->lang->title, | |
| 69 | + [ | |
| 70 | + 'category', | |
| 71 | + 'categoryId' => $item->id, | |
| 72 | + 'word' => $searchForm->word, | |
| 73 | + ], | |
| 74 | + [ | |
| 75 | + 'class' => ( ( $category->id === $item->id ) ? 'active' : '' ), | |
| 76 | + ] | |
| 77 | + ) | |
| 78 | + ); | |
| 79 | + } | |
| 80 | + ?> | |
| 81 | + </ul> | |
| 82 | + </div> | |
| 83 | + </div> | |
| 84 | + </div> | |
| 85 | + <div class="col-xs-9"> | |
| 86 | + <?php | |
| 87 | + if ($dataProvider) { | |
| 88 | + echo ListView::widget( | |
| 89 | + [ | |
| 90 | + 'options' => [ | |
| 91 | + 'class' => 'row products', | |
| 92 | + ], | |
| 93 | + 'itemOptions' => [ | |
| 94 | + 'tag' => false, | |
| 95 | + ], | |
| 96 | + 'layout' => '{items}', | |
| 97 | + 'dataProvider' => $dataProvider, | |
| 98 | + 'itemView' => function ($model) use ($view) { | |
| 99 | + /** | |
| 100 | + * @var Product $model | |
| 101 | + */ | |
| 102 | + return $view->render( | |
| 103 | + '@frontend/views/category/_product_item', | |
| 104 | + [ | |
| 105 | + 'product' => $model, | |
| 106 | + ] | |
| 107 | + ); | |
| 108 | + }, | |
| 109 | + ] | |
| 110 | + ); | |
| 111 | + echo Html::tag( | |
| 112 | + 'div', | |
| 113 | + LinkPager::widget( | |
| 114 | + [ | |
| 115 | + 'pagination' => $dataProvider->pagination, | |
| 116 | + ] | |
| 117 | + ), | |
| 118 | + [ | |
| 119 | + 'class' => 'pages', | |
| 120 | + ] | |
| 121 | + ); | |
| 122 | + } else { | |
| 123 | + echo \Yii::t('app', 'Введите текст для поиска в строке выше.'); | |
| 124 | + } | |
| 125 | + ?> | |
| 126 | + </div> | |
| 127 | + </div> | |
| 128 | + </div> | |
| 129 | +</div> | |
| 0 | 130 | \ No newline at end of file | ... | ... |
| 1 | +<?php | |
| 2 | + /** | |
| 3 | + * @var \yii\web\View $this | |
| 4 | + * @var \common\models\SearchForm $searchForm | |
| 5 | + * @var \yii\data\ActiveDataProvider $dataProvider | |
| 6 | + * @var \artbox\catalog\models\Category[] $categories | |
| 7 | + */ | |
| 8 | + use artbox\catalog\models\Product; | |
| 9 | + use yii\bootstrap\ActiveForm; | |
| 10 | + use yii\bootstrap\Html; | |
| 11 | + use yii\widgets\LinkPager; | |
| 12 | + use yii\widgets\ListView; | |
| 13 | + | |
| 14 | + $view = $this; | |
| 15 | + | |
| 16 | + $this->params[ 'breadcrumbs' ][] = \Yii::t('app', 'Поиск'); | |
| 17 | +?> | |
| 18 | +<div id="content"> | |
| 19 | + <div class="container"> | |
| 20 | + <div class="row"> | |
| 21 | + <div class="col-sm-offset-3 col-sm-6"> | |
| 22 | + <?php | |
| 23 | + $form = ActiveForm::begin( | |
| 24 | + [ | |
| 25 | + 'action' => [ 'index' ], | |
| 26 | + 'method' => 'get', | |
| 27 | + ] | |
| 28 | + ); | |
| 29 | + echo $form->field($searchForm, 'word') | |
| 30 | + ->textInput( | |
| 31 | + [ | |
| 32 | + 'placeholder' => $searchForm->getAttributeLabel('word'), | |
| 33 | + 'name' => 'word', | |
| 34 | + ] | |
| 35 | + ) | |
| 36 | + ->label(false); | |
| 37 | + $form::end(); | |
| 38 | + ?> | |
| 39 | + </div> | |
| 40 | + </div> | |
| 41 | + <div class="row"> | |
| 42 | + <div class="col-xs-3"> | |
| 43 | + <div class="panel panel-default sidebar-menu"> | |
| 44 | + <div class="panel-body"> | |
| 45 | + <ul class="nav nav-pills nav-stacked category-menu"> | |
| 46 | + <?php | |
| 47 | + foreach ($categories as $category) { | |
| 48 | + echo Html::tag( | |
| 49 | + 'li', | |
| 50 | + Html::a( | |
| 51 | + $category->lang->title, | |
| 52 | + [ | |
| 53 | + 'category', | |
| 54 | + 'categoryId' => $category->id, | |
| 55 | + 'word' => $searchForm->word, | |
| 56 | + ] | |
| 57 | + ) | |
| 58 | + ); | |
| 59 | + } | |
| 60 | + ?> | |
| 61 | + </ul> | |
| 62 | + </div> | |
| 63 | + </div> | |
| 64 | + </div> | |
| 65 | + <div class="col-xs-9"> | |
| 66 | + <?php | |
| 67 | + if ($dataProvider) { | |
| 68 | + echo ListView::widget( | |
| 69 | + [ | |
| 70 | + 'options' => [ | |
| 71 | + 'class' => 'row products', | |
| 72 | + ], | |
| 73 | + 'itemOptions' => [ | |
| 74 | + 'tag' => false, | |
| 75 | + ], | |
| 76 | + 'layout' => '{items}', | |
| 77 | + 'dataProvider' => $dataProvider, | |
| 78 | + 'itemView' => function ($model) use ($view) { | |
| 79 | + /** | |
| 80 | + * @var Product $model | |
| 81 | + */ | |
| 82 | + return $view->render( | |
| 83 | + '@frontend/views/category/_product_item', | |
| 84 | + [ | |
| 85 | + 'product' => $model, | |
| 86 | + ] | |
| 87 | + ); | |
| 88 | + }, | |
| 89 | + ] | |
| 90 | + ); | |
| 91 | + echo Html::tag( | |
| 92 | + 'div', | |
| 93 | + LinkPager::widget( | |
| 94 | + [ | |
| 95 | + 'pagination' => $dataProvider->pagination, | |
| 96 | + ] | |
| 97 | + ), | |
| 98 | + [ | |
| 99 | + 'class' => 'pages', | |
| 100 | + ] | |
| 101 | + ); | |
| 102 | + } else { | |
| 103 | + echo \Yii::t('app', 'Введите текст для поиска в строке выше.'); | |
| 104 | + } | |
| 105 | + ?> | |
| 106 | + </div> | |
| 107 | + </div> | |
| 108 | + </div> | |
| 109 | +</div> | |
| 0 | 110 | \ No newline at end of file | ... | ... |
frontend/views/site/_slider_product.php
frontend/web/css/style.css
| ... | ... | @@ -4653,4 +4653,14 @@ a.list-group-item.active > .badge, |
| 4653 | 4653 | .alert-cart.active{ |
| 4654 | 4654 | top: 30px; |
| 4655 | 4655 | box-shadow: 0px 0px 25px rgba(0, 0, 0, 0.3); |
| 4656 | +} | |
| 4657 | + | |
| 4658 | +.panel.sidebar-menu ul.nav.category-menu li a.active { | |
| 4659 | + background-color: #fd6721; | |
| 4660 | + color: white; | |
| 4661 | +} | |
| 4662 | + | |
| 4663 | +.panel.sidebar-menu ul.nav.category-menu li a.active:hover { | |
| 4664 | + background-color: #fd9131; | |
| 4665 | + color: white; | |
| 4656 | 4666 | } |
| 4657 | 4667 | \ No newline at end of file | ... | ... |