Tecnologías Móviles

clase 12

Ejercicio de la clase anterior

Agregar Tabs

Mostrar Mensajes

Mostrar Chats

Ejercicio de la clase anterior

public class ApiClientService extends Service implements SharedPreferences.OnSharedPreferenceChangeListener {
    ...
    protected void setEmail() {
        mEmail = mPreferences.getString("email", "amoyano@iua.edu.ar");
        mBinder.login();
    }

    public void onCreate() {
        ...
        setEmail();
    }

    public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) {
        ...
        if (key.equals("email")) {
            setEmail();
        }
    }

    ...
}

ApiClientService

Ejercicio de la clase anterior

public class ApiClientService extends Service implements SharedPreferences.OnSharedPreferenceChangeListener {
    ...
    public class APIBinder extends Binder {
        ...
        public void login()  {
            JSONObject data = new JSONObject();
            try {
                data.putOpt("mail", mEmail);
                RestClient.post(mUrl + "/login", data, new RestClient.Result() {
                    @Override
                    public void onResult(Object result) {
                        mLoggedIn=true;
                        db.setCurrentUser((JSONObject) result);
                        findUsers();
                        findChats();
                    }

                    @Override
                    public void onError(String message) {

                        mLoggedIn=false;
                    }
                });
            } catch (JSONException e) {
                mLoggedIn=false;
                e.printStackTrace();
            } catch (IOException e) {
                mLoggedIn=false;
                e.printStackTrace();
            }

        }
        ...
    }
    ...
}

ApiClientService

Ejercicio de la clase anterior

public class ApiClientService extends Service implements 
                SharedPreferences.OnSharedPreferenceChangeListener {
    ...
    public class APIBinder extends Binder {
        ...
        public void findChats(final RestClient.Result handler) {
            RestClient.get(mUrl + "chat/", new RestClient.Result() {
                @Override
                public void onResult(Object result) {
                    JSONArray resArray = (JSONArray) result;
                    db.setChats(resArray);
                    for(int i = 0; i < resArray.length(); i++) {
                        JSONObject chat = null;
                        try {
                            chat = resArray.getJSONObject(i);
                            findMsgs(chat.getString("_id"));
                        } catch (JSONException e) {
                            e.printStackTrace();
                        }
                    }
                    if(handler != null) handler.onResult(null);
                }

                @Override
                public void onError(String message) {}
            });
        }
        ...
    }
    ...
}

ApiClientService

Ejercicio de la clase anterior

public class ApiClientService extends Service implements 
                SharedPreferences.OnSharedPreferenceChangeListener {
    ...
    public class APIBinder extends Binder {
        ...
        public void findMsgs(final String chatId) { ... }

        public void findUsers() { ... }

        public void addChat(final JSONObject chat, final RestClient.Result handler) { ... }

        public void addMsg(final String chatId, final JSONObject msg) { ... }

        public void addUser(final JSONObject usuario) { ... }

        public void setUser(JSONObject usuario) { ... }

        public void rmUser(JSONObject usuario) { ... }
        ...
    }
    ...
}

ApiClientService

Ejercicio de la clase anterior

public class UserContract {
    ...
    public class ChatTable implements BaseColumns {
        public static final String TABLE_NAME = "chat";
        public static final String MONGO_ID = "mongo_id";
        public static final String TO = "toUser";
        public static final String DATE = "date";
        public static final String MSG = "msg";
        public static final String FROM = "fromUser";
        public static final String TABLE_CREATE =
                "CREATE TABLE " + TABLE_NAME + " (" +
        ...
    }

    public class MsgTable implements BaseColumns {
        public static final String TABLE_NAME = "msg";
        public static final String MONGO_ID = "mongo_id";
        public static final String MSG = "msg";
        public static final String DATE = "date";
        public static final String CHAT = "chat";
        public static final String FROM = "fromUser";
        public static final String TABLE_CREATE =
                "CREATE TABLE " + TABLE_NAME + " (" +
        ...
    }
}

UserContract

Ejercicio de la clase anterior

public class UserDBHelper extends SQLiteOpenHelper {
    ...
    public void setCurrentUser(JSONObject user) {
        try {
            mCurrentUser = user.getString("_id");
            mPreferences.edit().putString("current-user", mCurrentUser).commit();
        } catch (JSONException e) {
            e.printStackTrace();
        }
    }

    public String getCurrentUser() { ... }

    public JSONObject getChat(String id) { ... }

    public JSONObject getChatByUser(String id) { ... }

    private JSONObject chatRowToJSON(Cursor cursor) { ... }

