001/* @(#)FastaSequence.java 1.0 September 2009 002 * 003 * Copyright (c) 2009 Peter Troshin 004 * 005 * BioJava development code 006 * 007 * This code may be freely distributed and modified under the 008 * terms of the GNU Lesser General Public Licence. This should 009 * be distributed with the code. If you do not have a copy, 010 * see: 011 * 012 * http://www.gnu.org/copyleft/lesser.html 013 * 014 * Copyright for this code is held jointly by the individual 015 * authors. These should be listed in @author doc comments. 016 * 017 * For more information on the BioJava project and its aims, 018 * or to join the biojava-l mailing list, visit the home page 019 * at: 020 * 021 * http://www.biojava.org/ 022 * 023 */ 024 025package org.biojava.nbio.data.sequence; 026 027import jakarta.xml.bind.annotation.XmlAccessType; 028import jakarta.xml.bind.annotation.XmlAccessorType; 029import java.util.regex.Matcher; 030import java.util.regex.Pattern; 031 032/** 033 * A FASTA formatted sequence. Please note that this class does not make any 034 * assumptions as to what sequence it store e.g. it could be nucleotide, protein 035 * or even gapped alignment sequence! The only guarantee it makes is that the 036 * sequence does not contain white space characters e.g. spaces, new lines etc 037 * 038 * @author pvtroshin 039 * @version 1.0 040 * @since 3.0.2 041 */ 042 043@XmlAccessorType(XmlAccessType.FIELD) 044public final class FastaSequence implements Comparable<FastaSequence>{ 045 046 /** 047 * Sequence id 048 */ 049 private String id; 050 051 // TODO what about gapped sequence here! should be indicated 052 /** 053 * Returns the string representation of sequence 054 */ 055 private String sequence; 056 057 @SuppressWarnings("unused") 058 private FastaSequence() { 059 // Default constructor for JaxB 060 } 061 062 /** 063 * Upon construction the any whitespace characters are removed from the 064 * sequence 065 * 066 * @param id 067 * @param sequence 068 */ 069 public FastaSequence(final String id, final String sequence) { 070 this.id = id.trim(); 071 this.sequence = SequenceUtil.cleanSequence(sequence); 072 } 073 074 /** 075 * Gets the value of id 076 * 077 * @return the value of id 078 */ 079 public String getId() { 080 return id; 081 } 082 083 /** 084 * Gets the value of sequence 085 * 086 * @return the value of sequence 087 */ 088 public String getSequence() { 089 return sequence; 090 } 091 092 public static int countMatchesInSequence(final String theString, 093 final String theRegExp) { 094 final Pattern p = Pattern.compile(theRegExp); 095 final Matcher m = p.matcher(theString); 096 int cnt = 0; 097 while (m.find()) { 098 cnt++; 099 } 100 return cnt; 101 } 102 103 public String getFormattedFasta() { 104 return getFormatedSequence(80); 105 } 106 107 /** 108 * 109 * @return one line name, next line sequence, no matter what the sequence 110 * length is 111 */ 112 public String getOnelineFasta() { 113 String fasta = ">" + getId() + "\n"; 114 fasta += getSequence() + "\n"; 115 return fasta; 116 } 117 118 /** 119 * Format sequence per width letter in one string. Without spaces. 120 * 121 * @return multiple line formated sequence, one line width letters length 122 * 123 */ 124 public String getFormatedSequence(final int width) { 125 if (sequence == null) { 126 return ""; 127 } 128 129 assert width >= 0 : "Wrong width parameter "; 130 131 final StringBuilder sb = new StringBuilder(sequence); 132 int nchunks = sequence.length() / width; 133 // add up inserted new line chars 134 nchunks = (nchunks + sequence.length()) / width; 135 int nlineCharcounter = 0; 136 for (int i = 1; i <= nchunks; i++) { 137 final int insPos = width * i + nlineCharcounter; 138 // to prevent inserting new line in the very end of a sequence then 139 // it would have failed. 140 // Also covers the case when the sequences shorter than width 141 if (sb.length() <= insPos) { 142 break; 143 } 144 sb.insert(insPos, "\n"); 145 nlineCharcounter++; 146 } 147 return sb.toString(); 148 } 149 150 /** 151 * 152 * @return sequence length 153 */ 154 public int getLength() { 155 return sequence.length(); 156 } 157 158 /** 159 * Same as oneLineFasta 160 */ 161 @Override 162 public String toString() { 163 return this.getOnelineFasta(); 164 } 165 166 @Override 167 public int hashCode() { 168 final int prime = 31; 169 int result = 1; 170 result = prime * result + ((id == null) ? 0 : id.hashCode()); 171 result = prime * result 172 + ((sequence == null) ? 0 : sequence.hashCode()); 173 return result; 174 } 175 176 @Override 177 public boolean equals(final Object obj) { 178 if (this == obj) { 179 return true; 180 } 181 if (obj == null) { 182 return false; 183 } 184 if (getClass() != obj.getClass()) { 185 return false; 186 } 187 final FastaSequence other = (FastaSequence) obj; 188 if (id == null) { 189 if (other.id != null) { 190 return false; 191 } 192 } else if (!id.equals(other.id)) { 193 return false; 194 } 195 if (sequence == null) { 196 if (other.sequence != null) { 197 return false; 198 } 199 } else if (!sequence.equals(other.sequence)) { 200 return false; 201 } 202 return true; 203 } 204 205 @Override 206 public int compareTo(FastaSequence o) { 207 if(o==null || o.id==null) 208 return 1; 209 210 return this.getId().compareTo(o.id); 211 } 212 213}