Background
My app has several packages that I've developed on my own. My project structure is as follows:
/my_app
- /lib
- /my_packages
- /package_1
- /package_2
- ...
Several of the packages change their behavior depending on the environment (iOS, Android, Web, dev/prod). For example, packages query "localhost" when in dev, but "www.example.com" in production.
Problem
I'd like to share the environment between packages. I figured that the best way to do that would be to develop another package. That way, I can import the package library to each of the other packages. The package looks as follows:
File Structure
/environment_pkg
- /assets
- /configs
- config1.json
- config2.json
- ...
pubspec.yaml
flutter:
assets:
- assets/configs/config1.json
- assets/configs/config2.json
- ...
Loading Data
static Future<void> initialize(String name) async {
final configString = await rootBundle.loadString('assets/configs/${name}.json');
_config = json.decode(configString) as Map<String, dynamic>;
}
Error
Unhandled Exception: Unable to load asset: assets/configs/config1.json
Summary
I am trying to load JSON files from the assets folder with a flutter package using rootBundle. Flutter is unable to locate the files.
https://stackoverflow.com/a/74128378/12379401
Add packages/{packagename}/ before path
static Future<void> initialize(String name) async {
final configString = await rootBundle.loadString('packages/{packagename}/assets/configs/${name}.json');
_config = json.decode(configString) as Map<String, dynamic>; }
Related
I faced following problem:
My Android Flutter app has a serie of dart:ui calls, basically to do an application files backup. Lets just check how to test that it find the correct path of the app files. The function to test:
static Future<String> getUserPath() async {
final appDocDir = await getApplicationDocumentsDirectory();
final pathComponents = path.split(appDocDir.path);
pathComponents.removeLast();
final filepaths = path.join(path.joinAll(pathComponents), 'files', '.user');
if (!await Directory(filepaths).exists()) throw FileSystemException('User directory not found');
return filepaths;
}
Easy, it call different dart:ui functions to check where the user files are. It works now, but I want to write the test for this funcion. Basically I found two ways to test this is: using Flutter Drive tests to write integration tests, i followed this tutorial: https://flutter-website-staging.firebaseapp.com/testing/ . And using Flutter integration tests, using this guide: https://docs.flutter.dev/testing/integration-tests.
The test I wrote is next:
testWidgets('test_correct_path', (WidgetTester tester) async {
print('user folder: ' + await getUserPath());
assert('.user' == basename(await getUserPath()));
});
Using driver tests
I created the following folder on my project:
test_driver
├── backup.dart
└── backup_test.dart
On backup.dart
// This line imports the extension
import 'package:flutter_driver/driver_extension.dart';
import 'package:elRepoIo/main.dart' as app;
void main() {
// This line enables the extension
enableFlutterDriverExtension();
}
And on backup_test.dart I just added the needed functions after test main declaration:
void main() {
FlutterDriver driver;
// connect flutter driver to the app before executing the runs
setUpAll(() async {
driver = await FlutterDriver.connect();
});
// disconnect flutter driver from the app after executing the runs
tearDownAll(() async {
if (driver != null) {
driver.close();
}
});
// Tests down here
Then when I execute the tests using flutter drive --target=test_driver/backup.dart I receive a bunch of errors such flutter/packages/flutter_test/lib/src/frame_timing_summarizer.dart:5:8: Error: Not found: 'dart:ui' import 'dart:ui';
Using integration tests
Under
integration_test
├── backup_test.dart
I wrote the tests adding the following line:
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized(); // NEW
Then I run the test using flutter test integration_test/backup_test.dart -d $device_id where I get device_id using flutter devices.
If I debug this I found that the path is correct, but await Directory(filepaths).exists() is false.
It seem that if I run the app to create the folder and inmetially I run the tests the directory exist. Are integration tests deleting all file data? And how can I keep it?
The following code causes this error Null check operator used on a null value but I cannot figure out why.
import 'package:tflite_flutter/tflite_flutter.dart';
final modelFile = 'model.tflite';
void main() async {
foo();
}
void foo() async {
Interpreter inter = await Interpreter.fromAsset(modelFile);
}
The model exists at the correct location and I have tried multiple other locations just in case. The file is in my pubspec.yaml and I have tried multiple variations just in case. Not sure if I need to provide more parameters or something but I can't figure it out. Any help would be appreciated.
Just to test if it was an issue with me not placing the file in the correct location, or not referencing it in the pubspec.yaml I tested other assets such as Images and Text files in the same location and they loaded perfectly fine.
It looks like you have not exported the file. To fix to go to pubspec.yaml and find these lines:
flutter:
assets:
- model.tflite
And make sure you are specifying the correct file location when adding it through pubscpec. After that uninstall the app, do flutter clean, flutter pub get and it should work
I assumer the directory containing the model is named as "assets".
import 'package:tflite_flutter/tflite_flutter.dart';
final modelFile = 'assets/model.tflite';
void main() async {
foo();
}
void foo() async {
Interpreter inter = await Interpreter.fromAsset(modelFile);
}
Code for pubspec:-
flutter:
assets:
- assets/model.tflite
Recently, I was working on the portfolio website where I need to show the user's CV which was in the PDF format on another tab.
Since this was a standalone Web Project, I didn't want to handle PDF viewing for iOS and Android.
First, you need to install this package: universal_html
flutter pub add universal_html
import the package in your file
import 'package:universal_html/html.dart' as html;
And here is the method:
Future<void> showCV() async {
var bytes = await rootBundle.load("assets/files/cv.pdf"); // location of your asset file
final blob = html.Blob([bytes], 'application/pdf');
final url = html.Url.createObjectUrlFromBlob(blob);
html.window.open(url, "_blank");
html.Url.revokeObjectUrl(url);
}
Voila !
var bytes = await rootBundle.load("assets/files/cv.pdf"); // location of your
asset file
final blob = html.Blob([bytes], 'application/pdf');
final url = html.Url.createObjectUrlFromBlob(blob);
html.window.open(url, "_blank");
html.Url.revokeObjectUrl(url);
Does it work in web hosting also ? because in my case this code is working in local host but not in web host.
Package tried: https://pub.dev/packages/file_picker
Tried the example code implementation shared via GitHub. But the file path is returned as null for
web platform. Where same implementation in mobile platform return the path as expected.
https://github.com/miguelpruivo/flutter_file_picker/blob/master/example/lib/src/file_picker_demo.dart
Things aware of:
Path_Provider - Not supported for web
dart-io-library - Not supported for web
Open issue in Flutter Repo
Goal is to read the file from the path and not to upload. Any workaround to achieve this will be helpful.
Flutter channel: beta
As mentioned in the file_picker FAQ:
Paths aren't inaccessible from browsers since those provide fake paths. If you want to create a File instance to upload it somewhere, like FireStorage, you can do so with the bytes directly.
final result = await FilePicker.platform.pickFiles(type: FileType.any, allowMultiple: false);
if (result.files.first != null){
var fileBytes = result.files.first.bytes;
var fileName = result.files.first.name;
print(String.fromCharCodes(fileBytes));
}
I have a function for picking image from computer. Might work for you.
import 'dart:html';
void uploadImage({#required Function(File file) onSelected}) {
InputElement uploadInput = FileUploadInputElement()..accept = 'image/*';
uploadInput.click();
uploadInput.onChange.listen((event) {
final file = uploadInput.files.first;
final reader = FileReader();
reader.readAsDataUrl(file);
reader.onLoadEnd.listen((event) {
onSelected(file);
});
});
}
I am using some assets during my development (fake API responses and random images), and for that, I am loading some assets.
Obviously I do not want those assets to be bundled with the release APK, how do you come around doing so?
For now, I have all assets in one pubspec.yaml.
The end goal is to be able to run the app flutter run and have fake backend. I know of mocking HTTP calls, but it is very difficult to wrap a full app, so now I have two different data source (one from assets and another http-based), and I call the one depending on kReleaseMode value.
Thanks!
I had the same problem (which also seems to be an orphaned flutter issue), but solved it using a Builder running in the dart build_runner - not to be confused with the flutter Builder pattern for widgets (which spoils the search results). This works as sketched here:
collect dev-resources in a separate directory out of assets (in my case lib/devdata/)
create a Builder implementation and a build.yaml file
in build.yaml create separate dev_properties that allow the builder to behave according to the build mode, i.e. either copy the files to assets/devdata/ or delete them in assets/devdata/ and this directory to .gitignore
add assets/devdata/ to assets in pubspec.yaml
run build_runner either in dev or in prod mode to copy or delete the files
consume the files in the app using rootBundle if they are present / catch the resulting exception if they are absent
this command invokes the build_runner in dev / release mode respectively:
$ flutter pub run build_runner build
$ flutter pub run build_runner build --release
the builder looks roughly as follows (saved in lib/devdata/devdata.dart):
import 'dart:async';
import 'dart:io';
import 'package:build/build.dart';
Builder devData(BuilderOptions options) => DevDataBuilder(options);
class DevDataBuilder implements Builder {
final BuilderOptions options;
DevDataBuilder(this.options);
#override
FutureOr<void> build(BuildStep buildStep) async {
var mode = options.config['mode'];
var inputId = buildStep.inputId;
var sourcePath = inputId.path;
var targetPath = sourcePath.replaceFirst('lib/devdata/', 'assets/devdata/');
final outputId = AssetId(
inputId.package,
targetPath,
);
if (mode == 'dev') {
log.info('copying $sourcePath to $targetPath');
var contents = await buildStep.readAsString(inputId);
await buildStep.writeAsString(outputId, contents);
} else {
var file = File(targetPath);
if (await file.exists()) {
log.info('deleting $targetPath');
file.delete();
}
}
}
#override
Map<String, List<String>> get buildExtensions {
return {
'lib/devdata/{{file}}.json': const ['assets/devdata/{{file}}.json'],
};
}
}
the corresponding build.yaml is here:
builders:
test_data:
import: 'package:my_app_package/builder/devdata.dart'
builder_factories: ['devData']
build_extensions: {"lib/devdata/{{file}}.json": ["assets/testdata/{{file}}.json"]}
build_to: source
auto_apply: root_package
defaults:
generate_for:
include:
- lib/devdata/**
global_options:
my_app_package:test_data:
options:
mode: prod
dev_options:
mode: dev
here are some additional resources that led to this approach:
https://pub.dev/packages/build
https://pub.dev/packages/build_config
https://pub.dev/packages/build_runner
https://github.com/dart-lang/build/tree/master/example
https://github.com/dart-lang/build/blob/master/docs/build_yaml_format.md
https://pedromarquez.dev/blog/2022/8/flutter-code-gen-1
https://pedromarquez.dev/blog/2022/9/flutter-code-gen-2
https://github.com/flutter/flutter/issues/5813
https://github.com/flutter/flutter/pull/25487
an alternative approach without a builder can be found here (using the pr #25487 mentioned above):
https://github.com/sestegra/flutter_assets_flavors