index.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. <template>
  2. <view class="van-circle">
  3. <canvas class="van-circle__canvas" :type="type" :style="'width: ' + utils.addUnit(size) + ';height:' + utils.addUnit(size)" id="van-circle" canvas-id="van-circle"></canvas>
  4. <view v-if="!text" class="van-circle__text">
  5. <slot></slot>
  6. </view>
  7. <cover-view v-else class="van-circle__text">{{ text }}</cover-view>
  8. </view>
  9. </template>
  10. <script module="utils" lang="wxs" src="@/node_modules/@vant/weapp/lib/wxs/utils.wxs"></script>
  11. <script>
  12. 'use strict';
  13. Object.defineProperty(exports, '__esModule', {
  14. value: true
  15. });
  16. var component_1 = require('../common/component');
  17. var utils_1 = require('../common/utils');
  18. var color_1 = require('../common/color');
  19. var canvas_1 = require('./canvas');
  20. function format(rate) {
  21. return Math.min(Math.max(rate, 0), 100);
  22. }
  23. var PERIMETER = 2 * Math.PI;
  24. var BEGIN_ANGLE = -Math.PI / 2;
  25. var STEP = 1;
  26. component_1.VantComponent({
  27. props: {
  28. text: String,
  29. lineCap: {
  30. type: String,
  31. value: 'round'
  32. },
  33. value: {
  34. type: Number,
  35. value: 0,
  36. observer: 'reRender'
  37. },
  38. speed: {
  39. type: Number,
  40. value: 50
  41. },
  42. size: {
  43. type: Number,
  44. value: 100,
  45. observer: function () {
  46. this.drawCircle(this.currentValue);
  47. }
  48. },
  49. fill: String,
  50. layerColor: {
  51. type: String,
  52. value: color_1.WHITE
  53. },
  54. color: {
  55. type: [String, Object],
  56. value: color_1.BLUE,
  57. observer: function () {
  58. var that = this;
  59. this.setHoverColor().then(function () {
  60. that.drawCircle(that.currentValue);
  61. });
  62. }
  63. },
  64. type: {
  65. type: String,
  66. value: ''
  67. },
  68. strokeWidth: {
  69. type: Number,
  70. value: 4
  71. },
  72. clockwise: {
  73. type: Boolean,
  74. value: true
  75. }
  76. },
  77. data: {
  78. hoverColor: color_1.BLUE
  79. },
  80. methods: {
  81. getContext: function () {
  82. var that = this;
  83. var _a = this;
  84. var type = _a.type;
  85. var size = _a.size;
  86. if (type === '') {
  87. var ctx = uni.createCanvasContext('van-circle', this);
  88. return Promise.resolve(ctx);
  89. }
  90. var dpr = uni.getSystemInfoSync().pixelRatio;
  91. return new Promise(function (resolve) {
  92. uni.createSelectorQuery()
  93. .in(that)
  94. .select('#van-circle')
  95. .node()
  96. .exec(function (res) {
  97. var canvas = res[0].node;
  98. var ctx = canvas.getContext(type);
  99. if (!that.inited) {
  100. that.inited = true;
  101. canvas.width = size * dpr;
  102. canvas.height = size * dpr;
  103. ctx.scale(dpr, dpr);
  104. }
  105. resolve(canvas_1.adaptor(ctx));
  106. });
  107. });
  108. },
  109. setHoverColor: function () {
  110. var that = this;
  111. var _a = this;
  112. var color = _a.color;
  113. var size = _a.size;
  114. if (utils_1.isObj(color)) {
  115. return this.getContext().then(function (context) {
  116. var LinearColor = context.createLinearGradient(size, 0, 0, 0);
  117. Object.keys(color)
  118. .sort(function (a, b) {
  119. return parseFloat(a) - parseFloat(b);
  120. })
  121. .map(function (key) {
  122. return LinearColor.addColorStop(parseFloat(key) / 100, color[key]);
  123. });
  124. that.hoverColor = LinearColor;
  125. });
  126. }
  127. this.hoverColor = color;
  128. return Promise.resolve();
  129. },
  130. presetCanvas: function (context, strokeStyle, beginAngle, endAngle, fill) {
  131. var _a = this;
  132. var strokeWidth = _a.strokeWidth;
  133. var lineCap = _a.lineCap;
  134. var clockwise = _a.clockwise;
  135. var size = _a.size;
  136. var position = size / 2;
  137. var radius = position - strokeWidth / 2;
  138. context.setStrokeStyle(strokeStyle);
  139. context.setLineWidth(strokeWidth);
  140. context.setLineCap(lineCap);
  141. context.beginPath();
  142. context.arc(position, position, radius, beginAngle, endAngle, !clockwise);
  143. context.stroke();
  144. if (fill) {
  145. context.setFillStyle(fill);
  146. context.fill();
  147. }
  148. },
  149. renderLayerCircle: function (context) {
  150. var _a = this;
  151. var layerColor = _a.layerColor;
  152. var fill = _a.fill;
  153. this.presetCanvas(context, layerColor, 0, PERIMETER, fill);
  154. },
  155. renderHoverCircle: function (context, formatValue) {
  156. var clockwise = this.clockwise;
  157. // 结束角度
  158. var progress = PERIMETER * (formatValue / 100);
  159. var endAngle = clockwise ? BEGIN_ANGLE + progress : 3 * Math.PI - (BEGIN_ANGLE + progress);
  160. this.presetCanvas(context, this.hoverColor, BEGIN_ANGLE, endAngle);
  161. },
  162. drawCircle: function (currentValue) {
  163. var that = this;
  164. var size = this.size;
  165. this.getContext().then(function (context) {
  166. context.clearRect(0, 0, size, size);
  167. that.renderLayerCircle(context);
  168. var formatValue = format(currentValue);
  169. if (formatValue !== 0) {
  170. that.renderHoverCircle(context, formatValue);
  171. }
  172. context.draw();
  173. });
  174. },
  175. reRender: function () {
  176. var that = this;
  177. // tofector 动画暂时没有想到好的解决方案
  178. var _a = this;
  179. var value = _a.value;
  180. var speed = _a.speed;
  181. if (speed <= 0 || speed > 1000) {
  182. this.drawCircle(value);
  183. return;
  184. }
  185. this.clearInterval();
  186. this.currentValue = this.currentValue || 0;
  187. this.interval = setInterval(function () {
  188. if (that.currentValue !== value) {
  189. if (that.currentValue < value) {
  190. that.currentValue += STEP;
  191. } else {
  192. that.currentValue -= STEP;
  193. }
  194. that.drawCircle(that.currentValue);
  195. } else {
  196. that.clearInterval();
  197. }
  198. }, 1000 / speed);
  199. },
  200. clearInterval: function () {
  201. if (this.interval) {
  202. clearInterval(this.interval);
  203. this.interval = null;
  204. }
  205. }
  206. },
  207. mounted: function () {
  208. var that = this;
  209. this.currentValue = this.value;
  210. this.setHoverColor().then(function () {
  211. that.drawCircle(that.currentValue);
  212. });
  213. },
  214. destroyed: function () {
  215. this.clearInterval();
  216. }
  217. });
  218. </script>
  219. <style>
  220. @import './index.css';
  221. </style>