index.vue 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. <template>
  2. <view class="custom-class van-sticky" :style="computed.containerStyle({ fixed, height, zIndex })">
  3. <view :class="utils.bem('sticky-wrap', { fixed })" :style="computed.wrapStyle({ fixed, offsetTop, transform, zIndex })">
  4. <slot />
  5. </view>
  6. </view>
  7. </template>
  8. <script module="utils" lang="wxs" src="@/node_modules/@vant/weapp/dist/wxs/utils.wxs"></script>
  9. <script module="computed" lang="wxs" src="@/node_modules/@vant/weapp/dist/sticky/index.wxs"></script>
  10. <script>
  11. import { VantComponent } from '../common/component';
  12. import { pageScrollMixin } from '../mixins/page-scroll';
  13. const ROOT_ELEMENT = '.van-sticky';
  14. export default {
  15. data() {
  16. return {
  17. height: 0,
  18. fixed: false,
  19. transform: 0
  20. };
  21. },
  22. props: {
  23. zIndex: {
  24. type: Number,
  25. default: 99
  26. },
  27. offsetTop: {
  28. type: Number,
  29. default: 0
  30. },
  31. disabled: {
  32. type: Boolean
  33. },
  34. container: {
  35. type: null
  36. },
  37. scrollTop: {
  38. type: null
  39. }
  40. },
  41. mixins: [
  42. pageScrollMixin(function (event) {
  43. if (this.scrollTop != null) {
  44. return;
  45. }
  46. this.onScroll(event);
  47. })
  48. ],
  49. mounted() {
  50. this.onScroll();
  51. },
  52. methods: {
  53. onScroll({ scrollTop } = {}) {
  54. const { container, offsetTop, disabled } = this;
  55. if (disabled) {
  56. this.setDataAfterDiff({
  57. fixed: false,
  58. transform: 0
  59. });
  60. return;
  61. }
  62. this.scrollTop = scrollTop || this.scrollTop;
  63. if (typeof container === 'function') {
  64. Promise.all([this.getRect(ROOT_ELEMENT), this.getContainerRect()]).then(([root, container]) => {
  65. if (offsetTop + root.height > container.height + container.top) {
  66. this.setDataAfterDiff({
  67. fixed: false,
  68. transform: container.height - root.height
  69. });
  70. } else if (offsetTop >= root.top) {
  71. this.setDataAfterDiff({
  72. fixed: true,
  73. height: root.height,
  74. transform: 0
  75. });
  76. } else {
  77. this.setDataAfterDiff({
  78. fixed: false,
  79. transform: 0
  80. });
  81. }
  82. });
  83. return;
  84. }
  85. this.getRect(ROOT_ELEMENT).then((root) => {
  86. if (offsetTop >= root.top) {
  87. this.setDataAfterDiff({
  88. fixed: true,
  89. height: root.height
  90. });
  91. this.transform = 0;
  92. } else {
  93. this.setDataAfterDiff({
  94. fixed: false
  95. });
  96. }
  97. });
  98. },
  99. setDataAfterDiff(data) {
  100. this.$nextTick(() => {
  101. const diff = Object.keys(data).reduce((prev, key) => {
  102. if (data[key] !== this[key]) {
  103. prev[key] = data[key];
  104. }
  105. return prev;
  106. }, {});
  107. this.setData(diff);
  108. this.$emit('scroll', {
  109. scrollTop: this.scrollTop,
  110. isFixed: data.fixed || this.fixed
  111. });
  112. });
  113. },
  114. getContainerRect() {
  115. const nodesRef = this.container();
  116. return new Promise((resolve) => nodesRef.boundingClientRect(resolve).exec());
  117. }
  118. },
  119. watch: {
  120. offsetTop: {
  121. handler: function ({ scrollTop } = {}) {
  122. const { container, offsetTop, disabled } = this;
  123. if (disabled) {
  124. this.setDataAfterDiff({
  125. fixed: false,
  126. transform: 0
  127. });
  128. return;
  129. }
  130. this.scrollTop = scrollTop || this.scrollTop;
  131. if (typeof container === 'function') {
  132. Promise.all([this.getRect(ROOT_ELEMENT), this.getContainerRect()]).then(([root, container]) => {
  133. if (offsetTop + root.height > container.height + container.top) {
  134. this.setDataAfterDiff({
  135. fixed: false,
  136. transform: container.height - root.height
  137. });
  138. } else if (offsetTop >= root.top) {
  139. this.setDataAfterDiff({
  140. fixed: true,
  141. height: root.height,
  142. transform: 0
  143. });
  144. } else {
  145. this.setDataAfterDiff({
  146. fixed: false,
  147. transform: 0
  148. });
  149. }
  150. });
  151. return;
  152. }
  153. this.getRect(ROOT_ELEMENT).then((root) => {
  154. if (offsetTop >= root.top) {
  155. this.setDataAfterDiff({
  156. fixed: true,
  157. height: root.height
  158. });
  159. this.transform = 0;
  160. } else {
  161. this.setDataAfterDiff({
  162. fixed: false
  163. });
  164. }
  165. });
  166. },
  167. immediate: true
  168. },
  169. disabled: {
  170. handler: function ({ scrollTop } = {}) {
  171. const { container, offsetTop, disabled } = this;
  172. if (disabled) {
  173. this.setDataAfterDiff({
  174. fixed: false,
  175. transform: 0
  176. });
  177. return;
  178. }
  179. this.scrollTop = scrollTop || this.scrollTop;
  180. if (typeof container === 'function') {
  181. Promise.all([this.getRect(ROOT_ELEMENT), this.getContainerRect()]).then(([root, container]) => {
  182. if (offsetTop + root.height > container.height + container.top) {
  183. this.setDataAfterDiff({
  184. fixed: false,
  185. transform: container.height - root.height
  186. });
  187. } else if (offsetTop >= root.top) {
  188. this.setDataAfterDiff({
  189. fixed: true,
  190. height: root.height,
  191. transform: 0
  192. });
  193. } else {
  194. this.setDataAfterDiff({
  195. fixed: false,
  196. transform: 0
  197. });
  198. }
  199. });
  200. return;
  201. }
  202. this.getRect(ROOT_ELEMENT).then((root) => {
  203. if (offsetTop >= root.top) {
  204. this.setDataAfterDiff({
  205. fixed: true,
  206. height: root.height
  207. });
  208. this.transform = 0;
  209. } else {
  210. this.setDataAfterDiff({
  211. fixed: false
  212. });
  213. }
  214. });
  215. },
  216. immediate: true
  217. },
  218. container: {
  219. handler: function ({ scrollTop } = {}) {
  220. const { container, offsetTop, disabled } = this;
  221. if (disabled) {
  222. this.setDataAfterDiff({
  223. fixed: false,
  224. transform: 0
  225. });
  226. return;
  227. }
  228. this.scrollTop = scrollTop || this.scrollTop;
  229. if (typeof container === 'function') {
  230. Promise.all([this.getRect(ROOT_ELEMENT), this.getContainerRect()]).then(([root, container]) => {
  231. if (offsetTop + root.height > container.height + container.top) {
  232. this.setDataAfterDiff({
  233. fixed: false,
  234. transform: container.height - root.height
  235. });
  236. } else if (offsetTop >= root.top) {
  237. this.setDataAfterDiff({
  238. fixed: true,
  239. height: root.height,
  240. transform: 0
  241. });
  242. } else {
  243. this.setDataAfterDiff({
  244. fixed: false,
  245. transform: 0
  246. });
  247. }
  248. });
  249. return;
  250. }
  251. this.getRect(ROOT_ELEMENT).then((root) => {
  252. if (offsetTop >= root.top) {
  253. this.setDataAfterDiff({
  254. fixed: true,
  255. height: root.height
  256. });
  257. this.transform = 0;
  258. } else {
  259. this.setDataAfterDiff({
  260. fixed: false
  261. });
  262. }
  263. });
  264. },
  265. immediate: true
  266. },
  267. scrollTop: {
  268. handler: function (val) {
  269. this.onScroll({
  270. scrollTop: val
  271. });
  272. },
  273. immediate: true
  274. }
  275. }
  276. };
  277. </script>
  278. <style>
  279. @import './index.css';
  280. </style>