付煜
2022-03-24 0ac2c8c62a4ba1b92bcae9f11f7f4d9c6f54875b
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
<!--
Strike by Appiphony
 
Version: 1.0.0
Website: http://www.lightningstrike.io
GitHub: https://github.com/appiphony/Strike-Components
License: BSD 3-Clause License
-->
<aura:component controller="strike_lookupController" access="global">
    <!-- Required -->
    <aura:attribute name="label" type="String" required="true" description="String value displayed above the menu used to describe the component"/>
    <aura:attribute name="object" type="String" required="true" description="String value that determines which records to search"/>
    <aura:attribute name="searchField" type="String" required="true" description="String value that determines which field on the object will be displayed and searched against"/>
 
    <aura:attribute name="helpText" type="String" description="String value when hovering over the help box"/>
    <aura:attribute name="helpTextPlacement" type="String" default="auto top" description="Determines position of the helpText"/>
    <aura:attribute name="class" type="String" description="A CSS class that will be applied to the outer element. This style is in addition to base classes associated with the component"/>
    <aura:attribute name="placeholder" type="String" description="String value that will appear when no record is selected"/>
    <aura:attribute name="value" type="String" description="String value that holds the Id of the selected record"/>
 
    <aura:attribute name="iconName" type="String" description="String value that determines the icon name or path"/>
    <aura:attribute name="customIcon" type="Boolean" description="Determines whether the icon is custom or standard"/>
 
    <!-- Query -->
    <aura:attribute name="subtitleField" type="String" description="String value that determines which field to display second from search results"/>
    <aura:attribute name="subTitleFormat" type="String" description="subTitleFormart"/>
    <aura:attribute name="filter" type="String" description="String value that determines how the SOQL search will be filtered"/>
    <aura:attribute name="order" type="String" description="String value that determines how the SOQL search will be ordered"/>
    <aura:attribute name="limit" type="String" description="String value that determines how many records are returned in the SOQL search"/>
 
    <!-- Messages -->
    <aura:attribute name="loadingMessage" type="String" default="Loading..." description="String value that will appear when the component is initializing"/>
    <aura:attribute name="searchingMessage" type="String" default="Searching..." description="String value that will appear when the component is searching"/>
    <aura:attribute name="noResultsMessage" type="String" default="没有查找到 {0} 的结果" description="String value that will appear when the component finds 0 results"/>
    <aura:attribute name="errorMessage" type="String" description="String value displayed when the input is in an error state"/>
 
    <!-- New Records -->
    <aura:attribute name="allowNewRecords" type="Boolean" description="Determines whether the New Record option is displayed"/>
    <aura:attribute name="overrideNewEvent" type="Boolean" description="Determines whether the standard new event should fire"/>
 
    <aura:attribute name="showRecentRecords" type="Boolean" description="Determines whether the five most recently viewed records will be displayed when the search field is empty"/>
 
    <aura:attribute name="disabled" type="Boolean" description="Determines whether the input is disabled"/>
    <aura:attribute name="required" type="Boolean" description="Determines whether a '*' is displayed on the component"/>
    <aura:attribute name="error" type="Boolean" description="Determines whether the input is in an error state"/>
 
    <!-- Methods -->
    <aura:method name="showError" action="{!c.showError}" description="A method which triggers the error state">
        <aura:attribute name="errorMessage" type="String"/>
    </aura:method>
    <aura:method name="hideError" action="{!c.hideError}" description="A method which removes the error state"/>
 
    <!-- Internal -->
    <aura:attribute name="focusIndex" type="Integer" description="Integer value that determines which record is focused" access="private"/>
    <aura:attribute name="idNumber" type="Integer" description="Random id assigned to pair labels with inputs" access="private"/>
    <aura:attribute name="initCallsRunning" type="Integer" default="{!0}" description="Integer value that determines how many calls are running during init" access="private"/>
    <aura:attribute name="lastSearchTerm" type="String" description="String value that holds the last searched term" access="private"/>
    <aura:attribute name="openMenu" type="Boolean" default="{!false}" description="Determines whether the menu is open" access="private"/>
    <aura:attribute name="objectLabel" type="String" description="String value that holds the label of the object" access="private"/>
    <aura:attribute name="recentRecords" type="Object[]" description="List of the most recent records" access="private"/>
    <aura:attribute name="records" type="Object[]" description="List of records returned from the SOQL search" access="private"/>
    <aura:attribute name="searching" type="Boolean" description="Boolean value marked when searching" access="private"/>
    <aura:attribute name="searchTimeout" type="Object" description="Object that holds the search Timeout" access="private"/>
    <aura:attribute name="valueLabel" type="String" description="String value that holds the searchField of the selected record"/>
    <aura:attribute name="valueSublabel" type="String" description="String value that holds the subtitleField of the selected record" access="private"/>
    <aura:attribute name="isMobile" type="Boolean" default="{!false}" description="Determines if the user is in Salesforce1" access="private"/>
 
    <!-- Event registration and handlers -->
    <aura:handler name="init" value="{!this}" action="{!c.onInit}"/>
    <aura:handler name="change" value="{!v.focusIndex}" action="{!c.handleFocusIndexChange}"/>
    <aura:handler name="change" value="{!v.value}" action="{!c.handleValueChange}"/>
 
    <aura:handler name="change" value="{!v.object}" action="{!c.handleObjectChange}"/>
    <aura:handler name="change" value="{!v.searchField}" action="{!c.handleSearchfieldChange}"/>
    <aura:handler name="change" value="{!v.subtitleField}" action="{!c.handleSubtitlefieldChange}"/>
    <aura:handler name="change" value="{!v.filter}" action="{!c.handleFilterChange}"/>
    <aura:handler name="change" value="{!v.order}" action="{!c.handleOrderChange}"/>
    <aura:handler name="change" value="{!v.limit}" action="{!c.handleLimitChange}"/>
 
    <aura:registerEvent name="strike_evt_addNewRecord" type="c:strike_evt" description="Fires an event that notifies when the strike_evt_addNewRecord button is pressed"/>
    <aura:dependency resource="force:createRecord" type="EVENT"/>
 
    <!-- Strike Lookup -->
    <ui:scrollerWrapper aura:id="lookup" class="{!'slds-form-element slds-lookup slds-is-open ' + if(v.error, 'slds-has-error ', '') + if(v.isMobile, 'sl-lookup--mobile ', '') + if(v.openMenu, 'sl-lookup--open ', '') + v.class}">
 
        <aura:if isTrue="{!not(empty(v.helpText))}">
            <c:strike_tooltip placement="{!v.helpTextPlacement}" text="{!v.helpText}" class="sl-help-text-icon">
                <lightning:icon iconName="utility:info" size="xx-small"/>
            </c:strike_tooltip>
        </aura:if>
        <div class="{!'slds-form-element__control slds-input-has-icon slds-input-has-icon--right' + if(v.disabled, ' sl-disabled', '')}" onclick="{!c.cancelLookup}">
            <div class="{!'slds-pill_container' + if(or(not(empty(v.value)), greaterthan(v.initCallsRunning, 0)), '', ' slds-hide')}">
                <span class="{!'slds-pill slds-size--1-of-1' + if(greaterthan(v.initCallsRunning, 0), ' sl-pill--loading', '')}">
                    <aura:if isTrue="{!and(not(empty(v.iconName)), equals(v.initCallsRunning, 0))}">
                        <aura:if isTrue="{!v.customIcon}">
                            <span class="slds-icon_container">
                                <c:strike_svg aura:id="svgComp" xlinkHref="{!v.iconName}" ariaHidden="true" class="slds-icon-text-default slds-icon slds-icon--small slds-pill__icon_container"/>
                            </span>
                            <aura:set attribute="else">
                                <lightning:icon class="slds-pill__icon_container" iconName="{!v.iconName}" size="small"/>
                            </aura:set>
                        </aura:if>
                    </aura:if>
                    <span class="slds-pill__label sl-pill__label" title="{!if(v.initCallsRunning > 0, v.loadingMessage, v.valueLabel)}">{!if(v.initCallsRunning > 0, v.loadingMessage, v.valueLabel)}</span>
                    <button class="{!'slds-button slds-button--icon slds-pill__remove' + if(or(v.disabled, greaterthan(v.initCallsRunning, 0)), ' slds-hide', '')}" title="Remove" onclick="{!c.handlePillClick}">
                        <lightning:icon iconName="utility:close" size="xx-small"/>
                    </button>
                </span>
            </div>
            <div class="{!'sl-lookup__input_container' + if(v.isMobile == true, ' slds-grid slds-grid--pull-padded-xx-small', '') + if(and(equals(v.initCallsRunning, 0), empty(v.value)), '', ' slds-hide')}">
                <div class="{!'sl-lookup__input' + if(v.isMobile == true, ' slds-col slds-p-horizontal--xx-small', '')}">
                    <lightning:icon iconName="utility:search" size="x-small" class="sl-search-icon"/>
                    <input
                        aura:id="lookupInput"
                        id="{!'strike-lookup-' + v.idNumber}"
                        name="{!'input-' + v.idNumber}"
                        class="slds-lookup__search-input slds-input js-template-lookup"
                        type="text"
                        aria-autocomplete="list"
                        role="combobox"
                        aria-expanded="false"
                        placeholder="{!v.placeholder}"
                        onclick="{!c.handleInputClick}"
                        onfocus="{!c.handleInputFocus}"
                        onkeydown="{!c.handleInputKeyDown}"
                        onkeyup="{!c.handleInputKeyUp}"
                        onkeypress="{!c.handleInputKeyPress}"
                        disabled='{!v.disabled}'/>
                </div>
                <aura:if isTrue="{!v.isMobile}">
                    <div class="sl-lookup--mobile__cancel slds-col slds-no-flex slds-p-horizontal--xx-small">
                        <button class="slds-button slds-button--neutral" onclick="{!c.cancelLookup}">Cancel</button>
                    </div>
                </aura:if>
            </div>
            <div class="{!if(v.searching, 'slds-lookup__menu', 'slds-hide')}" role="listbox" onclick="{!c.handleSearchingClick}">
                <ul class="slds-lookup__list" role="listbox">
                    <li role="presentation">
                        <span class="slds-lookup__item-action slds-media sl-searching" role="option">
                            <div class="slds-media__body">
                                <div class="slds-lookup__result-text">{!v.searchingMessage}</div>
                            </div>
                        </span>
                    </li>
                </ul>
            </div>
            <div class="{!if(and(v.searching == false, v.openMenu), 'slds-lookup__menu', 'slds-hide')}" role="listbox">
                <div class="{!if(or(v.records == null, v.records.length == 0), '', 'slds-hide')}">
                    <ul class="slds-lookup__list" role="listbox">
                        <li role="presentation">
                            <span class="slds-lookup__item-action slds-media sl-no-results" role="option">
                                <div class="slds-media__body">
                                    <div class="slds-lookup__result-text">
                                        {!format(v.noResultsMessage, '"' + v.lastSearchTerm + '"')}
                                    </div>
                                </div>
                            </span>
                        </li>
                    </ul>
                </div>
                <ul aura:id="lookupMenu" class="{!if(v.searching, 'slds-hide', 'slds-lookup__list')}" role="listbox">
                    <aura:iteration items="{!v.records}" var="record" indexVar="index">
                        <li role="presentation" class="{!if(index == v.focusIndex, 'slds-has-focus', '')}" data-index="{!index}" onclick="{!c.handleRecordClick}">
                            <span class="slds-lookup__item-action slds-media" role="option">
                                <aura:if isTrue="{!and(not(empty(v.iconName)), not(v.customIcon))}">
                                    <lightning:icon class="slds-media__figure" iconName="{!v.iconName}" size="small"/>
                                </aura:if>
                                <aura:if isTrue="{!v.customIcon}">
                                    <span class="slds-icon_container">
                                        <c:strike_svg aura:id="svgComp" xlinkHref="{!v.iconName}" ariaHidden="true" class="slds-icon-text-default slds-media__figure slds-icon slds-icon--x-small"/>
                                    </span>
                                </aura:if>
                                <div class="slds-media__body">
                                    <div class="slds-lookup__result-text">{!record.label}</div>
                                    <span class="{!'slds-lookup__result-meta slds-text-body--small' + if(empty(record.sublabel, ' slds-hide', ''))}">{!record.sublabel}</span>
                                </div>
                            </span>
                        </li>
                    </aura:iteration>
                    <aura:if isTrue="{!v.allowNewRecords}">
                        <li role="presentation" class="{!'sl-lookup__new' + if(index == -1, ' slds-has-focus', '')}" onclick="{!c.handleNewRecordClick}">
                            <span class="slds-lookup__item-action slds-lookup__item-action--label" role="option">
                                <lightning:icon iconName="utility:add" size="x-small"/>
                                <span class="slds-truncate">新建 {!v.objectLabel}</span>
                            </span>
                        </li>
                    </aura:if>
                </ul>
            </div>
        </div>
 
        <aura:if isTrue="{!and(v.error, not(empty(v.errorMessage)))}">
            <div class="slds-form-element__help">{!v.errorMessage}</div>
        </aura:if>
    </ui:scrollerWrapper>
</aura:component>
<!--
Copyright 2017 Appiphony, LLC
 
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
 
1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote 
products derived from this software without specific prior written permission.
 
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
-->