CatchQuery.php 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. <?php
  2. declare(strict_types=1);
  3. namespace catcher;
  4. use catcher\base\CatchModel;
  5. use think\db\Query;
  6. use think\helper\Str;
  7. use think\Paginator;
  8. class CatchQuery extends Query
  9. {
  10. /**
  11. *
  12. * @time 2020年01月13日
  13. * @param string $model
  14. * @param string $joinField
  15. * @param string $currentJoinField
  16. * @param array $field
  17. * @param string $type
  18. * @param array $bind
  19. * @return CatchQuery
  20. */
  21. public function catchJoin(string $model, string $joinField, string $currentJoinField, array $field = [], string $type = 'INNER', array $bind = []): CatchQuery
  22. {
  23. $table = app($model)->getTable();
  24. // 合并字段
  25. $this->options['field'] = array_merge($this->options['field'] ?? [], array_map(function ($value) use ($table) {
  26. return $table . '.' . $value;
  27. }, $field));
  28. return $this->join($table, sprintf('%s.%s=%s.%s', $table, $joinField, $this->getAlias(), $currentJoinField), $type, $bind);
  29. }
  30. /**
  31. *
  32. * @time 2020年01月13日
  33. * @param string $model
  34. * @param string $joinField
  35. * @param string $currentJoinField
  36. * @param array $field
  37. * @param array $bind
  38. * @return CatchQuery
  39. */
  40. public function catchLeftJoin(string $model, string $joinField, string $currentJoinField, array $field = [], array $bind = []): CatchQuery
  41. {
  42. return $this->catchJoin($model, $joinField, $currentJoinField, $field,'LEFT', $bind);
  43. }
  44. /**
  45. *
  46. * @time 2020年01月13日
  47. * @param string $model
  48. * @param string $joinField
  49. * @param string $currentJoinField
  50. * @param array $field
  51. * @param array $bind
  52. * @return CatchQuery
  53. */
  54. public function catchRightJoin(string $model, string $joinField, string $currentJoinField, array $field = [], array $bind = []): CatchQuery
  55. {
  56. return $this->catchJoin($model, $joinField, $currentJoinField, $field,'RIGHT', $bind);
  57. }
  58. /**
  59. * rewrite
  60. *
  61. * @time 2020年01月13日
  62. * @param array|string $field
  63. * @param bool $needAlias
  64. * @return $this|Query
  65. */
  66. public function withoutField($field, $needAlias = false)
  67. {
  68. if (empty($field)) {
  69. return $this;
  70. }
  71. if (is_string($field)) {
  72. $field = array_map('trim', explode(',', $field));
  73. }
  74. // 过滤软删除字段
  75. $field[] = $this->model->getDeleteAtField();
  76. // 字段排除
  77. $fields = $this->getTableFields();
  78. $field = $fields ? array_diff($fields, $field) : $field;
  79. if (isset($this->options['field'])) {
  80. $field = array_merge((array) $this->options['field'], $field);
  81. }
  82. $this->options['field'] = array_unique($field);
  83. if ($needAlias) {
  84. $alias = $this->getAlias();
  85. $this->options['field'] = array_map(function ($field) use ($alias) {
  86. return $alias . '.' . $field;
  87. }, $this->options['field']);
  88. }
  89. return $this;
  90. }
  91. /**
  92. *
  93. * @time 2020年01月13日
  94. * @param array $params
  95. * @return CatchQuery
  96. */
  97. public function catchSearch($params = []): CatchQuery
  98. {
  99. $params = empty($params) ? \request()->param() : $params;
  100. if (empty($params)) {
  101. return $this;
  102. }
  103. foreach ($params as $field => $value) {
  104. $method = 'search' . Str::studly($field) . 'Attr';
  105. // value in [null, '']
  106. if ($value !== null && $value !== '' && method_exists($this->model, $method)) {
  107. $this->model->$method($this, $value, $params);
  108. }
  109. }
  110. return $this;
  111. }
  112. /**
  113. * 追加查询条件
  114. */
  115. public function appendQueryCond(Array $conds)
  116. {
  117. if (!empty($conds)) {
  118. // 特殊快捷操作
  119. $arr = [
  120. 'whereExp','whereNull','whereNotNull','whereLike','whereNotLike',
  121. 'whereIn','whereNotIn','whereBetween','whereNotBetween',
  122. 'whereBetweenTime','whereNotBetweenTime','whereOr',
  123. 'whereYear','whereMonth','whereDay',
  124. ];
  125. foreach($conds as $operate => $exps) {
  126. if (!in_array($operate, $arr) || !is_array($exps)) {
  127. $this->$operate($exps);
  128. continue;
  129. }
  130. foreach($exps as $exp) {
  131. $this->$operate(...$exp);
  132. }
  133. }
  134. }
  135. return $this;
  136. }
  137. /**
  138. *
  139. * @time 2020年01月13日
  140. * @return mixed
  141. */
  142. public function getAlias()
  143. {
  144. return isset($this->options['alias']) ? $this->options['alias'][$this->getTable()] : $this->getTable();
  145. }
  146. /**
  147. * rewrite
  148. *
  149. * @time 2020年01月13日
  150. * @param string $field
  151. * @param mixed $condition
  152. * @param string $option
  153. * @param string $logic
  154. * @return Query
  155. */
  156. public function whereLike(string $field, $condition, string $logic = 'AND', $option = 'both'): Query
  157. {
  158. switch ($option) {
  159. case 'both':
  160. $condition = '%' . $condition . '%';
  161. break;
  162. case 'left':
  163. $condition = '%' . $condition;
  164. break;
  165. default:
  166. $condition .= '%';
  167. }
  168. if (strpos($field, '.') === false) {
  169. $field = $this->getAlias() . '.' . $field;
  170. }
  171. return parent::whereLike($field, $condition, $logic);
  172. }
  173. /**
  174. * 额外的字段
  175. *
  176. * @time 2020年01月13日
  177. * @param $fields
  178. * @return CatchQuery
  179. */
  180. public function addFields($fields): CatchQuery
  181. {
  182. if (is_string($fields)) {
  183. $this->options['field'][] = $fields;
  184. return $this;
  185. }
  186. $this->options['field'] = array_merge($this->options['field'], $fields);
  187. return $this;
  188. }
  189. public function paginate($listRows = null, $simple = false): Paginator
  190. {
  191. if (!$listRows) {
  192. $limit = \request()->param('limit');
  193. $listRows = $limit ? : CatchModel::LIMIT;
  194. }
  195. if(gettype($listRows)=='string'){
  196. $listRows=(int)$listRows;
  197. }
  198. return parent::paginate($listRows, $simple); // TODO: Change the autogenerated stub
  199. }
  200. /**
  201. * 默认排序
  202. *
  203. * @time 2020年06月17日
  204. * @param string $order
  205. * @return $this
  206. */
  207. public function catchOrder($order = 'desc')
  208. {
  209. if (in_array('sort', array_keys($this->getFields()))) {
  210. $this->order($this->getTable() . '.sort', $order);
  211. }
  212. $this->order($this->getTable() . '.' . $this->getPk(), $order);
  213. return $this;
  214. }
  215. /**
  216. * 新增 Select 子查询
  217. *
  218. * @time 2020年06月17日
  219. * @param callable $callable
  220. * @param string $as
  221. * @return $this
  222. */
  223. public function addSelectSub(callable $callable, string $as)
  224. {
  225. $this->field(sprintf('%s as %s', $callable()->buildSql(), $as));
  226. return $this;
  227. }
  228. /**
  229. * 字段增加
  230. *
  231. * @time 2020年11月04日
  232. * @param $field
  233. * @param int $amount
  234. * @throws \think\db\exception\DbException
  235. * @return int
  236. */
  237. public function increment($field, $amount = 1)
  238. {
  239. return $this->inc($field, $amount)->update();
  240. }
  241. /**
  242. * 字段减少
  243. *
  244. * @time 2020年11月04日
  245. * @param $field
  246. * @param int $amount
  247. * @throws \think\db\exception\DbException
  248. * @return int
  249. */
  250. public function decrement($field, $amount = 1)
  251. {
  252. return $this->dec($field, $amount)->update();
  253. }
  254. }