commit
9b542f7653
@ -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;
|
||||||
|
@ -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();
|
||||||
|
@ -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(', ')}'
|
||||||
|
@ -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 {
|
||||||
|
@ -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
|
||||||
|
@ -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 {
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user