Merge pull request 'Version 0.4.2' (#83) from inex/selfprivacy.org.app:fixes-2021-dec into master

Reviewed-on: https://git.selfprivacy.org/kherel/selfprivacy.org.app/pulls/83
This commit is contained in:
Illia Chub 2022-01-10 07:04:30 +02:00
commit 56fe8fd329
21 changed files with 231 additions and 58 deletions

BIN
assets/images/gifs/Backblaze.gif Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 MiB

BIN
assets/images/gifs/CloudFlare.gif Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 MiB

BIN
assets/images/gifs/Hetzner.gif Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.3 MiB

View file

@ -0,0 +1,8 @@
### How to get Backblaze API Token
1. Visit the following link and authorize: https://secure.backblaze.com/user_signin.htm
2. On the left side of the interface, select **App Keys** in the **B2 Cloud Storage** subcategory.
3. Click on the blue **Generate New Master Application Key** button.
4. In the appeared pop-up window confirm the generation.
5. Save _keyID_ and _applicationKey_ in the safe place. For example, in the password manager.
![Backblaze token setup](resource:assets/images/gifs/Backblaze.gif)

View file

@ -0,0 +1,8 @@
### Как получить Backblaze API Token
1. Переходим по ссылке https://secure.backblaze.com/user_signin.htm и авторизуемся.
2. В левой части интерфейса выбираем **App Keys** в подкатегории **"Account"**.
3. Нажимаем на синюю кнопку **Generate New Master Application Key**.
4. Во всплывающем окне подтверждаем генерацию.
5. Сохраняем _keyID_ и _applicationKey_ в надёжном месте. Например в менеджере паролей.
![Backblaze token setup](resource:assets/images/gifs/Backblaze.gif)

View file

@ -0,0 +1,17 @@
### How to get Cloudflare API Token
1. Visit the following link: https://dash.cloudflare.com/
2. the right corner, click on the profile icon (a man in a circle). For the mobile version of the site, in the upper left corner, click the **Menu** button (three horizontal bars), in the dropdown menu, click on **My Profile**
3. There are four configuration categories to choose from: *Communication*, *Authentication*, **API Tokens**, *Session*. Choose **API Tokens**.
4. Click on **Create Token** button.
5. Go down to the bottom and see the **Create Custom Token** field and press **Get Started** button on the right side.
6. In the **Token Name** field, give your token a name.
7. Next we have Permissions. In the leftmost field, select **Zone**. In the longest field, center, select **DNS**. In the rightmost field, select **Edit**.
8. Next, right under this line, click Add More. Similar field will appear.
9. In the leftmost field of the new line, select, similar to the last line — **Zone**. In the center — a little different. Here choose the same as in the left — **Zone**. In the rightmost field, select **Read**.
10. Next look at **Zone Resources**. Under this inscription there is a line with two fields. The left must have **Include** and the right must have **Specific Zone**. Once you select Specific Zone, another field appears on the right. Choose your domain in it.
11. Flick to the bottom and press the blue **Continue to Summary** button.
12. Check if you got everything right. A similar string must be present: *Domain — DNS:Edit, Zone:Read*.
13. Click on **Create Token**.
14. We copy the created token, and save it in a reliable place (preferably in the password manager).
![Cloudflare token setup](resource:assets/images/gifs/CloudFlare.gif)

View file

@ -0,0 +1,15 @@
### Как получить Cloudflare API Token
1. Переходим по [ссылке](https://dash.cloudflare.com/) и авторизуемся в ранее созданном аккаунте. https://dash.cloudflare.com/
В правом углу кликаем на иконку профиля (человечек в кружочке). Для мобильной версии сайта, в верхнем левом углу, нажимаем кнопку **Меню** (три горизонтальных полоски), в выпавшем меню, ищем пункт **My Profile**.
3. Нам предлагается на выбор, четыре категории настройки: **Preferences**, **Authentication**, **API Tokens**, **Sessions**. Выбираем **API Tokens**.
4. Самым первым пунктом видим кнопку **Create Token**. С полной уверенностью в себе и желанием обрести приватность, нажимаем на неё.
5. Спускаемся в самый низ и видим поле **Create Custom Token** и кнопку **Get Started** с правой стороны. Нажимаем.
6. В поле **Token Name** даём своему токену имя. Можете покреативить и отнестись к этому как к наименованию домашнего зверька :)
7. Далее, у нас **Permissions**. В первом поле выбираем Zone. Во втором поле, по центру, выбираем **DNS**. В последнем поле выбираем **Edit**.
8. Далее смотрим на **Zone Resources**. Под этой надписью есть строка с двумя полями. В первом должно быть **Include**, а во втором — **Specific Zone**. Как только Вы выберите **Specific Zone**, справа появится ещё одно поле. В нём выбираем наш домен.
9. Листаем в самый низ и нажимаем на синюю кнопку **Continue to Summary**.
10. Проверяем, всё ли мы правильно выбрали. Должна присутствовать подобная строка: ваш.домен — **DNS:Edit, Zone:Read**.
11. Нажимаем **Create Token**.
12. Копируем созданный токен, и сохраняем его в надёжном месте (желательно — в менеджере паролей).
![Cloudflare token setup](resource:assets/images/gifs/CloudFlare.gif)

View file

@ -1,3 +1,4 @@
### How to get Hetzner API Token
1. Visit the following [link](https://console.hetzner.cloud/) and sign 1. Visit the following [link](https://console.hetzner.cloud/) and sign
into newly created account. into newly created account.
2. Enter into previously created project. If you haven't created one, 2. Enter into previously created project. If you haven't created one,
@ -17,4 +18,6 @@
**permissions**. Pick **Read & Write**. **permissions**. Pick **Read & Write**.
8. Click **Generate API Token.** 8. Click **Generate API Token.**
9. After that, our key will be shown. Store it in the reliable place, 9. After that, our key will be shown. Store it in the reliable place,
or in the password manager, which is better. or in the password manager, which is better.
![Hetzner token setup](resource:assets/images/gifs/Hetzner.gif)

View file

@ -4,4 +4,6 @@
3. Наводим мышкой на боковую панель. Она должна раскрыться, показав нам пункты меню. Нас интересует последний — Security (с иконкой ключика). 3. Наводим мышкой на боковую панель. Она должна раскрыться, показав нам пункты меню. Нас интересует последний — Security (с иконкой ключика).
4. Далее, в верхней части интерфейса видим примерно такой список: SSH Keys, API Tokens, Certificates, Members. Нам нужен API Tokens. Переходим по нему. 4. Далее, в верхней части интерфейса видим примерно такой список: SSH Keys, API Tokens, Certificates, Members. Нам нужен API Tokens. Переходим по нему.
5. В правой части интерфейса, нас будет ожидать кнопка Generate API token. Если же Вы используете мобильную версию сайта, в нижнем правом углу Вы увидите красный плюсик. Нажимаем на эту кнопку. 5. В правой части интерфейса, нас будет ожидать кнопка Generate API token. Если же Вы используете мобильную версию сайта, в нижнем правом углу Вы увидите красный плюсик. Нажимаем на эту кнопку.
6. В поле Description, даём нашему токену название (это может быть любое название, которые Вам нравиться. Сути оно не меняет. 6. В поле Description, даём нашему токену название (это может быть любое название, которые Вам нравиться. Сути оно не меняет.
![Hetzner token setup](resource:assets/images/gifs/Hetzner.gif)

View file

@ -127,6 +127,7 @@
"mail": { "mail": {
"title": "E-Mail", "title": "E-Mail",
"subtitle": "E-Mail for company and family.", "subtitle": "E-Mail for company and family.",
"login_info": "Use username and password from users tab. IMAP port is 143 with STARTTLS, SMTP port is 587 with STARTTLS.",
"bottom_sheet": { "bottom_sheet": {
"1": "To connect to the mailserver, please use {} domain alongside with username and password, that you created. Also feel free to invite", "1": "To connect to the mailserver, please use {} domain alongside with username and password, that you created. Also feel free to invite",
"2": "new users" "2": "new users"
@ -135,6 +136,7 @@
"messenger": { "messenger": {
"title": "Messenger", "title": "Messenger",
"subtitle": "Telegram or Signal not so private as Delta.Chat that uses your private server.", "subtitle": "Telegram or Signal not so private as Delta.Chat that uses your private server.",
"login_info": "Use the same username and password as for e-mail.",
"bottom_sheet": { "bottom_sheet": {
"1": "For connection, please use {} domain and credentials that you created." "1": "For connection, please use {} domain and credentials that you created."
} }
@ -142,6 +144,7 @@
"password_manager": { "password_manager": {
"title": "Password Manager", "title": "Password Manager",
"subtitle": "Base of your security. Bitwarden will help you to create, store and move passwords between devices, as well as input them, when requested using autocompletion.", "subtitle": "Base of your security. Bitwarden will help you to create, store and move passwords between devices, as well as input them, when requested using autocompletion.",
"login_info": "You will have to create an account on the website.",
"bottom_sheet": { "bottom_sheet": {
"1": "You can connect to the service and create a user via this link:" "1": "You can connect to the service and create a user via this link:"
} }
@ -149,6 +152,7 @@
"video": { "video": {
"title": "Videomeet", "title": "Videomeet",
"subtitle": "Zoom and Google Meet are good, but Jitsi Meet is a worth alternative that also gives you confidence that you're not being listened.", "subtitle": "Zoom and Google Meet are good, but Jitsi Meet is a worth alternative that also gives you confidence that you're not being listened.",
"login_info": "No account needed.",
"bottom_sheet": { "bottom_sheet": {
"1": "Using Jitsi as simple as just visiting this link:" "1": "Using Jitsi as simple as just visiting this link:"
} }
@ -156,6 +160,7 @@
"cloud": { "cloud": {
"title": "Cloud Storage", "title": "Cloud Storage",
"subtitle": "Do not allow cloud services to read your data by using NextCloud.", "subtitle": "Do not allow cloud services to read your data by using NextCloud.",
"login_info": "Login is admin, password is the same as with your main user. Create new accounts in Nextcloud interface.",
"bottom_sheet": { "bottom_sheet": {
"1": "You can connect and create a new user here:" "1": "You can connect and create a new user here:"
} }
@ -163,6 +168,7 @@
"social_network": { "social_network": {
"title": "Social Network", "title": "Social Network",
"subtitle": "It's hard to believe, but it became possible to create your own social network, with your own rules and target audience.", "subtitle": "It's hard to believe, but it became possible to create your own social network, with your own rules and target audience.",
"login_info": "You will have to create an account on the website.",
"bottom_sheet": { "bottom_sheet": {
"1": "You can connect and create new social user here:" "1": "You can connect and create new social user here:"
} }
@ -170,6 +176,7 @@
"git": { "git": {
"title": "Git Server", "title": "Git Server",
"subtitle": "Private alternative to the Github, that belongs to you, but not a Microsoft.", "subtitle": "Private alternative to the Github, that belongs to you, but not a Microsoft.",
"login_info": "You will have to create an account on the website. First user will become an admin.",
"bottom_sheet": { "bottom_sheet": {
"1": "You can connect and create a new user here:" "1": "You can connect and create a new user here:"
} }
@ -248,7 +255,8 @@
"title": "Jobs list", "title": "Jobs list",
"start": "Start", "start": "Start",
"empty": "No jobs", "empty": "No jobs",
"createUser": "Create", "createUser": "Create user",
"deleteUser": "Delete user",
"serviceTurnOff": "Turn off", "serviceTurnOff": "Turn off",
"serviceTurnOn": "Turn on", "serviceTurnOn": "Turn on",
"jobAdded": "Job added", "jobAdded": "Job added",

View file

@ -128,6 +128,7 @@
"mail": { "mail": {
"title": "Почта", "title": "Почта",
"subtitle": "Электронная почта для семьи или компании.", "subtitle": "Электронная почта для семьи или компании.",
"login_info": "Используйте логин и пароль из вкладки пользователей. IMAP порт: 143, STARTTLS. SMTP порт: 587, STARTTLS.",
"bottom_sheet": { "bottom_sheet": {
"1": "Для подключения почтового ящика используйте {} и профиль, который Вы создали. Так же приглашайте", "1": "Для подключения почтового ящика используйте {} и профиль, который Вы создали. Так же приглашайте",
"2": "новых пользователей." "2": "новых пользователей."
@ -136,6 +137,7 @@
"messenger": { "messenger": {
"title": "Мессенджер", "title": "Мессенджер",
"subtitle": "Telegram и Signal не так приватны, как Delta.Chat — он использует Ваш личный сервер.", "subtitle": "Telegram и Signal не так приватны, как Delta.Chat — он использует Ваш личный сервер.",
"login_info": "Используйте те же логин и пароль, что и для почты.",
"bottom_sheet": { "bottom_sheet": {
"1": "Для подключения используйте {} и логин пароль, который Вы создали." "1": "Для подключения используйте {} и логин пароль, который Вы создали."
} }
@ -143,6 +145,7 @@
"password_manager": { "password_manager": {
"title": "Менеджер паролей", "title": "Менеджер паролей",
"subtitle": "Это фундамент Вашей безопасности. Создавать, хранить, копировать пароли между устройствами и вбивать их в формы поможет Bitwarden.", "subtitle": "Это фундамент Вашей безопасности. Создавать, хранить, копировать пароли между устройствами и вбивать их в формы поможет Bitwarden.",
"login_info": "Аккаунт нужно создать на сайте.",
"bottom_sheet": { "bottom_sheet": {
"1": "Подключиться к серверу и создать пользователя можно по адресу:." "1": "Подключиться к серверу и создать пользователя можно по адресу:."
} }
@ -150,6 +153,7 @@
"video": { "video": {
"title": "Видеоконференция", "title": "Видеоконференция",
"subtitle": "Jitsi meet — отличный аналог Zoom и Google meet который помимо удобства ещё и гарантирует Вам защищённые высококачественные видеоконференции.", "subtitle": "Jitsi meet — отличный аналог Zoom и Google meet который помимо удобства ещё и гарантирует Вам защищённые высококачественные видеоконференции.",
"login_info": "Аккаунт не требуется.",
"bottom_sheet": { "bottom_sheet": {
"1": "Для использования просто перейдите по ссылке:." "1": "Для использования просто перейдите по ссылке:."
} }
@ -157,6 +161,7 @@
"cloud": { "cloud": {
"title": "Файловое облако", "title": "Файловое облако",
"subtitle": "Не позволяйте облачным сервисам просматривать ваши данные. Используйте NextCloud — надёжный дом для всех Ваших данных.", "subtitle": "Не позволяйте облачным сервисам просматривать ваши данные. Используйте NextCloud — надёжный дом для всех Ваших данных.",
"login_info": "Логин администратора: admin, пароль такой же как у основного пользователя. Создавайте новых пользователей в интерфейсе администратора NextCloud.",
"bottom_sheet": { "bottom_sheet": {
"1": "Подключиться к серверу и создать пользователя можно по адресу:." "1": "Подключиться к серверу и создать пользователя можно по адресу:."
} }
@ -164,6 +169,7 @@
"social_network": { "social_network": {
"title": "Социальная сеть", "title": "Социальная сеть",
"subtitle": "Сложно поверить, но стало возможным создать свою собственную социальную сеть, со своими правилами и аудиторией.", "subtitle": "Сложно поверить, но стало возможным создать свою собственную социальную сеть, со своими правилами и аудиторией.",
"login_info": "Аккаунт нужно создать на сайте.",
"bottom_sheet": { "bottom_sheet": {
"1": "Подключиться к серверу и создать пользователя можно по адресу:." "1": "Подключиться к серверу и создать пользователя можно по адресу:."
} }
@ -171,6 +177,7 @@
"git": { "git": {
"title": "Git-сервер", "title": "Git-сервер",
"subtitle": "Приватная альтернатива Github, которая принадлежит вам, а не Microsoft.", "subtitle": "Приватная альтернатива Github, которая принадлежит вам, а не Microsoft.",
"login_info": "Аккаунт нужно создать на сайте. Первый зарегистрированный пользователь становится администратором.",
"bottom_sheet": { "bottom_sheet": {
"1": "Подключиться к серверу и создать пользователя можно по адресу:." "1": "Подключиться к серверу и создать пользователя можно по адресу:."
} }
@ -249,7 +256,8 @@
"title": "Задачи", "title": "Задачи",
"start": "Начать выполенение", "start": "Начать выполенение",
"empty": "Пусто.", "empty": "Пусто.",
"createUser": "Создать запись", "createUser": "Создать пользователя",
"deleteUser": "Удалить пользователя",
"serviceTurnOff": "Остановить", "serviceTurnOff": "Остановить",
"serviceTurnOn": "Запустить", "serviceTurnOn": "Запустить",
"jobAdded": "Задача добавленна", "jobAdded": "Задача добавленна",

View file

@ -114,7 +114,8 @@ class HetznerApi extends ApiMap {
final apiToken = StringGenerators.apiToken(); final apiToken = StringGenerators.apiToken();
final hostname = domainName.split('.')[0]; // Replace all non-alphanumeric characters with an underscore
final hostname = domainName.split('.')[0].replaceAll(RegExp(r'[^a-zA-Z0-9]'), '-');
/// add ssh key when you need it: e.g. "ssh_keys":["kherel"] /// add ssh key when you need it: e.g. "ssh_keys":["kherel"]
/// check the branch name, it could be "development" or "master". /// check the branch name, it could be "development" or "master".

View file

@ -74,6 +74,28 @@ class ServerApi extends ApiMap {
return res; return res;
} }
Future<bool> deleteUser(User user) async {
bool res;
Response response;
var client = await getClient();
try {
response = await client.delete(
'/users/${user.login}',
options: Options(
contentType: 'application/json',
),
);
res = response.statusCode == HttpStatus.ok;
} catch (e) {
print(e);
res = false;
}
close(client);
return res;
}
String get rootAddress => String get rootAddress =>
throw UnimplementedError('not used in with implementation'); throw UnimplementedError('not used in with implementation');

View file

@ -69,6 +69,46 @@ extension ServiceTypesExt on ServiceTypes {
} }
} }
String get loginInfo {
switch (this) {
case ServiceTypes.mail:
return 'services.mail.login_info'.tr();
case ServiceTypes.messenger:
return 'services.messenger.login_info'.tr();
case ServiceTypes.passwordManager:
return 'services.password_manager.login_info'.tr();
case ServiceTypes.video:
return 'services.video.login_info'.tr();
case ServiceTypes.cloud:
return 'services.cloud.login_info'.tr();
case ServiceTypes.socialNetwork:
return 'services.social_network.login_info'.tr();
case ServiceTypes.git:
return 'services.git.login_info'.tr();
case ServiceTypes.vpn:
return '';
}
}
String get subdomain {
switch (this) {
case ServiceTypes.passwordManager:
return 'password';
case ServiceTypes.video:
return 'meet';
case ServiceTypes.cloud:
return 'cloud';
case ServiceTypes.socialNetwork:
return 'social';
case ServiceTypes.git:
return 'git';
case ServiceTypes.vpn:
case ServiceTypes.messenger:
default:
return '';
}
}
IconData get icon { IconData get icon {
switch (this) { switch (this) {
case ServiceTypes.mail: case ServiceTypes.mail:

View file

@ -6,6 +6,7 @@ import 'package:selfprivacy/logic/models/backblaze_bucket.dart';
import 'package:selfprivacy/logic/models/backup.dart'; import 'package:selfprivacy/logic/models/backup.dart';
import 'package:selfprivacy/logic/api_maps/server.dart'; import 'package:selfprivacy/logic/api_maps/server.dart';
import 'package:selfprivacy/logic/api_maps/backblaze.dart'; import 'package:selfprivacy/logic/api_maps/backblaze.dart';
import 'package:easy_localization/easy_localization.dart';
part 'backups_state.dart'; part 'backups_state.dart';
@ -85,7 +86,7 @@ class BackupsCubit extends AppConfigDependendCubit<BackupsState> {
Future<void> createBucket() async { Future<void> createBucket() async {
emit(state.copyWith(preventActions: true)); emit(state.copyWith(preventActions: true));
final domain = final domain =
appConfigCubit.state.cloudFlareDomain!.domainName.replaceAll('.', '-'); appConfigCubit.state.cloudFlareDomain!.domainName.replaceAll(RegExp(r'[^a-zA-Z0-9]'), '-');
final serverId = appConfigCubit.state.hetznerServer!.id; final serverId = appConfigCubit.state.hetznerServer!.id;
var bucketName = 'selfprivacy-$domain-$serverId'; var bucketName = 'selfprivacy-$domain-$serverId';
// If bucket name is too long, shorten it // If bucket name is too long, shorten it
@ -151,7 +152,8 @@ class BackupsCubit extends AppConfigDependendCubit<BackupsState> {
Future<void> forceUpdateBackups() async { Future<void> forceUpdateBackups() async {
emit(state.copyWith(preventActions: true)); emit(state.copyWith(preventActions: true));
await api.forceBackupListReload(); await api.forceBackupListReload();
getIt<NavigationService>().showSnackBar('providers.backup.refetchingList'); getIt<NavigationService>()
.showSnackBar('providers.backup.refetchingList'.tr());
emit(state.copyWith(preventActions: false)); emit(state.copyWith(preventActions: false));
} }

View file

@ -69,15 +69,18 @@ class JobsCubit extends Cubit<JobsState> {
} }
Future<void> rebootServer() async { Future<void> rebootServer() async {
emit(JobsStateLoading());
final isSuccessful = await api.reboot(); final isSuccessful = await api.reboot();
if (isSuccessful) { if (isSuccessful) {
getIt<NavigationService>().showSnackBar('jobs.rebootSuccess'.tr()); getIt<NavigationService>().showSnackBar('jobs.rebootSuccess'.tr());
} else { } else {
getIt<NavigationService>().showSnackBar('jobs.rebootFailed'.tr()); getIt<NavigationService>().showSnackBar('jobs.rebootFailed'.tr());
} }
emit(JobsStateEmpty());
} }
Future<void> upgradeServer() async { Future<void> upgradeServer() async {
emit(JobsStateLoading());
final isPullSuccessful = await api.pullConfigurationUpdate(); final isPullSuccessful = await api.pullConfigurationUpdate();
final isSuccessful = await api.upgrade(); final isSuccessful = await api.upgrade();
if (isSuccessful) { if (isSuccessful) {
@ -89,6 +92,7 @@ class JobsCubit extends Cubit<JobsState> {
} else { } else {
getIt<NavigationService>().showSnackBar('jobs.upgradeFailed'.tr()); getIt<NavigationService>().showSnackBar('jobs.upgradeFailed'.tr());
} }
emit(JobsStateEmpty());
} }
Future<void> applyAll() async { Future<void> applyAll() async {
@ -101,7 +105,12 @@ class JobsCubit extends Cubit<JobsState> {
if (job is CreateUserJob) { if (job is CreateUserJob) {
newUsers.add(job.user); newUsers.add(job.user);
await api.createUser(job.user); await api.createUser(job.user);
} else if (job is ServiceToggleJob) { }
if (job is DeleteUserJob) {
final deleted = await api.deleteUser(job.user);
if (deleted) usersCubit.remove(job.user);
}
if (job is ServiceToggleJob) {
hasServiceJobs = true; hasServiceJobs = true;
await api.switchService(job.type, job.needToTurnOn); await api.switchService(job.type, job.needToTurnOn);
} }

View file

@ -31,6 +31,17 @@ class CreateUserJob extends Job {
List<Object> get props => [id, title, user]; List<Object> get props => [id, title, user];
} }
class DeleteUserJob extends Job {
DeleteUserJob({
required this.user,
}) : super(title: '${"jobs.deleteUser".tr()} ${user.login}');
final User user;
@override
List<Object> get props => [id, title, user];
}
class ServiceToggleJob extends Job { class ServiceToggleJob extends Job {
ServiceToggleJob({ ServiceToggleJob({
required this.type, required this.type,

View file

@ -137,7 +137,8 @@ class InitializingPage extends StatelessWidget {
), ),
SizedBox(height: 10), SizedBox(height: 10),
BrandButton.text( BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()), onPressed: () =>
_showModal(context, _HowTo(fileName: 'how_hetzner')),
title: 'initializing.how'.tr(), title: 'initializing.how'.tr(),
), ),
], ],
@ -192,7 +193,11 @@ class InitializingPage extends StatelessWidget {
), ),
SizedBox(height: 10), SizedBox(height: 10),
BrandButton.text( BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()), onPressed: () => _showModal(
context,
_HowTo(
fileName: 'how_cloudflare',
)),
title: 'initializing.how'.tr(), title: 'initializing.how'.tr(),
), ),
], ],
@ -243,7 +248,11 @@ class InitializingPage extends StatelessWidget {
), ),
SizedBox(height: 10), SizedBox(height: 10),
BrandButton.text( BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()), onPressed: () => _showModal(
context,
_HowTo(
fileName: 'how_backblaze',
)),
title: 'initializing.how'.tr(), title: 'initializing.how'.tr(),
), ),
], ],
@ -334,12 +343,9 @@ class InitializingPage extends StatelessWidget {
text: 'initializing.10'.tr(), text: 'initializing.10'.tr(),
), ),
], ],
SizedBox(height: 10), SizedBox(
Spacer(), height: 10,
SizedBox(height: 10), width: double.infinity,
BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()),
title: 'initializing.how'.tr(),
), ),
], ],
); );
@ -403,11 +409,6 @@ class InitializingPage extends StatelessWidget {
: () => context.read<RootUserFormCubit>().trySubmit(), : () => context.read<RootUserFormCubit>().trySubmit(),
text: 'basis.connect'.tr(), text: 'basis.connect'.tr(),
), ),
SizedBox(height: 10),
BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()),
title: 'initializing.how'.tr(),
),
], ],
); );
}), }),
@ -431,11 +432,6 @@ class InitializingPage extends StatelessWidget {
: () => appConfigCubit.createServerAndSetDnsRecords(), : () => appConfigCubit.createServerAndSetDnsRecords(),
text: isLoading ? 'basis.loading'.tr() : 'initializing.11'.tr(), text: isLoading ? 'basis.loading'.tr() : 'initializing.11'.tr(),
), ),
Spacer(flex: 2),
BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()),
title: 'initializing.what'.tr(),
),
], ],
); );
}); });
@ -482,13 +478,6 @@ class InitializingPage extends StatelessWidget {
], ],
), ),
if (state.isLoading) BrandText.body2('initializing.17'.tr()), if (state.isLoading) BrandText.body2('initializing.17'.tr()),
Spacer(
flex: 2,
),
BrandButton.text(
onPressed: () => _showModal(context, _HowHetzner()),
title: 'initializing.what'.tr(),
),
], ],
); );
}); });
@ -503,11 +492,14 @@ class InitializingPage extends StatelessWidget {
} }
} }
class _HowHetzner extends StatelessWidget { class _HowTo extends StatelessWidget {
const _HowHetzner({ const _HowTo({
Key? key, Key? key,
required this.fileName,
}) : super(key: key); }) : super(key: key);
final String fileName;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BrandBottomSheet( return BrandBottomSheet(
@ -515,7 +507,7 @@ class _HowHetzner extends StatelessWidget {
child: Padding( child: Padding(
padding: paddingH15V0, padding: paddingH15V0,
child: BrandMarkdown( child: BrandMarkdown(
fileName: 'how_hetzner', fileName: fileName,
), ),
), ),
); );

View file

@ -38,6 +38,23 @@ class ServicesPage extends StatefulWidget {
_ServicesPageState createState() => _ServicesPageState(); _ServicesPageState createState() => _ServicesPageState();
} }
void _launchURL(url) async {
var _possible = await canLaunch(url);
if (_possible) {
try {
await launch(
url,
enableJavaScript: true,
);
} catch (e) {
print(e);
}
} else {
throw 'Could not launch $url';
}
}
class _ServicesPageState extends State<ServicesPage> { class _ServicesPageState extends State<ServicesPage> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -94,6 +111,9 @@ class _Card extends StatelessWidget {
(!switchableServices.contains(serviceType) || (!switchableServices.contains(serviceType) ||
serviceState.isEnableByType(serviceType)); serviceState.isEnableByType(serviceType));
var config = context.watch<AppConfigCubit>().state;
var domainName = UiHelpers.getDomainName(config);
return GestureDetector( return GestureDetector(
onTap: isSwithOn onTap: isSwithOn
? () => showDialog<void>( ? () => showDialog<void>(
@ -163,6 +183,30 @@ class _Card extends StatelessWidget {
SizedBox(height: 10), SizedBox(height: 10),
BrandText.h2(serviceType.title), BrandText.h2(serviceType.title),
SizedBox(height: 10), SizedBox(height: 10),
if (serviceType.subdomain != '')
Column(
children: [
GestureDetector(
onTap: () => _launchURL(
'https://${serviceType.subdomain}.$domainName'),
child: Text(
'${serviceType.subdomain}.$domainName',
style: linkStyle,
),
),
SizedBox(height: 10),
],
),
if (serviceType == ServiceTypes.mail)
Column(children: [
Text(
domainName,
style: linkStyle,
),
SizedBox(height: 10),
]),
BrandText.body2(serviceType.loginInfo),
SizedBox(height: 10),
BrandText.body2(serviceType.subtitle), BrandText.body2(serviceType.subtitle),
SizedBox(height: 10), SizedBox(height: 10),
], ],
@ -438,21 +482,4 @@ class _ServiceDetails extends StatelessWidget {
), ),
); );
} }
void _launchURL(url) async {
var _possible = await canLaunch(url);
if (_possible) {
try {
await launch(
url,
enableJavaScript: true,
);
} catch (e) {
print(e);
}
} else {
throw 'Could not launch $url';
}
}
} }

View file

@ -75,9 +75,8 @@ class _UserDetails extends StatelessWidget {
), ),
), ),
onPressed: () { onPressed: () {
context context.read<JobsCubit>().addJob(
.read<UsersCubit>() DeleteUserJob(user: user));
.remove(user);
Navigator.of(context) Navigator.of(context)
..pop() ..pop()
..pop(); ..pop();

View file

@ -1,7 +1,7 @@
name: selfprivacy name: selfprivacy
description: selfprivacy.org description: selfprivacy.org
publish_to: 'none' publish_to: 'none'
version: 0.4.1+9 version: 0.4.2+10
environment: environment:
sdk: '>=2.13.4 <3.0.0' sdk: '>=2.13.4 <3.0.0'
@ -61,6 +61,7 @@ flutter:
- assets/images/ - assets/images/
- assets/images/onboarding/ - assets/images/onboarding/
- assets/images/logos/ - assets/images/logos/
- assets/images/gifs/
- assets/translations/ - assets/translations/
- assets/markdown/ - assets/markdown/
fonts: fonts: