Status Update
Comments
da...@google.com <da...@google.com> #2
Please include a sample project that reproduces your issue.
ap...@google.com <ap...@google.com> #3
Hello again!
It was very tough but trying to replicate the issue in a sample project (since original project I cannot share it) I finally found the source of the issue and it's related to the length of the names of the objects used when generating code by Navigation component and line wrapping in those files
Please, try to compile project that I've attached and you will see the error. So no problem with inheritance or other possible causes that I commented in first message.
In project example, I've created:
MyNewDialogWithAVeryLongNameToTriggerAnError
that inhertis fromDialogFragment
Then I've added it to in nav_graph_test.xml
and with arguments that have also long IDs too:
<dialog
android:id="@+id/very_large_name_for_my_new_dialog_widget_for_trigger_an_error"
android:name="com.example.navtestapp.MyNewDialogWithAVeryLongNameToTriggerAnError"
tools:layout="@layout/fragment_my_dialog">
<argument
android:name="very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error"
app:argType="string"
app:nullable="true"/>
<argument
android:name="second_very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error"
app:argType="string"
app:nullable="true"/>
<action
android:id="@+id/action_dialog"
app:destination="@id/my_fragment" />
</dialog>
Then, when we try to compile project, we can't. It returns the error comented in first message:
This function must return a value of type
MyNewDialogWithAVeryLongNameToTriggerAnErrorArgs
If we look at generated code we can see:
data class MyNewDialogWithAVeryLongNameToTriggerAnErrorArgs(
val veryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError: String?,
val secondVeryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError: String?
) : NavArgs {
fun toBundle(): Bundle {
val result = Bundle()
result.putString("very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error",
this.veryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError)
result.putString("second_very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error",
this.secondVeryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError)
return result
}
companion object {
@JvmStatic
fun fromBundle(bundle: Bundle): MyNewDialogWithAVeryLongNameToTriggerAnErrorArgs {
bundle.setClassLoader(MyNewDialogWithAVeryLongNameToTriggerAnErrorArgs::class.java.classLoader)
val __veryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError : String?
if
(bundle.containsKey("very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error")) {
__veryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError =
bundle.getString("very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error")
} else {
throw IllegalArgumentException("Required argument \"very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error\" is missing and does not have an android:defaultValue")
}
val __secondVeryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError : String?
if
(bundle.containsKey("second_very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error")) {
__secondVeryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError =
bundle.getString("second_very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error")
} else {
throw IllegalArgumentException("Required argument \"second_very_large_name_for argument_in_my_new_dialog_widget_for_trigger_an_error\" is missing and does not have an android:defaultValue")
}
return
MyNewDialogWithAVeryLongNameToTriggerAnErrorArgs(__veryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError,
__secondVeryLargeNameForArgumentInMyNewDialogWidgetForTriggerAnError)
}
}
}
Pay attention to return
in fromBundle()
function that is split into two lines, preventing project to compile since compiler expects an expresion for returning it but it's in the next line. Removing breakline between return
and MyNewDialogWithAVeryLongNameToTriggerAnErrorArgs
fixes the problem, but since this is generated code, we are not able to do it.
Shouldn't navigation component take into consideration that objects could have long names and ensure that code generated finally is able to compile it?
Thank you!
da...@google.com <da...@google.com> #4
Thanks, your reproduction project and additional analysis is super helpful.
Safe Args uses a project called
We'll look at if this is still reproducible with the latest version of KotlinPoet. If upgrading fixes it, we'll include that upgrade in Navigation 2.3.1.
Otherwise, we'll file an issue against KotlinPoet and have to wait for a release that includes a fix.
an...@google.com <an...@google.com> #5
Branch: androidx-master-dev
commit 766a3ad47da73e1179ce6419693a825c4d665cfd
Author: Jeremy Woods <jbwoods@google.com>
Date: Wed Sep 23 14:44:52 2020
Keep long arg names from wrapping
When using safe-args with a long argument name, kotlin does an
automatic line wrap at 100 characters which makes it place the argument
return and variable on different lines, which fails.
We should add `ยท` to emit a space that never wraps.
Test: Added KotlinNavWriterTest
Bug: 168584987
Change-Id: Ibc31f91fd6b1c4a94f0b7c05b9b9679ffa09b625
M navigation/navigation-safe-args-generator/src/main/kotlin/androidx/navigation/safe/args/generator/kotlin/KotlinNavWriter.kt
M navigation/navigation-safe-args-generator/src/test/kotlin/androidx/navigation/safe/args/generator/KotlinNavWriterTest.kt
A navigation/navigation-safe-args-generator/src/test/test-data/expected/kotlin_nav_writer_test/ReallyReallyReallyReallyReallyReallyReallyReallyReallyReallyReallyReallyReallyReallyLongNameMainFragmentArgs.kt
Description
Version used: 2.2.0
Devices/Android versions reproduced on:
- device usb:336662528X product:xcover4ltexx model:SM_G390F device:xcover4lte transport_id:5
- device product:sdk_gphone_x86 model:Android_SDK_built_for_x86 device:generic_x86 transport_id:4
- device usb:336662528X product:zerofltexx model:SM_G920F device:zeroflte transport_id:15
In Version 2.1.0 annotation @Relation was only allowed for list types, if you used it on a non-list type the following runtime error occured:
error: Fields annotated with @Relation must be a List or Set.
Since Version 2.2.0 @Relation can be used on non-list types, in order to model relations with a single object reference.
This is handy if you have 1-to-1 relations and you don't have to explicitly get the first item from the result list, instead you get the object directly.
release notes:
One-to-One Relations: The restriction in POJO fields annotated with @Relation to be of type List or Set has been lifted, effectively allowing single-value relations to be represented.
It seems that this new feature is only supported up to a certain amount of table rows. In more details it means that if you run a query on tables with 100 rows, the returned POJOs have all valid non-null references to the object specified by the @Relation annotation. If you run the same query on tables with more than 999 rows (999 still works, 1000 will fail) the result will be that all returned POJOs have NULL as referenced object (see my example project).
If you use the list type with @Relation annotation you don't have this limitation.
Please refer to the attached example project unit test.
In the example project the same entities are used but with two different POJO types. One is using a List<> type annotated with @Relation, and the other is using the newly supported object type.
Android unit test show that the list type always works but the object type works up to 999 rows, after that it starts to fail.