@rivet-health/design-system 32.1.1 → 32.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -366,9 +366,15 @@ export class MentionsInputComponent {
366
366
  const newRange = window.getSelection()?.getRangeAt(0);
367
367
  if (newRange) {
368
368
  newRange.insertNode(mentionSpan);
369
- //Set the range start and end to the same location (rendering the cursor in the right position)
369
+ //collapse range to position cursor after mention
370
370
  newRange.setStartAfter(mentionSpan);
371
- newRange.setEndAfter(mentionSpan);
371
+ newRange.collapse(true);
372
+ //Add a space after the mention
373
+ const spaceNode = document.createTextNode('\u00A0');
374
+ newRange.insertNode(spaceNode);
375
+ //Position cursor after the space
376
+ newRange.setStartAfter(spaceNode);
377
+ newRange.collapse(true);
372
378
  selection.removeAllRanges();
373
379
  //Apply this range to the DOM
374
380
  selection.addRange(newRange);
@@ -518,4 +524,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
518
524
  }
519
525
  MentionsInputComponent.contentToPlainText = contentToPlainText;
520
526
  })(MentionsInputComponent || (MentionsInputComponent = {}));
521
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mentions-input.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/mentions-input/mentions-input.component.ts","../../../../../../projects/riv/src/lib/input/mentions-input/mentions-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\n\n/**\n * A rich text input component that allows users to type text and mention other users by typing `@` followed by a name.\n * Uses a `contenteditable` div instead of a traditional input/textarea.\n *\n * ## Data Model\n * Works with a structured content model rather than plain strings:\n * - **Input/Output**: `ContentItem[]` - an array of content pieces\n * - **ContentItem types**:\n *   - `UserMention`: `{ type: 'mention', id: number, name: string }`\n *   - `Text`: `{ type: 'text', text: string }`\n *\n * ## Rendering Flow\n * 1. **Content → DOM**: `renderContent()` converts the content input array to HTML. Mentions become `<span>` elements with `data-id` and `data-type` attributes. Text is escaped and rendered as plain text.\n * 2. **DOM → Content**: `domToContentItems()` parses the DOM back into `ContentItem[]`, walking through child nodes extracting mentions and text. Emitted via `contentChange` event on input.\n *\n * ## User Interaction\n * - **Typing `@` triggers mentions**: Monitors input for `@` character, extracts text after `@` (up to first space) as search text, positions a callout dropdown at the cursor location\n * - **Keyboard navigation**: Arrow keys navigate filtered users list, Enter selects highlighted user, Escape closes mentions dropdown\n * - **Selecting a mention**: Deletes the `@` and search text from DOM, creates a non-editable `<span>` element for the mention, inserts at cursor position, repositions cursor after the mention\n *\n * ## Technical Details\n * - **Cursor management**: Uses Selection API and Range objects to track/manipulate cursor position\n * - **Text extraction**: `getTextBeforeCursor()` gets plain text before cursor, treating mention spans as `@name`\n * - **Character position mapping**: `findDomPositionAtCharIndex()` converts character indices to DOM node positions using TreeWalker\n * - **Change detection**: Uses OnPush with manual `markForCheck()` calls\n * - **Readonly mode**: Swaps between contenteditable and non-editable divs, styles mentions differently\n *\n * ## State\n * - `mentionsVisible`: controls dropdown visibility\n * - `mentionSearchText`: current filter text after @\n * - `mentionAnchor`: DOMRect for positioning dropdown\n * - `selectedUserIndex`: keyboard navigation state\n * - `savedRange`: preserves cursor position for click-based selection\n */\n@Component({\n  selector: 'riv-mentions-input',\n  templateUrl: `./mentions-input.component.html`,\n  styleUrls: ['./mentions-input.component.css'],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MentionsInputComponent implements OnChanges, AfterViewInit {\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  @Input() content: MentionsInputComponent.ContentItem[] = [];\n  @Input() users: MentionsInputComponent.User[] = [];\n  @Input() readonly = false;\n  @Input() placeholder: string = 'Add a comment';\n\n  @Output() readonly contentChange = new EventEmitter<\n    MentionsInputComponent.ContentItem[]\n  >();\n\n  @ViewChild('input', { static: false })\n  private readonly inputElement?: ElementRef<HTMLDivElement>;\n\n  mentionsVisible = false;\n  mentionSearchText = '';\n  mentionAnchor: DOMRect | null = null;\n  selectedUserIndex = 0;\n  private savedRange: Range | null = null;\n\n  ngAfterViewInit(): void {\n    this.renderContent();\n  }\n\n  //DAHNE NOTE - ngOnChanges is called whenever any @Input() property changes, where the changed input contains information about those changes\n  //So if @Input() content changes and it's not the first change (the first render is handled by ngAfterViewInit()), we render the content\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['content'] && !changes['content'].firstChange) {\n      this.renderContent();\n    }\n  }\n\n  private renderContent(): void {\n    //This is necessary because the @ViewChild reference isn't available until after the view initializes\n    if (!this.inputElement?.nativeElement) {\n      return;\n    }\n\n    const html = this.content\n      .map(item =>\n        MentionsInputComponent.contentItemToHtml(item, this.readonly),\n      )\n      .join('');\n    this.inputElement.nativeElement.innerHTML = html;\n  }\n\n  private static contentItemToHtml(\n    item: MentionsInputComponent.ContentItem,\n    readonly: boolean,\n  ): string {\n    switch (item.type) {\n      //DAHNE TODO - maybe we escape HTML here too? Although I don't think it will be possible for users to input an XSS attack in item.name\n      case 'mention':\n        return `<span class=\"${\n          readonly ? 'readonly-mention' : 'mention'\n        }\" data-id=\"${\n          item.id\n        }\" data-type=\"mention\" contenteditable=\"false\">@${MentionsInputComponent.escapeHtml(\n          item.name,\n        )}</span>`;\n      case 'text':\n        return MentionsInputComponent.escapeHtml(item.text);\n      case 'newline':\n        return '<br>';\n    }\n  }\n\n  private static escapeHtml(text: string): string {\n    const div = document.createElement('div');\n    div.textContent = text;\n    return div.innerHTML;\n  }\n\n  onInput(): void {\n    this.checkForMentionTrigger();\n    this.emitContentChange();\n  }\n\n  onKeyDown(event: KeyboardEvent): void {\n    if (this.mentionsVisible) {\n      const filteredUsers = this.getFilteredUsers();\n      if (filteredUsers.length === 0) {\n        this.closeMentions();\n        return;\n      }\n\n      switch (event.key) {\n        case 'ArrowDown':\n          event.preventDefault();\n          this.selectedUserIndex = Math.min(\n            this.selectedUserIndex + 1,\n            filteredUsers.length - 1,\n          );\n          this.cdr.markForCheck();\n          break;\n        case 'ArrowUp':\n          event.preventDefault();\n          this.selectedUserIndex = Math.max(this.selectedUserIndex - 1, 0);\n          this.cdr.markForCheck();\n          break;\n        case 'Enter':\n        case 'Tab':\n          event.preventDefault();\n          if (filteredUsers[this.selectedUserIndex]) {\n            this.selectUser(filteredUsers[this.selectedUserIndex]);\n          }\n          break;\n        case 'Escape':\n          event.preventDefault();\n          this.closeMentions();\n          break;\n      }\n      return;\n    }\n\n    //Handle newlines when mentions aren't visible\n    if (event.key === 'Enter') {\n      event.preventDefault();\n      this.insertLineBreak();\n      this.emitContentChange();\n    }\n  }\n\n  onPaste(event: ClipboardEvent): void {\n    event.preventDefault();\n\n    if (!event.clipboardData) {\n      return;\n    }\n\n    //Look for HTML first\n    const html = event.clipboardData.getData('text/html');\n    const plainText = event.clipboardData.getData('text/plain');\n\n    if (html) {\n      const sanitizedNodes = this.parsePastedHtml(html);\n      this.insertNodesAtCursor(sanitizedNodes);\n    } else if (plainText) {\n      const nodes = this.plainTextToNodes(plainText);\n      this.insertNodesAtCursor(nodes);\n    }\n\n    this.emitContentChange();\n  }\n\n  private plainTextToNodes(text: string): Node[] {\n    const nodes: Node[] = [];\n    const lines = text.split('\\n');\n\n    lines.forEach((line, index) => {\n      if (line) {\n        nodes.push(document.createTextNode(line));\n      }\n      if (index < lines.length - 1) {\n        nodes.push(document.createElement('br'));\n      }\n    });\n\n    return nodes;\n  }\n\n  private parsePastedHtml(html: string): Node[] {\n    // Create a temporary container to parse the HTML\n    const tempContainer = document.createElement('div');\n    tempContainer.innerHTML = html;\n\n    const nodes: Node[] = [];\n\n    const walkNodes = (node: Node): void => {\n      if (node.nodeType === Node.ELEMENT_NODE) {\n        const element = node as HTMLElement;\n        // DAHNE NOTE - we only want mention spans, br elements, and text\n        if (\n          element.tagName === 'SPAN' &&\n          element.getAttribute('data-type') === 'mention' &&\n          element.hasAttribute('data-id')\n        ) {\n          //Create mention to ultimately insert into the DOM\n          const mentionSpan = document.createElement('span');\n          mentionSpan.className = this.readonly\n            ? 'readonly-mention'\n            : 'mention';\n          mentionSpan.setAttribute('data-id', element.getAttribute('data-id')!);\n          mentionSpan.setAttribute('data-type', 'mention');\n          mentionSpan.setAttribute('contenteditable', 'false');\n          mentionSpan.textContent = element.textContent;\n          nodes.push(mentionSpan);\n        } else if (element.tagName === 'BR') {\n          // Preserve line breaks\n          nodes.push(document.createElement('br'));\n        } else {\n          //Recursively look for nested children\n          for (const child of Array.from(element.childNodes)) {\n            walkNodes(child);\n          }\n        }\n      } else if (node.nodeType === Node.TEXT_NODE) {\n        const text = node.textContent || '';\n        if (text) {\n          nodes.push(document.createTextNode(text));\n        }\n      }\n    };\n\n    for (const child of Array.from(tempContainer.childNodes)) {\n      walkNodes(child);\n    }\n\n    //DAHNE NOTE - the result here should be an array of text/span-mention/br nodes\n    return nodes;\n  }\n\n  private insertNodesAtCursor(nodes: Node[]): void {\n    //Get selection object, representing user cursor position/selection\n    //Get Range object at start of selection, which should represent cursor position\n    //If there is a selection, delete it\n    //Insert the parsed nodes from above, move range ends to after node\n    //Then remove the range and reset it to move the cursor position to the end\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    range.deleteContents();\n\n    for (const node of nodes) {\n      range.insertNode(node);\n      range.setStartAfter(node);\n      range.setEndAfter(node);\n    }\n\n    selection.removeAllRanges();\n    selection.addRange(range);\n  }\n\n  private insertTextAtCursor(text: string): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    range.deleteContents();\n\n    const textNode = document.createTextNode(text);\n    range.insertNode(textNode);\n    range.setStartAfter(textNode);\n    range.setEndAfter(textNode);\n\n    selection.removeAllRanges();\n    selection.addRange(range);\n  }\n\n  private insertLineBreak(): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    range.deleteContents();\n\n    const br = document.createElement('br');\n    range.insertNode(br);\n    range.setStartAfter(br);\n    range.setEndAfter(br);\n\n    selection.removeAllRanges();\n    selection.addRange(range);\n  }\n\n  private checkForMentionTrigger(): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount || !this.inputElement) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    const textBeforeCursor = this.getTextBeforeCursor(range);\n\n    const lastAtIndex = textBeforeCursor.lastIndexOf('@');\n\n    if (lastAtIndex === -1) {\n      this.closeMentions();\n      return;\n    }\n\n    //If we allow for a single space, we need to check to make sure that this logic won't pick up previous\n    //'@' mentions and keep the mention dropdown open\n    const atPosition = this.findDomPositionAtCharIndex(lastAtIndex);\n    if (atPosition && this.isInsideMentionSpan(atPosition.node)) {\n      this.closeMentions();\n      return;\n    }\n\n    const textAfterAt = textBeforeCursor.substring(lastAtIndex + 1);\n    const char1 = textAfterAt.charCodeAt(0);\n    const charN = textAfterAt.charCodeAt(textAfterAt.length - 1);\n    const charN_1 = textAfterAt.charCodeAt(textAfterAt.length - 2);\n    //These are the Unicode representations of normal space and non-breaking space\n    const unicodeSpaceChars = [32, 160];\n\n    // Close mentions if:\n    // - There's a double space (allows single spaces in the middle of search)\n    // - The text starts with a space (typing \"@ \" should not trigger mentions)\n    if (\n      textAfterAt.includes('  ') ||\n      textAfterAt.startsWith(' ') ||\n      char1 == 160 ||\n      (unicodeSpaceChars.includes(charN) && unicodeSpaceChars.includes(charN_1))\n    ) {\n      this.closeMentions();\n      return;\n    }\n\n    // Trim trailing/leading spaces from search text while preserving spaces in the middle\n    this.mentionSearchText = textAfterAt.trim();\n\n    // Only show mentions if there are filtered users\n    const filteredUsers = this.getFilteredUsers();\n    if (filteredUsers.length === 0) {\n      this.closeMentions();\n      return;\n    }\n\n    this.mentionsVisible = true;\n    this.selectedUserIndex = 0;\n\n    //Save the current range for later restoration (needed for click selection)\n    //DAHNE NOTE - this is annoyingly neceessary because clicks and enter keys have different effects on the\n    //focus of the contenteditable div\n    this.savedRange = range.cloneRange();\n\n    if (atPosition) {\n      const atRange = document.createRange();\n      atRange.setStart(atPosition.node, atPosition.offset);\n      atRange.setEnd(atPosition.node, atPosition.offset + 1);\n      this.mentionAnchor = atRange.getBoundingClientRect();\n    }\n    this.cdr.markForCheck();\n  }\n\n  private getTextBeforeCursor(range: Range): string {\n    const clonedRange = range.cloneRange();\n    //We select all of the input contents, and then move the end point to the current cursor position\n    clonedRange.selectNodeContents(this.inputElement!.nativeElement);\n    clonedRange.setEnd(range.endContainer, range.endOffset);\n\n    //Create a temporary div so that we can extract text content from it\n    const fragment = clonedRange.cloneContents();\n    const div = document.createElement('div');\n    div.appendChild(fragment);\n\n    //Get text content, replacing mention spans with @name\n    return div.textContent || '';\n  }\n\n  getFilteredUsers(): MentionsInputComponent.User[] {\n    const users = !this.mentionSearchText\n      ? this.users\n      : this.users.filter(user =>\n          user.name\n            .toLowerCase()\n            .includes(this.mentionSearchText.toLowerCase()),\n        );\n\n    return users.slice().sort((a, b) => a.name.localeCompare(b.name));\n  }\n\n  onUserHover(index: number): void {\n    this.selectedUserIndex = index;\n    this.cdr.markForCheck();\n  }\n\n  selectUser(user: MentionsInputComponent.User): void {\n    if (!this.inputElement) {\n      return;\n    }\n\n    //This gets DOM info from user (e.g. highlighted selections, and cursor position)\n    const selection = window.getSelection();\n    if (!selection) {\n      return;\n    }\n\n    // Restore the saved range if available (needed when selecting via click)\n    if (this.savedRange) {\n      selection.removeAllRanges();\n      selection.addRange(this.savedRange);\n    }\n\n    if (!selection.rangeCount) {\n      return;\n    }\n\n    // Delete the \"@\" and search text\n    this.deleteAtMentionSearch();\n\n    //DAHNE NOTE - this would ideally be the same as contentItemToHTML, but we need an actual element reference here in order to append it to the DOM\n    const mentionSpan = document.createElement('span');\n    mentionSpan.className = this.readonly ? 'readonly-mention' : 'mention';\n    mentionSpan.setAttribute('data-id', String(user.id));\n    mentionSpan.setAttribute('data-type', 'mention');\n    mentionSpan.setAttribute('contenteditable', 'false');\n    mentionSpan.textContent = `@${user.name}`;\n\n    //Find current cursor position\n    const newRange = window.getSelection()?.getRangeAt(0);\n    if (newRange) {\n      newRange.insertNode(mentionSpan);\n      //Set the range start and end to the same location (rendering the cursor in the right position)\n      newRange.setStartAfter(mentionSpan);\n      newRange.setEndAfter(mentionSpan);\n      selection.removeAllRanges();\n      //Apply this range to the DOM\n      selection.addRange(newRange);\n    }\n\n    this.closeMentions();\n    this.emitContentChange();\n  }\n\n  private deleteAtMentionSearch(): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    //This represents the current position of the cursor\n    const cursorRange = selection.getRangeAt(0);\n\n    //This will return all of the raw text (HTML-stripped) before the cursor\n    const textBeforeCursor = this.getTextBeforeCursor(cursorRange);\n    //We do that^ so that we can find the correct index of the most recent '@'\n    const lastAtIndex = textBeforeCursor.lastIndexOf('@');\n\n    if (lastAtIndex === -1) {\n      return;\n    }\n\n    //Knowing the position of the last '@' character we can now find the actual DOM node + offset where \"@\" lives\n    const atPosition = this.findDomPositionAtCharIndex(lastAtIndex);\n\n    if (!atPosition) {\n      return;\n    }\n\n    // Create range from \"@\" to cursor and delete it\n    const deleteRange = document.createRange();\n    deleteRange.setStart(atPosition.node, atPosition.offset);\n    deleteRange.setEnd(cursorRange.endContainer, cursorRange.endOffset);\n    deleteRange.deleteContents();\n  }\n\n  private isInsideMentionSpan(node: Node): boolean {\n    let currentNode: Node | null = node;\n    while (currentNode && currentNode !== this.inputElement?.nativeElement) {\n      if (\n        currentNode.nodeType === Node.ELEMENT_NODE &&\n        (currentNode as HTMLElement).getAttribute('data-type') === 'mention'\n      ) {\n        return true;\n      }\n      currentNode = currentNode.parentNode;\n    }\n    return false;\n  }\n\n  //This will go text node by node looking for the\n  //@ mention and give its position in the context of that node\n  private findDomPositionAtCharIndex(\n    charIndex: number,\n  ): { node: Node; offset: number } | null {\n    if (!this.inputElement) {\n      return null;\n    }\n\n    // Walk through all text nodes in the input, counting characters\n    const treeWalker = document.createTreeWalker(\n      this.inputElement.nativeElement,\n      NodeFilter.SHOW_TEXT,\n      null,\n    );\n\n    let charCount = 0;\n\n    while (treeWalker.nextNode()) {\n      const node = treeWalker.currentNode;\n      const text = node.textContent || '';\n\n      // Check if our target character is in this text node\n      if (charCount + text.length > charIndex) {\n        // The character at charIndex is in this node\n        const offsetInNode = charIndex - charCount;\n        return { node, offset: offsetInNode };\n      }\n\n      charCount += text.length;\n    }\n\n    return null;\n  }\n\n  private closeMentions(): void {\n    this.mentionsVisible = false;\n    this.mentionSearchText = '';\n    this.mentionAnchor = null;\n    this.selectedUserIndex = 0;\n    this.savedRange = null;\n    this.cdr.markForCheck();\n  }\n\n  private emitContentChange(): void {\n    if (!this.inputElement) {\n      return;\n    }\n\n    const contentItems = this.domToContentItems();\n    this.contentChange.emit(contentItems);\n  }\n\n  private domToContentItems(): MentionsInputComponent.ContentItem[] {\n    if (!this.inputElement?.nativeElement) {\n      return [];\n    }\n\n    const items: MentionsInputComponent.ContentItem[] = [];\n    const childNodes = this.inputElement.nativeElement.childNodes;\n\n    for (const node of Array.from(childNodes)) {\n      if (node.nodeType === Node.ELEMENT_NODE) {\n        const element = node as HTMLElement;\n        if (\n          element.tagName === 'SPAN' &&\n          element.getAttribute('data-type') === 'mention'\n        ) {\n          const id = Number(element.getAttribute('data-id'));\n          const name = element.textContent?.replace('@', '') || '';\n          items.push({ type: 'mention', id, name });\n        } else if (element.tagName === 'BR') {\n          items.push({ type: 'newline' });\n        }\n      } else if (node.nodeType === Node.TEXT_NODE) {\n        const text = node.textContent || '';\n        if (text) {\n          items.push({ type: 'text', text });\n        }\n      }\n    }\n    return items;\n  }\n\n  onCalloutClose(): void {\n    this.closeMentions();\n  }\n}\n\nexport namespace MentionsInputComponent {\n  export const ContentTypes = ['mention', 'text', 'newline'] as const;\n  export type ContentType =\n    (typeof MentionsInputComponent.ContentTypes)[number];\n  export interface User {\n    id: number;\n    name: string;\n  }\n  export type Content<T extends ContentType> = { type: T };\n\n  export type UserMention = Content<'mention'> & User;\n  export type Text = Content<'text'> & { text: string };\n  export type Newline = Content<'newline'>;\n\n  export type ContentItem = UserMention | Text | Newline;\n\n  export function contentToPlainText(content: ContentItem[]): string {\n    return content\n      .map(item => {\n        switch (item.type) {\n          case 'mention':\n            return `@${item.name}`;\n          case 'text':\n            return item.text;\n          case 'newline':\n            return '\\n';\n        }\n      })\n      .join('');\n  }\n}\n","<div class=\"mentions-container\" #container>\n  <div\n    *ngIf=\"!readonly\"\n    #input\n    class=\"text-content editable\"\n    contenteditable=\"true\"\n    [attr.data-placeholder]=\"placeholder\"\n    (input)=\"onInput()\"\n    (keydown)=\"onKeyDown($event)\"\n    (paste)=\"onPaste($event)\"\n  ></div>\n\n  <div *ngIf=\"readonly\" #input class=\"text-content\"></div>\n\n  <riv-callout\n    *ngIf=\"!readonly && mentionsVisible && mentionAnchor\"\n    [anchor]=\"mentionAnchor\"\n    [isModal]=\"false\"\n    [showCaret]=\"false\"\n    [allowedPositions]=\"['bottom-right', 'top-right']\"\n    [theme]=\"'light'\"\n    (close)=\"onCalloutClose()\"\n  >\n    <div class=\"mentions-content\">\n      <div class=\"mentions-header\">Mention someone</div>\n      <div class=\"mentions-list\">\n        <button\n          *ngFor=\"let user of getFilteredUsers(); let i = index\"\n          [class.selected]=\"i === selectedUserIndex\"\n          (mouseenter)=\"onUserHover(i)\"\n          (click)=\"selectUser(user)\"\n        >\n          {{ user.name }}\n        </button>\n      </div>\n    </div>\n  </riv-callout>\n</div>\n"]}
527
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mentions-input.component.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/input/mentions-input/mentions-input.component.ts","../../../../../../projects/riv/src/lib/input/mentions-input/mentions-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\n\n/**\n * A rich text input component that allows users to type text and mention other users by typing `@` followed by a name.\n * Uses a `contenteditable` div instead of a traditional input/textarea.\n *\n * ## Data Model\n * Works with a structured content model rather than plain strings:\n * - **Input/Output**: `ContentItem[]` - an array of content pieces\n * - **ContentItem types**:\n *   - `UserMention`: `{ type: 'mention', id: number, name: string }`\n *   - `Text`: `{ type: 'text', text: string }`\n *\n * ## Rendering Flow\n * 1. **Content → DOM**: `renderContent()` converts the content input array to HTML. Mentions become `<span>` elements with `data-id` and `data-type` attributes. Text is escaped and rendered as plain text.\n * 2. **DOM → Content**: `domToContentItems()` parses the DOM back into `ContentItem[]`, walking through child nodes extracting mentions and text. Emitted via `contentChange` event on input.\n *\n * ## User Interaction\n * - **Typing `@` triggers mentions**: Monitors input for `@` character, extracts text after `@` (up to first space) as search text, positions a callout dropdown at the cursor location\n * - **Keyboard navigation**: Arrow keys navigate filtered users list, Enter selects highlighted user, Escape closes mentions dropdown\n * - **Selecting a mention**: Deletes the `@` and search text from DOM, creates a non-editable `<span>` element for the mention, inserts at cursor position, repositions cursor after the mention\n *\n * ## Technical Details\n * - **Cursor management**: Uses Selection API and Range objects to track/manipulate cursor position\n * - **Text extraction**: `getTextBeforeCursor()` gets plain text before cursor, treating mention spans as `@name`\n * - **Character position mapping**: `findDomPositionAtCharIndex()` converts character indices to DOM node positions using TreeWalker\n * - **Change detection**: Uses OnPush with manual `markForCheck()` calls\n * - **Readonly mode**: Swaps between contenteditable and non-editable divs, styles mentions differently\n *\n * ## State\n * - `mentionsVisible`: controls dropdown visibility\n * - `mentionSearchText`: current filter text after @\n * - `mentionAnchor`: DOMRect for positioning dropdown\n * - `selectedUserIndex`: keyboard navigation state\n * - `savedRange`: preserves cursor position for click-based selection\n */\n@Component({\n  selector: 'riv-mentions-input',\n  templateUrl: `./mentions-input.component.html`,\n  styleUrls: ['./mentions-input.component.css'],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MentionsInputComponent implements OnChanges, AfterViewInit {\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  @Input() content: MentionsInputComponent.ContentItem[] = [];\n  @Input() users: MentionsInputComponent.User[] = [];\n  @Input() readonly = false;\n  @Input() placeholder: string = 'Add a comment';\n\n  @Output() readonly contentChange = new EventEmitter<\n    MentionsInputComponent.ContentItem[]\n  >();\n\n  @ViewChild('input', { static: false })\n  private readonly inputElement?: ElementRef<HTMLDivElement>;\n\n  mentionsVisible = false;\n  mentionSearchText = '';\n  mentionAnchor: DOMRect | null = null;\n  selectedUserIndex = 0;\n  private savedRange: Range | null = null;\n\n  ngAfterViewInit(): void {\n    this.renderContent();\n  }\n\n  //DAHNE NOTE - ngOnChanges is called whenever any @Input() property changes, where the changed input contains information about those changes\n  //So if @Input() content changes and it's not the first change (the first render is handled by ngAfterViewInit()), we render the content\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['content'] && !changes['content'].firstChange) {\n      this.renderContent();\n    }\n  }\n\n  private renderContent(): void {\n    //This is necessary because the @ViewChild reference isn't available until after the view initializes\n    if (!this.inputElement?.nativeElement) {\n      return;\n    }\n\n    const html = this.content\n      .map(item =>\n        MentionsInputComponent.contentItemToHtml(item, this.readonly),\n      )\n      .join('');\n    this.inputElement.nativeElement.innerHTML = html;\n  }\n\n  private static contentItemToHtml(\n    item: MentionsInputComponent.ContentItem,\n    readonly: boolean,\n  ): string {\n    switch (item.type) {\n      //DAHNE TODO - maybe we escape HTML here too? Although I don't think it will be possible for users to input an XSS attack in item.name\n      case 'mention':\n        return `<span class=\"${\n          readonly ? 'readonly-mention' : 'mention'\n        }\" data-id=\"${\n          item.id\n        }\" data-type=\"mention\" contenteditable=\"false\">@${MentionsInputComponent.escapeHtml(\n          item.name,\n        )}</span>`;\n      case 'text':\n        return MentionsInputComponent.escapeHtml(item.text);\n      case 'newline':\n        return '<br>';\n    }\n  }\n\n  private static escapeHtml(text: string): string {\n    const div = document.createElement('div');\n    div.textContent = text;\n    return div.innerHTML;\n  }\n\n  onInput(): void {\n    this.checkForMentionTrigger();\n    this.emitContentChange();\n  }\n\n  onKeyDown(event: KeyboardEvent): void {\n    if (this.mentionsVisible) {\n      const filteredUsers = this.getFilteredUsers();\n      if (filteredUsers.length === 0) {\n        this.closeMentions();\n        return;\n      }\n\n      switch (event.key) {\n        case 'ArrowDown':\n          event.preventDefault();\n          this.selectedUserIndex = Math.min(\n            this.selectedUserIndex + 1,\n            filteredUsers.length - 1,\n          );\n          this.cdr.markForCheck();\n          break;\n        case 'ArrowUp':\n          event.preventDefault();\n          this.selectedUserIndex = Math.max(this.selectedUserIndex - 1, 0);\n          this.cdr.markForCheck();\n          break;\n        case 'Enter':\n        case 'Tab':\n          event.preventDefault();\n          if (filteredUsers[this.selectedUserIndex]) {\n            this.selectUser(filteredUsers[this.selectedUserIndex]);\n          }\n          break;\n        case 'Escape':\n          event.preventDefault();\n          this.closeMentions();\n          break;\n      }\n      return;\n    }\n\n    //Handle newlines when mentions aren't visible\n    if (event.key === 'Enter') {\n      event.preventDefault();\n      this.insertLineBreak();\n      this.emitContentChange();\n    }\n  }\n\n  onPaste(event: ClipboardEvent): void {\n    event.preventDefault();\n\n    if (!event.clipboardData) {\n      return;\n    }\n\n    //Look for HTML first\n    const html = event.clipboardData.getData('text/html');\n    const plainText = event.clipboardData.getData('text/plain');\n\n    if (html) {\n      const sanitizedNodes = this.parsePastedHtml(html);\n      this.insertNodesAtCursor(sanitizedNodes);\n    } else if (plainText) {\n      const nodes = this.plainTextToNodes(plainText);\n      this.insertNodesAtCursor(nodes);\n    }\n\n    this.emitContentChange();\n  }\n\n  private plainTextToNodes(text: string): Node[] {\n    const nodes: Node[] = [];\n    const lines = text.split('\\n');\n\n    lines.forEach((line, index) => {\n      if (line) {\n        nodes.push(document.createTextNode(line));\n      }\n      if (index < lines.length - 1) {\n        nodes.push(document.createElement('br'));\n      }\n    });\n\n    return nodes;\n  }\n\n  private parsePastedHtml(html: string): Node[] {\n    // Create a temporary container to parse the HTML\n    const tempContainer = document.createElement('div');\n    tempContainer.innerHTML = html;\n\n    const nodes: Node[] = [];\n\n    const walkNodes = (node: Node): void => {\n      if (node.nodeType === Node.ELEMENT_NODE) {\n        const element = node as HTMLElement;\n        // DAHNE NOTE - we only want mention spans, br elements, and text\n        if (\n          element.tagName === 'SPAN' &&\n          element.getAttribute('data-type') === 'mention' &&\n          element.hasAttribute('data-id')\n        ) {\n          //Create mention to ultimately insert into the DOM\n          const mentionSpan = document.createElement('span');\n          mentionSpan.className = this.readonly\n            ? 'readonly-mention'\n            : 'mention';\n          mentionSpan.setAttribute('data-id', element.getAttribute('data-id')!);\n          mentionSpan.setAttribute('data-type', 'mention');\n          mentionSpan.setAttribute('contenteditable', 'false');\n          mentionSpan.textContent = element.textContent;\n          nodes.push(mentionSpan);\n        } else if (element.tagName === 'BR') {\n          // Preserve line breaks\n          nodes.push(document.createElement('br'));\n        } else {\n          //Recursively look for nested children\n          for (const child of Array.from(element.childNodes)) {\n            walkNodes(child);\n          }\n        }\n      } else if (node.nodeType === Node.TEXT_NODE) {\n        const text = node.textContent || '';\n        if (text) {\n          nodes.push(document.createTextNode(text));\n        }\n      }\n    };\n\n    for (const child of Array.from(tempContainer.childNodes)) {\n      walkNodes(child);\n    }\n\n    //DAHNE NOTE - the result here should be an array of text/span-mention/br nodes\n    return nodes;\n  }\n\n  private insertNodesAtCursor(nodes: Node[]): void {\n    //Get selection object, representing user cursor position/selection\n    //Get Range object at start of selection, which should represent cursor position\n    //If there is a selection, delete it\n    //Insert the parsed nodes from above, move range ends to after node\n    //Then remove the range and reset it to move the cursor position to the end\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    range.deleteContents();\n\n    for (const node of nodes) {\n      range.insertNode(node);\n      range.setStartAfter(node);\n      range.setEndAfter(node);\n    }\n\n    selection.removeAllRanges();\n    selection.addRange(range);\n  }\n\n  private insertTextAtCursor(text: string): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    range.deleteContents();\n\n    const textNode = document.createTextNode(text);\n    range.insertNode(textNode);\n    range.setStartAfter(textNode);\n    range.setEndAfter(textNode);\n\n    selection.removeAllRanges();\n    selection.addRange(range);\n  }\n\n  private insertLineBreak(): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    range.deleteContents();\n\n    const br = document.createElement('br');\n    range.insertNode(br);\n    range.setStartAfter(br);\n    range.setEndAfter(br);\n\n    selection.removeAllRanges();\n    selection.addRange(range);\n  }\n\n  private checkForMentionTrigger(): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount || !this.inputElement) {\n      return;\n    }\n\n    const range = selection.getRangeAt(0);\n    const textBeforeCursor = this.getTextBeforeCursor(range);\n\n    const lastAtIndex = textBeforeCursor.lastIndexOf('@');\n\n    if (lastAtIndex === -1) {\n      this.closeMentions();\n      return;\n    }\n\n    //If we allow for a single space, we need to check to make sure that this logic won't pick up previous\n    //'@' mentions and keep the mention dropdown open\n    const atPosition = this.findDomPositionAtCharIndex(lastAtIndex);\n    if (atPosition && this.isInsideMentionSpan(atPosition.node)) {\n      this.closeMentions();\n      return;\n    }\n\n    const textAfterAt = textBeforeCursor.substring(lastAtIndex + 1);\n    const char1 = textAfterAt.charCodeAt(0);\n    const charN = textAfterAt.charCodeAt(textAfterAt.length - 1);\n    const charN_1 = textAfterAt.charCodeAt(textAfterAt.length - 2);\n    //These are the Unicode representations of normal space and non-breaking space\n    const unicodeSpaceChars = [32, 160];\n\n    // Close mentions if:\n    // - There's a double space (allows single spaces in the middle of search)\n    // - The text starts with a space (typing \"@ \" should not trigger mentions)\n    if (\n      textAfterAt.includes('  ') ||\n      textAfterAt.startsWith(' ') ||\n      char1 == 160 ||\n      (unicodeSpaceChars.includes(charN) && unicodeSpaceChars.includes(charN_1))\n    ) {\n      this.closeMentions();\n      return;\n    }\n\n    // Trim trailing/leading spaces from search text while preserving spaces in the middle\n    this.mentionSearchText = textAfterAt.trim();\n\n    // Only show mentions if there are filtered users\n    const filteredUsers = this.getFilteredUsers();\n    if (filteredUsers.length === 0) {\n      this.closeMentions();\n      return;\n    }\n\n    this.mentionsVisible = true;\n    this.selectedUserIndex = 0;\n\n    //Save the current range for later restoration (needed for click selection)\n    //DAHNE NOTE - this is annoyingly neceessary because clicks and enter keys have different effects on the\n    //focus of the contenteditable div\n    this.savedRange = range.cloneRange();\n\n    if (atPosition) {\n      const atRange = document.createRange();\n      atRange.setStart(atPosition.node, atPosition.offset);\n      atRange.setEnd(atPosition.node, atPosition.offset + 1);\n      this.mentionAnchor = atRange.getBoundingClientRect();\n    }\n    this.cdr.markForCheck();\n  }\n\n  private getTextBeforeCursor(range: Range): string {\n    const clonedRange = range.cloneRange();\n    //We select all of the input contents, and then move the end point to the current cursor position\n    clonedRange.selectNodeContents(this.inputElement!.nativeElement);\n    clonedRange.setEnd(range.endContainer, range.endOffset);\n\n    //Create a temporary div so that we can extract text content from it\n    const fragment = clonedRange.cloneContents();\n    const div = document.createElement('div');\n    div.appendChild(fragment);\n\n    //Get text content, replacing mention spans with @name\n    return div.textContent || '';\n  }\n\n  getFilteredUsers(): MentionsInputComponent.User[] {\n    const users = !this.mentionSearchText\n      ? this.users\n      : this.users.filter(user =>\n          user.name\n            .toLowerCase()\n            .includes(this.mentionSearchText.toLowerCase()),\n        );\n\n    return users.slice().sort((a, b) => a.name.localeCompare(b.name));\n  }\n\n  onUserHover(index: number): void {\n    this.selectedUserIndex = index;\n    this.cdr.markForCheck();\n  }\n\n  selectUser(user: MentionsInputComponent.User): void {\n    if (!this.inputElement) {\n      return;\n    }\n\n    //This gets DOM info from user (e.g. highlighted selections, and cursor position)\n    const selection = window.getSelection();\n    if (!selection) {\n      return;\n    }\n\n    // Restore the saved range if available (needed when selecting via click)\n    if (this.savedRange) {\n      selection.removeAllRanges();\n      selection.addRange(this.savedRange);\n    }\n\n    if (!selection.rangeCount) {\n      return;\n    }\n\n    // Delete the \"@\" and search text\n    this.deleteAtMentionSearch();\n\n    //DAHNE NOTE - this would ideally be the same as contentItemToHTML, but we need an actual element reference here in order to append it to the DOM\n    const mentionSpan = document.createElement('span');\n    mentionSpan.className = this.readonly ? 'readonly-mention' : 'mention';\n    mentionSpan.setAttribute('data-id', String(user.id));\n    mentionSpan.setAttribute('data-type', 'mention');\n    mentionSpan.setAttribute('contenteditable', 'false');\n    mentionSpan.textContent = `@${user.name}`;\n\n    //Find current cursor position\n    const newRange = window.getSelection()?.getRangeAt(0);\n    if (newRange) {\n      newRange.insertNode(mentionSpan);\n      //collapse range to position cursor after mention\n      newRange.setStartAfter(mentionSpan);\n      newRange.collapse(true);\n\n      //Add a space after the mention\n      const spaceNode = document.createTextNode('\\u00A0');\n      newRange.insertNode(spaceNode);\n      //Position cursor after the space\n      newRange.setStartAfter(spaceNode);\n      newRange.collapse(true);\n\n      selection.removeAllRanges();\n      //Apply this range to the DOM\n      selection.addRange(newRange);\n    }\n\n    this.closeMentions();\n    this.emitContentChange();\n  }\n\n  private deleteAtMentionSearch(): void {\n    const selection = window.getSelection();\n    if (!selection || !selection.rangeCount) {\n      return;\n    }\n\n    //This represents the current position of the cursor\n    const cursorRange = selection.getRangeAt(0);\n\n    //This will return all of the raw text (HTML-stripped) before the cursor\n    const textBeforeCursor = this.getTextBeforeCursor(cursorRange);\n    //We do that^ so that we can find the correct index of the most recent '@'\n    const lastAtIndex = textBeforeCursor.lastIndexOf('@');\n\n    if (lastAtIndex === -1) {\n      return;\n    }\n\n    //Knowing the position of the last '@' character we can now find the actual DOM node + offset where \"@\" lives\n    const atPosition = this.findDomPositionAtCharIndex(lastAtIndex);\n\n    if (!atPosition) {\n      return;\n    }\n\n    // Create range from \"@\" to cursor and delete it\n    const deleteRange = document.createRange();\n    deleteRange.setStart(atPosition.node, atPosition.offset);\n    deleteRange.setEnd(cursorRange.endContainer, cursorRange.endOffset);\n    deleteRange.deleteContents();\n  }\n\n  private isInsideMentionSpan(node: Node): boolean {\n    let currentNode: Node | null = node;\n    while (currentNode && currentNode !== this.inputElement?.nativeElement) {\n      if (\n        currentNode.nodeType === Node.ELEMENT_NODE &&\n        (currentNode as HTMLElement).getAttribute('data-type') === 'mention'\n      ) {\n        return true;\n      }\n      currentNode = currentNode.parentNode;\n    }\n    return false;\n  }\n\n  //This will go text node by node looking for the\n  //@ mention and give its position in the context of that node\n  private findDomPositionAtCharIndex(\n    charIndex: number,\n  ): { node: Node; offset: number } | null {\n    if (!this.inputElement) {\n      return null;\n    }\n\n    // Walk through all text nodes in the input, counting characters\n    const treeWalker = document.createTreeWalker(\n      this.inputElement.nativeElement,\n      NodeFilter.SHOW_TEXT,\n      null,\n    );\n\n    let charCount = 0;\n\n    while (treeWalker.nextNode()) {\n      const node = treeWalker.currentNode;\n      const text = node.textContent || '';\n\n      // Check if our target character is in this text node\n      if (charCount + text.length > charIndex) {\n        // The character at charIndex is in this node\n        const offsetInNode = charIndex - charCount;\n        return { node, offset: offsetInNode };\n      }\n\n      charCount += text.length;\n    }\n\n    return null;\n  }\n\n  private closeMentions(): void {\n    this.mentionsVisible = false;\n    this.mentionSearchText = '';\n    this.mentionAnchor = null;\n    this.selectedUserIndex = 0;\n    this.savedRange = null;\n    this.cdr.markForCheck();\n  }\n\n  private emitContentChange(): void {\n    if (!this.inputElement) {\n      return;\n    }\n\n    const contentItems = this.domToContentItems();\n    this.contentChange.emit(contentItems);\n  }\n\n  private domToContentItems(): MentionsInputComponent.ContentItem[] {\n    if (!this.inputElement?.nativeElement) {\n      return [];\n    }\n\n    const items: MentionsInputComponent.ContentItem[] = [];\n    const childNodes = this.inputElement.nativeElement.childNodes;\n\n    for (const node of Array.from(childNodes)) {\n      if (node.nodeType === Node.ELEMENT_NODE) {\n        const element = node as HTMLElement;\n        if (\n          element.tagName === 'SPAN' &&\n          element.getAttribute('data-type') === 'mention'\n        ) {\n          const id = Number(element.getAttribute('data-id'));\n          const name = element.textContent?.replace('@', '') || '';\n          items.push({ type: 'mention', id, name });\n        } else if (element.tagName === 'BR') {\n          items.push({ type: 'newline' });\n        }\n      } else if (node.nodeType === Node.TEXT_NODE) {\n        const text = node.textContent || '';\n        if (text) {\n          items.push({ type: 'text', text });\n        }\n      }\n    }\n    return items;\n  }\n\n  onCalloutClose(): void {\n    this.closeMentions();\n  }\n}\n\nexport namespace MentionsInputComponent {\n  export const ContentTypes = ['mention', 'text', 'newline'] as const;\n  export type ContentType =\n    (typeof MentionsInputComponent.ContentTypes)[number];\n  export interface User {\n    id: number;\n    name: string;\n  }\n  export type Content<T extends ContentType> = { type: T };\n\n  export type UserMention = Content<'mention'> & User;\n  export type Text = Content<'text'> & { text: string };\n  export type Newline = Content<'newline'>;\n\n  export type ContentItem = UserMention | Text | Newline;\n\n  export function contentToPlainText(content: ContentItem[]): string {\n    return content\n      .map(item => {\n        switch (item.type) {\n          case 'mention':\n            return `@${item.name}`;\n          case 'text':\n            return item.text;\n          case 'newline':\n            return '\\n';\n        }\n      })\n      .join('');\n  }\n}\n","<div class=\"mentions-container\" #container>\n  <div\n    *ngIf=\"!readonly\"\n    #input\n    class=\"text-content editable\"\n    contenteditable=\"true\"\n    [attr.data-placeholder]=\"placeholder\"\n    (input)=\"onInput()\"\n    (keydown)=\"onKeyDown($event)\"\n    (paste)=\"onPaste($event)\"\n  ></div>\n\n  <div *ngIf=\"readonly\" #input class=\"text-content\"></div>\n\n  <riv-callout\n    *ngIf=\"!readonly && mentionsVisible && mentionAnchor\"\n    [anchor]=\"mentionAnchor\"\n    [isModal]=\"false\"\n    [showCaret]=\"false\"\n    [allowedPositions]=\"['bottom-right', 'top-right']\"\n    [theme]=\"'light'\"\n    (close)=\"onCalloutClose()\"\n  >\n    <div class=\"mentions-content\">\n      <div class=\"mentions-header\">Mention someone</div>\n      <div class=\"mentions-list\">\n        <button\n          *ngFor=\"let user of getFilteredUsers(); let i = index\"\n          [class.selected]=\"i === selectedUserIndex\"\n          (mouseenter)=\"onUserHover(i)\"\n          (click)=\"selectUser(user)\"\n        >\n          {{ user.name }}\n        </button>\n      </div>\n    </div>\n  </riv-callout>\n</div>\n"]}
@@ -21,6 +21,15 @@ export class SelectComponent extends InputLabelComponent {
21
21
  this.size = 'medium';
22
22
  this.disabled = false;
23
23
  this.locked = false;
24
+ this.allowedCalloutPositions = [
25
+ 'top-left',
26
+ 'top-center',
27
+ 'top-right',
28
+ 'bottom-right',
29
+ 'bottom-center',
30
+ 'bottom-left',
31
+ ];
32
+ this.preferredCalloutPosition = 'bottom-right';
24
33
  this.orderByOptionGroups = [
25
34
  {
26
35
  options: [
@@ -114,10 +123,10 @@ export class SelectComponent extends InputLabelComponent {
114
123
  }
115
124
  }
116
125
  SelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
117
- SelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SelectComponent, selector: "riv-select", inputs: { manager: "manager", size: "size", disabled: "disabled", locked: "locked" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }, { propertyName: "customSelectedOptionsTemplate", first: true, predicate: ["customSelectedOptions"], descendants: true }, { propertyName: "optionTemplate", first: true, predicate: ["option"], descendants: true }, { propertyName: "headerTemplate", first: true, predicate: ["header"], descendants: true }, { propertyName: "footerTemplate", first: true, predicate: ["footer"], descendants: true }], viewQueries: [{ propertyName: "customTriggerButton", first: true, predicate: ["customTriggerButton"], descendants: true }, { propertyName: "standardTriggerButton", first: true, predicate: ["standardTriggerButton"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<ng-container *ngIf=\"manager?.state | async; let s\">\n <ng-template #content>\n <riv-loading-cover [loading]=\"s.load.loading\">\n <div\n class=\"content-body\"\n [class.inline]=\"s.display.inline\"\n [ngStyle]=\"{\n 'max-height': s.display.maxHeight,\n 'max-width': s.display.maxWidth,\n 'min-height': s.display.minHeight,\n 'min-width': s.display.minWidth\n }\"\n >\n <header *ngIf=\"headerTemplate\">\n <ng-container\n *ngTemplateOutlet=\"headerTemplate; context: { state: s }\"\n ></ng-container>\n </header>\n <riv-search\n *ngIf=\"s.query.showSearch\"\n [value]=\"s.query.search\"\n [placeholder]=\"s.query.searchPlaceholder\"\n [autoFocus]=\"true\"\n (valueChange)=\"searchChange($event)\"\n ></riv-search>\n <div *ngIf=\"s.query.showOrder\">\n <span class=\"order-by\">Sort by:</span>\n <riv-select [manager]=\"orderByManager\">\n <ng-template #trigger let-state=\"state\">\n <button\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'small'\"\n [icon]=\"state.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [iconPosition]=\"'last'\"\n >\n {{ getSelectedOrderByOption(s)?.title }}\n </button>\n </ng-template>\n </riv-select>\n </div>\n <div class=\"options\">\n <riv-select-node\n *ngIf=\"\n s.selection.allowMultiSelect &&\n s.selection.allowSelectAll &&\n !s.display.zeroStateMessage\n \"\n [mode]=\"'multi'\"\n [node]=\"{\n id: '__NOOP__',\n title: s.query.search ? 'Select all matching' : 'Select all',\n selected: s.selection.visibleSelectionState,\n selectable: true,\n expandable: false,\n expanded: false\n }\"\n (selectChange)=\"\n manager?.actions?.next({ type: 'visibleSelectedChange' })\n \"\n ></riv-select-node>\n <div\n *ngFor=\"let group of s.fullOptionGroups; trackBy: trackBySymbol\"\n class=\"group\"\n [class.divider]=\"s.display.dividers\"\n >\n <span *ngIf=\"group.header; let header\" class=\"header\">\n {{ header }}\n </span>\n <ng-container\n *ngFor=\"let option of group.options; trackBy: trackByOption\"\n >\n <ng-container *ngIf=\"optionTemplate; else standardTemplate\">\n <ng-container\n [ngTemplateOutlet]=\"optionTemplate\"\n [ngTemplateOutletContext]=\"{ node: option }\"\n ></ng-container>\n </ng-container>\n <ng-template #standardTemplate>\n <riv-select-node\n [mode]=\"s.selection.allowMultiSelect ? 'multi' : 'single'\"\n [node]=\"option\"\n [showSingleSelected]=\"s.display.showSingleSelected\"\n (selectChange)=\"\n selectChange(s.selection.allowMultiSelect, $event)\n \"\n (expandChange)=\"\n manager?.actions?.next({\n type: 'toggleOptionExpanded',\n id: $event\n })\n \"\n ></riv-select-node>\n </ng-template>\n </ng-container>\n </div>\n <span\n *ngIf=\"s.display.displayLimitMessage; let message\"\n class=\"display-limit\"\n >\n {{ message }}\n </span>\n <riv-zero-state\n *ngIf=\"s.display.zeroStateMessage; let message\"\n [message]=\"message\"\n ></riv-zero-state>\n </div>\n <footer *ngIf=\"footerTemplate\" class=\"custom-footer\">\n <ng-container\n *ngTemplateOutlet=\"footerTemplate; context: { state: s }\"\n ></ng-container>\n </footer>\n </div>\n </riv-loading-cover>\n </ng-template>\n\n <ng-container *ngIf=\"!s.display.inline; else content\">\n <ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n class=\"custom-trigger-button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ state: s }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n [class.small]=\"size === 'small'\"\n [class.large]=\"size === 'large'\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n *ngIf=\"s.selection.selected.size; else placeholderValue\"\n >\n <ng-container\n *ngIf=\"\n customSelectedOptionsTemplate;\n else standardSelectedOptionsTemplate\n \"\n >\n <span class=\"value\">\n <ng-container\n [ngTemplateOutlet]=\"customSelectedOptionsTemplate\"\n [ngTemplateOutletContext]=\"{ selected: s.selection.selected }\"\n ></ng-container>\n </span>\n </ng-container>\n <ng-template #standardSelectedOptionsTemplate>\n <span class=\"value\">{{\n s.display.formattedSelectedOptions\n }}</span>\n <span *ngIf=\"s.display.pillCount > 1\" class=\"selected-count\">\n {{ s.display.pillCount | rivNumber }}\n </span>\n </ng-template>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ s.display.placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon\n *ngIf=\"!locked\"\n [name]=\"s.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [size]=\"20\"\n ></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"20\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n </ng-template>\n\n <ng-container *ngIf=\"s.display.open\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"manager?.actions?.next({ type: 'openChange', open: false })\"\n >\n <ng-container [ngTemplateOutlet]=\"content\"></ng-container>\n </riv-callout>\n </ng-container>\n </ng-container>\n</ng-container>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small);background-color:var(--surface-light-0);cursor:pointer;padding-left:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1);cursor:default}.selected-count{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);font-weight:var(--font-weight-normal);color:var(--type-dark-body);border-radius:calc(var(--base-grid-size) * 3);background-color:var(--brand);padding:calc(var(--base-grid-size) / 4) calc(var(--base-grid-size) * 2);align-self:center}.value{font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) 0;flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.trigger.small .value{font:var(--input-small)}.trigger.large .value{font:var(--input-large);padding:var(--size-medium) var(--size-xsmall)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-xsmall) calc(var(--base-grid-size) * 1.5)}.trigger.warning{border-color:var(--surface-dark-caution)}.trigger.error{border-color:var(--surface-dark-danger);box-shadow:inset 0 0 0 var(--border-width-large) var(--surface-dark-danger)}.custom-trigger-button{max-width:100%}.order-by{font:var(--input-medium);color:var(--type-light-low-contrast)}.content-body{padding:var(--size-large);display:flex;flex-direction:column;gap:var(--size-large)}.content-body:not(.inline){max-height:50vh;min-width:calc(var(--base-grid-size) * 75);max-width:50vw}.options{flex-grow:1;overflow-y:auto;gap:var(--size-medium)}.options,.group{display:flex;flex-direction:column}.group{gap:var(--size-xsmall)}.group.divider:not(:last-child){padding-bottom:var(--size-medium);border-bottom:var(--border-width) solid var(--border-light)}.display-limit{font:var(--body-small);color:var(--type-light-disabled);padding:var(--size-small);text-align:center}.header{font:var(--input-medium);color:var(--type-light-low-contrast);padding:var(--size-xsmall) 0}.custom-footer{padding-top:var(--size-large);border-top:var(--border-width) solid var(--border-light)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.ButtonComponent, selector: "[rivButton]", inputs: ["locked", "disabled", "loading", "full", "size", "variant", "icon", "iconPosition", "current"] }, { kind: "component", type: i3.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "component", type: i4.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "component", type: i5.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText", "errorMessage", "state"], outputs: ["labelAction"] }, { kind: "component", type: i6.LoadingCoverComponent, selector: "riv-loading-cover", inputs: ["loading", "loadingSize", "errorMessage"] }, { kind: "directive", type: i7.OverlayDirective, selector: "[riv-overlay]" }, { kind: "component", type: i8.SearchComponent, selector: "riv-search", inputs: ["placeholder", "name", "labelTemplate"] }, { kind: "component", type: SelectComponent, selector: "riv-select", inputs: ["manager", "size", "disabled", "locked"] }, { kind: "component", type: i9.SelectNodeComponent, selector: "riv-select-node", inputs: ["mode", "node", "showSingleSelected"], outputs: ["selectChange", "expandChange"] }, { kind: "component", type: i10.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message", "title", "icon"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i11.NumberPipe, name: "rivNumber" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
126
+ SelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SelectComponent, selector: "riv-select", inputs: { manager: "manager", size: "size", disabled: "disabled", locked: "locked", allowedCalloutPositions: "allowedCalloutPositions", preferredCalloutPosition: "preferredCalloutPosition" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }, { propertyName: "customSelectedOptionsTemplate", first: true, predicate: ["customSelectedOptions"], descendants: true }, { propertyName: "optionTemplate", first: true, predicate: ["option"], descendants: true }, { propertyName: "headerTemplate", first: true, predicate: ["header"], descendants: true }, { propertyName: "footerTemplate", first: true, predicate: ["footer"], descendants: true }], viewQueries: [{ propertyName: "customTriggerButton", first: true, predicate: ["customTriggerButton"], descendants: true }, { propertyName: "standardTriggerButton", first: true, predicate: ["standardTriggerButton"], descendants: true }], usesInheritance: true, ngImport: i0, template: "<ng-container *ngIf=\"manager?.state | async; let s\">\n <ng-template #content>\n <riv-loading-cover [loading]=\"s.load.loading\">\n <div\n class=\"content-body\"\n [class.inline]=\"s.display.inline\"\n [ngStyle]=\"{\n 'max-height': s.display.maxHeight,\n 'max-width': s.display.maxWidth,\n 'min-height': s.display.minHeight,\n 'min-width': s.display.minWidth\n }\"\n >\n <header *ngIf=\"headerTemplate\">\n <ng-container\n *ngTemplateOutlet=\"headerTemplate; context: { state: s }\"\n ></ng-container>\n </header>\n <riv-search\n *ngIf=\"s.query.showSearch\"\n [value]=\"s.query.search\"\n [placeholder]=\"s.query.searchPlaceholder\"\n [autoFocus]=\"true\"\n (valueChange)=\"searchChange($event)\"\n ></riv-search>\n <div *ngIf=\"s.query.showOrder\">\n <span class=\"order-by\">Sort by:</span>\n <riv-select [manager]=\"orderByManager\">\n <ng-template #trigger let-state=\"state\">\n <button\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'small'\"\n [icon]=\"state.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [iconPosition]=\"'last'\"\n >\n {{ getSelectedOrderByOption(s)?.title }}\n </button>\n </ng-template>\n </riv-select>\n </div>\n <div class=\"options\">\n <riv-select-node\n *ngIf=\"\n s.selection.allowMultiSelect &&\n s.selection.allowSelectAll &&\n !s.display.zeroStateMessage\n \"\n [mode]=\"'multi'\"\n [node]=\"{\n id: '__NOOP__',\n title: s.query.search ? 'Select all matching' : 'Select all',\n selected: s.selection.visibleSelectionState,\n selectable: true,\n expandable: false,\n expanded: false\n }\"\n (selectChange)=\"\n manager?.actions?.next({ type: 'visibleSelectedChange' })\n \"\n ></riv-select-node>\n <div\n *ngFor=\"let group of s.fullOptionGroups; trackBy: trackBySymbol\"\n class=\"group\"\n [class.divider]=\"s.display.dividers\"\n >\n <span *ngIf=\"group.header; let header\" class=\"header\">\n {{ header }}\n </span>\n <ng-container\n *ngFor=\"let option of group.options; trackBy: trackByOption\"\n >\n <ng-container *ngIf=\"optionTemplate; else standardTemplate\">\n <ng-container\n [ngTemplateOutlet]=\"optionTemplate\"\n [ngTemplateOutletContext]=\"{ node: option }\"\n ></ng-container>\n </ng-container>\n <ng-template #standardTemplate>\n <riv-select-node\n [mode]=\"s.selection.allowMultiSelect ? 'multi' : 'single'\"\n [node]=\"option\"\n [showSingleSelected]=\"s.display.showSingleSelected\"\n (selectChange)=\"\n selectChange(s.selection.allowMultiSelect, $event)\n \"\n (expandChange)=\"\n manager?.actions?.next({\n type: 'toggleOptionExpanded',\n id: $event\n })\n \"\n ></riv-select-node>\n </ng-template>\n </ng-container>\n </div>\n <span\n *ngIf=\"s.display.displayLimitMessage; let message\"\n class=\"display-limit\"\n >\n {{ message }}\n </span>\n <riv-zero-state\n *ngIf=\"s.display.zeroStateMessage; let message\"\n [message]=\"message\"\n ></riv-zero-state>\n </div>\n <footer *ngIf=\"footerTemplate\" class=\"custom-footer\">\n <ng-container\n *ngTemplateOutlet=\"footerTemplate; context: { state: s }\"\n ></ng-container>\n </footer>\n </div>\n </riv-loading-cover>\n </ng-template>\n\n <ng-container *ngIf=\"!s.display.inline; else content\">\n <ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n class=\"custom-trigger-button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ state: s }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n [class.small]=\"size === 'small'\"\n [class.large]=\"size === 'large'\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n *ngIf=\"s.selection.selected.size; else placeholderValue\"\n >\n <ng-container\n *ngIf=\"\n customSelectedOptionsTemplate;\n else standardSelectedOptionsTemplate\n \"\n >\n <span class=\"value\">\n <ng-container\n [ngTemplateOutlet]=\"customSelectedOptionsTemplate\"\n [ngTemplateOutletContext]=\"{ selected: s.selection.selected }\"\n ></ng-container>\n </span>\n </ng-container>\n <ng-template #standardSelectedOptionsTemplate>\n <span class=\"value\">{{\n s.display.formattedSelectedOptions\n }}</span>\n <span *ngIf=\"s.display.pillCount > 1\" class=\"selected-count\">\n {{ s.display.pillCount | rivNumber }}\n </span>\n </ng-template>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ s.display.placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon\n *ngIf=\"!locked\"\n [name]=\"s.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [size]=\"20\"\n ></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"20\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n </ng-template>\n\n <ng-container *ngIf=\"s.display.open\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"allowedCalloutPositions\"\n [preferredPosition]=\"preferredCalloutPosition\"\n (close)=\"manager?.actions?.next({ type: 'openChange', open: false })\"\n >\n <ng-container [ngTemplateOutlet]=\"content\"></ng-container>\n </riv-callout>\n </ng-container>\n </ng-container>\n</ng-container>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small);background-color:var(--surface-light-0);cursor:pointer;padding-left:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1);cursor:default}.selected-count{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);font-weight:var(--font-weight-normal);color:var(--type-dark-body);border-radius:calc(var(--base-grid-size) * 3);background-color:var(--brand);padding:calc(var(--base-grid-size) / 4) calc(var(--base-grid-size) * 2);align-self:center}.value{font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) 0;flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.trigger.small .value{font:var(--input-small)}.trigger.large .value{font:var(--input-large);padding:var(--size-medium) var(--size-xsmall)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-xsmall) calc(var(--base-grid-size) * 1.5)}.trigger.warning{border-color:var(--surface-dark-caution)}.trigger.error{border-color:var(--surface-dark-danger);box-shadow:inset 0 0 0 var(--border-width-large) var(--surface-dark-danger)}.custom-trigger-button{max-width:100%}.order-by{font:var(--input-medium);color:var(--type-light-low-contrast)}.content-body{padding:var(--size-large);display:flex;flex-direction:column;gap:var(--size-large)}.content-body:not(.inline){max-height:50vh;min-width:calc(var(--base-grid-size) * 75);max-width:50vw}.options{flex-grow:1;overflow-y:auto;gap:var(--size-medium)}.options,.group{display:flex;flex-direction:column}.group{gap:var(--size-xsmall)}.group.divider:not(:last-child){padding-bottom:var(--size-medium);border-bottom:var(--border-width) solid var(--border-light)}.display-limit{font:var(--body-small);color:var(--type-light-disabled);padding:var(--size-small);text-align:center}.header{font:var(--input-medium);color:var(--type-light-low-contrast);padding:var(--size-xsmall) 0}.custom-footer{padding-top:var(--size-large);border-top:var(--border-width) solid var(--border-light)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.ButtonComponent, selector: "[rivButton]", inputs: ["locked", "disabled", "loading", "full", "size", "variant", "icon", "iconPosition", "current"] }, { kind: "component", type: i3.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "component", type: i4.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "component", type: i5.InputLabelComponent, selector: "riv-input-label", inputs: ["label", "help", "required", "labelActionText", "errorMessage", "state"], outputs: ["labelAction"] }, { kind: "component", type: i6.LoadingCoverComponent, selector: "riv-loading-cover", inputs: ["loading", "loadingSize", "errorMessage"] }, { kind: "directive", type: i7.OverlayDirective, selector: "[riv-overlay]" }, { kind: "component", type: i8.SearchComponent, selector: "riv-search", inputs: ["placeholder", "name", "labelTemplate"] }, { kind: "component", type: SelectComponent, selector: "riv-select", inputs: ["manager", "size", "disabled", "locked", "allowedCalloutPositions", "preferredCalloutPosition"] }, { kind: "component", type: i9.SelectNodeComponent, selector: "riv-select-node", inputs: ["mode", "node", "showSingleSelected"], outputs: ["selectChange", "expandChange"] }, { kind: "component", type: i10.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message", "title", "icon"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i11.NumberPipe, name: "rivNumber" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
118
127
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectComponent, decorators: [{
119
128
  type: Component,
120
- args: [{ selector: 'riv-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"manager?.state | async; let s\">\n <ng-template #content>\n <riv-loading-cover [loading]=\"s.load.loading\">\n <div\n class=\"content-body\"\n [class.inline]=\"s.display.inline\"\n [ngStyle]=\"{\n 'max-height': s.display.maxHeight,\n 'max-width': s.display.maxWidth,\n 'min-height': s.display.minHeight,\n 'min-width': s.display.minWidth\n }\"\n >\n <header *ngIf=\"headerTemplate\">\n <ng-container\n *ngTemplateOutlet=\"headerTemplate; context: { state: s }\"\n ></ng-container>\n </header>\n <riv-search\n *ngIf=\"s.query.showSearch\"\n [value]=\"s.query.search\"\n [placeholder]=\"s.query.searchPlaceholder\"\n [autoFocus]=\"true\"\n (valueChange)=\"searchChange($event)\"\n ></riv-search>\n <div *ngIf=\"s.query.showOrder\">\n <span class=\"order-by\">Sort by:</span>\n <riv-select [manager]=\"orderByManager\">\n <ng-template #trigger let-state=\"state\">\n <button\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'small'\"\n [icon]=\"state.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [iconPosition]=\"'last'\"\n >\n {{ getSelectedOrderByOption(s)?.title }}\n </button>\n </ng-template>\n </riv-select>\n </div>\n <div class=\"options\">\n <riv-select-node\n *ngIf=\"\n s.selection.allowMultiSelect &&\n s.selection.allowSelectAll &&\n !s.display.zeroStateMessage\n \"\n [mode]=\"'multi'\"\n [node]=\"{\n id: '__NOOP__',\n title: s.query.search ? 'Select all matching' : 'Select all',\n selected: s.selection.visibleSelectionState,\n selectable: true,\n expandable: false,\n expanded: false\n }\"\n (selectChange)=\"\n manager?.actions?.next({ type: 'visibleSelectedChange' })\n \"\n ></riv-select-node>\n <div\n *ngFor=\"let group of s.fullOptionGroups; trackBy: trackBySymbol\"\n class=\"group\"\n [class.divider]=\"s.display.dividers\"\n >\n <span *ngIf=\"group.header; let header\" class=\"header\">\n {{ header }}\n </span>\n <ng-container\n *ngFor=\"let option of group.options; trackBy: trackByOption\"\n >\n <ng-container *ngIf=\"optionTemplate; else standardTemplate\">\n <ng-container\n [ngTemplateOutlet]=\"optionTemplate\"\n [ngTemplateOutletContext]=\"{ node: option }\"\n ></ng-container>\n </ng-container>\n <ng-template #standardTemplate>\n <riv-select-node\n [mode]=\"s.selection.allowMultiSelect ? 'multi' : 'single'\"\n [node]=\"option\"\n [showSingleSelected]=\"s.display.showSingleSelected\"\n (selectChange)=\"\n selectChange(s.selection.allowMultiSelect, $event)\n \"\n (expandChange)=\"\n manager?.actions?.next({\n type: 'toggleOptionExpanded',\n id: $event\n })\n \"\n ></riv-select-node>\n </ng-template>\n </ng-container>\n </div>\n <span\n *ngIf=\"s.display.displayLimitMessage; let message\"\n class=\"display-limit\"\n >\n {{ message }}\n </span>\n <riv-zero-state\n *ngIf=\"s.display.zeroStateMessage; let message\"\n [message]=\"message\"\n ></riv-zero-state>\n </div>\n <footer *ngIf=\"footerTemplate\" class=\"custom-footer\">\n <ng-container\n *ngTemplateOutlet=\"footerTemplate; context: { state: s }\"\n ></ng-container>\n </footer>\n </div>\n </riv-loading-cover>\n </ng-template>\n\n <ng-container *ngIf=\"!s.display.inline; else content\">\n <ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n class=\"custom-trigger-button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ state: s }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n [class.small]=\"size === 'small'\"\n [class.large]=\"size === 'large'\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n *ngIf=\"s.selection.selected.size; else placeholderValue\"\n >\n <ng-container\n *ngIf=\"\n customSelectedOptionsTemplate;\n else standardSelectedOptionsTemplate\n \"\n >\n <span class=\"value\">\n <ng-container\n [ngTemplateOutlet]=\"customSelectedOptionsTemplate\"\n [ngTemplateOutletContext]=\"{ selected: s.selection.selected }\"\n ></ng-container>\n </span>\n </ng-container>\n <ng-template #standardSelectedOptionsTemplate>\n <span class=\"value\">{{\n s.display.formattedSelectedOptions\n }}</span>\n <span *ngIf=\"s.display.pillCount > 1\" class=\"selected-count\">\n {{ s.display.pillCount | rivNumber }}\n </span>\n </ng-template>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ s.display.placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon\n *ngIf=\"!locked\"\n [name]=\"s.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [size]=\"20\"\n ></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"20\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n </ng-template>\n\n <ng-container *ngIf=\"s.display.open\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"[\n 'top-left',\n 'top-center',\n 'top-right',\n 'bottom-right',\n 'bottom-center',\n 'bottom-left'\n ]\"\n (close)=\"manager?.actions?.next({ type: 'openChange', open: false })\"\n >\n <ng-container [ngTemplateOutlet]=\"content\"></ng-container>\n </riv-callout>\n </ng-container>\n </ng-container>\n</ng-container>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small);background-color:var(--surface-light-0);cursor:pointer;padding-left:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1);cursor:default}.selected-count{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);font-weight:var(--font-weight-normal);color:var(--type-dark-body);border-radius:calc(var(--base-grid-size) * 3);background-color:var(--brand);padding:calc(var(--base-grid-size) / 4) calc(var(--base-grid-size) * 2);align-self:center}.value{font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) 0;flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.trigger.small .value{font:var(--input-small)}.trigger.large .value{font:var(--input-large);padding:var(--size-medium) var(--size-xsmall)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-xsmall) calc(var(--base-grid-size) * 1.5)}.trigger.warning{border-color:var(--surface-dark-caution)}.trigger.error{border-color:var(--surface-dark-danger);box-shadow:inset 0 0 0 var(--border-width-large) var(--surface-dark-danger)}.custom-trigger-button{max-width:100%}.order-by{font:var(--input-medium);color:var(--type-light-low-contrast)}.content-body{padding:var(--size-large);display:flex;flex-direction:column;gap:var(--size-large)}.content-body:not(.inline){max-height:50vh;min-width:calc(var(--base-grid-size) * 75);max-width:50vw}.options{flex-grow:1;overflow-y:auto;gap:var(--size-medium)}.options,.group{display:flex;flex-direction:column}.group{gap:var(--size-xsmall)}.group.divider:not(:last-child){padding-bottom:var(--size-medium);border-bottom:var(--border-width) solid var(--border-light)}.display-limit{font:var(--body-small);color:var(--type-light-disabled);padding:var(--size-small);text-align:center}.header{font:var(--input-medium);color:var(--type-light-low-contrast);padding:var(--size-xsmall) 0}.custom-footer{padding-top:var(--size-large);border-top:var(--border-width) solid var(--border-light)}\n"] }]
129
+ args: [{ selector: 'riv-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"manager?.state | async; let s\">\n <ng-template #content>\n <riv-loading-cover [loading]=\"s.load.loading\">\n <div\n class=\"content-body\"\n [class.inline]=\"s.display.inline\"\n [ngStyle]=\"{\n 'max-height': s.display.maxHeight,\n 'max-width': s.display.maxWidth,\n 'min-height': s.display.minHeight,\n 'min-width': s.display.minWidth\n }\"\n >\n <header *ngIf=\"headerTemplate\">\n <ng-container\n *ngTemplateOutlet=\"headerTemplate; context: { state: s }\"\n ></ng-container>\n </header>\n <riv-search\n *ngIf=\"s.query.showSearch\"\n [value]=\"s.query.search\"\n [placeholder]=\"s.query.searchPlaceholder\"\n [autoFocus]=\"true\"\n (valueChange)=\"searchChange($event)\"\n ></riv-search>\n <div *ngIf=\"s.query.showOrder\">\n <span class=\"order-by\">Sort by:</span>\n <riv-select [manager]=\"orderByManager\">\n <ng-template #trigger let-state=\"state\">\n <button\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'small'\"\n [icon]=\"state.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [iconPosition]=\"'last'\"\n >\n {{ getSelectedOrderByOption(s)?.title }}\n </button>\n </ng-template>\n </riv-select>\n </div>\n <div class=\"options\">\n <riv-select-node\n *ngIf=\"\n s.selection.allowMultiSelect &&\n s.selection.allowSelectAll &&\n !s.display.zeroStateMessage\n \"\n [mode]=\"'multi'\"\n [node]=\"{\n id: '__NOOP__',\n title: s.query.search ? 'Select all matching' : 'Select all',\n selected: s.selection.visibleSelectionState,\n selectable: true,\n expandable: false,\n expanded: false\n }\"\n (selectChange)=\"\n manager?.actions?.next({ type: 'visibleSelectedChange' })\n \"\n ></riv-select-node>\n <div\n *ngFor=\"let group of s.fullOptionGroups; trackBy: trackBySymbol\"\n class=\"group\"\n [class.divider]=\"s.display.dividers\"\n >\n <span *ngIf=\"group.header; let header\" class=\"header\">\n {{ header }}\n </span>\n <ng-container\n *ngFor=\"let option of group.options; trackBy: trackByOption\"\n >\n <ng-container *ngIf=\"optionTemplate; else standardTemplate\">\n <ng-container\n [ngTemplateOutlet]=\"optionTemplate\"\n [ngTemplateOutletContext]=\"{ node: option }\"\n ></ng-container>\n </ng-container>\n <ng-template #standardTemplate>\n <riv-select-node\n [mode]=\"s.selection.allowMultiSelect ? 'multi' : 'single'\"\n [node]=\"option\"\n [showSingleSelected]=\"s.display.showSingleSelected\"\n (selectChange)=\"\n selectChange(s.selection.allowMultiSelect, $event)\n \"\n (expandChange)=\"\n manager?.actions?.next({\n type: 'toggleOptionExpanded',\n id: $event\n })\n \"\n ></riv-select-node>\n </ng-template>\n </ng-container>\n </div>\n <span\n *ngIf=\"s.display.displayLimitMessage; let message\"\n class=\"display-limit\"\n >\n {{ message }}\n </span>\n <riv-zero-state\n *ngIf=\"s.display.zeroStateMessage; let message\"\n [message]=\"message\"\n ></riv-zero-state>\n </div>\n <footer *ngIf=\"footerTemplate\" class=\"custom-footer\">\n <ng-container\n *ngTemplateOutlet=\"footerTemplate; context: { state: s }\"\n ></ng-container>\n </footer>\n </div>\n </riv-loading-cover>\n </ng-template>\n\n <ng-container *ngIf=\"!s.display.inline; else content\">\n <ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button\n #customTriggerButton\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n class=\"custom-trigger-button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"triggerTemplate\"\n [ngTemplateOutletContext]=\"{ state: s }\"\n ></ng-container>\n </button>\n </ng-container>\n <ng-template #standardTrigger>\n <riv-input-label\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n >\n <button\n #standardTriggerButton\n class=\"trigger\"\n [class.small]=\"size === 'small'\"\n [class.large]=\"size === 'large'\"\n [class.warning]=\"state === 'warning'\"\n [class.error]=\"state === 'error'\"\n (click)=\"allowedOpen()\"\n [disabled]=\"disabled || locked\"\n type=\"button\"\n >\n <ng-container\n *ngIf=\"s.selection.selected.size; else placeholderValue\"\n >\n <ng-container\n *ngIf=\"\n customSelectedOptionsTemplate;\n else standardSelectedOptionsTemplate\n \"\n >\n <span class=\"value\">\n <ng-container\n [ngTemplateOutlet]=\"customSelectedOptionsTemplate\"\n [ngTemplateOutletContext]=\"{ selected: s.selection.selected }\"\n ></ng-container>\n </span>\n </ng-container>\n <ng-template #standardSelectedOptionsTemplate>\n <span class=\"value\">{{\n s.display.formattedSelectedOptions\n }}</span>\n <span *ngIf=\"s.display.pillCount > 1\" class=\"selected-count\">\n {{ s.display.pillCount | rivNumber }}\n </span>\n </ng-template>\n </ng-container>\n <ng-template #placeholderValue>\n <span class=\"value placeholder\">{{ s.display.placeholder }}</span>\n </ng-template>\n <span class=\"chevron\">\n <riv-icon\n *ngIf=\"!locked\"\n [name]=\"s.display.open ? 'ChevronUp' : 'ChevronDown'\"\n [size]=\"20\"\n ></riv-icon>\n <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"20\"></riv-icon>\n </span>\n </button>\n </riv-input-label>\n </ng-template>\n\n <ng-container *ngIf=\"s.display.open\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [theme]=\"'light'\"\n [showCaret]=\"false\"\n [allowedPositions]=\"allowedCalloutPositions\"\n [preferredPosition]=\"preferredCalloutPosition\"\n (close)=\"manager?.actions?.next({ type: 'openChange', open: false })\"\n >\n <ng-container [ngTemplateOutlet]=\"content\"></ng-container>\n </riv-callout>\n </ng-container>\n </ng-container>\n</ng-container>\n", styles: [".trigger{width:100%;border:var(--border-width) solid var(--border-light);border-radius:var(--border-radius-small);display:flex;gap:var(--size-small);background-color:var(--surface-light-0);cursor:pointer;padding-left:var(--size-small)}.trigger:focus{outline:none;border:var(--border-width) solid var(--purp-60)}.trigger:disabled{color:var(--type-light-disabled);background-color:var(--surface-light-1);cursor:default}.selected-count{font-size:var(--type-1-font-size);line-height:var(--type-1-line-height-0);font-weight:var(--font-weight-normal);color:var(--type-dark-body);border-radius:calc(var(--base-grid-size) * 3);background-color:var(--brand);padding:calc(var(--base-grid-size) / 4) calc(var(--base-grid-size) * 2);align-self:center}.value{font:var(--input-medium);color:var(--type-light-high-contrast);padding:var(--size-small) 0;flex-grow:1;text-align:left;overflow:hidden;text-overflow:ellipsis;white-space:pre}.value.placeholder{color:var(--type-light-disabled)}.trigger.small .value{font:var(--input-small)}.trigger.large .value{font:var(--input-large);padding:var(--size-medium) var(--size-xsmall)}.chevron{display:flex;justify-content:center;align-items:center;padding:var(--size-xsmall) calc(var(--base-grid-size) * 1.5)}.trigger.warning{border-color:var(--surface-dark-caution)}.trigger.error{border-color:var(--surface-dark-danger);box-shadow:inset 0 0 0 var(--border-width-large) var(--surface-dark-danger)}.custom-trigger-button{max-width:100%}.order-by{font:var(--input-medium);color:var(--type-light-low-contrast)}.content-body{padding:var(--size-large);display:flex;flex-direction:column;gap:var(--size-large)}.content-body:not(.inline){max-height:50vh;min-width:calc(var(--base-grid-size) * 75);max-width:50vw}.options{flex-grow:1;overflow-y:auto;gap:var(--size-medium)}.options,.group{display:flex;flex-direction:column}.group{gap:var(--size-xsmall)}.group.divider:not(:last-child){padding-bottom:var(--size-medium);border-bottom:var(--border-width) solid var(--border-light)}.display-limit{font:var(--body-small);color:var(--type-light-disabled);padding:var(--size-small);text-align:center}.header{font:var(--input-medium);color:var(--type-light-low-contrast);padding:var(--size-xsmall) 0}.custom-footer{padding-top:var(--size-large);border-top:var(--border-width) solid var(--border-light)}\n"] }]
121
130
  }], propDecorators: { manager: [{
122
131
  type: Input
123
132
  }], size: [{
@@ -126,6 +135,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
126
135
  type: Input
127
136
  }], locked: [{
128
137
  type: Input
138
+ }], allowedCalloutPositions: [{
139
+ type: Input
140
+ }], preferredCalloutPosition: [{
141
+ type: Input
129
142
  }], triggerTemplate: [{
130
143
  type: ContentChild,
131
144
  args: ['trigger']
@@ -151,4 +164,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
151
164
  (function (SelectComponent) {
152
165
  SelectComponent.Sizes = ['small', 'medium', 'large'];
153
166
  })(SelectComponent || (SelectComponent = {}));
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/riv/src/lib/input/select/select/select.component.ts","../../../../../../../projects/riv/src/lib/input/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  ElementRef,\n  Input,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { debounce } from 'lodash';\nimport { distinctUntilChanged, map, skip, Subscription } from 'rxjs';\nimport { InputLabelComponent } from '../../input-label/input-label.component';\nimport { RivSelect } from '../state';\n\n@Component({\n  selector: 'riv-select',\n  templateUrl: './select.component.html',\n  styleUrls: ['./select.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SelectComponent<O extends RivSelect.BaseOption>\n  extends InputLabelComponent\n  implements OnInit, OnDestroy\n{\n  @Input()\n  manager?: RivSelect.Manager<O>;\n\n  @Input()\n  size: SelectComponent.Size = 'medium';\n\n  @Input()\n  disabled: boolean = false;\n\n  @Input()\n  locked: boolean = false;\n\n  @ContentChild('trigger')\n  triggerTemplate?: TemplateRef<{ state: RivSelect.FullState<O> }>;\n\n  @ContentChild('customSelectedOptions')\n  customSelectedOptionsTemplate?: TemplateRef<{\n    selected: RivSelect.OptionSet<O>;\n  }>;\n\n  @ContentChild('option')\n  optionTemplate?: TemplateRef<{ node: RivSelect.FullOption<O> }>;\n\n  @ContentChild('header')\n  headerTemplate?: TemplateRef<{ state: RivSelect.FullState<O> }>;\n\n  @ContentChild('footer')\n  footerTemplate?: TemplateRef<{ state: RivSelect.FullState<O> }>;\n\n  @ViewChild('customTriggerButton')\n  customTriggerButton?: ElementRef;\n\n  @ViewChild('standardTriggerButton')\n  standardTriggerButton?: ElementRef;\n\n  orderByManager?: RivSelect.Manager<SelectComponent.OrderByOption>;\n  private orderBySubscription?: Subscription;\n\n  ngOnInit(): void {\n    this.orderByManager = RivSelect.createInMemoryManager(\n      () => this.orderByOptionGroups,\n      {\n        showSingleSelected: true,\n        initialState: {\n          selection: {\n            selected: new RivSelect.OptionSet([\n              this.orderByOptionGroups[0].options.find(\n                ({ id }) => id === 'title',\n              )!,\n            ]),\n          },\n        },\n      },\n    );\n\n    this.orderBySubscription = this.orderByManager.state\n      .pipe(\n        map(s => [...s.selection.selected][0]?.id ?? null),\n        distinctUntilChanged((prev, current) => prev === current),\n        skip(1),\n      )\n      .subscribe(order => {\n        this.manager?.actions.next({\n          type: 'orderChange',\n          payload: order,\n        });\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.orderBySubscription?.unsubscribe();\n  }\n\n  getTrigger(): Element | null {\n    return (\n      this.customTriggerButton?.nativeElement ??\n      this.standardTriggerButton?.nativeElement\n    );\n  }\n\n  allowedOpen(): void {\n    if (!this.disabled && !this.locked) {\n      this.manager?.actions.next({\n        type: 'openChange',\n        open: true,\n      });\n    }\n  }\n\n  selectChange(multi: boolean, id: O['id']) {\n    if (multi) {\n      this.manager?.actions.next({\n        type: 'toggleOptionSelected',\n        id,\n      });\n    } else {\n      this.manager?.actions.next({\n        type: 'setSelectedOption',\n        id,\n      });\n      this.manager?.actions.next({\n        type: 'openChange',\n        open: false,\n      });\n    }\n  }\n\n  readonly orderByOptionGroups: RivSelect.OptionGroup<SelectComponent.OrderByOption>[] =\n    [\n      {\n        options: [\n          {\n            id: 'title',\n            title: 'A-Z',\n            help: 'Sorts values in alphanumeric order (A-Z, 0-9)',\n          },\n          {\n            id: 'count',\n            title: 'Volume',\n            help: 'Sorts values by count (highest to lowest)',\n          },\n        ],\n      },\n    ];\n\n  getSelectedOrderByOption(\n    state: RivSelect.FullState<O>,\n  ): SelectComponent.OrderByOption | null {\n    return (\n      this.orderByOptionGroups[0].options.find(\n        ({ id }) => id === state.query.order,\n      ) ?? null\n    );\n  }\n\n  searchChange = debounce((query: string) => {\n    this.manager?.actions?.next({ type: 'searchChange', payload: query });\n  }, 500);\n\n  // trackBySymbol for our option groups that don't have a good identity.\n  // this lazily attaches a Javascript symbol to any value and returns it.\n  // Symbols are guaranteed unique and they do not exist in JSON, so they\n  // do not interfere with any types or serialization for API calls. The\n  // only purpose they serve is to tell ngFor which objects are still in\n  // the array. Note: this is only safe on an array of objects.\n  trackBySymbol(_: number, value: any) {\n    if (!value._trackingSymbol) {\n      value._trackingSymbol = Symbol();\n    }\n    return value._trackingSymbol;\n  }\n\n  trackByOption(_: number, option: RivSelect.FullOption<O>): string | number {\n    return option.id;\n  }\n}\n\nexport namespace SelectComponent {\n  export type OrderByOption = {\n    id: RivSelect.OrderBy;\n    title: string;\n    help: string;\n  };\n\n  export const Sizes = ['small', 'medium', 'large'] as const;\n  export type Size = (typeof Sizes)[number];\n}\n","<ng-container *ngIf=\"manager?.state | async; let s\">\n  <ng-template #content>\n    <riv-loading-cover [loading]=\"s.load.loading\">\n      <div\n        class=\"content-body\"\n        [class.inline]=\"s.display.inline\"\n        [ngStyle]=\"{\n          'max-height': s.display.maxHeight,\n          'max-width': s.display.maxWidth,\n          'min-height': s.display.minHeight,\n          'min-width': s.display.minWidth\n        }\"\n      >\n        <header *ngIf=\"headerTemplate\">\n          <ng-container\n            *ngTemplateOutlet=\"headerTemplate; context: { state: s }\"\n          ></ng-container>\n        </header>\n        <riv-search\n          *ngIf=\"s.query.showSearch\"\n          [value]=\"s.query.search\"\n          [placeholder]=\"s.query.searchPlaceholder\"\n          [autoFocus]=\"true\"\n          (valueChange)=\"searchChange($event)\"\n        ></riv-search>\n        <div *ngIf=\"s.query.showOrder\">\n          <span class=\"order-by\">Sort by:</span>\n          <riv-select [manager]=\"orderByManager\">\n            <ng-template #trigger let-state=\"state\">\n              <button\n                rivButton\n                [variant]=\"'ghost'\"\n                [size]=\"'small'\"\n                [icon]=\"state.display.open ? 'ChevronUp' : 'ChevronDown'\"\n                [iconPosition]=\"'last'\"\n              >\n                {{ getSelectedOrderByOption(s)?.title }}\n              </button>\n            </ng-template>\n          </riv-select>\n        </div>\n        <div class=\"options\">\n          <riv-select-node\n            *ngIf=\"\n              s.selection.allowMultiSelect &&\n              s.selection.allowSelectAll &&\n              !s.display.zeroStateMessage\n            \"\n            [mode]=\"'multi'\"\n            [node]=\"{\n              id: '__NOOP__',\n              title: s.query.search ? 'Select all matching' : 'Select all',\n              selected: s.selection.visibleSelectionState,\n              selectable: true,\n              expandable: false,\n              expanded: false\n            }\"\n            (selectChange)=\"\n              manager?.actions?.next({ type: 'visibleSelectedChange' })\n            \"\n          ></riv-select-node>\n          <div\n            *ngFor=\"let group of s.fullOptionGroups; trackBy: trackBySymbol\"\n            class=\"group\"\n            [class.divider]=\"s.display.dividers\"\n          >\n            <span *ngIf=\"group.header; let header\" class=\"header\">\n              {{ header }}\n            </span>\n            <ng-container\n              *ngFor=\"let option of group.options; trackBy: trackByOption\"\n            >\n              <ng-container *ngIf=\"optionTemplate; else standardTemplate\">\n                <ng-container\n                  [ngTemplateOutlet]=\"optionTemplate\"\n                  [ngTemplateOutletContext]=\"{ node: option }\"\n                ></ng-container>\n              </ng-container>\n              <ng-template #standardTemplate>\n                <riv-select-node\n                  [mode]=\"s.selection.allowMultiSelect ? 'multi' : 'single'\"\n                  [node]=\"option\"\n                  [showSingleSelected]=\"s.display.showSingleSelected\"\n                  (selectChange)=\"\n                    selectChange(s.selection.allowMultiSelect, $event)\n                  \"\n                  (expandChange)=\"\n                    manager?.actions?.next({\n                      type: 'toggleOptionExpanded',\n                      id: $event\n                    })\n                  \"\n                ></riv-select-node>\n              </ng-template>\n            </ng-container>\n          </div>\n          <span\n            *ngIf=\"s.display.displayLimitMessage; let message\"\n            class=\"display-limit\"\n          >\n            {{ message }}\n          </span>\n          <riv-zero-state\n            *ngIf=\"s.display.zeroStateMessage; let message\"\n            [message]=\"message\"\n          ></riv-zero-state>\n        </div>\n        <footer *ngIf=\"footerTemplate\" class=\"custom-footer\">\n          <ng-container\n            *ngTemplateOutlet=\"footerTemplate; context: { state: s }\"\n          ></ng-container>\n        </footer>\n      </div>\n    </riv-loading-cover>\n  </ng-template>\n\n  <ng-container *ngIf=\"!s.display.inline; else content\">\n    <ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n      <button\n        #customTriggerButton\n        (click)=\"allowedOpen()\"\n        [disabled]=\"disabled || locked\"\n        type=\"button\"\n        class=\"custom-trigger-button\"\n      >\n        <ng-container\n          [ngTemplateOutlet]=\"triggerTemplate\"\n          [ngTemplateOutletContext]=\"{ state: s }\"\n        ></ng-container>\n      </button>\n    </ng-container>\n    <ng-template #standardTrigger>\n      <riv-input-label\n        [label]=\"label\"\n        [help]=\"help\"\n        [required]=\"required\"\n        [state]=\"state\"\n        [errorMessage]=\"errorMessage\"\n        [labelActionText]=\"labelActionText\"\n        (labelAction)=\"labelAction.emit($event)\"\n      >\n        <button\n          #standardTriggerButton\n          class=\"trigger\"\n          [class.small]=\"size === 'small'\"\n          [class.large]=\"size === 'large'\"\n          [class.warning]=\"state === 'warning'\"\n          [class.error]=\"state === 'error'\"\n          (click)=\"allowedOpen()\"\n          [disabled]=\"disabled || locked\"\n          type=\"button\"\n        >\n          <ng-container\n            *ngIf=\"s.selection.selected.size; else placeholderValue\"\n          >\n            <ng-container\n              *ngIf=\"\n                customSelectedOptionsTemplate;\n                else standardSelectedOptionsTemplate\n              \"\n            >\n              <span class=\"value\">\n                <ng-container\n                  [ngTemplateOutlet]=\"customSelectedOptionsTemplate\"\n                  [ngTemplateOutletContext]=\"{ selected: s.selection.selected }\"\n                ></ng-container>\n              </span>\n            </ng-container>\n            <ng-template #standardSelectedOptionsTemplate>\n              <span class=\"value\">{{\n                s.display.formattedSelectedOptions\n              }}</span>\n              <span *ngIf=\"s.display.pillCount > 1\" class=\"selected-count\">\n                {{ s.display.pillCount | rivNumber }}\n              </span>\n            </ng-template>\n          </ng-container>\n          <ng-template #placeholderValue>\n            <span class=\"value placeholder\">{{ s.display.placeholder }}</span>\n          </ng-template>\n          <span class=\"chevron\">\n            <riv-icon\n              *ngIf=\"!locked\"\n              [name]=\"s.display.open ? 'ChevronUp' : 'ChevronDown'\"\n              [size]=\"20\"\n            ></riv-icon>\n            <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"20\"></riv-icon>\n          </span>\n        </button>\n      </riv-input-label>\n    </ng-template>\n\n    <ng-container *ngIf=\"s.display.open\">\n      <riv-callout\n        *riv-overlay\n        [anchor]=\"getTrigger()\"\n        [theme]=\"'light'\"\n        [showCaret]=\"false\"\n        [allowedPositions]=\"[\n          'top-left',\n          'top-center',\n          'top-right',\n          'bottom-right',\n          'bottom-center',\n          'bottom-left'\n        ]\"\n        (close)=\"manager?.actions?.next({ type: 'openChange', open: false })\"\n      >\n        <ng-container [ngTemplateOutlet]=\"content\"></ng-container>\n      </riv-callout>\n    </ng-container>\n  </ng-container>\n</ng-container>\n"]}
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/riv/src/lib/input/select/select/select.component.ts","../../../../../../../projects/riv/src/lib/input/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  ElementRef,\n  Input,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { debounce } from 'lodash';\nimport { distinctUntilChanged, map, skip, Subscription } from 'rxjs';\nimport { InputLabelComponent } from '../../input-label/input-label.component';\nimport { RivSelect } from '../state';\nimport { CalloutComponent } from '../../../overlay/callout/callout.component';\n\n@Component({\n  selector: 'riv-select',\n  templateUrl: './select.component.html',\n  styleUrls: ['./select.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class SelectComponent<O extends RivSelect.BaseOption>\n  extends InputLabelComponent\n  implements OnInit, OnDestroy\n{\n  @Input()\n  manager?: RivSelect.Manager<O>;\n\n  @Input()\n  size: SelectComponent.Size = 'medium';\n\n  @Input()\n  disabled: boolean = false;\n\n  @Input()\n  locked: boolean = false;\n\n  @Input()\n  allowedCalloutPositions: CalloutComponent.AnchoredPosition[] = [\n    'top-left',\n    'top-center',\n    'top-right',\n    'bottom-right',\n    'bottom-center',\n    'bottom-left',\n  ];\n\n  @Input()\n  preferredCalloutPosition: CalloutComponent.AnchoredPosition = 'bottom-right';\n\n  @ContentChild('trigger')\n  triggerTemplate?: TemplateRef<{ state: RivSelect.FullState<O> }>;\n\n  @ContentChild('customSelectedOptions')\n  customSelectedOptionsTemplate?: TemplateRef<{\n    selected: RivSelect.OptionSet<O>;\n  }>;\n\n  @ContentChild('option')\n  optionTemplate?: TemplateRef<{ node: RivSelect.FullOption<O> }>;\n\n  @ContentChild('header')\n  headerTemplate?: TemplateRef<{ state: RivSelect.FullState<O> }>;\n\n  @ContentChild('footer')\n  footerTemplate?: TemplateRef<{ state: RivSelect.FullState<O> }>;\n\n  @ViewChild('customTriggerButton')\n  customTriggerButton?: ElementRef;\n\n  @ViewChild('standardTriggerButton')\n  standardTriggerButton?: ElementRef;\n\n  orderByManager?: RivSelect.Manager<SelectComponent.OrderByOption>;\n  private orderBySubscription?: Subscription;\n\n  ngOnInit(): void {\n    this.orderByManager = RivSelect.createInMemoryManager(\n      () => this.orderByOptionGroups,\n      {\n        showSingleSelected: true,\n        initialState: {\n          selection: {\n            selected: new RivSelect.OptionSet([\n              this.orderByOptionGroups[0].options.find(\n                ({ id }) => id === 'title',\n              )!,\n            ]),\n          },\n        },\n      },\n    );\n\n    this.orderBySubscription = this.orderByManager.state\n      .pipe(\n        map(s => [...s.selection.selected][0]?.id ?? null),\n        distinctUntilChanged((prev, current) => prev === current),\n        skip(1),\n      )\n      .subscribe(order => {\n        this.manager?.actions.next({\n          type: 'orderChange',\n          payload: order,\n        });\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.orderBySubscription?.unsubscribe();\n  }\n\n  getTrigger(): Element | null {\n    return (\n      this.customTriggerButton?.nativeElement ??\n      this.standardTriggerButton?.nativeElement\n    );\n  }\n\n  allowedOpen(): void {\n    if (!this.disabled && !this.locked) {\n      this.manager?.actions.next({\n        type: 'openChange',\n        open: true,\n      });\n    }\n  }\n\n  selectChange(multi: boolean, id: O['id']) {\n    if (multi) {\n      this.manager?.actions.next({\n        type: 'toggleOptionSelected',\n        id,\n      });\n    } else {\n      this.manager?.actions.next({\n        type: 'setSelectedOption',\n        id,\n      });\n      this.manager?.actions.next({\n        type: 'openChange',\n        open: false,\n      });\n    }\n  }\n\n  readonly orderByOptionGroups: RivSelect.OptionGroup<SelectComponent.OrderByOption>[] =\n    [\n      {\n        options: [\n          {\n            id: 'title',\n            title: 'A-Z',\n            help: 'Sorts values in alphanumeric order (A-Z, 0-9)',\n          },\n          {\n            id: 'count',\n            title: 'Volume',\n            help: 'Sorts values by count (highest to lowest)',\n          },\n        ],\n      },\n    ];\n\n  getSelectedOrderByOption(\n    state: RivSelect.FullState<O>,\n  ): SelectComponent.OrderByOption | null {\n    return (\n      this.orderByOptionGroups[0].options.find(\n        ({ id }) => id === state.query.order,\n      ) ?? null\n    );\n  }\n\n  searchChange = debounce((query: string) => {\n    this.manager?.actions?.next({ type: 'searchChange', payload: query });\n  }, 500);\n\n  // trackBySymbol for our option groups that don't have a good identity.\n  // this lazily attaches a Javascript symbol to any value and returns it.\n  // Symbols are guaranteed unique and they do not exist in JSON, so they\n  // do not interfere with any types or serialization for API calls. The\n  // only purpose they serve is to tell ngFor which objects are still in\n  // the array. Note: this is only safe on an array of objects.\n  trackBySymbol(_: number, value: any) {\n    if (!value._trackingSymbol) {\n      value._trackingSymbol = Symbol();\n    }\n    return value._trackingSymbol;\n  }\n\n  trackByOption(_: number, option: RivSelect.FullOption<O>): string | number {\n    return option.id;\n  }\n}\n\nexport namespace SelectComponent {\n  export type OrderByOption = {\n    id: RivSelect.OrderBy;\n    title: string;\n    help: string;\n  };\n\n  export const Sizes = ['small', 'medium', 'large'] as const;\n  export type Size = (typeof Sizes)[number];\n}\n","<ng-container *ngIf=\"manager?.state | async; let s\">\n  <ng-template #content>\n    <riv-loading-cover [loading]=\"s.load.loading\">\n      <div\n        class=\"content-body\"\n        [class.inline]=\"s.display.inline\"\n        [ngStyle]=\"{\n          'max-height': s.display.maxHeight,\n          'max-width': s.display.maxWidth,\n          'min-height': s.display.minHeight,\n          'min-width': s.display.minWidth\n        }\"\n      >\n        <header *ngIf=\"headerTemplate\">\n          <ng-container\n            *ngTemplateOutlet=\"headerTemplate; context: { state: s }\"\n          ></ng-container>\n        </header>\n        <riv-search\n          *ngIf=\"s.query.showSearch\"\n          [value]=\"s.query.search\"\n          [placeholder]=\"s.query.searchPlaceholder\"\n          [autoFocus]=\"true\"\n          (valueChange)=\"searchChange($event)\"\n        ></riv-search>\n        <div *ngIf=\"s.query.showOrder\">\n          <span class=\"order-by\">Sort by:</span>\n          <riv-select [manager]=\"orderByManager\">\n            <ng-template #trigger let-state=\"state\">\n              <button\n                rivButton\n                [variant]=\"'ghost'\"\n                [size]=\"'small'\"\n                [icon]=\"state.display.open ? 'ChevronUp' : 'ChevronDown'\"\n                [iconPosition]=\"'last'\"\n              >\n                {{ getSelectedOrderByOption(s)?.title }}\n              </button>\n            </ng-template>\n          </riv-select>\n        </div>\n        <div class=\"options\">\n          <riv-select-node\n            *ngIf=\"\n              s.selection.allowMultiSelect &&\n              s.selection.allowSelectAll &&\n              !s.display.zeroStateMessage\n            \"\n            [mode]=\"'multi'\"\n            [node]=\"{\n              id: '__NOOP__',\n              title: s.query.search ? 'Select all matching' : 'Select all',\n              selected: s.selection.visibleSelectionState,\n              selectable: true,\n              expandable: false,\n              expanded: false\n            }\"\n            (selectChange)=\"\n              manager?.actions?.next({ type: 'visibleSelectedChange' })\n            \"\n          ></riv-select-node>\n          <div\n            *ngFor=\"let group of s.fullOptionGroups; trackBy: trackBySymbol\"\n            class=\"group\"\n            [class.divider]=\"s.display.dividers\"\n          >\n            <span *ngIf=\"group.header; let header\" class=\"header\">\n              {{ header }}\n            </span>\n            <ng-container\n              *ngFor=\"let option of group.options; trackBy: trackByOption\"\n            >\n              <ng-container *ngIf=\"optionTemplate; else standardTemplate\">\n                <ng-container\n                  [ngTemplateOutlet]=\"optionTemplate\"\n                  [ngTemplateOutletContext]=\"{ node: option }\"\n                ></ng-container>\n              </ng-container>\n              <ng-template #standardTemplate>\n                <riv-select-node\n                  [mode]=\"s.selection.allowMultiSelect ? 'multi' : 'single'\"\n                  [node]=\"option\"\n                  [showSingleSelected]=\"s.display.showSingleSelected\"\n                  (selectChange)=\"\n                    selectChange(s.selection.allowMultiSelect, $event)\n                  \"\n                  (expandChange)=\"\n                    manager?.actions?.next({\n                      type: 'toggleOptionExpanded',\n                      id: $event\n                    })\n                  \"\n                ></riv-select-node>\n              </ng-template>\n            </ng-container>\n          </div>\n          <span\n            *ngIf=\"s.display.displayLimitMessage; let message\"\n            class=\"display-limit\"\n          >\n            {{ message }}\n          </span>\n          <riv-zero-state\n            *ngIf=\"s.display.zeroStateMessage; let message\"\n            [message]=\"message\"\n          ></riv-zero-state>\n        </div>\n        <footer *ngIf=\"footerTemplate\" class=\"custom-footer\">\n          <ng-container\n            *ngTemplateOutlet=\"footerTemplate; context: { state: s }\"\n          ></ng-container>\n        </footer>\n      </div>\n    </riv-loading-cover>\n  </ng-template>\n\n  <ng-container *ngIf=\"!s.display.inline; else content\">\n    <ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n      <button\n        #customTriggerButton\n        (click)=\"allowedOpen()\"\n        [disabled]=\"disabled || locked\"\n        type=\"button\"\n        class=\"custom-trigger-button\"\n      >\n        <ng-container\n          [ngTemplateOutlet]=\"triggerTemplate\"\n          [ngTemplateOutletContext]=\"{ state: s }\"\n        ></ng-container>\n      </button>\n    </ng-container>\n    <ng-template #standardTrigger>\n      <riv-input-label\n        [label]=\"label\"\n        [help]=\"help\"\n        [required]=\"required\"\n        [state]=\"state\"\n        [errorMessage]=\"errorMessage\"\n        [labelActionText]=\"labelActionText\"\n        (labelAction)=\"labelAction.emit($event)\"\n      >\n        <button\n          #standardTriggerButton\n          class=\"trigger\"\n          [class.small]=\"size === 'small'\"\n          [class.large]=\"size === 'large'\"\n          [class.warning]=\"state === 'warning'\"\n          [class.error]=\"state === 'error'\"\n          (click)=\"allowedOpen()\"\n          [disabled]=\"disabled || locked\"\n          type=\"button\"\n        >\n          <ng-container\n            *ngIf=\"s.selection.selected.size; else placeholderValue\"\n          >\n            <ng-container\n              *ngIf=\"\n                customSelectedOptionsTemplate;\n                else standardSelectedOptionsTemplate\n              \"\n            >\n              <span class=\"value\">\n                <ng-container\n                  [ngTemplateOutlet]=\"customSelectedOptionsTemplate\"\n                  [ngTemplateOutletContext]=\"{ selected: s.selection.selected }\"\n                ></ng-container>\n              </span>\n            </ng-container>\n            <ng-template #standardSelectedOptionsTemplate>\n              <span class=\"value\">{{\n                s.display.formattedSelectedOptions\n              }}</span>\n              <span *ngIf=\"s.display.pillCount > 1\" class=\"selected-count\">\n                {{ s.display.pillCount | rivNumber }}\n              </span>\n            </ng-template>\n          </ng-container>\n          <ng-template #placeholderValue>\n            <span class=\"value placeholder\">{{ s.display.placeholder }}</span>\n          </ng-template>\n          <span class=\"chevron\">\n            <riv-icon\n              *ngIf=\"!locked\"\n              [name]=\"s.display.open ? 'ChevronUp' : 'ChevronDown'\"\n              [size]=\"20\"\n            ></riv-icon>\n            <riv-icon *ngIf=\"locked\" [name]=\"'Lock'\" [size]=\"20\"></riv-icon>\n          </span>\n        </button>\n      </riv-input-label>\n    </ng-template>\n\n    <ng-container *ngIf=\"s.display.open\">\n      <riv-callout\n        *riv-overlay\n        [anchor]=\"getTrigger()\"\n        [theme]=\"'light'\"\n        [showCaret]=\"false\"\n        [allowedPositions]=\"allowedCalloutPositions\"\n        [preferredPosition]=\"preferredCalloutPosition\"\n        (close)=\"manager?.actions?.next({ type: 'openChange', open: false })\"\n      >\n        <ng-container [ngTemplateOutlet]=\"content\"></ng-container>\n      </riv-callout>\n    </ng-container>\n  </ng-container>\n</ng-container>\n"]}
@@ -103,7 +103,7 @@ export class SimpleSelectComponent extends InputLabelComponent {
103
103
  }
104
104
  }
105
105
  SimpleSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SimpleSelectComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
106
- SimpleSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SimpleSelectComponent, selector: "riv-simple-select", inputs: { optionGroups: "optionGroups", selectedOption: "selectedOption", size: "size", disabled: "disabled", locked: "locked", filterabilityOptions: "filterabilityOptions", orderBy: "orderBy", placeholder: "placeholder", maxHeight: "maxHeight", maxWidth: "maxWidth", minHeight: "minHeight", minWidth: "minWidth", dividers: "dividers" }, outputs: { selectedOptionChange: "selectedOptionChange" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }, { propertyName: "optionTemplate", first: true, predicate: ["option"], descendants: true }, { propertyName: "headerTemplate", first: true, predicate: ["header"], descendants: true }, { propertyName: "footerTemplate", first: true, predicate: ["footer"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<riv-select\n [manager]=\"manager\"\n [size]=\"size\"\n [disabled]=\"disabled\"\n [locked]=\"locked\"\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <ng-container *ngIf=\"triggerTemplate; let triggerTpl\">\n <ng-template #trigger let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"triggerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"optionTemplate; let optionTpl\">\n <ng-template #option let-node=\"node\">\n <button\n class=\"custom-node\"\n [disabled]=\"node.disabled\"\n (click)=\"\n manager?.actions?.next({\n type: 'setSelectedOption',\n id: node.id\n });\n manager?.actions?.next({\n type: 'openChange',\n open: false\n })\n \"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"optionTpl\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"headerTemplate; let headerTpl\">\n <ng-template #header let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"headerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"footerTemplate; let footerTpl\">\n <ng-template #footer let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"footerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n</riv-select>\n", styles: [".custom-node{display:block;width:100%;text-align:left}.custom-node:not(:disabled){cursor:pointer}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.SelectComponent, selector: "riv-select", inputs: ["manager", "size", "disabled", "locked"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
106
+ SimpleSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: SimpleSelectComponent, selector: "riv-simple-select", inputs: { optionGroups: "optionGroups", selectedOption: "selectedOption", size: "size", disabled: "disabled", locked: "locked", filterabilityOptions: "filterabilityOptions", orderBy: "orderBy", placeholder: "placeholder", maxHeight: "maxHeight", maxWidth: "maxWidth", minHeight: "minHeight", minWidth: "minWidth", dividers: "dividers" }, outputs: { selectedOptionChange: "selectedOptionChange" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }, { propertyName: "optionTemplate", first: true, predicate: ["option"], descendants: true }, { propertyName: "headerTemplate", first: true, predicate: ["header"], descendants: true }, { propertyName: "footerTemplate", first: true, predicate: ["footer"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<riv-select\n [manager]=\"manager\"\n [size]=\"size\"\n [disabled]=\"disabled\"\n [locked]=\"locked\"\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <ng-container *ngIf=\"triggerTemplate; let triggerTpl\">\n <ng-template #trigger let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"triggerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"optionTemplate; let optionTpl\">\n <ng-template #option let-node=\"node\">\n <button\n class=\"custom-node\"\n [disabled]=\"node.disabled\"\n (click)=\"\n manager?.actions?.next({\n type: 'setSelectedOption',\n id: node.id\n });\n manager?.actions?.next({\n type: 'openChange',\n open: false\n })\n \"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"optionTpl\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"headerTemplate; let headerTpl\">\n <ng-template #header let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"headerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"footerTemplate; let footerTpl\">\n <ng-template #footer let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"footerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n</riv-select>\n", styles: [".custom-node{display:block;width:100%;text-align:left}.custom-node:not(:disabled){cursor:pointer}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.SelectComponent, selector: "riv-select", inputs: ["manager", "size", "disabled", "locked", "allowedCalloutPositions", "preferredCalloutPosition"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
107
107
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SimpleSelectComponent, decorators: [{
108
108
  type: Component,
109
109
  args: [{ selector: 'riv-simple-select', changeDetection: ChangeDetectionStrategy.OnPush, template: "<riv-select\n [manager]=\"manager\"\n [size]=\"size\"\n [disabled]=\"disabled\"\n [locked]=\"locked\"\n [label]=\"label\"\n [help]=\"help\"\n [required]=\"required\"\n [state]=\"state\"\n [errorMessage]=\"errorMessage\"\n [labelActionText]=\"labelActionText\"\n (labelAction)=\"labelAction.emit($event)\"\n>\n <ng-container *ngIf=\"triggerTemplate; let triggerTpl\">\n <ng-template #trigger let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"triggerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"optionTemplate; let optionTpl\">\n <ng-template #option let-node=\"node\">\n <button\n class=\"custom-node\"\n [disabled]=\"node.disabled\"\n (click)=\"\n manager?.actions?.next({\n type: 'setSelectedOption',\n id: node.id\n });\n manager?.actions?.next({\n type: 'openChange',\n open: false\n })\n \"\n type=\"button\"\n >\n <ng-container\n [ngTemplateOutlet]=\"optionTpl\"\n [ngTemplateOutletContext]=\"{ node }\"\n ></ng-container>\n </button>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"headerTemplate; let headerTpl\">\n <ng-template #header let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"headerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n\n <ng-container *ngIf=\"footerTemplate; let footerTpl\">\n <ng-template #footer let-state=\"state\">\n <ng-container\n [ngTemplateOutlet]=\"footerTpl\"\n [ngTemplateOutletContext]=\"{ state }\"\n ></ng-container>\n </ng-template>\n </ng-container>\n</riv-select>\n", styles: [".custom-node{display:block;width:100%;text-align:left}.custom-node:not(:disabled){cursor:pointer}\n"] }]