3

短縮版:

autocompleteextenderによって生成された個々のアイテムのcssスタイルを変更する方法を知りたいのですが、コントロールでは、すべてのアイテムのスタイルをグループとして設定することしかできません。

ロングバージョン:

AutoCompleteExtenderを利用する検索ボックスがあります。入力すると、検索結果が検索ボックスの下のボックスにポップアップ表示されます。

<asp:TextBox ID="txtSearch" OnChange="javascript: Changed(this);"  runat="server" style="width:360px;" />
<cc1:AutoCompleteExtender ID="Search_AutoCompleteExtender" runat="server" 
                    BehaviorID="Search_AutoCompleteExtender"
                    MinimumPrefixLength="3" 
                    DelimiterCharacters="" 
                    Enabled="True" 
                    ServicePath="~/Services/Search.asmx" 
                    ServiceMethod="GetResults"
                    TargetControlID="txtSearch"
                    FirstRowSelected="true"
                    CompletionListItemCssClass="AutoExtenderList"
                    CompletionListHighlightedItemCssClass="AutoExtenderHighlight"
                    CompletionInterval="1" 
                    EnableCaching="false"
                    CompletionSetCount="20"
                    CompletionListElementID="autocompleteDropDownPanel1" />

<asp:Panel ID="autocompleteDropDownPanel1" runat="server" ScrollBars="Vertical" Height="250" style="overflow-y:scroll;position:absolute;left:0;top:0" /

CompletionListItemCssClassAutoCompleteExtenderには、すべてのアイテムとCompletionListHighlightedItemCssClass強調表示されたアイテムにCSSクラスを割り当てるプロパティがあります。私が知りたいのは、特定の基準に基づいて特定のアイテムを個別にスタイル設定する方法です。

私のサービスでは、ストアドプロシージャを実行し、結果を少し操作してから、文字列配列をautocompleteextenderに戻しています。各レコードに特定のフラグを含めるようにストアドプロシージャを設定しました。そのフラグを使用して、検索結果の特定の行を強調表示します(たとえば、テキストを赤にします)。

これどうやってするの?個々のアイテムにアクセスしてスタイルを変更する方法が見つからないようです。

私はずっと助けを求めていて、手ぶらでやって来ています。ありがとうございました。

4

1 に答える 1

3

このような動作はデフォルトではサポートされていませんが、AutoCompleteExtenderソースを微調整してこの機能を追加できます。実際には2つの方法があります。1つはオブジェクトを拡張する方法で、アイテムcssクラスの新しいプロパティを使用してWebサービスからエクステンダーに返され、このプロパティ値をエクステンダークライアントスクリプトに適用します。そして、2番目に利用可能なアプローチは、アイテム作成用の新しいイベントを導入し、それをページで処理し、その値に応じてアイテムにカスタムロジックを適用することです。ここで両方のアプローチを実装しましょう。まだダウンロードしていない場合は、最初にAjaxControlToolkitソースをダウンロードしてClient/MicrosoftAjax.Extended/AutoComplete/AutoCompleteBehavior.pre.jsファイルを開く必要があります。_updateこのファイルでは、次のようにメソッドを変更する必要があります。

