#7 Finished google and password logins
parent
6f8841a7f8
commit
ed72d03033
|
@ -283,6 +283,7 @@ public class LocationService extends Service implements GoogleApiClient.Connecti
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
Log.v(TAG, "Can't set location because user isn't logged in.");
|
Log.v(TAG, "Can't set location because user isn't logged in.");
|
||||||
|
//TODO: So stop the damn service!
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,6 +9,7 @@ import android.os.Bundle;
|
||||||
import android.text.method.LinkMovementMethod;
|
import android.text.method.LinkMovementMethod;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
|
import android.widget.EditText;
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
|
|
||||||
import com.google.android.gms.auth.api.Auth;
|
import com.google.android.gms.auth.api.Auth;
|
||||||
|
@ -16,6 +17,7 @@ import com.google.android.gms.auth.api.signin.GoogleSignInAccount;
|
||||||
import com.google.android.gms.auth.api.signin.GoogleSignInOptions;
|
import com.google.android.gms.auth.api.signin.GoogleSignInOptions;
|
||||||
import com.google.android.gms.auth.api.signin.GoogleSignInResult;
|
import com.google.android.gms.auth.api.signin.GoogleSignInResult;
|
||||||
import com.google.android.gms.common.ConnectionResult;
|
import com.google.android.gms.common.ConnectionResult;
|
||||||
|
import com.google.android.gms.common.SignInButton;
|
||||||
import com.google.android.gms.common.api.GoogleApiClient;
|
import com.google.android.gms.common.api.GoogleApiClient;
|
||||||
import com.google.android.gms.common.api.OptionalPendingResult;
|
import com.google.android.gms.common.api.OptionalPendingResult;
|
||||||
import com.google.android.gms.common.api.ResultCallback;
|
import com.google.android.gms.common.api.ResultCallback;
|
||||||
|
@ -32,9 +34,11 @@ import javax.net.ssl.TrustManagerFactory;
|
||||||
import javax.net.ssl.X509TrustManager;
|
import javax.net.ssl.X509TrustManager;
|
||||||
|
|
||||||
import okhttp3.Call;
|
import okhttp3.Call;
|
||||||
|
import okhttp3.FormBody;
|
||||||
import okhttp3.OkHttpClient;
|
import okhttp3.OkHttpClient;
|
||||||
import okhttp3.Request;
|
import okhttp3.Request;
|
||||||
import okhttp3.Callback;
|
import okhttp3.Callback;
|
||||||
|
import okhttp3.RequestBody;
|
||||||
import okhttp3.Response;
|
import okhttp3.Response;
|
||||||
|
|
||||||
public class LoginActivity extends AppCompatActivity implements
|
public class LoginActivity extends AppCompatActivity implements
|
||||||
|
@ -64,9 +68,9 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
setContentView(R.layout.activity_login);
|
setContentView(R.layout.activity_login);
|
||||||
setTitle(R.string.login_name);
|
setTitle(R.string.login_name);
|
||||||
TextView loginDescription = (TextView) findViewById(R.id.login_description);
|
TextView loginDescription = (TextView) findViewById(R.id.login_description);
|
||||||
TextView forgotPassword = (TextView) findViewById(R.id.login_forgot_password);
|
TextView forgotPassword = (TextView) findViewById(R.id.login_forgot_password);
|
||||||
loginDescription.setMovementMethod(LinkMovementMethod.getInstance());
|
loginDescription.setMovementMethod(LinkMovementMethod.getInstance());
|
||||||
forgotPassword.setMovementMethod(LinkMovementMethod.getInstance());
|
forgotPassword.setMovementMethod(LinkMovementMethod.getInstance());
|
||||||
|
|
||||||
// Configure sign-in to request the user's ID and basic profile, included in DEFAULT_SIGN_IN.
|
// Configure sign-in to request the user's ID and basic profile, included in DEFAULT_SIGN_IN.
|
||||||
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
|
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
|
||||||
|
@ -82,11 +86,14 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
// Set up buttons
|
// Set up buttons
|
||||||
// SignInButton signInButton = (SignInButton) findViewById(R.id.login_button_google);
|
SignInButton signInButton = (SignInButton) findViewById(R.id.login_button_google);
|
||||||
// signInButton.setStyle(SignInButton.SIZE_WIDE, SignInButton.COLOR_AUTO);
|
signInButton.setStyle(SignInButton.SIZE_WIDE, SignInButton.COLOR_AUTO);
|
||||||
|
|
||||||
// Button listeners
|
// Button listeners
|
||||||
|
findViewById(R.id.login_button).setOnClickListener(this);
|
||||||
findViewById(R.id.login_button_google).setOnClickListener(this);
|
findViewById(R.id.login_button_google).setOnClickListener(this);
|
||||||
|
// findViewById(R.id.login_button_facebook).setOnClickListener(this);
|
||||||
|
// findViewById(R.id.login_button_twitter).setOnClickListener(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -103,7 +110,7 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
// and the GoogleSignInResult will be available instantly.
|
// and the GoogleSignInResult will be available instantly.
|
||||||
Log.d(TAG, "Got cached sign-in");
|
Log.d(TAG, "Got cached sign-in");
|
||||||
GoogleSignInResult result = opr.get();
|
GoogleSignInResult result = opr.get();
|
||||||
handleSignInResult(result);
|
handleGoogleSignInResult(result);
|
||||||
} else {
|
} else {
|
||||||
// If the user has not previously signed in on this device or the sign-in has expired,
|
// If the user has not previously signed in on this device or the sign-in has expired,
|
||||||
// this asynchronous branch will attempt to sign in the user silently. Cross-device
|
// this asynchronous branch will attempt to sign in the user silently. Cross-device
|
||||||
|
@ -112,8 +119,8 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
opr.setResultCallback(new ResultCallback<GoogleSignInResult>() {
|
opr.setResultCallback(new ResultCallback<GoogleSignInResult>() {
|
||||||
@Override
|
@Override
|
||||||
public void onResult(GoogleSignInResult googleSignInResult) {
|
public void onResult(GoogleSignInResult googleSignInResult) {
|
||||||
hideProgressDialog();
|
hideProgressDialog();
|
||||||
handleSignInResult(googleSignInResult);
|
handleGoogleSignInResult(googleSignInResult);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -130,16 +137,17 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
if (requestCode == RC_SIGN_IN) {
|
if (requestCode == RC_SIGN_IN) {
|
||||||
Log.v(TAG, "requestCode was RC_SIGN_IN");
|
Log.v(TAG, "requestCode was RC_SIGN_IN");
|
||||||
GoogleSignInResult result = Auth.GoogleSignInApi.getSignInResultFromIntent(data);
|
GoogleSignInResult result = Auth.GoogleSignInApi.getSignInResultFromIntent(data);
|
||||||
handleSignInResult(result);
|
handleGoogleSignInResult(result);
|
||||||
}
|
}
|
||||||
|
// User just logged out. Don't log in again, stupid
|
||||||
else if (requestCode == SIGN_OUT) {
|
else if (requestCode == SIGN_OUT) {
|
||||||
Log.v(TAG, "requestCode was SIGN_OUT");
|
Log.v(TAG, "requestCode was SIGN_OUT");
|
||||||
DONT_LOG_IN = true;
|
DONT_LOG_IN = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void AuthenticateGoogle(final String token) throws Exception {
|
private void authenticateWithTracmanServer(final Request request) throws Exception {
|
||||||
|
|
||||||
// Needed to support TLS 1.1 and 1.2
|
// Needed to support TLS 1.1 and 1.2
|
||||||
TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(
|
TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(
|
||||||
TrustManagerFactory.getDefaultAlgorithm());
|
TrustManagerFactory.getDefaultAlgorithm());
|
||||||
|
@ -155,14 +163,10 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
.sslSocketFactory(new TLSSocketFactory(), trustManager)
|
.sslSocketFactory(new TLSSocketFactory(), trustManager)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
Request request = new Request.Builder()
|
|
||||||
.url(SERVER_ADDRESS+"login/app/google?id_token="+token)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
client.newCall(request).enqueue(new Callback() {
|
client.newCall(request).enqueue(new Callback() {
|
||||||
@Override
|
@Override
|
||||||
public void onFailure(Call call, IOException e) {
|
public void onFailure(Call call, IOException e) {
|
||||||
Log.e(TAG, "Failed to connect to server: " + SERVER_ADDRESS + "login/app/google?id_token=" + token);
|
Log.e(TAG, "Failed to connect to Tracman server!");
|
||||||
showError(R.string.server_connection_error);
|
showError(R.string.server_connection_error);
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
|
@ -203,23 +207,31 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
editor.commit();
|
editor.commit();
|
||||||
|
|
||||||
startActivityForResult(new Intent(LoginActivity.this, SettingsActivity.class), SIGN_OUT);
|
startActivityForResult(new Intent(LoginActivity.this, SettingsActivity.class), SIGN_OUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleSignInResult(GoogleSignInResult result) {
|
private void handleGoogleSignInResult(GoogleSignInResult result) {
|
||||||
// Log.d(TAG, "handleSignInResult:" + result.isSuccess());
|
// Log.d(TAG, "handleSignInResult:" + result.isSuccess());
|
||||||
if (result.isSuccess()) { // Signed in successfully
|
if (result.isSuccess()) { // Signed in successfully
|
||||||
GoogleSignInAccount acct = result.getSignInAccount();
|
GoogleSignInAccount acct = result.getSignInAccount();
|
||||||
String googleToken = acct.getIdToken();
|
|
||||||
// Log.v(TAG, "Google token: " + googleToken);
|
// Log.v(TAG, "Google token: " + googleToken);
|
||||||
try {
|
try {
|
||||||
AuthenticateGoogle(acct.getIdToken());
|
|
||||||
} catch (Exception e) {
|
// Build request
|
||||||
// Log.e(TAG, "Error sending ID token to backend.", e);
|
Request request = new Request.Builder()
|
||||||
|
.url(SERVER_ADDRESS+"login/app/google?id_token="+acct.getIdToken())
|
||||||
|
.build();
|
||||||
|
|
||||||
|
// Send to server
|
||||||
|
authenticateWithTracmanServer(request);
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "Error sending ID token to backend.", e);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Log.e(TAG, "Failed to log in: "+result.getStatus().getStatusCode());
|
// Log.e(TAG, "Failed to log in: "+result.getStatus().getStatusCode());
|
||||||
|
@ -229,11 +241,57 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void signIn() {
|
public void signInWithPassword() {
|
||||||
|
Log.d(TAG, "signInWithPassword() called");
|
||||||
|
|
||||||
|
// Get params from form
|
||||||
|
EditText emailText = (EditText)findViewById(R.id.login_email);
|
||||||
|
String email = emailText.getText().toString();
|
||||||
|
EditText passwordText = (EditText)findViewById(R.id.login_password);
|
||||||
|
String password = passwordText.getText().toString();
|
||||||
|
|
||||||
|
// Build formdata
|
||||||
|
RequestBody formData = new FormBody.Builder()
|
||||||
|
.add("email", email)
|
||||||
|
.add("password", password)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
// Build request
|
||||||
|
Request request = new Request.Builder()
|
||||||
|
.url(SERVER_ADDRESS+"login/app")
|
||||||
|
.post(formData)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
// Send formdata to endpoint
|
||||||
|
try {
|
||||||
|
Log.v(TAG, "Sending local login POST to server...");
|
||||||
|
authenticateWithTracmanServer(request);
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "Error sending local login to backend:",e);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void signInWithGoogle() {
|
||||||
|
Log.v(TAG, "signInWithGoogle() called");
|
||||||
Intent signInIntent = Auth.GoogleSignInApi.getSignInIntent(mGoogleApiClient);
|
Intent signInIntent = Auth.GoogleSignInApi.getSignInIntent(mGoogleApiClient);
|
||||||
startActivityForResult(signInIntent, RC_SIGN_IN);
|
startActivityForResult(signInIntent, RC_SIGN_IN);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// private void signInWithFacebook() {
|
||||||
|
// Log.v(TAG, "signInWithFacebook() called");
|
||||||
|
//
|
||||||
|
// //TODO: Facebook login to /login/app/facebook
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
|
||||||
|
// private void signInWithTwitter() {
|
||||||
|
// Log.v(TAG, "signInWithTwitter() called");
|
||||||
|
//
|
||||||
|
// //TODO: Twitter login to /login/app/twitter
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onConnectionFailed(ConnectionResult connectionResult) {
|
public void onConnectionFailed(ConnectionResult connectionResult) {
|
||||||
showError(R.string.disconnected);
|
showError(R.string.disconnected);
|
||||||
|
@ -267,10 +325,24 @@ public class LoginActivity extends AppCompatActivity implements
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onClick(View v) {
|
public void onClick(View v) {
|
||||||
|
Log.v(TAG, "onClick() called");
|
||||||
switch (v.getId()) {
|
switch (v.getId()) {
|
||||||
case R.id.login_button_google:
|
case R.id.login_button:
|
||||||
signIn();
|
Log.v(TAG, "Password login button pressed");
|
||||||
|
signInWithPassword();
|
||||||
break;
|
break;
|
||||||
|
case R.id.login_button_google:
|
||||||
|
Log.v(TAG, "Google login button pressed");
|
||||||
|
signInWithGoogle();
|
||||||
|
break;
|
||||||
|
// case R.id.login_button_facebook:
|
||||||
|
// Log.v(TAG, "Facebook login button pressed");
|
||||||
|
// signInWithFacebook();
|
||||||
|
// break;
|
||||||
|
// case R.id.login_button_twitter:
|
||||||
|
// Log.v(TAG, "Twitter login button pressed");
|
||||||
|
// signInWithTwitter();
|
||||||
|
// break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -209,14 +209,17 @@ public class SettingsActivity extends AppCompatPreferenceActivity {
|
||||||
public void onBackPressed() {
|
public void onBackPressed() {
|
||||||
Log.v(TAG, "onBackPressed() called");
|
Log.v(TAG, "onBackPressed() called");
|
||||||
|
|
||||||
|
// Get sharedPrefs
|
||||||
|
SharedPreferences sharedPref = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
|
||||||
|
SharedPreferences.Editor editor = sharedPref.edit();
|
||||||
|
|
||||||
// Stop LocationService
|
// Stop LocationService
|
||||||
Log.v(TAG, "Stopping location service...");
|
Log.v(TAG, "Stopping location service...");
|
||||||
stopService(new Intent(SettingsActivity.this, LocationService.class));
|
stopService(new Intent(SettingsActivity.this, LocationService.class));
|
||||||
|
editor.putBoolean("gps_switch",false);
|
||||||
|
|
||||||
// Remove saved loggedInUser
|
// Remove saved loggedInUser
|
||||||
Log.v(TAG, "Removing saved user...");
|
Log.v(TAG, "Removing saved user...");
|
||||||
SharedPreferences sharedPref = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
|
|
||||||
SharedPreferences.Editor editor = sharedPref.edit();
|
|
||||||
editor.remove("loggedInUser");
|
editor.remove("loggedInUser");
|
||||||
editor.remove("loggedInUserId");
|
editor.remove("loggedInUserId");
|
||||||
editor.remove("loggedInUserName");
|
editor.remove("loggedInUserName");
|
||||||
|
|
|
@ -60,43 +60,52 @@
|
||||||
<!-- Social login buttons -->
|
<!-- Social login buttons -->
|
||||||
<LinearLayout android:layout_width="match_parent"
|
<LinearLayout android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_weight="2"
|
android:layout_weight="1"
|
||||||
android:layout_marginBottom="30px" >
|
android:layout_marginBottom="30px" >
|
||||||
|
|
||||||
<!-- Google -->
|
<com.google.android.gms.common.SignInButton
|
||||||
<ImageButton
|
|
||||||
android:id="@+id/login_button_google"
|
android:id="@+id/login_button_google"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="match_parent"
|
||||||
android:layout_weight="1"
|
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:background="#ce4d39"
|
android:layout_gravity="center" />
|
||||||
android:layout_marginLeft="30px"
|
|
||||||
android:layout_marginRight="30px"
|
|
||||||
android:src="@drawable/social_google" />
|
|
||||||
|
|
||||||
<!-- Facebook -->
|
|
||||||
<ImageButton
|
|
||||||
android:id="@+id/login_button_facebook"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_weight="1"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:background="#305891"
|
|
||||||
android:layout_marginLeft="30px"
|
|
||||||
android:layout_marginRight="30px"
|
|
||||||
android:src="@drawable/social_facebook" />
|
|
||||||
|
|
||||||
<!-- Twitter -->
|
|
||||||
<ImageButton
|
|
||||||
android:id="@+id/login_button_twitter"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_weight="1"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:background="#2ca8d2"
|
|
||||||
android:layout_marginLeft="30px"
|
|
||||||
android:layout_marginRight="30px"
|
|
||||||
android:src="@drawable/social_twitter" />
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
|
<!--<!– Google –>-->
|
||||||
|
<!--<ImageButton-->
|
||||||
|
<!--android:id="@+id/login_button_google"-->
|
||||||
|
<!--android:layout_width="wrap_content"-->
|
||||||
|
<!--android:layout_weight="1"-->
|
||||||
|
<!--android:layout_height="match_parent"-->
|
||||||
|
<!--android:background="#ce4d39"-->
|
||||||
|
<!--android:layout_marginLeft="30px"-->
|
||||||
|
<!--android:layout_marginRight="30px"-->
|
||||||
|
<!--android:src="@drawable/social_google" />-->
|
||||||
|
|
||||||
|
<!--<!– Facebook –>-->
|
||||||
|
<!--<ImageButton-->
|
||||||
|
<!--android:id="@+id/login_button_facebook"-->
|
||||||
|
<!--android:layout_width="wrap_content"-->
|
||||||
|
<!--android:layout_weight="1"-->
|
||||||
|
<!--android:layout_height="match_parent"-->
|
||||||
|
<!--android:background="#305891"-->
|
||||||
|
<!--android:layout_marginLeft="30px"-->
|
||||||
|
<!--android:layout_marginRight="30px"-->
|
||||||
|
<!--android:src="@drawable/social_facebook" />-->
|
||||||
|
|
||||||
|
<!--<!– Twitter –>-->
|
||||||
|
<!--<ImageButton-->
|
||||||
|
<!--android:id="@+id/login_button_twitter"-->
|
||||||
|
<!--android:layout_width="wrap_content"-->
|
||||||
|
<!--android:layout_weight="1"-->
|
||||||
|
<!--android:layout_height="match_parent"-->
|
||||||
|
<!--android:background="#2ca8d2"-->
|
||||||
|
<!--android:layout_marginLeft="30px"-->
|
||||||
|
<!--android:layout_marginRight="30px"-->
|
||||||
|
<!--android:src="@drawable/social_twitter" />-->
|
||||||
|
|
||||||
|
<!--</LinearLayout>-->
|
||||||
|
|
||||||
<!-- Login form -->
|
<!-- Login form -->
|
||||||
<LinearLayout android:layout_width="match_parent"
|
<LinearLayout android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
|
@ -108,7 +117,8 @@
|
||||||
android:id="@+id/login_email"
|
android:id="@+id/login_email"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:text="@string/email"
|
android:hint="@string/email"
|
||||||
|
android:inputType="textEmailAddress"
|
||||||
android:textSize="24sp"/>
|
android:textSize="24sp"/>
|
||||||
|
|
||||||
<!-- Password -->
|
<!-- Password -->
|
||||||
|
@ -116,16 +126,18 @@
|
||||||
android:id="@+id/login_password"
|
android:id="@+id/login_password"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:text="@string/password"
|
android:inputType="textPassword"
|
||||||
|
android:hint="@string/password"
|
||||||
android:textSize="24sp"/>
|
android:textSize="24sp"/>
|
||||||
|
|
||||||
<!-- Submit -->
|
<!-- Submit -->
|
||||||
<Button
|
<Button
|
||||||
|
android:id="@+id/login_button"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_weight="3"
|
android:padding="15dp"
|
||||||
android:text="@string/login_button_text"
|
android:text="@string/login_button_text"
|
||||||
android:textSize="32sp" />
|
android:textSize="34sp" />
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue