diff --git a/app/build.gradle b/app/build.gradle
index c9c6c400..cad0b874 100644
--- a/app/build.gradle
+++ b/app/build.gradle
@@ -35,4 +35,10 @@ dependencies {
     compile 'com.github.peter9870:sparkbutton:master'
     testCompile 'junit:junit:4.12'
     compile 'com.mikhaellopez:circularfillableloaders:1.2.0'
+    compile 'com.squareup.retrofit2:retrofit:2.2.0'
+    compile 'com.squareup.retrofit2:converter-gson:2.1.0'
+    compile('com.mikepenz:materialdrawer:5.8.2@aar') {
+        transitive = true
+    }
+    compile 'com.github.chrisbanes:PhotoView:1.3.1'
 }
diff --git a/app/src/main/java/com/keylesspalace/tusky/Account.java b/app/src/main/java/com/keylesspalace/tusky/Account.java
deleted file mode 100644
index ae88d2f3..00000000
--- a/app/src/main/java/com/keylesspalace/tusky/Account.java
+++ /dev/null
@@ -1,92 +0,0 @@
-/* Copyright 2017 Andrew Dawson
- *
- * This file is part of Tusky.
- *
- * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
- * General Public License as published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
- * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
- * Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along with Tusky. If not, see
- * <http://www.gnu.org/licenses/>. */
-
-package com.keylesspalace.tusky;
-
-import android.text.Spanned;
-
-import org.json.JSONArray;
-import org.json.JSONException;
-import org.json.JSONObject;
-
-import java.util.ArrayList;
-import java.util.List;
-
-class Account {
-    String id;
-    String username;
-    String displayName;
-    Spanned note;
-    String url;
-    String avatar;
-    String header;
-    String followersCount;
-    String followingCount;
-    String statusesCount;
-
-    public static Account parse(JSONObject object) throws JSONException {
-        Account account = new Account();
-        account.id = object.getString("id");
-        account.username = object.getString("acct");
-        account.displayName = object.getString("display_name");
-        if (account.displayName.isEmpty()) {
-            account.displayName = object.getString("username");
-        }
-        account.note = HtmlUtils.fromHtml(object.getString("note"));
-        account.url = object.getString("url");
-        String avatarUrl = object.getString("avatar");
-        if (!avatarUrl.equals("/avatars/original/missing.png")) {
-            account.avatar = avatarUrl;
-        } else {
-            account.avatar = null;
-        }
-        String headerUrl = object.getString("header");
-        if (!headerUrl.equals("/headers/original/missing.png")) {
-            account.header = headerUrl;
-        } else {
-            account.header = null;
-        }
-        account.followersCount = object.getString("followers_count");
-        account.followingCount = object.getString("following_count");
-        account.statusesCount = object.getString("statuses_count");
-        return account;
-    }
-
-    public static List<Account> parse(JSONArray array) throws JSONException {
-        List<Account> accounts = new ArrayList<>();
-        for (int i = 0; i < array.length(); i++) {
-            JSONObject object = array.getJSONObject(i);
-            Account account = parse(object);
-            accounts.add(account);
-        }
-        return accounts;
-    }
-
-    @Override
-    public int hashCode() {
-        return id.hashCode();
-    }
-
-    @Override
-    public boolean equals(Object other) {
-        if (this.id == null) {
-            return this == other;
-        } else if (!(other instanceof Account)) {
-            return false;
-        }
-        Account account = (Account) other;
-        return account.id.equals(this.id);
-    }
-}
diff --git a/app/src/main/java/com/keylesspalace/tusky/AccountActivity.java b/app/src/main/java/com/keylesspalace/tusky/AccountActivity.java
index db7d2864..14e6834e 100644
--- a/app/src/main/java/com/keylesspalace/tusky/AccountActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/AccountActivity.java
@@ -39,30 +39,25 @@ import android.view.View;
 import android.widget.ImageView;
 import android.widget.TextView;
 
-import com.android.volley.AuthFailureError;
-import com.android.volley.Request;
-import com.android.volley.Response;
-import com.android.volley.VolleyError;
-import com.android.volley.toolbox.JsonArrayRequest;
-import com.android.volley.toolbox.JsonObjectRequest;
+import com.keylesspalace.tusky.entity.Account;
+import com.keylesspalace.tusky.entity.Relationship;
 import com.pkmmte.view.CircularImageView;
 import com.squareup.picasso.Picasso;
 
-import org.json.JSONArray;
-import org.json.JSONException;
-import org.json.JSONObject;
+import java.util.ArrayList;
+import java.util.List;
 
-import java.util.HashMap;
-import java.util.Map;
+import retrofit2.Call;
+import retrofit2.Callback;
+import retrofit2.Response;
 
 public class AccountActivity extends BaseActivity {
     private static final String TAG = "AccountActivity"; // Volley request tag and logging tag
 
-    private String domain;
-    private String accessToken;
     private String accountId;
     private boolean following = false;
     private boolean blocking = false;
+    private boolean muting = false;
     private boolean isSelf;
     private String openInWebUrl;
     private TabLayout tabLayout;
@@ -77,8 +72,6 @@ public class AccountActivity extends BaseActivity {
 
         SharedPreferences preferences = getSharedPreferences(
                 getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
-        domain = preferences.getString("domain", null);
-        accessToken = preferences.getString("accessToken", null);
         String loggedInAccountId = preferences.getString("loggedInAccountId", null);
 
         final Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
@@ -169,37 +162,17 @@ public class AccountActivity extends BaseActivity {
     }
 
     private void obtainAccount() {
-        String endpoint = String.format(getString(R.string.endpoint_accounts), accountId);
-        String url = "https://" + domain + endpoint;
-        JsonObjectRequest request = new JsonObjectRequest(Request.Method.GET, url, null,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        Account account;
-                        try {
-                            account = Account.parse(response);
-                        } catch (JSONException e) {
-                            onObtainAccountFailure();
-                            return;
-                        }
-                        onObtainAccountSuccess(account);
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onObtainAccountFailure();
-                    }
-                }) {
+        mastodonAPI.account(accountId).enqueue(new Callback<Account>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<Account> call, retrofit2.Response<Account> response) {
+                onObtainAccountSuccess(response.body());
             }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<Account> call, Throwable t) {
+                onObtainAccountFailure();
+            }
+        });
     }
 
     private void onObtainAccountSuccess(Account account) {
@@ -263,47 +236,28 @@ public class AccountActivity extends BaseActivity {
     }
 
     private void obtainRelationships() {
-        String endpoint = getString(R.string.endpoint_relationships);
-        String url = String.format("https://%s%s?id=%s", domain, endpoint, accountId);
-        JsonArrayRequest request = new JsonArrayRequest(url,
-                new Response.Listener<JSONArray>() {
-                    @Override
-                    public void onResponse(JSONArray response) {
-                        boolean following;
-                        boolean blocking;
-                        try {
-                            JSONObject object = response.getJSONObject(0);
-                            following = object.getBoolean("following");
-                            blocking = object.getBoolean("blocking");
-                        } catch (JSONException e) {
-                            onObtainRelationshipsFailure(e);
-                            return;
-                        }
-                        onObtainRelationshipsSuccess(following, blocking);
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onObtainRelationshipsFailure(error);
-                    }
-                }) {
+        List<String> ids = new ArrayList<>(1);
+        ids.add(accountId);
+        mastodonAPI.relationships(ids).enqueue(new Callback<List<Relationship>>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<List<Relationship>> call, retrofit2.Response<List<Relationship>> response) {
+                Relationship relationship = response.body().get(0);
+                onObtainRelationshipsSuccess(relationship.following, relationship.blocking, relationship.muting);
             }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<List<Relationship>> call, Throwable t) {
+                onObtainRelationshipsFailure((Exception) t);
+            }
+        });
     }
 
