web-dev-qa-db-ja.com

MVCモデルにはtrueが必要

ブールプロパティをtrueに設定することを要求するデータアノテーションを通じて方法はありますか?

public class MyAwesomeObj{
    public bool ThisMustBeTrue{get;set;}
}
74
Marty Trenouth

独自のバリデータを作成できます。

public class IsTrueAttribute : ValidationAttribute
{
    #region Overrides of ValidationAttribute

    /// <summary>
    /// Determines whether the specified value of the object is valid. 
    /// </summary>
    /// <returns>
    /// true if the specified value is valid; otherwise, false. 
    /// </returns>
    /// <param name="value">The value of the specified validation object on which the <see cref="T:System.ComponentModel.DataAnnotations.ValidationAttribute"/> is declared.
    ///                 </param>
    public override bool IsValid(object value)
    {
        if (value == null) return false;
        if (value.GetType() != typeof(bool)) throw new InvalidOperationException("can only be used on boolean properties.");

        return (bool) value;
    }

    #endregion
}
47
HackedByChinese

サーバー側とクライアント側の両方にバリデーターを作成します。 MVCと控えめなフォーム検証を使用すると、次のことを行うだけでこれを実現できます。

まず、プロジェクトにクラスを作成して、次のようにサーバー側の検証を実行します。

public class EnforceTrueAttribute : ValidationAttribute, IClientValidatable
{
    public override bool IsValid(object value)
    {
        if (value == null) return false;
        if (value.GetType() != typeof(bool)) throw new InvalidOperationException("can only be used on boolean properties.");
        return (bool)value == true;
    }

    public override string FormatErrorMessage(string name)
    {
        return "The " + name + " field must be checked in order to continue.";
    }

    public IEnumerable<ModelClientValidationRule> GetClientValidationRules(ModelMetadata metadata, ControllerContext context)
    {
        yield return new ModelClientValidationRule
        {
            ErrorMessage = String.IsNullOrEmpty(ErrorMessage) ? FormatErrorMessage(metadata.DisplayName) : ErrorMessage,
            ValidationType = "enforcetrue"
        };
    }
}

これに続いて、モデル内の適切なプロパティに注釈を付けます。

[EnforceTrue(ErrorMessage=@"Error Message")]
public bool ThisMustBeTrue{ get; set; }

最後に、次のスクリプトをビューに追加して、クライアント側の検証を有効にします。

<script type="text/javascript">
    jQuery.validator.addMethod("enforcetrue", function (value, element, param) {
        return element.checked;
    });
    jQuery.validator.unobtrusive.adapters.addBool("enforcetrue");
</script>

注:モデルからビューに注釈をプッシュするメソッドGetClientValidationRulesを既に作成しました。

リソースファイルを使用して国際化のエラーメッセージを提供する場合は、FormatErrorMessage呼び出しを削除(または単にベースを呼び出し)し、次のようにGetClientValidationRulesメソッドを調整します。

public IEnumerable<ModelClientValidationRule> GetClientValidationRules(ModelMetadata metadata, ControllerContext context)
{
    string errorMessage = String.Empty;
    if(String.IsNullOrWhiteSpace(ErrorMessage))
    {
        // Check if they supplied an error message resource
        if(ErrorMessageResourceType != null && !String.IsNullOrWhiteSpace(ErrorMessageResourceName))
        {
            var resMan = new ResourceManager(ErrorMessageResourceType.FullName, ErrorMessageResourceType.Assembly);
            errorMessage = resMan.GetString(ErrorMessageResourceName);
        }
    }
    else
    {
        errorMessage = ErrorMessage;
    }

    yield return new ModelClientValidationRule
    {
        ErrorMessage = errorMessage,
        ValidationType = "enforcetrue"
    };
}
127
dazbradbury

私はこれが古い投稿であることを知っていますが、これを行う簡単なサーバー側の方法を共有したかったです。 trueに設定されたパブリックプロパティを作成し、boolとそのプロパティを比較します。 boolがチェックされていない場合(デフォルトではfalse)、フォームは検証されません。

public bool isTrue
{ get { return true; } }

[Required]
[Display(Name = "I agree to the terms and conditions")]
[Compare("isTrue", ErrorMessage = "Please agree to Terms and Conditions")]
public bool iAgree { get; set; }
80
fields.cage

