index.vue 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. <template>
  2. <view class="van-index-bar">
  3. <slot />
  4. <view
  5. v-if="showSidebar"
  6. class="van-index-bar__sidebar"
  7. @tap.stop.prevent="onClick"
  8. @touchmove.stop.prevent="onTouchMove"
  9. @touchend.stop.prevent="onTouchStop"
  10. @touchcancel.stop.prevent="onTouchStop"
  11. >
  12. <view
  13. class="van-index-bar__index"
  14. :style="'z-index: ' + (zIndex + 1) + '; color: ' + (activeAnchorIndex === index ? highlightColor : '')"
  15. :data-index="index"
  16. v-for="(item, index) in indexList"
  17. :key="index"
  18. >
  19. {{ item }}
  20. </view>
  21. </view>
  22. </view>
  23. </template>
  24. <script>
  25. 'use strict';
  26. Object.defineProperty(exports, '__esModule', {
  27. value: true
  28. });
  29. var component_1 = require('../common/component');
  30. var color_1 = require('../common/color');
  31. var page_scroll_1 = require('../mixins/page-scroll');
  32. var indexList = function () {
  33. var indexList = [];
  34. var charCodeOfA = 'A'.charCodeAt(0);
  35. for (var i = 0; i < 26; i++) {
  36. indexList.push(String.fromCharCode(charCodeOfA + i));
  37. }
  38. return indexList;
  39. };
  40. component_1.VantComponent({
  41. relation: {
  42. name: 'index-anchor',
  43. type: 'descendant',
  44. current: 'index-bar',
  45. linked: function () {
  46. this.updateData();
  47. },
  48. unlinked: function () {
  49. this.updateData();
  50. }
  51. },
  52. props: {
  53. sticky: {
  54. type: Boolean,
  55. value: true
  56. },
  57. zIndex: {
  58. type: Number,
  59. value: 1
  60. },
  61. highlightColor: {
  62. type: String,
  63. value: color_1.GREEN
  64. },
  65. stickyOffsetTop: {
  66. type: Number,
  67. value: 0
  68. },
  69. indexList: {
  70. type: Array,
  71. value: indexList()
  72. }
  73. },
  74. mixins: [
  75. page_scroll_1.pageScrollMixin(function (event) {
  76. this.scrollTop = event.scrollTop || 0;
  77. this.onScroll();
  78. })
  79. ],
  80. data: {
  81. activeAnchorIndex: null,
  82. showSidebar: false
  83. },
  84. created: function () {
  85. this.scrollTop = 0;
  86. },
  87. methods: {
  88. updateData: function () {
  89. var that = this;
  90. this.$nextTick(function () {
  91. if (that.timer != null) {
  92. clearTimeout(that.timer);
  93. }
  94. that.timer = setTimeout(function () {
  95. that.setData({
  96. showSidebar: !!that.children.length
  97. });
  98. that.setRect().then(function () {
  99. that.onScroll();
  100. });
  101. }, 0);
  102. });
  103. },
  104. setRect: function () {
  105. return Promise.all([this.setAnchorsRect(), this.setListRect(), this.setSiderbarRect()]);
  106. },
  107. setAnchorsRect: function () {
  108. var that = this;
  109. return Promise.all(
  110. this.children.map(function (anchor) {
  111. return anchor.getRect('.van-index-anchor-wrapper').then(function (rect) {
  112. Object.assign(anchor, {
  113. height: rect.height,
  114. top: rect.top + that.scrollTop
  115. });
  116. });
  117. })
  118. );
  119. },
  120. setListRect: function () {
  121. var that = this;
  122. return this.getRect('.van-index-bar').then(function (rect) {
  123. Object.assign(that, {
  124. height: rect.height,
  125. top: rect.top + that.scrollTop
  126. });
  127. });
  128. },
  129. setSiderbarRect: function () {
  130. var that = this;
  131. return this.getRect('.van-index-bar__sidebar').then(function (res) {
  132. that.sidebar = {
  133. height: res.height,
  134. top: res.top
  135. };
  136. });
  137. },
  138. setDiffData: function (_a) {
  139. var target = _a.target;
  140. var data = _a.data;
  141. var diffData = {};
  142. Object.keys(data).forEach(function (key) {
  143. if (target.data[key] !== data[key]) {
  144. diffData[key] = data[key];
  145. }
  146. });
  147. if (Object.keys(diffData).length) {
  148. target.setData(diffData);
  149. }
  150. },
  151. getAnchorRect: function (anchor) {
  152. return anchor.getRect('.van-index-anchor-wrapper').then(function (rect) {
  153. return {
  154. height: rect.height,
  155. top: rect.top
  156. };
  157. });
  158. },
  159. getActiveAnchorIndex: function () {
  160. var that = this;
  161. var children = that.children;
  162. var scrollTop = that.scrollTop;
  163. var _b = this;
  164. var sticky = _b.sticky;
  165. var stickyOffsetTop = _b.stickyOffsetTop;
  166. for (var i = this.children.length - 1; i >= 0; i--) {
  167. var preAnchorHeight = i > 0 ? children[i - 1].height : 0;
  168. var reachTop = sticky ? preAnchorHeight + stickyOffsetTop : 0;
  169. if (reachTop + scrollTop >= children[i].top) {
  170. return i;
  171. }
  172. }
  173. return -1;
  174. },
  175. onScroll: function () {
  176. var that = this;
  177. var _a = this;
  178. var _b = _a.children;
  179. var children = _b === void 0 ? [] : _b;
  180. var scrollTop = _a.scrollTop;
  181. if (!children.length) {
  182. return;
  183. }
  184. var _c = this;
  185. var sticky = _c.sticky;
  186. var stickyOffsetTop = _c.stickyOffsetTop;
  187. var zIndex = _c.zIndex;
  188. var highlightColor = _c.highlightColor;
  189. var active = this.getActiveAnchorIndex();
  190. this.setDiffData({
  191. target: this,
  192. data: {
  193. activeAnchorIndex: active
  194. }
  195. });
  196. if (sticky) {
  197. var isActiveAnchorSticky_1 = false;
  198. if (active !== -1) {
  199. isActiveAnchorSticky_1 = children[active].top <= stickyOffsetTop + scrollTop;
  200. }
  201. children.forEach(function (item, index) {
  202. if (index === active) {
  203. var wrapperStyle = '';
  204. var anchorStyle = '\n color: ' + highlightColor + ';\n ';
  205. if (isActiveAnchorSticky_1) {
  206. wrapperStyle = '\n height: ' + children[index].height + 'px;\n ';
  207. anchorStyle =
  208. '\n position: fixed;\n top: ' +
  209. stickyOffsetTop +
  210. 'px;\n z-index: ' +
  211. zIndex +
  212. ';\n color: ' +
  213. highlightColor +
  214. ';\n ';
  215. }
  216. that.setDiffData({
  217. target: item,
  218. data: {
  219. active: true,
  220. anchorStyle: anchorStyle,
  221. wrapperStyle: wrapperStyle
  222. }
  223. });
  224. } else if (index === active - 1) {
  225. var currentAnchor = children[index];
  226. var currentOffsetTop = currentAnchor.top;
  227. var targetOffsetTop = index === children.length - 1 ? that.top : children[index + 1].top;
  228. var parentOffsetHeight = targetOffsetTop - currentOffsetTop;
  229. var translateY = parentOffsetHeight - currentAnchor.height;
  230. var anchorStyle =
  231. '\n position: relative;\n transform: translate3d(0, ' +
  232. translateY +
  233. 'px, 0);\n z-index: ' +
  234. zIndex +
  235. ';\n color: ' +
  236. highlightColor +
  237. ';\n ';
  238. that.setDiffData({
  239. target: item,
  240. data: {
  241. active: true,
  242. anchorStyle: anchorStyle
  243. }
  244. });
  245. } else {
  246. that.setDiffData({
  247. target: item,
  248. data: {
  249. active: false,
  250. anchorStyle: '',
  251. wrapperStyle: ''
  252. }
  253. });
  254. }
  255. });
  256. }
  257. },
  258. onClick: function (event) {
  259. this.scrollToAnchor(event.target.dataset.index);
  260. },
  261. onTouchMove: function (event) {
  262. var sidebarLength = this.children.length;
  263. var touch = event.touches[0];
  264. var itemHeight = this.sidebar.height / sidebarLength;
  265. var index = Math.floor((touch.clientY - this.sidebar.top) / itemHeight);
  266. if (index < 0) {
  267. index = 0;
  268. } else if (index > sidebarLength - 1) {
  269. index = sidebarLength - 1;
  270. }
  271. this.scrollToAnchor(index);
  272. },
  273. onTouchStop: function () {
  274. this.scrollToAnchorIndex = null;
  275. },
  276. scrollToAnchor: function (index) {
  277. var that = this;
  278. if (typeof index !== 'number' || this.scrollToAnchorIndex === index) {
  279. return;
  280. }
  281. this.scrollToAnchorIndex = index;
  282. var anchor = this.children.find(function (item) {
  283. return item.data.index === that.indexList[index];
  284. });
  285. if (anchor) {
  286. anchor.scrollIntoView(this.scrollTop);
  287. this.$emit('select', anchor.data.index);
  288. }
  289. }
  290. }
  291. });
  292. </script>
  293. <style>
  294. @import './index.css';
  295. </style>