Skip to content

Commit 3cc6cc6

Browse files
committedSep 26, 2023
Initial commit
0 parents  commit 3cc6cc6

17 files changed

+46235
-0
lines changed
 

‎.gitignore

+273
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,273 @@
1+
# Streamlit
2+
.streamlit/*
3+
!.streamlit/config.toml
4+
!*/frontend/.env
5+
6+
# Visual Studio Code
7+
.vscode/*
8+
#!.vscode/settings.json
9+
!.vscode/tasks.json
10+
!.vscode/launch.json
11+
!.vscode/extensions.json
12+
*.code-workspace
13+
14+
# Local History for Visual Studio Code
15+
.history/
16+
17+
# Byte-compiled / optimized / DLL files
18+
__pycache__/
19+
*.py[cod]
20+
*$py.class
21+
22+
# C extensions
23+
*.so
24+
25+
# Distribution / packaging
26+
.Python
27+
build/
28+
develop-eggs/
29+
dist/
30+
downloads/
31+
eggs/
32+
.eggs/
33+
lib/
34+
lib64/
35+
parts/
36+
sdist/
37+
var/
38+
wheels/
39+
share/python-wheels/
40+
*.egg-info/
41+
.installed.cfg
42+
*.egg
43+
MANIFEST
44+
45+
# PyInstaller
46+
# Usually these files are written by a python script from a template
47+
# before PyInstaller builds the exe, so as to inject date/other infos into it.
48+
*.manifest
49+
*.spec
50+
51+
# Installer logs
52+
pip-log.txt
53+
pip-delete-this-directory.txt
54+
55+
# Unit test / coverage reports
56+
htmlcov/
57+
.tox/
58+
.nox/
59+
.coverage
60+
.coverage.*
61+
.cache
62+
nosetests.xml
63+
coverage.xml
64+
*.cover
65+
*.py,cover
66+
.hypothesis/
67+
.pytest_cache/
68+
cover/
69+
70+
# Translations
71+
*.mo
72+
*.pot
73+
74+
# Django stuff:
75+
*.log
76+
local_settings.py
77+
db.sqlite3
78+
db.sqlite3-journal
79+
80+
# Flask stuff:
81+
instance/
82+
.webassets-cache
83+
84+
# Scrapy stuff:
85+
.scrapy
86+
87+
# Sphinx documentation
88+
docs/_build/
89+
90+
# PyBuilder
91+
.pybuilder/
92+
target/
93+
94+
# Jupyter Notebook
95+
.ipynb_checkpoints
96+
97+
# IPython
98+
profile_default/
99+
ipython_config.py
100+
101+
# pyenv
102+
# For a library or package, you might want to ignore these files since the code is
103+
# intended to run in multiple environments; otherwise, check them in:
104+
.python-version
105+
106+
# pipenv
107+
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
108+
# However, in case of collaboration, if having platform-specific dependencies or dependencies
109+
# having no cross-platform support, pipenv may install dependencies that don't work, or not
110+
# install all needed dependencies.
111+
#Pipfile.lock
112+
113+
# PEP 582; used by e.g. github.com/David-OConnor/pyflow
114+
__pypackages__/
115+
116+
# Celery stuff
117+
celerybeat-schedule
118+
celerybeat.pid
119+
120+
# SageMath parsed files
121+
*.sage.py
122+
123+
# Environments
124+
.env
125+
.venv
126+
env/
127+
venv/
128+
ENV/
129+
env.bak/
130+
venv.bak/
131+
132+
# Spyder project settings
133+
.spyderproject
134+
.spyproject
135+
136+
# Rope project settings
137+
.ropeproject
138+
139+
# mkdocs documentation
140+
/site
141+
142+
# mypy
143+
.mypy_cache/
144+
.dmypy.json
145+
dmypy.json
146+
147+
# Pyre type checker
148+
.pyre/
149+
150+
# pytype static type analyzer
151+
.pytype/
152+
153+
# Cython debug symbols
154+
cython_debug/
155+
156+
# Logs
157+
logs
158+
*.log
159+
npm-debug.log*
160+
yarn-debug.log*
161+
yarn-error.log*
162+
lerna-debug.log*
163+
164+
# Diagnostic reports (https://nodejs.org/api/report.html)
165+
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
166+
167+
# Runtime data
168+
pids
169+
*.pid
170+
*.seed
171+
*.pid.lock
172+
173+
# Directory for instrumented libs generated by jscoverage/JSCover
174+
lib-cov
175+
176+
# Coverage directory used by tools like istanbul
177+
coverage
178+
*.lcov
179+
180+
# nyc test coverage
181+
.nyc_output
182+
183+
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
184+
.grunt
185+
186+
# Bower dependency directory (https://bower.io/)
187+
bower_components
188+
189+
# node-waf configuration
190+
.lock-wscript
191+
192+
# Compiled binary addons (https://nodejs.org/api/addons.html)
193+
build/Release
194+
195+
# Dependency directories
196+
node_modules/
197+
jspm_packages/
198+
199+
# Snowpack dependency directory (https://snowpack.dev/)
200+
web_modules/
201+
202+
# TypeScript cache
203+
*.tsbuildinfo
204+
205+
# Optional npm cache directory
206+
.npm
207+
208+
# Optional eslint cache
209+
.eslintcache
210+
211+
# Microbundle cache
212+
.rpt2_cache/
213+
.rts2_cache_cjs/
214+
.rts2_cache_es/
215+
.rts2_cache_umd/
216+
217+
# Optional REPL history
218+
.node_repl_history
219+
220+
# Output of 'npm pack'
221+
*.tgz
222+
223+
# Yarn Integrity file
224+
.yarn-integrity
225+
226+
# dotenv environment variables file
227+
.env
228+
.env.test
229+
230+
# parcel-bundler cache (https://parceljs.org/)
231+
.cache
232+
.parcel-cache
233+
234+
# Next.js build output
235+
.next
236+
out
237+
238+
# Nuxt.js build / generate output
239+
.nuxt
240+
dist
241+
242+
# Gatsby files
243+
.cache/
244+
# Comment in the public line in if your project uses Gatsby and not Next.js
245+
# https://nextjs.org/blog/next-9-1#public-directory-support
246+
# public
247+
248+
# vuepress build output
249+
.vuepress/dist
250+
251+
# Serverless directories
252+
.serverless/
253+
254+
# FuseBox cache
255+
.fusebox/
256+
257+
# DynamoDB Local files
258+
.dynamodb/
259+
260+
# TernJS port file
261+
.tern-port
262+
263+
# Stores VSCode versions used for testing VSCode extensions
264+
.vscode-test
265+
266+
# yarn v2
267+
.yarn/cache
268+
.yarn/unplugged
269+
.yarn/build-state.yml
270+
.yarn/install-state.gz
271+
.pnp.*
272+
273+
/.idea/

‎LICENSE

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
Copyright (c) 2018-2021 Streamlit Inc.
2+
3+
Permission is hereby granted, free of charge, to any person obtaining a copy
4+
of this software and associated documentation files (the "Software"), to deal
5+
in the Software without restriction, including without limitation the rights
6+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
7+
copies of the Software, and to permit persons to whom the Software is
8+
furnished to do so, subject to the following conditions:
9+
10+
The above copyright notice and this permission notice shall be included in all
11+
copies or substantial portions of the Software.
12+
13+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
19+
SOFTWARE.

‎MANIFEST.in

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
recursive-include streamlit_condition_tree/frontend/build *

‎README.md

+114
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,114 @@
1+
2+
Based on [react-awesome-query-builder](https://github.com/ukrbublik/react-awesome-query-builder)
3+
4+
Check out [live demo](https://app-condition-tree-demo-dkclrsnqxpcgzkjzlv3mqu.streamlit.app/) !
5+
6+
<img src="preview.jpg" width="500" alt="preview">
7+
8+
9+
## Install
10+
11+
`pip install streamlit-condition-tree`
12+
13+
14+
## Features
15+
- Highly configurable
16+
- Fields can be of type:
17+
- simple (string, number, bool, date/time/datetime, list)
18+
- structs (will be displayed in selectbox as tree)
19+
- Comparison operators can be:
20+
- binary (== != < > ..)
21+
- unary (is empty, is null)
22+
- 'between' (for numbers, dates, times)
23+
- complex operators like 'proximity'
24+
- RHS can be:
25+
- values
26+
- another fields (of same type)
27+
- functions (arguments also can be values/fields/funcs)
28+
- LHS can be field or function
29+
- Reordering (drag-n-drop) support for rules and groups of rules
30+
- Export to MongoDb, SQL, JsonLogic, SpEL or ElasticSearch
31+
32+
## Basic usage
33+
34+
```python
35+
import streamlit as st
36+
from streamlit_condition_tree import condition_tree
37+
38+
39+
config = {
40+
'fields': {
41+
'name': {
42+
'label': 'Name',
43+
'type': 'text',
44+
},
45+
'qty': {
46+
'label': "Age",
47+
'type': "number",
48+
'fieldSettings': {
49+
'min': 0
50+
},
51+
'preferWidgets': ['number']
52+
},
53+
'like_tomatoes': {
54+
'label': 'Likes tomatoes',
55+
'type': "boolean",
56+
'operators': ["equal"],
57+
}
58+
}
59+
}
60+
61+
return_val = condition_tree(
62+
config,
63+
return_type='sql'
64+
)
65+
66+
st.write(return_val)
67+
```
68+
69+
## API
70+
71+
### Parameters
72+
73+
```python
74+
def condition_tree(
75+
config: Dict
76+
return_type: str
77+
tree: Dict
78+
min_height: int
79+
placeholder: str
80+
key: str
81+
)
82+
```
83+
84+
- **config**: Python dictionary that resembles the JSON counterpart of
85+
the React component [config](https://github.com/ukrbublik/react-awesome-query-builder/blob/master/CONFIG.adoc).
86+
*Note*: Javascript functions (ex: validators) are not yet supported.
87+
88+
89+
- **return_type**: Format of the returned value :
90+
- queryString
91+
- mongodb
92+
- sql
93+
- spel
94+
- elasticSearch
95+
- jsonLogic
96+
97+
98+
- **tree**: Input condition tree (see section below)
99+
100+
101+
- **min_height**: Minimum height of the component frame
102+
103+
104+
- **placeholder**: Text displayed when the condition tree is empty
105+
106+
107+
- **key**: Fixed identity if you want to change its arguments over time and not have it be re-created.
108+
Can also be used to access the generated condition tree (see section below).
109+
110+
111+
### Export & import a condition tree
112+
113+
When a key is defined for the component, the condition tree generated is accessible through `st.session_state[key]` as a dictionary.
114+
It can be loaded as an input tree through the `tree` parameter.

‎preview.jpg

67.5 KB
Loading

‎setup.py

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
from pathlib import Path
2+
3+
import setuptools
4+
5+
this_directory = Path(__file__).parent
6+
long_description = (this_directory / "README.md").read_text()
7+
8+
setuptools.setup(
9+
name="streamlit-condition-tree",
10+
version="0.1.1",
11+
author="Cédric Villette",
12+
author_email="cedric_villette@hotmail.fr",
13+
description="Condition Tree Builder for Streamlit",
14+
long_description=long_description,
15+
long_description_content_type="text/markdown",
16+
url="https://github.com/cedricvlt/streamlit-condition-tree",
17+
packages=setuptools.find_packages(),
18+
include_package_data=True,
19+
classifiers=[],
20+
python_requires=">=3.7",
21+
install_requires=[
22+
# By definition, a Custom Component depends on Streamlit.
23+
# If your component has other Python dependencies, list
24+
# them here.
25+
"streamlit >= 0.63",
26+
],
27+
extras_require={
28+
"devel": [
29+
"wheel",
30+
"pytest==7.4.0",
31+
"playwright==1.36.0",
32+
"requests==2.31.0",
33+
"pytest-playwright-snapshot==1.0",
34+
"pytest-rerunfailures==12.0",
35+
]
36+
}
37+
)

‎streamlit_condition_tree/__init__.py

+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
import os
2+
import streamlit.components.v1 as components
3+
import streamlit as st
4+
5+
_RELEASE = False
6+
7+
if not _RELEASE:
8+
_component_func = components.declare_component(
9+
"streamlit_condition_tree",
10+
url="http://localhost:3001",
11+
)
12+
else:
13+
parent_dir = os.path.dirname(os.path.abspath(__file__))
14+
build_dir = os.path.join(parent_dir, "frontend/build")
15+
_component_func = components.declare_component("streamlit_condition_tree", path=build_dir)
16+
17+
18+
def condition_tree(config: dict,
19+
return_type: str = 'queryString',
20+
tree: dict = None,
21+
min_height: int = 400,
22+
placeholder: str = '',
23+
key: str = None):
24+
"""Create a new instance of condition_tree.
25+
26+
Parameters
27+
----------
28+
config: dict
29+
Configuration defining the value types, supported operators and how
30+
they are rendered, imported and exported.
31+
return_type: str or None
32+
Format in which output should be returned to streamlit.
33+
Possible values : queryString | mongodb | sql | spel |
34+
elasticSearch | jsonLogic.
35+
tree: dict or None
36+
Input condition tree
37+
min_height: int
38+
Minimum height of the component frame
39+
placeholder: str
40+
Text displayed when the condition tree is empty
41+
key: str or None
42+
An optional key that uniquely identifies this component. If this is
43+
None, and the component's arguments are changed, the component will
44+
be re-mounted in the Streamlit frontend and lose its current state.
45+
Can also be used to access the condition tree through st.session_state.
46+
47+
Returns
48+
-------
49+
dict or object
50+
The output conditions with the selected format
51+
52+
"""
53+
54+
output_tree, component_value = _component_func(
55+
config=config,
56+
return_type=return_type,
57+
tree=tree,
58+
key='_' + key if key else None,
59+
min_height=min_height,
60+
placeholder=placeholder,
61+
default=['', '']
62+
)
63+
st.session_state[key] = output_tree
64+
65+
return component_value
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
{
2+
"endOfLine": "lf",
3+
"semi": false,
4+
"trailingComma": "es5"
5+
}

‎streamlit_condition_tree/frontend/package-lock.json

+35,908
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
{
2+
"name": "streamlit-condition-tree",
3+
"version": "0.1.1",
4+
"private": true,
5+
"dependencies": {
6+
"@fontsource/source-sans-pro": "^5.0.8",
7+
"@react-awesome-query-builder/antd": "^6.4.1",
8+
"@react-awesome-query-builder/ui": "^6.4.1",
9+
"baseui": "^13.0.0",
10+
"react": "^16.13.1",
11+
"react-dom": "^16.13.1",
12+
"streamlit-component-lib": "^2.0.0",
13+
"styletron-engine-atomic": "^1.5.0",
14+
"styletron-react": "^6.1.0"
15+
},
16+
"scripts": {
17+
"start": "react-scripts start",
18+
"build": "react-scripts build",
19+
"test": "react-scripts test",
20+
"eject": "react-scripts eject"
21+
},
22+
"eslintConfig": {
23+
"extends": "react-app"
24+
},
25+
"browserslist": {
26+
"production": [
27+
">0.2%",
28+
"not dead",
29+
"not op_mini all"
30+
],
31+
"development": [
32+
"last 1 chrome version",
33+
"last 1 firefox version",
34+
"last 1 safari version"
35+
]
36+
},
37+
"homepage": ".",
38+
"devDependencies": {
39+
"@types/node": "^12.0.0",
40+
"@types/react": "^16.9.0",
41+
"@types/react-dom": "^16.9.0",
42+
"react-scripts": "^5.0.1",
43+
"typescript": "^4.2.0"
44+
}
45+
}

‎streamlit_condition_tree/frontend/public/bootstrap.min.css

+9,501
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<title>Streamlit Component</title>
5+
<meta charset="UTF-8" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1" />
7+
<meta name="theme-color" content="#000000" />
8+
<meta name="description" content="Streamlit Component" />
9+
<link rel="stylesheet" href="bootstrap.min.css" />
10+
</head>
11+
<body>
12+
<noscript>You need to enable JavaScript to run this app.</noscript>
13+
<div id="root"></div>
14+
<!--
15+
This HTML file is a template.
16+
If you open it directly in the browser, you will see an empty page.
17+
18+
You can add webfonts, meta tags, or analytics to this file.
19+
The build step will place the bundled scripts into the <body> tag.
20+
21+
To begin the development, run `npm start` or `yarn start`.
22+
To create a production bundle, use `npm run build` or `yarn build`.
23+
-->
24+
</body>
25+
</html>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,146 @@
1+
import {ComponentProps, Streamlit, StreamlitComponentBase, withStreamlitConnection} from "streamlit-component-lib"
2+
import React, {ReactNode} from "react"
3+
4+
import type {BuilderProps, Config, ImmutableTree, JsonGroup, JsonTree} from '@react-awesome-query-builder/antd';
5+
import {AntdConfig, Builder, Query, Utils as QbUtils} from '@react-awesome-query-builder/antd';
6+
import {ConfigProvider, theme as antdTheme} from 'antd';
7+
import '@react-awesome-query-builder/antd/css/styles.css';
8+
import './style.css'
9+
import "@fontsource/source-sans-pro";
10+
11+
12+
interface State {
13+
tree: ImmutableTree,
14+
config: Config
15+
}
16+
17+
const defaultTree : JsonGroup = {
18+
type: "group",
19+
id: QbUtils.uuid()
20+
};
21+
22+
23+
const exportFunctions: Record<string, Function> = {
24+
queryString: QbUtils.queryString,
25+
mongodb: QbUtils.mongodbFormat,
26+
sql: QbUtils.sqlFormat,
27+
spel: QbUtils.spelFormat,
28+
elasticSearch: QbUtils.elasticSearchFormat,
29+
jsonLogic: QbUtils.jsonLogicFormat
30+
}
31+
32+
const formatTree = (tree: any) => {
33+
// Recursively add uuid and rename 'children' key
34+
tree.id = QbUtils.uuid()
35+
if (tree.children) {
36+
tree.children1 = tree.children;
37+
delete tree.children;
38+
tree.children1.forEach(formatTree);
39+
}
40+
};
41+
42+
const unformatTree = (tree: any) => {
43+
// Recursively remove uuid and rename 'children1' key
44+
delete tree.id;
45+
if (tree.children1) {
46+
tree.children = tree.children1;
47+
delete tree.children1;
48+
tree.children.forEach(unformatTree);
49+
}
50+
};
51+
52+
class ConditionTree extends StreamlitComponentBase<State> {
53+
54+
public constructor(props: ComponentProps) {
55+
super(props);
56+
57+
const config: Config = {
58+
...(AntdConfig),
59+
...props.args['config']
60+
};
61+
62+
// Load input tree
63+
let tree : ImmutableTree = QbUtils.loadTree(defaultTree)
64+
if (props.args['tree'] != null) {
65+
try {
66+
let input_tree = props.args['tree']
67+
formatTree(input_tree)
68+
tree = QbUtils.checkTree(QbUtils.loadTree(input_tree), config)
69+
} catch (error) {
70+
console.log(error);
71+
}
72+
}
73+
74+
this.state = { config, tree }
75+
this.setStreamlitValue(tree)
76+
}
77+
78+
public render = (): ReactNode => {
79+
const {theme} = this.props
80+
const tree = QbUtils.getTree(this.state.tree)
81+
const empty = !tree.children1 || !tree.children1.length
82+
83+
return(
84+
<div>
85+
<ConfigProvider
86+
theme={theme ? {
87+
token: {
88+
colorPrimary: theme['primaryColor'],
89+
colorText: theme['textColor'],
90+
fontFamily: theme['font'],
91+
fontSize: 16,
92+
controlHeight: 38,
93+
},
94+
algorithm: theme['base'] === 'dark' ?
95+
antdTheme.darkAlgorithm : antdTheme.defaultAlgorithm
96+
} : {}}
97+
>
98+
<Query
99+
{...this.state.config}
100+
value={this.state.tree}
101+
onChange={this.onChange}
102+
renderBuilder={this.renderBuilder}
103+
/>
104+
<p>{empty && this.props.args['placeholder']}</p>
105+
</ConfigProvider>
106+
</div>
107+
)
108+
}
109+
110+
private onChange = (immutableTree: ImmutableTree) => {
111+
this.setState({ tree: immutableTree })
112+
this.setStreamlitValue(immutableTree)
113+
}
114+
115+
private setStreamlitValue = (tree: ImmutableTree) => {
116+
const exportFunc = exportFunctions[this.props.args['return_type']]
117+
const exportValue = exportFunc ? exportFunc(tree, this.state.config) : ''
118+
119+
let output_tree : JsonTree = QbUtils.getTree(tree)
120+
unformatTree(output_tree)
121+
Streamlit.setComponentValue([output_tree, exportValue])
122+
}
123+
124+
private renderBuilder = (props: BuilderProps) => (
125+
<div className="query-builder-container">
126+
<div className="query-builder qb-lite">
127+
<Builder {...props} />
128+
</div>
129+
</div>
130+
)
131+
132+
componentDidUpdate = () => {
133+
// Class to apply custom css on rule_groups with a single child
134+
document
135+
.querySelectorAll('.rule_group>.group--children:has(> :nth-child(1):last-child)')
136+
.forEach((x)=> x.classList.add('single-child'))
137+
138+
// Set frame height
139+
const height = Math.max(
140+
document.body.scrollHeight+20,
141+
this.props.args['min_height']
142+
);
143+
Streamlit.setFrameHeight(height);
144+
}
145+
}
146+
export default withStreamlitConnection(ConditionTree);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import React from "react"
2+
import ReactDOM from "react-dom"
3+
import ConditionTree from "./ConditionTree"
4+
5+
ReactDOM.render(
6+
<React.StrictMode>
7+
<ConditionTree />
8+
</React.StrictMode>,
9+
document.getElementById("root")
10+
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
/// <reference types="react-scripts" />
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
.query-builder>.group-container,
2+
.query-builder>.group-container>.group>div,
3+
.query-builder>.group-container>.group>.group--children>.group-or-rule-container {
4+
padding-right: 0;
5+
}
6+
7+
.query-builder {
8+
margin: 0 0 0 -6px;
9+
}
10+
11+
.query-builder-container {
12+
margin-top: 25px;
13+
margin-bottom: 25px;
14+
}
15+
16+
.query-builder>.group-container>.group {
17+
border-color: var(--background-color);
18+
}
19+
20+
.rule, .rule_group {
21+
background-color: var(--secondary-background-color);
22+
}
23+
24+
.group {
25+
background-color: var(--background-color);
26+
border: 1px solid rgba(49, 51, 63, 0.2);
27+
border-radius: 8px;
28+
}
29+
30+
.rule_group:has(.single-child) {
31+
background-color: var(--secondary-background-color);
32+
border-color: var(--background-color);
33+
}
34+
35+
.group--children.single-child {
36+
margin-top: 0;
37+
margin-bottom: 0;
38+
}
39+
40+
.rule--value .rule--widget .widget--valuesrc {
41+
margin-top: 8px;
42+
}
43+
44+
.ant-btn:not(.type) {
45+
outline: 0;
46+
height: 33px;
47+
}
48+
49+
.rule--header>.ant-btn-group {
50+
margin-top: -2.5px;
51+
margin-bottom: -2.5px;
52+
}
53+
54+
.group--conjunctions {
55+
margin-left: -1px;
56+
}
57+
58+
.ant-btn-group>button:nth-child(3) {
59+
margin-left: -2px !important;
60+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
{
2+
"compilerOptions": {
3+
"target": "es5",
4+
"lib": [
5+
"dom",
6+
"dom.iterable",
7+
"esnext"
8+
],
9+
"allowJs": true,
10+
"skipLibCheck": true,
11+
"esModuleInterop": true,
12+
"allowSyntheticDefaultImports": true,
13+
"strict": true,
14+
"forceConsistentCasingInFileNames": true,
15+
"module": "esnext",
16+
"moduleResolution": "node",
17+
"resolveJsonModule": true,
18+
"isolatedModules": true,
19+
"noEmit": true,
20+
"jsx": "preserve"
21+
},
22+
"include": [
23+
"src"
24+
]
25+
}

0 commit comments

Comments
 (0)
Please sign in to comment.