Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion example/lib/main.dart
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
// ignore: avoid_web_libraries_in_flutter
import 'dart:html' as html;

import 'package:flutter/material.dart';
Expand Down Expand Up @@ -106,9 +107,10 @@ class _UploadExampleState extends State<UploadExample> {
if (pickedFile != null && pickedThumbnail != null) {
_largeFileUploader.upload(
uploadUrl: url,
name: '1',
headers: {"Authorization": "Bearer $accessToken"},
data: {"title": "Sample Title", "thumbnail": pickedThumbnail, "file": pickedFile},
onSendProgress: (progress) => debugPrint(progress.toString()),
onSendProgress: (progress, id) => debugPrint('$id: $progress'),
onComplete: (response) => debugPrint(response.toString()),
);

Expand Down
29 changes: 18 additions & 11 deletions example/pubspec.lock
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ packages:
name: async
url: "https://pub.dartlang.org"
source: hosted
version: "2.8.1"
version: "2.8.2"
boolean_selector:
dependency: transitive
description:
Expand All @@ -21,7 +21,7 @@ packages:
name: characters
url: "https://pub.dartlang.org"
source: hosted
version: "1.1.0"
version: "1.2.0"
charcode:
dependency: transitive
description:
Expand All @@ -42,7 +42,7 @@ packages:
name: collection
url: "https://pub.dartlang.org"
source: hosted
version: "1.15.0"
version: "1.16.0"
crypto:
dependency: transitive
description:
Expand Down Expand Up @@ -70,7 +70,7 @@ packages:
name: fake_async
url: "https://pub.dartlang.org"
source: hosted
version: "1.2.0"
version: "1.3.0"
flutter:
dependency: "direct main"
description: flutter
Expand Down Expand Up @@ -101,7 +101,7 @@ packages:
path: ".."
relative: true
source: path
version: "0.0.3"
version: "0.0.7"
lints:
dependency: transitive
description:
Expand All @@ -115,7 +115,14 @@ packages:
name: matcher
url: "https://pub.dartlang.org"
source: hosted
version: "0.12.10"
version: "0.12.11"
material_color_utilities:
dependency: transitive
description:
name: material_color_utilities
url: "https://pub.dartlang.org"
source: hosted
version: "0.1.4"
meta:
dependency: transitive
description:
Expand All @@ -129,7 +136,7 @@ packages:
name: path
url: "https://pub.dartlang.org"
source: hosted
version: "1.8.0"
version: "1.8.1"
sky_engine:
dependency: transitive
description: flutter
Expand All @@ -141,7 +148,7 @@ packages:
name: source_span
url: "https://pub.dartlang.org"
source: hosted
version: "1.8.1"
version: "1.8.2"
stack_trace:
dependency: transitive
description:
Expand Down Expand Up @@ -176,7 +183,7 @@ packages:
name: test_api
url: "https://pub.dartlang.org"
source: hosted
version: "0.4.2"
version: "0.4.9"
typed_data:
dependency: transitive
description:
Expand Down Expand Up @@ -204,7 +211,7 @@ packages:
name: vector_math
url: "https://pub.dartlang.org"
source: hosted
version: "2.1.0"
version: "2.1.2"
sdks:
dart: ">=2.12.0 <3.0.0"
dart: ">=2.17.0-0 <3.0.0"
flutter: ">=1.20.0"
53 changes: 0 additions & 53 deletions example/web/upload_worker.js

This file was deleted.

53 changes: 53 additions & 0 deletions lib/js/upload_worker.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
self.addEventListener('message', async (event) => {
var method = event.data.method;
var uploadUrl = event.data.uploadUrl;
var data = event.data.data;
var headers = event.data.headers;
uploadFile(method, uploadUrl, data, headers);
});

function uploadFile(method, uploadUrl, data, headers) {
var xhr = new XMLHttpRequest();
var formdata = new FormData();
var uploadPercent;

setData(formdata, data);

xhr.upload.addEventListener('progress', function (d) {
if (d.lengthComputable) {
uploadPercent = Math.floor((d.loaded / d.total) * 100);
postMessage(uploadPercent);
}
}, false);
xhr.onreadystatechange = function () {
if (xhr.readyState == XMLHttpRequest.DONE) {
postMessage("done");
}
}

xhr.onload = () => {
postMessage(xhr.response);
};

xhr.onerror = function () {
// only triggers if the request couldn't be made at all
postMessage("request failed");
};

xhr.open(method, uploadUrl, true);
setHeaders(xhr, headers);

xhr.send(formdata);
}

function setData(formdata, data) {
for (let key in data) {
formdata.append(key, data[key])
}
}

function setHeaders(xhr, headers) {
for (let key in headers) {
xhr.setRequestHeader(key, headers[key])
}
}
65 changes: 46 additions & 19 deletions lib/src/large_file_uploader.dart
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import 'package:large_file_uploader/src/enum/file_types.dart';
import 'package:universal_html/html.dart' as html;

/// Callback exposing currently upload progress.
typedef UploadProgressListener = Function(int progress);
typedef UploadProgressListener = Function(int progress, String id);

/// Callback exposing upload fail event.
typedef UploadFailureListener = Function();
Expand All @@ -23,10 +23,13 @@ class LargeFileUploader {
Timer? _timer;
int _fakeProgress = 0;

StreamSubscription? _streamSubscription;

void selectFileAndUpload({
String method = 'POST',
FileTypes type = FileTypes.file,
String? customFileType,
bool allowMultiple = false,
required String uploadUrl,
Map<String, dynamic>? data,
Map<String, dynamic>? headers,
Expand All @@ -40,43 +43,60 @@ class LargeFileUploader {
pick(
type: type,
customFileType: customFileType,
allowMultiple: allowMultiple,
callback: (file) {
data ??= {};
data!["file"] = file;
upload(uploadUrl: uploadUrl, onSendProgress: onSendProgress, data: data!);
upload(
method: method,
name: file.name,
uploadUrl: uploadUrl,
data: data!,
headers: headers,
onSendProgress: onSendProgress,
fakePreProcessMaxProgress: fakePreProcessMaxProgress,
fakePreProcessProgressPeriodInMillisecond: fakePreProcessProgressPeriodInMillisecond,
onSendWithFakePreProcessProgress: onSendWithFakePreProcessProgress,
onFailure: onFailure,
onComplete: onComplete,
);
},
);
}

void pick({
FileTypes type = FileTypes.file,
String? customFileType,
bool allowMultiple = false,
required OnFileSelectedListener callback,
}) {
html.FileUploadInputElement fileUploadInputElement = html.FileUploadInputElement();
fileUploadInputElement.accept = customFileType ?? type.value;
fileUploadInputElement.multiple = false;
fileUploadInputElement.multiple = allowMultiple;
fileUploadInputElement.click();

fileUploadInputElement.onChange.listen((_) {
if (fileUploadInputElement.files != null) {
callback.call(fileUploadInputElement.files!.first);
for (final file in fileUploadInputElement.files!) {
callback.call(file);
}
}
});
}

void upload({
String method = 'POST',
required String name,
required String uploadUrl,
required UploadProgressListener onSendProgress,
required Map<String, dynamic> data,
String method = 'POST',
Map<String, dynamic>? headers,
required UploadProgressListener onSendProgress,
int fakePreProcessMaxProgress = 30,
int fakePreProcessProgressPeriodInMillisecond = 500,
UploadProgressListener? onSendWithFakePreProcessProgress,
UploadFailureListener? onFailure,
UploadCompleteListener? onComplete,
}) {
}) async {
_worker.postMessage({
'method': method,
'uploadUrl': uploadUrl,
Expand All @@ -88,25 +108,29 @@ class LargeFileUploader {
_timer = Timer.periodic(Duration(milliseconds: fakePreProcessProgressPeriodInMillisecond), (Timer timer) {
if (_fakeProgress != fakePreProcessMaxProgress) {
_fakeProgress++;
onSendWithFakePreProcessProgress.call(_fakeProgress);
onSendWithFakePreProcessProgress.call(_fakeProgress, 'fake');
} else {
_disposeTimerAndFakeProgress();
}
});
}

_worker.onMessage.listen((data) {
_handleCallbacks(data.data,
onSendProgress: onSendProgress,
fakePreProcessMaxProgress: fakePreProcessMaxProgress,
onSendWithFakePreProcessProgress: onSendWithFakePreProcessProgress,
onFailure: onFailure,
onComplete: onComplete);
_streamSubscription = _worker.onMessage.listen((data) {
_handleCallbacks(
data.data,
name: name,
onSendProgress: onSendProgress,
fakePreProcessMaxProgress: fakePreProcessMaxProgress,
onSendWithFakePreProcessProgress: onSendWithFakePreProcessProgress,
onFailure: onFailure,
onComplete: onComplete,
);
});
}

void _handleCallbacks(
data, {
required String name,
required UploadProgressListener onSendProgress,
required int fakePreProcessMaxProgress,
UploadProgressListener? onSendWithFakePreProcessProgress,
Expand All @@ -116,19 +140,22 @@ class LargeFileUploader {
if (data == null) return;

if (data is int) {
onSendProgress.call(data);
onSendProgress.call(data, name);
if (data != 0) {
_disposeTimerAndFakeProgress();
onSendWithFakePreProcessProgress
?.call((fakePreProcessMaxProgress + (data * ((100 - fakePreProcessMaxProgress) / 100))).toInt());
onSendWithFakePreProcessProgress?.call(
(fakePreProcessMaxProgress + (data * ((100 - fakePreProcessMaxProgress) / 100))).toInt(),
name,
);
}
} else if (data.toString() == 'request failed') {
_disposeTimerAndFakeProgress();
onFailure?.call();
} else {
onSendWithFakePreProcessProgress?.call(100);
onSendWithFakePreProcessProgress?.call(100, name);
_disposeTimerAndFakeProgress();
onComplete?.call(data);
_streamSubscription?.cancel();
}
}

Expand Down