Encrypted messaging app
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

298 lines
8.5 KiB

3 years ago
  1. import 'dart:convert';
  2. import 'package:flutter/material.dart';
  3. import 'package:http/http.dart' as http;
  4. import '/components/flash_message.dart';
  5. import '/models/my_profile.dart';
  6. import '/utils/storage/session_cookie.dart';
  7. class LoginResponse {
  8. final String userId;
  9. final String username;
  10. final String publicKey;
  11. final String privateKey;
  12. final String symmetricKey;
  13. final String? imageLink;
  14. const LoginResponse({
  15. required this.publicKey,
  16. required this.privateKey,
  17. required this.symmetricKey,
  18. required this.userId,
  19. required this.username,
  20. this.imageLink,
  21. });
  22. factory LoginResponse.fromJson(Map<String, dynamic> json) {
  23. return LoginResponse(
  24. userId: json['user_id'],
  25. username: json['username'],
  26. publicKey: json['asymmetric_public_key'],
  27. privateKey: json['asymmetric_private_key'],
  28. symmetricKey: json['symmetric_key'],
  29. imageLink: json['image_link'],
  30. );
  31. }
  32. }
  33. class Login extends StatelessWidget {
  34. const Login({Key? key}) : super(key: key);
  35. @override
  36. Widget build(BuildContext context) {
  37. return Scaffold(
  38. appBar: AppBar(
  39. title: null,
  40. automaticallyImplyLeading: true,
  41. leading: IconButton(icon: const Icon(Icons.arrow_back),
  42. onPressed:() => {
  43. Navigator.pop(context)
  44. }
  45. ),
  46. backgroundColor: Colors.transparent,
  47. shadowColor: Colors.transparent,
  48. ),
  49. body: const SafeArea(
  50. child: LoginWidget(),
  51. ),
  52. );
  53. }
  54. }
  55. class LoginWidget extends StatefulWidget {
  56. const LoginWidget({Key? key}) : super(key: key);
  57. @override
  58. State<LoginWidget> createState() => _LoginWidgetState();
  59. }
  60. class _LoginWidgetState extends State<LoginWidget> {
  61. final _formKey = GlobalKey<FormState>();
  62. final TextEditingController _usernameController = TextEditingController();
  63. final TextEditingController _passwordController = TextEditingController();
  64. final TextEditingController _urlController = TextEditingController();
  65. bool _changedUrl = false;
  66. String _initialServerUrl = '';
  67. @override
  68. void initState() {
  69. MyProfile.getBaseServerUrl().then((String url) {
  70. _urlController.text = url;
  71. _initialServerUrl = url;
  72. });
  73. super.initState();
  74. }
  75. @override
  76. Widget build(BuildContext context) {
  77. const TextStyle inputTextStyle = TextStyle(
  78. fontSize: 18,
  79. );
  80. final OutlineInputBorder inputBorderStyle = OutlineInputBorder(
  81. borderRadius: BorderRadius.circular(5),
  82. borderSide: const BorderSide(
  83. color: Colors.transparent,
  84. )
  85. );
  86. final ButtonStyle buttonStyle = ElevatedButton.styleFrom(
  87. backgroundColor: Theme.of(context).colorScheme.surface,
  88. foregroundColor: Theme.of(context).colorScheme.onSurface,
  89. minimumSize: const Size.fromHeight(50),
  90. padding: const EdgeInsets.symmetric(horizontal: 10, vertical: 10),
  91. textStyle: TextStyle(
  92. fontSize: 20,
  93. fontWeight: FontWeight.bold,
  94. color: Theme.of(context).colorScheme.error,
  95. ),
  96. );
  97. return Center(
  98. child: Form(
  99. key: _formKey,
  100. child: Center(
  101. child: Padding(
  102. padding: const EdgeInsets.only(
  103. left: 20,
  104. right: 20,
  105. top: 0,
  106. bottom: 80,
  107. ),
  108. child: Column(
  109. mainAxisAlignment: MainAxisAlignment.center,
  110. crossAxisAlignment: CrossAxisAlignment.center,
  111. children: [
  112. Text(
  113. 'Login',
  114. style: TextStyle(
  115. fontSize: 35,
  116. color: Theme.of(context).colorScheme.onBackground,
  117. ),
  118. ),
  119. const SizedBox(height: 30),
  120. TextFormField(
  121. controller: _usernameController,
  122. decoration: InputDecoration(
  123. hintText: 'Username',
  124. enabledBorder: inputBorderStyle,
  125. focusedBorder: inputBorderStyle,
  126. ),
  127. style: inputTextStyle,
  128. // The validator receives the text that the user has entered.
  129. validator: (value) {
  130. if (value == null || value.isEmpty) {
  131. return 'Enter a username';
  132. }
  133. return null;
  134. },
  135. ),
  136. const SizedBox(height: 10),
  137. TextFormField(
  138. controller: _passwordController,
  139. obscureText: true,
  140. enableSuggestions: false,
  141. autocorrect: false,
  142. decoration: InputDecoration(
  143. hintText: 'Password',
  144. enabledBorder: inputBorderStyle,
  145. focusedBorder: inputBorderStyle,
  146. ),
  147. style: inputTextStyle,
  148. // The validator receives the text that the user has entered.
  149. validator: (value) {
  150. if (value == null || value.isEmpty) {
  151. return 'Enter a password';
  152. }
  153. return null;
  154. },
  155. ),
  156. const SizedBox(height: 15),
  157. changeUrl(),
  158. const SizedBox(height: 20),
  159. ElevatedButton(
  160. style: buttonStyle,
  161. onPressed: () {
  162. if (_formKey.currentState!.validate()) {
  163. login()
  164. .then((val) {
  165. Navigator.
  166. pushNamedAndRemoveUntil(
  167. context,
  168. '/home',
  169. ModalRoute.withName('/home'),
  170. );
  171. }).catchError((error) {
  172. showMessage(
  173. 'Could not login to Envelope, please try again later.',
  174. context,
  175. );
  176. });
  177. }
  178. },
  179. child: const Text('Submit'),
  180. ),
  181. ],
  182. )
  183. )
  184. )
  185. )
  186. );
  187. }
  188. Widget changeUrl() {
  189. if (_changedUrl) {
  190. return TextFormField(
  191. controller: _urlController,
  192. decoration: InputDecoration(
  193. hintText: 'URL',
  194. enabledBorder: OutlineInputBorder(
  195. borderRadius: BorderRadius.circular(5),
  196. borderSide: const BorderSide(
  197. color: Colors.transparent,
  198. )
  199. ),
  200. focusedBorder: OutlineInputBorder(
  201. borderRadius: BorderRadius.circular(5),
  202. borderSide: const BorderSide(
  203. color: Colors.transparent,
  204. )
  205. ),
  206. ),
  207. style: const TextStyle(
  208. fontSize: 18,
  209. ),
  210. // The validator receives the text that the user has entered.
  211. validator: (value) {
  212. if (value == null || value.isEmpty) {
  213. return 'Enter a URL';
  214. }
  215. return null;
  216. },
  217. );
  218. }
  219. return GestureDetector(
  220. onTap: () {
  221. setState(() {
  222. _changedUrl = true;
  223. });
  224. },
  225. child: Row(
  226. mainAxisAlignment: MainAxisAlignment.end,
  227. children: [
  228. Text(
  229. 'Change URL',
  230. style: TextStyle(
  231. color: Theme.of(context).disabledColor,
  232. ),
  233. ),
  234. const SizedBox(width: 10),
  235. Icon(
  236. Icons.edit,
  237. color: Theme.of(context).disabledColor,
  238. size: 14,
  239. ),
  240. const SizedBox(width: 10),
  241. ],
  242. )
  243. );
  244. }
  245. Future<dynamic> login() async {
  246. if (_urlController.text != _initialServerUrl) {
  247. await MyProfile.setServerUrl(_urlController.text);
  248. }
  249. final resp = await http.post(
  250. await MyProfile.getServerUrl('api/v1/login'),
  251. headers: <String, String>{
  252. 'Content-Type': 'application/json; charset=UTF-8',
  253. },
  254. body: jsonEncode(<String, String>{
  255. 'username': _usernameController.text,
  256. 'password': _passwordController.text,
  257. }),
  258. );
  259. if (resp.statusCode != 200) {
  260. throw Exception(resp.body);
  261. }
  262. String? rawCookie = resp.headers['set-cookie'];
  263. if (rawCookie != null) {
  264. int index = rawCookie.indexOf(';');
  265. setSessionCookie((index == -1) ? rawCookie : rawCookie.substring(0, index));
  266. }
  267. return await MyProfile.login(
  268. json.decode(resp.body),
  269. _passwordController.text,
  270. );
  271. }
  272. }