_update: function (prefixText, completionItems, cacheResults) {
    /// <summary>
    /// Method to update the status of the autocomplete behavior
    /// </summary>
    /// <param name="prefixText" type="String" DomElement="false" mayBeNull="true" />
    /// <param name="completionItems" type="Object" DomElement="false" mayBeNull="true" />
    /// <param name="cacheResults" type="Object" DomElement="false" mayBeNull="true" />
    /// <returns />       
    if (cacheResults && this.get_enableCaching()) {
        if (!this._cache) {
            this._cache = {};
        }
        this._cache[prefixText] = completionItems;
    }

    // If the target control loses focus or 
    // if the value in textbox has changed before the webservice returned
    // completion items we don't need to show popup 
    if ((!this._textBoxHasFocus) || (prefixText != this._currentCompletionWord())) {
        this._hideCompletionList();
        return;
    }

    if (completionItems && completionItems.length) {
        this._completionListElement.innerHTML = '';
        this._selectIndex = -1;
        var _firstChild = null;
        var text = null;
        var value = null;
        var cssClass = null;
        var dataItem = null;

        //remove this line if you don't need to implement itemDataBinding event
        var itemDataBindingHandler = this.get_events().getHandler('itemDataBinding');

        for (var i = 0; i < completionItems.length; i++) {
            // Create the item                
            var itemElement = null;
            if (this._completionListElementID) {
                // the completion element has been created by the user and li won't necessarily work
                itemElement = document.createElement('div');
            } else {
                itemElement = document.createElement('li');
            }

            // set the first child if it is null
            if (_firstChild == null) {
                _firstChild = itemElement;
            }

            // Get the text/value for the item
            try {
                dataItem = Sys.Serialization.JavaScriptSerializer.deserialize( completionItems[i] );
                if (dataItem && dataItem.First) {
                    // Use the text and value pair returned from the web service
                    text = dataItem.First;
                    value = dataItem.Second;

                    if (dataItem.CssClass) {
                        cssClass = dataItem.CssClass;
                    }
                }
                else {
                    // If the web service only returned a regular string, use it for
                    // both the text and the value
                    text = completionItems[i];
                    value = text;
                }
            } catch (ex) {
                text = completionItems[i];
                value = completionItems[i];
            }


            // Set the text/value for the item
            // ShowOnlyCurrentWordInCompletionListItem support
            var optionText = this._showOnlyCurrentWordInCompletionListItem ? text : this._getTextWithInsertedWord(text);
            itemElement.appendChild(document.createTextNode(optionText));
            itemElement._value = value;
            itemElement.__item = '';

            if (this._completionListItemCssClass) {
                Sys.UI.DomElement.addCssClass(itemElement, this._completionListItemCssClass);
            } else {
                var itemElementStyle = itemElement.style;
                itemElementStyle.padding = '0px';
                itemElementStyle.textAlign = 'left';
                itemElementStyle.textOverflow = 'ellipsis';
                // workaround for safari since normal colors do not
                // show well there.
                if (Sys.Browser.agent === Sys.Browser.Safari) {
                    itemElementStyle.backgroundColor = 'white';
                    itemElementStyle.color = 'black';
                } else {
                    itemElementStyle.backgroundColor = this._textBackground;
                    itemElementStyle.color = this._textColor;
                }
            }

            if (cssClass) {
                Sys.UI.DomElement.addCssClass(itemElement, cssClass);   
            }

            //remove this if you don't need to implement itemDataBinding event
            if (itemDataBindingHandler) {
                itemDataBindingHandler(itemElement, dataItem || completionItems[i]);
            }

            this._completionListElement.appendChild(itemElement);
        }
        var elementBounds = $common.getBounds(this.get_element());
        this._completionListElement.style.width = Math.max(1, elementBounds.width - 2) + 'px';
        this._completionListElement.scrollTop = 0;

        this.raisePopulated(Sys.EventArgs.Empty);

        var eventArgs = new Sys.CancelEventArgs();
        this.raiseShowing(eventArgs);
        if (!eventArgs.get_cancel()) {
            this.showPopup();
            // Check if the first Row is to be selected by default and if yes highlight it and updated selectIndex.
            if (this._firstRowSelected && (_firstChild != null)) {
                this._highlightItem(_firstChild);
                this._selectIndex = 0;
            }
        }
    } else {
        this._hideCompletionList();
    }
}

新しいクライアント側イベントでシナリオを実装するつもりがなかった場合、これは必要なプロジェクトのすべての変更です。アイテムのテキストと値とともに、Webサーバーメソッドからアイテムのcssクラスを既に渡すことができます。唯一の違いは、デフォルトでAjaxControlToolkitコントロールで使用されるPairの代わりにカスタムクラスを使用することです。

[WebMethod]
public string[] GetCompletionList(string prefixText, int count)
{
    var serializer = new JavaScriptSerializer();
    var items = Enumerable.Range(1, count)
        .Select(id => serializer.Serialize(
                        new
                        {
                            Second = id,
                            First = prefixText + "_" + Guid.NewGuid().ToString(),
                            CssClass = id % 2 == 0 ? "even" : "odd"
                        }));


    return items.ToArray();
}

カスタムクライアントイベント(実際に柔軟性を高める)も実装する場合は、AutoCompleteBehavior.pre.jsファイルにコードを追加する必要があります。Sys.Extended.UI.AutoCompleteBehavior.prototypeオブジェクトのどこかに以下のコードを追加します。

add_itemDataBinding: function (handler) {
    this.get_events().addHandler('itemDataBinding', handler);
},

remove_itemDataBinding: function (handler) {
    this.get_events().removeHandler('itemDataBinding', handler);
},

また、Server/AjaxControlToolkit/AutoComplete/AutoCompleteExtender.csファイルを開き、このプロパティをAutoCompleteExtenderクラスに追加します。

/// <summary>
/// Handler to attach to the client-side item data binding event
/// </summary>
[DefaultValue("")]
[ExtenderControlEvent]
[ClientPropertyName("itemDataBinding")]
public string OnClientItemDataBinding
{
    get { return GetPropertyValue("OnClientItemDataBinding", string.Empty); }
    set { SetPropertyValue("OnClientItemDataBinding", value); }
}

プロジェクトを再構築した後、追加機能を備えたカスタムAjaxControlToolkitアセンブリを使用できます。

<script type="text/javascript">
    function itemDataBinding(item, dataItem) {
        var value = parseInt(dataItem.Second, 10);
        if (!isNaN(value)) {
            item.className = value % 2 == 0 ? "even" : "odd";
        }
    }
</script>

<ajaxToolkit:AutoCompleteExtender runat="server" ID="autoComplete1" 
    TargetControlID="myTextBox" ServicePath="AutoComplete.asmx" ServiceMethod="GetCompletionList" 
    OnClientItemDataBinding="itemDataBinding">
于 2012-09-21T19:52:10.473 に答える