diff --git a/app/build.gradle b/app/build.gradle index b94b1cd0..899893a6 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -10,7 +10,7 @@ apply plugin: 'com.android.application' android { compileSdkVersion 22 - buildToolsVersion '21.1.2' + buildToolsVersion '22.0.1' defaultConfig { applicationId "at.bitfire.davdroid" diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index c5fa1ca9..80df7b32 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -26,6 +26,12 @@ + + + + + + - @@ -64,11 +69,30 @@ - + + + + + + + + + + + + { public CalDavCalendar(CloseableHttpClient httpClient, String baseURL, String user, String password, boolean preemptiveAuth) throws URISyntaxException { super(httpClient, baseURL, user, password, preemptiveAuth); - } - @Override protected String memberAcceptedMimeTypes() { diff --git a/app/src/main/java/at/bitfire/davdroid/resource/CalDavNotebook.java b/app/src/main/java/at/bitfire/davdroid/resource/CalDavNotebook.java new file mode 100644 index 00000000..45623324 --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/resource/CalDavNotebook.java @@ -0,0 +1,72 @@ +package at.bitfire.davdroid.resource; + +import android.util.Log; + +import org.apache.http.impl.client.CloseableHttpClient; +import org.simpleframework.xml.Serializer; +import org.simpleframework.xml.core.Persister; + +import java.io.StringWriter; +import java.net.URISyntaxException; + +import at.bitfire.davdroid.webdav.DavCalendarQuery; +import at.bitfire.davdroid.webdav.DavCompFilter; +import at.bitfire.davdroid.webdav.DavFilter; +import at.bitfire.davdroid.webdav.DavMultiget; +import at.bitfire.davdroid.webdav.DavProp; + +public class CalDavNotebook extends RemoteCollection { + private final static String TAG = "davdroid.CalDAVNotebook"; + + public CalDavNotebook(CloseableHttpClient httpClient, String baseURL, String user, String password, boolean preemptiveAuth) throws URISyntaxException { + super(httpClient, baseURL, user, password, preemptiveAuth); + } + + @Override + protected String memberAcceptedMimeTypes() + { + return "text/calendar"; + } + + @Override + protected DavMultiget.Type multiGetType() { + return DavMultiget.Type.CALENDAR; + } + + @Override + protected Note newResourceSkeleton(String name, String ETag) { + return new Note(name, ETag); + } + + + @Override + public String getMemberETagsQuery() { + DavCalendarQuery query = new DavCalendarQuery(); + + // prop + DavProp prop = new DavProp(); + prop.setGetetag(new DavProp.GetETag()); + query.setProp(prop); + + // filter + DavFilter filter = new DavFilter(); + query.setFilter(filter); + + DavCompFilter compFilter = new DavCompFilter("VCALENDAR"); + filter.setCompFilter(compFilter); + + compFilter.setCompFilter(new DavCompFilter("VJOURNAL")); + + Serializer serializer = new Persister(); + StringWriter writer = new StringWriter(); + try { + serializer.write(query, writer); + } catch (Exception e) { + Log.e(TAG, "Couldn't prepare REPORT query", e); + return null; + } + + return writer.toString(); + } + +} diff --git a/app/src/main/java/at/bitfire/davdroid/resource/CalDavTaskList.java b/app/src/main/java/at/bitfire/davdroid/resource/CalDavTaskList.java new file mode 100644 index 00000000..abc6d271 --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/resource/CalDavTaskList.java @@ -0,0 +1,72 @@ +package at.bitfire.davdroid.resource; + +import android.util.Log; + +import org.apache.http.impl.client.CloseableHttpClient; +import org.simpleframework.xml.Serializer; +import org.simpleframework.xml.core.Persister; + +import java.io.StringWriter; +import java.net.URISyntaxException; + +import at.bitfire.davdroid.webdav.DavCalendarQuery; +import at.bitfire.davdroid.webdav.DavCompFilter; +import at.bitfire.davdroid.webdav.DavFilter; +import at.bitfire.davdroid.webdav.DavMultiget; +import at.bitfire.davdroid.webdav.DavProp; + +public class CalDavTaskList extends RemoteCollection { + private final static String TAG = "davdroid.CalDAVTaskList"; + + public CalDavTaskList(CloseableHttpClient httpClient, String baseURL, String user, String password, boolean preemptiveAuth) throws URISyntaxException { + super(httpClient, baseURL, user, password, preemptiveAuth); + } + + @Override + protected String memberAcceptedMimeTypes() + { + return "text/calendar"; + } + + @Override + protected DavMultiget.Type multiGetType() { + return DavMultiget.Type.CALENDAR; + } + + @Override + protected Task newResourceSkeleton(String name, String ETag) { + return new Task(name, ETag); + } + + + @Override + public String getMemberETagsQuery() { + DavCalendarQuery query = new DavCalendarQuery(); + + // prop + DavProp prop = new DavProp(); + prop.setGetetag(new DavProp.GetETag()); + query.setProp(prop); + + // filter + DavFilter filter = new DavFilter(); + query.setFilter(filter); + + DavCompFilter compFilter = new DavCompFilter("VCALENDAR"); + filter.setCompFilter(compFilter); + + compFilter.setCompFilter(new DavCompFilter("VTODO")); + + Serializer serializer = new Persister(); + StringWriter writer = new StringWriter(); + try { + serializer.write(query, writer); + } catch (Exception e) { + Log.e(TAG, "Couldn't prepare REPORT query", e); + return null; + } + + return writer.toString(); + } + +} diff --git a/app/src/main/java/at/bitfire/davdroid/resource/DavResourceFinder.java b/app/src/main/java/at/bitfire/davdroid/resource/DavResourceFinder.java index 1e5b1009..78790b0e 100644 --- a/app/src/main/java/at/bitfire/davdroid/resource/DavResourceFinder.java +++ b/app/src/main/java/at/bitfire/davdroid/resource/DavResourceFinder.java @@ -131,25 +131,35 @@ public class DavResourceFinder implements Closeable { for (WebDavResource resource : possibleCalendars) if (resource.isCalendar()) { Log.i(TAG, "Found calendar: " + resource.getLocation().getPath()); - if (resource.getSupportedComponents() != null) { + ServerInfo.ResourceInfo info = new ServerInfo.ResourceInfo( + ServerInfo.ResourceInfo.Type.CALENDAR, + resource.isReadOnly(), + resource.getLocation().toString(), + resource.getDisplayName(), + resource.getDescription(), resource.getColor() + ); + info.setTimezone(resource.getTimezone()); + + if (resource.getSupportedComponents() == null) { + // no info about supported components, assuming all components are supported + info.setSupportingEvents(true); + info.setSupportingNotes(true); + } else { // CALDAV:supported-calendar-component-set available - boolean supportsEvents = false; for (String supportedComponent : resource.getSupportedComponents()) - if (supportedComponent.equalsIgnoreCase("VEVENT")) - supportsEvents = true; - if (!supportsEvents) { // ignore collections without VEVENT support - Log.i(TAG, "Ignoring this calendar because of missing VEVENT support"); + if ("VEVENT".equalsIgnoreCase(supportedComponent)) + info.setSupportingEvents(true); + else if ("VJOURNAL".equalsIgnoreCase(supportedComponent)) + info.setSupportingNotes(true); + else if ("VTODO".equalsIgnoreCase(supportedComponent)) + info.setSupportingTasks(true); + + if (!info.isSupportingEvents() && !info.isSupportingNotes() && !info.isSupportingTasks()) { + Log.i(TAG, "Ignoring this calendar because it supports neither VEVENT nor VJOURNAL nor VTODO"); continue; } } - ServerInfo.ResourceInfo info = new ServerInfo.ResourceInfo( - ServerInfo.ResourceInfo.Type.CALENDAR, - resource.isReadOnly(), - resource.getLocation().toString(), - resource.getDisplayName(), - resource.getDescription(), resource.getColor() - ); - info.setTimezone(resource.getTimezone()); + calendars.add(info); } serverInfo.setCalendars(calendars); diff --git a/app/src/main/java/at/bitfire/davdroid/resource/Event.java b/app/src/main/java/at/bitfire/davdroid/resource/Event.java index 1e34c680..6b86c46a 100644 --- a/app/src/main/java/at/bitfire/davdroid/resource/Event.java +++ b/app/src/main/java/at/bitfire/davdroid/resource/Event.java @@ -244,7 +244,7 @@ public class Event extends Resource { public ByteArrayOutputStream toEntity() throws IOException { net.fortuna.ical4j.model.Calendar ical = new net.fortuna.ical4j.model.Calendar(); ical.getProperties().add(Version.VERSION_2_0); - ical.getProperties().add(new ProdId("-//bitfire web engineering//DAVdroid " + Constants.APP_VERSION + " (ical4j 1.0.x)//EN")); + ical.getProperties().add(Constants.ICAL_PRODID); // "master event" (without exceptions) ComponentList components = ical.getComponents(); diff --git a/app/src/main/java/at/bitfire/davdroid/resource/LocalAddressBook.java b/app/src/main/java/at/bitfire/davdroid/resource/LocalAddressBook.java index ca2ebe9b..b058c0aa 100644 --- a/app/src/main/java/at/bitfire/davdroid/resource/LocalAddressBook.java +++ b/app/src/main/java/at/bitfire/davdroid/resource/LocalAddressBook.java @@ -654,12 +654,15 @@ public class LocalAddressBook extends LocalCollection { /* content builder methods */ @Override - protected Builder buildEntry(Builder builder, Resource resource) { + protected Builder buildEntry(Builder builder, Resource resource, boolean update) { Contact contact = (Contact)resource; + if (!update) + builder = builder + .withValue(RawContacts.ACCOUNT_NAME, account.name) + .withValue(RawContacts.ACCOUNT_TYPE, account.type); + return builder - .withValue(RawContacts.ACCOUNT_NAME, account.name) - .withValue(RawContacts.ACCOUNT_TYPE, account.type) .withValue(entryColumnRemoteName(), contact.getName()) .withValue(entryColumnUID(), contact.getUid()) .withValue(entryColumnETag(), contact.getETag()) diff --git a/app/src/main/java/at/bitfire/davdroid/resource/LocalCalendar.java b/app/src/main/java/at/bitfire/davdroid/resource/LocalCalendar.java index a661d6fc..5212cfe5 100644 --- a/app/src/main/java/at/bitfire/davdroid/resource/LocalCalendar.java +++ b/app/src/main/java/at/bitfire/davdroid/resource/LocalCalendar.java @@ -87,30 +87,25 @@ import lombok.Getter; public class LocalCalendar extends LocalCollection { private static final String TAG = "davdroid.LocalCalendar"; - @Getter protected long id; @Getter protected String url; + @Getter protected long id; protected static String COLLECTION_COLUMN_CTAG = Calendars.CAL_SYNC1; /* database fields */ + @Override protected Uri entriesURI() { return syncAdapterURI(Events.CONTENT_URI); } + @Override protected String entryColumnAccountType() { return Events.ACCOUNT_TYPE; } + @Override protected String entryColumnAccountName() { return Events.ACCOUNT_NAME; } + @Override protected String entryColumnParentID() { return Events.CALENDAR_ID; } + @Override protected String entryColumnID() { return Events._ID; } + @Override protected String entryColumnRemoteName() { return Events._SYNC_ID; } + @Override protected String entryColumnETag() { return Events.SYNC_DATA1; } + @Override protected String entryColumnDirty() { return Events.DIRTY; } + @Override protected String entryColumnDeleted() { return Events.DELETED; } + @Override - protected Uri entriesURI() { - return syncAdapterURI(Events.CONTENT_URI); - } - - protected String entryColumnAccountType() { return Events.ACCOUNT_TYPE; } - protected String entryColumnAccountName() { return Events.ACCOUNT_NAME; } - - protected String entryColumnParentID() { return Events.CALENDAR_ID; } - protected String entryColumnID() { return Events._ID; } - protected String entryColumnRemoteName() { return Events._SYNC_ID; } - protected String entryColumnETag() { return Events.SYNC_DATA1; } - - protected String entryColumnDirty() { return Events.DIRTY; } - protected String entryColumnDeleted() { return Events.DELETED; } - @TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR1) protected String entryColumnUID() { return (android.os.Build.VERSION.SDK_INT >= 17) ? @@ -164,7 +159,7 @@ public class LocalCalendar extends LocalCollection { if (info.getTimezone() != null) values.put(Calendars.CALENDAR_TIME_ZONE, info.getTimezone()); - Log.i(TAG, "Inserting calendar: " + values.toString() + " -> " + calendarsURI(account).toString()); + Log.i(TAG, "Inserting calendar: " + values.toString()); try { return client.insert(calendarsURI(account), values); } catch (RemoteException e) { @@ -176,8 +171,8 @@ public class LocalCalendar extends LocalCollection { @Cleanup Cursor cursor = providerClient.query(calendarsURI(account), new String[] { Calendars._ID, Calendars.NAME }, Calendars.DELETED + "=0 AND " + Calendars.SYNC_EVENTS + "=1", null, null); - - LinkedList calendars = new LinkedList(); + + LinkedList calendars = new LinkedList<>(); while (cursor != null && cursor.moveToNext()) calendars.add(new LocalCalendar(account, providerClient, cursor.getInt(0), cursor.getString(1))); return calendars.toArray(new LocalCalendar[0]); @@ -198,9 +193,9 @@ public class LocalCalendar extends LocalCollection { try { @Cleanup Cursor c = providerClient.query(ContentUris.withAppendedId(calendarsURI(), id), new String[] { COLLECTION_COLUMN_CTAG }, null, null, null); - if (c.moveToFirst()) { + if (c != null && c.moveToFirst()) return c.getString(0); - } else + else throw new LocalStorageException("Couldn't query calendar CTag"); } catch(RemoteException e) { throw new LocalStorageException(e); @@ -528,9 +523,14 @@ public class LocalCalendar extends LocalCollection { /* content builder methods */ @Override - protected Builder buildEntry(Builder builder, Resource resource) { + protected Builder buildEntry(Builder builder, Resource resource, boolean update) { final Event event = (Event)resource; + if (!update) + builder = builder + .withValue(Events.ACCOUNT_TYPE, account.type) + .withValue(Events.ACCOUNT_NAME, account.name); + builder = builder .withValue(Events.CALENDAR_ID, id) .withValue(Events.ALL_DAY, event.isAllDay() ? 1 : 0) @@ -648,7 +648,7 @@ public class LocalCalendar extends LocalCollection { protected Builder buildException(Builder builder, Event master, Event exception) { - buildEntry(builder, exception); + buildEntry(builder, exception, false); builder.withValue(Events.ORIGINAL_SYNC_ID, exception.getName()); // Some servers (iCloud, for instance) return RECURRENCE-ID with DATE-TIME even if @@ -747,9 +747,11 @@ public class LocalCalendar extends LocalCollection { /* private helper methods */ protected static Uri calendarsURI(Account account) { - return Calendars.CONTENT_URI.buildUpon().appendQueryParameter(Calendars.ACCOUNT_NAME, account.name) + return Calendars.CONTENT_URI.buildUpon() + .appendQueryParameter(Calendars.ACCOUNT_NAME, account.name) .appendQueryParameter(Calendars.ACCOUNT_TYPE, account.type) - .appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true").build(); + .appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true") + .build(); } protected Uri calendarsURI() { diff --git a/app/src/main/java/at/bitfire/davdroid/resource/LocalCollection.java b/app/src/main/java/at/bitfire/davdroid/resource/LocalCollection.java index 57c16ca4..e396b88d 100644 --- a/app/src/main/java/at/bitfire/davdroid/resource/LocalCollection.java +++ b/app/src/main/java/at/bitfire/davdroid/resource/LocalCollection.java @@ -15,14 +15,20 @@ import android.content.ContentUris; import android.content.ContentValues; import android.content.OperationApplicationException; import android.database.Cursor; +import android.database.DatabaseUtils; import android.net.Uri; import android.os.RemoteException; import android.provider.CalendarContract; import android.util.Log; +import org.apache.commons.lang.StringUtils; + import java.util.ArrayList; +import java.util.LinkedList; +import java.util.List; import lombok.Cleanup; +import lombok.Getter; /** * Represents a locally-stored synchronizable collection (for instance, the @@ -67,6 +73,10 @@ public abstract class LocalCollection { /** column name of an entry's UID */ abstract protected String entryColumnUID(); + + /** ID of the collection (for instance, CalendarContract.Calendars._ID) */ + // protected long id; + /** SQL filter expression */ String sqlFilter; @@ -114,7 +124,7 @@ public abstract class LocalCollection { for (int idx = 0; cursor.moveToNext(); idx++) { long id = cursor.getLong(0); - // new record: generate UID + remote file name so that we can upload + // new record: we have to generate UID + remote file name for uploading T resource = findById(id, false); resource.initialize(); // write generated UID + remote file name into database @@ -218,9 +228,9 @@ public abstract class LocalCollection { /** * Finds a specific resource by remote file name. Only records matching sqlFilter are taken into account. - * @param localID remote file name of the resource - * @param populate true: populates all data fields (for instance, contact or event details); - * false: only remote file name and ETag are populated + * @param remoteName remote file name of the resource + * @param populate true: populates all data fields (for instance, contact or event details); + * false: only remote file name and ETag are populated * @return resource with either ID/remote file/name/ETag or all fields populated * @throws RecordNotFoundException when the resource couldn't be found * @throws LocalStorageException when the content provider couldn't be queried @@ -255,7 +265,7 @@ public abstract class LocalCollection { * Creates a new resource object in memory. No content provider operations involved. * @param localID the ID of the resource * @param resourceName the (remote) file name of the resource - * @param ETag of the resource + * @param eTag ETag of the resource * @return the new resource object */ abstract public T newResource(long localID, String resourceName, String eTag); @@ -263,9 +273,9 @@ public abstract class LocalCollection { public void add(Resource resource) { int idx = pendingOperations.size(); pendingOperations.add( - buildEntry(ContentProviderOperation.newInsert(entriesURI()), resource) - .withYieldAllowed(true) - .build()); + buildEntry(ContentProviderOperation.newInsert(entriesURI()), resource, false) + .withYieldAllowed(true) + .build()); addDataRows(resource, -1, idx); } @@ -275,7 +285,7 @@ public abstract class LocalCollection { public void updateByRemoteName(Resource remoteResource) throws LocalStorageException { T localResource = findByRemoteName(remoteResource.getName(), false); pendingOperations.add( - buildEntry(ContentProviderOperation.newUpdate(ContentUris.withAppendedId(entriesURI(), localResource.getLocalID())), remoteResource) + buildEntry(ContentProviderOperation.newUpdate(ContentUris.withAppendedId(entriesURI(), localResource.getLocalID())), remoteResource, true) .withValue(entryColumnETag(), remoteResource.getETag()) .withYieldAllowed(true) .build()); @@ -296,8 +306,28 @@ public abstract class LocalCollection { * Enqueues deleting all resources except the give ones from the local collection. Requires commit(). * @param remoteResources resources with these remote file names will be kept */ - public abstract void deleteAllExceptRemoteNames(Resource[] remoteResources); - + public void deleteAllExceptRemoteNames(Resource[] remoteResources) { + final String where; + + if (remoteResources.length != 0) { + // delete all except certain entries + final List sqlFileNames = new LinkedList<>(); + for (final Resource res : remoteResources) + sqlFileNames.add(DatabaseUtils.sqlEscapeString(res.getName())); + where = entryColumnRemoteName() + " NOT IN (" + StringUtils.join(sqlFileNames, ",") + ')'; + } else + // delete all entries + where = entryColumnRemoteName() + " IS NOT NULL"; + + ContentProviderOperation.Builder builder = ContentProviderOperation.newDelete(entriesURI()) + .withSelection( // restrict deletion to parent collection + entryColumnParentID() + "=? AND (" + where + ')', + new String[] { String.valueOf(getId()) } + ); + pendingOperations.add(builder.withYieldAllowed(true).build()); + } + + /** Updates the locally-known ETag of a resource. */ public void updateETag(Resource res, String eTag) throws LocalStorageException { Log.d(TAG, "Setting ETag of local resource " + res.getName() + " to " + eTag); @@ -365,9 +395,11 @@ public abstract class LocalCollection { * Builds the main entry (for instance, a ContactsContract.RawContacts row) from a resource. * The entry is built for insertion to the location identified by entriesURI(). * - * @param builder Builder to be extended by all resource data that can be stored without extra data rows. + * @param builder Builder to be extended by all resource data that can be stored without extra data rows. + * @param resource Event, task or note resource whose contents shall be inserted/updated + * @param update false when the entry is built the first time (when creating the row), true if it's an update */ - protected abstract Builder buildEntry(Builder builder, Resource resource); + protected abstract Builder buildEntry(Builder builder, Resource resource, boolean update); /** Enqueues adding extra data rows of the resource to the local collection. */ protected abstract void addDataRows(Resource resource, long localID, int backrefIdx); diff --git a/app/src/main/java/at/bitfire/davdroid/resource/LocalNotebook.java b/app/src/main/java/at/bitfire/davdroid/resource/LocalNotebook.java new file mode 100644 index 00000000..e8f915fc --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/resource/LocalNotebook.java @@ -0,0 +1,179 @@ +package at.bitfire.davdroid.resource; + +import android.accounts.Account; +import android.content.ContentProviderClient; +import android.content.ContentProviderOperation; +import android.content.ContentResolver; +import android.content.ContentUris; +import android.content.ContentValues; +import android.database.Cursor; +import android.database.DatabaseUtils; +import android.net.Uri; +import android.os.RemoteException; +import android.provider.CalendarContract; +import android.util.Log; + +import net.fortuna.ical4j.model.DateTime; +import net.fortuna.ical4j.model.property.Created; +import net.fortuna.ical4j.model.property.DtStamp; + +import org.apache.commons.lang.StringUtils; + +import java.util.LinkedList; +import java.util.List; + +import at.bitfire.davdroid.Constants; +import at.bitfire.notebooks.provider.NoteContract; +import lombok.Cleanup; +import lombok.Getter; + +public class LocalNotebook extends LocalCollection { + private final static String TAG = "davdroid.LocalNotebook"; + + @Getter protected final String url; + @Getter protected final long id; + + protected static String COLLECTION_COLUMN_CTAG = NoteContract.Notebooks.SYNC1; + + @Override protected Uri entriesURI() { return syncAdapterURI(NoteContract.Notes.CONTENT_URI); } + @Override protected String entryColumnAccountType() { return NoteContract.Notes.ACCOUNT_TYPE; } + @Override protected String entryColumnAccountName() { return NoteContract.Notes.ACCOUNT_NAME; } + @Override protected String entryColumnParentID() { return NoteContract.Notes.NOTEBOOK_ID; } + @Override protected String entryColumnID() { return NoteContract.Notes._ID; } + @Override protected String entryColumnRemoteName() { return NoteContract.Notes._SYNC_ID; } + @Override protected String entryColumnETag() { return NoteContract.Notes.SYNC1; } + @Override protected String entryColumnDirty() { return NoteContract.Notes.DIRTY; } + @Override protected String entryColumnDeleted() { return NoteContract.Notes.DELETED; } + @Override protected String entryColumnUID() { return NoteContract.Notes.UID; } + + + public static Uri create(Account account, ContentResolver resolver, ServerInfo.ResourceInfo info) throws LocalStorageException { + final ContentProviderClient client = resolver.acquireContentProviderClient(NoteContract.AUTHORITY); + if (client == null) + throw new LocalStorageException("No notes provider found"); + + ContentValues values = new ContentValues(); + values.put(NoteContract.Notebooks._SYNC_ID, info.getURL()); + values.put(NoteContract.Notebooks.NAME, info.getTitle()); + + Log.i(TAG, "Inserting notebook: " + values.toString()); + try { + return client.insert(notebooksURI(account), values); + } catch (RemoteException e) { + throw new LocalStorageException(e); + } + } + + public static LocalNotebook[] findAll(Account account, ContentProviderClient providerClient) throws RemoteException { + @Cleanup Cursor cursor = providerClient.query(notebooksURI(account), + new String[] { NoteContract.Notebooks._ID, NoteContract.Notebooks._SYNC_ID }, + NoteContract.Notebooks.DELETED + "=0", null, null); + + LinkedList notebooks = new LinkedList<>(); + while (cursor != null && cursor.moveToNext()) + notebooks.add(new LocalNotebook(account, providerClient, cursor.getInt(0), cursor.getString(1))); + return notebooks.toArray(new LocalNotebook[0]); + } + + public LocalNotebook(Account account, ContentProviderClient providerClient, long id, String url) throws RemoteException { + super(account, providerClient); + this.id = id; + this.url = url; + + } + + + @Override + public String getCTag() throws LocalStorageException { + try { + @Cleanup Cursor c = providerClient.query(ContentUris.withAppendedId(notebooksURI(account), id), + new String[] { COLLECTION_COLUMN_CTAG }, null, null, null); + if (c != null && c.moveToFirst()) + return c.getString(0); + else + throw new LocalStorageException("Couldn't query notebook CTag"); + } catch(RemoteException e) { + throw new LocalStorageException(e); + } + } + + @Override + public void setCTag(String cTag) throws LocalStorageException { + ContentValues values = new ContentValues(1); + values.put(COLLECTION_COLUMN_CTAG, cTag); + try { + providerClient.update(ContentUris.withAppendedId(notebooksURI(account), id), values, null, null); + } catch(RemoteException e) { + throw new LocalStorageException(e); + } + } + + @Override + public Note newResource(long localID, String resourceName, String eTag) { + return new Note(localID, resourceName, eTag); + } + + + @Override + public void populate(Resource record) throws LocalStorageException { + try { + @Cleanup final Cursor cursor = providerClient.query(entriesURI(), + new String[] { + /* 0 */ entryColumnUID(), NoteContract.Notes.CREATED_AT, NoteContract.Notes.UPDATED_AT, NoteContract.Notes.DTSTART, + /* 4 */ NoteContract.Notes.SUMMARY, NoteContract.Notes.DESCRIPTION, NoteContract.Notes.COMMENT, + /* 7 */ NoteContract.Notes.ORGANIZER, NoteContract.Notes.STATUS, NoteContract.Notes.CLASSIFICATION, + /* 10 */ NoteContract.Notes.CONTACT, NoteContract.Notes.URL + }, entryColumnID() + "=?", new String[]{ String.valueOf(record.getLocalID()) }, null); + + Note note = (Note)record; + if (cursor != null && cursor.moveToFirst()) { + note.setUid(cursor.getString(0)); + + if (!cursor.isNull(1)) + note.setCreated(new Created(new DateTime(cursor.getLong(1)))); + + note.setSummary(cursor.getString(4)); + note.setDescription(cursor.getString(5)); + } + + } catch (RemoteException e) { + throw new LocalStorageException("Couldn't process locally stored note", e); + } + } + + @Override + protected ContentProviderOperation.Builder buildEntry(ContentProviderOperation.Builder builder, Resource resource, boolean update) { + final Note note = (Note)resource; + builder = builder + .withValue(entryColumnParentID(), id) + .withValue(entryColumnRemoteName(), note.getName()) + .withValue(entryColumnUID(), note.getUid()) + .withValue(entryColumnETag(), note.getETag()) + .withValue(NoteContract.Notes.SUMMARY, note.getSummary()) + .withValue(NoteContract.Notes.DESCRIPTION, note.getDescription()); + + if (note.getCreated() != null) + builder = builder.withValue(NoteContract.Notes.CREATED_AT, note.getCreated().getDateTime().getTime()); + + return builder; + } + + @Override + protected void addDataRows(Resource resource, long localID, int backrefIdx) { + } + + @Override + protected void removeDataRows(Resource resource) { + } + + + // helpers + + protected static Uri notebooksURI(Account account) { + return NoteContract.Notebooks.CONTENT_URI.buildUpon() + .appendQueryParameter(NoteContract.Notebooks.ACCOUNT_TYPE, account.type) + .appendQueryParameter(NoteContract.Notebooks.ACCOUNT_NAME, account.name) + .appendQueryParameter(NoteContract.CALLER_IS_SYNCADAPTER, "true") + .build(); + } +} diff --git a/app/src/main/java/at/bitfire/davdroid/resource/LocalTaskList.java b/app/src/main/java/at/bitfire/davdroid/resource/LocalTaskList.java new file mode 100644 index 00000000..dd883279 --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/resource/LocalTaskList.java @@ -0,0 +1,288 @@ +package at.bitfire.davdroid.resource; + +import android.accounts.Account; +import android.content.ContentProviderClient; +import android.content.ContentProviderOperation; +import android.content.ContentResolver; +import android.content.ContentUris; +import android.content.ContentValues; +import android.database.Cursor; +import android.net.Uri; +import android.os.RemoteException; +import android.util.Log; + +import net.fortuna.ical4j.model.Date; +import net.fortuna.ical4j.model.DateTime; +import net.fortuna.ical4j.model.property.Clazz; +import net.fortuna.ical4j.model.property.Completed; +import net.fortuna.ical4j.model.property.DtStart; +import net.fortuna.ical4j.model.property.Status; + +import org.dmfs.provider.tasks.TaskContract; + +import java.util.LinkedList; + +import lombok.Cleanup; +import lombok.Getter; + +public class LocalTaskList extends LocalCollection { + private static final String TAG = "davdroid.LocalTaskList"; + + @Getter protected String url; + @Getter protected long id; + + protected static String COLLECTION_COLUMN_CTAG = TaskContract.TaskLists.SYNC1; + + @Override protected Uri entriesURI() { return syncAdapterURI(TaskContract.Tasks.CONTENT_URI); } + @Override protected String entryColumnAccountType() { return TaskContract.Tasks.ACCOUNT_TYPE; } + @Override protected String entryColumnAccountName() { return TaskContract.Tasks.ACCOUNT_NAME; } + @Override protected String entryColumnParentID() { return TaskContract.Tasks.LIST_ID; } + @Override protected String entryColumnID() { return TaskContract.Tasks._ID; } + @Override protected String entryColumnRemoteName() { return TaskContract.Tasks._SYNC_ID; } + @Override protected String entryColumnETag() { return TaskContract.Tasks.SYNC1; } + @Override protected String entryColumnDirty() { return TaskContract.Tasks._DIRTY; } + @Override protected String entryColumnDeleted() { return TaskContract.Tasks._DELETED; } + @Override protected String entryColumnUID() { return TaskContract.Tasks.SYNC2; } + + + public static Uri create(Account account, ContentResolver resolver, ServerInfo.ResourceInfo info) throws LocalStorageException { + final ContentProviderClient client = resolver.acquireContentProviderClient(TaskContract.AUTHORITY); + if (client == null) + throw new LocalStorageException("No tasks provider found"); + + ContentValues values = new ContentValues(); + values.put(TaskContract.TaskLists.ACCOUNT_NAME, account.name); + values.put(TaskContract.TaskLists.ACCOUNT_TYPE, /*account.type*/"davdroid.new"); + values.put(TaskContract.TaskLists._SYNC_ID, info.getURL()); + values.put(TaskContract.TaskLists.LIST_NAME, info.getTitle()); + values.put(TaskContract.TaskLists.OWNER, account.name); + values.put(TaskContract.TaskLists.ACCESS_LEVEL, 0); + values.put(TaskContract.TaskLists.SYNC_ENABLED, 1); + values.put(TaskContract.TaskLists.VISIBLE, 1); + + Log.i(TAG, "Inserting task list: " + values.toString()); + try { + return client.insert(taskListsURI(account), values); + } catch (RemoteException e) { + throw new LocalStorageException(e); + } + } + + public static LocalTaskList[] findAll(Account account, ContentProviderClient providerClient) throws RemoteException { + @Cleanup Cursor cursor = providerClient.query(taskListsURI(account), + new String[] { TaskContract.TaskLists._ID, TaskContract.TaskLists._SYNC_ID }, + null, null, null); + + LinkedList taskList = new LinkedList<>(); + while (cursor != null && cursor.moveToNext()) + taskList.add(new LocalTaskList(account, providerClient, cursor.getInt(0), cursor.getString(1))); + return taskList.toArray(new LocalTaskList[0]); + } + + public LocalTaskList(Account account, ContentProviderClient providerClient, long id, String url) throws RemoteException { + super(account, providerClient); + this.id = id; + this.url = url; + } + + + @Override + public String getCTag() throws LocalStorageException { + try { + @Cleanup Cursor c = providerClient.query(ContentUris.withAppendedId(taskListsURI(account), id), + new String[] { COLLECTION_COLUMN_CTAG }, null, null, null); + if (c != null && c.moveToFirst()) + return c.getString(0); + else + throw new LocalStorageException("Couldn't query task list CTag"); + } catch(RemoteException e) { + throw new LocalStorageException(e); + } + } + + @Override + public void setCTag(String cTag) throws LocalStorageException { + ContentValues values = new ContentValues(1); + values.put(COLLECTION_COLUMN_CTAG, cTag); + try { + providerClient.update(ContentUris.withAppendedId(taskListsURI(account), id), values, null, null); + } catch(RemoteException e) { + throw new LocalStorageException(e); + } + } + + @Override + public Task newResource(long localID, String resourceName, String eTag) { + return new Task(localID, resourceName, eTag); + } + + + @Override + public void populate(Resource record) throws LocalStorageException { + try { + @Cleanup final Cursor cursor = providerClient.query(entriesURI(), + new String[] { + /* 0 */ entryColumnUID(), TaskContract.Tasks.TITLE, TaskContract.Tasks.LOCATION, TaskContract.Tasks.DESCRIPTION, TaskContract.Tasks.URL, + /* 5 */ TaskContract.Tasks.CLASSIFICATION, TaskContract.Tasks.STATUS, TaskContract.Tasks.PERCENT_COMPLETE, + /* 8 */ TaskContract.Tasks.DTSTART, TaskContract.Tasks.IS_ALLDAY, /*TaskContract.Tasks.COMPLETED, TaskContract.Tasks.COMPLETED_IS_ALLDAY*/ + }, entryColumnID() + "=?", new String[]{ String.valueOf(record.getLocalID()) }, null); + + Task task = (Task)record; + if (cursor != null && cursor.moveToFirst()) { + task.setUid(cursor.getString(0)); + + task.setSummary(cursor.getString(1)); + task.setLocation(cursor.getString(2)); + task.setDescription(cursor.getString(3)); + task.setUrl(cursor.getString(4)); + + if (!cursor.isNull(5)) + switch (cursor.getInt(5)) { + case TaskContract.Tasks.CLASSIFICATION_PUBLIC: + task.setClassification(Clazz.PUBLIC); + break; + case TaskContract.Tasks.CLASSIFICATION_CONFIDENTIAL: + task.setClassification(Clazz.CONFIDENTIAL); + break; + default: + task.setClassification(Clazz.PRIVATE); + } + + if (!cursor.isNull(6)) + switch (cursor.getInt(6)) { + case TaskContract.Tasks.STATUS_IN_PROCESS: + task.setStatus(Status.VTODO_IN_PROCESS); + break; + case TaskContract.Tasks.STATUS_COMPLETED: + task.setStatus(Status.VTODO_COMPLETED); + break; + case TaskContract.Tasks.STATUS_CANCELLED: + task.setStatus(Status.VTODO_CANCELLED); + break; + default: + task.setStatus(Status.VTODO_NEEDS_ACTION); + } + if (!cursor.isNull(7)) + task.setPercentComplete(cursor.getInt(7)); + + if (!cursor.isNull(8) && !cursor.isNull(9)) { + long ts = cursor.getLong(8); + boolean allDay = cursor.getInt(9) != 0; + task.setDtStart(new DtStart(allDay ? new Date(ts) : new DateTime(ts))); + } + + /*if (!cursor.isNull(10) && !cursor.isNull(11)) { + long ts = cursor.getLong(10); + // boolean allDay = cursor.getInt(11) != 0; + task.setCompletedAt(new Completed(allDay ? new Date(ts) : new DateTime(ts))); + }*/ + } + + } catch (RemoteException e) { + throw new LocalStorageException("Couldn't process locally stored task", e); + } + } + + @Override + protected ContentProviderOperation.Builder buildEntry(ContentProviderOperation.Builder builder, Resource resource, boolean update) { + final Task task = (Task)resource; + + if (!update) + builder = builder + .withValue(entryColumnParentID(), id) + .withValue(entryColumnRemoteName(), task.getName()); + + builder = builder + .withValue(entryColumnUID(), task.getUid()) + .withValue(entryColumnETag(), task.getETag()) + .withValue(TaskContract.Tasks.TITLE, task.getSummary()) + .withValue(TaskContract.Tasks.LOCATION, task.getLocation()) + .withValue(TaskContract.Tasks.DESCRIPTION, task.getDescription()) + .withValue(TaskContract.Tasks.URL, task.getUrl()); + + if (task.getClassification() != null) { + int classCode = TaskContract.Tasks.CLASSIFICATION_PRIVATE; + if (task.getClassification() == Clazz.PUBLIC) + classCode = TaskContract.Tasks.CLASSIFICATION_PUBLIC; + else if (task.getClassification() == Clazz.CONFIDENTIAL) + classCode = TaskContract.Tasks.CLASSIFICATION_CONFIDENTIAL; + builder = builder.withValue(TaskContract.Tasks.CLASSIFICATION, classCode); + } + + int statusCode = TaskContract.Tasks.STATUS_DEFAULT; + if (task.getStatus() != null) { + if (task.getStatus() == Status.VTODO_NEEDS_ACTION) + statusCode = TaskContract.Tasks.STATUS_NEEDS_ACTION; + else if (task.getStatus() == Status.VTODO_IN_PROCESS) + statusCode = TaskContract.Tasks.STATUS_IN_PROCESS; + else if (task.getStatus() == Status.VTODO_COMPLETED) + statusCode = TaskContract.Tasks.STATUS_COMPLETED; + else if (task.getStatus() == Status.VTODO_CANCELLED) + statusCode = TaskContract.Tasks.STATUS_CANCELLED; + } + builder = builder + .withValue(TaskContract.Tasks.STATUS, statusCode) + .withValue(TaskContract.Tasks.PERCENT_COMPLETE, task.getPercentComplete()); + + /*if (task.getCreatedAt() != null) + builder = builder.withValue(TaskContract.Tasks.CREATED, task.getCreatedAt().getDate().getTime());/* + + if (task.getDtStart() != null) { + Date start = task.getDtStart().getDate(); + boolean allDay; + if (start instanceof DateTime) + allDay = false; + else { + task.getDtStart().setUtc(true); + allDay = true; + } + long ts = start.getTime(); + builder = builder.withValue(TaskContract.Tasks.DTSTART, ts); + builder = builder.withValue(TaskContract.Tasks.IS_ALLDAY, allDay ? 1 : 0); + } + + /*if (task.getCompletedAt() != null) { + Date completed = task.getCompletedAt().getDate(); + boolean allDay; + if (completed instanceof DateTime) + allDay = false; + else { + task.getCompletedAt().setUtc(true); + allDay = true; + } + long ts = completed.getTime(); + builder = builder.withValue(TaskContract.Tasks.COMPLETED, ts); + builder = builder.withValue(TaskContract.Tasks.COMPLETED_IS_ALLDAY, allDay ? 1 : 0); + }*/ + + return builder; + } + + @Override + protected void addDataRows(Resource resource, long localID, int backrefIdx) { + } + + @Override + protected void removeDataRows(Resource resource) { + } + + + // helpers + + @Override + protected Uri syncAdapterURI(Uri baseURI) { + return baseURI.buildUpon() + .appendQueryParameter(entryColumnAccountType(), /*account.type*/"davdroid.new") + .appendQueryParameter(entryColumnAccountName(), account.name) + .appendQueryParameter(TaskContract.CALLER_IS_SYNCADAPTER, "true") + .build(); + } + + protected static Uri taskListsURI(Account account) { + return TaskContract.TaskLists.CONTENT_URI.buildUpon() + .appendQueryParameter(TaskContract.TaskLists.ACCOUNT_TYPE, /*account.type*/"davdroid.new") + .appendQueryParameter(TaskContract.TaskLists.ACCOUNT_NAME, account.name) + .appendQueryParameter(TaskContract.CALLER_IS_SYNCADAPTER, "true") + .build(); + } +} diff --git a/app/src/main/java/at/bitfire/davdroid/resource/Note.java b/app/src/main/java/at/bitfire/davdroid/resource/Note.java new file mode 100644 index 00000000..2843bfd0 --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/resource/Note.java @@ -0,0 +1,124 @@ +package at.bitfire.davdroid.resource; + +import android.util.Log; + +import net.fortuna.ical4j.data.CalendarBuilder; +import net.fortuna.ical4j.data.CalendarOutputter; +import net.fortuna.ical4j.data.ParserException; +import net.fortuna.ical4j.model.Component; +import net.fortuna.ical4j.model.ComponentList; +import net.fortuna.ical4j.model.PropertyList; +import net.fortuna.ical4j.model.ValidationException; +import net.fortuna.ical4j.model.component.VJournal; +import net.fortuna.ical4j.model.component.VToDo; +import net.fortuna.ical4j.model.property.Created; +import net.fortuna.ical4j.model.property.Description; +import net.fortuna.ical4j.model.property.DtStamp; +import net.fortuna.ical4j.model.property.Location; +import net.fortuna.ical4j.model.property.ProdId; +import net.fortuna.ical4j.model.property.Summary; +import net.fortuna.ical4j.model.property.Uid; +import net.fortuna.ical4j.model.property.Version; +import net.fortuna.ical4j.util.SimpleHostInfo; +import net.fortuna.ical4j.util.UidGenerator; + +import org.apache.commons.lang.StringUtils; + +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.io.InputStream; + +import at.bitfire.davdroid.Constants; +import at.bitfire.davdroid.syncadapter.DavSyncAdapter; +import lombok.Getter; +import lombok.Setter; + +public class Note extends Resource { + private final static String TAG = "davdroid.Note"; + + @Getter @Setter Created created; + @Getter @Setter String summary, description; + + + public Note(String name, String ETag) { + super(name, ETag); + } + + public Note(long localId, String name, String ETag) + { + super(localId, name, ETag); + } + + @Override + public void initialize() { + UidGenerator generator = new UidGenerator(new SimpleHostInfo(DavSyncAdapter.getAndroidID()), String.valueOf(android.os.Process.myPid())); + uid = generator.generateUid().getValue(); + name = uid + ".ics"; + } + + + @Override + public void parseEntity(InputStream entity, AssetDownloader downloader) throws IOException, InvalidResourceException { + net.fortuna.ical4j.model.Calendar ical; + try { + CalendarBuilder builder = new CalendarBuilder(); + ical = builder.build(entity); + + if (ical == null) + throw new InvalidResourceException("No iCalendar found"); + } catch (ParserException e) { + throw new InvalidResourceException(e); + } + + ComponentList notes = ical.getComponents(Component.VJOURNAL); + if (notes == null || notes.isEmpty()) + throw new InvalidResourceException("No VJOURNAL found"); + VJournal note = (VJournal)notes.get(0); + + if (note.getUid() != null) + uid = note.getUid().getValue(); + + if (note.getCreated() != null) + created = note.getCreated(); + + if (note.getSummary() != null) + summary = note.getSummary().getValue(); + if (note.getDescription() != null) + description = note.getDescription().getValue(); + } + + + @Override + public String getMimeType() { + return "text/calendar"; + } + + @Override + public ByteArrayOutputStream toEntity() throws IOException { + final net.fortuna.ical4j.model.Calendar ical = new net.fortuna.ical4j.model.Calendar(); + ical.getProperties().add(Version.VERSION_2_0); + ical.getProperties().add(Constants.ICAL_PRODID); + + final VJournal note = new VJournal(); + ical.getComponents().add(note); + final PropertyList props = note.getProperties(); + + if (uid != null) + props.add(new Uid(uid)); + + if (summary != null) + props.add(new Summary(summary)); + if (description != null) + props.add(new Description(description)); + + CalendarOutputter output = new CalendarOutputter(false); + ByteArrayOutputStream os = new ByteArrayOutputStream(); + try { + output.output(ical, os); + } catch (ValidationException e) { + Log.e(TAG, "Generated invalid iCalendar"); + } + return os; + } + +} diff --git a/app/src/main/java/at/bitfire/davdroid/resource/ServerInfo.java b/app/src/main/java/at/bitfire/davdroid/resource/ServerInfo.java index d9b5d2ec..207c9515 100644 --- a/app/src/main/java/at/bitfire/davdroid/resource/ServerInfo.java +++ b/app/src/main/java/at/bitfire/davdroid/resource/ServerInfo.java @@ -69,7 +69,9 @@ public class ServerInfo implements Serializable { VCardVersion vCardVersion; String timezone; - + boolean supportingEvents = false, + supportingNotes = false, + supportingTasks = false; public String getTitle() { if (title == null) { diff --git a/app/src/main/java/at/bitfire/davdroid/resource/Task.java b/app/src/main/java/at/bitfire/davdroid/resource/Task.java new file mode 100644 index 00000000..47aa6e3b --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/resource/Task.java @@ -0,0 +1,175 @@ +package at.bitfire.davdroid.resource; + +import android.util.Log; + +import net.fortuna.ical4j.data.CalendarBuilder; +import net.fortuna.ical4j.data.CalendarOutputter; +import net.fortuna.ical4j.data.ParserException; +import net.fortuna.ical4j.model.Component; +import net.fortuna.ical4j.model.ComponentList; +import net.fortuna.ical4j.model.PropertyList; +import net.fortuna.ical4j.model.ValidationException; +import net.fortuna.ical4j.model.component.VToDo; +import net.fortuna.ical4j.model.property.Clazz; +import net.fortuna.ical4j.model.property.Completed; +import net.fortuna.ical4j.model.property.Created; +import net.fortuna.ical4j.model.property.Description; +import net.fortuna.ical4j.model.property.DtStart; +import net.fortuna.ical4j.model.property.Location; +import net.fortuna.ical4j.model.property.PercentComplete; +import net.fortuna.ical4j.model.property.Priority; +import net.fortuna.ical4j.model.property.Status; +import net.fortuna.ical4j.model.property.Summary; +import net.fortuna.ical4j.model.property.Uid; +import net.fortuna.ical4j.model.property.Url; +import net.fortuna.ical4j.model.property.Version; +import net.fortuna.ical4j.util.SimpleHostInfo; +import net.fortuna.ical4j.util.UidGenerator; + +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.net.URI; +import java.net.URISyntaxException; +import java.net.URL; + +import at.bitfire.davdroid.Constants; +import at.bitfire.davdroid.syncadapter.DavSyncAdapter; +import lombok.Getter; +import lombok.Setter; + +public class Task extends Resource { + private final static String TAG = "davdroid.Task"; + + @Getter @Setter String summary, location, description, url; + @Getter @Setter int priority; + @Getter @Setter Clazz classification; + @Getter @Setter Status status; + + @Getter @Setter Created createdAt; + @Getter @Setter DtStart dtStart; + @Getter @Setter Completed completedAt; + @Getter @Setter Integer percentComplete; + + + public Task(String name, String ETag) { + super(name, ETag); + } + + public Task(long localId, String name, String ETag) + { + super(localId, name, ETag); + } + + @Override + public void initialize() { + UidGenerator generator = new UidGenerator(new SimpleHostInfo(DavSyncAdapter.getAndroidID()), String.valueOf(android.os.Process.myPid())); + uid = generator.generateUid().getValue(); + name = uid + ".ics"; + } + + + @Override + public void parseEntity(InputStream entity, AssetDownloader downloader) throws IOException, InvalidResourceException { + net.fortuna.ical4j.model.Calendar ical; + try { + CalendarBuilder builder = new CalendarBuilder(); + ical = builder.build(entity); + + if (ical == null) + throw new InvalidResourceException("No iCalendar found"); + } catch (ParserException e) { + throw new InvalidResourceException(e); + } + + ComponentList notes = ical.getComponents(Component.VTODO); + if (notes == null || notes.isEmpty()) + throw new InvalidResourceException("No VTODO found"); + VToDo todo = (VToDo)notes.get(0); + + if (todo.getUid() != null) + uid = todo.getUid().getValue(); + + if (todo.getSummary() != null) + summary = todo.getSummary().getValue(); + if (todo.getLocation() != null) + location = todo.getLocation().getValue(); + if (todo.getDescription() != null) + description = todo.getDescription().getValue(); + if (todo.getUrl() != null) + url = todo.getUrl().getValue(); + + priority = (todo.getPriority() != null) ? todo.getPriority().getLevel() : 0; + if (todo.getClassification() != null) + classification = todo.getClassification(); + if (todo.getStatus() != null) + status = todo.getStatus(); + + if (todo.getCreated() != null) + createdAt = todo.getCreated(); + if (todo.getStartDate() != null) + dtStart = todo.getStartDate(); + if (todo.getDateCompleted() != null) + completedAt = todo.getDateCompleted(); + if (todo.getPercentComplete() != null) + percentComplete = todo.getPercentComplete().getPercentage(); + } + + + @Override + public String getMimeType() { + return "text/calendar"; + } + + @Override + public ByteArrayOutputStream toEntity() throws IOException { + final net.fortuna.ical4j.model.Calendar ical = new net.fortuna.ical4j.model.Calendar(); + ical.getProperties().add(Version.VERSION_2_0); + ical.getProperties().add(Constants.ICAL_PRODID); + + final VToDo todo = new VToDo(); + ical.getComponents().add(todo); + final PropertyList props = todo.getProperties(); + + if (uid != null) + props.add(new Uid(uid)); + + if (summary != null) + props.add(new Summary(summary)); + if (location != null) + props.add(new Location(location)); + if (description != null) + props.add(new Description(description)); + if (url != null) + try { + props.add(new Url(new URI(url))); + } catch (URISyntaxException e) { + Log.e(TAG, "Ignoring invalid task URL: " + url, e); + } + if (priority != 0) + props.add(new Priority(priority)); + if (classification != null) + props.add(classification); + if (status != null) + props.add(status); + + if (createdAt != null) + props.add(createdAt); + if (dtStart != null) + props.add(dtStart); + if (completedAt != null) + props.add(completedAt); + if (percentComplete != null) + props.add(new PercentComplete(percentComplete)); + + CalendarOutputter output = new CalendarOutputter(false); + ByteArrayOutputStream os = new ByteArrayOutputStream(); + try { + output.output(ical, os); + } catch (ValidationException e) { + Log.e(TAG, "Generated invalid iCalendar"); + } + return os; + } + +} diff --git a/app/src/main/java/at/bitfire/davdroid/syncadapter/CalendarsSyncAdapterService.java b/app/src/main/java/at/bitfire/davdroid/syncadapter/CalendarsSyncAdapterService.java index 11e37a6d..72f86b11 100644 --- a/app/src/main/java/at/bitfire/davdroid/syncadapter/CalendarsSyncAdapterService.java +++ b/app/src/main/java/at/bitfire/davdroid/syncadapter/CalendarsSyncAdapterService.java @@ -28,7 +28,6 @@ import at.bitfire.davdroid.resource.RemoteCollection; public class CalendarsSyncAdapterService extends Service { private static SyncAdapter syncAdapter; - @Override public void onCreate() { if (syncAdapter == null) @@ -48,9 +47,8 @@ public class CalendarsSyncAdapterService extends Service { private static class SyncAdapter extends DavSyncAdapter { - private final static String TAG = "davdroid.CalDAVSync"; + private final static String TAG = "davdroid.CalendarsSync"; - private SyncAdapter(Context context) { super(context); } diff --git a/app/src/main/java/at/bitfire/davdroid/syncadapter/ContactsSyncAdapterService.java b/app/src/main/java/at/bitfire/davdroid/syncadapter/ContactsSyncAdapterService.java index 21d08230..aa00eb37 100644 --- a/app/src/main/java/at/bitfire/davdroid/syncadapter/ContactsSyncAdapterService.java +++ b/app/src/main/java/at/bitfire/davdroid/syncadapter/ContactsSyncAdapterService.java @@ -27,7 +27,6 @@ import at.bitfire.davdroid.resource.RemoteCollection; public class ContactsSyncAdapterService extends Service { private static ContactsSyncAdapter syncAdapter; - @Override public void onCreate() { if (syncAdapter == null) @@ -47,9 +46,8 @@ public class ContactsSyncAdapterService extends Service { private static class ContactsSyncAdapter extends DavSyncAdapter { - private final static String TAG = "davdroid.CardDAVSync"; + private final static String TAG = "davdroid.ContactsSync"; - private ContactsSyncAdapter(Context context) { super(context); } diff --git a/app/src/main/java/at/bitfire/davdroid/syncadapter/NotesSyncAdapterService.java b/app/src/main/java/at/bitfire/davdroid/syncadapter/NotesSyncAdapterService.java new file mode 100644 index 00000000..a26510bb --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/syncadapter/NotesSyncAdapterService.java @@ -0,0 +1,73 @@ +package at.bitfire.davdroid.syncadapter; + +import android.accounts.Account; +import android.app.Service; +import android.content.ContentProviderClient; +import android.content.Context; +import android.content.Intent; +import android.os.IBinder; +import android.os.RemoteException; +import android.util.Log; + +import java.net.URISyntaxException; +import java.util.HashMap; +import java.util.Map; + +import at.bitfire.davdroid.resource.CalDavNotebook; +import at.bitfire.davdroid.resource.LocalCollection; +import at.bitfire.davdroid.resource.LocalNotebook; +import at.bitfire.davdroid.resource.RemoteCollection; + +public class NotesSyncAdapterService extends Service { + private static NotesSyncAdapter syncAdapter; + + @Override + public void onCreate() { + if (syncAdapter == null) + syncAdapter = new NotesSyncAdapter(getApplicationContext()); + } + + @Override + public void onDestroy() { + syncAdapter.close(); + syncAdapter = null; + } + + @Override + public IBinder onBind(Intent intent) { + return syncAdapter.getSyncAdapterBinder(); + } + + + private static class NotesSyncAdapter extends DavSyncAdapter { + private final static String TAG = "davdroid.NotesSync"; + + private NotesSyncAdapter(Context context) { + super(context); + } + + @Override + protected Map, RemoteCollection> getSyncPairs(Account account, ContentProviderClient provider) { + AccountSettings settings = new AccountSettings(getContext(), account); + String userName = settings.getUserName(), + password = settings.getPassword(); + boolean preemptive = settings.getPreemptiveAuth(); + + try { + Map, RemoteCollection> map = new HashMap, RemoteCollection>(); + + for (LocalNotebook noteList : LocalNotebook.findAll(account, provider)) { + RemoteCollection dav = new CalDavNotebook(httpClient, noteList.getUrl(), userName, password, preemptive); + map.put(noteList, dav); + } + return map; + } catch (RemoteException ex) { + Log.e(TAG, "Couldn't find local notebooks", ex); + } catch (URISyntaxException ex) { + Log.e(TAG, "Couldn't build calendar URI", ex); + } + + return null; + } + } +} diff --git a/app/src/main/java/at/bitfire/davdroid/syncadapter/TasksSyncAdapterService.java b/app/src/main/java/at/bitfire/davdroid/syncadapter/TasksSyncAdapterService.java new file mode 100644 index 00000000..f70625b9 --- /dev/null +++ b/app/src/main/java/at/bitfire/davdroid/syncadapter/TasksSyncAdapterService.java @@ -0,0 +1,82 @@ +/* + * Copyright (c) 2013 – 2015 Ricki Hirner (bitfire web engineering). + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the GNU Public License v3.0 + * which accompanies this distribution, and is available at + * http://www.gnu.org/licenses/gpl.html + */ +package at.bitfire.davdroid.syncadapter; + +import android.accounts.Account; +import android.app.Service; +import android.content.ContentProviderClient; +import android.content.Context; +import android.content.Intent; +import android.os.IBinder; +import android.os.RemoteException; +import android.util.Log; + +import java.net.URISyntaxException; +import java.util.HashMap; +import java.util.Map; + +import at.bitfire.davdroid.resource.CalDavCalendar; +import at.bitfire.davdroid.resource.CalDavTaskList; +import at.bitfire.davdroid.resource.LocalCalendar; +import at.bitfire.davdroid.resource.LocalCollection; +import at.bitfire.davdroid.resource.LocalTaskList; +import at.bitfire.davdroid.resource.RemoteCollection; + +public class TasksSyncAdapterService extends Service { + private static SyncAdapter syncAdapter; + + @Override + public void onCreate() { + if (syncAdapter == null) + syncAdapter = new SyncAdapter(getApplicationContext()); + } + + @Override + public void onDestroy() { + syncAdapter.close(); + syncAdapter = null; + } + + @Override + public IBinder onBind(Intent intent) { + return syncAdapter.getSyncAdapterBinder(); + } + + + private static class SyncAdapter extends DavSyncAdapter { + private final static String TAG = "davdroid.TasksSync"; + + private SyncAdapter(Context context) { + super(context); + } + + @Override + protected Map, RemoteCollection> getSyncPairs(Account account, ContentProviderClient provider) { + AccountSettings settings = new AccountSettings(getContext(), account); + String userName = settings.getUserName(), + password = settings.getPassword(); + boolean preemptive = settings.getPreemptiveAuth(); + + try { + Map, RemoteCollection> map = new HashMap, RemoteCollection>(); + + for (LocalTaskList calendar : LocalTaskList.findAll(account, provider)) { + RemoteCollection dav = new CalDavTaskList(httpClient, calendar.getUrl(), userName, password, preemptive); + map.put(calendar, dav); + } + return map; + } catch (RemoteException ex) { + Log.e(TAG, "Couldn't find local task lists", ex); + } catch (URISyntaxException ex) { + Log.e(TAG, "Couldn't build task list URI", ex); + } + + return null; + } + } +} diff --git a/app/src/main/java/at/bitfire/davdroid/ui/setup/AccountDetailsFragment.java b/app/src/main/java/at/bitfire/davdroid/ui/setup/AccountDetailsFragment.java index e03cdad7..9aeff395 100644 --- a/app/src/main/java/at/bitfire/davdroid/ui/setup/AccountDetailsFragment.java +++ b/app/src/main/java/at/bitfire/davdroid/ui/setup/AccountDetailsFragment.java @@ -26,12 +26,18 @@ import android.widget.EditText; import android.widget.TextView; import android.widget.Toast; +import org.dmfs.provider.tasks.TaskContract; + import at.bitfire.davdroid.Constants; import at.bitfire.davdroid.R; import at.bitfire.davdroid.resource.LocalCalendar; +import at.bitfire.davdroid.resource.LocalNotebook; import at.bitfire.davdroid.resource.LocalStorageException; +import at.bitfire.davdroid.resource.LocalTaskList; import at.bitfire.davdroid.resource.ServerInfo; +import at.bitfire.davdroid.resource.Task; import at.bitfire.davdroid.syncadapter.AccountSettings; +import at.bitfire.notebooks.provider.NoteContract; public class AccountDetailsFragment extends Fragment implements TextWatcher { public static final String KEY_SERVER_INFO = "server_info"; @@ -102,22 +108,54 @@ public class AccountDetailsFragment extends Fragment implements TextWatcher { ContentResolver.setIsSyncable(account, ContactsContract.AUTHORITY, 0); if (accountManager.addAccountExplicitly(account, serverInfo.getPassword(), userData)) { - // account created, now create calendars + // account created, now create calendars ... boolean syncCalendars = false; for (ServerInfo.ResourceInfo calendar : serverInfo.getCalendars()) - if (calendar.isEnabled()) + if (calendar.isEnabled() && calendar.isSupportingEvents()) try { LocalCalendar.create(account, getActivity().getContentResolver(), calendar); syncCalendars = true; } catch (LocalStorageException e) { - Toast.makeText(getActivity(), "Couldn't create calendar(s): " + e.getMessage(), Toast.LENGTH_LONG).show(); + Toast.makeText(getActivity(), "Couldn't create calendar: " + e.getMessage(), Toast.LENGTH_LONG).show(); } if (syncCalendars) { ContentResolver.setIsSyncable(account, CalendarContract.AUTHORITY, 1); ContentResolver.setSyncAutomatically(account, CalendarContract.AUTHORITY, true); } else ContentResolver.setIsSyncable(account, CalendarContract.AUTHORITY, 0); - + + // ... and notes + boolean syncNotes = false; + for (ServerInfo.ResourceInfo calendar : serverInfo.getCalendars()) + if (calendar.isEnabled() && calendar.isSupportingNotes()) + try { + LocalNotebook.create(account, getActivity().getContentResolver(), calendar); + syncNotes = true; + } catch (LocalStorageException e) { + Toast.makeText(getActivity(), "Couldn't create notebook: " + e.getMessage(), Toast.LENGTH_LONG).show(); + } + if (syncNotes) { + ContentResolver.setIsSyncable(account, NoteContract.AUTHORITY, 1); + ContentResolver.setSyncAutomatically(account, NoteContract.AUTHORITY, true); + } else + ContentResolver.setIsSyncable(account, NoteContract.AUTHORITY, 0); + + // ... and tasks + boolean syncTasks = false; + for (ServerInfo.ResourceInfo calendar : serverInfo.getCalendars()) + if (calendar.isEnabled() && calendar.isSupportingTasks()) + try { + LocalTaskList.create(account, getActivity().getContentResolver(), calendar); + syncTasks = true; + } catch (LocalStorageException e) { + Toast.makeText(getActivity(), "Couldn't create task list: " + e.getMessage(), Toast.LENGTH_LONG).show(); + } + if (syncTasks) { + ContentResolver.setIsSyncable(account, TaskContract.AUTHORITY, 1); + ContentResolver.setSyncAutomatically(account, TaskContract.AUTHORITY, true); + } else + ContentResolver.setIsSyncable(account, TaskContract.AUTHORITY, 0); + getActivity().finish(); } else Toast.makeText(getActivity(), "Couldn't create account (account with this name already existing?)", Toast.LENGTH_LONG).show(); diff --git a/app/src/main/java/at/bitfire/davdroid/webdav/DavCompFilter.java b/app/src/main/java/at/bitfire/davdroid/webdav/DavCompFilter.java index ea833dc5..ecd64e18 100644 --- a/app/src/main/java/at/bitfire/davdroid/webdav/DavCompFilter.java +++ b/app/src/main/java/at/bitfire/davdroid/webdav/DavCompFilter.java @@ -19,5 +19,4 @@ public class DavCompFilter { @Element(required=false,name="comp-filter") @Getter @Setter DavCompFilter compFilter; - } diff --git a/app/src/main/java/at/bitfire/davdroid/webdav/DavFilter.java b/app/src/main/java/at/bitfire/davdroid/webdav/DavFilter.java index e33cf475..dc5b9dfc 100644 --- a/app/src/main/java/at/bitfire/davdroid/webdav/DavFilter.java +++ b/app/src/main/java/at/bitfire/davdroid/webdav/DavFilter.java @@ -8,6 +8,6 @@ import lombok.Setter; @Namespace(prefix="C",reference="urn:ietf:params:xml:ns:caldav") public class DavFilter { - @Element(required=false) + @Element(required=false,name="comp-filter") @Getter @Setter DavCompFilter compFilter; } diff --git a/app/src/main/java/at/bitfire/davdroid/webdav/WebDavResource.java b/app/src/main/java/at/bitfire/davdroid/webdav/WebDavResource.java index c1fb361f..bbe6d5ec 100644 --- a/app/src/main/java/at/bitfire/davdroid/webdav/WebDavResource.java +++ b/app/src/main/java/at/bitfire/davdroid/webdav/WebDavResource.java @@ -453,7 +453,7 @@ public class WebDavResource { } if (multiStatus.response == null) // empty response - throw new DavNoContentException(); + return; // member list will be built from response List members = new LinkedList(); diff --git a/app/src/main/java/at/bitfire/notebooks/provider/NoteContract.java b/app/src/main/java/at/bitfire/notebooks/provider/NoteContract.java new file mode 120000 index 00000000..896ac4dc --- /dev/null +++ b/app/src/main/java/at/bitfire/notebooks/provider/NoteContract.java @@ -0,0 +1 @@ +../../../../../../../../../notebooks/app/src/main/java/at/bitfire/notebooks/provider/NoteContract.java \ No newline at end of file diff --git a/app/src/main/java/org/dmfs/provider/tasks/TaskContract.java b/app/src/main/java/org/dmfs/provider/tasks/TaskContract.java new file mode 100644 index 00000000..cd116557 --- /dev/null +++ b/app/src/main/java/org/dmfs/provider/tasks/TaskContract.java @@ -0,0 +1,1282 @@ +/* + * Copyright (C) 2012 Marten Gajda + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + */ + +package org.dmfs.provider.tasks; + +import android.content.ContentResolver; +import android.net.Uri; + +/** + * Task contract. This class defines the interface to the task provider. + *

+ * TODO: Add missing javadoc. + *

+ *

+ * TODO: Specify extended properties + *

+ *

+ * TODO: Add CONTENT_URI for the attachment store. + *

+ *

+ * TODO: Also, we could use some refactoring... + *

+ * + * @author Marten Gajda + * @author Tobias Reinsch + */ +public final class TaskContract { + + /** + * Task provider authority. + */ + // TODO how to do this better? + public static final String AUTHORITY = "de.azapps.mirakel.provider"; + public static final String AUTHORITY_DMFS = "org.dmfs.tasks"; + + /** + * Base content URI. + */ + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY); + + /** + * URI parameter to signal that the caller is a sync adapter. + */ + public static final String CALLER_IS_SYNCADAPTER = "caller_is_syncadapter"; + + /** + * URI parameter to signal the request of the extended properties of a task. + */ + public static final String LOAD_PROPERTIES = "load_properties"; + + /** + * URI parameter to submit the account name of the account we operate on. + */ + public static final String ACCOUNT_NAME = "account_name"; + + /** + * URI parameter to submit the account type of the account we operate on. + */ + public static final String ACCOUNT_TYPE = "account_type"; + + /** + * Account type for local, unsynced task lists. + */ + public static final String LOCAL_ACCOUNT = "LOCAL"; + + + /** + * Private constructor to prevent instantiation. + */ + private TaskContract() { + } + + /** + * A set of columns for synchronization purposes. These columns exist in {@link Tasks} and in {@link TaskLists} but have different meanings. Only sync + * adapters are allowed to change these values. + * + * @author Marten Gajda + */ + public interface CommonSyncColumns { + + /** + * A unique Sync ID as set by the sync adapter. + *

+ * Value: String + *

+ */ + public static final String _SYNC_ID = "_sync_id"; + + /** + * Sync version as set by the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC_VERSION = "sync_version"; + + /** + * Indicates that a task or a task list has been changed. + *

+ * Value: Integer + *

+ */ + public static final String _DIRTY = "_dirty"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC1 = "sync1"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC2 = "sync2"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC3 = "sync3"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC4 = "sync4"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC5 = "sync5"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC6 = "sync6"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC7 = "sync7"; + + /** + * A general purpose column for the sync adapter. + *

+ * Value: String + *

+ */ + public static final String SYNC8 = "sync8"; + + } + + /** + * Additional sync columns for task lists. + * + * @author Marten Gajda + */ + public interface TaskListSyncColumns { + + /** + * The name of the account this list belongs to. This field is write-once. + *

+ * Value: String + *

+ */ + public static final String ACCOUNT_NAME = "account_name"; + + /** + * The type of the account this list belongs to. This field is write-once. + *

+ * Value: String + *

+ */ + public static final String ACCOUNT_TYPE = "account_type"; + } + + /** + * Additional sync columns for tasks. + * + * @author Marten Gajda + */ + public interface TaskSyncColumns { + /** + * The UID of a task. This is field can be changed by a sync adapter only. + *

+ * Value: String + *

+ */ + public static final String _UID = "_uid"; + + /** + * Deleted flag of a task. This is set to 1 by the content provider when a task app deletes a task. The sync adapter has to remove the task + * again to finish the removal. This value is read-only. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String _DELETED = "_deleted"; + } + + /** + * Data columns of task lists. + * + * @author Marten Gajda + */ + public interface TaskListColumns { + + /** + * List ID. + *

+ * Value: Long + *

+ *

+ * read-only + *

+ */ + public static final String _ID = "_id"; + + /** + * The name of the task list. + *

+ * Value: String + *

+ */ + public static final String LIST_NAME = "list_name"; + + /** + * The color of this list as integer (0xaarrggbb). Only the sync adapter can change this. + *

+ * Value: Integer + *

+ */ + public static final String LIST_COLOR = "list_color"; + + /** + * The access level a user has on this list. This value is not used yet, sync adapters should set it to 0. + *

+ * Value: Integer + *

+ */ + public static final String ACCESS_LEVEL = "list_access_level"; + + /** + * Indicates that a task list is set to be visible. + *

+ * Value: Integer (0 or 1) + *

+ */ + public static final String VISIBLE = "visible"; + + /** + * Indicates that a task list is set to be synced. + *

+ * Value: Integer (0 or 1) + *

+ */ + public static final String SYNC_ENABLED = "sync_enabled"; + + /** + * The email address of the list owner. + *

+ * Value: String + *

+ */ + public static final String OWNER = "list_owner"; + + } + + /** + * The task list table holds one entry for each task list. + * + * @author Marten Gajda + */ + public static final class TaskLists implements TaskListColumns, TaskListSyncColumns, + CommonSyncColumns { + public static final String CONTENT_URI_PATH = "tasklists"; + + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + CONTENT_URI_PATH); + + public static final String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/" + + AUTHORITY + "." + CONTENT_URI_PATH; + + public static final String CONTENT_TYPE = ContentResolver.CURSOR_DIR_BASE_TYPE + "/" + AUTHORITY + + "." + CONTENT_URI_PATH; + + /** + * The default sort order. + */ + public static final String DEFAULT_SORT_ORDER = ACCOUNT_NAME + ", " + LIST_NAME; + + /** + * An array of columns only a sync adapter is allowed to change. + */ + public static final String[] SYNC_ADAPTER_COLUMNS = new String[] { ACCESS_LEVEL, _DIRTY, OWNER, SYNC1, SYNC2, SYNC3, SYNC4, SYNC5, SYNC6, SYNC7, SYNC8, + _SYNC_ID, SYNC_VERSION, + }; + + } + + /** + * Task data columns. Defines all the values a task can have at most once. + * + * @author Marten Gajda + */ + public interface TaskColumns { + + /** + * The row id of a task. This value is read-only + *

+ * Value: Integer + *

+ */ + public static final String _ID = "_id"; + + /** + * The id of the list this task belongs to. This value is write-once and must not be null. + *

+ * Value: Integer + *

+ */ + public static final String LIST_ID = "list_id"; + + /** + * The title of the task. + *

+ * Value: String + *

+ */ + public static final String TITLE = "title"; + + /** + * The location of the task. + *

+ * Value: String + *

+ */ + public static final String LOCATION = "location"; + + /** + * A geographic location related to the task. The should be a string in the format "longitude,latitude". + *

+ * Value: String + *

+ */ + public static final String GEO = "geo"; + + /** + * The description of a task. + *

+ * Value: String + *

+ */ + public static final String DESCRIPTION = "description"; + + /** + * An URL for this task. Must be a valid URL if not null- + *

+ * Value: String + *

+ */ + public static final String URL = "url"; + + /** + * The email address of the organizer if any, {@code null} otherwise. + *

+ * Value: String + *

+ */ + public static final String ORGANIZER = "organizer"; + + /** + * The priority of a task. This is an Integer between zero and 9. Zero means there is no priority set. 1 is the highest priority and 9 the lowest. + *

+ * Value: Integer + *

+ */ + public static final String PRIORITY = "priority"; + + /** + * The default value of {@link #PRIORITY}. + */ + public static final int PRIORITY_DEFAULT = 0; + + /** + * The classification of a task. This value must be either null or one of {@link #CLASSIFICATION_PUBLIC}, {@link #CLASSIFICATION_PRIVATE}, + * {@link #CLASSIFICATION_CONFIDENTIAL}. + *

+ * Value: Integer + *

+ */ + public static final String CLASSIFICATION = "class"; + + /** + * Classification value for public tasks. + */ + public static final int CLASSIFICATION_PUBLIC = 0; + + /** + * Classification value for private tasks. + */ + public static final int CLASSIFICATION_PRIVATE = 1; + + /** + * Classification value for confidential tasks. + */ + public static final int CLASSIFICATION_CONFIDENTIAL = 2; + + /** + * Default value of {@link #CLASSIFICATION}. + */ + public static final Integer CLASSIFICATION_DEFAULT = null; + + /** + * Date of completion of this task in milliseconds since the epoch or {@code null} if this task has not been completed yet. + *

+ * Value: Long + *

+ */ + public static final String COMPLETED = "completed"; + + /** + * Indicates that the date of completion is an all-day date. + *

+ * Value: Integer + *

+ */ + public static final String COMPLETED_IS_ALLDAY = "completed_is_allday"; + + /** + * A number between 0 and 100 that indicates the progress of the task or null. + *

+ * Value: Integer (0-100) + *

+ */ + public static final String PERCENT_COMPLETE = "percent_complete"; + + /** + * The status of this task. One of {@link #STATUS_NEEDS_ACTION},{@link #STATUS_IN_PROCESS}, {@link #STATUS_COMPLETED}, {@link #STATUS_CANCELLED}. + *

+ * Value: Integer + *

+ */ + public static final String STATUS = "status"; + + /** + * A specific status indicating that nothing has been done yet. + */ + public static final int STATUS_NEEDS_ACTION = 0; + + /** + * A specific status indicating that some work has been done. + */ + public static final int STATUS_IN_PROCESS = 1; + + /** + * A specific status indicating that the task is completed. + */ + public static final int STATUS_COMPLETED = 2; + + /** + * A specific status indicating that the task has been cancelled. + */ + public static final int STATUS_CANCELLED = 3; + + /** + * The default status is "needs action". + */ + public static final int STATUS_DEFAULT = STATUS_NEEDS_ACTION; + + /** + * A flag that indicates a task is new (i.e. not work has been done yet). This flag is read-only. Its value is 1 when + * {@link #STATUS} equals {@link #STATUS_NEEDS_ACTION} and 0 otherwise. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String IS_NEW = "is_new"; + + /** + * A flag that indicates a task is closed (no more work has to be done). This flag is read-only. Its value is 1 when + * {@link #STATUS} equals {@link #STATUS_COMPLETED} or {@link #STATUS_CANCELLED} and 0 otherwise. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String IS_CLOSED = "is_closed"; + + /** + * An individual color for this task in the format 0xaarrggbb or {@code null} to use {@link TaskListColumns#LIST_COLOR} instead. + *

+ * Value: Integer + *

+ */ + public static final String TASK_COLOR = "task_color"; + + /** + * When this task starts in milliseconds since the epoch. + *

+ * Value: Long + *

+ */ + public static final String DTSTART = "dtstart"; + + /** + * Boolean: flag that indicates that this is an all-day task. + */ + public static final String IS_ALLDAY = "is_allday"; + + /** + * When this task has been created in milliseconds since the epoch. + *

+ * Value: Long + *

+ */ + public static final String CREATED = "created"; + + /** + * When this task had been modified the last time in milliseconds since the epoch. + *

+ * Value: Long + *

+ */ + public static final String LAST_MODIFIED = "last_modified"; + + /** + * String: An Olson Id of the time zone of this task. If this value is null, it's automatically replaced by the local time zone. + */ + public static final String TZ = "tz"; + + /** + * When this task is due in milliseconds since the epoch. Only one of {@link #DUE} or {@link #DURATION} must be supplied (or none of both if the task + * has no due date). + *

+ * Value: Long + *

+ */ + public static final String DUE = "due"; + + /** + * The duration of this task. Only one of {@link #DUE} or {@link #DURATION} must be supplied (or none of both if the task has no due date). Setting a + * {@link #DURATION} is not allowed when {@link #DTSTART} is null. The Value must be a duration string as in RFC 5545 Section 3.3.6. + *

+ * Value: String + *

+ */ + public static final String DURATION = "duration"; + + /** + * A comma separated list of time Strings in RFC 5545 format (see RFC 5545 Section 3.3.4 + * and RFC 5545 Section 3.3.5) that contains dates of instances of e recurring task. + * All-day tasks must use the DATE format specified in section 3.3.4 of RFC 5545. + * + * This value must be {@code null} for exception instances. + *

+ * Value: String + *

+ */ + public static final String RDATE = "rdate"; + + /** + * A comma separated list of time Strings in RFC 5545 format (see RFC 5545 Section 3.3.4 + * and RFC 5545 Section 3.3.5) that contains dates of exceptions of a recurring task. + * All-day tasks must use the DATE format specified in section 3.3.4 of RFC 5545. + * + * This value must be {@code null} for exception instances. + *

+ * Value: String + *

+ */ + public static final String EXDATE = "exdate"; + + /** + * A recurrence rule as specified in RFC 5545 Section 3.3.10. + * + * This value must be {@code null} for exception instances. + *

+ * Value: String + *

+ */ + public static final String RRULE = "rrule"; + + /** + * The _sync_id of the original event if this is an exception, null otherwise. Only one of {@link #ORIGINAL_INSTANCE_SYNC_ID} or + * {@link #ORIGINAL_INSTANCE_ID} must be set if this task is an exception. The other one will be updated by the content provider. + *

+ * Value: String + *

+ */ + public static final String ORIGINAL_INSTANCE_SYNC_ID = "original_instance_sync_id"; + + /** + * The row id of the original event if this is an exception, null otherwise. Only one of {@link #ORIGINAL_INSTANCE_SYNC_ID} or + * {@link #ORIGINAL_INSTANCE_ID} must be set if this task is an exception. The other one will be updated by the content provider. + *

+ * Value: Long + *

+ */ + public static final String ORIGINAL_INSTANCE_ID = "original_instance_id"; + + /** + * The time in milliseconds since the Epoch of the original instance that is overridden by this instance or null if this task is not an + * exception. + *

+ * Value: Long + *

+ */ + public static final String ORIGINAL_INSTANCE_TIME = "original_instance_time"; + + /** + * A flag indicating that the original instance was an all-day task. + *

+ * Value: Integer + *

+ */ + public static final String ORIGINAL_INSTANCE_ALLDAY = "original_instance_allday"; + + /** + * The row id of the parent task. null if the task has no parent task. + *

+ * Value: Long + *

+ */ + public static final String PARENT_ID = "parent_id"; + + /** + * The sorting of this task under it's parent task. + *

+ * Value: String + *

+ */ + public static final String SORTING = "sorting"; + + /** + * Indicates how many alarms a task has. 0 means the task has no alarms. + *

+ * Value: Integer + *

+ * + */ + public static final String HAS_ALARMS = "has_alarms"; + } + + /** + * The task table stores the data of all tasks. + * + * @author Marten Gajda + */ + public static final class Tasks implements TaskColumns, CommonSyncColumns, TaskSyncColumns { + /** + * The name of the account the task belongs to. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String ACCOUNT_NAME = TaskLists.ACCOUNT_NAME; + + /** + * The type of the account the task belongs to. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String ACCOUNT_TYPE = TaskLists.ACCOUNT_TYPE; + + /** + * The name of the list this task belongs to as integer (0xaarrggbb). This is auto-derived from the list the task belongs to. Do not write this value + * here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String LIST_NAME = TaskLists.LIST_NAME; + /** + * The color of the list this task belongs to as integer (0xaarrggbb). This is auto-derived from the list the task belongs to. Do not write this value + * here. To change the color of an individual task use {@code TASK_COLOR} instead. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String LIST_COLOR = TaskLists.LIST_COLOR; + + /** + * The owner of the list this task belongs. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String LIST_OWNER = TaskLists.OWNER; + + /** + * The access level of the list this task belongs. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String LIST_ACCESS_LEVEL = TaskLists.ACCESS_LEVEL; + + /** + * The visibility of the list this task belongs. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String VISIBLE = "visible"; + + public static final String CONTENT_URI_PATH = "tasks"; + + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + CONTENT_URI_PATH); + + public static final String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/" + + AUTHORITY + "." + CONTENT_URI_PATH; + + public static final String CONTENT_TYPE = ContentResolver.CURSOR_DIR_BASE_TYPE + "/" + AUTHORITY + + "." + CONTENT_URI_PATH; + + public static final String DEFAULT_SORT_ORDER = DUE; + + public static final String[] SYNC_ADAPTER_COLUMNS = new String[] { _DIRTY, SYNC1, SYNC2, SYNC3, SYNC4, SYNC5, SYNC6, SYNC7, SYNC8, _SYNC_ID, + SYNC_VERSION, + }; + } + + /** + * Columns of a task instance. + * + * @author Yannic Ahrens + * @author Marten Gajda + */ + public interface InstanceColumns { + /** + * _ID of task this instance belongs to. + *

+ * Value: Long + *

+ */ + public static final String TASK_ID = "task_id"; + + /** + * The start date of an instance in milliseconds since the epoch or null if the instance has no start date. At present this is read only. + *

+ * Value: Long + *

+ */ + public static final String INSTANCE_START = "instance_start"; + + /** + * The due date of an instance in milliseconds since the epoch or null if the instance has no due date. At present this is read only. + *

+ * Value: Long + *

+ */ + public static final String INSTANCE_DUE = "instance_due"; + + /** + * This column should be used in an order clause to sort instances by due date. It contains a slightly modified start date that takes allday tasks into + * account. + *

+ * Value: Long + *

+ *

+ * read-only + *

+ */ + public static final String INSTANCE_START_SORTING = "instance_start_sorting"; + + /** + * This column should be used in an order clause to sort instances by due date. It contains a slightly modified due date that takes allday tasks into + * account. + *

+ * Value: Long + *

+ *

+ * read-only + *

+ */ + public static final String INSTANCE_DUE_SORTING = "instance_due_sorting"; + + /** + * The duration of an instance in milliseconds or null if the instance has only one of start or due date or none of both. At present this + * is read only. + *

+ * Value: Long + *

+ */ + public static final String INSTANCE_DURATION = "instance_duration"; + + } + + /** + * Instances of a task. At present this table is read only. Currently it contains exactly one entry per task (and task exception), so it's merely a copy of + * {@link Tasks}. + *

+ * TODO: Insert all instances of recurring the tasks. + *

+ *

+ * TODO: In later releases it's planned to provide a convenient interface to add, change or delete task instances via this URI. + *

+ * + * @author Yannic Ahrens + */ + public static final class Instances implements TaskColumns, InstanceColumns { + + /** + * The name of the account the task belongs to. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String ACCOUNT_NAME = TaskLists.ACCOUNT_NAME; + + /** + * The type of the account the task belongs to. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String ACCOUNT_TYPE = TaskLists.ACCOUNT_TYPE; + + /** + * The name of the list this task belongs to as integer (0xaarrggbb). This is auto-derived from the list the task belongs to. Do not write this value + * here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String LIST_NAME = TaskLists.LIST_NAME; + /** + * The color of the list this task belongs to as integer (0xaarrggbb). This is auto-derived from the list the task belongs to. Do not write this value + * here. To change the color of an individual task use {@code TASK_COLOR} instead. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String LIST_COLOR = TaskLists.LIST_COLOR; + + /** + * The owner of the list this task belongs. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: String + *

+ *

+ * read-only + *

+ */ + public static final String LIST_OWNER = TaskLists.OWNER; + + /** + * The access level of the list this task belongs. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String LIST_ACCESS_LEVEL = TaskLists.ACCESS_LEVEL; + + /** + * The visibility of the list this task belongs. This is auto-derived from the list the task belongs to. Do not write this value here. + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public static final String VISIBLE = "visible"; + + public static final String CONTENT_URI_PATH = "instances"; + + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + CONTENT_URI_PATH); + + public static final String CONTENT_TYPE = ContentResolver.CURSOR_DIR_BASE_TYPE + "/" + AUTHORITY + + "." + CONTENT_URI_PATH; + + public static final String DEFAULT_SORT_ORDER = INSTANCE_DUE_SORTING; + + } + + /* + * ================================================================================== + * + * Everything below this line is not used yet and subject to change. Don't use it. + * + * ================================================================================== + */ + + /** + * Available values in Categories. + * + * Categories are per account. It's up to the front-end to ensure consistency of category colors across accounts. + * + * @author Marten Gajda + */ + public interface CategoriesColumns { + + public static final String _ID = "_id"; + + public static final String ACCOUNT_NAME = "account_name"; + + public static final String ACCOUNT_TYPE = "account_type"; + + public static final String NAME = "name"; + + public static final String COLOR = "color"; + } + + public static final class Categories implements CategoriesColumns { + + public static final String CONTENT_URI_PATH = "categories"; + + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + CONTENT_URI_PATH); + + public static final String DEFAULT_SORT_ORDER = NAME; + + } + + public interface AlarmsColumns { + public static final String ALARM_ID = "alarm_id"; + + public static final String LAST_TRIGGER = "last_trigger"; + + public static final String NEXT_TRIGGER = "next_trigger"; + } + + public static final class Alarms implements AlarmsColumns { + + public static final String CONTENT_URI_PATH = "alarms"; + + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + CONTENT_URI_PATH); + + } + + public interface PropertySyncColumns { + public static final String SYNC1 = "prop_sync1"; + + public static final String SYNC2 = "prop_sync2"; + + public static final String SYNC3 = "prop_sync3"; + + public static final String SYNC4 = "prop_sync4"; + + public static final String SYNC5 = "prop_sync5"; + + public static final String SYNC6 = "prop_sync6"; + + public static final String SYNC7 = "prop_sync7"; + + public static final String SYNC8 = "prop_sync8"; + } + + public interface PropertyColumns { + + public static final String PROPERTY_ID = "property_id"; + + public static final String TASK_ID = "task_id"; + + public static final String MIMETYPE = "mimetype"; + + public static final String VERSION = "prop_version"; + + public static final String DATA0 = "data0"; + + public static final String DATA1 = "data1"; + + public static final String DATA2 = "data2"; + + public static final String DATA3 = "data3"; + + public static final String DATA4 = "data4"; + + public static final String DATA5 = "data5"; + + public static final String DATA6 = "data6"; + + public static final String DATA7 = "data7"; + + public static final String DATA8 = "data8"; + + public static final String DATA9 = "data9"; + + public static final String DATA10 = "data10"; + + public static final String DATA11 = "data11"; + + public static final String DATA12 = "data12"; + + public static final String DATA13 = "data13"; + + public static final String DATA14 = "data14"; + + public static final String DATA15 = "data15"; + } + + public static final class Properties implements PropertySyncColumns, PropertyColumns { + + public static final String CONTENT_URI_PATH = "properties"; + + public static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/" + CONTENT_URI_PATH); + + public static final String DEFAULT_SORT_ORDER = DATA0; + + } + + public interface Property { + /** + * Attached documents. + *

+ * Note: Attachments are write-once. To change an attachment you'll have to remove and re-add it. + *

+ * + * @author Marten Gajda + */ + public static interface Attachment extends PropertyColumns { + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + + "/attachment"; + + /** + * ID of the attachment. Use this id to store and retrieve the attachment in the attachments table. + *

+ * Value: Long + *

+ */ + public final static String ATTACHMENT_ID = DATA1; + + /** + * Content-type of the attachment. + *

+ * Value: String + *

+ */ + public final static String FORMAT = DATA2; + } + + public static interface Attendee extends PropertyColumns { + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/attendee"; + + /** + * Name of the contact, if known. + *

+ * Value: String + *

+ */ + public final static String NAME = DATA0; + + /** + * Email address of the contact. + *

+ * Value: String + *

+ */ + public final static String EMAIL = DATA1; + + public final static String ROLE = DATA2; + + public final static String STATUS = DATA3; + + public final static String RSVP = DATA4; + } + + /** + * Categories are immutable. For creation is either the category id or name necessary + * + */ + public static interface Category extends PropertyColumns { + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/category"; + + /** + * Row id of the category. + *

+ * Value: Long + *

+ */ + public final static String CATEGORY_ID = DATA0; + + /** + * The name of the category + *

+ * Value: String + *

+ */ + public final static String CATEGORY_NAME = DATA1; + + /** + * The decimal coded color of the category + *

+ * Value: Integer + *

+ *

+ * read-only + *

+ */ + public final static String CATEGORY_COLOR = DATA2; + } + + public static interface Comment extends PropertyColumns { + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/comment"; + + /** + * Comment text. + *

+ * Value: String + *

+ */ + public final static String COMMENT = DATA0; + + /** + * Language code of the comment as defined in RFC5646 or null. + *

+ * Value: String + *

+ */ + public final static String LANGUAGE = DATA1; + } + + public static interface Contact extends PropertyColumns { + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/contact"; + + public final static String NAME = DATA0; + + public final static String LANGUAGE = DATA1; + } + + public static interface Relation extends PropertyColumns { + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/relation"; + + public final static String RELATED_ID = DATA1; + + public final static String RELATED_TYPE = DATA2; + } + + public static interface Alarm extends PropertyColumns { + + public static final int ALARM_TYPE_NOTHING = 0; + + public static final int ALARM_TYPE_MESSAGE = 1; + + public static final int ALARM_TYPE_EMAIL = 2; + + public static final int ALARM_TYPE_SMS = 3; + + public static final int ALARM_TYPE_SOUND = 4; + + public static final int ALARM_REFERENCE_DUE_DATE = 1; + + public static final int ALARM_REFERENCE_START_DATE = 2; + + /** + * The mime-type of this property. + */ + public final static String CONTENT_ITEM_TYPE = ContentResolver.CURSOR_ITEM_BASE_TYPE + "/alarm"; + + /** + * Number of minutes from the reference date when the alarm goes off. If the value is < 0 the alarm will go off after the reference date. + *

+ * Value: Integer + *

+ */ + public final static String MINUTES_BEFORE = DATA0; + + /** + * The reference date for the alarm. Either {@link ALARM_REFERENCE_DUE_DATE} or {@link ALARM_REFERENCE_START_DATE}. + *

+ * Value: Integer + *

+ */ + public final static String REFERENCE = DATA1; + + /** + * A message that appears with the alarm. + *

+ * Value: String + *

+ */ + public final static String MESSAGE = DATA2; + + /** + * The type of the alarm. Use the provided alarm types {@link ALARM_TYPE_MESSAGE}, {@link ALARM_TYPE_SOUND}, {@link ALARM_TYPE_NOTHING}, + * {@link ALARM_TYPE_EMAIL} and {@link ALARM_TYPE_SMS}. + *

+ * Value: Integer + *

+ */ + public final static String ALARM_TYPE = DATA3; + } + + } + +} diff --git a/app/src/main/res/xml/sync_notes.xml b/app/src/main/res/xml/sync_notes.xml new file mode 100644 index 00000000..51074ba3 --- /dev/null +++ b/app/src/main/res/xml/sync_notes.xml @@ -0,0 +1,15 @@ + + + diff --git a/app/src/main/res/xml/sync_tasks.xml b/app/src/main/res/xml/sync_tasks.xml new file mode 100644 index 00000000..0ed0aaeb --- /dev/null +++ b/app/src/main/res/xml/sync_tasks.xml @@ -0,0 +1,15 @@ + + +