UNPKG

kepler.gl

Version:

kepler.gl is a webgl based application to visualize large scale location data in the browser

337 lines (335 loc) 57.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _typeof = require("@babel/runtime/helpers/typeof"); Object.defineProperty(exports, "__esModule", { value: true }); exports.AiAssistantConfig = AiAssistantConfig; var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator")); var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator")); var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray")); var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _react = _interopRequireWildcard(require("react")); var _styledComponents = _interopRequireDefault(require("styled-components")); var _reactRedux = require("react-redux"); var _components = require("@kepler.gl/components"); var _apiKey = _interopRequireDefault(require("../icons/api-key")); var _models = _interopRequireDefault(require("../config/models.json")); var _usehooksTs = require("usehooks-ts"); var _core = require("@openassistant/core"); var _actions = require("../actions"); var _reactIntl = require("react-intl"); var _templateObject, _templateObject2, _templateObject3, _templateObject4, _templateObject5, _templateObject6, _templateObject7, _templateObject8; // SPDX-License-Identifier: MIT // Copyright contributors to the kepler.gl project function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; } var SectionTitle = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n font-size: ", ";\n color: ", ";\n text-transform: capitalize;\n"])), function (props) { return props.theme.inputFontSize; }, function (props) { return props.theme.effectPanelTextSecondary1; }); var StyledAiAssistantConfig = _styledComponents["default"].div(_templateObject2 || (_templateObject2 = (0, _taggedTemplateLiteral2["default"])(["\n padding: 12px;\n font-size: ", ";\n display: flex;\n flex-direction: column;\n gap: 12px;\n width: 100%;\n height: 100%;\n\n .api-key-input {\n box-shadow: ", ";\n width: 100%;\n .api-key-input__icon {\n position: absolute;\n height: ", "px;\n width: 30px;\n padding-left: 6px;\n display: flex;\n align-items: center;\n justify-content: center;\n color: ", ";\n }\n\n input {\n padding: 4px 36px;\n height: ", "px;\n caret-color: unset;\n }\n }\n"])), function (props) { return props.theme.primaryBtnFontSizeDefault; }, function (props) { return props.theme.boxShadow; }, function (props) { return props.theme.geocoderInputHeight; }, function (props) { return props.theme.subtextColor; }, function (props) { return props.theme.geocoderInputHeight; }); // Ollama model input wrapper: checkbox + 'Input Model Name:' + input // all children element have width based on the content var OllamaModelInputWrapper = _styledComponents["default"].div(_templateObject3 || (_templateObject3 = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n flex-direction: row;\n gap: 4px;\n align-items: center;\n"]))); var StyledWrapper = _styledComponents["default"].div(_templateObject4 || (_templateObject4 = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n justify-content: space-between;\n align-items: center;\n"]))); var StyledItemSelector = (0, _styledComponents["default"])(_components.ItemSelector)(_templateObject5 || (_templateObject5 = (0, _taggedTemplateLiteral2["default"])(["\n .item-selector__dropdown {\n padding-left: 10px;\n border-radius: 4px;\n }\n .active {\n border-color: ", ";\n border-radius: 4px 4px 0px 0px !important;\n }\n width: 100%;\n"])), function (props) { return props.theme.activeColor; }); var StyleSliderWrapper = _styledComponents["default"].div(_templateObject6 || (_templateObject6 = (0, _taggedTemplateLiteral2["default"])(["\n width: 100%;\n align-self: flex-start;\n height: 32px;\n display: flex;\n align-items: center;\n .kg-range-slider__input {\n height: 32px;\n text-align: center;\n padding: 3px 6px;\n }\n .kg-slider {\n padding-left: 6px;\n }\n .kg-range-slider {\n padding: 0px !important;\n }\n"]))); var StyledButton = _styledComponents["default"].div(_templateObject7 || (_templateObject7 = (0, _taggedTemplateLiteral2["default"])(["\n width: 100%;\n align-self: flex-start;\n margin-top: 12px;\n\n button div {\n display: flex;\n align-items: center;\n gap: 4px;\n margin-right: 4px;\n }\n"]))); var StyleErrorMessage = _styledComponents["default"].div(_templateObject8 || (_templateObject8 = (0, _taggedTemplateLiteral2["default"])(["\n font-size: ", ";\n background-color: ", ";\n border-radius: 4px;\n padding: 4px 8px;\n color: ", ";\n"])), function (props) { return props.theme.primaryBtnFontSizeDefault; }, function (props) { return props.theme.errorColor; }, function (props) { return props.theme.errorTextColor; }); var RangeSlider = _components.appInjector.get(_components.RangeSliderFactory); function AiAssistantConfig() { var dispatch = (0, _reactRedux.useDispatch)(); var aiAssistantConfig = (0, _reactRedux.useSelector)(function (state) { return state.demo.aiAssistant.config; }); var intl = (0, _reactIntl.useIntl)(); var _useLocalStorage = (0, _usehooksTs.useLocalStorage)('ai-assistant-provider', aiAssistantConfig.provider || 'openai'), _useLocalStorage2 = (0, _slicedToArray2["default"])(_useLocalStorage, 2), provider = _useLocalStorage2[0], setProvider = _useLocalStorage2[1]; var _useLocalStorage3 = (0, _usehooksTs.useLocalStorage)('ai-assistant-model', aiAssistantConfig.model || _models["default"][provider][0]), _useLocalStorage4 = (0, _slicedToArray2["default"])(_useLocalStorage3, 2), model = _useLocalStorage4[0], setModel = _useLocalStorage4[1]; var _useLocalStorage5 = (0, _usehooksTs.useLocalStorage)('ai-assistant-api-key', aiAssistantConfig.apiKey || ''), _useLocalStorage6 = (0, _slicedToArray2["default"])(_useLocalStorage5, 2), apiKey = _useLocalStorage6[0], setApiKey = _useLocalStorage6[1]; var _useLocalStorage7 = (0, _usehooksTs.useLocalStorage)('ai-assistant-temperature', aiAssistantConfig.temperature || 0.0), _useLocalStorage8 = (0, _slicedToArray2["default"])(_useLocalStorage7, 2), temperature = _useLocalStorage8[0], setTemperature = _useLocalStorage8[1]; var _useLocalStorage9 = (0, _usehooksTs.useLocalStorage)('ai-assistant-top-p', aiAssistantConfig.topP || 1.0), _useLocalStorage10 = (0, _slicedToArray2["default"])(_useLocalStorage9, 2), topP = _useLocalStorage10[0], setTopP = _useLocalStorage10[1]; var _useLocalStorage11 = (0, _usehooksTs.useLocalStorage)('ai-assistant-base-url', aiAssistantConfig.baseUrl || 'http://localhost:11434/api'), _useLocalStorage12 = (0, _slicedToArray2["default"])(_useLocalStorage11, 2), baseUrl = _useLocalStorage12[0], setBaseUrl = _useLocalStorage12[1]; var _useLocalStorage13 = (0, _usehooksTs.useLocalStorage)('ai-assistant-mapbox-token', aiAssistantConfig.mapboxToken || ''), _useLocalStorage14 = (0, _slicedToArray2["default"])(_useLocalStorage13, 2), mapboxToken = _useLocalStorage14[0], setMapboxToken = _useLocalStorage14[1]; var _useState = (0, _react.useState)(false), _useState2 = (0, _slicedToArray2["default"])(_useState, 2), ollamaModelInputChecked = _useState2[0], setOllamaModelInputChecked = _useState2[1]; var _useState3 = (0, _react.useState)(''), _useState4 = (0, _slicedToArray2["default"])(_useState3, 2), ollamaModelInputValue = _useState4[0], setOllamaModelInputValue = _useState4[1]; var _useState5 = (0, _react.useState)(false), _useState6 = (0, _slicedToArray2["default"])(_useState5, 2), connectionError = _useState6[0], setConnectionError = _useState6[1]; var _useState7 = (0, _react.useState)(''), _useState8 = (0, _slicedToArray2["default"])(_useState7, 2), errorMessage = _useState8[0], setErrorMessage = _useState8[1]; var _useState9 = (0, _react.useState)(false), _useState10 = (0, _slicedToArray2["default"])(_useState9, 2), isRunning = _useState10[0], setIsRunning = _useState10[1]; var onAiProviderSelect = function onAiProviderSelect(value) { if (typeof value === 'string') { setProvider(value); setModel(_models["default"][value][0]); setConnectionError(false); setErrorMessage(''); } }; var onLLMModelSelect = function onLLMModelSelect(value) { if (typeof value === 'string') { setModel(value); } }; var onApiKeyChange = function onApiKeyChange(e) { setApiKey(e.target.value); // reset previous key error if any setConnectionError(false); setErrorMessage(''); }; var onTemperatureChange = function onTemperatureChange(value) { setTemperature(value[1]); }; var onTopPChange = function onTopPChange(value) { setTopP(value[1]); }; var onBaseUrlChange = function onBaseUrlChange(e) { setBaseUrl(e.target.value); setConnectionError(false); setErrorMessage(''); }; var onMapboxTokenChange = function onMapboxTokenChange(e) { setMapboxToken(e.target.value); }; var onOllamaModelInputChecked = function onOllamaModelInputChecked(e) { setOllamaModelInputChecked(e.target.checked); if (!e.target.checked) { // use model from selector setModel(''); } }; var onOllamaModelInputValueChange = function onOllamaModelInputValueChange(e) { setOllamaModelInputValue(e.target.value); setModel(e.target.value); }; var onStartChat = /*#__PURE__*/function () { var _ref = (0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee() { var timeoutPromise, AssistantModel, success, _errorMessage; return _regenerator["default"].wrap(function _callee$(_context) { while (1) switch (_context.prev = _context.next) { case 0: setIsRunning(true); _context.prev = 1; timeoutPromise = new Promise(function (_, reject) { setTimeout(function () { return reject(new Error('Connection timeout after 15 seconds')); }, 15000); }); AssistantModel = (0, _core.GetAssistantModelByProvider)({ provider: provider }); _context.next = 6; return Promise.race([AssistantModel === null || AssistantModel === void 0 ? void 0 : AssistantModel.testConnection(apiKey, model), timeoutPromise]); case 6: success = _context.sent; _errorMessage = !success ? provider === 'ollama' ? 'Connection failed: maybe invalid Ollama Base URL' : 'Connection failed: maybe invalid API Key' : ''; setConnectionError(!success); setErrorMessage(_errorMessage); dispatch((0, _actions.updateAiAssistantConfig)({ provider: provider, model: model, apiKey: apiKey, baseUrl: baseUrl, isReady: success, temperature: temperature, topP: topP, mapboxToken: mapboxToken })); _context.next = 17; break; case 13: _context.prev = 13; _context.t0 = _context["catch"](1); setConnectionError(true); setErrorMessage(_context.t0 instanceof Error ? _context.t0.message : 'Connection failed'); case 17: _context.prev = 17; setIsRunning(false); return _context.finish(17); case 20: case "end": return _context.stop(); } }, _callee, null, [[1, 13, 17, 20]]); })); return function onStartChat() { return _ref.apply(this, arguments); }; }(); return /*#__PURE__*/_react["default"].createElement(StyledAiAssistantConfig, { className: "ai-assistant-config__type" }, /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.aiProvider" }))), /*#__PURE__*/_react["default"].createElement(StyledWrapper, null, /*#__PURE__*/_react["default"].createElement(StyledItemSelector, { selectedItems: provider, options: Object.keys(_models["default"]), multiSelect: false, disabled: false, onChange: onAiProviderSelect, filterOption: "name", getOptionValue: function getOptionValue(op) { return op; }, displayOption: function displayOption(op) { return op; }, searchable: false, showArrow: true })), /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.llmModel.title" }))), (provider === 'ollama' && !ollamaModelInputChecked || provider !== 'ollama') && /*#__PURE__*/_react["default"].createElement(StyledWrapper, null, /*#__PURE__*/_react["default"].createElement(StyledItemSelector, { selectedItems: model, options: _models["default"][provider], multiSelect: false, disabled: provider === 'ollama' ? ollamaModelInputChecked : false, placeholder: "Select LLM Model", onChange: onLLMModelSelect, filterOption: "name", getOptionValue: function getOptionValue(op) { return op; }, displayOption: function displayOption(op) { return op; }, searchable: false, showArrow: true })), provider === 'ollama' && /*#__PURE__*/_react["default"].createElement(OllamaModelInputWrapper, null, /*#__PURE__*/_react["default"].createElement("div", { style: { width: '250px' } }, /*#__PURE__*/_react["default"].createElement(_components.Checkbox, { id: "ollama-model-input", label: "Input Model Name", onChange: onOllamaModelInputChecked, checked: ollamaModelInputChecked })), /*#__PURE__*/_react["default"].createElement(_components.Input, { type: "text", onChange: onOllamaModelInputValueChange, placeholder: "Enter Model Name", value: ollamaModelInputValue, disabled: !ollamaModelInputChecked })), provider !== 'ollama' ? /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.apiKey.placeholder" }))), /*#__PURE__*/_react["default"].createElement("div", { className: "api-key-input" }, /*#__PURE__*/_react["default"].createElement("div", { className: "api-key-input__icon" }, /*#__PURE__*/_react["default"].createElement(_apiKey["default"], { height: "20px" })), /*#__PURE__*/_react["default"].createElement(_components.Input, { type: "text", onChange: onApiKeyChange, placeholder: intl.formatMessage({ id: 'aiAssistantManager.apiKey.placeholder' }), value: apiKey }))) : /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.baseUrl.placeholder" }))), /*#__PURE__*/_react["default"].createElement("div", { className: "api-key-input" }, /*#__PURE__*/_react["default"].createElement("div", { className: "api-key-input__icon" }, /*#__PURE__*/_react["default"].createElement(_apiKey["default"], { height: "20px" })), /*#__PURE__*/_react["default"].createElement(_components.Input, { type: "text", onChange: onBaseUrlChange, placeholder: intl.formatMessage({ id: 'aiAssistantManager.baseUrl.placeholder' }), value: baseUrl }))), connectionError && /*#__PURE__*/_react["default"].createElement(StyleErrorMessage, { className: "error-message" }, errorMessage), /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.temperature.title" }))), /*#__PURE__*/_react["default"].createElement(StyleSliderWrapper, null, /*#__PURE__*/_react["default"].createElement(RangeSlider, { showInput: true, isRanged: false, value0: 0, value1: temperature, onChange: onTemperatureChange, range: [0, 2], step: 0.1 })), /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.topP.title" }))), /*#__PURE__*/_react["default"].createElement(StyleSliderWrapper, null, /*#__PURE__*/_react["default"].createElement(RangeSlider, { showInput: true, isRanged: false, value0: 0, value1: topP, onChange: onTopPChange, range: [0, 1], step: 0.1 })), /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, /*#__PURE__*/_react["default"].createElement(_components.PanelLabelWrapper, null, /*#__PURE__*/_react["default"].createElement(SectionTitle, null, "Mapbox Token (optional for route/isochrone)")), /*#__PURE__*/_react["default"].createElement("div", { className: "api-key-input" }, /*#__PURE__*/_react["default"].createElement("div", { className: "api-key-input__icon" }, /*#__PURE__*/_react["default"].createElement(_apiKey["default"], { height: "20px" })), /*#__PURE__*/_react["default"].createElement(_components.Input, { type: "text", onChange: onMapboxTokenChange, placeholder: "Enter your Mapbox Token", value: mapboxToken }))), /*#__PURE__*/_react["default"].createElement(StyledButton, null, /*#__PURE__*/_react["default"].createElement(_components.Button, { onClick: onStartChat, width: '100%' }, isRunning && /*#__PURE__*/_react["default"].createElement(_components.LoadingSpinner, { size: 12 }), /*#__PURE__*/_react["default"].createElement(_reactIntl.FormattedMessage, { id: "aiAssistantManager.startChat" })))); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_styledComponents","_interopRequireDefault","_reactRedux","_components","_apiKey","_models","_usehooksTs","_core","_actions","_reactIntl","_templateObject","_templateObject2","_templateObject3","_templateObject4","_templateObject5","_templateObject6","_templateObject7","_templateObject8","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","_typeof","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","SectionTitle","styled","div","_taggedTemplateLiteral2","props","theme","inputFontSize","effectPanelTextSecondary1","StyledAiAssistantConfig","primaryBtnFontSizeDefault","boxShadow","geocoderInputHeight","subtextColor","OllamaModelInputWrapper","StyledWrapper","StyledItemSelector","ItemSelector","activeColor","StyleSliderWrapper","StyledButton","StyleErrorMessage","errorColor","errorTextColor","RangeSlider","appInjector","RangeSliderFactory","AiAssistantConfig","dispatch","useDispatch","aiAssistantConfig","useSelector","state","demo","aiAssistant","config","intl","useIntl","_useLocalStorage","useLocalStorage","provider","_useLocalStorage2","_slicedToArray2","setProvider","_useLocalStorage3","model","PROVIDER_MODELS","_useLocalStorage4","setModel","_useLocalStorage5","apiKey","_useLocalStorage6","setApiKey","_useLocalStorage7","temperature","_useLocalStorage8","setTemperature","_useLocalStorage9","topP","_useLocalStorage10","setTopP","_useLocalStorage11","baseUrl","_useLocalStorage12","setBaseUrl","_useLocalStorage13","mapboxToken","_useLocalStorage14","setMapboxToken","_useState","useState","_useState2","ollamaModelInputChecked","setOllamaModelInputChecked","_useState3","_useState4","ollamaModelInputValue","setOllamaModelInputValue","_useState5","_useState6","connectionError","setConnectionError","_useState7","_useState8","errorMessage","setErrorMessage","_useState9","_useState10","isRunning","setIsRunning","onAiProviderSelect","value","onLLMModelSelect","onApiKeyChange","target","onTemperatureChange","onTopPChange","onBaseUrlChange","onMapboxTokenChange","onOllamaModelInputChecked","checked","onOllamaModelInputValueChange","onStartChat","_ref","_asyncToGenerator2","_regenerator","mark","_callee","timeoutPromise","AssistantModel","success","_errorMessage","wrap","_callee$","_context","prev","next","Promise","_","reject","setTimeout","Error","GetAssistantModelByProvider","race","testConnection","sent","updateAiAssistantConfig","isReady","t0","message","finish","stop","apply","arguments","createElement","className","PanelLabelWrapper","FormattedMessage","id","selectedItems","options","keys","multiSelect","disabled","onChange","filterOption","getOptionValue","op","displayOption","searchable","showArrow","placeholder","style","width","Checkbox","label","Input","type","Fragment","height","formatMessage","showInput","isRanged","value0","value1","range","step","Button","onClick","LoadingSpinner","size"],"sources":["../../src/components/ai-assistant-config.tsx"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport React, {useState} from 'react';\nimport styled from 'styled-components';\nimport {useSelector, useDispatch} from 'react-redux';\nimport {\n  Input,\n  PanelLabelWrapper,\n  ItemSelector,\n  RangeSliderFactory,\n  Button,\n  LoadingSpinner,\n  appInjector,\n  Checkbox\n} from '@kepler.gl/components';\nimport {State} from '../index';\nimport ApiKey from '../icons/api-key';\nimport PROVIDER_MODELS from '../config/models.json';\nimport {useLocalStorage} from 'usehooks-ts';\nimport {GetAssistantModelByProvider} from '@openassistant/core';\nimport {updateAiAssistantConfig} from '../actions';\nimport {FormattedMessage, useIntl} from 'react-intl';\n\nconst SectionTitle = styled.div`\n  font-size: ${props => props.theme.inputFontSize};\n  color: ${props => props.theme.effectPanelTextSecondary1};\n  text-transform: capitalize;\n`;\n\nconst StyledAiAssistantConfig = styled.div`\n  padding: 12px;\n  font-size: ${props => props.theme.primaryBtnFontSizeDefault};\n  display: flex;\n  flex-direction: column;\n  gap: 12px;\n  width: 100%;\n  height: 100%;\n\n  .api-key-input {\n    box-shadow: ${props => props.theme.boxShadow};\n    width: 100%;\n    .api-key-input__icon {\n      position: absolute;\n      height: ${props => props.theme.geocoderInputHeight}px;\n      width: 30px;\n      padding-left: 6px;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      color: ${props => props.theme.subtextColor};\n    }\n\n    input {\n      padding: 4px 36px;\n      height: ${props => props.theme.geocoderInputHeight}px;\n      caret-color: unset;\n    }\n  }\n`;\n\n// Ollama model input wrapper: checkbox + 'Input Model Name:' + input\n// all children element have width based on the content\nconst OllamaModelInputWrapper = styled.div`\n  display: flex;\n  flex-direction: row;\n  gap: 4px;\n  align-items: center;\n`;\n\nconst StyledWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: center;\n`;\n\nconst StyledItemSelector = styled(ItemSelector)`\n  .item-selector__dropdown {\n    padding-left: 10px;\n    border-radius: 4px;\n  }\n  .active {\n    border-color: ${props => props.theme.activeColor};\n    border-radius: 4px 4px 0px 0px !important;\n  }\n  width: 100%;\n`;\n\nconst StyleSliderWrapper = styled.div`\n  width: 100%;\n  align-self: flex-start;\n  height: 32px;\n  display: flex;\n  align-items: center;\n  .kg-range-slider__input {\n    height: 32px;\n    text-align: center;\n    padding: 3px 6px;\n  }\n  .kg-slider {\n    padding-left: 6px;\n  }\n  .kg-range-slider {\n    padding: 0px !important;\n  }\n`;\n\nconst StyledButton = styled.div`\n  width: 100%;\n  align-self: flex-start;\n  margin-top: 12px;\n\n  button div {\n    display: flex;\n    align-items: center;\n    gap: 4px;\n    margin-right: 4px;\n  }\n`;\n\nconst StyleErrorMessage = styled.div`\n  font-size: ${props => props.theme.primaryBtnFontSizeDefault};\n  background-color: ${props => props.theme.errorColor};\n  border-radius: 4px;\n  padding: 4px 8px;\n  color: ${props => props.theme.errorTextColor};\n`;\n\nconst RangeSlider = appInjector.get(RangeSliderFactory);\n\nexport function AiAssistantConfig() {\n  const dispatch = useDispatch();\n  const aiAssistantConfig = useSelector((state: State) => state.demo.aiAssistant.config);\n  const intl = useIntl();\n\n  const [provider, setProvider] = useLocalStorage(\n    'ai-assistant-provider',\n    aiAssistantConfig.provider || 'openai'\n  );\n  const [model, setModel] = useLocalStorage(\n    'ai-assistant-model',\n    aiAssistantConfig.model || PROVIDER_MODELS[provider][0]\n  );\n  const [apiKey, setApiKey] = useLocalStorage(\n    'ai-assistant-api-key',\n    aiAssistantConfig.apiKey || ''\n  );\n  const [temperature, setTemperature] = useLocalStorage(\n    'ai-assistant-temperature',\n    aiAssistantConfig.temperature || 0.0\n  );\n  const [topP, setTopP] = useLocalStorage('ai-assistant-top-p', aiAssistantConfig.topP || 1.0);\n  const [baseUrl, setBaseUrl] = useLocalStorage(\n    'ai-assistant-base-url',\n    aiAssistantConfig.baseUrl || 'http://localhost:11434/api'\n  );\n  const [mapboxToken, setMapboxToken] = useLocalStorage(\n    'ai-assistant-mapbox-token',\n    aiAssistantConfig.mapboxToken || ''\n  );\n  const [ollamaModelInputChecked, setOllamaModelInputChecked] = useState(false);\n  const [ollamaModelInputValue, setOllamaModelInputValue] = useState('');\n  const [connectionError, setConnectionError] = useState(false);\n  const [errorMessage, setErrorMessage] = useState('');\n  const [isRunning, setIsRunning] = useState(false);\n\n  const onAiProviderSelect = (value: string | number | boolean | object | null) => {\n    if (typeof value === 'string') {\n      setProvider(value);\n      setModel(PROVIDER_MODELS[value][0]);\n      setConnectionError(false);\n      setErrorMessage('');\n    }\n  };\n\n  const onLLMModelSelect = (value: string | number | boolean | object | null) => {\n    if (typeof value === 'string') {\n      setModel(value);\n    }\n  };\n\n  const onApiKeyChange = (e: React.ChangeEvent<HTMLInputElement>) => {\n    setApiKey(e.target.value);\n    // reset previous key error if any\n    setConnectionError(false);\n    setErrorMessage('');\n  };\n\n  const onTemperatureChange = (value: number[]) => {\n    setTemperature(value[1]);\n  };\n\n  const onTopPChange = (value: number[]) => {\n    setTopP(value[1]);\n  };\n\n  const onBaseUrlChange = (e: React.ChangeEvent<HTMLInputElement>) => {\n    setBaseUrl(e.target.value);\n    setConnectionError(false);\n    setErrorMessage('');\n  };\n\n  const onMapboxTokenChange = (e: React.ChangeEvent<HTMLInputElement>) => {\n    setMapboxToken(e.target.value);\n  };\n\n  const onOllamaModelInputChecked = (e: React.ChangeEvent<HTMLInputElement>) => {\n    setOllamaModelInputChecked(e.target.checked);\n    if (!e.target.checked) {\n      // use model from selector\n      setModel('');\n    }\n  };\n\n  const onOllamaModelInputValueChange = (e: React.ChangeEvent<HTMLInputElement>) => {\n    setOllamaModelInputValue(e.target.value);\n    setModel(e.target.value);\n  };\n\n  const onStartChat = async () => {\n    setIsRunning(true);\n    try {\n      const timeoutPromise = new Promise((_, reject) => {\n        setTimeout(() => reject(new Error('Connection timeout after 15 seconds')), 15000);\n      });\n\n      const AssistantModel = GetAssistantModelByProvider({\n        provider: provider\n      });\n\n      const success = (await Promise.race([\n        AssistantModel?.testConnection(apiKey, model),\n        timeoutPromise\n      ])) as boolean;\n\n      const errorMessage = !success\n        ? provider === 'ollama'\n          ? 'Connection failed: maybe invalid Ollama Base URL'\n          : 'Connection failed: maybe invalid API Key'\n        : '';\n      setConnectionError(!success);\n      setErrorMessage(errorMessage);\n      dispatch(\n        updateAiAssistantConfig({\n          provider: provider,\n          model: model,\n          apiKey: apiKey,\n          baseUrl: baseUrl,\n          isReady: success,\n          temperature: temperature,\n          topP: topP,\n          mapboxToken: mapboxToken\n        })\n      );\n    } catch (error) {\n      setConnectionError(true);\n      setErrorMessage(error instanceof Error ? error.message : 'Connection failed');\n    } finally {\n      setIsRunning(false);\n    }\n  };\n\n  return (\n    <StyledAiAssistantConfig className=\"ai-assistant-config__type\">\n      <PanelLabelWrapper>\n        <SectionTitle>\n          <FormattedMessage id=\"aiAssistantManager.aiProvider\" />\n        </SectionTitle>\n      </PanelLabelWrapper>\n      <StyledWrapper>\n        <StyledItemSelector\n          selectedItems={provider}\n          options={Object.keys(PROVIDER_MODELS)}\n          multiSelect={false}\n          disabled={false}\n          onChange={onAiProviderSelect}\n          filterOption=\"name\"\n          getOptionValue={op => op}\n          displayOption={op => op}\n          searchable={false}\n          showArrow={true}\n        />\n      </StyledWrapper>\n      <PanelLabelWrapper>\n        <SectionTitle>\n          <FormattedMessage id=\"aiAssistantManager.llmModel.title\" />\n        </SectionTitle>\n      </PanelLabelWrapper>\n      {((provider === 'ollama' && !ollamaModelInputChecked) || provider !== 'ollama') && (\n        <StyledWrapper>\n          <StyledItemSelector\n            selectedItems={model}\n            options={PROVIDER_MODELS[provider]}\n            multiSelect={false}\n            disabled={provider === 'ollama' ? ollamaModelInputChecked : false}\n            placeholder=\"Select LLM Model\"\n            onChange={onLLMModelSelect}\n            filterOption=\"name\"\n            getOptionValue={op => op}\n            displayOption={op => op}\n            searchable={false}\n            showArrow={true}\n          />\n        </StyledWrapper>\n      )}\n      {provider === 'ollama' && (\n        <OllamaModelInputWrapper>\n          <div style={{width: '250px'}}>\n            <Checkbox\n              id=\"ollama-model-input\"\n              label=\"Input Model Name\"\n              onChange={onOllamaModelInputChecked}\n              checked={ollamaModelInputChecked}\n            />\n          </div>\n          <Input\n            type=\"text\"\n            onChange={onOllamaModelInputValueChange}\n            placeholder=\"Enter Model Name\"\n            value={ollamaModelInputValue}\n            disabled={!ollamaModelInputChecked}\n          />\n        </OllamaModelInputWrapper>\n      )}\n      {provider !== 'ollama' ? (\n        <>\n          <PanelLabelWrapper>\n            <SectionTitle>\n              <FormattedMessage id=\"aiAssistantManager.apiKey.placeholder\" />\n            </SectionTitle>\n          </PanelLabelWrapper>\n          <div className=\"api-key-input\">\n            <div className=\"api-key-input__icon\">\n              <ApiKey height=\"20px\" />\n            </div>\n            <Input\n              type=\"text\"\n              onChange={onApiKeyChange}\n              placeholder={intl.formatMessage({id: 'aiAssistantManager.apiKey.placeholder'})}\n              value={apiKey}\n            />\n          </div>\n        </>\n      ) : (\n        <>\n          <PanelLabelWrapper>\n            <SectionTitle>\n              <FormattedMessage id=\"aiAssistantManager.baseUrl.placeholder\" />\n            </SectionTitle>\n          </PanelLabelWrapper>\n          <div className=\"api-key-input\">\n            <div className=\"api-key-input__icon\">\n              <ApiKey height=\"20px\" />\n            </div>\n            <Input\n              type=\"text\"\n              onChange={onBaseUrlChange}\n              placeholder={intl.formatMessage({id: 'aiAssistantManager.baseUrl.placeholder'})}\n              value={baseUrl}\n            />\n          </div>\n        </>\n      )}\n      {connectionError && (\n        <StyleErrorMessage className=\"error-message\">{errorMessage}</StyleErrorMessage>\n      )}\n      <PanelLabelWrapper>\n        <SectionTitle>\n          <FormattedMessage id=\"aiAssistantManager.temperature.title\" />\n        </SectionTitle>\n      </PanelLabelWrapper>\n      <StyleSliderWrapper>\n        <RangeSlider\n          showInput={true}\n          isRanged={false}\n          value0={0}\n          value1={temperature}\n          onChange={onTemperatureChange}\n          range={[0, 2]}\n          step={0.1}\n        />\n      </StyleSliderWrapper>\n      <PanelLabelWrapper>\n        <SectionTitle>\n          <FormattedMessage id=\"aiAssistantManager.topP.title\" />\n        </SectionTitle>\n      </PanelLabelWrapper>\n      <StyleSliderWrapper>\n        <RangeSlider\n          showInput={true}\n          isRanged={false}\n          value0={0}\n          value1={topP}\n          onChange={onTopPChange}\n          range={[0, 1]}\n          step={0.1}\n        />\n      </StyleSliderWrapper>\n      <>\n        <PanelLabelWrapper>\n          <SectionTitle>Mapbox Token (optional for route/isochrone)</SectionTitle>\n        </PanelLabelWrapper>\n        <div className=\"api-key-input\">\n          <div className=\"api-key-input__icon\">\n            <ApiKey height=\"20px\" />\n          </div>\n          <Input\n            type=\"text\"\n            onChange={onMapboxTokenChange}\n            placeholder=\"Enter your Mapbox Token\"\n            value={mapboxToken}\n          />\n        </div>\n      </>\n      <StyledButton>\n        <Button onClick={onStartChat} width={'100%'}>\n          {isRunning && <LoadingSpinner size={12} />}\n          <FormattedMessage id=\"aiAssistantManager.startChat\" />\n        </Button>\n      </StyledButton>\n    </StyledAiAssistantConfig>\n  );\n}\n"],"mappings":"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