Compare commits

...

60 Commits

Author SHA1 Message Date
ccf84e3171 feat(chat): Добавлена возможность назначать пользователей администраторами
All checks were successful
Android Release / release (push) Successful in 5m51s
Это изменение добавляет новую функцию, которая позволяет назначать выбранных пользователей администраторами в выбранных чатах.

Ключевые изменения:

- В главное меню добавлен новый пункт "Назначить администратором".
- Проведен рефакторинг для повышения надежности: для хранения пользователей теперь используется Map<Integer, String> вместо двух отдельных списков, что исключает рассинхронизацию ID и имен.
- Введен enum UserAction (ADD, REMOVE, SET_ADMIN) для более чистого управления действиями с пользователями.
- В VkApiClient добавлен метод setMemberRole для взаимодействия с VK API.
- Обновлены диалоги подтверждения и итоговые отчеты для поддержки новой функции.
2026-01-25 02:35:15 +03:00
7ed50df7df Remove all fdroid v2
All checks were successful
Android Release / release (push) Successful in 29s
2026-01-19 01:59:34 +03:00
38c6b79bc2 Remove all fdroid
Some checks failed
Android Release / release (push) Failing after 3m32s
2026-01-19 01:50:27 +03:00
3dc72e991d change metadata
Some checks failed
Android Release / release (push) Failing after 5m16s
2026-01-19 01:43:47 +03:00
f65a12a113 change metadata
Some checks failed
Android Release / release (push) Failing after 5m20s
2026-01-19 01:37:52 +03:00
3be25d8161 change metadata
Some checks failed
Android Release / release (push) Failing after 5m20s
2026-01-19 01:30:38 +03:00
62a92a65f1 change config.yml
Some checks failed
Android Release / release (push) Failing after 6m22s
2026-01-19 01:16:30 +03:00
6d1c6541b3 change config.yml 2026-01-19 01:12:13 +03:00
9c32bf7d7d change docker to package
Some checks failed
Android Release / release (push) Failing after 5m53s
2026-01-19 00:49:59 +03:00
9aaced3d7a change docker to package
Some checks failed
Android Release / release (push) Failing after 6m7s
2026-01-19 00:41:50 +03:00
e1a4f36543 change docker dir
Some checks failed
Android Release / release (push) Failing after 3m42s
2026-01-19 00:35:05 +03:00
6ebf5b02dd remove --clean
Some checks failed
Android Release / release (push) Failing after 5m16s
2026-01-19 00:28:21 +03:00
553f591918 Move Release
Some checks failed
Android Release / release (push) Failing after 3m46s
2026-01-19 00:23:21 +03:00
c422294d7c Debug add and fixes
Some checks failed
Android Release / release (push) Has been cancelled
2026-01-19 00:20:44 +03:00
45938f7cec Debug add
Some checks failed
Android Release / release (push) Failing after 4m27s
2026-01-19 00:12:49 +03:00
0c08aae266 Fix directory location
Some checks failed
Android Release / release (push) Failing after 3m40s
2026-01-19 00:05:10 +03:00
1b3712eeca Fix Sign
Some checks failed
Android Release / release (push) Failing after 4m1s
2026-01-18 23:58:51 +03:00
82468c7e10 Fix Repo
Some checks failed
Android Release / release (push) Failing after 4m25s
2026-01-18 23:45:53 +03:00
e29de59384 Fix Repo
Some checks are pending
Android Release / release (push) Has started running
2026-01-18 23:45:31 +03:00
2658560210 Fix Repo
All checks were successful
Android Release / release (push) Successful in 31s
2026-01-18 23:43:05 +03:00
68ce7346b1 Fix Repo
Some checks failed
Android Release / release (push) Failing after 3m46s
2026-01-18 23:20:23 +03:00
aa01fcabfb Fix version Check
Some checks failed
Android Release / release (push) Failing after 4m11s
2026-01-18 23:09:45 +03:00
a83b432db6 Fix version Check
Some checks are pending
Android Release / release (push) Has started running
2026-01-18 23:08:59 +03:00
4a328d33e0 APK Rename
Some checks failed
Android Release / release (push) Failing after 4m2s
2026-01-18 22:57:57 +03:00
febd4f138c Some fixes
Some checks failed
Android Release / release (push) Has been cancelled
2026-01-18 22:54:38 +03:00
23771a9b59 Add F-Droid sign + fix run container
Some checks failed
Android Release / release (push) Failing after 4m18s
2026-01-18 22:42:40 +03:00
97de9776a1 Fix Container url
Some checks failed
Android Release / release (push) Failing after 6m15s
2026-01-18 22:25:50 +03:00
7c8e2d58f5 Fix EOL Stop
Some checks failed
Android Release / release (push) Failing after 3m38s
2026-01-18 22:20:17 +03:00
3c77188985 Fix Stop if version already released
Some checks failed
Android Release / release (push) Failing after 3m29s
2026-01-18 22:13:12 +03:00
422ff51b9e Fix setup-android
Some checks failed
Android Release / release (push) Failing after 1m8s
2026-01-18 22:10:27 +03:00
50090c8ed7 Bump to check CI/CD
Some checks failed
Android Release / release (push) Failing after 5s
2026-01-18 22:09:19 +03:00
6f5168bc27 Bump to check CI/CD
Some checks failed
Android Release / release (push) Failing after 5m16s
2026-01-18 19:28:22 +03:00
5684eb62b2 Fix Build Release APK (Maybe)
All checks were successful
Android Release / release (push) Successful in 3m11s
2026-01-18 01:35:00 +03:00
21ec78154d Fix Build Release APK (Maybe)
Some checks failed
Android Release / release (push) Failing after 49s
2026-01-18 01:32:43 +03:00
7f6fcc6c79 Fix Build Release APK (Maybe)
Some checks failed
Android Release / release (push) Failing after 2m48s
2026-01-18 01:27:39 +03:00
e8a1df940c Fix Build Release APK (Maybe)
Some checks failed
Android Release / release (push) Failing after 1m51s
2026-01-18 01:24:13 +03:00
5dc5dbbe57 Fix Build Release APK (Maybe)
Some checks failed
Android Release / release (push) Failing after 2m37s
2026-01-18 01:20:06 +03:00
53247d46e1 Fix Build Release APK (Maybe)
Some checks failed
Android Release / release (push) Failing after 1m23s
2026-01-18 01:18:36 +03:00
b5ade63bf9 Fix Build Release APK (Maybe)
Some checks failed
Android Release / release (push) Failing after 3m26s
2026-01-18 01:10:42 +03:00
683cc5759c Bump version to check CI
Some checks are pending
Android Release / release (push) Has started running
2026-01-18 01:09:22 +03:00
7b9aed8842 Fix tag check
Some checks are pending
Android Release / release (push) Has started running
2026-01-18 01:08:55 +03:00
5137be71c7 Fix tag check
All checks were successful
Android Release / release (push) Successful in 30s
2026-01-18 01:06:24 +03:00
1c7441c76b Fix tag check
Some checks failed
Android Release / release (push) Failing after 23s
2026-01-18 01:05:17 +03:00
c785569077 Fix tag check
Some checks are pending
Android Release / release (push) Has started running
2026-01-18 01:03:02 +03:00
5b1a0849f2 Fix tag check
Some checks failed
Android Release / release (push) Has been cancelled
2026-01-18 01:01:21 +03:00
a4183e6900 Fix tag check
Some checks failed
Android Release / release (push) Failing after 2m7s
2026-01-18 00:59:08 +03:00
cf39fec1ad Fix app/app/keystore
Some checks failed
Android Release / release (push) Failing after 2m33s
2026-01-18 00:55:50 +03:00
4e312a9685 Ready Fixes
Some checks failed
Android Release / release (push) Failing after 3m18s
2026-01-18 00:51:21 +03:00
b05f23c107 Ready Fixes
Some checks failed
Android Release / release (push) Failing after 1m44s
2026-01-18 00:43:35 +03:00
7b0f078fa3 Ready Fixes
Some checks failed
Android Release / release (push) Failing after 2m42s
2026-01-18 00:39:43 +03:00
84b83c1607 Ready Fixes
Some checks failed
Android Release / release (push) Failing after 14s
2026-01-18 00:38:30 +03:00
2bb469496b Ready Fixes
Some checks failed
Android Release / release (push) Failing after 21s
2026-01-18 00:37:20 +03:00
ba142431b6 Fixes Keystore Location
Some checks are pending
Android Release / release (push) Has started running
2026-01-18 00:35:54 +03:00
f9bbd337dc Fixes Android SDK
Some checks failed
Android Release / release (push) Failing after 2m37s
2026-01-18 00:30:43 +03:00
efb867b979 Fixes
Some checks failed
Android Release / release (push) Failing after 21s
2026-01-18 00:28:33 +03:00
08bd35bf8f Fixes
Some checks failed
Android Release / release (push) Failing after 1m13s
2026-01-18 00:08:20 +03:00
83e2c2f908 use out inctance
Some checks failed
Android Release / release (push) Failing after 17s
2026-01-18 00:01:53 +03:00
b07b79388f use gitea repos
Some checks failed
Android Release / release (push) Failing after 8s
2026-01-17 23:59:54 +03:00
dcde55fd1a Исправление actions
Some checks failed
Android Release / release (push) Failing after 8s
2026-01-17 23:46:17 +03:00
19f6ba4d65 Добавлены Actions для автоматической сборки проекта
Some checks failed
Android Release / release (push) Failing after 1m3s
2026-01-17 23:37:17 +03:00
9 changed files with 195 additions and 44 deletions

