LineGraph.java 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662
  1. /*
  2. * Created by Daniel Nadeau
  3. * daniel.nadeau01@gmail.com
  4. * danielnadeau.blogspot.com
  5. *
  6. * Licensed to the Apache Software Foundation (ASF) under one
  7. or more contributor license agreements. See the NOTICE file
  8. distributed with this work for additional information
  9. regarding copyright ownership. The ASF licenses this file
  10. to you under the Apache License, Version 2.0 (the
  11. "License"); you may not use this file except in compliance
  12. with the License. You may obtain a copy of the License at
  13. http://www.apache.org/licenses/LICENSE-2.0
  14. Unless required by applicable law or agreed to in writing,
  15. software distributed under the License is distributed on an
  16. "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
  17. KIND, either express or implied. See the License for the
  18. specific language governing permissions and limitations
  19. under the License.
  20. */
  21. package com.echo.holographlibrary;
  22. import android.content.Context;
  23. import android.graphics.Bitmap;
  24. import android.graphics.Bitmap.Config;
  25. import android.graphics.Canvas;
  26. import android.graphics.Color;
  27. import android.graphics.Paint;
  28. import android.graphics.Path;
  29. import android.graphics.Path.Direction;
  30. import android.graphics.Point;
  31. import android.graphics.PorterDuffXfermode;
  32. import android.graphics.Region;
  33. import android.util.AttributeSet;
  34. import android.util.TypedValue;
  35. import android.view.MotionEvent;
  36. import android.view.View;
  37. import java.util.ArrayList;
  38. public class LineGraph extends View {
  39. public interface AxisDataConverter {
  40. public String convertDataForX_Position(double x);
  41. public String convertDataForY_Position(double y);
  42. }
  43. private final static int AXIS_LABEL_FONT_SIZE = 10;
  44. private ArrayList<Line> lines = new ArrayList<Line>();
  45. Paint paint = new Paint();
  46. private double minY = 0, minX = 0;
  47. private double maxY = 0, maxX = 0;
  48. private double rangeYRatio = 0;
  49. private double rangeXRatio = 0;
  50. private boolean isMaxYUserSet = false;
  51. private boolean isMaxXUserSet = false;
  52. private int lineToFill = -1;
  53. private int indexSelected = -1;
  54. private OnPointClickedListener listener;
  55. private Bitmap fullImage;
  56. private boolean shouldUpdate = false;
  57. static final float bottomPadding = 40, topPadding = 10;
  58. static final float rightPadding = 10;
  59. static final float leftPadding = 50;
  60. static final float sidePadding = rightPadding + leftPadding;
  61. private float xAxisStep = 4;
  62. private float yAxisStep = 4;
  63. private AxisDataConverter converter;
  64. private Context mContext;
  65. public void setxAxisStep(float step){
  66. this.xAxisStep = step;
  67. }
  68. public void setYAxisStep(float step){
  69. this.yAxisStep = step;
  70. }
  71. public void setConverter(AxisDataConverter conv){
  72. this.converter = conv;
  73. }
  74. public LineGraph(Context context){
  75. super(context);
  76. this.mContext = context;
  77. this.setWillNotDraw(false);
  78. }
  79. public LineGraph(Context context, AttributeSet attrs) {
  80. super(context, attrs);
  81. this.mContext = context;
  82. this.setWillNotDraw(false);
  83. }
  84. public void setMinY(float minY){
  85. }
  86. public void removeAllLines(){
  87. while (lines.size() > 0){
  88. lines.remove(0);
  89. }
  90. shouldUpdate = true;
  91. postInvalidate();
  92. }
  93. public void addLine(Line line) {
  94. lines.add(line);
  95. shouldUpdate = true;
  96. postInvalidate();
  97. }
  98. public void addPointToLine(int lineIndex, double x, double y){
  99. addPointToLine(lineIndex, (float) x, (float) y);
  100. }
  101. public void addPointToLine(int lineIndex, float x, float y){
  102. LinePoint p = new LinePoint(x, y);
  103. addPointToLine(lineIndex, p);
  104. }
  105. public double getRangeYRatio(){
  106. return rangeYRatio;
  107. }
  108. public void setRangeYRatio(double rr){
  109. this.rangeYRatio = rr;
  110. }
  111. public double getRangeXRatio(){
  112. return rangeXRatio;
  113. }
  114. public void setRangeXRatio(double rr){
  115. this.rangeXRatio = rr;
  116. }
  117. public void addPointToLine(int lineIndex, LinePoint point){
  118. Line line = getLine(lineIndex);
  119. line.addPoint(point);
  120. lines.set(lineIndex, line);
  121. resetLimits();
  122. shouldUpdate = true;
  123. postInvalidate();
  124. }
  125. public void addPointsToLine(int lineIndex, LinePoint[] points){
  126. Line line = getLine(lineIndex);
  127. for(LinePoint point : points){
  128. line.addPoint(point);
  129. }
  130. lines.set(lineIndex, line);
  131. resetLimits();
  132. shouldUpdate = true;
  133. postInvalidate();
  134. }
  135. public void removeAllPointsAfter(int lineIndex, double x){
  136. removeAllPointsBetween(lineIndex, x, getMaxX());
  137. }
  138. public void removeAllPointsBefore(int lineIndex, double x){
  139. removeAllPointsBetween(lineIndex, getMinX(), x);
  140. }
  141. public void removeAllPointsBetween(int lineIndex, double startX, double finishX){
  142. Line line = getLine(lineIndex);
  143. LinePoint[] pts = new LinePoint[line.getPoints().size()];
  144. pts = line.getPoints().toArray(pts);
  145. for(LinePoint point : pts){
  146. if(point.getX() >= startX && point.getX() <= finishX)
  147. line.removePoint(point);
  148. }
  149. lines.set(lineIndex, line);
  150. resetLimits();
  151. shouldUpdate = true;
  152. postInvalidate();
  153. }
  154. public void removePointsFromLine(int lineIndex, LinePoint[] points){
  155. Line line = getLine(lineIndex);
  156. for(LinePoint point : points){
  157. line.removePoint(point);
  158. }
  159. lines.set(lineIndex, line);
  160. resetLimits();
  161. shouldUpdate = true;
  162. postInvalidate();
  163. }
  164. public void removePointFromLine(int lineIndex, float x, float y){
  165. LinePoint p = null;
  166. Line line = getLine(lineIndex);
  167. p = line.getPoint(x, y);
  168. removePointFromLine(lineIndex, p);
  169. }
  170. public void removePointFromLine(int lineIndex, LinePoint point){
  171. Line line = getLine(lineIndex);
  172. line.removePoint(point);
  173. lines.set(lineIndex, line);
  174. resetLimits();
  175. shouldUpdate = true;
  176. postInvalidate();
  177. }
  178. public void resetYLimits(){
  179. double range = getMaxY() - getMinY();
  180. setRangeY(getMinY()-range*getRangeYRatio(), getMaxY()+range*getRangeYRatio());
  181. isMaxYUserSet = false;
  182. }
  183. public void resetXLimits(){
  184. double range = getMaxX() - getMinX();
  185. setRangeX(getMinX()-range*getRangeXRatio(), getMaxX()+range*getRangeXRatio());
  186. isMaxXUserSet = false;
  187. }
  188. public void resetLimits() {
  189. resetYLimits();
  190. resetXLimits();
  191. }
  192. public ArrayList<Line> getLines() {
  193. return lines;
  194. }
  195. public void setLineToFill(int indexOfLine) {
  196. this.lineToFill = indexOfLine;
  197. shouldUpdate = true;
  198. postInvalidate();
  199. }
  200. public int getLineToFill(){
  201. return lineToFill;
  202. }
  203. public void setLines(ArrayList<Line> lines) {
  204. this.lines = lines;
  205. }
  206. public Line getLine(int index) {
  207. return lines.get(index);
  208. }
  209. public int getSize(){
  210. return lines.size();
  211. }
  212. public void setRangeY(float min, float max) {
  213. minY = min;
  214. maxY = max;
  215. isMaxYUserSet = true;
  216. }
  217. public void setRangeY(double min, double max){
  218. minY = min;
  219. maxY = max;
  220. isMaxYUserSet = true;
  221. }
  222. public void setRangeX(float min, float max) {
  223. minX = min;
  224. maxX = max;
  225. isMaxXUserSet = true;
  226. }
  227. public void setRangeX(double min, double max){
  228. minX = min;
  229. maxX = max;
  230. isMaxXUserSet = true;
  231. }
  232. public double getMaxY(){
  233. if (isMaxYUserSet)return maxY;
  234. double max = lines.get(0).getPoint(0).getY();
  235. for (Line line : lines){
  236. for (LinePoint point : line.getPoints()){
  237. max = point.getY() > max ? point.getY() : max;
  238. }
  239. }
  240. maxY = max;
  241. return maxY;
  242. }
  243. public double getMinY(){
  244. if (isMaxYUserSet)return minY;
  245. double min = lines.get(0).getPoint(0).getY();
  246. for (Line line : lines){
  247. for (LinePoint point : line.getPoints()){
  248. min = point.getY() < min ? point.getY() : min;
  249. }
  250. }
  251. minY = min;
  252. return minY;
  253. }
  254. public double getMinLimY(){
  255. return minY;
  256. }
  257. public double getMaxLimY(){
  258. return maxY;
  259. }
  260. public double getMinLimX(){
  261. if (isMaxXUserSet) {
  262. return minX;
  263. }
  264. else {
  265. return getMinX();
  266. }
  267. }
  268. public double getMaxLimX(){
  269. if (isMaxXUserSet) {
  270. return maxX;
  271. }
  272. else {
  273. return getMaxX();
  274. }
  275. }
  276. public double getMaxX(){
  277. double max = lines.size() > 0 ? lines.get(0).getPoint(0).getX() : 0;
  278. for (Line line : lines){
  279. for (LinePoint point : line.getPoints()){
  280. max =Math.max(point.getX(), max);// point.getX() > max ? point.getX() : max;
  281. }
  282. }
  283. maxX = max;
  284. return maxX;
  285. }
  286. public double getMinX(){
  287. double min = lines.size() > 0 ? lines.get(0).getPoint(0).getX() : 0;
  288. for (Line line : lines){
  289. for (LinePoint point : line.getPoints()){
  290. min =Math.min(point.getX(), min);// point.getX() < min ? point.getX() : min;
  291. }
  292. }
  293. minX = min;
  294. return minX;
  295. }
  296. private String getX_AxisTitle(double x){
  297. if (this.converter == null)return "" + (long)x;
  298. return this.converter.convertDataForX_Position(x);
  299. }
  300. private String getY_AxisTitle(double y){
  301. if (this.converter == null)return "" + (long)y;
  302. return this.converter.convertDataForY_Position(y);
  303. }
  304. public void onDraw(Canvas ca) {
  305. super.onDraw(ca);
  306. if (fullImage == null || shouldUpdate) {
  307. fullImage = Bitmap.createBitmap(getWidth(), getHeight(), Config.ARGB_8888);
  308. Canvas canvas = new Canvas(fullImage);
  309. paint.reset();
  310. Path path = new Path();
  311. float usableHeight = getHeight() - bottomPadding - topPadding;
  312. float usableWidth = getWidth() - 2*sidePadding;
  313. double maxY = getMaxLimY();
  314. double minY = getMinLimY();
  315. double maxX = getMaxLimX();
  316. double minX = getMinLimX();
  317. // DRAW THE BACKGROUND
  318. //this.drawBackground(canvas);
  319. // DRAW THE AXIS
  320. this.drawAxis(canvas);
  321. paint.reset();
  322. paint.setAntiAlias(true);
  323. // DRAW LINES
  324. for (Line line : lines){
  325. int count = 0;
  326. float lastXPixels = 0, newYPixels = 0;
  327. float lastYPixels = 0, newXPixels = 0;
  328. paint.setColor(line.getColor());
  329. paint.setStrokeWidth(getStrokeWidth(line));
  330. for (LinePoint p : line.getPoints()){
  331. float yPercent =(float) ((p.getY()-minY)/(maxY - minY));
  332. float xPercent = (float)((p.getX()-minX)/(maxX - minX));
  333. if (count == 0){
  334. lastXPixels = sidePadding + (xPercent*usableWidth);
  335. lastYPixels = getHeight() - bottomPadding - (usableHeight*yPercent);
  336. } else {
  337. newXPixels = sidePadding + (xPercent*usableWidth);
  338. newYPixels = getHeight() - bottomPadding - (usableHeight*yPercent);
  339. canvas.drawLine(lastXPixels, lastYPixels, newXPixels, newYPixels, paint);
  340. lastXPixels = newXPixels;
  341. lastYPixels = newYPixels;
  342. }
  343. count++;
  344. }
  345. }
  346. int pointCount = 0;
  347. // DRAW POINTS
  348. for (Line line : lines){
  349. paint.setColor(line.getColor());
  350. paint.setStrokeWidth(getStrokeWidth(line));
  351. paint.setStrokeCap(Paint.Cap.ROUND);
  352. if (line.isShowingPoints()){
  353. for (LinePoint p : line.getPoints()){
  354. float yPercent =(float) ((p.getY()-minY)/(maxY - minY));
  355. float xPercent =(float) ((p.getX()-minX)/(maxX - minX));
  356. float xPixels = sidePadding + (xPercent*usableWidth);
  357. float yPixels = getHeight() - bottomPadding - (usableHeight*yPercent);
  358. int outerRadius;
  359. if (line.isUsingDips()) {
  360. outerRadius = getPixelForDip(line.getStrokeWidth() + 4);
  361. }
  362. else {
  363. outerRadius = line.getStrokeWidth() + 4;
  364. }
  365. int innerRadius = outerRadius / 2;
  366. paint.setColor(p.getColor());
  367. canvas.drawCircle(xPixels,(float) yPixels, outerRadius, paint);
  368. paint.setColor(Color.WHITE);
  369. canvas.drawCircle(xPixels,(float) yPixels, innerRadius, paint);
  370. Path path2 = new Path();
  371. path2.addCircle(xPixels,(float) yPixels, 30, Direction.CW);
  372. p.setPath(path2);
  373. p.setRegion(new Region((int)(xPixels-30), (int)(yPixels-30), (int)(xPixels+30), (int)(yPixels+30)));
  374. if (indexSelected == pointCount && listener != null){
  375. paint.setColor(p.getColor());
  376. paint.setAlpha(100);
  377. canvas.drawPath(p.getPath(), paint);
  378. paint.setAlpha(255);
  379. }
  380. pointCount++;
  381. }
  382. }
  383. }
  384. shouldUpdate = false;
  385. }
  386. ca.drawBitmap(fullImage, 0, 0, null);
  387. }
  388. private void drawAxis(Canvas canvas){
  389. paint.reset();
  390. double maxX = getMaxLimX();
  391. double minX = getMinLimX();
  392. float usableWidth = getWidth() - 2*sidePadding;
  393. float usableHeight = getHeight() - bottomPadding - topPadding;
  394. float yPixels = getHeight() - (bottomPadding*3/5);
  395. // DRAW SEPERATOR
  396. paint.setColor(Color.BLACK);
  397. paint.setAlpha(50);
  398. paint.setAntiAlias(true);
  399. // x Axis
  400. canvas.drawLine(leftPadding ,yPixels, getWidth()-sidePadding, yPixels, paint);
  401. // y Axis
  402. canvas.drawLine(leftPadding ,topPadding, leftPadding, yPixels, paint);
  403. paint.setAlpha(255);
  404. this.paint.setTextSize(AXIS_LABEL_FONT_SIZE * mContext.getResources().getDisplayMetrics().scaledDensity);
  405. // Draw y-axis label text
  406. //double skippedValue = (maxY - minY ) / (Math.max(1., yAxisStep));
  407. double step = Math.max(1., (maxY - minY) / (Math.max(1., yAxisStep)));
  408. for (double y = minY; y <= maxY; y+=step){
  409. double yPercent = (y-minY)/(maxY - minY);
  410. double newYPixels = topPadding + (yPercent*usableHeight);
  411. canvas.drawLine((float)leftPadding,(float)newYPixels,(float)leftPadding-5.f,(float)newYPixels, paint);
  412. String title = this.getY_AxisTitle(maxY - y);
  413. float textwidth = (this.paint.measureText(title));
  414. canvas.drawText(title, 5.f ,(float)newYPixels + (textwidth/2), this.paint);
  415. //value+=skippedValue;
  416. }
  417. // Draw x-axis label text
  418. //value = minX;
  419. //skippedValue = (maxX - minX ) / (Math.max(1,xAxisStep));
  420. step = Math.max(1, (maxX - minX) / (Math.max(1, xAxisStep)));
  421. for (double x = minX; x <= maxX; x+=step){
  422. double xPercent = (x-minX)/(maxX - minX);
  423. double newXPixels = sidePadding + (xPercent*usableWidth);
  424. canvas.drawLine((float)newXPixels,(float) yPixels + 5,(float) newXPixels,(float) yPixels, paint);
  425. String title = this.getX_AxisTitle(x);
  426. float textwidth = (this.paint.measureText(title));
  427. //this.paint.setTextSize(AXIS_LABEL_FONT_SIZE * mContext.getResources().getDisplayMetrics().scaledDensity);
  428. canvas.drawText(title,(float) newXPixels - (textwidth/2),(float) yPixels + (bottomPadding / 2), this.paint);
  429. //value+=skippedValue;
  430. }
  431. paint.reset();
  432. }
  433. private void drawBackground(Canvas canvas){
  434. paint.reset();
  435. float usableHeight = getHeight() - bottomPadding - topPadding;
  436. float usableWidth = getWidth() - 2*sidePadding;
  437. double maxY = getMaxLimY();
  438. double minY = getMinLimY();
  439. double maxX = getMaxLimX();
  440. double minX = getMinLimX();
  441. Path path = new Path();
  442. // DRAW THE BACKGROUND
  443. int lineCount = 0;
  444. for (Line line : lines){
  445. int count = 0;
  446. float firstXPixels = 0, lastXPixels = 0, newYPixels = 0;
  447. float lastYPixels = 0, newXPixels = 0;
  448. if (lineCount == lineToFill){
  449. paint.setColor(Color.BLACK);
  450. paint.setAlpha(30);
  451. paint.setStrokeWidth(2);
  452. for (int i = 10; i-getWidth() < getHeight(); i = i+20){
  453. canvas.drawLine(i, getHeight()-bottomPadding, 0, getHeight()-bottomPadding-i, paint);
  454. }
  455. paint.setXfermode(new PorterDuffXfermode(android.graphics.PorterDuff.Mode.CLEAR));
  456. for (LinePoint p : line.getPoints()){
  457. float yPercent =(float) ((p.getY()-minY)/(maxY - minY));
  458. float xPercent =(float) ((p.getX()-minX)/(maxX - minX));
  459. if (count == 0){
  460. lastXPixels = sidePadding + (xPercent*usableWidth);
  461. lastYPixels = getHeight() - bottomPadding - (usableHeight*yPercent);
  462. firstXPixels = lastXPixels;
  463. path.moveTo(lastXPixels, lastYPixels);
  464. } else {
  465. newXPixels = sidePadding + (xPercent*usableWidth);
  466. newYPixels = getHeight() - bottomPadding - (usableHeight*yPercent);
  467. path.lineTo(newXPixels, newYPixels);
  468. Path pa = new Path();
  469. pa.moveTo(lastXPixels, lastYPixels);
  470. pa.lineTo(newXPixels, newYPixels);
  471. pa.lineTo(newXPixels, 0);
  472. pa.lineTo(lastXPixels, 0);
  473. pa.close();
  474. canvas.drawPath(pa, paint);
  475. lastXPixels = newXPixels;
  476. lastYPixels = newYPixels;
  477. }
  478. count++;
  479. }
  480. path.reset();
  481. path.moveTo(0, getHeight()-bottomPadding);
  482. path.lineTo(sidePadding, getHeight()-bottomPadding);
  483. path.lineTo(sidePadding, 0);
  484. path.lineTo(0, 0);
  485. path.close();
  486. canvas.drawPath(path, paint);
  487. path.reset();
  488. path.moveTo(getWidth(), getHeight()-bottomPadding);
  489. path.lineTo(getWidth()-sidePadding, getHeight()-bottomPadding);
  490. path.lineTo(getWidth()-sidePadding, 0);
  491. path.lineTo(getWidth(), 0);
  492. path.close();
  493. canvas.drawPath(path, paint);
  494. }
  495. lineCount++;
  496. }
  497. this.paint.reset();
  498. }
  499. private int getStrokeWidth(Line line) {
  500. int strokeWidth;
  501. if (line.isUsingDips()) {
  502. strokeWidth = getPixelForDip(line.getStrokeWidth());
  503. }
  504. else {
  505. strokeWidth = line.getStrokeWidth();
  506. }
  507. return strokeWidth;
  508. }
  509. private int getPixelForDip(int dipValue) {
  510. return (int) TypedValue.applyDimension(
  511. TypedValue.COMPLEX_UNIT_DIP,
  512. dipValue,
  513. getResources().getDisplayMetrics());
  514. }
  515. @Override
  516. public boolean onTouchEvent(MotionEvent event) {
  517. Point point = new Point();
  518. point.x = (int) event.getX();
  519. point.y = (int) event.getY();
  520. int count = 0;
  521. int lineCount = 0;
  522. int pointCount = 0;
  523. Region r = new Region();
  524. for (Line line : lines){
  525. pointCount = 0;
  526. for (LinePoint p : line.getPoints()){
  527. if (p.getPath() != null && p.getRegion() != null){
  528. r.setPath(p.getPath(), p.getRegion());
  529. if (r.contains((int)point.x,(int) point.y) && event.getAction() == MotionEvent.ACTION_DOWN){
  530. indexSelected = count;
  531. } else if (event.getAction() == MotionEvent.ACTION_UP){
  532. if (r.contains((int)point.x,(int) point.y) && listener != null){
  533. listener.onClick(lineCount, pointCount);
  534. }
  535. indexSelected = -1;
  536. }
  537. }
  538. pointCount++;
  539. count++;
  540. }
  541. lineCount++;
  542. }
  543. if (event.getAction() == MotionEvent.ACTION_DOWN || event.getAction() == MotionEvent.ACTION_UP){
  544. shouldUpdate = true;
  545. postInvalidate();
  546. }
  547. return true;
  548. }
  549. public void setOnPointClickedListener(OnPointClickedListener listener) {
  550. this.listener = listener;
  551. }
  552. public interface OnPointClickedListener {
  553. abstract void onClick(int lineIndex, int pointIndex);
  554. }
  555. }