    public UserDBHelper setChats(JSONArray chats) { ... }

    private void doCreateChat(JSONObject chat) { ... }

    public UserDBHelper createChat(JSONObject chat, RestClient.Result handler) { ... }

    public int countChat() { ... }

    public JSONObject readChat(Integer pos) { ... }

    public UserDBHelper setMsgs(String chatId, JSONArray msgs) { ... }

    private void doCreateMsg(String chatId, JSONObject msg) { ... }

    public int countMsg(String chatId) { ... }

    public JSONObject readMsg(String chatId, int position) { ... }

    private JSONObject msgRowToJSON(Cursor cursor) { ... }

    public UserDBHelper createMsg(String chatId, JSONObject msg) { ... }
}

UserDBHelper

Ejercicio de la clase anterior

public class ListFragment extends Fragment {
    ...
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        ...
        list.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                final String uid = adapter.getItem(position).optString("mongo_id");
                JSONObject chat = mDB.getChatByUser(uid);
                if(chat != null) {
                    main.openMsgs(chat.optString("mongo_id"));
                } else {
                    RestClient.Result handler = new RestClient.Result() {
                        @Override
                        public void onResult(Object result) {
                            JSONObject chat = mDB.getChatByUser(uid);
                            main.openMsgs(chat.optString("mongo_id"));
                        }

                        @Override
                        public void onError(String message) {
                            Toast.makeText(getActivity(), message, Toast.LENGTH_SHORT);
                        }
                    };
                    JSONObject newChat = new JSONObject();
                    try {
                        newChat.put("to", uid);
                        mDB.createChat(newChat, handler);
                    } catch (JSONException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
        ...
    }
}

ListFragment

Ejercicio de la clase anterior

public class ChatFragment extends Fragment {

    MainActivity main;
    public ChatFragment() {
        // Required empty public constructor
    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment

        main = (MainActivity) getActivity();
        View root = inflater.inflate(R.layout.fragment_chat, container, false);
        ListView chats = (ListView) root.findViewById(R.id.chatList);
        chats.setAdapter(ChatAdapter.getInstance(main));
        chats.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                JSONObject chat = ChatAdapter.getInstance().getItem(position);
                try {
                    main.openMsgs(chat.getString("mongo_id"));
                } catch (JSONException e) {
                    e.printStackTrace();
                }
            }
        });
        return root;
    }


}

ChatFragment

Ejercicio de la clase anterior

public class MainActivity extends AppCompatActivity {

    ...

    public void openMsgs(String chatId) {
        MsgFragment msg = new MsgFragment();
        Bundle args = new Bundle();
        args.putString("chatId", chatId);
        msg.setArguments(args);
        FragmentTransaction transaction = manager.beginTransaction();
        transaction.replace(R.id.container, msg);
        transaction.addToBackStack("messages");
        transaction.commit();
    }

    ...
}

MainActivity

Ejercicio de la clase anterior

public class MsgFragment extends Fragment {
    ...
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        MainActivity main = (MainActivity) getActivity();
        Bundle args = getArguments();
        final String chatId = args.getString("chatId");
        View root = inflater.inflate(R.layout.fragment_msg, container, false);
        ListView msgs = (ListView) root.findViewById(R.id.msgList);
        msgs.setAdapter(MsgAdapter.getInstance(main, chatId));
        final EditText msgText = (EditText) root.findViewById(R.id.msg);
        ImageView send = (ImageView) root.findViewById(R.id.send);
        send.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                UserDBHelper mDB = UserDBHelper.getInstance(getActivity());
                JSONObject msg = new JSONObject();
                try {
                    msg.put("chat", chatId);
                    msg.put("from", mDB.getCurrentUser());
                    msg.put("message", msgText.getText().toString());
                    mDB.createMsg(chatId, msg);
                    msgText.setText("");
                } catch (JSONException e) {
                    e.printStackTrace();
                }

            }
        });
        return root;
    }

}

MsgFragment

Ejercicio de la clase anterior

<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
    ...    
    <EditTextPreference
        android:defaultValue="amoyano@iua.edu.ar"
        android:inputType="textEmailAddress"
        android:key="email"
        android:title="@string/email_title" />
</PreferenceScreen>

res/xml/settings.xml

public class SettingsFragment extends PreferenceFragmentCompat {
    ...
    public void onCreate(Bundle savedInstanceState) {
        ...
        bindPreferenceSummaryToValue(findPreference("email"));
    }
}

