Signal drop!
Relay (operand.online) is unreachable.
Usually, a dropped signal means an upgrade is happening. Hold on!
Sorry, no connección.
Hang in there while we get back on track
gram: card
> ./src/ui/AutosuggestInput.tsx
Lenses
(coming soon!)
import React, {useState} from "react";
import Autosuggest from 'react-autosuggest';
import {functions} from '../formula'
const autosuggestTheme = {
container: {
display: 'inline-block',
position: 'relative',
minWidth: '50%',
marginLeft: '10px',
zIndex: '2500',
color: 'black',
fontFamily: '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol"',
fontSize: '12px',
},
input: {
padding: '5px',
border: 'solid thin #ddd',
height: '1.5em',
width: '100%',
fontFamily: '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol"',
fontSize: '14px',
'&:focus': {
border: 'none'
}
},
suggestionsContainer: {
display: 'none',
},
suggestionsContainerOpen: {
display: 'block',
position: 'absolute',
width: '100%',
minHeight: '80px',
maxHeight: '240px',
background: 'rgba(255,255,255,0.8)',
boxShadow: '0px 0px 3px gray',
overflowY: 'auto',
},
suggestionsList: {
padding: '0px',
margin: '0px',
listStyleType: 'none',
},
suggestion: {
margin: '0',
padding: '0px 5px',
cursor: 'pointer'
},
suggestionHighlighted: {
background: 'rgb(200,200,200,0.4)',
},
sectionContainer: {
padding: '0px 5px',
borderTop: '1px solid #ccc'
},
sectionTitle: {
fontVariantCaps: 'all-small-caps',
fontSize: '10px',
color: 'gray'
}
};
const AutosuggestInput = ({activeCellValue, setActiveCellValue, suggestions, setSuggestions, cellEditorRef, attributes, onCellEditorKeyPress, commitActiveCellValue}) => {
const [prefix, setPrefix] = useState('')
// This pattern matches every thing after and including any one of these symbols: = ( + - * / ,
const regex = /[=(\+\-\*\/,][^=(\+\-\*\/,]*$/;
// Teach Autosuggest how to calculate suggestions for any given input value.
const attributeNames = attributes.map(attribute => attribute.name);
const mathSymbols = {"Plus": "+", "Minus": "-", "Multiply": "*", "Divide": "/"};
const mathSuggestions = Object.keys(functions)
.filter(functionName => Object.keys(mathSymbols).indexOf(functionName) !== -1)
.reduce((obj, functionName) => {
const symbol = mathSymbols[functionName];
obj[symbol] = functions[functionName];
return obj;
}, {});
const allFunctionNames = Object.keys(functions)
.sort()
.filter(functionName => Object.keys(mathSymbols).indexOf(functionName) === -1)
const allSuggestions = [
{title: "Functions", suggestions: allFunctionNames},
{title: "Columns", suggestions: attributeNames}
]
const getSuggestions = value => {
const inputValue = value.trim()
const matchIndex = inputValue.search(regex);
// save the prefix of the input (everything up to the suggestion)
const curPrefix = value.slice(0, matchIndex+1);
if (curPrefix !== prefix) {
setPrefix(curPrefix);
}
if (matchIndex === inputValue.length - 1) {
// If at the start of a new expression, include all possible suggestions
return allSuggestions;
}
else if (matchIndex === -1) {
return [];
}
else {
// Use everything after the regex match index (matchValue) as the prefix to determine suggestions
const matchValue = inputValue.slice(matchIndex+1, inputValue.length).toLowerCase().trim();
let filteredSuggestions = allSuggestions
.map(section => {
return {
title: section.title,
suggestions: section.suggestions.filter(suggestion => suggestion.toLowerCase().slice(0, matchValue.length) === matchValue)
};
})
.filter(section => section.suggestions.length > 0);
// Math operations are suggested after a numeric attribute name or after ")"
// const attributeIndex = attributeNames.indexOf(matchValue);
// const isNumericAttribute = attributeIndex != -1 && attributes[attributeIndex].type === "numeric";
// const lastChar = matchValue[matchValue.length - 1];
// if (isNumericAttribute || lastChar === ")"){
// suggestions = suggestions.concat(Object.keys(mathSuggestions))
// }
return filteredSuggestions;
}
};
const getSuggestionValue = function(suggestion) {
return prefix[prefix.length-1] === "," ? prefix + " " + suggestion : prefix + suggestion;
}
// Determine how individual suggestions are rendered into HTML.
const renderSuggestion = function(suggestion, {query}) {
return(<div>{suggestion}</div>)
}
// Render helper text for functions in the footer
const renderSuggestionsContainer = function({ containerProps, children }) {
let inputValue = activeCellValue.toString().trim()
const lastChar = inputValue[inputValue.length-1]
if (lastChar === "(" || lastChar === ",") {
inputValue = inputValue.slice(0, inputValue.lastIndexOf("("))
}
const matchIndex = inputValue.search(regex);
const matchValue = inputValue.slice(matchIndex+1, inputValue.length).trim();
const attributeIndex = attributeNames.indexOf(matchValue);
let footer = undefined;
if (matchValue in functions) { // function
const params = functions[matchValue]["help"];
footer = (
<div>
{matchValue + "("} <b>{Object.keys(params).join(", ")}</b> {")"}
<div>{Object.keys(params).map(key => <div><b>{key}</b>: {params[key]}</div>)}</div>
</div>
)
}
else if (lastChar in mathSuggestions) { // math symbol
const helpText = mathSuggestions[lastChar]["help"];
footer = (
<div>
numeric1 <b>{lastChar}</b> numeric2
<div>{helpText}</div>
</div>
)
}
else if (attributeIndex != -1) { // attribute name
const attribute = attributes[attributeIndex];
footer = (
<div>
<b>{attribute.name}</b> is a column with type <b>{attribute.type}</b>.
</div>
)
}
return (
<div {...containerProps}>
{children}
<div style={footer ? {position: 'absolute', top: '0', right: '0', bottom: '0', width: '70%', height: '150%', borderTop: '1px solid rgb(204, 204, 204)', fontSize: '12px'} : {display: 'none'}}>
<div style={{position: 'sticky', top: '0', padding: '5px', border: '1px solid rgb(204, 204, 204)'}}>
<div style={{fontVariantCaps: 'all-small-caps', fontSize: '10px', color: 'gray'}} >Documentation</div>
{footer}
</div>
</div>
</div>
)
}
const onChange = function(event, { newValue }) {
setActiveCellValue(newValue);
}
// Autosuggest will call this function every time we need to update suggestions.
const onSuggestionsFetchRequested = function({ value }) {
setSuggestions(getSuggestions(value));
};
// Autosuggest will call this function every time we need to clear suggestions.
const onSuggestionsClearRequested = function() {
setSuggestions([]);
};
const renderSectionTitle = function(section) {
return <strong>{section.title}</strong>;
}
const getSectionSuggestions = function(section) {
return section.suggestions;
}
return <Autosuggest
suggestions={suggestions}
onSuggestionsFetchRequested={onSuggestionsFetchRequested}
onSuggestionsClearRequested={onSuggestionsClearRequested}
getSuggestionValue={getSuggestionValue}
renderSuggestion={renderSuggestion}
renderSuggestionsContainer={renderSuggestionsContainer}
multiSection={true}
renderSectionTitle={renderSectionTitle}
getSectionSuggestions={getSectionSuggestions}
inputProps={{
ref: cellEditorRef,
value: activeCellValue.toString(),
onChange: onChange,
onKeyPress: onCellEditorKeyPress,
placeholder: "Enter cell value, or enter \"=\" to begin a formula...",
onBlur: commitActiveCellValue}}
theme={autosuggestTheme}
/>
}
export default AutosuggestInput;