View File

@@ -0,0 +1,98 @@
name: Android Release
on:
push:
branches:
- master
jobs:
release:
runs-on: ubuntu-latest
steps:
# ------------------- Checkout -------------------
- name: Checkout
uses: https://git.daemonlord.ru/actions/checkout@v4
with:
fetch-depth: 0
tags: true
# ------------------- Setup JDK -------------------
- name: Set up JDK 17
uses: https://git.daemonlord.ru/actions/setup-java@v4
with:
distribution: temurin
java-version: 17
# ------------------- Install Node.js -------------------
- name: Install Node.js
run: |
curl -fsSL https://deb.nodesource.com/setup_22.x | bash -
apt-get install -y nodejs
# ------------------- Extract version -------------------
- name: Extract versionName
id: extract_version
run: |
VERSION=$(grep -oP 'versionName\s+"[^"]+"' app/build.gradle | head -n1 | cut -d'"' -f2 | tr -d '\r\n')
echo "version=$VERSION" >> $GITHUB_OUTPUT
echo "Detected version: $VERSION"
# ------------------- Stop if already released -------------------
- name: Stop if version already released
id: stop
run: |
VERSION=${{ steps.extract_version.outputs.version }}
if git show-ref --tags --quiet --verify "refs/tags/$VERSION"; then
echo "Version $VERSION already released, stopping job."
echo "CONTINUE=false" >> $GITHUB_ENV
else
echo "Version $VERSION not released yet, continuing workflow..."
echo "CONTINUE=true" >> $GITHUB_ENV
fi
# ------------------- Decode keystore -------------------
- name: Decode keystore
if: env.CONTINUE == 'true'
run: |
echo "${{ secrets.KEYSTORE_BASE64 }}" | base64 -d > app/release.keystore
- name: Make Gradlew executable
if: env.CONTINUE == 'true'
run: chmod +x ./gradlew
# ------------------- Set up Android SDK -------------------
- name: Set up Android SDK
if: env.CONTINUE == 'true'
uses: https://git.daemonlord.ru/actions/setup-android@v3
# ------------------- Build Release APK -------------------
- name: Build Release APK
if: env.CONTINUE == 'true'
env:
KEYSTORE_PASSWORD: ${{ secrets.KEYSTORE_PASSWORD }}
KEY_ALIAS: ${{ secrets.KEY_ALIAS }}
KEY_PASSWORD: ${{ secrets.KEY_PASSWORD }}
run: ./gradlew --no-daemon assembleRelease
# ------------------- Git tag -------------------
- name: Create git tag
if: env.CONTINUE == 'true'
run: |
git tag $VERSION
git push origin $VERSION
# ------------------- Gitea release -------------------
- name: Create Gitea Release
if: env.CONTINUE == 'true'
uses: https://git.daemonlord.ru/actions/gitea-release-action@v1
with:
server_url: https://git.daemonlord.ru
repository: ${{ gitea.repository }}
token: ${{ secrets.API_TOKEN }}
tag_name: ${{ steps.extract_version.outputs.version }}
name: Release ${{ steps.extract_version.outputs.version }}
body: |
Android release ${{ steps.extract_version.outputs.version }}
files: |
app/build/outputs/apk/release/*.apk

View File

@@ -4,26 +4,39 @@ plugins {
android {
namespace 'com.anabasis.vkchatmanager'
compileSdk {
version = release(36)
}
compileSdk 36
defaultConfig {
applicationId "com.anabasis.vkchatmanager"
minSdk 26
targetSdk 36
versionCode 3
versionName "1.1.1"
versionCode 10200
versionName "1.2.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
// --- Настройка signingConfigs ---
signingConfigs {
release {
// Путь к keystore — от корня проекта
storeFile file("release.keystore")
storePassword System.getenv("KEYSTORE_PASSWORD") ?: "change_me"
keyAlias System.getenv("KEY_ALIAS") ?: "change_me"
keyPassword System.getenv("KEY_PASSWORD") ?: "change_me"
}
}
buildTypes {
release {
minifyEnabled true
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
// Используем signingConfig release
signingConfig signingConfigs.release
}
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_17
targetCompatibility JavaVersion.VERSION_17
@@ -31,7 +44,6 @@ android {
}
dependencies {
implementation libs.recyclerview
implementation libs.viewpager2
implementation libs.material

View File

@@ -17,7 +17,6 @@ import android.widget.Toast;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
@@ -42,21 +41,23 @@ import org.json.JSONArray;
import org.json.JSONObject;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
public class MainActivity extends AppCompatActivity {
private enum UserAction {
ADD, REMOVE, SET_ADMIN
}
private VkApiClient api;
private final List<VkChat> office = new ArrayList<>();
private final List<VkChat> retail = new ArrayList<>();
private final List<VkChat> warehouse = new ArrayList<>();
private final List<VkChat> coffee = new ArrayList<>();
private final List<VkChat> other = new ArrayList<>();
private final List<Integer> userIdsToProcess = new ArrayList<>();
private final List<String> userNamesToProcess = new ArrayList<>();
private final Map<Integer, String> usersToProcess = new LinkedHashMap<>();
private LinearProgressIndicator progressBar;
private androidx.swiperefreshlayout.widget.SwipeRefreshLayout swipeRefresh;
@@ -74,13 +75,13 @@ public class MainActivity extends AppCompatActivity {
result -> {
if (result.getResultCode() == RESULT_OK) {
recreate();
} else {
finish();
}
}
);
});
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
WindowCompat.setDecorFitsSystemWindows(getWindow(), false);
setContentView(R.layout.activity_main);
@@ -154,8 +155,8 @@ public class MainActivity extends AppCompatActivity {
}
});
addBtn.setOnClickListener(v -> processUsers(true));
removeBtn.setOnClickListener(v -> processUsers(false));
addBtn.setOnClickListener(v -> processUsers(UserAction.ADD));
removeBtn.setOnClickListener(v -> processUsers(UserAction.REMOVE));
showUsersBtn.setOnClickListener(v -> showUsers());
}
@@ -187,6 +188,11 @@ public class MainActivity extends AppCompatActivity {
return true;
}
if (id == R.id.action_set_admin) {
processUsers(UserAction.SET_ADMIN);
return true;
}
if (id == R.id.action_token_status) {
showTokenStatus();
return true;
@@ -383,8 +389,7 @@ public class MainActivity extends AppCompatActivity {
/* ===================== LINKS ===================== */
private void processLinksList(List<String> links) {
userIdsToProcess.clear();
userNamesToProcess.clear();
usersToProcess.clear();
setUiEnabled(false);
@@ -394,8 +399,7 @@ public class MainActivity extends AppCompatActivity {
String screen = extractScreenName(link);
int uid = api.resolveUserId(screen);
if (uid > 0) {
userIdsToProcess.add(uid);
userNamesToProcess.add(api.getUserName(uid));
usersToProcess.put(uid, api.getUserName(uid));
}
} catch (TokenExpiredException e) {
runOnUiThread(this::handleTokenExpired);
@@ -406,7 +410,7 @@ public class MainActivity extends AppCompatActivity {
setUiEnabled(true);
Toast.makeText(
this,
getString(R.string.users_loaded_message, userIdsToProcess.size()),
getString(R.string.users_loaded_message, usersToProcess.size()),
Toast.LENGTH_SHORT
).show();
});
@@ -431,16 +435,16 @@ public class MainActivity extends AppCompatActivity {
return link;
}
/* ===================== ADD / REMOVE ===================== */
/* ===================== ADD / REMOVE / SET_ADMIN ===================== */
private void showUsers() {
if (userNamesToProcess.isEmpty()) {
if (usersToProcess.isEmpty()) {
Toast.makeText(this, getString(R.string.user_list_empty), Toast.LENGTH_SHORT).show();
return;
}
StringBuilder msg = new StringBuilder();
for (String name : userNamesToProcess) {
for (String name : usersToProcess.values()) {
msg.append("").append(name).append("\n");
}
@@ -451,10 +455,13 @@ public class MainActivity extends AppCompatActivity {
.show();
}
private void processUsers(boolean add) {
private void processUsers(UserAction action) {
List<VkChat> chats = getCurrentFragment().getSelected();
if (chats.isEmpty() || userIdsToProcess.isEmpty()) {
ChatListFragment currentFragment = getCurrentFragment();
if (currentFragment == null) return;
List<VkChat> chats = currentFragment.getSelected();
if (chats.isEmpty() || usersToProcess.isEmpty()) {
Toast.makeText(this,
getString(R.string.no_chats_or_users_selected_message),
Toast.LENGTH_SHORT).show();
@@ -462,10 +469,20 @@ public class MainActivity extends AppCompatActivity {
}
StringBuilder msg = new StringBuilder();
msg.append(add ? getString(R.string.add_users_dialog_message) : getString(R.string.remove_users_dialog_message));
switch (action) {
case ADD:
msg.append(getString(R.string.add_users_dialog_message));
break;
case REMOVE:
msg.append(getString(R.string.remove_users_dialog_message));
break;
case SET_ADMIN:
msg.append(getString(R.string.set_admin_dialog_message));
break;
}
msg.append("\n\n");
for (String name : userNamesToProcess) {
for (String name : usersToProcess.values()) {
msg.append("").append(name).append("\n");
}
@@ -480,18 +497,18 @@ public class MainActivity extends AppCompatActivity {
.setTitle(getString(R.string.confirmation_dialog_title))
.setMessage(msg.toString())
.setPositiveButton(getString(R.string.confirm_button), (d, w) ->
executeUsers(add, chats))
executeUsers(action, chats))
.setNegativeButton(getString(R.string.cancel_button), null)
.show();
}
private void executeUsers(boolean add, List<VkChat> chats) {
private void executeUsers(UserAction action, List<VkChat> chats) {
setUiEnabled(false);
progressBar.setVisibility(View.VISIBLE);
progressBar.setIndeterminate(false);
progressBar.setProgress(0, true);
int totalOps = chats.size() * userIdsToProcess.size();
int totalOps = chats.size() * usersToProcess.size();
progressBar.setMax(totalOps);
new Thread(() -> {
@@ -499,19 +516,27 @@ public class MainActivity extends AppCompatActivity {
int done = 0;
for (VkChat c : chats) {
for (int i = 0; i < userIdsToProcess.size(); i++) {
int uid = userIdsToProcess.get(i);
String userName = userNamesToProcess.get(i);
for (Map.Entry<Integer, String> user : usersToProcess.entrySet()) {
int uid = user.getKey();
String userName = user.getValue();
String resultMessage;
try {
if (add) {
api.addUser(c.id, uid, true);
resultMessage = getString(R.string.op_success_add_format, userName, c.title);
} else {
api.removeUser(c.id, uid);
resultMessage = getString(R.string.op_success_remove_format, userName, c.title);
}
resultMessage = switch (action) {
case ADD -> {
api.addUser(c.id, uid, true);
yield getString(R.string.op_success_add_format, userName, c.title);
}
case REMOVE -> {
api.removeUser(c.id, uid);
yield getString(R.string.op_success_remove_format, userName, c.title);
}
case SET_ADMIN -> {
api.setMemberRole(c.id, uid, "admin");
yield getString(R.string.op_success_set_admin_format, userName, c.title);
}
};
} catch (TokenExpiredException e) {
runOnUiThread(this::handleTokenExpired);
return;

View File

@@ -59,6 +59,13 @@ public class VkApiClient {
"&member_id=" + userId);
}
public void setMemberRole(int chatId, int userId, String role) throws Exception {
call("messages.setMemberRole",
"peer_id=" + (2000000000 + chatId) +
"&member_id=" + userId +
"&role=" + role);
}
public int resolveUserId(String screenName) throws Exception {
JSONObject resp = call(

View File

@@ -12,6 +12,9 @@
<item
android:id="@+id/action_deselect_all"
android:title="@string/action_deselect_all" />
<item
android:id="@+id/action_set_admin"
android:title="@string/action_set_admin" />
<item
android:id="@+id/action_token_status"
android:title="@string/action_token_status" />

View File

@@ -19,6 +19,7 @@
<string name="no_chats_or_users_selected_message">Не выбраны чаты или пользователи</string>
<string name="add_users_dialog_message">Добавить пользователей:</string>
<string name="remove_users_dialog_message">Удалить пользователей:</string>
<string name="set_admin_dialog_message">Назначить администратором:</string>
<string name="in_chats_dialog_message">В чаты:</string>
<string name="confirmation_dialog_title">Подтверждение</string>
<string name="confirm_button">Подтвердить</string>
@@ -27,6 +28,7 @@
<string name="action_refresh">Обновить</string>
<string name="action_select_all">Выбрать все</string>
<string name="action_deselect_all">Снять выбор</string>
<string name="action_set_admin">Назначить администратором</string>
<string name="action_token_status">Статус токена</string>
<string name="token_status_title">Статус токена</string>
<string name="token_status_perpetual">Токен бессрочный</string>
@@ -39,6 +41,7 @@
<string name="op_unknown_error">неизвестная ошибка</string>
<string name="op_success_add_format">✅ %1$s в %2$s: Успешно</string>
<string name="op_success_remove_format">✅ %1$s из %2$s: Успешно</string>
<string name="op_success_set_admin_format">✅ %1$s в %2$s: Назначен администратором</string>
<string name="op_failure_format">❌ %1$s в %2$s: %3$s</string>
<string name="action_sign_out">Выйти</string>
<string name="api_error">Ошибка API: %s</string>

View File

@@ -19,6 +19,7 @@
<string name="no_chats_or_users_selected_message">No chats or users selected</string>
<string name="add_users_dialog_message">Add users:</string>
<string name="remove_users_dialog_message">Remove users:</string>
<string name="set_admin_dialog_message">Set as admin:</string>
<string name="in_chats_dialog_message">In chats:</string>
<string name="confirmation_dialog_title">Confirmation</string>
<string name="confirm_button">Confirm</string>
@@ -27,6 +28,7 @@
<string name="action_refresh">Refresh</string>
<string name="action_select_all">Select All</string>
<string name="action_deselect_all">Deselect All</string>
<string name="action_set_admin">Set as Admin</string>
<string name="action_token_status">Token Status</string>
<string name="token_status_title">Token Status</string>
<string name="token_status_perpetual">Token is perpetual</string>
@@ -39,6 +41,7 @@
<string name="op_unknown_error">unknown error</string>
<string name="op_success_add_format">✅ %1$s in %2$s: Success</string>
<string name="op_success_remove_format">✅ %1$s from %2$s: Success</string>
<string name="op_success_set_admin_format">✅ %1$s in %2$s: Is now an admin</string>
<string name="op_failure_format">❌ %1$s in %2$s: %3$s</string>
<string name="action_sign_out">Sign Out</string>
<string name="api_error">API error: %s</string>

BIN
assets/app_icon.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.5 KiB

BIN
assets/icon.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.5 KiB