SettingsFragment

    ...
        <ListView
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:id="@+id/msgList"
            android:stackFromBottom="true"
            android:transcriptMode="alwaysScroll">
        </ListView>
    ...

res/layout/fragment_msg.xml

WebSockets

Es una tecnología que proporciona un canal de comunicación bidireccional y full-duplex sobre un único socket TCP.

 

Está diseñada para ser implementada en navegadores y servidores web, pero puede utilizarse por cualquier aplicación cliente/servidor.

 

Permite la emisión de eventos del lado del servidor.

 

Una librería que implementa WebSockets para muchas plataformas, incluyendo a Android, es Socket.IO

WebSockets

app/build.gradle
dependencies {
    ...
    compile 'com.github.nkzawa:socket.io-client:0.3.0'
}

AndroidManifest.xml

<!-- app/AndroidManifest.xml -->
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
    <uses-permission android:name="android.permission.INTERNET" />
    ...
</manifest>

WebSockets

MainActivity o Fragment
import com.github.nkzawa.socketio.client.IO;
import com.github.nkzawa.socketio.client.Socket;

private Socket mSocket;
{
    try {
        mSocket = IO.socket("http://chat.socket.io");
    } catch (URISyntaxException e) {}
}

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    mSocket.connect();
}

WebSockets

Emitiendo Eventos
private EditText mInputMessageView;

private void attemptSend() {
    String message = mInputMessageView.getText().toString().trim();
    if (TextUtils.isEmpty(message)) {
        return;
    }

    mInputMessageView.setText("");
    mSocket.emit("new message", message);
}
Escuchando Eventos
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    mSocket.on("new message", onNewMessage);
    mSocket.connect();
}

WebSockets

Listener
private Emitter.Listener onNewMessage = new Emitter.Listener() {
    @Override
    public void call(final Object.. args) {
        getActivity().runOnUiThread(new Runnable() {
            @Override
            public void run() {
                JSONObject data = (JSONObject) args[0];
                String username;
                String message;
                try {
                    username = data.getString("username");
                    message = data.getString("message");
                } catch (JSONException e) {
                    return;
                }

                // add the message to view
                addMessage(username, message);
            }
        });
    }
};

WebSockets

Desconectar
@Override
public void onDestroy() {
    super.onDestroy();

    mSocket.disconnect();
    mSocket.off("new message", onNewMessage);
}

Ejercicio

Basado en el ejercicio de la clase anterior, realizar los siguientes puntos teniendo en cuenta la base a donde apunta la api (https://iua-tm-api.herokuapp.com/) utilizar los siguientes eventos:

  • Eventos Escuchados por el servidor:
    • 'post:login' (String mail): Realiza el Login de acuerdo al mail enviado.
    • 'get:user' (JSONObject query): Realiza una búsqueda de usuarios de acuerdo al criterio especificado (query).
    • 'get:chat' (): Realiza la búsqueda de chats a los cuales pertenece el usuario logueado.
    • 'post:chat' (String to): Inserta un nuevo chat, y pone como participante al usuario identificado por su id (to).
    • 'post:message' (String chatId, String msg): Inserta el mensaje "msg" en el chat "chatId".
    • 'get:message' (String chatId): Realiza la búsqueda de todos los mensajes del chat "chatId".
    • 'post:user' (JSONObject data): Crea un nuevo usuario.

Ejercicio

  • Eventos Escuchados por el Servidor:
    • 'put:user' (JSONObject data): Modifica el usuario actual.
    • 'delete:user' (): Elimina el usuario actual.
  • Eventos Emitidos por el Servidor:
    • 'api:error' (String err): Devuelve un String con el error.
    • 'post:login' (JSONObject user): Respuesta al evento post:login.
    • 'get:user' (JSONArray users): Respuesta al evento get:user.
    • 'get:chat' (JSONArray chats): Respuesta al evento get:chat.
    • 'post:chat' (JSONObject chat): Respuesta al evento post:chat.
    • 'post:message' (JSONObject msg): Respuesta al evento post:message.
    • 'get:message' (JSONArray messages): Respuesta al evento get:message.
    • 'post:user', (String insertedId): Respuesta al evento post:user.
    • 'put:user', (String updatedId): Respuesta al evento put:user.

Ejercicio

  • El fragment del chat debería contener un EditText en donde agrego el mensaje, y un botón "enviar" para agregar mensajes al chat (POST /chat/<id de chat>/message con {"message": "<texto del mensaje>"} como body)

Tecnologías Móviles - Clase 12

By Agustin Moyano

Tecnologías Móviles - Clase 12

  • 757