title:
어드민 사용자가 보는 페이지의 상단 왼쪽에 보여지는 제목입니다.
title: 어드민 제목
layout: {}
layout.style
어드민 화면 최상단 영역의 스타일을 바꿀 수 있습니다. 우리 팀에게 맞는 배경색으로 바꿔보세요.
📘 title 키와 동일한 위치에서 사용합니다.
layout:
style:
backgroundColor: "#3f63bf !important"
borderBottom: solid 1px rgba(0,0,0,0.05)
layout.login
Beta
셀렉트로 파트너센터나 고객 포털 등을 제작하실 때 로그인 화면을 일부 커스텀, 화이트라벨링을 할 수 있습니다.
INFO
베타(Beta) 기간 후에 이용 요금제에 따라 관련 기능 지원 정책이 달라집니다. 궁금하신 사항은 이메일 등으로 문의해주세요.
layout:
login:
methods:
google:
enabled: true
text: 회사계정 로그인
email:
enabled: true
text: 이메일 비밀번호 로그인
header: Selectfromuser.com
# header: |
# <img src="https://align.selectfromuser.com/lib_DmHvtrBlUMtUhGvc/sh11bex09q31frsy.png" style="height: 50px; margin: 0 auto;" />
favicon: https://files.umso.co/lib_DmHvtrBlUMtUhGvc/emvdae7wl2lyppo0.png
title: 파트너센터 로그인
subtitle: 입점한 셀러분들만 이용할 수 있어요.
width: 600
buttons:
- label: 도움말
href: https://docs.selectfromuser.com
target: _blank
- label: 입점하기
href: https://docs.selectfromuser.com
target: _blank
footer: |
<b>© 2023 selectfromuser Inc.</b>
주식회사 셀렉트프롬유저 | 서울시 송파구 법원로 128, B동 1408호 (문정동, 문정역SKV1)
사업자등록번호 386-88-02410 | 통신판매신고 제2023-서울송파-0491호
layout.head
셀렉트 어드민 <head>
태그에 필요한 스크립트(script)를 임베드할 수 있습니다. 구글 애널리틱스(Google Analytics)나 앰플리튜드(Amplitude) 등 분석 서비스의 스크립트를 활용해보세요.
INFO
베타(Beta) 기간 후에 이용 요금제에 따라 관련 기능 지원 정책이 달라집니다. 궁금하신 사항은 이메일 등으로 문의해주세요.
구글 애널리틱스(Google Analytics, GA) 예제
GA 태그
<!-- Google tag (gtag.js) -->
<script async src="https://www.googletagmanager.com/gtag/js?id=G-XXXXXXXXXX"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'G-XXXXXXXXXX');
</script>
셀렉트 임베드 방법
layout:
head:
- tag: script
async: true
src: https://www.googletagmanager.com/gtag/js?id=G-XXXXXXXXXX
- tag: script
innerHTML: |
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'G-XXXXXXXXXX');
layout.hideLeftMenu
상단 메뉴만 유지하고 싶을때 해당 옵션을 적용해주세요. menus.type: nav
와 함께 사용시 유용합니다.
layout:
hideLeftMenu: true
menus: []
어드민 사용자 페이지의 메뉴를 설정할 수 있습니다.
menus.path
메뉴에 접근하는 경로입니다. 링크를 공유했을 때 사용자가 이해하기 쉬운 네이밍을 한다면 좋습니다.
menus.name
메뉴에 표시되는 이름입니다.
menus:
- path: user
name: 사용자
menus.group
group이 없는 경우 모든 메뉴는 같은 그룹으로 인식됩니다. 메뉴들을 따로 묶고 싶으신 경우 group을 이용해주세요.
menus:
- path: user
name: 사용자
group: u
- path: user/active
name: 활성 유저
group: u
menus.placement
메뉴는 기본적으로 왼쪽 영역과 오른쪽 탭에 둘다 나타납니다. 왼쪽 또는 탭에만 메뉴를 보여주고 싶은 경우 placement를 이용해보세요.
placement: menu-only
placement: tab-only
placement: expand-only
placement: page-only # 탭메뉴 표시 안함
menus:
- path: user
name: 사용자
group: u
- path: user/active
name: 활성 유저
group: u
placement: tab-only
menus.redirect
메뉴 클릭 시 redirect 시킬 path 또는 링크를 지정합니다.
menus:
- path: user
name: 사용자
group: u
redirect: user/all
placement: menu-only
- path: user/all
name: 전체 사용자
group: u
placement: tab-only
- path: user/active
name: 활성 유저
group: u
placement: tab-only
menus.roles
메뉴와 페이지 접근에 대한 권한을 설정합니다. 그룹별 또는 개별 이메일로 설정 가능합니다.
자세한 사항은 '메뉴 권한 설정' 가이드에서 확인하실 수 있어요.
menus:
- path: /approvals
name: 심사
roles:
list:
- 팀장
- 임원
view:
- 팀장
- 임원
- email::test@email.com
list (메뉴에 보이기) | view (페이지 접근) | 용도/결과 |
---|---|---|
O | O | 해당 메뉴와 페이지에 접근 권한 있음 (일반적) |
X | X | 해당 메뉴와 페이지에 권한 없음 (일반적) |
O | X | 권한은 없지만 메뉴의 존재 자체는 알 수 있음 |
X | O | URL만으로 비공개 페이지 공유 또는 테스트할 때 |
menus.type: divider
메뉴 사이에 시각적으로 영역을 구분짓고 싶을 때 사용할 수 있습니다.
menus:
- path: user
name: 사용자
group: u
- type: divider
- path: payment
name: 결제
group: p
menus.type: search
메뉴들 사이에 퀵서치Quick Search 용도로 검색창을 넣을 수 있습니다.
redirect
로 path를 지정하여 이동합니다.- 검색 시 입력값은 trim(공백제거) 처리합니다.
label
로 검색 버튼 이름을 바꿔보세요.- 검색창에 입력한 값은
value
로 전달됩니다.
menus:
- type: search
label: 검색
placeholder: 고객이름
redirect: search-user?name={{value}}
- type: search
label: 찾기
placeholder: 010-0000-0000
redirect: search-user?phone={{value}}
- path: search-user
placement: top
검색창을 최상단 메뉴바 영역에도 보이게 하고, 간단한 스타일도 적용할 수 있습니다.
menus:
- type: search
label: 검색
placeholder: 010-0000-0000
redirect: search-user?phone={{value}}
placement: top
style:
width: 500px
datalistFromQuery
검색창 안에 보여줄 특정 데이터들을 datalistFromQuery로 가져올 수 있습니다.
- 가져온 데이터 필드들을
format
으로 치환해서 보여줍니다. html: true
를 사용하면 format 안에 html 양식을 사용할 수 있습니다.- redirect가 String이 아니라 Object인 경우 결과값의 category와 일치하는쪽으로 가져와서 redirect 합니다.
menus:
- type: search
placement: top
placeholder: 상품코드/재고코드/공급코드
style:
width: 300px
datalistFromQuery:
resource: mysql.qa
type: query
sqlType: select
sql: >
(
SELECT id, name, 'properties' AS category
FROM properties
WHERE id LIKE CONCAT('%', :value, '%')
OR (LENGTH(:value) > 2 AND name LIKE CONCAT('%', :value, '%'))
ORDER BY id DESC
LIMIT 10
)
UNION ALL
(
SELECT id, name AS name2, 'properties2' AS category
FROM properties2
WHERE id LIKE CONCAT('%', :value, '%')
OR (LENGTH(:value) > 2 AND name LIKE CONCAT('%', :value, '%'))
ORDER BY id DESC
LIMIT 10
)
format: >
ID: {{ id}} Name: {{name}}{{ name2 }} ({{category}})
# html: true
# format: >
# <span style="color: royalblue">ID: {{ id}}</span>
# {{name}}{{ name2 }}
# ({{category}})
# redirect: search-user-id?p={{category}}&user_id={{id}}
redirect:
properties: search-user-id?p={{category}}&user_id={{id}}
properties2: >
search-user-id?p={{category}}&cc_id={{id}}#{{hero_image_url}}
menus.type: nav
placement: top
과 함께 사용하여 메뉴를 최상단 내비게이션으로 이용할 수 있고, 하위 menus를 추가할 수도 있습니다.
menus:
- name: 시약관리
type: nav
placement: top
menus:
- path: pages/search
name: 시약검색
placement: top
- path: pages/codes
name: 코드관리
placement: top
- path: pages/qc
name: 품질관리
placement: top
menus.icon
메뉴 이름(name) 앞에 U, NEW, TIP 같은 내용이나 약어를 텍스트로 추가하거나 mdi 아이콘을 추가할 수 있어요.
- 지원하는 아이콘: https://pictogrammers.github.io/@mdi/font/7.2.96/
- 필요한 아이콘을 찾기 쉬운 사이트: https://pictogrammers.com/library/mdi/
menus:
- path: user
icon: U
name: 사용자
- path: payment
icon: mdi-cash
name: 결제관리
menus.iconEnd
메뉴 이름(name) 뒤쪽에 텍스트를 추가하거나 mdi 아이콘을 추가할 수 있어요. icon과 함께 사용할 수도 있습니다.
menus:
- path: payment
iconEnd: mdi-cash
name: 결제관리
menus.menus
메뉴 아래에 메뉴를 추가하고 접고 펼 수 있습니다.
menus:
- path: user
name: 사용자 관리
placement: expand-only
menus:
- path: active-user
name: 활성 사용자
menus.badgeOptions
메뉴에 쿼리 결과의 카운트 숫자를 배지 형태로 표기할 수 있어요.
지원 색상은 아래와 같습니다. Customizing Colors - Tailwind CSS
- slate, gray, zinc, neutral, stone, red, orange, amber, yellow, lime, green, emerald, teal, cyan, sky, blue, indigo, violet, purple, fuchsia, pink, rose
menus:
- path: orders/pending
name: 상품 준비중 관리
badgeOptions:
enabled: true
type: query
resource: mysql.qa
sql: >
SELECT COUNT(id) FROM ProductOrder
postfix: 건
color: green
pages: []
페이지들을 구성하기 위한 최상위 키
pages.path
개별 페이지와 메뉴를 path 정보로 연결합니다.
menus:
- path: user
- path: payment
pages:
- path: user
- path: payment
pages.blocks
페이지 안을 블록으로 구성합니다.
menus:
- path: user
pages:
- path: user
blocks:
pages.autofocus
페이지 단위로 입력 필드의 autofocus 기능을 끌 수 있습니다.
pages:
- path: user
autofocus: false
pages.layout
페이지 단위로 layout style을 설정하고, 블록을 어떤곳에 넣을지 지정합니다.
layout name: page1
에 blocks layout: page1
블록이 들어갑니다.
지정된 블록(type: top | left | center | right | bottom
)보다 레이아웃을 더 자유롭게 설정할 수 있습니다.
pages:
- path: property-detail
layout:
style:
# max-width: 1400px
margin: 0px auto
display: flex
flex-wrap: wrap
div:
- name: page1
style:
width: 100%
- name: page2-1
style:
width: 50%
padding-right: 20px
- name: page2-2
style:
width: 50%
- name: page3
style:
width: 100%
blocks:
- layout: page1
type: markdown
- layout: page2-1
type: markdown
- layout: page2-2
type: markdown
- layout: page3
type: markdown
pages.layout: dashboard
특정 페이지를 layout: dashboard
로 설정하면 대시보드 레이아웃이 적용됩니다.
pages:
- path: sample/layout/dashboard
layout: dashboard
blocks:
- type: markdown
content: |
layout dashboard 1
- type: markdown
content: |
layout dashboard 2
- type: markdown
content: |
layout dashboard 3
pages.roles
- menus와 마찬가지로 roles를 적용할 수 있습니다.
- menus.roles와 pages.roles가 이중 레이어로 적용됩니다. (어느 한쪽으로 우선적용되지 않습니다.)
pages:
- path: pages/Ze1f_b
roles:
list:
- email::test@selectfromuser.com
view:
- email::test@selectfromuser.com
메뉴를 노출하지 않으면서 메뉴단위로 roles를 지정하고 싶다면 placement: page-only
와 menus.roles
를 사용할 수 있습니다.
menus:
- path: pages/Ze1f_b
placement: page-only
roles:
list:
- email::test@selectfromuser.com
view:
- email::test@selectfromuser.com
pages:
- path: pages/Ze1f_b
blocks: []
페이지 안은 블록들로 이루어져 있습니다. 1개 또는 여러개를 추가할 수 있으며, 블록의 종류는 query, http, markdown 이 있습니다.
blocks:
- type: query
- type: http
- type: markdown
blocks.type
blocks:
- type: query
name: 샘플 쿼리
resource: sample.db
sqlType: select # update, insert
- type: http
name: 샘플 http
axios: GET
- type: markdown
content: type anything
type: query
SQL 쿼리의 select, update, insert를 모두 사용할 수 있습니다.
sqlType: select
sqlType: update
sqlType: insert
- type: query
resource: sample.db
sqlType: select
sql: SELECT * FROM user LIMIT 10
- type: query
resource: sample.db
sqlType: update
sql: UPDATE user SET status = :status WHERE id = :id AND email = :email
params:
- key: status
- key: id
- key: email
sqlConfirm
실행될 쿼리를 확인하고 진행합니다. (update, delete 경우에 활용)
- type: query
resource: sample.db
sqlType: select
sql: UPDATE user SET status = :status WHERE id = :id AND email = :email
sqlConfirm: true
params:
- key: status
- key: id
- key: email
confirm: false
쿼리 실행시 나타나는 확인(confirm)창을 끄고 싶은 경우 이용해주세요.
- type: query
resource: mysql.qa
sqlType: insert
sql: INSERT INTO properties SET created_at = NOW(), name = :name
params:
- key: name
confirm: false
toast, toastOptions
블록 실행 시 알림창(toast)을 띄울 수 있습니다.
toast
: 알림창 안 내용을 입력합니다.toastOptions
type
: 알림창(toast)의 타입을 설정합니다.position
: 알림창 위치를 설정합니다.duration
: 알림창이 떠있는 시간을 설정합니다. (단위: ms)
- type: query
resource: sample.db
sqlType: select
sql: UPDATE user SET status = :status WHERE id = :id AND email = :email
sqlConfirm: true
params:
- key: status
- key: id
- key: email
toast: 상태를 변경했습니다.
toastOptions:
type: success
# success, error, warning, info
position: top-right
# top-right, top-center, top-left, bottom-right, bottom-center, bottom-left
duration: 3000
쿼리 실행후 toast를 통해 응답 정보를 표기할 수도 있습니다. 데이터베이스 결과값에 따라 맞춰 활용 가능합니다.
- type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO stock_wine
SET code = :code
params:
- key: code
confirm: false
toast: |
완료
<br />
<p>affectedRows: {{affectedRows}}</p>
<p>insertId: <a href="#{{insertId}}">{{insertId}}</a></p>
MySQL 응답 예제:
{
"fieldCount": 0,
"affectedRows": 1,
"insertId": 1949,
"info": "",
"serverStatus": 2,
"warningStatus": 0
}
type: http
REST API와 GraphQL API 모두 사용 가능합니다. axios 용법을 그대로 따릅니다. axios에 대한 자세한 설명은 여기서 확인해보세요.
method: GET
- type: http
axios:
method: GET
url: https://testapi.com/v1/33iac2d
rowsPath: rows
method: POST
- type: http
axios:
method: POST
url: https://testapi.com/v1/33iac2d
data:
'user_id': user-{{id}}
'v': "{{API_VERSION}}"
params:
- key: id
- key: API_VERSION
valueFromEnv: true
graphQL
- type: http
name: http 샘플
axios:
method: POST
url: https://testapi.com/v1/33iac2d
data:
query: >
{
getUser(id: "test") {
id
name
email
}
}
type: markdown
마크다운 용법에 맞춰서 내용을 작성할 수 있습니다. 페이지에 대한 설명이나 유의사항 등을 작성해보세요.
- type: markdown
content: >
write something
type: top | left | center | right | bottom
페이지 안에 영역을 구분하는 타입입니다. 해당 타입을 입력하고 아래에 쿼리, API, 마크다운 블록을 추가할 수 있습니다. 간단한 레이아웃(layout)을 빠르게 구성할 수 있어요.
blocks:
- type: top # left, center, right, bottom
title: title
subtitle: subtitle
blocks:
- type: markdown
- type: query
예제
pages:
- path: company-layouts
blocks:
-
type: top
title: title
blocks:
- type: markdown
content: >
> TOP
-
type: left
title: title
subtitle: subtitle
blocks:
- type: markdown
content: >
> LEFT
-
type: center
style:
width: 50%
height: 80vh
maxHeight: calc(100vh - 300px)
overflow: scroll
blocks:
- type: markdown
content: >
> CENTER
- type: query
title: 내역은 최근 30일
subtitle: 내역은 최근 30일
description: 영수증 목록
resource: mysql.qa
sql: SELECT * FROM test_10k LIMIT 300
sqlType: select
-
type: right
blocks:
- type: markdown
content: >
> RIGHT
-
type: bottom
blocks:
- type: markdown
content: >
> BOTTOM
type: header
사이트 이동 경로(breadcrumb)을 페이지 최상단에 표시할 수 있습니다. items 아래 순서대로 표기됩니다.
아이콘icon
은 아래 방식들로 지원합니다.
- 지원하는 아이콘: https://pictogrammers.github.io/@mdi/font/7.2.96/
- 필요한 아이콘을 찾기 쉬운 사이트: https://pictogrammers.com/library/mdi/
blocks:
- type: header # 사이트 이동 경로(breadcrumb) 표시 영역
items:
- path: company
label: 업체목록
icon: home
- label: 업체상세
type: tab
blocks 아래에 탭을 자유롭게 추가할 수 있어요. 탭 스타일도 지정할 수 있습니다.
- tabOptions.type: button, button full, full, plain
- 탭 타입(스타일)을 지정합니다.
- tabOptions.keep: true
- 선택했던 탭을 유지합니다.
blocks:
- type: tab
tabOptions:
type: button # 버튼형 탭
# type: button full # 전체폭 버튼형 탭
# type: full # 전체폭 탭
# type: plain
# keep: true
autoload: 1
tabs:
- name: 주문
blocks:
- name: 결제
blocks:
type: toggle
토글(toggle) 타입 블록을 통해 내용을 접었다 펼칠 수 있습니다.
class
: 토글의 기본 스타일을 지정 (텍스트 크기, 여백, 그림자, 색상 등)toggledClass
: 토글이 활성화되었을 때 적용될 스타일 (글꼴 두께, 배경색 등)- Tailwind CSS 클래스를 활용하여 스타일링
icon
: 토글 옆에 표시될 아이콘toggled
: 초기 토글 상태
blocks:
- type: toggle
name: toggle sample
icon: tree
class: text-lg p-2 shadow rounded text-green-700
toggledClass: font-medium text-green-700 bg-green-600/10
# toggled: true
blocks:
- type: query
type: iframe
어드민 페이지 안에 글이나 영상을 iframe 방식으로 임베딩embedding 할 수 있습니다.
pages:
- path: iframe-sample
blocks:
- type: iframe
src: https://www.selectfromuser.com/
style:
width: 50%
minWidth: 550px
height: 80vh
type: comment
사용자, 담당자 로그가 필요한 화면에 댓글, 노트, 메모를 남겨보세요.
path
: 코멘트의 저장위치를 설정합니다.- 계정정보의 이름, 이메일 등이 기록됩니다.
- 코멘트 추가, 수정, 삭제 지원
- "나에게만 표시" 기능 지원
- type: query
resource: mysql.qa
sql: SELECT id, name, created_at FROM customers LIMIT 5
columns:
id:
hidden: false
viewModal:
mode: side
useColumn: name
blocks:
- type: comment
path: customers/{{id}}
params:
- key: id
valueFromRow: true
type: userProperty
일반 사용자(view 권한)가 사용자 프로퍼티 값을 입력할 수 있는 블록(blocks.type)을 추가할 수 있습니다.
해당 기능을 이용하면 관리자(admin) 권한이 없어도 규정된 환경에서 사용자 프로퍼티(user property)를 입력할 수 있습니다.
설정 > 계정 페이지에 접속하지 않고, 일반 어드민 화면에서 입력이 필요한 경우 이용해보세요.
blocks:
- type: userProperty
예제 YAML 및 유의사항
- type: userProperty와 동위에 roles.edit을 입력해야합니다. (roles가 없는경우 에러를 띄워줍니다.)
- blocks.roles는 설정 > 권한 그룹에서 추가할 수 있습니다.
- 사용자는 properties 키에 설정된 필드와 값들만 입력할 수 있습니다.
menus:
- path: pages/LiyQhD
name: 사용자 속성 설정
pages:
- path: pages/LiyQhD
containerStyle: |
width: 800px
# class: container
title: 상점 권한 관리
blocks:
- type: markdown
content: |
- 파트너 권한
- type: userProperty
roles:
edit:
- 계정관리
properties:
shopCode:
type: string
placeholder: 상점코드
dropdown:
# - A
# - B
A: 타입 A
B: 타입 B
shopId:
type: string
label: 상점 ID
placeholder: 상점ID 8자리
description: 상점ID를 입력해주세요.
A:
type: string
B:
type: string
# required: true
blocks.reloadAfterSubmit
데이터 추가, 수정 후에 관련 페이지를 새로고침 하고 싶을 때 이용해요.
insert, update 쿼리 블록이나 post api 블록으로 반영된 데이터를 바로 조회하고 싶을 때 적합해요.
- type: query
resource: mysql
sqlType: insert
sql: >
INSERT INTO products
SET created_at = NOW()
, name = :name
params:
- key: name
label: 상품명
reloadAfterSubmit: true
blocks.resetAfterSubmit
INSERT 쿼리 블록이나 POST API 블록 등을 실행한 다음에 입력 값을 초기화하여 빈칸으로 만들고 싶을 때 사용해요.
- type: query
resource: mysql
sqlType: insert
sql: >
INSERT INTO properties
SET created_at = NOW()
, name = :name
params:
- key: name
label: 자산명
resetAfterSubmit: true
blocks.redirectAfterSubmit
블록을 실행한 다음 특정 페이지로 바로 이동하고 싶을때 사용해요.
- type: query
name: 추가
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO FaqContent
SET name = :name,
type2 = :type2,
created_at = NOW()
redirectAfterSubmit: "/faq-contents"
# redirectAfterSubmit: |
# /faq-contents#{"modal":{"id":{{insertId}},"$name":"view","p":"pages.142.blocks.0"},"tableSelectedRows":[]}
blocks.closeAfterSubmit
모달을 띄운 다음, 데이터 수정 등 작업 제출시 창을 닫을 수 있습니다.
modals:
- path: modal1-:id
header: false
height: 400px
blocks:
- type: http
name: 포스트 업데이트
axios:
method: POST
url: https://jsonplaceholder.typicode.com/posts
closeAfterSubmit: true
params:
- key: published_at
blocks.autosubmit
SELECT 쿼리 블록이나 GET API 등 데이터 조회 블록에서 params와 함께 필터 검색을 할 때 사용됩니다.
조회버튼을 누르지않고 값을 입력하면 바로 제출하게 되는 옵션입니다.
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, name FROM wine_stock
WHERE name LIKE CONCAT('%', :category ,'%')
autosubmit: true
params:
- key: category
dropdown:
- 블랑
- 트라피체
blocks.containerClass
블록 영역의 container 스타일을 바꿀 수 있습니다. (예시: containerClass: container mx-auto
)
pages:
- path: select/UserTemplateLog
blocks:
- type: query
resource: selectCloud
containerClass: container mx-auto
sqlType: select
sql: ...
blocks.showDownload
showDownload: false
데이터 다운로드 기능을 끌 수 있습니다.
pages:
- path: show-download-false
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT *
FROM test
LIMIT 10
showDownload: false
showDownload: csv
조회한 데이터를 CSV 파일로 다운로드할 수 있습니다.
pages:
- path: show-download-csv
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT *
FROM test
LIMIT 10
showDownload: csv
showDownload: csv formatted
데이터 다운로드는 '데이터 원본'을 기본으로 합니다. 셀렉트 어드민에 보여지는대로 다운로드 받고 싶으시다면 formatted를 추가해주세요.
showDownload: csv xlsx
조회한 데이터를 엑셀 파일(xlsx)로 다운로드 할 수 있습니다.
showDownload: csv formatted xlsx
엑셀 파일을 csv formatted와 동일하게 셀렉트 어드민에 보이는대로(formatted) 다운로드 받을 수 있습니다.
showDownload: csv selected
selectOptions 적용시 조회한 데이터의 일부를 선택하여 다운로드할 수 있습니다.
showDownload: csv paginated
paginationOptions 적용시 조회 데이터의 본 페이지 내역만 다운로드할 수 있습니다.
blocks.showRefresh
각 block에 showRefresh를 추가할 수 있습니다. 블록 영역 마우스오버시 “새로고침” 버튼이 보이게 되고 클릭시 해당 블록을 새로고침하게 됩니다.
blocks:
- type: query
id: query1
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM properties3 WHERE id = :id LIMIT 3
params:
- key: id
value: 1
showRefresh: true
responseFn: |
// query2.trigger() // 이 내용 새로고침시 query2도 새로고침
- type: query
id: query2
resource: mysql.qa
sqlType: select
sql: >
select NOW()
showRefresh: true
params:
- key: id2
value: 2
blocks.downloadAfterSubmit
- 데이터의 화면 표시가 불필요하거나 대량 데이터를 다운로드 받아야할때
downloadAfterSubmit: true
를 추가해보세요. - 바로 다운로드하고 화면에 렌더링하지 않습니다. (10만건이상 처리)
- query SELECT, http GET 일때만 작동합니다.
- 기본적으로
autoload: false
를 함께 추가해야합니다.
blocks:
-
type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM test_10k
WHERE id BETWEEN :from AND :to
LIMIT 100000
# type: http
# axios:
# url: https://api.selectfromuser.com/sample-api/products
# method: GET
# rowsPath: rows
params:
- key: from
value: 1
- key: to
value: 5000
submitButton:
label: 다운로드
autoload: false
showDownload: csv xlsx
downloadAfterSubmit: true
blocks.log
쿼리 실행을 로깅하여 모니터링하고 싶을 때 이용해요.
포맷과 예제
- log: [key] [memo]
- log: 호텔변경 id= before= after=BNB
- log: 월드관리_색상변경
blocks:
- type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO test SET name = :name, created_at = NOW()
params:
- key: name
# log: true
log: 데이터추가 name={{name}}
log.prompt
데이터 수정시 사유입력 프롬프트를 제공합니다.
prompt: true 적용시 변경사유를 남기고 실행하는 구조입니다. 조회성 블록 외, 수정하는 query, http api 에서만 켜집니다.
- type: right
blocks:
- type: http
axios:
url: https://httpbin.selectfromuser.com/anything
method: POST
log:
name: 수정_와인목록_API
prompt: true
# promptComments:
# - 고객요청
# - 파기
params:
- key: name
value: HELLO
- type: left
blocks:
- type: query
resource: mysql.qa
sqlType: insert
sql: SELECT 1
log:
name: 수정_와인목록
prompt: true
promptComments:
- 고객요청
- 파기
params:
- key: name
value: HELLO
blocks.roles
메뉴 단위보다 더 세부적인 접근제어가 필요한 경우 block 단위 권한지정을 사용해보세요.
- 모든 block 요청시 roles.edit에 권한지정시 서버단에서 차단
- menus roles처럼 권한그룹, email::, select::edit 사용가능
- 차단되면 '사용자에게 수정 권한이 없습니다' 표시
- type: http
axios:
url: https://httpbin.selectfromuser.com/anything
method: POST
roles:
edit:
- email::testsupport@selectfromuser.com
blocks.paginationOptions
주로 조회할 때 페이지네이션을 추가해서 필요한 정보만 빨리 볼 수 있게 도와줍니다.
perPage
페이지 당 보여지는 기본 row 개수를 지정합니다.
- type: query
...
paginationOptions:
enabled: true
perPage: 10
total
별도의 쿼리없이 서버사이드 페이지네이션을 적용할 수 있습니다.
테이블 내역의 전체개수를 total로 설정합니다.
반드시 page_offset
, page_limit
, mode: remote
과 함께 써야합니다.
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT *
FROM properties
LIMIT :page_offset, :page_limit
paginationOptions:
enabled: true
perPage: 10
total: 5000
mode: remote
- type: query
resource: mongodb
sqlType: select
query:
collection: properties
find: {}
limit: ${{page_limit}}
skip: ${{page_offset}}
paginationOptions:
enabled: true
perPage: 10
total: 5000
mode: remote
jumpPage
숫자를 입력해 특정 페이지로 바로 이동할 수 있어요.
paginationOptions:
enabled: true
jumpPage: true
position: top
페이지네이션 UI를 테이블 위쪽으로 옮길 수 있습니다.
paginationOptions:
enabled: true
position: top
blocks.searchOptions
조회한 내역들을 검색할 수 있게 지원합니다. 프론트 레벨에서 검색하기 때문에 서버 부하를 줄일 수 있는 장점이 있습니다.
- type: query
...
searchOptions:
enabled: true
trigger
테이블 검색 시, 내용을 입력하고 엔터(enter)를 눌렀을 때만 조회하게 설정할 수 있습니다. 데이터를 안정적으로 조회할 수 있는 장점이 있습니다.
searchOptions:
enabled: true
trigger: enter
placeholder
테이블의 검색 필드 안에 placeholder 값을 입력할 수 있습니다. 사용자가 참고할 수 있는 정보를 기입할 수 있습니다.
searchOptions:
enabled: true
placeholder: '입력한 검색어로 찾습니다.'
ignoreCase
검색시 알파벳 대소문자 무시 설정을 합니다. (기본값은 ignoreCase: true
)
searchOptions:
enabled: true
ignoreCase: false
ignoreWhitespace
검색시 공백 무시 설정을 합니다. (기본값은 ignoreWhitespace: true
)
searchOptions:
enabled: true
ignoreWhitespace: false
blocks.filterOptions
테이블의 특정 컬럼만 검색하는 창을 만듭니다.
sql: SELECT * FROM test_10k
columns:
name:
filterOptions:
enabled: true
placeholder: 이름 검색
blocks.tableOptions
fixed
: 테이블의 머리행(header)을 고정할 수 있습니다.height
값 입력 시 테이블 높이를 설정하게 됩니다. 300~400px 또는 60~80vh 로 설정하시면 기본적인 범위를 아실 수 있습니다.
condensed
: 테이블 행(row) 높이를 좁게 만들어 한페이지에서 더 많은 데이터를 볼 수 있습니다.bordered
: 테이블 컬럼 사이의 세로줄을 없앨 수 있습니다.striped
: 테이블 행들을 구분하기 쉽게 번갈아가며 색상을 바꿔줍니다.small
: 테이블 안의 글씨 크기를 작게 만듭니다.hidden: true
: 테이블의 전체 항목을 가릴 수 있습니다. 각 column에 값이 있으면 무시합니다. 원하는 항목만 표시할 때 사용해요. (columns.hidden: false
)sortable: false
: 테이블 전체 컬럼의 정렬 기능을 끌 수 있습니다. 원하는 컬럼만 정렬을 켜보세요.thClass: text-center
,tdClass: text-center
: 테이블 전체에 부트스트랩 클래스를 적용할 수 있어요.
- type: query
...
tableOptions:
fixed: true
height: 300px
# height: 80vh
condensed: true
bordered: false
striped: true
small: true
sortable: false
# hidden: true
thClass: text-center
tdClass: text-center
tableOptions.cell
cell: true
를 추가하면 특정셀을 마우스/엔터로 선택하거나 키보드 화살표로 이동할 수 있어요.
columns, updateOptions와 함께 사용하면 특정셀을 수정하고 저장할 수도 있습니다.
tableOptions:
cell: true
columns:
inflow:
updateOptions:
type: query
resource: mysql.qa
sql: UPDATE wine_stock SET inflow = :value WHERE id = :id
toast: 저장
blocks.sortOptions
데이터 조회 결과의 정렬 기본값을 설정할 수 있습니다.
blocks:
- type: query
resource: mysql
sqlType: select
sql: >
...
sortOptions:
enabled: true
initialSortBy:
field: Ratio # 필드 이름
type: desc # desc, asc
multipleColumns: true
로 여러개 컬럼도 설정 가능합니다.
blocks:
- type: query
resource: mysql
sqlType: select
sql: >
...
multipleColumns: true
sortOptions:
enabled: true
initialSortBy:
- field: Ratio
type: desc
- field: ClickCount
type: desc
blocks.columnOptions
주로 데이터 조회 결과를 다르게 표현하고 싶을 때 이용합니다. columnOptions와 columns 사용 시 값의 데이터 타입이 text로 지정됩니다. 데이터 타입을 바꾸고 싶은 경우 type키를 이용해주세요.
- type: query
...
columnOptions:
- field: id
- field: name
label: 이름
- field: json
format: json
columnOptionsAppend
전체 필드를 보여주면서, 일부 필드만 columnOptions로 수정하고 싶을 때 사용하세요.
columnOptionsAppend: true
columnOptions:
- field: name
label: 이름
blocks.columns
columnOptions와 동일한 기능이고 양식이 다릅니다.
- type: query
...
columns:
id:
name:
label: 이름
rank:
label: 랭킹
type: number
created_at:
blocks.params
블록 안에서 parameter를 쓸 수 있습니다. 데이터를 조회하거나 생성, 수정할 때 입력폼(input) 역할을 하게 됩니다.
- type: query
...
params:
- key: id
label: 아이디(ID)
blocks.tabOptions
블록 안에 세부 탭을 여러개 추가할 수 있습니다.
blocks:
- type: query
...
tabOptions:
autoload: 1
tabs:
- name: 주문
blocks:
- name: 결제
blocks:
blocks.chartOptions
조회 데이터 결과를 막대(bar), 선(line), 원형(doughnut, polarArea) 등 차트나 그래프로 표현할 수 있습니다.
blocks:
- type: query
name: 상품 카테고리 비율
resource: mysql.qa
sqlType: select
sql: >
SELECT `type`, COUNT(id) AS count
FROM products
WHERE `type` IS NOT NULL
GROUP BY `type`
chartOptions:
type: bar # line, doughnut, polarArea
x: type
y: count
showButtonWithResult: true
chartOptions.y
차트의 y축에 해당하는 항목을 복수로 선택할 수 있어요.
chartOptions:
type: bar
x: date
y:
- company
- user
chartOptions.backgroundColor
차트의 배경색을 바꿀 수 있어요.
chartOptions:
type: bar
x: date
y:
- company
- user
backgroundColor:
company: "RoyalBlue"
user: "#0D6EFD"
blocks.display
display: form
조회한 데이터를 form 양식으로 보여주거나, 데이터를 입력하는데 사용할 수 있어요.
주로 viewModal, modals나 tabOptions, type: left/right 등을 사용해 부모블록의 하위에 쓰입니다.
조회 예제
viewModal:
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT id, name, email, created_at FROM users WHERE id = :id
params:
- key: id
valueFromRow: id
display: form
수정 예제
viewModal:
blocks:
- type: query
resource: mysql.qa
sqlType: update
sql: >
UPDATE users
SET updated_at = NOW()
, name = :name
, email = :email
WHERE id = :id
params:
- key: id
valueFromRow: id
- key: name
- key: email
display: form
display: form inline
form 양식 데이터를 inline 형태로 나열해줍니다.
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT id, name, email, created_at FROM users WHERE id = :id
params:
- key: id
valueFromRow: id
display: form inline
display: form + INSERT
데이터를 추가하는 사용자에게 더 알맞는 입력폼을 만들어보세요.
formOptions로 양식 너비를 조절할 수 있어요.
firstLabelWidth
: 줄 가장 왼쪽 최소 너비labelWidth
: 레이블 최소 너비width
: 입력 폼 너비
추가 예제
blocks:
- type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO products
SET created_at = NOW()
, main_image = :main_image
, name = :name
, code = :code
, price = :price
, sell_status = :sell_status
display: form
formOptions:
firstLabelWidth: 200px
labelWidth: 100px
width: 400px
params:
- key: main_image
label: 대표이미지
title: 상품 추가
required: true
- key: name
label: 상품명
group: prod
- key: code
label: 상품코드
group: prod
- key: price
label: 판매가
format: number
group: sell
- key: sell_status
label: 판매여부
group: sell
width: 200px
divider, title, subtitle
필드와 속성이 여러개면 구분이 필요합니다. 구분선이나 묶음 이름을 넣을 수 있어요.
viewModal:
useColumns: id
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
select *
from properties
where id = :id
params:
- key: id
valueFromRow: id
display: form
columns:
id:
title: 기본정보
subtitle: 상품에 대한 기본정보
name:
address:
divider: true
title: 상세정보
type:
hero_image_url:
memo:
display: col-1 | col-2 | col-3
조회한 데이터를 피봇 테이블로 보여줍니다.
blocks:
- type: query
sqlType: select
...
display: col-1 # col-2, col-3
# thStyle:
# width: 150px
display: col-1/2 + updateOptions
updateOptions, columns.editable 과 함께 사용 시 display: col-1/2 뷰를 유지하면서 데이터를 수정할 수 있게 됩니다.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, name
FROM test
WHERE id = :id
display: col-1
updateOptions:
type: query
resource: mysql.qa
sql: >
UPDATE test
SET created_at = NOW()
, name = :name
WHERE id = :id
params:
- key: id
valueFromRow: id
columns:
id:
name:
editable: true
display: post
- 내용을 위에서 아래로 문단 표시합니다.
- thClass, thStyle, thClass, tdStyle 활용 가능
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT title, body FROM app_notice
WHERE id = :id
params:
- key: id
valueFromRow: id
display: post
columns:
title:
# tdClass: text-lg font-bold text-slate-500
label: " "
tdClass: px-8 leading-8
tdStyle:
fontSize: 1.5rem !important
body:
# tdClass: p-5 text-sm font-semibold text-slate-700
tdClass: text-sm p-8
label: " "
display: card
데이터를 카드형태로 보여줍니다.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, name, memo
FROM properties
ORDER BY id ASC
LIMIT 100
display: card
display: metric
지표 데이터를 보여줄 때 유용합니다.
metricOptions.type: bar
blocks:
- type: query
resource: mysql.qa
name: category barlist
sql: >
SELECT COALESCE(`type`, 'NONE') AS 't', count(id) * 334 AS 'c'
FROM properties
GROUP BY `type`
ORDER BY 2 DESC
display: metric
metricOptions:
type: bar
name: t
href: "#id={{id}}"
nameLabel: 소스
value: c
valueLabel: 방문자수
total: 전체 방문 # value 값들의 전체 합과 라벨
metricOptions.type: category
blocks:
- type: query
resource: mysql.qa
name: category
sql: >
SELECT count(id) as 'c'
FROM users
GROUP BY ISNULL(`signed_at`)
ORDER BY 1 DESC
display: metric
metricOptions:
type: category
names:
- Active users
- Inactive users
value: c
total: 전체 사용자 # value 값들의 전체 합과 라벨
display: calendar
조회한 데이터를 달력 형태로 보고, 날짜별 상세 내역을 클릭하여 모달을 띄울 수 있어요.
cache: true
: 캐시를 남겨 불필요한 로딩을 방지해줍니다. 이전달, 다음달을 오고갈때 더 자연스러워요.color
: tailwindcss color class를 지원합니다. https://tailwindcss.com/docs/customizing-colorsformatFn: numberPart
: 숫자에 구분점(,)을 찍어줍니다.openModal
: 상세 내역을 클릭하여 모달을 띄울 수 있어요.modals
와 함께 이용해보세요.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT
DATE_FORMAT(created_at, '%Y-%m-%d') as 'date',
CONCAT(COUNT(id), ' 건') AS count_order,
CONCAT(SUM(amount), ' 원') AS sum_order_amount,
CONCAT('취소 ', COUNT(IF(status = 'cancel', id, NULL)), ' 건') AS count_order_cancel
FROM orders
WHERE created_at BETWEEN :calendar1 AND :calendar2
GROUP BY 1
params:
- key: calendar
range: true
valueFromCalendar: true
display: calendar
autoload: true
cache: true
columns:
count_order:
label: 총 주문수
color: blue-600
formatFn: numberPart
sum_order_amount:
label: 주문금액 합계
color: green-600
formatFn: numberPart
openModal: order-list
count_order_cancel:
label: 취소수량
color: gray-500
display: timeline
조회한 데이터를 날짜로 정렬해서 타임라인 뷰로 볼 수 있습니다. 변경내역 조회 등에 활용할 수 있어요.
timelineOptions
로 정렬 기준 컬럼을 선택하고 template을 지정할 수 있습니다.- template은 HTML 적용 가능합니다.
- 조회한 값을 template 안에서
{{value}}
방식으로 사용해요.
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT payments.created_at, users.name AS user_name,
payments.plan_name,
payments.json->"$.before" as `before`,
payments.json->"$.after" as `after`
FROM payments, users
where payments.team_id = :team_id
and users.id = payments.user_id
ORDER BY payments.id DESC LIMIT 10
display: timeline
timelineOptions:
useColumn: created_at
template: |
<b>{{user_name}}</b>
{{plan_name}}: <b>{{after}}</b>로 변경했습니다.
# template: <b>{{user_name}}</b>님이 플랜을 변경했습니다. <b>{{plan_name}}</b> {{before}} → {{after}}
# template:
# Payment create: <b>{{user_name}}</b>님이 새로운 플랜을 결제하였습니다.
params:
- key: team_id
display: map
위도, 경도 데이터를 바탕으로 마커(marker)를 추가하거나 불러온 데이터를 통해 폴리곤(Polygon) 설정이 가능합니다. 마커를 클릭했을때 모달을 띄워 상세 정보를 조회할 수도 있습니다.
지도 기능
- display: map
- displayFn 으로 네이버지도 API 그대로 추가 가능합니다.
- 마커, 인포, 폴리곤, 폴리라인
- 클릭시 모달 띄우기, 액션 띄우기
예제1: 지도에 마커 표시하기
menus:
- path: pages/EBulXa
name: 지도 마커
pages:
- path: pages/EBulXa
title: 마커를 클릭하면 모달 열기
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM test_map
display: map
displayFn: |
for (const row of rows) {
// 마커 추가
const marker = new naver.maps.Marker({
position: new naver.maps.LatLng(row.lat, row.long),
title: row.name,
map: map,
});
naver.maps.Event.addListener(marker, 'click', function(e) {
// console.log('>>>선택한 로우 데이터', row)
// 모달 열기
openModal('view', row)
});
}
// 지도를 첫번째 마커로 옮기기
map.setCenter(new naver.maps.LatLng(rows[0].lat, rows[0].long))
height: 600px
width: 100%
mapOptions:
zoom: 15
zoomControl: true
modals:
- path: view
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM test_map
WHERE id = :id
params:
- key: id
valueFromRow: true
예제2: 지도에 상세보기, 폴리곤, 폴리라인 표시하기
menus:
- path: pages/JGi9HR
name: openAction, polygon
pages:
- path: pages/JGi9HR
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM test_map
display: map
displayFn: |
// 폴리곤 표시
var polygon = new naver.maps.Polygon({
map: map,
paths: [
[
new naver.maps.LatLng(37.37544345085402, 127.11224555969238),
new naver.maps.LatLng(37.37230584065902, 127.10791110992432),
new naver.maps.LatLng(37.35975408751081, 127.10795402526855),
new naver.maps.LatLng(37.359924641705476, 127.11576461791992),
new naver.maps.LatLng(37.35931064479073, 127.12211608886719),
new naver.maps.LatLng(37.36043630196386, 127.12293148040771),
new naver.maps.LatLng(37.36354029942161, 127.12310314178465),
new naver.maps.LatLng(37.365211629488016, 127.12456226348876),
new naver.maps.LatLng(37.37544345085402, 127.11224555969238)
]
],
fillColor: '#ff0000',
fillOpacity: 0.3,
strokeColor: '#ff0000',
strokeOpacity: 0.6,
strokeWeight: 3
});
// 폴리라인 표시
var polyline = new naver.maps.Polyline({
map: map,
path: [
new naver.maps.LatLng(37.359924641705476, 127.1148204803467),
new naver.maps.LatLng(37.36343797188166, 127.11486339569092),
new naver.maps.LatLng(37.368520071054576, 127.11473464965819),
new naver.maps.LatLng(37.3685882848096, 127.1088123321533),
new naver.maps.LatLng(37.37295383612657, 127.10876941680907),
new naver.maps.LatLng(37.38001321351567, 127.11851119995116),
new naver.maps.LatLng(37.378546827477855, 127.11984157562254),
new naver.maps.LatLng(37.376637072444105, 127.12052822113036),
new naver.maps.LatLng(37.37530703574853, 127.12190151214598),
new naver.maps.LatLng(37.371657839593894, 127.11645126342773),
new naver.maps.LatLng(37.36855417793982, 127.1207857131958)
]
});
for (const row of rows) {
// 마커 추가
const marker = new naver.maps.Marker({
position: new naver.maps.LatLng(row.lat, row.long),
title: row.name,
map: map,
});
// 인포창 필요없으면 생략해도됨 (클릭시 바로 모달만 열기)
// 인포창 시작 >>>
const contentString = `
<div class="p-5">
<h3>${row.name}</h3>
<p>${row.address}</p>
</div>`
const infowindow = new naver.maps.InfoWindow({
content: contentString
});
naver.maps.Event.addListener(marker, 'mouseover mouseout', function(e) {
if (infowindow.getMap()) {
infowindow.close();
} else {
infowindow.open(map, marker);
}
})
// 인포창 끝 <<<
naver.maps.Event.addListener(marker, 'click', function(e) {
// console.log('>>>선택한 로우 데이터', row)
// 모달 열기
openModal('view', row)
// 액션 열기
// openAction('UPDATE_STATUS', row)
});
}
// 지도를 첫번째 마커로 옮기기
map.setCenter(new naver.maps.LatLng(rows[0].lat, rows[0].long))
height: 600px
# width: 100%
# mapOptions:
# zoom: 15
# zoomControl: true
modals:
- path: view
mode: side
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM test_map
WHERE id = :id
params:
- key: id
valueFromRow: true
actions:
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT 1
name: UPDATE_STATUS
modal: true
참고자료
display: shadow
display: shadow
는 shadow DOM 개념과 유사한 기능입니다. 공통 상세정보를 가져와 하위 여러개 블록을 표시해보세요.
TIP
단계별로 데이터를 조회하는게 알맞지 않거나, 모달 URL 구성시 모달 하위에 사용하기 위한 params 값에 긴 내용으로 인해 URL 최대 길이를 넘거나 인코딩이 실패*할 수 있습니다.
*셀렉트에서는 모달 생성시 useColumn 된 내용들을 가급적 URL에 저장하고 있습니다. (새로고침이나 뒤로가기등 지원)
이때 display: shadow를 통해 tabOptions나 한번더 감싸지 않고 조회된 값을 하위 블록으로 보내는 방법을 이용하실 수 있습니다.
- path: pages/CG6psi
title: Large valueFromRow modal
subtitle: 내용
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM wine_stock
columns:
name:
openModal: test
modals:
- path: test
useColumn:
- id
usePage: pages/CG6psi/modal
- path: pages/CG6psi/modal
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM wine_stock WHERE id = :id
params:
- key: id
valueFromRow: true
display: shadow
blocks:
- type: http
axios:
url: https://httpbin.selectfromuser.com/anything
method: POST
params:
- key: name
defaultValueFromRow: name
- key: memo
defaultValueFromRow: memo
format: textarea
display: html table
데이터 조회 결과를 HTML 표 형태로 출력합니다. 표는 사용자 정의 스타일을 적용해 결과를 직관적으로 시각화합니다.
테이블 구성 요소
thead
: 테이블 헤더 정의. 컬럼 그룹화 및 설명 추가 가능.tbody
: 데이터 행 반복 출력. 데이터 매핑은{{param}}
을 사용.tfoot
: 총계 또는 요약 정보 추가. 계산 결과를 포함할 수 있음.
데이터 후처리
responseFn
: 조회된 데이터를 렌더링 전에 수정하는 로직을 정의합니다.totalFn
: tfoot에 출력할 총계 데이터를 계산합니다.
중요 참고사항
- 데이터 바인딩:
{{param}}
구문을 사용해 SQL 조회 결과와 UI를 연결합니다. - 스타일 적용: 각 셀 및 행에 class 속성을 추가해 UI 스타일을 커스터마이징하세요.
- 후처리 함수:
responseFn
과totalFn
을 활용해 데이터 포맷팅 및 계산 로직을 작성할 수 있습니다.
pages:
- path: pages/3qQ2sR
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT
id, name, vintage, inflow, outflow, price, safeflow, deleted_at
FROM wine_stock
WHERE inflow IS NOT NULL
display: html table
# display: table html
actions:
- label: 확정
responseFn: |
return rows.map(e => {
e.deleted_at = e.deleted_at ? moment(e.deleted_at).format('YY.MM.DD') : '-'
return e
})
thead:
rows:
- class: bg-neutral-100 text-neutral-800 font-medium divide-x
cells:
- th: { rowspan: 2, content: "번호" }
- th: { rowspan: 2, content: "상품명" }
- th: { colspan: 2, content: "재고량" }
- th: { rowspan: 2, content: "판매기간", help: "해당기간까지 게시됩니다.", width: 200px}
-
class: "bg-neutral-100 text-neutral-800 font-medium divide-x"
cells:
- th: { class: "border", content: "입고" }
- th: { class: "border", content: "출고" }
tbody:
rows:
- class: "text-center divide-x hover:bg-neutral-100"
cells:
- td: { content: "{{id}}" }
- td: { content: "{{name}}", class: "text-left" }
- td: { content: "{{inflow}}", class: "text-right" }
- td: { content: "{{outflow}}", class: "text-right" }
- td: { content: "{{deleted_at}}" }
tfoot:
rows:
- class: "font-medium divide-x text-center"
cells:
- th: { colspan: 2, class: "bg-neutral-100 text-neutral-800", content: "수량합계" }
- td:
class: "text-right"
content: <span class="text-blue-600">{{inflow}}</span> Box
- td:
class: "text-right"
content: <span class="text-blue-600">{{outflow}}</span> Box
- td: ""
- class: "font-base divide-x text-center"
cells:
- th: { colspan: 2, class: "bg-neutral-100 text-neutral-800", content: "금액합계" }
- td: { class: "text-right", content: "{{inflow_amount}} 원" }
- td: { class: "text-right", content: "{{outflow_amount}} 원" }
- td: ""
- class: "font-base divide-x text-center"
cells:
- th: { colspan: 2, class: "bg-neutral-100 text-neutral-800", content: "재고금액합계" }
- td: { colspan: 2, content: "{{amount}} 원" }
- td: ""
totalFn: |
total.inflow = _.sumBy(rows, 'inflow')
total.outflow = _.sumBy(rows, 'outflow')
total.inflow_amount = _.sum(rows.map(e => (+e.price || 0) * +e.inflow))
total.outflow_amount = _.sum(rows.map(e => (+e.price || 0) * +e.outflow))
total.amount = total.inflow_amount - total.outflow_amount
total.inflow_amount = filters.number(total.inflow_amount)
total.outflow_amount = filters.number(total.outflow_amount)
total.amount = filters.number(total.amount)
blocks.formOptions
params와 함께 쓰이며, 입력 양식 스타일을 다양하게 수정할 수 있습니다.
display: inline
formOptions:
display: inline
params:
- key: customer_name
label: 고객이름
form group
formOptions:
labelWidth: 140px
firstLabelWidth: 140px
width: 300px
params:
- key: customer_name
label: 고객이름
group: 1
display: col
formOptions:
display: col
labelWidth: 140px
firstLabelWidth: 140px
params:
- key: customer_name
label: 고객이름
group: 1
- key: product_name
label: 상품명
group: 2
form group class
formOptions:
groupClass: flex-col py-5 border-b
groupLabelClass: text-start
blocks.submitButton
params와 주로 쓰입니다. 값을 입력하고 제출할 때 버튼의 이름이나 색상을 바꿀 수 있어요.
params:
- key: name
submitButton:
label: 검색
type: primary
submitButton.type
버튼 색상은 아래와 같이 바꿀 수 있습니다.
type: default
# type: primary
# type: primary-light
# type: danger
# type: danger-light
# type: warning
# type: warning-light
# type: success
# type: success-light
blocks.resetButton
params 필드에 입력한 값들을 일괄적으로 빈값으로 바꾸고 싶을 때 이용합니다.params.key.defaultValue
에 상관없이 항상 빈값으로 처리하고 싶은 경우 clear: true
를 추가합니다.
# resetButton: true
resetButton:
label: 초기화
type: light
# clear: true
blocks.viewModal
테이블 마지막에 '조회' 버튼이 추가됩니다. 클릭시 새로운 모달 팝업이 노출됩니다.
- type: query
...
viewModal:
blocks:
- type: query
viewModal.useColumn
특정 컬럼을 모달 조회 링크로 지정할 수 있습니다.
viewModal.params.valueFromRow
조회한 row 데이터를 모달 안에서 사용할 수 있습니다.
blocks.modals
blocks.selectOptions
selectOptions는 테이블의 row를 선택할 때 쓰입니다. actions와 함께 사용할 때 유용합니다.
selectOnCheckboxOnly: true
테이블의 row를 선택할 때, 체크박스 영역을 눌러야만 선택할 수 있게 설정하게 됩니다.
blocks:
- type: query
selectOptions:
enabled: true
selectOnCheckboxOnly: true
selectOptions.template
selectOptions로 테이블 row 내역을 선택했을때 template 기능을 활용하여 해당 내역을 테이블 상단에 표기할 수 있습니다.
selectOptions:
enabled: true
# template: |
# `선택됨: ${ tableSelectedRows.map(e => e.id).join(',') }`
template: |
`선택됨: ${ tableSelectedRows.map(e => `<span class="rounded bg-slate-100 p-1">${e.name}</span>`).join(' ') }`
blocks.refs
특정 컬럼의 값을 링크로 만들고 클릭하여 새창을 열 수 있습니다.
조회 결과 row의 다른 컬럼 값을 가져와서 변수에 넣을 수 있어요.
sql: SELECT id, name, code FROM coupons LIMIT 10
columns:
refs:
- column: name
href: https://search.naver.com/search.naver?query={{code}}
blocks.actions
선택한 1개 또는 여러 항목으로 다음 화면을 진행합니다.
blocks.onRowClick
특정 행을 클릭하여 모달을 열거나 액션을 실행할 수 있습니다.
<주의>
- 행row 클릭시 1개의 openModal이나 openAction만 실행할 수 있어요.
- actions로 생기는 버튼은 헷갈리지 않게 hidden으로 숨겨주세요.
viewModal 모달 열기
onRowClick:
openModal: true
viewModal:
blocks:
- type: query
modals 모달 열기
onRowClick:
openModal: modal-path
modals:
- path: modal-path
actions 실행
onRowClick:
openAction: action1
actions:
- name: action1
hidden: true
actions로 URL 열기
onRowClick:
openAction: action2
actions:
- name: action2
openUrl: https://www.selectfromuser.com
hidden: true
columns: {}
columnOptions와 동일한 기능이고 양식이 다릅니다.
- type: query
...
columns:
id:
name:
label: 이름
rank:
label: 랭킹
type: number
created_at:
columns.format
개별 컬럼의 데이터 포맷(format)을 설정할 수 있습니다.
- type: query
...
columns:
id:
name:
created_at:
format: date
agree:
format: checkbox
trueValue: 1
falseValue: 0
format: url
특정 컬럼의 값을 활성화된 URL 링크(link)로 표시할 수 있습니다.
- path: manage-animal
blocks:
- type: query
resource: acme
sqlType: select
sql: >
SELECT id, name, image_url
FROM animals limit 10
columns:
image_url:
format: url
format: image
필드 안의 값을 바탕으로 이미지(image)로 표시할 수 있습니다.
thumbnail: true
: 이미지를 썸네일 형태로 설정해줍니다.flex: true
: 이미지가 여러개일 때 가로로 정렬합니다.thumbnailWidth
: 썸네일의 가로 크기를 설정합니다. (기본값은 column 100%)thumbnailHeight
: 썸네일의 세로 크기를 설정합니다.
- path: manage-animal
blocks:
- type: query
resource: acme
sqlType: select
sql: >
SELECT id, name, image_url, image_url->'$.data[*].url' AS thumbnail, full_address, created_at, registered_at
FROM animals
LIMIT 10
columns:
image_url:
format: url
thumbnail:
format: image
width: 400px
height: 120px
thumbnail: true
thumbnailWidth: 100px
flex: true
style: >
border-radius: 10px
이미지를 세로 정렬하고 스크롤하고 싶을 때는 tdStyle, height, overflow를 이용해주세요.
columns:
thumbnail:
format: image
thumbnail: true
thumbnailWidth: 100px
tdStyle:
height: 200px
overflow: scroll
format: json, json-inline
json 값을 1줄 또는 여러줄로 표기할 수 있습니다.
columns:
col1:
format: json # 여러줄로 표기
col2:
format: json-inline # 1줄로 표기
format: table
columns 안에서 사용 가능하며, JSON 데이터를 필드 안에 테이블 형태로 변환하여 보여줍니다.
sql: >
SELECT id, json->'$.data' AS json_to_table FROM users LIMIT 100
columns:
json_to_table:
format: table
Object 피봇 테이블 형태로 보여집니다.
{
"data": {
"id": 1234,
"amount": 99000
}
}
Array 일반적인 표 형태로 보여집니다.
{
"data": [
{
"id": 1234,
"amount": 99000,
"status": "DONE"
},
{
"id": 2234,
"amount": 179000,
"status": "DONE"
}
]
}
format: viewer
에디터로 입력한 데이터를 viewer 기능으로 보여줍니다. display: col-1, 2, post와 함께 적용할 수도 있어요.
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, wine_id, amount, memo FROM wine_stock LIMIT 5
columns:
memo:
format: viewer
width: 50vw
columns.valueAs
각 컬럼의 값들을 다른 텍스트로 표시할 수 있습니다. 데이터를 다른 이름으로 바꿔서 표기할 수 있어요.
# columns 용법
columns:
customer_link: # column name
format: url
valueAs: 링크
# columnOptions 용법
columnOptions:
- field: customer_link
format: url
valueAs: 링크
multiple valueAs
valueAs 아래에 값: 라벨
형태로 여러개를 설정할 수 있습니다. 실제 값이 바뀌는 것이 아닌, 표기가 바뀝니다. 표기된 데이터 필드에 마우스를 올리고 2~3초 기다리면 본래 값이 보여집니다.
- type: query
resource: testdb
sqlType: select
sql: >
SELECT *
FROM property
columns:
type:
valueAs:
HOTEL: 호텔
PENSION: 펜션
RESORT: 리조
format: image, url + valueAs
valueAs에 URL과 파라미터를 통해 데이터 기반으로 이미지, URL 결과를 보여줄 수 있습니다.
- path: manage-animal
blocks:
- type: query
resource: acme
sqlType: select
sql: >
SELECT id, name, postfix
FROM animals
LIMIT 10
columns:
postfix:
format: image
valueAs: https://placekitten.com/{{postfix}}?id={{id}}
columns.template
template 키와 HTML로 데이터 값을 더 복잡한 구성으로 표기할 수 있어요.
- type: query
resource: pgsql
sqlType: select
sql: >
SELECT
'media/cc0-audio/t-rex-roar.mp3' as url,
'media/cc0-videos/flower.webm' as video_url
# display: form
columns:
url:
template: |
<audio controls src="https://interactive-examples.mdn.mozilla.net/{{url}}"></audio>
video_url:
template: |-
<video controls width="250">
<source src="https://interactive-examples.mdn.mozilla.net/{{video_url}}" type="video/webm" />
</video>
columns.formatFn
formatFn을 통해 JavaScript 코드를 이용할 수 있어요. (column 단위로 적용)
return
이 없어도 inline으로 인식합니다.lodash
함수를 사용할 수 있습니다.- formatFn과 valueAs를 함께 쓰면 formatFn이 먼저 적용되어요.
- type: query
resource: testdb
sqlType: select
sql: >
SELECT *
FROM property
columns:
type:
formatFn: |
'$' + value
type2:
formatFn: |
_.capitalize(value || '')
type3:
formatFn: |
if (value == 'HOTEL') { return '호텔'}
if (value == 'PENSION') { return '펜션'}
if (value == 'RESORT') { return '리조트'}
type4:
formatFn: |
return {
HOTEL: '호텔',
PENSION: '펜션',
RESORT: '리조트'
}[value] || value
formatFn: datetime, datetime-local, datetimeA, datetimeA-local
datetime을 통해 날짜/시간 데이터를 원본 그대로 표기하거나, datetime-local로 로컬 타임존에 맞춰서 표기할 수 있어요.
datetimeA로 AM, PM으로 표기도 가능합니다.
columns:
created_at:
formatFn: datetime
# formatFn: datetime-local
# formatFn: datetimeA
# formatFn: datetimeA-local
formatFn: number, number0, numberPart
숫자에 천단위로 콤마(comma) 표시를 할 수 있습니다. 금액 정보 등을 표기할때 유용해요.
number
: 콤마 표시 (빈값이면 빈칸)number0
: 콤마 표시 (빈값이면 0)numberPart
: 콤마 표시 (앞뒤 문자열은 유지. 예를 들면 KRW 1500 -> KRW 1,500)
columns:
paid_amount:
formatFn: number
# formatFn: number0
# formatFn: numberPart
formatFn: number + prefix, postfix
formatFn number류 뒤에 prefix(접두어)와 postfix(접미어) 내용을 순서대로 붙일 수 있어요.
columns:
won_price:
type: number
formatFn:
- number0
- "12개월 " # prefix
- " 만" # postfix
formatFn: splitComma
필드 안의 데이터가 comma (,)
로 구분된 경우 보기 좋게 뱃지로 표기해줍니다. 태그나 카테고리 정보를 보여줄 때 유용해요.
columns:
tags:
label: 태그
formatFn: splitComma
sortable: false
formatFn: maskCenter4
필드 데이터의 중간 4자리를 가립니다.
phone:
label: 연락처
formatFn: maskCenter4
sortable: false
formatFn: maskEnd4
필드 데이터의 뒤 4자리를 가립니다.
phone:
label: 연락처
formatFn: maskEnd4
sortable: false
columns.hidden
특정 컬럼을 숨길 수 있습니다.
columns:
hidden_info:
hidden: true
columns.sortable
테이블 특정 컬럼의 정렬 기능을 끌 수 있습니다.
columns:
phone:
label: 연락처
sortable: false
columns.subtitle
특정 컬럼을 숨기고, 다른 컬럼 아래에 작게 표기할 수 있습니다.
columns:
id:
name:
label: 상품 이름
sortable: false
subtitle: store_path
store_path:
hidden: true
columns.color
특정 컬럼에서 데이터 값에 따라 색깔을 부여할 수 있습니다. 결제상태나 이슈 등을 구분하기 쉽게 설정해보세요.
📘 안내 사항
color:
아래에필드값
:색상
방식으로 추가할 수 있습니다.- 필드값은 대소문자를 구분합니다.
- 지원하는 색상들은 아래와 같습니다.
green
,yellow
,blue
,red
,gray
,pink
,orange
,purple
,brown
,light-gray
columns:
status:
label: 상태
color:
paid: green
canceled: red
columns.copy
특정 컬럼에 복사 아이콘을 추가합니다. 아이콘 클릭 한번으로 데이터를 클립보드에 복사할 수 있어요. 다른곳에 붙여넣기(ctrl+v) 해보세요.
columns:
email:
label: 이메일
copy: true
columns.openModal
특정 컬럼의 값들을 링크로 만들고, 링크를 클릭하여 모달을 열 수 있습니다.
columns:
id:
name:
openModal: orders-:id
modals:
- path: orders-:id
blocks:
columns.openUrl
특정 컬럼의 값을 링크로 만들고, 링크를 클릭하여 새창으로 특정 URL 주소를 열 수 있습니다.
클릭한 필드 row의 다른 값을 가져와서 변수에 사용 가능합니다.
columns:
id:
openUrl: https://search.naver.com/search.naver?query={{name}}
# target: _self
name:
columns.buttons
컬럼 안에 버튼을 추가할 수 있습니다.
columns:
조회:
buttons:
- label: 주문 상세
- label: 고객 상세
buttons.openModal
버튼을 눌렀을 때 모달을 띄워주는 방법입니다.
buttons.openModal
과modals.path
의 값으로 연결됩니다.
columns:
조회:
buttons:
- label: 상세
openModal: user-:id
type: primary
modals:
- path: user-:id
blocks:
- type: query
resource: mysql
sqlType: select
sql: >
SELECT * FROM users WHERE id = :id
params:
- key: id
valueFromRow: id
buttons.openAction
버튼buttons을 눌렀을 때 액션actions을 실행합니다.
buttons.openAction
과actions.name
으로 연결됩니다.
columns:
작업:
append: true
buttons:
- type: danger-light
label: 삭제
openAction: delete
actions:
- name: delete
label: 삭제
hidden: true
type: query
resource: mysql
sql: DELETE FROM category WHERE id = :id
single: true
confirmText: |
정말로 삭제하시겠습니까?
params:
- key: id
valueFromRow: id
buttons.openUrl
버튼을 눌렀을 때 특정 URL로 방문합니다.
- 어드민 내부 페이지로 이동할 때는
https://app.selectfromuser.com/admin/0000
어드민 기본 URL 뒤쪽을 작성합니다. - 외부 사이트로 이동할 때는 URL을 그대로 입력해주세요.
- 소속 테이블 데이터를 가져와서 URL에 파라미터로 쓸 수 있습니다.
{{field_name}}
sql: SELECT user_id, product_name FROM orders LIMIT 10
columns:
이동:
buttons:
- label: 주문
openUrl: orders?user_id={{user_id}}
# openUrl: https://search.naver.com/search.naver?where=nexearch&sm=top_hty&fbm=0&ie=utf8&query={{product_name}}
# target: _self # 기본은 새창
buttons.openPage
버튼을 눌렀을 때 (셀렉트 내부) 특정 페이지로 이동합니다.
openUrl과 차이점
- 셀렉트 앱(어드민) 안에서의 이동이기 때문에 전체 새로고침이 되지 않습니다.
- URL에 값을 넣어줍니다.
sql: SELECT id, name, created_at FROM companies LIMIT 100
columns:
name:
buttons:
- label: 회사 상세
openPage: company/{{id}}
buttons.openPopper
openPopper는 특정 버튼 클릭 시 팝업 레이어(Popper)를 열어 사용자가 추가 작업을 수행하거나 데이터를 확인할 수 있도록 하는 기능입니다. modals와 유사하지만, 더 가볍고 위치 설정이 유연합니다.
openPopper
: button 클릭시 Popper를 엽니다.popperOptions
: Popper의 위치 및 간격 조정- placement: right (Popper 우측)
- distance, skidding: 간격 조절가능
popperStyle
: Popper의 크기와 스타일을 정의- width, height: Popper의 크기
- overflow: 내용이 넘칠 때 스크롤 설정 (auto, scroll)
- padding: Popper 내부 여백
columns:
vintage:
buttons:
- label: 빈티지 조회
openPopper: true
popperOptions:
placement: right
popperStyle:
width: 500px
height: 400px
overflow: auto
padding: 20px
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT DISTINCT vintage FROM wine_stock WHERE id = :id
params:
- key: id
valueFromRow: true
columns:
vintage:
copy: true
showDownload: false
buttons.visible
컬럼 필드의 값에 따라 버튼이 보이거나 보이지 않게 설정할 수 있습니다.
menus:
- path: pages/columns-buttons-visible
name: columns
pages:
- path: pages/columns-buttons-visible
blocks:
- type: query
resource: mysql
sqlType: select
sql: >
SELECT 1 AS field, 2 AS field2
columns:
field:
label: field label
buttons:
- label: button label
visible: "{{ row.field == '1' }}"
field2:
label: field2 label
buttons:
- label: button2 label
visible: "{{ row.field !== '1' }}"
columns.updateOptions
한번에 여러 필드를 수정하지 않고, 낱개로 처리할 수 있습니다.
columns:
status:
updateOptions:
type: query
resource: mysql.qa
sql: UPDATE test_list SET status = :value WHERE id = :id
confirm: true
confirm: true
쿼리 또는 API 블록을 실행할 때, 한번 더 확인할 수 있게 합니다.
dropdown 적용
columns:
status:
dropdown:
- pending
- confirm
- closed
updateOptions:
type: query
resource: mysql.qa
sql: UPDATE test_list SET name = :value WHERE id = :id
confirm: true
format.checkbox 적용
columns:
is_used:
format: checkbox
trueValue: 1
falseValue: 0
updateOptions:
type: query
resource: mysql.qa
sql: UPDATE images SET is_used = :value WHERE id = :id
showActionGroup
columns.updateOptions안에 action 버튼을 보여주고 실행할 수 있습니다.
columns.showActionGroup
과actions.group
의 값으로 연결됩니다.actions.visible
로 필드 값과 조건에 따라 action 버튼을 보여주거나 가릴 수 있습니다.
sql: |
SELECT id, booking_status, canceled_at
FROM classBooking
LIMIT 100
columns:
booking_status:
updateOptions:
showActionGroup: bookingStatus
actions:
- label: 예약복구
single: true
placement: bottom right
button:
type: default
confirm: 예약을 정말 복구하시겠습니까?
type: query
resource: mysql
sql: >
UPDATE classBooking
SET booking_status = 'restored', canceled_at = NULL
WHERE id = :id
params:
- key: id
valueFromRow: id
reloadAfterSubmit: true
group: bookingStatus
visible: "{{ row.booking_status == 'canceled' }}"
format: toggle
조회한 테이블의 특정 컬럼 값을 수정할때 toggle UI를 적용할 수 있습니다.
columns:
status:
label: 상태
updateOptions:
type: query
resource: mysql.qa
sqlType: update
sql: >
UPDATE properties SET status = :value WHERE id = :id
params:
- key: id
valueFromRow: id
confirm: true
format: toggle
trueValue: 1
falseValue: 0
trueLabel: 활성
falseLabel: 비활성
format: table
params format: table
과 동일하게 columns updateOptions에서 JSON 형태의 데이터를 수정하는 UI 화면을 구성할 수 있어요.
- type: query
resource: mysql.qa
title: id=65 wine_stock의 property_json의 일부분을 수정
sqlType: select
sql: |
select id, property_data from wine_stock
where id = 65
display: form
responseFn: |
rows = rows.map(e => {
e.shippingRules = (e.property_data || {}).shippingRules || []
e.shippingRules = e.shippingRules.map(row => {
for (const key in row) {
row[key] = { value: row[key] }
}
return row
})
return e
})
return rows
columns:
property_data:
hidden: true
shippingRules:
label: 배송비 규정
updateOptions:
type: query
resource: mysql.qa
sqlType: update
sql: |
UPDATE wine_stock
SET property_data = :value
where id = :id
requestFn: |
const value = params.find(e => e.key == 'value')
const shippingRules = params.find(e => e.key == 'shippingRules')
value.value = JSON.stringify({
shippingRules: shippingRules.value.map(row => {
return {
minAmount: row.minAmount.value,
maxAmount: row.maxAmount.value,
shippingCost: row.shippingCost.value,
}
})
})
format: table
style:
width: 700px
class: table text-xs
headers:
minAmount:
label: 배송비시작
format: number
postfix: 원 이상
postfixStyle:
width: 50px
maxAmount:
label: 금액 (없으면 최대)
prefix: "~"
format: number
postfix: 원 미만
postfixStyle:
width: 50px
placeholder: 비어있으면 최대금액
shippingCost:
label: 배송비부과
format: number
postfix: 원
columns.updateOptions + display: form
Update 쿼리 블록을 조회/수정 모드를 껐다 켜는 방식으로 사용할 수 있습니다. 적용시 수정버튼이 추가됩니다.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM app_notice
WHERE id = :id
params:
- key: id
valueFromRow: id
columns:
title:
editable: true
author:
editable: true
body:
editable: true
format: textarea
date:
editable: true
published_at:
editable: true
format: date
display: form
updateOptions:
type: query
resource: mysql.qa
sqlType: update
sql: >
UPDATE app_notice
SET title = :title
, author = :author
, body = :body
, `date` = :date
, published_at = :published_at
WHERE id = :id
columns.searchOptions
개별 항목을 수정할때, 데이터를 먼저 검색하고 찾은 결과값을 반영할 수 있습니다. datalistFromQuery와 유사하지만 사용자에게 더 풍부한 검색 경험을 제공할 수 있습니다.
적용 가능한 상황
- display: form
- 개별 컬럼 updateOptions
blocks:
- type: query
resource: mysql
sqlType: select
sql: >
SELECT * FROM bookings LIMIT 10
columns:
id:
openModal: bookings/:id
modals:
- path: bookings/:id
blocks:
- type: query
resource: mysql
sqlType: select
sql: SELECT * FROM bookings WHERE id = :id
display: form
columns:
id:
assignee_id:
label: 담당자
updateOptions:
type: query
resource: mysql
sqlType: update
sql: >
UPDATE bookings
SET assignee_id = :value
WHERE id = :id
searchOptions:
enabled: true
# allowEmpty: false # 내역을 선택해야 적용 가능
# allowEdit: true # 선택지에 없는 텍스트 입력 가능
type: query
resource: mysql
sqlType: select
sql: >
SELECT
id AS value,
name AS '이름', status AS '상태'
FROM assingees
WHERE (name LIKE CONCAT('%', :value, '%') OR id = :value)
AND deleted_at IS NULL
columns:
value:
hidden: true
columns.prepend, append
특정 컬럼을 가장 앞(prepend)에 두거나, 가장 뒤로(append) 둘 수 있습니다.
columns:
field_first:
prepend: true
field_last:
append: true
columns.sticky
prepend: true
와 같이 사용되며, 특정 컬럼을 고정하여 좌우 스크롤하여도 계속 보이게 만듭니다.
columns:
column1:
prepend: true
sticky: true
column2:
columns.value, openModal
특정 컬럼에 고정된 값을 입력하고, 모달을 오픈하는 텍스트 링크로 만들 수 있습니다.
columns:
field:
value: 조회
openModal: column-openmodal
modals:
- path: column-openmodal
blocks: ...
columns.style
특정 컬럼에 CSS Style을 입힐 수 있어요. 원하는 스타일을 적용해보세요.
columns:
user_name:
style: |
text-align: center;
blocks
params: []
📘 params의 key 값은 영어 알파벳을 지원합니다.
key 의 값에 한글 등 알파벳이 아닌 문자를 입력하면 정상적으로 작동하지 않게 됩니다. 꼭 key: id 와 같이 영어 알파벳을 이용해주세요.
params.value 계열
parameter에 특정 값을 미리 입력하거나, 다른 곳에서 가져와서 입력하고 싶을 때 아래와 같은 키를 사용하세요.
key(키) | 설명 |
---|---|
valueFromRow | viewModal 안에서 사용 |
valueFromSelectedRows | selectOptions가 있는 actions 안에서 사용 |
defaultValue | 기본값을 설정하고 싶을 때 이용 |
defaultValueFromRow | viewModal 안에서 사용 |
valueFromEnv | API를 호출할 때 안전한 환경변수에서 이용 |
valueFromUserProperty | 사용자 속성에서 값을 가져와서 이용 |
params.defaultValueFromQuery
쿼리로 데이터 가져와서 기본값으로 넣는 방법입니다. 페이지 조회 시 1회 실행되며, sql 쿼리 결과의 첫번째 값 또는 AS value 값을 가져옵니다.
params:
- key: created_at
defaultValueFromQuery:
type: query
resource: acme
sql: select date_format(now(), '%Y-%m-%d')
params.defaultValueFn
params:
- key: created_at
defaultValueFn: |
// return moment().add(3, 'days').format('YYYY-MM-DD')
// return DateTime.now().toFormat('yyyy-MM-dd')
return '2022-08-12'
범위 날짜 사용시 (format: date, range: true)
return [] 형태로 입력해주세요.
params:
- key: created_at
format: date
range: true
defaultValueFn: |
return [
moment().add(-1, 'month').format('YYYY-MM-DD'),
moment().add(1, 'day').format('YYYY-MM-DD')
]
params.valueFromUserProperty
설정 > 계정의 사용자 이름(name)과 이메일(email), 아이디(id) 시스템 데이터를 변수로 이용할 수 있습니다.
params:
- key: user_name
valueFromUserProperty: "{{name}}"
- key: email
valueFromUserProperty: "{{email}}"
- key: user_id
valueFromUserProperty: "{{id}}"
params.valueFromSheet
CSV 데이터를 업로드하고 params로 매칭한 다음, DB에 데이터를 저장할 수 있습니다.
sheetOptions.append: false
: 처음에 추가한 파일만 적용됩니다.multiple: true
: 여러개의 파일을 한번에 선택하고 올릴 수 있습니다.accept
: 선택 가능한 파일 유형을 제한할 수 있어요.
actions:
- type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO products
SET name = :name
, code = :code
, unit = :unit
, created_at = NOW()
forEach: true
single: true
modal: true
params:
- key: sheet
format: sheet
# sheetOptions:
# append: false
# multiple: true
# accept: .csv,.xls,.xlsx
- key: name
valueFromSheet: 상품명
- key: code
valueFromSheet: 상품약어
- key: unit
valueFromSheet: 수량
sheetOptions.convertDate
엑셀에서는 날짜 데이터가 자체 시리얼날짜포맷으로 저장됩니다. 해당 시리얼넘버를 YYYY-MM-DD 형태로 저장하고 싶을때 별도의 코딩없이 convertDate 키를 활용하여 변환할 항목을 지정해줍니다.
엑셀 파일을 통해 INSERT 또는 UPDATE 쿼리를 벌크로 실행할 때 유용해요.
params:
- key: sheet
format: sheet
label: 엑셀 가져오기
accept: .xlsx
sheetOptions:
append: false
convertDate:
- 시작일
- 종료일
- key: id
valueFromSheet: id
- key: start_date
valueFromSheet: 시작일
- key: end_date
valueFromSheet: 종료일
params.valueFromSearch
웹 주소(URL)의 쿼리 스트링(Query string)을 파라미터로 사용할 수 있어요. updateParams와 함께 사용할 수 있습니다.
- e.g. xxx.com?property_id=123 → params.valueFromSearch=property_id
sql: >
SELECT id, property_id FROM bookings LIMIT 100
columns:
property_id:
updateParams:
id: "{{property_id}}" # id 라는 parameter에 property_id 값을 기입
viewModal:
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT *
FROM properties
WHERE id = :id
display: form
params:
- key: id
valueFromSearch: id
params.required
특정 파라미터를 필수값으로 설정하는 방법
params:
- key: email
required: true
params.format
input 안에 값을 입력할 때 format을 지정합니다.
format: number
format: date
format: time
format: datetime
format: text
format: editor
format: color
format.date | datetime | year | month | week
날짜 데이터의 종류별로 입력 포맷을 선택할 수 있습니다.
params:
- key: from
label: 시작일
format: date
# format: datetime
# format: year
# format: month
# format: week
placeholder: 주문생성일 기준
shortcuts
- format.date 과 함께 사용하는 손쉬운 입력 방법(shortcuts)을 설정해보세요.
params:
- key: from
label: 시작일
format: date
placeholder: 주문생성일 기준
shortcuts:
- label: 최초시점
date: 2024-01-01
- label: 어제
offset: -1
period: day
- label: 오늘
today: true
- label: 지난주
offset: -1
period: week
- label: 지난달
offset: -1
period: month
- label: 월초
startOf: month
- label: 월말
endOf: month
showButtons: true
- shortcuts의 label들을 바깥에 버튼으로 빼서 보여줍니다.
range: true
- 기간을 조회하기 쉬운 UI를 띄워주고
from
과to
값을 입력 받습니다. - key의 값(e.g. dateFromTo)에 숫자를 붙여 array로 보내줍니다.
- 예시:
dateFromTo1: "2023-03-10", dateFromTo2: "2023-03-20"
- SQL 쿼리에서
:dateFrom1
,:dateFrom2
와 같이 이용해보세요.
- 예시:
- shortcuts 사용 시,
from
과to
로 UI에 값을 입력합니다.
sql: >
SELECT *
FROM orders
WHERE created_at >= :dateFromTo1
AND created_at < :dateFromTo2
params:
- key: dateFromTo
label: 조회기간
format: date
showButtons: true
range: true
shortcuts:
- label: 최근 일주일
from:
offset: -7
period: day
to:
offset: 0
period: day
- label: 이번달
from:
offset: 0
startOf: month
to:
offset: 0
endOf: month
relative: true
- shortcuts 이용시 상대적(relative)으로 값을 조절할 수 있습니다.
- 여러번 눌러 반복적으로 적용할 수 있습니다.
예제1: format date 이용시
shortcuts:
- label: 어제 D-1
offset: -1
period: day
relative: true
- label: 내일 D+1
offset: 1
period: day
relative: true
- label: 지난주
offset: -1
period: week
- label: 지난달
offset: -1
period: month
- label: 월초
startOf: month
- label: 월말
endOf: month
예제2: date range 이용시
shortcuts:
- label: 지난주
relative: true
from:
offset: -7
period: day
to:
offset: -7
period: day
- label: 다음주
relative: true
from:
offset: 7
startOf: day
to:
offset: 7
endOf: day
- label: 초기화
reset: true
reset: true
, clear: true
기본값으로 초기화하고 싶을 때는 reset
, 기본값도 지우고 싶을 때는 clear
를 사용해요.
shortcuts:
- label: 초기화
reset: true
# clear: true
disabledDate
, disabledTime
조건에 따라 선택가능한 날짜를 제한합니다. 예시는 오늘 이후의 날짜만 선택하면서, 30분 단위로 시간을 입력 받습니다.
params:
- key: date
format: datetime
disabledDate: |
/**
* 날짜를 비활성화하는 함수
* @param {Date} date - 사용자에게 표시된 날짜 객체
* @param {Date} currentValue - 선택된 날짜 (범위선택시 사용)
* @returns {boolean} - true면 해당 날짜는 비활성화됨
*/
// 현재 날짜(new Date) 이전의 날짜를 비활성화
return date < new Date
minuteStep: 30
showSecond: false
format.time
format: time
을 추가하여 시간값을 선택 입력할 수 있습니다.
timeOptions
start
: 시간 선택 범위의 시작점end
: 시간 선택 범위의 끝점step
: 선택 가능한 시간들의 단위 (e.g. 00:10 → 10분 단위)format
: 시간 선택 시, 표시 방법- HH:mm는 24시간제 표기이고 HH:mm A는 AM/PM 표기입니다.
params:
- key: checkin_time
format: time
timeOptions:
start: 08:00
step: 00:10
end: 22:00
format: HH:mm
# format: HH:mm A
format.color
색상(color)을 선택하고 Hex Code로 입력할 수 있습니다.
pages:
- path: color
blocks:
- type: query
name: 색상 추가
resource: mysql.qa
sqlType: insert
sql: INSERT INTO color (hex) VALUES (:hex)
params:
- key: hex
format: color
reloadAfterSubmit: true
format: range
슬라이더(slider) UI로 범위 사이 값을 입력할 수 있습니다.
params:
- key: range
format: range
label: 범위값
min: -5
max: 5
format: s3
s3를 사용하시는 경우 format: s3
로 쉽고 안정적으로 파일을 업로드해보세요.
- 파일 여러개를 그때 그때 업로드할 수 있고, 삭제 가능합니다.
- 해당 key에 Array(List)로 들어갑니다.
- 해당 key를 하나씩 forEach로 처리 가능 (String으로 하나씩 들어갑니다.)
- 이미지는 1일 내 자동으로 삭제됩니다. (경로는 무작위 생성, 이름은 원본 그대로 유지)
menus:
- path: file-cloud-s3
name: file-cloud-s3
pages:
- path: file-cloud-s3
blocks:
- type: http
axios:
method: POST
url: https://api.selectfromuser.com/sample-api/upload-each-s3
params:
'imageCode': "{{imageCode}}"
'classId': "ATES"
data:
file_urls: "{{ file_urls }}"
params:
- key: imageCode
radio:
MAIN: 메인 사진
SUB: 서브 사진도
- key: file_urls
format: s3
multiple: true
forEach: true
format: storage
별도의 저장소 설정(S3, API)없이도 파일 업로드를 지원합니다.
- format: storage를 적용할 params.key는 중복되지 않는 이름으로 자유롭게 설정해주세요.
path
: 파일 저장위치를 지정합니다.
현재 지원 스펙
- 이미지 업로드 지원 (추후 다양한 파일 지원예정)
- 어드민 사용자 대상 프라이빗 URL만 지원
- (외부공유는 차단) 추후 public 용도(공유용, 공지, 게시글)로 활용할수있게 오픈예정
세부옵션
multiple
: 여러개 업로드max
: 최대 개수 제한accept
: 파일 타입 제한
- type: query
resource: mysql
sqlType: update
sql: UPDATE wines SET name = :name WHERE id = :id
params:
- key: id
valueFromRow: true
- key: name
label: 상품명
- key: data
label: 상품 이미지
format: storage
path: /wine/{{id}}/images
multiple: true
# max: 2
accept: image/*
# accept: .jpg, .png
format: address
주소와 우편번호를 입력해야할 때, 별도의 설치나 개발없이 셀렉트에 내장된 Daum Postcode Service로 주소를 검색하고 우편번호까지 한번에 입력할 수 있어요.
format: address
파라미터 영역에 주소 검색 버튼이 생성됩니다.- 셀렉트에서 roadAddress(도로명주소)와 zonecode(우편번호)를 내려줍니다.
- 셀렉트에서 x축 좌표(경도, longitude), y축 좌표(위도, latitude)값을 내려줍니다.
params:
- key: address
label: 주소
format: address
updateParams:
address: roadAddress # address 파라미터에 검색 결과인 roadAddress 값을 넣습니다.
postcode: zonecode # postcode 파라미터에 zonecode 값을 넣습니다.
longitude: x # longitude 파라미터에 x축 값을 넣습니다.
latitude: y # latitude 파라미터에 y축 값을 넣습니다.
- key: postcode
label: 우편번호
- key: longitude
label: 경도
- key: latitude
label: 위도
format: tiptap
상용 에디터 tiptap 옵션을 추가하였습니다.
- 외부이미지, 상품상세를 디비에 바로 넣을 수 있어요.
- type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT wine_notes SET created_at = NOW(), memo = :memo WHERE wine_id = :wine_id
params:
- key: wine_id
- key: memo
format: tiptap
width: 800px
format: table
format: table
을 통해 JSON 형태의 데이터를 테이블 UI로 표시 및 입력하는 UI 화면을 구성할 수 있어요.
예제 가이드
headers
: 테이블 열의 입력값 형식, 유효성, 표시방식을 설정requestFn
: 입력 데이터를 서버에 보낼 JSON 형식으로 가공- style
width
: 700px: 테이블 너비 설정 showHeaders
: 테이블 헤더 표시여부class: table text-xs
: 테이블의 클래스 지정(작은 글자 크기)- headers
label, format
: 헤더 컬럼 라벨, 포맷 - headers
prefix, postfix
: 입력 필드 전후 텍스트 - headers
postfixStyle
: postfix 스타일
menus:
- path: pages/sOCGir
name: format table
pages:
- path: pages/sOCGir
title: id=65 wine_stock의 property_json.shippingRules[] 를 수정합니다.
blocks:
- type: query
resource: mysql.qa
sqlType: update
sql: |
UPDATE wine_stock
SET property_data = :shippingRules
where id = 65
params:
- key: shippingRules
label: 배송비 규정
format: table
style:
width: 700px
showHeaders: false
class: table text-xs
headers:
minAmount:
label: 배송비시작
format: number
required: true
postfix: 원 이상
postfixStyle:
width: 50px
maxAmount:
prefix: "~"
format: number
postfix: 원 미만
postfixStyle:
width: 50px
placeholder: 비어있으면 최대금액
shippingCost:
format: number
postfix: 원
value: # 기본값이 필요한 경우에 입력 (예제 데이터)
- minAmount:
value: 0
maxAmount:
value: 50000
shippingCost:
value: 3000
- minAmount:
value: 5000
maxAmount:
value: ""
placeholder: (최대)
shippingCost:
value: 0
reloadAfterSubmit: true
requestFn: |
const shippingRules = params.find(e => e.key == 'shippingRules')
shippingRules.value = JSON.stringify({
shippingRules: shippingRules.value.map(row => {
return {
minAmount: row.minAmount.value,
maxAmount: row.maxAmount.value,
shippingCost: row.shippingCost.value,
}
})
})
format: listbox
format: listbox
는 선택 옵션을 제공하는 UI 컴포넌트입니다. 드롭다운과 유사하지만 여러 항목을 한번에 볼 수 있고, 다중 선택이 가능한 특징이 있습니다.
listStyle
: 리스트박스의 UI 크기와 스크롤을 정의하는 스타일 속성입니다.datalistFromQuery
: 리스트박스에 표시될 데이터를 DB에서 조회하는 쿼리를 정의합니다.template
: 리스트박스의 각 항목이 어떻게 표시될지 정의하는 HTML 템플릿입니다. 쿼리에서 조회한 필드들을{{필드명}}
형태로 사용할 수 있습니다.
params:
- key: display_json
format: listbox
multiple: true
listStyle:
minWidth: 300px
height: 300px
overflow: scroll
datalistFromQuery:
type: query
resource: mysql.qa
sql: SELECT DISTINCT name AS value, name AS label, price FROM wine_stock
template: |
{{value}}
<span class="text-xs font-bold bg-slate-400 text-white px-1 rounded">{{price}}원</span>
showOptions
: 선택가능한 옵션을 감추고, 정렬 기능만 이용합니다.buttons
: 추가, 삭제등 버튼을 추가합니다.
params:
- key: display_json
format: listbox
multiple: true
listStyle:
minWidth: 300px
height: 300px
overflow: scroll
showOptions: false
buttons:
- label: 상품추가
openModal:
width: 600px
height: 500px
blocks:
- type: query
resource: mysql.qa
title: 전시상품검색
id: query1
sqlType: select
sql: |
SELECT name AS value, name AS label, vintage, price FROM wine_stock
selectOptions:
enabled: true
showDownload: false
params:
- key: t1
footers:
- type: cancel
label: 취소
placement: center
- type: ok
label: 선택
button:
type: primary
placement: center
- label: 상품제거
deleteItem: true
template: |
{{value}}
<span class="text-xs font-bold bg-slate-400 text-white px-1 rounded">{{price}}원</span>
params.formatString
입력 포맷을 특정 방식으로 설정할 수 있습니다.
params:
- key: amount_decimal
formatString: 0.00 %
format: text
required: true
params.radio
라디오버튼으로 값을 선택하여 입력합니다.
params:
- key: status
radio:
- draft
- published
값 value와 표시 label을 분리해서 이용할 수도 있습니다.
params:
- key: status
defaultValue: draft
radio:
- value: draft
label: 초안
- value: published
label: 배포 완료
params:
- key: status
defaultValue: draft
radio:
- draft: 초안
- published: 배포 완료
radioButtonGroup
라디오 버튼을 시각적으로 더 큰 버튼 묶음으로 만들어줍니다.
radioButtonGroup: true
radio:
- all
- lead
- customer
params.checkbox
체크박스로 값을 선택하여 입력합니다.
params:
- key: is_guest
defaultValue: Y
checkbox:
true: Y
false: N
params:
- key: is_guest
checkbox:
true: ACTIVE
false: ''
params.dropdown
드롭다운으로 값을 선택할 수 있습니다.
params:
- key: status
dropdown:
- draft
- published
드롭다운 선택 시, 실제 반영되는 값과 UI에 보여지는 내용을 다르게 할 수도 있습니다.
params:
- key: status
dropdown:
- value: draft
label: 초안
- value: published
label: 배포완료
params:
- key: status
dropdown:
- draft: 초안
- published: 배포완료
dropdownSize
dropdown 선택지에 노출되는 개수(size)를 조절합니다.
params:
- key: name
- key: status
dropdown:
- pinned: 고정
- event: 이벤트
- ad: 광고
dropdownSize: 3 # 보여지는 선택 내역 개수
dropdownMultiple
dropdown 선택지에서 여러건을 선택할 수 있습니다.
params:
- key: name
- key: status
dropdown:
- pinned: 고정
- event: 이벤트
- ad: 광고
dropdownSize: 3
dropdownMultiple: true
dropdown selectOptions
Dropdown 기능의 자동완성, 복수 선택, 키보드 사용을 지원하는 등 개선된 경험을 제공합니다. (datalist
도 같은 방식으로 지원합니다.)
params:
- key: item
label: 신규 카테고리 추가
selectOptions:
enabled: true
dropdown:
- 호텔
- 리조트
- 캠핑
- 독채
- 수영장
params.selectOptions
multiple
: 여러개 선택taggable
: 태그 형태로 관리pushTags
: 입력한 값을 옵션에 계속 남겨둘지 여부
selectOptions:
enabled: true
multiple: true
taggable: true
pushTags: true
params.datalist
값 선택 시, 보여지는 내용과 실제 저장되는 데이터를 구분해서 처리할 수 있습니다.
params:
- key: code
datalist:
- value: A000
label: 분류1
- value: A002
label: 분류2
value
: 데이터에 저장되는 값label
: 선택 시 보여지는 내용
params.datalist.datalistFromQuery
쿼리로 조회한 datalist에서 선택하여 params에 입력합니다.
params:
- key: code
datalist: true
datalistFromQuery:
type: query
resource: mysql.qa
sqlType: select
sql: >
select id as 'value', code_name as 'label' from codes
params.datalist.datalistPreview
datalist에서 선택한 값의 label을 미리보기로 표기해줍니다.
params:
- key: code
datalist: true
datalistPreview: true
datalistFromQuery: ...
params.datalist.datalistLength
datalist에서 선택할 수 있는 값의 최대 개수를 지정합니다.
params:
- key: code
datalist: true
datalistLength: 3
datalistPreview: true
datalistFromQuery: ...
params.options
여러 단계로 값을 입력하는등 복잡한 데이터 처리가 필요할 때
- requestSubmitFn과 함께 사용해보세요.
- prefix, postfix: 입력한 값에 붙는 접두어, 접미어
- display 추가 옵션: document, table
- display가 없으면 기본값(plain) 적용
예제id: form
으로 입력받아서 JS코드로 처리하고 id: query_add_wine
의 쿼리를 실행
- type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO wine_stock
SET name = :name,
vintage = :vintage,
price = :price
params:
- key: name
- key: vintage
- key: price
id: query_add_wine
hidden: true
toast: 추가했습니다.
- type: query
resource: mysql.qa
sqlType: insert
sql: >
SELECT 1
requestSubmitFn: |
for (const i in form.params.vintage.value) {
const vintage_value = form.params.vintage.value[i]
query_add_wine.params.name.value = form.params.name.value
query_add_wine.params.vintage.value = vintage_value
query_add_wine.params.price.value = form.params.vintage.options.price.value[i] || null
console.log(form.params.vintage)
await query_add_wine.trigger()
}
id: form
params:
- key: vintage
datalist: []
selectOptions:
enabled: true
multiple: true
taggable: true
group: vintage
# display: document
options:
price:
label: 가격
placeholder: 00,000
prefix: 정가
postfix: 원
class: text-right
params.searchOptions
- params datalist 대신에 쓸 수 있습니다. 풍부한 검색 경험을 제공하고 싶은 경우 이용해보세요.
- 검색창이 뜨고, 전체 내역에서 선택한 내용은 구분됩니다.
- 1건 선택지의 경우 엔터시 바로 선택됩니다.
- 검색창 안에(searchOptions) 조회되는 컬럼은 모두 표시합니다.
params:
- key: vintage
showSearchButton: true
# searchButtonLabel: 선택
searchOptions:
enabled: true
# allowEmpty: false # 내역을 선택해야 적용 가능
# allowEdit: true # 선택지에 없는 텍스트 입력 가능
resource: mysql.qa
type: query
sql: |
SELECT vintage AS value
FROM wine_stock
WHERE vintage > 2000 AND vintage LIKE CONCAT(:value, '%')
GROUP BY vintage
- value를 label 처리하고 싶을때는 아래와 같이 적용하시면 됩니다.
params:
- key: vintage
...
searchOptions:
enabled: true
resource: mysql.qa
type: query
sql: |
SELECT vintage AS value, vintage AS label
...
columns:
vintage:
hidden: true
params.filterOptions
- 디비에서 가져오지만 거의 변하지않는 코드/필터검색에 가까운경우 적합합니다. (완전검색 X)
- Array param의 경우 쿼리편의를 위해 mapFirstValue 추가 (첫번째 값을 항상 빼줍니다. name → name1)
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM properties
WHERE (!LENGTH(:property_type1) OR `type` IN ('', :property_type))
LIMIT 30
params:
- key: property_type
label: 숙소타입
mapFirstValue: true
filterOptions:
enabled: true
# 기본 개수 표시 (없으면 '필터'로 표시)
label: 해당 타입
## 값 표시 원하는경우
# style:
# width: 200px
# showValues: true
resource: mysql.qa
type: query
sql: |
SELECT `type` as value, COUNT(id) as count, `memo`
FROM properties
GROUP BY `type`
template: |
<p class="flex">{{value}}
<span class="ml-auto text-base text-sky-700">{{count}}개<span>
</p>
<p class="opacity-50 text-sm">{{memo}}</p>
params.categoryOptions
지역, 카테고리, 코드등을 순차적으로 선택하고 입력해야할때 사용하실 수 있습니다.
예제 가이드
- params와 columns.updateOptions에서 사용할 수 있어요.
- template: 카테고리 → 문자열로 변경 (지정 안하면 object로 기입. responseFn 쓰고싶은 경우)
- valueFn: 문자열 → 카테고리로 변경 (디비에 저장된 값을 불러와 입력폼에 반영. 001002003 같은 코드는 자동으로 식별불가능)
- 조회할때 보통 카테고리정보도 같이 불러와 넘겨줄 수 있습니다.
- row
object
: 보고있는 내용 전체 JSON - value
string
: 전달받은 값 - list
array
: 불러온 모든 카테고리 목록
blocks:
- type: query
resource: mysql.qa
sqlType: update
sql: >
UPDATE wine_stock
SET name = :category
WHERE id = 23
params:
- key: name
group: 1
- key: category
label: 상품카테고리
format: category
group: 2
style:
width: 800px
columns:
- label: 대분류
key: code1
style:
height: 200px
- label: 중분류
key: code2
style:
height: 200px
# maxWidth: 300px
- label: 소분류
key: code3
style:
height: 200px
# width: 500px
categoryOptions:
type: query
resource: mysql.qa
sql: |
SELECT * FROM category
template: "{{basecode1}}{{basecode2}}{{basecode3}}"
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT * FROM wine_stock
WHERE id = 23
display: form
columns:
name:
label: 상품카테고리
format: category
updateOptions:
type: query
resource: mysql.qa
sql: |
UPDATE wine_stock
SET name = :value
WHERE id = 23
columns:
- label: 대분류
key: code1
style:
height: 200px
- label: 중분류
key: code2
style:
height: 200px
- label: 소분류
key: code3
style:
height: 200px
categoryOptions:
type: query
resource: mysql.qa
sql: |
SELECT * FROM category
template: "{{basecode1}}{{basecode2}}{{basecode3}}"
valueFn: |
const code1 = value.slice(0, 3)
const code2 = value.slice(3, 6)
const code3 = value.slice(6, 9)
return list.find(e => {
return e.basecode1 == code1 && e.basecode2 == code2 && e.basecode3 == code3
})
params.progressStep
데이터 입력시 스텝을 나눠 각 단계에 집중할 수 있게 돕습니다.
actions:
- label: 등록
single: true
type: query
resource: mysql.qa
sqlType: insert
sql: >
INSERT INTO leads
SET name = :name,
birth = :birth,
region = :region,
job = :job,
created_at = NOW()
modal: true
height: 500px
params:
- key: name
label: 이름
progressStep: 기본정보
- key: birth
label: 생년월일
progressStep: 상세정보
format: date
- key: region
label: 지역
progressStep: 상세정보
- key: job
label: 직업
progressStep: 상세정보
params.disabled
입력 필드를 비활성화 시킵니다. 마우스 커서도 올릴 수 없게 됩니다.
params:
- key: status
disabled: true
defaultValue: draft
params.readonly
일반 텍스트 필드를 '읽기 전용' 상태로 설정할 수 있습니다. 마우스 커서로 내용을 복사할 수 있습니다. dropdown 이나 radio 등 선택이나 옵션 계열의 필드는 적용되지 않습니다.
params:
- key: code_number
readonly: true
params.hidden
입력 필드를 숨길 수 있습니다.
params:
- key: hidden_field
hidden: true
params.hiddenIfNotOption | disableIfNotOption
선택지(option)가 없는 경우 해당 param을 숨기거나 비활성화할 수 있습니다. 부모 카테고리/코드 기반으로 데이터를 조회할때 적용할 수 있어요.
blocks:
- type: query
resource: mysql
sqlType: insert
sql: >
INSERT INTO products (name, code_group1, code_group2)
values (:name, :code1, :code2)
reloadAfterSubmit: true
class: p-2
params:
- key: name
- key: code1
required: true
datalistDropdown: true
dropdownSize: 10
datalistFromQuery:
type: query
resource: mysql
sql: >
SELECT DISTINCT level1 AS value
FROM category
ORDER BY level1 ASC
- key: code2
datalistDropdown: true
dropdownSize: 10
hiddenIfNotOption: true
# disableIfNotOption: true
datalistFromQuery:
type: query
resource: mysql
sql: >
SELECT DISTINCT level2 AS value
FROM category
WHERE level1 = :code1
ORDER BY level2 ASC
params.placeholder
입력 필드 안에 placeholder 를 입력해 필드에 대한 가이드를 줄 수 있습니다.
params:
- key: type1
label: 업종
placeholder: 키워드 입력
params.raw
파라미터에 원본 SQL 값을 넣고 싶을 때는 raw를 사용해주세요. (e.g. NULL
, NOW()
)
pages:
- path: select/users2
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT *
FROM test_list
WHERE name IS :name
LIMIT 100
params:
- key: name
label: 이름
raw:
Y: NOT NULL
N: NULL
radio:
- value: Y
label: 있음
- value: N
label: 없음
defaultValue: Y
params.query
{{query}}
와 params를 활용해서 sql 쿼리 WHERE절에 일부를 옵션으로 넣을 수 있습니다.
pages:
- path: properties
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT *
FROM properties
WHERE name LIKE CONCAT('%', :name, '%')
AND {{query}}
LIMIT 1000
params:
- key: name
label: 숙소명
- key: types
label: 숙소타입
radio:
- value: ''
label: 전체
- value: HOTEL
label: 호텔
- value: NON HOTEL
label: 호텔 외
query:
'': >
1=1
'HOTEL': >
`type` = 'HOTEL'
'NON HOTEL': >
`type` IN ('GUESTHOUSE', 'BNB') OR `type` IS NULL
params.orderBy
테이블 정렬을 서버에서 할 수 있어요. 여러개의 정렬 옵션을 추가할 수 있습니다.
- SQL 예시: ORDER BY created_at ASC
- params에서 선택한 값에 따라 orderBy 키와 매칭합니다.
- 매칭된 orderBy의 값을
{{ orderBy }}
에 넣습니다.
- type: query
resource: mysql
name: 회원목록
sqlType: select
sql: >
SELECT *
FROM user_profiles
{{ orderBy }}
LIMIT 100
autoload: false
params:
- key: order
label: 정렬
defaultValue: 최근가입순
dropdown:
- 최근가입순
- 오래된가입순
orderBy:
'최근가입순': ORDER BY created_at DESC
'오래된가입순': ORDER BY created_at ASC
pages.params
페이지 단위로 파라미터(parameter)를 지정하여, 여러 블록의 데이터를 필터 조회할 수 있습니다.
pages:
- path: new-charts
title: 마케팅 대시보드
subtitle: 주요 마케팅 지표를 확인할 수 있습니다.
containerClass: container-sm w-50
params:
- key: category
blocks:
- type: query
- type: query
params.validateFn
입력한 값을 사전에 검증하는 기능입니다. validateFn
안에 Javascript를 사용할 수 있어요.
전체 params에서 params.key가 amount
인 내역의 값을 찾고 key: cancel_amount
의 값(param.value)을 비교하는 예제입니다.
params:
- key: amount
valueFromRow: amount
- key: cancel_amount
label: 취소 금액
validateFn: |
const amount = params.find(e => e.key == 'amount')
if (param.value > amount.value) {
return '취소 불가능'
} else {
return ''
}
params.validateFromQuery
쿼리를 validateFn과 함께 사용할 수 있습니다.
- showValidateButton: 버튼 노출시 엔터 또는 클릭해야 검증
- validateButtonLabel: 버튼 라벨
- validText: 서버단 체크했을때 메시지
params:
- key: business_number
label: 사업자번호
required: true
validateFromQuery:
type: query
resource: mysql.qa
sql: >
SELECT COUNT(id) AS count
FROM properties
WHERE business_number = :value
validateFn: |
if (+validateFromQuery.count > 0) {
return '중복된 사업자번호 입니다.'
}
return true
validateFn: |
if (param.value.length != 10) {
return '사업자번호(10자리)를 입력해주세요.'
}
if (!isFinite(+param.value)) {
return '사업자 번호만 입력해주세요.'
}
return true
# showValidateButton: true
# validateButtonLabel: 사업자조회
# validText: 새로운 사업자번호 사용가능
params.minlength, maxlength
입력값의 최소, 최대 글자수를 제한할 수 있습니다.
params:
- key: phone
minlength: 1
maxlength: 12
blocks.viewModal
viewModal: {}
viewModal
불러온 데이터를 클릭하여 모달을 열게됩니다.
1개의 모달을 지원합니다. 여러개가 필요한 경우 modals
를 이용해주세요.
viewModal.name
viewModal.mode
center
, full
, side
viewModal.displayParentRow Deprecated
viewModal:
displayParentRow: true
viewModal.dismissible
모달이 띄워졌을 때 esc로 꺼지지 않게 하고 싶을 때 dismissble을 이용해주세요.
viewModal:
dismissible: false
viewModal.header
모달 header에 있는 전후이동, 링크복사 등 셀렉트 기능을 숨길 수 있습니다.
viewModal:
header: false
viewModal.blocks
여러개 블록을 추가 가능합니다.
viewModal.blocks.display
기본적으로 데이터를 조회하면 테이블(표) 형태로 나타납니다. 이를 특정한 형태로 바꾸고 싶을 때 display를 사용합니다.
viewModal.useColumn
특정 컬럼을 모달 조회 링크로 지정할 수 있습니다.
viewModal:
useColumn: id
blocks:
viewModal.params.valueFromRow
조회한 row 데이터를 모달 안에서 사용할 수 있습니다.
sql: select id, name, email, created_at from user limit 10
viewModal:
blocks:
- type: query
...
sql: select * from order where user_id = :user_id
params:
- key: user_id
valueFromRow: id
viewModal.usePage
pages:
- path: company
blocks:
- type: query
resource: mysql.qa
sql: SELECT * FROM properties LIMIT 8
columns:
id:
name:
type:
address:
viewModal:
usePage: company/detail/{{id}} # sql 쿼리 결과의 id를 가져와서 사용합니다.
- path: company/detail/{{id}} # 받아온 id 값을 아래 sql 쿼리 파라미터에 사용합니다.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, name
FROM properties
WHERE id = :id
viewModal.width, height
모달의 너비(width), 높이(height)를 설정할 수 있습니다. 숫자(px)와 퍼센트(%)를 지원합니다.
viewModal:
width: 1000px
# width: 70%
height: 800px
blocks.modals
modals: []
modals
INFO
여러개 모달을 지원하는 부분 외에는 viewModal
와 동일합니다.
불러온 데이터를 클릭하거나 버튼을 통해 모달을 열게됩니다.
modals와 viewModal 비교
- type: query
resource: mysql
sqlType: select
sql: SELECT id, name, email, created_at FROM users LIMIT 10
columns:
id:
openModal: user
modals:
- path: user
useColumn: id
blocks:
- type: query
- type: query
resource: mysql
sqlType: select
sql: SELECT id, name, email, created_at FROM users LIMIT 10
columns:
viewModal:
useColumn: id
blocks:
- type: query
modals.path
모달의 고유 주소를 지정합니다. 해당 고유주소로 페이지를 띄웁니다.
view-:id
경로에 따라 id
를 파라미터로 가져옵니다.
view-:id-:code
경로에 따라 id
, code
를 파라미터로 가져옵니다.
modals.useColumn
명시된 파라미터가 주소에 남고 모달안에서 쓸 수 있도록 가져옵니다. path
에 파라미터를 넣거나 여기에 추가합니다.
useColumn: id # 1개 가져오기
useColumn: # 2개 가져오기
- id
- code
modals.usePage
pages:
- path: company
blocks:
- type: query
resource: mysql.qa
sql: SELECT * FROM properties LIMIT 8
columns:
id:
name:
openModal: name
type:
address:
modals:
- path: name
usePage: company/detail/{{id}} # sql 쿼리 결과의 id를 가져와서 사용합니다.
- path: company/detail/{{id}} # 받아온 id 값을 아래 sql 쿼리 파라미터에 사용합니다.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, name
FROM properties
WHERE id = :id
modals.roles
계정 또는 권한 그룹(roles)에 따라 모달 안 데이터를 보이지 않게 할 수 있습니다. 서버 레벨에서 막아야하는 경우 blocks.roles를 이용해주세요.
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: >
SELECT id, name FROM wine_stock
columns:
name:
openModal: view
modals:
- path: view
roles:
message: 해당 내역 수정 권한이 없습니다.
view:
- email::test@selectfromuser.com33
blocks:
blocks
actions: []
selectOptions을 통해 선택된 테이블의 row가 필요합니다.
만약 row 선택 없이 단일 액션이 필요한 경우 single
을 이용합니다.
actions
버튼을 통해 특정 액션을 실행할 때 쓰입니다.
blocks:
- type: query
selectOptions:
enabled: true
actions:
- label: 버튼
type: query
actions.type
type: query
actions:
- type: query
label: 전시
resource: acme
sqlType: update
sql: update product set status = 'published' where id = :id
params:
- key: id
type: http
actions:
- type: http
label: 결제취소
axios:
method: POST
url: https://testapi.com/v1/33iac2d/payment/cancel
data:
'payment_id': {{id}}
'v': "{{API_VERSION}}"
'status': 'cancel'
params:
- key: id
- key: API_VERSION
valueFromEnv: true
actions.label
액션 버튼이 어떤 역할을 하는지 구분하기 위해 label을 입력하시는걸 권장합니다.
actions:
- label: 전시
placement: right top
actions.placement
액션 버튼의 위치를 위(상단), 아래(하단), 왼쪽(좌), 오른쪽(우) 변경하고 싶을 때
placement: right top
placement: right bottom
placement: left top
placement: left bottom
actions.button.type
actions:
- label: 제출
placement: bottom right
button:
type: default
- label: 삭제
confirmText: 계속하시겠습니까?
placement: bottom right
button:
type: danger-light
actions.button.icon
버튼 안에 아이콘을 입력할 수 있습니다.
- 지원하는 아이콘: https://pictogrammers.github.io/@mdi/font/7.2.96/
- 필요한 아이콘을 찾기 쉬운 사이트: https://pictogrammers.com/library/mdi/
actions:
- label: 불참
single: true
placement: bottom left
button:
type: danger-light
icon: handball
actions.single
actions 안에서 체크박스 없이, 실행 버튼만을 만들고 싶을 때 single 키를 이용하세요.
actions:
- label: 전체 초기화
placement: right top
single: true
actions.openUrl
액션 버튼을 눌러 특정 URL의 페이지로 이동합니다.
actions:
- label: 페이지 이동
target: _self
openUrl: https://dev.selectfromuser.com
single: true
actions.openModal
액션 버튼을 눌러 모달을 열 수 있습니다.
modal의 path에 parameter(e.g. :id
)를 넣어 부모 데이터나 URL에서 가져와서 사용할 수 있습니다.
actions:
- label: 모달 열기
single: true
openModal: modal1-:id
modals:
- path: modal1-:id
blocks:
- type: query
resource: mysql
actions.openPopper
single: true
면 block.buttons.openPopper와 동일- selectOptions로 선택된 경우
- forEach true면 개별로우(row)마다 표시 (선택된 로우만큼 늘어남)
- single true인데 forEach false이면 아무것도 안뜸
- (기본) tableSelectedRows param으로 넘겨줌
- forEach true면 개별로우(row)마다 표시 (선택된 로우만큼 늘어남)
pages:
- path: sample/actions/openpopper
title: Popper와 Actions 활용 예제
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT * FROM wine_stock WHERE id BETWEEN :id AND :id+2
params:
- key: id
value: 64
selectOptions:
enabled: true
actions:
- label: 빈티지 확인하기
openPopper: true
forEach: true # 주의
popperOptions:
placement: right
popperStyle:
width: 600px
height: 600px
overflow: scroll
padding: 20px
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT id, name, vintage FROM wine_stock WHERE id = :id
params:
- key: id
valueFromRow: true
title: "{{name}}"
columns:
vintage:
copy: true
showDownload: false
- label: 빈티지 확인하기 (tableSelectedRows)
openPopper: true
popperOptions:
placement: right
popperStyle:
width: 600px
height: 600px
overflow: scroll
padding: 20px
blocks:
- type: query
resource: mysql.qa
sqlType: select
sql: SELECT id, name, vintage FROM wine_stock WHERE id IN (0, :...ids)
params:
- key: ids
hidden: true
title: 빈티지 조회
showDownload: false
requestFn: |
const tableSelectedRows = params.find(e => e.key == 'tableSelectedRows')
const ids = tableSelectedRows.value.map(e => e.id)
for (const param of params) {
if (param.key == 'ids') {
param.value = ids
}
}
(popperStyle 팁)
openPopper시 꽉차게 하고 스크롤 만드는 예제
popperOptions:
placement: right
popperStyle:
width: 500px
height: 90vh
maxHeight: 90vh
overflow: scroll
padding: 20px
actions.showDownload
데이터 다운로드 기능을 액션 버튼에 추가할 있어요.
액션 버튼 추가 시, 헷갈리지 않게 기존의 showDownload 옵션은 끄는걸 권장해요. (showDownload: false
)
actions:
- label: csv 다운로드
showDownload: csv
single: true
showDownload
actions.params.valueFromSelectedRows
선택한 row의 값을 가져와서, 파라미터에 이용할 수 있습니다.
selectOptions:
enabled: true
actions:
- label: 버튼
type: query
resource: mysql.qa
sqlType: update
sql: UPDATE products SET sell_status = 'true' WHERE id = :id
params:
- key: id
valueFromSelectedRows: id
selectOptions:
enabled: true
actions:
- label: 버튼
type: http
axios:
method: POST
url: https://httpbin.selectfromuser.com/anything
data:
id: "{{id}}"
params:
- key: id
valueFromSelectedRows: id
actions.confirmText
actions:
- label: 삭제
confirmText: 정말 삭제하시겠습니까? 삭제 후 복구가 어려울 수 있습니다.
actions.params.promptText
prompt를 띄워서 parameter에 값을 입력할 수 있습니다.
params:
- key: memo
valueFromPrompt: true
promptText: 메모를 입력해주세요.
actions.modal
액션 버튼을 눌러 모달을 띄울 수 있습니다.
actions:
- label: 티켓 추가
placement: top right
modal: true
type: query
...
actions.forEach
액션 실행 시, 선택된 row의 값들을 하나씩 연속으로 실행합니다.
actions:
- label: 티켓 추가
placement: top left
modal: true
type: query
resource: acme
sqlType: insert
sql: >
insert into tickets (created_at, written_by, title, description, status, type, type_id)
values ( current_timestamp, :written_by, :title, :description, :status, 'order', :type_id)
forEach: true
params:
- key: written_by
- key: title
help: >
필드에 대한 도움말
- key: description
- key: status
- key: type_id
valueFromSelectedRows: true
valueFromSelectedRowsAs: id