/android/export-contacts

To get this branch, use:
bzr branch http://bzr.ed.am/android/export-contacts

« back to all changes in this revision

Viewing changes to src/am/ed/exportcontacts/VcardExporter.java

  • Committer: edam
  • Date: 2010-07-04 14:54:18 UTC
  • Revision ID: edam@waxworlds.org-20100704145418-vcpya2sxsop0dlrq
- initial checkin
- copied intro and basic vcf-configure activities from import-contacts
- copied WizzardActivity class from import-contacts

Show diffs side-by-side

added added

removed removed

1
 
/*
2
 
 * Exporter.java
3
 
 *
4
 
 * Copyright (C) 2011 to 2013 Tim Marston <tim@ed.am>
5
 
 *
6
 
 * This file is part of the Export Contacts program (hereafter referred
7
 
 * to as "this program").  For more information, see
8
 
 * http://ed.am/dev/android/export-contacts
9
 
 *
10
 
 * This program is free software: you can redistribute it and/or modify
11
 
 * it under the terms of the GNU General Public License as published by
12
 
 * the Free Software Foundation, either version 3 of the License, or
13
 
 * (at your option) any later version.
14
 
 *
15
 
 * This program is distributed in the hope that it will be useful,
16
 
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17
 
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18
 
 * GNU General Public License for more details.
19
 
 *
20
 
 * You should have received a copy of the GNU General Public License
21
 
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
22
 
 */
23
 
 
24
 
package am.ed.exportcontacts;
25
 
 
26
 
import java.io.File;
27
 
import java.io.FileNotFoundException;
28
 
import java.io.FileOutputStream;
29
 
import java.io.IOException;
30
 
import java.util.AbstractCollection;
31
 
import java.util.ArrayList;
32
 
import java.util.Iterator;
33
 
 
34
 
import android.content.SharedPreferences;
35
 
 
36
 
public class VcardExporter extends Exporter
37
 
