From f8f31542b9caac0d3f0dd6f257e4b5cb40556ef6 Mon Sep 17 00:00:00 2001 From: A117870935 Date: Wed, 7 Jul 2021 17:21:49 +0530 Subject: [PATCH] Implemented privacy settings --- .../nmc/android/ui/ClickableSpanTestHelper.kt | 115 ++ .../ui/LoginPrivacySettingsActivityIT.kt | 109 ++ .../android/ui/PrivacySettingsActivityIT.kt | 93 ++ app/src/main/AndroidManifest.xml | 9 + .../nextcloud/client/di/ComponentsModule.java | 8 + .../client/preferences/AppPreferences.java | 18 + .../preferences/AppPreferencesImpl.java | 25 + .../com/nmc/android/ui/LauncherActivity.kt | 47 +- .../ui/LoginPrivacySettingsActivity.kt | 107 ++ .../nmc/android/ui/PrivacySettingsActivity.kt | 87 ++ .../android/ui/PrivacySettingsInterface.kt | 13 + .../ui/PrivacySettingsInterfaceImpl.kt | 14 + .../com/nmc/android/ui/PrivacyUserAction.kt | 9 + .../nmc/android/utils/CheckableThemeUtils.kt | 117 ++ .../java/com/nmc/android/utils/Extensions.kt | 42 + .../authentication/AuthenticatorActivity.java | 9 +- .../android/ui/activity/SettingsActivity.java | 15 + .../main/res/drawable/ic_privacy_settings.xml | 12 + .../activity_login_privacy_settings.xml | 84 ++ .../res/layout/activity_privacy_settings.xml | 128 ++ app/src/main/res/values-de/strings.xml | 1047 +++++++++++++++++ app/src/main/res/values-night/colors.xml | 64 + app/src/main/res/values/colors.xml | 89 ++ app/src/main/res/values/dimens.xml | 31 + app/src/main/res/values/setup.xml | 2 +- app/src/main/res/values/strings.xml | 27 + 26 files changed, 2306 insertions(+), 15 deletions(-) create mode 100644 app/src/androidTest/java/com/nmc/android/ui/ClickableSpanTestHelper.kt create mode 100644 app/src/androidTest/java/com/nmc/android/ui/LoginPrivacySettingsActivityIT.kt create mode 100644 app/src/androidTest/java/com/nmc/android/ui/PrivacySettingsActivityIT.kt create mode 100644 app/src/main/java/com/nmc/android/ui/LoginPrivacySettingsActivity.kt create mode 100644 app/src/main/java/com/nmc/android/ui/PrivacySettingsActivity.kt create mode 100644 app/src/main/java/com/nmc/android/ui/PrivacySettingsInterface.kt create mode 100644 app/src/main/java/com/nmc/android/ui/PrivacySettingsInterfaceImpl.kt create mode 100644 app/src/main/java/com/nmc/android/ui/PrivacyUserAction.kt create mode 100644 app/src/main/java/com/nmc/android/utils/CheckableThemeUtils.kt create mode 100644 app/src/main/java/com/nmc/android/utils/Extensions.kt create mode 100644 app/src/main/res/drawable/ic_privacy_settings.xml create mode 100644 app/src/main/res/layout/activity_login_privacy_settings.xml create mode 100644 app/src/main/res/layout/activity_privacy_settings.xml create mode 100644 app/src/main/res/values-de/strings.xml create mode 100644 app/src/main/res/values/dimens.xml diff --git a/app/src/androidTest/java/com/nmc/android/ui/ClickableSpanTestHelper.kt b/app/src/androidTest/java/com/nmc/android/ui/ClickableSpanTestHelper.kt new file mode 100644 index 000000000000..59c91cd42dbf --- /dev/null +++ b/app/src/androidTest/java/com/nmc/android/ui/ClickableSpanTestHelper.kt @@ -0,0 +1,115 @@ +package com.nmc.android.ui + +import android.text.Spannable +import android.text.style.ClickableSpan +import android.view.View +import android.widget.TextView +import androidx.test.espresso.UiController +import androidx.test.espresso.ViewAction +import androidx.test.espresso.ViewInteraction +import androidx.test.espresso.assertion.ViewAssertions.matches +import androidx.test.espresso.matcher.BoundedMatcher +import androidx.test.espresso.matcher.ViewMatchers +import androidx.test.espresso.matcher.ViewMatchers.isAssignableFrom +import androidx.test.espresso.matcher.ViewMatchers.isDisplayed +import org.hamcrest.Description +import org.hamcrest.Matcher + +object ClickableSpanTestHelper { + + /** + * method to get clickable span form a text view + * example: val clickableSpan = getClickableSpan("Link text", onView(withId(R.id.text_id))) + */ + fun getClickableSpan(spanText: String, matcher: ViewInteraction?): ClickableSpan? { + val clickableSpans = arrayOf(null) + + // Get the SpannableString from the TextView + matcher?.check(matches(isDisplayed())) + matcher?.perform(object : ViewAction { + override fun getConstraints(): Matcher { + return isAssignableFrom(TextView::class.java) + } + + override fun getDescription(): String { + return "get text from TextView" + } + + override fun perform(uiController: UiController, view: View) { + val textView = view as TextView + val text = textView.text + if (text is Spannable) { + val spans = text.getSpans( + 0, text.length, + ClickableSpan::class.java + ) + for (span in spans) { + val start = text.getSpanStart(span) + val end = text.getSpanEnd(span) + val spanString = text.subSequence(start, end).toString() + if (spanString == spanText) { + clickableSpans[0] = span + return + } + } + } + throw java.lang.RuntimeException("ClickableSpan not found") + } + }) + return clickableSpans[0] + } + + /** + * perform click on the spanned string + * @link getClickableSpan() method to get clickable span + */ + fun performClickSpan(clickableSpan: ClickableSpan?): ViewAction { + return object : ViewAction { + override fun getConstraints(): Matcher { + return ViewMatchers.isAssignableFrom(TextView::class.java) + } + + override fun getDescription(): String { + return "clicking on a span" + } + + override fun perform(uiController: UiController, view: View) { + val textView = view as TextView + val spannable = textView.text as Spannable + val spans = spannable.getSpans( + 0, spannable.length, + ClickableSpan::class.java + ) + for (span in spans) { + if (span == clickableSpan) { + span.onClick(textView) + return + } + } + throw RuntimeException("ClickableSpan not found") + } + } + } + + fun verifyClickSpan(clickableSpan: ClickableSpan?): Matcher { + return object : BoundedMatcher(TextView::class.java) { + override fun describeTo(description: Description) { + description.appendText("clickable span") + } + + override fun matchesSafely(textView: TextView): Boolean { + val spannable = textView.text as Spannable + val spans = spannable.getSpans( + 0, spannable.length, + ClickableSpan::class.java + ) + for (span in spans) { + if (span == clickableSpan) { + return true + } + } + return false + } + } + } +} \ No newline at end of file diff --git a/app/src/androidTest/java/com/nmc/android/ui/LoginPrivacySettingsActivityIT.kt b/app/src/androidTest/java/com/nmc/android/ui/LoginPrivacySettingsActivityIT.kt new file mode 100644 index 000000000000..3232d588769e --- /dev/null +++ b/app/src/androidTest/java/com/nmc/android/ui/LoginPrivacySettingsActivityIT.kt @@ -0,0 +1,109 @@ +package com.nmc.android.ui + +import androidx.test.espresso.Espresso.onView +import androidx.test.espresso.Espresso.pressBack +import androidx.test.espresso.action.ViewActions.click +import androidx.test.espresso.assertion.ViewAssertions.matches +import androidx.test.espresso.intent.Intents +import androidx.test.espresso.intent.Intents.intended +import androidx.test.espresso.intent.matcher.IntentMatchers.hasComponent +import androidx.test.espresso.matcher.ViewMatchers.isClickable +import androidx.test.espresso.matcher.ViewMatchers.isCompletelyDisplayed +import androidx.test.espresso.matcher.ViewMatchers.withId +import androidx.test.ext.junit.rules.ActivityScenarioRule +import androidx.test.ext.junit.runners.AndroidJUnit4 +import com.nextcloud.client.preferences.AppPreferencesImpl +import com.nmc.android.ui.ClickableSpanTestHelper.getClickableSpan +import com.owncloud.android.AbstractIT +import com.owncloud.android.R +import com.owncloud.android.ui.activity.ExternalSiteWebView +import com.owncloud.android.ui.activity.FileDisplayActivity +import org.junit.Assert.* +import org.junit.Rule +import org.junit.Test +import org.junit.runner.RunWith + +@RunWith(AndroidJUnit4::class) +class LoginPrivacySettingsActivityIT : AbstractIT() { + + @get:Rule + val activityRule = ActivityScenarioRule(LoginPrivacySettingsActivity::class.java) + + @Test + fun verifyNothingHappensOnBackPress() { + pressBack() + shortSleep() + + //check any one view to check the activity is not destroyed + onView(withId(R.id.tv_privacy_setting_title)).check(matches(isCompletelyDisplayed())) + } + + @Test + fun verifyUIElements() { + onView(withId(R.id.ic_privacy)).check(matches(isCompletelyDisplayed())) + + onView(withId(R.id.tv_privacy_setting_title)).check(matches(isCompletelyDisplayed())) + + onView(withId(R.id.tv_login_privacy_intro_text)).check(matches(isCompletelyDisplayed())) + + onView(withId(R.id.privacy_accept_btn)).check(matches(isCompletelyDisplayed())) + onView(withId(R.id.privacy_accept_btn)).check(matches(isClickable())) + } + + @Test + fun verifyAcceptButtonRedirection() { + Intents.init() + onView(withId(R.id.privacy_accept_btn)).perform(click()) + + //check if the policy action saved correct --> 2 for Accept action + assertEquals(2, AppPreferencesImpl.fromContext(targetContext).privacyPolicyAction) + + intended(hasComponent(FileDisplayActivity::class.java.canonicalName)) + Intents.release() + } + + @Test + fun verifySettingsTextClick() { + Intents.init() + val settingsClickableSpan = getClickableSpan("Settings", onView(withId(R.id.tv_login_privacy_intro_text))) + onView(withId(R.id.tv_login_privacy_intro_text)).perform( + ClickableSpanTestHelper.performClickSpan( + settingsClickableSpan + ) + ) + intended(hasComponent(PrivacySettingsActivity::class.java.canonicalName)) + Intents.release() + } + + @Test + fun verifyPrivacyPolicyTextClick() { + Intents.init() + val privacyPolicyClickableSpan = + getClickableSpan("Privacy Policy", onView(withId(R.id.tv_login_privacy_intro_text))) + onView(withId(R.id.tv_login_privacy_intro_text)).perform( + ClickableSpanTestHelper.performClickSpan( + privacyPolicyClickableSpan + ) + ) + intended(hasComponent(ExternalSiteWebView::class.java.canonicalName)) + Intents.release() + } + + @Test + fun verifyRejectTextClick() { + Intents.init() + val rejectClickableSpan = + getClickableSpan("reject", onView(withId(R.id.tv_login_privacy_intro_text))) + onView(withId(R.id.tv_login_privacy_intro_text)).perform( + ClickableSpanTestHelper.performClickSpan( + rejectClickableSpan + ) + ) + + //check if the policy action saved correct --> 1 for Reject action + assertEquals(1, AppPreferencesImpl.fromContext(targetContext).privacyPolicyAction) + + intended(hasComponent(FileDisplayActivity::class.java.canonicalName)) + Intents.release() + } +} \ No newline at end of file diff --git a/app/src/androidTest/java/com/nmc/android/ui/PrivacySettingsActivityIT.kt b/app/src/androidTest/java/com/nmc/android/ui/PrivacySettingsActivityIT.kt new file mode 100644 index 000000000000..5de5d075991b --- /dev/null +++ b/app/src/androidTest/java/com/nmc/android/ui/PrivacySettingsActivityIT.kt @@ -0,0 +1,93 @@ +package com.nmc.android.ui + +import android.content.Intent +import androidx.test.core.app.ActivityScenario +import androidx.test.core.app.launchActivity +import androidx.test.espresso.Espresso.onView +import androidx.test.espresso.action.ViewActions.click +import androidx.test.espresso.assertion.ViewAssertions.matches +import androidx.test.espresso.matcher.ViewMatchers.isChecked +import androidx.test.espresso.matcher.ViewMatchers.isClickable +import androidx.test.espresso.matcher.ViewMatchers.isCompletelyDisplayed +import androidx.test.espresso.matcher.ViewMatchers.isDisplayed +import androidx.test.espresso.matcher.ViewMatchers.isEnabled +import androidx.test.espresso.matcher.ViewMatchers.withId +import androidx.test.ext.junit.runners.AndroidJUnit4 +import com.owncloud.android.AbstractIT +import com.owncloud.android.R +import org.hamcrest.CoreMatchers.not +import org.junit.After +import org.junit.Assert.* +import org.junit.Before +import org.junit.Test +import org.junit.runner.RunWith + +@RunWith(AndroidJUnit4::class) +class PrivacySettingsActivityIT : AbstractIT() { + + private fun getIntent(showSettingsButton: Boolean): Intent = + Intent(targetContext, PrivacySettingsActivity::class.java) + .putExtra("show_settings_button", showSettingsButton) + + lateinit var activityRule: ActivityScenario + + @Before + fun setUp() { + activityRule = launchActivity(getIntent(false)) + } + + @Test + fun verifyUIElements() { + onView(withId(R.id.tv_privacy_intro_text)).check(matches(isCompletelyDisplayed())) + + onView(withId(R.id.switch_data_collection)).check(matches(isCompletelyDisplayed())) + onView(withId(R.id.switch_data_collection)).check(matches(not(isEnabled()))) + onView(withId(R.id.switch_data_collection)).check(matches(isChecked())) + + onView(withId(R.id.switch_data_analysis)).check(matches(isCompletelyDisplayed())) + onView(withId(R.id.switch_data_analysis)).check(matches(isEnabled())) + //by-default the analysis switch will be checked as per #AppPreferences.isDataAnalysisEnabled will return true + onView(withId(R.id.switch_data_analysis)).check(matches(isChecked())) + onView(withId(R.id.switch_data_analysis)).check(matches(isClickable())) + + onView(withId(R.id.privacy_save_settings_btn)).check(matches(not(isDisplayed()))) + } + + @Test + fun verifyDataCollectionSwitchToggle() { + //since this button is disabled performing click operation should do nothing + //and switch will be in checked state only + onView(withId(R.id.switch_data_collection)).perform(click()) + onView(withId(R.id.switch_data_collection)).check(matches(isChecked())) + + onView(withId(R.id.switch_data_collection)).perform(click()) + onView(withId(R.id.switch_data_collection)).check(matches(isChecked())) + } + + @Test + fun verifyDataAnalysisSwitchToggle() { + onView(withId(R.id.switch_data_analysis)).perform(click()) + onView(withId(R.id.switch_data_analysis)).check(matches(not(isChecked()))) + + onView(withId(R.id.switch_data_analysis)).perform(click()) + onView(withId(R.id.switch_data_analysis)).check(matches(isChecked())) + } + + @Test + fun verifySaveSettingsButton() { + //button not shown on the basis of extras passed to intent + onView(withId(R.id.privacy_save_settings_btn)).check(matches(not(isDisplayed()))) + //close the activity already open + activityRule.close() + + //launch activity with extras as true + activityRule = launchActivity(getIntent(true)) + //button will be shown if extras is true + onView(withId(R.id.privacy_save_settings_btn)).check(matches(isDisplayed())) + } + + @After + fun tearDown() { + activityRule.close() + } +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index f9cc7c311c33..c6c07c647082 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -503,6 +503,15 @@ android:name=".ui.preview.PreviewBitmapActivity" android:exported="false" android:theme="@style/Theme.ownCloud.OverlayGrey" /> + + + Note: use userAccountManager.currentAccount for user validation + // because setting enableAccountHandling false will not set user or account values under SessionMixin class + + // if user is null then go to authenticator activity + if (userAccountManager.currentAccount == null) { + startActivity(Intent(this, AuthenticatorActivity::class.java)) + } + //if user is logged in but did not accepted the privacy policy then take him there + //show him the privacy policy screen again + //check if app has been updated, if yes then also we have to show the privacy policy screen + else if (userAccountManager.currentAccount != null && (appPreferences.privacyPolicyAction == PrivacyUserAction.NO_ACTION + || appPreferences.lastSeenVersionCode < BuildConfig.VERSION_CODE) + ) { + LoginPrivacySettingsActivity.openPrivacySettingsActivity(this) + } else { + startActivity(Intent(this, FileDisplayActivity::class.java)) + } + finish() + } + override fun onCreate(savedInstanceState: Bundle?) { // Mandatory to call this before super method to show system launch screen for api level 31+ installSplashScreen() + //Fix of NMC-2464 + //this is mandatory to call before super() function + //setting false to show launcher screen properly if user is not logged in + enableAccountHandling = false + super.onCreate(savedInstanceState) binding = ActivitySplashBinding.inflate(layoutInflater) @@ -75,17 +103,18 @@ class LauncherActivity : BaseActivity() { } private fun scheduleSplashScreen() { - Handler(Looper.getMainLooper()).postDelayed({ - if (!user.isPresent) { - startActivity(Intent(this, AuthenticatorActivity::class.java)) - } else { - startActivity(Intent(this, FileDisplayActivity::class.java)) - } - finish() - }, SPLASH_DURATION) + handler.postDelayed( + runnable, + SPLASH_DURATION + ) + } + + override fun onPause() { + super.onPause() + handler.removeCallbacks(runnable) } companion object { const val SPLASH_DURATION = 1500L } -} +} \ No newline at end of file diff --git a/app/src/main/java/com/nmc/android/ui/LoginPrivacySettingsActivity.kt b/app/src/main/java/com/nmc/android/ui/LoginPrivacySettingsActivity.kt new file mode 100644 index 000000000000..b9c9c203d736 --- /dev/null +++ b/app/src/main/java/com/nmc/android/ui/LoginPrivacySettingsActivity.kt @@ -0,0 +1,107 @@ +package com.nmc.android.ui + +import android.content.Context +import android.content.Intent +import android.os.Bundle +import android.view.View +import com.nextcloud.client.preferences.AppPreferences +import com.nmc.android.utils.makeLinks +import com.owncloud.android.BuildConfig +import com.owncloud.android.R +import com.owncloud.android.databinding.ActivityLoginPrivacySettingsBinding +import com.owncloud.android.ui.activity.ExternalSiteWebView +import com.owncloud.android.ui.activity.FileDisplayActivity +import com.owncloud.android.ui.activity.ToolbarActivity +import javax.inject.Inject + +class LoginPrivacySettingsActivity : ToolbarActivity() { + + companion object { + @JvmStatic + fun openPrivacySettingsActivity(context: Context) { + val intent = Intent(context, LoginPrivacySettingsActivity::class.java) + context.startActivity(intent) + } + } + + private lateinit var binding: ActivityLoginPrivacySettingsBinding + + @Inject + lateinit var preferences: AppPreferences + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + binding = ActivityLoginPrivacySettingsBinding.inflate(layoutInflater) + setContentView(binding.root) + setupToolbar() + resetPreferenceForPrivacy() + //don't show back button + supportActionBar?.setDisplayHomeAsUpEnabled(false) + updateActionBarTitleAndHomeButtonByString(resources.getString(R.string.privacy_settings)) + setUpPrivacyText() + binding.privacyAcceptBtn.setOnClickListener { + //on accept finish the activity + //update the accept privacy action to preferences + preferences.privacyPolicyAction = PrivacyUserAction.ACCEPT_ACTION + openFileDisplayActivity() + } + } + + private fun resetPreferenceForPrivacy() { + preferences.setDataAnalysis(false) + preferences.privacyPolicyAction = PrivacyUserAction.NO_ACTION + } + + private fun setUpPrivacyText() { + val privacyText = String.format( + resources.getString(R.string.login_privacy_settings_intro_text), resources + .getString(R.string.login_privacy_policy), resources + .getString(R.string.login_privacy_reject), resources + .getString(R.string.login_privacy_settings) + ) + binding.tvLoginPrivacyIntroText.text = privacyText + + //make links clickable + binding.tvLoginPrivacyIntroText.makeLinks( + Pair(resources.getString(R.string.login_privacy_policy), View.OnClickListener { + //open privacy policy url + val intent = Intent(this, ExternalSiteWebView::class.java) + intent.putExtra( + ExternalSiteWebView.EXTRA_TITLE, + resources.getString(R.string.privacy_policy) + ) + intent.putExtra(ExternalSiteWebView.EXTRA_URL, resources.getString(R.string.privacy_url)) + intent.putExtra(ExternalSiteWebView.EXTRA_SHOW_SIDEBAR, false) + intent.putExtra(ExternalSiteWebView.EXTRA_MENU_ITEM_ID, -1) + startActivity(intent) + }), Pair(resources + .getString(R.string.login_privacy_reject), View.OnClickListener { + //disable data analysis option and close the activity + preferences.setDataAnalysis(false) + //update the reject privacy action to preferences + preferences.privacyPolicyAction = PrivacyUserAction.REJECT_ACTION + openFileDisplayActivity() + }), Pair(resources + .getString(R.string.login_privacy_settings), View.OnClickListener { + //open privacy settings screen + PrivacySettingsActivity.openPrivacySettingsActivity(this, true) + }) + ) + } + + private fun openFileDisplayActivity() { + //update the version code when user has accepted or rejected privacy policy + //this will be used to help to check app up-gradation + preferences.lastSeenVersionCode = BuildConfig.VERSION_CODE + + val i = Intent(this, FileDisplayActivity::class.java) + i.action = FileDisplayActivity.RESTART + i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) + startActivity(i) + finish() + } + + override fun onBackPressed() { + //user cannot close this screen without accepting or rejecting the privacy policy + } +} diff --git a/app/src/main/java/com/nmc/android/ui/PrivacySettingsActivity.kt b/app/src/main/java/com/nmc/android/ui/PrivacySettingsActivity.kt new file mode 100644 index 000000000000..e3572062d287 --- /dev/null +++ b/app/src/main/java/com/nmc/android/ui/PrivacySettingsActivity.kt @@ -0,0 +1,87 @@ +package com.nmc.android.ui + +import android.content.Context +import android.content.Intent +import android.graphics.drawable.ColorDrawable +import android.os.Bundle +import android.view.MenuItem +import android.view.View +import com.nextcloud.client.preferences.AppPreferences +import com.nmc.android.utils.CheckableThemeUtils +import com.owncloud.android.R +import com.owncloud.android.databinding.ActivityPrivacySettingsBinding +import com.owncloud.android.ui.activity.ToolbarActivity +import javax.inject.Inject + +class PrivacySettingsActivity : ToolbarActivity() { + + companion object { + private const val EXTRA_SHOW_SETTINGS = "show_settings_button" + + @JvmStatic + fun openPrivacySettingsActivity(context: Context, isShowSettings: Boolean) { + val intent = Intent(context, PrivacySettingsActivity::class.java) + intent.putExtra(EXTRA_SHOW_SETTINGS, isShowSettings) + context.startActivity(intent) + } + } + + private lateinit var binding: ActivityPrivacySettingsBinding + + /** + * variable to check if save settings button needs to be shown or not + * currently we are showing only when user opens this activity from LoginPrivacySettingsActivity + */ + private var isShowSettingsButton = false + + @Inject + lateinit var preferences: AppPreferences + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + binding = ActivityPrivacySettingsBinding.inflate(layoutInflater) + setContentView(binding.root) + setupToolbar() + setupActionBar() + setUpViews() + showHideSettingsButton() + } + + private fun setupActionBar() { + supportActionBar?.let { + viewThemeUtils.platform.themeStatusBar(this) + it.setDisplayHomeAsUpEnabled(true) + it.setDisplayShowTitleEnabled(true) + //custom color for back arrow for NMC + viewThemeUtils.files.themeActionBar(this, it, resources.getString(R.string.privacy_settings)) + it.setBackgroundDrawable(ColorDrawable(resources.getColor(R.color.bg_default, null))) + } + } + + private fun showHideSettingsButton() { + isShowSettingsButton = intent.getBooleanExtra(EXTRA_SHOW_SETTINGS, false) + binding.privacySaveSettingsBtn.visibility = if (isShowSettingsButton) View.VISIBLE else View.GONE + } + + private fun setUpViews() { + CheckableThemeUtils.tintSwitch(binding.switchDataCollection) + CheckableThemeUtils.tintSwitch(binding.switchDataAnalysis) + binding.switchDataAnalysis.isChecked = preferences.isDataAnalysisEnabled + binding.switchDataAnalysis.setOnCheckedChangeListener { _, isChecked -> + preferences.setDataAnalysis(isChecked) + } + binding.privacySaveSettingsBtn.setOnClickListener { + //finish the activity as we are changing the setting on switch check change + finish() + } + } + + override fun onOptionsItemSelected(item: MenuItem): Boolean { + when (item.itemId) { + android.R.id.home -> { + finish() + } + } + return super.onOptionsItemSelected(item) + } +} diff --git a/app/src/main/java/com/nmc/android/ui/PrivacySettingsInterface.kt b/app/src/main/java/com/nmc/android/ui/PrivacySettingsInterface.kt new file mode 100644 index 000000000000..860ffada08ab --- /dev/null +++ b/app/src/main/java/com/nmc/android/ui/PrivacySettingsInterface.kt @@ -0,0 +1,13 @@ +package com.nmc.android.ui + +import android.content.Context + +/** + * interface to open privacy settings activity from nmc/1921-settings branch + * for implementation look nmc/1878-privacy branch + * this class will have the declaration for it since it has the PrivacySettingsActivity.java in place + * since we don't have privacy settings functionality in this branch so to handle the redirection we have used interface + */ +interface PrivacySettingsInterface { + fun openPrivacySettingsActivity(context: Context) +} \ No newline at end of file diff --git a/app/src/main/java/com/nmc/android/ui/PrivacySettingsInterfaceImpl.kt b/app/src/main/java/com/nmc/android/ui/PrivacySettingsInterfaceImpl.kt new file mode 100644 index 000000000000..e087bc888e6f --- /dev/null +++ b/app/src/main/java/com/nmc/android/ui/PrivacySettingsInterfaceImpl.kt @@ -0,0 +1,14 @@ +package com.nmc.android.ui + +import android.content.Context + +/** + * interface impl to launch PrivacySettings Activity + * this class will have the implementation for it since it has the PrivacySettingsActivity in place + * calling of this method will be done from nmc/1921-settings + */ +class PrivacySettingsInterfaceImpl : PrivacySettingsInterface { + override fun openPrivacySettingsActivity(context: Context) { + PrivacySettingsActivity.openPrivacySettingsActivity(context, false) + } +} \ No newline at end of file diff --git a/app/src/main/java/com/nmc/android/ui/PrivacyUserAction.kt b/app/src/main/java/com/nmc/android/ui/PrivacyUserAction.kt new file mode 100644 index 000000000000..6dca412cc29b --- /dev/null +++ b/app/src/main/java/com/nmc/android/ui/PrivacyUserAction.kt @@ -0,0 +1,9 @@ +package com.nmc.android.ui + +//class to handle user action for privacy +object PrivacyUserAction { + //privacy user action to maintain the state of privacy policy + const val NO_ACTION = 0 //user has taken no action + const val REJECT_ACTION = 1 //user rejected the privacy policy + const val ACCEPT_ACTION = 2 //user has accepted the privacy policy +} \ No newline at end of file diff --git a/app/src/main/java/com/nmc/android/utils/CheckableThemeUtils.kt b/app/src/main/java/com/nmc/android/utils/CheckableThemeUtils.kt new file mode 100644 index 000000000000..a3b8a1149948 --- /dev/null +++ b/app/src/main/java/com/nmc/android/utils/CheckableThemeUtils.kt @@ -0,0 +1,117 @@ +package com.nmc.android.utils + +import android.content.res.ColorStateList +import androidx.appcompat.widget.AppCompatCheckBox +import androidx.appcompat.widget.SwitchCompat +import androidx.core.content.res.ResourcesCompat +import com.owncloud.android.R + +object CheckableThemeUtils { + @JvmStatic + fun tintCheckbox(vararg checkBoxes: AppCompatCheckBox) { + for (checkBox in checkBoxes) { + val checkEnabled = ResourcesCompat.getColor( + checkBox.context.resources, + R.color.checkbox_checked_enabled, + checkBox.context.theme + ) + val checkDisabled = ResourcesCompat.getColor( + checkBox.context.resources, + R.color.checkbox_checked_disabled, + checkBox.context.theme + ) + val uncheckEnabled = ResourcesCompat.getColor( + checkBox.context.resources, + R.color.checkbox_unchecked_enabled, + checkBox.context.theme + ) + val uncheckDisabled = ResourcesCompat.getColor( + checkBox.context.resources, + R.color.checkbox_unchecked_disabled, + checkBox.context.theme + ) + + val states = arrayOf( + intArrayOf(android.R.attr.state_enabled, android.R.attr.state_checked), + intArrayOf(-android.R.attr.state_enabled, android.R.attr.state_checked), + intArrayOf(android.R.attr.state_enabled, -android.R.attr.state_checked), + intArrayOf(-android.R.attr.state_enabled, -android.R.attr.state_checked) + ) + val colors = intArrayOf( + checkEnabled, + checkDisabled, + uncheckEnabled, + uncheckDisabled + ) + checkBox.buttonTintList = ColorStateList(states, colors) + } + } + + @JvmStatic + @JvmOverloads + fun tintSwitch(switchView: SwitchCompat, color: Int = 0, colorText: Boolean = false) { + if (colorText) { + switchView.setTextColor(color) + } + + val states = arrayOf( + intArrayOf(android.R.attr.state_enabled, android.R.attr.state_checked), + intArrayOf(android.R.attr.state_enabled, -android.R.attr.state_checked), + intArrayOf(-android.R.attr.state_enabled) + ) + + val thumbColorCheckedEnabled = ResourcesCompat.getColor( + switchView.context.resources, + R.color.switch_thumb_checked_enabled, + switchView.context.theme + ) + val thumbColorUncheckedEnabled = + ResourcesCompat.getColor( + switchView.context.resources, + R.color.switch_thumb_unchecked_enabled, + switchView.context.theme + ) + val thumbColorDisabled = + ResourcesCompat.getColor( + switchView.context.resources, + R.color.switch_thumb_disabled, + switchView.context.theme + ) + + val thumbColors = intArrayOf( + thumbColorCheckedEnabled, + thumbColorUncheckedEnabled, + thumbColorDisabled + ) + val thumbColorStateList = ColorStateList(states, thumbColors) + + val trackColorCheckedEnabled = ResourcesCompat.getColor( + switchView.context.resources, + R.color.switch_track_checked_enabled, + switchView.context.theme + ) + val trackColorUncheckedEnabled = + ResourcesCompat.getColor( + switchView.context.resources, + R.color.switch_track_unchecked_enabled, + switchView.context.theme + ) + val trackColorDisabled = + ResourcesCompat.getColor( + switchView.context.resources, + R.color.switch_track_disabled, + switchView.context.theme + ) + + val trackColors = intArrayOf( + trackColorCheckedEnabled, + trackColorUncheckedEnabled, + trackColorDisabled + ) + + val trackColorStateList = ColorStateList(states, trackColors) + + switchView.thumbTintList = thumbColorStateList + switchView.trackTintList = trackColorStateList + } +} \ No newline at end of file diff --git a/app/src/main/java/com/nmc/android/utils/Extensions.kt b/app/src/main/java/com/nmc/android/utils/Extensions.kt new file mode 100644 index 000000000000..035690e7af04 --- /dev/null +++ b/app/src/main/java/com/nmc/android/utils/Extensions.kt @@ -0,0 +1,42 @@ +package com.nmc.android.utils + +import android.text.Selection +import android.text.Spannable +import android.text.SpannableString +import android.text.Spanned +import android.text.TextPaint +import android.text.method.LinkMovementMethod +import android.text.style.ClickableSpan +import android.view.View +import android.widget.TextView + +fun TextView.makeLinks(vararg links: Pair) { + val spannableString = SpannableString(this.text) + var startIndexOfLink = -1 + for (link in links) { + val clickableSpan = object : ClickableSpan() { + override fun updateDrawState(textPaint: TextPaint) { + // use this to change the link color + textPaint.color = textPaint.linkColor + // toggle below value to enable/disable + // the underline shown below the clickable text + //textPaint.isUnderlineText = true + } + + override fun onClick(view: View) { + Selection.setSelection((view as TextView).text as Spannable, 0) + view.invalidate() + link.second.onClick(view) + } + } + startIndexOfLink = this.text.toString().indexOf(link.first, startIndexOfLink + 1) + spannableString.setSpan( + clickableSpan, startIndexOfLink, startIndexOfLink + link.first.length, + Spanned.SPAN_EXCLUSIVE_EXCLUSIVE + ) + } + this.movementMethod = + LinkMovementMethod.getInstance() // without LinkMovementMethod, link can not click + this.setText(spannableString, TextView.BufferType.SPANNABLE) +} + diff --git a/app/src/main/java/com/owncloud/android/authentication/AuthenticatorActivity.java b/app/src/main/java/com/owncloud/android/authentication/AuthenticatorActivity.java index f54db136715e..60d81de82c5c 100644 --- a/app/src/main/java/com/owncloud/android/authentication/AuthenticatorActivity.java +++ b/app/src/main/java/com/owncloud/android/authentication/AuthenticatorActivity.java @@ -84,6 +84,7 @@ import com.nextcloud.client.device.DeviceInfo; import com.nextcloud.client.di.Injectable; import com.nextcloud.client.onboarding.FirstRunActivity; +import com.nmc.android.ui.LoginPrivacySettingsActivity; import com.nextcloud.client.onboarding.OnboardingService; import com.nextcloud.client.preferences.AppPreferences; import com.nextcloud.java.util.Optional; @@ -120,7 +121,6 @@ import com.owncloud.android.services.OperationsService; import com.owncloud.android.services.OperationsService.OperationsServiceBinder; import com.owncloud.android.ui.NextcloudWebViewClient; -import com.owncloud.android.ui.activity.FileDisplayActivity; import com.owncloud.android.ui.dialog.IndeterminateProgressDialog; import com.owncloud.android.ui.dialog.SslUntrustedCertDialog; import com.owncloud.android.ui.dialog.SslUntrustedCertDialog.OnSslUntrustedCertListener; @@ -1219,10 +1219,9 @@ private void endSuccess() { if (onlyAdd) { finish(); } else { - Intent i = new Intent(this, FileDisplayActivity.class); - i.setAction(FileDisplayActivity.RESTART); - i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); - startActivity(i); + //login privacy settings to accept/reject by the user after login + LoginPrivacySettingsActivity.openPrivacySettingsActivity(this); + finish(); } } diff --git a/app/src/main/java/com/owncloud/android/ui/activity/SettingsActivity.java b/app/src/main/java/com/owncloud/android/ui/activity/SettingsActivity.java index da23f3168a8c..ca36179bcf13 100644 --- a/app/src/main/java/com/owncloud/android/ui/activity/SettingsActivity.java +++ b/app/src/main/java/com/owncloud/android/ui/activity/SettingsActivity.java @@ -57,11 +57,13 @@ import com.nextcloud.client.logger.ui.LogsActivity; import com.nextcloud.client.network.ClientFactory; import com.nextcloud.client.network.ConnectivityService; +import com.nmc.android.ui.PrivacySettingsInterface; import com.nextcloud.client.preferences.AppPreferences; import com.nextcloud.client.preferences.AppPreferencesImpl; import com.nextcloud.client.preferences.DarkMode; import com.owncloud.android.BuildConfig; import com.owncloud.android.MainApp; +import com.nmc.android.ui.PrivacySettingsInterfaceImpl; import com.owncloud.android.R; import com.owncloud.android.authentication.AuthenticatorActivity; import com.owncloud.android.datamodel.ArbitraryDataProvider; @@ -152,6 +154,16 @@ public class SettingsActivity extends PreferenceActivity @Inject ViewThemeUtils viewThemeUtils; @Inject ConnectivityService connectivityService; + /** + * Things to note about both the branches. + * 1. nmc/1921-settings branch: + * --> interface won't be initialised + * --> calling of interface method will be done here + * 2. nmc/1878-privacy + * --> interface will be initialised + * --> calling of interface method won't be done here + */ + private PrivacySettingsInterface privacySettingsInterface; @SuppressWarnings("deprecation") @Override @@ -164,6 +176,9 @@ public void onCreate(Bundle savedInstanceState) { setupActionBar(); + //NMC customization will be initialised in nmc/1878-privacy + privacySettingsInterface = new PrivacySettingsInterfaceImpl(); + // Register context menu for list of preferences. registerForContextMenu(getListView()); diff --git a/app/src/main/res/drawable/ic_privacy_settings.xml b/app/src/main/res/drawable/ic_privacy_settings.xml new file mode 100644 index 000000000000..c3bb3fb01899 --- /dev/null +++ b/app/src/main/res/drawable/ic_privacy_settings.xml @@ -0,0 +1,12 @@ + + + diff --git a/app/src/main/res/layout/activity_login_privacy_settings.xml b/app/src/main/res/layout/activity_login_privacy_settings.xml new file mode 100644 index 000000000000..3a51e9457b3d --- /dev/null +++ b/app/src/main/res/layout/activity_login_privacy_settings.xml @@ -0,0 +1,84 @@ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/activity_privacy_settings.xml b/app/src/main/res/layout/activity_privacy_settings.xml new file mode 100644 index 000000000000..33ef22c67720 --- /dev/null +++ b/app/src/main/res/layout/activity_privacy_settings.xml @@ -0,0 +1,128 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/values-de/strings.xml b/app/src/main/res/values-de/strings.xml new file mode 100644 index 000000000000..6c2819da242f --- /dev/null +++ b/app/src/main/res/values-de/strings.xml @@ -0,0 +1,1047 @@ + + + %1$s Android App + Über + Version %1$s + Version %1$s, Build #%2$s + Kontoerstellung fehlgeschlagen + Konto-Icon + Kein Konto gefunden + Fehlgeschlagene Uploads entfernen + Bearbeiten + Alle Benachrichtigungen löschen + Papierkorb leeren + Senden/Teilen + Rasteransicht + Listenansicht + Kontakte & Kalender wiederherstellen + Kopieren + Neuer Ordner + Verschieben + Verschieben oder kopieren + Öffnen mit + Suche + Details + Senden + Einstellungen + Sortieren + Aktiver Nutzer + Bislang keine Aktivitäten + Bisher keine Ereignisse wie Hinzugekommenes, Änderungen und Freigaben + Senden + Link senden an… + Aktivitäten + Weiteren Link hinzufügen + Öffentlichen Link zum Teilen hinzufügen + Neue sichere Dateiablage hinzufügen + Hinzufügen zu %1$s + Erweiterte Einstellungen + Weiterteilen erlauben + Zeigt ein Widget aus dem Dashboard an + Suche in %s + Verknüpftes Konto nicht gefunden! + Zugriffsfehler: %1$s + Das Konto ist bislang auf dem Gerät nicht vorhanden + Ein Konto für den gleichen Benutzer und Server existiert auf diesem Gerät bereits + Der eingegebene Benutzer passt nicht zu dem Benutzer dieses Kontos + Unbekannte Serverversion + Verbindung hergestellt + Ihr Server gibt keine korrekte Benutzer-ID zurück, bitte kontaktieren Sie einen Administrator. + Serveradresse https://… + Falsches Adressformat für den Server + Server nicht gefunden + Keine Netzwerkverbindung + Sichere Verbindung nicht verfügbar. + Fehlerhafte Serverkonfiguration + Autorisierung nicht erfolgreich + Zugriff durch den Autorisierungsserver abgelehnt + Die gesicherte Verbindung wird durch eine ungesicherte Route geleitet. + Sichere Verbindung hergestellt + SSL-Initialisierung fehlgeschlagen + SSL-Server-Identität konnte nicht überprüft werden + Verbindung testen + Der Server braucht zu lange für eine Antwort. + Versuche Anmeldung… + Falscher Benutzername oder Passwort + Unbekannter Fehler: %1$s + Es ist ein unbekannter HTTP-Fehler aufgetreten! + Es ist ein unbekannter Fehler aufgetreten. + Konnte den Host nicht finden + %1$s unterstützt nicht mehrere Benutzerkonten + Verbindung konnte nicht hergestellt werden + im Original-Verzeichnis belassen, da nur lesbar + Nur über gebührenfreies WLAN hochladen + /AutoUpload + Einrichten + Erstellen Sie ein Setup für den eigenen Ordner + Erstellen Sie einen eigenen Ordner + Energiesparmodus deaktivieren + Ordner verstecken + Avatar + Abwesend + Sicherungseinstellungen + Kontakte & Kalender-Sicherung + Schliessen + Deaktivieren + Ihr Gerät hat möglicherweise Batterieoptimierung aktiv. AutoUpload funktioniert nur dann richtig, wenn Sie diese App davon ausschließen. + Batterie-Optimierung + Durch zu viele Fehlversuche verzögert + Kalender + Kalender + Beim Laden des Zertifikats ist ein Fehler aufgetreten. + Änderungen in der Entwicklerversion + Schauen Sie später noch einmal vorbei oder laden Sie neu. + Checkbox + Wähle einen lokalen Ordner… + Ort wählen + Wähle einen entfernten Ordner … + Bitte eine Vorlage auswählen und einen Dateinamen eingeben. + Wählen Sie, welche Datei behalten werden soll! + Widget auswählen + Löschen der Benachrichtigungen fehlgeschlagen. + Statusnachricht löschen + Statusnachricht löschen nach + Text von %1$s kopiert + Kein Text zum Kopieren in die Zwischenablage empfangen + Link kopiert + Unerwarteter Fehler beim Kopieren in die Zwischenablage + Zurück + Abbrechen + Synchronisierung abbrechen + Konto auswählen + Bestätigen + Kopieren + Löschen + Fehler + Nicht genügend Speicher + Unbekannter Fehler + Lade… + Nächste + Nein + OK + Ausstehend + Löschen + Umbenennen + Speichern + Auswählen + Senden + Teilen + Überspringen + Konto wechseln + Zu Konto wechseln + Ja + Testen Sie die Entwickler-Version + Dies beinhaltet neue Funktionalitäten und ist nicht komplett qualitätsgesichert. Es können daher Fehler/Bugs auftreten, melden Sie uns solche bitte. + Forum + Helfen Sie anderen auf + Überprüfen, Ändern und Schreiben von Code, weitere Enzelheiten finden Sie hier: %1$s. + Aktiv etwas beitragen + die App + Übersetzen + Entwicklungs-Version direkt herunterladen + Entwicklungs-Version über die F-Droid-App beziehen + Release-Kandidat über die F-Droid-App beziehen + Release-Kandidat über den Google-Play-Store beziehen + Vorabversion + Der Release-Kandidat (RC) ist eine Vorschau auf die kommende Version und sollte stabil sein. Das Testen des RC unter Ihrer individuellen Umgebung kann helfen, die Stabilität und Qualität zu sichern. Registrieren Sie sich als Tester im Play-Store oder schauen Sie in die \"Version\"-Sektion auf F-Droid. + Fehler gefunden? Komisches Verhalten? + Helfen Sie durch Testen + Fehlerbericht auf GitHub erstellen + Konfigurieren + Lokale Verschlüsselung entfernen + Wollen Sie %1$s wirklich löschen? + Möchten Sie die ausgewählten Elemente wirklich löschen? + Wollen Sie %1$s und deren Inhalte wirklich löschen? + Möchten Sie die ausgewählten Elemente und deren inhalt wirklich löschen? + Nur lokal + Fehler beim Erstellen des Konfliktdialoges + Konflikt-Datei %1$s + Lokale Datei + Falls beide Versionen gewählt werden, wird bei der lokalen Datei eine Zahl am Ende des Dateinamens hinzugefügt. + Server-Datei + Kontakte-Backup + Nutzericon für Kontaktliste + Keine Berechtigung vergeben, es wurde nichts importiert. + Kontakte + Jetzt sichern + Sicherung geplant und wird in Kürze starten + Import geplant und wird in Kürze starten + Keine Datei gefunden + Wir können Ihr letztes Backup nicht finden! + In die Zwischenablage kopiert + Es ist ein Fehler beim Kopieren der Datei oder des Ordners aufgetreten. + Es ist nicht möglich, einen Ordner in einen seiner Unterordner zu kopieren + Die Datei existiert bereits im Zielverzeichnis + Kopieren nicht möglich. Bitte prüfen Sie, ob die Datei existiert. + Link kopieren + In verschlüsselten Ordner kopieren/verschieben wird aktuell nicht unterstützt. + Kopieren nach… + Originalbild konnte nicht heruntergeladen werden + Konnte die Freigaben nicht abrufen + Konnte URL nicht abrufen + Erstellen + Verzeichnis konnte nicht erstellt werden + Neu erstellen + Neues Dokument erstellen + Neuen Ordner erstellen + Neue Präsentation erstellen + Neue Tabelle erstellen + Ordnerinfo hinzufügen + Ordnerinfo erstellen + Zugangsdaten ausgeschaltet + Tägliche Sicherung + Zu sichernde Daten + Zugangsdaten falsch + Benutzerkonto löschen + Konto %s und alle lokalen Dateien löschen?\n\nLöschung kann nicht rückgängig gemacht werden. + Einträgen entfernen + Link löschen + Auswahl aufheben + Ziel-Dateiname + Neue Version verfügbar + Keine Information verfügbar. + Keine neue Version verfügbar. + Schließen + Es wurde nicht auf Duplikate geprüft. + Dieser Algorithmus-Auszug ist auf Ihrem Handy nicht verfügbar. + Anmeldung über Direktlink fehlgeschlagen! + Mit %1$s an %2$s anmelden + Ausschalten + Ausblenden + Benachrichtigung verwerfen + Zeigt Ihre 12-Wörter-Passphrase an + Bitte nicht stören + Mehrere Bilder + PDF-Datei + Exporttyp wählen + PDF-Erstellung fehlgeschlagen + Erstelle PDF… + Fertig + Nicht löschen + Datei kann lokal nicht erstellt werden + Ungültiger Dateiname für lokale Datei + Die neueste Entwicklerversion herunterladen + Konnte %1$s nicht herunterladen + Herunterladen fehlgeschlagen, erneute Anmeldung erforderlich + Herunterladen fehlgeschlagen + Diese Datei steht auf dem Server nicht mehr zur Verfügung + %1$d%% Herunterladen %2$s + Lade herunter… + %1$s heruntergeladen + Heruntergeladen + Noch nicht Heruntergeladen + Hauptmenü schließen + Gemeinschaft + Hintergrundbild des Kopfbereiches + Aktivitäten + Alle Dateien + Favoriten + Medien + Gruppenordner + Home + Benachrichtigungen + Auf dem Gerät + Kürzlich geändert + Geteilt + Gelöschte Dateien + Uploads + Abmelden + Hauptmenü öffnen + %1$s von %2$s verwendet + %1$s verwendet + Automatisches Hochladen + E2E bislang nicht eingerichtet + Ohne Internetverbindung nicht möglich + Mehr + Notizen + Talk + Weitere Nextcloud-Apps + Nextcloud Notizen + Nextcloud Talk + Als verschlüsselt festlegen + Verschlüsselung einrichten + Entschlüssele… + Schließen + Geben Sie bitte das Passwort ein, um den privaten Schlüssel zu entschlüsseln. + Dieser Ordner ist nicht leer. + Neue Schlüssel generieren… + Diese 12 Wörter-Phrase ist wie ein sehr starkes Passwort: Sie bietet vollen Zugriff um Ihre verschlüsselten Dateien anzusehen und zu verwenden. Bitte schreiben Sie sie auf und verwahren sie an einem sicheren Ort. + Ende-zu-Ende Verschlüsselung ist auf diesem Server deaktiviert. + Notieren Sie Ihre Verschlüsselungs-Passphrase + Passwort… + Schlüssel abrufen… + Schlüssel speichern + Verschlüsselung einrichten + Die Schlüssel konnten nicht gespeichert werden. Bitte erneut versuchen. + Fehler beim Entschlüsseln. Falsches Passwort? + Ziel-Dateiname eingeben + Bitte Dateinamen eingeben + %1$s konnte nicht in den lokalen %2$s Ordner kopiert werden + Kritischer Fehler: Operationen können nicht ausgeführt werden + Fehler beim Auswählen des Datums + Fehler beim Kommentieren der Datei + %1$s abgestürzt + Fehler beim Erzeugen einer Datei aus der Vorlage + Fehler beim Anzeigen von Dateiaktionen + Fehler beim Ändern des Sperr-Status + Melden + Problem melden? (benötigt ein GitHub-Konto) + Fehler beim Abruf der Datei + Fehler beim Laden der Vorlagen + Fehler beim Anzeigen des Setup-Dialogs für Verschlüsselung! + Fehler beim Starten der Kamera + Start des Dokumentenscans fehlgeschlagen + Konten + Aufgabenname + Fortschritt + Gestartet + Status + Benutzer + UUID + Hintergrund-Aufgaben + Alle Aufgaben abbrechen + Inaktive Aufgaben bereinigen + Test-Aufgabe planen + Test-Aufgabe starten + Test-Aufgabe stoppen + Migrationen (App-Aktualisierung) + Einstellungen + Technischer Testmodus + Dateiübertragung + Test-Download in die Warteschlange stellen + Test-Upload in die Warteschlange stellen + Entfernter Pfad + Übertragen + Herunterladen + Hochladen + Hinzufügen oder hochladen + Datei konnte nicht an den Downloadmanager übergeben werden + Drucken der Datei fehlgeschlagen + Starten des Editors fehlgeschlagen + Oberfläche konnte nicht aktualisiert werden + Zu den Favoriten hinzufügen + Favoriten + Dateiname bereits vorhanden + Löschen + Fehler beim Abrufen der Aktivitäten für die Datei + Fehler beim Laden der Details + Herunterladen \u0020 + Datei + Behalten + Laden Sie Inhalt hoch oder synchronisieren Sie mit Ihren Geräten. + Noch keine Favoriten vorhanden + Dateien und Ordner, die als Favoriten markiert werden, erscheinen hier. + Keine Bilder oder Videos gefunden + Keine Dateien vorhanden + Keine Ergebnisse in diesem Ordner + Keine Ergebnisse + Noch nichts vorhanden. Sie können einen Ordner hinzufügen. + Heruntergeladene Dateien und Ordner werden hier angezeigt werden. + Keine, innerhalb der letzten 7 Tage, geänderten Dateien gefunden + Vielleicht befindet es sich in einem anderen Ordner? + Von Ihnen geteilte Dateien und Ordner werden hier angezeigt. + Noch nichts geteilt + Keine Ergebnisse für Ihre Suche gefunden + Ordner + Lade… + Es wurde keine App für diesen Dateityp gefunden. + Gerade eben + Berechtigungen erforderlich + Speicherberechtigungen + %1$s funktioniert am besten mit Berechtigungen für den Zugriff auf den Speicher. Sie können vollen Zugriff auf alle Dateien oder schreibgeschützten Zugriff auf Fotos und Videos wählen. + %1$s benötigt Dateiverwaltungsberechtigungen, um Dateien hochzuladen. Sie können vollen Zugriff auf alle Dateien oder schreibgeschützten Zugriff auf Fotos und Videos wählen.  + Prüfe Zielort… + Aufräumen… + Aktualisiere Speicherordner + Daten-Verzeichnis existiert bereits. Bitte auswählen: + Nextcloud-Verzeichnis existiert bereits + Mehr Speicherplatz benötigt + Quelldatei kann nicht gelesen werden + Für das Ziel existieren keine Schreibrechte + Migration fehlgeschlagen + Aktualisierung des Index fehlgeschlagen + Verschiebe Daten… + Abgeschlossen + Überschreiben + Bereite Migration vor… + Konfiguration der Konten wiederherstellen… + Sichere Konfiguration der Konten… + Möchten Sie wirklich den Speicherordner in %1$s ändern?\n\nHinweis: Alle Daten müssen dann erneut heruntergeladen werden. + Quellverzeichnis kann nicht gelesen werden! + Aktualisiere Index… + Verwende bestehenden + Warte auf die Fertigstellung aller Synchronisierungen … + Datei nicht gefunden + Die gewünschte Datei konnte nicht synchronisiert werden. Die letzte verfügbare Version wird angezeigt. + Umbenennen + Fehler beim Wiederherstellen der Dateiversion! + Dateiversion erfolgreich wiederhergestellt + Details + Herunterladen + Exportieren + Datei während des Hochladens in %1$s umbenannt + Synchronisieren + Keine Datei gewählt + Dateiname darf nicht leer sein + Verbotene Zeichen: / \\ < > : \" | ? * + Der Dateiname enthält mindestens ein ungültiges Zeichen + Dateiname + Halten Sie Ihre Daten sicher und unter Ihrer Kontrolle + Sichere Zusammenarbeit & Dateiaustausch + Benutzerfreundliche Webmailer, Kalender & Kontakte + Bildschirm teilen, Online-Meetings& Web-Konferenzen + Ordner existiert bereits + Erstellen + Keine Ordner vorhanden + Auswählen + Zielordner auswählen + Kopieren + Verschieben + Ihnen fehlt die Erlaubnis %s + diese Datei zu kopieren + diese Datei zu erstellen + diese Datei zu löschen + um diese Datei zu verschieben + diese Datei umzubenennen + Dateien werden heruntergeladen… + Dateien werden hochgeladen… + Einige Dateien konnten nicht verschoben werden + Lokal: %1$s + Alle verschieben + Remote: %1$s + Alle Dateien wurden verschoben + Weiterleiten + 4 Stunden + Dateiname führt dazu, das die Datei versteckt sein wird. + Name + Notiz + Passwort + Server nicht verfügbar + Eigenen Server betreiben + Symbol für leere Liste + Symbol des Dashboard-Widgets + Symbol des Widget-Eintrags + Bearbeitet + Horizontal spiegeln + Vertikal spiegeln + Gegen den Uhrzeigersinn drehen + Im Uhrzeigersinn drehen + Bild konnte nicht bearbeitet werden. + Dateidetails + Bedingungen für die Bildaufnahme + ƒ/%s + ISO %s + %s MP + %s mm + %s s + in Ordner %1$s + Vorhandene Dateien ebenfalls hochladen + Nur während des Ladens der Batterie hochladen + /SofortUpload + Ungültige URL + Unsichtbar + Bezeichnung darf nicht leer sein + Letzte Sicherung:%1$s + Link + Link-Name + Hochladen und Bearbeiten erlauben + Bearbeitung + Dateien ablegen (nur Hochladen) + Nur anzeigen + Layout der Liste + Weitere Ergebnisse laden + Es befinden sich keine Dateien in diesem Ordner. + Die Datei wurde im lokalen Dateisystem nicht gefunden + %1$s/%2$s + Hier gibt es keine weiteren Ordner. + Ordner finden + Läuft ab: %1$s + Datei sperren + Gesperrt von %1$s + Gesperrt von der App %1$s + %1$s Android-App Meldungen + Keine App zum Senden von Protokolldateien gefunden. Bitte installieren Sie einen E-Mail-Client. + Angemeldet als %1$s + Anmelden + Der Link zu Ihrer %1$s Webseite, wenn Sie diese im Browser öffnen. + Lösche Logs + Aktualisieren + Suche Logs + Protokolldateien per E-Mail versenden + Logs: %1$d kB, Abfragetreffer %2$d / %3$d in %4$d ms + Lade… + Logs: %1$d kB, kein Filter + Protokolle (Logs) + Server befindet sich im Wartungsmodus + Daten löschen + Einstellungen, Datenbank und Server-Zertifikate von %1$s\'s Daten werden dauerhaft gelöscht.\n\nHerunter geladene Dateien bleiben unangetastet.\n\nDieser Vorgang kann eine Zeit dauern. + Verwalte Speicherplatz + Die Mediendatei kann nicht gestreamt werden + Konnte die Mediendatei nicht lesen + Mediendatei ist nicht korrekt encodiert + Wartezeit beim Abspielversuch abgelaufen + Der interne Player kann die Mediendatei nicht abspielen + Medien-Codec wird nicht unterstützt + Vorspulknopf + %1$s Musik Player + Wiedergabe-/Pause Knopf + Rückspulknopf + %1$s (wird abgespielt) + Neue zuerst + Älteste zuerst + A - Z + Z - A + Größte zuerst + Kleinste zuerst + Mehr + Es ist ein Fehler beim Verschieben der Datei oder des Ordners aufgetreten. + Es ist nicht möglich, einen Ordner in einen seiner Unterordner zu verschieben + Die Datei ist bereits im Zielordner vorhanden + Verschieben nicht möglich. Bitte prüfen Sie, ob die Datei existiert. + Verschieben nach… + Es ist ein Fehler beim Warten auf den Server aufgetreten, die Operation konnte nicht abgeschlossen werden. + Es ist ein Fehler beim Verbinden mit dem Server aufgetreten. + Es ist ein Fehler beim Warten auf den Server aufgetreten, die Operation konnte nicht abgeschlossen werden + Die Operation kann nicht abgeschlossen werden, der Server ist nicht erreichbar. + Neuer Kommentar… + Neuer Medienordner %1$s gefunden. + Foto + Video + Neue Benachrichtigung + Neue Version erstellt + Keine Aktionen für diesen Benutzer + Keine App verfügbar um Links zu öffnen + Es existieren keine Kalender + Keine App verfügbar, um E-Mail-Adressen zu verarbeiten + Keine Elemente + Keine App zur Kartenverwaltung verfügbar + Nur ein Konto zulässig + Keine App verfügbar um PDFs anzuzeigen + Keine App zum Senden der ausgewählten Dateien verfügbar + Bitte mindestens eine Berechtigung zum Teilen auswählen. + Senden + Notiz konnte nicht versandt werden + Notiz-Symbol + Ausführen der Aktion fehlgeschlagen. + Zeigt den Herunterlade-Fortschritt an + Downloads + Zeigt den Fortschritt der Dateisynchronisierung und die Ergebnisse an + Dateisynchronisierung + Für neue Medienordner und ähnliches Benachrichtigungen anzeigen + Allgemeine Benachrichtigungen + Musik-Player Fortschritt + Media-Player + Vom Server gesendete Push-Benachrichtigungen anzeigen: Erwähnungen in Kommentaren, Zugriff auf externe Freigaben erhalten, Ankündigungen von Administratoren erhalten, etc. + Push-Benachrichtigungen + Zeigt den Hochlade-Fortschritt an + Uploads + Benachrichtigungssymbol + Keine Benachrichtigungen + Bitte später noch einmal nachsehen. + Keine Internetverbindung + 1 Stunde + Online + Online-Status + Die Serverversion wird nicht mehr unterstützt, bitte aktualisieren Sie diesen! + Weiteres Menü + PIN eingeben + Die PIN wird jedes mal beim Start der App abgefragt + Bitte geben Sie Ihre PIN ein + Die PINs stimmen nicht überein + Bitte Ihre PIN nochmals eingeben + Ihre PIN löschen + PIN gelöscht + PIN gespeichert + PIN nicht korrekt + Passwortgeschützte PDF-Datei konnte nicht geöffnet werden. Bitte verwenden Sie einen externen PDF-Viewer. + Auf eine Seite klicken um hereinzuzoomen + Zulassen + Ablehnen + Sie haben nicht die erforderlichen Rechte, um Dateien hoch- oder herunterzuladen. + Keine App gefunden, mit der ein Bid gesetzt werden könnte + An Startbildschirm anheften + %1$s öffnen + 389 KB + platzhalter.txt + 12:23:45 + Dies ist ein Platzhalter + 18.5.2012 12:23 + Stopp + Umschalten + Bei deaktiviertem Energiesparmodus werden Daten auch bei niedrigem Batterieladezustand hochgeladen. + gelöscht + im Originalordner behalten + in den App-Ordner verschoben + Was tun, wenn die Datei bereits vorhanden ist? + Mich jedesmal fragen + Hochladen überspringen + Version auf dem Server überschreiben + Neue Version umbenennen + Was tun, wenn die Datei bereits vorhanden ist? + Konto hinzufügen + Kalender & Kontakte synchronisieren + Weder F-Droid noch Google-Play sind installiert + DAVx5 (zuvor bekannt als DAVdroid) (v1.3.0+) für aktuelles Konto einrichten + Kalender- & Adress-Synchronisierung konfigurieren + Über + Details + Dev + Allgemein + Mehr + Tägliche Sicherung Ihrer Kalender & Kontakte + Tägliche Sicherung Ihrer Kontakte + Unerwarteter Fehler beim Einrichten von DAVx5 (früher bekannt als DAVdroid) + Ende-zu-Ende-Verschlüsselung ist eingerichtet! + E2E-Gedächtnisstütze + Um die Gedächtnisstütze anzuzeigen, aktivieren Sie bitte Geräte-Zugangsdaten + Benachrichtigungen der Mediensuche anzeigen + Über neu gefundene Medienordner informieren + GNU General Public Lizenz, version 2 + Hilfe + Impressum + Originaldatei wird… + Originaldatei wird… + Nach Datum in Unterordnern speichern + Unterordner benutzen + Unterordneroptionen + Diesem Client Ende-zu-Ende-Verschlüsselung hinzufügen + Lizenz + App-PIN + Geräte-Anmeldedaten aktiviert + Keine Geräte-Anmeldedaten eingerichtet. + Nichts + App schützen mittels + Geräte-Anmeldedaten + PIN + Konten verwalten + Einem Freund empfehlen + Verschlüsselung lokal entfernen + Ende-zu-Ende-Verschlüsselung einrichten + App-Umschalter anzeigen + Nextcloud-App-Vorschläge in der Navigationsüberschrift + Versteckte Dateien anzeigen + Zum Programmcode + Speicherordner + Ordner für \"Automatisches Hochladen\" verwalten + Lokaler Ordner + Remote-Ordner + Design + Dunkel + Hell + Systemvorgaben verwenden + + + Zur Optimierung unserer App erfassen wir anonymisierte Daten. Hierzu nutzen wir Software Lösungen verschiedener Partner. Wir möchten Ihnen volle Transparenz und Entscheidungsgewalt über die Verarbeitung und Erfassung Ihrer anonymisierten Nutzungsdaten geben. Ihre Einstellungen können Sie auch später jederzeit in den Einstellungen unter Datenschutz ändern. Bitte beachten Sie jedoch, dass die Datenerfassungen einen erheblichen Beitrag zur Optimierung dieser App leisten und Sie diese Optimierungen durch die Unterbindung der Datenübermittlung verhindern. + Erfoderliche Datenerfassung + Die Erfassung dieser Daten ist notwendig, um wesentliche Funktionen der App nutzen zu können. + Analyse-Datenerfassung zur bedarfsgerechten Gestaltung + Diese Daten helfen uns, die App Nutzung für Sie zu optimieren und Systemabstürze und Fehler schneller zu identifizieren. + Diese App verwendet Cookies und ähnliche Technologien (Tools). + Mit einem Klick auf Zustimmen akzeptieren Sie die Verarbeitung und auch die Weitergabe Ihrer Daten an + Drittanbieter. Die Daten werden für Analysen, Retargeting und zur Ausspielung von personalisierten Inhalten + und Werbung auf Seiten der Telekom, sowie auf Drittanbieterseiten genutzt. Weitere Informationen, auch zur + Datenverarbeitung durch Drittanbieter, finden Sie in den Einstellungen sowie in unseren %s. Sie können die + Verwendung der Tools %s oder jederzeit über ihre %s anpassen. + Datenschutz-Einstellungen + Akzeptieren + Einstellungen speichern + Datenschutzhinweise + ablehnen + Einstellungen + + + Bildvorschau + Keine lokale Datei für die Vorschau vorhanden + Bild kann nicht angezeigt werden + Entschuldigung + Datenschutz + Neuer Name + Push Benachrichtigungen deaktiviert aufgrund der Abhängigkeiten zu proprietären Google Play Diensten. + Keine Push Benachrichtigungen aufgrund veralteter Benutzersitzung. Bitte erwägen Sie das erneute Hinzufügen Ihres Kontos. + Push Benachrichtigungen aktuell nicht verfügbar. + QR-Code konnte nicht gelesen werden! + Probieren Sie %1$s auf Ihrem Smartphone + Ich empfehle Ihnen %1$s auf Ihrem Smartphone zu nutzen.\nHerunterladen: %2$s + %1$s oder %2$s + Inhalt aktualisieren + Neu laden + (entfernt) + Datei nicht gefunden! + Sie können die Ende-zu-Ende-Verschlüsselung lokal auf diesem Client entfernen + Sie können die Ende-zu-Ende-Verschlüsselung lokal auf diesem Client entfernen. Die verschlüsselten Dateien bleiben auf dem Server, werden aber nicht mehr mit diesem Computer synchronisiert. + Löschung fehlgeschlagen + Entfernen der Benachrichtigungen fehlgeschlagen. + Entfernen + Gelöscht + Geben Sie einen neuen Namen ein + Die lokale Kopie konnte nicht umbenannt werden. Versuche es mit einem anderen Namen. + Umbenennen nicht möglich. Der Name wird bereits verwendet. + Das Weiterverteilen ist nicht erlaubt + Resharing/Wiederteilen ist nicht erlaubt. + Kein verkleinertes Bild verfügbar. Vollbild herunterladen? + Datei wiederherstellen + Letzte Sicherung wiederherstellen + Gelöschte Datei wiederherstellen + Ausgewählte wiederherstellen + Abrufen der Datei… + Laden des Dokuments fehlgeschlagen! + Anmelden mit QR-Code + Seite scannen + Schützt Ihre Daten + Selbst gehostete Produktivität + Durchsuchen und teilen + Alle Aktionen schnell erreichbar + Aktivitäten, Freigaben, … + Alles schnell erreichbar + All ihre Konten + an einem Ort + Automatisches Hochladen + ihrer Fotos & Videos + Kalender & Kontakte + Synchronisiere mit DAVx5 + Fehler beim Abrufen der Suchergebnisse + Alle auswählen + Medien-Ordner auswählen + Bitte eine Vorlage auswählen + Vorlage auswählen + Senden + Notiz an Empfänger senden + Freigabe senden + Icon für den Senden-Button + Setze als + Notiz setzen + Nutze Bild als + Status setzen + Statusnachricht setzen + Während der Einrichtung der Ende-zu-Ende-Verschlüsselung erhalten Sie eine zufällige 12-Wörter-Gedächtnisstütze, die Sie benötigen, um Ihre Dateien auf anderen Geräten zu öffnen. Diese wird nur auf diesem Gerät gespeichert und kann in diesem Bildschirm erneut angezeigt werden. Bitte notieren Sie es an einem sicheren Ort! + Teilen + Teilen & Link kopieren + Teilen + %1$s + Läuft am %1$s ab + Teile %1$s + %1$s (Gruppe) + Internen Link teilen + Der interne Freigabelink funktioniert nur für Benutzer mit Zugriff auf diese Datei + Der interne Freigabelink funktioniert nur für Benutzer mit Zugriff auf diesen Ordner + auf %1$s + Link teilen + Sie müssen ein Passwort eingeben + Es ist ein Fehler beim Freigeben der Datei oder des Ordners aufgetreten. + Teilen nicht möglich. Bitte prüfen Sie, ob die Datei existiert. + diese Datei zu teilen + Passwort eingeben (optional) + Passwort eingeben + Teile Link (%1$s) + Ablaufdatum setzen + Passwort setzen + Passwortgeschützt + Kann bearbeiten + Dateiablage + Sichere Dateiablage + Nur anzeigen + Berechtigungen zum Teilen + %1$s (remote) + %1$s (Unterhaltung) + Name, Federated-Cloud-ID oder E-Mail-Adresse … + Neue E-Mail senden + Notiz an Empfänger + Einstellungen + Download verbergen + Link teilen + Link teilen + Entfernen + Teilen mit… + Avatar des geteilten Nutzers + Teilen + geteilt + Über einen Link geteilt + Geteilt mit Ihnen von %1$s + Freigeben fehlgeschlagen + Fotos anzeigen + Videos anzeigen + Mit Provider anmelden + Zulassen, dass %1$s auf Ihr Nextcloud Konto %2$s zugreifen darf? + Sortiere nach + Neue zuerst + Älteste zuerst + A - Z + Z - A + Kleinste zuerst + Größste zuerst + Ausblenden + Details + Die Identität des Servers konnte nicht verifiziert werden + Land: + Üblicher Name: + Ort: + Organisation: + Organisationseinheit: + Bundesland: + Fingerabdruck: + Ausgestellt von: + Signatur: + Algorithmus: + Ausgestellt für: + Gültigkeit: + Von: + Bis: + - Keine Informationen über den Fehler + Das Zertifikat konnte nicht gespeichert werden + Das Zertifikat konnte nicht angezeigt werden. + Soll diesem Zertifikat trotzdem vertraut werden? + - Das Zertifikat des Servers ist abgelaufen + - Das Zertifikat des Servers ist nicht vertrauenswürdig + - Das Gültigkeitsdatum des Serverzertifikats liegt in der Zukunft + - Die Adresse stimmt nicht mit dem im Zertifikat angegebenen Hostnamen überein + Statusnachricht + Kamera + Speicherort auswählen + Standard + Dokumente + Downloads + Interner Speicher + Filme + Musik + Voller Zugriff + Medien schreibgeschützt + Bilder + Die selbst gehostete Produktivitätsplattform gibt Ihnen die volle Kontrolle.\n\nFunktionen:\n* Einfache, moderne Oberfläche. Angepasst an das Design Ihres Servers\n* +.\n\nEigenschaften:\n* Einfache, moderne an Ihr Design angepasste Oberfläche\n* Laden Sie Ihre Dateien auf Ihren Nextcloud Server\n* Teilen Sie Ihre Dateien mit anderen\n* Halten Sie Ihre favorisierten Dateien und Ordner synchron\n* Suche über alle Ordner auf Ihrem Server\n* Automatisches Hochladen von Fotos und Videos von Ihrem Gerät\n* Bleiben Sie auf dem Laufenden durch Benachrichtigungen\n* Mehrere Konten gleichzeitig\n* Geschützter Zugriff durch Fingerabdruck oder PIN\n* Integration mit DAVx5 (ehemals DAVdroid) für einfaches Konfigurieren der Kalender- & Kontakte-Synchronisierung\n\nBitte melden Sie alle Probleme unter https://github.com/nextcloud/android/issues oder diskutieren Sie über die App auf https://help.nextcloud.com/c/clients/android\n\nWas ist die Nextcloud? Die Nextcloud ist ein privater Dateiaustausch & -synchronisierungs Server. Nextcloud ist Open Source und kann auf einem eigenen Server installiert werden. So behalten Sie die Kontrolle über Ihre Fotos, Ihre Kalender und Kontakte, Ihre Dokumente und alle anderen Daten.\n\nSchauen Sie sich Nextcloud an unter https://nextcloud.com. + Die selbst gehostete Produktivitätsplattform gibt Ihnen die volle Kontrolle.\nDies ist die offizielle Entwicklungsversion und bietet neue, ungetestete Funkitionen. Dies kann zu Instabilitäten und Datenverlust führen. Die App ist für alle Benutzer, die Testen wollen und falls nötig Fehlerberichte erstellen. Nutzen Sie diese Version nicht für Ihre Produktivarbeit!\n\nSowohl die reguläre als auch die Entwicklerversion sind auf F-droid verfügbar und können parallel installiert werden. + Die selbst gehostete Produktivitätsplattform gibt Ihnen die volle Kontrolle + Die selbst gehostete Produktivitätsplattform gibt Ihnen die volle Kontrolle (Vorab- Entwicklungsversion) + Streamen mit… + Internes Streamen nicht möglich + Laden Sie sich den Inhalt herunter oder nutzen Sie eine externe App. + Strikter Modus: keine HTTP-Verbindung erlaubt! + Jahr/Monat/Tag + Jahr/Monat + Jahr + \"%1$s\" wurde mit Ihnen geteilt + %1$s hat \"%2$s\" mit Ihnen geteilt + Nur Fotos + Fotos & Videos + Nur Videos + Vorschlagen + Konflikte gefunden + Der Ordner %1$s existiert nicht mehr + Synchronisierung von %1$s konnte nicht abgeschlossen werden + Falsches Passwort für %1$s + Synchronhalten schlug fehl. + Synchronisierung fehlgeschlagen + Synchronisierung fehlgeschlagen. Sie müssen sich erneut anmelden + Dateiinhalte bereits synchronisiert + Synchronisation von Ordner %1$s konnte nicht fertiggestellt werden + Seit Version 1.3.16 werden Dateien, die von diesem Gerät aus hochgeladen werden, in den lokalen Ordner %1$s kopiert, um Datenverlust zu vermeiden, wenn eine einzelne Datei mit mehreren Konten synchronisiert wird.\n\nInfolge dieser Änderung wurden alle Dateien, die mit vorherigen Versionen dieser App hochgeladen wurden, in den Ordner %2$s verschoben. Jedoch ist während der Konto-Synchronisierung ein Fehler aufgetreten, der das Abschließen dieses Vorgangs verhindert hat. Sie können die Datei(en) entweder wie sie sind belassen und den Link zu %3$s löschen oder die Datei(en) in den Ordner %1$s verschieben und den Link zu %4$s beibehalten.\n\nUnten befindet sich eine Liste der lokalen Datei(en) und der mit ihnen verbundenen entfernten Datei(en) in %5$s. + Einige lokale Dateien wurden vergessen + Neueste Version der Datei wird geladen. + Zu synchronisierende Elemente auswählen + Speicherplatz freimachen + %1$s ist %2$s, jedoch ist nur noch %3$s auf diesem Gerät verfügbar. + Nicht genügend Speicherplatz + Status-synchronisieren-Button + Dateien + Einstellungs-Button + Ordner konfigurieren + Die Sofort-Uploads wurden vollständig überarbeitet. Konfiguriere Sie Ihren automatischen Uploader im Hauptmenü.\n\nGenießen Sie den verbesserten Auto-Upload. + Keine Medien-Ordner gefunden + Für %1$s + Typ + Synchronisiert + Schlagworte + Prüfe Server-Verbindung + 30 Minuten + Diese Woche + Vorschaubild + Miniaturbild für vorhandene Datei + Miniaturbild für neue Datei + Laden dauert länger als erwartet + Heute + Gelöschte Dateien + Keine gelöschten Dateien + Hier können Sie gelöschte Dateien wiederherstellen. + Datei %1$s konnte nicht gelöscht werden! + Datei %1$s konnte nicht wiederhergestellt werden! + Endgültig löschen + Laden des Papierkorbs fehlgeschlagen! + Dateien konnten nicht endgültig gelöscht werden! + Datei entsperren + Es gibt ungelesene Kommentare + Verschlüsselung aufheben + Aus den Favoriten entfernen + Es ist ein Fehler beim Entfernen der Freigabe für diese Datei oder den Ordner aufgetreten. + Entfernen der Freigabe nicht möglich. Bitte prüfen Sie, ob die Datei existiert. + diese Datei nicht mehr zu teilen + Freigabe aufheben fehlgeschlagen + Zugriff über nicht vertrauenswürdige Domain. Weitere Informationen finden sich in der Dokumentation. + Es ist ein Fehler bei der Aktualisierungd der Freigabe aufgetreten. + Aktualisierung nicht möglich. Bitte prüfen Sie, ob die Datei existiert. + zum Aktualisieren dieser Freigabe + Aktualisierung der Freigabe fehlgeschlagen + Datei kann lokal nicht erstellt werden + Dateien hochladen von… + Inhalt anderer Apps hochladen + Von Kamera hochladen + Dateiname + Dateityp + Google Maps Abkürzungs-Datei (%s) + Internet Abkürzungs-Datei (%s) + Schnipsel Textdatei (.txt) + Dateinamen und -typ zum Hochladen eingeben + Dateien hochladen + Button, um Objekt hochzuladen + Löschen + Keine Uploads verfügbar + Laden Sie Inhalte hoch oder aktivieren Sie den Auto Upload. + Konflikt lösen + Lokaler Speicher voll + Datei konnte nicht in den lokalen Speicher kopiert werden + Fehler beim Sperren des Ordners + Verschlüsselung ist nur möglich mit >= Android 5.0 + Es steht nicht genügend Speicherplatz zur Verfügung, um die ausgewählten Dateien in das Verzeichnis %1$s zu kopieren. Sollen diese stattdessen verschoben werden? + Dokument von der Kamera scannen + Synchronisierungskonflikt, bitte manuell beheben + Unbekannter Fehler + Auswählen + Hochladen + Empfangene Daten enthalten keine gültige Datei. + %1$s darf eine empfangene Datei nicht lesen + Es ist ein Fehler beim Kopieren dieser Datei in einen temporären Ordner aufgetreten. Bitte versuchen Sie erneut die Datei zu senden. + Die zum Hochladen ausgewählte Datei kann nicht gefunden werden. Bitte prüfen Sie ob die Datei existiert. + Diese Datei kann nicht hochgeladen werden + Keine Datei zum Hochladen + Ordnername + Kontoinformationen + Datenschutz + Datenschutz + Datenschutzbestimmungen + Verwendete OpenSource Software + Info + Bedienung + Die Ende-zu-Ende Verschlüsselung wurde bereits auf einem anderen Gerät eingerichtet. Bitte geben Sie Ihre Passphrase ein, damit die Dateien synchronisiert und entschlüsselt werden. + Hochladeordner auswählen + Konnte %1$s nicht hochladen + Hochladen fehlgeschlagen, bitte erneut anmelden + Dateikonflikt beim Hochladen + Wählen Sie, welche Version von %1$s behalten werden soll + Hochladen fehlgeschlagen + Upload Option: + Datei in Ordner %1$s verschieben + Quellverzeichnis ist nur lesbar; Datei wird nur hochgeladen + Hochladen und im Quellordner behalten + Hochladen und Datei im Quellordner löschen + diesen Ordner hochzuladen + %1$d%% Hochladen %2$s + Lade hoch… + %1$s hochgeladen + Beenden + Einrichten + Es sind keine %1$s Konten auf Ihrem Gerät eingerichtet. Bitte erstellen Sie zuerst ein Konto. + Kein Konto gefunden + Aktuell + Fehlgeschlagene/Wartende neustarten + Hochgeladen + Warte auf Hochladen + Uploads + Abgebrochen + Konflikt + Verbindungsfehler + Zugangsdaten falsch + Dateifehler + Ordnerfehler + Lokale Datei nicht gefunden + Berechtigungsfehler + Serverzertifikat ist nicht vertrauenswürdig + Serverversion wird abgerufen … + App beendet + Fertiggestellt + Unbekannter Fehler + Virus gefunden. Das Hochladen kann nicht abgeschlossen werden! + Warte auf Verlassen des Energiesparmodus + Warte auf das Aufladen des Gerätes + Warte auf nicht gebührenpflichtiges WLAN + Nutzer + Adresse + E-Mail + Telefonnummer + Twitter + Webseite + Fehler beim Laden der Nutzer-Informationen + Keine persönlichen Infos eingestellt + Fügen Sie Name, Bild und Kontaktdaten auf Ihrer Profilseite hinzu. + Benutzername + Herunterladen + Video Überlagerungsicon + Bitte warten… + Überprüfe gespeicherte Anmeldeinformationen + Kopiere Datei von privatem Speicher + Bitte aktualisieren Sie die Android System WebView-App für eine Anmeldung + Aktualisieren + Android System WebView aktualisieren + Was-gibt\'s-Neues-Bild + Überspringen + Neu in %1$s + Wie ist Ihr Status? + Widgets sind nur auf %1$s 25 oder neuer verfügbar + Nicht verfügbar + E-Mail senden + Speicherordner existiert nicht! + Ursache könnte die Wiederherstellung einer Sicherungskopie auf einem anderen Gerät sein. Der Standard-Ordner wird jetzt wieder verwendet. Bitte überprüfen Sie die Einstellungen bezüglich des Speicherortes. + + Inhalte von %1$d Datei konnten nicht synchronisiert werden (Konflikte: %2$d) + Inhalte von %1$d Dateien konnten nicht synchronisiert werden (Konflikte: %2$d) + + + %1$d Datei im %2$s-Verzeichnis konnten nicht kopiert werden nach + %1$d Dateien im %2$s-Verzeichnis konnten nicht kopiert werden nach + + + %1$d Termin in %2$s geschrieben + %1$d Termine in %2$s geschrieben + + + %1$d neue UID erstellt + %1$d neue UIDs erstellt + + + %d Eintrag verarbeitet. + %d Einträge verarbeitet. + + + %d doppelter Eintrag gefunden. + %d doppelte Einträge gefunden. + + + %d Datei exportiert + %d Dateien exportiert + + + %d Datei konnte nicht exportiert werden + %d Dateien konnten nicht exportiert werden + + + %d Datei exportiert, der Rest wurde aufgrund von Fehlern übersprungen + %d Dateien exportiert, der Rest wurde aufgrund von Fehlern übersprungen + + + %d Datei wird exportiert. Einzelheiten finden Sie in der Benachrichtigung. + %d Dateien werden exportiert. Einzelheiten finden Sie in der Benachrichtigung. + + + %1$d Ordner + %1$d Ordner + + + %1$d Datei + %1$d Dateien + + + Zeige %1$dversteckten Ordner + Zeige %1$d versteckte Ordner + + + %d ausgewählt + %d ausgewählt + + diff --git a/app/src/main/res/values-night/colors.xml b/app/src/main/res/values-night/colors.xml index 08bf64d552ad..e926fdcc432f 100644 --- a/app/src/main/res/values-night/colors.xml +++ b/app/src/main/res/values-night/colors.xml @@ -48,4 +48,68 @@ #1E1E1E @android:color/white + + + #FFFFFF + @color/grey_30 + @color/grey_30 + #CCCCCC + @color/grey_70 + @color/grey_80 + #2D2D2D + @color/grey_70 + @color/grey_70 + + + @color/grey_80 + @color/grey_0 + + + @color/grey_80 + @color/grey_0 + + + @color/grey_60 + @color/grey_0 + @color/grey_0 + @color/grey_30 + #FFFFFF + @color/grey_30 + @color/grey_80 + #FFFFFF + + + @color/grey_80 + @color/grey_30 + @color/grey_0 + + + @color/grey_80 + @color/grey_0 + @color/grey_80 + + + @color/grey_70 + @color/grey_60 + + + @color/grey_70 + @color/grey_70 + + + #FFFFFF + @color/grey_30 + @color/grey_0 + @color/grey_0 + @color/grey_0 + @color/grey_0 + @color/grey_60 + @color/grey_0 + #FFFFFF + + + #121212 + @color/grey_0 + @color/grey_80 + @color/grey_80 diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml index 58fcdecf1fc2..b05b582157a7 100644 --- a/app/src/main/res/values/colors.xml +++ b/app/src/main/res/values/colors.xml @@ -82,4 +82,93 @@ @android:color/white #666666 #A5A5A5 + + + #191919 + @color/primary + #191919 + #191919 + @color/grey_30 + @android:color/white + #FFFFFF + @color/grey_0 + #CCCCCC + #77c4ff + #B3FFFFFF + @color/grey_10 + + + #101010 + #F2F2F2 + #E5E5E5 + #B2B2B2 + #666666 + #4C4C4C + #333333 + + + @color/design_snackbar_background_color + @color/white + + + #FFFFFF + #191919 + + + @color/grey_0 + #191919 + @color/primary + #191919 + @color/primary + @color/grey_30 + @color/white + #191919 + + + #FFFFFF + #191919 + #191919 + + + #FFFFFF + #191919 + #FFFFFF + + + @color/primary + #F399C7 + #FFFFFF + @color/grey_30 + @color/grey_10 + @color/grey_0 + + + @color/primary + @color/grey_30 + @color/grey_30 + #CCCCCC + + + #191919 + @color/grey_30 + #191919 + #191919 + #191919 + #191919 + @color/grey_30 + #191919 + #000000 + #191919 + #F6E5EB + #C16F81 + #0D39DF + #0099ff + + + @color/grey_0 + #191919 + @color/grey_0 + @color/grey_30 + #77b6bb + #5077b6bb diff --git a/app/src/main/res/values/dimens.xml b/app/src/main/res/values/dimens.xml new file mode 100644 index 000000000000..cc9e25255a10 --- /dev/null +++ b/app/src/main/res/values/dimens.xml @@ -0,0 +1,31 @@ + + + 4dp + 16dp + 24dp + 6dp + 18sp + 15sp + 15dp + 56dp + 86dp + 80dp + 11sp + 30dp + 55dp + 258dp + 17sp + 20dp + 160dp + 50dp + 150dp + 55dp + 48dp + 48dp + 24dp + 26dp + 20sp + 145dp + 1dp + 13sp + \ No newline at end of file diff --git a/app/src/main/res/values/setup.xml b/app/src/main/res/values/setup.xml index 6786cf27db48..a012a6a3a782 100644 --- a/app/src/main/res/values/setup.xml +++ b/app/src/main/res/values/setup.xml @@ -73,7 +73,7 @@ true false true - https://github.com/nextcloud/android + https://static.magentacloud.de/licences/android.html true https://www.gnu.org/licenses/gpl-2.0.html diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index c60ee567ec2a..aa67ccc8dab3 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -201,6 +201,27 @@ Created %1$d fresh UID Created %1$d fresh UIDs + + To optimize your app, we collect anonymous data. For this we use + software solutions of different partners. We would like to give you full transparency and decision-making power over the processing and collection of your anonymized usage data. You can also change your settings at any time later in the app settings under data protection. Please note, however, that data collection makes a considerable contribution to the optimization of this app and you prevent this optimization by preventing data transmission. + Required data collection + The collection of this data is necessary to be able to use essential + functions of the app. + Analysis-data acquisition for the design + This data helps us to optimize the app usage for you and to identify system crashes and errors more quickly. + This app uses Cookies and similar technologies (tools). By + clicking Accept, you accept the processing and also the Transfer of your data to third parties. The data will + be used for Analysis, retargeting and to Display personalized Content and Advertising on sites and + third-party sites. You can find further information, including Information on data processing by third-party + Providers, in the Settings and in our %s. You can %s the use of the Tools or customize them at any time in the + %s. + Privacy Settings + Accept + Save Settings + Privacy Policy + reject + Settings + Processed %d entry. Processed %d entries. @@ -1085,6 +1106,12 @@ Not possible without internet connection Scan page Done + Info + Data Privacy + Privacy Settings + Privacy Policy + Used OpenSource Software + Service Generating PDF… Error starting document scan PDF generation failed