HTML의 폼 관련 태그들

목차

1. 시작

HTML 폼은 사용자가 웹사이트와 상호작용할 수 있는 많은 기능을 제공한다. 회원가입이나 로그인 등에 필요한 get, post 요청도 날릴 수 있다. 하지만 이런 트윗이 올라올 정도로 그렇게 주목받지 못하고 있다. 솔직히 나도 몇 번 써본 적이 없다.

html-form

따라서 MDN의 HTML 폼 가이드와 그간 주워들은 몇 가지를 정리하는 시간을 가지고자 한다. 만약 이 글을 읽게 되는 사람이 있다면 기본적인 HTML 지식과 약간의 개발 경험이 있다고 간주한다.

2. 간단한 서버 열기

express-server라는 폴더를 만들고 내부에서 npm init

그리고 express와 body-parser를 설치한다.

npm install express --save
npm install body-parser

폴더 내부의 index.js 파일을 다음과 같이 작성한다.

const express = require('express');
const bodyParser = require('body-parser')
const app = express();
const port = 8080;

app.use(bodyParser.urlencoded({ extended: false }))

// parse application/json
app.use(bodyParser.json())

app.get('/', (req, res) => res.send('Hello World!'));

app.post('/', (req, res) => {
  console.log(req.body);
  res.send('Got a POST request');
});

app.listen(port, () => console.log(`Example app listening on port ${port}!`));

이제 node index.js를 실행하면 localhost:8080으로 서버에 접속할 수 있다. 그리고 localhost:8080에서는 get, post 요청을 처리할 수 있게 될 것이다.

3. HTML 폼 기본

다음과 같이 localhost:8080으로 post 요청을 보내는 폼을 만들어보자. 간단한 로그인 폼이다.

<form action="http://localhost:8080/" method="post">
  <label for="userid">아이디</label>
  <input type="text" id="userid" name="userid" placeholder="id" />
  <label for="userpw">비밀번호</label>
  <input type="password" id="userpw" name="userpw" placeholder="pw" />
  <button type="submit">로그인</button>
</form>

form의 action 속성은 데이터를 보낼 URL을, method 속성은 어떤 HTTP 메서드를 사용할지를 지정하는 데 쓰인다.

이 폼을 만들고 제출하게 되면 localhost:8080에서는 다음과 같은 로그가 출력될 것이다.

{ userid: '입력한 id', userpw: '입력한 비밀번호' }

input에 지정한 name은 서버에서 받을 때 사용할 key가 된 것을 알 수 있다.

4. 폼 구조화 태그

사용자 폼을 만들기 위한 태그들은 꽤 많고, form 태그를 써본 적 없는 사람이라도 한번쯤은 써봤을 많은 유명한 태그들이 있다.

예를 들어서 <input>이나 <button>과 같은 태그들은 사실 폼 관련 태그로 분류된다. 이들은 모두 form 태그 바깥에서도 사용할 수 있기 때문에 다른 곳에서도 쓰이는 경우가 많다. 그 경우 해당 요소들은 폼과 관련이 없으며 JS로 관련 동작을 정의해 주게 된다.

하지만 우린 지금 폼에 대해서 알아보고 있으므로, 이 요소들을 폼과 관련해서 간략히 알아보도록 하자.

4.1. form

form 태그는 폼을 정의하는 태그이다. HTML로 폼을 정의할 때는 언제나 이 요소로 시작해야 한다. 단 form을 다른 form 태그로 둘러싸는 것은 제한되어 있으며 그렇게 할 경우 예측할 수 없는 방식으로 작동할 수 있다.

가질 수 있는 속성들은 다음과 같다.

  • accept-charset: 폼을 서버로 제출할 때 사용할 문자 인코딩을 지정한다. 기본값은 unknown이다.
  • action: 폼을 통해서 서버로 정보를 전송할 때 사용할 URL을 지정한다.
  • autocomplete(기본값은 on): 브라우저가 form 항목을 사용자의 과거 입력값에 기반하여 채워넣을 수 있는지. 기본값은 on
  • enctype: 폼을 전송할 때 사용할 콘텐츠 MIME 타입을 지정한다. 기본값은 application/x-www-form-urlencoded
  • method: 폼을 전송할 때 사용할 HTTP 메서드를 지정한다. 기본값은 get
  • name: 폼의 이름을 지정한다. 반드시 문서 폼 가운데 고유해야 한다.
  • novalidate: 폼을 서버로 제출할 때 유효성 검사를 하지 않도록 지정한다. 기본값은 false
  • target: 폼 요청 전송 후 응답을 어떻게 받을지를 지정한다. 기본값은 _self

