1
0
mirror of https://github.com/nerzhul/ownCloud-SMS-App.git synced 2025-06-10 01:16:15 +00:00

Compare commits

...

666 Commits

Author SHA1 Message Date
Nextcloud bot
bbb6f939ba
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-05-18 03:09:05 +00:00
Nextcloud bot
edbf7d74e9
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-05-04 03:09:07 +00:00
Nextcloud bot
a93d1ce4bc
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-05-02 03:09:49 +00:00
Nextcloud bot
733727c478
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-05-01 03:10:02 +00:00
Nextcloud bot
a2b4715f5c
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-04-30 03:10:07 +00:00
Nextcloud bot
8a6d42477e
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-04-29 03:10:15 +00:00
Nextcloud bot
7cd8d5f66f
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-04-25 03:08:27 +00:00
Nextcloud bot
57302f6689
fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-04-19 03:10:01 +00:00
Nextcloud bot
d34c0474b8
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-04-04 03:10:58 +00:00
Nextcloud bot
2144f32b65
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-03-18 03:19:33 +00:00
Nextcloud bot
a83ffff1e0
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-03-17 03:09:25 +00:00
Nextcloud bot
3c37a8a22f
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-26 03:09:30 +00:00
Nextcloud bot
46f56742cd
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-25 03:18:13 +00:00
Nextcloud bot
33a1b5542e
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-24 03:04:37 +00:00
Nextcloud bot
8a0c8bb569
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-20 03:06:57 +00:00
Nextcloud bot
7b7e55853d
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-16 03:34:10 +00:00
Nextcloud bot
7c41ec2e42
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-13 03:07:17 +00:00
Nextcloud bot
e5f96a7a9e
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-09 03:08:10 +00:00
Nextcloud bot
e773058e6a
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-02-06 03:05:55 +00:00
Nextcloud bot
44773ffc75
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-01-29 03:02:25 +00:00
Nextcloud bot
edfbc5a77a
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-01-24 03:04:07 +00:00
Dmitriy Bogdanov
7d677a9fe2
Fix notifications not being shown on modern Androids (#221)
* Add notification channels support

* Update Travis CI config
2025-01-23 18:47:17 +01:00
Nextcloud bot
5ea03be8d6
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2025-01-21 03:05:12 +00:00
Nextcloud bot
4c7dd797f2
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-12-28 03:05:42 +00:00
Nextcloud bot
cd45e0ba95
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-11-18 03:01:21 +00:00
Nextcloud bot
679e39c019
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-11-11 02:59:30 +00:00
Nextcloud bot
5a975b2843
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-11-04 03:02:00 +00:00
Nextcloud bot
cdcb26b2de
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-24 03:21:30 +00:00
Nextcloud bot
3c889b8727
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-22 02:58:07 +00:00
Nextcloud bot
0fcd5e1b90
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-19 03:07:40 +00:00
Nextcloud bot
874444e8eb
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-18 03:02:38 +00:00
Nextcloud bot
b9a78a0262
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-13 02:58:53 +00:00
Nextcloud bot
a8479b59bf
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-12 02:52:12 +00:00
Nextcloud bot
fa56c85969
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-11 03:02:02 +00:00
Nextcloud bot
fd592fb39e
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-10-04 02:54:14 +00:00
Nextcloud bot
ba24fa7fed
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-09-18 02:55:21 +00:00
Nextcloud bot
d4cd898a05
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-09-11 03:00:37 +00:00
Nextcloud bot
cda7c306c8
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-09-02 02:57:18 +00:00
Nextcloud bot
989946263d
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-08-28 02:57:27 +00:00
Nextcloud bot
3233007191
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-08-02 02:46:55 +00:00
Nextcloud bot
6364bccfba
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-07-15 02:44:58 +00:00
Nextcloud bot
130b45a901
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-07-14 02:46:39 +00:00
Nextcloud bot
90358e07f7
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-06-02 02:44:16 +00:00
Nextcloud bot
2b18901a97
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-05-25 02:44:31 +00:00
Nextcloud bot
3834633d78
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-05-21 02:40:18 +00:00
Nextcloud bot
74b74d1265
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-05-17 02:39:43 +00:00
Nextcloud bot
18982eed14
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-04-25 02:48:04 +00:00
Nextcloud bot
8f7ce34a02
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-04-19 02:49:10 +00:00
Nextcloud bot
8f6cec4501
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-04-12 02:45:38 +00:00
Nextcloud bot
b5f144cb16
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-04-05 02:46:03 +00:00
Nextcloud bot
802fc1b68d
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-03-30 02:55:35 +00:00
Nextcloud bot
8765a0bbbf
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-03-24 02:48:37 +00:00
Nextcloud bot
e60e1d5805
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-02-25 02:46:18 +00:00
Nextcloud bot
904c9fe870
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-01-13 02:49:54 +00:00
Nextcloud bot
eb58db193d
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-01-07 02:47:48 +00:00
Nextcloud bot
cc49a6e828
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2024-01-04 02:45:46 +00:00
Nextcloud bot
c5cd46cdef
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-12-23 02:47:16 +00:00
Nextcloud bot
4da04c6cb3
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-11-29 02:47:21 +00:00
Nextcloud bot
f16702b7c2
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-11-15 02:55:26 +00:00
Nextcloud bot
c2a2afaa88
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-11-11 02:47:28 +00:00
Nextcloud bot
02967d3640
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-10-31 02:53:51 +00:00
Nextcloud bot
9be7af764d
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-10-06 02:51:33 +00:00
Nextcloud bot
14063ef66a
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-08-09 03:11:21 +00:00
Nextcloud bot
769425ab94
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-08-05 02:49:16 +00:00
Nextcloud bot
8566985222
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-08-03 02:48:09 +00:00
Nextcloud bot
8e91e061ef
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-30 02:41:06 +00:00
Nextcloud bot
fb19382fd2
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-26 03:17:01 +00:00
Nextcloud bot
68ea2f80a3
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-20 02:42:42 +00:00
Nextcloud bot
8d2a1bab74
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-19 02:42:26 +00:00
Nextcloud bot
a5a9c7967d
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-17 02:38:54 +00:00
Nextcloud bot
2bae3bcc93
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-13 02:47:05 +00:00
Nextcloud bot
f25b6ea1e3
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-07-04 02:42:24 +00:00
Nextcloud bot
c652d1a6e6
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-25 02:45:10 +00:00
Nextcloud bot
2450a74c32
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-20 02:42:07 +00:00
Nextcloud bot
10cbe87f51
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-19 02:36:07 +00:00
Nextcloud bot
c0a6f565c7
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-18 02:29:45 +00:00
Nextcloud bot
eac20f4094
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-17 02:30:57 +00:00
Nextcloud bot
873678d1ec
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-11 02:49:49 +00:00
Nextcloud bot
7a446f3243
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-06-04 02:27:47 +00:00
Nextcloud bot
c7605ede18
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-05-30 02:31:29 +00:00
Nextcloud bot
1a1a88bd44
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-05-21 02:30:43 +00:00
Nextcloud bot
010db03f37
Fix(l10n): Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-05-12 04:44:33 +00:00
Nextcloud bot
24f3bba827
Fix(l10n): 🔠 Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-03-31 04:49:51 +00:00
Nextcloud bot
fbfb72acdf
Fix(l10n): 🔠 Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-03-16 04:39:30 +00:00
Nextcloud bot
237c695922
Fix(l10n): 🔠 Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-01-05 03:40:22 +00:00
Nextcloud bot
22218bb556
Fix(l10n): 🔠 Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2023-01-04 06:17:04 +00:00
Nextcloud bot
c25ebc819a
Fix(l10n): 🔠 Update translations from Transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-12-31 08:06:31 +00:00
Nextcloud bot
8d29a90222
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-11-08 03:24:46 +00:00
Nextcloud bot
a056d07cdc
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-11-05 03:28:27 +00:00
Nextcloud bot
b0353c3de9
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-27 03:23:00 +00:00
Nextcloud bot
577ac531ee
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-20 18:09:42 +00:00
Nextcloud bot
a373de0b5d
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-19 03:27:23 +00:00
Nextcloud bot
72ff4c0d21
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-19 03:27:05 +00:00
Nextcloud bot
18b3380cdc
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-18 03:24:20 +00:00
Nextcloud bot
c1aa168642
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-18 03:24:02 +00:00
Nextcloud bot
534d4b5a3f
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-17 03:22:53 +00:00
Nextcloud bot
9e8733e4f6
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-17 03:22:37 +00:00
Nextcloud bot
e000f06939
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-16 03:22:41 +00:00
Nextcloud bot
2658f9ca62
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-16 03:22:26 +00:00
Nextcloud bot
083fc202c7
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-15 03:22:49 +00:00
Nextcloud bot
0c36ff37ac
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-15 03:22:33 +00:00
Nextcloud bot
106b615201
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-14 03:27:19 +00:00
Nextcloud bot
8040bd9b43
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-14 03:27:02 +00:00
Nextcloud bot
ddc837e580
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-13 03:22:28 +00:00
Nextcloud bot
e19b1048c5
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-13 03:22:11 +00:00
Nextcloud bot
30cfebaf87
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-12 03:22:34 +00:00
Nextcloud bot
95e6c294ae
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-12 03:22:18 +00:00
Nextcloud bot
10acdd47f8
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-11 03:20:57 +00:00
Nextcloud bot
fe052f0fdc
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-11 03:20:40 +00:00
Nextcloud bot
63b8c9c512
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-10 03:27:33 +00:00
Nextcloud bot
98698068b1
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-10 03:27:17 +00:00
Nextcloud bot
7101e5e714
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-09 03:38:10 +00:00
Nextcloud bot
7d8867da60
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-09 03:37:54 +00:00
Nextcloud bot
df68852fd5
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-08 03:22:53 +00:00
Nextcloud bot
72bb1858a6
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-08 03:22:37 +00:00
Nextcloud bot
6f56028b00
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-07 03:23:50 +00:00
Nextcloud bot
7a851a1684
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-07 03:23:35 +00:00
Nextcloud bot
527c6f535c
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-06 03:24:41 +00:00
Nextcloud bot
1cb7a81a47
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-06 03:24:25 +00:00
Nextcloud bot
93f83c216a
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-05 03:18:02 +00:00
Nextcloud bot
80ec02b767
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-05 03:17:46 +00:00
Nextcloud bot
5309e46d58
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-04 03:18:39 +00:00
Nextcloud bot
b61d82fe1e
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-04 03:18:23 +00:00
Nextcloud bot
0c6d16063c
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-03 03:23:08 +00:00
Nextcloud bot
74dd4b33a4
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-03 03:22:52 +00:00
Nextcloud bot
f0d6588c07
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-02 03:21:01 +00:00
Nextcloud bot
8302493940
[tx-robot] Update transifex configuration
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-10-02 03:20:41 +00:00
Nextcloud bot
fc5241d6f9
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-09-20 04:37:23 +00:00
Nextcloud bot
65e3c246d7
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-08-07 04:17:59 +00:00
Nextcloud bot
5ddf363b20
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-05-05 04:16:54 +00:00
Nextcloud bot
dbe2543fcf
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2022-04-24 04:17:58 +00:00
Nextcloud bot
3dd9acddc7
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-12-27 03:56:13 +00:00
Nextcloud bot
0a9d1cc191
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-11-19 04:44:13 +00:00
Nextcloud bot
643203c874
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-11-13 03:56:21 +00:00
Nextcloud bot
5fe15716ee
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-11-12 03:51:59 +00:00
Nextcloud bot
599fe53515
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-10-20 04:15:44 +00:00
Nextcloud bot
8a87e4d63a
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-09-21 04:05:52 +00:00
Nextcloud bot
ffd5119c98
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-09-15 04:04:00 +00:00
Nextcloud bot
11409cbead
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-09-02 04:00:32 +00:00
Nextcloud bot
78f3d79630
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-08-06 04:08:44 +00:00
Nextcloud bot
6687f92346
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-06-28 04:02:46 +00:00
Nextcloud bot
2836ac6869
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-06-21 04:04:46 +00:00
Nextcloud bot
59acbd9ef5
[tx-robot] updated from transifex
Signed-off-by: Nextcloud bot <bot@nextcloud.com>
2021-06-03 04:12:36 +00:00
Nextcloud bot
fcbaf713f3
[tx-robot] updated from transifex 2021-05-14 04:00:30 +00:00
Nextcloud bot
e515419d63
[tx-robot] updated from transifex 2021-05-04 04:03:32 +00:00
Nextcloud bot
17374d2183
[tx-robot] updated from transifex 2021-05-02 04:00:24 +00:00
Nextcloud bot
915617d79a
[tx-robot] updated from transifex 2021-04-28 04:04:27 +00:00
Nextcloud bot
786ee2b842
[tx-robot] updated from transifex 2021-04-27 04:06:19 +00:00
Nextcloud bot
a3d804325f
[tx-robot] updated from transifex 2021-04-18 03:59:35 +00:00
Nextcloud bot
991bde7231
[tx-robot] updated from transifex 2021-04-17 04:01:43 +00:00
Nextcloud bot
58205cc06c
[tx-robot] updated from transifex 2021-04-15 04:02:45 +00:00
Nextcloud bot
c07dabf789
[tx-robot] updated from transifex 2021-04-13 04:02:11 +00:00
Nextcloud bot
bd8b23f1e0
[tx-robot] updated from transifex 2021-04-12 04:02:11 +00:00
Nextcloud bot
3b682e25d6
[tx-robot] updated from transifex 2021-04-08 04:02:01 +00:00
Nextcloud bot
ea17951c0a
[tx-robot] updated from transifex 2021-04-07 04:03:04 +00:00
Nextcloud bot
b7e5690a81
[tx-robot] updated from transifex 2021-04-04 03:59:48 +00:00
Nextcloud bot
eabab3be02
[tx-robot] updated from transifex 2021-03-23 03:58:37 +00:00
Nextcloud bot
710e471811
[tx-robot] updated from transifex 2021-03-22 03:57:59 +00:00
Nextcloud bot
7c2eef2e88
[tx-robot] updated from transifex 2021-03-20 03:56:27 +00:00
Nextcloud bot
477e3f8319
[tx-robot] updated from transifex 2021-03-16 03:55:53 +00:00
Nextcloud bot
f49f575971
[tx-robot] updated from transifex 2021-03-04 04:00:10 +00:00
Nextcloud bot
0ceb01b2bd
[tx-robot] updated from transifex 2021-02-27 03:57:20 +00:00
Nextcloud bot
a240c1cbef
[tx-robot] updated from transifex 2021-02-24 04:02:48 +00:00
Nextcloud bot
29ff6635f2
[tx-robot] updated from transifex 2021-02-18 03:59:03 +00:00
Nextcloud bot
a63d9599e1
[tx-robot] updated from transifex 2021-02-15 03:49:58 +00:00
Nextcloud bot
5261961673
[tx-robot] updated from transifex 2021-02-13 03:51:13 +00:00
Nextcloud bot
8adc71c413
[tx-robot] updated from transifex 2021-02-09 03:50:58 +00:00
Nextcloud bot
3a0601d217
[tx-robot] updated from transifex 2021-02-08 03:51:03 +00:00
Nextcloud bot
cb1c6fb71a
[tx-robot] updated from transifex 2021-02-06 03:54:45 +00:00
Nextcloud bot
2e287d4dcb
[tx-robot] updated from transifex 2021-02-05 03:55:03 +00:00
Nextcloud bot
64e26551a6
[tx-robot] updated from transifex 2021-01-27 03:54:31 +00:00
Nextcloud bot
4ecf711a65
[tx-robot] updated from transifex 2021-01-26 03:52:56 +00:00
Nextcloud bot
8e6e5b64e4
[tx-robot] updated from transifex 2021-01-19 03:51:52 +00:00
Nextcloud bot
86ac855e9a
[tx-robot] updated from transifex 2021-01-18 03:51:00 +00:00
Nextcloud bot
aa530a204c
[tx-robot] updated from transifex 2021-01-14 03:53:41 +00:00
Nextcloud bot
bd0c9c0c97
[tx-robot] updated from transifex 2021-01-02 03:59:33 +00:00
Nextcloud bot
00c2c6e8ea
[tx-robot] updated from transifex 2020-12-30 03:55:05 +00:00
Nextcloud bot
18650d4d7b
[tx-robot] updated from transifex 2020-12-19 03:52:53 +00:00
Nextcloud bot
3b23a320bc
[tx-robot] updated from transifex 2020-12-15 03:54:07 +00:00
Nextcloud bot
1a18eb48a1
[tx-robot] updated from transifex 2020-12-10 03:59:56 +00:00
Nextcloud bot
4d17c9c5af
[tx-robot] updated from transifex 2020-12-08 03:54:22 +00:00
Nextcloud bot
07fb0e7a87
[tx-robot] updated from transifex 2020-11-30 03:49:06 +00:00
Nextcloud bot
e2c8f185f5
[tx-robot] updated from transifex 2020-11-09 03:49:02 +00:00
Nextcloud bot
c88c9bb5be
[tx-robot] updated from transifex 2020-11-08 03:49:00 +00:00
Nextcloud bot
dc04f5997b
[tx-robot] updated from transifex 2020-11-05 03:47:24 +00:00
Nextcloud bot
5e9bcc88ca
[tx-robot] updated from transifex 2020-11-02 03:48:01 +00:00
Nextcloud bot
b99951c444
[tx-robot] updated from transifex 2020-11-01 03:47:16 +00:00
Nextcloud bot
353145693d
[tx-robot] updated from transifex 2020-10-31 03:48:53 +00:00
Nextcloud bot
c57f5237ed
[tx-robot] updated from transifex 2020-10-29 03:47:24 +00:00
Nextcloud bot
be3a183633
[tx-robot] updated from transifex 2020-10-27 03:48:38 +00:00
Nextcloud bot
511e11e454
[tx-robot] updated from transifex 2020-10-19 03:43:59 +00:00
Nextcloud bot
837a3effb1
[tx-robot] updated from transifex 2020-10-17 03:44:32 +00:00
Nextcloud bot
739ac51931
[tx-robot] updated from transifex 2020-10-15 03:43:43 +00:00
Nextcloud bot
1b58f45a9f
[tx-robot] updated from transifex 2020-10-14 03:47:57 +00:00
Nextcloud bot
e5f6719d22
[tx-robot] updated from transifex 2020-10-13 03:46:24 +00:00
Nextcloud bot
151b1488d4
[tx-robot] updated from transifex 2020-10-11 03:41:24 +00:00
Nextcloud bot
171890e507
[tx-robot] updated from transifex 2020-10-10 03:48:31 +00:00
Nextcloud bot
6a4c2363e7
[tx-robot] updated from transifex 2020-09-30 03:45:27 +00:00
Nextcloud bot
598f3413cf
[tx-robot] updated from transifex 2020-09-28 03:43:36 +00:00
Nextcloud bot
ec9729ebeb
[tx-robot] updated from transifex 2020-09-22 03:43:06 +00:00
Nextcloud bot
ab60df2369
[tx-robot] updated from transifex 2020-09-21 03:37:46 +00:00
Nextcloud bot
e11bed3f31
[tx-robot] updated from transifex 2020-09-20 03:37:19 +00:00
Nextcloud bot
02375e41c5
[tx-robot] updated from transifex 2020-09-18 03:36:44 +00:00
Nextcloud bot
05380b2d39
[tx-robot] updated from transifex 2020-09-15 03:27:54 +00:00
Nextcloud bot
c19248555d
[tx-robot] updated from transifex 2020-09-14 03:25:55 +00:00
Nextcloud bot
b0785093a9
[tx-robot] updated from transifex 2020-09-12 03:25:24 +00:00
Nextcloud bot
5d244ca55a
[tx-robot] updated from transifex 2020-09-10 03:32:13 +00:00
Nextcloud bot
46327119f8
[tx-robot] updated from transifex 2020-09-09 03:25:38 +00:00
Nextcloud bot
f682438980
[tx-robot] updated from transifex 2020-09-07 03:34:43 +00:00
Nextcloud bot
dd09e43818
[tx-robot] updated from transifex 2020-09-05 03:35:03 +00:00
Nextcloud bot
150284e27b
[tx-robot] updated from transifex 2020-09-04 03:35:18 +00:00
rakekniven
f8b9665963
l10n: Triple dot to ellipsis (#224)
Reported at Transifex.

Signed-off-by: rakekniven <mark.ziegler@rakekniven.de>
2020-09-03 13:26:46 +02:00
Nextcloud bot
2aacbb13b5
[tx-robot] updated from transifex 2020-08-26 03:36:10 +00:00
Nextcloud bot
e43a172079
[tx-robot] updated from transifex 2020-08-14 03:50:28 +00:00
Nextcloud bot
d96b8a7875
[tx-robot] updated from transifex 2020-08-06 03:37:22 +00:00
Nextcloud bot
478e3718fb
[tx-robot] updated from transifex 2020-08-04 03:36:16 +00:00
Nextcloud bot
3c19e328a2
[tx-robot] updated from transifex 2020-07-04 03:33:29 +00:00
Nextcloud bot
c9ed554c6d
[tx-robot] updated from transifex 2020-07-02 03:41:28 +00:00
Nextcloud bot
5bc84f9e7f
[tx-robot] updated from transifex 2020-07-01 03:31:43 +00:00
Nextcloud bot
a6ee12a195
[tx-robot] updated from transifex 2020-06-18 03:32:33 +00:00
Nextcloud bot
86d2dce73b
[tx-robot] updated from transifex 2020-06-06 03:34:09 +00:00
Nextcloud bot
236d348aba
[tx-robot] updated from transifex 2020-05-31 03:30:58 +00:00
Nextcloud bot
98daefe3df
[tx-robot] updated from transifex 2020-05-23 03:33:35 +00:00
Nextcloud bot
abce140b8f
[tx-robot] updated from transifex 2020-05-21 03:35:26 +00:00
Nextcloud bot
548e0d871f
[tx-robot] updated from transifex 2020-05-18 03:29:07 +00:00
Nextcloud bot
633901ea99
[tx-robot] updated from transifex 2020-05-17 03:29:47 +00:00
Nextcloud bot
889e365e03
[tx-robot] updated from transifex 2020-05-08 03:31:04 +00:00
e-alfred
b1688dbfed
Added screenshots (#231) 2020-05-02 08:17:22 +02:00
Nextcloud bot
93c954ec7c
[tx-robot] updated from transifex 2020-04-11 03:38:10 +00:00
Nextcloud bot
a7d74d4686
[tx-robot] updated from transifex 2020-03-30 03:38:08 +00:00
Nextcloud bot
961c731f60
[tx-robot] updated from transifex 2020-03-25 03:38:21 +00:00
Nextcloud bot
72702ab53e
[tx-robot] updated from transifex 2020-03-16 03:41:48 +00:00
Nextcloud bot
c199f6f60d
[tx-robot] updated from transifex 2020-02-25 03:42:19 +00:00
Nextcloud bot
dc41bf3798
[tx-robot] updated from transifex 2020-02-24 03:38:37 +00:00
Nextcloud bot
53d9431c15
[tx-robot] updated from transifex 2020-02-19 03:34:19 +00:00
Nextcloud bot
4d00647cc1
[tx-robot] updated from transifex 2020-02-12 03:33:27 +00:00
Nextcloud bot
8130af570f
[tx-robot] updated from transifex 2020-02-07 03:48:09 +00:00
Nextcloud bot
fee8902acb
[tx-robot] updated from transifex 2020-02-06 03:34:17 +00:00
Nextcloud bot
333a31b056
[tx-robot] updated from transifex 2020-02-05 03:35:49 +00:00
Nextcloud bot
48dca0fbb4
[tx-robot] updated from transifex 2020-02-01 03:31:48 +00:00
Nextcloud bot
6c63d8f89b
[tx-robot] updated from transifex 2020-01-28 03:37:13 +00:00
Maxime Sibellas
9bca7b4ca2 Fix typo in french translation for value title_activity_select_account (#222) 2020-01-27 17:07:13 +01:00
Nextcloud bot
79a56e2ada
[tx-robot] updated from transifex 2020-01-26 03:32:24 +00:00
Nextcloud bot
859147b69b
[tx-robot] updated from transifex 2020-01-24 03:31:38 +00:00
Nextcloud bot
00850485e9
[tx-robot] updated from transifex 2020-01-22 03:35:35 +00:00
Nextcloud bot
7bf6b842a0
[tx-robot] updated from transifex 2020-01-20 03:32:56 +00:00
Nextcloud bot
6ad684b650
[tx-robot] updated from transifex 2020-01-19 03:34:16 +00:00
Nextcloud bot
6554f343ae
[tx-robot] updated from transifex 2020-01-16 03:37:16 +00:00
Nextcloud bot
4fc26ff2d4
[tx-robot] updated from transifex 2020-01-14 03:34:11 +00:00
Nextcloud bot
d15945c1ca
[tx-robot] updated from transifex 2020-01-11 03:34:09 +00:00
Nextcloud bot
8bf4ab7c93
[tx-robot] updated from transifex 2020-01-09 04:29:15 +00:00
Niedermann IT-Dienstleistungen
600cb29c9c Update app icon background layer (#218)
see nextcloud/talk-android#695

Please do not forget to update the Play Store logo :)
2020-01-05 16:43:04 +01:00
Nextcloud bot
6840c982ca
[tx-robot] updated from transifex 2020-01-04 04:29:34 +00:00
Nextcloud bot
04b35c398d
[tx-robot] updated from transifex 2020-01-03 04:26:00 +00:00
Nextcloud bot
070a53167a
[tx-robot] updated from transifex 2020-01-02 04:26:26 +00:00
Nextcloud bot
badb5aaaa1
[tx-robot] updated from transifex 2020-01-01 04:26:08 +00:00
Nextcloud bot
7e73e2336b
[tx-robot] updated from transifex 2019-12-25 04:26:01 +00:00
Nextcloud bot
598f1fcf9a
[tx-robot] updated from transifex 2019-12-23 04:29:33 +00:00
Nextcloud bot
13f164b867
[tx-robot] updated from transifex 2019-12-22 04:32:42 +00:00
Nextcloud bot
0c5290118f
[tx-robot] updated from transifex 2019-12-06 03:20:24 +00:00
Nextcloud bot
58a4a7bc4e
[tx-robot] updated from transifex 2019-12-03 03:18:56 +00:00
Nextcloud bot
5bfbb005a5
[tx-robot] updated from transifex 2019-12-02 03:16:40 +00:00
Nextcloud bot
ad2f7c10f7
[tx-robot] updated from transifex 2019-11-28 03:19:20 +00:00
Nextcloud bot
82f43a885f
[tx-robot] updated from transifex 2019-11-22 03:15:02 +00:00
Nextcloud bot
76f2388f02
[tx-robot] updated from transifex 2019-11-21 03:14:32 +00:00
Nextcloud bot
e6e1f1a721
[tx-robot] updated from transifex 2019-11-20 03:15:31 +00:00
Nextcloud bot
961a760234
[tx-robot] updated from transifex 2019-11-03 03:17:31 +00:00
Nextcloud bot
5ea297104c
[tx-robot] updated from transifex 2019-10-28 03:12:20 +00:00
Nextcloud bot
855de0f86e
[tx-robot] updated from transifex 2019-10-25 03:13:09 +00:00
Nextcloud bot
191fabbca4
[tx-robot] updated from transifex 2019-09-24 03:14:06 +00:00
Nextcloud bot
d5149f729b
[tx-robot] updated from transifex 2019-09-21 03:09:17 +00:00
Nextcloud bot
5f5cc55259
[tx-robot] updated from transifex 2019-09-16 03:03:26 +00:00
Nextcloud bot
50d4de987e
[tx-robot] updated from transifex 2019-09-08 03:17:59 +00:00
Nextcloud bot
584524a1c9
[tx-robot] updated from transifex 2019-09-06 03:10:56 +00:00
Nextcloud bot
1730a48f23
[tx-robot] updated from transifex 2019-09-02 03:11:14 +00:00
Nextcloud bot
9f4219bd68
[tx-robot] updated from transifex 2019-08-30 03:14:03 +00:00
Nextcloud bot
3cdf1337f8
[tx-robot] updated from transifex 2019-08-28 03:18:15 +00:00
Nextcloud bot
9a1f187058
[tx-robot] updated from transifex 2019-08-20 03:10:47 +00:00
Nextcloud bot
b09f60cc4e
[tx-robot] updated from transifex 2019-08-18 03:15:36 +00:00
Nextcloud bot
9e5297f2aa
[tx-robot] updated from transifex 2019-08-12 03:10:29 +00:00
e-alfred
080ad35048 Remove dead Google Play links (#210) 2019-08-07 11:30:26 +02:00
Nextcloud bot
c64745d885
[tx-robot] updated from transifex 2019-07-19 03:22:56 +00:00
Nextcloud bot
f66e101e76
[tx-robot] updated from transifex 2019-06-26 03:03:11 +00:00
Nextcloud bot
33a62654cb
[tx-robot] updated from transifex 2019-06-18 03:02:18 +00:00
Nextcloud bot
dc13d4a21c
[tx-robot] updated from transifex 2019-06-08 02:53:19 +00:00
Nextcloud bot
bbef52e03a
[tx-robot] updated from transifex 2019-06-01 02:53:10 +00:00
Nextcloud bot
812c801042
[tx-robot] updated from transifex 2019-05-28 03:07:10 +00:00
Nextcloud bot
926182f997
[tx-robot] updated from transifex 2019-05-22 03:02:36 +00:00
Nextcloud bot
f75a6d40f1
[tx-robot] updated from transifex 2019-05-21 07:15:29 +00:00
Nextcloud bot
b3aca1c6b3
[tx-robot] updated from transifex 2019-05-13 00:53:15 +00:00
Nextcloud bot
5a02dfa045
[tx-robot] updated from transifex 2019-05-08 01:11:40 +00:00
Nextcloud bot
76f901c81f
[tx-robot] updated from transifex 2019-05-07 00:53:48 +00:00
Nextcloud bot
6a13d7ac83
[tx-robot] updated from transifex 2019-05-01 00:53:07 +00:00
Nextcloud bot
f1e5df9e02
[tx-robot] updated from transifex 2019-04-30 00:54:06 +00:00
Nextcloud bot
6bbd5da44f
[tx-robot] updated from transifex 2019-04-28 00:55:09 +00:00
Nextcloud bot
523603c1e3
[tx-robot] updated from transifex 2019-04-26 00:52:07 +00:00
Nextcloud bot
f953e6cd09
[tx-robot] updated from transifex 2019-04-20 00:53:01 +00:00
Nextcloud bot
123ce0adb3
[tx-robot] updated from transifex 2019-04-19 00:54:31 +00:00
Nextcloud bot
7c14bff700
[tx-robot] updated from transifex 2019-04-18 00:54:49 +00:00
Loïc Blot
d92ed7d4b1
Make gp_translation_version non translatable 2019-04-17 09:51:42 +02:00
Nextcloud bot
4f265773b8
[tx-robot] updated from transifex 2019-04-17 00:55:18 +00:00
Nextcloud bot
da0082ccac
[tx-robot] updated from transifex 2019-04-16 01:05:57 +00:00
Nextcloud bot
363248f580
[tx-robot] updated from transifex 2019-04-15 00:54:35 +00:00
Nextcloud bot
13a44da126
[tx-robot] updated from transifex 2019-04-13 00:56:52 +00:00
Nextcloud bot
9409ca64e2
[tx-robot] updated from transifex 2019-04-12 00:54:52 +00:00
Nextcloud bot
27942004b5
[tx-robot] updated from transifex 2019-04-11 00:54:39 +00:00
rakekniven
15aa29910d Update google_playstore_strings.xml (#204)
Changed project names.

Reported at Transifex
2019-04-10 11:20:57 +02:00
Nextcloud bot
6ceee7b012
[tx-robot] updated from transifex 2019-04-09 00:54:46 +00:00
Nextcloud bot
b5957f0c06
[tx-robot] updated from transifex 2019-04-01 00:57:54 +00:00
Nextcloud bot
d4d7fce3e8
[tx-robot] updated from transifex 2019-03-31 01:57:04 +00:00
Nextcloud bot
a7b33e58ac
[tx-robot] updated from transifex 2019-03-29 01:56:44 +00:00
Nextcloud bot
7d8db76067
[tx-robot] updated from transifex 2019-03-25 01:50:12 +00:00
Nextcloud bot
5106432c2b
[tx-robot] updated from transifex 2019-03-24 01:50:33 +00:00
Nextcloud bot
42a2a55cae
[tx-robot] updated from transifex 2019-03-22 01:50:19 +00:00
Nextcloud bot
a9cf377063
[tx-robot] updated from transifex 2019-03-19 01:58:13 +00:00
Loïc Blot
bccfbae82c
Mark pref_lastmsgdate as non translatable
Fix #202
2019-03-18 11:28:09 +01:00
Nextcloud bot
cfb605928d
[tx-robot] updated from transifex 2019-03-18 01:55:44 +00:00
Nextcloud bot
5cae31082b
[tx-robot] updated from transifex 2019-03-15 01:58:43 +00:00
Nextcloud bot
0cda521540
[tx-robot] updated from transifex 2019-03-14 01:51:18 +00:00
Nextcloud bot
d26712cd89
[tx-robot] updated from transifex 2019-03-12 01:50:08 +00:00
Nextcloud bot
a05a30b0e2
[tx-robot] updated from transifex 2019-03-05 02:01:04 +00:00
Nextcloud bot
d9ca696db0
[tx-robot] updated from transifex 2019-03-04 01:47:38 +00:00
Nextcloud bot
a81c5ad464
[tx-robot] updated from transifex 2019-03-03 01:48:24 +00:00
Nextcloud bot
dc593807d4
[tx-robot] updated from transifex 2019-03-02 01:48:18 +00:00
Nextcloud bot
3e5f5705d2
[tx-robot] updated from transifex 2019-03-01 01:55:28 +00:00
Nextcloud bot
8a10e28a1a
[tx-robot] updated from transifex 2019-02-24 01:57:16 +00:00
Nextcloud bot
2e0ce0706d
[tx-robot] updated from transifex 2019-02-17 01:55:59 +00:00
Nextcloud bot
484c25dabe
[tx-robot] updated from transifex 2019-02-16 01:53:12 +00:00
Nextcloud bot
5fc47f03e8
[tx-robot] updated from transifex 2019-02-14 01:53:18 +00:00
Nextcloud bot
b012971a8c
[tx-robot] updated from transifex 2019-02-10 01:56:16 +00:00
Nextcloud bot
816e975ca0
[tx-robot] updated from transifex 2019-02-09 01:54:53 +00:00
Nextcloud bot
649c97d31e
[tx-robot] updated from transifex 2019-02-07 01:57:15 +00:00
Nextcloud bot
4f139906a5
[tx-robot] updated from transifex 2019-02-06 01:48:33 +00:00
Nextcloud bot
1e206b55c8
[tx-robot] updated from transifex 2019-02-03 01:49:09 +00:00
Nextcloud bot
5b184b0490
[tx-robot] updated from transifex 2019-02-02 01:48:21 +00:00
Nextcloud bot
9847559732
[tx-robot] updated from transifex 2019-01-30 01:50:07 +00:00
Nextcloud bot
ceea85925f
[tx-robot] updated from transifex 2019-01-29 01:47:51 +00:00
Nextcloud bot
e81ed347e7
[tx-robot] updated from transifex 2019-01-28 01:48:48 +00:00
Nextcloud bot
32d6864e7c
[tx-robot] updated from transifex 2019-01-27 01:51:11 +00:00
Nextcloud bot
50683049fd
[tx-robot] updated from transifex 2019-01-26 01:52:45 +00:00
Nextcloud bot
edb117a020
[tx-robot] updated from transifex 2019-01-23 01:53:54 +00:00
Nextcloud bot
1a89b1d5f1
[tx-robot] updated from transifex 2019-01-21 01:52:57 +00:00
Nextcloud bot
67f60015ec
[tx-robot] updated from transifex 2019-01-19 01:55:08 +00:00
Nextcloud bot
90b651f3f1
[tx-robot] updated from transifex 2019-01-18 01:57:14 +00:00
Nextcloud bot
55e17986a9
[tx-robot] updated from transifex 2019-01-17 01:54:26 +00:00
Loïc Blot
2b4a1ecf9a
Update PRIVACY_POLICY.md 2019-01-16 17:23:21 +01:00
Loïc Blot
77ab787db7 Add privacy policy inside application and repository
This should permit to prevent Google play removal due to this missing information.
2019-01-16 16:56:53 +01:00
Loïc Blot
d88836e7d2
Update README.md links 2019-01-16 13:59:58 +01:00
Nextcloud bot
f5147e1bc0
[tx-robot] updated from transifex 2019-01-15 01:57:23 +00:00
Loic Blot
6bd1ce5ae9
Version 2.0.6 - ncsmsgo updated to latest golang runtime (1.11.4) 2019-01-14 19:54:10 +01:00
Nextcloud bot
58a753c921
[tx-robot] updated from transifex 2019-01-14 01:54:42 +00:00
Nextcloud bot
3668bbff51
[tx-robot] updated from transifex 2019-01-09 01:55:43 +00:00
Loic Blot
990de7450b
Version fixes 2018-12-22 18:38:14 +01:00
Loic Blot
6b9c825ed9
Version 2.0.5 2018-12-22 18:27:56 +01:00
Loic Blot
c4598208ac
Update maxSdkVersion 2018-12-22 18:24:31 +01:00
Loic Blot
03d18ae1e2
Update target version & dependencies 2018-12-22 18:23:40 +01:00
Loic Blot
3bb087663a
Gradle updates 2018-12-22 18:23:40 +01:00
Nextcloud bot
a43e3f410c
[tx-robot] updated from transifex 2018-12-01 01:50:21 +00:00
Nextcloud bot
a836408cae
[tx-robot] updated from transifex 2018-11-29 01:53:15 +00:00
Nextcloud bot
e2e7ff783d
[tx-robot] updated from transifex 2018-11-13 01:55:21 +00:00
Nextcloud bot
201da17578
[tx-robot] updated from transifex 2018-11-10 01:52:17 +00:00
Nextcloud bot
6e8a2d10b9
[tx-robot] updated from transifex 2018-11-06 01:52:33 +00:00
Nextcloud bot
dfc9f81695
[tx-robot] updated from transifex 2018-10-13 00:44:57 +00:00
Nextcloud bot
8d6b6b20e8
[tx-robot] updated from transifex 2018-09-27 00:54:37 +00:00
Nextcloud bot
261da28edb
[tx-robot] updated from transifex 2018-09-26 00:54:46 +00:00
Nextcloud bot
7d00089898
[tx-robot] updated from transifex 2018-09-25 00:54:42 +00:00
Nextcloud bot
ceefd6c5d7
[tx-robot] updated from transifex 2018-09-22 00:54:44 +00:00
Nextcloud bot
54ba895d1d
[tx-robot] updated from transifex 2018-09-21 01:01:52 +00:00
Nextcloud bot
75eb21fb4a
[tx-robot] updated from transifex 2018-09-17 00:55:16 +00:00
Nextcloud bot
3243292a66
[tx-robot] updated from transifex 2018-09-13 00:53:33 +00:00
Nextcloud bot
bc7d23970e
[tx-robot] updated from transifex 2018-09-08 00:53:13 +00:00
Nextcloud bot
28cfd0ce2a
[tx-robot] updated from transifex 2018-08-31 00:51:23 +00:00
Nextcloud bot
c50ce0e89c
[tx-robot] updated from transifex 2018-08-08 00:53:12 +00:00
Nextcloud bot
6594fee930
[tx-robot] updated from transifex 2018-07-23 00:49:25 +00:00
Nextcloud bot
753505141f
[tx-robot] updated from transifex 2018-07-20 00:49:11 +00:00
Nextcloud bot
c2cbee5cef
[tx-robot] updated from transifex 2018-07-17 00:48:35 +00:00
Nextcloud bot
cf3c7da645
[tx-robot] updated from transifex 2018-07-16 00:47:35 +00:00
Nextcloud bot
b500d02695
[tx-robot] updated from transifex 2018-07-14 00:47:55 +00:00
Nextcloud bot
e5573e7f56
[tx-robot] updated from transifex 2018-07-13 00:50:40 +00:00
Nextcloud bot
397a766663
[tx-robot] updated from transifex 2018-07-10 09:15:10 +00:00
Nextcloud bot
b06b50a8be
[tx-robot] updated from transifex 2018-07-06 00:44:19 +00:00
Nextcloud bot
8a628abb29
[tx-robot] updated from transifex 2018-06-21 00:53:11 +00:00
Nextcloud bot
cb2a8835d3
[tx-robot] updated from transifex 2018-06-15 00:52:29 +00:00
Loic Blot
7b9f035cd0
Update gradle tool 2018-06-10 16:16:25 +02:00
Loic Blot
96cdeaa2f0
Update deps & gradle 2018-06-10 16:08:42 +02:00
Nextcloud bot
92ef7cf592
[tx-robot] updated from transifex 2018-05-31 00:51:28 +00:00
Nextcloud bot
3b8f63d419
[tx-robot] updated from transifex 2018-05-20 00:51:48 +00:00
Nextcloud bot
c1fc5571bf
[tx-robot] updated from transifex 2018-05-18 00:56:43 +00:00
Nextcloud bot
f76798cb9e
[tx-robot] updated from transifex 2018-05-12 00:50:50 +00:00
Nextcloud bot
9ff749182b
[tx-robot] updated from transifex 2018-04-25 00:47:27 +00:00
Nextcloud bot
94818906ae
[tx-robot] updated from transifex 2018-04-22 00:46:31 +00:00
Niedermann IT-Dienstleistungen
9e597fb58c Adaptive icons and drawer adjustments (#188)
* Added adaptive icon

* Adjust drawer header to Notes app
2018-04-10 22:51:03 +02:00
Nextcloud bot
4a526730fa
[tx-robot] updated from transifex 2018-04-10 00:46:16 +00:00
Nextcloud bot
0bed763021
[tx-robot] updated from transifex 2018-04-09 00:51:35 +00:00
Nextcloud bot
5c8ba4b370
[tx-robot] updated from transifex 2018-03-27 00:44:02 +00:00
Nextcloud bot
77dd567593
[tx-robot] updated from transifex 2018-03-25 01:41:43 +00:00
Nextcloud bot
7d017c9919
[tx-robot] updated from transifex 2018-03-23 01:43:48 +00:00
Nextcloud bot
79cb5983a2
[tx-robot] updated from transifex 2018-03-20 01:43:37 +00:00
Loic Blot
e1fad991a8
Version 2.0.4 2018-03-18 12:26:16 +01:00
Loic Blot
160688d06f
Update android sdk & build tools 2018-03-18 12:25:31 +01:00
Loic Blot
449a6b2a81
Message date is a long, not a int 2018-03-18 12:16:25 +01:00
Loic Blot
15c9617a2b
Fix iteration on restore + add a log to understand where we are in restoration process 2018-03-18 11:37:36 +01:00
Loic Blot
7af398786e
Fix some restore issue (not complete yet) 2018-03-18 11:37:33 +01:00
Nextcloud bot
99560474fb
[tx-robot] updated from transifex 2018-03-17 01:44:00 +00:00
Nextcloud bot
9a1b28cb6d
[tx-robot] updated from transifex 2018-03-02 01:47:46 +00:00
Nextcloud bot
ae910f363b
[tx-robot] updated from transifex 2018-02-27 01:53:01 +00:00
Nextcloud bot
dd9be98ac8
[tx-robot] updated from transifex 2018-02-22 01:45:08 +00:00
Nextcloud bot
b1ab14f2b0
[tx-robot] updated from transifex 2018-02-21 01:45:41 +00:00
Nextcloud bot
cf7611999c
[tx-robot] updated from transifex 2018-02-20 01:48:56 +00:00
Loic Blot
60ac0afbea
Fix aar date issues due to overflow in 32b golang int in the JNI interface.
Use int64
2018-02-17 18:37:10 +01:00
Nextcloud bot
a1db1d3c03
[tx-robot] updated from transifex 2018-02-17 01:45:49 +00:00
Nextcloud bot
1451943e68
[tx-robot] updated from transifex 2018-02-14 01:43:43 +00:00
Nextcloud bot
6d82702a8d
[tx-robot] updated from transifex 2018-02-12 23:01:39 +00:00
Loic Blot
f2348d0a46
Version 2.0.2: fetch last http error issue properly from low level go client 2018-02-12 22:57:43 +01:00
Loic Blot
95d2bfe8f9
Bump version 2.0.1 2018-02-10 22:48:39 +01:00
Loïc Blot
61ed3d2791
SMS restore now used GoLang HTTP client. (#180)
This permits to remove the whole android HTTP client from source
2018-02-10 15:59:46 +01:00
Loic Blot
8ad3b251b0
Migrate getAllSmsIds to golang implementation 2018-02-10 11:23:46 +01:00
Loic Blot
db2cc05a3a
OCHttpClient::getPhoneList now used GoLang aar
* add handleEarlyHTTPStatus to properly handle http return codes in java part
* cleanup dead code
2018-02-10 10:44:53 +01:00
Loic Blot
1f26787983
Improve ASyncContactLoad performance by using addAll instead of looping + add 2018-02-10 09:49:41 +01:00
Loic Blot
5d0b5d377b
Prepare 2.0.0 version & add Go mobile mention 2018-02-10 00:10:01 +01:00
Loic Blot
d57810b5d8 Replace C++ with Cmake with GoLang gomobile
* Add ncsmsgo.arr which is the GoLang gomobile part
See: https://gitlab.com/nerzhul/ncsmsgo
* Android-NDK is not needed anymore
* Android app getVersion() now uses the Golang part
* Use the next getLastHTTPStatus() call in getVersion()
* Android app pushSms() now uses the Golang part
* JNI SmsBuffer has been replaced with GoLang aar bindings
-> This permits to remove some java code & increase performance
* Prepare a insecure switch flag to the Golang client
2018-02-09 23:59:01 +01:00
Nextcloud bot
5317dc4fd9
[tx-robot] updated from transifex 2018-02-09 01:43:46 +00:00
Loic Blot
62747ae33e
Update gradle version 2018-02-08 22:31:43 +01:00
Loic Blot
a8fad0bd3f
Fix the build 2018-02-08 22:31:30 +01:00
Nextcloud bot
f099ca95d2
[tx-robot] updated from transifex 2018-01-24 01:42:37 +00:00
Nextcloud bot
314016b5ca
[tx-robot] updated from transifex 2018-01-15 01:42:39 +00:00
Nextcloud bot
a7917c0b45
[tx-robot] updated from transifex 2018-01-05 01:43:58 +00:00
Nextcloud bot
da2a42ba91
[tx-robot] updated from transifex 2018-01-04 01:36:44 +00:00
Nextcloud bot
108c619fe7
[tx-robot] updated from transifex 2017-12-29 01:31:45 +00:00
Tobias Kaminsky
59db354042 Fix transifex
Transifex is using values-es_419 for spanish "latin america", but this is not supported on android. This changes "values-es_419" to "values-es-rUS", which works.
2017-12-28 16:20:43 +01:00
Nextcloud bot
92930ec758
[tx-robot] updated from transifex 2017-12-21 01:31:49 +00:00
Nextcloud bot
b877807f76
[tx-robot] updated from transifex 2017-12-20 01:32:20 +00:00
Nextcloud bot
8acf838b88
[tx-robot] updated from transifex 2017-12-19 01:33:19 +00:00
Nextcloud bot
3d27756f02
[tx-robot] updated from transifex 2017-12-18 01:31:22 +00:00
Nextcloud bot
c90d3c0fc2
[tx-robot] updated from transifex 2017-12-17 01:31:47 +00:00
Nextcloud bot
5b2c22e4ba
[tx-robot] updated from transifex 2017-12-11 01:40:45 +00:00
Nextcloud bot
14916aa092
[tx-robot] updated from transifex 2017-12-09 01:40:12 +00:00
Nextcloud bot
1553bc2e0e
[tx-robot] updated from transifex 2017-12-08 14:00:40 +00:00
Nextcloud bot
98f24bdf78
[tx-robot] updated from transifex 2017-12-08 01:38:58 +00:00
Nextcloud bot
3a04abe2d7
[tx-robot] updated from transifex 2017-12-07 01:39:40 +00:00
Loic Blot
89bf53de39
Release 1.3.4 2017-12-06 21:12:26 +01:00
Loic Blot
d12bd8b835
Add a new connection error type & fix periodic sync 2017-12-06 21:11:42 +01:00
Loic Blot
56857c16d4
Fix a wrong pref_slow_sync_frequency_titles in values-en 2017-12-06 19:44:26 +01:00
Loic Blot
011b78ba30
Update sdk to v27 & build tools to 27.0.2 2017-12-06 19:04:52 +01:00
Nextcloud bot
7ede7bb350
[tx-robot] updated from transifex 2017-12-06 01:40:18 +00:00
Nextcloud bot
f5b605c5f6
[tx-robot] updated from transifex 2017-12-05 01:40:02 +00:00
Nextcloud bot
33349000d4
[tx-robot] updated from transifex 2017-12-04 01:39:30 +00:00
Nextcloud bot
5f337bd8b0
[tx-robot] updated from transifex 2017-12-03 01:40:00 +00:00
Nextcloud bot
290312e4c9
[tx-robot] updated from transifex 2017-12-02 01:40:04 +00:00
Loic Blot
2fbb0e6ca5
Release 1.3.3
Help diagnose error + better HTTP options
2017-12-01 22:46:34 +01:00
Loic Blot
580d4c1588
Enhance connection error messages
Handle each exception in OCHttpClient::execute separately to help diagnose
2017-12-01 22:45:58 +01:00
Loic Blot
141dfa40ed
Add Transfer-Encoding option chunked in requests 2017-12-01 19:45:16 +01:00
Loic Blot
ff845bc079
update tools in travis 2017-12-01 19:44:59 +01:00
Loic Blot
05d1355b76
Fix build with NDK 16 2017-12-01 19:34:13 +01:00
Loic Blot
32ab866e55
update gradle build version 2017-12-01 19:07:45 +01:00
Nextcloud bot
3514305b60
[tx-robot] updated from transifex 2017-11-28 01:37:06 +00:00
Loic Blot
f4b20b5873
Release 1.3.2 2017-11-25 12:10:42 +01:00
Nextcloud bot
84134671be
[tx-robot] updated from transifex 2017-11-14 01:36:42 +00:00
Nextcloud bot
7ca2dbe089
[tx-robot] updated from transifex 2017-11-11 01:41:44 +00:00
Nextcloud bot
3ff9208004
[tx-robot] updated from transifex 2017-11-10 08:38:34 +00:00
Nextcloud bot
94856096ed
[tx-robot] updated from transifex 2017-11-10 01:40:33 +00:00
Nextcloud bot
395142b6b0
[tx-robot] updated from transifex 2017-11-09 01:36:37 +00:00
Loic Blot
2efc7d1436
Bump version code for play store (strange...) 2017-11-08 23:48:57 +01:00
Loic Blot
9f31d3ae53
Bump to 1.3.1 2017-11-08 23:41:13 +01:00
Loic Blot
08aba120e8
Translation fix 2017-11-08 23:40:35 +01:00
Loic Blot
fe2a27f4cb
Fix some HTTP client errors, especially closing write output
This fixes issue #162
2017-11-08 23:39:35 +01:00
Nextcloud bot
37c8f29f1e
[tx-robot] updated from transifex 2017-11-08 01:38:03 +00:00
Nextcloud bot
cda995fb57
[tx-robot] updated from transifex 2017-11-07 01:34:35 +00:00
Nextcloud bot
abe6d6b40a
[tx-robot] updated from transifex 2017-11-04 01:33:04 +00:00
Loic Blot
62df914e95
Bump to 1.3.0 2017-11-03 12:21:43 +01:00
Loic Blot
a17bffb68c
Re-initiate the SSL trust manager and remove org.apache.commons.httpclient
+ fix a wrong string
2017-11-03 12:21:16 +01:00
Loic Blot
5cb6c63f16
Refactor the whole OCHttpClient to use modern android http client 2017-11-03 12:16:54 +01:00
Nextcloud bot
605fbf5ad7
[tx-robot] updated from transifex 2017-11-03 01:38:04 +00:00
Nextcloud bot
d2e7621b88
[tx-robot] updated from transifex 2017-10-31 01:30:49 +00:00
Loic Blot
c08e0fbd51
use java.net.URL instead of URI 2017-10-30 09:17:02 +01:00
Loic Blot
5c0eb4ca14
Release 1.2.5 2017-10-30 08:43:06 +01:00
Loic Blot
1701065643
Revert "removed wrongly existing launcher icon"
This reverts commit 87538e351813f41b45a852b4bdf2b27c210cd960.
2017-10-30 08:40:10 +01:00
Loic Blot
d8855fd2c6
json::escape_string: if str is null return empty string 2017-10-30 08:38:20 +01:00
Loic Blot
8e15e18b09
Code style fixes 2017-10-30 08:37:08 +01:00
Loic Blot
8d0ba298ed
Update tools 2017-10-30 08:34:31 +01:00
Loic Blot
eae84dadfb
Fix a NullPointerException reported by Play store 2017-10-30 08:34:17 +01:00
Loic Blot
a72583894e
Translation fix 2017-10-30 08:33:53 +01:00
Nextcloud bot
ad70ad4100
[tx-robot] updated from transifex 2017-10-24 00:33:21 +00:00
Nextcloud bot
2da06bb1a9
[tx-robot] updated from transifex 2017-10-23 00:30:48 +00:00
AndyScherzinger
46650c4660 use mimap instead of drawable 2017-10-22 15:58:19 +02:00
AndyScherzinger
87538e3518 removed wrongly existing launcher icon 2017-10-22 15:58:19 +02:00
Nextcloud bot
94a57138a4
[tx-robot] updated from transifex 2017-10-20 00:31:13 +00:00
Nextcloud bot
81b60c4bd8
[tx-robot] updated from transifex 2017-10-11 00:30:18 +00:00
Nextcloud bot
7ddca6ee33
[tx-robot] updated from transifex 2017-10-10 00:32:17 +00:00
Nextcloud bot
1a57f78b5c
[tx-robot] updated from transifex 2017-09-29 00:31:11 +00:00
Nextcloud bot
35bc4f0017
[tx-robot] updated from transifex 2017-09-27 00:30:03 +00:00
Nextcloud bot
395f368052
[tx-robot] updated from transifex 2017-09-22 00:30:01 +00:00
Nextcloud bot
db86583571
[tx-robot] updated from transifex 2017-09-19 00:30:01 +00:00
Nextcloud bot
6ea0b5d317
[tx-robot] updated from transifex 2017-09-16 00:29:15 +00:00
Nextcloud bot
ee5929e9eb
[tx-robot] updated from transifex 2017-09-15 00:28:33 +00:00
Nextcloud bot
70c2d6ba34
[tx-robot] updated from transifex 2017-09-14 00:28:25 +00:00
Nextcloud bot
8bb64db9ed
[tx-robot] updated from transifex 2017-09-13 00:28:49 +00:00
Loic Blot
587099dba7
Fullsync: add a complete sync message when sync is done instead of telling nothing to sync 2017-09-12 08:10:58 +02:00
Loic Blot
47abfac7c3
Fix wrong Thread created Toast
This fixes #153
2017-09-12 08:03:06 +02:00
Loic Blot
b47a36dd99
Bump 1.2.3 2017-09-11 21:57:42 +02:00
Loic Blot
3edd898d7d
Full sync is now fixed
Full sync was previously not really good, Sms were ordered by date DESC, then the last date was good and when multiple loop iterations occurs, only 1 iteration happened
SmsBuffer now owns LastMessageDate instead of AndroidSmsFetcher
Full sync is now owned by AsyncsmsSync instead of MainActivity
Last message date is more accurate and less error prone
2017-09-11 21:56:34 +02:00
Loic Blot
bd2e70c7e7
Increase versioncode too 2017-09-04 22:37:43 +02:00
Loic Blot
3986e12313
Bump to 1.2.1 2017-09-04 22:37:31 +02:00
Loic Blot
fe67a6e012
Fix date type problem on some platforms 2017-09-04 22:36:22 +02:00
Loic Blot
56eecb98bc
Remove a useless try/catch 2017-09-04 22:28:42 +02:00
Loic Blot
0407f719ef
Add SmsEntry object to remove a JsonObject unoptimized storage object 2017-09-04 22:27:36 +02:00
Loic Blot
ebfd1ccfbf
SmsBuffer: add push helper function to factorize code 2017-09-04 20:26:25 +02:00
Nextcloud bot
d950fc73ed
[tx-robot] updated from transifex 2017-09-03 00:28:55 +00:00
Loic Blot
d092ba4a4d Fix JNI implementation, don't use static methods in Java 2017-08-24 08:27:24 +02:00
Loic Blot
38e529e3f6 Drop SDK cmake 2017-08-24 08:27:24 +02:00
Loic Blot
e037a1c9b2 try cmake like this 2017-08-24 08:27:24 +02:00
Loic Blot
139e5644eb Add cmake component to travis 2017-08-24 08:27:24 +02:00
Loic Blot
b216dc3ebb Try to accept cmake license 2017-08-24 08:27:24 +02:00
Loic Blot
11bbed02b8 Don't let Java choose when cleaning SmsBuffer, do it ourself 2017-08-24 08:27:24 +02:00
Loic Blot
5e6a1fc28e SmsBuffer is now working, replace json objects with the SmsBuffer on sync 2017-08-24 08:27:24 +02:00
Loic Blot
47c2923d0e Update tools versions & prepare JNI work to reduce memory & CPU usage 2017-08-24 08:27:24 +02:00
Nextcloud bot
c52168c939
[tx-robot] updated from transifex 2017-08-24 00:30:50 +00:00
Loïc Blot
c1abafc72c SmsDataProvider: factorize field names 2017-08-22 16:50:16 +02:00
Loïc Blot
c716107c3b Add more files to .gitignore 2017-08-22 15:39:45 +02:00
Loïc Blot
1986b941a6 Update build tools to 26.0.1 2017-08-22 14:39:01 +02:00
Loïc Blot
0a785692b4 Manage versionCode and versionString in gradle directly 2017-08-22 12:27:12 +02:00
Loïc Blot
7303534911 Update travis.yml components 2017-08-22 11:38:30 +02:00
Loic Blot
7c38679287
Re-bump version (play store problem) 2017-08-22 10:19:25 +02:00
Loic Blot
4aa0d0a500
Bump versioncode 2017-08-22 10:18:39 +02:00
Loic Blot
cd3b6bb02b
Bump android versioncode 2017-08-22 10:17:46 +02:00
Loic Blot
6c0de1ddef
Prepare for Android O 2017-08-22 10:12:56 +02:00
Nextcloud bot
70ed3a30c6
[tx-robot] updated from transifex 2017-08-22 00:29:49 +00:00
Nextcloud bot
8e27201e7f
[tx-robot] updated from transifex 2017-08-21 00:29:18 +00:00
Nextcloud bot
941654684b
[tx-robot] updated from transifex 2017-08-19 00:30:13 +00:00
Nextcloud bot
33152d3152
[tx-robot] updated from transifex 2017-08-18 00:29:29 +00:00
Nextcloud bot
94fffc2af3
[tx-robot] updated from transifex 2017-08-17 00:29:14 +00:00
Nextcloud bot
b0f97b3243
[tx-robot] updated from transifex 2017-08-16 00:29:28 +00:00
Nextcloud bot
c12acea373
[tx-robot] updated from transifex 2017-08-02 00:29:06 +00:00
Loic Blot
923a10b137
Update gradle version 2017-07-29 11:41:11 +02:00
Nextcloud bot
155da3c368
[tx-robot] updated from transifex 2017-07-28 00:27:55 +00:00
Nextcloud bot
8b944550e0
[tx-robot] updated from transifex 2017-07-22 00:27:49 +00:00
Nextcloud bot
c5e6bf4f15
[tx-robot] updated from transifex 2017-07-14 00:29:25 +00:00
Nextcloud bot
23d2aae40f
[tx-robot] updated from transifex 2017-07-13 00:29:02 +00:00
Nextcloud bot
e20de7ef54
[tx-robot] updated from transifex 2017-07-12 00:28:57 +00:00
Nextcloud bot
2bef4c1576
[tx-robot] updated from transifex 2017-07-11 00:30:01 +00:00
Nextcloud bot
4f56b9249f
[tx-robot] updated from transifex 2017-07-10 00:28:53 +00:00
Nextcloud bot
f665aea164
[tx-robot] updated from transifex 2017-07-08 00:28:26 +00:00
Nextcloud bot
e76d92daa8
[tx-robot] updated from transifex 2017-07-03 00:28:12 +00:00
Nextcloud bot
4a04a506e9
[tx-robot] updated from transifex 2017-06-30 00:28:30 +00:00
Nextcloud bot
7cd99f04b8
[tx-robot] updated from transifex 2017-06-26 00:28:38 +00:00
Nextcloud bot
179774d34c
[tx-robot] updated from transifex 2017-06-20 00:28:37 +00:00
Nextcloud bot
e1bc4d443c
[tx-robot] updated from transifex 2017-06-16 00:30:02 +00:00
Nextcloud bot
5c2194e8cb
[tx-robot] updated from transifex 2017-06-15 00:29:19 +00:00
Nextcloud bot
8b9dc8204a
[tx-robot] updated from transifex 2017-06-14 00:28:04 +00:00
Nextcloud bot
9701637007
[tx-robot] updated from transifex 2017-06-11 00:27:46 +00:00
Nextcloud bot
21db9c38f3
[tx-robot] updated from transifex 2017-06-10 00:28:07 +00:00
Nextcloud bot
97eb626795
[tx-robot] updated from transifex 2017-06-09 00:28:24 +00:00
Nextcloud bot
11361dfac9
[tx-robot] updated from transifex 2017-06-08 00:28:40 +00:00
Nextcloud bot
8f79d11e53
[tx-robot] updated from transifex 2017-06-07 00:30:32 +00:00
Loic Blot
c125201c4b
Code style fixes 2017-06-07 00:26:05 +02:00
Loic Blot
4f3a5d85dd
Version 1.1.0 2017-06-07 00:01:13 +02:00
Loic Blot
0eb1af2da8
BindObjectPref now takes resId and resolve setting directly 2017-06-07 00:00:43 +02:00
Loic Blot
1d45d0a318
Add an option to disable sync notification
Fix #150
2017-06-06 23:50:20 +02:00
Loic Blot
ceba0324e1
Fix TLS context create, TLSv1.2 > 1.1 > 1.0 > SSL 2017-06-06 23:35:04 +02:00
Nextcloud bot
697a02bfed
[tx-robot] updated from transifex 2017-06-05 00:27:29 +00:00
Nextcloud bot
35297c3d2a
[tx-robot] updated from transifex 2017-06-03 00:26:35 +00:00
Nextcloud bot
c03e7f3f4b
[tx-robot] updated from transifex 2017-06-02 00:28:31 +00:00
Nextcloud bot
24ce187e95
[tx-robot] updated from transifex 2017-06-01 00:26:32 +00:00
Nextcloud bot
af15291bcd
[tx-robot] updated from transifex 2017-05-29 00:28:36 +00:00
Nextcloud bot
75c3f70dbc
[tx-robot] updated from transifex 2017-05-28 00:27:10 +00:00
Nextcloud bot
53472e8a05
[tx-robot] updated from transifex 2017-05-26 00:26:54 +00:00
Nextcloud bot
ebba06ce79
[tx-robot] updated from transifex 2017-05-25 00:26:30 +00:00
Nextcloud bot
ec7dea1c3c
[tx-robot] updated from transifex 2017-05-23 00:27:18 +00:00
Loïc Blot
d4904797f7 Contactload code cleanup 2017-05-22 14:25:31 +02:00
Loïc Blot
a5a03dbfcd More code cleanup 2017-05-22 14:10:36 +02:00
Loïc Blot
e2c65cf867 Code cleanups 2017-05-22 13:26:04 +02:00
Loïc Blot
e46a7bccf2 Add codebeat badge 2017-05-22 12:40:02 +02:00
Loic Blot
ffa033ee95
Update android build tools & gradle 2017-05-21 15:16:01 +02:00
Nextcloud bot
4b02caf75a
[tx-robot] updated from transifex 2017-05-20 00:25:43 +00:00
Nextcloud bot
55db7a3340
[tx-robot] updated from transifex 2017-05-19 00:26:56 +00:00
Nextcloud bot
f67174b55a
[tx-robot] updated from transifex 2017-05-18 00:25:17 +00:00
Nextcloud bot
559026ca82
[tx-robot] updated from transifex 2017-05-08 00:22:38 +00:00
Nextcloud bot
349f87a888
[tx-robot] updated from transifex 2017-05-07 00:22:29 +00:00
Loic Blot
d617dade24
Release 1.0.1 2017-05-03 23:11:07 +02:00
Loic Blot
4c5a43c336
Update android support library version 2017-05-03 23:09:00 +02:00
Loïc Blot
f48c09f5ec Update gradle wrapper version 2017-05-03 16:02:32 +02:00
Loïc Blot
b681967b6a Add .idea to git ignore 2017-05-03 15:47:42 +02:00
Loïc Blot
c682d7fd0d Properly handle InvalidArgumentException when URI is not valid in HTTP connector
This fixes issue #149
2017-05-03 15:47:29 +02:00
Nextcloud bot
bc9f36d80b
[tx-robot] updated from transifex 2017-05-01 00:22:30 +00:00
Nextcloud bot
fddef1167b
[tx-robot] updated from transifex 2017-04-27 00:22:11 +00:00
Loic Blot
fafb8c7388
Update gradle tools 2017-04-26 07:27:34 +02:00
Nextcloud bot
c67370ac17
[tx-robot] updated from transifex 2017-04-26 00:22:56 +00:00
Nextcloud bot
201b463904
[tx-robot] updated from transifex 2017-04-24 00:21:03 +00:00
Nextcloud bot
d20292a137
[tx-robot] updated from transifex 2017-04-23 00:20:53 +00:00
Nextcloud bot
ed592a187b
[tx-robot] updated from transifex 2017-04-21 00:21:19 +00:00
Nextcloud bot
dabc11bb75
[tx-robot] updated from transifex 2017-04-20 00:21:40 +00:00
Nextcloud bot
3def96b8bb
[tx-robot] updated from transifex 2017-04-18 00:21:30 +00:00
Nextcloud bot
aaf64a7eb9
[tx-robot] updated from transifex 2017-04-17 00:21:07 +00:00
Nextcloud bot
e943ed86d2
[tx-robot] updated from transifex 2017-04-16 00:21:10 +00:00
Nextcloud bot
a6bb55450b
[tx-robot] updated from transifex 2017-04-15 00:22:53 +00:00
Nextcloud bot
3781d2f029
[tx-robot] updated from transifex 2017-04-14 00:21:59 +00:00
Nextcloud bot
5fbc7d2664
[tx-robot] updated from transifex 2017-04-13 00:22:24 +00:00
Nextcloud bot
49933a65cd
[tx-robot] updated from transifex 2017-04-06 00:22:14 +00:00
Nextcloud bot
5d7fbb6d8a
[tx-robot] updated from transifex 2017-04-01 00:22:10 +00:00
Nextcloud bot
6ed75c90f3
[tx-robot] updated from transifex 2017-03-31 00:22:29 +00:00
Nextcloud bot
f250ebb4a8
[tx-robot] updated from transifex 2017-03-29 00:21:03 +00:00
Nextcloud bot
349fbe0f97
[tx-robot] updated from transifex 2017-03-28 00:21:23 +00:00
Nextcloud bot
f072484ea5
[tx-robot] updated from transifex 2017-03-27 00:21:29 +00:00
Nextcloud bot
f89ebdc70e
[tx-robot] updated from transifex 2017-03-26 01:21:45 +00:00
Nextcloud bot
e891f70f79
[tx-robot] updated from transifex 2017-03-25 01:22:17 +00:00
Nextcloud bot
3303942782
[tx-robot] updated from transifex 2017-03-24 01:20:42 +00:00
Loic Blot
bd5824395e
Travis build fix 2017-03-23 21:17:43 +01:00
Loic Blot
15e16237f1
Release 1.0.0
Application is stable, reliable and new design has been done by a great contributors.
2017-03-23 20:52:47 +01:00
Loic Blot
afa0fc2fac
Add a feature to reinitialize synchronization cursor 2017-03-23 20:43:32 +01:00
Andy Scherzinger
f83eebe4dd Drawer, Toolbar and minor UI optimizations (#145)
* initial add of toolbar and hamburger menu to main activity
* toolbar added to preferences and accounts screens
* fixed original string (name)
* Layout fix for content
* initial add of toolbar and hamburger menu to main activity
* remove unnecessary padding bottom
* fix styling and paint login activity in blue
* toolbar added to preferences and accounts screens
* Layout fix for content
* remove unnecessary padding bottom
* fix styling and paint login activity in blue
* fixes after rebase
* consolidate style since minVersion=16, make drawer to be behind the systembar
* optimize drawer header
* add back navigation to login activity, fix spinner background
* add back navigation
* more launcher icon to mipmap folders
* fix identation
* fix license header
* revert code style deletion, change identation
* proper notification icon
* proper resolution for login logo
* fix large notification icon
* removed unused Eclair switch since support is v16+
* only use small notification icon
* spaces->tabs
* fix toolbar style for Android 4.x
2017-03-23 20:16:31 +01:00
Loic Blot
0bc671901e
Update libraries and tools 2017-03-23 20:12:49 +01:00
Nextcloud bot
9351c7e65c
[tx-robot] updated from transifex 2017-03-23 01:20:22 +00:00
Nextcloud bot
40c4ea4543
[tx-robot] updated from transifex 2017-03-15 01:19:46 +00:00
Nextcloud bot
6214c0f92d
[tx-robot] updated from transifex 2017-03-12 01:21:49 +00:00
Nextcloud bot
a72175252a
[tx-robot] updated from transifex 2017-03-11 01:20:55 +00:00
Nextcloud bot
cc62030de2
[tx-robot] updated from transifex 2017-03-10 01:19:59 +00:00
Nextcloud bot
d3985b1c92
[tx-robot] updated from transifex 2017-03-04 01:19:00 +00:00
Nextcloud bot
f2ed7e0558
[tx-robot] updated from transifex 2017-02-28 01:19:33 +00:00
Nextcloud bot
f6684138eb
[tx-robot] updated from transifex 2017-02-27 01:32:33 +00:00
Nextcloud bot
0bcefdc30f
[tx-robot] updated from transifex 2017-02-24 01:34:52 +00:00
Nextcloud bot
f2f5597b63
[tx-robot] updated from transifex 2017-02-23 01:31:48 +00:00
Nextcloud bot
9f71d7fe9e
[tx-robot] updated from transifex 2017-02-22 01:24:03 +00:00
Nextcloud bot
2495c97a35
[tx-robot] updated from transifex 2017-02-15 01:18:37 +00:00
Nextcloud bot
4a62335d52
[tx-robot] updated from transifex 2017-02-14 01:19:07 +00:00
Loïc Blot
6c90218d69 Translation fixes (#144)
* Make some translations at not translatable
* Remove them from all translation files

This fixes #143
2017-02-13 10:49:52 +01:00
Daniel Hansson
181fa7f46a Update issue_template.md (#142) 2017-02-13 10:21:30 +01:00
Nextcloud bot
5a15fea9bf
[tx-robot] updated from transifex 2017-02-13 01:18:17 +00:00
Nextcloud bot
0e152ae380
[tx-robot] updated from transifex 2017-02-11 01:18:35 +00:00
Nextcloud bot
d8aff2dd92
[tx-robot] updated from transifex 2017-02-10 01:17:44 +00:00
Nextcloud bot
5d55417531
[tx-robot] updated from transifex 2017-02-09 01:18:06 +00:00
Nextcloud bot
1981c44988
[tx-robot] updated from transifex 2017-02-08 01:18:24 +00:00
Nextcloud bot
96cc3d6bdd
[tx-robot] updated from transifex 2017-02-07 01:20:23 +00:00
Nextcloud bot
b7b07c3bbe
[tx-robot] updated from transifex 2017-02-06 01:17:31 +00:00
Nextcloud bot
f7718a9343
[tx-robot] updated from transifex 2017-02-05 01:17:53 +00:00
Nextcloud bot
77589f9ae5
[tx-robot] updated from transifex 2017-02-04 01:19:04 +00:00
Nextcloud bot
c9c2701355
[tx-robot] updated from transifex 2017-02-03 19:00:37 +00:00
Nextcloud Bot
e76b459f75 Merge pull request #140 from MorrisJobke/transifex
add transifex config
2017-02-03 12:49:46 -06:00
Morris Jobke
3afa30cb79
add transifex config
Signed-off-by: Morris Jobke <hey@morrisjobke.de>
2017-02-03 12:47:54 -06:00
Daniel Hansson
86b909472f Translation too long, and get's cut off (#138)
A possible solution for cut off text. But, would be better to fix in the code somehow.
2017-01-31 07:59:29 +01:00
Loic Blot
60ad4f5fa5 Bump version to 0.24.3
Minor build fixes
2017-01-26 07:43:05 +01:00
Loic Blot
2da14a0bdd Bump version to 0.24.2 2017-01-26 07:40:30 +01:00
Daniel Hansson
c79faeb279 Small fix (#137)
Same as 7fd6e874eb/src/main/res/values/strings.xml
2017-01-23 23:29:03 +01:00
Daniel Hansson
f9d4025193 Update Swedish strings (#136) 2017-01-23 23:20:27 +01:00
Andy Scherzinger
32475e72e6 set new drawer menu icons and also ship them (#133) 2017-01-23 22:36:11 +01:00
Daniel Hansson
d64b702fbb Update to Nextcloud (#134) 2017-01-23 22:34:52 +01:00
Loïc Blot
5785c44484 Bump 0.24.1 2017-01-20 11:35:25 +01:00
Loïc Blot
8b482f51c4 HTTPClient: add redirection follower 2017-01-20 11:35:19 +01:00
Loïc Blot
82d1de3463 HTTPClient: send application name & version code in User Agent
Also ignore cookies like owncloud android lib & set http version to 1.1
2017-01-20 11:35:11 +01:00
Loic Blot
dccfee4377 Bump version to 0.24.0 2017-01-18 23:05:31 +01:00
Loic Blot
90d758bb4a documentation fixes 2017-01-18 23:03:32 +01:00
Loic Blot
ef1922494b Fix a layout problem with nav view in main activity 2017-01-18 23:00:24 +01:00
Loic Blot
bcefc1fdf2 HTTPClient: Permit circular redirection.
This should fixes #128
2017-01-18 23:00:18 +01:00
Ner'zhul
f1188819f2 Add travis build support (#126)
Also update build tools to 25.0.1
2016-12-24 12:52:31 +01:00
Ner'zhul
e831b91f44 README.md: drop dead link 2016-12-24 11:24:04 +01:00
Niedermann IT-Dienstleistungen
28a28907a8 Update README.md (#124)
- Restructured
- Emojis
- ownCloud -> Nextcloud
- Typos
2016-12-24 11:23:20 +01:00
Loic Blot
6b5e809f55 Notify we didn't need owncloud Android library anymore 2016-12-21 20:58:20 +01:00
Ner'zhul
14a3cb1a3b Typo fix 2016-12-21 20:56:29 +01:00
Niedermann IT-Dienstleistungen
c3232dca60 Move to Nextcloud (#118)
* Move to Nextcloud (#116)

- Changed colors to Nextcloud Theme

* Move to Nextcloud (#116)

- Adjusted strings and some translations

* Move to Nextcloud (#116)

- Adjust icon color
2016-12-21 20:55:17 +01:00
Loic Blot
1257de1a0a Bump version to 0.23.2 2016-12-20 22:18:22 +01:00
Loic Blot
62ad5b1d34 Import some sources from legacy HTTP Client to permit using self signed certificates like before
It's not the perfect solution but this works

Rename HTTPRequestBuilder to OCHttpClient & make it child of HttpClient
2016-12-20 22:17:44 +01:00
Loic Blot
f0ec0f8b9f Use commons-httpclient from maven instead of embedded 2016-12-20 21:54:49 +01:00
Loic Blot
25e2d13409 Rewrite our HTTP Client to remove the owncloud library dependency and doing authentication ourselves
* Fixes issue #122
* Cleanup the code
* Drop owncloud library support, this will help the unofficial F-Droid build
2016-12-20 21:34:52 +01:00
Loic Blot
f2082e5215 Gradle update 2016-12-20 21:23:26 +01:00
Ner'zhul
03d7ac2b3e Update README.md 2016-12-14 23:34:27 +01:00
Ner'zhul
87fa2b70bb Remove outdated mention to jenkins 2016-12-14 23:33:29 +01:00
Boris Kraut
54f67e5200 Fix classpath dependencies (#115)
com.android.tools.build:gradle:2.2.2 is not in mavenCentral, but only in jcenter.
2016-12-11 00:48:14 +01:00
270 changed files with 16906 additions and 2268 deletions

3
.gitignore vendored
View File

@ -4,3 +4,6 @@ ownCloudSMS.iml
ownCloudSMS-release.apk
lint.xml
.gradle/
.idea/
local.properties
ownCloud-SMS-App.iml

231
.idea/codeStyleSettings.xml generated Normal file
View File

@ -0,0 +1,231 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectCodeStyleSettingsManager">
<option name="PER_PROJECT_SETTINGS">
<value>
<option name="CLASS_COUNT_TO_USE_IMPORT_ON_DEMAND" value="99" />
<option name="NAMES_COUNT_TO_USE_IMPORT_ON_DEMAND" value="99" />
<option name="PACKAGES_TO_USE_IMPORT_ON_DEMAND">
<value />
</option>
<option name="IMPORT_LAYOUT_TABLE">
<value>
<package name="android" withSubpackages="true" static="false" />
<emptyLine />
<package name="com" withSubpackages="true" static="false" />
<emptyLine />
<package name="junit" withSubpackages="true" static="false" />
<emptyLine />
<package name="net" withSubpackages="true" static="false" />
<emptyLine />
<package name="org" withSubpackages="true" static="false" />
<emptyLine />
<package name="java" withSubpackages="true" static="false" />
<emptyLine />
<package name="javax" withSubpackages="true" static="false" />
<emptyLine />
<package name="" withSubpackages="true" static="false" />
<emptyLine />
<package name="" withSubpackages="true" static="true" />
<emptyLine />
</value>
</option>
<option name="RIGHT_MARGIN" value="100" />
<AndroidXmlCodeStyleSettings>
<option name="USE_CUSTOM_SETTINGS" value="true" />
</AndroidXmlCodeStyleSettings>
<Objective-C-extensions>
<option name="GENERATE_INSTANCE_VARIABLES_FOR_PROPERTIES" value="ASK" />
<option name="RELEASE_STYLE" value="IVAR" />
<option name="TYPE_QUALIFIERS_PLACEMENT" value="BEFORE" />
<file>
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Import" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Macro" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Typedef" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Enum" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Constant" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Global" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Struct" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="FunctionPredecl" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Function" />
</file>
<class>
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Property" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="Synthesize" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="InitMethod" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="StaticMethod" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="InstanceMethod" />
<option name="com.jetbrains.cidr.lang.util.OCDeclarationKind" value="DeallocMethod" />
</class>
<extensions>
<pair source="cpp" header="h" />
<pair source="c" header="h" />
</extensions>
</Objective-C-extensions>
<XML>
<option name="XML_KEEP_LINE_BREAKS" value="false" />
<option name="XML_ALIGN_ATTRIBUTES" value="false" />
<option name="XML_SPACE_INSIDE_EMPTY_TAG" value="true" />
</XML>
<codeStyleSettings language="XML">
<option name="FORCE_REARRANGE_MODE" value="1" />
<indentOptions>
<option name="CONTINUATION_INDENT_SIZE" value="4" />
</indentOptions>
<arrangement>
<rules>
<section>
<rule>
<match>
<AND>
<NAME>xmlns:android</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>xmlns:.*</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:id</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:name</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>name</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>style</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:layout_width</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:layout_height</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:layout_.*</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:width</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:height</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_NAMESPACE>.*</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
</rules>
</arrangement>
</codeStyleSettings>
</value>
</option>
<option name="PREFERRED_PROJECT_CODE_STYLE" value="Default (1)" />
</component>
</project>

28
.travis.yml Normal file
View File

@ -0,0 +1,28 @@
---
language: android
before_cache:
- rm -f $HOME/.gradle/caches/modules-2/modules-2.lock
- rm -fr $HOME/.gradle/caches/*/plugin-resolution/
cache:
directories:
- $HOME/.gradle/caches/
- $HOME/.gradle/wrapper/
jdk:
- oraclejdk8
before_install:
- yes | sdkmanager "platforms;android-28"
android:
components:
- tools
- platform-tools
- extra
- build-tools-28.0.3
- android-28
licenses:
- 'android-sdk-license-.+'
- 'google-gdk-license-.+'
- 'android-sdk-preview-license-.+'
- 'android-.*'

17
.tx/config Normal file
View File

@ -0,0 +1,17 @@
[main]
host = https://www.transifex.com
[o:nextcloud:p:nextcloud:r:android-sms]
file_filter = src/main/res/values-<lang>/strings.xml
source_file = src/main/res/values/strings.xml
source_lang = en
type = ANDROID
lang_map = ar_EG: ar-rEG, ar_KW: ar-rKW, da_DK: da-rDK, hu_HU: hu-rHU, lb_LU: lb-rLU, tzm_DZ: tzm-rDZ, ar_DZ: ar-rDZ, ar_SA: ar-rSA, arn_CL: arn-rCL, de_LI: de-rLI, fr_MC: fr-rMC, se_FI: se-rFI, ta_LK: ta-rLK, tt_RU: tt-rRU, xh_ZA: xh-rZA, co_FR: co-rFR, cy_GB: cy-rGB, en_SG: en-rSG, es_PA: es-rPA, es_UY: es-rUY, ku_IQ: ku-rIQ, rm_CH: rm-rCH, smj_SE: smj-rSE, sq_AL: sq-rAL, tr_TR: tr-rTR, uz_UZ: uz-rUZ, en_AU: en-rAU, ar_OM: ar-rOM, el_GR: el-rGR, es_EC: es-rEC, ha_NG: ha-rNG, hr_HR: hr-rHR, or_IN: or-rIN, pt_BR: pt-rBR, se_NO: se-rNO, am_ET: am-rET, en_US: en-rUS, es_PE: es-rPE, fa_IR: fa-rIR, fr_CA: fr-rCA, fy_NL: fy-rNL, hr_BA: hr-rBA, hy_AM: hy-rAM, lt_LT: lt-rLT, ar_SY: ar-rSY, ca_ES: ca-rES, en_JM: en-rJM, es_AR: es-rAR, es_PY: es-rPY, it_CH: it-rCH, kk_KZ: kk-rKZ, vi_VN: vi-rVN, es_MX: es-rMX, pt_PT: pt-rPT, uk_UA: uk-rUA, zh_CN.GB2312: zh-rBG, zu_ZA: zu-rZA, bs_BA: bs-rBA, is_IS: is-rIS, my_MM: my, quz_PE: quz-rPE, ur_PK: ur-rPK, ar_AE: ar-rAE, ar_LY: ar-rLY, ar_QA: ar-rQA, ja_JP: ja-rJP, nl_BE: nl-rBE, nso_ZA: nso-rZA, rw_RW: rw-rRW, sr_BA: sr-rBA, te_IN: te-rIN, de_AT: de-rAT, dv_MV: dv-rMV, ro_RO: ro-rRO, sv_SE: sv-rSE, fr_FR: fr-rFR, he_IL: he-rIL, ne_NP: ne-rNP, sms_FI: sms-rFI, ar_TN: ar-rTN, az_AZ: az-rAZ, de_LU: de-rLU, es_CO: es-rCO, es_NI: es-rNI, id_ID: id-rID, quz_BO: quz-rBO, sr@latin: sr-rSP, en_GB: en-rGB, es_PR: es-rPR, es_SV: es-rSV, kn_IN: kn-rIN, ar_MA: ar-rMA, bo_CN: bo-rCN, dsb_DE: dsb-rDE, ig_NG: ig-rNG, mn_CN: mn-rCN, moh_CA: moh-rCA, pa_IN: pa-rIN, ps_AF: ps-rAF, smn_FI: smn-rFI, zh_MO: zh-rMO, en@pirate: en-rpirate, gl_ES: gl-rES, th_TH: th-rTH, fr_BE: fr-rBE, nb_NO: nb-rNO, prs_AF: prs-rAF, qut_GT: qut-rGT, en_ZW: en-rZW, eu_ES: eu-rES, hsb_DE: hsb-rDE, lo_LA: lo-rLA, mk_MK: mk-rMK, sk_SK: sk-rSK, sma_SE: sma-rSE, zh_HK: zh-rHK, as_IN: as-rIN, en_MY: en-rMY, en_NZ: en-rNZ, es_GT: es-rGT, es_HN: es-rHN, hi_IN: hi-rIN, mt_MT: mt-rMT, oc_FR: oc-rFR, sa_IN: sa-rIN, tk_TM: tk-rTM, ba_RU: ba-rRU, be_BY: be-rBY, kl_GL: kl-rGL, lv_LV: lv-rLV, sah_RU: sah-rRU, yo_NG: yo-rNG, de_DE: de-rDE, es_VE: es-rVE, gd_GB: gd-rGB, ko_KR: ko-rKR, sl_SI: sl-rSI, ug_CN: ug-rCN, ar_YE: ar-rYE, en_IN: en-rIN, es_BO: es-rBO, fr_LU: fr-rLU, bn_BD: bn-rBD, bn_IN: bn-rIN, gu_IN: gu-rIN, mr_IN: mr-rIN, ar_IQ: ar-rIQ, en_CA: en-rCA, es_CR: es-rCR, es_ES: es-rES, ga_IE: ga-rIE, gsw_FR: gsw-rFR, mn_MN: mn-rMN, ru_RU: ru-rRU, sr_CS: sr-rCS, tg_TJ: tg-rTJ, bg_BG: bg-rBG, iu_CA: iu-rCA, nl_NL: nl-rNL, quz_EC: quz-rEC, sma_NO: sma-rNO, sv_FI: sv-rFI, en_IE: en-rIE, fr_CH: fr-rCH, zh_TW: zh-rTW, ar_LB: ar-rLB, br_FR: br-rFR, cs_CZ: cs-rCZ, en_BZ: en-rBZ, en_TT: en-rTT, et_EE: et-rEE, fi_FI: fi-rFI, ii_CN: ii-rCN, km_KH: km-rKH, kok_IN: kok-rIN, ml_IN: ml-rIN, se_SE: se-rSE, syr_SY: syr-rSY, af_ZA: af-rZA, en_ZA: en-rZA, es_CL: es-rCL, mi_NZ: mi-rNZ, smj_NO: smj-rNO, wo_SN: wo-rSN, ar_BH: ar-rBH, fo_FO: fo-rFO, ky_KG: ky-rKG, ms_BN: ms-rBN, nn_NO: nn-rNO, zh_SG: zh-rSG, ar_JO: ar-rJO, en_PH: en-rPH, es_DO: es-rDO, ms_MY: ms-rMY, pl_PL: pl-rPL, sr_RS: sr-rRS, zh_CN: zh-rCN, es_419: es-rUS, it_IT: it-rIT, ka_GE: ka-rGE, si_LK: si-rLK, tn_ZA: tn-rZA, de_CH: de-rCH, fil_PH: fil-rPH, sr_ME: sr-rME, sw_KE: sw-rKE, ta_IN: ta-rIN
[o:nextcloud:p:nextcloud:r:android-sms-playstore]
file_filter = src/main/res/values-<lang>/google_playstore_strings.xml
source_file = src/main/res/values/google_playstore_strings.xml
source_lang = en
type = ANDROID
lang_map = ka_GE: ka-rGE, kn_IN: kn-rIN, sr_CS: sr-rCS, tr_TR: tr-rTR, bs_BA: bs-rBA, es_VE: es-rVE, gd_GB: gd-rGB, es_BO: es-rBO, es_HN: es-rHN, es_PE: es-rPE, eu_ES: eu-rES, rw_RW: rw-rRW, de_CH: de-rCH, de_DE: de-rDE, en_GB: en-rGB, sk_SK: sk-rSK, smj_NO: smj-rNO, zu_ZA: zu-rZA, es_ES: es-rES, mn_MN: mn-rMN, sa_IN: sa-rIN, ta_LK: ta-rLK, uk_UA: uk-rUA, fil_PH: fil-rPH, hr_BA: hr-rBA, ml_IN: ml-rIN, tg_TJ: tg-rTJ, uz_UZ: uz-rUZ, ar_LY: ar-rLY, ar_YE: ar-rYE, my_MM: my, ar_KW: ar-rKW, en_ZW: en-rZW, es_PA: es-rPA, it_CH: it-rCH, ku_IQ: ku-rIQ, ar_AE: ar-rAE, ar_BH: ar-rBH, ar_DZ: ar-rDZ, prs_AF: prs-rAF, pt_BR: pt-rBR, ug_CN: ug-rCN, oc_FR: oc-rFR, sma_NO: sma-rNO, es_PY: es-rPY, ms_MY: ms-rMY, mt_MT: mt-rMT, es_CR: es-rCR, fr_BE: fr-rBE, or_IN: or-rIN, quz_EC: quz-rEC, sr@latin: sr-rSP, zh_HK: zh-rHK, ar_JO: ar-rJO, ar_MA: ar-rMA, bn_BD: bn-rBD, en_ZA: en-rZA, it_IT: it-rIT, ko_KR: ko-rKR, mk_MK: mk-rMK, bo_CN: bo-rCN, co_FR: co-rFR, dsb_DE: dsb-rDE, nl_NL: nl-rNL, sah_RU: sah-rRU, se_SE: se-rSE, zh_CN.GB2312: zh-rBG, zh_TW: zh-rTW, az_AZ: az-rAZ, cy_GB: cy-rGB, nb_NO: nb-rNO, es_UY: es-rUY, fo_FO: fo-rFO, ig_NG: ig-rNG, lo_LA: lo-rLA, mi_NZ: mi-rNZ, en_MY: en-rMY, en_TT: en-rTT, es_SV: es-rSV, nso_ZA: nso-rZA, th_TH: th-rTH, ms_BN: ms-rBN, en_SG: en-rSG, es_EC: es-rEC, id_ID: id-rID, nl_BE: nl-rBE, fr_CH: fr-rCH, hi_IN: hi-rIN, is_IS: is-rIS, en_AU: en-rAU, et_EE: et-rEE, pt_PT: pt-rPT, hy_AM: hy-rAM, lv_LV: lv-rLV, tk_TM: tk-rTM, ur_PK: ur-rPK, en_NZ: en-rNZ, es_DO: es-rDO, es_GT: es-rGT, sv_FI: sv-rFI, tzm_DZ: tzm-rDZ, vi_VN: vi-rVN, ar_OM: ar-rOM, fa_IR: fa-rIR, hu_HU: hu-rHU, de_AT: de-rAT, en_IN: en-rIN, iu_CA: iu-rCA, qut_GT: qut-rGT, smj_SE: smj-rSE, ar_SY: ar-rSY, ar_TN: ar-rTN, cs_CZ: cs-rCZ, te_IN: te-rIN, zh_MO: zh-rMO, quz_BO: quz-rBO, ta_IN: ta-rIN, zh_CN: zh-rCN, de_LI: de-rLI, en_BZ: en-rBZ, ga_IE: ga-rIE, fy_NL: fy-rNL, ha_NG: ha-rNG, kk_KZ: kk-rKZ, rm_CH: rm-rCH, ro_RO: ro-rRO, ar_SA: ar-rSA, en_PH: en-rPH, es_419: es-rUS, hr_HR: hr-rHR, pa_IN: pa-rIN, pl_PL: pl-rPL, sma_SE: sma-rSE, xh_ZA: xh-rZA, dv_MV: dv-rMV, fi_FI: fi-rFI, fr_FR: fr-rFR, zh_SG: zh-rSG, ne_NP: ne-rNP, da_DK: da-rDK, gu_IN: gu-rIN, km_KH: km-rKH, mr_IN: mr-rIN, smn_FI: smn-rFI, el_GR: el-rGR, es_CL: es-rCL, ky_KG: ky-rKG, ps_AF: ps-rAF, ru_RU: ru-rRU, sms_FI: sms-rFI, sq_AL: sq-rAL, arn_CL: arn-rCL, es_AR: es-rAR, fr_LU: fr-rLU, sr_ME: sr-rME, sr_RS: sr-rRS, sw_KE: sw-rKE, yo_NG: yo-rNG, be_BY: be-rBY, es_MX: es-rMX, es_NI: es-rNI, ii_CN: ii-rCN, si_LK: si-rLK, sl_SI: sl-rSI, tn_ZA: tn-rZA, ar_EG: ar-rEG, ar_IQ: ar-rIQ, fr_MC: fr-rMC, moh_CA: moh-rCA, se_NO: se-rNO, en_CA: en-rCA, fr_CA: fr-rCA, kl_GL: kl-rGL, en_IE: en-rIE, en_JM: en-rJM, gl_ES: gl-rES, hsb_DE: hsb-rDE, lb_LU: lb-rLU, af_ZA: af-rZA, ba_RU: ba-rRU, bn_IN: bn-rIN, sv_SE: sv-rSE, wo_SN: wo-rSN, es_CO: es-rCO, br_FR: br-rFR, he_IL: he-rIL, ja_JP: ja-rJP, lt_LT: lt-rLT, tt_RU: tt-rRU, ar_LB: ar-rLB, ar_QA: ar-rQA, as_IN: as-rIN, de_LU: de-rLU, es_PR: es-rPR, gsw_FR: gsw-rFR, kok_IN: kok-rIN, mn_CN: mn-rCN, am_ET: am-rET, bg_BG: bg-rBG, ca_ES: ca-rES, quz_PE: quz-rPE, se_FI: se-rFI, sr_BA: sr-rBA, syr_SY: syr-rSY, en_US: en-rUS, en@pirate: en-rpirate, nn_NO: nn-rNO

20
PRIVACY_POLICY.md Normal file
View File

@ -0,0 +1,20 @@
# Overview
Nextcloud SMS is a free software developed by its contributors. This privacy policy
is intended to inform you about data gathered by this application."
# Information we collect
Only SMS and call log are collected by the application.
# Where information is sent.
Information is neither sent to Nextcloud team servers nor Nextcloud SMS team servers nor
any government nor another entity you don't want.
When you configure a Nextcloud account in the application, you agree with the Nextcloud
instance owner that your SMS and call log data will be stored in his infrastructure
under his responsibility.
We __don't__ recommend to use a public or a company Nextcloud instance account. Your privacy
must be under your control on your own Nextcloud instance.

View File

@ -1,59 +1,45 @@
# ownCloud SMS Android Application Offical Repository
# Nextcloud SMS (Android)
## Introduction
[![codebeat badge](https://codebeat.co/badges/df05cef7-6724-4a2f-b170-96ed1ab793f6)](https://codebeat.co/projects/github-com-nerzhul-owncloud-sms-app-master)
ownCloud SMS app push your Android devices conversation into your ownCloud instance, using ocsms app.
Nextcloud SMS app pushes your Android devices conversation into your Nextcloud instance, using [ocsms app](https://github.com/nerzhul/ocsms).
<a href="https://play.google.com/store/apps/details?id=fr.unix_experience.owncloud_sms">
<img src="http://www.android.com/images/brand/android_app_on_play_large.png" alt="Download from Google Play" />
</a>
<a href="https://f-droid.org/repository/browse/?fdid=fr.unix_experience.owncloud_sms">
<img src="https://camo.githubusercontent.com/7df0eafa4433fa4919a56f87c3d99cf81b68d01c/68747470733a2f2f662d64726f69642e6f72672f77696b692f696d616765732f632f63342f462d44726f69642d627574746f6e5f617661696c61626c652d6f6e2e706e67" alt="ownCloud Notes App on fdroid.org" />
</a>
## :arrow_forward: Access
ocsms app sources are available here: https://github.com/nerzhul/ocsms/
[![Nextcloud Notes App on fdroid.org](https://camo.githubusercontent.com/7df0eafa4433fa4919a56f87c3d99cf81b68d01c/68747470733a2f2f662d64726f69642e6f72672f77696b692f696d616765732f632f63342f462d44726f69642d627574746f6e5f617661696c61626c652d6f6e2e706e67)](https://f-droid.org/repository/browse/?fdid=fr.unix_experience.owncloud_sms)
## Application documentation
ocsms app sources are available here: https://github.com/nextcloud/ocsms
You can found application documentation here: https://github.com/nerzhul/ownCloud-SMS-App/wiki
## :notebook: Application documentation
## Licence
You can find the application documentation here: https://github.com/nerzhul/ncsms-android/wiki
ownCloud SMS Android Application licence is in reflexion, then sources are partial.
## :link: Requirements
- [Nextcloud](https://nextcloud.com/) instance running
- [ocsms](https://github.com/nextcloud/ocsms) app enabled
- App locales and layouts are under BSD 2 clause licence
- App DataProviders are under AGPLv3
## :exclamation: Reporting issues
## Contributions
- **Client:** https://github.com/nerzhul/ncsms-android/issues
- **Server:** https://github.com/nextcloud/ocsms/issues
We are searching for translations in others langs
## :rocket: Contributions
To contribute please download `res/values/strings.xml` and `res/values/google_playstore_strings.xml` and give us a translated version!
- We are searching for **translations** into others languages. To contribute please download `res/values/strings.xml` and `res/values/google_playstore_strings.xml` and provide a Pull Request with a translated version!
- You can also contribute by adding **patches** in Java code or cleanups.
- Application uses a [GoMobile AAR](https://gitlab.com/nerzhul/ncsmsgo) to have the best performance on phones with modern technologies like HTTP/2.0
You can also contribute by adding patches in Java code or cleanups.
## Requirements
- An ownCloud instance with ocsms app
## Build requirements
- nrz-androidlib (last version)
- ownCloud-Android-Library v1.0
## Reporting issues
Please create your issues for the **client** here:
https://github.com/nerzhul/ownCloud-SMS-App/issues
And for the **server** app here:
https://github.com/nerzhul/ocsms/issues
## Developers
You can find our continuous integration here: http://jenkins.unix-experience.fr/job/ownCloud%20SMS%20%28Android%29/
### Build requirements
- gradle
### Coding guidelines
- No empty lines at EOF
- No trailing whitespaces
## :notebook: License
Nextcloud SMS Android Application license is in reflexion, then sources are partial.
- App locales and layouts are under BSD 2 clause license
- App DataProviders are under AGPLv3

View File

@ -1,18 +1,26 @@
buildscript {
repositories {
mavenCentral()
jcenter()
google()
}
dependencies {
classpath 'com.android.tools.build:gradle:2.2.2'
classpath 'com.android.tools.build:gradle:3.2.1'
}
}
allprojects {
repositories {
jcenter()
google()
}
}
apply plugin: 'com.android.application'
android {
compileSdkVersion 25
buildToolsVersion "25.0.0"
useLibrary 'org.apache.http.legacy'
compileSdkVersion 28
buildToolsVersion "28.0.3"
lintOptions {
abortOnError false
@ -20,9 +28,17 @@ android {
defaultConfig {
applicationId "fr.unix_experience.owncloud_sms"
versionCode 70
versionName "2.0.6"
minSdkVersion 16
targetSdkVersion 25
maxSdkVersion 25
targetSdkVersion 28
maxSdkVersion 28
ndk {
// Specifies the ABI configurations of your native
// libraries Gradle should build and package with your APK.
abiFilters 'x86', 'x86_64', 'armeabi', 'armeabi-v7a',
'arm64-v8a', 'mips', 'mips64'
}
}
buildTypes {
@ -41,14 +57,15 @@ repositories {
url "https://jitpack.io"
}
mavenCentral()
google()
}
dependencies {
compile project(':owncloudAndroidLibrary')
compile 'com.android.support:support-v13:25.0.1'
compile 'com.android.support:appcompat-v7:25.0.1'
compile 'com.android.support:design:25.0.1'
compile 'in.srain.cube:ultra-ptr:1.0.11'
compile 'com.github.dmytrodanylyk.android-process-button:library:1.0.4'
compile 'com.android.support:support-v4:25.0.1'
implementation 'com.android.support:support-v13:28.0.0'
implementation 'com.android.support:appcompat-v7:28.0.0'
implementation 'com.android.support:design:28.0.0'
implementation 'in.srain.cube:ultra-ptr:1.0.11'
implementation 'com.github.dmytrodanylyk.android-process-button:library:1.0.4'
implementation 'com.android.support:support-v4:28.0.0'
implementation project(':ncsmsgo')
}

BIN
gradle/wrapper/gradle-wrapper.jar vendored Normal file

Binary file not shown.

View File

@ -0,0 +1,6 @@
#Sat Dec 22 18:14:31 CET 2018
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-4.6-all.zip

164
gradlew vendored Executable file
View File

@ -0,0 +1,164 @@
#!/usr/bin/env bash
##############################################################################
##
## Gradle start up script for UN*X
##
##############################################################################
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS=""
APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"`
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD="maximum"
warn ( ) {
echo "$*"
}
die ( ) {
echo
echo "$*"
echo
exit 1
}
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
case "`uname`" in
CYGWIN* )
cygwin=true
;;
Darwin* )
darwin=true
;;
MINGW* )
msys=true
;;
esac
# For Cygwin, ensure paths are in UNIX format before anything is touched.
if $cygwin ; then
[ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
fi
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
PRG="$0"
# Need this for relative symlinks.
while [ -h "$PRG" ] ; do
ls=`ls -ld "$PRG"`
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
PRG="$link"
else
PRG=`dirname "$PRG"`"/$link"
fi
done
SAVED="`pwd`"
cd "`dirname \"$PRG\"`/" >&-
APP_HOME="`pwd -P`"
cd "$SAVED" >&-
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD="$JAVA_HOME/jre/sh/java"
else
JAVACMD="$JAVA_HOME/bin/java"
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD="java"
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
# Increase the maximum file descriptors if we can.
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
MAX_FD_LIMIT=`ulimit -H -n`
if [ $? -eq 0 ] ; then
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
MAX_FD="$MAX_FD_LIMIT"
fi
ulimit -n $MAX_FD
if [ $? -ne 0 ] ; then
warn "Could not set maximum file descriptor limit: $MAX_FD"
fi
else
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
fi
fi
# For Darwin, add options to specify how the application appears in the dock
if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi
# For Cygwin, switch paths to Windows format before running java
if $cygwin ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
# We build the pattern for arguments to be converted via cygpath
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
SEP=""
for dir in $ROOTDIRSRAW ; do
ROOTDIRS="$ROOTDIRS$SEP$dir"
SEP="|"
done
OURCYGPATTERN="(^($ROOTDIRS))"
# Add a user-defined pattern to the cygpath arguments
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
fi
# Now convert the arguments - kludge to limit ourselves to /bin/sh
i=0
for arg in "$@" ; do
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
else
eval `echo args$i`="\"$arg\""
fi
i=$((i+1))
done
case $i in
(0) set -- ;;
(1) set -- "$args0" ;;
(2) set -- "$args0" "$args1" ;;
(3) set -- "$args0" "$args1" "$args2" ;;
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac
fi
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
function splitJvmOpts() {
JVM_OPTS=("$@")
}
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"

90
gradlew.bat vendored Normal file
View File

@ -0,0 +1,90 @@
@if "%DEBUG%" == "" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS=
set DIRNAME=%~dp0
if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if "%ERRORLEVEL%" == "0" goto init
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto init
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:init
@rem Get command-line arguments, handling Windowz variants
if not "%OS%" == "Windows_NT" goto win9xME_args
if "%@eval[2+2]" == "4" goto 4NT_args
:win9xME_args
@rem Slurp the command line arguments.
set CMD_LINE_ARGS=
set _SKIP=2
:win9xME_args_slurp
if "x%~1" == "x" goto execute
set CMD_LINE_ARGS=%*
goto execute
:4NT_args
@rem Get arguments from the 4NT Shell from JP Software
set CMD_LINE_ARGS=%$
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
:end
@rem End local scope for the variables with windows NT shell
if "%ERRORLEVEL%"=="0" goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
exit /b 1
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

View File

@ -48,7 +48,7 @@
y="32"
x="32"
id="rect4"
style="fill:#1d2d44" /><path
style="fill:#0082C9" /><path
style="fill:#ffffff;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:0.25007507;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0"
d="m 122.44396,89.125042 c -18.45887,0 -33.318923,14.115008 -33.318923,31.647818 l 0,143.74017 c 0,17.53283 14.860053,31.64784 33.318923,31.64784 l 38.24891,0 152.52844,126.43415 -49.39316,-126.43415 125.72788,0 c 18.45889,0 33.31893,-14.11501 33.31893,-31.64784 l 0,-143.74017 c 0,-17.53281 -14.86004,-31.647818 -33.31893,-31.647818 z m 9.05472,27.207518 249.00265,0 c 9.08866,0 16.40549,5.85418 16.40549,13.12558 l 0,8.53918 c 0,7.27142 -7.31683,13.12389 -16.40549,13.12389 l -249.00265,0 c -9.08867,0 -16.40551,-5.85247 -16.40551,-13.12389 l 0,-8.53918 c 0,-7.2714 7.31684,-13.12558 16.40551,-13.12558 z m 0,60.50813 249.00265,0 c 9.08866,0 16.40549,5.85418 16.40549,13.12559 l 0,8.53748 c 0,7.27141 -7.31683,13.12559 -16.40549,13.12559 l -249.00265,0 c -9.08867,0 -16.40551,-5.85418 -16.40551,-13.12559 l 0,-8.53748 c 0,-7.27141 7.31684,-13.12559 16.40551,-13.12559 z m 0,60.01043 89.95439,0 c 9.08866,0 16.40548,7.9877 16.40548,15.25911 l 0,6.40397 c 0,7.27139 -7.31682,13.12558 -16.40548,13.12558 l -89.95439,0 c -9.08867,0 -16.40551,-5.85419 -16.40551,-13.12558 l 0,-6.40397 c 0,-7.27141 7.31684,-15.25911 16.40551,-15.25911 z"
id="rect3350"

Before

Width:  |  Height:  |  Size: 3.0 KiB

After

Width:  |  Height:  |  Size: 3.0 KiB

View File

@ -17,11 +17,11 @@ Tell us what happens instead
### Server configuration
**ownCloud version:**
**Nextcloud version:**
**PHP version:**
**HTTPd server:**
**Webserver:**
**HTTPS:**
@ -31,10 +31,15 @@ Tell us what happens instead
**Phone:**
**ownCloud SMS app version:**
**Nextcloud SMS app version:**
### Logs
```
Insert your log here
```
## Screenshots
<!--
Upload your screenshots here if any
-->

2
ncsmsgo/build.gradle Normal file
View File

@ -0,0 +1,2 @@
configurations.maybeCreate("default")
artifacts.add("default", file('ncsmsgo.aar'))

BIN
ncsmsgo/ncsmsgo.aar Normal file

Binary file not shown.

25
ncsmsgo/ncsmsgo.iml Normal file
View File

@ -0,0 +1,25 @@
<?xml version="1.0" encoding="UTF-8"?>
<module external.linked.project.id=":ncsmsgo" external.linked.project.path="$MODULE_DIR$" external.root.project.path="$MODULE_DIR$/.." external.system.id="GRADLE" type="JAVA_MODULE" version="4">
<component name="FacetManager">
<facet type="android-gradle" name="Android-Gradle">
<configuration>
<option name="GRADLE_PROJECT_PATH" value=":ncsmsgo" />
</configuration>
</facet>
<facet type="java-gradle" name="Java-Gradle">
<configuration>
<option name="BUILD_FOLDER_PATH" value="$MODULE_DIR$/build" />
<option name="BUILDABLE" value="false" />
</configuration>
</facet>
</component>
<component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_7" inherit-compiler-output="true">
<exclude-output />
<content url="file://$MODULE_DIR$">
<excludeFolder url="file://$MODULE_DIR$/.gradle" />
<excludeFolder url="file://$MODULE_DIR$/build" />
</content>
<orderEntry type="inheritedJdk" />
<orderEntry type="sourceFolder" forTests="false" />
</component>
</module>

51
notification_icon.svg Normal file
View File

@ -0,0 +1,51 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg
xmlns:dc="http://purl.org/dc/elements/1.1/"
xmlns:cc="http://creativecommons.org/ns#"
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
xmlns:svg="http://www.w3.org/2000/svg"
xmlns="http://www.w3.org/2000/svg"
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
enable-background="new 0 0 595.275 311.111"
xml:space="preserve"
height="546.13336"
width="546.13336"
version="1.1"
y="0px"
x="0px"
id="svg2"
viewBox="0 0 512 512"
inkscape:version="0.92.1 r15371"
sodipodi:docname="notification_icon.svg"
inkscape:export-filename="C:\DEV\src\Android\Nextcloud\ownCloud-SMS-App\src\main\res\drawable-xxxhdpi\notification_icon.png"
inkscape:export-xdpi="16.879999"
inkscape:export-ydpi="16.879999"><metadata
id="metadata10"><rdf:RDF><cc:Work
rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type
rdf:resource="http://purl.org/dc/dcmitype/StillImage" /></cc:Work></rdf:RDF></metadata><defs
id="defs8" /><sodipodi:namedview
pagecolor="#ffffff"
bordercolor="#666666"
borderopacity="1"
objecttolerance="10"
gridtolerance="10"
guidetolerance="10"
inkscape:pageopacity="0"
inkscape:pageshadow="2"
inkscape:window-width="1920"
inkscape:window-height="1005"
id="namedview6"
showgrid="false"
inkscape:zoom="1.4739747"
inkscape:cx="140.31712"
inkscape:cy="253.09893"
inkscape:window-x="-9"
inkscape:window-y="-9"
inkscape:window-maximized="1"
inkscape:current-layer="svg2" /><path
style="fill:#ffffff;fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:0.30745259;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0"
d="m 76.647972,90.431718 c -24.788364,0 -44.743887,15.887432 -44.743887,35.621852 v 161.78969 c 0,19.73444 19.955523,35.62188 44.743887,35.62188 H 128.01232 L 332.84234,465.77569 266.5124,323.46514 h 168.83961 c 24.7884,0 44.7439,-15.88744 44.7439,-35.62188 V 126.05357 c 0,-19.73442 -19.9555,-35.621852 -44.7439,-35.621852 z M 88.80753,121.0557 h 334.38495 c 12.20514,0 22.03089,6.58929 22.03089,14.77376 v 9.61145 c 0,8.1845 -9.82575,14.77187 -22.03089,14.77187 H 88.80753 c -12.205149,0 -22.030913,-6.58737 -22.030913,-14.77187 v -9.61145 c 0,-8.18447 9.825764,-14.77376 22.030913,-14.77376 z m 0,68.10616 h 334.38495 c 12.20514,0 22.03089,6.5893 22.03089,14.77378 v 9.60954 c 0,8.18448 -9.82575,14.77377 -22.03089,14.77377 H 88.80753 c -12.205149,0 -22.030913,-6.58929 -22.030913,-14.77377 v -9.60954 c 0,-8.18448 9.825764,-14.77378 22.030913,-14.77378 z m 0,67.54597 h 120.79949 c 12.20514,0 22.03088,8.99072 22.03088,17.17521 v 7.20812 c 0,8.18446 -9.82574,14.77376 -22.03088,14.77376 H 88.80753 c -12.205149,0 -22.030913,-6.5893 -22.030913,-14.77376 v -7.20812 c 0,-8.18449 9.825764,-17.17521 22.030913,-17.17521 z"
id="rect3350"
inkscape:connector-curvature="0"
sodipodi:nodetypes="sssscccssssssssssssssssssssssssssssssss" /></svg>

After

Width:  |  Height:  |  Size: 3.1 KiB

BIN
screenshots/login.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

BIN
screenshots/settings.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 44 KiB

BIN
screenshots/settings2.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 45 KiB

1
settings.gradle Normal file
View File

@ -0,0 +1 @@
include ':ncsmsgo'

View File

@ -1,43 +1,43 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest package="fr.unix_experience.owncloud_sms"
xmlns:android="http://schemas.android.com/apk/res/android"
android:versionCode="42"
android:versionName="0.23.0"> <!-- From Android 4.1 to 7.0 -->
<uses-sdk android:maxSdkVersion="25"/>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="fr.unix_experience.owncloud_sms"> <!-- From Android 4.1 to O -->
<uses-sdk android:maxSdkVersion="26" />
<uses-permission android:name="android.permission.READ_SMS"/>
<uses-permission android:name="android.permission.READ_SMS" />
<uses-permission android:name="android.permission.WRITE_SMS" />
<!-- For SMS Restore & Sending -->
<uses-permission android:name="android.permission.SEND_SMS"/>
<uses-permission android:name="android.permission.SEND_SMS" />
<!-- For SMS Broadcaster -->
<uses-permission android:name="android.permission.RECEIVE_SMS"/>
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
<uses-permission android:name="android.permission.RECEIVE_SMS" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!-- For syncer -->
<uses-permission android:name="android.permission.READ_SYNC_STATS"/>
<uses-permission android:name="android.permission.READ_SYNC_SETTINGS"/>
<uses-permission android:name="android.permission.WRITE_SYNC_SETTINGS"/>
<uses-permission android:name="android.permission.WAKE_LOCK"/>
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<uses-permission android:name="android.permission.READ_SYNC_STATS" />
<uses-permission android:name="android.permission.READ_SYNC_SETTINGS" />
<uses-permission android:name="android.permission.WRITE_SYNC_SETTINGS" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<!-- For account management -->
<uses-permission android:name="android.permission.GET_ACCOUNTS"/>
<uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS"/>
<uses-permission android:name="android.permission.MANAGE_ACCOUNTS"/>
<uses-permission android:name="android.permission.USE_CREDENTIALS"/>
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS" />
<uses-permission android:name="android.permission.MANAGE_ACCOUNTS" />
<uses-permission android:name="android.permission.USE_CREDENTIALS" />
<!-- For backup restauration -->
<uses-permission android:name="android.permission.READ_CONTACTS"/>
<uses-permission android:name="android.permission.VIBRATE"/>
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<uses-permission android:name="android.permission.READ_CONTACTS" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:theme="@style/OcSmsTheme">
<!-- Related to periodic sync -->
@ -46,55 +46,59 @@
android:exported="true"
android:process=":sync">
<intent-filter>
<action android:name="android.content.SyncAdapter"/>
<action android:name="android.content.SyncAdapter" />
</intent-filter>
<meta-data
android:name="android.content.SyncAdapter"
android:resource="@xml/sync_adapter"/>
android:resource="@xml/sync_adapter" />
</service>
<provider
android:name=".providers.SmsDataProvider"
android:authorities="@string/account_authority"
android:label="@string/pref_title_sync">
</provider>
android:label="@string/pref_title_sync"></provider>
<!-- Related to Login -->
<service android:name=".authenticators.OwnCloudAuthenticatorService">
<intent-filter>
<action android:name="android.accounts.AccountAuthenticator"/>
<action android:name="android.accounts.AccountAuthenticator" />
</intent-filter>
<meta-data
android:name="android.accounts.AccountAuthenticator"
android:resource="@xml/owncloud_account_authenticator"/>
android:resource="@xml/owncloud_account_authenticator" />
</service>
<receiver android:name=".broadcast_receivers.IncomingSms"
<receiver
android:name=".broadcast_receivers.IncomingSms"
android:permission="android.permission.BROADCAST_SMS">
<intent-filter>
<action android:name="android.provider.Telephony.SMS_RECEIVED"/>
<action android:name="android.provider.Telephony.SMS_RECEIVED" />
<action android:name="android.provider.Telephony.SMS_DELIVER" />
</intent-filter>
</receiver>
<!-- BroadcastReceiver that listens for incoming MMS messages. Note: useless class, used only for restoring SMS -->
<receiver android:name=".misc.MmsReceiver"
<receiver
android:name=".misc.MmsReceiver"
android:permission="android.permission.BROADCAST_WAP_PUSH">
<intent-filter>
<action android:name="android.provider.Telephony.WAP_PUSH_DELIVER" />
<data android:mimeType="application/vnd.wap.mms-message" />
</intent-filter>
</receiver>
<!-- Activity that allows the user to send new SMS/MMS messages Note: useless class, used only for restoring SMS -->
<activity android:name=".misc.ComposeSmsActivity" >
<activity android:name=".misc.ComposeSmsActivity">
<intent-filter>
<action android:name="android.intent.action.SEND" />
<action android:name="android.intent.action.SENDTO" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
<data android:scheme="sms" />
<data android:scheme="smsto" />
<data android:scheme="mms" />
@ -103,12 +107,15 @@
</activity>
<!-- Service that delivers messages from the phone "quick response" Note: useless class, used only for restoring SMS -->
<service android:name=".misc.HeadlessSmsSendService"
android:permission="android.permission.SEND_RESPOND_VIA_MESSAGE"
android:exported="true" >
<service
android:name=".misc.HeadlessSmsSendService"
android:exported="true"
android:permission="android.permission.SEND_RESPOND_VIA_MESSAGE">
<intent-filter>
<action android:name="android.intent.action.RESPOND_VIA_MESSAGE" />
<category android:name="android.intent.category.DEFAULT" />
<data android:scheme="sms" />
<data android:scheme="smsto" />
<data android:scheme="mms" />
@ -118,43 +125,43 @@
<receiver android:name=".broadcast_receivers.ConnectivityChanged">
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
</receiver>
<activity
android:name=".activities.LoginActivity"
android:label="@string/title_activity_login">
</activity>
android:label="@string/title_activity_login"
android:theme="@style/OcSmsTheme.Login"></activity>
<activity
android:name=".activities.remote_account.AccountListActivity"
android:label="@string/title_activity_select_account">
</activity>
android:label="@string/title_activity_select_account"></activity>
<activity
android:name=".activities.OCSMSSettingsActivity"
android:label="@string/title_activity_general_settings">
</activity>
android:label="@string/title_activity_general_settings"></activity>
<activity
android:name=".activities.MainActivity"
android:label="@string/app_name">
android:label="@string/app_name"
android:theme="@style/OcSmsTheme.Drawer">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER"/>
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".activities.remote_account.ContactListActivity"
android:label="@string/title_activity_select_contact">
</activity>
android:label="@string/title_activity_select_contact"></activity>
<activity
android:name=".activities.remote_account.AccountActionsActivity"
android:label="@string/account_actions">
</activity>
android:label="@string/account_actions"></activity>
<activity
android:name=".activities.remote_account.RestoreMessagesActivity"
android:label="@string/restore_all_messages">
</activity>
android:label="@string/restore_all_messages"></activity>
<activity
android:name=".activities.PrivacyPolicyActivity"
android:label="@string/action_appinfo_privacy_policy"
android:theme="@style/OcSmsTheme.NoActionBar"></activity>
</application>
</manifest>

Binary file not shown.

After

Width:  |  Height:  |  Size: 37 KiB

View File

@ -208,7 +208,7 @@ public class AppCompatListActivity extends AppCompatActivity {
super.onContentChanged();
View emptyView = findViewById(R.id.empty);
mList = (ListView)findViewById(R.id.list);
mList = findViewById(R.id.list);
if (mList == null) {
throw new RuntimeException(
"Your content must have a ListView whose id attribute is " +

View File

@ -24,15 +24,17 @@ import android.animation.AnimatorListenerAdapter;
import android.annotation.TargetApi;
import android.content.ContentResolver;
import android.content.Intent;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Build;
import android.os.Bundle;
import android.provider.Settings;
import android.support.v7.app.ActionBar;
import android.support.v7.app.AppCompatActivity;
import android.text.TextUtils;
import android.util.Log;
import android.util.Pair;
import android.view.KeyEvent;
import android.view.MenuItem;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.inputmethod.EditorInfo;
@ -41,20 +43,23 @@ import android.widget.Spinner;
import android.widget.TextView;
import com.dd.processbutton.iml.ActionProcessButton;
import com.owncloud.android.lib.common.OwnCloudClient;
import com.owncloud.android.lib.common.OwnCloudClientFactory;
import com.owncloud.android.lib.common.OwnCloudCredentialsFactory;
import java.net.MalformedURLException;
import java.net.URL;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.authenticators.OwnCloudAuthenticator;
import fr.unix_experience.owncloud_sms.defines.DefaultPrefs;
import fr.unix_experience.owncloud_sms.enums.LoginReturnCode;
import fr.unix_experience.owncloud_sms.engine.OCHttpClient;
import fr.unix_experience.owncloud_sms.exceptions.OCSyncException;
/**
* A login screen that offers login via email/password.
*/
@TargetApi(Build.VERSION_CODES.HONEYCOMB)
public class LoginActivity extends AppCompatActivity {
private static final String TAG = LoginActivity.class.getCanonicalName();
/**
* Keep track of the login task to ensure we can cancel it if requested.
*/
@ -74,12 +79,15 @@ public class LoginActivity extends AppCompatActivity {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
// Set up the login form.
_protocolView = (Spinner) findViewById(R.id.oc_protocol);
_serverView = (EditText) findViewById(R.id.oc_server);
_loginView = (EditText) findViewById(R.id.oc_login);
ActionBar actionBar = getSupportActionBar();
actionBar.setDisplayHomeAsUpEnabled(true);
_passwordView = (EditText) findViewById(R.id.oc_password);
// Set up the login form.
_protocolView = findViewById(R.id.oc_protocol);
_serverView = findViewById(R.id.oc_server);
_loginView = findViewById(R.id.oc_login);
_passwordView = findViewById(R.id.oc_password);
_passwordView
.setOnEditorActionListener(new TextView.OnEditorActionListener() {
@Override
@ -93,7 +101,7 @@ public class LoginActivity extends AppCompatActivity {
}
});
_signInButton = (ActionProcessButton) findViewById(R.id.oc_signin_button);
_signInButton = findViewById(R.id.oc_signin_button);
_signInButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View view) {
@ -105,6 +113,19 @@ public class LoginActivity extends AppCompatActivity {
mProgressView = findViewById(R.id.login_progress);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
boolean retval = true;
switch (item.getItemId()) {
case android.R.id.home:
finish();
break;
default:
retval = super.onOptionsItemSelected(item);
}
return retval;
}
/**
* Attempts to sign in or register the account specified by the login form.
* If there are form errors (invalid email, missing fields, etc.), the
@ -174,8 +195,12 @@ public class LoginActivity extends AppCompatActivity {
_signInButton.setProgress(25);
showProgress(true);
String serverURL = protocol + serverAddr;
try {
mAuthTask = new UserLoginTask(serverURL, login, password);
mAuthTask.execute((Void) null);
} catch (MalformedURLException e) {
Log.e(TAG, "Invalid server URL " + serverURL);
}
}
}
@ -231,34 +256,31 @@ public class LoginActivity extends AppCompatActivity {
*/
public class UserLoginTask extends AsyncTask<Void, Void, Boolean> {
UserLoginTask(String serverURI, String login, String password) {
Log.i(TAG, "_serverURI = " + serverURI);
_serverURI = Uri.parse(serverURI);
UserLoginTask(String serverURL, String login, String password) throws MalformedURLException {
_serverURL = new URL(serverURL);
Log.i(TAG, "_serverURL = " + serverURL);
_login = login;
_password = password;
_last_http_error = null;
}
@Override
protected Boolean doInBackground(Void... params) {
// Create client object to perform remote operations
OwnCloudClient ocClient = OwnCloudClientFactory.createOwnCloudClient(
_serverURI, getBaseContext(),
// Activity or Service context
true
);
_returnCode = 0;
OCHttpClient http = new OCHttpClient(getBaseContext(), _serverURL, _login, _password);
try {
Pair<Integer, Integer> vPair = http.getVersion();
_returnCode = vPair.first;
} catch (IllegalArgumentException e) {
Log.w(TAG, "Failed to getVersion, IllegalArgumentException occured: " + e.getMessage());
_returnCode = 597;
} catch (OCSyncException e) {
Log.w(TAG, "Failed to login, OCSyncException occured: " + e.getMessage());
_returnCode = 599;
}
// Set basic credentials
ocClient.setCredentials(
OwnCloudCredentialsFactory.newBasicCredentials(_login, _password)
);
// Send an authentication test to ownCloud
OwnCloudAuthenticator at = new OwnCloudAuthenticator(getBaseContext());
at.setClient(ocClient);
_returnCode = at.testCredentials();
return (_returnCode == LoginReturnCode.OK);
_last_http_error = http.getLastError();
return (_returnCode == 200);
}
@Override
@ -275,13 +297,13 @@ public class LoginActivity extends AppCompatActivity {
}
// Generate a label
String accountLabel = _login + "@" + _serverURI.getHost();
String accountLabel = _login + "@" + _serverURL.getHost();
// We create the account
Account account = new Account(accountLabel, accountType);
Bundle accountBundle = new Bundle();
accountBundle.putString("ocLogin", _login);
accountBundle.putString("ocURI", _serverURI.toString());
accountBundle.putString("ocURI", _serverURL.toString());
// And we push it to Android
AccountManager accMgr = AccountManager.get(getApplicationContext());
@ -304,26 +326,33 @@ public class LoginActivity extends AppCompatActivity {
} else {
boolean serverViewRequestFocus = true;
switch (_returnCode) {
case INVALID_ADDR:
_serverView.setError(getString(R.string.error_invalid_server_address));
case 0:
if (!_last_http_error.isEmpty()) {
_serverView.setError("Low level error: " + _last_http_error);
}
else {
_serverView.setError("Unknown error");
}
break;
case HTTP_CONN_FAILED:
_serverView.setError(getString(R.string.error_http_connection_failed));
break;
case CONN_FAILED:
_serverView.setError(getString(R.string.error_connection_failed));
break;
case CONN_FAILED_NOT_FOUND:
case 404:
_serverView.setError(getString(R.string.error_connection_failed_not_found));
break;
case UNKNOWN_ERROR:
_serverView.setError("UNK");
case 597:
_serverView.setError(getString(R.string.error_invalid_server_address));
break;
case INVALID_LOGIN:
case 400:
case 598:
_serverView.setError(getString(R.string.error_connection_failed));
break;
case 599:
_serverView.setError(getString(R.string.error_http_connection_failed));
break;
case 401:
case 403:
_passwordView.setError(getString(R.string.error_invalid_login));
_passwordView.requestFocus();
// Warning, there is no break here to disable serverViewRequestFocus too
case OK:
case 200:
default:
serverViewRequestFocus = false;
break;
@ -334,7 +363,7 @@ public class LoginActivity extends AppCompatActivity {
}
// If not ok, reset the progress
if (_returnCode != LoginReturnCode.OK) {
if (_returnCode != 200) {
_signInButton.setProgress(0);
}
}
@ -346,12 +375,13 @@ public class LoginActivity extends AppCompatActivity {
showProgress(false);
}
private final Uri _serverURI;
private final URL _serverURL;
private final String _login;
private final String _password;
private LoginReturnCode _returnCode;
private String _last_http_error;
private int _returnCode;
public static final String PARAM_AUTHTOKEN_TYPE = "auth.token";
static final String PARAM_AUTHTOKEN_TYPE = "auth.token";
private final String TAG = UserLoginTask.class.getCanonicalName();
}
}

View File

@ -38,30 +38,29 @@ import android.support.v4.view.GravityCompat;
import android.support.v4.widget.DrawerLayout;
import android.support.v7.app.ActionBarDrawerToggle;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.util.Log;
import android.view.MenuItem;
import android.view.Window;
import android.widget.Toast;
import org.json.JSONArray;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.activities.remote_account.AccountListActivity;
import fr.unix_experience.owncloud_sms.engine.ASyncSMSSync.SyncTask;
import fr.unix_experience.owncloud_sms.engine.AndroidSmsFetcher;
import fr.unix_experience.owncloud_sms.engine.ConnectivityMonitor;
import fr.unix_experience.owncloud_sms.enums.OCSMSNotificationType;
import fr.unix_experience.owncloud_sms.enums.PermissionID;
import fr.unix_experience.owncloud_sms.notifications.OCSMSNotificationUI;
import fr.unix_experience.owncloud_sms.prefs.PermissionChecker;
import static fr.unix_experience.owncloud_sms.enums.PermissionID.REQUEST_MAX;
import static fr.unix_experience.owncloud_sms.enums.PermissionID.REQUEST_SMS;
public class MainActivity extends AppCompatActivity
implements NavigationView.OnNavigationItemSelectedListener{
implements NavigationView.OnNavigationItemSelectedListener {
private ConnectivityMonitor _ConnectivityMonitor = null;
private DrawerLayout drawer;
@Override
protected void onCreate(Bundle savedInstanceState) {
if (_ConnectivityMonitor == null) {
@ -72,61 +71,134 @@ public class MainActivity extends AppCompatActivity
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
setupToolbar();
if (getSupportActionBar() != null) {
getSupportActionBar().setHomeButtonEnabled(true);
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
}
drawer = findViewById(R.id.drawer_layout);
setupDrawer();
drawer.openDrawer(GravityCompat.START);
}
protected void setupToolbar() {
setSupportActionBar((Toolbar) findViewById(R.id.toolbar));
}
private void setupDrawer() {
ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
this, drawer, null, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
assert drawer != null;
drawer.addDrawerListener(toggle);
toggle.syncState();
toggle.setDrawerIndicatorEnabled(true);
NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
NavigationView navigationView = findViewById(R.id.nav_view);
assert navigationView != null;
navigationView.setNavigationItemSelectedListener(this);
}
/**
* checks if the drawer exists and is opened.
*
* @return <code>true</code> if the drawer is open, else <code>false</code>
*/
public boolean isDrawerOpen() {
return drawer != null && drawer.isDrawerOpen(GravityCompat.START);
}
/**
* closes the drawer.
*/
public void closeDrawer() {
if (drawer != null) {
drawer.closeDrawer(GravityCompat.START);
}
}
/**
* opens the drawer.
*/
public void openDrawer() {
if (drawer != null) {
drawer.openDrawer(GravityCompat.START);
}
}
@Override
public void onBackPressed() {
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
assert drawer != null;
if (drawer.isDrawerOpen(GravityCompat.START)) {
drawer.closeDrawer(GravityCompat.START);
if (isDrawerOpen()) {
closeDrawer();
} else {
super.onBackPressed();
}
}
@Override
public boolean onNavigationItemSelected(MenuItem item) {
public boolean onOptionsItemSelected(MenuItem item) {
boolean retval = true;
switch (item.getItemId()) {
case android.R.id.home: {
if (isDrawerOpen()) {
closeDrawer();
} else {
openDrawer();
}
break;
}
default:
retval = super.onOptionsItemSelected(item);
}
return retval;
}
@Override
public boolean onNavigationItemSelected(@NonNull MenuItem item) {
int id = item.getItemId();
boolean res = true;
switch (id) {
case R.id.nav_sync: syncAllMessages(); break;
case R.id.nav_manage: res = openAppSettings(); break;
case R.id.nav_rateus: res = openGooglePlayStore(); break;
case R.id.nav_add_account: res = openAddAccount(); break;
case R.id.nav_my_accounts: res = openMyAccounts(); break;
case R.id.nav_appinfo_perms: res = openAppInfos(); break;
case R.id.nav_sync:
syncAllMessages();
break;
case R.id.nav_manage:
res = openAppSettings();
break;
case R.id.nav_rateus:
res = openGooglePlayStore();
break;
case R.id.nav_add_account:
res = openAddAccount();
break;
case R.id.nav_my_accounts:
res = openMyAccounts();
break;
case R.id.nav_appinfo_perms:
res = openAppInfos();
break;
case R.id.nav_appinfo_privacy_policy:
res = openPrivacyPolicy();
break;
default:
Log.e(TAG, "Unhandled navigation item " + Integer.toString(id));
}
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
assert drawer != null;
drawer.closeDrawer(GravityCompat.START);
closeDrawer();
return res;
}
private boolean openAppSettings () {
private boolean openAppSettings() {
startActivity(new Intent(this, OCSMSSettingsActivity.class));
return true;
}
private boolean openAddAccount () {
private boolean openAddAccount() {
startActivity(new Intent(Settings.ACTION_ADD_ACCOUNT));
return true;
}
public void syncAllMessages () {
public void syncAllMessages() {
Log.v(MainActivity.TAG, "Launch syncAllMessages()");
if (!PermissionChecker.checkPermission(this, Manifest.permission.READ_SMS,
REQUEST_SMS)) {
@ -134,36 +206,26 @@ public class MainActivity extends AppCompatActivity
}
Context ctx = getApplicationContext();
if (_ConnectivityMonitor.isValid()) {
// Now fetch messages since last stored date
JSONArray smsList = new JSONArray();
new AndroidSmsFetcher(ctx).bufferMessagesSinceDate(smsList, (long) 0);
if (smsList.length() > 0) {
OCSMSNotificationUI.notify(ctx, ctx.getString(R.string.sync_title),
ctx.getString(R.string.sync_inprogress), OCSMSNotificationType.SYNC.ordinal());
new SyncTask(getApplicationContext(), smsList).execute();
}
else {
Toast.makeText(ctx, ctx.getString(R.string.nothing_to_sync), Toast.LENGTH_SHORT).show();
}
}
else {
if (!_ConnectivityMonitor.isValid()) {
Toast.makeText(ctx, ctx.getString(R.string.err_sync_no_connection_available), Toast.LENGTH_SHORT).show();
Log.v(MainActivity.TAG, "Finish syncAllMessages(): invalid connection");
return;
}
new SyncTask(this).execute();
Log.v(MainActivity.TAG, "Finish syncAllMessages()");
}
private boolean openMyAccounts () {
private boolean openMyAccounts() {
startActivity(new Intent(this, AccountListActivity.class));
return true;
}
private boolean openGooglePlayStore () {
private boolean openGooglePlayStore() {
Intent intent;
try {
intent = new Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=" + getPackageName()));
} catch (android.content.ActivityNotFoundException anfe) {
} catch (android.content.ActivityNotFoundException e) {
intent = new Intent(Intent.ACTION_VIEW, Uri.parse("https://play.google.com/store/apps/details?id=" + getPackageName()));
}
@ -171,7 +233,7 @@ public class MainActivity extends AppCompatActivity
return true;
}
private boolean openAppInfos () {
private boolean openAppInfos() {
Intent intent = new Intent();
intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
Uri uri = Uri.fromParts("package", getPackageName(), null);
@ -180,6 +242,11 @@ public class MainActivity extends AppCompatActivity
return true;
}
private boolean openPrivacyPolicy() {
startActivity(new Intent(this, PrivacyPolicyActivity.class));
return true;
}
/*
* Permissions
*/

View File

@ -1,7 +1,7 @@
package fr.unix_experience.owncloud_sms.activities;
/*
* Copyright (c) 2014-2015, Loic Blot <loic.blot@unix-experience.fr>
* Copyright (c) 2014-2017, Loic Blot <loic.blot@unix-experience.fr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
@ -24,9 +24,14 @@ import android.content.ContentResolver;
import android.content.PeriodicSync;
import android.os.Bundle;
import android.preference.ListPreference;
import android.support.v7.app.ActionBar;
import android.support.v7.app.AppCompatDelegate;
import android.util.Log;
import android.util.Pair;
import android.view.MenuItem;
import java.util.List;
import java.util.Vector;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.activities.virtual.VirtualSettingsActivity;
@ -42,6 +47,29 @@ public class OCSMSSettingsActivity extends VirtualSettingsActivity {
private static AccountManager _accountMgr;
private static String _accountAuthority;
private static String _accountType;
private static Vector<Pair<Integer, Boolean>> _boolSettings;
private AppCompatDelegate mDelegate;
@SuppressWarnings("deprecation")
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActionBar actionBar = getSupportActionBar();
actionBar.setDisplayHomeAsUpEnabled(true);
}
private AppCompatDelegate getDelegate() {
if (mDelegate == null) {
mDelegate = AppCompatDelegate.create(this, null);
}
return mDelegate;
}
public ActionBar getSupportActionBar() {
return getDelegate().getSupportActionBar();
}
@Override
protected void onPostCreate(Bundle savedInstanceState) {
@ -51,18 +79,30 @@ public class OCSMSSettingsActivity extends VirtualSettingsActivity {
VirtualSettingsActivity._prefsRessourceFile = R.xml.pref_data_sync;
// Bind our boolean preferences
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("push_on_receive", DefaultPrefs.pushOnReceive));
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("sync_wifi", DefaultPrefs.syncWifi));
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("sync_4g", DefaultPrefs.sync4G));
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("sync_3g", DefaultPrefs.sync3G));
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("sync_gprs", DefaultPrefs.syncGPRS));
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("sync_2g", DefaultPrefs.sync2G));
VirtualSettingsActivity._boolPrefs.add(new BindObjectPref("sync_others", DefaultPrefs.syncOthers));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_push_on_receive, DefaultPrefs.pushOnReceive));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_show_sync_notifications, DefaultPrefs.showSyncNotifications));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_sync_wifi, DefaultPrefs.syncWifi));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_sync_4g, DefaultPrefs.sync4G));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_sync_3g, DefaultPrefs.sync3G));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_sync_gprs, DefaultPrefs.syncGPRS));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_sync_2g, DefaultPrefs.sync2G));
VirtualSettingsActivity._boolPrefs.add(
new BindObjectPref(R.string.setting_sync_others, DefaultPrefs.syncOthers));
// Bind our string preferences
VirtualSettingsActivity._stringPrefs.add(new BindObjectPref("sync_frequency", "15"));
VirtualSettingsActivity._stringPrefs.add(new BindObjectPref("sync_bulk_messages", "-1"));
VirtualSettingsActivity._stringPrefs.add(new BindObjectPref("minimum_sync_chars", "1"));
VirtualSettingsActivity._stringPrefs.add(
new BindObjectPref(R.string.setting_sync_frequency, "15"));
VirtualSettingsActivity._stringPrefs.add(
new BindObjectPref(R.string.setting_sync_bulk_messages, "-1"));
VirtualSettingsActivity._stringPrefs.add(
new BindObjectPref(R.string.setting_minimum_sync_chars, "1"));
// Must be at the end, after preference bind
super.onPostCreate(savedInstanceState);
@ -70,7 +110,7 @@ public class OCSMSSettingsActivity extends VirtualSettingsActivity {
protected void handleCheckboxPreference(String key, Boolean value) {
// Network types allowed for sync
if ("push_on_receive".equals(key) ||
if ("push_on_receive".equals(key) || "show_sync_notifications".equals(key) ||
"sync_wifi".equals(key) || "sync_2g".equals(key) ||
"sync_3g".equals(key) || "sync_gprs".equals(key) ||
"sync_4g".equals(key) || "sync_others".equals(key)) {
@ -88,9 +128,7 @@ public class OCSMSSettingsActivity extends VirtualSettingsActivity {
int index = preference.findIndexOfValue(value);
// Set the summary to reflect the new value.
preference
.setSummary((index >= 0) ? preference.getEntries()[index]
: null);
preference.setSummary((index >= 0) ? preference.getEntries()[index] : null);
Log.i(OCSMSSettingsActivity.TAG, "Modifying listPreference " + key);
@ -107,12 +145,12 @@ public class OCSMSSettingsActivity extends VirtualSettingsActivity {
long syncFreq = Long.parseLong(value);
// Get ownCloud SMS account list
for (Account acct: myAccountList) {
for (Account acct : myAccountList) {
// And get all authorities for this account
List<PeriodicSync> syncList = ContentResolver.getPeriodicSyncs(acct, OCSMSSettingsActivity._accountAuthority);
boolean foundSameSyncCycle = false;
for (PeriodicSync ps: syncList) {
for (PeriodicSync ps : syncList) {
if ((ps.period == syncFreq) && (ps.extras.getInt("synctype") == 1)) {
foundSameSyncCycle = true;
}
@ -130,9 +168,22 @@ public class OCSMSSettingsActivity extends VirtualSettingsActivity {
prefs.putLong(key, syncFreq);
}
}
else if ("sync_bulk_messages".equals(key) || "minimum_sync_chars".equals(key)) {
} else if ("sync_bulk_messages".equals(key) || "minimum_sync_chars".equals(key)) {
prefs.putInteger(key, Integer.parseInt(value));
}
}
@Override
public boolean onMenuItemSelected(int featureId, MenuItem item) {
super.onMenuItemSelected(featureId, item);
switch (item.getItemId()) {
case android.R.id.home:
onBackPressed();
break;
default:
return false;
}
return true;
}
}

View File

@ -0,0 +1,21 @@
package fr.unix_experience.owncloud_sms.activities;
import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.view.View;
import fr.unix_experience.owncloud_sms.R;
public class PrivacyPolicyActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_privacy_policy);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
}
}

View File

@ -1,8 +1,13 @@
package fr.unix_experience.owncloud_sms.activities.remote_account;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AlertDialog;
import android.support.v7.app.ActionBar;
import android.util.Log;
import android.view.MenuItem;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.ListView;
@ -11,6 +16,7 @@ import java.util.ArrayList;
import fr.unix_experience.android_lib.AppCompatListActivity;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.prefs.OCSMSSharedPrefs;
public class AccountActionsActivity extends AppCompatListActivity {
@Override
@ -19,6 +25,9 @@ public class AccountActionsActivity extends AppCompatListActivity {
setContentView(R.layout.activity_account_actions);
ActionBar actionBar = getSupportActionBar();
actionBar.setDisplayHomeAsUpEnabled(true);
ArrayList<String> itemList = new ArrayList<>();
ArrayAdapter<String> adp = new ArrayAdapter<>(getBaseContext(),
android.R.layout.simple_dropdown_item_1line, itemList);
@ -26,6 +35,7 @@ public class AccountActionsActivity extends AppCompatListActivity {
// Create item list
itemList.add(getBaseContext().getString(R.string.restore_all_messages));
itemList.add(getBaseContext().getString(R.string.reinit_sync_cursor));
adp.notifyDataSetChanged();
@ -33,6 +43,19 @@ public class AccountActionsActivity extends AppCompatListActivity {
_accountName = getIntent().getStringExtra("account");
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
boolean retval = true;
switch (item.getItemId()) {
case android.R.id.home:
finish();
break;
default:
retval = super.onOptionsItemSelected(item);
}
return retval;
}
@Override
protected void onListItemClick(ListView l, View v, int position, long id) {
switch (position) {
@ -46,6 +69,23 @@ public class AccountActionsActivity extends AppCompatListActivity {
Log.e(AccountActionsActivity.TAG, e.getMessage());
}
break;
case 1:
final Context me = this;
new AlertDialog.Builder(this)
.setIcon(android.R.drawable.ic_dialog_alert)
.setTitle(R.string.reinit_sync_cursor)
.setMessage(R.string.reinit_sync_cursor_confirm)
.setPositiveButton(R.string.yes_confirm, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
(new OCSMSSharedPrefs(me)).setLastMessageDate(0L);
Log.i(AccountActionsActivity.TAG, "Synchronization cursor reinitialized");
}
})
.setNegativeButton(R.string.no_confirm, null)
.show();
break;
default: break; // Unhandled
}
}

View File

@ -3,13 +3,14 @@ package fr.unix_experience.owncloud_sms.activities.remote_account;
import android.accounts.Account;
import android.accounts.AccountManager;
import android.os.Bundle;
import android.view.MenuItem;
import java.util.ArrayList;
import java.util.Collections;
import fr.unix_experience.owncloud_sms.adapters.AndroidAccountAdapter;
import fr.unix_experience.android_lib.AppCompatListActivity;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.adapters.AndroidAccountAdapter;
public class AccountListActivity extends AppCompatListActivity {
@ -21,6 +22,9 @@ public class AccountListActivity extends AppCompatListActivity {
setContentView(R.layout.restore_activity_accountlist);
getSupportActionBar().setHomeButtonEnabled(true);
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
ArrayList<Account> itemList = new ArrayList<>();
AndroidAccountAdapter adapter = new AndroidAccountAdapter(this,
@ -35,7 +39,18 @@ public class AccountListActivity extends AppCompatListActivity {
Collections.addAll(itemList, accountList);
adapter.notifyDataSetChanged();
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home:
onBackPressed();
break;
default:
return super.onOptionsItemSelected(item);
}
return true;
}
}

View File

@ -49,33 +49,70 @@ public class ContactListActivity extends AppCompatActivity implements ASyncConta
assert getIntent().getExtras() != null;
String accountName = getIntent().getExtras().getString("account");
// accountName cannot be null, devel error
assert accountName != null;
ContactListActivity.mAccountMgr = AccountManager.get(getBaseContext());
Account[] myAccountList =
ContactListActivity.mAccountMgr.getAccountsByType(getString(R.string.account_type));
// Init view
mObjects = new ArrayList<>();
setContentView(R.layout.restore_activity_contactlist);
mLayout = (SwipeRefreshLayout) findViewById(R.id.contactlist_swipe_container);
mLayout = findViewById(R.id.contactlist_swipe_container);
mAdapter = new ContactListAdapter(getBaseContext(), mObjects);
final Spinner sp = (Spinner) findViewById(R.id.contact_spinner);
mContactInfos = (LinearLayout) findViewById(R.id.contactinfos_layout);
final ProgressBar contactProgressBar = (ProgressBar) findViewById(R.id.contactlist_pgbar);
ListView contactPhoneListView = (ListView) findViewById(R.id.contact_phonelistView);
mContactInfos = findViewById(R.id.contactinfos_layout);
ListView contactPhoneListView = findViewById(R.id.contact_phonelistView);
mContactPhoneListAdapter = new RecoveryPhoneNumberListViewAdapter(getBaseContext());
assert contactPhoneListView != null;
contactPhoneListView.setAdapter(mContactPhoneListAdapter);
mContactInfos.setVisibility(View.INVISIBLE);
initSpinner();
createAccountList();
}
private void createAccountList() {
final ProgressBar contactProgressBar = findViewById(R.id.contactlist_pgbar);
assert contactProgressBar != null;
String accountName = getIntent().getExtras().getString("account");
assert accountName != null;
Account[] myAccountList =
ContactListActivity.mAccountMgr.getAccountsByType(getString(R.string.account_type));
for (final Account element : myAccountList) {
if (element.name.equals(accountName)) {
// Load "contacts"
contactProgressBar.setVisibility(View.VISIBLE);
new ContactLoadTask(element, getBaseContext(), mAdapter, mObjects, mLayout,
contactProgressBar, mContactInfos).execute();
// Add refresh handler
mLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
@Override
public void onRefresh() {
mLayout.setRefreshing(true);
mContactInfos.setVisibility(View.INVISIBLE);
contactProgressBar.setVisibility(View.VISIBLE);
(new Handler()).post(new Runnable() {
@Override
public void run() {
mObjects.clear();
mAdapter.notifyDataSetChanged();
new ContactLoadTask(element, getBaseContext(), mAdapter, mObjects,
mLayout, contactProgressBar, mContactInfos).execute();
}
});
}
});
return;
}
}
}
private void initSpinner() {
final Spinner sp = findViewById(R.id.contact_spinner);
assert sp != null;
sp.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
@Override
@ -95,34 +132,6 @@ public class ContactListActivity extends AppCompatActivity implements ASyncConta
});
sp.setAdapter(mAdapter);
for (final Account element : myAccountList) {
if (element.name.equals(accountName)) {
// Load "contacts"
assert contactProgressBar != null;
contactProgressBar.setVisibility(View.VISIBLE);
new ContactLoadTask(element, getBaseContext(), mAdapter, mObjects, mLayout, contactProgressBar, mContactInfos).execute();
// Add refresh handler
mLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
@Override
public void onRefresh() {
mLayout.setRefreshing(true);
mContactInfos.setVisibility(View.INVISIBLE);
contactProgressBar.setVisibility(View.VISIBLE);
(new Handler()).post(new Runnable() {
@Override
public void run() {
mObjects.clear();
mAdapter.notifyDataSetChanged();
new ContactLoadTask(element, getBaseContext(), mAdapter, mObjects, mLayout, contactProgressBar, mContactInfos).execute();
}
});
}
});
return;
}
}
}
private void fetchContact(String name) {

View File

@ -26,8 +26,10 @@ import android.content.pm.PackageManager;
import android.os.Bundle;
import android.provider.Telephony;
import android.support.v4.app.ActivityCompat;
import android.support.v7.app.ActionBar;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
@ -49,6 +51,9 @@ public class RestoreMessagesActivity extends AppCompatActivity {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_restore_messages);
ActionBar actionBar = getSupportActionBar();
actionBar.setDisplayHomeAsUpEnabled(true);
assert getIntent().getExtras() != null;
String accountName = getIntent().getExtras().getString("account");
@ -68,9 +73,9 @@ public class RestoreMessagesActivity extends AppCompatActivity {
}
initInterface();
Button fix_button = (Button) findViewById(R.id.button_fix_permissions);
final Button launch_restore = (Button) findViewById(R.id.button_launch_restore);
final ProgressBar pb = (ProgressBar) findViewById(R.id.progressbar_restore);
Button fix_button = findViewById(R.id.button_fix_permissions);
final Button launch_restore = findViewById(R.id.button_launch_restore);
final ProgressBar pb = findViewById(R.id.progressbar_restore);
final RestoreMessagesActivity me = this;
fix_button.setOnClickListener(new View.OnClickListener() {
@ -111,14 +116,27 @@ public class RestoreMessagesActivity extends AppCompatActivity {
});
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
boolean retval = true;
switch (item.getItemId()) {
case android.R.id.home:
finish();
break;
default:
retval = super.onOptionsItemSelected(item);
}
return retval;
}
private void initInterface() {
TextView tv_error = (TextView) findViewById(R.id.tv_error_default_smsapp);
TextView tv_error = findViewById(R.id.tv_error_default_smsapp);
tv_error.setText(R.string.error_make_default_sms_app);
findViewById(R.id.tv_restore_finished).setVisibility(View.INVISIBLE);
findViewById(R.id.tv_progress_value).setVisibility(View.INVISIBLE);
Button fix_button = (Button) findViewById(R.id.button_fix_permissions);
Button launch_restore = (Button) findViewById(R.id.button_launch_restore);
ProgressBar pb = (ProgressBar) findViewById(R.id.progressbar_restore);
Button fix_button = findViewById(R.id.button_fix_permissions);
Button launch_restore = findViewById(R.id.button_launch_restore);
ProgressBar pb = findViewById(R.id.progressbar_restore);
pb.setVisibility(View.INVISIBLE);
if (android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.KITKAT) {
@ -132,8 +150,7 @@ public class RestoreMessagesActivity extends AppCompatActivity {
tv_error.setVisibility(View.VISIBLE);
fix_button.setVisibility(View.VISIBLE);
launch_restore.setVisibility(View.INVISIBLE);
}
else {
} else {
tv_error.setVisibility(View.INVISIBLE);
fix_button.setVisibility(View.INVISIBLE);
launch_restore.setVisibility(View.VISIBLE);
@ -141,10 +158,10 @@ public class RestoreMessagesActivity extends AppCompatActivity {
}
private void errorNotification(int err) {
TextView tv = (TextView) findViewById(R.id.tv_error_default_smsapp);
Button fix_button = (Button) findViewById(R.id.button_fix_permissions);
Button launch_restore = (Button) findViewById(R.id.button_launch_restore);
ProgressBar pb = (ProgressBar) findViewById(R.id.progressbar_restore);
TextView tv = findViewById(R.id.tv_error_default_smsapp);
Button fix_button = findViewById(R.id.button_fix_permissions);
Button launch_restore = findViewById(R.id.button_launch_restore);
ProgressBar pb = findViewById(R.id.progressbar_restore);
tv.setText(err);
tv.setVisibility(View.VISIBLE);
fix_button.setVisibility(View.INVISIBLE);
@ -165,9 +182,9 @@ public class RestoreMessagesActivity extends AppCompatActivity {
switch (requestCode) {
case RestoreMessagesActivity.REQUEST_DEFAULT_SMSAPP:
if (resultCode == Activity.RESULT_OK) {
TextView tv = (TextView) findViewById(R.id.tv_error_default_smsapp);
Button fix_button = (Button) findViewById(R.id.button_fix_permissions);
Button launch_restore = (Button) findViewById(R.id.button_launch_restore);
TextView tv = findViewById(R.id.tv_error_default_smsapp);
Button fix_button = findViewById(R.id.button_fix_permissions);
Button launch_restore = findViewById(R.id.button_launch_restore);
tv.setVisibility(View.INVISIBLE);
fix_button.setVisibility(View.INVISIBLE);
launch_restore.setVisibility(View.VISIBLE);
@ -208,7 +225,7 @@ public class RestoreMessagesActivity extends AppCompatActivity {
}
public void onProgressUpdate(Integer value) {
TextView tv_progress = (TextView) findViewById(R.id.tv_progress_value);
TextView tv_progress = findViewById(R.id.tv_progress_value);
if (tv_progress.getVisibility() == View.INVISIBLE) {
tv_progress.setVisibility(View.VISIBLE);
}

View File

@ -177,9 +177,9 @@ public class VirtualSettingsActivity extends PreferenceActivity {
// The preference object, it's only a key value pair
protected class BindObjectPref {
public String name;
public Object value;
public BindObjectPref(String prefName, Object prefVal) {
name = prefName;
Object value;
public BindObjectPref(int resId, Object prefVal) {
name = getString(resId);
value = prefVal;
}
}

View File

@ -43,7 +43,7 @@ public class AndroidAccountAdapter extends ArrayAdapter<Account> {
final Account account = _accounts.get(position);
if (account != null) {
TextView label = (TextView) v.findViewById(AndroidAccountAdapter._accountFieldId);
TextView label = v.findViewById(AndroidAccountAdapter._accountFieldId);
if (label != null) {
label.setText(account.name + " >");
v.setOnClickListener(new OnClickListener() {

View File

@ -39,7 +39,7 @@ public class ContactListAdapter extends ArrayAdapter<String> {
String element = _objects.get(position);
if (element != null) {
TextView label = (TextView) v.findViewById(ContactListAdapter._fieldId);
TextView label = v.findViewById(ContactListAdapter._fieldId);
if (label != null) {
label.setText(element);
}

View File

@ -30,7 +30,7 @@ public class RecoveryPhoneNumberListViewAdapter extends ArrayAdapter<String> {
v = inflater.inflate(RecoveryPhoneNumberListViewAdapter._itemLayout, null);
}
TextView label = (TextView) v.findViewById(RecoveryPhoneNumberListViewAdapter._fieldId);
TextView label = v.findViewById(RecoveryPhoneNumberListViewAdapter._fieldId);
if (label != null) {
final String l = getItem(position).toString();
label.setText(getItem(position).toString());

View File

@ -25,19 +25,8 @@ import android.accounts.NetworkErrorException;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import com.owncloud.android.lib.common.OwnCloudClient;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.http.HttpStatus;
import org.json.JSONObject;
import java.io.IOException;
import fr.unix_experience.owncloud_sms.activities.LoginActivity;
import fr.unix_experience.owncloud_sms.enums.LoginReturnCode;
public class OwnCloudAuthenticator extends AbstractAccountAuthenticator {
// Simple constructor
@ -104,94 +93,7 @@ public class OwnCloudAuthenticator extends AbstractAccountAuthenticator {
return null;
}
/*
* Return codes
* 1: invalid address
* 2: HTTP failed
* 3: connexion failed
* 4: invalid login
* 5: unknown error
*/
public LoginReturnCode testCredentials() {
LoginReturnCode bRet = LoginReturnCode.OK;
GetMethod get;
int status;
try {
get = new GetMethod(_client.getBaseUri() + "/index.php/ocs/cloud/user?format=json");
} catch (IllegalArgumentException e) {
return LoginReturnCode.INVALID_ADDR;
}
get.addRequestHeader("OCS-APIREQUEST", "true");
try {
status = _client.executeMethod(get);
} catch (IllegalArgumentException e) {
return LoginReturnCode.INVALID_ADDR;
} catch (HttpException e) {
return LoginReturnCode.HTTP_CONN_FAILED;
} catch (IOException e) {
return LoginReturnCode.CONN_FAILED;
}
try {
if(OwnCloudAuthenticator.isSuccess(status)) {
String response = get.getResponseBodyAsString();
Log.i(OwnCloudAuthenticator.TAG, "Successful response: " + response);
// Parse the response
JSONObject respJSON = new JSONObject(response);
JSONObject respOCS = respJSON.getJSONObject(OwnCloudAuthenticator.NODE_OCS);
JSONObject respData = respOCS.getJSONObject(OwnCloudAuthenticator.NODE_DATA);
String id = respData.getString(OwnCloudAuthenticator.NODE_ID);
String displayName = respData.getString(OwnCloudAuthenticator.NODE_DISPLAY_NAME);
String email = respData.getString(OwnCloudAuthenticator.NODE_EMAIL);
Log.i(OwnCloudAuthenticator.TAG, "*** Parsed user information: " + id + " - " + displayName + " - " + email);
} else {
String response = get.getResponseBodyAsString();
Log.e(OwnCloudAuthenticator.TAG, "Failed response while getting user information ");
if (response != null) {
Log.e(OwnCloudAuthenticator.TAG, "*** status code: " + status + " ; response message: " + response);
} else {
Log.e(OwnCloudAuthenticator.TAG, "*** status code: " + status);
}
switch (status) {
case 401: bRet = LoginReturnCode.INVALID_LOGIN; break;
case 404: bRet = LoginReturnCode.CONN_FAILED_NOT_FOUND; break;
default: bRet = LoginReturnCode.UNKNOWN_ERROR; break;
}
}
} catch (Exception e) {
Log.e(OwnCloudAuthenticator.TAG, "Exception while getting OC user information", e);
bRet = LoginReturnCode.UNKNOWN_ERROR;
} finally {
get.releaseConnection();
}
return bRet;
}
private static boolean isSuccess(int status) {
return (status == HttpStatus.SC_OK);
}
public void setClient(OwnCloudClient oc) {
_client = oc;
}
private final Context _context;
private OwnCloudClient _client;
private static final String TAG = OwnCloudAuthenticator.class.getSimpleName();
private static final String NODE_OCS = "ocs";
private static final String NODE_DATA = "data";
private static final String NODE_ID = "id";
private static final String NODE_DISPLAY_NAME= "display-name";
private static final String NODE_EMAIL= "email";
}

View File

@ -25,8 +25,6 @@ import android.content.Context;
import android.content.Intent;
import android.util.Log;
import org.json.JSONArray;
import java.util.concurrent.atomic.AtomicReference;
import fr.unix_experience.owncloud_sms.R;
@ -36,6 +34,7 @@ import fr.unix_experience.owncloud_sms.engine.ConnectivityMonitor;
import fr.unix_experience.owncloud_sms.enums.PermissionID;
import fr.unix_experience.owncloud_sms.prefs.OCSMSSharedPrefs;
import fr.unix_experience.owncloud_sms.prefs.PermissionChecker;
import ncsmsgo.SmsBuffer;
public class ConnectivityChanged extends BroadcastReceiver implements ASyncSMSSync {
@ -82,14 +81,14 @@ public class ConnectivityChanged extends BroadcastReceiver implements ASyncSMSSy
Log.i(ConnectivityChanged.TAG,"Synced Last:" + lastMessageSynced);
// Now fetch messages since last stored date
JSONArray smsList = new JSONArray();
new AndroidSmsFetcher(context).bufferMessagesSinceDate(smsList, lastMessageSynced);
SmsBuffer smsBuffer = new SmsBuffer();
new AndroidSmsFetcher(context).bufferMessagesSinceDate(smsBuffer, lastMessageSynced);
AtomicReference<ConnectivityMonitor> cMon = new AtomicReference<>(new ConnectivityMonitor(context));
// Synchronize if network is valid and there are SMS
if (cMon.get().isValid() && (smsList.length() > 0)) {
new SyncTask(context, smsList).execute();
if (cMon.get().isValid() && !smsBuffer.empty()) {
new SyncTask(context, smsBuffer).execute();
}
}

View File

@ -4,6 +4,7 @@ public class DefaultPrefs {
public final static Integer syncInterval = 15;
public final static Integer minimumCharsForSync = 0;
public final static Boolean pushOnReceive = true;
public final static Boolean showSyncNotifications = true;
public final static Boolean syncWifi = true;
public final static Boolean sync2G = true;

View File

@ -12,15 +12,13 @@ import android.view.View;
import android.widget.LinearLayout;
import android.widget.ProgressBar;
import org.json.JSONArray;
import org.json.JSONException;
import java.util.ArrayList;
import java.util.Collections;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.adapters.ContactListAdapter;
import fr.unix_experience.owncloud_sms.exceptions.OCSyncException;
import ncsmsgo.SmsPhoneListResponse;
public interface ASyncContactLoad {
class ContactLoadTask extends AsyncTask<Void, Void, Boolean> {
@ -68,19 +66,46 @@ public interface ASyncContactLoad {
ArrayList<String> serverPhoneList = new ArrayList<>();
JSONArray phoneNumbers = _client.getServerPhoneNumbers();
for (int i = 0; i < phoneNumbers.length(); i++) {
String phone = phoneNumbers.getString(i);
serverPhoneList.add(phone);
SmsPhoneListResponse splr = _client.getServerPhoneNumbers();
if (splr == null) {
_objects.add(_context.getString(R.string.err_fetch_phonelist));
return false;
}
String phoneNumber;
while (!(phoneNumber = splr.getNextEntry()).equals("")) {
serverPhoneList.add(phoneNumber);
}
// Read all contacts
readContacts(serverPhoneList);
_objects.addAll(serverPhoneList);
// Sort phone numbers
Collections.sort(_objects);
} catch (OCSyncException e) {
_objects.add(_context.getString(e.getErrorId()));
return false;
}
return true;
}
private void readContacts(ArrayList<String> serverPhoneList) {
ContentResolver cr = _context.getContentResolver();
Cursor cur = cr.query(ContactsContract.Contacts.CONTENT_URI,
null, null, null, null);
if (((cur != null) ? cur.getCount() : 0) > 0) {
if (cur == null) {
return;
}
if (cur.getCount() == 0) {
cur.close();
return;
}
String id, name;
while ((cur != null) && cur.moveToNext()) {
while (cur.moveToNext()) {
id = cur.getString(cur.getColumnIndex(ContactsContract.Contacts._ID));
name = cur.getString(cur.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME));
if (Integer.parseInt(cur.getString(
@ -107,27 +132,10 @@ public interface ASyncContactLoad {
}
}
}
}
if (cur != null) {
cur.close();
}
for (String phone : serverPhoneList) {
_objects.add(phone);
}
// Sort phone numbers
Collections.sort(_objects);
} catch (JSONException e) {
_objects.add(_context.getString(R.string.err_fetch_phonelist));
return false;
} catch (OCSyncException e) {
_objects.add(_context.getString(e.getErrorId()));
return false;
}
return true;
}
protected void onPostExecute(Boolean success) {
_adapter.notifyDataSetChanged();
_layout.setRefreshing(false);

View File

@ -7,14 +7,11 @@ import android.os.AsyncTask;
import android.provider.Telephony;
import android.util.Log;
import org.json.JSONException;
import org.json.JSONObject;
import java.util.Iterator;
import fr.unix_experience.owncloud_sms.activities.remote_account.RestoreMessagesActivity;
import fr.unix_experience.owncloud_sms.enums.MailboxID;
import fr.unix_experience.owncloud_sms.providers.SmsDataProvider;
import ncsmsgo.SmsMessage;
import ncsmsgo.SmsMessagesResponse;
/*
* Copyright (c) 2014-2016, Loic Blot <loic.blot@unix-experience.fr>
@ -60,45 +57,38 @@ public interface ASyncSMSRecovery {
OCSMSOwnCloudClient client = new OCSMSOwnCloudClient(_context, _account);
SmsDataProvider smsDataProvider = new SmsDataProvider(_context);
JSONObject obj = client.retrieveSomeMessages(start, 500);
SmsMessagesResponse obj = client.retrieveSomeMessages(start, 500);
if (obj == null) {
Log.i(ASyncSMSRecovery.TAG, "Retrieved returns failure");
return null;
}
Integer nb = 0;
try {
while (obj.getLong("last_id") != start) {
JSONObject messages = obj.getJSONObject("messages");
Iterator<?> keys = messages.keys();
while (keys.hasNext()) {
String key = (String)keys.next();
if (messages.get(key) instanceof JSONObject) {
JSONObject msg = messages.getJSONObject(key);
int mbid = msg.getInt("mailbox");
while ((obj != null) && (obj.getLastID() != start)) {
Log.i(TAG, "Retrieving messages from " + Long.toString(start)
+ " to " + Long.toString(obj.getLastID()));
SmsMessage message;
while ((message = obj.getNextMessage()) != null) {
int mbid = (int) message.getMailbox();
// Ignore invalid mailbox
if (mbid > MailboxID.ALL.getId()) {
Log.e(ASyncSMSRecovery.TAG, "Invalid mailbox found: " + msg.getString("mailbox"));
Log.e(ASyncSMSRecovery.TAG, "Invalid mailbox found: " + mbid);
continue;
}
String address;
String body;
int type;
try {
address = msg.getString("address");
body = msg.getString("msg");
type = msg.getInt("type");
}
catch (JSONException e) {
Log.e(ASyncSMSRecovery.TAG, "Invalid SMS data found: " + e.getMessage());
String address = message.getAddress();
String body = message.getMessage();
int type = (int) message.getType();
if (address.isEmpty() || body.isEmpty()) {
Log.e(ASyncSMSRecovery.TAG, "Invalid SMS message found: " + message.toString());
continue;
}
MailboxID mailbox_id = MailboxID.fromInt(mbid);
String date = Long.toString(message.getDate());
// Ignore already existing messages
if (smsDataProvider.messageExists(address, body, key, mailbox_id)) {
if (smsDataProvider.messageExists(address, body, date, mailbox_id)) {
publishProgress(nb);
continue;
}
@ -106,22 +96,20 @@ public interface ASyncSMSRecovery {
ContentValues values = new ContentValues();
values.put(Telephony.Sms.ADDRESS, address);
values.put(Telephony.Sms.BODY, body);
values.put(Telephony.Sms.DATE, key);
values.put(Telephony.Sms.DATE, date);
values.put(Telephony.Sms.TYPE, type);
values.put(Telephony.Sms.SEEN, 1);
values.put(Telephony.Sms.READ, 1);
// @TODO verify message exists before inserting it
_context.getContentResolver().insert(Uri.parse(mailbox_id.getURI()), values);
nb++;
if ((nb % 5) == 0) {
if ((nb % 10) == 0) {
publishProgress(nb);
}
}
}
start = obj.getLong("last_id");
start = obj.getLastID();
if (!new ConnectivityMonitor(_context).isValid()) {
Log.e(ASyncSMSRecovery.TAG, "Restore connectivity problems, aborting");
@ -129,12 +117,10 @@ public interface ASyncSMSRecovery {
}
obj = client.retrieveSomeMessages(start, 500);
}
} catch (JSONException e) {
Log.e(ASyncSMSRecovery.TAG, "Missing last_id field!");
}
// Force this refresh to fix dates
_context.getContentResolver().delete(Uri.parse("content://sms/conversations/-1"), null, null);
_context.getContentResolver().delete(Uri.parse("content://sms/conversations/-1"),
null, null);
publishProgress(nb);

View File

@ -19,28 +19,94 @@ package fr.unix_experience.owncloud_sms.engine;
import android.accounts.Account;
import android.accounts.AccountManager;
import android.app.Activity;
import android.content.Context;
import android.os.AsyncTask;
import android.util.Log;
import org.json.JSONArray;
import android.widget.Toast;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.enums.OCSMSNotificationType;
import fr.unix_experience.owncloud_sms.exceptions.OCSyncException;
import fr.unix_experience.owncloud_sms.notifications.OCSMSNotificationUI;
import fr.unix_experience.owncloud_sms.prefs.OCSMSSharedPrefs;
import ncsmsgo.SmsBuffer;
public interface ASyncSMSSync {
class SyncTask extends AsyncTask<Void, Void, Void> {
public SyncTask(Context context, JSONArray smsList) {
public SyncTask(Activity context) {
_activity = context;
_context = context;
_smsList = smsList;
_smsBuffer = null;
}
public SyncTask(Context context, SmsBuffer buffer) {
_activity = null;
_context = context;
_smsBuffer = buffer;
}
@Override
protected Void doInBackground(Void... params) {
Log.i(ASyncSMSSync.TAG, "Starting background sync");
// If no smsBuffer given it's a full sync
if (_smsBuffer == null) {
doFullSync();
}
else {
performSync(_smsBuffer);
}
Log.i(ASyncSMSSync.TAG, "Stopping background sync");
return null;
}
private void doFullSync() {
OCSMSSharedPrefs prefs = new OCSMSSharedPrefs(_context);
long syncStartupDate = prefs.getLastMessageDate();
Log.i(ASyncSMSSync.TAG, "Current message date is " + syncStartupDate);
boolean shouldSync = true;
boolean hasSyncSomething = false;
AndroidSmsFetcher fetcher = new AndroidSmsFetcher(_context);
while (shouldSync) {
SmsBuffer smsBuffer = new SmsBuffer();
fetcher.bufferMessagesSinceDate(smsBuffer, syncStartupDate);
if (smsBuffer.empty()) {
if (_activity != null) {
final boolean syncComplete = hasSyncSomething;
_activity.runOnUiThread(new Runnable() {
public void run() {
Toast.makeText(_context,
_context.getString(syncComplete ? R.string.sync_complete : R.string.nothing_to_sync),
Toast.LENGTH_SHORT).show();
}
});
}
Log.i(ASyncSMSSync.TAG, "Finish syncAllMessages(): no more sms");
smsBuffer.clear();
shouldSync = false;
continue;
}
if (prefs.showSyncNotifications()) {
OCSMSNotificationUI.notify(_context, _context.getString(R.string.sync_title),
_context.getString(R.string.sync_inprogress), OCSMSNotificationType.SYNC);
}
try {
syncStartupDate = smsBuffer.getLastMessageDate();
performSync(smsBuffer);
hasSyncSomething = true;
} finally {
OCSMSNotificationUI.cancel(_context, OCSMSNotificationType.SYNC);
}
}
}
private void performSync(SmsBuffer smsBuffer) {
// Get ownCloud SMS account list
AccountManager _accountMgr = AccountManager.get(_context);
Account[] myAccountList = _accountMgr.getAccountsByType(_context.getString(R.string.account_type));
@ -49,24 +115,25 @@ public interface ASyncSMSSync {
for (Account element : myAccountList) {
try {
OCSMSOwnCloudClient _client = new OCSMSOwnCloudClient(_context, element);
_client.doPushRequest(_smsList);
OCSMSNotificationUI.cancel(_context);
// Fetch API version first to do some early verifications
Log.i(ASyncSMSSync.TAG, "Server API version: " + _client.getServerAPIVersion());
_client.doPushRequest(smsBuffer);
OCSMSNotificationUI.cancel(_context, OCSMSNotificationType.SYNC_FAILED);
} catch (IllegalStateException e) { // Fail to read account data
OCSMSNotificationUI.notify(_context, _context.getString(R.string.fatal_error),
e.getMessage(), OCSMSNotificationType.SYNC_FAILED.ordinal());
e.getMessage(), OCSMSNotificationType.SYNC_FAILED);
} catch (OCSyncException e) {
Log.e(ASyncSMSSync.TAG, _context.getString(e.getErrorId()));
OCSMSNotificationUI.notify(_context, _context.getString(R.string.fatal_error),
e.getMessage(), OCSMSNotificationType.SYNC_FAILED.ordinal());
e.getMessage(), OCSMSNotificationType.SYNC_FAILED);
}
}
OCSMSNotificationUI.cancel(_context);
Log.i(ASyncSMSSync.TAG, "Stopping background sync");
return null;
smsBuffer.clear();
}
private final SmsBuffer _smsBuffer;
private final Context _context;
private final JSONArray _smsList;
private final Activity _activity;
}
String TAG = ASyncSMSSync.class.getSimpleName();

View File

@ -23,14 +23,13 @@ import android.util.Log;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import fr.unix_experience.owncloud_sms.enums.MailboxID;
import fr.unix_experience.owncloud_sms.providers.SmsDataProvider;
import ncsmsgo.SmsBuffer;
public class AndroidSmsFetcher {
public AndroidSmsFetcher(Context ct) {
_lastMsgDate = (long) 0;
_context = ct;
_existingInboxMessages = null;
@ -38,13 +37,35 @@ public class AndroidSmsFetcher {
_existingDraftsMessages = null;
}
void fetchAllMessages(JSONArray result) {
void fetchAllMessages(SmsBuffer result) {
bufferMailboxMessages(result, MailboxID.INBOX);
bufferMailboxMessages(result, MailboxID.SENT);
bufferMailboxMessages(result, MailboxID.DRAFTS);
}
private void bufferMailboxMessages(JSONArray result, MailboxID mbID) {
private void readMailBox(Cursor c, SmsBuffer smsBuffer, MailboxID mbID) {
do {
SmsEntry entry = new SmsEntry();
for (int idx = 0; idx < c.getColumnCount(); idx++) {
handleProviderColumn(c, idx, entry);
}
// Mailbox ID is required by server
entry.mailboxId = mbID.ordinal();
smsBuffer.push(entry.id,
mbID.ordinal(),
entry.type,
entry.date,
entry.address,
entry.body,
entry.read ? "true" : "false",
entry.seen ? "true" : "false");
}
while (c.moveToNext());
}
private void bufferMailboxMessages(SmsBuffer smsBuffer, MailboxID mbID) {
if ((_context == null)) {
return;
}
@ -64,30 +85,14 @@ public class AndroidSmsFetcher {
}
// Reading mailbox
do {
JSONObject entry = new JSONObject();
try {
for (int idx = 0; idx < c.getColumnCount(); idx++) {
handleProviderColumn(c, idx, entry);
}
// Mailbox ID is required by server
entry.put("mbox", mbID.ordinal());
result.put(entry);
} catch (JSONException e) {
Log.e(AndroidSmsFetcher.TAG, "JSON Exception when reading SMS Mailbox", e);
}
}
while (c.moveToNext());
readMailBox(c, smsBuffer, mbID);
Log.i(AndroidSmsFetcher.TAG, c.getCount() + " messages read from " + mbID.getURI());
c.close();
}
// Used by Content Observer
public JSONArray getLastMessage(MailboxID mbID) {
public SmsBuffer getLastMessage(MailboxID mbID) {
if ((_context == null)) {
return null;
}
@ -99,10 +104,9 @@ public class AndroidSmsFetcher {
}
// We create a list of strings to store results
JSONArray results = new JSONArray();
JSONObject entry = new JSONObject();
SmsEntry entry = new SmsEntry();
SmsBuffer results = new SmsBuffer();
try {
Integer mboxId = -1;
for (int idx = 0; idx < c.getColumnCount(); idx++) {
Integer rid = handleProviderColumn(c, idx, entry);
@ -116,12 +120,15 @@ public class AndroidSmsFetcher {
* mboxId is greater than server mboxId by 1 because types
* aren't indexed in the same mean
*/
entry.put("mbox", (mboxId - 1));
results.put(entry);
} catch (JSONException e) {
Log.e(AndroidSmsFetcher.TAG, "JSON Exception when reading SMS Mailbox", e);
}
entry.mailboxId = mboxId - 1;
results.push(entry.id,
mbID.ordinal(),
entry.type,
entry.date,
entry.address,
entry.body,
entry.read ? "true" : "false",
entry.seen ? "true" : "false");
c.close();
@ -129,14 +136,14 @@ public class AndroidSmsFetcher {
}
// Used by ConnectivityChanged Event
public void bufferMessagesSinceDate(JSONArray result, Long sinceDate) {
bufferMessagesSinceDate(result, MailboxID.INBOX, sinceDate);
bufferMessagesSinceDate(result, MailboxID.SENT, sinceDate);
bufferMessagesSinceDate(result, MailboxID.DRAFTS, sinceDate);
public void bufferMessagesSinceDate(SmsBuffer smsBuffer, Long sinceDate) {
bufferMessagesSinceDate(smsBuffer, MailboxID.INBOX, sinceDate);
bufferMessagesSinceDate(smsBuffer, MailboxID.SENT, sinceDate);
bufferMessagesSinceDate(smsBuffer, MailboxID.DRAFTS, sinceDate);
}
// Used by ConnectivityChanged Event
private void bufferMessagesSinceDate(JSONArray result, MailboxID mbID, Long sinceDate) {
private void bufferMessagesSinceDate(SmsBuffer smsBuffer, MailboxID mbID, Long sinceDate) {
Log.i(AndroidSmsFetcher.TAG, "bufferMessagesSinceDate for " + mbID.toString() + " sinceDate " + sinceDate.toString());
if ((_context == null)) {
return;
@ -150,37 +157,23 @@ public class AndroidSmsFetcher {
return;
}
do {
JSONObject entry = new JSONObject();
try {
for (int idx = 0; idx < c.getColumnCount(); idx++) {
handleProviderColumn(c, idx, entry);
}
// Mailbox ID is required by server
entry.put("mbox", mbID.ordinal());
result.put(entry);
} catch (JSONException e) {
Log.e(AndroidSmsFetcher.TAG, "JSON Exception when reading SMS Mailbox", e);
}
}
while (c.moveToNext());
// Read Mailbox
readMailBox(c, smsBuffer, mbID);
Log.i(AndroidSmsFetcher.TAG, c.getCount() + " messages read from " + mbID.getURI());
c.close();
}
private Integer handleProviderColumn(Cursor c, int idx, JSONObject entry) throws JSONException {
private Integer handleProviderColumn(Cursor c, int idx, SmsEntry entry) {
String colName = c.getColumnName(idx);
// Id column is must be an integer
switch (colName) {
case "_id":
entry.put(colName, c.getInt(idx));
entry.id = c.getInt(idx);
break;
case "type":
entry.put(colName, c.getInt(idx));
entry.type = c.getInt(idx);
return c.getInt(idx);
/* For debug purpose
case "length(address)":
@ -188,19 +181,22 @@ public class AndroidSmsFetcher {
break;*/
// Seen and read must be pseudo boolean
case "read":
entry.read = (c.getInt(idx) > 0);
break;
case "seen":
entry.put(colName, (c.getInt(idx) > 0) ? "true" : "false");
entry.seen = (c.getInt(idx) > 0);
break;
case "date":
// Special case for date, we need to record last without searching
Long tmpDate = c.getLong(idx);
if (tmpDate > _lastMsgDate) {
_lastMsgDate = tmpDate;
}
entry.put(colName, c.getString(idx));
entry.date = c.getLong(idx);
break;
case "address":
entry.address = c.getString(idx);
break;
case "body":
entry.body = c.getString(idx);
break;
default:
entry.put(colName, c.getString(idx));
// Unhandled column
break;
}
@ -249,16 +245,10 @@ public class AndroidSmsFetcher {
_existingDraftsMessages = draftMessages;
}
Long getLastMessageDate() {
return _lastMsgDate;
}
private final Context _context;
private JSONArray _existingInboxMessages;
private JSONArray _existingSentMessages;
private JSONArray _existingDraftsMessages;
private Long _lastMsgDate;
private static final String TAG = AndroidSmsFetcher.class.getSimpleName();
}

View File

@ -1,93 +0,0 @@
package fr.unix_experience.owncloud_sms.engine;
/*
* Copyright (c) 2014-2016, Loic Blot <loic.blot@unix-experience.fr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
import android.content.Context;
import android.net.Uri;
import com.owncloud.android.lib.common.OwnCloudClient;
import com.owncloud.android.lib.common.OwnCloudClientFactory;
import com.owncloud.android.lib.common.OwnCloudCredentialsFactory;
import org.apache.commons.httpclient.HttpMethod;
import org.apache.commons.httpclient.methods.GetMethod;
import org.apache.commons.httpclient.methods.PostMethod;
import org.apache.commons.httpclient.methods.StringRequestEntity;
import java.io.IOException;
class HTTPRequestBuilder {
private final OwnCloudClient _ocClient;
// API v1 calls
private static final String OC_GET_VERSION = "/index.php/apps/ocsms/get/apiversion?format=json";
private static final String OC_GET_ALL_SMS_IDS = "/index.php/apps/ocsms/get/smsidlist?format=json";
private static final String OC_GET_LAST_MSG_TIMESTAMP = "/index.php/apps/ocsms/get/lastmsgtime?format=json";
private static final String OC_PUSH_ROUTE = "/index.php/apps/ocsms/push?format=json";
// API v2 calls
private static final String OC_V2_GET_PHONELIST = "/index.php/apps/ocsms/api/v2/phones/list?format=json";
private static final String OC_V2_GET_MESSAGES ="/index.php/apps/ocsms/api/v2/messages/[START]/[LIMIT]?format=json";
private static final String OC_V2_GET_MESSAGES_PHONE ="/index.php/apps/ocsms/api/v2/messages/[PHONENUMBER]/[START]/[LIMIT]?format=json";
private static final String OC_V2_GET_MESSAGES_SENDQUEUE = "/index.php/apps/ocsms/api/v2/messages/sendqueue?format=json";
HTTPRequestBuilder(Context context, Uri serverURI, String accountName, String accountPassword) {
_ocClient = OwnCloudClientFactory.createOwnCloudClient(
serverURI, context, true);
// Set basic credentials
_ocClient.setCredentials(
OwnCloudCredentialsFactory.newBasicCredentials(accountName, accountPassword)
);
}
private GetMethod get(String oc_call) {
GetMethod get = new GetMethod(_ocClient.getBaseUri() + oc_call);
get.addRequestHeader("OCS-APIREQUEST", "true");
return get;
}
GetMethod getAllSmsIds() {
return get(HTTPRequestBuilder.OC_GET_ALL_SMS_IDS);
}
public GetMethod getVersion() {
return get(HTTPRequestBuilder.OC_GET_VERSION);
}
PostMethod pushSms(StringRequestEntity ent) {
PostMethod post = new PostMethod(_ocClient.getBaseUri() + HTTPRequestBuilder.OC_PUSH_ROUTE);
post.addRequestHeader("OCS-APIREQUEST", "true");
post.setRequestEntity(ent);
return post;
}
GetMethod getPhoneList() {
return get(HTTPRequestBuilder.OC_V2_GET_PHONELIST);
}
GetMethod getMessages(Long start, Integer limit) {
return get(HTTPRequestBuilder.OC_V2_GET_MESSAGES.
replace("[START]", start.toString()).replace("[LIMIT]", limit.toString()));
}
int execute(HttpMethod req) throws IOException {
return _ocClient.executeMethod(req);
}
}

View File

@ -0,0 +1,114 @@
package fr.unix_experience.owncloud_sms.engine;
/*
* Copyright (c) 2014-2016, Loic Blot <loic.blot@unix-experience.fr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
import android.content.Context;
import android.util.Pair;
import java.net.URL;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.enums.OCSyncErrorType;
import fr.unix_experience.owncloud_sms.exceptions.OCSyncException;
import fr.unix_experience.owncloud_sms.providers.AndroidVersionProvider;
import ncsmsgo.SmsBuffer;
import ncsmsgo.SmsHTTPClient;
import ncsmsgo.SmsIDListResponse;
import ncsmsgo.SmsMessagesResponse;
import ncsmsgo.SmsPhoneListResponse;
import ncsmsgo.SmsPushResponse;
public class OCHttpClient {
private SmsHTTPClient _smsHttpClient;
public OCHttpClient(Context context, URL serverURL, String accountName, String accountPassword) {
_smsHttpClient = new SmsHTTPClient();
// @TODO: at a point add a flag to permit insecure connections somewhere instead of trusting them
_smsHttpClient.init(serverURL.toString(), new AndroidVersionProvider(context).getVersionCode(),
accountName, accountPassword, false);
}
private void handleEarlyHTTPStatus(int httpStatus) throws OCSyncException {
switch (httpStatus) {
case 403: {
// Authentication failed
throw new OCSyncException(R.string.err_sync_auth_failed, OCSyncErrorType.AUTH);
}
}
}
public String getLastError() {
return _smsHttpClient.getLastError();
}
Pair<Integer, SmsIDListResponse> getAllSmsIds() throws OCSyncException {
SmsIDListResponse silr = _smsHttpClient.doGetSmsIDList();
int httpStatus = (int) _smsHttpClient.getLastHTTPStatus();
handleEarlyHTTPStatus(httpStatus);
return new Pair<>(httpStatus, silr);
}
// Perform the GoLang doVersionCall and handle return
public Pair<Integer, Integer> getVersion() throws OCSyncException {
Integer serverAPIVersion = (int) _smsHttpClient.doVersionCall();
int httpStatus = (int) _smsHttpClient.getLastHTTPStatus();
handleEarlyHTTPStatus(httpStatus);
// If last status is not 200, send the wrong status now
if (httpStatus != 200) {
return new Pair<>(httpStatus, 0);
}
if (serverAPIVersion > 0) {
return new Pair<>(200, serverAPIVersion);
}
else if (serverAPIVersion == 0) {
// Return default version
return new Pair<>(200, 1);
}
else if (serverAPIVersion == -1) {
// This return code from API means I/O error
throw new OCSyncException(R.string.err_sync_http_request_ioexception, OCSyncErrorType.IO);
}
else {
throw new OCSyncException(R.string.err_sync_http_request_returncode_unhandled, OCSyncErrorType.SERVER_ERROR);
}
}
Pair<Integer, SmsPushResponse> pushSms(SmsBuffer smsBuf) throws OCSyncException {
SmsPushResponse spr = _smsHttpClient.doPushCall(smsBuf);
int httpStatus = (int) _smsHttpClient.getLastHTTPStatus();
handleEarlyHTTPStatus(httpStatus);
return new Pair<>(httpStatus, spr);
}
Pair<Integer, SmsPhoneListResponse> getPhoneList() throws OCSyncException {
SmsPhoneListResponse splr = _smsHttpClient.doGetPhoneList();
int httpStatus = (int) _smsHttpClient.getLastHTTPStatus();
handleEarlyHTTPStatus(httpStatus);
return new Pair<>(httpStatus, splr);
}
Pair<Integer, SmsMessagesResponse> getMessages(Long start, Integer limit) throws OCSyncException {
SmsMessagesResponse smr = _smsHttpClient.doGetMessagesCall(start, limit);
int httpStatus = (int) _smsHttpClient.getLastHTTPStatus();
handleEarlyHTTPStatus(httpStatus);
return new Pair<>(httpStatus, smr);
}
}

View File

@ -20,26 +20,21 @@ package fr.unix_experience.owncloud_sms.engine;
import android.accounts.Account;
import android.accounts.AccountManager;
import android.content.Context;
import android.net.Uri;
import android.util.Log;
import android.util.Pair;
import org.apache.commons.httpclient.HttpException;
import org.apache.commons.httpclient.HttpMethod;
import org.apache.commons.httpclient.methods.PostMethod;
import org.apache.commons.httpclient.methods.StringRequestEntity;
import org.apache.http.HttpStatus;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.ConnectException;
import java.net.MalformedURLException;
import java.net.URL;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.enums.OCSyncErrorType;
import fr.unix_experience.owncloud_sms.exceptions.OCSyncException;
import fr.unix_experience.owncloud_sms.prefs.OCSMSSharedPrefs;
import ncsmsgo.SmsBuffer;
import ncsmsgo.SmsIDListResponse;
import ncsmsgo.SmsMessagesResponse;
import ncsmsgo.SmsPhoneListResponse;
import ncsmsgo.SmsPushResponse;
@SuppressWarnings("deprecation")
public class OCSMSOwnCloudClient {
@ -56,327 +51,104 @@ public class OCSMSOwnCloudClient {
throw new IllegalStateException(context.getString(R.string.err_sync_account_unparsable));
}
Uri serverURI = Uri.parse(ocURI);
_http = new HTTPRequestBuilder(context, serverURI,
accountManager.getUserData(account, "ocLogin"),
try {
URL serverURL = new URL(ocURI);
_http = new OCHttpClient(context,
serverURL, accountManager.getUserData(account, "ocLogin"),
accountManager.getPassword(account));
_connectivityMonitor = new ConnectivityMonitor(_context);
} catch (MalformedURLException e) {
throw new IllegalStateException(context.getString(R.string.err_sync_account_unparsable));
}
}
public Integer getServerAPIVersion() throws OCSyncException {
doHttpRequest(_http.getVersion(), true);
if (_jsonQueryBuffer == null) {
// Return default version
return 1;
}
try {
_serverAPIVersion = _jsonQueryBuffer.getInt("version");
}
catch (JSONException e) {
Log.e(OCSMSOwnCloudClient.TAG, "No version received from server, assuming version 1", e);
_serverAPIVersion = 1;
}
Pair<Integer, Integer> vPair = _http.getVersion();
_serverAPIVersion = vPair.second;
if (vPair.first == 200 && _serverAPIVersion > 0) {
return _serverAPIVersion;
}
JSONArray getServerPhoneNumbers() throws OCSyncException {
doHttpRequest(_http.getPhoneList(), true);
if (_jsonQueryBuffer == null) {
return 0;
}
SmsPhoneListResponse getServerPhoneNumbers() throws OCSyncException {
Pair<Integer, SmsPhoneListResponse> response = _http.getPhoneList();
if (response.second == null || response.first != 200) {
return null;
}
try {
return _jsonQueryBuffer.getJSONArray("phoneList");
} catch (JSONException e) {
Log.e(OCSMSOwnCloudClient.TAG, "No phonelist received from server, empty it", e);
return null;
}
return response.second;
}
public void doPushRequest(JSONArray smsList) throws OCSyncException {
/**
public void doPushRequest(SmsBuffer smsBuffer) throws OCSyncException {
/*
* If we need other API push, set it here
*/
switch (_serverAPIVersion) {
case 1:
default: doPushRequestV1(smsList); break;
default: doPushRequestV1(smsBuffer); break;
}
}
private void doPushRequestV1(JSONArray smsList) throws OCSyncException {
// We need to save this date as a step for connectivity change
Long lastMsgDate = (long) 0;
if (smsList == null) {
doHttpRequest(_http.getAllSmsIds());
if (_jsonQueryBuffer == null) {
private void doPushRequestV1(SmsBuffer smsBuffer) throws OCSyncException {
if (smsBuffer == null) {
Pair<Integer, SmsIDListResponse> response = _http.getAllSmsIds();
if (response.second == null) {
return;
}
JSONObject smsBoxes = new JSONObject();
JSONArray inboxSmsList = null, sentSmsList = null, draftsSmsList = null;
try {
smsBoxes = _jsonQueryBuffer.getJSONObject("smslist");
} catch (JSONException e) {
try {
_jsonQueryBuffer.getJSONArray("smslist");
} catch (JSONException e2) {
Log.e(OCSMSOwnCloudClient.TAG, "Invalid datas received from server (doPushRequest, get SMS list)", e);
throw new OCSyncException(R.string.err_sync_get_smslist, OCSyncErrorType.PARSE);
}
// Create new SmsBuffer to get results
smsBuffer = new SmsBuffer();
}
try {
inboxSmsList = smsBoxes.getJSONArray("inbox");
} catch (JSONException e) {
Log.i(OCSMSOwnCloudClient.TAG, "No inbox Sms received from server (doPushRequest, get SMS list)");
}
try {
sentSmsList = smsBoxes.getJSONArray("sent");
} catch (JSONException e) {
Log.i(OCSMSOwnCloudClient.TAG, "No sent Sms received from server (doPushRequest, get SMS list)");
}
try {
draftsSmsList = smsBoxes.getJSONArray("drafts");
} catch (JSONException e) {
Log.i(OCSMSOwnCloudClient.TAG, "No drafts Sms received from server (doPushRequest, get SMS list)");
}
AndroidSmsFetcher fetcher = new AndroidSmsFetcher(_context);
fetcher.setExistingInboxMessages(inboxSmsList);
fetcher.setExistingSentMessages(sentSmsList);
fetcher.setExistingDraftsMessages(draftsSmsList);
smsList = new JSONArray();
fetcher.fetchAllMessages(smsList);
// Get maximum message date present in smsList to keep a step when connectivity changes
lastMsgDate = fetcher.getLastMessageDate();
}
if (smsList.length() == 0) {
if (smsBuffer.empty()) {
Log.i(OCSMSOwnCloudClient.TAG, "No new SMS to sync, sync done");
return;
}
PostMethod post = createPushRequest(smsList);
if (post == null) {
Log.e(OCSMSOwnCloudClient.TAG,"Push request for POST is null");
throw new OCSyncException(R.string.err_sync_craft_http_request, OCSyncErrorType.IO);
}
Pair<Integer, SmsPushResponse> response = _http.pushSms(smsBuffer);
doHttpRequest(post);
if (_jsonQueryBuffer == null) {
Log.e(OCSMSOwnCloudClient.TAG,"Request failed. It doesn't return a valid JSON Object");
if (response.second == null) {
Log.e(OCSMSOwnCloudClient.TAG,"Push request failed. GoLang response is empty.");
throw new OCSyncException(R.string.err_sync_push_request, OCSyncErrorType.IO);
}
Boolean pushStatus;
String pushMessage;
try {
pushStatus = _jsonQueryBuffer.getBoolean("status");
pushMessage = _jsonQueryBuffer.getString("msg");
}
catch (JSONException e) {
Log.e(OCSMSOwnCloudClient.TAG, "Invalid datas received from server", e);
throw new OCSyncException(R.string.err_sync_push_request_resp, OCSyncErrorType.PARSE);
}
// Push was OK, we can save the lastMessageDate which was saved to server
(new OCSMSSharedPrefs(_context)).setLastMessageDate(lastMsgDate);
(new OCSMSSharedPrefs(_context)).setLastMessageDate(smsBuffer.getLastMessageDate());
Log.i(OCSMSOwnCloudClient.TAG, "SMS Push request said: status " + pushStatus + " - " + pushMessage);
Log.i(OCSMSOwnCloudClient.TAG, "SMS Push request said: status " +
response.second.getStatus() + " - " + response.second.getMessage());
Log.i(OCSMSOwnCloudClient.TAG, "LastMessageDate set to: " + smsBuffer.getLastMessageDate());
}
private PostMethod createPushRequest(JSONArray smsList) throws OCSyncException {
JSONObject obj = createPushJSONObject(smsList);
if (obj == null) {
return null;
}
StringRequestEntity ent = createJSONRequestEntity(obj);
if (ent == null) {
return null;
}
return _http.pushSms(ent);
}
private JSONObject createPushJSONObject(JSONArray smsList) throws OCSyncException {
if (smsList == null) {
Log.e(OCSMSOwnCloudClient.TAG,"NULL SMS List");
throw new OCSyncException(R.string.err_sync_create_json_null_smslist, OCSyncErrorType.IO);
}
JSONObject reqJSON = new JSONObject();
try {
reqJSON.put("smsDatas", smsList);
reqJSON.put("smsCount", smsList.length());
} catch (JSONException e) {
Log.e(OCSMSOwnCloudClient.TAG,"JSON Exception when creating JSON request object");
throw new OCSyncException(R.string.err_sync_create_json_put_smslist, OCSyncErrorType.PARSE);
}
return reqJSON;
}
private StringRequestEntity createJSONRequestEntity(JSONObject obj) throws OCSyncException {
StringRequestEntity requestEntity;
try {
requestEntity = new StringRequestEntity(
obj.toString(),
"application/json",
"UTF-8");
} catch (UnsupportedEncodingException e) {
Log.e(OCSMSOwnCloudClient.TAG,"Unsupported encoding when generating request");
throw new OCSyncException(R.string.err_sync_create_json_request_encoding, OCSyncErrorType.PARSE);
}
return requestEntity;
}
JSONObject retrieveSomeMessages(Long start, Integer limit) {
SmsMessagesResponse retrieveSomeMessages(Long start, Integer limit) {
// This is not allowed by server
if (limit > OCSMSOwnCloudClient.SERVER_RECOVERY_MSG_LIMIT) {
Log.e(OCSMSOwnCloudClient.TAG, "Message recovery limit exceeded");
return null;
}
Pair<Integer, SmsMessagesResponse> response;
try {
doHttpRequest(_http.getMessages(start, limit));
response = _http.getMessages(start, limit);
} catch (OCSyncException e) {
_jsonQueryBuffer = null;
Log.e(OCSMSOwnCloudClient.TAG, "Request failed.");
return null;
}
if (!_jsonQueryBuffer.has("messages") || !_jsonQueryBuffer.has("last_id")) {
if (response.second == null) {
Log.e(OCSMSOwnCloudClient.TAG,
"Invalid response received from server, either messages or last_id field is missing.");
return null;
}
return _jsonQueryBuffer;
return response.second;
}
private void doHttpRequest(HttpMethod req) throws OCSyncException {
doHttpRequest(req, false);
}
// skipError permit to skip invalid JSON datas
private void doHttpRequest(HttpMethod req, Boolean skipError) throws OCSyncException {
// Reinit the queryBuffer
_jsonQueryBuffer = null;
int status = 0;
// We try maximumHttpReqTries because sometimes network is slow or unstable
int tryNb = 0;
while (tryNb < OCSMSOwnCloudClient.maximumHttpReqTries) {
tryNb++;
if (!_connectivityMonitor.isValid()) {
if (tryNb == OCSMSOwnCloudClient.maximumHttpReqTries) {
req.releaseConnection();
throw new OCSyncException(R.string.err_sync_no_connection_available, OCSyncErrorType.IO);
}
continue;
}
try {
status = _http.execute(req);
Log.i(OCSMSOwnCloudClient.TAG, "HTTP Request done at try " + tryNb);
// Force loop exit
tryNb = OCSMSOwnCloudClient.maximumHttpReqTries;
} catch (ConnectException e) {
Log.e(OCSMSOwnCloudClient.TAG, "Unable to perform a connection to ownCloud instance", e);
// If it's the last try
if (tryNb == OCSMSOwnCloudClient.maximumHttpReqTries) {
req.releaseConnection();
throw new OCSyncException(R.string.err_sync_http_request_connect, OCSyncErrorType.IO);
}
} catch (HttpException e) {
Log.e(OCSMSOwnCloudClient.TAG, "Unable to perform a connection to ownCloud instance", e);
// If it's the last try
if (tryNb == OCSMSOwnCloudClient.maximumHttpReqTries) {
req.releaseConnection();
throw new OCSyncException(R.string.err_sync_http_request_httpexception, OCSyncErrorType.IO);
}
} catch (IOException e) {
Log.e(OCSMSOwnCloudClient.TAG, "Unable to perform a connection to ownCloud instance", e);
// If it's the last try
if (tryNb == OCSMSOwnCloudClient.maximumHttpReqTries) {
req.releaseConnection();
throw new OCSyncException(R.string.err_sync_http_request_ioexception, OCSyncErrorType.IO);
}
}
}
if (status == HttpStatus.SC_OK) {
String response;
try {
response = req.getResponseBodyAsString();
} catch (IOException e) {
Log.e(OCSMSOwnCloudClient.TAG, "Unable to parse server response", e);
throw new OCSyncException(R.string.err_sync_http_request_resp, OCSyncErrorType.IO);
}
// Parse the response
try {
_jsonQueryBuffer = new JSONObject(response);
} catch (JSONException e) {
if (!skipError) {
if (response.contains("ownCloud") && response.contains("DOCTYPE")) {
Log.e(OCSMSOwnCloudClient.TAG, "OcSMS app not enabled or ownCloud upgrade is required");
throw new OCSyncException(R.string.err_sync_ocsms_not_installed_or_oc_upgrade_required,
OCSyncErrorType.SERVER_ERROR);
}
else {
Log.e(OCSMSOwnCloudClient.TAG, "Unable to parse server response", e);
throw new OCSyncException(R.string.err_sync_http_request_parse_resp, OCSyncErrorType.PARSE);
}
}
}
} else if (status == HttpStatus.SC_FORBIDDEN) {
// Authentication failed
throw new OCSyncException(R.string.err_sync_auth_failed, OCSyncErrorType.AUTH);
} else {
// Unk error
String response;
try {
response = req.getResponseBodyAsString();
} catch (IOException e) {
Log.e(OCSMSOwnCloudClient.TAG, "Unable to parse server response", e);
throw new OCSyncException(R.string.err_sync_http_request_resp, OCSyncErrorType.PARSE);
}
Log.e(OCSMSOwnCloudClient.TAG, "Server set unhandled HTTP return code " + status);
if (response != null) {
Log.e(OCSMSOwnCloudClient.TAG, "Status code: " + status + ". Response message: " + response);
} else {
Log.e(OCSMSOwnCloudClient.TAG, "Status code: " + status);
}
throw new OCSyncException(R.string.err_sync_http_request_returncode_unhandled, OCSyncErrorType.SERVER_ERROR);
}
}
private static final int maximumHttpReqTries = 3;
private final HTTPRequestBuilder _http;
private final OCHttpClient _http;
private final Context _context;
private final ConnectivityMonitor _connectivityMonitor;
private Integer _serverAPIVersion;
private JSONObject _jsonQueryBuffer;
private static final String TAG = OCSMSOwnCloudClient.class.getSimpleName();

View File

@ -0,0 +1,29 @@
package fr.unix_experience.owncloud_sms.engine;
/*
* Copyright (c) 2014-2016, Loic Blot <loic.blot@unix-experience.fr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
public class SmsEntry {
public int id;
public int mailboxId;
public int type;
public boolean read;
public boolean seen;
public long date;
public String address;
public String body;
}

View File

@ -0,0 +1,48 @@
package fr.unix_experience.owncloud_sms.enums;
import android.app.NotificationManager;
import android.os.Build;
import android.support.annotation.StringRes;
import fr.unix_experience.owncloud_sms.R;
public enum OCSMSNotificationChannel {
DEFAULT("OCSMS_DEFAULT", R.string.notification_channel_name_default, null),
SYNC("OCSMS_SYNC", R.string.notification_channel_name_sync, null);
static {
// well, that's a bit of a hack :/
// can be inlined in the future
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
DEFAULT.importance = NotificationManager.IMPORTANCE_DEFAULT;
SYNC.importance = NotificationManager.IMPORTANCE_LOW;
}
}
private final String channelId;
private final int nameResId;
private final Integer descResId;
private int importance;
OCSMSNotificationChannel(String channelId, @StringRes int nameResId, @StringRes Integer descResId) {
this.channelId = channelId;
this.nameResId = nameResId;
this.descResId = descResId;
}
public String getChannelId() {
return channelId;
}
public int getNameResId() {
return nameResId;
}
public Integer getDescResId() {
return descResId;
}
public int getImportance() {
return importance;
}
}

View File

@ -18,8 +18,23 @@ package fr.unix_experience.owncloud_sms.enums;
*/
public enum OCSMSNotificationType {
SYNC,
SYNC_FAILED,
DEBUG,
PERMISSION,
SYNC(OCSMSNotificationChannel.SYNC, 0),
SYNC_FAILED(OCSMSNotificationChannel.DEFAULT, 1),
PERMISSION(OCSMSNotificationChannel.DEFAULT, 2);
private final OCSMSNotificationChannel channel;
private final int notificationId;
OCSMSNotificationType(OCSMSNotificationChannel channel, int notificationId) {
this.channel = channel;
this.notificationId = notificationId;
}
public OCSMSNotificationChannel getChannel() {
return channel;
}
public int getNotificationId() {
return notificationId;
}
}

View File

@ -1,19 +1,33 @@
package fr.unix_experience.owncloud_sms.notifications;
import android.annotation.TargetApi;
/*
* Copyright (c) 2014-2017, Loic Blot <loic.blot@unix-experience.fr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
import android.app.Notification;
import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.content.res.Resources;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Build;
import android.support.v4.app.NotificationCompat;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.enums.OCSMSNotificationChannel;
import fr.unix_experience.owncloud_sms.enums.OCSMSNotificationType;
/**
* Helper class for showing and canceling ui
@ -28,29 +42,34 @@ public class OCSMSNotificationUI {
*/
private static final String NOTIFICATION_TAG = "OCSMS_NOTIFICATION";
public static void notify(Context context, String titleString,
String contentString, OCSMSNotificationType type) {
notify(context, titleString, contentString,
type.getChannel().getChannelId(), type.getNotificationId());
}
/**
* Shows the notification, or updates a previously shown notification of
* this type, with the given parameters.
*
* @see #cancel(Context)
* @see #cancel(Context, OCSMSNotificationType)
*/
public static void notify(Context context, String titleString,
String contentString, int number) {
public static void notify(Context context, String titleString, String contentString,
String channelId, int notificationId) {
Resources res = context.getResources();
// This image is used as the notification's large icon (thumbnail).
// TODO: Remove this if your notification has no relevant thumbnail.
Bitmap picture = BitmapFactory.decodeResource(res, R.drawable.ic_launcher);
// Bitmap picture = BitmapFactory.decodeResource(res, R.mipmap.ic_launcher);
String ticker = (titleString.length() > 20) ? titleString.substring(0, 20) : titleString;
// String ticker = (titleString.length() > 20) ? titleString.substring(0, 20) : titleString;
String title = res.getString(R.string.ui_notification_title_template, titleString);
NotificationCompat.Builder builder = new NotificationCompat.Builder(context)
NotificationCompat.Builder builder = new NotificationCompat.Builder(context, channelId)
// Set appropriate defaults for the notification light, sound,
// and vibration.
.setSmallIcon(R.drawable.ic_launcher)
.setSmallIcon(R.drawable.notification_icon)
.setContentTitle(title)
.setContentText(contentString)
@ -60,48 +79,55 @@ public class OCSMSNotificationUI {
// 4.1 or later)
.setPriority(NotificationCompat.PRIORITY_DEFAULT)
// Provide a large icon, shown with the notification in the
// notification drawer on devices running Android 3.0 or later.
.setLargeIcon(picture)
// Set ticker text (preview) information for this notification.
//.setTicker(ticker)
// Show a number. This is useful when stacking notifications of
// a single type.
.setNumber(number)
.setStyle(new NotificationCompat.BigTextStyle()
.bigText(contentString)
.setBigContentTitle(title)
.setSummaryText(titleString))
.setAutoCancel(true);
.setAutoCancel(true)
.setColor(context.getResources().getColor(R.color.oc_primary));
OCSMSNotificationUI.notify(context, builder.build());
notify(context, builder.build(), notificationId);
}
@TargetApi(Build.VERSION_CODES.ECLAIR)
private static void notify(Context context, Notification notification) {
private static void notify(Context context, Notification notification, int notificationId) {
NotificationManager nm = (NotificationManager) context
.getSystemService(Context.NOTIFICATION_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ECLAIR) {
nm.notify(OCSMSNotificationUI.NOTIFICATION_TAG, 0, notification);
} else {
nm.notify(OCSMSNotificationUI.NOTIFICATION_TAG.hashCode(), notification);
}
createNotificationChannels(context, nm);
nm.notify(OCSMSNotificationUI.NOTIFICATION_TAG, notificationId, notification);
}
/**
* Cancels any notifications of this type previously shown using
* {@link #notify(Context, String, int)}.
* {@link #notify(Context, String, String, OCSMSNotificationType)}.
*/
@TargetApi(Build.VERSION_CODES.ECLAIR)
public static void cancel(Context context) {
public static void cancel(Context context, OCSMSNotificationType type) {
cancel(context, type.getNotificationId());
}
public static void cancel(Context context, int notificationId) {
NotificationManager nm = (NotificationManager) context
.getSystemService(Context.NOTIFICATION_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ECLAIR) {
nm.cancel(OCSMSNotificationUI.NOTIFICATION_TAG, 0);
} else {
nm.cancel(OCSMSNotificationUI.NOTIFICATION_TAG.hashCode());
nm.cancel(OCSMSNotificationUI.NOTIFICATION_TAG, notificationId);
}
private static void createNotificationChannels(Context context, NotificationManager nm) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
for (OCSMSNotificationChannel ocsmsChannel : OCSMSNotificationChannel.values()) {
NotificationChannel channel = new NotificationChannel(
ocsmsChannel.getChannelId(),
context.getString(ocsmsChannel.getNameResId()),
ocsmsChannel.getImportance());
if (ocsmsChannel.getDescResId() != null) {
channel.setDescription(context.getString(ocsmsChannel.getDescResId()));
}
nm.createNotificationChannel(channel);
}
}
}
}

View File

@ -25,8 +25,6 @@ import android.database.ContentObserver;
import android.os.Handler;
import android.util.Log;
import org.json.JSONArray;
import fr.unix_experience.owncloud_sms.R;
import fr.unix_experience.owncloud_sms.engine.ASyncSMSSync;
import fr.unix_experience.owncloud_sms.engine.AndroidSmsFetcher;
@ -35,6 +33,7 @@ import fr.unix_experience.owncloud_sms.engine.OCSMSOwnCloudClient;
import fr.unix_experience.owncloud_sms.enums.MailboxID;
import fr.unix_experience.owncloud_sms.enums.PermissionID;
import fr.unix_experience.owncloud_sms.prefs.PermissionChecker;
import ncsmsgo.SmsBuffer;
public class SmsObserver extends ContentObserver implements ASyncSMSSync {
@ -60,13 +59,13 @@ public class SmsObserver extends ContentObserver implements ASyncSMSSync {
}
AndroidSmsFetcher fetcher = new AndroidSmsFetcher(_context);
JSONArray smsList = fetcher.getLastMessage(MailboxID.ALL);
SmsBuffer smsBuffer = fetcher.getLastMessage(MailboxID.ALL);
ConnectivityMonitor cMon = new ConnectivityMonitor(_context);
// Synchronize if network is valid and there are SMS
if (cMon.isValid() && (smsList != null)) {
new SyncTask(_context, smsList).execute();
if (cMon.isValid() && (smsBuffer != null)) {
new SyncTask(_context, smsBuffer).execute();
}
}

View File

@ -43,6 +43,10 @@ public class OCSMSSharedPrefs extends SharedPrefs {
return _sPrefs.getBoolean("push_on_receive", DefaultPrefs.pushOnReceive);
}
public Boolean showSyncNotifications() {
return _sPrefs.getBoolean("show_sync_notifications", DefaultPrefs.showSyncNotifications);
}
public Boolean syncInWifi() {
return _sPrefs.getBoolean("sync_wifi", DefaultPrefs.syncWifi);
}

View File

@ -67,7 +67,7 @@ public class PermissionChecker {
// For context only show a notification
OCSMSNotificationUI.notify(context, context.getString(R.string.notif_permission_required),
context.getString(R.string.notif_permission_required_content),
OCSMSNotificationType.PERMISSION.ordinal());
OCSMSNotificationType.PERMISSION);
return false;
}

View File

@ -0,0 +1,47 @@
package fr.unix_experience.owncloud_sms.providers;
import android.content.Context;
import android.content.pm.PackageManager;
/*
* Copyright (c) 2014-2017, Loic Blot <loic.blot@unix-experience.fr>
* All rights reserved.
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
public class AndroidVersionProvider {
private Context _context;
public AndroidVersionProvider(Context context) {
assert (context != null);
_context = context;
}
public String getVersionCode() {
try {
return _context.getPackageManager().
getPackageInfo(_context.getPackageName(), 0).versionName;
} catch (PackageManager.NameNotFoundException e) {
return "unknown version";
}
}
}

View File

@ -29,6 +29,17 @@ import fr.unix_experience.owncloud_sms.enums.MailboxID;
import fr.unix_experience.owncloud_sms.prefs.OCSMSSharedPrefs;
public class SmsDataProvider extends ContentProvider {
static String[] messageFields = {
"read",
"date",
"address",
"seen",
"body",
"_id",
"type",
//"length(address)" // For debug purposes
};
// WARNING: mandatory
public SmsDataProvider() {}
public SmsDataProvider (Context ct) {
@ -43,7 +54,7 @@ public class SmsDataProvider extends ContentProvider {
public Cursor query(String mailBox) {
return query(Uri.parse(mailBox),
new String[] { "read", "date", "address", "seen", "body", "_id", "type", },
SmsDataProvider.messageFields,
null, null, null
);
}
@ -52,16 +63,7 @@ public class SmsDataProvider extends ContentProvider {
public Cursor queryNonExistingMessages(String mailBox, String existingIds) {
Log.i(SmsDataProvider.TAG, "queryNonExistingMessages !");
if (!existingIds.isEmpty()) {
return query(Uri.parse(mailBox),
new String[] {
"read",
"date",
"address",
"seen",
"body",
"_id",
"type",
},
return query(Uri.parse(mailBox), SmsDataProvider.messageFields,
"_id NOT IN (" + existingIds + ")", null, null
);
}
@ -70,18 +72,8 @@ public class SmsDataProvider extends ContentProvider {
}
public Cursor queryMessagesSinceDate(String mailBox, Long sinceDate) {
return query(Uri.parse(mailBox),
new String[] {
"read",
"date",
"address",
"seen",
"body",
"_id",
"type",
//"length(address)" // For debug purposes
},
"date > ?", new String[] { sinceDate.toString() }, null
return query(Uri.parse(mailBox), SmsDataProvider.messageFields,
"date > ?", new String[] { sinceDate.toString() }, "date ASC"
);
}

View File

@ -31,6 +31,7 @@ import fr.unix_experience.owncloud_sms.enums.OCSMSNotificationType;
import fr.unix_experience.owncloud_sms.enums.OCSyncErrorType;
import fr.unix_experience.owncloud_sms.exceptions.OCSyncException;
import fr.unix_experience.owncloud_sms.notifications.OCSMSNotificationUI;
import fr.unix_experience.owncloud_sms.prefs.OCSMSSharedPrefs;
class SmsSyncAdapter extends AbstractThreadedSyncAdapter {
@ -41,8 +42,11 @@ class SmsSyncAdapter extends AbstractThreadedSyncAdapter {
@Override
public void onPerformSync(Account account, Bundle extras, String authority,
ContentProviderClient provider, SyncResult syncResult) {
if (new OCSMSSharedPrefs(getContext()).showSyncNotifications()) {
OCSMSNotificationUI.notify(getContext(), getContext().getString(R.string.sync_title),
getContext().getString(R.string.sync_inprogress), OCSMSNotificationType.SYNC.ordinal());
getContext().getString(R.string.sync_inprogress), OCSMSNotificationType.SYNC);
}
try {
OCSMSOwnCloudClient _client = new OCSMSOwnCloudClient(getContext(), account);
@ -52,14 +56,13 @@ class SmsSyncAdapter extends AbstractThreadedSyncAdapter {
// and push datas
_client.doPushRequest(null);
OCSMSNotificationUI.cancel(getContext());
OCSMSNotificationUI.cancel(getContext(), OCSMSNotificationType.SYNC_FAILED);
} catch (IllegalStateException e) {
OCSMSNotificationUI.notify(getContext(), getContext().getString(R.string.fatal_error),
e.getMessage(), OCSMSNotificationType.SYNC_FAILED.ordinal());
e.getMessage(), OCSMSNotificationType.SYNC_FAILED);
} catch (OCSyncException e) {
OCSMSNotificationUI.cancel(getContext());
OCSMSNotificationUI.notify(getContext(), getContext().getString(R.string.fatal_error),
getContext().getString(e.getErrorId()), OCSMSNotificationType.SYNC_FAILED.ordinal());
getContext().getString(e.getErrorId()), OCSMSNotificationType.SYNC_FAILED);
if (e.getErrorType() == OCSyncErrorType.IO) {
syncResult.stats.numIoExceptions++;
}
@ -72,6 +75,8 @@ class SmsSyncAdapter extends AbstractThreadedSyncAdapter {
else {
Log.w(SmsSyncAdapter.TAG, "onPerformSync: unhandled response");
}
} finally {
OCSMSNotificationUI.cancel(getContext(), OCSMSNotificationType.SYNC);
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 657 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 558 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 748 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1004 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 840 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 873 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 586 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 398 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 384 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 459 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 646 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 576 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 639 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 446 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 677 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 686 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 900 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 604 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 701 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 910 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 994 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

View File

@ -0,0 +1,28 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:aapt="http://schemas.android.com/aapt"
android:width="108dp"
android:height="108dp"
android:autoMirrored="true"
android:viewportWidth="1344"
android:viewportHeight="1344">
<path
android:fillType="evenOdd"
android:pathData="M0,0h1344v1344h-1344z"
android:strokeLineJoin="round">
<aapt:attr name="android:fillColor">
<gradient
android:endX="1344"
android:endY="1.2959057E-4"
android:startX="163.34073"
android:startY="1344"
android:type="linear">
<item
android:color="#FF0082C9"
android:offset="0" />
<item
android:color="#FF1CAFFF"
android:offset="1" />
</gradient>
</aapt:attr>
</path>
</vector>

View File

@ -0,0 +1,22 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="108dp"
android:height="108dp"
android:viewportWidth="816.5869"
android:viewportHeight="816.5869">
<group android:translateX="20.414673"
android:translateY="20.414673">
<group android:translateX="131.87878"
android:translateY="131.87878">
<path
android:pathData="m122.444,89.125c-18.459,0 -33.319,14.115 -33.319,31.648l0,143.74c0,17.533 14.86,31.648 33.319,31.648l38.249,0 152.528,126.434 -49.393,-126.434 125.728,0c18.459,0 33.319,-14.115 33.319,-31.648l0,-143.74c0,-17.533 -14.86,-31.648 -33.319,-31.648zM131.499,116.333 L380.501,116.333c9.089,0 16.405,5.854 16.405,13.126l0,8.539c0,7.271 -7.317,13.124 -16.405,13.124l-249.003,0c-9.089,0 -16.406,-5.852 -16.406,-13.124l0,-8.539c0,-7.271 7.317,-13.126 16.406,-13.126zM131.499,176.841 L380.501,176.841c9.089,0 16.405,5.854 16.405,13.126l0,8.537c0,7.271 -7.317,13.126 -16.405,13.126l-249.003,0c-9.089,0 -16.406,-5.854 -16.406,-13.126l0,-8.537c0,-7.271 7.317,-13.126 16.406,-13.126zM131.499,236.851 L221.453,236.851c9.089,0 16.405,7.988 16.405,15.259l0,6.404c0,7.271 -7.317,13.126 -16.405,13.126l-89.954,0c-9.089,0 -16.406,-5.854 -16.406,-13.126l0,-6.404c0,-7.271 7.317,-15.259 16.406,-15.259z"
android:strokeAlpha="0"
android:strokeLineJoin="miter"
android:strokeWidth="0.25007507"
android:fillColor="#ffffff"
android:strokeColor="#000000"
android:fillType="evenOdd"
android:fillAlpha="1"
android:strokeLineCap="butt"/>
</group>
</group>
</vector>

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.7 KiB

After

Width:  |  Height:  |  Size: 2.5 KiB

View File

@ -61,7 +61,9 @@
android:id="@+id/ocsms_logo"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:src="@drawable/login_logo"
android:paddingTop="24dp"
android:paddingBottom="36dp"
android:src="@drawable/logo"
android:contentDescription="@string/login_logo" />
<LinearLayout
@ -73,8 +75,9 @@
android:id="@+id/oc_protocol"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:entries="@array/protocol_array">
</Spinner>
android:layout_gravity="center_vertical"
android:popupBackground="@color/oc_primary"
android:entries="@array/protocol_array" />
<android.support.design.widget.TextInputLayout
android:layout_width="match_parent"
@ -84,6 +87,7 @@
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:hint="@string/prompt_serverURI"
android:textColor="@color/oc_white"
android:inputType="textUri" >
<requestFocus />
</EditText>
@ -99,9 +103,9 @@
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="@string/prompt_login"
android:textColor="@color/oc_white"
android:inputType="textEmailAddress"
android:maxLines="1"
android:singleLine="true" />
android:maxLines="1" />
</android.support.design.widget.TextInputLayout>
<android.support.design.widget.TextInputLayout
@ -117,7 +121,7 @@
android:imeOptions="actionUnspecified"
android:inputType="textPassword"
android:maxLines="1"
android:singleLine="true" />
android:textColor="@color/oc_white" />
</android.support.design.widget.TextInputLayout>
<com.dd.processbutton.iml.ActionProcessButton

View File

@ -34,9 +34,14 @@
android:fitsSystemWindows="true"
tools:openDrawer="start"
xmlns:app="http://schemas.android.com/apk/res-auto">
<RelativeLayout
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
android:layout_height="match_parent"
android:orientation="vertical">
<include
layout="@layout/toolbar" />
<TextView
android:id="@+id/main_title_welcome"
@ -44,7 +49,6 @@
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/ma_title_welcome"
android:gravity="top"
android:textAppearance="?android:attr/textAppearanceLarge" />
<TextView
@ -53,33 +57,16 @@
android:paddingRight="16dp"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="top"
android:layout_alignParentRight="true"
android:layout_alignParentEnd="true"
android:layout_below="@+id/main_title_welcome"
android:text="@string/ma_content_welcome"
android:textAppearance="?android:attr/textAppearanceMedium" />
<TextView
android:id="@+id/main_tv_swipe"
android:paddingLeft="16dp"
android:paddingRight="16dp"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="top"
android:layout_alignParentRight="true"
android:layout_alignParentEnd="true"
android:layout_below="@+id/main_tv_welcome"
android:text="@string/ma_content_swipeaction"
android:textAppearance="?android:attr/textAppearanceMedium" />
</RelativeLayout>
</LinearLayout>
<android.support.design.widget.NavigationView
android:id="@+id/nav_view"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="start"
android:fitsSystemWindows="true"
app:headerLayout="@layout/nav_header_main"
app:menu="@menu/activity_main_drawer"/>
app:menu="@menu/activity_main_drawer"
android:fitsSystemWindows="true"/>
</android.support.v4.widget.DrawerLayout>

Some files were not shown because too many files have changed in this diff Show More