mirror of https://github.com/sveltejs/svelte
				
				
				
			feat: improve `bind:group` behavior (#7892)
	
		
	
				
					
				
			track all `#each` variables that could result in a change to the inputs and also update the `$$binding_groups` variable which holds the references to the inputs of each group accordingly. Fixes #7633 Fixes #6112 Fixes #7884pull/8349/head
							parent
							
								
									6476e9b34f
								
							
						
					
					
						commit
						0966d1d282
					
				| @ -0,0 +1,52 @@ | ||||
| // https://github.com/sveltejs/svelte/issues/7633
 | ||||
| export default { | ||||
| 	async test({ assert, target, component, window }) { | ||||
| 		let inputs = target.querySelectorAll('input'); | ||||
| 
 | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 
 | ||||
| 		component.moveDown(0); | ||||
| 		component.moveDown(1); | ||||
| 		await Promise.resolve(); | ||||
| 
 | ||||
| 		assert.htmlEqual( | ||||
| 			target.innerHTML, | ||||
| 			` | ||||
| 				<div class="item"> | ||||
| 					b <label><input name="current" type="radio" value="b"> current</label> | ||||
| 				</div> | ||||
| 				<div class="item"> | ||||
| 					c <label><input name="current" type="radio" value="c"> current</label> | ||||
| 				</div> | ||||
| 				<div class="item"> | ||||
| 					a <label><input name="current" type="radio" value="a"> current</label> | ||||
| 				</div> | ||||
| 			` | ||||
| 		); | ||||
| 
 | ||||
| 		// after shifting order, should still keep the correct radio checked
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		assert.equal(inputs[0].checked, false); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, true); | ||||
| 
 | ||||
| 		(component.current = 'b'); | ||||
| 		await Promise.resolve(); | ||||
| 
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 
 | ||||
| 		component.moveDown(1); | ||||
| 		await Promise.resolve(); | ||||
| 
 | ||||
| 		// after shifting order, should still keep the correct radio checked
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 	} | ||||
| }; | ||||
| @ -0,0 +1,36 @@ | ||||
| <script> | ||||
| 	export let list = [ | ||||
| 		{ name: "a", text: "This is a test." }, | ||||
| 		{ name: "b", text: "This is another test." }, | ||||
| 		{ name: "c", text: "This is also a test." }, | ||||
| 	]; | ||||
| 	export let current = "a"; | ||||
| 	export function moveUp(i) { | ||||
| 		list = [ | ||||
| 			...list.slice(0, Math.max(i - 1, 0)), | ||||
| 			list[i], | ||||
| 			list[i - 1], | ||||
| 			...list.slice(i + 1), | ||||
| 		]; | ||||
| 	} | ||||
| 	export function moveDown(i) { | ||||
| 		moveUp(i + 1); | ||||
| 	} | ||||
| </script> | ||||
| 
 | ||||