いくつかのソリューションを試しましたが、クライアント側とサーバー側の両方の検証を取得するのに完全に機能するものはありませんでした。だから私はそれを動作させるために私のMVC 5アプリケーションでやったこと:

ViewModel内(サーバー側の検証用):

public bool IsTrue => true;

[Required]
[Display(Name = "I agree to the terms and conditions")]
[Compare(nameof(IsTrue), ErrorMessage = "Please agree to Terms and Conditions")]
public bool HasAcceptedTermsAndConditions { get; set; }

Razorページ(クライアント側の検証用):

<div class="form-group">
   @Html.CheckBoxFor(m => m.HasAcceptedTermsAndConditions)
   @Html.LabelFor(m => m.HasAcceptedTermsAndConditions)
   @Html.ValidationMessageFor(m => m.HasAcceptedTermsAndConditions)

   @Html.Hidden(nameof(Model.IsTrue), "true")
</div>
11
Kapé

次のフィドルに人々を誘導したいと思います: https://dotnetfiddle.net/JbPh0X

ユーザーは[Range(typeof(bool), "true", "true", ErrorMessage = "You gotta tick the box!")]をブールプロパティに追加し、サーバー側の検証を機能させました。

クライアント側の検証も機能させるため、次のスクリプトを追加しました。

// extend jquery range validator to work for required checkboxes
var defaultRangeValidator = $.validator.methods.range;
$.validator.methods.range = function(value, element, param) {
    if(element.type === 'checkbox') {
        // if it's a checkbox return true if it is checked
        return element.checked;
    } else {
        // otherwise run the default validation function
        return defaultRangeValidator.call(this, value, element, param);
    }
}
8
Harvey

文字列表現がTrueと等しいかどうかを確認するだけです:

[RegularExpression("True")]
public bool TermsAndConditions { get; set; }
6
ta.speot.is

独自の属性を作成するか、 CustomValidationAttribute を使用できます。

これは、CustomValidationAttributeの使用方法です。

[CustomValidation(typeof(BoolValidation), "ValidateBool")]

boolValidationは次のように定義されます。

public class BoolValidation
{
  public static ValidationResult ValidateBool(bool boolToBeTrue)
  {
    if (boolToBeTrue)
    {
      return ValidationResult.Success;
    }
    else
    {
      return new ValidationResult(
          "Bool must be true.");
    }
  }
5
Matthew Manela

[Required]属性はany valueを要求することを意味します-trueまたはfalseのいずれかです。そのためには別の検証を使用する必要があります。

5

ASP.NET Core MVCの場合は、dazbradburyのソリューションに基づいたクライアントとサーバーの検証です

public class EnforceTrueAttribute : ValidationAttribute, IClientModelValidator
{
    public override bool IsValid(object value)
    {
        if (value == null) return false;
        if (value.GetType() != typeof(bool)) throw new InvalidOperationException("can only be used on boolean properties.");
        return (bool)value;
    }

    public void AddValidation(ClientModelValidationContext context)
    {
        MergeAttribute(context.Attributes, "data-val", "true");
        var errorMessage = ErrorMessage ?? 
            $"The value for field {context.ModelMetadata.GetDisplayName()} must be true.";
        MergeAttribute(context.Attributes, "data-val-enforcetrue", errorMessage);
    }

    private void MergeAttribute(IDictionary<string, string> attributes,
        string key,
        string value)
    {
        if (attributes.ContainsKey(key))
        {
            return;
        }
        attributes.Add(key, value);
    }
}

そして、クライアント上で:

$.validator.addMethod("enforcetrue", function (value, element, param) {
    return element.checked;
});

$.validator.unobtrusive.adapters.addBool("enforcetrue");

次に、使用法は次のとおりです。

[EnforceTrue(ErrorMessage = "Please tick the checkbox")]
public bool IsAccepted { get; set; }
2
Matt

web.configで設定された適切なアイテム がありますか?

これにより、検証が機能しなくなる可能性があります。

カスタム検証属性を作成することもできます([Required]は、それが存在するかどうかだけを気にし、あなたは値を気にします):

[AttributeUsage(AttributeTargets.Property | AttributeTargets.Field, AllowMultiple = false)]
sealed public class RequiredTrueAttribute : ValidationAttribute
{
    // Internal field to hold the mask value.
    readonly bool accepted;

