Merge pull request #5476 from 21pages/ab

change tag color
This commit is contained in:
RustDesk 2023-08-22 19:51:46 +08:00 committed by GitHub
commit 9b542f7653
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
40 changed files with 152 additions and 50 deletions

View File

@ -1077,7 +1077,7 @@ Color str2color(String str, [alpha = 0xFF]) {
return Color((hash & 0xFF7FFF) | (alpha << 24)); return Color((hash & 0xFF7FFF) | (alpha << 24));
} }
Color str2color2(String str, [alpha = 0xFF]) { Color str2color2(String str, {List<int> existing = const []}) {
Map<String, Color> colorMap = { Map<String, Color> colorMap = {
"red": Colors.red, "red": Colors.red,
"green": Colors.green, "green": Colors.green,
@ -1094,10 +1094,10 @@ Color str2color2(String str, [alpha = 0xFF]) {
}; };
final color = colorMap[str.toLowerCase()]; final color = colorMap[str.toLowerCase()];
if (color != null) { if (color != null) {
return color.withAlpha(alpha); return color.withAlpha(0xFF);
} }
if (str.toLowerCase() == 'yellow') { if (str.toLowerCase() == 'yellow') {
return Colors.yellow.withAlpha(alpha); return Colors.yellow.withAlpha(0xFF);
} }
var hash = 0; var hash = 0;
for (var i = 0; i < str.length; i++) { for (var i = 0; i < str.length; i++) {
@ -1105,7 +1105,15 @@ Color str2color2(String str, [alpha = 0xFF]) {
} }
List<Color> colorList = colorMap.values.toList(); List<Color> colorList = colorMap.values.toList();
hash = hash % colorList.length; hash = hash % colorList.length;
return colorList[hash].withAlpha(alpha); var result = colorList[hash].withAlpha(0xFF);
if (existing.contains(result.value)) {
Color? notUsed =
colorList.firstWhereOrNull((e) => !existing.contains(e.value));
if (notUsed != null) {
result = notUsed;
}
}
return result;
} }
const K = 1024; const K = 1024;

View File

@ -7,6 +7,7 @@ import 'package:flutter_hbb/models/ab_model.dart';
import 'package:flutter_hbb/models/platform_model.dart'; import 'package:flutter_hbb/models/platform_model.dart';
import '../../desktop/widgets/material_mod_popup_menu.dart' as mod_menu; import '../../desktop/widgets/material_mod_popup_menu.dart' as mod_menu;
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:flex_color_picker/flex_color_picker.dart';
import '../../common.dart'; import '../../common.dart';
import 'dialog.dart'; import 'dialog.dart';
@ -513,7 +514,7 @@ class AddressBookTag extends StatelessWidget {
child: Obx(() => Container( child: Obx(() => Container(
decoration: BoxDecoration( decoration: BoxDecoration(
color: tags.contains(name) color: tags.contains(name)
? str2color2(name, 0xFF) ? gFFI.abModel.getTagColor(name)
: Theme.of(context).colorScheme.background, : Theme.of(context).colorScheme.background,
borderRadius: BorderRadius.circular(4)), borderRadius: BorderRadius.circular(4)),
margin: const EdgeInsets.symmetric(horizontal: 4.0, vertical: 4.0), margin: const EdgeInsets.symmetric(horizontal: 4.0, vertical: 4.0),
@ -528,7 +529,7 @@ class AddressBookTag extends StatelessWidget {
shape: BoxShape.circle, shape: BoxShape.circle,
color: tags.contains(name) color: tags.contains(name)
? Colors.white ? Colors.white
: str2color2(name)), : gFFI.abModel.getTagColor(name)),
).marginOnly(right: radius / 2), ).marginOnly(right: radius / 2),
Expanded( Expanded(
child: Text(name, child: Text(name,
@ -568,6 +569,23 @@ class AddressBookTag extends StatelessWidget {
Future.delayed(Duration.zero, () => Get.back()); Future.delayed(Duration.zero, () => Get.back());
}); });
}), }),
getEntry(translate(translate('Change Color')), () async {
final model = gFFI.abModel;
Color oldColor = model.getTagColor(name);
Color newColor = await showColorPickerDialog(
context,
oldColor,
pickersEnabled: {
ColorPickerType.accent: false,
ColorPickerType.wheel: true,
},
showColorCode: true,
);
if (oldColor != newColor) {
model.setTagColor(name, newColor);
model.pushAb();
}
}),
getEntry(translate("Delete"), () { getEntry(translate("Delete"), () {
gFFI.abModel.deleteTag(name); gFFI.abModel.deleteTag(name);
gFFI.abModel.pushAb(); gFFI.abModel.pushAb();

View File

@ -201,7 +201,8 @@ class _PeerCardState extends State<_PeerCard>
) )
], ],
); );
final colors = _frontN(peer.tags, 25).map((e) => str2color2(e)).toList(); final colors =
_frontN(peer.tags, 25).map((e) => gFFI.abModel.getTagColor(e)).toList();
return Tooltip( return Tooltip(
message: isMobile message: isMobile
? '' ? ''
@ -311,7 +312,8 @@ class _PeerCardState extends State<_PeerCard>
), ),
); );
final colors = _frontN(peer.tags, 25).map((e) => str2color2(e)).toList(); final colors =
_frontN(peer.tags, 25).map((e) => gFFI.abModel.getTagColor(e)).toList();
return Tooltip( return Tooltip(
message: peer.tags.isNotEmpty message: peer.tags.isNotEmpty
? '${translate('Tags')}: ${peer.tags.join(', ')}' ? '${translate('Tags')}: ${peer.tags.join(', ')}'

View File

@ -28,6 +28,7 @@ class AbModel {
final pullError = "".obs; final pullError = "".obs;
final pushError = "".obs; final pushError = "".obs;
final tags = [].obs; final tags = [].obs;
final RxMap<String, int> tagColors = Map<String, int>.fromEntries([]).obs;
final peers = List<Peer>.empty(growable: true).obs; final peers = List<Peer>.empty(growable: true).obs;
final sortTags = shouldSortTags().obs; final sortTags = shouldSortTags().obs;
final retrying = false.obs; final retrying = false.obs;
@ -80,10 +81,11 @@ class AbModel {
if (resp.body.toLowerCase() == "null") { if (resp.body.toLowerCase() == "null") {
// normal reply, emtpy ab return null // normal reply, emtpy ab return null
tags.clear(); tags.clear();
tagColors.clear();
peers.clear(); peers.clear();
} else if (resp.body.isNotEmpty) { } else if (resp.body.isNotEmpty) {
Map<String, dynamic> json = Map<String, dynamic> json =
_jsonDecode(utf8.decode(resp.bodyBytes), resp.statusCode); _jsonDecodeResp(utf8.decode(resp.bodyBytes), resp.statusCode);
if (json.containsKey('error')) { if (json.containsKey('error')) {
throw json['error']; throw json['error'];
} else if (json.containsKey('data')) { } else if (json.containsKey('data')) {
@ -93,26 +95,7 @@ class AbModel {
} catch (e) {} } catch (e) {}
final data = jsonDecode(json['data']); final data = jsonDecode(json['data']);
if (data != null) { if (data != null) {
final oldOnlineIDs = _deserialize(data);
peers.where((e) => e.online).map((e) => e.id).toList();
tags.clear();
peers.clear();
if (data['tags'] is List) {
tags.value = data['tags'];
}
if (data['peers'] is List) {
for (final peer in data['peers']) {
peers.add(Peer.fromJson(peer));
}
}
if (isFull(false)) {
peers.removeRange(licensedDevices, peers.length);
}
// restore online
peers
.where((e) => oldOnlineIDs.contains(e.id))
.map((e) => e.online = true)
.toList();
_saveCache(); // save on success _saveCache(); // save on success
} }
} }
@ -242,10 +225,7 @@ class AbModel {
final api = "${await bind.mainGetApiServer()}/api/ab"; final api = "${await bind.mainGetApiServer()}/api/ab";
var authHeaders = getHttpHeaders(); var authHeaders = getHttpHeaders();
authHeaders['Content-Type'] = "application/json"; authHeaders['Content-Type'] = "application/json";
final peersJsonData = peers.map((e) => e.toAbUploadJson()).toList(); final body = jsonEncode(_serialize());
final body = jsonEncode({
"data": jsonEncode({"tags": tags, "peers": peersJsonData})
});
http.Response resp; http.Response resp;
// support compression // support compression
if (licensedDevices > 0 && body.length > 1024) { if (licensedDevices > 0 && body.length > 1024) {
@ -261,7 +241,7 @@ class AbModel {
ret = true; ret = true;
_saveCache(); _saveCache();
} else { } else {
Map<String, dynamic> json = _jsonDecode(resp.body, resp.statusCode); Map<String, dynamic> json = _jsonDecodeResp(resp.body, resp.statusCode);
if (json.containsKey('error')) { if (json.containsKey('error')) {
throw json['error']; throw json['error'];
} else if (resp.statusCode == 200) { } else if (resp.statusCode == 200) {
@ -318,6 +298,7 @@ class AbModel {
void deleteTag(String tag) { void deleteTag(String tag) {
gFFI.abModel.selectedTags.remove(tag); gFFI.abModel.selectedTags.remove(tag);
tags.removeWhere((element) => element == tag); tags.removeWhere((element) => element == tag);
tagColors.remove(tag);
for (var peer in peers) { for (var peer in peers) {
if (peer.tags.isEmpty) { if (peer.tags.isEmpty) {
continue; continue;
@ -353,6 +334,11 @@ class AbModel {
} }
}).toList(); }).toList();
} }
int? oldColor = tagColors[oldTag];
if (oldColor != null) {
tagColors.remove(oldTag);
tagColors.addAll({newTag: oldColor});
}
} }
void unsetSelectedTags() { void unsetSelectedTags() {
@ -368,6 +354,20 @@ class AbModel {
} }
} }
Color getTagColor(String tag) {
int? colorValue = tagColors[tag];
if (colorValue != null) {
return Color(colorValue);
}
return str2color2(tag, existing: tagColors.values.toList());
}
setTagColor(String tag, Color color) {
if (tags.contains(tag)) {
tagColors[tag] = color.value;
}
}
void merge(Peer r, Peer p) { void merge(Peer r, Peer p) {
p.hash = r.hash.isEmpty ? p.hash : r.hash; p.hash = r.hash.isEmpty ? p.hash : r.hash;
p.username = r.username.isEmpty ? p.username : r.username; p.username = r.username.isEmpty ? p.username : r.username;
@ -467,12 +467,10 @@ class AbModel {
_saveCache() { _saveCache() {
try { try {
final peersJsonData = peers.map((e) => e.toAbUploadJson()).toList(); var m = _serialize();
final m = <String, dynamic>{ m.addAll(<String, dynamic>{
"access_token": bind.mainGetLocalOption(key: 'access_token'), "access_token": bind.mainGetLocalOption(key: 'access_token'),
"peers": peersJsonData, });
"tags": tags.map((e) => e.toString()).toList(),
};
bind.mainSaveAb(json: jsonEncode(m)); bind.mainSaveAb(json: jsonEncode(m));
} catch (e) { } catch (e) {
debugPrint('ab save:$e'); debugPrint('ab save:$e');
@ -488,22 +486,13 @@ class AbModel {
final cache = await bind.mainLoadAb(); final cache = await bind.mainLoadAb();
final data = jsonDecode(cache); final data = jsonDecode(cache);
if (data == null || data['access_token'] != access_token) return; if (data == null || data['access_token'] != access_token) return;
tags.clear(); _deserialize(data);
peers.clear();
if (data['tags'] is List) {
tags.value = data['tags'];
}
if (data['peers'] is List) {
for (final peer in data['peers']) {
peers.add(Peer.fromJson(peer));
}
}
} catch (e) { } catch (e) {
debugPrint("load ab cache: $e"); debugPrint("load ab cache: $e");
} }
} }
Map<String, dynamic> _jsonDecode(String body, int statusCode) { Map<String, dynamic> _jsonDecodeResp(String body, int statusCode) {
try { try {
Map<String, dynamic> json = jsonDecode(body); Map<String, dynamic> json = jsonDecode(body);
return json; return json;
@ -516,6 +505,50 @@ class AbModel {
} }
} }
Map<String, dynamic> _serialize() {
final peersJsonData = peers.map((e) => e.toAbUploadJson()).toList();
final tagColorJsonData = jsonEncode(tagColors);
return {
"tags": tags,
"peers": peersJsonData,
"tag_colors": tagColorJsonData
};
}
_deserialize(dynamic data) {
if (data == null) return;
final oldOnlineIDs = peers.where((e) => e.online).map((e) => e.id).toList();
tags.clear();
tagColors.clear();
peers.clear();
if (data['tags'] is List) {
tags.value = data['tags'];
}
if (data['peers'] is List) {
for (final peer in data['peers']) {
peers.add(Peer.fromJson(peer));
}
}
if (isFull(false)) {
peers.removeRange(licensedDevices, peers.length);
}
// restore online
peers
.where((e) => oldOnlineIDs.contains(e.id))
.map((e) => e.online = true)
.toList();
if (data['tag_colors'] is String) {
Map<String, dynamic> map = jsonDecode(data['tag_colors']);
tagColors.value = Map<String, int>.from(map);
}
// add color to tag
final tagsWithoutColor =
tags.toList().where((e) => !tagColors.containsKey(e)).toList();
for (var t in tagsWithoutColor) {
tagColors[t] = str2color2(t, existing: tagColors.values.toList()).value;
}
}
reSyncToast(Future<bool> future) { reSyncToast(Future<bool> future) {
if (!shouldSyncAb()) return; if (!shouldSyncAb()) return;
Future.delayed(Duration.zero, () async { Future.delayed(Duration.zero, () async {

View File

@ -97,6 +97,7 @@ dependencies:
dropdown_button2: ^2.0.0 dropdown_button2: ^2.0.0
uuid: ^3.0.7 uuid: ^3.0.7
auto_size_text_field: ^2.2.1 auto_size_text_field: ^2.2.1
flex_color_picker: ^3.3.0
dev_dependencies: dev_dependencies:
icons_launcher: ^2.0.4 icons_launcher: ^2.0.4

View File

@ -1525,6 +1525,12 @@ pub struct Ab {
pub peers: Vec<AbPeer>, pub peers: Vec<AbPeer>,
#[serde(default, deserialize_with = "deserialize_vec_string")] #[serde(default, deserialize_with = "deserialize_vec_string")]
pub tags: Vec<String>, pub tags: Vec<String>,
#[serde(
default,
deserialize_with = "deserialize_string",
skip_serializing_if = "String::is_empty"
)]
pub tag_colors: String,
} }
impl Ab { impl Ab {

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "未成功获取地址簿"), ("pull_ab_failed_tip", "未成功获取地址簿"),
("push_ab_failed_tip", "未成功上传地址簿"), ("push_ab_failed_tip", "未成功上传地址簿"),
("synced_peer_readded_tip", "最近会话中存在的设备将会被重新同步到地址簿。"), ("synced_peer_readded_tip", "最近会话中存在的设备将会被重新同步到地址簿。"),
("Change Color", "更改颜色"),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "Nepodařilo se obnovit adresář"), ("pull_ab_failed_tip", "Nepodařilo se obnovit adresář"),
("push_ab_failed_tip", "Nepodařilo se synchronizovat adresář se serverem"), ("push_ab_failed_tip", "Nepodařilo se synchronizovat adresář se serverem"),
("synced_peer_readded_tip", "Zařízení, která byla přítomna v posledních relacích, budou synchronizována zpět do adresáře."), ("synced_peer_readded_tip", "Zařízení, která byla přítomna v posledních relacích, budou synchronizována zpět do adresáře."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "Aktualisierung des Adressbuchs fehlgeschlagen"), ("pull_ab_failed_tip", "Aktualisierung des Adressbuchs fehlgeschlagen"),
("push_ab_failed_tip", "Synchronisierung des Adressbuchs mit dem Server fehlgeschlagen"), ("push_ab_failed_tip", "Synchronisierung des Adressbuchs mit dem Server fehlgeschlagen"),
("synced_peer_readded_tip", "Die Geräte, die in den letzten Sitzungen vorhanden waren, werden erneut zum Adressbuch hinzugefügt."), ("synced_peer_readded_tip", "Die Geräte, die in den letzten Sitzungen vorhanden waren, werden erneut zum Adressbuch hinzugefügt."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "No se ha podido refrescar el directorio"), ("pull_ab_failed_tip", "No se ha podido refrescar el directorio"),
("push_ab_failed_tip", "No se ha podido sincronizar el directorio con el servidor"), ("push_ab_failed_tip", "No se ha podido sincronizar el directorio con el servidor"),
("synced_peer_readded_tip", "Los dispositivos presentes en sesiones recientes se sincronizarán con el directorio."), ("synced_peer_readded_tip", "Los dispositivos presentes en sesiones recientes se sincronizarán con el directorio."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "Impossible d'actualiser le carnet d'adresses"), ("pull_ab_failed_tip", "Impossible d'actualiser le carnet d'adresses"),
("push_ab_failed_tip", "Échec de la synchronisation du carnet d'adresses"), ("push_ab_failed_tip", "Échec de la synchronisation du carnet d'adresses"),
("synced_peer_readded_tip", "Les appareils qui étaient présents dans les sessions récentes seront synchronisés avec le carnet d'adresses."), ("synced_peer_readded_tip", "Les appareils qui étaient présents dans les sessions récentes seront synchronisés avec le carnet d'adresses."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "Gagal memuat ulang buku alamat"), ("pull_ab_failed_tip", "Gagal memuat ulang buku alamat"),
("push_ab_failed_tip", "Gagal menyinkronkan buku alamat ke server"), ("push_ab_failed_tip", "Gagal menyinkronkan buku alamat ke server"),
("synced_peer_readded_tip", "Perangkat yang terdaftar dalam sesi-sesi terbaru akan di-sinkronkan kembali ke buku alamat."), ("synced_peer_readded_tip", "Perangkat yang terdaftar dalam sesi-sesi terbaru akan di-sinkronkan kembali ke buku alamat."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "Impossibile aggiornare la rubrica"), ("pull_ab_failed_tip", "Impossibile aggiornare la rubrica"),
("push_ab_failed_tip", "Impossibile sincronizzare la rubrica con il server"), ("push_ab_failed_tip", "Impossibile sincronizzare la rubrica con il server"),
("synced_peer_readded_tip", "I dispositivi presenti nelle sessioni recenti saranno sincronizzati di nuovo nella rubrica."), ("synced_peer_readded_tip", "I dispositivi presenti nelle sessioni recenti saranno sincronizzati di nuovo nella rubrica."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "Невозможно обновить адресную книгу"), ("pull_ab_failed_tip", "Невозможно обновить адресную книгу"),
("push_ab_failed_tip", "Невозможно синхронизировать адресную книгу с сервером"), ("push_ab_failed_tip", "Невозможно синхронизировать адресную книгу с сервером"),
("synced_peer_readded_tip", "Устройства, присутствовавшие в последних сеансах, будут синхронизированы с адресной книгой."), ("synced_peer_readded_tip", "Устройства, присутствовавшие в последних сеансах, будут синхронизированы с адресной книгой."),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", "未成功獲取地址簿"), ("pull_ab_failed_tip", "未成功獲取地址簿"),
("push_ab_failed_tip", "未成功上傳地址簿"), ("push_ab_failed_tip", "未成功上傳地址簿"),
("synced_peer_readded_tip", "最近會話中存在的設備將會被重新同步到地址簿。"), ("synced_peer_readded_tip", "最近會話中存在的設備將會被重新同步到地址簿。"),
("Change Color", "更改顏色"),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }

View File

@ -538,5 +538,6 @@ pub static ref T: std::collections::HashMap<&'static str, &'static str> =
("pull_ab_failed_tip", ""), ("pull_ab_failed_tip", ""),
("push_ab_failed_tip", ""), ("push_ab_failed_tip", ""),
("synced_peer_readded_tip", ""), ("synced_peer_readded_tip", ""),
("Change Color", ""),
].iter().cloned().collect(); ].iter().cloned().collect();
} }