diff --git a/conda/js/src/components/experiment/ExperimentEditor.jsx b/conda/js/src/components/experiment/ExperimentEditor.jsx
index 78f3e502588f2e4468686f34b64f534d82757b90..acc55144d1bbae5b3b94671868d37af4f5c9e5f8 100644
--- a/conda/js/src/components/experiment/ExperimentEditor.jsx
+++ b/conda/js/src/components/experiment/ExperimentEditor.jsx
@@ -4,7 +4,7 @@ import {
 	Container,
 	Row,
 	Col,
-	Button,
+	Button, ButtonGroup,
 	Form,
 	FormGroup,
 	Label,
@@ -141,14 +141,15 @@ const isValidEntity = (
 	possibleObj: {inputs: any, outputs: any} = {inputs:{},outputs:{}},
 	inferredTypes = {}
 ): boolean => {
-	if(!blockTcEntity.hasOwnProperty('inputs') && Object.keys(possibleObj.inputs).length > 0)
+	const hasInputs = Array.isArray(blockTcEntity.inputs);
+	const hasOutputs = Array.isArray(blockTcEntity.outputs);
+	if(!hasInputs && Object.keys(possibleObj.inputs).length > 0)
 		return false;
-	if(!blockTcEntity.hasOwnProperty('outputs') && Object.keys(possibleObj.outputs).length > 0)
+	if(!hasOutputs && Object.keys(possibleObj.outputs).length > 0)
 		return false;
-	if(blockTcEntity.hasOwnProperty('inputs') && blockTcEntity.inputs.length !== Object.keys(possibleObj.inputs).length)
+	if(hasInputs && blockTcEntity.inputs.length !== Object.keys(possibleObj.inputs).length)
 		return false;
-	if(blockTcEntity.hasOwnProperty('inputs') &&
-		blockTcEntity.hasOwnProperty('outputs') &&
+	if(hasInputs && hasOutputs &&
 		blockTcEntity.outputs.length !== Object.keys(possibleObj.outputs).length
 	)
 		return false;
@@ -199,6 +200,9 @@ const levMapStrings = (keyArr: string[], valArr: string[]): { [string]: string }
 	const vals = [...valArr];
 	const map = {};
 
+	//console.log(keys);
+	//console.log(vals);
+
 	while(keys.length > 0){
 		const scores = {};
 		const matches = {};
@@ -226,6 +230,7 @@ const levMapStrings = (keyArr: string[], valArr: string[]): { [string]: string }
 		map[chosenKey] = matches[chosenKey];
 		keys.splice(keys.indexOf(chosenKey), 1);
 		vals.splice(vals.indexOf(matches[chosenKey]), 1);
+		//console.log(map);
 	}
 
 	const orderedMap = {};
@@ -233,6 +238,7 @@ const levMapStrings = (keyArr: string[], valArr: string[]): { [string]: string }
 		orderedMap[k] = map[k];
 	}
 
+	//console.log(orderedMap);
 	return orderedMap;
 };
 
@@ -586,13 +592,14 @@ export class ExperimentEditor extends React.Component<Props, State> {
 					onChange={e => {
 						const str = e.target.value;
 
-						//Object.keys(this.props.data.contents.datasets).map(dsName
 						const newDs = JSON.parse(str);
 						//console.log(newDs);
 						this.setContents({...this.props.data.contents, datasets: newDs});
+						/*
 						for(const dataset in this.props.data.contents.datasets){
 							this.setLockMap(dataset, true);
 						}
+						*/
 					}}
 				>
 					<option value=''>Protocol...</option>
@@ -625,7 +632,17 @@ export class ExperimentEditor extends React.Component<Props, State> {
 								const tcDs = tcDss.pop();
 								let setsIdx = sets.findIndex(set => set.set === tcDs.name && dbSetIsValidForTcDataset(set, tcDs));
 								if(setsIdx === -1){
-									setsIdx = sets.findIndex(set => dbSetIsValidForTcDataset(set, tcDs));
+									const validSets = sets.filter(set => dbSetIsValidForTcDataset(set, tcDs));
+									let bestScore = 10000;
+									let bestMatch = 'notfound';
+									for(const s of validSets){
+										const score = lev.get(tcDs.name, s.set);
+										if(score < bestScore){
+											bestScore = score;
+											bestMatch = s;
+										}
+									}
+									setsIdx = sets.findIndex(set => set == bestMatch);
 								}
 								if(setsIdx === -1)
 									return [dbProtStr, false];
@@ -692,7 +709,7 @@ export class ExperimentEditor extends React.Component<Props, State> {
 
 										const newDs = {...this.props.data.contents.datasets, [name]: ds};
 										this.setContents({...this.props.data.contents, datasets: newDs});
-										this.setLockMap(name, true);
+										//this.setLockMap(name, true);
 									}}
 								>
 									<option value=''>Dataset...</option>
@@ -765,6 +782,17 @@ export class ExperimentEditor extends React.Component<Props, State> {
 		const queue = block.queue || this.props.data.contents.globals.queue;
 		const envDisabled = !block.hasOwnProperty('environment');
 
+		const getClearedGlobals = () => {
+			const globals = copyObj(this.props.data.contents.globals);
+			// if the old algorithm had parameters and no other block is using that alg,
+			// delete the global param defaults for the old alg
+			if(globals.hasOwnProperty(block.algorithm) &&
+				[...this.props.toolchain.contents.analyzers, ...this.props.toolchain.contents.blocks]
+				.filter(b => b.algorithm === block.algorithm).length <= 1)
+				delete globals[block.algorithm];
+			return globals;
+		};
+
 		return (
 			<FormGroup key={key} className={`block${ key } block_${ blockName }`}>
 				<h4>
@@ -874,7 +902,8 @@ export class ExperimentEditor extends React.Component<Props, State> {
 										thisBlock.outputs = levMapStrings(Object.keys(outputs), tcBlock.outputs);
 
 									// setup the parameters (erase & create stuff)
-									const globals = copyObj(this.props.data.contents.globals);
+									const globals = getClearedGlobals();
+
 									// if the new alg has parameters, gen the global defaults for the params
 									if(alg.contents.parameters && Object.keys(alg.contents.parameters).length > 0)
 										globals[alg.name] = {
@@ -884,15 +913,8 @@ export class ExperimentEditor extends React.Component<Props, State> {
 											...this.props.data.contents.globals[alg.name],
 										};
 
-									// if the old algorithm had parameters and no other block is using that alg,
-									// delete the global param defaults for the old alg
-									if(globals.hasOwnProperty(block.algorithm) &&
-										[...this.props.toolchain.contents.analyzers, ...this.props.toolchain.contents.blocks]
-										.filter(b => b.algorithm === block.algorithm).length <= 1)
-										delete globals[block.algorithm];
-
 									updateBlock(thisBlock, globals);
-									this.setLockMap(blockName, true);
+									//this.setLockMap(blockName, true);
 								}}
 							>
 								<option value=''>Algorithm...</option>
@@ -905,44 +927,66 @@ export class ExperimentEditor extends React.Component<Props, State> {
 								}
 							</Input>
 						</Col>
-						{ !isAnalyzer &&
-								<Col sm='auto'>
-									<Button
-										color='primary'
-										disabled={block.algorithm === ''}
-										title={`Copies the algorithm & IO mappings to all unconfigured blocks with the same inputs & outputs`}
-										onClick={() => {
-											// copy the algorithm and input/output mappings to blocks
-											// that are the same except for the block name and dont have an algorithm assigned already
-											const targetBlocks = this.props.toolchain.contents.blocks
-											// not the block being copied from
-											.filter(b => b.name !== blockName)
-											// same IO
-											.filter(b => JSON.stringify(b.inputs) === JSON.stringify(tcBlock.inputs))
-											.filter(b => JSON.stringify(b.outputs) === JSON.stringify(tcBlock.outputs))
-											.map(b => b.name)
-											;
-											//console.log(targetBlocks);
-
-											// because the target blocks have the exact same inputs/outputs as the current block
-											// we dont need to re-compute the block object that we will copy!
-											// instead, just reuse the current block object and assign it to all
-											// the target blocks.
-											// also, because all these blocks are the same algorithm as the current block
-											// we dont need to mess with the globals.
-
-											const newBlocks = copyObj(this.props.data.contents.blocks);
-											targetBlocks.forEach(bName => { newBlocks[bName] = copyObj(block); });
-											this.setContents({
-												...this.props.data.contents,
-												blocks: newBlocks,
-											});
-										}}
-									>
-										Copy to similar blocks
-									</Button>
-								</Col>
-						}
+						<Col sm='auto'>
+							{ !isAnalyzer &&
+									<React.Fragment>
+										<Button
+											color='primary'
+											disabled={block.algorithm === ''}
+											title={`Copies the algorithm & IO mappings to all unconfigured blocks with the same inputs & outputs`}
+											onClick={() => {
+												// copy the algorithm and input/output mappings to blocks
+												// that are the same except for the block name and dont have an algorithm assigned already
+												const targetBlocks = this.props.toolchain.contents.blocks
+												// not the block being copied from
+												.filter(b => b.name !== blockName)
+												// same IO
+												.filter(b => JSON.stringify(b.inputs) === JSON.stringify(tcBlock.inputs))
+												.filter(b => JSON.stringify(b.outputs) === JSON.stringify(tcBlock.outputs))
+												.map(b => b.name)
+												;
+												//console.log(targetBlocks);
+
+												// because the target blocks have the exact same inputs/outputs as the current block
+												// we dont need to re-compute the block object that we will copy!
+												// instead, just reuse the current block object and assign it to all
+												// the target blocks.
+												// also, because all these blocks are the same algorithm as the current block
+												// we dont need to mess with the globals.
+
+												const newBlocks = copyObj(this.props.data.contents.blocks);
+												targetBlocks.forEach(bName => { newBlocks[bName] = copyObj(block); });
+												this.setContents({
+													...this.props.data.contents,
+													blocks: newBlocks,
+												});
+											}}
+										>
+											Copy to similar blocks
+										</Button>
+										{' '}
+									</React.Fragment>
+							}
+							<Button
+								color='secondary'
+								title={`Clears the block data, resetting the selected block.`}
+								onClick={() => {
+									const newBlock = copyObj(block);
+									const newGlobals = getClearedGlobals();
+									newBlock.algorithm = '';
+									if(newBlock.inputs)
+										newBlock.inputs = {};
+									if(newBlock.outputs)
+										newBlock.outputs = {};
+									if(newBlock.parameters)
+										newBlock.parameters = {};
+
+									updateBlock(newBlock, newGlobals);
+								}}
+							>
+								Reset
+							</Button>
+						</Col>
 					</FormGroup>
 				</FormGroup>
 				{ Object.entries(alg.contents.parameters || {}).length > 0 && <h5>Parameters</h5> }
@@ -1305,9 +1349,9 @@ export class ExperimentEditor extends React.Component<Props, State> {
 						{ this.state.disableTypeInference ? 'Enable' : 'Disable' } Type Inference
 					</Button>
 				</FormGroup>
-				{ this.renderBlocks() }
-				{ this.renderAnalyzers() }
-				{ this.renderDatasets() }
+				{ this.state.activeBlockInfo.set === 'blocks' && this.renderBlocks() }
+				{ this.state.activeBlockInfo.set === 'analyzers' && this.renderAnalyzers() }
+				{ this.state.activeBlockInfo.set === 'datasets' && this.renderDatasets() }
 				{ this.renderGlobals() }
 			</Form>
 		</div>
diff --git a/conda/js/src/components/experiment/ExperimentEditor.spec.jsx b/conda/js/src/components/experiment/ExperimentEditor.spec.jsx
index 71feeed1c40d5cedff0b0f4211b814eabb2da511..017ee392ddfbcb7caab93b133df3df2c3334fce9 100644
--- a/conda/js/src/components/experiment/ExperimentEditor.spec.jsx
+++ b/conda/js/src/components/experiment/ExperimentEditor.spec.jsx
@@ -98,6 +98,7 @@ describe('<ExperimentEditor />', () => {
 
 			//console.log('finished name change, doing dataset');
 
+			wrapper.find('svg #block_set').simulate('click');
 			wrapper.find('div.dataset0 select').simulate('change', { target: { value: 'protocol/set (simple/1)'}});
 			expect(updateFunc.callCount).to.equal(1);
 			expect(wrapper.props().data.contents).to.have.deep.property('datasets', {
@@ -237,18 +238,19 @@ describe('<ExperimentEditor />', () => {
 
 			//console.log('doing dataset');
 
-			wrapper.find('div.datasets select').at(0).simulate('change', { target: { value: '{"testing_data":{"database":"iris/1","protocol":"Main","set":"training"},"training_data":{"database":"iris/1","protocol":"Main","set":"testing"}}'}});
+			wrapper.find('svg #block_training_data').simulate('click');
+			wrapper.find('div.datasets select').at(0).simulate('change', { target: { value: '{"testing_data":{"database":"iris/1","protocol":"Main","set":"testing"},"training_data":{"database":"iris/1","protocol":"Main","set":"training"}}'}});
 			expect(updateFunc.callCount).to.equal(1);
 			expect(wrapper.props().data.contents).to.have.deep.property('datasets', {
 				'testing_data': {
 					'database': 'iris/1',
 					'protocol': 'Main',
-					'set': 'training'
+					'set': 'testing'
 				},
 				'training_data': {
 					'database': 'iris/1',
 					'protocol': 'Main',
-					'set': 'testing'
+					'set': 'training'
 				}
 			});
 
@@ -490,12 +492,12 @@ describe('<ExperimentEditor />', () => {
 						'testing_data': {
 							'database': 'iris/1',
 							'protocol': 'Main',
-							'set': 'training'
+							'set': 'testing'
 						},
 						'training_data': {
 							'database': 'iris/1',
 							'protocol': 'Main',
-							'set': 'testing'
+							'set': 'training'
 						}
 					},
 					'globals': {
@@ -509,4 +511,184 @@ describe('<ExperimentEditor />', () => {
 			});
 		});
 	});
+
+	describe('Regression Tests', () => {
+		it(`clears the 'echo' & 'analysis' blocks from user/user/single/1/single_add`, () => {
+			const expName = 'user/user/single/1/single_add';
+			const saveFunc = sinon.spy();
+			const _updateFunc = (obj) => {
+				wrapper.setProps && wrapper.setProps({ data: obj });
+			};
+			const updateFunc = sinon.spy(_updateFunc);
+			const exp = testExps.find(exp => expName === exp.name);
+			const tc = testTcs.find(tc => expName.includes(tc.name));
+			wrapper = mount(
+				<C
+					data={getValidObj(exp, tc, [...normalBlocks, ...analyzerBlocks])}
+					experiments={[]}
+					normalBlocks={normalBlocks}
+					analyzerBlocks={analyzerBlocks}
+					datasets={datasets}
+					toolchain={tc}
+					saveFunc={saveFunc}
+					environments={envs}
+					updateFunc={updateFunc}
+				/>
+			);
+
+			expect(wrapper.props().data).to.have.property('name', expName);
+			expect(wrapper.props().data.contents).to.have.deep.property('blocks', {
+				'echo': {
+					'inputs': {
+						'in_data': 'in'
+					},
+					'algorithm': 'user/integers_add/1',
+					'outputs': {
+						'out_data': 'out'
+					},
+					parameters: {}
+				}
+			});
+
+			wrapper.find('svg #block_echo').simulate('click');
+			wrapper.update();
+			wrapper.find('.algorithm button.btn-secondary').simulate('click');
+			expect(updateFunc.callCount).to.equal(1);
+			expect(wrapper.props().data.contents).to.have.deep.property('blocks',
+				{ echo: {
+					inputs: {},
+					algorithm: '',
+					outputs: {},
+					parameters: {}
+				}}
+			);
+
+			wrapper.find('svg #block_analysis').simulate('click');
+			wrapper.update();
+			wrapper.find('.algorithm button.btn-secondary').simulate('click');
+			expect(updateFunc.callCount).to.equal(2);
+			expect(wrapper.props().data.contents).to.have.deep.property('analyzers',
+				{ analysis: {
+					inputs: {},
+					algorithm: '',
+					parameters: {}
+				}}
+			);
+
+			expect(wrapper.props().data).to.be.deep.equal({
+				'name': 'user/user/single/1/single_add',
+				'contents': {
+					description: '',
+					'analyzers': {
+						'analysis': {
+							inputs: {},
+							algorithm: '',
+							parameters: {}
+						}
+					},
+					'datasets': {
+						'set': {
+							'set': 'set',
+							'protocol': 'protocol',
+							'database': 'simple/1'
+						}
+					},
+					'blocks': {
+						'echo': {
+							algorithm: '',
+							inputs: {},
+							outputs: {},
+							parameters: {}
+						}
+					},
+					'globals': {
+						'queue': 'queue',
+						'environment': {
+							'version': '1.2.0',
+							'name': 'Python 2.7'
+						},
+					}
+				}
+			});
+		});
+
+		it(`doesnt change the lockMap after an algorithm or dataset is assigned`, () => {
+			const expName = 'user/user/single/1/single_add';
+			const saveFunc = sinon.spy();
+			const _updateFunc = (obj) => {
+				wrapper.setProps && wrapper.setProps({ data: obj });
+			};
+			const updateFunc = sinon.spy(_updateFunc);
+			const tc = testTcs.find(tc => expName.includes(tc.name));
+			wrapper = mount(
+				<C
+					data={getValidObj({name: expName, contents: {}}, tc, [...normalBlocks, ...analyzerBlocks])}
+					experiments={[]}
+					normalBlocks={normalBlocks}
+					analyzerBlocks={analyzerBlocks}
+					datasets={datasets}
+					toolchain={tc}
+					saveFunc={saveFunc}
+					environments={envs}
+					updateFunc={updateFunc}
+				/>
+			);
+
+			expect(wrapper.props().data).to.have.property('name', expName);
+
+			//console.log('finished name change, doing dataset');
+
+			wrapper.find('svg #block_set').simulate('click');
+			wrapper.find('div.dataset0 select').simulate('change', { target: { value: 'protocol/set (simple/1)'}});
+			expect(updateFunc.callCount).to.equal(1);
+			expect(wrapper.props().data.contents).to.have.deep.property('datasets', {
+				'set': {
+					'set': 'set',
+					'protocol': 'protocol',
+					'database': 'simple/1'
+				}
+			});
+
+			//console.log('finished dataset, doing block');
+
+			wrapper.find('svg #block_echo').simulate('click');
+			expect(wrapper.find({ name: 'echo', set: 'blocks'}).find('.tcBlockBackground').prop('className')).to.include('highlighted');
+			wrapper.find('div.block0 div.algorithm select').at(0).simulate('change', { target: { value: 'user/integers_add/1'}});
+			expect(updateFunc.callCount).to.equal(2);
+			expect(wrapper.props().data.contents).to.have.deep.property('blocks', {
+				'echo': {
+					'inputs': {
+						'in_data': 'in'
+					},
+					'algorithm': 'user/integers_add/1',
+					'outputs': {
+						'out_data': 'out'
+					},
+					parameters: {}
+				}
+			});
+
+			//console.log('finished block, doing analyzer');
+
+			wrapper.find('svg #block_analysis').simulate('click');
+			expect(wrapper.find({ name: 'analysis', set: 'analyzers'}).find('.tcBlockBackground').prop('className')).to.include('highlighted');
+			wrapper.find('div.block0 div.algorithm select').at(0).simulate('change', { target: { value: 'user/integers_echo_analyzer/1'}});
+			expect(updateFunc.callCount).to.equal(3);
+			expect(wrapper.props().data.contents).to.have.deep.property('analyzers', {
+				'analysis': {
+					'inputs': {
+						'in_data': 'in'
+					},
+					'algorithm': 'user/integers_echo_analyzer/1',
+					parameters: {},
+				}
+			});
+
+			expect(wrapper.state().lockMap).to.deep.equal({
+				set: false,
+				echo: false,
+				analysis: false,
+			});
+		});
+	});
 });