Compare commits
No commits in common. "dd52ffb39811045e10e45aa93f6474b9073f89d8" and "f9fba42a796f7f155219299725ce8882caed7dec" have entirely different histories.
dd52ffb398
...
f9fba42a79
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -28,4 +28,3 @@ migrate_working_dir/
|
||||||
.dart_tool/
|
.dart_tool/
|
||||||
.packages
|
.packages
|
||||||
build/
|
build/
|
||||||
|
|
||||||
|
|
15
CHANGELOG.md
15
CHANGELOG.md
|
@ -1,18 +1,3 @@
|
||||||
## 0.0.3
|
|
||||||
|
|
||||||
Lots of improvements!
|
|
||||||
|
|
||||||
- `CardTextField` now has customizable styles. Stripe integration is natively handled now, returning a card token, if stripe keys are provided.
|
|
||||||
- `README` revamped with emojis, screen recordings, the whole nine yards.
|
|
||||||
- `LICENSE` changed from BSD-3.0 to MIT license for pub points, I guess it wasn't being recognized correctly...
|
|
||||||
- Added `http` depency for handling Stripe token api call.
|
|
||||||
- Added Widget tests because that should be a thing that gets checked.
|
|
||||||
- Fix for backspacing on mobile not changing focus.
|
|
||||||
- Fix for text field spacing when in small form factor
|
|
||||||
- Much improved usability on mobile, added manually scrolling to element
|
|
||||||
- Added Icon Size param for card Provider Icon
|
|
||||||
|
|
||||||
|
|
||||||
## 0.0.2
|
## 0.0.2
|
||||||
|
|
||||||
Added dartdoc comments for more pub points!
|
Added dartdoc comments for more pub points!
|
||||||
|
|
12
LICENSE
12
LICENSE
|
@ -1,7 +1,11 @@
|
||||||
Copyright 2023 - Nathan Anderson
|
Copyright 2023 Nathan Anderson
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
BSD 3-Clause License
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
|
||||||
|
Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
|
||||||
|
The names of its contributors may not be used to endorse or promote products derived from this software without specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
56
README.md
56
README.md
|
@ -3,41 +3,10 @@ A simple and clean Stripe Element Card clone, rebuilt in native Flutter widgets.
|
||||||
This is not an officially maintained package by Stripe, but using the html stripe
|
This is not an officially maintained package by Stripe, but using the html stripe
|
||||||
elements they provide with flutter is less than ideal.
|
elements they provide with flutter is less than ideal.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
Got to use emojis and taglines for attention grabbing and algorithm hacking:
|
- Card number validation
|
||||||
|
- No more depending on Flutter Webview
|
||||||
⚡Blazingly fast ( its as fast as the rest of flutter )
|
|
||||||
🧹Cleaner ( fewer dependencies than the official stripe elements )
|
|
||||||
🛡️Safe and Supports all Flutter Targets ( its native flutter with minimal dependencies )
|
|
||||||
☑️Seemless UI/UX ( hard to match stripe quality, but I think I got close )
|
|
||||||
🔄Built-in Stripe Integration ( guess that one is obvious )
|
|
||||||
☯️Chi Energy Boost ( alright I'm fishing... )
|
|
||||||
|
|
||||||
### Why StripeNativeCardField?
|
|
||||||
|
|
||||||
- Fewer dependencies: no more depending on Flutter Webview
|
|
||||||
- Customizable: the entire field can inherit native Flutter styling, i.e. `BoxDecoration()`
|
|
||||||
- Native Implementation: compiles and loads like the rest of your app, unlike embeded html
|
|
||||||
- Automatic validation: no `inputFormatters` or `RegExp` needed on your side
|
|
||||||
|
|
||||||
The card data can either be retrieved with the `onCardDetailsComplete` callback, or
|
|
||||||
you can have the element automatically create a Stripe card token when the fields
|
|
||||||
are filled out, and return the token with the `onTokenReceived` callback.
|
|
||||||
|
|
||||||
### Mobile
|
|
||||||
|
|
||||||
![mobile showcase](./example/loading.gif)
|
|
||||||
|
|
||||||
### Desktop
|
|
||||||
|
|
||||||
![desktop showcase](./example/loading.gif)
|
|
||||||
|
|
||||||
### Customizable
|
|
||||||
|
|
||||||
![cumstomization showcase](./example/loading.gif)
|
|
||||||
|
|
||||||
## Getting started
|
## Getting started
|
||||||
|
|
||||||
|
@ -47,13 +16,10 @@ are filled out, and return the token with the `onTokenReceived` callback.
|
||||||
|
|
||||||
Include the package in a file:
|
Include the package in a file:
|
||||||
|
|
||||||
|
|
||||||
```dart
|
```dart
|
||||||
import 'package:stripe_native_card_field/stripe_native_card_field.dart';
|
import 'package:stripe_native_card_field/stripe_native_card_field.dart';
|
||||||
```
|
```
|
||||||
|
|
||||||
### For just Card Data
|
|
||||||
|
|
||||||
```dart
|
```dart
|
||||||
CardTextField(
|
CardTextField(
|
||||||
width: 500,
|
width: 500,
|
||||||
|
@ -64,24 +30,6 @@ CardTextField(
|
||||||
);
|
);
|
||||||
```
|
```
|
||||||
|
|
||||||
### For Stripe Token
|
|
||||||
|
|
||||||
```dart
|
|
||||||
CardTextField(
|
|
||||||
width: 500,
|
|
||||||
stripePublishableKey: 'pk_test_abc123', // Your stripe key here
|
|
||||||
onTokenReceived: (token) {
|
|
||||||
// Save the stripe token to send to your backend
|
|
||||||
setState(() => _token = token);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
```
|
|
||||||
|
|
||||||
### Cumstomization
|
|
||||||
|
|
||||||
For documentation on all of the available customizable aspects of the `CardTextField`, go
|
|
||||||
to the [API docs here](https://pub.dev/documentation/stripe_native_card_field/latest/stripe_native_card_field/CardTextField-class.html).
|
|
||||||
|
|
||||||
## Additional information
|
## Additional information
|
||||||
|
|
||||||
Repository located [here](https://git.fosscat.com/n8r/stripe_native_card_field)
|
Repository located [here](https://git.fosscat.com/n8r/stripe_native_card_field)
|
||||||
|
|
|
@ -1,19 +0,0 @@
|
||||||
<component name="libraryTable">
|
|
||||||
<library name="Dart SDK">
|
|
||||||
<CLASSES>
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/async" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/collection" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/convert" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/core" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/developer" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/html" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/io" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/isolate" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/math" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/mirrors" />
|
|
||||||
<root url="file:///home/nate/Tooling/flutter/bin/cache/dart-sdk/lib/typed_data" />
|
|
||||||
</CLASSES>
|
|
||||||
<JAVADOC />
|
|
||||||
<SOURCES />
|
|
||||||
</library>
|
|
||||||
</component>
|
|
|
@ -1,15 +0,0 @@
|
||||||
<component name="libraryTable">
|
|
||||||
<library name="KotlinJavaRuntime">
|
|
||||||
<CLASSES>
|
|
||||||
<root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib.jar!/" />
|
|
||||||
<root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-reflect.jar!/" />
|
|
||||||
<root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-test.jar!/" />
|
|
||||||
</CLASSES>
|
|
||||||
<JAVADOC />
|
|
||||||
<SOURCES>
|
|
||||||
<root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib-sources.jar!/" />
|
|
||||||
<root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-reflect-sources.jar!/" />
|
|
||||||
<root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-test-sources.jar!/" />
|
|
||||||
</SOURCES>
|
|
||||||
</library>
|
|
||||||
</component>
|
|
|
@ -1,9 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="ProjectModuleManager">
|
|
||||||
<modules>
|
|
||||||
<module fileurl="file://$PROJECT_DIR$/example.iml" filepath="$PROJECT_DIR$/example.iml" />
|
|
||||||
<module fileurl="file://$PROJECT_DIR$/android/example_android.iml" filepath="$PROJECT_DIR$/android/example_android.iml" />
|
|
||||||
</modules>
|
|
||||||
</component>
|
|
||||||
</project>
|
|
|
@ -1,6 +0,0 @@
|
||||||
<component name="ProjectRunConfigurationManager">
|
|
||||||
<configuration default="false" name="main.dart" type="FlutterRunConfigurationType" factoryName="Flutter">
|
|
||||||
<option name="filePath" value="$PROJECT_DIR$/lib/main.dart" />
|
|
||||||
<method />
|
|
||||||
</configuration>
|
|
||||||
</component>
|
|
|
@ -1,36 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<project version="4">
|
|
||||||
<component name="FileEditorManager">
|
|
||||||
<leaf>
|
|
||||||
<file leaf-file-name="main.dart" pinned="false" current-in-tab="true">
|
|
||||||
<entry file="file://$PROJECT_DIR$/lib/main.dart">
|
|
||||||
<provider selected="true" editor-type-id="text-editor">
|
|
||||||
<state relative-caret-position="0">
|
|
||||||
<caret line="0" column="0" lean-forward="false" selection-start-line="0" selection-start-column="0" selection-end-line="0" selection-end-column="0" />
|
|
||||||
</state>
|
|
||||||
</provider>
|
|
||||||
</entry>
|
|
||||||
</file>
|
|
||||||
</leaf>
|
|
||||||
</component>
|
|
||||||
<component name="ToolWindowManager">
|
|
||||||
<editor active="true" />
|
|
||||||
<layout>
|
|
||||||
<window_info id="Project" active="false" anchor="left" auto_hide="false" internal_type="DOCKED" type="DOCKED" visible="true" show_stripe_button="true" weight="0.25" sideWeight="0.5" order="0" side_tool="false" content_ui="combo" />
|
|
||||||
</layout>
|
|
||||||
</component>
|
|
||||||
<component name="ProjectView">
|
|
||||||
<navigator currentView="ProjectPane" proportions="" version="1">
|
|
||||||
</navigator>
|
|
||||||
<panes>
|
|
||||||
<pane id="ProjectPane">
|
|
||||||
<option name="show-excluded-files" value="false" />
|
|
||||||
</pane>
|
|
||||||
</panes>
|
|
||||||
</component>
|
|
||||||
<component name="PropertiesComponent">
|
|
||||||
<property name="last_opened_file_path" value="$PROJECT_DIR$" />
|
|
||||||
<property name="dart.analysis.tool.window.force.activate" value="true" />
|
|
||||||
<property name="show.migrate.to.gradle.popup" value="false" />
|
|
||||||
</component>
|
|
||||||
</project>
|
|
|
@ -1,19 +0,0 @@
|
||||||
package io.flutter.plugins;
|
|
||||||
|
|
||||||
import androidx.annotation.Keep;
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
import io.flutter.Log;
|
|
||||||
|
|
||||||
import io.flutter.embedding.engine.FlutterEngine;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generated file. Do not edit.
|
|
||||||
* This file is generated by the Flutter tool based on the
|
|
||||||
* plugins that support the Android platform.
|
|
||||||
*/
|
|
||||||
@Keep
|
|
||||||
public final class GeneratedPluginRegistrant {
|
|
||||||
private static final String TAG = "GeneratedPluginRegistrant";
|
|
||||||
public static void registerWith(@NonNull FlutterEngine flutterEngine) {
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,29 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<module type="JAVA_MODULE" version="4">
|
|
||||||
<component name="FacetManager">
|
|
||||||
<facet type="android" name="Android">
|
|
||||||
<configuration>
|
|
||||||
<option name="ALLOW_USER_CONFIGURATION" value="false" />
|
|
||||||
<option name="GEN_FOLDER_RELATIVE_PATH_APT" value="/gen" />
|
|
||||||
<option name="GEN_FOLDER_RELATIVE_PATH_AIDL" value="/gen" />
|
|
||||||
<option name="MANIFEST_FILE_RELATIVE_PATH" value="/app/src/main/AndroidManifest.xml" />
|
|
||||||
<option name="RES_FOLDER_RELATIVE_PATH" value="/app/src/main/res" />
|
|
||||||
<option name="ASSETS_FOLDER_RELATIVE_PATH" value="/app/src/main/assets" />
|
|
||||||
<option name="LIBS_FOLDER_RELATIVE_PATH" value="/app/src/main/libs" />
|
|
||||||
<option name="PROGUARD_LOGS_FOLDER_RELATIVE_PATH" value="/app/src/main/proguard_logs" />
|
|
||||||
</configuration>
|
|
||||||
</facet>
|
|
||||||
</component>
|
|
||||||
<component name="NewModuleRootManager" inherit-compiler-output="true">
|
|
||||||
<exclude-output />
|
|
||||||
<content url="file://$MODULE_DIR$">
|
|
||||||
<sourceFolder url="file://$MODULE_DIR$/app/src/main/java" isTestSource="false" />
|
|
||||||
<sourceFolder url="file://$MODULE_DIR$/app/src/main/kotlin" isTestSource="false" />
|
|
||||||
<sourceFolder url="file://$MODULE_DIR$/gen" isTestSource="false" generated="true" />
|
|
||||||
</content>
|
|
||||||
<orderEntry type="jdk" jdkName="Android API 29 Platform" jdkType="Android SDK" />
|
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
|
||||||
<orderEntry type="library" name="Flutter for Android" level="project" />
|
|
||||||
<orderEntry type="library" name="KotlinJavaRuntime" level="project" />
|
|
||||||
</component>
|
|
||||||
</module>
|
|
BIN
example/android/gradle/wrapper/gradle-wrapper.jar
vendored
BIN
example/android/gradle/wrapper/gradle-wrapper.jar
vendored
Binary file not shown.
160
example/android/gradlew
vendored
160
example/android/gradlew
vendored
|
@ -1,160 +0,0 @@
|
||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
##############################################################################
|
|
||||||
##
|
|
||||||
## Gradle start up script for UN*X
|
|
||||||
##
|
|
||||||
##############################################################################
|
|
||||||
|
|
||||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
|
||||||
DEFAULT_JVM_OPTS=""
|
|
||||||
|
|
||||||
APP_NAME="Gradle"
|
|
||||||
APP_BASE_NAME=`basename "$0"`
|
|
||||||
|
|
||||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
|
||||||
MAX_FD="maximum"
|
|
||||||
|
|
||||||
warn ( ) {
|
|
||||||
echo "$*"
|
|
||||||
}
|
|
||||||
|
|
||||||
die ( ) {
|
|
||||||
echo
|
|
||||||
echo "$*"
|
|
||||||
echo
|
|
||||||
exit 1
|
|
||||||
}
|
|
||||||
|
|
||||||
# OS specific support (must be 'true' or 'false').
|
|
||||||
cygwin=false
|
|
||||||
msys=false
|
|
||||||
darwin=false
|
|
||||||
case "`uname`" in
|
|
||||||
CYGWIN* )
|
|
||||||
cygwin=true
|
|
||||||
;;
|
|
||||||
Darwin* )
|
|
||||||
darwin=true
|
|
||||||
;;
|
|
||||||
MINGW* )
|
|
||||||
msys=true
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# Attempt to set APP_HOME
|
|
||||||
# Resolve links: $0 may be a link
|
|
||||||
PRG="$0"
|
|
||||||
# Need this for relative symlinks.
|
|
||||||
while [ -h "$PRG" ] ; do
|
|
||||||
ls=`ls -ld "$PRG"`
|
|
||||||
link=`expr "$ls" : '.*-> \(.*\)$'`
|
|
||||||
if expr "$link" : '/.*' > /dev/null; then
|
|
||||||
PRG="$link"
|
|
||||||
else
|
|
||||||
PRG=`dirname "$PRG"`"/$link"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
SAVED="`pwd`"
|
|
||||||
cd "`dirname \"$PRG\"`/" >/dev/null
|
|
||||||
APP_HOME="`pwd -P`"
|
|
||||||
cd "$SAVED" >/dev/null
|
|
||||||
|
|
||||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
|
||||||
|
|
||||||
# Determine the Java command to use to start the JVM.
|
|
||||||
if [ -n "$JAVA_HOME" ] ; then
|
|
||||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
|
||||||
# IBM's JDK on AIX uses strange locations for the executables
|
|
||||||
JAVACMD="$JAVA_HOME/jre/sh/java"
|
|
||||||
else
|
|
||||||
JAVACMD="$JAVA_HOME/bin/java"
|
|
||||||
fi
|
|
||||||
if [ ! -x "$JAVACMD" ] ; then
|
|
||||||
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
|
||||||
|
|
||||||
Please set the JAVA_HOME variable in your environment to match the
|
|
||||||
location of your Java installation."
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
JAVACMD="java"
|
|
||||||
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
|
||||||
|
|
||||||
Please set the JAVA_HOME variable in your environment to match the
|
|
||||||
location of your Java installation."
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Increase the maximum file descriptors if we can.
|
|
||||||
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
|
|
||||||
MAX_FD_LIMIT=`ulimit -H -n`
|
|
||||||
if [ $? -eq 0 ] ; then
|
|
||||||
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
|
||||||
MAX_FD="$MAX_FD_LIMIT"
|
|
||||||
fi
|
|
||||||
ulimit -n $MAX_FD
|
|
||||||
if [ $? -ne 0 ] ; then
|
|
||||||
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# For Darwin, add options to specify how the application appears in the dock
|
|
||||||
if $darwin; then
|
|
||||||
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
|
||||||
fi
|
|
||||||
|
|
||||||
# For Cygwin, switch paths to Windows format before running java
|
|
||||||
if $cygwin ; then
|
|
||||||
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
|
||||||
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
|
||||||
JAVACMD=`cygpath --unix "$JAVACMD"`
|
|
||||||
|
|
||||||
# We build the pattern for arguments to be converted via cygpath
|
|
||||||
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
|
||||||
SEP=""
|
|
||||||
for dir in $ROOTDIRSRAW ; do
|
|
||||||
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
|
||||||
SEP="|"
|
|
||||||
done
|
|
||||||
OURCYGPATTERN="(^($ROOTDIRS))"
|
|
||||||
# Add a user-defined pattern to the cygpath arguments
|
|
||||||
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
|
||||||
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
|
||||||
fi
|
|
||||||
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
|
||||||
i=0
|
|
||||||
for arg in "$@" ; do
|
|
||||||
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
|
||||||
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
|
||||||
|
|
||||||
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
|
||||||
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
|
||||||
else
|
|
||||||
eval `echo args$i`="\"$arg\""
|
|
||||||
fi
|
|
||||||
i=$((i+1))
|
|
||||||
done
|
|
||||||
case $i in
|
|
||||||
(0) set -- ;;
|
|
||||||
(1) set -- "$args0" ;;
|
|
||||||
(2) set -- "$args0" "$args1" ;;
|
|
||||||
(3) set -- "$args0" "$args1" "$args2" ;;
|
|
||||||
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
|
||||||
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
|
||||||
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
|
||||||
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
|
||||||
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
|
||||||
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
|
||||||
esac
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
|
|
||||||
function splitJvmOpts() {
|
|
||||||
JVM_OPTS=("$@")
|
|
||||||
}
|
|
||||||
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
|
|
||||||
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
|
|
||||||
|
|
||||||
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
|
|
90
example/android/gradlew.bat
vendored
90
example/android/gradlew.bat
vendored
|
@ -1,90 +0,0 @@
|
||||||
@if "%DEBUG%" == "" @echo off
|
|
||||||
@rem ##########################################################################
|
|
||||||
@rem
|
|
||||||
@rem Gradle startup script for Windows
|
|
||||||
@rem
|
|
||||||
@rem ##########################################################################
|
|
||||||
|
|
||||||
@rem Set local scope for the variables with windows NT shell
|
|
||||||
if "%OS%"=="Windows_NT" setlocal
|
|
||||||
|
|
||||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
|
||||||
set DEFAULT_JVM_OPTS=
|
|
||||||
|
|
||||||
set DIRNAME=%~dp0
|
|
||||||
if "%DIRNAME%" == "" set DIRNAME=.
|
|
||||||
set APP_BASE_NAME=%~n0
|
|
||||||
set APP_HOME=%DIRNAME%
|
|
||||||
|
|
||||||
@rem Find java.exe
|
|
||||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
|
||||||
|
|
||||||
set JAVA_EXE=java.exe
|
|
||||||
%JAVA_EXE% -version >NUL 2>&1
|
|
||||||
if "%ERRORLEVEL%" == "0" goto init
|
|
||||||
|
|
||||||
echo.
|
|
||||||
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
|
||||||
echo.
|
|
||||||
echo Please set the JAVA_HOME variable in your environment to match the
|
|
||||||
echo location of your Java installation.
|
|
||||||
|
|
||||||
goto fail
|
|
||||||
|
|
||||||
:findJavaFromJavaHome
|
|
||||||
set JAVA_HOME=%JAVA_HOME:"=%
|
|
||||||
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
|
||||||
|
|
||||||
if exist "%JAVA_EXE%" goto init
|
|
||||||
|
|
||||||
echo.
|
|
||||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
|
||||||
echo.
|
|
||||||
echo Please set the JAVA_HOME variable in your environment to match the
|
|
||||||
echo location of your Java installation.
|
|
||||||
|
|
||||||
goto fail
|
|
||||||
|
|
||||||
:init
|
|
||||||
@rem Get command-line arguments, handling Windowz variants
|
|
||||||
|
|
||||||
if not "%OS%" == "Windows_NT" goto win9xME_args
|
|
||||||
if "%@eval[2+2]" == "4" goto 4NT_args
|
|
||||||
|
|
||||||
:win9xME_args
|
|
||||||
@rem Slurp the command line arguments.
|
|
||||||
set CMD_LINE_ARGS=
|
|
||||||
set _SKIP=2
|
|
||||||
|
|
||||||
:win9xME_args_slurp
|
|
||||||
if "x%~1" == "x" goto execute
|
|
||||||
|
|
||||||
set CMD_LINE_ARGS=%*
|
|
||||||
goto execute
|
|
||||||
|
|
||||||
:4NT_args
|
|
||||||
@rem Get arguments from the 4NT Shell from JP Software
|
|
||||||
set CMD_LINE_ARGS=%$
|
|
||||||
|
|
||||||
:execute
|
|
||||||
@rem Setup the command line
|
|
||||||
|
|
||||||
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
|
||||||
|
|
||||||
@rem Execute Gradle
|
|
||||||
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
|
||||||
|
|
||||||
:end
|
|
||||||
@rem End local scope for the variables with windows NT shell
|
|
||||||
if "%ERRORLEVEL%"=="0" goto mainEnd
|
|
||||||
|
|
||||||
:fail
|
|
||||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
|
||||||
rem the _cmd.exe /c_ return code!
|
|
||||||
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
|
||||||
exit /b 1
|
|
||||||
|
|
||||||
:mainEnd
|
|
||||||
if "%OS%"=="Windows_NT" endlocal
|
|
||||||
|
|
||||||
:omega
|
|
|
@ -1,5 +0,0 @@
|
||||||
sdk.dir=/home/nate/Android/Sdk
|
|
||||||
flutter.sdk=/home/nate/Tooling/flutter
|
|
||||||
flutter.buildMode=debug
|
|
||||||
flutter.versionName=1.0.0
|
|
||||||
flutter.versionCode=1
|
|
|
@ -1,17 +0,0 @@
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<module type="JAVA_MODULE" version="4">
|
|
||||||
<component name="NewModuleRootManager" inherit-compiler-output="true">
|
|
||||||
<exclude-output />
|
|
||||||
<content url="file://$MODULE_DIR$">
|
|
||||||
<sourceFolder url="file://$MODULE_DIR$/lib" isTestSource="false" />
|
|
||||||
<sourceFolder url="file://$MODULE_DIR$/test" isTestSource="true" />
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/.dart_tool" />
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/.idea" />
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/build" />
|
|
||||||
</content>
|
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
|
||||||
<orderEntry type="library" name="Dart SDK" level="project" />
|
|
||||||
<orderEntry type="library" name="Flutter Plugins" level="project" />
|
|
||||||
<orderEntry type="library" name="Dart Packages" level="project" />
|
|
||||||
</component>
|
|
||||||
</module>
|
|
|
@ -1,19 +0,0 @@
|
||||||
//
|
|
||||||
// Generated file. Do not edit.
|
|
||||||
//
|
|
||||||
|
|
||||||
// clang-format off
|
|
||||||
|
|
||||||
#ifndef GeneratedPluginRegistrant_h
|
|
||||||
#define GeneratedPluginRegistrant_h
|
|
||||||
|
|
||||||
#import <Flutter/Flutter.h>
|
|
||||||
|
|
||||||
NS_ASSUME_NONNULL_BEGIN
|
|
||||||
|
|
||||||
@interface GeneratedPluginRegistrant : NSObject
|
|
||||||
+ (void)registerWithRegistry:(NSObject<FlutterPluginRegistry>*)registry;
|
|
||||||
@end
|
|
||||||
|
|
||||||
NS_ASSUME_NONNULL_END
|
|
||||||
#endif /* GeneratedPluginRegistrant_h */
|
|
|
@ -1,14 +0,0 @@
|
||||||
//
|
|
||||||
// Generated file. Do not edit.
|
|
||||||
//
|
|
||||||
|
|
||||||
// clang-format off
|
|
||||||
|
|
||||||
#import "GeneratedPluginRegistrant.h"
|
|
||||||
|
|
||||||
@implementation GeneratedPluginRegistrant
|
|
||||||
|
|
||||||
+ (void)registerWithRegistry:(NSObject<FlutterPluginRegistry>*)registry {
|
|
||||||
}
|
|
||||||
|
|
||||||
@end
|
|
|
@ -1,6 +1,5 @@
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:stripe_native_card_field/card_details.dart';
|
|
||||||
import 'package:stripe_native_card_field/stripe_native_card_field.dart';
|
import 'package:stripe_native_card_field/stripe_native_card_field.dart';
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
|
@ -14,9 +13,23 @@ class MyApp extends StatelessWidget {
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return MaterialApp(
|
return MaterialApp(
|
||||||
debugShowCheckedModeBanner: false,
|
title: 'Flutter Demo',
|
||||||
title: 'Native Stripe Field Demo',
|
|
||||||
theme: ThemeData(
|
theme: ThemeData(
|
||||||
|
// This is the theme of your application.
|
||||||
|
//
|
||||||
|
// TRY THIS: Try running your application with "flutter run". You'll see
|
||||||
|
// the application has a blue toolbar. Then, without quitting the app,
|
||||||
|
// try changing the seedColor in the colorScheme below to Colors.green
|
||||||
|
// and then invoke "hot reload" (save your changes or press the "hot
|
||||||
|
// reload" button in a Flutter-supported IDE, or press "r" if you used
|
||||||
|
// the command line to start the app).
|
||||||
|
//
|
||||||
|
// Notice that the counter didn't reset back to zero; the application
|
||||||
|
// state is not lost during the reload. To reset the state, use hot
|
||||||
|
// restart instead.
|
||||||
|
//
|
||||||
|
// This works for code too, not just values: Most code changes can be
|
||||||
|
// tested with just a hot reload.
|
||||||
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
|
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
|
||||||
useMaterial3: true,
|
useMaterial3: true,
|
||||||
),
|
),
|
||||||
|
@ -28,6 +41,15 @@ class MyApp extends StatelessWidget {
|
||||||
class MyHomePage extends StatefulWidget {
|
class MyHomePage extends StatefulWidget {
|
||||||
const MyHomePage({super.key, required this.title});
|
const MyHomePage({super.key, required this.title});
|
||||||
|
|
||||||
|
// This widget is the home page of your application. It is stateful, meaning
|
||||||
|
// that it has a State object (defined below) that contains fields that affect
|
||||||
|
// how it looks.
|
||||||
|
|
||||||
|
// This class is the configuration for the state. It holds the values (in this
|
||||||
|
// case the title) provided by the parent (in this case the App widget) and
|
||||||
|
// used by the build method of the State. Fields in a Widget subclass are
|
||||||
|
// always marked "final".
|
||||||
|
|
||||||
final String title;
|
final String title;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
@ -35,9 +57,6 @@ class MyHomePage extends StatefulWidget {
|
||||||
}
|
}
|
||||||
|
|
||||||
class _MyHomePageState extends State<MyHomePage> {
|
class _MyHomePageState extends State<MyHomePage> {
|
||||||
CardDetailsValidState? state;
|
|
||||||
String? errorText;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
|
@ -49,35 +68,15 @@ class _MyHomePageState extends State<MyHomePage> {
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
const Padding(
|
const Text(
|
||||||
padding: EdgeInsets.all(8.0),
|
|
||||||
child: Text(
|
|
||||||
'Enter your card details below:',
|
'Enter your card details below:',
|
||||||
),
|
),
|
||||||
),
|
|
||||||
CardTextField(
|
CardTextField(
|
||||||
width: 300,
|
width: 500,
|
||||||
onCardDetailsComplete: (details) {
|
onCardDetailsComplete: (details) {
|
||||||
if (kDebugMode) {
|
if (kDebugMode) print('Got card details: $details');
|
||||||
print(details);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
// textStyle: TextStyle(fontSize: 24.0),
|
|
||||||
// cardFieldWidth: 260,
|
|
||||||
// expFieldWidth: 100.0,
|
|
||||||
// securityFieldWidth: 60.0,
|
|
||||||
// postalFieldWidth: 130.0,
|
|
||||||
// iconSize: Size(50.0, 35.0),
|
|
||||||
overrideValidState: state,
|
|
||||||
errorText: errorText,
|
|
||||||
),
|
),
|
||||||
ElevatedButton(
|
|
||||||
child: const Text('Set manual error'),
|
|
||||||
onPressed: () => setState(() {
|
|
||||||
errorText = 'There is a problem';
|
|
||||||
state = CardDetailsValidState.invalidCard;
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 18 KiB |
|
@ -91,22 +91,6 @@ packages:
|
||||||
description: flutter
|
description: flutter
|
||||||
source: sdk
|
source: sdk
|
||||||
version: "0.0.0"
|
version: "0.0.0"
|
||||||
http:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: http
|
|
||||||
sha256: "759d1a329847dd0f39226c688d3e06a6b8679668e350e2891a6474f8b4bb8525"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "1.1.0"
|
|
||||||
http_parser:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: http_parser
|
|
||||||
sha256: "2aa08ce0341cc9b354a498388e30986515406668dbcc4f7c950c3e715496693b"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "4.0.2"
|
|
||||||
lints:
|
lints:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
@ -206,7 +190,7 @@ packages:
|
||||||
path: ".."
|
path: ".."
|
||||||
relative: true
|
relative: true
|
||||||
source: path
|
source: path
|
||||||
version: "0.0.2"
|
version: "0.0.1"
|
||||||
term_glyph:
|
term_glyph:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
@ -223,14 +207,6 @@ packages:
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.6.0"
|
version: "0.6.0"
|
||||||
typed_data:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: typed_data
|
|
||||||
sha256: facc8d6582f16042dd49f2463ff1bd6e2c9ef9f3d5da3d9b087e244a7b564b3c
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "1.3.2"
|
|
||||||
vector_graphics:
|
vector_graphics:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|
|
@ -7,11 +7,12 @@ import 'package:flutter/foundation.dart';
|
||||||
/// when fields are filled and validated as correct.
|
/// when fields are filled and validated as correct.
|
||||||
class CardDetails {
|
class CardDetails {
|
||||||
CardDetails({
|
CardDetails({
|
||||||
required String? cardNumber,
|
required dynamic cardNumber,
|
||||||
required this.securityCode,
|
required String? securityCode,
|
||||||
required this.expirationString,
|
required this.expirationString,
|
||||||
required this.postalCode,
|
required this.postalCode,
|
||||||
}) : _cardNumber = cardNumber {
|
}) : _cardNumber = cardNumber {
|
||||||
|
this.securityCode = int.tryParse(securityCode ?? '');
|
||||||
checkIsValid();
|
checkIsValid();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,26 +28,21 @@ class CardDetails {
|
||||||
set cardNumber(String? num) => _cardNumber = num;
|
set cardNumber(String? num) => _cardNumber = num;
|
||||||
|
|
||||||
String? _cardNumber;
|
String? _cardNumber;
|
||||||
String? securityCode;
|
int? securityCode;
|
||||||
String? postalCode;
|
String? postalCode;
|
||||||
String? expirationString;
|
String? expirationString;
|
||||||
DateTime? expirationDate;
|
DateTime? expirationDate;
|
||||||
bool _complete = false;
|
bool _complete = false;
|
||||||
CardDetailsValidState _validState = CardDetailsValidState.blank;
|
ValidState _validState = ValidState.blank;
|
||||||
int _lastCheckHash = 0;
|
int _lastCheckHash = 0;
|
||||||
CardProvider? provider;
|
CardProvider? provider;
|
||||||
|
|
||||||
set overrideValidState(CardDetailsValidState state) => _validState = state;
|
|
||||||
|
|
||||||
/// Checks the validity of the `CardDetails` and returns the result.
|
/// Checks the validity of the `CardDetails` and returns the result.
|
||||||
CardDetailsValidState get validState {
|
ValidState get validState {
|
||||||
checkIsValid();
|
checkIsValid();
|
||||||
return _validState;
|
return _validState;
|
||||||
}
|
}
|
||||||
|
|
||||||
String get expMonth => isComplete ? expirationString!.split('/').first : '';
|
|
||||||
String get expYear => isComplete ? expirationString!.split('/').last : '';
|
|
||||||
|
|
||||||
// TODO rename to be more clear
|
// TODO rename to be more clear
|
||||||
/// Returns true if `_cardNumber` is null, or
|
/// Returns true if `_cardNumber` is null, or
|
||||||
/// if the _cardNumber matches the detected `provider`'s
|
/// if the _cardNumber matches the detected `provider`'s
|
||||||
|
@ -80,7 +76,7 @@ class CardDetails {
|
||||||
_lastCheckHash = currentHash;
|
_lastCheckHash = currentHash;
|
||||||
if (_cardNumber == null && expirationString == null && securityCode == null && postalCode == null) {
|
if (_cardNumber == null && expirationString == null && securityCode == null && postalCode == null) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.blank;
|
_validState = ValidState.blank;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final nums = _cardNumber!
|
final nums = _cardNumber!
|
||||||
|
@ -92,71 +88,66 @@ class CardDetails {
|
||||||
.toList();
|
.toList();
|
||||||
if (!_luhnAlgorithmCheck(nums)) {
|
if (!_luhnAlgorithmCheck(nums)) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.invalidCard;
|
_validState = ValidState.invalidCard;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (_cardNumber == null || !cardNumberFilled) {
|
if (_cardNumber == null || !cardNumberFilled) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.missingCard;
|
_validState = ValidState.missingCard;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (expirationString == null) {
|
if (expirationString == null) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.missingDate;
|
_validState = ValidState.missingDate;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final expSplits = expirationString!.split('/');
|
final expSplits = expirationString!.split('/');
|
||||||
if (expSplits.length != 2 || expSplits.last == '') {
|
if (expSplits.length != 2 || expSplits.last == '') {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.missingDate;
|
_validState = ValidState.missingDate;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final month = int.parse(expSplits.first[0] == '0' ? expSplits.first[1] : expSplits.first);
|
final month = int.parse(expSplits.first[0] == '0' ? expSplits.first[1] : expSplits.first);
|
||||||
if (month < 1 || month > 12) {
|
if (month < 1 || month > 12) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.invalidMonth;
|
_validState = ValidState.invalidMonth;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final year = 2000 + int.parse(expSplits.last);
|
final year = 2000 + int.parse(expSplits.last);
|
||||||
final date = DateTime(year, month);
|
final date = DateTime(year, month);
|
||||||
if (date.isBefore(DateTime.now())) {
|
if (date.isBefore(DateTime.now())) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.dateTooEarly;
|
_validState = ValidState.dateTooEarly;
|
||||||
return;
|
return;
|
||||||
} else if (date.isAfter(DateTime.now().add(const Duration(days: 365 * 50)))) {
|
} else if (date.isAfter(DateTime.now().add(const Duration(days: 365 * 50)))) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.dateTooLate;
|
_validState = ValidState.dateTooLate;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
expirationDate = date;
|
expirationDate = date;
|
||||||
if (securityCode == null) {
|
if (securityCode == null) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.missingCVC;
|
_validState = ValidState.missingCVC;
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (provider != null && securityCode!.length != provider!.cvcLength) {
|
|
||||||
_complete = false;
|
|
||||||
_validState = CardDetailsValidState.invalidCVC;
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (postalCode == null) {
|
if (postalCode == null) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.missingZip;
|
_validState = ValidState.missingZip;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!RegExp(r'^\d{5}(-\d{4})?$').hasMatch(postalCode!)) {
|
if (!RegExp(r'^\d{5}(-\d{4})?$').hasMatch(postalCode!)) {
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.invalidZip;
|
_validState = ValidState.invalidZip;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_complete = true;
|
_complete = true;
|
||||||
_validState = CardDetailsValidState.ok;
|
_validState = ValidState.ok;
|
||||||
} catch (err, st) {
|
} catch (err, st) {
|
||||||
if (kDebugMode) {
|
if (kDebugMode) {
|
||||||
print('Error while validating CardDetails: $err\n$st');
|
print('Error while validating CardDetails: $err\n$st');
|
||||||
}
|
}
|
||||||
_complete = false;
|
_complete = false;
|
||||||
_validState = CardDetailsValidState.error;
|
_validState = ValidState.error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -245,7 +236,7 @@ class CardDetails {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Enum of validation states a `CardDetails` object can have.
|
/// Enum of validation states a `CardDetails` object can have.
|
||||||
enum CardDetailsValidState {
|
enum ValidState {
|
||||||
ok,
|
ok,
|
||||||
error,
|
error,
|
||||||
blank,
|
blank,
|
||||||
|
|
|
@ -7,10 +7,9 @@ import 'package:flutter_svg/flutter_svg.dart';
|
||||||
///
|
///
|
||||||
/// To see a list of supported card providers, see `CardDetails.provider`.
|
/// To see a list of supported card providers, see `CardDetails.provider`.
|
||||||
class CardProviderIcon extends StatefulWidget {
|
class CardProviderIcon extends StatefulWidget {
|
||||||
const CardProviderIcon({required this.cardDetails, this.size, super.key});
|
const CardProviderIcon({required this.cardDetails, super.key});
|
||||||
|
|
||||||
final CardDetails? cardDetails;
|
final CardDetails? cardDetails;
|
||||||
final Size? size;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<CardProviderIcon> createState() => _CardProviderIconState();
|
State<CardProviderIcon> createState() => _CardProviderIconState();
|
||||||
|
@ -35,29 +34,22 @@ class _CardProviderIconState extends State<CardProviderIcon> {
|
||||||
CardProviderID.jcb.name:
|
CardProviderID.jcb.name:
|
||||||
'<svg enable-background="new 0 0 780 500" height="500" viewBox="0 0 780 500" width="780" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"><linearGradient id="a" gradientTransform="matrix(132.87 0 0 323.02 -120270 -100930)" gradientUnits="userSpaceOnUse" x1="908.72" x2="909.72" y1="313.21" y2="313.21"><stop offset="0" stop-color="#007b40"/><stop offset="1" stop-color="#55b330"/></linearGradient><linearGradient id="b" gradientTransform="matrix(133.43 0 0 323.02 -121080 -100920)" gradientUnits="userSpaceOnUse" x1="908.73" x2="909.73" y1="313.21" y2="313.21"><stop offset="0" stop-color="#1d2970"/><stop offset="1" stop-color="#006dba"/></linearGradient><linearGradient id="c" gradientTransform="matrix(132.96 0 0 323.03 -120500 -100930)" gradientUnits="userSpaceOnUse" x1="908.72" x2="909.72" y1="313.21" y2="313.21"><stop offset="0" stop-color="#6e2b2f"/><stop offset="1" stop-color="#e30138"/></linearGradient><path d="m632.24 361.27c0 41.615-33.729 75.36-75.357 75.36h-409.13v-297.88c0-41.626 33.73-75.371 75.364-75.371h409.12l-.001 297.89z" fill="#fff"/><path d="m498.86 256.54c11.686.254 23.438-.516 35.077.4 11.787 2.199 14.628 20.043 4.156 25.887-7.145 3.85-15.633 1.434-23.379 2.113h-15.854zm41.834-32.145c2.596 9.164-6.238 17.392-15.064 16.13h-26.77c.188-8.642-.367-18.022.272-26.209 10.724.302 21.547-.616 32.209.48 4.581 1.151 8.415 4.917 9.353 9.599zm64.425-135.9c.498 17.501.072 35.927.215 53.783-.033 72.596.07 145.19-.057 217.79-.47 27.207-24.582 50.848-51.601 51.391-27.045.11-54.094.017-81.143.047v-109.75c29.471-.152 58.957.309 88.416-.23 13.666-.858 28.635-9.875 29.271-24.914 1.609-15.104-12.631-25.551-26.151-27.201-5.197-.135-5.045-1.515 0-2.117 12.895-2.787 23.021-16.133 19.227-29.499-3.233-14.058-18.771-19.499-31.695-19.472-26.352-.179-52.709-.025-79.062-.077.17-20.489-.355-41 .283-61.474 2.088-26.716 26.807-48.748 53.446-48.27 26.287-.004 52.57-.004 78.851-.005z" fill="url(#a)"/><path d="m174.74 139.54c.673-27.164 24.888-50.611 51.872-51.008 26.945-.083 53.894-.012 80.839-.036-.074 90.885.146 181.78-.111 272.66-1.038 26.834-24.989 49.834-51.679 50.309-26.996.098-53.995.014-80.992.041v-113.45c26.223 6.195 53.722 8.832 80.474 4.723 15.991-2.573 33.487-10.426 38.901-27.016 3.984-14.191 1.741-29.126 2.334-43.691v-33.825h-46.297c-.208 22.371.426 44.781-.335 67.125-1.248 13.734-14.849 22.46-27.802 21.994-16.064.17-47.897-11.642-47.897-11.642-.08-41.914.466-94.405.693-136.18z" fill="url(#b)"/><path d="m324.72 211.89c-2.437.517-.49-8.301-1.113-11.646.166-21.15-.347-42.323.283-63.458 2.082-26.829 26.991-48.916 53.738-48.288h78.768c-.074 90.885.145 181.78-.111 272.66-1.039 26.834-24.992 49.833-51.683 50.309-26.997.102-53.997.016-80.996.042v-124.3c18.439 15.129 43.5 17.484 66.472 17.525 17.318-.006 34.535-2.676 51.353-6.67v-22.772c-18.953 9.446-41.233 15.446-62.243 10.019-14.656-3.648-25.295-17.812-25.058-32.937-1.698-15.729 7.522-32.335 22.979-37.011 19.191-6.008 40.107-1.413 58.096 6.398 3.854 2.018 7.766 4.521 6.225-1.921v-17.899c-30.086-7.158-62.104-9.792-92.33-2.005-8.749 2.468-17.273 6.211-24.38 11.956z" fill="url(#c)"/></svg>',
|
'<svg enable-background="new 0 0 780 500" height="500" viewBox="0 0 780 500" width="780" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"><linearGradient id="a" gradientTransform="matrix(132.87 0 0 323.02 -120270 -100930)" gradientUnits="userSpaceOnUse" x1="908.72" x2="909.72" y1="313.21" y2="313.21"><stop offset="0" stop-color="#007b40"/><stop offset="1" stop-color="#55b330"/></linearGradient><linearGradient id="b" gradientTransform="matrix(133.43 0 0 323.02 -121080 -100920)" gradientUnits="userSpaceOnUse" x1="908.73" x2="909.73" y1="313.21" y2="313.21"><stop offset="0" stop-color="#1d2970"/><stop offset="1" stop-color="#006dba"/></linearGradient><linearGradient id="c" gradientTransform="matrix(132.96 0 0 323.03 -120500 -100930)" gradientUnits="userSpaceOnUse" x1="908.72" x2="909.72" y1="313.21" y2="313.21"><stop offset="0" stop-color="#6e2b2f"/><stop offset="1" stop-color="#e30138"/></linearGradient><path d="m632.24 361.27c0 41.615-33.729 75.36-75.357 75.36h-409.13v-297.88c0-41.626 33.73-75.371 75.364-75.371h409.12l-.001 297.89z" fill="#fff"/><path d="m498.86 256.54c11.686.254 23.438-.516 35.077.4 11.787 2.199 14.628 20.043 4.156 25.887-7.145 3.85-15.633 1.434-23.379 2.113h-15.854zm41.834-32.145c2.596 9.164-6.238 17.392-15.064 16.13h-26.77c.188-8.642-.367-18.022.272-26.209 10.724.302 21.547-.616 32.209.48 4.581 1.151 8.415 4.917 9.353 9.599zm64.425-135.9c.498 17.501.072 35.927.215 53.783-.033 72.596.07 145.19-.057 217.79-.47 27.207-24.582 50.848-51.601 51.391-27.045.11-54.094.017-81.143.047v-109.75c29.471-.152 58.957.309 88.416-.23 13.666-.858 28.635-9.875 29.271-24.914 1.609-15.104-12.631-25.551-26.151-27.201-5.197-.135-5.045-1.515 0-2.117 12.895-2.787 23.021-16.133 19.227-29.499-3.233-14.058-18.771-19.499-31.695-19.472-26.352-.179-52.709-.025-79.062-.077.17-20.489-.355-41 .283-61.474 2.088-26.716 26.807-48.748 53.446-48.27 26.287-.004 52.57-.004 78.851-.005z" fill="url(#a)"/><path d="m174.74 139.54c.673-27.164 24.888-50.611 51.872-51.008 26.945-.083 53.894-.012 80.839-.036-.074 90.885.146 181.78-.111 272.66-1.038 26.834-24.989 49.834-51.679 50.309-26.996.098-53.995.014-80.992.041v-113.45c26.223 6.195 53.722 8.832 80.474 4.723 15.991-2.573 33.487-10.426 38.901-27.016 3.984-14.191 1.741-29.126 2.334-43.691v-33.825h-46.297c-.208 22.371.426 44.781-.335 67.125-1.248 13.734-14.849 22.46-27.802 21.994-16.064.17-47.897-11.642-47.897-11.642-.08-41.914.466-94.405.693-136.18z" fill="url(#b)"/><path d="m324.72 211.89c-2.437.517-.49-8.301-1.113-11.646.166-21.15-.347-42.323.283-63.458 2.082-26.829 26.991-48.916 53.738-48.288h78.768c-.074 90.885.145 181.78-.111 272.66-1.039 26.834-24.992 49.833-51.683 50.309-26.997.102-53.997.016-80.996.042v-124.3c18.439 15.129 43.5 17.484 66.472 17.525 17.318-.006 34.535-2.676 51.353-6.67v-22.772c-18.953 9.446-41.233 15.446-62.243 10.019-14.656-3.648-25.295-17.812-25.058-32.937-1.698-15.729 7.522-32.335 22.979-37.011 19.191-6.008 40.107-1.413 58.096 6.398 3.854 2.018 7.766 4.521 6.225-1.921v-17.899c-30.086-7.158-62.104-9.792-92.33-2.005-8.749 2.468-17.273 6.211-24.38 11.956z" fill="url(#c)"/></svg>',
|
||||||
};
|
};
|
||||||
|
final double height = 20;
|
||||||
late final Size _size;
|
final double width = 30;
|
||||||
|
|
||||||
@override
|
|
||||||
initState() {
|
|
||||||
super.initState();
|
|
||||||
|
|
||||||
_size = widget.size ?? const Size(30.0, 20.0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
late Widget child;
|
late Widget child;
|
||||||
if (widget.cardDetails?.cardNumber != null &&
|
if (widget.cardDetails?.cardNumber != null &&
|
||||||
widget.cardDetails!.cardNumberFilled &&
|
widget.cardDetails!.cardNumberFilled &&
|
||||||
widget.cardDetails!.validState == CardDetailsValidState.invalidCard) {
|
widget.cardDetails!.validState == ValidState.invalidCard) {
|
||||||
child = Padding(
|
child = Padding(
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 5.0),
|
padding: const EdgeInsets.symmetric(horizontal: 5.0),
|
||||||
child: SvgPicture.string(
|
child: SvgPicture.string(
|
||||||
key: const Key('invalid-card'),
|
key: const Key('invalid-card'),
|
||||||
cardProviderSvg['error']!,
|
cardProviderSvg['error']!,
|
||||||
height: _size.height,
|
height: height,
|
||||||
width: _size.width,
|
width: width,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
@ -67,8 +59,8 @@ class _CardProviderIconState extends State<CardProviderIcon> {
|
||||||
child: SvgPicture.string(
|
child: SvgPicture.string(
|
||||||
key: const Key('credit_card'),
|
key: const Key('credit_card'),
|
||||||
cardProviderSvg['credit-card']!,
|
cardProviderSvg['credit-card']!,
|
||||||
height: _size.height,
|
height: height,
|
||||||
width: _size.width,
|
width: width,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
@ -94,8 +86,8 @@ class _CardProviderIconState extends State<CardProviderIcon> {
|
||||||
return SvgPicture.string(
|
return SvgPicture.string(
|
||||||
key: Key('${id.name}-card'),
|
key: Key('${id.name}-card'),
|
||||||
cardProviderSvg[id.name]!,
|
cardProviderSvg[id.name]!,
|
||||||
height: _size.height,
|
height: height,
|
||||||
width: _size.width,
|
width: width,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,141 +1,42 @@
|
||||||
library stripe_native_card_field;
|
library stripe_native_card_field;
|
||||||
|
|
||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
import 'dart:convert';
|
|
||||||
import 'dart:developer';
|
|
||||||
import 'dart:io';
|
|
||||||
|
|
||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:http/http.dart' as http;
|
|
||||||
|
|
||||||
import 'card_details.dart';
|
import 'card_details.dart';
|
||||||
import 'card_provider_icon.dart';
|
import 'card_provider_icon.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
|
|
||||||
/// Enum to track each step of the card detail
|
/// Enum to track each step of the card detail
|
||||||
/// entry process.
|
/// entry process.
|
||||||
enum CardEntryStep { number, exp, cvc, postal }
|
enum CardEntryStep { number, exp, cvc, postal }
|
||||||
|
|
||||||
// enum LoadingLocation { ontop, rightInside }
|
|
||||||
|
|
||||||
/// A uniform text field for entering card details, based
|
/// A uniform text field for entering card details, based
|
||||||
/// on the behavior of Stripe's various html elements.
|
/// on the behavior of Stripe's various html elements.
|
||||||
///
|
///
|
||||||
/// Required `width`.
|
/// Required `width` and `onCardDetailsComplete`.
|
||||||
///
|
///
|
||||||
/// If the provided `width < 450.0`, the `CardTextField`
|
/// If the provided `width < 450.0`, the `CardTextField`
|
||||||
/// will scroll its content horizontally with the cursor
|
/// will scroll its content horizontally with the cursor
|
||||||
/// to compensate.
|
/// to compensate.
|
||||||
class CardTextField extends StatefulWidget {
|
class CardTextField extends StatefulWidget {
|
||||||
CardTextField({
|
const CardTextField(
|
||||||
Key? key,
|
{Key? key,
|
||||||
|
required this.onCardDetailsComplete,
|
||||||
required this.width,
|
required this.width,
|
||||||
this.onStripeResponse,
|
|
||||||
this.onCardDetailsComplete,
|
|
||||||
this.stripePublishableKey,
|
|
||||||
this.height,
|
this.height,
|
||||||
this.textStyle,
|
this.inputDecoration,
|
||||||
this.hintTextStyle,
|
|
||||||
this.errorTextStyle,
|
|
||||||
this.boxDecoration,
|
this.boxDecoration,
|
||||||
this.errorBoxDecoration,
|
this.errorBoxDecoration})
|
||||||
this.loadingWidget,
|
: super(key: key);
|
||||||
this.showInternalLoadingWidget = true,
|
|
||||||
this.delayToShowLoading = const Duration(milliseconds: 750),
|
|
||||||
this.onCallToStripe,
|
|
||||||
this.overrideValidState,
|
|
||||||
this.errorText,
|
|
||||||
this.cardFieldWidth,
|
|
||||||
this.expFieldWidth,
|
|
||||||
this.securityFieldWidth,
|
|
||||||
this.postalFieldWidth,
|
|
||||||
this.iconSize,
|
|
||||||
// this.loadingWidgetLocation = LoadingLocation.rightInside,
|
|
||||||
}) : super(key: key) {
|
|
||||||
if (stripePublishableKey != null) {
|
|
||||||
assert(stripePublishableKey!.startsWith('pk_'));
|
|
||||||
if (kReleaseMode && !stripePublishableKey!.startsWith('pk_live_')) {
|
|
||||||
log('StripeNativeCardField: *WARN* You are not using a live publishableKey in production.');
|
|
||||||
} else if ((kDebugMode || kProfileMode) && stripePublishableKey!.startsWith('pk_live_')) {
|
|
||||||
log('StripeNativeCardField: *WARN* You are using a live stripe key in a debug environment, proceed with caution!');
|
|
||||||
log('StripeNativeCardField: *WARN* Ideally you should be using your test keys whenever not in production.');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (onStripeResponse != null) {
|
|
||||||
log('StripeNativeCardField: *ERROR* You provided the onTokenReceived callback, but did not provide a stripePublishableKey.');
|
|
||||||
assert(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Width of the entire CardTextField
|
final InputDecoration? inputDecoration; // TODO unapplied style
|
||||||
|
final BoxDecoration? boxDecoration; // TODO unapplied style
|
||||||
|
final BoxDecoration? errorBoxDecoration; // TODO unapplied style
|
||||||
final double width;
|
final double width;
|
||||||
|
|
||||||
/// Height of the entire CardTextField, defaults to 60.0
|
|
||||||
final double? height;
|
|
||||||
|
|
||||||
/// Width of card number field, only override if changing the default `textStyle.fontSize`, defaults to 180.0
|
|
||||||
final double? cardFieldWidth;
|
|
||||||
|
|
||||||
/// Width of expiration date field, only override if changing the default `textStyle.fontSize`, defaults to 70.0
|
|
||||||
final double? expFieldWidth;
|
|
||||||
|
|
||||||
/// Width of security number field, only override if changing the default `textStyle.fontSize`, defaults to 40.0
|
|
||||||
final double? securityFieldWidth;
|
|
||||||
|
|
||||||
/// Width of postal code field, only override if changing the default `textStyle.fontSize`, defaults to 95.0
|
|
||||||
final double? postalFieldWidth;
|
|
||||||
|
|
||||||
/// Overrides the default box decoration of the text field
|
|
||||||
final BoxDecoration? boxDecoration;
|
|
||||||
|
|
||||||
/// Overrides the default box decoration of the text field when there is a validation error
|
|
||||||
final BoxDecoration? errorBoxDecoration;
|
|
||||||
|
|
||||||
/// Shown and overrides CircularProgressIndicator() if the request to stripe takes longer than `delayToShowLoading`
|
|
||||||
final Widget? loadingWidget;
|
|
||||||
|
|
||||||
/// Overrides default icon size of the card provider, defaults to `Size(30.0, 20.0)`
|
|
||||||
final Size? iconSize;
|
|
||||||
|
|
||||||
/// Determines where the loading indicator appears when contacting stripe
|
|
||||||
// final LoadingLocation loadingWidgetLocation;
|
|
||||||
|
|
||||||
/// Default TextStyle
|
|
||||||
final TextStyle? textStyle;
|
|
||||||
|
|
||||||
/// Default TextStyle for the hint text in each TextFormField.
|
|
||||||
/// If null, inherits from the `textStyle`.
|
|
||||||
final TextStyle? hintTextStyle;
|
|
||||||
|
|
||||||
/// TextStyle used when any TextFormField's have a validation error
|
|
||||||
/// If null, inherits from the `textStyle`.
|
|
||||||
final TextStyle? errorTextStyle;
|
|
||||||
|
|
||||||
/// Time to wait until showing the loading indicator when retrieving Stripe token
|
|
||||||
final Duration delayToShowLoading;
|
|
||||||
|
|
||||||
/// Whether to show the internal loading widget on calls to Stripe
|
|
||||||
final bool showInternalLoadingWidget;
|
|
||||||
|
|
||||||
/// Stripe publishable key, starts with 'pk_'
|
|
||||||
final String? stripePublishableKey;
|
|
||||||
|
|
||||||
/// Callback when the http request is made to Stripe
|
|
||||||
final void Function()? onCallToStripe;
|
|
||||||
|
|
||||||
/// Callback that returns the stripe token for the card
|
|
||||||
final void Function(Map<String, dynamic>)? onStripeResponse;
|
|
||||||
|
|
||||||
/// Callback that returns the completed CardDetails object
|
/// Callback that returns the completed CardDetails object
|
||||||
final void Function(CardDetails)? onCardDetailsComplete;
|
final void Function(CardDetails) onCardDetailsComplete;
|
||||||
|
final double? height;
|
||||||
/// Can manually override the ValidState to surface errors returned from Stripe
|
|
||||||
final CardDetailsValidState? overrideValidState;
|
|
||||||
|
|
||||||
/// Can manually override the errorText displayed to surface errors returned from Stripe
|
|
||||||
final String? errorText;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<CardTextField> createState() => CardTextFieldState();
|
State<CardTextField> createState() => CardTextFieldState();
|
||||||
|
@ -151,51 +52,20 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
late TextEditingController _securityCodeController;
|
late TextEditingController _securityCodeController;
|
||||||
late TextEditingController _postalCodeController;
|
late TextEditingController _postalCodeController;
|
||||||
|
|
||||||
// Not made private for access in widget tests
|
|
||||||
late FocusNode cardNumberFocusNode;
|
late FocusNode cardNumberFocusNode;
|
||||||
late FocusNode expirationFocusNode;
|
late FocusNode expirationFocusNode;
|
||||||
late FocusNode securityCodeFocusNode;
|
late FocusNode securityCodeFocusNode;
|
||||||
late FocusNode postalCodeFocusNode;
|
late FocusNode postalCodeFocusNode;
|
||||||
|
|
||||||
// Not made private for access in widget tests
|
final double _cardFieldWidth = 180.0;
|
||||||
late final bool isWideFormat;
|
final double _expirationFieldWidth = 70.0;
|
||||||
|
final double _securityFieldWidth = 40.0;
|
||||||
// Widget configurable styles
|
final double _postalFieldWidth = 100.0;
|
||||||
late final BoxDecoration _normalBoxDecoration;
|
|
||||||
late final BoxDecoration _errorBoxDecoration;
|
|
||||||
late final TextStyle _errorTextStyle;
|
|
||||||
late final TextStyle _normalTextStyle;
|
|
||||||
late final TextStyle _hintTextSyle;
|
|
||||||
|
|
||||||
/// Width of the card number text field
|
|
||||||
late final double _cardFieldWidth;
|
|
||||||
|
|
||||||
/// Width of the expiration text field
|
|
||||||
late final double _expirationFieldWidth;
|
|
||||||
|
|
||||||
/// Width of the security code text field
|
|
||||||
late final double _securityFieldWidth;
|
|
||||||
|
|
||||||
/// Width of the postal code text field
|
|
||||||
late final double _postalFieldWidth;
|
|
||||||
|
|
||||||
/// Width of the internal scrollable field, is potentially larger than the provided `widget.width`
|
|
||||||
late final double _internalFieldWidth;
|
late final double _internalFieldWidth;
|
||||||
|
late final bool _isWideFormat;
|
||||||
|
|
||||||
/// Width of the gap between card number and expiration text fields when expanded
|
|
||||||
late final double _expanderWidthExpanded;
|
|
||||||
|
|
||||||
/// Width of the gap between card number and expiration text fields when collapsed
|
|
||||||
late final double _expanderWidthCollapsed;
|
|
||||||
|
|
||||||
String? _validationErrorText;
|
|
||||||
bool _showBorderError = false;
|
bool _showBorderError = false;
|
||||||
final _isMobile = Platform.isAndroid || Platform.isIOS;
|
String? _validationErrorText;
|
||||||
|
|
||||||
/// If a request to Stripe is being made
|
|
||||||
bool _loading = false;
|
|
||||||
final CardDetails _cardDetails = CardDetails.blank();
|
|
||||||
int _prevErrorOverrideHash = 0;
|
|
||||||
|
|
||||||
final _currentCardEntryStepController = StreamController<CardEntryStep>();
|
final _currentCardEntryStepController = StreamController<CardEntryStep>();
|
||||||
final _horizontalScrollController = ScrollController();
|
final _horizontalScrollController = ScrollController();
|
||||||
|
@ -203,86 +73,51 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
|
|
||||||
final _formFieldKey = GlobalKey<FormState>();
|
final _formFieldKey = GlobalKey<FormState>();
|
||||||
|
|
||||||
@override
|
final CardDetails _cardDetails = CardDetails.blank();
|
||||||
void initState() {
|
|
||||||
_cardFieldWidth = widget.cardFieldWidth ?? 180.0;
|
|
||||||
_expirationFieldWidth = widget.expFieldWidth ?? 70.0;
|
|
||||||
_securityFieldWidth = widget.securityFieldWidth ?? 40.0;
|
|
||||||
_postalFieldWidth = widget.postalFieldWidth ?? 95.0;
|
|
||||||
|
|
||||||
// No way to get backspace events on soft keyboards, so add invisible character to detect delete
|
final normalBoxDecoration = BoxDecoration(
|
||||||
_cardNumberController = TextEditingController();
|
|
||||||
_expirationController = TextEditingController(text: _isMobile ? '\u200b' : '');
|
|
||||||
_securityCodeController = TextEditingController(text: _isMobile ? '\u200b' : '');
|
|
||||||
_postalCodeController = TextEditingController(text: _isMobile ? '\u200b' : '');
|
|
||||||
|
|
||||||
// Otherwise, use `RawKeyboard` listener
|
|
||||||
if (!_isMobile) {
|
|
||||||
RawKeyboard.instance.addListener(_backspaceTransitionListener);
|
|
||||||
}
|
|
||||||
|
|
||||||
cardNumberFocusNode = FocusNode();
|
|
||||||
expirationFocusNode = FocusNode();
|
|
||||||
securityCodeFocusNode = FocusNode();
|
|
||||||
postalCodeFocusNode = FocusNode();
|
|
||||||
|
|
||||||
_errorTextStyle = const TextStyle(color: Colors.red, fontSize: 14, inherit: true)
|
|
||||||
.merge(widget.errorTextStyle ?? widget.textStyle);
|
|
||||||
_normalTextStyle = const TextStyle(color: Colors.black87, fontSize: 14, inherit: true).merge(widget.textStyle);
|
|
||||||
_hintTextSyle = const TextStyle(color: Colors.black54, fontSize: 14, inherit: true)
|
|
||||||
.merge(widget.hintTextStyle ?? widget.textStyle);
|
|
||||||
|
|
||||||
_normalBoxDecoration = BoxDecoration(
|
|
||||||
color: const Color(0xfff6f9fc),
|
color: const Color(0xfff6f9fc),
|
||||||
border: Border.all(
|
border: Border.all(
|
||||||
color: const Color(0xffdde0e3),
|
color: const Color(0xffdde0e3),
|
||||||
width: 2.0,
|
width: 2.0,
|
||||||
),
|
),
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
borderRadius: BorderRadius.circular(8.0),
|
||||||
).copyWith(
|
|
||||||
backgroundBlendMode: widget.boxDecoration?.backgroundBlendMode,
|
|
||||||
border: widget.boxDecoration?.border,
|
|
||||||
borderRadius: widget.boxDecoration?.borderRadius,
|
|
||||||
boxShadow: widget.boxDecoration?.boxShadow,
|
|
||||||
color: widget.boxDecoration?.color,
|
|
||||||
gradient: widget.boxDecoration?.gradient,
|
|
||||||
image: widget.boxDecoration?.image,
|
|
||||||
shape: widget.boxDecoration?.shape,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
_errorBoxDecoration = BoxDecoration(
|
final errorBoxDecoration = BoxDecoration(
|
||||||
color: const Color(0xfff6f9fc),
|
color: const Color(0xfff6f9fc),
|
||||||
border: Border.all(
|
border: Border.all(
|
||||||
color: Colors.red,
|
color: Colors.red,
|
||||||
width: 2.0,
|
width: 2.0,
|
||||||
),
|
),
|
||||||
borderRadius: BorderRadius.circular(8.0),
|
borderRadius: BorderRadius.circular(8.0),
|
||||||
).copyWith(
|
|
||||||
backgroundBlendMode: widget.errorBoxDecoration?.backgroundBlendMode,
|
|
||||||
border: widget.errorBoxDecoration?.border,
|
|
||||||
borderRadius: widget.errorBoxDecoration?.borderRadius,
|
|
||||||
boxShadow: widget.errorBoxDecoration?.boxShadow,
|
|
||||||
color: widget.errorBoxDecoration?.color,
|
|
||||||
gradient: widget.errorBoxDecoration?.gradient,
|
|
||||||
image: widget.errorBoxDecoration?.image,
|
|
||||||
shape: widget.errorBoxDecoration?.shape,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
final TextStyle _errorTextStyle = const TextStyle(color: Colors.red, fontSize: 14);
|
||||||
|
final TextStyle _normalTextStyle = const TextStyle(color: Colors.black87, fontSize: 14);
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
_cardNumberController = TextEditingController();
|
||||||
|
_expirationController = TextEditingController();
|
||||||
|
_securityCodeController = TextEditingController();
|
||||||
|
_postalCodeController = TextEditingController();
|
||||||
|
|
||||||
|
cardNumberFocusNode = FocusNode();
|
||||||
|
expirationFocusNode = FocusNode();
|
||||||
|
securityCodeFocusNode = FocusNode();
|
||||||
|
postalCodeFocusNode = FocusNode();
|
||||||
|
|
||||||
_currentCardEntryStepController.stream.listen(
|
_currentCardEntryStepController.stream.listen(
|
||||||
_onStepChange,
|
_onStepChange,
|
||||||
);
|
);
|
||||||
|
RawKeyboard.instance.addListener(_backspaceTransitionListener);
|
||||||
isWideFormat =
|
_isWideFormat = widget.width >= 450;
|
||||||
widget.width >= _cardFieldWidth + _expirationFieldWidth + _securityFieldWidth + _postalFieldWidth + 60.0;
|
if (_isWideFormat) {
|
||||||
if (isWideFormat) {
|
_internalFieldWidth = widget.width + 80;
|
||||||
_internalFieldWidth = widget.width + _postalFieldWidth + 35;
|
|
||||||
_expanderWidthExpanded = widget.width - _cardFieldWidth - _expirationFieldWidth - _securityFieldWidth - 35;
|
|
||||||
_expanderWidthCollapsed =
|
|
||||||
widget.width - _cardFieldWidth - _expirationFieldWidth - _securityFieldWidth - _postalFieldWidth - 70;
|
|
||||||
} else {
|
} else {
|
||||||
_internalFieldWidth = _cardFieldWidth + _expirationFieldWidth + _securityFieldWidth + _postalFieldWidth + 80;
|
_internalFieldWidth = _cardFieldWidth + _expirationFieldWidth + _securityFieldWidth + _postalFieldWidth + 80;
|
||||||
}
|
}
|
||||||
|
|
||||||
super.initState();
|
super.initState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -296,22 +131,14 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
expirationFocusNode.dispose();
|
expirationFocusNode.dispose();
|
||||||
securityCodeFocusNode.dispose();
|
securityCodeFocusNode.dispose();
|
||||||
|
|
||||||
if (!_isMobile) {
|
|
||||||
RawKeyboard.instance.removeListener(_backspaceTransitionListener);
|
RawKeyboard.instance.removeListener(_backspaceTransitionListener);
|
||||||
}
|
|
||||||
|
|
||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
if ((widget.errorText != null || widget.overrideValidState != null) &&
|
|
||||||
Object.hashAll([widget.errorText, widget.overrideValidState]) != _prevErrorOverrideHash) {
|
|
||||||
_prevErrorOverrideHash = Object.hashAll([widget.errorText, widget.overrideValidState]);
|
|
||||||
_validateFields();
|
|
||||||
}
|
|
||||||
return Column(
|
return Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Form(
|
Form(
|
||||||
|
@ -321,34 +148,10 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
// Focuses to the current field
|
// Focuses to the current field
|
||||||
_currentCardEntryStepController.add(_currentStep);
|
_currentCardEntryStepController.add(_currentStep);
|
||||||
},
|
},
|
||||||
// Enable scrolling on mobile and if its narrow (not all fields visible)
|
|
||||||
onHorizontalDragUpdate: (details) {
|
|
||||||
if (!_isMobile || isWideFormat) return;
|
|
||||||
final newOffset = _horizontalScrollController.offset - details.delta.dx;
|
|
||||||
|
|
||||||
final max = _horizontalScrollController.position.maxScrollExtent;
|
|
||||||
if (newOffset < -30.0) {
|
|
||||||
_horizontalScrollController.jumpTo(-30.0);
|
|
||||||
} else if (newOffset > max + 30.0) {
|
|
||||||
_horizontalScrollController.jumpTo(max + 30.0);
|
|
||||||
} else {
|
|
||||||
_horizontalScrollController.jumpTo(newOffset);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
onHorizontalDragEnd: (details) {
|
|
||||||
if (!_isMobile || isWideFormat || details.primaryVelocity == null) return;
|
|
||||||
|
|
||||||
const dur = Duration(milliseconds: 300);
|
|
||||||
const cur = Curves.ease;
|
|
||||||
|
|
||||||
// final max = _horizontalScrollController.position.maxScrollExtent;
|
|
||||||
final newOffset = _horizontalScrollController.offset - details.primaryVelocity! * 0.15;
|
|
||||||
_horizontalScrollController.animateTo(newOffset, curve: cur, duration: dur);
|
|
||||||
},
|
|
||||||
child: Container(
|
child: Container(
|
||||||
width: widget.width,
|
width: widget.width,
|
||||||
height: widget.height ?? 60.0,
|
height: widget.height ?? 60.0,
|
||||||
decoration: _showBorderError ? _errorBoxDecoration : _normalBoxDecoration,
|
decoration: _showBorderError ? errorBoxDecoration : normalBoxDecoration,
|
||||||
child: ClipRect(
|
child: ClipRect(
|
||||||
child: IgnorePointer(
|
child: IgnorePointer(
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
|
@ -365,7 +168,6 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 6.0),
|
padding: const EdgeInsets.symmetric(horizontal: 6.0),
|
||||||
child: CardProviderIcon(
|
child: CardProviderIcon(
|
||||||
cardDetails: _cardDetails,
|
cardDetails: _cardDetails,
|
||||||
size: widget.iconSize,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(
|
SizedBox(
|
||||||
|
@ -375,11 +177,7 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
focusNode: cardNumberFocusNode,
|
focusNode: cardNumberFocusNode,
|
||||||
controller: _cardNumberController,
|
controller: _cardNumberController,
|
||||||
keyboardType: TextInputType.number,
|
keyboardType: TextInputType.number,
|
||||||
style: _isRedText([
|
style: _isRedText([ValidState.invalidCard, ValidState.missingCard, ValidState.blank])
|
||||||
CardDetailsValidState.invalidCard,
|
|
||||||
CardDetailsValidState.missingCard,
|
|
||||||
CardDetailsValidState.blank
|
|
||||||
])
|
|
||||||
? _errorTextStyle
|
? _errorTextStyle
|
||||||
: _normalTextStyle,
|
: _normalTextStyle,
|
||||||
validator: (content) {
|
validator: (content) {
|
||||||
|
@ -387,9 +185,9 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
_cardDetails.cardNumber = content;
|
_cardDetails.cardNumber = content;
|
||||||
if (_cardDetails.validState == CardDetailsValidState.invalidCard) {
|
if (_cardDetails.validState == ValidState.invalidCard) {
|
||||||
_setValidationState('Your card number is invalid.');
|
_setValidationState('You card number is invalid.');
|
||||||
} else if (_cardDetails.validState == CardDetailsValidState.missingCard) {
|
} else if (_cardDetails.validState == ValidState.missingCard) {
|
||||||
_setValidationState('Your card number is incomplete.');
|
_setValidationState('Your card number is incomplete.');
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -404,160 +202,103 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
_currentCardEntryStepController.add(CardEntryStep.exp);
|
_currentCardEntryStepController.add(CardEntryStep.exp);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
onFieldSubmitted: (_) => _currentCardEntryStepController.add(CardEntryStep.exp),
|
|
||||||
inputFormatters: [
|
inputFormatters: [
|
||||||
LengthLimitingTextInputFormatter(19),
|
LengthLimitingTextInputFormatter(19),
|
||||||
FilteringTextInputFormatter.allow(RegExp('[0-9 ]')),
|
FilteringTextInputFormatter.allow(RegExp('[0-9 ]')),
|
||||||
CardNumberInputFormatter(),
|
CardNumberInputFormatter(),
|
||||||
],
|
],
|
||||||
decoration: InputDecoration(
|
decoration: const InputDecoration(
|
||||||
hintText: 'Card number',
|
hintText: 'Card number',
|
||||||
contentPadding: EdgeInsets.zero,
|
|
||||||
hintStyle: _hintTextSyle,
|
|
||||||
fillColor: Colors.transparent,
|
fillColor: Colors.transparent,
|
||||||
border: InputBorder.none,
|
border: InputBorder.none,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
if (isWideFormat)
|
if (_isWideFormat)
|
||||||
Flexible(
|
Flexible(
|
||||||
fit: FlexFit.loose,
|
fit: FlexFit.loose,
|
||||||
// fit: _currentStep == CardEntryStep.number ? FlexFit.loose : FlexFit.tight,
|
// fit: _currentStep == CardEntryStep.number ? FlexFit.loose : FlexFit.tight,
|
||||||
child: AnimatedContainer(
|
child: AnimatedContainer(
|
||||||
curve: Curves.easeInOut,
|
curve: Curves.easeOut,
|
||||||
duration: const Duration(milliseconds: 400),
|
duration: const Duration(milliseconds: 400),
|
||||||
constraints: _currentStep == CardEntryStep.number
|
constraints: _currentStep == CardEntryStep.number
|
||||||
? BoxConstraints.loose(
|
? BoxConstraints.loose(const Size(400.0, 1.0))
|
||||||
Size(_expanderWidthExpanded, 0.0),
|
: BoxConstraints.tight(const Size(0, 0)))),
|
||||||
)
|
|
||||||
: BoxConstraints.tight(
|
|
||||||
Size(_expanderWidthCollapsed, 0.0),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
// Spacer(flex: _currentStep == CardEntryStep.number ? 100 : 1),
|
// Spacer(flex: _currentStep == CardEntryStep.number ? 100 : 1),
|
||||||
SizedBox(
|
AnimatedContainer(
|
||||||
|
duration: const Duration(milliseconds: 125),
|
||||||
width: _expirationFieldWidth,
|
width: _expirationFieldWidth,
|
||||||
child: Stack(
|
child: TextFormField(
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
children: [
|
|
||||||
// Must manually add hint label because they wont show on mobile with backspace hack
|
|
||||||
if (_isMobile && _expirationController.text == '\u200b')
|
|
||||||
Text('MM/YY', style: _hintTextSyle),
|
|
||||||
TextFormField(
|
|
||||||
key: const Key('expiration_field'),
|
key: const Key('expiration_field'),
|
||||||
focusNode: expirationFocusNode,
|
focusNode: expirationFocusNode,
|
||||||
controller: _expirationController,
|
controller: _expirationController,
|
||||||
keyboardType: TextInputType.number,
|
|
||||||
style: _isRedText([
|
style: _isRedText([
|
||||||
CardDetailsValidState.dateTooLate,
|
ValidState.dateTooLate,
|
||||||
CardDetailsValidState.dateTooEarly,
|
ValidState.dateTooEarly,
|
||||||
CardDetailsValidState.missingDate,
|
ValidState.missingDate,
|
||||||
CardDetailsValidState.invalidMonth
|
ValidState.invalidMonth
|
||||||
])
|
])
|
||||||
? _errorTextStyle
|
? _errorTextStyle
|
||||||
: _normalTextStyle,
|
: _normalTextStyle,
|
||||||
validator: (content) {
|
validator: (content) {
|
||||||
if (content == null || content.isEmpty || _isMobile && content == '\u200b') {
|
if (content == null || content.isEmpty) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_isMobile) {
|
|
||||||
setState(() => _cardDetails.expirationString = content.replaceAll('\u200b', ''));
|
|
||||||
} else {
|
|
||||||
setState(() => _cardDetails.expirationString = content);
|
setState(() => _cardDetails.expirationString = content);
|
||||||
}
|
if (_cardDetails.validState == ValidState.dateTooEarly) {
|
||||||
|
|
||||||
if (_cardDetails.validState == CardDetailsValidState.dateTooEarly) {
|
|
||||||
_setValidationState('Your card\'s expiration date is in the past.');
|
_setValidationState('Your card\'s expiration date is in the past.');
|
||||||
} else if (_cardDetails.validState == CardDetailsValidState.dateTooLate) {
|
} else if (_cardDetails.validState == ValidState.dateTooLate) {
|
||||||
_setValidationState('Your card\'s expiration year is invalid.');
|
_setValidationState('Your card\'s expiration year is invalid.');
|
||||||
} else if (_cardDetails.validState == CardDetailsValidState.missingDate) {
|
} else if (_cardDetails.validState == ValidState.missingDate) {
|
||||||
_setValidationState('You must include your card\'s expiration date.');
|
_setValidationState('You must include your card\'s expiration date.');
|
||||||
} else if (_cardDetails.validState == CardDetailsValidState.invalidMonth) {
|
} else if (_cardDetails.validState == ValidState.invalidMonth) {
|
||||||
_setValidationState('Your card\'s expiration month is invalid.');
|
_setValidationState('Invalid expiration month.');
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
onChanged: (str) {
|
onChanged: (str) {
|
||||||
if (_isMobile) {
|
|
||||||
if (str.isEmpty) {
|
|
||||||
_backspacePressed();
|
|
||||||
}
|
|
||||||
setState(() => _cardDetails.expirationString = str.replaceAll('\u200b', ''));
|
|
||||||
} else {
|
|
||||||
setState(() => _cardDetails.expirationString = str);
|
setState(() => _cardDetails.expirationString = str);
|
||||||
}
|
|
||||||
if (str.length == 5) {
|
if (str.length == 5) {
|
||||||
_currentCardEntryStepController.add(CardEntryStep.cvc);
|
_currentCardEntryStepController.add(CardEntryStep.cvc);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
onFieldSubmitted: (_) => _currentCardEntryStepController.add(CardEntryStep.cvc),
|
|
||||||
inputFormatters: [
|
inputFormatters: [
|
||||||
LengthLimitingTextInputFormatter(5),
|
LengthLimitingTextInputFormatter(5),
|
||||||
FilteringTextInputFormatter.allow(RegExp('[0-9/]')),
|
FilteringTextInputFormatter.allow(RegExp('[0-9/]')),
|
||||||
CardExpirationFormatter(),
|
CardExpirationFormatter(),
|
||||||
],
|
],
|
||||||
decoration: InputDecoration(
|
decoration: const InputDecoration(
|
||||||
contentPadding: EdgeInsets.zero,
|
hintText: 'MM/YY',
|
||||||
hintText: _isMobile ? '' : 'MM/YY',
|
|
||||||
hintStyle: _hintTextSyle,
|
|
||||||
fillColor: Colors.transparent,
|
fillColor: Colors.transparent,
|
||||||
border: InputBorder.none,
|
border: InputBorder.none,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
|
||||||
),
|
),
|
||||||
),
|
AnimatedContainer(
|
||||||
SizedBox(
|
duration: const Duration(milliseconds: 250),
|
||||||
width: _securityFieldWidth,
|
width: _securityFieldWidth,
|
||||||
child: Stack(
|
child: TextFormField(
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
children: [
|
|
||||||
if (_isMobile && _securityCodeController.text == '\u200b')
|
|
||||||
Text(
|
|
||||||
'CVC',
|
|
||||||
style: _hintTextSyle,
|
|
||||||
),
|
|
||||||
TextFormField(
|
|
||||||
key: const Key('security_field'),
|
key: const Key('security_field'),
|
||||||
focusNode: securityCodeFocusNode,
|
focusNode: securityCodeFocusNode,
|
||||||
controller: _securityCodeController,
|
controller: _securityCodeController,
|
||||||
keyboardType: TextInputType.number,
|
style: _isRedText([ValidState.invalidCVC, ValidState.missingCVC])
|
||||||
style:
|
|
||||||
_isRedText([CardDetailsValidState.invalidCVC, CardDetailsValidState.missingCVC])
|
|
||||||
? _errorTextStyle
|
? _errorTextStyle
|
||||||
: _normalTextStyle,
|
: _normalTextStyle,
|
||||||
validator: (content) {
|
validator: (content) {
|
||||||
if (content == null || content.isEmpty || _isMobile && content == '\u200b') {
|
if (content == null || content.isEmpty) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
setState(() => _cardDetails.securityCode = int.tryParse(content));
|
||||||
if (_isMobile) {
|
if (_cardDetails.validState == ValidState.invalidCVC) {
|
||||||
setState(() => _cardDetails.securityCode = content.replaceAll('\u200b', ''));
|
|
||||||
} else {
|
|
||||||
setState(() => _cardDetails.securityCode = content);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_cardDetails.validState == CardDetailsValidState.invalidCVC) {
|
|
||||||
_setValidationState('Your card\'s security code is invalid.');
|
_setValidationState('Your card\'s security code is invalid.');
|
||||||
} else if (_cardDetails.validState == CardDetailsValidState.missingCVC) {
|
} else if (_cardDetails.validState == ValidState.missingCVC) {
|
||||||
_setValidationState('Your card\'s security code is incomplete.');
|
_setValidationState('You card\'s security code is incomplete.');
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
onFieldSubmitted: (_) => _currentCardEntryStepController.add(CardEntryStep.postal),
|
|
||||||
onChanged: (str) {
|
onChanged: (str) {
|
||||||
if (_isMobile) {
|
|
||||||
if (str.isEmpty) {
|
|
||||||
_backspacePressed();
|
|
||||||
}
|
|
||||||
setState(() => _cardDetails.expirationString = str.replaceAll('\u200b', ''));
|
|
||||||
} else {
|
|
||||||
setState(() => _cardDetails.expirationString = str);
|
setState(() => _cardDetails.expirationString = str);
|
||||||
}
|
|
||||||
|
|
||||||
if (str.length == _cardDetails.provider?.cvcLength) {
|
if (str.length == _cardDetails.provider?.cvcLength) {
|
||||||
_currentCardEntryStepController.add(CardEntryStep.postal);
|
_currentCardEntryStepController.add(CardEntryStep.postal);
|
||||||
}
|
}
|
||||||
|
@ -567,83 +308,49 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
_cardDetails.provider == null ? 4 : _cardDetails.provider!.cvcLength),
|
_cardDetails.provider == null ? 4 : _cardDetails.provider!.cvcLength),
|
||||||
FilteringTextInputFormatter.allow(RegExp('[0-9]')),
|
FilteringTextInputFormatter.allow(RegExp('[0-9]')),
|
||||||
],
|
],
|
||||||
decoration: InputDecoration(
|
decoration: const InputDecoration(
|
||||||
contentPadding: EdgeInsets.zero,
|
hintText: 'CVC',
|
||||||
hintText: _isMobile ? '' : 'CVC',
|
|
||||||
hintStyle: _hintTextSyle,
|
|
||||||
fillColor: Colors.transparent,
|
fillColor: Colors.transparent,
|
||||||
border: InputBorder.none,
|
border: InputBorder.none,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
|
||||||
),
|
),
|
||||||
),
|
AnimatedContainer(
|
||||||
SizedBox(
|
duration: const Duration(milliseconds: 250),
|
||||||
width: _postalFieldWidth,
|
width: _postalFieldWidth,
|
||||||
child: Stack(
|
child: TextFormField(
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
children: [
|
|
||||||
if (_isMobile && _postalCodeController.text == '\u200b')
|
|
||||||
Text(
|
|
||||||
'Postal Code',
|
|
||||||
style: _hintTextSyle,
|
|
||||||
),
|
|
||||||
TextFormField(
|
|
||||||
key: const Key('postal_field'),
|
key: const Key('postal_field'),
|
||||||
focusNode: postalCodeFocusNode,
|
focusNode: postalCodeFocusNode,
|
||||||
controller: _postalCodeController,
|
controller: _postalCodeController,
|
||||||
keyboardType: TextInputType.number,
|
style: _isRedText([ValidState.invalidZip, ValidState.missingZip])
|
||||||
style:
|
|
||||||
_isRedText([CardDetailsValidState.invalidZip, CardDetailsValidState.missingZip])
|
|
||||||
? _errorTextStyle
|
? _errorTextStyle
|
||||||
: _normalTextStyle,
|
: _normalTextStyle,
|
||||||
validator: (content) {
|
validator: (content) {
|
||||||
if (content == null || content.isEmpty || _isMobile && content == '\u200b') {
|
if (content == null || content.isEmpty) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_isMobile) {
|
|
||||||
setState(() => _cardDetails.postalCode = content.replaceAll('\u200b', ''));
|
|
||||||
} else {
|
|
||||||
setState(() => _cardDetails.postalCode = content);
|
setState(() => _cardDetails.postalCode = content);
|
||||||
}
|
|
||||||
|
|
||||||
if (_cardDetails.validState == CardDetailsValidState.invalidZip) {
|
if (_cardDetails.validState == ValidState.invalidZip) {
|
||||||
_setValidationState('The postal code you entered is not correct.');
|
_setValidationState('The postal code you entered is not correct.');
|
||||||
} else if (_cardDetails.validState == CardDetailsValidState.missingZip) {
|
} else if (_cardDetails.validState == ValidState.missingZip) {
|
||||||
_setValidationState('You must enter your card\'s postal code.');
|
_setValidationState('You must enter your card\'s postal code.');
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
onChanged: (str) {
|
onChanged: (str) {
|
||||||
if (_isMobile) {
|
|
||||||
if (str.isEmpty) {
|
|
||||||
_backspacePressed();
|
|
||||||
}
|
|
||||||
setState(() => _cardDetails.postalCode = str.replaceAll('\u200b', ''));
|
|
||||||
} else {
|
|
||||||
setState(() => _cardDetails.postalCode = str);
|
setState(() => _cardDetails.postalCode = str);
|
||||||
}
|
|
||||||
},
|
},
|
||||||
textInputAction: TextInputAction.done,
|
|
||||||
onFieldSubmitted: (_) {
|
onFieldSubmitted: (_) {
|
||||||
_postalFieldSubmitted();
|
_validateFields();
|
||||||
|
widget.onCardDetailsComplete(_cardDetails);
|
||||||
},
|
},
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
contentPadding: EdgeInsets.zero,
|
hintText: _currentStep == CardEntryStep.number ? '' : 'Postal Code',
|
||||||
hintText: _isMobile ? '' : 'Postal Code',
|
|
||||||
hintStyle: _hintTextSyle,
|
|
||||||
fillColor: Colors.transparent,
|
fillColor: Colors.transparent,
|
||||||
border: InputBorder.none,
|
border: InputBorder.none,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
AnimatedOpacity(
|
|
||||||
duration: const Duration(milliseconds: 300),
|
|
||||||
opacity: _loading && widget.showInternalLoadingWidget ? 1.0 : 0.0,
|
|
||||||
child: widget.loadingWidget ?? const CircularProgressIndicator(),
|
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
@ -660,8 +367,7 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.only(top: 8.0, left: 14.0),
|
padding: const EdgeInsets.only(top: 8.0, left: 14.0),
|
||||||
child: Text(
|
child: Text(
|
||||||
// Spacing changes by like a pixel if its an empty string, slight jitter when error appears and disappears
|
_validationErrorText ?? '',
|
||||||
_validationErrorText ?? ' ',
|
|
||||||
style: const TextStyle(color: Colors.red),
|
style: const TextStyle(color: Colors.red),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
@ -670,47 +376,9 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _postalFieldSubmitted() async {
|
|
||||||
_validateFields();
|
|
||||||
if (_cardDetails.isComplete) {
|
|
||||||
if (widget.onCardDetailsComplete != null) {
|
|
||||||
widget.onCardDetailsComplete!(_cardDetails);
|
|
||||||
} else if (widget.onStripeResponse != null) {
|
|
||||||
bool returned = false;
|
|
||||||
|
|
||||||
Future.delayed(
|
|
||||||
const Duration(milliseconds: 750),
|
|
||||||
() => returned ? null : setState(() => _loading = true),
|
|
||||||
);
|
|
||||||
|
|
||||||
const stripeCardUrl = 'https://api.stripe.com/v1/tokens';
|
|
||||||
// Callback that stripe call is being made
|
|
||||||
if (widget.onCallToStripe != null) widget.onCallToStripe!();
|
|
||||||
final response = await http.post(
|
|
||||||
Uri.parse(stripeCardUrl),
|
|
||||||
body: {
|
|
||||||
'card[number]': _cardDetails.cardNumber,
|
|
||||||
'card[cvc]': _cardDetails.securityCode,
|
|
||||||
'card[exp_month]': _cardDetails.expMonth,
|
|
||||||
'card[exp_year]': _cardDetails.expYear,
|
|
||||||
'card[address_zip]': _cardDetails.postalCode,
|
|
||||||
'key': widget.stripePublishableKey,
|
|
||||||
},
|
|
||||||
headers: {"Content-Type": "application/x-www-form-urlencoded"},
|
|
||||||
);
|
|
||||||
|
|
||||||
returned = true;
|
|
||||||
final jsonBody = jsonDecode(response.body);
|
|
||||||
|
|
||||||
widget.onStripeResponse!(jsonBody);
|
|
||||||
if (_loading) setState(() => _loading = false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Provided a list of `ValidState`, returns whether
|
/// Provided a list of `ValidState`, returns whether
|
||||||
/// make the text field red
|
/// make the text field red
|
||||||
bool _isRedText(List<CardDetailsValidState> args) {
|
bool _isRedText(List<ValidState> args) {
|
||||||
return _showBorderError && args.contains(_cardDetails.validState);
|
return _showBorderError && args.contains(_cardDetails.validState);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -727,17 +395,11 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
/// the validation state
|
/// the validation state
|
||||||
void _validateFields() {
|
void _validateFields() {
|
||||||
_validationErrorText = null;
|
_validationErrorText = null;
|
||||||
if (widget.overrideValidState != null) {
|
|
||||||
_cardDetails.overrideValidState = widget.overrideValidState!;
|
|
||||||
_setValidationState(widget.errorText);
|
|
||||||
} else {
|
|
||||||
_formFieldKey.currentState!.validate();
|
_formFieldKey.currentState!.validate();
|
||||||
|
|
||||||
// Clear up validation state if everything is valid
|
// Clear up validation state if everything is valid
|
||||||
if (_validationErrorText == null) {
|
if (_validationErrorText == null) {
|
||||||
_setValidationState(null);
|
_setValidationState(null);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -790,26 +452,9 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
postalCodeFocusNode.requestFocus();
|
postalCodeFocusNode.requestFocus();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (!isWideFormat) {
|
if (!_isWideFormat) {
|
||||||
_scrollRow(step);
|
_scrollRow(step);
|
||||||
}
|
}
|
||||||
// If mobile, and keyboard is closed, unfocus, to allow refocus
|
|
||||||
// print(MediaQuery.of(context).viewInsets.bottom);
|
|
||||||
// if (_isMobile && _hasFocus() && MediaQuery.of(context).viewInsets.bottom == 0.0) {
|
|
||||||
// cardNumberFocusNode.unfocus();
|
|
||||||
// expirationFocusNode.unfocus();
|
|
||||||
// securityCodeFocusNode.unfocus();
|
|
||||||
// postalCodeFocusNode.unfocus();
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns true if any field in the `CardTextField` has focus.
|
|
||||||
// ignore: unused_element
|
|
||||||
bool _hasFocus() {
|
|
||||||
return cardNumberFocusNode.hasFocus ||
|
|
||||||
expirationFocusNode.hasFocus ||
|
|
||||||
securityCodeFocusNode.hasFocus ||
|
|
||||||
postalCodeFocusNode.hasFocus;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Function that is listening to the keyboard events.
|
/// Function that is listening to the keyboard events.
|
||||||
|
@ -825,49 +470,24 @@ class CardTextFieldState extends State<CardTextField> {
|
||||||
case CardEntryStep.number:
|
case CardEntryStep.number:
|
||||||
break;
|
break;
|
||||||
case CardEntryStep.exp:
|
case CardEntryStep.exp:
|
||||||
if (_expirationController.text.isNotEmpty) break;
|
final expStr = _expirationController.text;
|
||||||
case CardEntryStep.cvc:
|
if (expStr.isNotEmpty) break;
|
||||||
if (_securityCodeController.text.isNotEmpty) break;
|
|
||||||
case CardEntryStep.postal:
|
|
||||||
if (_postalCodeController.text.isNotEmpty) break;
|
|
||||||
}
|
|
||||||
_transitionStepFocus();
|
|
||||||
}
|
|
||||||
|
|
||||||
void _backspacePressed() {
|
|
||||||
// Put the empty char back into the controller
|
|
||||||
switch (_currentStep) {
|
|
||||||
case CardEntryStep.number:
|
|
||||||
break;
|
|
||||||
case CardEntryStep.exp:
|
|
||||||
_expirationController.text = '\u200b';
|
|
||||||
case CardEntryStep.cvc:
|
|
||||||
_securityCodeController.text = '\u200b';
|
|
||||||
case CardEntryStep.postal:
|
|
||||||
_postalCodeController.text = '\u200b';
|
|
||||||
}
|
|
||||||
_transitionStepFocus();
|
|
||||||
}
|
|
||||||
|
|
||||||
void _transitionStepFocus() {
|
|
||||||
switch (_currentStep) {
|
|
||||||
case CardEntryStep.number:
|
|
||||||
break;
|
|
||||||
case CardEntryStep.exp:
|
|
||||||
_currentCardEntryStepController.add(CardEntryStep.number);
|
_currentCardEntryStepController.add(CardEntryStep.number);
|
||||||
String numStr = _cardNumberController.text;
|
String numStr = _cardNumberController.text;
|
||||||
_cardNumberController.text = numStr.substring(0, numStr.length - 1);
|
_cardNumberController.text = numStr.substring(0, numStr.length - 1);
|
||||||
break;
|
break;
|
||||||
case CardEntryStep.cvc:
|
case CardEntryStep.cvc:
|
||||||
|
final cvcStr = _securityCodeController.text;
|
||||||
|
if (cvcStr.isNotEmpty) break;
|
||||||
_currentCardEntryStepController.add(CardEntryStep.exp);
|
_currentCardEntryStepController.add(CardEntryStep.exp);
|
||||||
final expStr = _expirationController.text;
|
final expStr = _expirationController.text;
|
||||||
_expirationController.text = expStr.substring(0, expStr.length - 1);
|
_expirationController.text = expStr.substring(0, expStr.length - 1);
|
||||||
break;
|
|
||||||
case CardEntryStep.postal:
|
case CardEntryStep.postal:
|
||||||
|
final String postalStr = _postalCodeController.text;
|
||||||
|
if (postalStr.isNotEmpty) break;
|
||||||
_currentCardEntryStepController.add(CardEntryStep.cvc);
|
_currentCardEntryStepController.add(CardEntryStep.cvc);
|
||||||
final String cvcStr = _securityCodeController.text;
|
final String cvcStr = _securityCodeController.text;
|
||||||
_securityCodeController.text = cvcStr.substring(0, cvcStr.length - 1);
|
_securityCodeController.text = cvcStr.substring(0, cvcStr.length - 1);
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
name: stripe_native_card_field
|
name: stripe_native_card_field
|
||||||
description: A native flutter implementation of the elegant Stripe Card Field.
|
description: A native flutter implementation of the elegant Stripe Card Field.
|
||||||
version: 0.0.3
|
version: 0.0.2
|
||||||
repository: https://git.fosscat.com/n8r/stripe_native_card_field
|
repository: https://git.fosscat.com/n8r/stripe_native_card_field
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
|
@ -11,7 +11,6 @@ dependencies:
|
||||||
flutter:
|
flutter:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
flutter_svg: ^2.0.9
|
flutter_svg: ^2.0.9
|
||||||
http: ^1.1.0
|
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
|
|
|
@ -20,20 +20,20 @@ void main() {
|
||||||
width: width,
|
width: width,
|
||||||
onCardDetailsComplete: (cd) => details = cd,
|
onCardDetailsComplete: (cd) => details = cd,
|
||||||
);
|
);
|
||||||
await tester.pumpWidget(baseCardFieldWidget(cardField));
|
await tester.pumpWidget(cardFieldWidget(cardField));
|
||||||
|
|
||||||
final input = TestTextInput();
|
final input = TestTextInput();
|
||||||
|
|
||||||
final cardState = tester.state(find.byType(CardTextField)) as CardTextFieldState;
|
final cardState = tester.state(find.byType(CardTextField)) as CardTextFieldState;
|
||||||
|
|
||||||
assertEmptyTextFields(tester, cardState.isWideFormat);
|
assertEmptyTextFields(tester, width);
|
||||||
|
|
||||||
await tester.tap(find.byType(CardTextField));
|
await tester.tap(find.byType(CardTextField));
|
||||||
expect(cardState.cardNumberFocusNode.hasFocus, true);
|
expect(cardState.cardNumberFocusNode.hasFocus, true);
|
||||||
|
|
||||||
// await enterTextByKey(tester, key: cardFieldKey, text: '4242424242424242');
|
// await enterTextByKey(tester, key: cardFieldKey, text: '4242424242424242');
|
||||||
input.enterText("4242424242424242");
|
input.enterText("4242424242424242");
|
||||||
await tester.pump();
|
await tester.pumpAndSettle();
|
||||||
|
|
||||||
expect(cardState.cardNumberFocusNode.hasFocus, false);
|
expect(cardState.cardNumberFocusNode.hasFocus, false);
|
||||||
expect(cardState.expirationFocusNode.hasFocus, true);
|
expect(cardState.expirationFocusNode.hasFocus, true);
|
||||||
|
@ -44,19 +44,18 @@ void main() {
|
||||||
// Backspace should move focus back to card number
|
// Backspace should move focus back to card number
|
||||||
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.backspace);
|
await tester.sendKeyDownEvent(LogicalKeyboardKey.backspace);
|
||||||
await tester.pump();
|
await tester.pumpAndSettle();
|
||||||
|
|
||||||
expect(getTextFormField(expirationFieldKey).controller?.text, '');
|
expect(getTextFormField(expirationFieldKey).controller?.text, '');
|
||||||
expect(getTextFormField(cardFieldKey).controller?.text, '4242 4242 4242 424');
|
expect(getTextFormField(cardFieldKey).controller?.text, '4242 4242 4242 424');
|
||||||
expect(cardState.cardNumberFocusNode.hasFocus, true);
|
expect(cardState.cardNumberFocusNode.hasFocus, true);
|
||||||
expect(cardState.expirationFocusNode.hasFocus, false);
|
expect(cardState.expirationFocusNode.hasFocus, false);
|
||||||
// Postal code should now be gone
|
// Postal code should now be gone
|
||||||
// FIXME this doesnt work
|
expect(find.text("Postal Code"), findsNothing);
|
||||||
// expect(find.text("Postal Code"), findsNothing);
|
|
||||||
|
|
||||||
// When using TestTextInput, any enterText() clears what is currently in focused field
|
// When using TestTextInput, any enterText() clears what is currently in focused field
|
||||||
input.enterText("4242424242424242");
|
input.enterText("4242424242424242");
|
||||||
await tester.pump();
|
await tester.pumpAndSettle();
|
||||||
|
|
||||||
expect(getTextFormField(cardFieldKey).controller?.text, '4242 4242 4242 4242');
|
expect(getTextFormField(cardFieldKey).controller?.text, '4242 4242 4242 4242');
|
||||||
expect(cardState.cardNumberFocusNode.hasFocus, false);
|
expect(cardState.cardNumberFocusNode.hasFocus, false);
|
||||||
|
@ -89,106 +88,13 @@ void main() {
|
||||||
await input.receiveAction(TextInputAction.done);
|
await input.receiveAction(TextInputAction.done);
|
||||||
await tester.pump();
|
await tester.pump();
|
||||||
|
|
||||||
final expectedCardDetails = CardDetails(
|
final expectedCardDetails = CardDetails(cardNumber: '4242 4242 4242 4242', securityCode: '333', expirationString: '10/28', postalCode: '91555');
|
||||||
cardNumber: '4242 4242 4242 4242', securityCode: '333', expirationString: '10/28', postalCode: '91555');
|
|
||||||
// print('${expectedCardDetails.toString()}\n${details?.toString()}');
|
|
||||||
expect(details?.hash, expectedCardDetails.hash);
|
expect(details?.hash, expectedCardDetails.hash);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
testWidgets(
|
|
||||||
'CardTextField: GIVEN the user enters invalid input WHEN each text field is filled THEN the correct error messages are displayed',
|
|
||||||
(tester) async {
|
|
||||||
const width = 500.0;
|
|
||||||
CardDetails? details;
|
|
||||||
|
|
||||||
final cardField = CardTextField(
|
|
||||||
width: width,
|
|
||||||
onCardDetailsComplete: (cd) => details = cd,
|
|
||||||
);
|
|
||||||
await tester.pumpWidget(baseCardFieldWidget(cardField));
|
|
||||||
|
|
||||||
final input = TestTextInput();
|
|
||||||
|
|
||||||
final cardState = tester.state(find.byType(CardTextField)) as CardTextFieldState;
|
|
||||||
|
|
||||||
assertEmptyTextFields(tester, cardState.isWideFormat);
|
|
||||||
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.tab);
|
|
||||||
expect(cardState.cardNumberFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('4242424242424222');
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(find.text('Your card number is invalid.'), findsOneWidget);
|
|
||||||
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.backspace);
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(cardState.cardNumberFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('4242424242424242');
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(cardState.expirationFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('0055');
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(find.text("Your card's expiration month is invalid."), findsOneWidget);
|
|
||||||
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.backspace);
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(cardState.expirationFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('1099');
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(find.text("Your card's expiration year is invalid."), findsOneWidget);
|
|
||||||
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.backspace);
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(cardState.expirationFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('0228');
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(cardState.securityCodeFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
// FIXME this isnt transitioning focus correctly in test
|
|
||||||
input.enterText('123');
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.tab);
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(cardState.postalCodeFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('1234');
|
|
||||||
// Pressing enter doesnt work here...
|
|
||||||
await input.receiveAction(TextInputAction.done);
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
expect(find.text("The postal code you entered is not correct."), findsOneWidget);
|
|
||||||
|
|
||||||
await tester.tap(find.byType(CardTextField));
|
|
||||||
|
|
||||||
// Tab from security field to get zipcode focus
|
|
||||||
await tester.sendKeyDownEvent(LogicalKeyboardKey.tab);
|
|
||||||
expect(cardState.postalCodeFocusNode.hasFocus, true);
|
|
||||||
|
|
||||||
input.enterText('12345');
|
|
||||||
await input.receiveAction(TextInputAction.done);
|
|
||||||
await tester.pump();
|
|
||||||
|
|
||||||
final expectedCardDetails = CardDetails(
|
|
||||||
cardNumber: '4242 4242 4242 4242', expirationString: '02/28', securityCode: '123', postalCode: '12345');
|
|
||||||
|
|
||||||
expect(details?.hash, expectedCardDetails.hash);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget baseCardFieldWidget(CardTextField cardField) => MaterialApp(
|
Widget cardFieldWidget(CardTextField cardField) => MaterialApp(
|
||||||
home: Scaffold(
|
home: Scaffold(
|
||||||
body: Center(
|
body: Center(
|
||||||
child: Column(
|
child: Column(
|
||||||
|
@ -200,13 +106,11 @@ Widget baseCardFieldWidget(CardTextField cardField) => MaterialApp(
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
void assertEmptyTextFields(WidgetTester tester, bool isWideFormat) {
|
void assertEmptyTextFields(WidgetTester tester, double width) {
|
||||||
if (isWideFormat) {
|
|
||||||
expect(find.text("Card number"), findsOneWidget);
|
expect(find.text("Card number"), findsOneWidget);
|
||||||
expect(find.text("MM/YY"), findsOneWidget);
|
expect(find.text("MM/YY"), findsOneWidget);
|
||||||
expect(find.text("CVC"), findsOneWidget);
|
expect(find.text("CVC"), findsOneWidget);
|
||||||
}
|
expect(find.text("Postal Code"), findsNothing);
|
||||||
// expect(find.text("Postal Code"), findsNothing);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> enterTextByKey(WidgetTester tester, {required String key, required String text}) async {
|
Future<void> enterTextByKey(WidgetTester tester, {required String key, required String text}) async {
|
||||||
|
|
Loading…
Reference in New Issue
Block a user