2016-05-02 09:52:09 +03:00
|
|
|
|
using System;
|
|
|
|
|
using Android.App;
|
|
|
|
|
using Android.Content.PM;
|
|
|
|
|
using Android.Views;
|
|
|
|
|
using Android.OS;
|
|
|
|
|
using Bit.App.Abstractions;
|
|
|
|
|
using XLabs.Ioc;
|
2016-05-21 19:32:34 +03:00
|
|
|
|
using Plugin.Settings.Abstractions;
|
2016-07-02 01:54:00 +03:00
|
|
|
|
using Plugin.Connectivity.Abstractions;
|
|
|
|
|
using Acr.UserDialogs;
|
2016-08-18 06:08:26 +03:00
|
|
|
|
using Android.Content;
|
2016-08-19 02:58:25 +03:00
|
|
|
|
using System.Reflection;
|
2016-08-20 03:42:33 +03:00
|
|
|
|
using Xamarin.Forms.Platform.Android;
|
|
|
|
|
using Xamarin.Forms;
|
2016-08-27 21:36:32 +03:00
|
|
|
|
using System.Threading.Tasks;
|
2017-01-28 07:13:28 +03:00
|
|
|
|
using Bit.App.Models.Page;
|
2017-06-08 23:22:11 +03:00
|
|
|
|
using Bit.App;
|
2017-06-28 06:33:13 +03:00
|
|
|
|
using Android.Nfc;
|
2017-06-29 19:11:07 +03:00
|
|
|
|
using Android.Views.InputMethods;
|
2017-07-22 22:38:08 +03:00
|
|
|
|
using System.IO;
|
2017-07-23 04:06:53 +03:00
|
|
|
|
using System.Linq;
|
2016-05-02 09:52:09 +03:00
|
|
|
|
|
|
|
|
|
namespace Bit.Android
|
|
|
|
|
{
|
2016-08-19 07:27:37 +03:00
|
|
|
|
[Activity(Label = "bitwarden",
|
2016-08-27 21:36:32 +03:00
|
|
|
|
Icon = "@drawable/icon",
|
2017-08-29 21:33:25 +03:00
|
|
|
|
ConfigurationChanges = ConfigChanges.ScreenSize | ConfigChanges.Orientation)]
|
2016-08-20 03:42:33 +03:00
|
|
|
|
public class MainActivity : FormsAppCompatActivity
|
2016-05-02 09:52:09 +03:00
|
|
|
|
{
|
2016-08-14 07:15:47 +03:00
|
|
|
|
private const string HockeyAppId = "d3834185b4a643479047b86c65293d42";
|
2017-06-08 23:22:11 +03:00
|
|
|
|
private DateTime? _lastAction;
|
2017-06-28 06:33:13 +03:00
|
|
|
|
private Java.Util.Regex.Pattern _otpPattern = Java.Util.Regex.Pattern.Compile("^.*?([cbdefghijklnrtuv]{32,64})$");
|
2017-07-21 18:39:22 +03:00
|
|
|
|
private IDeviceActionService _deviceActionService;
|
|
|
|
|
private ISettings _settings;
|
2016-08-14 07:15:47 +03:00
|
|
|
|
|
2016-08-27 22:00:12 +03:00
|
|
|
|
protected override void OnCreate(Bundle bundle)
|
2016-05-02 09:52:09 +03:00
|
|
|
|
{
|
2017-02-17 06:22:19 +03:00
|
|
|
|
var uri = Intent.GetStringExtra("uri");
|
2017-02-10 05:06:47 +03:00
|
|
|
|
if(!Resolver.IsSet)
|
2017-01-28 07:13:28 +03:00
|
|
|
|
{
|
|
|
|
|
MainApplication.SetIoc(Application);
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-07 06:27:14 +03:00
|
|
|
|
var policy = new StrictMode.ThreadPolicy.Builder().PermitAll().Build();
|
|
|
|
|
StrictMode.SetThreadPolicy(policy);
|
|
|
|
|
|
2016-08-20 03:42:33 +03:00
|
|
|
|
ToolbarResource = Resource.Layout.toolbar;
|
|
|
|
|
TabLayoutResource = Resource.Layout.tabs;
|
|
|
|
|
|
2016-05-02 09:52:09 +03:00
|
|
|
|
base.OnCreate(bundle);
|
2016-08-27 21:36:32 +03:00
|
|
|
|
|
|
|
|
|
// workaround for app compat bug
|
|
|
|
|
// ref https://forums.xamarin.com/discussion/62414/app-resuming-results-in-crash-with-formsappcompatactivity
|
2016-08-27 22:00:12 +03:00
|
|
|
|
Task.Delay(10).Wait();
|
2016-08-27 21:36:32 +03:00
|
|
|
|
|
2016-06-05 05:35:03 +03:00
|
|
|
|
Console.WriteLine("A OnCreate");
|
2017-07-24 22:04:31 +03:00
|
|
|
|
if(!App.Utilities.Helpers.InDebugMode())
|
|
|
|
|
{
|
|
|
|
|
Window.AddFlags(WindowManagerFlags.Secure);
|
|
|
|
|
}
|
2016-05-02 09:52:09 +03:00
|
|
|
|
|
2016-08-14 07:15:47 +03:00
|
|
|
|
var appIdService = Resolver.Resolve<IAppIdService>();
|
|
|
|
|
var authService = Resolver.Resolve<IAuthService>();
|
|
|
|
|
|
|
|
|
|
HockeyApp.Android.CrashManager.Register(this, HockeyAppId,
|
|
|
|
|
new HockeyAppCrashManagerListener(appIdService, authService));
|
2016-08-19 02:58:25 +03:00
|
|
|
|
|
2016-08-20 03:42:33 +03:00
|
|
|
|
Forms.Init(this, bundle);
|
2016-05-02 09:52:09 +03:00
|
|
|
|
|
2016-08-20 03:42:33 +03:00
|
|
|
|
typeof(Color).GetProperty("Accent", BindingFlags.Public | BindingFlags.Static)
|
|
|
|
|
.SetValue(null, Color.FromHex("d2d6de"));
|
2016-08-19 02:58:25 +03:00
|
|
|
|
|
2017-07-21 18:39:22 +03:00
|
|
|
|
_deviceActionService = Resolver.Resolve<IDeviceActionService>();
|
|
|
|
|
_settings = Resolver.Resolve<ISettings>();
|
2016-05-21 19:32:34 +03:00
|
|
|
|
LoadApplication(new App.App(
|
2017-01-28 07:13:28 +03:00
|
|
|
|
uri,
|
2016-05-21 19:32:34 +03:00
|
|
|
|
Resolver.Resolve<IAuthService>(),
|
2016-07-02 01:54:00 +03:00
|
|
|
|
Resolver.Resolve<IConnectivity>(),
|
|
|
|
|
Resolver.Resolve<IUserDialogs>(),
|
2016-05-21 19:32:34 +03:00
|
|
|
|
Resolver.Resolve<IDatabaseService>(),
|
2016-07-01 03:08:34 +03:00
|
|
|
|
Resolver.Resolve<ISyncService>(),
|
2017-07-21 18:39:22 +03:00
|
|
|
|
_settings,
|
2016-08-04 07:06:09 +03:00
|
|
|
|
Resolver.Resolve<ILockService>(),
|
2016-11-26 18:51:04 +03:00
|
|
|
|
Resolver.Resolve<IGoogleAnalyticsService>(),
|
2017-02-06 07:55:46 +03:00
|
|
|
|
Resolver.Resolve<ILocalizeService>(),
|
2017-04-28 18:07:26 +03:00
|
|
|
|
Resolver.Resolve<IAppInfoService>(),
|
2017-07-13 17:51:45 +03:00
|
|
|
|
Resolver.Resolve<IAppSettingsService>(),
|
2017-07-21 18:39:22 +03:00
|
|
|
|
_deviceActionService));
|
2016-08-18 06:08:26 +03:00
|
|
|
|
|
2017-06-29 19:11:07 +03:00
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application>(
|
|
|
|
|
Xamarin.Forms.Application.Current, "DismissKeyboard", (sender) =>
|
|
|
|
|
{
|
|
|
|
|
DismissKeyboard();
|
|
|
|
|
});
|
|
|
|
|
|
2016-08-27 21:36:32 +03:00
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application>(Xamarin.Forms.Application.Current, "RateApp", (sender) =>
|
2016-08-18 06:08:26 +03:00
|
|
|
|
{
|
|
|
|
|
RateApp();
|
|
|
|
|
});
|
2017-01-28 07:13:28 +03:00
|
|
|
|
|
2017-02-01 08:38:35 +03:00
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application>(Xamarin.Forms.Application.Current, "Accessibility", (sender) =>
|
|
|
|
|
{
|
|
|
|
|
OpenAccessibilitySettings();
|
|
|
|
|
});
|
|
|
|
|
|
2017-01-28 07:13:28 +03:00
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application, VaultListPageModel.Login>(
|
|
|
|
|
Xamarin.Forms.Application.Current, "Autofill", (sender, args) =>
|
|
|
|
|
{
|
|
|
|
|
ReturnCredentials(args);
|
|
|
|
|
});
|
2017-02-10 02:12:34 +03:00
|
|
|
|
|
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application>(Xamarin.Forms.Application.Current, "BackgroundApp", (sender) =>
|
|
|
|
|
{
|
|
|
|
|
MoveTaskToBack(true);
|
|
|
|
|
});
|
2017-06-08 23:22:11 +03:00
|
|
|
|
|
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application, string>(
|
|
|
|
|
Xamarin.Forms.Application.Current, "LaunchApp", (sender, args) =>
|
|
|
|
|
{
|
|
|
|
|
LaunchApp(args);
|
|
|
|
|
});
|
2017-06-28 06:33:13 +03:00
|
|
|
|
|
2017-06-29 05:24:04 +03:00
|
|
|
|
MessagingCenter.Subscribe<Xamarin.Forms.Application, bool>(
|
|
|
|
|
Xamarin.Forms.Application.Current, "ListenYubiKeyOTP", (sender, listen) =>
|
2017-06-28 06:33:13 +03:00
|
|
|
|
{
|
2017-06-29 05:24:04 +03:00
|
|
|
|
ListenYubiKey(listen);
|
2017-06-28 06:33:13 +03:00
|
|
|
|
});
|
2017-01-28 07:13:28 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ReturnCredentials(VaultListPageModel.Login login)
|
|
|
|
|
{
|
|
|
|
|
Intent data = new Intent();
|
2017-01-31 03:26:39 +03:00
|
|
|
|
if(login == null)
|
|
|
|
|
{
|
|
|
|
|
data.PutExtra("canceled", "true");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2017-07-21 18:39:22 +03:00
|
|
|
|
var isPremium = Resolver.Resolve<ITokenService>()?.TokenPremium ?? false;
|
|
|
|
|
var autoCopyEnabled = !_settings.GetValueOrDefault(Constants.SettingDisableTotpCopy, false);
|
|
|
|
|
if(isPremium && autoCopyEnabled && _deviceActionService != null && login.Totp.Value != null)
|
|
|
|
|
{
|
|
|
|
|
_deviceActionService.CopyToClipboard(App.Utilities.Crypto.Totp(login.Totp.Value));
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-31 03:26:39 +03:00
|
|
|
|
data.PutExtra("uri", login.Uri.Value);
|
|
|
|
|
data.PutExtra("username", login.Username);
|
|
|
|
|
data.PutExtra("password", login.Password.Value);
|
|
|
|
|
}
|
2017-01-28 07:13:28 +03:00
|
|
|
|
|
|
|
|
|
if(Parent == null)
|
|
|
|
|
{
|
|
|
|
|
SetResult(Result.Ok, data);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Parent.SetResult(Result.Ok, data);
|
|
|
|
|
}
|
2017-06-08 23:22:11 +03:00
|
|
|
|
|
2017-01-28 07:13:28 +03:00
|
|
|
|
Finish();
|
2016-05-02 09:52:09 +03:00
|
|
|
|
}
|
|
|
|
|
|
2016-06-05 05:35:03 +03:00
|
|
|
|
protected override void OnPause()
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("A OnPause");
|
|
|
|
|
base.OnPause();
|
2017-06-29 05:24:04 +03:00
|
|
|
|
ListenYubiKey(false);
|
2016-06-05 05:35:03 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnDestroy()
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("A OnDestroy");
|
|
|
|
|
base.OnDestroy();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnRestart()
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("A OnRestart");
|
|
|
|
|
base.OnRestart();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnStart()
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("A OnStart");
|
|
|
|
|
base.OnStart();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnStop()
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("A OnStop");
|
|
|
|
|
base.OnStop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnResume()
|
|
|
|
|
{
|
|
|
|
|
base.OnResume();
|
2017-02-04 07:21:40 +03:00
|
|
|
|
Console.WriteLine("A OnResume");
|
|
|
|
|
|
|
|
|
|
// workaround for app compat bug
|
|
|
|
|
// ref https://bugzilla.xamarin.com/show_bug.cgi?id=36907
|
|
|
|
|
Task.Delay(10).Wait();
|
2017-06-29 05:24:04 +03:00
|
|
|
|
|
|
|
|
|
if(Utilities.NfcEnabled())
|
|
|
|
|
{
|
|
|
|
|
MessagingCenter.Send(Xamarin.Forms.Application.Current, "ResumeYubiKey");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnNewIntent(Intent intent)
|
|
|
|
|
{
|
|
|
|
|
base.OnNewIntent(intent);
|
|
|
|
|
Console.WriteLine("A OnNewIntent");
|
|
|
|
|
ParseYubiKey(intent.DataString);
|
2016-06-05 05:35:03 +03:00
|
|
|
|
}
|
2016-08-18 06:08:26 +03:00
|
|
|
|
|
2017-07-23 04:06:53 +03:00
|
|
|
|
public async override void OnRequestPermissionsResult(int requestCode, string[] permissions, Permission[] grantResults)
|
2017-07-14 00:23:18 +03:00
|
|
|
|
{
|
2017-07-23 04:06:53 +03:00
|
|
|
|
if(requestCode == Constants.SelectFilePermissionRequestCode)
|
|
|
|
|
{
|
|
|
|
|
if(grantResults.Any(r => r != Permission.Granted))
|
|
|
|
|
{
|
|
|
|
|
MessagingCenter.Send(Xamarin.Forms.Application.Current, "SelectFileCameraPermissionDenied");
|
|
|
|
|
}
|
|
|
|
|
await _deviceActionService.SelectFileAsync();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-14 00:23:18 +03:00
|
|
|
|
ZXing.Net.Mobile.Forms.Android.PermissionsHandler.OnRequestPermissionsResult(requestCode, permissions, grantResults);
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-22 22:38:08 +03:00
|
|
|
|
protected override void OnActivityResult(int requestCode, Result resultCode, Intent data)
|
|
|
|
|
{
|
|
|
|
|
if(requestCode == Constants.SelectFileRequestCode && resultCode == Result.Ok)
|
|
|
|
|
{
|
|
|
|
|
global::Android.Net.Uri uri = null;
|
2017-07-23 04:06:53 +03:00
|
|
|
|
string fileName = null;
|
|
|
|
|
if(data != null && data.Data != null)
|
2017-07-22 22:38:08 +03:00
|
|
|
|
{
|
|
|
|
|
uri = data.Data;
|
2017-07-23 04:06:53 +03:00
|
|
|
|
fileName = Utilities.GetFileName(ApplicationContext, uri);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// camera
|
|
|
|
|
var root = new Java.IO.File(global::Android.OS.Environment.ExternalStorageDirectory, "bitwarden");
|
|
|
|
|
var file = new Java.IO.File(root, "temp_camera_photo.jpg");
|
|
|
|
|
uri = global::Android.Net.Uri.FromFile(file);
|
|
|
|
|
fileName = $"photo_{DateTime.UtcNow.ToString("yyyyMMddHHmmss")}.jpg";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(uri == null)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using(var stream = ContentResolver.OpenInputStream(uri))
|
|
|
|
|
using(var memoryStream = new MemoryStream())
|
|
|
|
|
{
|
|
|
|
|
stream.CopyTo(memoryStream);
|
|
|
|
|
MessagingCenter.Send(Xamarin.Forms.Application.Current, "SelectFileResult",
|
|
|
|
|
new Tuple<byte[], string>(memoryStream.ToArray(), fileName ?? "unknown_file_name"));
|
2017-07-22 22:38:08 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-18 06:08:26 +03:00
|
|
|
|
public void RateApp()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var rateIntent = RateIntentForUrl("market://details");
|
|
|
|
|
StartActivity(rateIntent);
|
|
|
|
|
}
|
|
|
|
|
catch(ActivityNotFoundException)
|
|
|
|
|
{
|
|
|
|
|
var rateIntent = RateIntentForUrl("https://play.google.com/store/apps/details");
|
|
|
|
|
StartActivity(rateIntent);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Intent RateIntentForUrl(string url)
|
|
|
|
|
{
|
|
|
|
|
var intent = new Intent(Intent.ActionView, global::Android.Net.Uri.Parse($"{url}?id={PackageName}"));
|
|
|
|
|
var flags = ActivityFlags.NoHistory | ActivityFlags.MultipleTask;
|
|
|
|
|
if((int)Build.VERSION.SdkInt >= 21)
|
|
|
|
|
{
|
|
|
|
|
flags |= ActivityFlags.NewDocument;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// noinspection deprecation
|
|
|
|
|
flags |= ActivityFlags.ClearWhenTaskReset;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
intent.AddFlags(flags);
|
|
|
|
|
return intent;
|
|
|
|
|
}
|
2017-02-01 08:38:35 +03:00
|
|
|
|
|
|
|
|
|
private void OpenAccessibilitySettings()
|
|
|
|
|
{
|
|
|
|
|
var intent = new Intent(global::Android.Provider.Settings.ActionAccessibilitySettings);
|
|
|
|
|
StartActivity(intent);
|
|
|
|
|
}
|
2017-06-08 23:22:11 +03:00
|
|
|
|
|
|
|
|
|
private void LaunchApp(string packageName)
|
|
|
|
|
{
|
|
|
|
|
if(_lastAction.LastActionWasRecent())
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
_lastAction = DateTime.UtcNow;
|
|
|
|
|
|
|
|
|
|
packageName = packageName.Replace("androidapp://", string.Empty);
|
|
|
|
|
var launchIntent = PackageManager.GetLaunchIntentForPackage(packageName);
|
|
|
|
|
if(launchIntent == null)
|
|
|
|
|
{
|
|
|
|
|
var dialog = Resolver.Resolve<IUserDialogs>();
|
|
|
|
|
dialog.Alert(string.Format(App.Resources.AppResources.CannotOpenApp, packageName));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
StartActivity(launchIntent);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-28 06:33:13 +03:00
|
|
|
|
|
2017-06-29 05:24:04 +03:00
|
|
|
|
private void ListenYubiKey(bool listen)
|
2017-06-28 06:33:13 +03:00
|
|
|
|
{
|
2017-06-29 05:24:04 +03:00
|
|
|
|
if(!Utilities.NfcEnabled())
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-06-28 06:33:13 +03:00
|
|
|
|
|
|
|
|
|
var adapter = NfcAdapter.GetDefaultAdapter(this);
|
2017-06-29 05:24:04 +03:00
|
|
|
|
if(listen)
|
|
|
|
|
{
|
|
|
|
|
var intent = new Intent(this, Class);
|
|
|
|
|
intent.AddFlags(ActivityFlags.SingleTop);
|
|
|
|
|
var pendingIntent = PendingIntent.GetActivity(this, 0, intent, 0);
|
|
|
|
|
|
|
|
|
|
// register for all NDEF tags starting with http och https
|
|
|
|
|
var ndef = new IntentFilter(NfcAdapter.ActionNdefDiscovered);
|
|
|
|
|
ndef.AddDataScheme("http");
|
|
|
|
|
ndef.AddDataScheme("https");
|
|
|
|
|
var filters = new IntentFilter[] { ndef };
|
|
|
|
|
|
|
|
|
|
// register for foreground dispatch so we'll receive tags according to our intent filters
|
|
|
|
|
adapter.EnableForegroundDispatch(this, pendingIntent, filters, null);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
adapter.DisableForegroundDispatch(this);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ParseYubiKey(string data)
|
|
|
|
|
{
|
|
|
|
|
if(data == null)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-06-28 06:33:13 +03:00
|
|
|
|
|
2017-06-29 05:24:04 +03:00
|
|
|
|
var otpMatch = _otpPattern.Matcher(data);
|
|
|
|
|
if(otpMatch.Matches())
|
2017-06-28 06:33:13 +03:00
|
|
|
|
{
|
2017-06-29 05:24:04 +03:00
|
|
|
|
var otp = otpMatch.Group(1);
|
|
|
|
|
MessagingCenter.Send(Xamarin.Forms.Application.Current, "GotYubiKeyOTP", otp);
|
2017-06-28 06:33:13 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 19:11:07 +03:00
|
|
|
|
|
|
|
|
|
private void DismissKeyboard()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var imm = (InputMethodManager)GetSystemService(InputMethodService);
|
|
|
|
|
imm.HideSoftInputFromWindow(CurrentFocus.WindowToken, 0);
|
|
|
|
|
}
|
|
|
|
|
catch { }
|
|
|
|
|
}
|
2016-05-02 09:52:09 +03:00
|
|
|
|
}
|
|
|
|
|
}
|