4.1.1. enctype

이 속성 중 enctype은 폼 데이터를 어떻게 인코딩해서 폼을 제출할지를 결정한다.

UTF-8 유니코드로 문자를 인코딩하는 encodeURI를 사용하는 application/x-www-form-urlencoded가 기본값이다.

multipart/form-data는 FormData API를 사용해서 폼을 제출하며 Ajax로 전송된다. 폼을 이용할 때는 반드시 이 enctype을 사용해야 한다.

text/plain은 폼을 제출할 때 폼 데이터를 인코딩하지 않고 일반 텍스트로 전송한다. HTML5에서 디버깅 용도로 추가되었으며 실제로 디버깅 용도로 대부분 사용된다.

4.2. fieldset, legend

<fieldset>은 같은 목적을 가진 폼 요소들을 묶을 때 사용된다. <legend>는 부모 <fieldset>의 요소를 설명하는 데에 쓰인다.

대부분의 스크린 리더들 또한 이를 인식하여 fieldset 내용을 읽기 전 legend를 읽어준다.

위의 로그인 폼을 이 요소들을 사용해서 보강해 보자.

<form action="http://localhost:8080/" method="post">
  <fieldset>
    <legend>로그인</legend>
    <label for="userid">아이디</label>
    <input type="text" id="userid" name="userid" placeholder="id" />
    <label for="userpw">비밀번호</label>
    <input type="password" id="userpw" name="userpw" placeholder="pw" />
    <button type="submit">로그인</button>
  </fieldset>
</form>

이렇게 하면 fieldset과 legend가 묶인 폼이 만들어진다. 아무 CSS를 넣지 않았지만 누가 보아도 묶인 form이라는 것을 알 수 있다.

fieldset-legend

이런 분류를 이용해서 <fieldset>으로 폼의 섹션을 나누고 <legend>로 섹션의 제목을 붙이는 것도 가능하다. 이는 MDN에서도 중요한 예제로 취급하는 유스케이스이다. 물론 남용하면 안 되겠지만.

fieldset 요소에 disabled 속성을 지정할 경우 일반적으로 회색으로 표시되며, 자손 컨트롤을 비활성화한다. 또한 모든 브라우저 이벤트를 받지 않게 된다. 예외적으로 <legend>안의 폼은 비활성화되지 않는다.

그리고 fieldset은 블록 레벨 요소이고, legend도 블록 레벨 요소이다.

이때 엣지, 크롬 브라우저에선 현재 fieldset 내부에서 flex, grid 디스플레이를 사용할 수 없는 버그가 존재한다.

4.3. label

label로 폼 내 UI의 설명을 나타낼 수 있다. for속성으로 지정한 id의 UI와 묶여진다.

이렇게 UI를 연관시키면 사용자가 label을 누를 시 해당하는 UI가 활성화되고 click 이벤트도 동작한다. 체크박스나 라디오버튼을 쓸 때 버튼의 영역보다 넓은 영역을 '버튼 선택을 위해 눌러야 하는 영역'으로 사용할 수 있어 특히 유용하다.

물론 label로 해당 UI를 둘러싸는 방식으로 둘을 연관시킬 수도 있다. 다음과 같이 말이다.

<label>
  <input type="checkbox" name="agree" />
  동의합니다.
</label>

하지만 이런 방식으로 label과 UI를 연관시킬 경우 몇 가지 종류의 보조 기술이 label과 해당 UI의 관계를 이해하지 못할 수 있어 for 속성으로 id를 지정하는 게 좋다. 위의 로그인 폼에서도 for속성을 사용하였다.

접근성을 고려할 때 label 내부에 button 같은 인터랙티브 요소를 배치하지 않는 게 좋다.

label은 인라인 요소이다.

여러 개의 라벨을 하나의 요소에 연결하는 건 좋지 않으며 그럴 경우 label 내부에 span 태그를 넣어서 해결하자.

또한 특별히 읽어져야 하는 요소가 있다면 aria-label 속성을 사용한다. 아래 같은 경우 필수 요소라는 의미를 갖는 *aria-label을 설정하였다.

<label for="username">Name: <span aria-label="required">*</span></label>
<input id="username" type="text" name="username" required />

4.4. output

사용자의 입력을 받아서 계산 출력을 보여주는 데 사용된다. for 속성으로 다른 요소의 id를 지정하여 어떤 요소의 출력인지를 지정할 수 있다.

