別のコンボボックスで選択されたものに基づいて、マップからの値をコンボボックスに動的に取り込む
-
03-07-2019 - |
質問
わかりました、ここにJava / JavaScriptの達人用の
私のアプリでは、コントローラーの1つがTreeMapをそのJSPに渡します。このマップには、キーとして自動車メーカーの名前が、値として自動車オブジェクトのリストがあります。これらのCarオブジェクトは、車の名前、ID、製造年などを含む単純なBeanです。 そのため、マップは次のようになります(これは単なる例であり、物事を少し明確にするためのものです):
キー:ポルシェ
値:3つのCarオブジェクトを含むリスト(たとえば、911、Carrera、Boxter、生産年数およびID)
キー:フィアット
値:2つのCarオブジェクトを含むリスト(たとえば、PuntoとUno)
など...
今、私のJSPには2つのコンボボックスがあります。 1つは自動車メーカーのリスト(地図のキー-この部分は方法を知っています)を受け取り、もう1つは動的に変更して、ユーザーが特定の車を選択したときに車の名前を表示する必要があります最初のコンボボックスのメーカー。そのため、たとえば、ユーザーは「ポルシェ」を選択します。最初のコンボボックスで、2番目はすぐに「911、カレラ、ボクスター」を表示します...
これを行う方法を見つけるために数日を費やした後、敗北を認める準備ができました。いろいろなことを試しましたが、途中で壁にぶつかるたびに。誰も私がこれにアプローチする方法を提案できますか?
はい、私はJavaScript初心者です。誰かが疑問に思っていたら...
編集:これをコードチャレンジとして再タグ付けしました。 JavaScriptフレームワーク(JQueryなど)を使用せずにこの問題を解決するすべての人に称賛を送ります。
解決 2
とにかく、私が言ったように、私は最終的に自分でそれをやることができたので、ここに私の答えがあります...
このようにコントローラーからマップを受け取ります(Springを使用していますが、他のフレームワークでこれがどのように機能するかわかりません):
<c:set var="manufacturersAndModels" scope="page" value="${MANUFACTURERS_AND_MODELS_MAP}"/>
これらは私のコンボです:
<select id="manufacturersList" name="manufacturersList" onchange="populateModelsCombo(this.options[this.selectedIndex].index);" >
<c:forEach var="manufacturersItem" items="<%= manufacturers%>">
<option value='<c:out value="${manufacturersItem}" />'><c:out value="${manufacturersItem}" /></option>
</c:forEach>
</select>
select id="modelsList" name="modelsList"
<c:forEach var="model" items="<%= models %>" >
<option value='<c:out value="${model}" />'><c:out value="${model}" /></option>
</c:forEach>
</select>
次のクラスをインポートしました(もちろん、一部の名前は変更されています):
<%@ page import="org.mycompany.Car,java.util.Map,java.util.TreeMap,java.util.List,java.util.ArrayList,java.util.Set,java.util.Iterator;" %>
そして、ここにすべてのハードワークを行うコードがあります:
<script type="text/javascript">
<%
Map mansAndModels = new TreeMap();
mansAndModels = (TreeMap) pageContext.getAttribute("manufacturersAndModels");
Set manufacturers = mansAndModels.keySet(); //We'll use this one to populate the first combo
Object[] manufacturersArray = manufacturers.toArray();
List cars;
List models = new ArrayList(); //We'll populate the second combo the first time the page is displayed with this list
//initial second combo population
cars = (List) mansAndModels.get(manufacturersArray[0]);
for(Iterator iter = cars.iterator(); iter.hasNext();) {
Car car = (Car) iter.next();
models.add(car.getModel());
}
%>
function populateModelsCombo(key) {
var modelsArray = new Array();
//Here goes the tricky part, we populate a two-dimensional javascript array with values from the map
<%
for(int i = 0; i < manufacturersArray.length; i++) {
cars = (List) mansAndModels.get(manufacturersArray[i]);
Iterator carsIterator = cars.iterator();
%>
singleManufacturerModelsArray = new Array();
<%
for(int j = 0; carsIterator.hasNext(); j++) {
Car car = (Car) carsIterator.next();
%>
singleManufacturerModelsArray[<%= j%>] = "<%= car.getModel()%>";
<%
}
%>
modelsArray[<%= i%>] = singleManufacturerModelsArray;
<%
}
%>
var modelsList = document.getElementById("modelsList");
//Empty the second combo
while(modelsList.hasChildNodes()) {
modelsList.removeChild(modelsList.childNodes[0]);
}
//Populate the second combo with new values
for (i = 0; i < modelsArray[key].length; i++) {
modelsList.options[i] = new Option(modelsArray[key][i], modelsArray[key][i]);
}
}
他のヒント
チャレンジが大好きです。
jQueryなし、単なるJavaScript、Safariでテスト済み。
事前に次のコメントを追加します。
- エラーのために長い間失敗しています チェック。
- 2つの部分が生成されます。 マップを持つ最初のスクリプトノード および製造元の内容 選択
- Works on My Machine(TM) (Safari / OS X)
- なし(css) スタイリングが適用されました。味が悪いので とにかく役に立たない。
。
<body>
<script>
// DYNAMIC
// Generate in JSP
// You can put the script tag in the body
var modelsPerManufacturer = {
'porsche' : [ 'boxter', '911', 'carrera' ],
'fiat': [ 'punto', 'uno' ]
};
</script>
<script>
// STATIC
function setSelectOptionsForModels(modelArray) {
var selectBox = document.myForm.models;
for (i = selectBox.length - 1; i>= 0; i--) {
// Bottom-up for less flicker
selectBox.remove(i);
}
for (i = 0; i< modelArray.length; i++) {
var text = modelArray[i];
var opt = new Option(text,text, false, false);
selectBox.add(opt);
}
}
function setModels() {
var index = document.myForm.manufacturer.selectedIndex;
if (index == -1) {
return;
}
var manufacturerOption = document.myForm.manufacturer.options[index];
if (!manufacturerOption) {
// Strange, the form does not have an option with given index.
return;
}
manufacturer = manufacturerOption.value;
var modelsForManufacturer = modelsPerManufacturer[manufacturer];
if (!modelsForManufacturer) {
// This modelsForManufacturer is not in the modelsPerManufacturer map
return; // or alert
}
setSelectOptionsForModels(modelsForManufacturer);
}
function modelSelected() {
var index = document.myForm.models.selectedIndex;
if (index == -1) {
return;
}
alert("You selected " + document.myForm.models.options[index].value);
}
</script>
<form name="myForm">
<select onchange="setModels()" id="manufacturer" size="5">
<!-- Options generated by the JSP -->
<!-- value is index of the modelsPerManufacturer map -->
<option value="porsche">Porsche</option>
<option value="fiat">Fiat</option>
</select>
<select onChange="modelSelected()" id="models" size="5">
<!-- Filled dynamically by setModels -->
</select>
</form>
</body>
Strutsを使用していますか
これを行うには、JavaScriptのトリック(またはAJAX)が必要になります。
行う必要があるのは、JavaScriptコードのどこか(その分だけ生成する方法は別として):
var map = {
'porsche': [ 'boxter', '911', 'carrera' ],
'fiat': ['punto', 'uno']
};
これは基本的にサーバー側のデータ構造のコピー、つまりメーカーがキーとするマップで、各値には車のタイプの配列があります。
次に、メーカーのonchangeイベントで、上記で定義したマップから配列を取得し、そこからオプションのリストを作成する必要があります。 (devguru.comをご覧ください-標準のJavaScriptオブジェクトに関する多くの有用な情報があります)。
車のリストの大きさにもよりますが、AJAXルートに行くのが最善かもしれません。
メーカーに指定された車種のリストを検索する新しいコントローラーを作成し、 JSON (JSONである必要はありませんが、私にとっては非常にうまく機能します。)
次に、 jQuery などのライブラリを使用して、onchangeイベントで車のリストを取得し、メーカー。 (jQueryは、知っておくべき優れたJavaScriptフレームワークです。JavaScriptを使用した開発がはるかに簡単になります。ドキュメントは非常に優れています。)
その一部が理にかなっていることを願っていますか
これは、タグライブラリや外部依存関係を一切伴わない、jspでの実用的なカットアンドペーストの回答です。モデルを含むマップはハードコーディングされていますが、問題は発生しません。
追加されたJSPは読みやすさを向上させないため、この回答を前回の回答から分離しました。そして、「実生活」では、JSPにすべての埋め込みロジックを負担するのではなく、どこかのクラスに配置します。またはタグを使用します。
すべての&quot; first&quot;ものは、スーパーフルオを防ぐためです生成されたコード内。 foreachを使用しても、要素の量に関する知識は得られないため、最後にチェックします。また、最初の要素の処理をスキップして、最後の&quot;、&quot;を削除することもできます。その後、ビルダーの長さを1減らします。
<%@ page language="java" contentType="text/html; charset=ISO-8859-1"
pageEncoding="ISO-8859-1"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<%@page import="java.util.Map"%>
<%@page import="java.util.TreeMap"%>
<%@page import="java.util.Arrays"%>
<%@page import="java.util.Collection"%>
<%@page import="java.util.List"%>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<title>Challenge</title>
</head>
<body onload="setModels()">
<% // You would get your map some other way.
Map<String,List<String>> map = new TreeMap<String,List<String>>();
map.put("porsche", Arrays.asList(new String[]{"911", "Carrera"}));
map.put("mercedes", Arrays.asList(new String[]{"foo", "bar"}));
%>
<%! // You may wish to put this in a class
public String modelsToJavascriptList(Collection<String> items) {
StringBuilder builder = new StringBuilder();
builder.append('[');
boolean first = true;
for (String item : items) {
if (!first) {
builder.append(',');
} else {
first = false;
}
builder.append('\'').append(item).append('\'');
}
builder.append(']');
return builder.toString();
}
public String mfMapToString(Map<String,List<String>> mfmap) {
StringBuilder builder = new StringBuilder();
builder.append('{');
boolean first = true;
for (String mf : mfmap.keySet()) {
if (!first) {
builder.append(',');
} else {
first = false;
}
builder.append('\'').append(mf).append('\'');
builder.append(" : ");
builder.append( modelsToJavascriptList(mfmap.get(mf)) );
}
builder.append("};");
return builder.toString();
}
%>
<script>
var modelsPerManufacturer =<%= mfMapToString(map) %>
function setSelectOptionsForModels(modelArray) {
var selectBox = document.myForm.models;
for (i = selectBox.length - 1; i>= 0; i--) {
// Bottom-up for less flicker
selectBox.remove(i);
}
for (i = 0; i< modelArray.length; i++) {
var text = modelArray[i];
var opt = new Option(text,text, false, false);
selectBox.add(opt);
}
}
function setModels() {
var index = document.myForm.manufacturer.selectedIndex;
if (index == -1) {
return;
}
var manufacturerOption = document.myForm.manufacturer.options[index];
if (!manufacturerOption) {
// Strange, the form does not have an option with given index.
return;
}
manufacturer = manufacturerOption.value;
var modelsForManufacturer = modelsPerManufacturer[manufacturer];
if (!modelsForManufacturer) {
// This modelsForManufacturer is not in the modelsPerManufacturer map
return; // or alert
}
setSelectOptionsForModels(modelsForManufacturer);
}
function modelSelected() {
var index = document.myForm.models.selectedIndex;
if (index == -1) {
return;
}
alert("You selected " + document.myForm.models.options[index].value);
}
</script>
<form name="myForm">
<select onchange="setModels()" id="manufacturer" size="5">
<% boolean first = true;
for (String mf : map.keySet()) { %>
<option value="<%= mf %>" <%= first ? "SELECTED" : "" %>><%= mf %></option>
<% first = false;
} %>
</select>
<select onChange="modelSelected()" id="models" size="5">
<!-- Filled dynamically by setModels -->
</select>
</form>
</body>
</html>
プロトタイプを使用して、このようなものはどうですか?まず、カテゴリの選択ボックス:
<SELECT onchange="changeCategory(this.options[this.selectedIndex].value); return false;">
<OPTION value="#categoryID#">#category#</OPTION>
...
次に、サブカテゴリごとに1つずつ、N個の異なる選択ボックスを出力します。
<SELECT name="myFormVar" class="categorySelect">
...
changeCategory javascript関数は、クラスcategorySelectのすべての選択を無効にし、現在のcategoryIDの選択のみを有効にします。
// Hide all category select boxes except the new one
function changeCategory(categoryID) {
$("select.categorySelect").each(function (select) {
select.hide();
select.disable();
});
$(categoryID).show();
$(categoryID).enable();
}
プロトタイプでこのように非表示/無効にすると、ページ上で非表示になるだけでなく、そのFORM変数が投稿されないようにします。したがって、同じFORM変数名(myFormVar)を使用してN個の選択を行っても、アクティブなもののみが投稿されます。
それほど前ではなく、似たようなことを考えました。
jQueryとTexoTelaアドオンを使用すると、それほど難しくありませんでした。
最初に、上記のマップのようなデータ構造があります:
var map = {
'porsche': [ 'boxter', '911', 'carrera' ],
'fiat': ['punto', 'uno']
};
HTMLは次のように見えるはずです:
<select size="4" id="manufacturers">
</select>
<select size="4" id="models">
</select>
次に、最初のコンボに次のようなjQueryコードを入力します。
$(document).ready(
function() {
$("#bronsysteem").change( manufacturerSelected() );
} );
);
manufacturerSelectedは、onChangeイベントで登録されたコールバックです
function manufacturerSelected() {
newSelection = $("#manufacturers").selectedValues();
if (newSelection.length != 1) {
alert("Expected a selection!");
return;
}
newSelection = newSelection[0];
fillModels(newSelection);
}
function fillModels(manufacterer) {
var models = map[manufacturer];
$("models").removeOption(/./); // Empty combo
for(modelId in models) {
model = models[modelId];
$("models").addOption(model,model); // Value, Text
}
}
これでうまくいくはずです。
構文エラーがある可能性があることに注意してください。私はあなたのユースケースを反映するようにコードを編集しましたが、削除しなければなりませんでした かなりたくさんあります。
これが役立つ場合、コメントをいただければ幸いです。
前回の投稿のアドオンとして。マップを反復処理するJSPにスクリプトタグを配置できます。マップの反復処理の例は、 Strutsのマップ。
達成したいこと(フォームの送信を気にしない場合)は次のようなものだと思います:
<script>
var map = {
<logic:iterate id="entry" name="myForm" property="myMap">
'<bean:write name=" user" property="key"/>' : [
<logic:iterate id="model" name="entry" property="value">
'<bean:write name=" model" property="name"/>' ,
</logic:iterate>
] ,
</logic:iterate>
};
</script>
まだいくつかの不自然な&quot;、&quot;があります。防止したいかもしれませんが、これでうまくいくと思います。