メインコンテンツまでスキップ

フォーム

クエリパラメータの利用

入力した内容によって、次に移るページで表示されるものが異なるウェブページがあります。例として、Googleでutcodeと検索した際に表示される検索結果のページを考えてみましょう。

クエリパラメータ

このページのURLは、次のようになっています。

https://www.google.com/search?q=utcode&sourceid=chrome&ie=UTF-8

このようなページでは、入力した内容によって異なるウェブページが表示されるよう、入力した内容が移った先のページのURLに反映されています。この例では、URLの末尾に?q=utcode&sourceid=chrome&ie=UTF-8というデータが付加されています。これがクエリパラメータです。

クエリパラメータは、キーと値の組み合わせにより表現されます。先ほどのGoogleの検索結果であれば、次のような3つのパラメータが存在することになります。

キー
qutcode
sourceidchrome
ieUTF-8

Expressからクエリパラメータを利用する場合、request.query (express.Request#queryプロパティ) にこれらの値がオブジェクトとして格納されています。

main.mjs
import express from "express";

const app = express();
app.get("/", (request, response) => {
response.send(JSON.stringify(request.query));
});
app.listen(3000);
JSONとは何か

JSONは「JavaScript Object Notation」の略でJavaScriptのオブジェクト表記法を倣ったデータ記述方法のことです。つまりデータの保存はオブジェクトの形式で行うという決まり事です。表記法の名前なので臆する必要はありません。

JSON.stringifyメソッド

JSON.stringifyメソッドはオブジェクトをJSON形式の文字列に変換するメソッドです。

URLとして使用できない文字の対処方法

URLとして使用できない文字(日本語文字など)はURLエンコードする必要があります。JavaScriptならencodeURIComponent関数で変換できます。

encodeURIComponent("日本語"); // "%E6%97%A5%E6%9C%AC%E8%AA%9E"

サーバーにデータを送信する

form要素を使うとユーザーの入力からクエリパラメータを生成してページ遷移できます。

  • formaction属性: フォーム送信時に移動し、データを送信するページ
  • input要素: テキストボックス
  • inputname属性: クエリパラメータのキー
  • button要素: form内のボタンをクリックするとformactionに指定したページに遷移します(送信ボタンになります)

以下のコードの、HTMLファイルと、JavaScriptファイルを作成して実行してみましょう。

public/index.html
<!doctype html>
<html lang="ja">
<head>
<meta charset="utf-8" />
<title>フォーム</title>
</head>
<body>
<form action="/send">
<input name="name" />
<input name="age" />
<button>送信</button>
</form>
</body>
</html>
main.mjs
import express from "express";

const app = express();
app.use(express.static("./public"));
app.get("/send", (request, response) => {
response.send(
`あなたの名前は${request.query.name}で、${request.query.age}歳ですね。`,
);
});
app.listen(3000);

上記コードを実行すると、以下のような画面がブラウザに表示されます。

初めの画面

以下のように入力して、送信ボタンをクリックすると、 初めの画面に入力

http://localhost:3000/sendに移り、以下のような画面が表示されます。

http://localhost:3000/send

このページのURLを見てみましょう。特に、クエリパラメータの部分に注目してみましょう。 URLのクエリパラメータ

演習問題1

npmの章で扱ったnode-emojiパッケージのemojify関数を使って、送られた文字列にemojify関数を適用するサーバーを作ってみましょう。

また、文字列を簡単にサーバーに送れるようにフォームも作りましょう。

たとえば、フォームでI :heart: :coffee:と送信したときに、I ❤️ ☕とブラウザに表示されれば成功です。

解答例: node-emoji

node-emojiexpressをインストールするのを忘れないようにしましょう。

public/index.html
<!doctype html>
<html lang="ja">
<head>
<meta charset="utf-8" />
<title>node-emoji (フォーム) の解答例</title>
</head>
<body>
<form action="/emojify">
<input name="text" />
<button type="submit">送信</button>
</form>
</body>
</html>
main.mjs
import express from "express";
import { emojify } from "node-emoji";

const app = express();
app.use(express.static("./public"));

app.get("/emojify", (request, response) => {
const text = request.query.text;
const emojifiedText = emojify(text);
response.send(emojifiedText);
});

app.listen(3000);

演習問題2(発展)

書籍検索システムを作ってみましょう。まずは、配列に本のデータを登録します。

const books = [
{ title: "吾輩は猫である", author: "夏目漱石" },
{ title: "こころ", author: "夏目漱石" },
{ title: "坊つちやん", author: "夏目漱石" },
{ title: "舞姫", author: "森鴎外" },
{ title: "高瀬舟", author: "森鴎外" },
];

著者名を入力して送信すると、その著者の出版物が一覧表示されるシステムを作ってみましょう。

Array#filterメソッド

Array#filterメソッドは、関数オブジェクトを引数としてとり、その関数がtrueとなる要素だけからなる新しい配列を返すメソッドです。

const numbers = [1, 2, 3, 4, 5, 6, 7, 8];

/// [2, 4, 6, 8]
const evenNumbers = numbers.filter((number) => number % 2 === 0);
解答例
public/index.html
<form action="/search">
<input name="author" />
<button>検索</button>
</form>
main.mjs
import express from "express";

const app = express();

const books = [
{ title: "吾輩は猫である", author: "夏目漱石" },
{ title: "こころ", author: "夏目漱石" },
{ title: "坊つちやん", author: "夏目漱石" },
{ title: "舞姫", author: "森鴎外" },
{ title: "高瀬舟", author: "森鴎外" },
];

app.use(express.static("./public"));

app.get("/search", (request, response) => {
const selectedBooks = books.filter(
(book) => book.author === request.query.author,
);
response.send(`
<!doctype html>
<html lang="ja">
<head>
<meta charset="utf-8" />
<title>Document</title>
</head>
<body>
<ul>
${selectedBooks.map((book) => `<li>${book.title}</li>`).join("")}
</ul>
</body>
</html>
`);
});

app.listen(3000);