예를 들어서 사용자가 입력한 ID를 위의 로그인 폼에서 실시간으로 보여주고 싶다면 다음과 같이 한다.

<form 
  action="http://localhost:8080/" 
  method="post" 
  id="loginForm"
  oninput="result.value='사용자가 입력한 ID '+userid.value"
>
  <fieldset form="loginForm">
    <legend>로그인</legend>
    <label for="userid">아이디</label>
    <input type="text" id="userid" name="userid" placeholder="id" />
    <label for="userpw">비밀번호</label>
    <input type="password" id="userpw" name="userpw" placeholder="pw" />
    <output name="result" for="loginForm">사용자가 입력한 ID </output>
  </fieldset>
  <button type="submit" form="loginForm">로그인</button>
</form>

사용자의 ID 인풋 값만 보여주면 되니까 input에 for를 지정해도 된다.

<form 
  action="http://localhost:8080/" 
  method="post" 
  id="loginForm"
  oninput="result.value=userid.value"
>
  <fieldset form="loginForm">
    <legend>로그인</legend>
    <label for="userid">아이디</label>
    <input type="text" id="userid" name="userid" placeholder="id" />
    <label for="userpw">비밀번호</label>
    <input type="password" id="userpw" name="userpw" placeholder="pw" />
    <output name="result" for="userid"></output>
  </fieldset>
  <button type="submit" form="loginForm">로그인</button>
</form>

https://css-tricks.com/the-output-element/

4.5. 폼 구조

앞서 말했듯이 폼 구조는 HTML만으로도 만들 수 있다.

몇 가지 사용례가 있다. ul이나 ol 태그 내부의 li 요소들을 통해서 폼 요소들을 감싸거나, p, div요소들도 흔한 래퍼로 쓰인다. 리스트는 체크박스나 라디오버튼을 묶을 때 흔히 쓰인다.

그리고 fieldset 내부에 복잡한 양식이 있을 때는 section 요소로 요소들을 분류하고 제목 태그를 다는 것도 흔하다. 폼 안에 여러 기능을 하는 섹션이 있다면 section으로 분류해주자.

5. input

사용자의 데이터를 받을 수 있는 요소를 생성한다. type 속성으로 어떤 종류의 데이터를 받을지를 지정할 수 있고 이외에도 다양한 특성을 가지고 있다.

너무 길어져서 글을 분리하였다. HTML input tag

6. 기타 폼 요소

기타 폼 요소 MDN 글

6.1. textarea

줄바꿈을 포함하는 여러 줄의 일반 텍스트를 입력할 수 있는 컨트롤을 생성한다. <input>태그는 한 줄의 텍스트만 입력 가능했다. 오직 텍스트 콘텐츠만 받기 때문에 어떤 콘텐츠를 넣어도 텍스트로 렌더링된다.

colsrows 속성으로 컨트롤의 크기를 지정할 수 있고 wrap 속성으로 줄바꿈 방식을 지정할 수 있다.

<textarea>태그 사이에 텍스트를 넣으면 콘텐츠의 기본값이 된다. value요소를 사용하는 input보다 기본값 지정이 간단하다.

minlengthmaxlength 속성으로 입력할 수 있는 문자의 최소, 최대 길이를 지정할 수 있다.

<textarea
  rows="5"
  cols="10"
  minlength="10"
  maxlength="100"
  name="message"
>
  메시지를 입력하세요
</textarea>

resize CSS로 textarea의 크기 조절 방식을 지정할 수 있다. 기본값은 가로, 세로 모두 크기가 조절되는 both이다.

6.2. select

<select><option>요소로 선택지들 중 하나를 고르는 상자를 만들 수 있다. 폼 제출시 <select>요소의 value는 선택된 <option>의 value 값이 된다.

그리고 <optgroup>요소로 option들을 그룹화할 수 있다. <optgroup>태그가 갖는 label 속성의 값은 해당 태그에 싸인 선택지들의 위에 표기된다.

<optgroup>가 disabled 속성을 통해 비활성화되면 거기 속한 옵션들도 모두 비활성화된다.

<form>
  <p>
    <label for="coffeeMenu">주문할 커피를 고르세요</label>
    <select id="coffeeMenu" name="coffee">
      <!-- There is a trick here you think you'll pick
          a banana but you'll eat an orange >:-) -->
      <option value="카라멜마끼아또">카라멜마끼아또</option>
      <option value="카페라떼">카페라떼</option>
      <optgroup label="아메리카노">
        <option>아메리카노</option>
        <option>꿀아메리카노</option>
        <option>헤이즐넛아메리카노</option>
      </optgroup>
    </select>
  </p>