{
38
 
        protected FileOutputStream _ostream = null;
39
 
        protected boolean _first_contact = true;
40
 
 
41
 
        public VcardExporter( Doit doit )
42
 
        {
43
 
                super( doit );
44
 
        }
45
 
 
46
 
        @Override
47
 
        protected void preExport() throws AbortExportException
48
 
        {
49
 
                SharedPreferences prefs = getSharedPreferences();
50
 
 
51
 
                // create output filename
52
 
                String filename = prefs.getString( "filename", "android-contacts.vcf" );
53
 
                File file = new File( "/sdcard" + prefs.getString( "location", "/" ) +
54
 
                        filename );
55
 
 
56
 
                // check if the output file already exists
57
 
                if( file.exists() && file.length() > 0 )
58
 
                        if( !showContinue( R.string.error_vcf_exists ) )
59
 
                                finish( ACTION_ABORT );
60
 
 
61
 
                // open file
62
 
                try {
63
 
                        _ostream = new FileOutputStream( file );
64
 
                }
65
 
                catch( FileNotFoundException e ) {
66
 
                        showError( R.string.error_filenotfound );
67
 
                }
68
 
        }
69
 
 
70
 
        /**
71
 
         * Do line folding at 75 chars
72
 
         * @param raw string
73
 
         * @return folded string
74
 
         */
75
 
        private String fold( String line )
76
 
        {
77
 
                StringBuilder ret = new StringBuilder( line.length() );
78
 
 
79
 
                // keep pulling off the first line's worth of chars, while the string is
80
 
                // still longer than a line should be
81
 
                while( line.length() > 75 )
82
 
                {
83
 
                        // length of the line we'll be pulling off
84
 
                        int len = 75;
85
 
 
86
 
                        // if splitting at this length would break apart a codepoint, use
87
 
                        // one less char
88
 
                        if( Character.isHighSurrogate( line.charAt( len - 1 ) ) )
89
 
                                len--;
90
 
 
91
 
                        // count how many backslashes would be at the end of the line we're
92
 
                        // pulling off
93
 
                        int count = 0;
94
 
                        for( int a = len - 1; a >= 0; a-- )
95
 
                                if( line.charAt( a ) == '\\' )
96
 
                                        count++;
97
 
                                else
98
 
                                        break;
99
 
 
100
 
                        // if there would be an odd number of slashes at the end of the line
101
 
                        // then pull off one fewer characters so that we don't break apart
102
 
                        // escape sequences
103
 
                        if( count % 2 == 1 )
104
 
                                len--;
105
 
 
106
 
                        // pull off the line and add it to the output, folded
107
 
                        ret.append( line.substring( 0, len ) + "\n " );
108
 
                        line = line.substring( len );
109
 
                }
110
 
 
111
 
                // add any remaining data
112
 
                ret.append( line );
113
 
 
114
 
                return ret.toString();
115
 
        }
116
 
 
117
 
        /**
118
 
         * Do unsafe character escaping
119
 
         * @param raw string
120
 
         * @return escaped string
121
 
         */
122
 
        private String escape( String str )
123
 
        {
124
 
                StringBuilder ret = new StringBuilder( str.length() );
125
 
                for( int a = 0; a < str.length(); a++ )
126
 
                {
127
 
                        int c = str.codePointAt( a );
128
 
                        switch( c )
129
 
                        {
130
 
                        case '\n':
131
 
                                // append escaped newline
132
 
                                ret.append( "\\n" );
133
 
                                break;
134
 
                        case ',':
135
 
                        case ';':
136
 
                        case '\\':
137
 
                                // append return character
138
 
                                ret.append( '\\' );
139
 
                                // fall through
140
 
                        default:
141
 
                                // append character
142
 
                                ret.append( Character.toChars( c ) );
143
 
                        }
144
 
                }
145
 
 
146
 
                return ret.toString();
147
 
        }
148
 
 
149
 
        /**
150
 
         * join
151
 
         */
152
 
        @SuppressWarnings( "rawtypes" )
153
 
        public static String join( AbstractCollection s, String delimiter)
154
 
        {
155
 
                StringBuffer buffer = new StringBuffer();
156
 
                Iterator iter = s.iterator();
157
 
                if( iter.hasNext() ) {
158
 
                        buffer.append( iter.next() );
159
 
                        while( iter.hasNext() ) {
160
 
                                buffer.append( delimiter );
161
 
                                buffer.append( iter.next() );
162
 
                        }
163
 
                }
164
 
                return buffer.toString();
165
 
        }
166
 
 
167
 
        /**
168
 
         * Is the provided value a valid date-and-or-time, as per the spec?
169
 
         *
170
 
         * @param value the value
171
 
         * @return true if it is
172
 
         */
173
 
        protected boolean isValidDateAndOrTime( String value )
174
 
        {
175
 
                // ISO 8601:2004 4.1.2 date with 4.1.2.3 a) and b) reduced accuracy
176
 
                String date =
177
 
                        "[0-9]{4}(?:-?[0-9]{2}(?:-?[0-9]{2})?)?";
178
 
 
179
 
                // ISO 8601:2000 5.2.1.3 d), e) and f) truncated date representation
180
 
                String date_trunc =
181
 
                        "--(?:[0-9]{2}(?:-?[0-9]{2})?|-[0-9]{2})";
182
 
 
183
 
                // ISO 8601:2004 4.2.2 time with 4.2.2.3 reduced accuracy, 4.2.4 UTC and
184
 
                // 4.2.5 zone offset, no 4.2.2.4 decimal fraction and no 4.2.3 24:00
185
 
                // midnight
186
 
                String time =
187
 
                        "(?:[0-1][0-9]|2[0-3])(?::?[0-5][0-9](?::?(?:60|[0-5][0-9]))?)?" +
188
 
                        "(?:Z|[-+](?:[0-1][0-9]|2[0-3])(?::?[0-5][0-9])?)?";
189
 
 
190
 
                // ISO 8601:2000 5.3.1.4 a), b) and c) truncated time representation
191
 
                String time_trunc =
192
 
                        "-(?:[0-5][0-9](?::?(?:60|[0-5][0-9]))?|-(?:60|[0-5][0-9]))";
193
 
 
194
 
                // RFC6350 (vCard 3.0) date-and-or-time with mandatory time designator
195
 
                String date_and_or_time =
196
 
                        "(?:" + date + "|" + date_trunc + ")?" +
197
 
                        "(?:T(?:" + time + "|" + time_trunc + "))?";
198
 
 
199
 
                return value.matches( date_and_or_time );
200
 
        }
201
 
 
202
 
        @Override
203
 
        protected boolean exportContact( ContactData contact )
204
 
                throws AbortExportException
205
 
        {
206
 
                StringBuilder out = new StringBuilder();
207
 
 
208
 
                // skip if the contact has no identifiable features
209
 
                if( contact.getPrimaryIdentifier() == null )
210
 
                        return false;
211
 
 
212
 
                // append newline
213
 
                if( _first_contact )
214
 
                        _first_contact = false;
215
 
                else
216
 
                        out.append( "\n" );
217
 
 
218
 
                // append header
219
 
                out.append( "BEGIN:VCARD\n" );
220
 
                out.append( "VERSION:3.0\n" );
221
 
 
222
 
                // append formatted name
223
 
                String name = contact.getName();
224
 
                if( name == null ) name = "";
225
 
                out.append( fold( "FN:" + escape( name ) ) + "\n" );
226
 
 
227
 
                // append name
228
 
                String[] bits = name.split( " +" );
229
 
                StringBuilder tmp = new StringBuilder();
230
 
                for( int a = 1; a < bits.length - 1; a++ ) {
231
 
                        if( a > 1 ) tmp.append( " " );
232
 
                        tmp.append( escape( bits[ a ] ) );
233
 
                }
234
 
                String value = escape( bits[ bits.length - 1 ] ) + ";" +
235
 
                        ( bits.length > 1? escape( bits[ 0 ] ) : "" ) + ";" +
236
 
                        tmp.toString() + ";;";
237
 
                out.append( fold( "N:" + value ) + "\n" );
238
 
 
239
 
                // append organisations and titles
240
 
                ArrayList< Exporter.ContactData.OrganisationDetail > organisations =
241
 
                        contact.getOrganisations();
242
 
                if( organisations != null ) {
243
 
                        for( int a = 0; a < organisations.size(); a++ ) {
244
 
                                if( organisations.get( a ).getOrganisation() != null )
245
 
                                        out.append( fold( "ORG:" + escape(
246
 
                                                organisations.get( a ).getOrganisation() ) ) + "\n" );
247
 
                                if( organisations.get( a ).getTitle() != null )
248
 
                                        out.append( fold( "TITLE:" + escape(
249
 
                                                organisations.get( a ).getTitle() ) ) + "\n" );
250
 
                        }
251
 
                }
252
 
 
253
 
                // append phone numbers
254
 
                ArrayList< Exporter.ContactData.NumberDetail > numbers =
255
 
                        contact.getNumbers();
256
 
                if( numbers != null ) {
257
 
                        for( int a = 0; a < numbers.size(); a++ ) {
258
 
                                ArrayList< String > types = new ArrayList< String >();
259
 
                                switch( numbers.get( a ).getType() ) {
260
 
                                case ContactData.TYPE_HOME:
261
 
                                        types.add( "VOICE" ); types.add( "HOME" ); break;
262
 
                                case ContactData.TYPE_WORK:
263
 
                                        types.add( "VOICE" ); types.add( "WORK" ); break;
264
 
                                case ContactData.TYPE_FAX_HOME:
265
 
                                        types.add( "FAX" ); types.add( "HOME" ); break;
266
 
                                case ContactData.TYPE_FAX_WORK:
267
 
                                        types.add( "FAX" ); types.add( "WORK" ); break;
268
 
                                case ContactData.TYPE_PAGER:
269
 
                                        types.add( "PAGER" ); break;
270
 
                                case ContactData.TYPE_MOBILE:
271
 
                                        types.add( "VOICE" ); types.add( "CELL" ); break;
272
 
                                }
273
 
                                if( a == 0 ) types.add( "PREF" );
274
 
                                out.append( fold( "TEL" +
275
 
                                        ( types.size() > 0? ";TYPE=" + join( types, "," ) : "" ) +
276
 
                                        ":" + escape( numbers.get( a ).getNumber() ) ) + "\n" );
277
 
                        }
278
 
                }
279
 
 
280
 
                // append email addresses
281
 
                ArrayList< Exporter.ContactData.EmailDetail > emails =
282
 
                        contact.getEmails();
283
 
                if( emails != null ) {
284
 
                        for( int a = 0; a < emails.size(); a++ ) {
285
 
                                ArrayList< String > types = new ArrayList< String >();
286
 
                                types.add( "INTERNET" );
287
 
                                switch( emails.get( a ).getType() ) {
288
 
                                case ContactData.TYPE_HOME:
289
 
                                        types.add( "HOME" ); break;
290
 
                                case ContactData.TYPE_WORK:
291
 
                                        types.add( "WORK" ); break;
292
 
                                }
293
 
                                out.append( fold( "EMAIL" +
294
 
                                        ( types.size() > 0? ";TYPE=" + join( types, "," ) : "" ) +
295
 
                                        ":" + escape( emails.get( a ).getEmail() ) ) + "\n" );
296
 
                        }
297
 
                }
298
 
 
299
 
                // append addresses
300
 
                ArrayList< Exporter.ContactData.AddressDetail > addresses =
301
 
                        contact.getAddresses();
302
 
                if( addresses != null ) {
303
 
                        for( int a = 0; a < addresses.size(); a++ ) {
304
 
                                ArrayList< String > types = new ArrayList< String >();
305
 
                                types.add( "POSTAL" );
306
 
                                switch( addresses.get( a ).getType() ) {
307
 
                                case ContactData.TYPE_HOME:
308
 
                                        types.add( "HOME" ); break;
309
 
                                case ContactData.TYPE_WORK:
310
 
                                        types.add( "WORK" ); break;
311
 
                                }
312
 
                                // we use LABEL because is accepts formatted text (whereas ADR
313
 
                                // expects semicolon-delimited fields with specific purposes)
314
 
                                out.append( fold( "LABEL" +
315
 
                                        ( types.size() > 0? ";TYPE=" + join( types, "," ) : "" ) +
316
 
                                        ":" + escape( addresses.get( a ).getAddress() ) ) + "\n" );
317
 
                        }
318
 
                }
319
 
 
320
 
                // append notes
321
 
                ArrayList< String > notes = contact.getNotes();
322
 
                if( notes != null )
323
 
                        for( int a = 0; a < notes.size(); a++ )
324
 
                                out.append( fold( "NOTE:" + escape( notes.get( a ) ) ) + "\n" );
325
 
 
326
 
                // append birthday
327
 
                String birthday = contact.getBirthday();
328
 
                if( birthday != null ) {
329
 
                        birthday.trim();
330
 
                        if( isValidDateAndOrTime( birthday ) )
331
 
                                out.append( fold( "BDAY:" + escape( birthday ) ) + "\n" );
332
 
                        else
333
 
                                out.append(
334
 
                                        fold( "BDAY;VALUE=text:" + escape( birthday ) ) + "\n" );
335
 
                }
336
 
 
337
 
                // append footer
338
 
                out.append( "END:VCARD\n" );
339
 
 
340
 
                // replace '\n' with "\r\n" (spec requires CRLF)
341
 
                int pos = 0;
342
 
                while( true ) {
343
 
                        pos = out.indexOf( "\n", pos );
344
 
                        if( pos == -1 ) break;
345
 
                        out.replace( pos, pos + 1, "\r\n" );
346
 
 
347
 
                        // skip our inserted string
348
 
                        pos += 2;
349
 
                }
350
 
 
351
 
                // write to file
352
 
                try {
353
 
                        _ostream.write( out.toString().getBytes() );
354
 
                        _ostream.flush();
355
 
                }
356
 
                catch( IOException e ) {
357
 
                        showError( R.string.error_ioerror );
358
 
                }
359
 
 
360
 
                return true;
361
 
        }
362
 
 
363
 
}