From d4818c556725dba9a2fe1b9d8de94603accb3292 Mon Sep 17 00:00:00 2001 From: J-Jamet Date: Wed, 12 Jan 2022 14:19:04 +0100 Subject: [PATCH] Select entry colors --- .../keepass/activities/EntryEditActivity.kt | 11 +++ .../dialogs/ColorPickerDialogFragment.kt | 73 ++++++++++++++ .../activities/fragments/EntryEditFragment.kt | 14 +++ .../keepass/adapters/NodesAdapter.kt | 27 +++++- .../keepass/database/element/Entry.kt | 42 ++++++++ .../com/kunzisoft/keepass/model/EntryInfo.kt | 12 +++ .../keepass/view/TemplateAbstractView.kt | 4 + .../keepass/view/TemplateEditView.kt | 24 +++++ .../viewmodels/ColorPickerViewModel.kt | 15 +++ .../keepass/viewmodels/NodeEditViewModel.kt | 29 ++++++ .../ic_color_background_white_24dp.xml | 13 +++ .../ic_color_foreground_white_24dp.xml | 13 +++ app/src/main/res/layout/activity_group.xml | 2 +- .../main/res/layout/fragment_color_picker.xml | 50 ++++++++++ .../main/res/layout/item_list_nodes_entry.xml | 53 +++++----- app/src/main/res/layout/view_template.xml | 68 +++++++++---- app/src/main/res/values/strings.xml | 3 + art/ic_color_background.svg | 97 +++++++++++++++++++ art/ic_color_foreground.svg | 97 +++++++++++++++++++ 19 files changed, 593 insertions(+), 54 deletions(-) create mode 100644 app/src/main/java/com/kunzisoft/keepass/activities/dialogs/ColorPickerDialogFragment.kt create mode 100644 app/src/main/java/com/kunzisoft/keepass/viewmodels/ColorPickerViewModel.kt create mode 100644 app/src/main/res/drawable/ic_color_background_white_24dp.xml create mode 100644 app/src/main/res/drawable/ic_color_foreground_white_24dp.xml create mode 100644 app/src/main/res/layout/fragment_color_picker.xml create mode 100644 art/ic_color_background.svg create mode 100644 art/ic_color_foreground.svg diff --git a/app/src/main/java/com/kunzisoft/keepass/activities/EntryEditActivity.kt b/app/src/main/java/com/kunzisoft/keepass/activities/EntryEditActivity.kt index db73bf28d..8fb0cad02 100644 --- a/app/src/main/java/com/kunzisoft/keepass/activities/EntryEditActivity.kt +++ b/app/src/main/java/com/kunzisoft/keepass/activities/EntryEditActivity.kt @@ -74,6 +74,7 @@ import com.kunzisoft.keepass.tasks.AttachmentFileBinderManager import com.kunzisoft.keepass.timeout.TimeoutHelper import com.kunzisoft.keepass.utils.UriUtil import com.kunzisoft.keepass.view.* +import com.kunzisoft.keepass.viewmodels.ColorPickerViewModel import com.kunzisoft.keepass.viewmodels.EntryEditViewModel import org.joda.time.DateTime import java.util.* @@ -103,6 +104,8 @@ class EntryEditActivity : DatabaseLockActivity(), private var mEntryLoaded: Boolean = false private var mTemplatesSelectorAdapter: TemplatesSelectorAdapter? = null + private val mColorPickerViewModel: ColorPickerViewModel by viewModels() + private var mAllowCustomFields = false private var mAllowOTP = false @@ -243,6 +246,14 @@ class EntryEditActivity : DatabaseLockActivity(), IconPickerActivity.launch(this@EntryEditActivity, iconImage, mIconSelectionActivityResultLauncher) } + mEntryEditViewModel.requestColorSelection.observe(this) { color -> + ColorPickerDialogFragment().show(supportFragmentManager, "ColorPickerFragment") + } + + mColorPickerViewModel.colorPicked.observe(this) { color -> + mEntryEditViewModel.selectColor(color) + } + mEntryEditViewModel.requestDateTimeSelection.observe(this) { dateInstant -> if (dateInstant.type == DateInstant.Type.TIME) { // Launch the time picker diff --git a/app/src/main/java/com/kunzisoft/keepass/activities/dialogs/ColorPickerDialogFragment.kt b/app/src/main/java/com/kunzisoft/keepass/activities/dialogs/ColorPickerDialogFragment.kt new file mode 100644 index 000000000..e11010cbc --- /dev/null +++ b/app/src/main/java/com/kunzisoft/keepass/activities/dialogs/ColorPickerDialogFragment.kt @@ -0,0 +1,73 @@ +package com.kunzisoft.keepass.activities.dialogs + +import android.app.Dialog +import android.graphics.Color +import android.os.Bundle +import android.widget.CompoundButton +import androidx.annotation.ColorInt +import androidx.appcompat.app.AlertDialog +import androidx.fragment.app.activityViewModels +import com.kunzisoft.androidclearchroma.view.ChromaColorView +import com.kunzisoft.keepass.R +import com.kunzisoft.keepass.viewmodels.ColorPickerViewModel + +class ColorPickerDialogFragment : DatabaseDialogFragment() { + + private val mColorPickerViewModel: ColorPickerViewModel by activityViewModels() + + private lateinit var enableSwitchView: CompoundButton + private lateinit var chromaColorView: ChromaColorView + + private var mDefaultColor = Color.WHITE + + override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { + + activity?.let { activity -> + val root = activity.layoutInflater.inflate(R.layout.fragment_color_picker, null) + enableSwitchView = root.findViewById(R.id.switch_element) + chromaColorView = root.findViewById(R.id.chroma_color_view) + + var activated = false + arguments?.apply { + if (containsKey(ARG_INITIAL_COLOR)) { + mDefaultColor = getInt(ARG_INITIAL_COLOR) + } + if (containsKey(ARG_ACTIVATED)) { + activated = getBoolean(ARG_ACTIVATED) + } + } + enableSwitchView.isChecked = activated + + val builder = AlertDialog.Builder(activity) + builder.setView(root) + .setPositiveButton(android.R.string.ok) { _, _ -> + val color: Int? = if (enableSwitchView.isChecked) + chromaColorView.currentColor + else + null + mColorPickerViewModel.pickColor(color) + } + .setNegativeButton(android.R.string.cancel) { _, _ -> + // Do nothing + } + + return builder.create() + } + return super.onCreateDialog(savedInstanceState) + } + + companion object { + private const val ARG_INITIAL_COLOR = "ARG_INITIAL_COLOR" + private const val ARG_ACTIVATED = "ARG_ACTIVATED" + + fun newInstance( + @ColorInt initialColor: Int?, + ): ColorPickerDialogFragment { + val colorPickerDialogFragment = ColorPickerDialogFragment() + val args = Bundle() + args.putInt(ARG_INITIAL_COLOR, initialColor ?: Color.WHITE) + args.putBoolean(ARG_ACTIVATED, initialColor != null) + return colorPickerDialogFragment + } + } +} \ No newline at end of file diff --git a/app/src/main/java/com/kunzisoft/keepass/activities/fragments/EntryEditFragment.kt b/app/src/main/java/com/kunzisoft/keepass/activities/fragments/EntryEditFragment.kt index 1d8bea4bf..4f31ad15a 100644 --- a/app/src/main/java/com/kunzisoft/keepass/activities/fragments/EntryEditFragment.kt +++ b/app/src/main/java/com/kunzisoft/keepass/activities/fragments/EntryEditFragment.kt @@ -99,6 +99,12 @@ class EntryEditFragment: DatabaseFragment() { setOnIconClickListener { mEntryEditViewModel.requestIconSelection(templateView.getIcon()) } + setOnBackgroundColorClickListener { + mEntryEditViewModel.requestBackgroundColorSelection(templateView.getBackgroundColor()) + } + setOnForegroundColorClickListener { + mEntryEditViewModel.requestForegroundColorSelection(templateView.getForegroundColor()) + } setOnCustomEditionActionClickListener { field -> mEntryEditViewModel.requestCustomFieldEdition(field) } @@ -147,6 +153,14 @@ class EntryEditFragment: DatabaseFragment() { templateView.setIcon(iconImage) } + mEntryEditViewModel.onBackgroundColorSelected.observe(this) { color -> + templateView.setBackgroundColor(color) + } + + mEntryEditViewModel.onForegroundColorSelected.observe(this) { color -> + templateView.setForegroundColor(color) + } + mEntryEditViewModel.onPasswordSelected.observe(viewLifecycleOwner) { passwordField -> templateView.setPasswordField(passwordField) } diff --git a/app/src/main/java/com/kunzisoft/keepass/adapters/NodesAdapter.kt b/app/src/main/java/com/kunzisoft/keepass/adapters/NodesAdapter.kt index 6018b49bd..892301c39 100644 --- a/app/src/main/java/com/kunzisoft/keepass/adapters/NodesAdapter.kt +++ b/app/src/main/java/com/kunzisoft/keepass/adapters/NodesAdapter.kt @@ -21,6 +21,7 @@ package com.kunzisoft.keepass.adapters import android.content.Context import android.graphics.Color +import android.graphics.PorterDuff import android.util.TypedValue import android.view.LayoutInflater import android.view.View @@ -31,6 +32,7 @@ import android.widget.TextView import android.widget.Toast import androidx.annotation.ColorInt import androidx.core.content.ContextCompat +import androidx.core.view.isVisible import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.SortedList import androidx.recyclerview.widget.SortedListAdapterCallback @@ -39,7 +41,9 @@ import com.kunzisoft.keepass.database.element.Database import com.kunzisoft.keepass.database.element.Entry import com.kunzisoft.keepass.database.element.Group import com.kunzisoft.keepass.database.element.SortNodeEnum -import com.kunzisoft.keepass.database.element.node.* +import com.kunzisoft.keepass.database.element.node.Node +import com.kunzisoft.keepass.database.element.node.NodeVersionedInterface +import com.kunzisoft.keepass.database.element.node.Type import com.kunzisoft.keepass.database.element.template.TemplateField import com.kunzisoft.keepass.otp.OtpElement import com.kunzisoft.keepass.otp.OtpType @@ -377,6 +381,25 @@ class NodesAdapter (private val context: Context, val entry = subNode as Entry database.startManageEntry(entry) + // Assign colors + val backgroundColor = entry.backgroundColor + if (backgroundColor != null) { + holder.backgroundView?.setColorFilter(backgroundColor, PorterDuff.Mode.SRC_ATOP) + holder.backgroundView?.isVisible = true + } else { + holder.backgroundView?.isVisible = false + } + val foregroundColor = entry.foregroundColor + if (foregroundColor != null) { + holder.foregroundView?.setColorFilter(foregroundColor, PorterDuff.Mode.SRC_ATOP) + holder.icon.apply { + database.iconDrawableFactory.assignDatabaseIcon(this, subNode.icon, foregroundColor) + } + holder.foregroundView?.isVisible = true + } else { + holder.foregroundView?.isVisible = false + } + holder.text.text = entry.getVisualTitle() // Add subText with username holder.subText?.apply { @@ -510,6 +533,8 @@ class NodesAdapter (private val context: Context, var container: View = itemView.findViewById(R.id.node_container) var imageIdentifier: ImageView? = itemView.findViewById(R.id.node_image_identifier) var icon: ImageView = itemView.findViewById(R.id.node_icon) + var backgroundView: ImageView? = itemView.findViewById(R.id.background_view) + var foregroundView: ImageView? = itemView.findViewById(R.id.foreground_view) var text: TextView = itemView.findViewById(R.id.node_text) var subText: TextView? = itemView.findViewById(R.id.node_subtext) var meta: TextView = itemView.findViewById(R.id.node_meta) diff --git a/app/src/main/java/com/kunzisoft/keepass/database/element/Entry.kt b/app/src/main/java/com/kunzisoft/keepass/database/element/Entry.kt index 83e2707a0..65bcef378 100644 --- a/app/src/main/java/com/kunzisoft/keepass/database/element/Entry.kt +++ b/app/src/main/java/com/kunzisoft/keepass/database/element/Entry.kt @@ -19,8 +19,10 @@ */ package com.kunzisoft.keepass.database.element +import android.graphics.Color import android.os.Parcel import android.os.Parcelable +import com.kunzisoft.androidclearchroma.ChromaUtil import com.kunzisoft.keepass.database.element.binary.AttachmentPool import com.kunzisoft.keepass.database.element.database.DatabaseKDBX import com.kunzisoft.keepass.database.element.database.DatabaseVersioned @@ -238,6 +240,42 @@ class Entry : Node, EntryVersionedInterface { entryKDBX?.notes = value } + var backgroundColor: Int? + get() { + var colorInt: Int? = null + entryKDBX?.backgroundColor?.let { + try { + colorInt = Color.parseColor(it) + } catch (e: Exception) {} + } + return colorInt + } + set(value) { + entryKDBX?.backgroundColor = if (value == null) { + "" + } else { + ChromaUtil.getFormattedColorString(value, false) + } + } + + var foregroundColor: Int? + get() { + var colorInt: Int? = null + entryKDBX?.foregroundColor?.let { + try { + colorInt = Color.parseColor(it) + } catch (e: Exception) {} + } + return colorInt + } + set(value) { + entryKDBX?.foregroundColor = if (value == null) { + "" + } else { + ChromaUtil.getFormattedColorString(value, false) + } + } + private fun isTan(): Boolean { return title == PMS_TAN_ENTRY && username.isNotEmpty() } @@ -419,6 +457,8 @@ class Entry : Node, EntryVersionedInterface { entryInfo.expiryTime = expiryTime entryInfo.url = url entryInfo.notes = notes + entryInfo.backgroundColor = backgroundColor + entryInfo.foregroundColor = foregroundColor entryInfo.customFields = getExtraFields().toMutableList() // Add otpElement to generate token entryInfo.otpModel = getOtpElement()?.otpModel @@ -453,6 +493,8 @@ class Entry : Node, EntryVersionedInterface { expiryTime = newEntryInfo.expiryTime url = newEntryInfo.url notes = newEntryInfo.notes + backgroundColor = newEntryInfo.backgroundColor + foregroundColor = newEntryInfo.foregroundColor addExtraFields(newEntryInfo.customFields) database?.attachmentPool?.let { binaryPool -> newEntryInfo.attachments.forEach { attachment -> diff --git a/app/src/main/java/com/kunzisoft/keepass/model/EntryInfo.kt b/app/src/main/java/com/kunzisoft/keepass/model/EntryInfo.kt index 9324ea9a8..5ae5fc0fc 100644 --- a/app/src/main/java/com/kunzisoft/keepass/model/EntryInfo.kt +++ b/app/src/main/java/com/kunzisoft/keepass/model/EntryInfo.kt @@ -40,6 +40,8 @@ class EntryInfo : NodeInfo { var password: String = "" var url: String = "" var notes: String = "" + var backgroundColor: Int? = null + var foregroundColor: Int? = null var customFields: MutableList = mutableListOf() var attachments: MutableList = mutableListOf() var otpModel: OtpModel? = null @@ -53,6 +55,10 @@ class EntryInfo : NodeInfo { password = parcel.readString() ?: password url = parcel.readString() ?: url notes = parcel.readString() ?: notes + val readBgColor = parcel.readInt() + backgroundColor = if (readBgColor == -1) null else readBgColor + val readFgColor = parcel.readInt() + foregroundColor = if (readFgColor == -1) null else readFgColor parcel.readList(customFields, Field::class.java.classLoader) parcel.readList(attachments, Attachment::class.java.classLoader) otpModel = parcel.readParcelable(OtpModel::class.java.classLoader) ?: otpModel @@ -70,6 +76,8 @@ class EntryInfo : NodeInfo { parcel.writeString(password) parcel.writeString(url) parcel.writeString(notes) + parcel.writeInt(backgroundColor ?: -1) + parcel.writeInt(foregroundColor ?: -1) parcel.writeList(customFields) parcel.writeList(attachments) parcel.writeParcelable(otpModel, flags) @@ -196,6 +204,8 @@ class EntryInfo : NodeInfo { if (password != other.password) return false if (url != other.url) return false if (notes != other.notes) return false + if (backgroundColor != other.backgroundColor) return false + if (foregroundColor != other.foregroundColor) return false if (customFields != other.customFields) return false if (attachments != other.attachments) return false if (otpModel != other.otpModel) return false @@ -211,6 +221,8 @@ class EntryInfo : NodeInfo { result = 31 * result + password.hashCode() result = 31 * result + url.hashCode() result = 31 * result + notes.hashCode() + result = 31 * result + backgroundColor.hashCode() + result = 31 * result + foregroundColor.hashCode() result = 31 * result + customFields.hashCode() result = 31 * result + attachments.hashCode() result = 31 * result + (otpModel?.hashCode() ?: 0) diff --git a/app/src/main/java/com/kunzisoft/keepass/view/TemplateAbstractView.kt b/app/src/main/java/com/kunzisoft/keepass/view/TemplateAbstractView.kt index 5a8697daa..c8271d7d3 100644 --- a/app/src/main/java/com/kunzisoft/keepass/view/TemplateAbstractView.kt +++ b/app/src/main/java/com/kunzisoft/keepass/view/TemplateAbstractView.kt @@ -46,6 +46,8 @@ abstract class TemplateAbstractView< protected var headerContainerView: ViewGroup protected var entryIconView: ImageView + protected var backgroundColorButton: ImageView + protected var foregroundColorButton: ImageView private var titleContainerView: ViewGroup protected var templateContainerView: ViewGroup private var customFieldsContainerView: SectionView @@ -57,6 +59,8 @@ abstract class TemplateAbstractView< headerContainerView = findViewById(R.id.template_header_container) entryIconView = findViewById(R.id.template_icon_button) + backgroundColorButton = findViewById(R.id.template_background_color_button) + foregroundColorButton = findViewById(R.id.template_foreground_color_button) titleContainerView = findViewById(R.id.template_title_container) templateContainerView = findViewById(R.id.template_fields_container) // To fix card view margin below Marshmallow diff --git a/app/src/main/java/com/kunzisoft/keepass/view/TemplateEditView.kt b/app/src/main/java/com/kunzisoft/keepass/view/TemplateEditView.kt index d3918be16..e2a32b0fd 100644 --- a/app/src/main/java/com/kunzisoft/keepass/view/TemplateEditView.kt +++ b/app/src/main/java/com/kunzisoft/keepass/view/TemplateEditView.kt @@ -54,6 +54,30 @@ class TemplateEditView @JvmOverloads constructor(context: Context, populateIconMethod?.invoke(entryIconView, iconImage) } + fun setOnBackgroundColorClickListener(onClickListener: OnClickListener) { + backgroundColorButton.setOnClickListener(onClickListener) + } + + fun getBackgroundColor(): Int? { + return mEntryInfo?.backgroundColor + } + + fun setBackgroundColor(color: Int?) { + mEntryInfo?.backgroundColor = color + } + + fun setOnForegroundColorClickListener(onClickListener: OnClickListener) { + foregroundColorButton.setOnClickListener(onClickListener) + } + + fun getForegroundColor(): Int? { + return mEntryInfo?.foregroundColor + } + + fun setForegroundColor(color: Int?) { + mEntryInfo?.foregroundColor = color + } + override fun preProcessTemplate() { headerContainerView.isVisible = true } diff --git a/app/src/main/java/com/kunzisoft/keepass/viewmodels/ColorPickerViewModel.kt b/app/src/main/java/com/kunzisoft/keepass/viewmodels/ColorPickerViewModel.kt new file mode 100644 index 000000000..282e3f425 --- /dev/null +++ b/app/src/main/java/com/kunzisoft/keepass/viewmodels/ColorPickerViewModel.kt @@ -0,0 +1,15 @@ +package com.kunzisoft.keepass.viewmodels + +import androidx.lifecycle.LiveData +import androidx.lifecycle.MutableLiveData +import androidx.lifecycle.ViewModel + +class ColorPickerViewModel: ViewModel() { + + val colorPicked : LiveData get() = _colorPicked + private val _colorPicked = MutableLiveData() + + fun pickColor(color: Int?) { + _colorPicked.value = color + } +} \ No newline at end of file diff --git a/app/src/main/java/com/kunzisoft/keepass/viewmodels/NodeEditViewModel.kt b/app/src/main/java/com/kunzisoft/keepass/viewmodels/NodeEditViewModel.kt index 80d9ea79a..4c55af246 100644 --- a/app/src/main/java/com/kunzisoft/keepass/viewmodels/NodeEditViewModel.kt +++ b/app/src/main/java/com/kunzisoft/keepass/viewmodels/NodeEditViewModel.kt @@ -14,6 +14,14 @@ abstract class NodeEditViewModel : ViewModel() { val onIconSelected : LiveData get() = _onIconSelected private val _onIconSelected = SingleLiveEvent() + private var mColorRequest: ColorRequest = ColorRequest.BACKGROUND + val requestColorSelection : LiveData get() = _requestColorSelection + private val _requestColorSelection = SingleLiveEvent() + val onBackgroundColorSelected : LiveData get() = _onBackgroundColorSelected + private val _onBackgroundColorSelected = SingleLiveEvent() + val onForegroundColorSelected : LiveData get() = _onForegroundColorSelected + private val _onForegroundColorSelected = SingleLiveEvent() + val requestDateTimeSelection : LiveData get() = _requestDateTimeSelection private val _requestDateTimeSelection = SingleLiveEvent() val onDateSelected : LiveData get() = _onDateSelected @@ -29,6 +37,23 @@ abstract class NodeEditViewModel : ViewModel() { _onIconSelected.value = iconImage } + fun requestBackgroundColorSelection(initialColor: Int?) { + mColorRequest = ColorRequest.BACKGROUND + _requestColorSelection.value = initialColor + } + + fun requestForegroundColorSelection(initialColor: Int?) { + mColorRequest = ColorRequest.FOREGROUND + _requestColorSelection.value = initialColor + } + + fun selectColor(color: Int?) { + when (mColorRequest) { + ColorRequest.BACKGROUND -> _onBackgroundColorSelected.value = color + ColorRequest.FOREGROUND -> _onForegroundColorSelected.value = color + } + } + fun requestDateTimeSelection(dateInstant: DateInstant) { _requestDateTimeSelection.value = dateInstant } @@ -40,4 +65,8 @@ abstract class NodeEditViewModel : ViewModel() { fun selectTime(hours: Int, minutes: Int) { _onTimeSelected.value = DataTime(hours, minutes) } + + private enum class ColorRequest { + BACKGROUND, FOREGROUND + } } \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_color_background_white_24dp.xml b/app/src/main/res/drawable/ic_color_background_white_24dp.xml new file mode 100644 index 000000000..ea4549a72 --- /dev/null +++ b/app/src/main/res/drawable/ic_color_background_white_24dp.xml @@ -0,0 +1,13 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_color_foreground_white_24dp.xml b/app/src/main/res/drawable/ic_color_foreground_white_24dp.xml new file mode 100644 index 000000000..558ef76ec --- /dev/null +++ b/app/src/main/res/drawable/ic_color_foreground_white_24dp.xml @@ -0,0 +1,13 @@ + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_group.xml b/app/src/main/res/layout/activity_group.xml index 7d0372d8d..3edfdcbd6 100644 --- a/app/src/main/res/layout/activity_group.xml +++ b/app/src/main/res/layout/activity_group.xml @@ -51,7 +51,7 @@ android:src="@drawable/background_rounded_square" android:layout_marginRight="6dp" android:layout_marginEnd="6dp" - android:contentDescription="@string/database"/> + android:contentDescription="@string/content_description_database_color"/> + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/item_list_nodes_entry.xml b/app/src/main/res/layout/item_list_nodes_entry.xml index 377d7e2aa..e32c28440 100644 --- a/app/src/main/res/layout/item_list_nodes_entry.xml +++ b/app/src/main/res/layout/item_list_nodes_entry.xml @@ -38,6 +38,20 @@ app:layout_constraintEnd_toEndOf="parent" android:background="?android:attr/selectableItemBackground" > + + - - + android:contentDescription="@string/content_description_entry_foreground_color" + android:src="@drawable/background_rounded_square" /> . --> - - + + android:layout_height="wrap_content"> + - - + + + + + + - - - - + android:layout_gravity="end"> + + + + Keyfile checkbox Repeat toggle password visibility Entry icon + Database color + Entry foreground color + Entry background color Validate Discard changes? Discard diff --git a/art/ic_color_background.svg b/art/ic_color_background.svg new file mode 100644 index 000000000..76548ca12 --- /dev/null +++ b/art/ic_color_background.svg @@ -0,0 +1,97 @@ + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + diff --git a/art/ic_color_foreground.svg b/art/ic_color_foreground.svg new file mode 100644 index 000000000..db19395c5 --- /dev/null +++ b/art/ic_color_foreground.svg @@ -0,0 +1,97 @@ + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + +