Skip to content
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

Feature Idea: dot syntax for function calls (and code completion ) #37993

Closed
pannous opened this issue Oct 12, 2020 · 7 comments
Closed

Feature Idea: dot syntax for function calls (and code completion ) #37993

pannous opened this issue Oct 12, 2020 · 7 comments

Comments

@pannous
Copy link

pannous commented Oct 12, 2020

One limitation of the functional approach is discoverability: which functions can be applied to a given type? This is not necessary however: We (should) know all the functions which take an object of given type as first argument. This leads to the possibility of code completion:

abstract type Animal end
struct Cat <: Animal end
struct Fish <: Animal end
swims(Cat)=false 
swims(Fish)=true
pussy=Cat()
pussy.swims() # <--- NEW syntax, same as swims(pussy), but discoverable via <tab>

The important point is that it doesn't change the fundamentals of the language at all**, it's just syntactic sugar with the huge benefit of discoverability in the REPL. It might also hugely increase adoption.

Corollary: An inverse 'methods' function which lists all the methods for a given type (can be huge).

(** I've read https://discourse.julialang.org/t/psa-julia-is-not-at-that-stage-of-development-anymore/44872 )

@timholy
Copy link
Member

timholy commented Oct 12, 2020

Your swims methods are busted (should be swims(::Cat) = false). After fixing that, methodswith(Cat) works.

But...why should we privilege the first argument? Julia itself doesn't, and it's a crucial advantage that it does not.

If your tab key is feeling neglected, I'd much rather see something like

?(x, yTAB

start listing options for ? depending on the types of x and y.

@JeffBezanson
Copy link
Member

I completely understand this argument, and it is often discussed as an advantage of "dot-style" object-oriented languages. But adding this to julia is really not realistic. Imagine I filed an issue with python saying "I don't like the dots, can you please allow function-call syntax for all methods?"

I don't think this is "just" syntactic sugar. It's arguably misleading for e.g. x.length() to look for length in the current scope instead of in x or its type. It's also not even clear how it should work, since x.y is already overloadable syntax. When do we call getproperty vs. looking for a y function in the calling scope? I mean that rhetorically --- even if some solution is possible I think that basic conflict would make the language significantly more confusing. Somewhat ironically, our python interop (PyCall.jl) relies on x.y() working the way it does currently --- rewriting it to y(x) would prevent PyCall from working.

@StefanKarpinski
Copy link
Member

One syntax that I've toyed with in my head is named infix operator syntax: i.e. the generalization of x in y. The tricky bit comes if you want to apply it to multiple arguments and if you want to chain it. There's also the issue that Julia functions don't generally take arguments in an obj.f(args...) friendly order. For example, you want to write collection.map(transform) but the order of arguments is the traditional functional order: map(transform, collections...). You'll note that this order is more flexible in general since you can map a transform over many collections. What would you do in object syntax? Write (a, b, c).map(transform)?

@timholy
Copy link
Member

timholy commented Oct 14, 2020

If the key point is tab-completion, doesn't ?(x, yTAB basically solve the problem? Just a question of someone PRing it.

@stevengj
Copy link
Member

Isn't this a duplicate of #31779?

@pannous
Copy link
Author

pannous commented Nov 18, 2020

Thanks for all the feedback.

I see that it would take more work than anticipated.

I guess the discoverability aspect could be solved in the REPL/IDE without changing the language:

pussy.swi<tab> => swims(pussy)

It's arguably misleading for e.g. x.length() to look for length in the current scope instead of in x or its type.

sorry I don't understand this objection.

(a, b, c).map(transform) looks good, as would
transform. map(a,b,c). These signatures are not mutually exclusive.

Since this idea is not received worth much excitement the issue can probably be closed. Just for the record, here is Haskells new take on the dot syntax:
ghc-proposals/ghc-proposals#282

So the argument in the 'duplicate' issue that this would turn Julia into an OOP language seems only partially valid.

A more philosophical question, maybe not applicable to Julia: could the concepts of functions and properties be merged? so that a.b=c is identical to b(a)=c?

@stevengj
Copy link
Member

Closing as a duplicate; in general, we don't like to have the same discussion in multiple issues.

You can add new comments to old issues, even if they are closed, if you feel that there is something that hasn't been said there.

timholy added a commit that referenced this issue Dec 9, 2020
timholy added a commit that referenced this issue Jul 25, 2021
timholy added a commit that referenced this issue Aug 9, 2021
* ?(x, y)TAB completes methods accepting x, y

Closes #30052
xref #38704
xref #37993

Co-authored-by: Jameson Nash <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

5 participants