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.cox;
022
023import java.text.DecimalFormat;
024import java.util.ArrayList;
025import java.util.LinkedHashMap;
026
027/**
028 * Information needed to represent a survival curve
029 *
030 * @author Scooter Willis 
031 */
032public class StrataInfo {
033
034        private ArrayList<Double> time = new ArrayList<>();
035        private ArrayList<Integer> status = new ArrayList<>();
036        private ArrayList<Double> nevent = new ArrayList<>();
037        private ArrayList<Double> ncens = new ArrayList<>();
038        private ArrayList<Double> nrisk = new ArrayList<>();
039        private ArrayList<Double> weight = new ArrayList<>();
040        private ArrayList<Double> surv = new ArrayList<>();
041        private ArrayList<Double> varhaz = new ArrayList<>();
042        private ArrayList<Double> stderr = new ArrayList<>();
043        private ArrayList<Double> stdlow = new ArrayList<>();
044        private ArrayList<Double> upper = new ArrayList<>();
045        private ArrayList<Double> lower = new ArrayList<>();
046        private LinkedHashMap<Double, Integer> ndead = new LinkedHashMap<>();
047        DecimalFormat df = new DecimalFormat("#.######");
048        DecimalFormat dfe = new DecimalFormat("0.000000E0");
049
050        /**
051         * Need to find the actual time for the nearest time represented as a
052         * percentage Would be used to then look up the number at risk at that
053         * particular time
054         *
055         * @param timePercentage
056         * @return
057         */
058        public Double getNearestTime(double timePercentage) {
059                //the arrays should be sorted by time so this step is probably not needed
060                Double minTime = null;
061                Double maxTime = null;
062                for (Double t : time) {
063                        if (minTime == null || t < minTime) {
064                                minTime = t;
065                        }
066                        if (maxTime == null || t > maxTime) {
067                                maxTime = t;
068                        }
069                }
070                Double timeRange = maxTime - minTime;
071                Double targetTime = minTime + timePercentage * timeRange;
072                Double previousTime = null;
073                for (Double t : time) {
074                        if (previousTime == null || t <= targetTime) {
075                                previousTime = t;
076                        } else {
077                                return previousTime;
078                        }
079                }
080                return previousTime;
081        }
082
083        /**
084         * Selection of number of risk will depend on the precision and rounding of
085         * time in the survival table. If you are asking for 12 and entry exists for
086         * 11.9999999 then 12 is greater than 11.99999 unless you round.
087         *
088         * @param t
089         * @return
090         */
091        public Double getNearestAtRisk(double t) {
092                Integer index = 0;
093/*       String timeValue = t + "";
094                String format = "#";
095                int numDecimals = 0;
096                int decimalIndex = timeValue.indexOf(".");
097                if (decimalIndex > 0) {
098                        for (int i = timeValue.length() - 1; i > decimalIndex; i--) {
099                                if (timeValue.charAt(i) == '0' && numDecimals == 0) {
100                                        continue;
101                                }
102                                if (i == decimalIndex - 1) {
103                                        format = format + ".#";
104                                } else {
105                                        format = format + "#";
106                                }
107                        }
108                }
109 */
110                DecimalFormat newFormat = new DecimalFormat("#.#"); //used to round on expected precision of time. Not correct but trying to match the other packages
111
112                for (int i = 0; i < time.size(); i++) {
113                        Double compareTime = time.get(i);
114                  //  compareTime = new Double(Math.round(compareTime)); //this is rounding up so that we stop on the first match trying to get this to match another report. Not correct or the other report is wrong
115                        compareTime = Double.valueOf(newFormat.format(compareTime));
116                        if (compareTime < t) {
117                                index = i + 1 ;
118                        } else if(compareTime == t){
119                                index = i;
120                                break;
121                        }else {
122                                break;
123                        }
124                }
125
126                //http://www.inside-r.org/packages/cran/rms/docs/survplot
127                //per validation using survplot from RMS package and ggkm they select the next
128                //time in the future which doesn't seem to be correct as the next time represents
129                //knowledge about the future but maybe nrisk at that point in time is defined
130                //as the nrisk prior to that time. This appears to be the case where at time 0
131                //you would expect that everyone is at risk and you should report that time which
132                //is the case in survplot. Added in index = 0 or if the time you are requesting has
133                //an exact match
134                //survplot(kma,n.risk=TRUE,time.inc=1090)
135                //ggkm(kma,timeby=1090)
136           //     if(index != 0 && time.get(index) != t){
137           //      index++;
138           //     }
139                if (index >= nrisk.size()) {
140                        return null;
141                } else {
142                        return nrisk.get(index);
143                }
144        }
145
146        /**
147         *
148         * @param d
149         * @return
150         */
151        public String f(Double d) {
152                String v = df.format(d);
153                int l = 10 - v.length();
154                for (int i = 0; i < l; i++) {
155                        v = v + " ";
156                }
157                return v;
158        }
159
160        @Override
161        public String toString() {
162                String o = "";
163                o = o + "n=" + nevent.size() + "\r\n";
164                o = o + "     time      nevent     ncens     nrisk     weight     surv   varhaz    stderr    stdlow    lower    upper\r\n";
165                for (int i = 0; i < nevent.size(); i++) {
166                        //    if(nevent.get(i) == 0)
167                        //        continue;
168                        o = o + (i + 1) + "    " + f(time.get(i)) + " " + f(nevent.get(i)) + " " + f(ncens.get(i)) + " " + f(nrisk.get(i)) + " " + f(weight.get(i)) + " " + f(surv.get(i)) + " " + (varhaz.get(i)) + "  " + stderr.get(i) + "  " + stdlow.get(i) + "  " + lower.get(i) + "  " + upper.get(i) + "\r\n";
169
170                }
171                o = o + "\r\n";
172                //   for(Integer i : ndead.values()){
173                //       o = o + i + "\r\n";
174                //   }
175
176                return o;
177        }
178
179        /**
180         * @return the time
181         */
182        public ArrayList<Double> getTime() {
183                return time;
184        }
185
186        /**
187         * @return the surv
188         */
189        public ArrayList<Double> getSurv() {
190                return surv;
191        }
192
193        /**
194         * @return the stderr
195         */
196        public ArrayList<Double> getStderr() {
197                return stderr;
198        }
199
200        /**
201         * @return the upper
202         */
203        public ArrayList<Double> getUpper() {
204                return upper;
205        }
206
207        /**
208         * @return the lower
209         */
210        public ArrayList<Double> getLower() {
211                return lower;
212        }
213
214        /**
215         * @return the status
216         */
217        public ArrayList<Integer> getStatus() {
218                return status;
219        }
220
221        /**
222         * @return the nevent
223         */
224        public ArrayList<Double> getNevent() {
225                return nevent;
226        }
227
228        /**
229         * @return the ncens
230         */
231        public ArrayList<Double> getNcens() {
232                return ncens;
233        }
234
235        /**
236         * @return the nrisk
237         */
238        public ArrayList<Double> getNrisk() {
239                return nrisk;
240        }
241
242        /**
243         * @return the weight
244         */
245        public ArrayList<Double> getWeight() {
246                return weight;
247        }
248
249        /**
250         * @return the ndead
251         */
252        public LinkedHashMap<Double, Integer> getNdead() {
253                return ndead;
254        }
255
256        /**
257         * @return the varhaz
258         */
259        public ArrayList<Double> getVarhaz() {
260                return varhaz;
261        }
262
263        /**
264         * @return the stdlow
265         */
266        public ArrayList<Double> getStdlow() {
267                return stdlow;
268        }
269
270        /**
271         * @param stdlow the stdlow to set
272         */
273        public void setStdlow(ArrayList<Double> stdlow) {
274                this.stdlow = stdlow;
275        }
276}