| {#each list as item (item.name)} | ||||
| 	<div class="item"> | ||||
| 		{item.name} | ||||
| 		{#if true} | ||||
| 			<label | ||||
| 				><input | ||||
| 					type="radio" | ||||
| 					name="current" | ||||
| 					bind:group={current} | ||||
| 					value={item.name} | ||||
| 				/> current</label | ||||
| 			> | ||||
| 		{/if} | ||||
| 	</div> | ||||
| {/each} | ||||
| 
 | ||||
| @ -0,0 +1,88 @@ | ||||
| // https://github.com/sveltejs/svelte/issues/6112
 | ||||
| export default { | ||||
| 	async test({ assert, target, component, window }) { | ||||
| 		let inputs = target.querySelectorAll('input'); | ||||
| 
 | ||||
| 		const check = (set) => { | ||||
| 			for (let i = 0; i < inputs.length; i++) { | ||||
| 				assert.equal(inputs[i].checked, set.has(i)); | ||||
| 			} | ||||
| 		}; | ||||
| 
 | ||||
| 		assert.htmlEqual( | ||||
| 			target.innerHTML, | ||||
| 			` | ||||
| 				<div>1</div> | ||||
| 				<div>2  | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="a"> | ||||
| 						<input type="radio" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="c"> | ||||
| 						<input type="radio" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 				<div>3  | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="a"> | ||||
| 						<input type="radio" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="c"> | ||||
| 						<input type="radio" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			` | ||||
| 		); | ||||
| 
 | ||||
| 		check(new Set([0, 2, 5, 6])); | ||||
| 
 | ||||
| 		const event = new window.Event('change'); | ||||
| 
 | ||||
| 		// dom to value
 | ||||
| 		inputs[3].checked = true; | ||||
| 		await inputs[3].dispatchEvent(event); | ||||
| 
 | ||||
| 		check(new Set([0, 3, 5, 6])); | ||||
| 		assert.equal(component.pipelineOperations[1].operation.args[1].value, 'd'); | ||||
| 
 | ||||
| 		// remove item
 | ||||
| 		component.pipelineOperations = component.pipelineOperations.slice(1); | ||||
| 		await Promise.resolve(); | ||||
| 
 | ||||
| 		assert.htmlEqual( | ||||
| 			target.innerHTML, | ||||
| 			` | ||||
| 				<div>2  | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="a"> | ||||
| 						<input type="radio" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="c"> | ||||
| 						<input type="radio" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 				<div>3  | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="a"> | ||||
| 						<input type="radio" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="radio" value="c"> | ||||
| 						<input type="radio" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			` | ||||
| 		); | ||||
| 
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		check(new Set([0, 3, 5, 6])); | ||||
| 
 | ||||
| 		inputs[2].checked = true; | ||||
| 		await inputs[2].dispatchEvent(event); | ||||
| 		 | ||||
| 		check(new Set([0, 2, 5, 6])); | ||||
| 	} | ||||
| }; | ||||
| @ -0,0 +1,60 @@ | ||||
| <script> | ||||
| 	export let pipelineOperations = [ | ||||
| 		{ | ||||
| 			operation: { | ||||
| 				name: "foo", | ||||
| 				args: [], | ||||
| 			}, | ||||
| 			id: 1, | ||||
| 		}, | ||||
| 		{ | ||||
| 			operation: { | ||||
| 				name: "bar", | ||||
| 				args: [ | ||||
| 					{ | ||||
| 						name: "bar_1", | ||||
| 						value: "a", | ||||
| 						options: [{ value: "a" }, { value: "b" }], | ||||
| 					}, | ||||
| 					{ | ||||
| 						name: "bar_2", | ||||
| 						value: "c", | ||||
| 						options: [{ value: "c" }, { value: "d" }], | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 			id: 2, | ||||
| 		}, | ||||
| 		{ | ||||
| 			operation: { | ||||
| 				name: "baz", | ||||
| 				args: [ | ||||
| 					{ | ||||
| 						name: "baz_1", | ||||
| 						value: "b", | ||||
| 						options: [{ value: "a" }, { value: "b" }], | ||||
| 					}, | ||||
| 					{ | ||||
| 						name: "baz_2", | ||||
| 						value: "c", | ||||
| 						options: [{ value: "c" }, { value: "d" }], | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 			id: 3, | ||||
| 		}, | ||||
| 	]; | ||||
| </script> | ||||
| 
 | ||||
| {#each pipelineOperations as { operation, id } (id)} | ||||
| 	<div> | ||||
| 		{id} | ||||
| 		{#each operation.args as arg} | ||||
| 			<div class="arg"> | ||||
| 				{#each arg.options as { value }} | ||||
| 					<input type="radio" bind:group={arg.value} {value} /> | ||||
| 				{/each} | ||||
| 			</div> | ||||
| 		{/each} | ||||
| 	</div> | ||||
| {/each} | ||||
| @ -0,0 +1,89 @@ | ||||
| // https://github.com/sveltejs/svelte/issues/6112
 | ||||
| export default { | ||||
| 	async test({ assert, target, component, window }) { | ||||
| 		let inputs = target.querySelectorAll('input'); | ||||
| 
 | ||||
| 		const check = (set) => { | ||||
| 			for (let i = 0; i < inputs.length; i++) { | ||||
| 				assert.equal(inputs[i].checked, set.has(i)); | ||||
| 			} | ||||
| 		}; | ||||
| 
 | ||||
| 		assert.htmlEqual( | ||||
| 			target.innerHTML, | ||||
| 			` | ||||
| 				<div>1</div> | ||||
| 				<div>2  | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="a"> | ||||
| 						<input type="checkbox" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="c"> | ||||
| 						<input type="checkbox" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 				<div>3  | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="a"> | ||||
| 						<input type="checkbox" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="c"> | ||||
| 						<input type="checkbox" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			` | ||||
| 		); | ||||
| 
 | ||||
| 		check(new Set([0, 2])); | ||||
| 
 | ||||
| 		const event = new window.Event('change'); | ||||
| 
 | ||||
| 		// dom to value
 | ||||
| 		inputs[3].checked = true; | ||||
| 		await inputs[3].dispatchEvent(event); | ||||
| 
 | ||||
| 		check(new Set([0, 2, 3])); | ||||
| 		assert.deepEqual(component.pipelineOperations[1].operation.args[1].value, ['c', 'd']); | ||||
| 
 | ||||
| 		// remove item
 | ||||
| 		component.pipelineOperations = component.pipelineOperations.slice(1); | ||||
| 		await Promise.resolve(); | ||||
| 
 | ||||
| 		assert.htmlEqual( | ||||
| 			target.innerHTML, | ||||
| 			` | ||||
| 				<div>2  | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="a"> | ||||
| 						<input type="checkbox" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="c"> | ||||
| 						<input type="checkbox" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 				<div>3  | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="a"> | ||||
| 						<input type="checkbox" value="b"> | ||||
| 					</div> | ||||
| 					<div class="arg"> | ||||
| 						<input type="checkbox" value="c"> | ||||
| 						<input type="checkbox" value="d"> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			` | ||||
| 		); | ||||
| 
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		check(new Set([0, 2, 3])); | ||||
| 
 | ||||
| 		inputs[5].checked = true; | ||||
| 		await inputs[5].dispatchEvent(event); | ||||
| 		 | ||||
| 		check(new Set([0, 2, 3, 5])); | ||||
| 		assert.deepEqual(component.pipelineOperations[1].operation.args[0].value, ['b']); | ||||
| 	} | ||||
| }; | ||||
| @ -0,0 +1,60 @@ | ||||
| <script> | ||||
| 	export let pipelineOperations = [ | ||||
| 		{ | ||||
| 			operation: { | ||||
| 				name: "foo", | ||||
| 				args: [], | ||||
| 			}, | ||||
| 			id: 1, | ||||
| 		}, | ||||
| 		{ | ||||
| 			operation: { | ||||
| 				name: "bar", | ||||
| 				args: [ | ||||
| 					{ | ||||
| 						name: "bar_1", | ||||
| 						value: ["a"], | ||||
| 						options: [{ value: "a" }, { value: "b" }], | ||||
| 					}, | ||||
| 					{ | ||||
| 						name: "bar_2", | ||||
| 						value: ["c"], | ||||
| 						options: [{ value: "c" }, { value: "d" }], | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 			id: 2, | ||||
| 		}, | ||||
| 		{ | ||||
| 			operation: { | ||||
| 				name: "baz", | ||||
| 				args: [ | ||||
| 					{ | ||||
| 						name: "baz_1", | ||||
| 						value: [], | ||||
| 						options: [{ value: "a" }, { value: "b" }], | ||||
| 					}, | ||||
| 					{ | ||||
| 						name: "baz_2", | ||||
| 						value: [], | ||||
| 						options: [{ value: "c" }, { value: "d" }], | ||||
| 					}, | ||||
| 				], | ||||
| 			}, | ||||
| 			id: 3, | ||||
| 		}, | ||||
| 	]; | ||||
| </script> | ||||
| 
 | ||||
| {#each pipelineOperations as { operation, id } (id)} | ||||
| 	<div> | ||||
| 		{id} | ||||
| 		{#each operation.args as arg} | ||||
| 			<div class="arg"> | ||||
| 				{#each arg.options as { value }} | ||||
| 					<input type="checkbox" bind:group={arg.value} {value} /> | ||||
| 				{/each} | ||||
| 			</div> | ||||
| 		{/each} | ||||
| 	</div> | ||||
| {/each} | ||||
| @ -0,0 +1,30 @@ | ||||
| export default { | ||||
| 	async test({ assert, target, window }) { | ||||
| 		const [input1, input2] = target.querySelectorAll('input[type=text]'); | ||||
| 		const radio = target.querySelector('input[type=radio]'); | ||||
| 
 | ||||
| 		assert.equal(radio.checked, false); | ||||
| 		 | ||||
| 		const event = new window.Event('input'); | ||||
| 		 | ||||
| 		input1.value = 'world'; | ||||
| 		await input1.dispatchEvent(event); | ||||
| 		assert.equal(radio.checked, true); | ||||
| 
 | ||||
| 		input2.value = 'foo'; | ||||
| 		await input2.dispatchEvent(event); | ||||
| 		assert.equal(radio.checked, false); | ||||
| 		 | ||||
| 		input1.value = 'foo'; | ||||
| 		await input1.dispatchEvent(event); | ||||
| 		assert.equal(radio.checked, true); | ||||
| 		 | ||||
| 		input1.value = 'bar'; | ||||
| 		await input1.dispatchEvent(event); | ||||
| 		assert.equal(radio.checked, false); | ||||
| 		 | ||||
| 		input2.value = 'bar'; | ||||
| 		await input2.dispatchEvent(event); | ||||
| 		assert.equal(radio.checked, true); | ||||
| 	} | ||||
| }; | ||||
| @ -0,0 +1,10 @@ | ||||
| <script> | ||||
| 	let name = 'world'; | ||||
| 	let current = ''; | ||||
| </script> | ||||
| 
 | ||||
| <input type="radio" name="current" bind:group={current} value={name}> | ||||
| 
 | ||||
| <input type="text" bind:value={current} /> | ||||
| 
 | ||||
| <input type="text" bind:value={name} /> | ||||
| @ -0,0 +1,78 @@ | ||||
| // https://github.com/sveltejs/svelte/issues/7884
 | ||||
| export default { | ||||
| 	async test({ assert, target, component, window }) { | ||||
| 		let inputs = target.querySelectorAll('input'); | ||||
| 
 | ||||
| 		assert.htmlEqual(target.innerHTML, ` | ||||
| 			<p>{"foo":[],"bar":[]}</p> | ||||
| 			<h2>foo</h2> | ||||
| 			<ul> | ||||
| 				<li><label><input name="foo" type="checkbox" value="1"> 1</label></li> | ||||
| 				<li><label><input name="foo" type="checkbox" value="2"> 2</label></li> | ||||
| 				<li><label><input name="foo" type="checkbox" value="3"> 3</label></li> | ||||
| 			</ul> | ||||
| 			<h2>bar</h2> | ||||
| 			<ul> | ||||
| 				<li><label><input name="bar" type="checkbox" value="1"> 1</label></li> | ||||
| 				<li><label><input name="bar" type="checkbox" value="2"> 2</label></li> | ||||
| 				<li><label><input name="bar" type="checkbox" value="3"> 3</label></li> | ||||
| 			</ul> | ||||
| 		`);
 | ||||
| 
 | ||||
| 		const event = new window.Event('change'); | ||||
| 
 | ||||
| 		inputs[0].checked = true; | ||||
| 		await inputs[0].dispatchEvent(event); | ||||
| 		inputs[2].checked = true; | ||||
| 		await inputs[2].dispatchEvent(event); | ||||
| 		inputs[3].checked = true; | ||||
| 		await inputs[3].dispatchEvent(event); | ||||
| 
 | ||||
| 		assert.htmlEqual(target.innerHTML, ` | ||||
| 			<p>{"foo":[1,3],"bar":[1]}</p> | ||||
| 			<h2>foo</h2> | ||||
| 			<ul> | ||||
| 				<li><label><input name="foo" type="checkbox" value="1"> 1</label></li> | ||||
| 				<li><label><input name="foo" type="checkbox" value="2"> 2</label></li> | ||||
| 				<li><label><input name="foo" type="checkbox" value="3"> 3</label></li> | ||||
| 			</ul> | ||||
| 			<h2>bar</h2> | ||||
| 			<ul> | ||||
| 				<li><label><input name="bar" type="checkbox" value="1"> 1</label></li> | ||||
| 				<li><label><input name="bar" type="checkbox" value="2"> 2</label></li> | ||||
| 				<li><label><input name="bar" type="checkbox" value="3"> 3</label></li> | ||||
| 			</ul>		 | ||||
| 		`);
 | ||||
| 		 | ||||
| 		await component.update(); | ||||
| 
 | ||||
| 		assert.htmlEqual(target.innerHTML, ` | ||||
| 			<p>{"foo":[1,3],"bar":[1],"qux":[]}</p> | ||||
| 			<h2>qux</h2> | ||||
| 			<ul> | ||||
| 				<li><label><input name="qux" type="checkbox" value="4"> 4</label></li> | ||||
| 				<li><label><input name="qux" type="checkbox" value="5"> 5</label></li> | ||||
| 				<li><label><input name="qux" type="checkbox" value="6"> 6</label></li> | ||||
| 			</ul> | ||||
| 		`);
 | ||||
| 
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		inputs[0].checked = true; | ||||
| 		await inputs[0].dispatchEvent(event); | ||||
| 
 | ||||
| 		assert.htmlEqual(target.innerHTML, ` | ||||
| 			<p>{"foo":[1,3],"bar":[1],"qux":[4]}</p> | ||||
| 			<h2>qux</h2> | ||||
| 			<ul> | ||||
| 				<li><label><input name="qux" type="checkbox" value="4"> 4</label></li> | ||||
| 				<li><label><input name="qux" type="checkbox" value="5"> 5</label></li> | ||||
| 				<li><label><input name="qux" type="checkbox" value="6"> 6</label></li> | ||||
| 			</ul> | ||||
| 		`);
 | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 	} | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| @ -0,0 +1,36 @@ | ||||
| <script> | ||||
| 	let keys = ["foo", "bar"]; | ||||
| 	let values = [1, 2, 3]; | ||||
| 
 | ||||
| 	const object = {}; | ||||
| 
 | ||||
| 	$: keys.forEach((key) => { | ||||
| 		// Make sure Svelte has an array to bind to | ||||
| 		if (!object[key]) { | ||||
| 			object[key] = []; | ||||
| 		} | ||||
| 	}); | ||||
| 
 | ||||
| 	export function update() { | ||||
| 		keys = ["qux"]; | ||||
| 		values = [4, 5, 6]; | ||||
| 	} | ||||
| </script> | ||||
| 
 | ||||
| <p> | ||||
| 	{JSON.stringify(object)} | ||||
| </p> | ||||
| 
 | ||||
| {#each keys as key (key)} | ||||
| 	<h2>{key}</h2> | ||||
| 	<ul> | ||||
| 		{#each values as value (value)} | ||||
| 			<li> | ||||
| 				<label> | ||||
| 					<input type="checkbox" name={key} {value} bind:group={object[key]} /> | ||||
| 					{value} | ||||
| 				</label> | ||||
| 			</li> | ||||
| 		{/each} | ||||
| 	</ul> | ||||
| {/each} | ||||
| @ -0,0 +1,49 @@ | ||||
| // https://github.com/sveltejs/svelte/issues/7633
 | ||||
| export default { | ||||
| 	async test({ assert, target, component, window }) { | ||||
| 		let inputs = target.querySelectorAll('input'); | ||||
| 
 | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 
 | ||||
| 		await component.moveDown(0); | ||||
| 		await component.moveDown(1); | ||||
| 		 | ||||
| 		assert.htmlEqual( | ||||
| 			target.innerHTML, | ||||
| 			` | ||||
| 				<div class="item"> | ||||
| 					b <label><input name="current" type="radio" value="b"> current</label> | ||||
| 				</div> | ||||
| 				<div class="item"> | ||||
| 					c <label><input name="current" type="radio" value="c"> current</label> | ||||
| 				</div> | ||||
| 				<div class="item"> | ||||
| 					a <label><input name="current" type="radio" value="a"> current</label> | ||||
| 				</div> | ||||
| 			` | ||||
| 		); | ||||
| 
 | ||||
| 		// after shifting order, should still keep the correct radio checked
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		assert.equal(inputs[0].checked, false); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, true); | ||||
| 
 | ||||
| 		await (component.current = 'b'); | ||||
| 		 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 
 | ||||
| 		await component.moveDown(1); | ||||
| 
 | ||||
| 		// after shifting order, should still keep the correct radio checked
 | ||||
| 		inputs = target.querySelectorAll('input'); | ||||
| 		assert.equal(inputs[0].checked, true); | ||||
| 		assert.equal(inputs[1].checked, false); | ||||
| 		assert.equal(inputs[2].checked, false); | ||||
| 	} | ||||
| }; | ||||
| @ -0,0 +1,36 @@ | ||||
| <script> | ||||
| 	export let list = [ | ||||
| 		{ name: "a", text: "This is a test." }, | ||||
| 		{ name: "b", text: "This is another test." }, | ||||
| 		{ name: "c", text: "This is also a test." }, | ||||
| 	]; | ||||
| 	export let current = "a"; | ||||
| 	export function moveUp(i) { | ||||
| 		list = [ | ||||
| 			...list.slice(0, Math.max(i - 1, 0)), | ||||
| 			list[i], | ||||
| 			list[i - 1], | ||||
| 			...list.slice(i + 1), | ||||
| 		]; | ||||
| 	} | ||||
| 	export function moveDown(i) { | ||||
| 		moveUp(i + 1); | ||||
| 	} | ||||
| </script> | ||||
| 
 | ||||
| {#each list as item} | ||||
| 	<div class="item"> | ||||
| 		{item.name} | ||||
| 		{#if true} | ||||
| 			<label | ||||
| 				><input | ||||
| 					type="radio" | ||||
| 					name="current" | ||||
| 					bind:group={current} | ||||
| 					value={item.name} | ||||
| 				/> current</label | ||||
| 			> | ||||
| 		{/if} | ||||
| 	</div> | ||||
| {/each} | ||||
| 
 | ||||
					Loading…
					
					
				
		Reference in new issue