</form>

이렇게 선택된 선택지는 select에 지정된 name 속성을 키로 하고 선택된 option의 value 속성을 값으로 하는 객체로 전송된다. 위의 경우 카페라떼가 선택되었다면, { coffee: '카페라떼' }가 전송된다.

따라서 모든 option은 value 속성을 가져야 하는데 이를 생략시 option 태그 내부의 텍스트 콘텐츠를 value로 사용한다. option에 selected 속성 지정시 해당 option이 선택된 상태로 페이지가 렌더링된다.

select에 multiple 속성 지정시 다수 항목이 선택 가능하며(이 경우 더이상 선택 상자가 드롭다운으로 나오지 않는다) size 속성을 통해 select 요소가 focus상태가 아닐 경우 한 번에 노출되는 항목 수도 지정 가능하다.

form 속성을 통해 select와 연결할 form 요소도 지정 가능하다. form 바깥에 있어도 해당 폼과 연결이 가능해지는 것이다.

6.2.1. CSS 스타일링

select 속성은 CSS 스타일링이 어렵기로 유명하다. appearance 등으로 기본 외형을 제거할 수도 있고 몇몇 속성은 잘 적용되지만 내부 구조도 복잡하고 일관적이지 않은 결과가 나올 수 있다.

따라서 일관된 스타일링이 중요하다면 JS와 WAI-ARIA 등을 이용해 별도의 드롭다운 메뉴를 만드는 것도 고려 가능한 옵션이다.

6.3. datalist

이 요소를 이용해서 폼 요소 입력 시 자동완성에 대한 선택지를 제공할 수 있다. 사용자가 직접 값을 입력할 시 이는 무시된다.

<datalist>는 다른 제어 요소에서 고를 수 있는 선택지를 나타내는 <option> 요소 여러 개를 담고, <input> 요소 등의 list 속성에 연결하고자 하는 <datalist> 태그의 id를 지정하면 연결된다.

list 속성으로 이게 연결되면 input요소 등에서 입력시 datalist의 option들이 자동 완성으로 나타난다. 보통은 드롭다운 박스로 나타난다.

MDN에서 제공하는 예제

이외에도 <input type="color">에 연결할 경우 색상을 선택 가능한 팔레트가 나타나게 하는 등의 동작을 할 수 있다.

6.3.1. fallback

대부분의 브라우저에서 <datalist>를 지원하지만 IE의 낮은 버전이나, 안드로이드 파이어폭스의 2020년 이전 버전과 같은 낡은 브라우저에서는 지원하지 않는 경우도 있다.

이럴 땐 <datalist>가 제대로 작동할 시 내부 요소 중 <option>이 아닌 요소들은 모두 무시한다는 점을 이용해서 <select> 요소를 이용한 fallback을 만들 수 있다.

<datalist id="coffeeList">
  <label for="suggestion">Pick Menu</label>
  <select id="suggestion" name="altCoffee">
    <option>아메리카노</option>
    <option>카페라떼</option>
    <option>카페모카</option>
  </select>
</datalist>

<datalist>가 작동할 시 label, select는 무시되고 option들만 표시될 것이고 작동하지 않을 시 선택 상자가 나올 것이다.

6.4. meter, progress

<meter> 요소는 하나의 값이 최소와 최댓값 사이에서 상대적으로 어느 정도 위치를 점하는지를 나타낸다.

<form>
  <label for="percent">비율</label>
  <meter
  id="percent"
  value="60"
  min="0"
  max="100"
  >
  60%
  </meter>
</form>

<progress> 요소는 0부터 max까지 중 어느 작업의 완료 정도를 나타낸다. 따라서 최솟값은 항상 0이며 max 미만의 value 속성 값에 따라서 프로그레스 바가 채워진다.

6.5. button

<button> 요소는 클릭 가능한 버튼을 나타낸다. form 외부에도 얼마든지 배치할 수 있다.

type은 submit, reset, button이 있다. 지정하지 않은 경우 submit이 기본값이며 제출 동작을 안 하게 하고 싶다면 type="button"을 지정하자.

또한 submit 속성을 지정한 input요소에서 그랬던 것과 같이 formaction, formenctype, formmethod, formnovalidate, formtarget 속성을 사용하여 form의 속성을 오버라이딩할 수 있다.

6.5.1. button과 input

