- 
                Notifications
    You must be signed in to change notification settings 
- Fork 76
          Provide filter overload to replace colGroups(predicate<ColumnGroup>)
          #1359
        
          New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Conversation
cbf5652    to
    c32a2ef      
    Compare
  
    I prefer composition to inheritance here because changing ColumnWithPath hierarchy seems difficult, a lot of code assumes only one possible implementation. So for the sake of only one `filter` implementation, let's introduce a simple new class
c32a2ef    to
    a0a3f44      
    Compare
  
    | */ | ||
| @Suppress("INAPPLICABLE_JVM_NAME") | ||
| @JvmName("filterColumnGroups") | ||
| public fun ColumnSet<AnyRow>.filter(predicate: Predicate<ColumnGroupWithPath<*>>): ColumnSet<*> = | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this intermediate type is not needed, right?
colGroups {} takes a Predicate<ColumnGroup<*>>; we can duplicate that here:
fun ColumnSet<AnyRow>.filter(predicate: Predicate<ColumnGroup<*>>): ColumnSet<*> =
    colsInternal { columnWithPath ->
        columnWithPath.isColumnGroup() && predicate(columnWithPath)
    }There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
.isColumnGroup() does an instance check and auto-cast so is safe :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes but it won't hurt to have path property, same as ColumnSet<*>.filter does
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
now I understand what you're trying to achieve :)
Still, we'll have to take a slightly further look, because I see we have the ColumnGroupWithPathImpl class. Adding a completely separate ColumnGroupWithPath would be confusing at best
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As alternative we could rename and expose ColumnGroupWithPathImpl, but
- ColumnWithPath is used extensively across whole column selection and ColumnGroupWithPathImpl is important implementation detail
- ColumnGroupWithPath is needed only for one filter overload, and is very simple itself - it's only a class with 2 properties.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What if we don't introduce a new class and just create:
@Suppress("UNCHECKED_CAST")
@get:JvmName("columnAsColumnGroup")
public val <T> ColumnWithPath<DataRow<T>>.column: ColumnGroup<T>
    get() = data as ColumnGroup<T>
@get:JvmName("columnAsFrameColumn")
public val <T> ColumnWithPath<DataFrame<T>>.column: FrameColumn<T>
    get() = data as FrameColumn<T>
public val <T> ColumnWithPath<T>.column: DataColumn<T>
    get() = dataI think the name "column" is easier to discover in the ColumnWithPath class than "data".
We could even try to find a way to hide data in ColumnWithPath, but that probably even won't be necessary.
(btw I tried calling them "data" with @kotlin.internal.HidesMembers, but unfortunately that doesn't seem to work :( )
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good idea, i'll try
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In CS DSL scope for DataColumn<DataRow<*>> is very polluted though =( There are functions named column. IMO totally impossible to find anything just from looking at completion.


But if person knows what they're looking for, they can use asColumnGroup even without column.
For comparison completion for ColumnGroupWithPath looks clearer:

I tried to limit visibility of functions from outer scope with DslMarker, but i'm afraid everything is still visible despite Dsl Scope violation errors.
So i'd go with ColumnGroupWithPath for better discoverability
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's a good point. Just try to be as transparant as possible in KDoc if you decide to go this route. I want to prevent confusion between ColumnGroupWithPathImp and ColumnGroupWithPath, especially if they are separate things.
Also, how's the situation for FrameColumn?
Before this PR this was possible:
df.select { colGroups { it.containsColumn("name") } }but this wasn't:
df.select { colGroups().filter { it.containsColumn("name") }Because DataColumn<DataRow> is not a ColumnGroup =( So the cast was needed.
With a little compromise now following will be possible:
df.select { colGroups().filter { it.data.containsColumn("name") }I decided against making complex inheritance hierarchy, so unlike with normal filter, our new ColumnGroupWithPath is not a ColumnGroup/DataColumn itself. It only contains a data: ColumnGroup, same as ColumnWithPath. Inheritance required deep and imo too complex changes!