001/*
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 */
021package org.biojava.nbio.survival.kaplanmeier.figure;
022
023import org.biojava.nbio.survival.cox.*;
024
025import javax.imageio.ImageIO;
026import javax.swing.*;
027import java.awt.*;
028import java.awt.geom.AffineTransform;
029import java.awt.image.BufferedImage;
030import java.io.File;
031import java.io.FileWriter;
032import java.text.DecimalFormat;
033import java.util.ArrayList;
034import java.util.Collections;
035import java.util.LinkedHashMap;
036
037/**
038 *
039 * @author Scooter Willis 
040 */
041public class KaplanMeierFigure extends JPanel {
042
043        private static final long serialVersionUID = 1L;
044
045        ArrayList<String> title = new ArrayList<>();
046        /**
047         *
048         */
049        private int top;
050        /**
051         *
052         */
053        private int bottom;
054        /**
055         *
056         */
057        private int left;
058        private int yaxisLabel = 20;
059        /**
060         *
061         */
062        private int right;
063        int titleHeight;
064        int xAxisLabelHeight;
065        int labelWidth;
066        double minTime = 0.0;
067        double maxTime = 10.0;
068        double minPercentage = 0.0;
069        double maxPercentage = 1.0;
070        FontMetrics fm;
071        KMFigureInfo kmfi = new KMFigureInfo();
072        LinkedHashMap<String, ArrayList<CensorStatus>> survivalData = new LinkedHashMap<>();
073        ArrayList<String> lineInfoList = new ArrayList<>();
074        SurvFitInfo sfi = new SurvFitInfo();
075        private String fileName = "";
076        private ArrayList<Double> xAxisTimeValues = new ArrayList<>();
077        private ArrayList<Integer> xAxisTimeCoordinates = new ArrayList<>();
078
079        /**
080         *
081         */
082        public KaplanMeierFigure() {
083                super();
084                setSize(500, 400);
085                setBackground(Color.WHITE);
086        }
087
088        /**
089         * Get the name of the groups that are being plotted in the figure
090         *
091         * @return
092         */
093        public ArrayList<String> getGroups() {
094                return new ArrayList<>(survivalData.keySet());
095        }
096
097        /**
098         * To get the median percentile for a particular group pass the value of
099         * .50.
100         *
101         * @param group
102         * @param percentile
103         * @return
104         */
105        public Double getSurvivalTimePercentile(String group, double percentile) {
106
107                StrataInfo si = sfi.getStrataInfoHashMap().get(group);
108                ArrayList<Double> percentage = si.getSurv();
109                Integer percentileIndex = null;
110                for (int i = 0; i < percentage.size(); i++) {
111                        if (percentage.get(i) == percentile) {
112                                if (i + 1 < percentage.size()) {
113                                        percentileIndex = i + 1;
114                                }
115                                break;
116                        } else if (percentage.get(i) < percentile) {
117                                percentileIndex = i;
118                                break;
119                        }
120                }
121                if (percentileIndex != null) {
122                        return si.getTime().get(percentileIndex);
123                } else {
124                        return null;
125                }
126        }
127
128        /**
129         *
130         * @param kmfi
131         */
132        public void setKMFigureInfo(KMFigureInfo kmfi) {
133                this.kmfi = kmfi;
134                if (kmfi.width != null && kmfi.height != null) {
135                        this.setSize(kmfi.width, kmfi.height);
136                }
137        }
138
139        public KMFigureInfo getKMFigureInfo() {
140                return kmfi;
141        }
142
143        /**
144         *
145         * @param lineInfoList
146         */
147        public void setFigureLineInfo(ArrayList<String> lineInfoList) {
148                this.lineInfoList = lineInfoList;
149                this.repaint();
150        }
151
152        /**
153         *
154         * @param title Title of figures
155         * @param ci
156         * @param strataVariable The column that based on value will do a figure
157         * line
158         * @param legendMap Map the value in the column to something readable
159         * @param useWeighted
160         * @throws Exception
161         */
162        public void setCoxInfo(ArrayList<String> title, CoxInfo ci, String strataVariable, LinkedHashMap<String, String> legendMap, Boolean useWeighted) throws Exception {
163                LinkedHashMap<String, ArrayList<CensorStatus>> survivalData = new LinkedHashMap<>();
164                ArrayList<SurvivalInfo> siList = ci.getSurvivalInfoList();
165                int n = 0;
166                int event = 0;
167                for (SurvivalInfo si : siList) {
168                        String strata = si.getOriginalMetaData(strataVariable);
169                        String legend = legendMap.get(strata);
170                        if (legend == null) {
171
172                                legend = strata;
173                        }
174                        ArrayList<CensorStatus> censorStatusList = survivalData.get(legend);
175                        if (censorStatusList == null) {
176                                censorStatusList = new ArrayList<>();
177                                survivalData.put(legend, censorStatusList);
178                        }
179                        CensorStatus cs = new CensorStatus(strata, si.getTime(), si.getStatus() + "");
180                        cs.weight = si.getWeight();
181                        censorStatusList.add(cs);
182                        n++;
183                        if (si.getStatus() == 1) {
184                                event++;
185                        }
186                }
187
188                setSurvivalData(title, survivalData, useWeighted);
189                CoxCoefficient cc = ci.getCoefficient(strataVariable);
190                //DecimalFormat df = new DecimalFormat("#.##");
191                String line1 = "HR=" + fmt(cc.getHazardRatio(), 2, 0) + " (CI:" + fmt(cc.getHazardRatioLoCI(), 2, 0) + "-" + fmt(cc.getHazardRatioHiCI(), 2, 0) + ")";
192                String line2 = "p=" + fmt(cc.getPvalue(), 3, 0);
193           // String line2 = "logrank P=" + fmt(ci.getScoreLogrankTestpvalue(), 3, 0);
194                String line3 = "n=" + n + " events=" + event;
195//        System.out.println("setCoxInfo=" + cc.pvalue + " " + title);
196
197
198                ArrayList<String> lines = new ArrayList<>();
199                lines.add(line1);
200                lines.add(line2);
201                lines.add(line3);
202                setFigureLineInfo(lines);
203        }
204
205        /**
206         *
207         * @param d
208         * @param precision
209         * @param pad
210         * @return
211         */
212        public static String fmt(Double d, int precision, int pad) {
213                String value = "";
214                DecimalFormat dfe = new DecimalFormat("0.00E0");
215                String dpad = "0.";
216                double p = 1.0;
217                for (int i = 0; i < (precision); i++) {
218                        dpad = dpad + "0";
219                        p = p / 10.0;
220                }
221                DecimalFormat df = new DecimalFormat(dpad);
222                if (Math.abs(d) >= p) {
223                        value = df.format(d);
224                } else {
225                        value = dfe.format(d);
226                }
227                int length = value.length();
228                int extra = pad - length;
229                if (extra > 0) {
230                        for (int i = 0; i < extra; i++) {
231                                value = " " + value;
232                        }
233                }
234                return value;
235        }
236
237        /**
238         *
239         * @return
240         */
241        public SurvFitInfo getSurvivalFitInfo() {
242                return sfi;
243        }
244
245        /**
246         * Allow setting of points in the figure where weighted correction has been
247         * done and percentage has already been calculated.
248         *
249         * @param title
250         * @param sfi
251         * @param userSetMaxTime
252         */
253        public void setSurvivalData(ArrayList<String> title, SurvFitInfo sfi, Double userSetMaxTime) {
254                this.title = title;
255                LinkedHashMap<String, StrataInfo> strataInfoHashMap = sfi.getStrataInfoHashMap();
256                Double mTime = null;
257                for (StrataInfo si : strataInfoHashMap.values()) {
258                        for (double t : si.getTime()) {
259                                if (mTime == null || t > mTime) {
260                                        mTime = t;
261                                }
262                        }
263                }
264
265                int evenCheck = Math.round(mTime.floatValue());
266                if (evenCheck % 2 == 1) {
267                        evenCheck = evenCheck + 1;
268                }
269                this.maxTime = evenCheck;
270
271                if (userSetMaxTime != null && userSetMaxTime > maxTime) {
272                        this.maxTime = userSetMaxTime;
273                }
274                this.sfi = sfi;
275                if (sfi.getStrataInfoHashMap().size() == 1) {
276                        return;
277                }
278                this.repaint();
279        }
280
281        /**
282         * The data will set the max time which will result in off time points for
283         * tick marks
284         *
285         * @param title
286         * @param survivalData
287         * @param useWeighted
288         * @throws Exception
289         */
290        public void setSurvivalData(ArrayList<String> title, LinkedHashMap<String, ArrayList<CensorStatus>> survivalData, Boolean useWeighted) throws Exception {
291                this.setSurvivalData(title, survivalData, null, useWeighted);
292        }
293
294        /**
295         *
296         * @param title
297         * @param survivalData
298         * @param userSetMaxTime
299         * @param useWeighted
300         * @throws Exception
301         */
302        public void setSurvivalData(ArrayList<String> title, LinkedHashMap<String, ArrayList<CensorStatus>> survivalData, Double userSetMaxTime, Boolean useWeighted) throws Exception {
303                this.title = title;
304                this.survivalData = survivalData;
305                Double mTime = null;
306                ArrayList<String> labels = new ArrayList<>(survivalData.keySet());
307                Collections.sort(labels);
308                for (String legend : labels) {
309                        ArrayList<CensorStatus> censorStatusList = survivalData.get(legend);
310                        for (CensorStatus cs : censorStatusList) {
311
312                                if (mTime == null || cs.time > mTime) {
313                                        mTime = cs.time;
314                                }
315                        }
316                }
317
318                int evenCheck = Math.round(mTime.floatValue());
319                if (evenCheck % 2 == 1) {
320                        evenCheck = evenCheck + 1;
321                }
322                this.maxTime = evenCheck;
323
324                if (userSetMaxTime != null && userSetMaxTime > maxTime) {
325                        this.maxTime = userSetMaxTime;
326                }
327
328                //calculate percentages
329                SurvFitKM survFitKM = new SurvFitKM();
330                sfi = survFitKM.process(survivalData, useWeighted);
331                this.repaint();
332        }
333
334        /**
335         * Save data from survival curve to text file
336         *
337         * @param fileName
338         * @throws Exception
339         */
340        public void saveSurvivalData(String fileName) throws Exception {
341                FileWriter fw = new FileWriter(fileName);
342                fw.write("index\tTIME\tSTATUS\tGROUP\r\n");
343                int index = 0;
344                for (String group : survivalData.keySet()) {
345                        ArrayList<CensorStatus> sd = survivalData.get(group);
346                        for (CensorStatus cs : sd) {
347                                String line = index + "\t" + cs.time + "\t" + cs.censored + "\t" + cs.group + "\r\n";
348                                index++;
349                                fw.write(line);
350                        }
351                }
352                fw.close();
353        }
354        DecimalFormat df = new DecimalFormat("#.#");
355
356        @Override
357        public void paintComponent(Graphics g) // draw graphics in the panel
358        {
359                int width = getWidth();             // width of window in pixels
360                int height = getHeight();           // height of window in pixels
361                setFigureDimensions();
362                g.setColor(Color.white);
363                g.clearRect(0, 0, width, height);
364
365                super.paintComponent(g);            // call superclass to make panel display correctly
366
367                drawLegend(g);
368                drawSurvivalCurves(g);
369                drawFigureLineInfo(g);
370                // Drawing code goes here
371        }
372
373        private void drawFigureLineInfo(Graphics g) {
374                Graphics2D g2 = (Graphics2D) g;
375                setRenderingHints(g2);
376                g2.setColor(Color.BLACK);
377                fm = getFontMetrics(getFont());
378                int yoffset = fm.getHeight() * lineInfoList.size();
379
380                int x = getTimeX(kmfi.figureLineInfoLowerPercentX * maxTime);
381                int y = getPercentageY(kmfi.figureLineInfoLowerPercentY) - yoffset;
382
383                for (String line : lineInfoList) {
384                        g2.drawString(line, x, y);
385                        y = y + fm.getHeight();
386                }
387
388        }
389
390        private void drawSurvivalCurves(Graphics g) {
391                Graphics2D g2 = (Graphics2D) g;
392                setRenderingHints(g2);
393                g2.setStroke(kmfi.kmStroke);
394
395
396                int colorIndex = 0;
397                ArrayList<String> labels = new ArrayList<>(sfi.getStrataInfoHashMap().keySet());
398                Collections.sort(labels);
399
400                LinkedHashMap<String, StrataInfo> strataInfoHashMap = sfi.getStrataInfoHashMap();
401
402                for (String legend : labels) {
403                        StrataInfo si = strataInfoHashMap.get(legend);
404                        g2.setColor(kmfi.legendColor[colorIndex]);
405                        colorIndex++;
406
407                        for (int i = 0; i < si.getSurv().size() - 1; i++) {
408                                double p0time = si.getTime().get(i);
409                                double p1time = si.getTime().get(i + 1);
410                                double p0percentage = si.getSurv().get(i);
411                                double p1percentage = si.getSurv().get(i + 1);
412                                if (i == 0) {
413                                        g2.drawLine(getTimeX(0), getPercentageY(1), getTimeX(p0time), getPercentageY(1));
414                                        g2.drawLine(getTimeX(p0time), getPercentageY(1), getTimeX(p0time), getPercentageY(p0percentage));
415                                }
416                                g2.drawLine(getTimeX(p0time), getPercentageY(p0percentage), getTimeX(p1time), getPercentageY(p0percentage));
417
418                                g2.drawLine(getTimeX(p1time), getPercentageY(p0percentage), getTimeX(p1time), getPercentageY(p1percentage));
419                                // if (si.getStatus().get(i) == 0) {
420                                if (i > 0 && si.getNcens().get(i) > 0) {
421                                        g2.drawLine(getTimeX(p0time), getPercentageY(p0percentage) - 4, getTimeX(p0time), getPercentageY(p0percentage) + 4);
422                                        g2.drawLine(getTimeX(p0time) - 4, getPercentageY(p0percentage), getTimeX(p0time) + 4, getPercentageY(p0percentage));
423                                }
424                        }
425
426
427                }
428
429                String maxString = "";
430                for (String legend : labels) {
431                        if (legend.length() > maxString.length()) {
432                                maxString = legend;
433                        }
434                }
435
436                int offset = fm.stringWidth(maxString);
437                int x = getTimeX(kmfi.legendUpperPercentX * maxTime) - offset;
438                int y = getPercentageY(kmfi.legendUpperPercentY);
439
440                colorIndex = 0;
441                for (String legend : labels) {
442                        g2.setColor(kmfi.legendColor[colorIndex]);
443                        colorIndex++;
444                        g2.drawLine(x - 20, y - (fm.getHeight() / 3), x - 5, y - (fm.getHeight() / 3));
445                        g2.drawString(legend, x, y);
446                        y = y + fm.getHeight();
447                }
448
449
450        }
451
452        /**
453         * Get the X coordinate based on a time value
454         *
455         * @param value
456         * @return
457         */
458        private int getTimeX(double value) {
459                double d = left + (((right - left) * value) / (maxTime - minTime));
460                return (int) d;
461        }
462
463        /**
464         * Get the Y coordinate based on percent value 0.0-1.0
465         *
466         * @param value
467         * @return
468         */
469        private int getPercentageY(double value) {
470                value = 1.0 - value;
471                double d = top + (((bottom - top) * value) / (maxPercentage - minPercentage));
472                return (int) d;
473        }
474
475        /**
476         * @return the fileName
477         */
478        public String getFileName() {
479                return fileName;
480        }
481
482        /**
483         * @return the top
484         */
485        public int getTop() {
486                return top;
487        }
488
489        /**
490         * @return the bottom
491         */
492        public int getBottom() {
493                return bottom;
494        }
495
496        /**
497         * @return the left
498         */
499        public int getLeft() {
500                return left;
501        }
502
503        /**
504         * @return the right
505         */
506        public int getRight() {
507                return right;
508        }
509
510        /**
511         * @return the xAxisTimeValues
512         */
513        public ArrayList<Double> getxAxisTimeValues() {
514                return xAxisTimeValues;
515        }
516
517        /**
518         * @return the xAxisTimeValues
519         */
520        public ArrayList<Integer> getxAxisTimeCoordinates() {
521                return xAxisTimeCoordinates;
522        }
523
524        class PlotInfo {
525
526                double time;
527                double atRisk;
528                double censored;
529                double events;
530                double percentage;
531
532                @Override
533                public String toString() {
534                        return time + "\t" + atRisk + "\t" + censored + "\t" + events + "\t" + (atRisk - events) + "\t" + percentage;
535                }
536        }
537
538        /**
539         * Do higher quality rendering options
540         *
541         * @param g
542         */
543        private void setRenderingHints(Graphics2D g) {
544                RenderingHints rh = new RenderingHints(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
545                rh.put(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_ENABLE);
546                rh.put(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_GASP);
547                rh.put(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
548
549                g.setRenderingHints(rh);
550
551        }
552
553        /**
554         * Setup the axis, labels etc
555         *
556         * @param g
557         */
558        private void drawLegend(Graphics g) {
559                Graphics2D g2 = (Graphics2D) g;
560                setRenderingHints(g2);
561                g2.setColor(Color.BLACK);
562                Font font = g2.getFont();
563                Font f = new Font(font.getFontName(), Font.BOLD, font.getSize());
564                g2.setFont(f);
565                fm = getFontMetrics(f);
566                int fontHeight = fm.getHeight();
567                for (int i = 0; i < title.size(); i++) {
568                        if (fm.stringWidth(title.get(i)) > .8 * this.getWidth()) {
569                                f = new Font(font.getFontName(), Font.BOLD, 10);
570                                g2.setFont(f);
571                                fm = getFontMetrics(f);
572                        }
573                        g2.drawString(title.get(i), (getSize().width - fm.stringWidth(title.get(i))) / 2, ((i + 1) * fontHeight));
574                        // g2.setFont(font);
575                }
576                // draw the maxPercentage and minPercentage values
577                String label = df.format(minPercentage);
578                g2.drawString(label, left - 5 - (fm.stringWidth(label)), bottom + titleHeight / 6);
579                g2.drawLine(left - 5, bottom, left, bottom);
580                double d = minPercentage + kmfi.yaxisPercentIncrement;
581                //double graphHeight = top - bottom;
582
583                while (d < maxPercentage) {
584                        int yvalue = bottom - (int) (d * (bottom - top));
585                        label = df.format(d * 100);
586                        g2.drawString(label, left - 5 - (fm.stringWidth(label)), yvalue + titleHeight / 6); //
587
588                        g2.drawLine(left - 5, yvalue, left, yvalue);
589                        d = d + kmfi.yaxisPercentIncrement;
590                }
591
592                label = df.format(maxPercentage * 100);
593                g2.drawString(label, left - 5 - (fm.stringWidth(label)), top + (titleHeight) / 6);
594                g2.drawLine(left - 5, top, left, top);
595
596                // Create a rotation transformation for the font.
597                AffineTransform fontAT = new AffineTransform();
598
599
600                // Derive a new font using a rotatation transform
601                fontAT.rotate(270 * java.lang.Math.PI / 180);
602                Font theDerivedFont = f.deriveFont(fontAT);
603
604                // set the derived font in the Graphics2D context
605                g2.setFont(theDerivedFont);
606
607                // Render a string using the derived font
608                int yaxisHeight = fm.stringWidth(kmfi.yAxisLegend);
609                g2.drawString(kmfi.yAxisLegend, yaxisLabel, (bottom - (int) (.5 * (bottom - top))) + yaxisHeight / 2);
610
611                // put the original font back
612                g2.setFont(f);
613
614
615
616                double timeDistance = maxTime - minTime;
617                double timeIncrement = timeDistance * kmfi.xaxisPercentIncrement;
618                double timeInt = (int) Math.floor(timeIncrement);
619                if (timeInt < 1.0) {
620                        timeInt = 1.0;
621                }
622                adjustedPercentIncrement = timeInt / timeDistance;
623
624                d = adjustedPercentIncrement; //kmfi.xaxisPercentIncrement;
625                xAxisTimeValues.clear();
626                xAxisTimeCoordinates.clear();
627
628                //if we don't have time values then use percentage to set time. Not perfect but allows different tics
629                if (kmfi.xAxisLabels.isEmpty()) {
630                        xAxisTimeValues.add(minTime);
631                        xAxisTimeCoordinates.add(left);
632                        while (d <= 1.0) {
633                                double xaxisTime = ((minTime * kmfi.timeScale) + d * ((maxTime - minTime) * kmfi.timeScale)); //
634                                xAxisTimeValues.add(xaxisTime);
635
636                                Integer coordinate = left + (int) (d * (right - left));
637                                xAxisTimeCoordinates.add(coordinate);
638                                //       System.out.println(d + " " + left + " " + right + " " + coordinate + " " + minTime + " " + maxTime);
639                                d = d + adjustedPercentIncrement; //kmfi.xaxisPercentIncrement;
640                        }
641                } else {
642                        minTime = kmfi.xAxisLabels.get(0);
643                        maxTime = kmfi.xAxisLabels.get(kmfi.xAxisLabels.size() - 1);
644                        for (Double xaxisTime : kmfi.xAxisLabels) {
645                                xAxisTimeValues.add(xaxisTime);
646                                d = (xaxisTime - minTime) / (maxTime - minTime);
647                                Integer coordinate = left + (int) (d * (right - left));
648                                xAxisTimeCoordinates.add(coordinate);
649                        }
650                }
651
652                for (int i = 0; i < xAxisTimeValues.size(); i++) {
653                        Double xaxisTime = xAxisTimeValues.get(i);
654                        Integer xCoordinate = xAxisTimeCoordinates.get(i);
655                        label = df.format(xaxisTime);
656                        if (i == xAxisTimeValues.size() - 1) {
657                                g2.drawString(label, xCoordinate - (fm.stringWidth(label)), bottom + fm.getHeight() + 5);
658                        } else {
659                                g2.drawString(label, xCoordinate - (fm.stringWidth(label) / 2), bottom + fm.getHeight() + 5);
660                        }
661                        g2.drawLine(xCoordinate, bottom, xCoordinate, bottom + 5);
662                }
663
664                // draw the vertical and horizontal lines
665                g2.setStroke(kmfi.axisStroke);
666                g2.drawLine(left, top, left, bottom);
667                g2.drawLine(left, bottom, right, bottom);
668
669                // draw xAxis legend
670                g2.drawString(kmfi.xAxisLegend, getSize().width / 2 - (fm.stringWidth(kmfi.xAxisLegend) / 2), bottom + 2 * fm.getHeight() + 10);
671        }
672        Double adjustedPercentIncrement = 0.0;
673
674        /**
675         * Get the percentage increment for the time axis
676         *
677         * @return
678         */
679        public Double getTimeAxisIncrementPercentage() {
680                return adjustedPercentIncrement;
681        }
682
683        /**
684         * Reset the various bounds used to draw graph
685         */
686        private void setFigureDimensions() {
687                fm = getFontMetrics(getFont());
688                titleHeight = kmfi.titleHeight;//fm.getHeight();
689                xAxisLabelHeight = titleHeight;
690                labelWidth = Math.max(fm.stringWidth(df.format(minPercentage)),
691                                fm.stringWidth(df.format(maxPercentage))) + 5;
692                top = kmfi.padding + titleHeight;
693                bottom = this.getHeight() - kmfi.padding - xAxisLabelHeight;
694                left = kmfi.padding + labelWidth + yaxisLabel;
695                right = this.getWidth() - kmfi.padding;
696
697        }
698
699        /**
700         * Combine the KM and Num risk into one image
701         *
702         * @param fileName
703         */
704        public void savePNGKMNumRisk(String fileName) {
705                if (fileName.startsWith("null") || fileName.startsWith("Null") || fileName.startsWith("NULL")) {
706                        return;
707                }
708                this.fileName = fileName;
709
710                NumbersAtRiskPanel numbersAtRiskPanel = new NumbersAtRiskPanel();
711                numbersAtRiskPanel.setKaplanMeierFigure(this);
712                numbersAtRiskPanel.setSize(this.getWidth(), numbersAtRiskPanel.getHeight());
713                BufferedImage imageKM = new BufferedImage(this.getWidth(), this.getHeight(), BufferedImage.TYPE_INT_RGB);
714                Graphics2D graphics2D = imageKM.createGraphics();
715
716                this.paint(graphics2D);
717
718                BufferedImage imageNumRisk = new BufferedImage(numbersAtRiskPanel.getWidth(), numbersAtRiskPanel.getHeight(), BufferedImage.TYPE_INT_RGB);
719                Graphics2D graphics2DNumRisk = imageNumRisk.createGraphics();
720                numbersAtRiskPanel.paint(graphics2DNumRisk);
721
722
723                BufferedImage image = new BufferedImage(numbersAtRiskPanel.getWidth(), numbersAtRiskPanel.getHeight() + this.getHeight(), BufferedImage.TYPE_INT_RGB);
724                Graphics2D g = image.createGraphics();
725
726                g.drawImage(imageKM, 0, 0, null);
727                g.drawImage(imageNumRisk, 0, this.getHeight(), null);
728
729                try {
730                        ImageIO.write(image, "png", new File(fileName));
731                } catch (Exception ex) {
732                        ex.printStackTrace();
733                }
734
735        }
736
737        /**
738         *
739         * @param fileName
740         */
741        public void savePNG(String fileName) {
742                if (fileName.startsWith("null") || fileName.startsWith("Null") || fileName.startsWith("NULL")) {
743                        return;
744                }
745                this.fileName = fileName;
746                BufferedImage image = new BufferedImage(this.getWidth(), this.getHeight(), BufferedImage.TYPE_INT_RGB);
747                Graphics2D graphics2D = image.createGraphics();
748
749                this.paint(graphics2D);
750                try {
751                        ImageIO.write(image, "png", new File(fileName));
752                } catch (Exception ex) {
753                        ex.printStackTrace();
754                }
755
756        }
757
758        /**
759         * @param args the command line arguments
760         */
761        public static void main(String[] args) {
762                // TODO code application logic here
763                try {
764
765                        KaplanMeierFigure kaplanMeierFigure = new KaplanMeierFigure();
766                        LinkedHashMap<String, ArrayList<CensorStatus>> survivalDataHashMap = new LinkedHashMap<>();
767
768//            if (false) { //http://sph.bu.edu/otlt/MPH-Modules/BS/BS704_Survival/
769//                ArrayList<CensorStatus> graph1 = new ArrayList<CensorStatus>();
770//                graph1.add(new CensorStatus("A", 24.0, "0"));
771//                graph1.add(new CensorStatus("A", 3.0, "1"));
772//                graph1.add(new CensorStatus("A", 11.0, "0"));
773//                graph1.add(new CensorStatus("A", 19.0, "0"));
774//                graph1.add(new CensorStatus("A", 24.0, "0"));
775//                graph1.add(new CensorStatus("A", 13.0, "0"));
776//
777//                graph1.add(new CensorStatus("A", 14.0, "1"));
778//                graph1.add(new CensorStatus("A", 2.0, "0"));
779//                graph1.add(new CensorStatus("A", 18.0, "0"));
780//                graph1.add(new CensorStatus("A", 17.0, "0"));
781//                graph1.add(new CensorStatus("A", 24.0, "0"));
782//                graph1.add(new CensorStatus("A", 21.0, "0"));
783//                graph1.add(new CensorStatus("A", 12.0, "0"));
784//
785//                graph1.add(new CensorStatus("A", 1.0, "1"));
786//                graph1.add(new CensorStatus("A", 10.0, "0"));
787//                graph1.add(new CensorStatus("A", 23.0, "1"));
788//                graph1.add(new CensorStatus("A", 6.0, "0"));
789//                graph1.add(new CensorStatus("A", 5.0, "1"));
790//                graph1.add(new CensorStatus("A", 9.0, "0"));
791//                graph1.add(new CensorStatus("A", 17.0, "1"));
792//
793//                survivalDataHashMap.put("Label 1", graph1);
794//
795//
796//
797//            }
798
799
800                        if (true) {
801
802
803
804                                ArrayList<CensorStatus> graph1 = new ArrayList<>();
805                                graph1.add(new CensorStatus("A", 1.0, "1"));
806                                graph1.add(new CensorStatus("A", 1.0, "1"));
807                                graph1.add(new CensorStatus("A", 1.0, "1"));
808                                graph1.add(new CensorStatus("A", 2.0, "1"));
809                                graph1.add(new CensorStatus("A", 2.0, "1"));
810                                graph1.add(new CensorStatus("A", 3.0, "1"));
811
812                                graph1.add(new CensorStatus("A", 4.0, "1"));
813                                graph1.add(new CensorStatus("A", 4.0, "1"));
814                                graph1.add(new CensorStatus("A", 4.0, "1"));
815                                graph1.add(new CensorStatus("A", 4.0, "1"));
816                                graph1.add(new CensorStatus("A", 4.0, "1"));
817                                graph1.add(new CensorStatus("A", 4.0, "1"));
818                                graph1.add(new CensorStatus("A", 4.0, "0"));
819
820                                graph1.add(new CensorStatus("A", 5.0, "1"));
821                                graph1.add(new CensorStatus("A", 5.0, "1"));
822
823                                graph1.add(new CensorStatus("A", 8.0, "0"));
824                                graph1.add(new CensorStatus("A", 8.0, "0"));
825                                graph1.add(new CensorStatus("A", 8.0, "0"));
826                                graph1.add(new CensorStatus("A", 8.0, "0"));
827                                graph1.add(new CensorStatus("A", 8.0, "0"));
828                                graph1.add(new CensorStatus("A", 8.0, "0"));
829                                graph1.add(new CensorStatus("A", 8.0, "1"));
830
831                                graph1.add(new CensorStatus("A", 9.0, "1"));
832                                graph1.add(new CensorStatus("A", 9.0, "1"));
833                                graph1.add(new CensorStatus("A", 9.0, "1"));
834                                graph1.add(new CensorStatus("A", 9.0, "1"));
835                                graph1.add(new CensorStatus("A", 9.0, "1"));
836
837
838                                graph1.add(new CensorStatus("A", 13.0, "0"));
839                                graph1.add(new CensorStatus("A", 13.0, "0"));
840                                graph1.add(new CensorStatus("A", 13.0, "1"));
841
842                                survivalDataHashMap.put("Label 1", graph1);
843
844                                ArrayList<CensorStatus> graph2 = new ArrayList<>();
845                                graph2.add(new CensorStatus("A", 1.0, "1"));
846                                graph2.add(new CensorStatus("A", 1.0, "1"));
847                                graph2.add(new CensorStatus("A", 1.0, "0"));
848                                graph2.add(new CensorStatus("A", 3.0, "0"));
849                                graph2.add(new CensorStatus("A", 3.0, "1"));
850                                graph2.add(new CensorStatus("A", 4.0, "1"));
851
852                                graph2.add(new CensorStatus("A", 4.0, "1"));
853                                graph2.add(new CensorStatus("A", 4.0, "1"));
854                                graph2.add(new CensorStatus("A", 4.0, "1"));
855                                graph2.add(new CensorStatus("A", 5.0, "1"));
856                                graph2.add(new CensorStatus("A", 5.0, "0"));
857                                graph2.add(new CensorStatus("A", 5.0, "0"));
858                                graph2.add(new CensorStatus("A", 5.0, "0"));
859
860                                graph2.add(new CensorStatus("A", 6.0, "1"));
861                                graph2.add(new CensorStatus("A", 6.0, "0"));
862
863                                graph2.add(new CensorStatus("A", 7.0, "0"));
864                                graph2.add(new CensorStatus("A", 7.0, "0"));
865                                graph2.add(new CensorStatus("A", 7.0, "0"));
866                                graph2.add(new CensorStatus("A", 7.0, "0"));
867                                graph2.add(new CensorStatus("A", 8.0, "1"));
868                                graph2.add(new CensorStatus("A", 8.0, "1"));
869                                graph2.add(new CensorStatus("A", 8.0, "1"));
870
871                                graph2.add(new CensorStatus("A", 8.0, "1"));
872                                graph2.add(new CensorStatus("A", 8.0, "1"));
873                                graph2.add(new CensorStatus("A", 8.0, "0"));
874                                graph2.add(new CensorStatus("A", 9.0, "0"));
875                                graph2.add(new CensorStatus("A", 9.0, "1"));
876
877
878                                graph2.add(new CensorStatus("A", 10.0, "0"));
879                                graph2.add(new CensorStatus("A", 10.0, "0"));
880                                graph2.add(new CensorStatus("A", 10.0, "0"));
881
882                                survivalDataHashMap.put("Label 2", graph2);
883                        }
884
885                        ArrayList<String> figureInfo = new ArrayList<>();
886                        //DecimalFormat dfe = new DecimalFormat("0.00E0");
887                        //DecimalFormat df = new DecimalFormat("0.00");
888
889
890
891                        JFrame application = new JFrame();
892                        application.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
893                        application.add(kaplanMeierFigure);
894                        kaplanMeierFigure.setSize(500, 400);
895
896                        application.setSize(500, 400);         // window is 500 pixels wide, 400 high
897                        application.setVisible(true);
898
899                        ArrayList<String> titles = new ArrayList<>();
900                        titles.add("Line 1");
901                        titles.add("Line 2");
902                        kaplanMeierFigure.setSurvivalData(titles, survivalDataHashMap, true);
903
904                        //   figureInfo.add("HR=2.1 95% CI(1.8-2.5)");
905                        //   figureInfo.add("p-value=.001");
906                        kaplanMeierFigure.setFigureLineInfo(figureInfo);
907
908                        kaplanMeierFigure.savePNGKMNumRisk("test.png");
909
910                } catch (Exception e) {
911                        e.printStackTrace();
912                }
913        }
914}