    public bool Accepted
    {
        get { return accepted; }
    }

    public RequiredTrueAttribute(bool accepted)
    {
        this.accepted = accepted;
    }

    public override bool IsValid(object value)
    {
        bool isAccepted = (bool)value;
        return (isAccepted == true);
    }

    public override string FormatErrorMessage(string name)
    {
        return String.Format(CultureInfo.CurrentCulture,
          ErrorMessageString, name, this.Accepted);
    }
}

次に、使用法:

[RequiredTrue(ErrorMessage="{0} requires acceptance to continue.")]
public bool Agreement {get; set;}

here から。

2
George Stocker

Ta.speot.isによる投稿とJerad Roseからのコメントのフォローアップ:

指定された投稿は、控えめな検証ではクライアント側で機能しません。これは両方のキャンプ(クライアントとサーバー)で動作するはずです:

[RegularExpression("(True|true)")]
public bool TermsAndConditions { get; set; }
2
lobotommy

これは私のために働いたものです。他に何もしませんでした。 Mvc 5:

モデル

public string True
{
  get
  {
    return "true";
  }
}

[Required]
[Compare("True", ErrorMessage = "Please agree to the Acknowlegement")]
public bool Acknowlegement { get; set; }

表示

  @Html.HiddenFor(m => m.True)
  @Html.EditorFor(model => model.Acknowlegement, new { htmlAttributes = Model.Attributes })
  @Html.ValidationMessageFor(model => model.Acknowlegement, "", new { @class = "text-danger" })

enter image description here

enter image description here

1
toddmo

DataAnnotationsを使用する方法はわかりませんが、これはコントローラーで簡単に実行できます。

public ActionResult Add(Domain.Something model)
{

    if (!model.MyCheckBox)
        ModelState.AddModelError("MyCheckBox", "You forgot to click accept");

    if (ModelState.IsValid) {
        //'# do your stuff
    }

}

他の唯一のオプションは、サーバー側のカスタムバリデーターとクライアント側の リモートバリデーター を構築することです(リモート検証はMVC3 +でのみ利用可能です)

1
Chase Florell
/// <summary> 
///  Summary : -CheckBox for or input type check required validation is not working the root cause and solution as follows
///
///  Problem :
///  The key to this problem lies in interpretation of jQuery validation 'required' rule. I digged a little and find a specific code inside a jquery.validate.unobtrusive.js file:
///  adapters.add("required", function (options) {
///  if (options.element.tagName.toUpperCase() !== "INPUT" || options.element.type.toUpperCase() !== "CHECKBOX") {
///    setValidationValues(options, "required", true);
///    }
///   });
///   
///  Fix: (Jquery script fix at page level added in to check box required area)
///  jQuery.validator.unobtrusive.adapters.add("brequired", function (options) {
///   if (options.element.tagName.toUpperCase() == "INPUT" && options.element.type.toUpperCase() == "CHECKBOX") {
///              options.rules["required"] = true;
///   if (options.message) {
///                   options.messages["required"] = options.message;
///                       }
///  Fix : (C# Code for MVC validation)
///  You can see it inherits from common RequiredAttribute. Moreover it implements IClientValidateable. This is to make assure that rule will be propagated to client side (jQuery validation) as well.
///  
///  Annotation example :
///   [BooleanRequired]
///   public bool iAgree { get; set' }
/// </summary>


public class BooleanRequired : RequiredAttribute, IClientValidatable
{

    public BooleanRequired()
    {
    }

    public override bool IsValid(object value)
    {
        return value != null && (bool)value == true;
    }

    public IEnumerable<ModelClientValidationRule> GetClientValidationRules(ModelMetadata metadata, ControllerContext context)
    {
        return new ModelClientValidationRule[] { new ModelClientValidationRule() { ValidationType = "brequired", ErrorMessage = this.ErrorMessage } };
    }
}

Foolproof validation here をご覧ください。 Nugetからダウンロード/インストールできます。

これは、この種のことのための素晴らしい小さなライブラリです。

0
DavidWainwright

これを処理する最良の方法は、ボックスがtrueの場合はコントローラーをチェックインすること、そうでない場合はモデルにエラーを追加してビューを再表示することです。

前に述べたように、[必須]はすべて、値があることを確認することであり、チェックされていない場合は、まだfalseになります。

0
samack