-    private void onObtainRelationshipsSuccess(boolean following, boolean blocking) {
+    private void onObtainRelationshipsSuccess(boolean following, boolean blocking, boolean muting) {
         this.following = following;
         this.blocking = blocking;
+        this.muting = muting;
 
-        if (!following || !blocking) {
+        if (!following || !blocking || !muting) {
             invalidateOptionsMenu();
         }
 
@@ -355,58 +309,42 @@ public class AccountActivity extends BaseActivity {
                 title = getString(R.string.action_block);
             }
             block.setTitle(title);
+            MenuItem mute = menu.findItem(R.id.action_mute);
+            if (muting) {
+                title = getString(R.string.action_unmute);
+            } else {
+                title = getString(R.string.action_mute);
+            }
+            mute.setTitle(title);
         } else {
             // It shouldn't be possible to block or follow yourself.
             menu.removeItem(R.id.action_follow);
             menu.removeItem(R.id.action_block);
+            menu.removeItem(R.id.action_mute);
         }
         return super.onPrepareOptionsMenu(menu);
     }
 
-    private void postRequest(String endpoint, Response.Listener<JSONObject> listener,
-            Response.ErrorListener errorListener) {
-        String url = "https://" + domain + endpoint;
-        JsonObjectRequest request = new JsonObjectRequest(Request.Method.POST, url, null, listener,
-                errorListener) {
+    private void follow(final String id) {
+        Callback<Relationship> cb = new Callback<Relationship>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<Relationship> call, retrofit2.Response<Relationship> response) {
+                following = response.body().following;
+                // TODO: display message/indicator when "requested" is true (i.e. when the follow is awaiting approval)
+                updateButtons();
+            }
+
+            @Override
+            public void onFailure(Call<Relationship> call, Throwable t) {
+                onFollowFailure(id);
             }
         };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
-    }
 
-    private void follow(final String id) {
-        int endpointId;
         if (following) {
-            endpointId = R.string.endpoint_unfollow;
+            mastodonAPI.unfollowAccount(id).enqueue(cb);
         } else {
-            endpointId = R.string.endpoint_follow;
+            mastodonAPI.followAccount(id).enqueue(cb);
         }
-        postRequest(String.format(getString(endpointId), id),
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        boolean followingValue;
-                        try {
-                            followingValue = response.getBoolean("following");
-                        } catch (JSONException e) {
-                            onFollowFailure(id);
-                            return;
-                        }
-                        following = followingValue;
-                        updateButtons();
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onFollowFailure(id);
-                    }
-                });
     }
 
     private void onFollowFailure(final String id) {
@@ -428,33 +366,23 @@ public class AccountActivity extends BaseActivity {
     }
 
     private void block(final String id) {
-        int endpointId;
+        Callback<Relationship> cb = new Callback<Relationship>() {
+            @Override
+            public void onResponse(Call<Relationship> call, retrofit2.Response<Relationship> response) {
+                blocking = response.body().blocking;
+                updateButtons();
+            }
+
+            @Override
+            public void onFailure(Call<Relationship> call, Throwable t) {
+                onBlockFailure(id);
+            }
+        };
         if (blocking) {
-            endpointId = R.string.endpoint_unblock;
+            mastodonAPI.unblockAccount(id).enqueue(cb);
         } else {
-            endpointId = R.string.endpoint_block;
+            mastodonAPI.blockAccount(id).enqueue(cb);
         }
-        postRequest(String.format(getString(endpointId), id),
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        boolean blockingValue;
-                        try {
-                            blockingValue = response.getBoolean("blocking");
-                        } catch (JSONException e) {
-                            onBlockFailure(id);
-                            return;
-                        }
-                        blocking = blockingValue;
-                        updateButtons();
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onBlockFailure(id);
-                    }
-                });
     }
 
     private void onBlockFailure(final String id) {
@@ -475,6 +403,50 @@ public class AccountActivity extends BaseActivity {
                 .show();
     }
 
+
+    private void mute(final String id) {
+        Callback<Relationship> cb = new Callback<Relationship>() {
+            @Override
+            public void onResponse(Call<Relationship> call, Response<Relationship> response) {
+                muting = response.body().muting;
+                updateButtons();
+            }
+
+            @Override
+            public void onFailure(Call<Relationship> call, Throwable t) {
+                onMuteFailure(id);
+            }
+        };
+
+        if (muting) {
+            mastodonAPI.unmuteAccount(id).enqueue(cb);
+        } else {
+            mastodonAPI.muteAccount(id).enqueue(cb);
+        }
+    }
+
+    private void onMuteFailure(final String id) {
+        int messageId;
+
+        if (muting) {
+            messageId = R.string.error_unmuting;
+        } else {
+            messageId = R.string.error_muting;
+        }
+
+        View.OnClickListener listener = new View.OnClickListener() {
+            @Override
+            public void onClick(View v) {
+                mute(id);
+            }
+        };
+
+        Snackbar.make(findViewById(R.id.activity_account), messageId, Snackbar.LENGTH_LONG)
+                .setAction(R.string.action_retry, listener)
+                .show();
+    }
+
+
     @Override
     public boolean onOptionsItemSelected(MenuItem item) {
         switch (item.getItemId()) {
@@ -496,6 +468,10 @@ public class AccountActivity extends BaseActivity {
                 block(accountId);
                 return true;
             }
+            case R.id.action_mute: {
+                mute(accountId);
+                return true;
+            }
         }
         return super.onOptionsItemSelected(item);
     }
diff --git a/app/src/main/java/com/keylesspalace/tusky/AccountAdapter.java b/app/src/main/java/com/keylesspalace/tusky/AccountAdapter.java
index 6fd60a6f..ab7653a6 100644
--- a/app/src/main/java/com/keylesspalace/tusky/AccountAdapter.java
+++ b/app/src/main/java/com/keylesspalace/tusky/AccountAdapter.java
@@ -17,6 +17,8 @@ package com.keylesspalace.tusky;
 
 import android.support.v7.widget.RecyclerView;
 
+import com.keylesspalace.tusky.entity.Account;
+
 import java.util.ArrayList;
 import java.util.List;
 
diff --git a/app/src/main/java/com/keylesspalace/tusky/AccountFragment.java b/app/src/main/java/com/keylesspalace/tusky/AccountFragment.java
index c3e7be81..09115c60 100644
--- a/app/src/main/java/com/keylesspalace/tusky/AccountFragment.java
+++ b/app/src/main/java/com/keylesspalace/tusky/AccountFragment.java
@@ -34,16 +34,17 @@ import com.android.volley.AuthFailureError;
 import com.android.volley.Request;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
-import com.android.volley.toolbox.JsonArrayRequest;
 import com.android.volley.toolbox.StringRequest;
-
-import org.json.JSONArray;
-import org.json.JSONException;
+import com.keylesspalace.tusky.entity.Account;
+import com.keylesspalace.tusky.entity.Relationship;
 
 import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
 
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class AccountFragment extends Fragment implements AccountActionListener,
         FooterActionListener {
     private static final String TAG = "Account"; // logging tag and Volley request tag
@@ -63,6 +64,7 @@ public class AccountFragment extends Fragment implements AccountActionListener,
     private EndlessOnScrollListener scrollListener;
     private AccountAdapter adapter;
     private TabLayout.OnTabSelectedListener onTabSelectedListener;
+    private MastodonAPI api;
 
     public static AccountFragment newInstance(Type type) {
         Bundle arguments = new Bundle();
@@ -92,6 +94,7 @@ public class AccountFragment extends Fragment implements AccountActionListener,
                 getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
         domain = preferences.getString("domain", null);
         accessToken = preferences.getString("accessToken", null);
+        api = ((BaseActivity) getActivity()).mastodonAPI;
     }
 
     @Override
@@ -170,55 +173,33 @@ public class AccountFragment extends Fragment implements AccountActionListener,
     }
 
     private void fetchAccounts(final String fromId) {
-        String endpoint;
+        Callback<List<Account>> cb = new Callback<List<Account>>() {
+            @Override
+            public void onResponse(Call<List<Account>> call, retrofit2.Response<List<Account>> response) {
+                onFetchAccountsSuccess(response.body(), fromId);
+            }
+
+            @Override
+            public void onFailure(Call<List<Account>> call, Throwable t) {
+                onFetchAccountsFailure((Exception) t);
+            }
+        };
+
         switch (type) {
             default:
             case FOLLOWS: {
-                endpoint = String.format(getString(R.string.endpoint_following), accountId);
+                api.accountFollowing(accountId, fromId, null, null).enqueue(cb);
                 break;
             }
             case FOLLOWERS: {
-                endpoint = String.format(getString(R.string.endpoint_followers), accountId);
+                api.accountFollowers(accountId, fromId, null, null).enqueue(cb);
                 break;
             }
             case BLOCKS: {
-                endpoint = getString(R.string.endpoint_blocks);
+                api.blocks(fromId, null, null).enqueue(cb);
                 break;
             }
         }
-        String url = "https://" + domain + endpoint;
-        if (fromId != null) {
-            url += "?max_id=" + fromId;
-        }
-        JsonArrayRequest request = new JsonArrayRequest(url,
-                new Response.Listener<JSONArray>() {
-                    @Override
-                    public void onResponse(JSONArray response) {
-                        List<Account> accounts;
-                        try {
-                            accounts = Account.parse(response);
-                        } catch (JSONException e) {
-                            onFetchAccountsFailure(e);
-                            return;
-                        }
-                        onFetchAccountsSuccess(accounts, fromId);
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onFetchAccountsFailure(error);
-                    }
-                }) {
-            @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
-            }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(getContext()).addToRequestQueue(request);
     }
 
     private void fetchAccounts() {
@@ -285,35 +266,23 @@ public class AccountFragment extends Fragment implements AccountActionListener,
     }
 
     public void onBlock(final boolean block, final String id, final int position) {
-        String endpoint;
-        if (!block) {
-            endpoint = String.format(getString(R.string.endpoint_unblock), id);
-        } else {
-            endpoint = String.format(getString(R.string.endpoint_block), id);
-        }
-        String url = "https://" + domain + endpoint;
-        StringRequest request = new StringRequest(Request.Method.POST, url,
-                new Response.Listener<String>() {
-                    @Override
-                    public void onResponse(String response) {
-                        onBlockSuccess(block, position);
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onBlockFailure(block, id);
-                    }
-                }) {
+        Callback<Relationship> cb = new Callback<Relationship>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<Relationship> call, retrofit2.Response<Relationship> response) {
+                onBlockSuccess(block, position);
+            }
+
+            @Override
+            public void onFailure(Call<Relationship> call, Throwable t) {
+                onBlockFailure(block, id);
             }
         };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(getContext()).addToRequestQueue(request);
+
+        if (!block) {
+            api.unblockAccount(id).enqueue(cb);
+        } else {
+            api.blockAccount(id).enqueue(cb);
+        }
     }
 
     private void onBlockSuccess(boolean blocked, int position) {
diff --git a/app/src/main/java/com/keylesspalace/tusky/BaseActivity.java b/app/src/main/java/com/keylesspalace/tusky/BaseActivity.java
index bb3a4fb5..fa529ed9 100644
--- a/app/src/main/java/com/keylesspalace/tusky/BaseActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/BaseActivity.java
@@ -15,7 +15,9 @@
 
 package com.keylesspalace.tusky;
 
+import android.content.Context;
 import android.content.Intent;
+import android.content.SharedPreferences;
 import android.graphics.Color;
 import android.graphics.PorterDuff;
 import android.graphics.drawable.Drawable;
@@ -23,17 +25,35 @@ import android.os.Bundle;
 import android.preference.PreferenceManager;
 import android.support.annotation.Nullable;
 import android.support.v7.app.AppCompatActivity;
+import android.text.Spanned;
 import android.util.TypedValue;
 import android.view.Menu;
 
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+
+import java.io.IOException;
+
+import okhttp3.Interceptor;
+import okhttp3.OkHttpClient;
+import okhttp3.Request;
+import okhttp3.Response;
+import retrofit2.Retrofit;
+import retrofit2.converter.gson.GsonConverterFactory;
+
 /* There isn't presently a way to globally change the theme of a whole application at runtime, just
  * individual activities. So, each activity has to set its theme before any views are created. And
  * the most expedient way to accomplish this was to put it in a base class and just have every
  * activity extend from it. */
 public class BaseActivity extends AppCompatActivity {
+    protected MastodonAPI mastodonAPI;
+
     @Override
     protected void onCreate(@Nullable Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);
+
+        createMastodonAPI();
+
         if (PreferenceManager.getDefaultSharedPreferences(this).getBoolean("lightTheme", false)) {
             setTheme(R.style.AppTheme_Light);
         }
@@ -59,6 +79,46 @@ public class BaseActivity extends AppCompatActivity {
         overridePendingTransition(R.anim.slide_from_left, R.anim.slide_to_right);
     }
 
+    protected String getAccessToken() {
+        SharedPreferences preferences = getSharedPreferences(getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
+        return preferences.getString("accessToken", null);
+    }
+
+    protected String getBaseUrl() {
+        SharedPreferences preferences = getSharedPreferences(getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
+        return "https://" + preferences.getString("domain", null);
+    }
+
+    protected void createMastodonAPI() {
+        OkHttpClient okHttpClient = new OkHttpClient.Builder()
+                .addInterceptor(new Interceptor() {
+                    @Override
+                    public Response intercept(Chain chain) throws IOException {
+                        Request originalRequest = chain.request();
+
+                        Request.Builder builder = originalRequest.newBuilder()
+                                .header("Authorization", String.format("Bearer %s", getAccessToken()));
+
+                        Request newRequest = builder.build();
+
+                        return chain.proceed(newRequest);
+                    }
+                })
+                .build();
+
+        Gson gson = new GsonBuilder()
+                .registerTypeAdapter(Spanned.class, new SpannedTypeAdapter())
+                .create();
+
+        Retrofit retrofit = new Retrofit.Builder()
+                .baseUrl(getBaseUrl())
+                .client(okHttpClient)
+                .addConverterFactory(GsonConverterFactory.create(gson))
+                .build();
+
+        mastodonAPI = retrofit.create(MastodonAPI.class);
+    }
+
     @Override
     public boolean onCreateOptionsMenu(Menu menu) {
         TypedValue value = new TypedValue();
diff --git a/app/src/main/java/com/keylesspalace/tusky/BlocksActivity.java b/app/src/main/java/com/keylesspalace/tusky/BlocksActivity.java
index 99e39193..534e5ea6 100644
--- a/app/src/main/java/com/keylesspalace/tusky/BlocksActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/BlocksActivity.java
@@ -21,6 +21,7 @@ import android.support.v4.app.Fragment;
 import android.support.v4.app.FragmentTransaction;
 import android.support.v7.app.ActionBar;
 import android.support.v7.widget.Toolbar;
+import android.view.MenuItem;
 
 public class BlocksActivity extends BaseActivity {
     @Override
@@ -33,6 +34,8 @@ public class BlocksActivity extends BaseActivity {
         ActionBar bar = getSupportActionBar();
         if (bar != null) {
             bar.setTitle(getString(R.string.title_blocks));
+            bar.setDisplayHomeAsUpEnabled(true);
+            bar.setDisplayShowHomeEnabled(true);
         }
 
         FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
@@ -40,4 +43,15 @@ public class BlocksActivity extends BaseActivity {
         fragmentTransaction.add(R.id.fragment_container, fragment);
         fragmentTransaction.commit();
     }
+
+    @Override
+    public boolean onOptionsItemSelected(MenuItem item) {
+        switch (item.getItemId()) {
+            case android.R.id.home: {
+                onBackPressed();
+                return true;
+            }
+        }
+        return super.onOptionsItemSelected(item);
+    }
 }
diff --git a/app/src/main/java/com/keylesspalace/tusky/BlocksAdapter.java b/app/src/main/java/com/keylesspalace/tusky/BlocksAdapter.java
index 0b440547..37bfc403 100644
--- a/app/src/main/java/com/keylesspalace/tusky/BlocksAdapter.java
+++ b/app/src/main/java/com/keylesspalace/tusky/BlocksAdapter.java
@@ -23,6 +23,7 @@ import android.widget.Button;
 import android.widget.ImageView;
 import android.widget.TextView;
 
+import com.keylesspalace.tusky.entity.Account;
 import com.squareup.picasso.Picasso;
 
 import java.util.HashSet;
diff --git a/app/src/main/java/com/keylesspalace/tusky/ComposeActivity.java b/app/src/main/java/com/keylesspalace/tusky/ComposeActivity.java
index 375731de..4c8f2154 100644
--- a/app/src/main/java/com/keylesspalace/tusky/ComposeActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/ComposeActivity.java
@@ -76,6 +76,8 @@ import com.android.volley.Request;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.JsonObjectRequest;
+import com.keylesspalace.tusky.entity.Media;
+import com.keylesspalace.tusky.entity.Status;
 
 import org.json.JSONArray;
 import org.json.JSONException;
@@ -95,6 +97,12 @@ import java.util.Locale;
 import java.util.Map;
 import java.util.Random;
 
+import okhttp3.MediaType;
+import okhttp3.MultipartBody;
+import okhttp3.RequestBody;
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class ComposeActivity extends BaseActivity {
     private static final String TAG = "ComposeActivity"; // logging tag, and volley request tag
     private static final int STATUS_CHARACTER_LIMIT = 500;
@@ -137,7 +145,7 @@ public class ComposeActivity extends BaseActivity {
         ImageView preview;
         Uri uri;
         String id;
-        Request uploadRequest;
+        Call<Media> uploadRequest;
         ReadyStage readyStage;
         byte[] content;
         long mediaSize;
@@ -629,53 +637,28 @@ public class ComposeActivity extends BaseActivity {
 
     private void sendStatus(String content, String visibility, boolean sensitive,
                             String spoilerText) {
-        String endpoint = getString(R.string.endpoint_status);
-        String url = "https://" + domain + endpoint;
-        JSONObject parameters = new JSONObject();
-        try {
-            parameters.put("status", content);
-            parameters.put("visibility", visibility);
-            parameters.put("sensitive", sensitive);
-            parameters.put("spoiler_text", spoilerText);
-            if (inReplyToId != null) {
-                parameters.put("in_reply_to_id", inReplyToId);
-            }
-            JSONArray mediaIds = new JSONArray();
-            for (QueuedMedia item : mediaQueued) {
-                mediaIds.put(item.id);
-            }
-            if (mediaIds.length() > 0) {
-                parameters.put("media_ids", mediaIds);
-            }
-        } catch (JSONException e) {
-            onSendFailure();
-            return;
+        ArrayList<String> mediaIds = new ArrayList<String>();
+
+        for (QueuedMedia item : mediaQueued) {
+            mediaIds.add(item.id);
         }
-        JsonObjectRequest request = new JsonObjectRequest(Request.Method.POST, url, parameters,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        onSendSuccess();
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onSendFailure();
-                    }
-                }) {
+
+        mastodonAPI.createStatus(content, inReplyToId, spoilerText, visibility, sensitive, mediaIds).enqueue(new Callback<Status>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<Status> call, retrofit2.Response<Status> response) {
+                onSendSuccess();
             }
-        };
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<Status> call, Throwable t) {
+                onSendFailure();
+            }
+        });
     }
 
     private void onSendSuccess() {
-        Toast.makeText(this, getString(R.string.confirmation_send), Toast.LENGTH_SHORT).show();
+        Snackbar bar = Snackbar.make(findViewById(R.id.activity_compose), getString(R.string.confirmation_send), Snackbar.LENGTH_SHORT);
+        bar.show();
         finish();
     }
 
@@ -941,9 +924,6 @@ public class ComposeActivity extends BaseActivity {
     private void uploadMedia(final QueuedMedia item) {
         item.readyStage = QueuedMedia.ReadyStage.UPLOADING;
 
-        String endpoint = getString(R.string.endpoint_media);
-        String url = "https://" + domain + endpoint;
-
         final String mimeType = getContentResolver().getType(item.uri);
         MimeTypeMap map = MimeTypeMap.getSingleton();
         String fileExtension = map.getExtensionFromMimeType(mimeType);
@@ -953,58 +933,42 @@ public class ComposeActivity extends BaseActivity {
                 randomAlphanumericString(10),
                 fileExtension);
 
-        MultipartRequest request = new MultipartRequest(Request.Method.POST, url, null,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        try {
-                            item.id = response.getString("id");
-                        } catch (JSONException e) {
-                            onUploadFailure(item);
-                            return;
-                        }
-                        waitForMediaLatch.countDown();
-                    }
-                }, new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onUploadFailure(item);
-                    }
-                }) {
+        byte[] content = item.content;
+
+        if (content == null) {
+            InputStream stream;
+
+            try {
+                stream = getContentResolver().openInputStream(item.uri);
+            } catch (FileNotFoundException e) {
+                return;
+            }
+
+            content = inputStreamGetBytes(stream);
+            IOUtils.closeQuietly(stream);
+
+            if (content == null) {
+                return;
+            }
+        }
+
+        RequestBody requestFile = RequestBody.create(MediaType.parse(mimeType), content);
+        MultipartBody.Part body = MultipartBody.Part.createFormData("file", filename, requestFile);
+
+        item.uploadRequest = mastodonAPI.uploadMedia(body);
+
+        item.uploadRequest.enqueue(new Callback<Media>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<Media> call, retrofit2.Response<Media> response) {
+                item.id = response.body().id;
+                waitForMediaLatch.countDown();
             }
 
             @Override
-            public DataItem getData() {
-                byte[] content = item.content;
-                if (content == null) {
-                    InputStream stream;
-                    try {
-                        stream = getContentResolver().openInputStream(item.uri);
-                    } catch (FileNotFoundException e) {
-                        return null;
-                    }
-                    content = inputStreamGetBytes(stream);
-                    IOUtils.closeQuietly(stream);
-                    if (content == null) {
-                        return null;
-                    }
-                }
-                DataItem data = new DataItem();
-                data.name = "file";
-                data.filename = filename;
-                data.mimeType = mimeType;
-                data.content = content;
-                return data;
+            public void onFailure(Call<Media> call, Throwable t) {
+                onUploadFailure(item);
             }
-        };
-        request.setTag(TAG);
-        item.uploadRequest = request;
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+        });
     }
 
     private void onUploadFailure(QueuedMedia item) {
diff --git a/app/src/main/java/com/keylesspalace/tusky/FavouritesActivity.java b/app/src/main/java/com/keylesspalace/tusky/FavouritesActivity.java
index 6b66b846..8241b950 100644
--- a/app/src/main/java/com/keylesspalace/tusky/FavouritesActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/FavouritesActivity.java
@@ -21,6 +21,7 @@ import android.support.v4.app.Fragment;
 import android.support.v4.app.FragmentTransaction;
 import android.support.v7.app.ActionBar;
 import android.support.v7.widget.Toolbar;
+import android.view.MenuItem;
 
 public class FavouritesActivity extends BaseActivity {
     @Override
@@ -33,6 +34,8 @@ public class FavouritesActivity extends BaseActivity {
         ActionBar bar = getSupportActionBar();
         if (bar != null) {
             bar.setTitle(getString(R.string.title_favourites));
+            bar.setDisplayHomeAsUpEnabled(true);
+            bar.setDisplayShowHomeEnabled(true);
         }
 
         FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
@@ -40,4 +43,15 @@ public class FavouritesActivity extends BaseActivity {
         fragmentTransaction.add(R.id.fragment_container, fragment);
         fragmentTransaction.commit();
     }
+
+    @Override
+    public boolean onOptionsItemSelected(MenuItem item) {
+        switch (item.getItemId()) {
+            case android.R.id.home: {
+                onBackPressed();
+                return true;
+            }
+        }
+        return super.onOptionsItemSelected(item);
+    }
 }
diff --git a/app/src/main/java/com/keylesspalace/tusky/FollowAdapter.java b/app/src/main/java/com/keylesspalace/tusky/FollowAdapter.java
index de70bcee..140e32d0 100644
--- a/app/src/main/java/com/keylesspalace/tusky/FollowAdapter.java
+++ b/app/src/main/java/com/keylesspalace/tusky/FollowAdapter.java
@@ -23,6 +23,7 @@ import android.view.ViewGroup;
 import android.widget.ImageView;
 import android.widget.TextView;
 
+import com.keylesspalace.tusky.entity.Account;
 import com.squareup.picasso.Picasso;
 
 /** Both for follows and following lists. */
diff --git a/app/src/main/java/com/keylesspalace/tusky/MainActivity.java b/app/src/main/java/com/keylesspalace/tusky/MainActivity.java
index 3fb5b338..d027952f 100644
--- a/app/src/main/java/com/keylesspalace/tusky/MainActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/MainActivity.java
@@ -21,6 +21,7 @@ import android.content.Context;
 import android.content.Intent;
 import android.content.SharedPreferences;
 import android.graphics.drawable.Drawable;
+import android.net.Uri;
 import android.os.Build;
 import android.os.SystemClock;
 import android.preference.PreferenceManager;
@@ -35,12 +36,26 @@ import android.transition.TransitionInflater;
 import android.view.Menu;
 import android.view.MenuItem;
 import android.view.View;
+import android.widget.ImageView;
 
 import com.android.volley.AuthFailureError;
 import com.android.volley.Request;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.JsonObjectRequest;
+import com.keylesspalace.tusky.entity.Account;
+import com.mikepenz.materialdrawer.AccountHeader;
+import com.mikepenz.materialdrawer.AccountHeaderBuilder;
+import com.mikepenz.materialdrawer.Drawer;
+import com.mikepenz.materialdrawer.DrawerBuilder;
+import com.mikepenz.materialdrawer.model.PrimaryDrawerItem;
+import com.mikepenz.materialdrawer.model.ProfileDrawerItem;
+import com.mikepenz.materialdrawer.model.SecondaryDrawerItem;
+import com.mikepenz.materialdrawer.model.interfaces.IDrawerItem;
+import com.mikepenz.materialdrawer.model.interfaces.IProfile;
+import com.mikepenz.materialdrawer.util.AbstractDrawerImageLoader;
+import com.mikepenz.materialdrawer.util.DrawerImageLoader;
+import com.squareup.picasso.Picasso;
 
 import org.json.JSONException;
 import org.json.JSONObject;
@@ -49,6 +64,9 @@ import java.util.HashMap;
 import java.util.Map;
 import java.util.Stack;
 
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class MainActivity extends BaseActivity {
     private static final String TAG = "MainActivity"; // logging tag and Volley request tag
 
@@ -59,6 +77,8 @@ public class MainActivity extends BaseActivity {
     private String loggedInAccountUsername;
     Stack<Integer> pageHistory = new Stack<Integer>();
     private ViewPager viewPager;
+    private AccountHeader headerResult;
+    private Drawer drawer;
 
     @Override
     protected void onCreate(Bundle savedInstanceState) {
@@ -80,6 +100,88 @@ public class MainActivity extends BaseActivity {
             }
         });
 
+        headerResult = new AccountHeaderBuilder()
+                .withActivity(this)
+                .withSelectionListEnabledForSingleProfile(false)
+                .withTranslucentStatusBar(true)
+                .withCompactStyle(true)
+                .withOnAccountHeaderProfileImageListener(new AccountHeader.OnAccountHeaderProfileImageListener() {
+                    @Override
+                    public boolean onProfileImageClick(View view, IProfile profile, boolean current) {
+                        Intent intent = new Intent(MainActivity.this, AccountActivity.class);
+                        intent.putExtra("id", loggedInAccountId);
+                        startActivity(intent);
+                        return false;
+                    }
+
+                    @Override
+                    public boolean onProfileImageLongClick(View view, IProfile profile, boolean current) {
+                        return false;
+                    }
+                })
+                .build();
+
+        DrawerImageLoader.init(new AbstractDrawerImageLoader() {
+            @Override
+            public void set(ImageView imageView, Uri uri, Drawable placeholder) {
+                Picasso.with(imageView.getContext()).load(uri).placeholder(placeholder).into(imageView);
+            }
+
+            @Override
+            public void cancel(ImageView imageView) {
+                Picasso.with(imageView.getContext()).cancelRequest(imageView);
+            }
+        });
+
+        drawer = new DrawerBuilder()
+                .withActivity(this)
+                .withToolbar(toolbar)
+                .withTranslucentStatusBar(true)
+                .withAccountHeader(headerResult)
+                .withHasStableIds(true)
+                .withSelectedItem(-1)
+                .addDrawerItems(
+                        new PrimaryDrawerItem().withIdentifier(1).withName(getString(R.string.action_view_favourites)).withSelectable(false),
+                        new PrimaryDrawerItem().withIdentifier(2).withName(getString(R.string.action_view_blocks)).withSelectable(false),
+                        new PrimaryDrawerItem().withIdentifier(3).withName(getString(R.string.action_view_preferences)).withSelectable(false),
+                        new PrimaryDrawerItem().withIdentifier(4).withName(getString(R.string.action_logout)).withSelectable(false)
+                )
+                .withOnDrawerItemClickListener(new Drawer.OnDrawerItemClickListener() {
+                    @Override
+                    public boolean onItemClick(View view, int position, IDrawerItem drawerItem) {
+                        if (drawerItem != null) {
+                            long drawerItemIdentifier = drawerItem.getIdentifier();
+
+                            if (drawerItemIdentifier == 1) {
+                                Intent intent = new Intent(MainActivity.this, FavouritesActivity.class);
+                                startActivity(intent);
+                            } else if (drawerItemIdentifier == 2) {
+                                Intent intent = new Intent(MainActivity.this, BlocksActivity.class);
+                                startActivity(intent);
+                            } else if (drawerItemIdentifier == 3) {
+                                Intent intent = new Intent(MainActivity.this, PreferencesActivity.class);
+                                startActivity(intent);
+                            } else if (drawerItemIdentifier == 4) {
+                                if (notificationServiceEnabled) {
+                                    alarmManager.cancel(serviceAlarmIntent);
+                                }
+                                SharedPreferences preferences = getSharedPreferences(
+                                        getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
+                                SharedPreferences.Editor editor = preferences.edit();
+                                editor.remove("domain");
+                                editor.remove("accessToken");
+                                editor.apply();
+                                Intent intent = new Intent(MainActivity.this, SplashActivity.class);
+                                startActivity(intent);
+                                finish();
+                            }
+                        }
+
+                        return false;
+                    }
+                })
+                .build();
+
         // Setup the tabs and timeline pager.
         TimelinePagerAdapter adapter = new TimelinePagerAdapter(getSupportFragmentManager());
         String[] pageTitles = {
@@ -148,48 +250,42 @@ public class MainActivity extends BaseActivity {
     private void fetchUserInfo() {
         SharedPreferences preferences = getSharedPreferences(
                 getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
-        String domain = preferences.getString("domain", null);
-        final String accessToken = preferences.getString("accessToken", null);
+        final String domain = preferences.getString("domain", null);
         String id = preferences.getString("loggedInAccountId", null);
         String username = preferences.getString("loggedInAccountUsername", null);
-        if (id != null && username != null) {
-            loggedInAccountId = id;
-            loggedInAccountUsername = username;
-        } else {
-            String endpoint = getString(R.string.endpoint_verify_credentials);
-            String url = "https://" + domain + endpoint;
-            JsonObjectRequest request = new JsonObjectRequest(Request.Method.GET, url, null,
-                    new Response.Listener<JSONObject>() {
-                        @Override
-                        public void onResponse(JSONObject response) {
-                            String username;
-                            String id;
-                            try {
-                                id = response.getString("id");
-                                username = response.getString("acct");
-                            } catch (JSONException e) {
-                                onFetchUserInfoFailure(e);
-                                return;
-                            }
-                            onFetchUserInfoSuccess(id, username);
-                        }
-                    },
-                    new Response.ErrorListener() {
-                        @Override
-                        public void onErrorResponse(VolleyError error) {
-                            onFetchUserInfoFailure(error);
-                        }
-                    }) {
+        //if (id != null && username != null) {
+        //    loggedInAccountId = id;
+        //    loggedInAccountUsername = username;
+        //} else {
+            mastodonAPI.accountVerifyCredentials().enqueue(new Callback<Account>() {
                 @Override
-                public Map<String, String> getHeaders() throws AuthFailureError {
-                    Map<String, String> headers = new HashMap<>();
-                    headers.put("Authorization", "Bearer " + accessToken);
-                    return headers;
+                public void onResponse(Call<Account> call, retrofit2.Response<Account> response) {
+                    Account me = response.body();
+                    ImageView background = headerResult.getHeaderBackgroundView();
+
+                    Picasso.with(MainActivity.this)
+                            .load(me.header)
+                            .placeholder(R.drawable.account_header_missing)
+                            .resize(background.getWidth(), background.getHeight())
+                            .centerCrop()
+                            .into(background);
+
+                    headerResult.addProfiles(
+                            new ProfileDrawerItem()
+                                    .withName(me.displayName)
+                                    .withEmail(String.format("%s@%s", me.username, domain))
+                                    .withIcon(me.avatar)
+                    );
+
+                    //onFetchUserInfoSuccess(response.body().id, response.body().username);
                 }
-            };
-            request.setTag(TAG);
-            VolleySingleton.getInstance(this).addToRequestQueue(request);
-        }
+
+                @Override
+                public void onFailure(Call<Account> call, Throwable t) {
+                    onFetchUserInfoFailure((Exception) t);
+                }
+            });
+        //}
     }
 
     private void onFetchUserInfoSuccess(String id, String username) {
@@ -207,59 +303,11 @@ public class MainActivity extends BaseActivity {
         Log.e(TAG, "Failed to fetch user info. " + exception.getMessage());
     }
 
-    @Override
-    public boolean onCreateOptionsMenu(Menu menu) {
-        getMenuInflater().inflate(R.menu.main_toolbar, menu);
-        return super.onCreateOptionsMenu(menu);
-    }
-
-    @Override
-    public boolean onOptionsItemSelected(MenuItem item) {
-        switch (item.getItemId()) {
-            case R.id.action_view_profile: {
-                Intent intent = new Intent(this, AccountActivity.class);
-                intent.putExtra("id", loggedInAccountId);
-                startActivity(intent);
-                return true;
-            }
-            case R.id.action_view_preferences: {
-                Intent intent = new Intent(this, PreferencesActivity.class);
-                startActivity(intent);
-                return true;
-            }
-            case R.id.action_view_favourites: {
-                Intent intent = new Intent(this, FavouritesActivity.class);
-                startActivity(intent);
-                return true;
-            }
-            case R.id.action_view_blocks: {
-                Intent intent = new Intent(this, BlocksActivity.class);
-                startActivity(intent);
-                return true;
-            }
-            case R.id.action_logout: {
-                if (notificationServiceEnabled) {
-                    alarmManager.cancel(serviceAlarmIntent);
-                }
-                SharedPreferences preferences = getSharedPreferences(
-                        getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
-                SharedPreferences.Editor editor = preferences.edit();
-                editor.remove("domain");
-                editor.remove("accessToken");
-                editor.apply();
-                Intent intent = new Intent(this, SplashActivity.class);
-                startActivity(intent);
-                finish();
-                return true;
-            }
-        }
-
-        return super.onOptionsItemSelected(item);
-    }
-
     @Override
     public void onBackPressed() {
-        if(pageHistory.empty()) {
+        if(drawer != null && drawer.isDrawerOpen()) {
+            drawer.closeDrawer();
+        } else if(pageHistory.empty()) {
             super.onBackPressed();
         } else {
             pageHistory.pop();
diff --git a/app/src/main/java/com/keylesspalace/tusky/MastodonAPI.java b/app/src/main/java/com/keylesspalace/tusky/MastodonAPI.java
new file mode 100644
index 00000000..66019ad1
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/MastodonAPI.java
@@ -0,0 +1,170 @@
+package com.keylesspalace.tusky;
+
+import com.keylesspalace.tusky.entity.Account;
+import com.keylesspalace.tusky.entity.Media;
+import com.keylesspalace.tusky.entity.Notification;
+import com.keylesspalace.tusky.entity.Relationship;
+import com.keylesspalace.tusky.entity.Status;
+import com.keylesspalace.tusky.entity.StatusContext;
+
+import java.util.List;
+
+import okhttp3.MultipartBody;
+import okhttp3.RequestBody;
+import okhttp3.ResponseBody;
+import retrofit2.Call;
+import retrofit2.http.DELETE;
+import retrofit2.http.Field;
+import retrofit2.http.FormUrlEncoded;
+import retrofit2.http.GET;
+import retrofit2.http.Multipart;
+import retrofit2.http.POST;
+import retrofit2.http.Part;
+import retrofit2.http.Path;
+import retrofit2.http.Query;
+
+public interface MastodonAPI {
+    @GET("api/v1/timelines/home")
+    Call<List<Status>> homeTimeline(
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @GET("api/v1/timelines/public")
+    Call<List<Status>> publicTimeline(
+            @Query("local") Boolean local,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @GET("api/v1/timelines/tag/{hashtag}")
+    Call<List<Status>> hashtagTimeline(
+            @Path("hashtag") String hashtag,
+            @Query("local") Boolean local,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+
+    @GET("api/v1/notifications")
+    Call<List<Notification>> notifications(
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @POST("api/v1/notifications/clear")
+    Call<ResponseBody> clearNotifications();
+    @GET("api/v1/notifications/{id}")
+    Call<Notification> notification(@Path("id") String notificationId);
+
+    @Multipart
+    @POST("api/v1/media")
+    Call<Media> uploadMedia(@Part("file") MultipartBody.Part file);
+
+    @FormUrlEncoded
+    @POST("api/v1/statuses")
+    Call<Status> createStatus(
+            @Field("status") String text,
+            @Field("in_reply_to_id") String inReplyToId,
+            @Field("spoiler_text") String warningText,
+            @Field("visibility") String visibility,
+            @Field("sensitive") Boolean sensitive,
+            @Field("media_ids[]") List<String> mediaIds);
+    @GET("api/v1/statuses/{id}")
+    Call<Status> status(@Path("id") String statusId);
+    @GET("api/v1/statuses/{id}/context")
+    Call<StatusContext> statusContext(@Path("id") String statusId);
+    @GET("api/v1/statuses/{id}/reblogged_by")
+    Call<List<Account>> statusRebloggedBy(
+            @Path("id") String statusId,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @GET("api/v1/statuses/{id}/favourited_by")
+    Call<List<Account>> statusFavouritedBy(
+            @Path("id") String statusId,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @DELETE("api/v1/statuses/{id}")
+    Call<ResponseBody> deleteStatus(@Path("id") String statusId);
+    @POST("api/v1/statuses/{id}/reblog")
+    Call<Status> reblogStatus(@Path("id") String statusId);
+    @POST("api/v1/statuses/{id}/unreblog")
+    Call<Status> unreblogStatus(@Path("id") String statusId);
+    @POST("api/v1/statuses/{id}/favourite")
+    Call<Status> favouriteStatus(@Path("id") String statusId);
+    @POST("api/v1/statuses/{id}/unfavourite")
+    Call<Status> unfavouriteStatus(@Path("id") String statusId);
+
+    @GET("api/v1/accounts/verify_credentials")
+    Call<Account> accountVerifyCredentials();
+    @GET("api/v1/accounts/search")
+    Call<List<Account>> searchAccounts(
+            @Query("q") String q,
+            @Query("resolve") Boolean resolve,
+            @Query("limit") Integer limit);
+    @GET("api/v1/accounts/{id}")
+    Call<Account> account(@Path("id") String accountId);
+    @GET("api/v1/accounts/{id}/statuses")
+    Call<List<Status>> accountStatuses(
+            @Path("id") String accountId,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @GET("api/v1/accounts/{id}/followers")
+    Call<List<Account>> accountFollowers(
+            @Path("id") String accountId,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @GET("api/v1/accounts/{id}/following")
+    Call<List<Account>> accountFollowing(
+            @Path("id") String accountId,
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @POST("api/v1/accounts/{id}/follow")
+    Call<Relationship> followAccount(@Path("id") String accountId);
+    @POST("api/v1/accounts/{id}/unfollow")
+    Call<Relationship> unfollowAccount(@Path("id") String accountId);
+    @POST("api/v1/accounts/{id}/block")
+    Call<Relationship> blockAccount(@Path("id") String accountId);
+    @POST("api/v1/accounts/{id}/unblock")
+    Call<Relationship> unblockAccount(@Path("id") String accountId);
+    @POST("api/v1/accounts/{id}/mute")
+    Call<Relationship> muteAccount(@Path("id") String accountId);
+    @POST("api/v1/accounts/{id}/unmute")
+    Call<Relationship> unmuteAccount(@Path("id") String accountId);
+
+    @GET("api/v1/accounts/relationships")
+    Call<List<Relationship>> relationships(@Query("id[]") List<String> accountIds);
+
+    @GET("api/v1/blocks")
+    Call<List<Account>> blocks(
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+
+    @GET("api/v1/mutes")
+    Call<List<Account>> mutes(
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+
+    @GET("api/v1/favourites")
+    Call<List<Status>> favourites(
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+
+    @GET("api/v1/follow_requests")
+    Call<List<Account>> followRequests(
+            @Query("max_id") String maxId,
+            @Query("since_id") String sinceId,
+            @Query("limit") Integer limit);
+    @POST("api/v1/follow_requests/{id}/authorize")
+    Call<Relationship> authorizeFollowRequest(@Path("id") String accountId);
+    @POST("api/v1/follow_requests/{id}/reject")
+    Call<Relationship> rejectFollowRequest(@Path("id") String accountId);
+
+    @FormUrlEncoded
+    @POST("api/v1/reports")
+    Call<ResponseBody> report(@Field("account_id") String accountId, @Field("status_ids[]") List<String> statusIds, @Field("comment") String comment);
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/Notification.java b/app/src/main/java/com/keylesspalace/tusky/Notification.java
deleted file mode 100644
index f7691172..00000000
--- a/app/src/main/java/com/keylesspalace/tusky/Notification.java
+++ /dev/null
@@ -1,133 +0,0 @@
-/* Copyright 2017 Andrew Dawson
- *
- * This file is part of Tusky.
- *
- * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
- * General Public License as published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
- * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
- * Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along with Tusky. If not, see
- * <http://www.gnu.org/licenses/>. */
-
-package com.keylesspalace.tusky;
-
-import android.support.annotation.Nullable;
-
-import org.json.JSONArray;
-import org.json.JSONException;
-import org.json.JSONObject;
-
-import java.util.ArrayList;
-import java.util.List;
-
-class Notification {
-    enum Type {
-        MENTION,
-        REBLOG,
-        FAVOURITE,
-        FOLLOW,
-    }
-    private Type type;
-    private String id;
-    private String displayName;
-    private String username;
-    private String avatar;
-    private String accountId;
-    /** Which of the user's statuses has been mentioned, reblogged, or favourited. */
-    private Status status;
-
-    private Notification(Type type, String id, String displayName, String username, String avatar,
-            String accountId) {
-        this.type = type;
-        this.id = id;
-        this.displayName = displayName;
-        this.username = username;
-        this.avatar = avatar;
-        this.accountId = accountId;
-    }
-
-    Type getType() {
-        return type;
-    }
-
-    String getId() {
-        return id;
-    }
-
-    String getDisplayName() {
-        return displayName;
-    }
-
-    String getUsername() {
-        return username;
-    }
-
-    String getAvatar() {
-        return avatar;
-    }
-
-    String getAccountId() {
-        return accountId;
-    }
-
-    @Nullable Status getStatus() {
-        return status;
-    }
-
-    void setStatus(Status status) {
-        this.status = status;
-    }
-
-    private boolean hasStatusType() {
-        return type == Type.MENTION
-                || type == Type.FAVOURITE
-                || type == Type.REBLOG;
-    }
-
-    static List<Notification> parse(JSONArray array) throws JSONException {
-        List<Notification> notifications = new ArrayList<>();
-        for (int i = 0; i < array.length(); i++) {
-            JSONObject object = array.getJSONObject(i);
-            String id = object.getString("id");
-            Notification.Type type = Notification.Type.valueOf(
-                    object.getString("type").toUpperCase());
-            JSONObject account = object.getJSONObject("account");
-            String displayName = account.getString("display_name");
-            if (displayName.isEmpty()) {
-                displayName = account.getString("username");
-            }
-            String username = account.getString("acct");
-            String avatar = account.getString("avatar");
-            String accountId = account.getString("id");
-            Notification notification = new Notification(type, id, displayName, username, avatar,
-                    accountId);
-            if (notification.hasStatusType()) {
-                JSONObject statusObject = object.getJSONObject("status");
-                Status status = Status.parse(statusObject, false);
-                notification.setStatus(status);
-            }
-            notifications.add(notification);
-        }
-        return notifications;
-    }
-
-    @Override
-    public int hashCode() {
-        return id.hashCode();
-    }
-
-    @Override
-    public boolean equals(Object other) {
-        if (this.id == null) {
-            return this == other;
-        } else if (!(other instanceof Notification)) {
-            return false;
-        }
-        Notification notification = (Notification) other;
-        return notification.getId().equals(this.id);
-    }
-}
diff --git a/app/src/main/java/com/keylesspalace/tusky/NotificationsAdapter.java b/app/src/main/java/com/keylesspalace/tusky/NotificationsAdapter.java
index c8306542..5b8ea652 100644
--- a/app/src/main/java/com/keylesspalace/tusky/NotificationsAdapter.java
+++ b/app/src/main/java/com/keylesspalace/tusky/NotificationsAdapter.java
@@ -28,6 +28,8 @@ import android.view.ViewGroup;
 import android.widget.ImageView;
 import android.widget.TextView;
 
+import com.keylesspalace.tusky.entity.Notification;
+import com.keylesspalace.tusky.entity.Status;
 import com.squareup.picasso.Picasso;
 
 import java.util.ArrayList;
@@ -86,26 +88,26 @@ class NotificationsAdapter extends RecyclerView.Adapter implements AdapterItemRe
     public void onBindViewHolder(RecyclerView.ViewHolder viewHolder, int position) {
         if (position < notifications.size()) {
             Notification notification = notifications.get(position);
-            Notification.Type type = notification.getType();
+            Notification.Type type = notification.type;
             switch (type) {
                 case MENTION: {
                     StatusViewHolder holder = (StatusViewHolder) viewHolder;
-                    Status status = notification.getStatus();
+                    Status status = notification.status;
                     holder.setupWithStatus(status, statusListener);
                     break;
                 }
                 case FAVOURITE:
                 case REBLOG: {
                     StatusNotificationViewHolder holder = (StatusNotificationViewHolder) viewHolder;
-                    holder.setMessage(type, notification.getDisplayName(),
-                            notification.getStatus());
+                    holder.setMessage(type, notification.account.displayName,
+                            notification.status);
                     break;
                 }
                 case FOLLOW: {
                     FollowViewHolder holder = (FollowViewHolder) viewHolder;
-                    holder.setMessage(notification.getDisplayName(), notification.getUsername(),
-                            notification.getAvatar());
-                    holder.setupButtons(followListener, notification.getAccountId());
+                    holder.setMessage(notification.account.displayName, notification.account.username,
+                            notification.account.avatar);
+                    holder.setupButtons(followListener, notification.account.id);
                     break;
                 }
             }
@@ -126,7 +128,7 @@ class NotificationsAdapter extends RecyclerView.Adapter implements AdapterItemRe
             return VIEW_TYPE_FOOTER;
         } else {
             Notification notification = notifications.get(position);
-            switch (notification.getType()) {
+            switch (notification.type) {
                 default:
                 case MENTION: {
                     return VIEW_TYPE_MENTION;
@@ -269,7 +271,7 @@ class NotificationsAdapter extends RecyclerView.Adapter implements AdapterItemRe
             str.setSpan(new android.text.style.StyleSpan(Typeface.BOLD), 0, displayName.length(),
                     Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
             message.setText(str);
-            statusContent.setText(status.getContent());
+            statusContent.setText(status.content);
         }
     }
 }
diff --git a/app/src/main/java/com/keylesspalace/tusky/NotificationsFragment.java b/app/src/main/java/com/keylesspalace/tusky/NotificationsFragment.java
index 0aeded4d..5e0f3f20 100644
--- a/app/src/main/java/com/keylesspalace/tusky/NotificationsFragment.java
+++ b/app/src/main/java/com/keylesspalace/tusky/NotificationsFragment.java
@@ -15,6 +15,7 @@
 
 package com.keylesspalace.tusky;
 
+import android.app.NotificationManager;
 import android.content.Context;
 import android.graphics.drawable.Drawable;
 import android.os.Bundle;
@@ -32,6 +33,8 @@ import com.android.volley.AuthFailureError;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.JsonArrayRequest;
+import com.keylesspalace.tusky.entity.Notification;
+import com.keylesspalace.tusky.entity.Status;
 
 import org.json.JSONArray;
 import org.json.JSONException;
@@ -40,6 +43,9 @@ import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
 
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class NotificationsFragment extends SFragment implements
         SwipeRefreshLayout.OnRefreshListener, StatusActionListener, FooterActionListener,
         NotificationsAdapter.FollowListener {
@@ -65,6 +71,14 @@ public class NotificationsFragment extends SFragment implements
         super.onDestroy();
     }
 
+    @Override
+    public void onCreate(@Nullable Bundle savedInstanceState) {
+        NotificationManager notificationManager =
+                (NotificationManager) getActivity().getSystemService(Context.NOTIFICATION_SERVICE);
+        notificationManager.cancel(PullNotificationService.NOTIFY_ID);
+        super.onCreate(savedInstanceState);
+    }
+
     @Nullable
     @Override
     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
@@ -92,7 +106,7 @@ public class NotificationsFragment extends SFragment implements
                 NotificationsAdapter adapter = (NotificationsAdapter) view.getAdapter();
                 Notification notification = adapter.getItem(adapter.getItemCount() - 2);
                 if (notification != null) {
-                    sendFetchNotificationsRequest(notification.getId());
+                    sendFetchNotificationsRequest(notification.id);
                 } else {
                     sendFetchNotificationsRequest();
                 }
@@ -135,37 +149,19 @@ public class NotificationsFragment extends SFragment implements
     }
 
     private void sendFetchNotificationsRequest(final String fromId) {
-        String endpoint = getString(R.string.endpoint_notifications);
-        String url = "https://" + domain + endpoint;
-        if (fromId != null) {
-            url += "?max_id=" + fromId;
-        }
-        JsonArrayRequest request = new JsonArrayRequest(url,
-                new Response.Listener<JSONArray>() {
-                    @Override
-                    public void onResponse(JSONArray response) {
-                        try {
-                            List<Notification> notifications = Notification.parse(response);
-                            onFetchNotificationsSuccess(notifications, fromId);
-                        } catch (JSONException e) {
-                            onFetchNotificationsFailure(e);
-                        }
-                    }
-                }, new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onFetchNotificationsFailure(error);
-                    }
-                }) {
+        MastodonAPI api = ((BaseActivity) getActivity()).mastodonAPI;
+
+        api.notifications(fromId, null, null).enqueue(new Callback<List<Notification>>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<List<Notification>> call, retrofit2.Response<List<Notification>> response) {
+                onFetchNotificationsSuccess(response.body(), fromId);
             }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(getContext()).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<List<Notification>> call, Throwable t) {
+                onFetchNotificationsFailure((Exception) t);
+            }
+        });
     }
 
     private void sendFetchNotificationsRequest() {
@@ -174,7 +170,7 @@ public class NotificationsFragment extends SFragment implements
 
     private static boolean findNotification(List<Notification> notifications, String id) {
         for (Notification notification : notifications) {
-            if (notification.getId().equals(id)) {
+            if (notification.id.equals(id)) {
                 return true;
             }
         }
@@ -218,7 +214,7 @@ public class NotificationsFragment extends SFragment implements
     public void onLoadMore() {
         Notification notification = adapter.getItem(adapter.getItemCount() - 2);
         if (notification != null) {
-            sendFetchNotificationsRequest(notification.getId());
+            sendFetchNotificationsRequest(notification.id);
         } else {
             sendFetchNotificationsRequest();
         }
@@ -226,22 +222,22 @@ public class NotificationsFragment extends SFragment implements
 
     public void onReply(int position) {
         Notification notification = adapter.getItem(position);
-        super.reply(notification.getStatus());
+        super.reply(notification.status);
     }
 
     public void onReblog(boolean reblog, int position) {
         Notification notification = adapter.getItem(position);
-        super.reblog(notification.getStatus(), reblog, adapter, position);
+        super.reblog(notification.status, reblog, adapter, position);
     }
 
     public void onFavourite(boolean favourite, int position) {
         Notification notification = adapter.getItem(position);
-        super.favourite(notification.getStatus(), favourite, adapter, position);
+        super.favourite(notification.status, favourite, adapter, position);
     }
 
     public void onMore(View view, int position) {
         Notification notification = adapter.getItem(position);
-        super.more(notification.getStatus(), view, adapter, position);
+        super.more(notification.status, view, adapter, position);
     }
 
     public void onViewMedia(String url, Status.MediaAttachment.Type type) {
@@ -250,7 +246,7 @@ public class NotificationsFragment extends SFragment implements
 
     public void onViewThread(int position) {
         Notification notification = adapter.getItem(position);
-        super.viewThread(notification.getStatus());
+        super.viewThread(notification.status);
     }
 
     public void onViewTag(String tag) {
diff --git a/app/src/main/java/com/keylesspalace/tusky/PreferencesActivity.java b/app/src/main/java/com/keylesspalace/tusky/PreferencesActivity.java
index 53b29cea..ffc2811a 100644
--- a/app/src/main/java/com/keylesspalace/tusky/PreferencesActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/PreferencesActivity.java
@@ -22,7 +22,7 @@ import android.preference.PreferenceManager;
 import android.support.annotation.Nullable;
 import android.support.v7.app.AppCompatActivity;
 
-public class PreferencesActivity extends AppCompatActivity
+public class PreferencesActivity extends BaseActivity
         implements SharedPreferences.OnSharedPreferenceChangeListener {
     private boolean themeSwitched;
 
diff --git a/app/src/main/java/com/keylesspalace/tusky/PullNotificationService.java b/app/src/main/java/com/keylesspalace/tusky/PullNotificationService.java
index d562275a..f3be9f6d 100644
--- a/app/src/main/java/com/keylesspalace/tusky/PullNotificationService.java
+++ b/app/src/main/java/com/keylesspalace/tusky/PullNotificationService.java
@@ -26,24 +26,38 @@ import android.provider.Settings;
 import android.support.annotation.Nullable;
 import android.support.v4.app.NotificationCompat;
 import android.support.v4.app.TaskStackBuilder;
+import android.text.Spanned;
 
 import com.android.volley.AuthFailureError;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.ImageRequest;
 import com.android.volley.toolbox.JsonArrayRequest;
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.keylesspalace.tusky.entity.*;
+import com.keylesspalace.tusky.entity.Notification;
 
 import org.json.JSONArray;
 import org.json.JSONException;
 
+import java.io.IOException;
 import java.util.ArrayList;
 import java.util.Date;
 import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
 
+import okhttp3.Interceptor;
+import okhttp3.OkHttpClient;
+import okhttp3.Request;
+import retrofit2.Call;
+import retrofit2.Callback;
+import retrofit2.Retrofit;
+import retrofit2.converter.gson.GsonConverterFactory;
+
 public class PullNotificationService extends IntentService {
-    private static final int NOTIFY_ID = 6; // This is an arbitrary number.
+    static final int NOTIFY_ID = 6; // This is an arbitrary number.
     private static final String TAG = "PullNotifications"; // logging tag and Volley request tag
 
     public PullNotificationService() {
@@ -62,82 +76,80 @@ public class PullNotificationService extends IntentService {
                 getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
         String domain = preferences.getString("domain", null);
         String accessToken = preferences.getString("accessToken", null);
-        long date = preferences.getLong("lastUpdate", 0);
-        Date lastUpdate = null;
-        if (date != 0) {
-            lastUpdate = new Date(date);
-        }
-        checkNotifications(domain, accessToken, lastUpdate);
+        String lastUpdateId = preferences.getString("lastUpdateId", null);
+        checkNotifications(domain, accessToken, lastUpdateId);
     }
 
     private void checkNotifications(final String domain, final String accessToken,
-            final Date lastUpdate) {
-        String endpoint = getString(R.string.endpoint_notifications);
-        String url = "https://" + domain + endpoint;
-        JsonArrayRequest request = new JsonArrayRequest(url,
-                new Response.Listener<JSONArray>() {
+            final String lastUpdateId) {
+        OkHttpClient okHttpClient = new OkHttpClient.Builder()
+                .addInterceptor(new Interceptor() {
                     @Override
-                    public void onResponse(JSONArray response) {
-                        List<Notification> notifications;
-                        try {
-                            notifications = Notification.parse(response);
-                        } catch (JSONException e) {
-                            onCheckNotificationsFailure(e);
-                            return;
-                        }
-                        onCheckNotificationsSuccess(notifications, lastUpdate);
+                    public okhttp3.Response intercept(Chain chain) throws IOException {
+                        Request originalRequest = chain.request();
+
+                        Request.Builder builder = originalRequest.newBuilder()
+                                .header("Authorization", String.format("Bearer %s", accessToken));
+
+                        Request newRequest = builder.build();
+
+                        return chain.proceed(newRequest);
                     }
-                }, new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onCheckNotificationsFailure(error);
-                    }
-                }) {
+                })
+                .build();
+
+        Gson gson = new GsonBuilder()
+                .registerTypeAdapter(Spanned.class, new SpannedTypeAdapter())
+                .create();
+
+        Retrofit retrofit = new Retrofit.Builder()
+                .baseUrl("https://" + domain)
+                .client(okHttpClient)
+                .addConverterFactory(GsonConverterFactory.create(gson))
+                .build();
+
+        MastodonAPI api = retrofit.create(MastodonAPI.class);
+
+        api.notifications(null, lastUpdateId, null).enqueue(new Callback<List<Notification>>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<List<Notification>> call, retrofit2.Response<List<Notification>> response) {
+                onCheckNotificationsSuccess(response.body(), lastUpdateId);
             }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<List<Notification>> call, Throwable t) {
+                onCheckNotificationsFailure((Exception) t);
+            }
+        });
     }
 
-    private void onCheckNotificationsSuccess(List<Notification> notifications, Date lastUpdate) {
-        Date newest = null;
+    private void onCheckNotificationsSuccess(List<com.keylesspalace.tusky.entity.Notification> notifications, String lastUpdateId) {
         List<MentionResult> mentions = new ArrayList<>();
-        for (Notification notification : notifications) {
-            if (notification.getType() == Notification.Type.MENTION) {
-                Status status = notification.getStatus();
+
+        for (com.keylesspalace.tusky.entity.Notification notification : notifications) {
+            if (notification.type == com.keylesspalace.tusky.entity.Notification.Type.MENTION) {
+                Status status = notification.status;
+
                 if (status != null) {
-                    Date createdAt = status.getCreatedAt();
-                    if (lastUpdate == null || createdAt.after(lastUpdate)) {
-                        MentionResult mention = new MentionResult();
-                        mention.content = status.getContent().toString();
-                        mention.displayName = notification.getDisplayName();
-                        mention.avatarUrl = status.getAvatar();
-                        mentions.add(mention);
-                    }
-                    if (newest == null || createdAt.after(newest)) {
-                        newest = createdAt;
-                    }
+                    MentionResult mention = new MentionResult();
+                    mention.content = status.content.toString();
+                    mention.displayName = notification.account.displayName;
+                    mention.avatarUrl = status.account.avatar;
+                    mentions.add(mention);
                 }
             }
         }
-        long now = new Date().getTime();
-        if (mentions.size() > 0) {
+
+        if (notifications.size() > 0) {
             SharedPreferences preferences = getSharedPreferences(
                     getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
             SharedPreferences.Editor editor = preferences.edit();
-            editor.putLong("lastUpdate", now);
+            editor.putString("lastUpdateId", notifications.get(0).id);
             editor.apply();
+        }
+
+        if (mentions.size() > 0) {
             loadAvatar(mentions, mentions.get(0).avatarUrl);
-        } else if (newest != null) {
-            long hoursAgo = (now - newest.getTime()) / (60 * 60 * 1000);
-            if (hoursAgo >= 1) {
-                dismissStaleNotifications();
-            }
         }
     }
 
@@ -227,10 +239,4 @@ public class PullNotificationService extends IntentService {
                 (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
         notificationManager.notify(NOTIFY_ID, builder.build());
     }
-
-    private void dismissStaleNotifications() {
-        NotificationManager notificationManager =
-                (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
-        notificationManager.cancel(NOTIFY_ID);
-    }
 }
diff --git a/app/src/main/java/com/keylesspalace/tusky/ReportActivity.java b/app/src/main/java/com/keylesspalace/tusky/ReportActivity.java
index fb0a8ccd..3c13570a 100644
--- a/app/src/main/java/com/keylesspalace/tusky/ReportActivity.java
+++ b/app/src/main/java/com/keylesspalace/tusky/ReportActivity.java
@@ -38,16 +38,22 @@ import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.JsonArrayRequest;
 import com.android.volley.toolbox.JsonObjectRequest;
+import com.keylesspalace.tusky.entity.Status;
 
 import org.json.JSONArray;
 import org.json.JSONException;
 import org.json.JSONObject;
 
 import java.util.ArrayList;
+import java.util.Arrays;
 import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
 
+import okhttp3.ResponseBody;
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class ReportActivity extends BaseActivity {
     private static final String TAG = "ReportActivity"; // logging tag and Volley request tag
 
@@ -141,45 +147,22 @@ public class ReportActivity extends BaseActivity {
 
     private void sendReport(final String accountId, final String[] statusIds,
             final String comment) {
-        JSONObject parameters = new JSONObject();
-        try {
-            parameters.put("account_id", accountId);
-            parameters.put("status_ids", makeStringArrayCompat(statusIds));
-            parameters.put("comment", comment);
-        } catch (JSONException e) {
-            Log.e(TAG, "Not all the report parameters have been properly set. " + e.getMessage());
-            onSendFailure(accountId, statusIds, comment);
-            return;
-        }
-        String endpoint = getString(R.string.endpoint_reports);
-        String url = "https://" + domain + endpoint;
-        JsonObjectRequest request = new JsonObjectRequest(Request.Method.POST, url, parameters,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        onSendSuccess();
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onSendFailure(accountId, statusIds, comment);
-                    }
-                }) {
+        mastodonAPI.report(accountId, Arrays.asList(statusIds), comment).enqueue(new Callback<ResponseBody>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<ResponseBody> call, retrofit2.Response<ResponseBody> response) {
+                onSendSuccess();
             }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<ResponseBody> call, Throwable t) {
+                onSendFailure(accountId, statusIds, comment);
+            }
+        });
     }
 
     private void onSendSuccess() {
-        Toast.makeText(this, getString(R.string.confirmation_reported), Toast.LENGTH_SHORT)
-                .show();
+        Snackbar bar = Snackbar.make(anyView, getString(R.string.confirmation_reported), Snackbar.LENGTH_SHORT);
+        bar.show();
         finish();
     }
 
@@ -197,46 +180,26 @@ public class ReportActivity extends BaseActivity {
     }
 
     private void fetchRecentStatuses(String accountId) {
-        String endpoint = String.format(getString(R.string.endpoint_statuses), accountId);
-        String url = "https://" + domain + endpoint;
-        JsonArrayRequest request = new JsonArrayRequest(url,
-                new Response.Listener<JSONArray>() {
-                    @Override
-                    public void onResponse(JSONArray response) {
-                        List<Status> statusList;
-                        try {
-                            statusList = Status.parse(response);
-                        } catch (JSONException e) {
-                            onFetchStatusesFailure(e);
-                            return;
-                        }
-                        // Add all the statuses except reblogs.
-                        List<ReportAdapter.ReportStatus> itemList = new ArrayList<>();
-                        for (Status status : statusList) {
-                            if (status.getRebloggedByDisplayName() == null) {
-                                ReportAdapter.ReportStatus item = new ReportAdapter.ReportStatus(
-                                        status.getId(), status.getContent(), false);
-                                itemList.add(item);
-                            }
-                        }
-                        adapter.addItems(itemList);
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onFetchStatusesFailure(error);
-                    }
-                }) {
+        mastodonAPI.accountStatuses(accountId, null, null, null).enqueue(new Callback<List<Status>>() {
             @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
+            public void onResponse(Call<List<Status>> call, retrofit2.Response<List<Status>> response) {
+                List<Status> statusList = response.body();
+                List<ReportAdapter.ReportStatus> itemList = new ArrayList<>();
+                for (Status status : statusList) {
+                    if (status.reblog != null) {
+                        ReportAdapter.ReportStatus item = new ReportAdapter.ReportStatus(
+                                status.id, status.content, false);
+                        itemList.add(item);
+                    }
+                }
+                adapter.addItems(itemList);
             }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(this).addToRequestQueue(request);
+
+            @Override
+            public void onFailure(Call<List<Status>> call, Throwable t) {
+                onFetchStatusesFailure((Exception) t);
+            }
+        });
     }
 
     private void onFetchStatusesFailure(Exception exception) {
diff --git a/app/src/main/java/com/keylesspalace/tusky/SFragment.java b/app/src/main/java/com/keylesspalace/tusky/SFragment.java
index ba1fea6e..6b7845f5 100644
--- a/app/src/main/java/com/keylesspalace/tusky/SFragment.java
+++ b/app/src/main/java/com/keylesspalace/tusky/SFragment.java
@@ -33,6 +33,8 @@ import com.android.volley.Request;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.JsonObjectRequest;
+import com.keylesspalace.tusky.entity.Relationship;
+import com.keylesspalace.tusky.entity.Status;
 
 import org.json.JSONObject;
 
@@ -41,6 +43,10 @@ import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
 
+import okhttp3.ResponseBody;
+import retrofit2.Call;
+import retrofit2.Callback;
+
 /* Note from Andrew on Jan. 22, 2017: This class is a design problem for me, so I left it with an
  * awkward name. TimelineFragment and NotificationFragment have significant overlap but the nature
  * of that is complicated by how they're coupled with Status and Notification and the corresponding
@@ -54,6 +60,7 @@ public class SFragment extends Fragment {
     protected String accessToken;
     protected String loggedInAccountId;
     protected String loggedInUsername;
+    private MastodonAPI api;
 
     @Override
     public void onCreate(@Nullable Bundle savedInstanceState) {
@@ -65,6 +72,7 @@ public class SFragment extends Fragment {
         accessToken = preferences.getString("accessToken", null);
         loggedInAccountId = preferences.getString("loggedInAccountId", null);
         loggedInUsername = preferences.getString("loggedInAccountUsername", null);
+        api = ((BaseActivity) getActivity()).mastodonAPI;
     }
 
     @Override
@@ -73,117 +81,119 @@ public class SFragment extends Fragment {
         super.onDestroy();
     }
 
-    protected void sendRequest(
-            int method, String endpoint, JSONObject parameters,
-            @Nullable Response.Listener<JSONObject> responseListener,
-            @Nullable Response.ErrorListener errorListener) {
-        if (responseListener == null) {
-            // Use a dummy listener if one wasn't specified so the request can be constructed.
-            responseListener = new Response.Listener<JSONObject>() {
-                @Override
-                public void onResponse(JSONObject response) {}
-            };
-        }
-        if (errorListener == null) {
-            errorListener = new Response.ErrorListener() {
-                @Override
-                public void onErrorResponse(VolleyError error) {
-                    Log.e(TAG, "Request Failed: " + error.getMessage());
-                }
-            };
-        }
-        String url = "https://" + domain + endpoint;
-        JsonObjectRequest request = new JsonObjectRequest(
-                method, url, parameters, responseListener, errorListener) {
-            @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
-            }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(getContext()).addToRequestQueue(request);
-    }
-
-    protected void postRequest(String endpoint) {
-        sendRequest(Request.Method.POST, endpoint, null, null, null);
-    }
-
     protected void reply(Status status) {
-        String inReplyToId = status.getId();
-        Status.Mention[] mentions = status.getMentions();
+        String inReplyToId = status.getActionableId();
+        Status.Mention[] mentions = status.mentions;
         List<String> mentionedUsernames = new ArrayList<>();
         for (Status.Mention mention : mentions) {
-            mentionedUsernames.add(mention.getUsername());
+            mentionedUsernames.add(mention.username);
         }
-        mentionedUsernames.add(status.getUsername());
+        mentionedUsernames.add(status.account.username);
         mentionedUsernames.remove(loggedInUsername);
         Intent intent = new Intent(getContext(), ComposeActivity.class);
         intent.putExtra("in_reply_to_id", inReplyToId);
-        intent.putExtra("reply_visibility", status.getVisibility().toString().toLowerCase());
+        intent.putExtra("reply_visibility", status.visibility.toString().toLowerCase());
         intent.putExtra("mentioned_usernames", mentionedUsernames.toArray(new String[0]));
         startActivity(intent);
     }
 
     protected void reblog(final Status status, final boolean reblog,
             final RecyclerView.Adapter adapter, final int position) {
-        String id = status.getId();
-        String endpoint;
+        String id = status.getActionableId();
+
+        Callback<Status> cb = new Callback<Status>() {
+            @Override
+            public void onResponse(Call<Status> call, retrofit2.Response<Status> response) {
+                status.reblogged = reblog;
+                adapter.notifyItemChanged(position);
+            }
+
+            @Override
+            public void onFailure(Call<Status> call, Throwable t) {
+
+            }
+        };
+
         if (reblog) {
-            endpoint = String.format(getString(R.string.endpoint_reblog), id);
+            api.reblogStatus(id).enqueue(cb);
         } else {
-            endpoint = String.format(getString(R.string.endpoint_unreblog), id);
+            api.unreblogStatus(id).enqueue(cb);
         }
-        sendRequest(Request.Method.POST, endpoint, null,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        status.setReblogged(reblog);
-                        adapter.notifyItemChanged(position);
-                    }
-                }, null);
     }
 
     protected void favourite(final Status status, final boolean favourite,
             final RecyclerView.Adapter adapter, final int position) {
-        String id = status.getId();
-        String endpoint;
-        if (favourite) {
-            endpoint = String.format(getString(R.string.endpoint_favourite), id);
-        } else {
-            endpoint = String.format(getString(R.string.endpoint_unfavourite), id);
-        }
-        sendRequest(Request.Method.POST, endpoint, null, new Response.Listener<JSONObject>() {
+        String id = status.getActionableId();
+
+        Callback<Status> cb = new Callback<Status>() {
             @Override
-            public void onResponse(JSONObject response) {
-                status.setFavourited(favourite);
+            public void onResponse(Call<Status> call, retrofit2.Response<Status> response) {
+                status.favourited = favourite;
                 adapter.notifyItemChanged(position);
             }
-        }, null);
+
+            @Override
+            public void onFailure(Call<Status> call, Throwable t) {
+
+            }
+        };
+
+        if (favourite) {
+            api.favouriteStatus(id).enqueue(cb);
+        } else {
+            api.unfavouriteStatus(id).enqueue(cb);
+        }
     }
 
     protected void follow(String id) {
-        String endpoint = String.format(getString(R.string.endpoint_follow), id);
-        postRequest(endpoint);
+        api.followAccount(id).enqueue(new Callback<Relationship>() {
+            @Override
+            public void onResponse(Call<Relationship> call, retrofit2.Response<Relationship> response) {
+
+            }
+
+            @Override
+            public void onFailure(Call<Relationship> call, Throwable t) {
+
+            }
+        });
     }
 
     private void block(String id) {
-        String endpoint = String.format(getString(R.string.endpoint_block), id);
-        postRequest(endpoint);
+        api.blockAccount(id).enqueue(new Callback<Relationship>() {
+            @Override
+            public void onResponse(Call<Relationship> call, retrofit2.Response<Relationship> response) {
+
+            }
+
+            @Override
+            public void onFailure(Call<Relationship> call, Throwable t) {
+
+            }
+        });
     }
 
     private void delete(String id) {
-        String endpoint = String.format(getString(R.string.endpoint_delete), id);
-        sendRequest(Request.Method.DELETE, endpoint, null, null, null);
+        api.deleteStatus(id).enqueue(new Callback<ResponseBody>() {
+            @Override
+            public void onResponse(Call<ResponseBody> call, retrofit2.Response<ResponseBody> response) {
+
+            }
+
+            @Override
+            public void onFailure(Call<ResponseBody> call, Throwable t) {
+
+            }
+        });
     }
 
     protected void more(Status status, View view, final AdapterItemRemover adapter,
             final int position) {
-        final String id = status.getId();
-        final String accountId = status.getAccountId();
-        final String accountUsename = status.getUsername();
-        final Spanned content = status.getContent();
+        final String id = status.getActionableId();
+        final String accountId = status.getActionableStatus().account.id;
+        final String accountUsename = status.getActionableStatus().account.username;
+        final Spanned content = status.getActionableStatus().content;
+        final String statusUrl = status.getActionableStatus().url;
         PopupMenu popup = new PopupMenu(getContext(), view);
         // Give a different menu depending on whether this is the user's own toot or not.
         if (loggedInAccountId == null || !loggedInAccountId.equals(accountId)) {
@@ -196,8 +206,12 @@ public class SFragment extends Fragment {
                     @Override
                     public boolean onMenuItemClick(MenuItem item) {
                         switch (item.getItemId()) {
-                            case R.id.status_follow: {
-                                follow(accountId);
+                            case R.id.status_share: {
+                                Intent sendIntent = new Intent();
+                                sendIntent.setAction(Intent.ACTION_SEND);
+                                sendIntent.putExtra(Intent.EXTRA_TEXT, statusUrl);
+                                sendIntent.setType("text/plain");
+                                startActivity(Intent.createChooser(sendIntent, getResources().getText(R.string.send_status_to)));
                                 return true;
                             }
                             case R.id.status_block: {
@@ -234,19 +248,17 @@ public class SFragment extends Fragment {
     protected void viewMedia(String url, Status.MediaAttachment.Type type) {
         switch (type) {
             case IMAGE: {
-                Fragment newFragment;
-                if (fileExtensionMatches(url, "gif")) {
-                    newFragment = ViewGifFragment.newInstance(url);
-                } else {
-                    newFragment = ViewMediaFragment.newInstance(url);
-                }
+                Fragment newFragment = ViewMediaFragment.newInstance(url);
+
                 FragmentManager manager = getFragmentManager();
                 manager.beginTransaction()
+                        .setCustomAnimations(R.anim.zoom_in, R.anim.zoom_out, R.anim.zoom_in, R.anim.zoom_out)
                         .add(R.id.overlay_fragment_container, newFragment)
                         .addToBackStack(null)
                         .commit();
                 break;
             }
+            case GIFV:
             case VIDEO: {
                 Intent intent = new Intent(getContext(), ViewVideoActivity.class);
                 intent.putExtra("url", url);
@@ -264,7 +276,7 @@ public class SFragment extends Fragment {
 
     protected void viewThread(Status status) {
         Intent intent = new Intent(getContext(), ViewThreadActivity.class);
-        intent.putExtra("id", status.getId());
+        intent.putExtra("id", status.id);
         startActivity(intent);
     }
 
diff --git a/app/src/main/java/com/keylesspalace/tusky/SpannedTypeAdapter.java b/app/src/main/java/com/keylesspalace/tusky/SpannedTypeAdapter.java
new file mode 100644
index 00000000..d1e2adda
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/SpannedTypeAdapter.java
@@ -0,0 +1,18 @@
+package com.keylesspalace.tusky;
+
+import android.text.Spanned;
+
+import com.emojione.Emojione;
+import com.google.gson.JsonDeserializationContext;
+import com.google.gson.JsonDeserializer;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonParseException;
+
+import java.lang.reflect.Type;
+
+class SpannedTypeAdapter implements JsonDeserializer<Spanned> {
+    @Override
+    public Spanned deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
+        return HtmlUtils.fromHtml(Emojione.shortnameToUnicode(json.getAsString(), false));
+    }
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/Status.java b/app/src/main/java/com/keylesspalace/tusky/Status.java
deleted file mode 100644
index 9ecb2d12..00000000
--- a/app/src/main/java/com/keylesspalace/tusky/Status.java
+++ /dev/null
@@ -1,357 +0,0 @@
-/* Copyright 2017 Andrew Dawson
- *
- * This file is part of Tusky.
- *
- * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
- * General Public License as published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
- * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
- * Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along with Tusky. If not, see
- * <http://www.gnu.org/licenses/>. */
-
-package com.keylesspalace.tusky;
-
-import android.support.annotation.Nullable;
-import android.text.Spanned;
-
-import com.emojione.Emojione;
-
-import org.json.JSONArray;
-import org.json.JSONException;
-import org.json.JSONObject;
-
-import java.text.ParseException;
-import java.text.SimpleDateFormat;
-import java.util.ArrayList;
-import java.util.Date;
-import java.util.List;
-
-public class Status {
-    enum Visibility {
-        PUBLIC,
-        UNLISTED,
-        PRIVATE,
-    }
-
-    private String id;
-    private String accountId;
-    private String displayName;
-    /** the username with the remote domain appended, like @domain.name, if it's a remote account */
-    private String username;
-    /** the main text of the status, marked up with style for links & mentions, etc */
-    private Spanned content;
-    /** the fully-qualified url of the avatar image */
-    private String avatar;
-    private String rebloggedByDisplayName;
-    /** when the status was initially created */
-    private Date createdAt;
-    /** whether the authenticated user has reblogged this status */
-    private boolean reblogged;
-    /** whether the authenticated user has favourited this status */
-    private boolean favourited;
-    private boolean sensitive;
-    private String spoilerText;
-    private Visibility visibility;
-    private MediaAttachment[] attachments;
-    private Mention[] mentions;
-
-    static final int MAX_MEDIA_ATTACHMENTS = 4;
-
-    public Status(String id, String accountId, String displayName, String username, Spanned content,
-                  String avatar, Date createdAt, boolean reblogged, boolean favourited,
-                  String visibility) {
-        this.id = id;
-        this.accountId = accountId;
-        this.displayName = displayName;
-        this.username = username;
-        this.content = content;
-        this.avatar = avatar;
-        this.createdAt = createdAt;
-        this.reblogged = reblogged;
-        this.favourited = favourited;
-        this.spoilerText = "";
-        this.visibility = Visibility.valueOf(visibility.toUpperCase());
-        this.attachments = new MediaAttachment[0];
-        this.mentions = new Mention[0];
-    }
-
-    String getId() {
-        return id;
-    }
-
-    String getAccountId() {
-        return accountId;
-    }
-
-    String getDisplayName() {
-        return displayName;
-    }
-
-    String getUsername() {
-        return username;
-    }
-
-    Spanned getContent() {
-        return content;
-    }
-
-    String getAvatar() {
-        return avatar;
-    }
-
-    Date getCreatedAt() {
-        return createdAt;
-    }
-
-    String getRebloggedByDisplayName() {
-        return rebloggedByDisplayName;
-    }
-
-    boolean getReblogged() {
-        return reblogged;
-    }
-
-    boolean getFavourited() {
-        return favourited;
-    }
-
-    boolean getSensitive() {
-        return sensitive;
-    }
-
-    String getSpoilerText() {
-        return spoilerText;
-    }
-
-    Visibility getVisibility() {
-        return visibility;
-    }
-
-    MediaAttachment[] getAttachments() {
-        return attachments;
-    }
-
-    Mention[] getMentions() {
-        return mentions;
-    }
-
-    private void setRebloggedByDisplayName(String name) {
-        rebloggedByDisplayName = name;
-    }
-
-    void setReblogged(boolean reblogged) {
-        this.reblogged = reblogged;
-    }
-
-    void setFavourited(boolean favourited) {
-        this.favourited = favourited;
-    }
-
-    private void setSpoilerText(String spoilerText) {
-        this.spoilerText = spoilerText;
-    }
-
-    private void setMentions(Mention[] mentions) {
-        this.mentions = mentions;
-    }
-
-    private void setAttachments(MediaAttachment[] attachments, boolean sensitive) {
-        this.attachments = attachments;
-        this.sensitive = sensitive;
-    }
-
-    @Override
-    public int hashCode() {
-        return id.hashCode();
-    }
-
-    @Override
-    public boolean equals(Object other) {
-        if (this.id == null) {
-            return this == other;
-        } else if (!(other instanceof Status)) {
-            return false;
-        }
-        Status status = (Status) other;
-        return status.id.equals(this.id);
-    }
-
-    @SuppressWarnings("SimpleDateFormat") // UTC needs to not specify a Locale
-    @Nullable
-    private static Date parseDate(String dateTime) {
-        Date date;
-        String s = dateTime.replace("Z", "+00:00");
-        SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSZ");
-        try {
-            date = format.parse(s);
-        } catch (ParseException e) {
-            e.printStackTrace();
-            return null;
-        }
-        return date;
-    }
-
-    private static MediaAttachment.Type parseMediaType(@Nullable String type) {
-        if (type == null) {
-            return MediaAttachment.Type.UNKNOWN;
-        }
-        switch (type.toUpperCase()) {
-            case "IMAGE": return MediaAttachment.Type.IMAGE;
-            case "GIFV":
-            case "VIDEO": return MediaAttachment.Type.VIDEO;
-            default: return MediaAttachment.Type.UNKNOWN;
-        }
-    }
-
-    public static Status parse(JSONObject object, boolean isReblog) throws JSONException {
-        String id = object.getString("id");
-        String content = object.getString("content");
-        Date createdAt = parseDate(object.getString("created_at"));
-        boolean reblogged = object.optBoolean("reblogged");
-        boolean favourited = object.optBoolean("favourited");
-        String spoilerText = object.getString("spoiler_text");
-        boolean sensitive = object.optBoolean("sensitive");
-        String visibility = object.getString("visibility");
-
-        JSONObject account = object.getJSONObject("account");
-        String accountId = account.getString("id");
-        String displayName = account.getString("display_name");
-        if (displayName.isEmpty()) {
-            displayName = account.getString("username");
-        }
-        String username = account.getString("acct");
-        String avatarUrl = account.getString("avatar");
-        String avatar;
-        if (!avatarUrl.equals("/avatars/original/missing.png")) {
-            avatar = avatarUrl;
-        } else {
-            avatar = "";
-        }
-
-        JSONArray mentionsArray = object.getJSONArray("mentions");
-        Mention[] mentions = null;
-        if (mentionsArray != null) {
-            int n = mentionsArray.length();
-            mentions = new Mention[n];
-            for (int i = 0; i < n; i++) {
-                JSONObject mention = mentionsArray.getJSONObject(i);
-                String url = mention.getString("url");
-                String mentionedUsername = mention.getString("acct");
-                String mentionedAccountId = mention.getString("id");
-                mentions[i] = new Mention(url, mentionedUsername, mentionedAccountId);
-            }
-        }
-
-        JSONArray mediaAttachments = object.getJSONArray("media_attachments");
-        MediaAttachment[] attachments = null;
-        if (mediaAttachments != null) {
-            int n = mediaAttachments.length();
-            attachments = new MediaAttachment[n];
-            for (int i = 0; i < n; i++) {
-                JSONObject attachment = mediaAttachments.getJSONObject(i);
-                String url = attachment.getString("url");
-                String previewUrl = attachment.getString("preview_url");
-                String type = attachment.getString("type");
-                attachments[i] = new MediaAttachment(url, previewUrl, parseMediaType(type));
-            }
-        }
-
-        Status reblog = null;
-        /* This case shouldn't be hit after the first recursion at all. But if this method is
-         * passed unusual data this check will prevent extra recursion */
-        if (!isReblog) {
-            JSONObject reblogObject = object.optJSONObject("reblog");
-            if (reblogObject != null) {
-                reblog = parse(reblogObject, true);
-            }
-        }
-
-        Status status;
-        if (reblog != null) {
-            status = reblog;
-            status.setRebloggedByDisplayName(displayName);
-        } else {
-            Spanned contentPlus = HtmlUtils.fromHtml(Emojione.shortnameToUnicode(content, false));
-            status = new Status(
-                    id, accountId, displayName, username, contentPlus, avatar, createdAt,
-                    reblogged, favourited, visibility);
-            if (mentions != null) {
-                status.setMentions(mentions);
-            }
-            if (attachments != null) {
-                status.setAttachments(attachments, sensitive);
-            }
-            if (!spoilerText.isEmpty()) {
-                status.setSpoilerText(spoilerText);
-            }
-        }
-        return status;
-    }
-
-    public static List<Status> parse(JSONArray array) throws JSONException {
-        List<Status> statuses = new ArrayList<>();
-        for (int i = 0; i < array.length(); i++) {
-            JSONObject object = array.getJSONObject(i);
-            statuses.add(parse(object, false));
-        }
-        return statuses;
-    }
-
-    static class MediaAttachment {
-        enum Type {
-            IMAGE,
-            VIDEO,
-            UNKNOWN,
-        }
-
-        private String url;
-        private String previewUrl;
-        private Type type;
-
-        MediaAttachment(String url, String previewUrl, Type type) {
-            this.url = url;
-            this.previewUrl = previewUrl;
-            this.type = type;
-        }
-
-        String getUrl() {
-            return url;
-        }
-
-        String getPreviewUrl() {
-            return previewUrl;
-        }
-
-        Type getType() {
-            return type;
-        }
-    }
-
-    static class Mention {
-        private String url;
-        private String username;
-        private String id;
-
-        Mention(String url, String username, String id) {
-            this.url = url;
-            this.username = username;
-            this.id = id;
-        }
-
-        String getUrl() {
-            return url;
-        }
-
-        String getUsername() {
-            return username;
-        }
-
-        String getId() {
-            return id;
-        }
-    }
-}
diff --git a/app/src/main/java/com/keylesspalace/tusky/StatusActionListener.java b/app/src/main/java/com/keylesspalace/tusky/StatusActionListener.java
index abf0a5b6..548bde06 100644
--- a/app/src/main/java/com/keylesspalace/tusky/StatusActionListener.java
+++ b/app/src/main/java/com/keylesspalace/tusky/StatusActionListener.java
@@ -17,6 +17,8 @@ package com.keylesspalace.tusky;
 
 import android.view.View;
 
+import com.keylesspalace.tusky.entity.Status;
+
 interface StatusActionListener {
     void onReply(int position);
     void onReblog(final boolean reblog, final int position);
diff --git a/app/src/main/java/com/keylesspalace/tusky/StatusViewHolder.java b/app/src/main/java/com/keylesspalace/tusky/StatusViewHolder.java
index 09521551..ac346b98 100644
--- a/app/src/main/java/com/keylesspalace/tusky/StatusViewHolder.java
+++ b/app/src/main/java/com/keylesspalace/tusky/StatusViewHolder.java
@@ -30,6 +30,7 @@ import android.widget.ImageView;
 import android.widget.TextView;
 import android.widget.ToggleButton;
 
+import com.keylesspalace.tusky.entity.Status;
 import com.squareup.picasso.Picasso;
 import com.varunest.sparkbutton.SparkButton;
 import com.varunest.sparkbutton.SparkEventListener;
@@ -124,8 +125,8 @@ class StatusViewHolder extends RecyclerView.ViewHolder {
                 final String accountUsername = text.subSequence(1, text.length()).toString();
                 String id = null;
                 for (Status.Mention mention: mentions) {
-                    if (mention.getUsername().equals(accountUsername)) {
-                        id = mention.getId();
+                    if (mention.username.equals(accountUsername)) {
+                        id = mention.id;
                     }
                 }
                 if (id != null) {
@@ -227,7 +228,7 @@ class StatusViewHolder extends RecyclerView.ViewHolder {
         final int n = Math.min(attachments.length, Status.MAX_MEDIA_ATTACHMENTS);
 
         for (int i = 0; i < n; i++) {
-            String previewUrl = attachments[i].getPreviewUrl();
+            String previewUrl = attachments[i].previewUrl;
 
             previews[i].setVisibility(View.VISIBLE);
 
@@ -236,8 +237,8 @@ class StatusViewHolder extends RecyclerView.ViewHolder {
                 .placeholder(mediaPreviewUnloadedId)
                 .into(previews[i]);
 
-            final String url = attachments[i].getUrl();
-            final Status.MediaAttachment.Type type = attachments[i].getType();
+            final String url = attachments[i].url;
+            final Status.MediaAttachment.Type type = attachments[i].type;
             previews[i].setOnClickListener(new View.OnClickListener() {
                 @Override
                 public void onClick(View v) {
@@ -339,33 +340,35 @@ class StatusViewHolder extends RecyclerView.ViewHolder {
     }
 
     void setupWithStatus(Status status, StatusActionListener listener) {
-        setDisplayName(status.getDisplayName());
-        setUsername(status.getUsername());
-        setCreatedAt(status.getCreatedAt());
-        setContent(status.getContent(), status.getMentions(), listener);
-        setAvatar(status.getAvatar());
-        setReblogged(status.getReblogged());
-        setFavourited(status.getFavourited());
-        String rebloggedByDisplayName = status.getRebloggedByDisplayName();
-        if (rebloggedByDisplayName == null) {
+        Status realStatus = status.getActionableStatus();
+
+        setDisplayName(realStatus.account.displayName);
+        setUsername(realStatus.account.username);
+        setCreatedAt(realStatus.createdAt);
+        setContent(realStatus.content, realStatus.mentions, listener);
+        setAvatar(realStatus.account.avatar);
+        setReblogged(realStatus.reblogged);
+        setFavourited(realStatus.favourited);
+        String rebloggedByDisplayName = status.account.displayName;
+        if (status.reblog == null) {
             hideRebloggedByDisplayName();
         } else {
             setRebloggedByDisplayName(rebloggedByDisplayName);
         }
-        Status.MediaAttachment[] attachments = status.getAttachments();
-        boolean sensitive = status.getSensitive();
+        Status.MediaAttachment[] attachments = realStatus.attachments;
+        boolean sensitive = realStatus.sensitive;
         setMediaPreviews(attachments, sensitive, listener);
         /* A status without attachments is sometimes still marked sensitive, so it's necessary to
          * check both whether there are any attachments and if it's marked sensitive. */
         if (!sensitive || attachments.length == 0) {
             hideSensitiveMediaWarning();
         }
-        setupButtons(listener, status.getAccountId());
-        setRebloggingEnabled(status.getVisibility() != Status.Visibility.PRIVATE);
-        if (status.getSpoilerText().isEmpty()) {
+        setupButtons(listener, realStatus.account.id);
+        setRebloggingEnabled(realStatus.visibility != Status.Visibility.PRIVATE);
+        if (realStatus.spoilerText.isEmpty()) {
             hideSpoilerText();
         } else {
-            setSpoilerText(status.getSpoilerText());
+            setSpoilerText(realStatus.spoilerText);
         }
     }
 }
\ No newline at end of file
diff --git a/app/src/main/java/com/keylesspalace/tusky/ThreadAdapter.java b/app/src/main/java/com/keylesspalace/tusky/ThreadAdapter.java
index 3e9ed57b..4cc2dc48 100644
--- a/app/src/main/java/com/keylesspalace/tusky/ThreadAdapter.java
+++ b/app/src/main/java/com/keylesspalace/tusky/ThreadAdapter.java
@@ -20,6 +20,8 @@ import android.view.LayoutInflater;
 import android.view.View;
 import android.view.ViewGroup;
 
+import com.keylesspalace.tusky.entity.Status;
+
 import java.util.ArrayList;
 import java.util.List;
 
diff --git a/app/src/main/java/com/keylesspalace/tusky/TimelineAdapter.java b/app/src/main/java/com/keylesspalace/tusky/TimelineAdapter.java
index 792afd40..428413cf 100644
--- a/app/src/main/java/com/keylesspalace/tusky/TimelineAdapter.java
+++ b/app/src/main/java/com/keylesspalace/tusky/TimelineAdapter.java
@@ -21,6 +21,8 @@ import android.view.LayoutInflater;
 import android.view.View;
 import android.view.ViewGroup;
 
+import com.keylesspalace.tusky.entity.Status;
+
 import java.util.ArrayList;
 import java.util.List;
 
diff --git a/app/src/main/java/com/keylesspalace/tusky/TimelineFragment.java b/app/src/main/java/com/keylesspalace/tusky/TimelineFragment.java
index 0f41e534..907d6c90 100644
--- a/app/src/main/java/com/keylesspalace/tusky/TimelineFragment.java
+++ b/app/src/main/java/com/keylesspalace/tusky/TimelineFragment.java
@@ -31,6 +31,7 @@ import com.android.volley.AuthFailureError;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
 import com.android.volley.toolbox.JsonArrayRequest;
+import com.keylesspalace.tusky.entity.Status;
 
 import org.json.JSONArray;
 import org.json.JSONException;
@@ -39,6 +40,9 @@ import java.util.HashMap;
 import java.util.List;
 import java.util.Map;
 
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class TimelineFragment extends SFragment implements
         SwipeRefreshLayout.OnRefreshListener, StatusActionListener, FooterActionListener {
     private static final String TAG = "Timeline"; // logging tag and Volley request tag
@@ -117,7 +121,7 @@ public class TimelineFragment extends SFragment implements
                 TimelineAdapter adapter = (TimelineAdapter) view.getAdapter();
                 Status status = adapter.getItem(adapter.getItemCount() - 2);
                 if (status != null) {
-                    sendFetchTimelineRequest(status.getId());
+                    sendFetchTimelineRequest(status.id);
                 } else {
                     sendFetchTimelineRequest();
                 }
@@ -168,67 +172,43 @@ public class TimelineFragment extends SFragment implements
     }
 
     private void sendFetchTimelineRequest(final String fromId) {
-        String endpoint;
+        MastodonAPI api = ((BaseActivity) getActivity()).mastodonAPI;
+
+        Callback<List<Status>> cb = new Callback<List<Status>>() {
+            @Override
+            public void onResponse(Call<List<Status>> call, retrofit2.Response<List<Status>> response) {
+                onFetchTimelineSuccess(response.body(), fromId);
+            }
+
+            @Override
+            public void onFailure(Call<List<Status>> call, Throwable t) {
+                onFetchTimelineFailure((Exception) t);
+            }
+        };
+
         switch (kind) {
             default:
             case HOME: {
-                endpoint = getString(R.string.endpoint_timelines_home);
-                break;
-            }
-            case MENTIONS: {
-                endpoint = getString(R.string.endpoint_timelines_mentions);
+                api.homeTimeline(fromId, null, null).enqueue(cb);
                 break;
             }
             case PUBLIC: {
-                endpoint = getString(R.string.endpoint_timelines_public);
+                api.publicTimeline(null, fromId, null, null).enqueue(cb);
                 break;
             }
             case TAG: {
-                endpoint = String.format(getString(R.string.endpoint_timelines_tag), hashtagOrId);
+                api.hashtagTimeline(hashtagOrId, null, fromId, null, null).enqueue(cb);
                 break;
             }
             case USER: {
-                endpoint = String.format(getString(R.string.endpoint_statuses), hashtagOrId);
+                api.accountStatuses(hashtagOrId, fromId, null, null).enqueue(cb);
                 break;
             }
             case FAVOURITES: {
-                endpoint = getString(R.string.endpoint_favourites);
+                api.favourites(fromId, null, null).enqueue(cb);
                 break;
             }
         }
-        String url = "https://" + domain + endpoint;
-        if (fromId != null) {
-            url += "?max_id=" + fromId;
-        }
-        JsonArrayRequest request = new JsonArrayRequest(url,
-                new Response.Listener<JSONArray>() {
-                    @Override
-                    public void onResponse(JSONArray response) {
-                        List<Status> statuses = null;
-                        try {
-                            statuses = Status.parse(response);
-                        } catch (JSONException e) {
-                            onFetchTimelineFailure(e);
-                        }
-                        if (statuses != null) {
-                            onFetchTimelineSuccess(statuses, fromId);
-                        }
-                    }
-                }, new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onFetchTimelineFailure(error);
-                    }
-        }) {
-            @Override
-            public Map<String, String> getHeaders() throws AuthFailureError {
-                Map<String, String> headers = new HashMap<>();
-                headers.put("Authorization", "Bearer " + accessToken);
-                return headers;
-            }
-        };
-        request.setTag(TAG);
-        VolleySingleton.getInstance(getContext()).addToRequestQueue(request);
     }
 
     private void sendFetchTimelineRequest() {
@@ -237,7 +217,7 @@ public class TimelineFragment extends SFragment implements
 
     private static boolean findStatus(List<Status> statuses, String id) {
         for (Status status : statuses) {
-            if (status.getId().equals(id)) {
+            if (status.id.equals(id)) {
                 return true;
             }
         }
@@ -281,7 +261,7 @@ public class TimelineFragment extends SFragment implements
     public void onLoadMore() {
         Status status = adapter.getItem(adapter.getItemCount() - 2);
         if (status != null) {
-            sendFetchTimelineRequest(status.getId());
+            sendFetchTimelineRequest(status.id);
         } else {
             sendFetchTimelineRequest();
         }
diff --git a/app/src/main/java/com/keylesspalace/tusky/ViewGifFragment.java b/app/src/main/java/com/keylesspalace/tusky/ViewGifFragment.java
deleted file mode 100644
index b75ca396..00000000
--- a/app/src/main/java/com/keylesspalace/tusky/ViewGifFragment.java
+++ /dev/null
@@ -1,58 +0,0 @@
-/* Copyright 2017 Andrew Dawson
- *
- * This file is part of Tusky.
- *
- * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
- * General Public License as published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
- * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
- * Public License for more details.
- *
- * You should have received a copy of the GNU General Public License along with Tusky. If not, see
- * <http://www.gnu.org/licenses/>. */
-
-package com.keylesspalace.tusky;
-
-import android.os.Bundle;
-import android.support.annotation.Nullable;
-import android.support.v4.app.Fragment;
-import android.view.LayoutInflater;
-import android.view.View;
-import android.view.ViewGroup;
-import android.webkit.WebView;
-
-public class ViewGifFragment extends Fragment {
-    public static ViewGifFragment newInstance(String url) {
-        Bundle arguments = new Bundle();
-        ViewGifFragment fragment = new ViewGifFragment();
-        arguments.putString("url", url);
-        fragment.setArguments(arguments);
-        return fragment;
-    }
-
-    @Nullable
-    @Override
-    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
-            @Nullable Bundle savedInstanceState) {
-        View rootView = inflater.inflate(R.layout.fragment_view_gif, container, false);
-
-        String url = getArguments().getString("url");
-        WebView gifView = (WebView) rootView.findViewById(R.id.gif_view);
-        gifView.loadUrl(url);
-
-        rootView.setOnClickListener(new View.OnClickListener() {
-            @Override
-            public void onClick(View v) {
-                dismiss();
-            }
-        });
-
-        return rootView;
-    }
-
-    private void dismiss() {
-        getFragmentManager().popBackStack();
-    }
-}
diff --git a/app/src/main/java/com/keylesspalace/tusky/ViewMediaFragment.java b/app/src/main/java/com/keylesspalace/tusky/ViewMediaFragment.java
index aeb78f6f..ffb74241 100644
--- a/app/src/main/java/com/keylesspalace/tusky/ViewMediaFragment.java
+++ b/app/src/main/java/com/keylesspalace/tusky/ViewMediaFragment.java
@@ -21,8 +21,11 @@ import android.view.LayoutInflater;
 import android.view.View;
 import android.view.ViewGroup;
 
-import com.android.volley.toolbox.ImageLoader;
-import com.android.volley.toolbox.NetworkImageView;
+import com.squareup.picasso.Callback;
+import com.squareup.picasso.Picasso;
+
+import uk.co.senab.photoview.PhotoView;
+import uk.co.senab.photoview.PhotoViewAttacher;
 
 public class ViewMediaFragment extends Fragment {
     public static ViewMediaFragment newInstance(String url) {
@@ -40,17 +43,36 @@ public class ViewMediaFragment extends Fragment {
 
         Bundle arguments = getArguments();
         String url = arguments.getString("url");
-        NetworkImageView image = (NetworkImageView) rootView.findViewById(R.id.view_media_image);
-        ImageLoader imageLoader = VolleySingleton.getInstance(getContext()).getImageLoader();
-        image.setImageUrl(url, imageLoader);
+        PhotoView photoView = (PhotoView) rootView.findViewById(R.id.view_media_image);
 
-        rootView.setOnClickListener(new View.OnClickListener() {
+        final PhotoViewAttacher attacher = new PhotoViewAttacher(photoView);
+
+        attacher.setOnPhotoTapListener(new PhotoViewAttacher.OnPhotoTapListener() {
             @Override
-            public void onClick(View v) {
+            public void onPhotoTap(View view, float x, float y) {
+
+            }
+
+            @Override
+            public void onOutsidePhotoTap() {
                 dismiss();
             }
         });
 
+        Picasso.with(getContext())
+                .load(url)
+                .into(photoView, new Callback() {
+                    @Override
+                    public void onSuccess() {
+                        attacher.update();
+                    }
+
+                    @Override
+                    public void onError() {
+
+                    }
+                });
+
         return rootView;
     }
 
diff --git a/app/src/main/java/com/keylesspalace/tusky/ViewThreadFragment.java b/app/src/main/java/com/keylesspalace/tusky/ViewThreadFragment.java
index 9389c0bf..f615eb04 100644
--- a/app/src/main/java/com/keylesspalace/tusky/ViewThreadFragment.java
+++ b/app/src/main/java/com/keylesspalace/tusky/ViewThreadFragment.java
@@ -30,12 +30,17 @@ import android.view.ViewGroup;
 import com.android.volley.Request;
 import com.android.volley.Response;
 import com.android.volley.VolleyError;
+import com.keylesspalace.tusky.entity.Status;
+import com.keylesspalace.tusky.entity.StatusContext;
 
 import org.json.JSONException;
 import org.json.JSONObject;
 
 import java.util.List;
 
+import retrofit2.Call;
+import retrofit2.Callback;
+
 public class ViewThreadFragment extends SFragment implements StatusActionListener {
     private RecyclerView recyclerView;
     private ThreadAdapter adapter;
@@ -78,54 +83,39 @@ public class ViewThreadFragment extends SFragment implements StatusActionListene
     }
 
     private void sendStatusRequest(final String id) {
-        String endpoint = String.format(getString(R.string.endpoint_get_status), id);
-        super.sendRequest(Request.Method.GET, endpoint, null,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        Status status;
-                        try {
-                            status = Status.parse(response, false);
-                        } catch (JSONException e) {
-                            onThreadRequestFailure(id);
-                            return;
-                        }
-                        int position = adapter.insertStatus(status);
-                        recyclerView.scrollToPosition(position);
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onThreadRequestFailure(id);
-                    }
-                });
+        MastodonAPI api = ((BaseActivity) getActivity()).mastodonAPI;
+
+        api.status(id).enqueue(new Callback<Status>() {
+            @Override
+            public void onResponse(Call<Status> call, retrofit2.Response<Status> response) {
+                int position = adapter.insertStatus(response.body());
+                recyclerView.scrollToPosition(position);
+            }
+
+            @Override
+            public void onFailure(Call<Status> call, Throwable t) {
+                onThreadRequestFailure(id);
+            }
+        });
     }
 
     private void sendThreadRequest(final String id) {
-        String endpoint = String.format(getString(R.string.endpoint_context), id);
-        super.sendRequest(Request.Method.GET, endpoint, null,
-                new Response.Listener<JSONObject>() {
-                    @Override
-                    public void onResponse(JSONObject response) {
-                        try {
-                            List<Status> ancestors =
-                                    Status.parse(response.getJSONArray("ancestors"));
-                            List<Status> descendants =
-                                    Status.parse(response.getJSONArray("descendants"));
-                            adapter.addAncestors(ancestors);
-                            adapter.addDescendants(descendants);
-                        } catch (JSONException e) {
-                            onThreadRequestFailure(id);
-                        }
-                    }
-                },
-                new Response.ErrorListener() {
-                    @Override
-                    public void onErrorResponse(VolleyError error) {
-                        onThreadRequestFailure(id);
-                    }
-                });
+        MastodonAPI api = ((BaseActivity) getActivity()).mastodonAPI;
+
+        api.statusContext(id).enqueue(new Callback<StatusContext>() {
+            @Override
+            public void onResponse(Call<StatusContext> call, retrofit2.Response<StatusContext> response) {
+                StatusContext context = response.body();
+
+                adapter.addAncestors(context.ancestors);
+                adapter.addDescendants(context.descendants);
+            }
+
+            @Override
+            public void onFailure(Call<StatusContext> call, Throwable t) {
+                onThreadRequestFailure(id);
+            }
+        });
     }
 
     private void onThreadRequestFailure(final String id) {
@@ -162,7 +152,7 @@ public class ViewThreadFragment extends SFragment implements StatusActionListene
 
     public void onViewThread(int position) {
         Status status = adapter.getItem(position);
-        if (thisThreadsStatusId.equals(status.getId())) {
+        if (thisThreadsStatusId.equals(status.id)) {
             // If already viewing this thread, don't reopen it.
             return;
         }
diff --git a/app/src/main/java/com/keylesspalace/tusky/entity/Account.java b/app/src/main/java/com/keylesspalace/tusky/entity/Account.java
new file mode 100644
index 00000000..d091512e
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/entity/Account.java
@@ -0,0 +1,65 @@
+/* Copyright 2017 Andrew Dawson
+ *
+ * This file is part of Tusky.
+ *
+ * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
+ * General Public License as published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
+ * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
+ * Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along with Tusky. If not, see
+ * <http://www.gnu.org/licenses/>. */
+
+package com.keylesspalace.tusky.entity;
+
+import android.text.Spanned;
+
+import com.google.gson.annotations.SerializedName;
+
+public class Account {
+    public String id;
+
+    @SerializedName("acct")
+    public String username;
+
+    @SerializedName("display_name")
+    public String displayName;
+
+    public Spanned note;
+
+    public String url;
+
+    public String avatar;
+
+    public String header;
+
+    public boolean locked;
+
+    @SerializedName("followers_count")
+    public String followersCount;
+
+    @SerializedName("following_count")
+    public String followingCount;
+
+    @SerializedName("statuses_count")
+    public String statusesCount;
+
+    @Override
+    public int hashCode() {
+        return id.hashCode();
+    }
+
+    @Override
+    public boolean equals(Object other) {
+        if (this.id == null) {
+            return this == other;
+        } else if (!(other instanceof Account)) {
+            return false;
+        }
+        Account account = (Account) other;
+        return account.id.equals(this.id);
+    }
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/entity/Media.java b/app/src/main/java/com/keylesspalace/tusky/entity/Media.java
new file mode 100644
index 00000000..1db867fb
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/entity/Media.java
@@ -0,0 +1,17 @@
+package com.keylesspalace.tusky.entity;
+
+import com.google.gson.annotations.SerializedName;
+
+public class Media {
+    public String id;
+
+    public String type;
+
+    public String url;
+
+    @SerializedName("preview_url")
+    public String previewUrl;
+
+    @SerializedName("text_url")
+    public String textUrl;
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/entity/Notification.java b/app/src/main/java/com/keylesspalace/tusky/entity/Notification.java
new file mode 100644
index 00000000..fb0b7b9a
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/entity/Notification.java
@@ -0,0 +1,55 @@
+/* Copyright 2017 Andrew Dawson
+ *
+ * This file is part of Tusky.
+ *
+ * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
+ * General Public License as published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
+ * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
+ * Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along with Tusky. If not, see
+ * <http://www.gnu.org/licenses/>. */
+
+package com.keylesspalace.tusky.entity;
+
+import com.google.gson.annotations.SerializedName;
+
+public class Notification {
+    public enum Type {
+        @SerializedName("mention")
+        MENTION,
+        @SerializedName("reblog")
+        REBLOG,
+        @SerializedName("favourite")
+        FAVOURITE,
+        @SerializedName("follow")
+        FOLLOW,
+    }
+
+    public Type type;
+
+    public String id;
+
+    public Account account;
+
+    public Status status;
+
+    @Override
+    public int hashCode() {
+        return id.hashCode();
+    }
+
+    @Override
+    public boolean equals(Object other) {
+        if (this.id == null) {
+            return this == other;
+        } else if (!(other instanceof Notification)) {
+            return false;
+        }
+        Notification notification = (Notification) other;
+        return notification.id.equals(this.id);
+    }
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/entity/Relationship.java b/app/src/main/java/com/keylesspalace/tusky/entity/Relationship.java
new file mode 100644
index 00000000..d0874405
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/entity/Relationship.java
@@ -0,0 +1,18 @@
+package com.keylesspalace.tusky.entity;
+
+import com.google.gson.annotations.SerializedName;
+
+public class Relationship {
+    public String id;
+
+    public boolean following;
+
+    @SerializedName("followed_by")
+    public boolean followedBy;
+
+    public boolean blocking;
+
+    public boolean muting;
+
+    public boolean requested;
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/entity/Status.java b/app/src/main/java/com/keylesspalace/tusky/entity/Status.java
new file mode 100644
index 00000000..39378ee9
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/entity/Status.java
@@ -0,0 +1,136 @@
+/* Copyright 2017 Andrew Dawson
+ *
+ * This file is part of Tusky.
+ *
+ * Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
+ * General Public License as published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
+ * the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
+ * Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along with Tusky. If not, see
+ * <http://www.gnu.org/licenses/>. */
+
+package com.keylesspalace.tusky.entity;
+
+import android.text.Spanned;
+
+import com.google.gson.annotations.SerializedName;
+
+import java.util.Date;
+
+public class Status {
+    private Status actionableStatus;
+
+    public String url;
+
+    @SerializedName("reblogs_count")
+    public String reblogsCount;
+
+    @SerializedName("favourites_count")
+    public String favouritesCount;
+
+    @SerializedName("in_reply_to_id")
+    public String inReplyToId;
+
+    @SerializedName("in_reply_to_account_id")
+    public String inReplyToAccountId;
+
+    public String getActionableId() {
+        return reblog == null ? id : reblog.id;
+    }
+
+    public Status getActionableStatus() {
+        return reblog == null ? this : reblog;
+    }
+
+    public enum Visibility {
+        @SerializedName("public")
+        PUBLIC,
+        @SerializedName("unlisted")
+        UNLISTED,
+        @SerializedName("private")
+        PRIVATE,
+    }
+
+    public String id;
+
+    public Account account;
+
+    public Spanned content;
+
+    public Status reblog;
+
+    @SerializedName("created_at")
+    public Date createdAt;
+
+    public boolean reblogged;
+
+    public boolean favourited;
+
+    public boolean sensitive;
+
+    @SerializedName("spoiler_text")
+    public String spoilerText;
+
+    public Visibility visibility;
+
+    @SerializedName("media_attachments")
+    public MediaAttachment[] attachments;
+
+    public Mention[] mentions;
+
+    public static final int MAX_MEDIA_ATTACHMENTS = 4;
+
+    @Override
+    public int hashCode() {
+        return id.hashCode();
+    }
+
+    @Override
+    public boolean equals(Object other) {
+        if (this.id == null) {
+            return this == other;
+        } else if (!(other instanceof Status)) {
+            return false;
+        }
+        Status status = (Status) other;
+        return status.id.equals(this.id);
+    }
+
+    public static class MediaAttachment {
+        public enum Type {
+            @SerializedName("image")
+            IMAGE,
+            @SerializedName("gifv")
+            GIFV,
+            @SerializedName("video")
+            VIDEO,
+            UNKNOWN,
+        }
+
+        public String url;
+
+        @SerializedName("preview_url")
+        public String previewUrl;
+
+        @SerializedName("text_url")
+        public String textUrl;
+
+        @SerializedName("remote_url")
+        public String remoteUrl;
+
+        public Type type;
+    }
+
+    public static class Mention {
+        public String id;
+
+        public String url;
+
+        @SerializedName("acct")
+        public String username;
+    }
+}
diff --git a/app/src/main/java/com/keylesspalace/tusky/entity/StatusContext.java b/app/src/main/java/com/keylesspalace/tusky/entity/StatusContext.java
new file mode 100644
index 00000000..338b5503
--- /dev/null
+++ b/app/src/main/java/com/keylesspalace/tusky/entity/StatusContext.java
@@ -0,0 +1,8 @@
+package com.keylesspalace.tusky.entity;
+
+import java.util.List;
+
+public class StatusContext {
+    public List<Status> ancestors;
+    public List<Status> descendants;
+}
diff --git a/app/src/main/res/anim/zoom_in.xml b/app/src/main/res/anim/zoom_in.xml
new file mode 100644
index 00000000..e607a315
--- /dev/null
+++ b/app/src/main/res/anim/zoom_in.xml
@@ -0,0 +1,20 @@
+<?xml version="1.0" encoding="utf-8"?>
+<set xmlns:android="http://schemas.android.com/apk/res/android">
+    <scale
+        android:interpolator="@android:anim/linear_interpolator"
+        android:fromXScale=".1"
+        android:toXScale="1"
+        android:fromYScale=".1"
+        android:toYScale="1"
+        android:pivotX="50%"
+        android:pivotY="50%"
+        android:duration="200"
+        android:fillAfter="true">
+    </scale>
+
+    <alpha
+        android:interpolator="@android:anim/linear_interpolator"
+        android:fromAlpha="0"
+        android:toAlpha="1"
+        android:duration="300" />
+</set>
\ No newline at end of file
diff --git a/app/src/main/res/anim/zoom_out.xml b/app/src/main/res/anim/zoom_out.xml
new file mode 100644
index 00000000..ab81c679
--- /dev/null
+++ b/app/src/main/res/anim/zoom_out.xml
@@ -0,0 +1,20 @@
+<?xml version="1.0" encoding="utf-8"?>
+<set xmlns:android="http://schemas.android.com/apk/res/android">
+    <scale
+        android:interpolator="@android:anim/linear_interpolator"
+        android:fromXScale="1"
+        android:toXScale=".1"
+        android:fromYScale="1"
+        android:toYScale=".1"
+        android:pivotX="50%"
+        android:pivotY="50%"
+        android:duration="200"
+        android:fillAfter="true">
+    </scale>
+
+    <alpha
+        android:interpolator="@android:anim/linear_interpolator"
+        android:fromAlpha="1"
+        android:toAlpha="0"
+        android:duration="300" />
+</set>
\ No newline at end of file
diff --git a/app/src/main/res/drawable/account_header_missing.xml b/app/src/main/res/drawable/account_header_missing.xml
index 0567c24b..51629946 100644
--- a/app/src/main/res/drawable/account_header_missing.xml
+++ b/app/src/main/res/drawable/account_header_missing.xml
@@ -2,7 +2,6 @@
 <shape
     xmlns:android="http://schemas.android.com/apk/res/android"
     android:shape="rectangle">
-
+    <size android:width="700px" android:height="335px" />
     <solid android:color="@color/color_background_dark" />
-
 </shape>
\ No newline at end of file
diff --git a/app/src/main/res/drawable/ic_back.xml b/app/src/main/res/drawable/ic_back.xml
deleted file mode 100644
index cceab543..00000000
--- a/app/src/main/res/drawable/ic_back.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="850.3937"
-    android:viewportWidth="850.3937" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="m410.4,48.8c-9.1,0 -18.1,3.5 -25.1,10.4L84.7,359.9c-1.6,1.6 -2.9,3.2 -4.1,5 -6,6.3 -9.7,14.9 -9.7,24.4l0,70.9c0,11.3 5.2,21.3 13.4,27.8 1.6,3.2 3.8,6.2 6.5,8.9L391.5,797.5c13.9,13.9 36.2,13.9 50.1,0l50.1,-50.1c13.9,-13.9 13.9,-36.2 0,-50.1l-201.7,-201.7 454.1,0c19.6,0 35.4,-15.8 35.4,-35.4l0,-70.9c0,-19.6 -15.8,-35.4 -35.4,-35.4l-452.9,0 194.4,-194.4c13.9,-13.9 13.9,-36.2 0,-50.1l-50.1,-50.1c-6.9,-6.9 -16,-10.4 -25.1,-10.4z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="10.62992096"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_block_24dp.xml b/app/src/main/res/drawable/ic_block_24dp.xml
new file mode 100644
index 00000000..7a184d9d
--- /dev/null
+++ b/app/src/main/res/drawable/ic_block_24dp.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+        android:width="24dp"
+        android:height="24dp"
+        android:viewportWidth="24.0"
+        android:viewportHeight="24.0">
+    <path
+        android:fillColor="@color/toolbar_icon_dark"
+        android:pathData="M12,2C6.48,2 2,6.48 2,12s4.48,10 10,10 10,-4.48 10,-10S17.52,2 12,2zM4,12c0,-4.42 3.58,-8 8,-8 1.85,0 3.55,0.63 4.9,1.69L5.69,16.9C4.63,15.55 4,13.85 4,12zM12,20c-1.85,0 -3.55,-0.63 -4.9,-1.69L18.31,7.1C19.37,8.45 20,10.15 20,12c0,4.42 -3.58,8 -8,8z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_compose.xml b/app/src/main/res/drawable/ic_compose.xml
deleted file mode 100644
index c5669bb7..00000000
--- a/app/src/main/res/drawable/ic_compose.xml
+++ /dev/null
@@ -1,11 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="708.66144"
-    android:viewportWidth="708.66144" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="m86.4,17.7c-37.3,0 -67.3,30 -67.3,67.3l0,538.6c0,37.3 30,67.3 67.3,67.3l537.2,0c37.3,0 67.3,-30 67.3,-67.3l0,-462.4c-17.6,17.9 -35.4,35.6 -53.2,53.3l0,352.4c0,39.3 -31.6,70.9 -70.9,70.9l-425.2,0c-39.3,0 -70.9,-31.6 -70.9,-70.9l0,-425.2c0,-39.3 31.6,-70.9 70.9,-70.9l358.9,0c18,-17.7 36,-35.3 53.8,-53.2l-468,0z"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="33.62945938"/>
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="m672.8,8.2 l25.1,25.1c13.9,13.9 13.9,36.2 0,50.1L361.6,420.4C347.1,434.2 199.5,537.2 185.6,523.3l-0.8,-0.8C170.9,508.6 272.1,359.2 286.6,344.7L622.7,8.2c13.9,-13.9 36.2,-13.9 50.1,0z"
-        android:strokeAlpha="1" android:strokeColor="#cccccc"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="0"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_exit_to_app_24dp.xml b/app/src/main/res/drawable/ic_exit_to_app_24dp.xml
new file mode 100644
index 00000000..93378231
--- /dev/null
+++ b/app/src/main/res/drawable/ic_exit_to_app_24dp.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+        android:width="24dp"
+        android:height="24dp"
+        android:viewportWidth="24.0"
+        android:viewportHeight="24.0">
+    <path
+        android:fillColor="@color/toolbar_icon_dark"
+        android:pathData="M10.09,15.59L11.5,17l5,-5 -5,-5 -1.41,1.41L12.67,11H3v2h9.67l-2.58,2.59zM19,3H5c-1.11,0 -2,0.9 -2,2v4h2V5h14v14H5v-4H3v4c0,1.1 0.89,2 2,2h14c1.1,0 2,-0.9 2,-2V5c0,-1.1 -0.9,-2 -2,-2z"/>
+</vector>
diff --git a/app/src/main/res/drawable/ic_extra.xml b/app/src/main/res/drawable/ic_extra.xml
deleted file mode 100644
index 62fddb87..00000000
--- a/app/src/main/res/drawable/ic_extra.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="42.519684"
-    android:viewportWidth="42.519684" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M7.81,16.57C7.36,16.55 6.87,16.73 6.43,17.17C6.27,17.34 6.13,17.51 5.99,17.68C5.86,17.79 5.73,17.91 5.63,18.04C5.49,18.21 5.4,18.42 5.29,18.62C5.29,18.62 5.29,18.63 5.28,18.63C5.04,19 4.81,19.37 4.6,19.76C4.1,20.87 3.96,22.05 4.07,23.25C4.23,24.34 4.77,25.2 5.58,25.93C6.51,26.7 7.61,26.67 8.71,26.34C9.63,26.03 10.41,25.46 11.14,24.84C11.78,24.32 12.3,23.72 12.71,23.01C13.13,22.11 13.19,21.16 13.02,20.2C12.92,19.29 12.36,18.64 11.77,18.01C11.12,17.34 10.29,17.03 9.38,16.89C9.15,16.86 8.91,16.84 8.68,16.85C8.42,16.68 8.13,16.58 7.81,16.57zM22.36,16.9C21.94,16.96 21.57,17.11 21.23,17.31C20.89,17.38 20.54,17.55 20.21,17.88C19.3,18.75 18.74,19.9 18.28,21.05C17.88,22.29 17.95,23.52 18.34,24.73C18.78,25.83 19.59,26.53 20.69,26.93C21.86,27.26 22.88,26.85 23.82,26.15C24.66,25.41 25.2,24.46 25.56,23.41C25.93,22.47 26.06,21.49 26.01,20.48C25.99,19.54 25.59,18.73 25.04,17.99C24.3,17.13 23.45,16.9 22.36,16.9zM35.14,17.19C34.99,17.2 34.86,17.24 34.73,17.28C34.26,17.25 33.75,17.42 33.29,17.88C32.46,18.61 31.87,19.5 31.43,20.52C31.01,21.82 31.15,23.13 31.64,24.39C32.06,25.39 32.78,26.08 33.75,26.53C35.12,26.87 35.98,26.36 36.91,25.4C37.5,24.76 37.97,24.04 38.36,23.26C38.82,22.34 38.9,21.37 38.77,20.37C38.61,19.46 38.13,18.74 37.49,18.11C37.03,17.63 36.44,17.38 35.81,17.23C35.56,17.19 35.34,17.18 35.14,17.19zM8.68,21.7C8.7,21.72 8.72,21.73 8.74,21.75C8.63,21.69 8.89,22.06 8.69,21.73C8.68,21.72 8.68,21.71 8.68,21.7zM35.02,23.23C35.03,23.23 35.03,23.24 35.04,23.24C35.28,23.41 35.15,23.37 35.02,23.23z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="square" android:strokeLineJoin="miter" android:strokeWidth="0.30000001"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_favourite.xml b/app/src/main/res/drawable/ic_favourite.xml
deleted file mode 100644
index 24478b8b..00000000
--- a/app/src/main/res/drawable/ic_favourite.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="42.519684"
-    android:viewportWidth="42.519684" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M21.68,5.81C20.21,5.78 17.68,14.98 16.48,15.81C15.27,16.65 5.77,15.82 5.29,17.2C4.81,18.59 12.77,23.84 13.2,25.24C13.62,26.64 9.89,35.42 11.06,36.3C12.23,37.19 19.68,31.24 21.14,31.27C22.61,31.3 29.81,37.56 31.01,36.72C32.21,35.88 28.86,26.96 29.34,25.57C29.82,24.19 37.99,19.28 37.57,17.88C37.15,16.47 27.62,16.91 26.45,16.02C25.29,15.14 23.14,5.84 21.68,5.81z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="square" android:strokeLineJoin="miter" android:strokeWidth="0.30000001"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_favourited.xml b/app/src/main/res/drawable/ic_favourited.xml
deleted file mode 100644
index fe780bc2..00000000
--- a/app/src/main/res/drawable/ic_favourited.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="16dp" android:viewportHeight="566.92914"
-    android:viewportWidth="566.92914" android:width="16dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M141.7,0C63.2,0 0,63.2 0,141.7L0,425.2C0,503.7 63.2,566.9 141.7,566.9L425.2,566.9C503.7,566.9 566.9,503.7 566.9,425.2L566.9,141.7C566.9,63.2 503.7,0 425.2,0L141.7,0zM283.6,24.8C287.6,24.9 291.2,27.1 293,30.7L363.4,173.4L520.9,196.3C529.6,197.6 533.1,208.3 526.8,214.4L412.8,325.5L439.7,482.3C441.2,491 432.1,497.6 424.3,493.5L283.5,419.5L142.6,493.5C134.8,497.6 125.7,491 127.2,482.3L154.1,325.5L40.2,214.4C33.8,208.3 37.3,197.6 46,196.3L203.5,173.4L273.9,30.7C275.7,27.1 279.5,24.8 283.6,24.8z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="0"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_followed.xml b/app/src/main/res/drawable/ic_followed.xml
deleted file mode 100644
index bdef7ee8..00000000
--- a/app/src/main/res/drawable/ic_followed.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="16dp" android:viewportHeight="566.92914"
-    android:viewportWidth="566.92914" android:width="16dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M141.7,0C63.2,0 0,63.2 0,141.7L0,425.2C0,503.7 63.2,566.9 141.7,566.9L425.2,566.9C503.7,566.9 566.9,503.7 566.9,425.2L566.9,141.7C566.9,63.2 503.7,0 425.2,0L141.7,0zM283.5,70.9A88.6,88.6 0,0 1,372 159.4A88.6,88.6 0,0 1,283.5 248A88.6,88.6 0,0 1,194.9 159.4A88.6,88.6 0,0 1,283.5 70.9zM194.9,311.3C194.9,311.3 229.1,336.6 283.5,336.6C338.4,336.6 370.5,311.3 370.5,311.3C496.1,407.5 460.6,478.3 460.6,478.3L106.3,478.3C106.3,478.3 70.9,407.5 194.9,311.3z"
-        android:strokeAlpha="1" android:strokeColor="#9d9d9d"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="0"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_lock_24dp.xml b/app/src/main/res/drawable/ic_lock_24dp.xml
deleted file mode 100644
index 6316164d..00000000
--- a/app/src/main/res/drawable/ic_lock_24dp.xml
+++ /dev/null
@@ -1,9 +0,0 @@
-<vector xmlns:android="http://schemas.android.com/apk/res/android"
-        android:width="24dp"
-        android:height="24dp"
-        android:viewportWidth="24.0"
-        android:viewportHeight="24.0">
-    <path
-        android:fillColor="@color/toolbar_icon_dark"
-        android:pathData="M18,8h-1L17,6c0,-2.76 -2.24,-5 -5,-5S7,3.24 7,6v2L6,8c-1.1,0 -2,0.9 -2,2v10c0,1.1 0.9,2 2,2h12c1.1,0 2,-0.9 2,-2L20,10c0,-1.1 -0.9,-2 -2,-2zM12,17c-1.1,0 -2,-0.9 -2,-2s0.9,-2 2,-2 2,0.9 2,2 -0.9,2 -2,2zM15.1,8L8.9,8L8.9,6c0,-1.71 1.39,-3.1 3.1,-3.1 1.71,0 3.1,1.39 3.1,3.1v2z"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_media.xml b/app/src/main/res/drawable/ic_media.xml
deleted file mode 100644
index ee873111..00000000
--- a/app/src/main/res/drawable/ic_media.xml
+++ /dev/null
@@ -1,31 +0,0 @@
-<vector android:height="48dp" android:viewportHeight="1133.894"
-    android:viewportWidth="1134.6519" android:width="48dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="M52.7,262.2L1081.9,262.2A38.6,38.6 0,0 1,1120.5 300.8L1120.5,833.1A38.6,38.6 0,0 1,1081.9 871.7L52.7,871.7A38.6,38.6 0,0 1,14.2 833.1L14.2,300.8A38.6,38.6 0,0 1,52.7 262.2z"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="28.34645653"/>
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="m19.6,458.3c104.2,-9.7 76.2,61 365.2,125.3 61.9,13.8 50,40.6 96.2,58 105.8,39.9 376.7,15.8 639.8,33.5"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="miter" android:strokeWidth="28.34645653"/>
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="m1011.8,494c0,0 -130.5,-8 -158.9,-39.2 -142.4,-156.4 -193.3,0.9 -217,-9.7 -74.7,-33.3 -65,21.3 -199.8,103.2 -20.5,12.4 -8.8,16.9 39.1,18.1 143.3,3.8 -74.6,16.2 24.6,18.4l115.8,2.6"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="miter" android:strokeWidth="28.34645653"/>
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="m1121.2,496.7c0,0 -254.5,-33.7 -505.7,90.3 -98.7,48.8 350.1,80.7 350.1,80.7"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="miter" android:strokeWidth="28.34645653"/>
-    <path android:fillColor="#00000000"
-        android:pathData="m459,531.9 l-245.5,-0" android:strokeAlpha="1"
-        android:strokeColor="#ffffff" android:strokeLineCap="butt"
-        android:strokeLineJoin="miter" android:strokeWidth="28.34645653"/>
-    <path android:fillColor="#00000000"
-        android:pathData="M14.2,639C390.1,602 473.1,743.8 1118.5,752.1"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="miter" android:strokeWidth="35.43307114"/>
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="M277.5,425.5m-62.9,0a62.9,62.9 0,1 1,125.7 0a62.9,62.9 0,1 1,-125.7 0"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="35.43307114"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_media_disabled.xml b/app/src/main/res/drawable/ic_media_disabled.xml
deleted file mode 100644
index 9c1a90e9..00000000
--- a/app/src/main/res/drawable/ic_media_disabled.xml
+++ /dev/null
@@ -1,11 +0,0 @@
-<vector android:height="48dp" android:viewportHeight="1133.894"
-    android:viewportWidth="1134.6519" android:width="48dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#9d9d9d"
-        android:pathData="m277.5,344.9c-44.3,0 -80.6,36.3 -80.6,80.6 -0,44.3 36.3,80.6 80.6,80.6 44.3,0 80.6,-36.3 80.6,-80.6 -0,-44.3 -36.3,-80.6 -80.6,-80.6zM277.5,380.3c25.1,0 45.1,20 45.1,45.1 0,25.1 -20,45.1 -45.1,45.1 -25.1,0 -45.1,-20 -45.1,-45.1 0,-25.1 20,-45.1 45.1,-45.1zM135.7,615.2c-38,-0.1 -78.6,1.8 -123.2,6.2l-0.1,35.3 306.3,31.4 2.1,-20.4c4.8,0.8 9.6,1.5 14.4,2.4l6.3,-34.9c-65,-11.8 -130.2,-19.8 -205.8,-20zM549.6,676.1 L549.2,679.4 542.6,712.8c135.7,27.1 307.8,53.6 575.7,57.1l2,-35.3 -570.7,-58.5zM213.5,517.8 L213.5,546.1 459,546.1 459,517.8 213.5,517.8zM1034.5,478.4c-83.3,0.3 -215.7,11.2 -354.9,65.1l10.2,26.4c134.7,-52.1 263.8,-62.9 344.7,-63.2 53,-0.2 84.7,4.1 84.7,4.1l3.7,-28.1c0,0 -34,-4.5 -88.6,-4.3zM628.4,605.1 L618,631.5c43.4,17 128.1,28.6 204.3,37.2 76.1,8.6 142.3,13.3 142.3,13.3l2,-28.3c0,0 -65.7,-4.7 -141.1,-13.2 -75.4,-8.5 -161.9,-21.6 -197.1,-35.4zM828.5,411.3 L810.2,432.9c10,8.5 20.8,18.8 32.2,31.4 10.5,11.6 25.7,17.8 43,23.3 14.5,4.6 30.4,7.9 46.1,10.7l-269,63.5 6.5,27.6 346,-81.6 -2.4,-27.9c0,0 -32,-2 -67.4,-7.7 -17.7,-2.9 -36.1,-6.7 -51.3,-11.5 -15.2,-4.8 -26.9,-11.3 -30.6,-15.3 -12.2,-13.4 -23.8,-24.6 -34.9,-34zM445.8,525.5c-5.4,3.5 -11.1,7.1 -17,10.7 -5.8,3.5 -10.5,5.8 -14.4,13.3 -2,3.8 -2.8,10.6 -0.3,15.7 2.5,5.1 6.3,7.4 9.3,8.8l9.1,4.5 31.1,-31.1 -17.7,-21.9zM37.7,443.2c-6.1,-0 -12.5,0.3 -19.4,1l2.6,28.2c6.1,-0.6 11.7,-0.8 16.8,-0.8 23.5,0 37.3,5.1 55.5,15l-0.1,-0.1c86.8,47.7 182.4,93.2 288.7,110.9 6.2,1.4 11.4,2.9 15.9,4.4l9.2,-26.8c-5.7,-1.9 -11.9,-3.7 -19.1,-5.3l-0.4,-0.1 -0.4,-0.1C285.7,552.8 192.7,508.9 106.9,461.7l-0,-0 -0,-0C86.5,450.7 65.9,443.2 37.7,443.2l-0,0zM652.9,568.5 L647.6,573.8 549.5,671.9 579.7,674.6c135,12.1 340.2,1.1 540.2,14.6l3.7,-28 -470.7,-92.7zM662.3,599.2 L941.1,654.1C817.5,652.1 701.5,654.1 612.8,648.7l49.4,-49.4zM52.7,248c-29,0 -52.7,23.8 -52.7,52.7l0,532.4c0,29 23.8,52.7 52.7,52.7l66.5,0a14.2,14.2 0,0 0,10 -4.2L738.7,272.2a14.2,14.2 0,0 0,-10 -24.2l-676,0zM979.2,248a14.2,14.2 0,0 0,-10 4.2L359.7,861.7a14.2,14.2 0,0 0,10 24.2l712.2,0c29,0 52.7,-23.8 52.7,-52.7l0,-532.4c0,-29 -23.8,-52.7 -52.7,-52.7l-102.7,0zM52.7,276.4 L694.5,276.4 113.3,857.5 52.7,857.5c-13.8,0 -24.4,-10.6 -24.4,-24.4l0,-532.4c0,-13.8 10.6,-24.4 24.4,-24.4zM985.1,276.4 L1081.9,276.4c13.8,0 24.4,10.6 24.4,24.4l0,532.4c0,13.8 -10.6,24.4 -24.4,24.4l-677.9,0 581.1,-581.1z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="35.43307114"/>
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="M915.1,130.3L985.8,201.1A38.6,38.6 58.5,0 1,985.8 255.6L238.6,1002.8A38.6,38.6 84,0 1,184.1 1002.8L113.3,932.1A38.6,38.6 92.7,0 1,113.3 877.6L860.6,130.3A38.6,38.6 79.6,0 1,915.1 130.3z"
-        android:strokeAlpha="1" android:strokeColor="#9d9d9d"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="35.43307515"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_notify_mention.xml b/app/src/main/res/drawable/ic_notify_mention.xml
deleted file mode 100644
index 6ee8c1ce..00000000
--- a/app/src/main/res/drawable/ic_notify_mention.xml
+++ /dev/null
@@ -1,11 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="637.7953"
-    android:viewportWidth="637.7953" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M638.4,462C518.5,688.5 338.4,648.4 266,598.8 159,525.6 124,422.3 242.9,288.3 455.8,48.3 302.5,13.2 302.5,13.2c0,0 182.3,30.4 27.3,256.1 -80.8,117.6 -110.4,189.8 -8,253.1 110,56.5 231,-61.9 257,-103z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="butt" android:strokeLineJoin="miter" android:strokeWidth="1"/>
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="m263.5,4.2c-41,12.5 -74,22 -103,59.8C60.4,194.3 29.6,305.7 128.9,407.1 251.6,489.6 425.3,370.6 425.3,370.6l38.5,66.6c0,0 -194.3,142.4 -338,52.1C79.6,460.2 -85.6,403.7 64.9,121.1 122.5,12.7 176.6,-10.9 263.5,4.2Z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="butt" android:strokeLineJoin="miter" android:strokeWidth="1"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_options.xml b/app/src/main/res/drawable/ic_options.xml
deleted file mode 100644
index b4f39bea..00000000
--- a/app/src/main/res/drawable/ic_options.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="1133.8583"
-    android:viewportWidth="1133.8583" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#00000000"
-        android:pathData="M704.8,566.9A137.9,137.9 0,0 1,566.9 704.8,137.9 137.9,0 0,1 429,566.9 137.9,137.9 0,0 1,566.9 429,137.9 137.9,0 0,1 704.8,566.9ZM566.9,1098.1c-184.7,0 -26,-116.8 -185.9,-209.2 -160,-92.4 -181.8,103.5 -274.1,-56.4 -92.4,-160 88.2,-80.9 88.2,-265.6 0,-184.7 -180.5,-105.6 -88.2,-265.6 92.4,-160 114.1,35.9 274.1,-56.4 160,-92.4 1.2,-209.2 185.9,-209.2 184.7,-0 26,116.8 185.9,209.2 160,92.4 181.8,-103.5 274.1,56.4 92.4,160 -88.2,80.9 -88.2,265.6 0,184.7 180.5,105.6 88.2,265.6C934.6,992.5 912.8,796.6 752.8,888.9 592.9,981.3 751.6,1098.1 566.9,1098.1Z"
-        android:strokeAlpha="1" android:strokeColor="#ffffff"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="68.95068359"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_person_outline_24dp.xml b/app/src/main/res/drawable/ic_person_outline_24dp.xml
deleted file mode 100644
index 4ad4f8d0..00000000
--- a/app/src/main/res/drawable/ic_person_outline_24dp.xml
+++ /dev/null
@@ -1,9 +0,0 @@
-<vector xmlns:android="http://schemas.android.com/apk/res/android"
-        android:width="24dp"
-        android:height="24dp"
-        android:viewportWidth="24.0"
-        android:viewportHeight="24.0">
-    <path
-        android:fillColor="@color/toolbar_icon_dark"
-        android:pathData="M12,5.9c1.16,0 2.1,0.94 2.1,2.1s-0.94,2.1 -2.1,2.1S9.9,9.16 9.9,8s0.94,-2.1 2.1,-2.1m0,9c2.97,0 6.1,1.46 6.1,2.1v1.1L5.9,18.1L5.9,17c0,-0.64 3.13,-2.1 6.1,-2.1M12,4C9.79,4 8,5.79 8,8s1.79,4 4,4 4,-1.79 4,-4 -1.79,-4 -4,-4zM12,13c-2.67,0 -8,1.34 -8,4v3h16v-3c0,-2.66 -5.33,-4 -8,-4z"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_reblog.xml b/app/src/main/res/drawable/ic_reblog.xml
deleted file mode 100644
index 3666855f..00000000
--- a/app/src/main/res/drawable/ic_reblog.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="42.519684"
-    android:viewportWidth="42.519684" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="m24.58,8.38c-0.38,0.01 -0.77,0.01 -1.15,0.02 -0.39,0.01 -0.78,0.01 -1.17,0.02 -0.4,0.01 -0.8,0.01 -1.2,0.02 -0.39,0.01 -0.79,0.02 -1.18,0.03 -0.38,0.01 -0.76,0.02 -1.15,0.03 -0.37,0.01 -0.74,0.02 -1.11,0.04 -0.34,0.01 -0.69,0.03 -1.03,0.04 -0.31,0.01 -0.63,0.03 -0.94,0.04 -0.3,0.01 -0.6,0.03 -0.9,0.05 -0.29,0.02 -0.58,0.04 -0.88,0.06 -0.27,0.02 -0.55,0.04 -0.82,0.06 -0.24,0.02 -0.47,0.04 -0.71,0.07 -0.16,0.02 -0.31,0.06 -0.47,0.07l0,0.01c-0.25,0 -0.2,3.54 0.05,3.54 0.13,-0.25 0.58,-0.15 0.99,-0.14 0.25,-0.03 0.5,-0.05 0.75,-0.07 0.28,-0.02 0.56,-0.04 0.83,-0.06 0.29,-0.02 0.59,-0.03 0.88,-0.05 0.3,-0.02 0.61,-0.03 0.91,-0.05 0.34,-0.02 0.67,-0.03 1.01,-0.04 0.36,-0.01 0.73,-0.02 1.09,-0.04 0.38,-0.01 0.76,-0.02 1.13,-0.03 0.39,-0.01 0.78,-0.02 1.17,-0.03 0.4,-0.01 0.79,-0.01 1.19,-0.02 0.39,-0.01 0.78,-0.01 1.17,-0.02 0.38,-0.01 0.77,-0.01 1.15,-0.02 0.35,-0.01 0.71,-0.01 1.06,-0.01 0.36,0.01 0.72,0.01 1.08,0.02 0.33,0.01 0.65,0.03 0.98,0.05 0.32,0.03 0.63,0.05 0.95,0.09 0.29,0.03 0.57,0.08 0.85,0.13 0.22,0.04 0.44,0.11 0.65,0.19 0.21,0.07 0.4,0.18 0.58,0.29 0.18,0.11 0.35,0.24 0.5,0.38 0.18,0.16 0.35,0.34 0.51,0.52 0.14,0.16 0.26,0.33 0.37,0.51 0.11,0.2 0.22,0.41 0.32,0.62 0.11,0.22 0.19,0.44 0.27,0.67 0.09,0.26 0.17,0.52 0.24,0.78 0.06,0.26 0.11,0.51 0.17,0.77 0.06,0.27 0.09,0.55 0.12,0.82 0.04,0.3 0.06,0.6 0.08,0.9 0.02,0.31 0.04,0.63 0.04,0.94 0,0.33 0.01,0.65 0.01,0.98 0,0.31 0.01,0.62 0.01,0.93l0.01,0.86c0,0.26 0,0.52 0.01,0.78 0.03,0.25 3.54,-0.23 3.5,-0.47 -0,-0.26 -0,-0.52 -0.01,-0.78l-0.01,-0.86c-0,-0.31 -0.01,-0.62 -0.01,-0.94 -0,-0.34 -0.01,-0.68 -0.02,-1.02 -0.01,-0.36 -0.04,-0.72 -0.06,-1.08 -0.03,-0.36 -0.07,-0.71 -0.12,-1.07 -0.05,-0.35 -0.11,-0.71 -0.19,-1.05 -0.07,-0.34 -0.15,-0.67 -0.25,-1 -0.1,-0.31 -0.19,-0.63 -0.31,-0.94 -0.12,-0.32 -0.25,-0.64 -0.4,-0.95 -0.15,-0.31 -0.31,-0.62 -0.5,-0.92 -0.2,-0.31 -0.42,-0.61 -0.67,-0.88 -0.24,-0.26 -0.49,-0.52 -0.75,-0.76C32.91,10.16 32.61,9.92 32.28,9.72 31.95,9.51 31.61,9.32 31.25,9.17 30.88,9.02 30.52,8.89 30.14,8.81 29.78,8.73 29.42,8.66 29.05,8.61 28.69,8.56 28.33,8.52 27.97,8.49c-0.38,-0.04 -0.76,-0.05 -1.14,-0.08 -0.36,-0.01 -0.72,-0.02 -1.09,-0.03 -0.39,-0.01 -0.77,-0.02 -1.16,-0.01zM8.94,10.1C8.9,10.03 8.31,9.3 8.31,9.3c0,0 -0.68,0.97 -0.71,1.03 -0.12,0.18 -0.24,0.36 -0.36,0.54 -0.08,0.12 -0.16,0.24 -0.25,0.36 -0.54,0.78 -0.64,0.88 -0.79,1.12 -0.07,0.1 -0.14,0.19 -0.21,0.29 -0.27,0.39 -0.55,0.77 -0.83,1.15 -0.28,0.38 -0.56,0.76 -0.85,1.12 -0.28,0.37 -0.55,0.73 -0.83,1.1 -0.25,0.33 -0.5,0.66 -0.76,0.99 -0.24,0.32 -0.48,0.63 -0.72,0.95 -0.24,0.28 -0.45,0.58 -0.67,0.87 -0.2,0.27 -0.4,0.54 -0.6,0.81 -0.19,0.24 -0.35,0.5 -0.54,0.74 -0.1,0.14 -0.05,0.07 -0.16,0.21 -0.1,0.23 3.14,1.63 3.24,1.4 0.1,-0.14 0.05,-0.07 0.16,-0.22 0.17,-0.24 0.34,-0.47 0.52,-0.71 0.2,-0.27 0.4,-0.53 0.6,-0.8 0.23,-0.29 0.46,-0.58 0.68,-0.88 0.24,-0.31 0.47,-0.61 0.71,-0.92 0.26,-0.33 0.51,-0.67 0.77,-1.01 0.27,-0.36 0.55,-0.72 0.82,-1.08 0.29,-0.4 0.59,-0.79 0.88,-1.19 0.01,-0.01 0.01,-0.02 0.02,-0.03 0.17,0.21 0.34,0.43 0.51,0.64 0.28,0.35 0.55,0.71 0.83,1.07 0.28,0.37 0.57,0.73 0.85,1.1 0.29,0.38 0.58,0.76 0.87,1.13 0.28,0.36 0.56,0.72 0.83,1.09 0.28,0.36 0.55,0.71 0.82,1.07 0.25,0.33 0.5,0.66 0.76,0.98 0.23,0.29 0.46,0.57 0.68,0.87 0.15,0.19 0.07,0.1 0.22,0.28 0.16,0.2 2.92,-2.01 2.76,-2.21 -0.13,-0.17 -0.07,-0.08 -0.2,-0.26 -0.23,-0.31 -0.47,-0.61 -0.71,-0.91 -0.25,-0.33 -0.5,-0.65 -0.76,-0.98 -0.27,-0.35 -0.54,-0.7 -0.81,-1.05 -0.28,-0.37 -0.56,-0.74 -0.85,-1.11 -0.29,-0.37 -0.57,-0.74 -0.86,-1.11 -0.29,-0.38 -0.58,-0.76 -0.87,-1.13 -0.28,-0.37 -0.56,-0.73 -0.85,-1.09 -0.27,-0.34 -0.54,-0.68 -0.81,-1.02 -0.1,-0.13 -0.21,-0.25 -0.31,-0.38 -0.36,-0.45 -0.51,-0.62 -0.77,-0.94 -0.08,-0.1 -0.17,-0.21 -0.25,-0.31 -0.14,-0.17 -0.28,-0.34 -0.41,-0.52 -0.07,-0.09 -0.13,-0.18 -0.19,-0.28zM40.94,20.11c-0.15,0.12 -0.3,0.23 -0.44,0.36 -0.24,0.2 -0.47,0.42 -0.7,0.63 -0.27,0.26 -0.54,0.51 -0.81,0.77 -0.29,0.28 -0.58,0.55 -0.87,0.83 -0.29,0.28 -0.58,0.55 -0.87,0.83 -0.27,0.27 -0.55,0.52 -0.81,0.79 -0.27,0.26 -0.55,0.52 -0.82,0.78 -0.24,0.24 -0.49,0.48 -0.74,0.71 -0.18,0.16 -0.34,0.33 -0.51,0.5 -0.11,-0.13 -0.22,-0.25 -0.32,-0.38 -0.24,-0.29 -0.48,-0.57 -0.72,-0.86 -0.26,-0.31 -0.52,-0.62 -0.78,-0.93 -0.27,-0.31 -0.54,-0.63 -0.8,-0.94 -0.27,-0.3 -0.53,-0.61 -0.81,-0.9 -0.24,-0.26 -0.48,-0.52 -0.71,-0.78 -0.19,-0.22 -0.38,-0.43 -0.58,-0.65 -0.13,-0.14 -0.26,-0.29 -0.4,-0.43 -0.18,-0.18 -2.68,2.32 -2.5,2.5 0.11,0.11 0.21,0.22 0.31,0.34 0.19,0.2 0.37,0.41 0.55,0.62 0.25,0.27 0.49,0.54 0.74,0.81 0.26,0.28 0.51,0.57 0.76,0.85 0.26,0.3 0.52,0.61 0.78,0.92 0.25,0.3 0.5,0.59 0.75,0.89 0.24,0.29 0.49,0.58 0.73,0.87 0.22,0.28 0.46,0.54 0.69,0.81 0.21,0.25 0.42,0.49 0.63,0.74 0.21,0.24 0.43,0.47 0.63,0.71 0.15,0.17 0.29,0.34 0.44,0.51 0.05,0.06 0.28,0.33 0.28,0.33 0,0 0.58,-0.52 0.62,-0.54 0.03,-0.03 0.05,-0.06 0.08,-0.08 0.18,-0.19 0.39,-0.36 0.57,-0.56 0.2,-0.2 0.4,-0.4 0.61,-0.59 0.24,-0.22 0.48,-0.44 0.7,-0.68 0.25,-0.25 0.52,-0.48 0.77,-0.73 0.26,-0.26 0.52,-0.51 0.78,-0.76 0.29,-0.26 0.55,-0.54 0.84,-0.8 0.28,-0.28 0.57,-0.55 0.86,-0.82 0.28,-0.28 0.58,-0.55 0.87,-0.83 0.26,-0.26 0.53,-0.51 0.8,-0.76 0.2,-0.19 0.4,-0.38 0.59,-0.57 0.12,-0.1 0.22,-0.22 0.34,-0.32 0.23,-0.11 -1.31,-3.29 -1.54,-3.18zM10.07,20.43c-1,-0.03 -2.98,0.14 -2.97,0.32 -0.02,0.12 -0.03,0.25 -0.05,0.38 -0.03,0.25 -0.05,0.51 -0.07,0.76 -0.01,0.31 -0.02,0.63 -0.03,0.94 -0.01,0.3 -0.01,0.61 -0.02,0.91 -0.01,0.32 0.01,0.65 0.02,0.97 0,0.32 0.02,0.64 0.04,0.96 0.01,0.33 0.05,0.65 0.09,0.98 0.03,0.31 0.07,0.63 0.11,0.94 0.04,0.3 0.09,0.61 0.14,0.91 0.06,0.31 0.14,0.62 0.23,0.93 0.1,0.36 0.27,0.7 0.45,1.03 0.18,0.33 0.4,0.65 0.64,0.94 0.21,0.27 0.45,0.52 0.69,0.76 0.25,0.26 0.53,0.5 0.81,0.73 0.3,0.25 0.62,0.46 0.95,0.67 0.34,0.21 0.7,0.39 1.07,0.56 0.38,0.18 0.77,0.34 1.17,0.49 0.42,0.16 0.84,0.28 1.27,0.4 0.46,0.12 0.92,0.23 1.38,0.32 0.5,0.09 1.01,0.17 1.52,0.23 0.5,0.06 1,0.11 1.5,0.16 0.51,0.05 1.03,0.08 1.54,0.11 0.53,0.03 1.05,0.04 1.58,0.06 0.51,0.01 1.02,0.02 1.53,0.03 0.49,-0 0.98,-0.01 1.47,-0.01 0.47,0 0.94,-0.01 1.41,-0.01 0.48,0.01 0.95,0.02 1.43,0.03 0.47,0.01 0.95,0.02 1.42,0.03 0.38,0 0.76,0.02 1.14,0.02 0.25,0.03 0.67,-3.48 0.42,-3.51 -0.36,0.01 -0.71,0.01 -1.07,0 -0.47,-0.01 -0.94,-0.02 -1.42,-0.03 -0.48,-0.01 -0.96,-0.02 -1.45,-0.03 -0.49,-0.01 -0.98,-0.01 -1.48,0 -0.48,0 -0.97,0.01 -1.45,0.02 -0.49,0 -0.97,-0 -1.46,-0.02 -0.49,-0.01 -0.98,-0.02 -1.48,-0.04 -0.48,-0.03 -0.95,-0.05 -1.43,-0.08 -0.48,-0.04 -0.96,-0.09 -1.44,-0.14 -0.45,-0.05 -0.89,-0.11 -1.34,-0.19 -0.39,-0.07 -0.78,-0.15 -1.16,-0.25 -0.34,-0.09 -0.69,-0.18 -1.02,-0.3 -0.31,-0.11 -0.62,-0.23 -0.93,-0.37 -0.27,-0.13 -0.54,-0.26 -0.8,-0.4 -0.22,-0.13 -0.44,-0.27 -0.64,-0.44 -0.2,-0.16 -0.39,-0.32 -0.56,-0.5 -0.17,-0.17 -0.34,-0.35 -0.48,-0.54 -0.11,-0.14 -0.22,-0.29 -0.3,-0.45 -0.08,-0.15 -0.15,-0.3 -0.18,-0.46 -0.06,-0.22 -0.11,-0.45 -0.14,-0.67 -0.04,-0.26 -0.08,-0.53 -0.12,-0.79 -0.04,-0.28 -0.06,-0.56 -0.09,-0.84 -0.03,-0.28 -0.04,-0.56 -0.06,-0.84 -0.01,-0.29 -0.01,-0.58 -0.02,-0.86 -0.01,-0.28 -0.01,-0.56 0,-0.84 0.01,-0.3 0.01,-0.6 0.02,-0.9 0.01,-0.25 0.01,-0.5 0.04,-0.75 0.02,-0.23 0.05,-0.45 0.08,-0.68 0.02,-0.17 0.03,-0.34 0.04,-0.51l0,-0.01c-0,-0.06 -0.23,-0.1 -0.56,-0.11z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="square" android:strokeLineJoin="miter" android:strokeWidth="0.30000001"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_reblog_disabled.xml b/app/src/main/res/drawable/ic_reblog_disabled.xml
deleted file mode 100644
index afafe08e..00000000
--- a/app/src/main/res/drawable/ic_reblog_disabled.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="42.519684"
-    android:viewportWidth="42.519684" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M37.79,3.12C36.3,4.63 34.82,6.16 33.35,7.7C30.19,11.06 26.99,14.39 23.77,17.7C20.33,21.24 16.82,24.71 13.31,28.17C10.93,30.56 8.49,32.88 5.98,35.13L5.26,35.74L7.3,38.15C7.57,37.93 7.83,37.7 8.09,37.48C10.63,35.2 13.11,32.84 15.53,30.41C19.05,26.94 22.58,23.46 26.03,19.9C29.26,16.59 32.46,13.25 35.63,9.88C37.08,8.36 38.55,6.85 40.03,5.36L37.79,3.12zM24.58,8.38C24.2,8.38 23.81,8.39 23.43,8.4C23.04,8.4 22.64,8.41 22.25,8.42C21.85,8.43 21.45,8.43 21.05,8.44C20.66,8.45 20.27,8.46 19.88,8.47C19.49,8.48 19.11,8.49 18.73,8.5C18.36,8.51 17.99,8.52 17.62,8.53C17.28,8.55 16.94,8.56 16.59,8.57C16.28,8.59 15.96,8.6 15.65,8.62C15.35,8.63 15.05,8.65 14.75,8.67C14.45,8.68 14.16,8.7 13.87,8.72C13.6,8.74 13.32,8.76 13.05,8.79C12.81,8.81 12.58,8.83 12.34,8.86C12.18,8.88 12.03,8.92 11.87,8.93L11.87,8.93C11.62,8.94 11.66,12.47 11.91,12.47C12.04,12.22 12.49,12.32 12.9,12.33C13.15,12.3 13.4,12.28 13.65,12.26C13.93,12.24 14.21,12.22 14.49,12.2C14.78,12.18 15.07,12.16 15.37,12.15C15.67,12.13 15.98,12.11 16.28,12.1C16.62,12.08 16.95,12.07 17.29,12.06C17.65,12.05 18.01,12.03 18.38,12.02C18.75,12.01 19.13,12 19.51,11.99C19.9,11.98 20.29,11.97 20.68,11.96C21.07,11.95 21.47,11.95 21.87,11.94C22.26,11.93 22.64,11.93 23.03,11.92C23.42,11.91 23.8,11.91 24.18,11.9C24.53,11.89 24.89,11.89 25.24,11.89C25.6,11.9 25.96,11.9 26.32,11.91C26.39,11.91 26.46,11.92 26.53,11.92C27.56,10.85 28.6,9.79 29.63,8.71C29.44,8.68 29.25,8.64 29.05,8.61C28.69,8.56 28.33,8.52 27.97,8.49C27.59,8.45 27.21,8.44 26.83,8.41C26.47,8.41 26.1,8.4 25.74,8.39C25.35,8.38 24.97,8.37 24.58,8.38zM8.31,9.3C8.31,9.3 7.63,10.27 7.6,10.33C7.48,10.51 7.36,10.69 7.24,10.87C7.15,10.99 7.07,11.11 6.99,11.23C6.45,12.01 6.35,12.11 6.2,12.35C6.13,12.45 6.07,12.55 6,12.64C5.73,13.03 5.45,13.41 5.17,13.79C4.89,14.17 4.61,14.55 4.32,14.92C4.04,15.29 3.76,15.65 3.49,16.02C3.24,16.35 2.98,16.68 2.73,17.01C2.49,17.33 2.25,17.64 2.01,17.96C1.78,18.24 1.56,18.54 1.34,18.83C1.14,19.1 0.94,19.37 0.73,19.64C0.54,19.88 0.39,20.14 0.2,20.38C0.09,20.51 0.14,20.45 0.04,20.59C-0.06,20.82 3.19,22.22 3.29,21.99C3.39,21.85 3.34,21.92 3.45,21.77C3.61,21.53 3.79,21.3 3.97,21.06C4.17,20.8 4.37,20.53 4.57,20.27C4.8,19.97 5.03,19.68 5.25,19.38C5.48,19.08 5.72,18.77 5.95,18.46C6.21,18.13 6.47,17.79 6.72,17.45C6.99,17.09 7.26,16.73 7.54,16.37C7.84,15.97 8.14,15.58 8.42,15.18C8.43,15.17 8.44,15.16 8.44,15.15C8.62,15.36 8.79,15.58 8.96,15.79C9.24,16.14 9.51,16.5 9.79,16.86C10.07,17.23 10.35,17.59 10.63,17.96C10.92,18.34 11.21,18.72 11.51,19.1C11.79,19.46 12.06,19.82 12.34,20.18C12.62,20.54 12.89,20.9 13.16,21.25C13.41,21.58 13.67,21.91 13.92,22.24C14.15,22.52 14.38,22.81 14.6,23.11C14.75,23.3 14.67,23.2 14.82,23.39C14.89,23.49 15.58,23.01 16.26,22.48C16.69,22.04 17.13,21.6 17.56,21.16C17.47,21.05 17.5,21.08 17.38,20.93C17.15,20.62 16.92,20.32 16.68,20.02C16.42,19.69 16.17,19.37 15.92,19.04C15.65,18.69 15.38,18.34 15.11,17.99C14.83,17.62 14.55,17.25 14.26,16.88C13.98,16.51 13.69,16.14 13.4,15.77C13.12,15.39 12.83,15.01 12.54,14.63C12.25,14.27 11.97,13.9 11.68,13.54C11.41,13.2 11.14,12.86 10.87,12.53C10.77,12.4 10.66,12.27 10.56,12.15C10.2,11.7 10.05,11.53 9.79,11.21C9.71,11.1 9.62,11 9.54,10.89C9.4,10.72 9.26,10.55 9.13,10.38C9.06,10.28 9,10.19 8.94,10.1C8.9,10.03 8.31,9.3 8.31,9.3zM35.23,13.22C34.33,14.17 33.42,15.1 32.51,16.04C32.52,16.07 32.53,16.11 32.54,16.14C32.6,16.4 32.66,16.66 32.71,16.91C32.77,17.18 32.8,17.46 32.83,17.74C32.88,18.03 32.9,18.33 32.91,18.63C32.94,18.95 32.95,19.26 32.96,19.58C32.96,19.9 32.96,20.23 32.96,20.56C32.97,20.87 32.97,21.18 32.98,21.49L32.98,22.35C32.99,22.61 32.99,22.87 32.99,23.13C33.03,23.38 36.53,22.9 36.5,22.65C36.49,22.4 36.49,22.14 36.49,21.88L36.48,21.02C36.48,20.7 36.47,20.39 36.47,20.08C36.46,19.74 36.46,19.4 36.45,19.06C36.44,18.7 36.41,18.34 36.39,17.98C36.36,17.62 36.32,17.27 36.27,16.91C36.22,16.56 36.17,16.21 36.09,15.86C36.01,15.53 35.94,15.19 35.84,14.86C35.74,14.55 35.64,14.23 35.53,13.93C35.44,13.69 35.34,13.46 35.23,13.22zM40.94,20.11C40.8,20.23 40.64,20.33 40.51,20.47C40.26,20.67 40.04,20.89 39.81,21.1C39.54,21.36 39.27,21.61 39,21.87C38.72,22.15 38.42,22.42 38.13,22.7C37.85,22.98 37.56,23.25 37.27,23.53C37,23.79 36.72,24.05 36.46,24.32C36.18,24.58 35.91,24.84 35.64,25.1C35.4,25.34 35.15,25.58 34.9,25.81C34.72,25.97 34.56,26.14 34.39,26.31C34.28,26.19 34.17,26.06 34.06,25.93C33.82,25.65 33.58,25.36 33.34,25.08C33.08,24.77 32.82,24.46 32.56,24.15C32.29,23.84 32.03,23.52 31.76,23.21C31.49,22.91 31.22,22.6 30.95,22.31C30.71,22.05 30.47,21.79 30.24,21.53C30.04,21.31 29.85,21.09 29.66,20.88C29.52,20.74 29.39,20.59 29.25,20.46C29.08,20.28 26.58,22.78 26.75,22.96C26.86,23.06 26.96,23.18 27.07,23.29C27.25,23.49 27.44,23.7 27.62,23.91C27.87,24.18 28.11,24.45 28.36,24.71C28.62,24.99 28.87,25.28 29.12,25.57C29.38,25.87 29.65,26.18 29.91,26.48C30.16,26.78 30.41,27.08 30.65,27.38C30.9,27.66 31.14,27.95 31.38,28.24C31.61,28.52 31.85,28.78 32.08,29.05C32.29,29.3 32.5,29.54 32.71,29.79C32.92,30.03 33.13,30.26 33.34,30.5C33.49,30.67 33.64,30.84 33.78,31.01C33.83,31.06 34.06,31.33 34.06,31.33C34.06,31.33 34.65,30.82 34.68,30.79C34.71,30.77 34.74,30.74 34.76,30.71C34.95,30.52 35.15,30.35 35.33,30.15C35.53,29.95 35.73,29.75 35.94,29.56C36.18,29.34 36.42,29.12 36.64,28.89C36.89,28.63 37.16,28.41 37.41,28.16C37.67,27.9 37.93,27.65 38.19,27.4C38.48,27.14 38.75,26.86 39.03,26.59C39.31,26.32 39.6,26.04 39.89,25.77C40.17,25.49 40.46,25.22 40.75,24.94C41.02,24.69 41.28,24.43 41.55,24.18C41.75,23.99 41.95,23.8 42.14,23.61C42.26,23.51 42.36,23.39 42.48,23.29C42.71,23.18 41.17,20 40.94,20.11zM10.07,20.43C9.08,20.4 7.09,20.57 7.11,20.75C7.09,20.88 7.07,21.01 7.05,21.13C7.02,21.39 7,21.64 6.98,21.9C6.97,22.21 6.96,22.52 6.95,22.84C6.94,23.14 6.94,23.45 6.93,23.75C6.93,24.08 6.94,24.4 6.95,24.73C6.95,25.05 6.96,25.37 6.99,25.69C7,26.02 7.04,26.34 7.07,26.67C7.1,26.98 7.14,27.29 7.19,27.61C7.23,27.91 7.27,28.21 7.33,28.51C7.39,28.83 7.47,29.14 7.56,29.44C7.66,29.8 7.83,30.14 8,30.47C8.94,29.6 9.87,28.73 10.79,27.84C10.78,27.75 10.75,27.66 10.73,27.56C10.69,27.3 10.66,27.03 10.62,26.77C10.58,26.49 10.56,26.21 10.53,25.93C10.5,25.64 10.48,25.36 10.47,25.08C10.46,24.8 10.45,24.51 10.45,24.22C10.44,23.94 10.43,23.66 10.45,23.38C10.45,23.08 10.46,22.79 10.47,22.49C10.48,22.24 10.48,21.98 10.51,21.73C10.53,21.51 10.56,21.28 10.59,21.06C10.61,20.89 10.62,20.72 10.63,20.54L10.63,20.54C10.63,20.48 10.41,20.44 10.07,20.43zM16.86,31.93C15.87,32.89 14.88,33.84 13.89,34.8C14.13,34.87 14.37,34.96 14.61,35.02C15.07,35.15 15.53,35.25 15.99,35.34C16.5,35.43 17.01,35.51 17.52,35.57C18.01,35.63 18.52,35.68 19.02,35.73C19.53,35.78 20.04,35.81 20.56,35.84C21.08,35.87 21.61,35.88 22.14,35.89C22.65,35.9 23.16,35.92 23.67,35.92C24.16,35.92 24.65,35.91 25.14,35.91C25.61,35.91 26.08,35.9 26.55,35.9C27.02,35.9 27.5,35.91 27.97,35.92C28.45,35.93 28.92,35.94 29.4,35.95C29.78,35.95 30.16,35.97 30.54,35.96C30.79,35.99 31.21,32.48 30.96,32.45C30.6,32.46 30.25,32.46 29.89,32.45C29.42,32.45 28.95,32.44 28.47,32.43C27.99,32.42 27.51,32.41 27.03,32.4C26.54,32.39 26.05,32.39 25.55,32.4C25.07,32.41 24.58,32.41 24.1,32.42C23.61,32.42 23.13,32.41 22.64,32.4C22.15,32.39 21.66,32.38 21.17,32.36C20.69,32.33 20.21,32.31 19.74,32.28C19.26,32.23 18.78,32.18 18.3,32.13C17.85,32.08 17.41,32.02 16.96,31.95C16.93,31.94 16.89,31.93 16.86,31.93z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="square" android:strokeLineJoin="miter" android:strokeWidth="0.30000001"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_reblogged.xml b/app/src/main/res/drawable/ic_reblogged.xml
deleted file mode 100644
index 1f9d8b64..00000000
--- a/app/src/main/res/drawable/ic_reblogged.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="16dp" android:viewportHeight="566.92914"
-    android:viewportWidth="566.92914" android:width="16dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M141.7,0C63.2,0 0,63.2 0,141.7L0,425.2C0,503.7 63.2,566.9 141.7,566.9L425.2,566.9C503.7,566.9 566.9,503.7 566.9,425.2L566.9,141.7C566.9,63.2 503.7,0 425.2,0L141.7,0zM177.2,124L354.3,124C432.9,124 496.1,182.6 496.1,265.7L496.1,336.6L549.2,336.6L460.6,425.2L372,336.6L425.2,336.6L425.2,265.7C425.2,226.4 393.6,194.9 354.3,194.9L248,194.9L177.2,124zM106.3,141.7L194.9,230.3L141.7,230.3L141.7,301.2C141.7,340.5 173.3,372 212.6,372L318.9,372L389.8,442.9L212.6,442.9C134,442.9 70.9,384.3 70.9,301.2L70.9,230.3L17.7,230.3L106.3,141.7z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="butt" android:strokeLineJoin="round" android:strokeWidth="0"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_reply.xml b/app/src/main/res/drawable/ic_reply.xml
deleted file mode 100644
index 3df673b9..00000000
--- a/app/src/main/res/drawable/ic_reply.xml
+++ /dev/null
@@ -1,7 +0,0 @@
-<vector android:height="24dp" android:viewportHeight="42.519684"
-    android:viewportWidth="42.519684" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
-    <path android:fillAlpha="1" android:fillColor="#ffffff"
-        android:pathData="M15.18,9.94C15.08,9.91 15.02,9.91 15.01,9.94C14.88,10.06 14.74,10.21 14.61,10.35C14.32,10.68 14.04,11 13.75,11.32C13.39,11.73 13.03,12.13 12.67,12.53C12.28,12.95 11.89,13.37 11.49,13.79C11.11,14.2 10.72,14.62 10.33,15.04C9.96,15.44 9.58,15.83 9.21,16.24C8.88,16.61 8.55,16.97 8.21,17.33C7.9,17.65 7.6,18 7.29,18.34C7.2,18.44 7.11,18.53 7.02,18.63C6.83,18.76 6.46,19.18 6.08,19.65C5.96,19.77 5.85,19.89 5.73,20.01C5.5,20.25 5.27,20.49 5.04,20.73C5,20.77 4.97,20.81 4.93,20.85C4.92,20.88 4.98,20.95 5.08,21.03C4.98,21.2 4.92,21.33 4.96,21.36C5.04,21.42 5.11,21.49 5.18,21.56C5.41,21.78 5.64,21.99 5.88,22.2C6.2,22.47 6.53,22.73 6.86,22.99C7.25,23.3 7.64,23.59 8.03,23.9C8.44,24.21 8.84,24.54 9.24,24.86C9.65,25.2 10.05,25.54 10.45,25.88C10.86,26.23 11.27,26.58 11.67,26.94C12.08,27.3 12.48,27.66 12.89,28.02C13.28,28.38 13.68,28.74 14.07,29.1C14.46,29.47 14.86,29.83 15.25,30.19C15.64,30.55 16.03,30.91 16.41,31.27C16.47,31.33 16.53,31.38 16.59,31.44C16.77,31.61 19.22,29.07 19.04,28.89C18.98,28.83 18.91,28.77 18.84,28.71C18.45,28.34 18.06,27.97 17.66,27.6C17.27,27.24 16.87,26.88 16.48,26.52C16.08,26.15 15.68,25.78 15.28,25.42C14.87,25.05 14.45,24.68 14.04,24.32C13.62,23.94 13.2,23.58 12.77,23.21C12.36,22.86 11.94,22.5 11.51,22.16C11.1,21.82 10.68,21.49 10.26,21.15C10.09,21.02 9.91,20.88 9.74,20.75C9.96,20.5 10.18,20.25 10.41,20.01C10.7,19.68 11.01,19.36 11.31,19.04C11.65,18.67 11.98,18.3 12.32,17.93C12.69,17.54 13.05,17.14 13.42,16.75C13.8,16.34 14.19,15.92 14.58,15.51C14.98,15.08 15.38,14.65 15.78,14.21C16.14,13.8 16.5,13.39 16.87,12.98C17.14,12.67 17.4,12.36 17.69,12.08C17.84,11.92 17.99,11.76 18.13,11.6C18.23,11.41 15.85,10.11 15.18,9.94zM14.04,18.52L14.04,18.53C13.79,18.49 13.35,22 13.6,22.03L14.05,22.03L14.82,22.03L15.82,22.03C16.22,22.03 16.62,22.03 17.03,22.04C17.5,22.06 17.97,22.07 18.44,22.08C18.93,22.1 19.43,22.11 19.92,22.12C20.4,22.13 20.88,22.15 21.37,22.17C21.85,22.19 22.34,22.23 22.82,22.26C23.3,22.29 23.77,22.33 24.25,22.37C24.7,22.41 25.15,22.46 25.6,22.52C26.03,22.58 26.47,22.65 26.89,22.73C27.3,22.82 27.71,22.92 28.11,23.03C28.48,23.13 28.85,23.26 29.21,23.4C29.54,23.54 29.85,23.71 30.16,23.89C30.48,24.08 30.78,24.3 31.08,24.53C31.37,24.76 31.63,25.02 31.88,25.29C32.12,25.55 32.33,25.84 32.52,26.14C32.72,26.48 32.88,26.84 33.02,27.21C33.18,27.66 33.31,28.13 33.42,28.59C33.54,29.13 33.64,29.67 33.73,30.21C33.82,30.81 33.89,31.42 33.96,32.02C34.03,32.67 34.1,33.32 34.15,33.97C34.16,34.08 34.16,34.19 34.17,34.3C34.22,34.55 37.69,33.87 37.64,33.62C37.63,33.51 37.63,33.4 37.62,33.29C37.56,32.6 37.48,31.92 37.4,31.24C37.32,30.58 37.24,29.92 37.13,29.27C37.03,28.66 36.91,28.04 36.76,27.44C36.61,26.85 36.44,26.27 36.21,25.71C35.99,25.16 35.75,24.63 35.43,24.13C35.13,23.68 34.81,23.24 34.44,22.84C34.08,22.47 33.72,22.11 33.31,21.79C32.92,21.49 32.53,21.19 32.1,20.94C31.67,20.67 31.22,20.42 30.75,20.22C30.29,20.04 29.83,19.85 29.36,19.72C28.89,19.58 28.41,19.46 27.92,19.35C27.43,19.25 26.94,19.16 26.44,19.08C25.95,19.01 25.45,18.95 24.95,18.9C24.46,18.86 23.96,18.81 23.47,18.79C22.96,18.75 22.45,18.71 21.95,18.69C21.44,18.66 20.93,18.63 20.43,18.62C19.94,18.61 19.45,18.6 18.96,18.58C18.49,18.57 18.02,18.56 17.55,18.54C17.12,18.53 16.69,18.51 16.26,18.52L15.26,18.52L14.5,18.52L14.04,18.52z"
-        android:strokeAlpha="1" android:strokeColor="#00000000"
-        android:strokeLineCap="square" android:strokeLineJoin="miter" android:strokeWidth="0.30000001"/>
-</vector>
diff --git a/app/src/main/res/drawable/ic_settings_24dp.xml b/app/src/main/res/drawable/ic_settings_24dp.xml
new file mode 100644
index 00000000..4c81fb46
--- /dev/null
+++ b/app/src/main/res/drawable/ic_settings_24dp.xml
@@ -0,0 +1,9 @@
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+        android:width="24dp"
+        android:height="24dp"
+        android:viewportWidth="24.0"
+        android:viewportHeight="24.0">
+    <path
+        android:fillColor="@color/toolbar_icon_dark"
+        android:pathData="M19.43,12.98c0.04,-0.32 0.07,-0.64 0.07,-0.98s-0.03,-0.66 -0.07,-0.98l2.11,-1.65c0.19,-0.15 0.24,-0.42 0.12,-0.64l-2,-3.46c-0.12,-0.22 -0.39,-0.3 -0.61,-0.22l-2.49,1c-0.52,-0.4 -1.08,-0.73 -1.69,-0.98l-0.38,-2.65C14.46,2.18 14.25,2 14,2h-4c-0.25,0 -0.46,0.18 -0.49,0.42l-0.38,2.65c-0.61,0.25 -1.17,0.59 -1.69,0.98l-2.49,-1c-0.23,-0.09 -0.49,0 -0.61,0.22l-2,3.46c-0.13,0.22 -0.07,0.49 0.12,0.64l2.11,1.65c-0.04,0.32 -0.07,0.65 -0.07,0.98s0.03,0.66 0.07,0.98l-2.11,1.65c-0.19,0.15 -0.24,0.42 -0.12,0.64l2,3.46c0.12,0.22 0.39,0.3 0.61,0.22l2.49,-1c0.52,0.4 1.08,0.73 1.69,0.98l0.38,2.65c0.03,0.24 0.24,0.42 0.49,0.42h4c0.25,0 0.46,-0.18 0.49,-0.42l0.38,-2.65c0.61,-0.25 1.17,-0.59 1.69,-0.98l2.49,1c0.23,0.09 0.49,0 0.61,-0.22l2,-3.46c0.12,-0.22 0.07,-0.49 -0.12,-0.64l-2.11,-1.65zM12,15.5c-1.93,0 -3.5,-1.57 -3.5,-3.5s1.57,-3.5 3.5,-3.5 3.5,1.57 3.5,3.5 -1.57,3.5 -3.5,3.5z"/>
+</vector>
diff --git a/app/src/main/res/layout/fragment_view_media.xml b/app/src/main/res/layout/fragment_view_media.xml
index 50de8b59..1126dbe0 100644
--- a/app/src/main/res/layout/fragment_view_media.xml
+++ b/app/src/main/res/layout/fragment_view_media.xml
@@ -3,12 +3,8 @@
     android:layout_width="match_parent"
     android:layout_height="match_parent"
     android:background="#60000000">
-
-    <com.android.volley.toolbox.NetworkImageView
+    <uk.co.senab.photoview.PhotoView
         android:id="@+id/view_media_image"
         android:layout_width="match_parent"
-        android:layout_height="match_parent"
-        android:layout_centerInParent="true"
-        android:scaleType="fitCenter" />
-
+        android:layout_height="match_parent" />
 </RelativeLayout>
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_account.xml b/app/src/main/res/layout/item_account.xml
index 7f1f9dc1..44f5739c 100644
--- a/app/src/main/res/layout/item_account.xml
+++ b/app/src/main/res/layout/item_account.xml
@@ -3,16 +3,18 @@
     android:orientation="vertical"
     android:layout_width="match_parent"
     android:layout_height="wrap_content"
-    android:padding="16dp"
+    android:paddingLeft="16dp"
+    android:paddingRight="16dp"
     android:id="@+id/account_container">
 
     <RelativeLayout
+        android:paddingTop="8dp"
         android:layout_width="match_parent"
         android:layout_height="wrap_content">
 
         <ImageView
-            android:layout_width="wrap_content"
-            android:layout_height="wrap_content"
+            android:layout_width="48dp"
+            android:layout_height="48dp"
             android:id="@+id/account_avatar"
             android:layout_marginRight="10dp" />
 
@@ -20,6 +22,7 @@
             android:layout_width="wrap_content"
             android:layout_height="wrap_content"
             android:orientation="vertical"
+
             android:layout_centerVertical="true"
             android:layout_toRightOf="@id/account_avatar">
 
@@ -27,12 +30,14 @@
                 android:layout_width="wrap_content"
                 android:layout_height="wrap_content"
                 android:id="@+id/account_display_name"
+                android:text="Display name"
                 android:textColor="?android:textColorPrimary"
                 android:textStyle="normal|bold" />
 
             <TextView
                 android:layout_width="wrap_content"
                 android:layout_height="wrap_content"
+                android:text="\@username"
                 android:textColor="?android:textColorSecondary"
                 android:id="@+id/account_username" />
 
@@ -45,6 +50,7 @@
         android:layout_height="wrap_content"
         android:id="@+id/account_note"
         android:paddingTop="4dp"
+        android:paddingBottom="8dp"
         android:textColor="?android:textColorTertiary" />
 
 </LinearLayout>
\ No newline at end of file
diff --git a/app/src/main/res/layout/tab_main.xml b/app/src/main/res/layout/tab_main.xml
deleted file mode 100644
index 5684fffd..00000000
--- a/app/src/main/res/layout/tab_main.xml
+++ /dev/null
@@ -1,14 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
-    android:layout_width="match_parent"
-    android:layout_height="match_parent">
-
-    <TextView
-        android:layout_width="wrap_content"
-        android:layout_height="wrap_content"
-        android:id="@+id/title"
-        android:layout_centerInParent="true"
-        android:textAllCaps="true"
-        android:textStyle="normal|bold" />
-
-</RelativeLayout>
\ No newline at end of file
diff --git a/app/src/main/res/menu/account_toolbar.xml b/app/src/main/res/menu/account_toolbar.xml
index 3188882e..64b2f22f 100644
--- a/app/src/main/res/menu/account_toolbar.xml
+++ b/app/src/main/res/menu/account_toolbar.xml
@@ -10,6 +10,10 @@
         android:title="@string/action_follow"
         app:showAsAction="never" />
 
+    <item android:id="@+id/action_mute"
+        android:title="@string/action_mute"
+        app:showAsAction="never" />
+
     <item android:id="@+id/action_block"
         android:title="@string/action_block"
         app:showAsAction="never" />
diff --git a/app/src/main/res/menu/main_toolbar.xml b/app/src/main/res/menu/main_toolbar.xml
deleted file mode 100644
index 2517ee5d..00000000
--- a/app/src/main/res/menu/main_toolbar.xml
+++ /dev/null
@@ -1,31 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<menu
-    xmlns:android="http://schemas.android.com/apk/res/android"
-    xmlns:app="http://schemas.android.com/apk/res-auto">
-
-    <item
-        android:id="@+id/action_view_profile"
-        android:title="@string/action_view_profile"
-        app:showAsAction="never" />
-
-    <item
-        android:id="@+id/action_view_preferences"
-        android:title="@string/action_view_preferences"
-        app:showAsAction="never" />
-
-    <item
-        android:id="@+id/action_view_favourites"
-        android:title="@string/action_view_favourites"
-        app:showAsAction="never" />
-
-    <item
-        android:id="@+id/action_view_blocks"
-        android:title="@string/action_view_blocks"
-        app:showAsAction="never" />
-
-    <item
-        android:id="@+id/action_logout"
-        android:title="@string/action_logout"
-        app:showAsAction="never" />
-
-</menu>
diff --git a/app/src/main/res/menu/status_more.xml b/app/src/main/res/menu/status_more.xml
index 6886ea86..33617c9d 100644
--- a/app/src/main/res/menu/status_more.xml
+++ b/app/src/main/res/menu/status_more.xml
@@ -1,8 +1,8 @@
 <?xml version="1.0" encoding="utf-8"?>
 <menu xmlns:android="http://schemas.android.com/apk/res/android">
-
-    <item android:id="@+id/status_follow"
-        android:title="@string/action_follow" />
+    <item
+        android:id="@+id/status_share"
+        android:title="@string/action_share"/>
     <item android:title="@string/action_block"
         android:id="@+id/status_block" />
     <item android:title="@string/action_report"
diff --git a/app/src/main/res/transition/activity_slide.xml b/app/src/main/res/transition/activity_slide.xml
deleted file mode 100644
index cd343707..00000000
--- a/app/src/main/res/transition/activity_slide.xml
+++ /dev/null
@@ -1,3 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<slide xmlns:android="http://schemas.android.com/apk/res/"
-    android:duration="1000"/>
\ No newline at end of file
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index 4fb113b8..c48f4a5e 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -91,7 +91,7 @@
     <string name="notification_follow_format">%s followed you</string>
 
     <string name="report_username_format">Reporting @%s</string>
-    <string name="report_comment_hint">Additional Comments?</string>
+    <string name="report_comment_hint">Additional comments?</string>
 
     <string name="action_compose">Compose</string>
     <string name="action_login">Login with Mastodon</string>
@@ -113,8 +113,8 @@
     <string name="action_view_profile">Profile</string>
     <string name="action_view_preferences">Preferences</string>
     <string name="action_view_favourites">Favourites</string>
-    <string name="action_view_blocks">Blocked Users</string>
-    <string name="action_open_in_web">Open In Web</string>
+    <string name="action_view_blocks">Blocked users</string>
+    <string name="action_open_in_web">Open in browser</string>
     <string name="action_set_time">Set</string>
 
     <string name="confirmation_send">Toot!</string>
@@ -143,18 +143,24 @@
     <string name="notification_service_one_mention">Mention from %s</string>
 
     <string name="pref_title_notification_settings">Notifications</string>
-    <string name="pref_title_pull_notifications">Enable Pull Notifcations</string>
-    <string name="pref_summary_pull_notifications">check for notifications periodically</string>
-    <string name="pref_title_pull_notification_check_interval">Check Interval</string>
-    <string name="pref_summary_pull_notification_check_interval">how often to pull</string>
+    <string name="pref_title_pull_notifications">Enable pull notifcations</string>
+    <string name="pref_summary_pull_notifications">Check for notifications periodically</string>
+    <string name="pref_title_pull_notification_check_interval">Check interval</string>
+    <string name="pref_summary_pull_notification_check_interval">How often to pull</string>
     <string name="pref_title_notification_alert_sound">Notify with a sound</string>
     <string name="pref_title_notification_style_vibrate">Notify with vibration</string>
     <string name="pref_title_notification_style_light">Notify with light</string>
     <string name="pref_title_appearance_settings">Appearance</string>
-    <string name="pref_title_light_theme">Use The Light Theme</string>
+    <string name="pref_title_light_theme">Use the Light Theme</string>
     <string name="action_submit">Submit</string>
     <string name="action_photo_pick">Add media</string>
     <string name="action_compose_options">Privacy options</string>
     <string name="login_success">Welcome back!</string>
+    <string name="action_share">Share</string>
+    <string name="send_status_to">Share toot URL to...</string>
+    <string name="action_mute">Mute</string>
+    <string name="action_unmute">Unmute</string>
+    <string name="error_unmuting">That user wasn\'t unmuted.</string>
+    <string name="error_muting">That user wasn\'t muted.</string>
 
 </resources>
diff --git a/app/src/main/res/values/styles.xml b/app/src/main/res/values/styles.xml
index df0430db..8546465d 100644
--- a/app/src/main/res/values/styles.xml
+++ b/app/src/main/res/values/styles.xml
@@ -55,6 +55,16 @@
         <item name="notification_icon_tint">@color/notification_icon_tint_dark</item>
         <item name="report_status_background_color">@color/report_status_background_dark</item>
         <item name="report_status_divider_drawable">@drawable/report_status_divider_dark</item>
+
+        <item name="material_drawer_background">@color/color_primary_dark</item>
+        <item name="material_drawer_primary_text">@color/text_color_primary_dark</item>
+        <item name="material_drawer_primary_icon">@color/toolbar_icon_dark</item>
+        <item name="material_drawer_secondary_text">@color/text_color_secondary_dark</item>
+        <item name="material_drawer_hint_text">@color/text_color_tertiary_dark</item>
+        <item name="material_drawer_divider">@color/color_primary_dark_dark</item>
+        <item name="material_drawer_selected">@color/window_background_dark</item>
+        <item name="material_drawer_selected_text">@color/text_color_primary_dark</item>
+        <item name="material_drawer_header_selection_text">@color/text_color_primary_dark</item>
     </style>
 
     <style name="AppTheme.ImageButton.Dark" parent="@style/Widget.AppCompat.Button.Borderless.Colored">