한때 <button> 태그는 <input type="button"> 태그에 비해 덜 쓰였다. 이는 IE6, IE7에서의 버그 때문이었다.

이 버그는 <button> 태그의 namevalue를 지정하고 폼을 제출하면 value가 전송되는 대신 버튼의 raw content를 전송하는 버그였다. 따라서 사람들은 이 버그를 방지하기 위해 <input>을 사용하였다.

그러나 IE8부터는 이 버그가 고쳐졌기 때문에 <button>을 마음놓고 사용하면 된다. <button><input>에 비해 내부에 HTML 컨텐츠도 넣을 수 있으므로 스타일링도 편하다.

7. 폼 밖에서 폼 연관시키기

HTML을 하다 보면 <button>과 같은 요소들을 폼과 관련 없는 부분에서도 사용할 때가 많다. 그러면 이들을 폼과 연관시키기 위해서는 어떤 방법이 필요할까?

폼 내부에 버튼을 포함시키면 된다. 하지만 그럴 수 없다면? 그럴 때 form 속성이 등장한다.

form 속성은 폼과 연관시킬 수 있는 요소들에 사용할 수 있다. 이 속성은 폼의 id를 가리키는데, 이를 통해서 폼과 연관시킬 수 있다. 가령 button 요소를 form 안에 넣지 않고 폼과 연관시켜 보겠다.

<form action="http://localhost:8080/" method="post" id="loginForm">
  <fieldset>
    <legend>로그인</legend>
    <label for="userid">아이디</label>
    <input type="text" id="userid" name="userid" placeholder="id" />
    <label for="userpw">비밀번호</label>
    <input type="password" id="userpw" name="userpw" placeholder="pw" />
  </fieldset>
</form>
<button type="submit" form="loginForm">로그인</button>

이는 form 바깥에 있는 fieldset 요소를 폼에 포함해야 할 때도 사용할 수 있다. fieldset 요소에 연결할 form 요소 id를 form 속성에 지정하면 된다.

<label> 요소도 form 속성을 통해서 외부에 있는 폼과 연관시킬 수 있다. 이렇게 하면 label이 폼 요소 내부에 있지 않아도 어디에서나 폼과 연관시킬 수 있다.

8. 레거시 브라우저의 폼

IE나 오래된 스마트폰의 브라우저와 같은 레거시 브라우저를 지원하는 건 쉽지 않다. <input>태그의 특정 type속성을 지원하지 않는 브라우저의 경우 type="text"로 자동으로 변환되는 등의 지원이 있긴 하지만 기본적으로는 쉽지 않다. 이런 상황을 위한 조언이 MDN에 있다.

스타일링은 레거시 브라우저 지원이 더 힘들다. 폼 요소의 프로퍼티 호환성 표같은 것을 보면 이를 확실히 느낄 수 있다. 이럴 때는 CSS의 @supports 쿼리를 이용해서 레거시 브라우저를 위한 스타일링을 따로 지정해주는 것도 좋다.

레거시 브라우저에서는 지원하지 않는 API를 폴리필을 통해 사용하는 경우도 있는데 레거시 브라우저의 엔진은 일반적으로 더 느리기까지 하다. 그러면 레거시 브라우저는 폴리필을 사용하는 것뿐 아니라 느린 엔진의 문제까지 겹쳐서 더 큰 UI 문제를 발생시킨다. HTML/CSS와 같은 구조와 JS동작을 엄격히 분리하는 unobtrusive JS 방법론을 이용해서 JS가 없이도 최소한의 기능과 접근성을 보장하는 방법을 고려해볼 수 있다.

혹은 커스텀 위젯을 직접 만드는 것도, 매우 큰 비용과 시간을 감수할 만하다면 시도해 볼 수 있다.

참고

HTML 참고서, 양식 부분 https://developer.mozilla.org/ko/docs/Web/HTML/Element#%EC%96%91%EC%8B%9D

HTML 폼 가이드 https://developer.mozilla.org/ko/docs/Learn/Forms

express 공식 문서 https://expressjs.com/ko/starter/hello-world.html

express와 body-parser https://expressjs.com/en/resources/middleware/body-parser.html

https://tech.devsisters.com/posts/functional-react-state-management/

https://dev.to/dailydevtips1/submit-button-outside-the-form-2m6f

output element https://css-tricks.com/the-output-element/

레거시 브라우저에서의 폼 https://developer.mozilla.org/en-US/docs/Learn/Forms/HTML_forms_in_legacy_browsers