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

[TextField] Can't pass a ref to TextField component #1083

Closed
theodoretan opened this issue Feb 21, 2019 · 16 comments
Closed

[TextField] Can't pass a ref to TextField component #1083

theodoretan opened this issue Feb 21, 2019 · 16 comments

Comments

@theodoretan
Copy link
Member

theodoretan commented Feb 21, 2019

Background

We have a use case where we have two TextFields and their heights are being matched. The current way we're accomplishing this is by using a ref to get the height of the TextFields from the DummyInput and passing back the larger of the two.

Problem

With the 3.6.0 version, TextField is now being wrapped with withAppProvider meaning that the ref is no longer being set on the component and it is breaking our section since we can't do ref.current.input to get the height.

Solution

I have two solutions in mind and have seen them used in the current code base:

  1. Use the withRef higher-order component with compose in TextField so that refs can be forwarded to the component. Keeping the functionality of our use case.

  2. Add a data-height attribute on the component that will give the user the height that the component calculates so that we don't have to use a ref.

I'm leaning towards solution 2 because, at least for our use case, we just need the calculated height of the textfields. The use case with using a ref would probably be where someone needs to modify the element and I'm not sure that's something we would want to let happen.

@danrosenthal
Copy link
Contributor

Is there any reason that wrapping the input in a div, placing the ref on that div, and getting the height from that would not work?

@AndrewMusgrave
Copy link
Member

Is there any reason that wrapping the input in a div, placing the ref on that div, and getting the height from that would not work?

clientHeight doesn't include margin or border so that may be a viable option.

If you can't use Dan's suggestion I'm in favour of option one. It's the "react" way of doing things and using a ref only isn't an option anymore because of "wrapper hell"

@theodoretan
Copy link
Member Author

Thanks, placing the ref of the div works 👍

@raunofreiberg
Copy link

raunofreiberg commented Jun 25, 2019

What if one wanted to use https://github.com/text-mask/text-mask/tree/master/react#customize-rendered-input-component with a TextField from Polaris? The library needs access to the input DOM node to manage masking.

@danrosenthal
Copy link
Contributor

@raunofreiberg I'm not familiar with text-mask, but on a quick glance it appears to be an abstraction on top of the pattern property. We expose a pattern prop on TextField, which translates directly to the pattern attribute on the input element. Is there any reason that does not fit your use-case?

@raunofreiberg
Copy link

The pattern attribute doesn't seem to mask the input. What I exactly have in mind is the date input here: https://text-mask.github.io/text-mask/

Typing in the input helps you just fill in the blanks for the date with a nice UX.

@danrosenthal
Copy link
Contributor

danrosenthal commented Jun 25, 2019

Yup, agreed that is a nice UX, and not achievable in the current implementation. This may even be something worth supporting natively in the component.

@AndrewMusgrave what would it look like to implement ref forwarding with WithRef HOC? Are there other components where access to the underlying DOM node might be useful?

@sijad
Copy link
Contributor

sijad commented Oct 9, 2019

it's also not possible to use https://react-hook-form.com/ because it requires to register inputs via ref prop.

please also see #1918

@andychongyz
Copy link

@sijad Do you in anyhow found any way to use react-hook-form? If not, what form validation library would you recommend to use?

@sijad
Copy link
Contributor

sijad commented Dec 4, 2019

@andychong1996 unfortunatly no. I'm using Formik v2 + Fonk in my project.
yup is more popular validator but Fonk was simpler.
for formik you can also use this projec: t: https://github.com/SatelCreative/formik-polaris

@AndrewMusgrave
Copy link
Member

Reaching into components is something that Polaris doesn't recommend since it's likely that a future upgrade will eventually break something. If your deadset on using react hook form, you can still access the input element using web api's. If you need to register the input element multiple times, you may want to look into using a mutation observer as well.

import React, { useCallback, useState, useEffect } from "react";
import { TextField } from "@shopify/polaris";

function Input({ children, id }) {
  const [inputNode, setInputNode] = useState();

  useEffect(() => {
      setInputNode(document.querySelector('input'));
  }, [id]);

  return children(inputNode);
}

export default function TextFieldExample() {
  const [value, setValue] = useState("Jaded Pixel");
  const id = "my-text-field";

  const handleChange = useCallback(newValue => setValue(newValue), []);

  return (
    <Input>
      {inputNode => {
        console.log(inputNode);
        return (
          <TextField id={id} label="Store name" value={value} onChange={handleChange} />
        );
      }}
    </Input>
  );
}

With that being said, we don't recommend reaching into components 😄

@sonatard
Copy link

Material UI supports ref.
mui/material-ui#23174

Are there any changes Polaris plan?

@airhorns
Copy link

It seems like the overall "don't reach into components" design principle is breaking compatibility with the most popular React form state management library. In this case, is it really adding that much future compatibility guarantees? TextField will always have to wrap some user-facing input element, and should always be a good citizen and emit browser-land focus and blur events. Those two bits are what react-hook-form wants to use a ref to couple to, and that seems pretty reasonable as both of those are a fundamental part of any text field's contract. Exposing a ref in the long term that commits to that (and only that) doesn't extend or change the contract if it preserves just those elements.

I also might call this DX hostile -- this decision for "a future upgrade that may break something" is causing real pain for devs now. Maybe a better option would be to expose the ref, and reserve the right to change which element exactly is referenced in the future, enabling us forced-to-use-your-thing consumers to still play nice with the rest of the ecosystem while bearing responsibility for the maintenance burden if we reach in too far? Knowing that there is a (super poorly performing) work around using the DOM that you folks documented too -- why make folks have to do the workaround?

@SearheiParkhamchuk
Copy link

Here is my workaround:

import { TextField } from '@shopify/polaris'

import { type ForwardedRef, forwardRef, useId, useImperativeHandle } from 'react'

import { type InputTextProps } from './@types'

function InputText({...props}: InputTextProps, ref: ForwardedRef<HTMLInputElement>) {
  const id = useId()

  useImperativeHandle(ref, () => document.getElementById(id))

  return (
    <TextField
     {...props}
      id={id}
    />
  )
}

export default forwardRef<HTMLInputElement, InputTextProps>(InputText)

@MTraveller
Copy link

This works:

import { Controller, useForm } from "react-hook-form";

type Inputs = {
  name: string;
};

export default function MyField() {
  const { control } = useForm<Inputs>();
 
  return(
    <Controller
      name="name"
      control={control}
      defaultValue=""
      rules={{ required: "Name is required" }}
      render={({ field, fieldState: { error } }) => (
        <TextField
          label="Name"
          name={field.name}
          value={field.value}
          onChange={(value) => field.onChange(value)}
          onBlur={field.onBlur}
          error={error && error.message}
          autoComplete="off"
        />
      )}
    />
  );
}

@muchisx
Copy link

muchisx commented Nov 14, 2024

While workarounds are possible, please consider re-opening this issue and exposing the refs, this is a standard react pattern that the most popular ui libraries follow, why is Polaris unique here? most importantly dealing with user-interactive elements such as form elements.

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