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

Shared functionality: ApplicationForm? Mixin? Boilerplate? #1

Open
jackjennings opened this issue Jun 15, 2017 · 8 comments
Open

Shared functionality: ApplicationForm? Mixin? Boilerplate? #1

jackjennings opened this issue Jun 15, 2017 · 8 comments

Comments

@jackjennings
Copy link
Member

Should microform also have an "install" generator that adds an ApplicationForm class to your project?

Or, taking a step back, how can some of the generic functionality like the respond_to? and other delegation method. This could either be a parent class, or a mix-in module, or simply repeated in each form class.

The next question is, if functionality is shared in either of those way, does that mean that the instance variable needs to get more generic:

def initialize project
  @project = project
end
# vs
def initialize record
  @record = record
end

Or, is there some way that the instance variable is configured. This could be too clever:

class ProjectForm
  include Micrform::Form[:project]
end

Or, the instance variable could be inferred from the form name, e.g. ProjectForm assumes there will be a @project instance variable assigned in the initializer.

Do these "solutions" jump trough too many hoops to just avoid renaming the attribute to the more generic record?

@jackjennings
Copy link
Member Author

If we're taking a cue from Pundit, they're using record and installing an ApplicationPolicy: https://github.com/elabs/pundit/blob/master/lib/generators/pundit/install/templates/application_policy.rb

@ghost
Copy link

ghost commented Jun 15, 2017

Hmm, yeah, an ApplicationForm and generic @record seems useful.

def initialize record
  @record = record
end

Repeating this would be repetitive, but I'm wondering if it would be the most readable to people? Hmm.

@jackjennings
Copy link
Member Author

record record record

Yeah, I like when variable names are specific, but willing to make concessions if we can hide some of the implementation details.

It could also be "subject", which at least points back towards the name of the form class for context.

Though I'm also aware of keeping consistency with previous artwork and the fact that pundit uses "record"…

@jackjennings
Copy link
Member Author

jackjennings commented Jun 20, 2017

I was wondering if maybe this takes the form of some modules that import class methods:

class ProjectForm
  extend Microform::Delegator

  attr_reader :project
  delegates_to :project

  # ...
end

Then that delegates_to class method defines the method_missing, respond_to? etc, passing them to the appropriately-named variable.

@jackjennings
Copy link
Member Author

Could open the door for other mixins as well… and make some functionality opt-in.

@jackjennings
Copy link
Member Author

I guess it does make me uncomfortable that it pushes functionality into the gem, instead of being "owned" by the project. In the case that there is a concrete ApplicationForm, at least that can be generated and live in the project instead of reaching into the library code to inject functionality like the above example.

@jackjennings jackjennings changed the title ApplicationForm? Shared functionality: ApplicationForm? Mixin? Boilerplate? Jun 20, 2017
@jackjennings
Copy link
Member Author

I'm leaning towards an ApplicationForm, after reading the original post in this PR about standardizing to Application* classes: rails/rails#28243

But I'm not sure what gets included there. A blank class? A class that somehow has the method_missing etc defined, or that defines a class method to do so in child forms. Would push for the simplest option, but I'm not sure yet what that is.

@jackjennings
Copy link
Member Author

jackjennings commented Jun 21, 2017

ApplicationForm could be a SimpleDelegator, and just sidestep any reference to the object at all:

class ApplicationForm < SimpleDelegator
end

class ProjectForm < ApplicationForm
  def submit changeset
    update changeset
  end
end

In effect that's the same that we're doing with method_missing now, but feels weird on a gut level. Seems like there might be some strangeness based on the readme of this library I just found: https://github.com/stevenharman/dumb_delegator

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

1 participant