Merge branch 'development' into feature/notes
13
README.md
@@ -29,20 +29,21 @@
|
||||
<br />
|
||||
<br />
|
||||
<p>
|
||||
<a href="https://play.google.com/store/apps/details?id=de.astubenbord.paperless_mobile">
|
||||
<img src="resources/get_it_on_google_play_en.svg" width="140px">
|
||||
<a href='https://play.google.com/store/apps/details?id=de.astubenbord.paperless_mobile&hl=de&gl=US&pcampaignid=pcampaignidMKT-Other-global-all-co-prtnr-py-PartBadge-Mar2515-1'>
|
||||
<img alt='Get it on Google Play' src='https://play.google.com/intl/en_us/badges/static/images/badges/en_badge_web_generic.png' height="80"/>
|
||||
</a>
|
||||
<a href="https://f-droid.org/packages/de.astubenbord.paperless_mobile">
|
||||
<img alt="Get it on F-Droid" src="https://fdroid.gitlab.io/artwork/badge/get-it-on.png" height="80">
|
||||
</a>
|
||||
</p>
|
||||
<!--<a href="https://github.com/astubenbord/paperless-mobile">View Demo</a>
|
||||
·-->
|
||||
<a href="https://github.com/astubenbord/paperless-mobile/issues">Report Bug</a>
|
||||
·
|
||||
<a href="https://github.com/astubenbord/paperless-mobile/discussions/categories/feature-requests">Request Feature</a>
|
||||
</p>
|
||||
</div>
|
||||
|
||||
## Important Notes
|
||||
* ⚠️ Breaking changes **are still expected** between updates! Please perform a **clean instal** after updating the app before filing an issue or leaving a 1-star review in the App-Store!
|
||||
## ⚠️ Important Notes
|
||||
- Breaking changes **are still expected** between updates! Please perform a **clean instal** after updating the app before filing an issue or leaving a 1-star review in the App-Store!
|
||||
|
||||
<!-- ABOUT THE PROJECT -->
|
||||
## About The Project
|
||||
|
||||
@@ -103,9 +103,7 @@ android.applicationVariants.all { variant ->
|
||||
def abiName = output.getFilter(OutputFile.ABI)
|
||||
def abiVersionCode = project.ext.abiCodes.get(abiName)
|
||||
if (abiVersionCode != null) {
|
||||
output.versionCodeOverride = abiVersionCode * 1000 + variant.versionCode
|
||||
println(abiName + ": " + output.versionCodeOverride)
|
||||
// output.versionCodeOverride = variant.versionCode * 10 + abiVersionCode
|
||||
output.versionCodeOverride = variant.versionCode * 10 + abiVersionCode
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
- Beheben eines Fehlers durch welchen Dokumente nicht mehr bearbeitet werden konnten
|
||||
@@ -0,0 +1,2 @@
|
||||
- Kleinere visuelle Änderungen
|
||||
- Aktualisieren der Store-Präsenz
|
||||
@@ -0,0 +1 @@
|
||||
- Beheben von versehentlich hardcoded Theme
|
||||
@@ -0,0 +1 @@
|
||||
- Fixed a bug which caused documents not being editable anymore
|
||||
@@ -0,0 +1,2 @@
|
||||
- Minor visual updates
|
||||
- Update store assets
|
||||
@@ -0,0 +1 @@
|
||||
- Fix accidentally hardcoded Theme mode
|
||||
@@ -1,10 +1,8 @@
|
||||
An (almost) fully fledged mobile Paperless client compatible with Paperless.
|
||||
|
||||
Using this app requires access to running a Paperless-ng*x instance.
|
||||
|
||||
* View and search documents
|
||||
* Add, delete or edit documents
|
||||
* Share, download, print or preview your documents
|
||||
* Share, download, print and preview your documents
|
||||
* Manage correspondents, document types, tags and storage paths
|
||||
* Scan and upload documents with preset correspondent, document type, tags and creation date
|
||||
* Review and quickly process newly added documents in the inbox
|
||||
@@ -14,4 +12,4 @@ Using this app requires access to running a Paperless-ng*x instance.
|
||||
* Modern, intuitive UI built according to the Material Design 3 specification
|
||||
* Light and dark theme
|
||||
* Support for dynamic color (Android 12+ only)
|
||||
* Available in English, German, French, Catalan, Polish, Czech and Turkish language
|
||||
* Available in English, German, French, Spanish, Catalan, Polish, Czech, Russian and Turkish language with more to come
|
||||
|
||||
|
Before Width: | Height: | Size: 45 KiB After Width: | Height: | Size: 68 KiB |
|
Before Width: | Height: | Size: 252 KiB After Width: | Height: | Size: 350 KiB |
|
Before Width: | Height: | Size: 252 KiB After Width: | Height: | Size: 353 KiB |
|
Before Width: | Height: | Size: 249 KiB After Width: | Height: | Size: 532 KiB |
|
Before Width: | Height: | Size: 263 KiB After Width: | Height: | Size: 405 KiB |
|
Before Width: | Height: | Size: 252 KiB After Width: | Height: | Size: 190 KiB |
|
Before Width: | Height: | Size: 194 KiB After Width: | Height: | Size: 256 KiB |
|
Before Width: | Height: | Size: 258 KiB After Width: | Height: | Size: 274 KiB |
|
Before Width: | Height: | Size: 161 KiB After Width: | Height: | Size: 353 KiB |
@@ -1 +1 @@
|
||||
An (almost) fully fledged mobile Paperless client.
|
||||
An (almost) fully fledged mobile paperless-ng*x client.
|
||||
@@ -115,11 +115,11 @@ class FileService {
|
||||
formatBytes(await getDirSizeInBytes(consumptionDir));
|
||||
|
||||
logger.ft(
|
||||
"Removing scans...",
|
||||
"Clearing scans directory...",
|
||||
className: runtimeType.toString(),
|
||||
methodName: "clearUserData",
|
||||
);
|
||||
await _temporaryScansDirectory.delete(recursive: true);
|
||||
await _temporaryScansDirectory.clear();
|
||||
logger.ft(
|
||||
"Removed $scanDirSize...",
|
||||
className: runtimeType.toString(),
|
||||
|
||||
@@ -2,6 +2,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:flutter_form_builder/flutter_form_builder.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
import 'package:paperless_api/paperless_api.dart';
|
||||
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
|
||||
import 'package:paperless_mobile/core/widgets/form_builder_fields/extended_date_range_form_field/extended_date_range_dialog.dart';
|
||||
import 'package:paperless_mobile/core/widgets/form_builder_fields/extended_date_range_form_field/relative_date_range_picker_helper.dart';
|
||||
import 'package:paperless_mobile/generated/l10n/app_localizations.dart';
|
||||
@@ -11,6 +12,7 @@ class FormBuilderExtendedDateRangePicker extends StatefulWidget {
|
||||
final String labelText;
|
||||
final DateRangeQuery initialValue;
|
||||
final void Function(DateRangeQuery? query)? onChanged;
|
||||
final EdgeInsets padding;
|
||||
|
||||
const FormBuilderExtendedDateRangePicker({
|
||||
super.key,
|
||||
@@ -18,6 +20,7 @@ class FormBuilderExtendedDateRangePicker extends StatefulWidget {
|
||||
required this.labelText,
|
||||
required this.initialValue,
|
||||
this.onChanged,
|
||||
required this.padding,
|
||||
});
|
||||
|
||||
@override
|
||||
@@ -49,29 +52,36 @@ class _FormBuilderExtendedDateRangePickerState
|
||||
builder: (field) {
|
||||
return Column(
|
||||
children: [
|
||||
TextFormField(
|
||||
controller: _textEditingController,
|
||||
readOnly: true,
|
||||
onTap: () => _showExtendedDateRangePicker(field),
|
||||
decoration: InputDecoration(
|
||||
prefixIcon: const Icon(Icons.date_range),
|
||||
labelText: widget.labelText,
|
||||
suffixIcon: _textEditingController.text.isNotEmpty
|
||||
? IconButton(
|
||||
icon: const Icon(Icons.clear),
|
||||
onPressed: () {
|
||||
field.didChange(const UnsetDateRangeQuery());
|
||||
},
|
||||
)
|
||||
: null,
|
||||
Padding(
|
||||
padding: widget.padding.copyWith(bottom: 0),
|
||||
child: TextFormField(
|
||||
controller: _textEditingController,
|
||||
readOnly: true,
|
||||
onTap: () => _showExtendedDateRangePicker(field),
|
||||
decoration: InputDecoration(
|
||||
prefixIcon: const Icon(Icons.date_range),
|
||||
labelText: widget.labelText,
|
||||
suffixIcon: _textEditingController.text.isNotEmpty
|
||||
? IconButton(
|
||||
icon: const Icon(Icons.clear),
|
||||
onPressed: () {
|
||||
field.didChange(const UnsetDateRangeQuery());
|
||||
},
|
||||
)
|
||||
: null,
|
||||
),
|
||||
),
|
||||
),
|
||||
MediaQuery.removePadding(
|
||||
context: context,
|
||||
removeLeft: true,
|
||||
removeRight: true,
|
||||
child: RelativeDateRangePickerHelper(field: field),
|
||||
),
|
||||
RelativeDateRangePickerHelper(
|
||||
field: field,
|
||||
padding: widget.padding,
|
||||
)
|
||||
// MediaQuery.removePadding(
|
||||
//context: context,
|
||||
//removeLeft: true,
|
||||
//removeRight: true,
|
||||
//child: ,
|
||||
//),
|
||||
],
|
||||
);
|
||||
},
|
||||
|
||||
@@ -6,11 +6,13 @@ import 'package:paperless_mobile/generated/l10n/app_localizations.dart';
|
||||
class RelativeDateRangePickerHelper extends StatefulWidget {
|
||||
final FormFieldState<DateRangeQuery> field;
|
||||
final void Function(DateRangeQuery value)? onChanged;
|
||||
final EdgeInsets padding;
|
||||
|
||||
const RelativeDateRangePickerHelper({
|
||||
super.key,
|
||||
required this.field,
|
||||
this.onChanged,
|
||||
required this.padding,
|
||||
});
|
||||
|
||||
@override
|
||||
@@ -24,25 +26,32 @@ class _RelativeDateRangePickerHelperState
|
||||
Widget build(BuildContext context) {
|
||||
return SizedBox(
|
||||
height: 64,
|
||||
child: ListView.separated(
|
||||
itemCount: _options.length,
|
||||
separatorBuilder: (context, index) => const SizedBox(width: 8.0),
|
||||
itemBuilder: (context, index) {
|
||||
final option = _options[index];
|
||||
return ColoredChipWrapper(
|
||||
child: FilterChip(
|
||||
label: Text(option.title),
|
||||
onSelected: (isSelected) {
|
||||
final value =
|
||||
isSelected ? option.value : const RelativeDateRangeQuery();
|
||||
widget.field.didChange(value);
|
||||
widget.onChanged?.call(value);
|
||||
},
|
||||
selected: widget.field.value == option.value,
|
||||
),
|
||||
);
|
||||
},
|
||||
child: CustomScrollView(
|
||||
scrollDirection: Axis.horizontal,
|
||||
slivers: [
|
||||
SliverToBoxAdapter(child: SizedBox(width: widget.padding.left)),
|
||||
SliverList.separated(
|
||||
itemCount: _options.length,
|
||||
separatorBuilder: (context, index) => const SizedBox(width: 8.0),
|
||||
itemBuilder: (context, index) {
|
||||
final option = _options[index];
|
||||
return ColoredChipWrapper(
|
||||
child: FilterChip(
|
||||
label: Text(option.title),
|
||||
onSelected: (isSelected) {
|
||||
final value = isSelected
|
||||
? option.value
|
||||
: const RelativeDateRangeQuery();
|
||||
widget.field.didChange(value);
|
||||
widget.onChanged?.call(value);
|
||||
},
|
||||
selected: widget.field.value == option.value,
|
||||
),
|
||||
);
|
||||
},
|
||||
),
|
||||
SliverToBoxAdapter(child: SizedBox(width: widget.padding.right))
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
@@ -63,7 +63,10 @@ class ChangelogDialog extends StatelessWidget {
|
||||
}
|
||||
|
||||
const _versionNumbers = {
|
||||
"59": "3.1.5",
|
||||
"4033": "3.1.8",
|
||||
"4023": "3.1.7",
|
||||
"4013": "3.1.6",
|
||||
"4003": "3.1.5",
|
||||
"58": "3.1.4",
|
||||
"57": "3.1.3",
|
||||
"56": "3.1.2",
|
||||
|
||||
@@ -3,6 +3,7 @@ import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:flutter_form_builder/flutter_form_builder.dart';
|
||||
import 'package:paperless_api/paperless_api.dart';
|
||||
import 'package:paperless_mobile/core/database/tables/local_user_account.dart';
|
||||
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
|
||||
import 'package:paperless_mobile/core/repository/label_repository.dart';
|
||||
import 'package:paperless_mobile/core/widgets/form_builder_fields/extended_date_range_form_field/form_builder_extended_date_range_picker.dart';
|
||||
import 'package:paperless_mobile/features/labels/tags/view/widgets/tags_form_field.dart';
|
||||
@@ -93,14 +94,14 @@ class _DocumentFilterFormState extends State<DocumentFilterForm> {
|
||||
|
||||
List<Widget> _buildFormFieldList(LabelRepository labelRepository) {
|
||||
return [
|
||||
_buildQueryFormField(),
|
||||
_buildQueryFormField().paddedSymmetrically(horizontal: 12),
|
||||
Align(
|
||||
alignment: Alignment.centerLeft,
|
||||
child: Text(
|
||||
S.of(context)!.advanced,
|
||||
style: Theme.of(context).textTheme.bodySmall,
|
||||
),
|
||||
),
|
||||
).paddedLTRB(12, 16, 12, 0),
|
||||
FormBuilderExtendedDateRangePicker(
|
||||
name: DocumentFilterForm.fkCreatedAt,
|
||||
initialValue: widget.initialFilter.created,
|
||||
@@ -108,6 +109,7 @@ class _DocumentFilterFormState extends State<DocumentFilterForm> {
|
||||
onChanged: (_) {
|
||||
_checkQueryConstraints();
|
||||
},
|
||||
padding: EdgeInsets.symmetric(vertical: 8, horizontal: 12),
|
||||
),
|
||||
FormBuilderExtendedDateRangePicker(
|
||||
name: DocumentFilterForm.fkAddedAt,
|
||||
@@ -116,17 +118,28 @@ class _DocumentFilterFormState extends State<DocumentFilterForm> {
|
||||
onChanged: (_) {
|
||||
_checkQueryConstraints();
|
||||
},
|
||||
padding: EdgeInsets.symmetric(vertical: 8, horizontal: 12),
|
||||
),
|
||||
_buildCorrespondentFormField(labelRepository.correspondents),
|
||||
_buildDocumentTypeFormField(labelRepository.documentTypes),
|
||||
_buildStoragePathFormField(labelRepository.storagePaths),
|
||||
_buildTagsFormField(labelRepository.tags),
|
||||
]
|
||||
.map((w) => SliverPadding(
|
||||
padding: widget.padding,
|
||||
sliver: SliverToBoxAdapter(child: w),
|
||||
))
|
||||
.toList();
|
||||
_buildCorrespondentFormField(labelRepository.correspondents)
|
||||
.paddedSymmetrically(
|
||||
horizontal: 16,
|
||||
vertical: 4,
|
||||
),
|
||||
_buildDocumentTypeFormField(labelRepository.documentTypes)
|
||||
.paddedSymmetrically(
|
||||
horizontal: 16,
|
||||
vertical: 4,
|
||||
),
|
||||
_buildStoragePathFormField(labelRepository.storagePaths)
|
||||
.paddedSymmetrically(
|
||||
horizontal: 16,
|
||||
vertical: 4,
|
||||
),
|
||||
_buildTagsFormField(labelRepository.tags).paddedSymmetrically(
|
||||
horizontal: 16,
|
||||
vertical: 4,
|
||||
),
|
||||
].map((e) => SliverToBoxAdapter(child: e)).toList();
|
||||
}
|
||||
|
||||
void _checkQueryConstraints() {
|
||||
|
||||
@@ -385,7 +385,7 @@ class _GoRouterShellState extends State<GoRouterShell> {
|
||||
);
|
||||
},
|
||||
routerConfig: _router,
|
||||
debugShowCheckedModeBanner: true,
|
||||
debugShowCheckedModeBanner: false,
|
||||
title: "Paperless Mobile",
|
||||
theme: buildTheme(
|
||||
brightness: Brightness.light,
|
||||
@@ -397,7 +397,7 @@ class _GoRouterShellState extends State<GoRouterShell> {
|
||||
dynamicScheme: darkDynamic,
|
||||
preferredColorScheme: settings.preferredColorSchemeOption,
|
||||
),
|
||||
themeMode: settings.preferredThemeMode,
|
||||
themeMode: settings.preferredThemeMode,
|
||||
supportedLocales: const [
|
||||
Locale('en'),
|
||||
Locale('de'),
|
||||
|
||||
@@ -73,7 +73,7 @@ class DocumentModel extends Equatable {
|
||||
this.userCanChange,
|
||||
this.permissions,
|
||||
this.customFields = const [],
|
||||
this.notes = const [],
|
||||
this.notes = const [] = const [],
|
||||
});
|
||||
|
||||
factory DocumentModel.fromJson(Map<String, dynamic> json) =>
|
||||
|
||||
@@ -14,5 +14,6 @@ class Permissions with _$Permissions {
|
||||
@HiveField(1) required UsersAndGroupsPermissions change,
|
||||
}) = _Permissions;
|
||||
|
||||
factory Permissions.fromJson(Map<String, dynamic> json) => _$PermissionsFromJson(json);
|
||||
factory Permissions.fromJson(Map<String, dynamic> json) =>
|
||||
_$PermissionsFromJson(json);
|
||||
}
|
||||
|
||||
@@ -15,7 +15,7 @@ publish_to: "none" # Remove this line if you wish to publish to pub.dev
|
||||
# In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion.
|
||||
# Read more about iOS versioning at
|
||||
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
|
||||
version: 3.1.5+59
|
||||
version: 3.1.8+403
|
||||
|
||||
environment:
|
||||
sdk: ">=3.1.0 <4.0.0"
|
||||
|
||||
|
Before Width: | Height: | Size: 4.9 KiB |
@@ -9,7 +9,7 @@ function mergeChangelogs () {
|
||||
__target_file=$__target_dir/changelogs_$1.md
|
||||
rm -f $__target_file
|
||||
touch $__target_file
|
||||
ls $__script_dir/../android/fastlane/metadata/android/$1/changelogs/[0-9]*.txt | tac | while read f; do
|
||||
ls -1v $__script_dir/../android/fastlane/metadata/android/$1/changelogs/[0-9]*.txt | tac | while read f; do
|
||||
__build_number="${f%.*}"
|
||||
echo "# $(basename -- $__build_number)" >> $__target_file
|
||||
cat $f >> $__target_file
|
||||
|
||||