Tag.php 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. <?php
  2. namespace addons\cms\model;
  3. use think\Db;
  4. use think\Model;
  5. /**
  6. * 标签模型
  7. */
  8. class Tag extends Model
  9. {
  10. protected $name = "cms_tag";
  11. // 开启自动写入时间戳字段
  12. protected $autoWriteTimestamp = 'int';
  13. // 定义时间戳字段名
  14. protected $createTime = '';
  15. protected $updateTime = '';
  16. // 追加属性
  17. protected $append = [
  18. 'url',
  19. 'fullurl'
  20. ];
  21. protected static $config = [];
  22. protected static $tagCount = 0;
  23. protected static function init()
  24. {
  25. $config = get_addon_config('cms');
  26. self::$config = $config;
  27. }
  28. public function model()
  29. {
  30. return $this->belongsTo("Modelx");
  31. }
  32. public function getUrlAttr($value, $data)
  33. {
  34. return $this->buildUrl($value, $data);
  35. }
  36. public function getFullurlAttr($value, $data)
  37. {
  38. return $this->buildUrl($value, $data, true);
  39. }
  40. private function buildUrl($value, $data, $domain = false)
  41. {
  42. $diyname = isset($data['name']) && $data['name'] ? $data['name'] : $data['id'];
  43. $time = $data['createtime'] ?? time();
  44. $vars = [
  45. ':id' => $data['id'],
  46. ':name' => $diyname,
  47. ':diyname' => $diyname,
  48. ':year' => date("Y", $time),
  49. ':month' => date("m", $time),
  50. ':day' => date("d", $time)
  51. ];
  52. $suffix = static::$config['moduleurlsuffix']['tag'] ?? static::$config['urlsuffix'];
  53. return addon_url('cms/tag/index', $vars, $suffix, $domain);
  54. }
  55. /**
  56. * 获取标签列表
  57. * @param $tag
  58. * @return false|\PDOStatement|string|\think\Collection
  59. */
  60. public static function getTagList($tag)
  61. {
  62. $config = get_addon_config('cms');
  63. $condition = empty($tag['condition']) ? '' : $tag['condition'];
  64. $field = empty($tag['field']) ? '*' : $tag['field'];
  65. $row = empty($tag['row']) ? 10 : (int)$tag['row'];
  66. $orderby = empty($tag['orderby']) ? 'nums' : $tag['orderby'];
  67. $orderway = empty($tag['orderway']) ? 'desc' : strtolower($tag['orderway']);
  68. $limit = empty($tag['limit']) ? $row : $tag['limit'];
  69. $cache = !isset($tag['cache']) ? $config['cachelifetime'] === 'true' ? true : (int)$config['cachelifetime'] : (int)$tag['cache'];
  70. $orderway = in_array($orderway, ['asc', 'desc']) ? $orderway : 'desc';
  71. $paginate = !isset($tag['paginate']) ? false : $tag['paginate'];
  72. $cache = !$cache ? false : $cache;
  73. self::$tagCount++;
  74. $where = [];
  75. $order = $orderby == 'rand' ? Db::raw('rand()') : (preg_match("/\,|\s/", $orderby) ? $orderby : "{$orderby} {$orderway}");
  76. $tagModel = self::where($where)
  77. ->where($condition)
  78. ->field($field)
  79. ->orderRaw($order);
  80. if ($paginate) {
  81. $paginateArr = explode(',', $paginate);
  82. $listRows = is_numeric($paginate) ? $paginate : (is_numeric($paginateArr[0]) ? $paginateArr[0] : $row);
  83. $config = [];
  84. $config['var_page'] = isset($paginateArr[2]) ? $paginateArr[2] : 'tpage' . self::$tagCount;
  85. $config['path'] = isset($paginateArr[3]) ? $paginateArr[3] : '';
  86. $config['fragment'] = isset($paginateArr[4]) ? $paginateArr[4] : '';
  87. $config['query'] = request()->get();
  88. $list = $tagModel->paginate($listRows, (isset($paginateArr[1]) ? $paginateArr[1] : false), $config);
  89. } else {
  90. $list = $tagModel->limit($limit)->cache($cache)->select();
  91. }
  92. foreach ($list as $k => $v) {
  93. $v['textlink'] = '<a href="' . $v['url'] . '">' . $v['name'] . '</a>';
  94. }
  95. return $list;
  96. }
  97. /**
  98. * 刷新标签
  99. * @param string $tags 标签,多个标签以,分隔
  100. * @param int $archives_id 文档ID
  101. * @return bool
  102. */
  103. public static function refresh($tags, $archives_id)
  104. {
  105. $field = "nums";
  106. $tags = str_replace(',', ',', $tags);
  107. $tagsArr = explode(',', $tags);
  108. $tagsArr = array_unique(array_filter(array_map('strtolower', $tagsArr)));
  109. //取出所有标签列表
  110. $tagsList = Tag::alias('t')
  111. ->join('cms_taggable ta', 'ta.tag_id = t.id', 'RIGHT')
  112. ->where('ta.archives_id', $archives_id)
  113. ->field('t.*,ta.archives_id,ta.id AS taggable_id')
  114. ->select();
  115. foreach ($tagsList as $index => $item) {
  116. $item['name'] = strtolower($item['name']);
  117. if (!in_array($item['name'], $tagsArr)) {
  118. //统计减1
  119. isset($item[$field]) && $item->setDec($field);
  120. //删除标签
  121. Taggable::where('id', $item['taggable_id'])->delete();
  122. } else {
  123. $tagsArr = array_diff($tagsArr, [$item['name']]);
  124. }
  125. }
  126. if (!$tagsArr) {
  127. return true;
  128. }
  129. $insertTagIds = [];
  130. //取出剩余标签
  131. $tagList = self::where('name', 'in', $tagsArr)->select();
  132. foreach ($tagList as $index => $item) {
  133. $item['name'] = strtolower($item['name']);
  134. $item->setInc($field);
  135. $tagsArr = array_diff($tagsArr, [$item['name']]);
  136. $insertTagIds[] = $item['id'];
  137. }
  138. //剩余未插入的话题
  139. if ($tagsArr) {
  140. $originTagsArr = explode(',', $tags);
  141. foreach ($originTagsArr as $index => $item) {
  142. $name = strtolower($item);
  143. if (in_array($name, $tagsArr)) {
  144. $tagsArr = array_diff($tagsArr, [$name]);
  145. $object = self::create(['name' => $item, $field => 1]);
  146. $id = $object->id;
  147. $insertTagIds[] = $id;
  148. }
  149. }
  150. }
  151. //插入到taggable表
  152. $insertList = [];
  153. foreach ($insertTagIds as $index => $tag_id) {
  154. $insertList[] = ['tag_id' => $tag_id, 'archives_id' => $archives_id, 'createtime' => time()];
  155. }
  156. if ($insertList) {
  157. (new Taggable())->insertAll($insertList);
  158. }
  159. return true;
  160. }
  161. }