首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >角度2:如何防止在按键输入时提交表格?

角度2:如何防止在按键输入时提交表格?
EN

Stack Overflow用户
提问于 2016-12-01 11:32:12
回答 3查看 21.1K关注 0票数 14

我有一个表单,其中一个字段作为自动完成。如果用户输入一个单词并按enter,则应将该字段的内容添加到该字段下面的列表中。

问题是:当用户点击enter时,自然会提交整个表单。

我已经对处理按键的函数进行了return false。但在调用此函数之前,表单似乎就已经提交了。

我怎样才能防止这种情况发生?

基本形式:

代码语言:javascript
复制
<div id="profileForm">
  <form [formGroup]="profileForm" (ngSubmit)="onSubmit()" method="post" *ngIf="!showSuccessMessage">

    <div class="row">

      <div class="form-group col-xs-12 col-sm-6">
        <label for="first_name">My Skills</label>
        <div class="autocomplete">
          <input formControlName="skill_string" [(ngModel)]="skillString" name="skill_string"
          type="text" class="form-control" id="skill_string" placeholder="Comma separated" (keyup.enter)="skillsHandleEnter(skillString)">
          <ul class="autocomplete-list" *ngIf="skillHints.length > 0">
            <li class="list-item" *ngFor="let skill of skillHints" (click)="addSkillFromAutocomplete(skill)">{{skill}}</li>
          </ul>
        </div>
        <div id="skill-cloud" class="tag-cloud">
          <span class="skill-tag tag label label-success" *ngFor="let skill of selectedSkills" (click)="removeSkill(skill)">{{skill}} x</span>
        </div>
      </div>

    </div>

    <div class="row">

      <hr>

      <div class="form-group submit-group">
        <div class="col-sm-12">
          <button type="submit" class="btn btn-primary pull-right" [disabled]="!profileForm.valid">Save</button>
        </div>
      </div>

    </div>

  </form>
</div>

基本组件(我在这里删除了很多发布它的逻辑):

代码语言:javascript
复制
import { Component, OnInit } from '@angular/core';
import { FormGroup, FormBuilder } from '@angular/forms';
import { ActivatedRoute, Router } from '@angular/router';

import { Subscription } from 'rxjs/Rx';
import 'rxjs/add/operator/debounceTime';
import * as _ from 'lodash';

import { MemberService } from '../shared/index';

@Component({
  moduleId: module.id,
  selector: 'signup',
  templateUrl: 'signup.component.html',
  styleUrls: ['signup.component.css']
})
export class SignupComponent implements OnInit {

  private profileForm:FormGroup;
  private validation_errors:Array<any>;
  private selectedSkills:Array<string>;
  private skillHints:Array<string>;
  private skillString:string;

  constructor(private route: ActivatedRoute,
    private formBuilder: FormBuilder,
    private memberService: MemberService,
    private router: Router ) {

      this.selectedSkills = [];
      this.skillHints = [];
      this.skillString = '';

      // Set up form
      this.profileForm = this.formBuilder.group({
        skill_string: ['']
      });

  }

  ngOnInit(): any {
    // Do something
  }



  updateSelectedSkills(skillString:string):void {
    if(skillString) ) {
      let cleanString = skillString.trim().replace(/[ ]{2,}/g, ' ');
      this.selectedSkills = _.compact(this.selectedSkills.concat(cleanString.split(',')));
      this.skillString = '';
      this.skillHints = [];
    }
  }

  skillsHandleEnter(skillString:string):void {
    console.log("ENTER");
    this.updateSelectedSkills(skillString);
    return false;
  }

  autocompleteSkills(term:string):void {
    this.memberService.autocompleteSkills(term).subscribe(
      res => {
        this.skillHints = [];
        for(let i = 0; i < res.data.length; i++) {
          this.skillHints.push(res.data[i].name);
        }
      }
    );
  }

  addSkillFromAutocomplete(skillString:string):void {
    this.selectedSkills.push(skillString);
    this.memberProfile.skill_string = '';
    this.skillHints = [];
    this.skillString = '';
  }


  onSubmit():void {
    this.memberService.saveProfile(this.memberProfile, this.selectedSkills).subscribe(
      res => {
        console.log(res);
      }
    );
  }

}
EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2016-12-01 11:57:57

试一试

代码语言:javascript
复制
<form (keydown.enter)="$event.target.tagName == 'TEXTAREA'" [formGroup]="profileForm" (ngSubmit)="onSubmit($event)">

它还将允许enterTextarea中使用。

票数 18
EN

Stack Overflow用户

发布于 2016-12-01 12:04:07

答案其实很简单..。这不是Event.preventDefault(),因为我正在监听输入字段中的Enter,而不是按钮。从type="submit"中删除button是不够的,因为默认情况下所有按钮都是类型提交。唯一必要的更改是对按钮元素进行更改,显式地添加type="button"并添加(click)侦听器:

代码语言:javascript
复制
<button type="button" (click)="onSubmit()" class="btn btn-primary pull-right" [disabled]="!profileForm.valid">Save</button>

唯一的问题是:现在提交带有enter的表单是行不通的。如果焦点在“自动完成输入”字段中,只会防止enter提交表单,就会稍微优雅一些。

编辑:

只有在游标位于“自动完成”字段中时,才能阻止enter提交表单,方法是使用Ankit Singh的解决方案并对其进行一些修改:

代码语言:javascript
复制
<form [formGroup]="profileForm" (ngSubmit)="onSubmit()" method="post" (keydown.enter)="$event.target.id != 'skill_string'" *ngIf="!showSuccessMessage">

(注:条件必须返回false以防止触发默认操作)

当然,我们还需要我们的常规提交按钮(不附加单击事件,否则表单将提交两次):

代码语言:javascript
复制
<button type="submit" class="btn btn-primary pull-right" [disabled]="!profileForm.valid">Save</button>

如果要使用event.target.classList类,也可以检查.autocomplete。或者将检查逻辑移到传递$event的函数中。

票数 7
EN

Stack Overflow用户

发布于 2016-12-01 11:40:58

角2中的事件表现为正常的DOM事件。要捕获事件对象,在模板的事件回调中将$event作为参数传递:

Html:

代码语言:javascript
复制
<button (keyup.enter)="skillsHandleEnter($event, skillString)"></button>

使用JavaScript的Event.preventDefault()

代码语言:javascript
复制
@Component(...)
class MyComponent {
  skillsHandleEnter(event, skillString) {
    event.preventDefault();
    // ... your logic
  }
}
票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/40909585

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档