mirror of
https://github.com/lingble/chatwoot.git
synced 2025-10-30 18:47:51 +00:00
Merge branch 'develop' into fix/hc-editor
This commit is contained in:
@@ -1,6 +1,7 @@
|
||||
version: 2.1
|
||||
orbs:
|
||||
node: circleci/node@6.1.0
|
||||
qlty-orb: qltysh/qlty-orb@0.0
|
||||
|
||||
defaults: &defaults
|
||||
working_directory: ~/build
|
||||
@@ -89,14 +90,6 @@ jobs:
|
||||
command: |
|
||||
source ~/.rvm/scripts/rvm
|
||||
bundle install
|
||||
# pnpm install
|
||||
|
||||
- run:
|
||||
name: Download cc-test-reporter
|
||||
command: |
|
||||
mkdir -p ~/tmp
|
||||
curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ~/tmp/cc-test-reporter
|
||||
chmod +x ~/tmp/cc-test-reporter
|
||||
|
||||
# Swagger verification
|
||||
- run:
|
||||
@@ -108,10 +101,11 @@ jobs:
|
||||
echo "ERROR: The swagger.json file is not in sync with the yaml specification. Run 'rake swagger:build' and commit 'swagger/swagger.json'."
|
||||
exit 1
|
||||
fi
|
||||
mkdir -p ~/tmp
|
||||
curl -L https://repo1.maven.org/maven2/org/openapitools/openapi-generator-cli/6.3.0/openapi-generator-cli-6.3.0.jar > ~/tmp/openapi-generator-cli-6.3.0.jar
|
||||
java -jar ~/tmp/openapi-generator-cli-6.3.0.jar validate -i swagger/swagger.json
|
||||
|
||||
# we remove the FRONTED_URL from the .env before running the tests
|
||||
# Configure environment and database
|
||||
- run:
|
||||
name: Database Setup and Configure Environment Variables
|
||||
command: |
|
||||
@@ -149,17 +143,11 @@ jobs:
|
||||
command: pnpm run eslint
|
||||
|
||||
- run:
|
||||
name: Run frontend tests
|
||||
name: Run frontend tests (with coverage)
|
||||
command: |
|
||||
mkdir -p ~/build/coverage/frontend
|
||||
~/tmp/cc-test-reporter before-build
|
||||
pnpm run test:coverage
|
||||
|
||||
- run:
|
||||
name: Code Climate Test Coverage (Frontend)
|
||||
command: |
|
||||
~/tmp/cc-test-reporter format-coverage -t lcov -o "~/build/coverage/frontend/codeclimate.frontend_$CIRCLE_NODE_INDEX.json"
|
||||
|
||||
# Run backend tests
|
||||
- run:
|
||||
name: Run backend tests
|
||||
@@ -167,18 +155,18 @@ jobs:
|
||||
mkdir -p ~/tmp/test-results/rspec
|
||||
mkdir -p ~/tmp/test-artifacts
|
||||
mkdir -p ~/build/coverage/backend
|
||||
~/tmp/cc-test-reporter before-build
|
||||
TESTFILES=$(circleci tests glob "spec/**/*_spec.rb" | circleci tests split --split-by=timings)
|
||||
bundle exec rspec --format progress \
|
||||
bundle exec rspec -I ./spec --require coverage_helper --require spec_helper --format progress \
|
||||
--format RspecJunitFormatter \
|
||||
--out ~/tmp/test-results/rspec.xml \
|
||||
-- ${TESTFILES}
|
||||
no_output_timeout: 30m
|
||||
|
||||
- run:
|
||||
name: Code Climate Test Coverage (Backend)
|
||||
command: |
|
||||
~/tmp/cc-test-reporter format-coverage -t simplecov -o "~/build/coverage/backend/codeclimate.$CIRCLE_NODE_INDEX.json"
|
||||
# Qlty coverage publish
|
||||
- qlty-orb/coverage_publish:
|
||||
files: |
|
||||
coverage/coverage.json
|
||||
coverage/lcov.info
|
||||
|
||||
- run:
|
||||
name: List coverage directory contents
|
||||
@@ -189,3 +177,7 @@ jobs:
|
||||
root: ~/build
|
||||
paths:
|
||||
- coverage
|
||||
|
||||
- store_artifacts:
|
||||
path: coverage
|
||||
destination: coverage
|
||||
|
||||
@@ -6,6 +6,13 @@
|
||||
# Use `rake secret` to generate this variable
|
||||
SECRET_KEY_BASE=replace_with_lengthy_secure_hex
|
||||
|
||||
# Active Record Encryption keys (required for MFA/2FA functionality)
|
||||
# Generate these keys by running: rails db:encryption:init
|
||||
# IMPORTANT: Use different keys for each environment (development, staging, production)
|
||||
# ACTIVE_RECORD_ENCRYPTION_PRIMARY_KEY=
|
||||
# ACTIVE_RECORD_ENCRYPTION_DETERMINISTIC_KEY=
|
||||
# ACTIVE_RECORD_ENCRYPTION_KEY_DERIVATION_SALT=
|
||||
|
||||
# Replace with the URL you are planning to use for your app
|
||||
FRONTEND_URL=http://0.0.0.0:3000
|
||||
# To use a dedicated URL for help center pages
|
||||
|
||||
99
.github/workflows/run_mfa_spec.yml
vendored
Normal file
99
.github/workflows/run_mfa_spec.yml
vendored
Normal file
@@ -0,0 +1,99 @@
|
||||
name: Run MFA Tests
|
||||
permissions:
|
||||
contents: read
|
||||
|
||||
on:
|
||||
pull_request:
|
||||
|
||||
# If two pushes happen within a short time in the same PR, cancel the run of the oldest push
|
||||
concurrency:
|
||||
group: pr-${{ github.workflow }}-${{ github.head_ref }}
|
||||
cancel-in-progress: true
|
||||
|
||||
jobs:
|
||||
test:
|
||||
runs-on: ubuntu-22.04
|
||||
# Only run if MFA test keys are available
|
||||
if: github.event_name == 'workflow_dispatch' || (github.repository == 'chatwoot/chatwoot' && github.actor != 'dependabot[bot]')
|
||||
|
||||
services:
|
||||
postgres:
|
||||
image: pgvector/pgvector:pg15
|
||||
env:
|
||||
POSTGRES_USER: postgres
|
||||
POSTGRES_PASSWORD: ''
|
||||
POSTGRES_DB: postgres
|
||||
POSTGRES_HOST_AUTH_METHOD: trust
|
||||
ports:
|
||||
- 5432:5432
|
||||
options: >-
|
||||
--mount type=tmpfs,destination=/var/lib/postgresql/data
|
||||
--health-cmd pg_isready
|
||||
--health-interval 10s
|
||||
--health-timeout 5s
|
||||
--health-retries 5
|
||||
redis:
|
||||
image: redis
|
||||
ports:
|
||||
- 6379:6379
|
||||
options: --entrypoint redis-server
|
||||
|
||||
env:
|
||||
RAILS_ENV: test
|
||||
POSTGRES_HOST: localhost
|
||||
# Active Record encryption keys required for MFA - test keys only, not for production use
|
||||
ACTIVE_RECORD_ENCRYPTION_PRIMARY_KEY: 'test_key_a6cde8f7b9c2d3e4f5a6b7c8d9e0f1a2b3c4d5e6f7a8b9c0d1e2f3a4b5c6d7'
|
||||
ACTIVE_RECORD_ENCRYPTION_DETERMINISTIC_KEY: 'test_key_b7def9a8c0d3e4f5a6b7c8d9e0f1a2b3c4d5e6f7a8b9c0d1e2f3a4b5c6d8'
|
||||
ACTIVE_RECORD_ENCRYPTION_KEY_DERIVATION_SALT: 'test_salt_c8efa0b9d1e4f5a6b7c8d9e0f1a2b3c4d5e6f7a8b9c0d1e2f3a4b5c6d9'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- uses: ruby/setup-ruby@v1
|
||||
with:
|
||||
bundler-cache: true
|
||||
|
||||
- name: Create database
|
||||
run: bundle exec rake db:create
|
||||
|
||||
- name: Install pgvector extension
|
||||
run: |
|
||||
PGPASSWORD="" psql -h localhost -U postgres -d chatwoot_test -c "CREATE EXTENSION IF NOT EXISTS vector;"
|
||||
|
||||
- name: Seed database
|
||||
run: bundle exec rake db:schema:load
|
||||
|
||||
- name: Run MFA-related backend tests
|
||||
run: |
|
||||
bundle exec rspec \
|
||||
spec/services/mfa/token_service_spec.rb \
|
||||
spec/services/mfa/authentication_service_spec.rb \
|
||||
spec/requests/api/v1/profile/mfa_controller_spec.rb \
|
||||
spec/controllers/devise_overrides/sessions_controller_spec.rb \
|
||||
--profile=10 \
|
||||
--format documentation
|
||||
env:
|
||||
NODE_OPTIONS: --openssl-legacy-provider
|
||||
|
||||
- name: Run MFA-related tests in user_spec
|
||||
run: |
|
||||
# Run specific MFA-related tests from user_spec
|
||||
bundle exec rspec spec/models/user_spec.rb \
|
||||
-e "two factor" \
|
||||
-e "2FA" \
|
||||
-e "MFA" \
|
||||
-e "otp" \
|
||||
-e "backup code" \
|
||||
--profile=10 \
|
||||
--format documentation
|
||||
env:
|
||||
NODE_OPTIONS: --openssl-legacy-provider
|
||||
|
||||
- name: Upload test logs
|
||||
uses: actions/upload-artifact@v4
|
||||
if: failure()
|
||||
with:
|
||||
name: mfa-test-logs
|
||||
path: |
|
||||
log/test.log
|
||||
tmp/screenshots/
|
||||
4
.gitignore
vendored
4
.gitignore
vendored
@@ -95,3 +95,7 @@ yarn-debug.log*
|
||||
.claude/settings.local.json
|
||||
.cursor
|
||||
CLAUDE.local.md
|
||||
|
||||
# Histoire deployment
|
||||
.netlify
|
||||
.histoire
|
||||
|
||||
19
AGENTS.md
19
AGENTS.md
@@ -55,4 +55,21 @@
|
||||
|
||||
## Ruby Best Practices
|
||||
|
||||
- Use compact `module/class` definitions; avoid nested styles
|
||||
- Use compact `module/class` definitions; avoid nested styles
|
||||
|
||||
## Enterprise Edition Notes
|
||||
|
||||
- Chatwoot has an Enterprise overlay under `enterprise/` that extends/overrides OSS code.
|
||||
- When you add or modify core functionality, always check for corresponding files in `enterprise/` and keep behavior compatible.
|
||||
- Follow the Enterprise development practices documented here:
|
||||
- https://chatwoot.help/hc/handbook/articles/developing-enterprise-edition-features-38
|
||||
|
||||
Practical checklist for any change impacting core logic or public APIs
|
||||
- Search for related files in both trees before editing (e.g., `rg -n "FooService|ControllerName|ModelName" app enterprise`).
|
||||
- If adding new endpoints, services, or models, consider whether Enterprise needs:
|
||||
- An override (e.g., `enterprise/app/...`), or
|
||||
- An extension point (e.g., `prepend_mod_with`, hooks, configuration) to avoid hard forks.
|
||||
- Avoid hardcoding instance- or plan-specific behavior in OSS; prefer configuration, feature flags, or extension points consumed by Enterprise.
|
||||
- Keep request/response contracts stable across OSS and Enterprise; update both sets of routes/controllers when introducing new APIs.
|
||||
- When renaming/moving shared code, mirror the change in `enterprise/` to prevent drift.
|
||||
- Tests: Add Enterprise-specific specs under `spec/enterprise`, mirroring OSS spec layout where applicable.
|
||||
|
||||
17
Gemfile
17
Gemfile
@@ -62,6 +62,10 @@ gem 'redis-namespace'
|
||||
# super fast record imports in bulk
|
||||
gem 'activerecord-import'
|
||||
|
||||
gem 'searchkick'
|
||||
gem 'opensearch-ruby'
|
||||
gem 'faraday_middleware-aws-sigv4'
|
||||
|
||||
##--- gems for server & infra configuration ---##
|
||||
gem 'dotenv-rails', '>= 3.0.0'
|
||||
gem 'foreman'
|
||||
@@ -74,9 +78,12 @@ gem 'barnes'
|
||||
gem 'devise', '>= 4.9.4'
|
||||
gem 'devise-secure_password', git: 'https://github.com/chatwoot/devise-secure_password', branch: 'chatwoot'
|
||||
gem 'devise_token_auth', '>= 1.2.3'
|
||||
# two-factor authentication
|
||||
gem 'devise-two-factor', '>= 5.0.0'
|
||||
# authorization
|
||||
gem 'jwt'
|
||||
gem 'pundit'
|
||||
|
||||
# super admin
|
||||
gem 'administrate', '>= 0.20.1'
|
||||
gem 'administrate-field-active_storage', '>= 1.0.3'
|
||||
@@ -89,7 +96,7 @@ gem 'wisper', '2.0.0'
|
||||
##--- gems for channels ---##
|
||||
gem 'facebook-messenger'
|
||||
gem 'line-bot-api'
|
||||
gem 'twilio-ruby', '~> 5.66'
|
||||
gem 'twilio-ruby'
|
||||
# twitty will handle subscription of twitter account events
|
||||
# gem 'twitty', git: 'https://github.com/chatwoot/twitty'
|
||||
gem 'twitty', '~> 0.1.5'
|
||||
@@ -108,7 +115,7 @@ gem 'google-cloud-translate-v3', '>= 0.7.0'
|
||||
##-- apm and error monitoring ---#
|
||||
# loaded only when environment variables are set.
|
||||
# ref application.rb
|
||||
gem 'ddtrace', require: false
|
||||
gem 'datadog', '~> 2.0', require: false
|
||||
gem 'elastic-apm', require: false
|
||||
gem 'newrelic_rpm', require: false
|
||||
gem 'newrelic-sidekiq-metrics', '>= 1.6.2', require: false
|
||||
@@ -167,6 +174,7 @@ gem 'audited', '~> 5.4', '>= 5.4.1'
|
||||
|
||||
# need for google auth
|
||||
gem 'omniauth', '>= 2.1.2'
|
||||
gem 'omniauth-saml'
|
||||
gem 'omniauth-google-oauth2', '>= 1.1.3'
|
||||
gem 'omniauth-rails_csrf_protection', '~> 1.0', '>= 1.0.2'
|
||||
|
||||
@@ -212,6 +220,8 @@ group :development do
|
||||
gem 'stackprof'
|
||||
# Should install the associated chrome extension to view query logs
|
||||
gem 'meta_request', '>= 0.8.3'
|
||||
|
||||
gem 'tidewave'
|
||||
end
|
||||
|
||||
group :test do
|
||||
@@ -221,6 +231,7 @@ group :test do
|
||||
gem 'webmock'
|
||||
# test profiling
|
||||
gem 'test-prof'
|
||||
gem 'simplecov_json_formatter', require: false
|
||||
end
|
||||
|
||||
group :development, :test do
|
||||
@@ -245,7 +256,7 @@ group :development, :test do
|
||||
gem 'rubocop-factory_bot', require: false
|
||||
gem 'seed_dump'
|
||||
gem 'shoulda-matchers'
|
||||
gem 'simplecov', '0.17.1', require: false
|
||||
gem 'simplecov', '>= 0.21', require: false
|
||||
gem 'spring'
|
||||
gem 'spring-watcher-listen'
|
||||
end
|
||||
|
||||
146
Gemfile.lock
146
Gemfile.lock
@@ -194,10 +194,14 @@ GEM
|
||||
activerecord (>= 5.a)
|
||||
database_cleaner-core (~> 2.0.0)
|
||||
database_cleaner-core (2.0.1)
|
||||
date (3.4.1)
|
||||
ddtrace (0.48.0)
|
||||
ffi (~> 1.0)
|
||||
datadog (2.19.0)
|
||||
datadog-ruby_core_source (~> 3.4, >= 3.4.1)
|
||||
libdatadog (~> 18.1.0.1.0)
|
||||
libddwaf (~> 1.24.1.0.3)
|
||||
logger
|
||||
msgpack
|
||||
datadog-ruby_core_source (3.4.1)
|
||||
date (3.4.1)
|
||||
debug (1.8.0)
|
||||
irb (>= 1.5.0)
|
||||
reline (>= 0.3.1)
|
||||
@@ -208,6 +212,11 @@ GEM
|
||||
railties (>= 4.1.0)
|
||||
responders
|
||||
warden (~> 1.2.3)
|
||||
devise-two-factor (6.1.0)
|
||||
activesupport (>= 7.0, < 8.1)
|
||||
devise (~> 4.0)
|
||||
railties (>= 7.0, < 8.1)
|
||||
rotp (~> 6.0)
|
||||
devise_token_auth (1.2.5)
|
||||
bcrypt (~> 3.0)
|
||||
devise (> 3.5.2, < 5)
|
||||
@@ -215,7 +224,7 @@ GEM
|
||||
diff-lcs (1.5.1)
|
||||
digest-crc (0.6.5)
|
||||
rake (>= 12.0.0, < 14.0.0)
|
||||
docile (1.4.0)
|
||||
docile (1.4.1)
|
||||
domain_name (0.5.20190701)
|
||||
unf (>= 0.0.5, < 1.0.0)
|
||||
dotenv (3.1.2)
|
||||
@@ -226,6 +235,35 @@ GEM
|
||||
addressable (~> 2.8)
|
||||
drb (2.2.3)
|
||||
dry-cli (1.1.0)
|
||||
dry-configurable (1.3.0)
|
||||
dry-core (~> 1.1)
|
||||
zeitwerk (~> 2.6)
|
||||
dry-core (1.1.0)
|
||||
concurrent-ruby (~> 1.0)
|
||||
logger
|
||||
zeitwerk (~> 2.6)
|
||||
dry-inflector (1.2.0)
|
||||
dry-initializer (3.2.0)
|
||||
dry-logic (1.6.0)
|
||||
bigdecimal
|
||||
concurrent-ruby (~> 1.0)
|
||||
dry-core (~> 1.1)
|
||||
zeitwerk (~> 2.6)
|
||||
dry-schema (1.14.1)
|
||||
concurrent-ruby (~> 1.0)
|
||||
dry-configurable (~> 1.0, >= 1.0.1)
|
||||
dry-core (~> 1.1)
|
||||
dry-initializer (~> 3.2)
|
||||
dry-logic (~> 1.5)
|
||||
dry-types (~> 1.8)
|
||||
zeitwerk (~> 2.6)
|
||||
dry-types (1.8.3)
|
||||
bigdecimal (~> 3.0)
|
||||
concurrent-ruby (~> 1.0)
|
||||
dry-core (~> 1.0)
|
||||
dry-inflector (~> 1.0)
|
||||
dry-logic (~> 1.4)
|
||||
zeitwerk (~> 2.6)
|
||||
ecma-re-validator (0.4.0)
|
||||
regexp_parser (~> 2.2)
|
||||
elastic-apm (4.6.2)
|
||||
@@ -248,8 +286,10 @@ GEM
|
||||
railties (>= 5.0.0)
|
||||
faker (3.2.0)
|
||||
i18n (>= 1.8.11, < 2)
|
||||
faraday (2.9.0)
|
||||
faraday-net_http (>= 2.0, < 3.2)
|
||||
faraday (2.13.1)
|
||||
faraday-net_http (>= 2.0, < 3.5)
|
||||
json
|
||||
logger
|
||||
faraday-follow_redirects (0.3.0)
|
||||
faraday (>= 1, < 3)
|
||||
faraday-mashify (0.1.1)
|
||||
@@ -257,13 +297,23 @@ GEM
|
||||
hashie
|
||||
faraday-multipart (1.0.4)
|
||||
multipart-post (~> 2)
|
||||
faraday-net_http (3.1.0)
|
||||
net-http
|
||||
faraday-net_http (3.4.0)
|
||||
net-http (>= 0.5.0)
|
||||
faraday-net_http_persistent (2.1.0)
|
||||
faraday (~> 2.5)
|
||||
net-http-persistent (~> 4.0)
|
||||
faraday-retry (2.2.1)
|
||||
faraday (~> 2.0)
|
||||
faraday_middleware-aws-sigv4 (1.0.1)
|
||||
aws-sigv4 (~> 1.0)
|
||||
faraday (>= 2.0, < 3)
|
||||
fast-mcp (1.5.0)
|
||||
addressable (~> 2.8)
|
||||
base64
|
||||
dry-schema (~> 1.14)
|
||||
json (~> 2.0)
|
||||
mime-types (~> 3.4)
|
||||
rack (~> 3.1)
|
||||
fcm (1.0.8)
|
||||
faraday (>= 1.0.0, < 3.0)
|
||||
googleauth (~> 1)
|
||||
@@ -402,7 +452,7 @@ GEM
|
||||
rails-dom-testing (>= 1, < 3)
|
||||
railties (>= 4.2.0)
|
||||
thor (>= 0.14, < 2.0)
|
||||
json (2.12.0)
|
||||
json (2.13.2)
|
||||
json_refs (0.1.8)
|
||||
hana
|
||||
json_schemer (0.2.24)
|
||||
@@ -417,7 +467,7 @@ GEM
|
||||
judoscale-sidekiq (1.8.2)
|
||||
judoscale-ruby (= 1.8.2)
|
||||
sidekiq (>= 5.0)
|
||||
jwt (2.8.1)
|
||||
jwt (2.10.1)
|
||||
base64
|
||||
kaminari (1.2.2)
|
||||
activesupport (>= 4.1.0)
|
||||
@@ -444,6 +494,16 @@ GEM
|
||||
logger (~> 1.6)
|
||||
letter_opener (1.10.0)
|
||||
launchy (>= 2.2, < 4)
|
||||
libdatadog (18.1.0.1.0)
|
||||
libdatadog (18.1.0.1.0-x86_64-linux)
|
||||
libddwaf (1.24.1.0.3)
|
||||
ffi (~> 1.0)
|
||||
libddwaf (1.24.1.0.3-arm64-darwin)
|
||||
ffi (~> 1.0)
|
||||
libddwaf (1.24.1.0.3-x86_64-darwin)
|
||||
ffi (~> 1.0)
|
||||
libddwaf (1.24.1.0.3-x86_64-linux)
|
||||
ffi (~> 1.0)
|
||||
line-bot-api (1.28.0)
|
||||
lint_roller (1.1.0)
|
||||
liquid (5.4.0)
|
||||
@@ -489,7 +549,7 @@ GEM
|
||||
mutex_m (0.3.0)
|
||||
neighbor (0.2.3)
|
||||
activerecord (>= 5.2)
|
||||
net-http (0.4.1)
|
||||
net-http (0.6.0)
|
||||
uri
|
||||
net-http-persistent (4.0.2)
|
||||
connection_pool (~> 2.2)
|
||||
@@ -534,8 +594,9 @@ GEM
|
||||
oj (3.16.10)
|
||||
bigdecimal (>= 3.0)
|
||||
ostruct (>= 0.2)
|
||||
omniauth (2.1.2)
|
||||
omniauth (2.1.3)
|
||||
hashie (>= 3.4.6)
|
||||
logger
|
||||
rack (>= 2.2.3)
|
||||
rack-protection
|
||||
omniauth-google-oauth2 (1.1.3)
|
||||
@@ -549,6 +610,12 @@ GEM
|
||||
omniauth-rails_csrf_protection (1.0.2)
|
||||
actionpack (>= 4.2)
|
||||
omniauth (~> 2.0)
|
||||
omniauth-saml (2.2.4)
|
||||
omniauth (~> 2.1)
|
||||
ruby-saml (~> 1.18)
|
||||
opensearch-ruby (3.4.0)
|
||||
faraday (>= 1.0, < 3)
|
||||
multi_json (>= 1.0)
|
||||
openssl (3.2.0)
|
||||
orm_adapter (0.5.0)
|
||||
os (1.1.4)
|
||||
@@ -577,7 +644,7 @@ GEM
|
||||
activesupport (>= 3.0.0)
|
||||
raabro (1.4.0)
|
||||
racc (1.8.1)
|
||||
rack (2.2.15)
|
||||
rack (3.2.0)
|
||||
rack-attack (6.7.0)
|
||||
rack (>= 1.0, < 4)
|
||||
rack-contrib (2.5.0)
|
||||
@@ -586,19 +653,20 @@ GEM
|
||||
rack (>= 2.0.0)
|
||||
rack-mini-profiler (3.2.0)
|
||||
rack (>= 1.2.0)
|
||||
rack-protection (3.2.0)
|
||||
rack-protection (4.1.1)
|
||||
base64 (>= 0.1.0)
|
||||
rack (~> 2.2, >= 2.2.4)
|
||||
logger (>= 1.6.0)
|
||||
rack (>= 3.0.0, < 4)
|
||||
rack-proxy (0.7.7)
|
||||
rack
|
||||
rack-session (1.0.2)
|
||||
rack (< 3)
|
||||
rack-session (2.1.1)
|
||||
base64 (>= 0.1.0)
|
||||
rack (>= 3.0.0)
|
||||
rack-test (2.1.0)
|
||||
rack (>= 1.3)
|
||||
rack-timeout (0.6.3)
|
||||
rackup (1.0.1)
|
||||
rack (< 3)
|
||||
webrick
|
||||
rackup (2.2.1)
|
||||
rack (>= 3)
|
||||
rails (7.1.5.2)
|
||||
actioncable (= 7.1.5.2)
|
||||
actionmailbox (= 7.1.5.2)
|
||||
@@ -659,7 +727,8 @@ GEM
|
||||
retriable (3.1.2)
|
||||
reverse_markdown (2.1.1)
|
||||
nokogiri
|
||||
rexml (3.4.1)
|
||||
rexml (3.4.4)
|
||||
rotp (6.3.0)
|
||||
rspec-core (3.13.0)
|
||||
rspec-support (~> 3.13.0)
|
||||
rspec-expectations (3.13.2)
|
||||
@@ -714,6 +783,9 @@ GEM
|
||||
faraday (>= 1)
|
||||
faraday-multipart (>= 1)
|
||||
ruby-progressbar (1.13.0)
|
||||
ruby-saml (1.18.1)
|
||||
nokogiri (>= 1.13.10)
|
||||
rexml
|
||||
ruby-vips (2.1.4)
|
||||
ffi (~> 1.12)
|
||||
ruby2_keywords (0.0.5)
|
||||
@@ -749,6 +821,9 @@ GEM
|
||||
parser
|
||||
scss_lint (0.60.0)
|
||||
sass (~> 3.5, >= 3.5.5)
|
||||
searchkick (5.5.2)
|
||||
activemodel (>= 7.1)
|
||||
hashie
|
||||
securerandom (0.4.1)
|
||||
seed_dump (3.3.1)
|
||||
activerecord (>= 4)
|
||||
@@ -795,11 +870,12 @@ GEM
|
||||
faraday (>= 0.17.5, < 3.a)
|
||||
jwt (>= 1.5, < 3.0)
|
||||
multi_json (~> 1.10)
|
||||
simplecov (0.17.1)
|
||||
simplecov (0.22.0)
|
||||
docile (~> 1.1)
|
||||
json (>= 1.8, < 3)
|
||||
simplecov-html (~> 0.10.0)
|
||||
simplecov-html (0.10.2)
|
||||
simplecov-html (~> 0.11)
|
||||
simplecov_json_formatter (~> 0.1)
|
||||
simplecov-html (0.13.2)
|
||||
simplecov_json_formatter (0.1.4)
|
||||
slack-ruby-client (2.5.2)
|
||||
faraday (>= 2.0)
|
||||
faraday-mashify
|
||||
@@ -829,13 +905,17 @@ GEM
|
||||
telephone_number (1.4.20)
|
||||
test-prof (1.2.1)
|
||||
thor (1.4.0)
|
||||
tidewave (0.2.0)
|
||||
fast-mcp (~> 1.5.0)
|
||||
rack (>= 2.0)
|
||||
rails (>= 7.1.0)
|
||||
tilt (2.3.0)
|
||||
time_diff (0.3.0)
|
||||
activesupport
|
||||
i18n
|
||||
timeout (0.4.3)
|
||||
trailblazer-option (0.1.2)
|
||||
twilio-ruby (5.77.0)
|
||||
twilio-ruby (7.6.0)
|
||||
faraday (>= 0.9, < 3.0)
|
||||
jwt (>= 1.5, < 3.0)
|
||||
nokogiri (>= 1.6, < 2.0)
|
||||
@@ -882,7 +962,6 @@ GEM
|
||||
addressable (>= 2.8.0)
|
||||
crack (>= 0.3.2)
|
||||
hashdiff (>= 0.4.0, < 2.0.0)
|
||||
webrick (1.9.1)
|
||||
websocket-driver (0.7.7)
|
||||
base64
|
||||
websocket-extensions (>= 0.1.0)
|
||||
@@ -928,10 +1007,11 @@ DEPENDENCIES
|
||||
commonmarker
|
||||
csv-safe
|
||||
database_cleaner
|
||||
ddtrace
|
||||
datadog (~> 2.0)
|
||||
debug (~> 1.8)
|
||||
devise (>= 4.9.4)
|
||||
devise-secure_password!
|
||||
devise-two-factor (>= 5.0.0)
|
||||
devise_token_auth (>= 1.2.3)
|
||||
dotenv-rails (>= 3.0.0)
|
||||
down
|
||||
@@ -940,6 +1020,7 @@ DEPENDENCIES
|
||||
facebook-messenger
|
||||
factory_bot_rails (>= 6.4.3)
|
||||
faker
|
||||
faraday_middleware-aws-sigv4
|
||||
fcm
|
||||
flag_shih_tzu
|
||||
foreman
|
||||
@@ -980,6 +1061,8 @@ DEPENDENCIES
|
||||
omniauth-google-oauth2 (>= 1.1.3)
|
||||
omniauth-oauth2
|
||||
omniauth-rails_csrf_protection (~> 1.0, >= 1.0.2)
|
||||
omniauth-saml
|
||||
opensearch-ruby
|
||||
pg
|
||||
pg_search
|
||||
pgvector
|
||||
@@ -1008,6 +1091,7 @@ DEPENDENCIES
|
||||
ruby_llm-schema
|
||||
scout_apm
|
||||
scss_lint
|
||||
searchkick
|
||||
seed_dump
|
||||
sentry-rails (>= 5.19.0)
|
||||
sentry-ruby
|
||||
@@ -1017,7 +1101,8 @@ DEPENDENCIES
|
||||
sidekiq (>= 7.3.1)
|
||||
sidekiq-cron (>= 1.12.0)
|
||||
sidekiq_alive
|
||||
simplecov (= 0.17.1)
|
||||
simplecov (>= 0.21)
|
||||
simplecov_json_formatter
|
||||
slack-ruby-client (~> 2.5.2)
|
||||
spring
|
||||
spring-watcher-listen
|
||||
@@ -1026,8 +1111,9 @@ DEPENDENCIES
|
||||
stripe
|
||||
telephone_number
|
||||
test-prof
|
||||
tidewave
|
||||
time_diff
|
||||
twilio-ruby (~> 5.66)
|
||||
twilio-ruby
|
||||
twitty (~> 0.1.5)
|
||||
tzinfo-data
|
||||
uglifier
|
||||
|
||||
3
Rakefile
3
Rakefile
@@ -2,5 +2,8 @@
|
||||
# for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
|
||||
|
||||
require_relative 'config/application'
|
||||
# Load Enterprise Edition rake tasks if they exist
|
||||
enterprise_tasks_path = Rails.root.join('enterprise/tasks_railtie.rb').to_s
|
||||
require enterprise_tasks_path if File.exist?(enterprise_tasks_path)
|
||||
|
||||
Rails.application.load_tasks
|
||||
|
||||
@@ -1 +1 @@
|
||||
3.4.2
|
||||
3.4.3
|
||||
|
||||
@@ -52,3 +52,5 @@ class AgentBuilder
|
||||
}.compact))
|
||||
end
|
||||
end
|
||||
|
||||
AgentBuilder.prepend_mod_with('AgentBuilder')
|
||||
|
||||
@@ -28,7 +28,7 @@ class V2::Reports::LabelSummaryBuilder < V2::Reports::BaseSummaryBuilder
|
||||
|
||||
{
|
||||
conversation_counts: fetch_conversation_counts(conversation_filter),
|
||||
resolved_counts: fetch_resolved_counts(conversation_filter),
|
||||
resolved_counts: fetch_resolved_counts,
|
||||
resolution_metrics: fetch_metrics(conversation_filter, 'conversation_resolved', use_business_hours),
|
||||
first_response_metrics: fetch_metrics(conversation_filter, 'first_response', use_business_hours),
|
||||
reply_metrics: fetch_metrics(conversation_filter, 'reply_time', use_business_hours)
|
||||
@@ -62,10 +62,21 @@ class V2::Reports::LabelSummaryBuilder < V2::Reports::BaseSummaryBuilder
|
||||
fetch_counts(conversation_filter)
|
||||
end
|
||||
|
||||
def fetch_resolved_counts(conversation_filter)
|
||||
# since the base query is ActsAsTaggableOn,
|
||||
# the status :resolved won't automatically be converted to integer status
|
||||
fetch_counts(conversation_filter.merge(status: Conversation.statuses[:resolved]))
|
||||
def fetch_resolved_counts
|
||||
# Count resolution events, not conversations currently in resolved status
|
||||
# Filter by reporting_event.created_at, not conversation.created_at
|
||||
reporting_event_filter = { name: 'conversation_resolved', account_id: account.id }
|
||||
reporting_event_filter[:created_at] = range if range.present?
|
||||
|
||||
ReportingEvent
|
||||
.joins(conversation: { taggings: :tag })
|
||||
.where(
|
||||
reporting_event_filter.merge(
|
||||
taggings: { taggable_type: 'Conversation', context: 'labels' }
|
||||
)
|
||||
)
|
||||
.group('tags.name')
|
||||
.count
|
||||
end
|
||||
|
||||
def fetch_counts(conversation_filter)
|
||||
@@ -84,9 +95,7 @@ class V2::Reports::LabelSummaryBuilder < V2::Reports::BaseSummaryBuilder
|
||||
|
||||
def fetch_metrics(conversation_filter, event_name, use_business_hours)
|
||||
ReportingEvent
|
||||
.joins('INNER JOIN conversations ON reporting_events.conversation_id = conversations.id')
|
||||
.joins('INNER JOIN taggings ON taggings.taggable_id = conversations.id')
|
||||
.joins('INNER JOIN tags ON taggings.tag_id = tags.id')
|
||||
.joins(conversation: { taggings: :tag })
|
||||
.where(
|
||||
conversations: conversation_filter,
|
||||
name: event_name,
|
||||
|
||||
@@ -38,27 +38,34 @@ class V2::Reports::Timeseries::CountReportBuilder < V2::Reports::Timeseries::Bas
|
||||
end
|
||||
|
||||
def scope_for_resolutions_count
|
||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
||||
scope.reporting_events.where(
|
||||
name: :conversation_resolved,
|
||||
conversations: { status: :resolved }, created_at: range
|
||||
).distinct
|
||||
account_id: account.id,
|
||||
created_at: range
|
||||
)
|
||||
end
|
||||
|
||||
def scope_for_bot_resolutions_count
|
||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
||||
scope.reporting_events.where(
|
||||
name: :conversation_bot_resolved,
|
||||
conversations: { status: :resolved }, created_at: range
|
||||
).distinct
|
||||
account_id: account.id,
|
||||
created_at: range
|
||||
)
|
||||
end
|
||||
|
||||
def scope_for_bot_handoffs_count
|
||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
||||
name: :conversation_bot_handoff,
|
||||
account_id: account.id,
|
||||
created_at: range
|
||||
).distinct
|
||||
end
|
||||
|
||||
def grouped_count
|
||||
# IMPORTANT: time_zone parameter affects both data grouping AND output timestamps
|
||||
# It converts timestamps to the target timezone before grouping, which means
|
||||
# the same event can fall into different day buckets depending on timezone
|
||||
# Example: 2024-01-15 00:00 UTC becomes 2024-01-14 16:00 PST (falls on different day)
|
||||
@grouped_values = object_scope.group_by_period(
|
||||
group_by,
|
||||
:created_at,
|
||||
|
||||
@@ -0,0 +1,20 @@
|
||||
class Api::V1::Accounts::AssignmentPolicies::InboxesController < Api::V1::Accounts::BaseController
|
||||
before_action :fetch_assignment_policy
|
||||
before_action -> { check_authorization(AssignmentPolicy) }
|
||||
|
||||
def index
|
||||
@inboxes = @assignment_policy.inboxes
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def fetch_assignment_policy
|
||||
@assignment_policy = Current.account.assignment_policies.find(
|
||||
params[:assignment_policy_id]
|
||||
)
|
||||
end
|
||||
|
||||
def permitted_params
|
||||
params.permit(:assignment_policy_id)
|
||||
end
|
||||
end
|
||||
@@ -0,0 +1,36 @@
|
||||
class Api::V1::Accounts::AssignmentPoliciesController < Api::V1::Accounts::BaseController
|
||||
before_action :fetch_assignment_policy, only: [:show, :update, :destroy]
|
||||
before_action :check_authorization
|
||||
|
||||
def index
|
||||
@assignment_policies = Current.account.assignment_policies
|
||||
end
|
||||
|
||||
def show; end
|
||||
|
||||
def create
|
||||
@assignment_policy = Current.account.assignment_policies.create!(assignment_policy_params)
|
||||
end
|
||||
|
||||
def update
|
||||
@assignment_policy.update!(assignment_policy_params)
|
||||
end
|
||||
|
||||
def destroy
|
||||
@assignment_policy.destroy!
|
||||
head :ok
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def fetch_assignment_policy
|
||||
@assignment_policy = Current.account.assignment_policies.find(params[:id])
|
||||
end
|
||||
|
||||
def assignment_policy_params
|
||||
params.require(:assignment_policy).permit(
|
||||
:name, :description, :assignment_order, :conversation_priority,
|
||||
:fair_distribution_limit, :fair_distribution_window, :enabled
|
||||
)
|
||||
end
|
||||
end
|
||||
@@ -0,0 +1,46 @@
|
||||
class Api::V1::Accounts::Inboxes::AssignmentPoliciesController < Api::V1::Accounts::BaseController
|
||||
before_action :fetch_inbox
|
||||
before_action :fetch_assignment_policy, only: [:create]
|
||||
before_action -> { check_authorization(AssignmentPolicy) }
|
||||
before_action :validate_assignment_policy, only: [:show, :destroy]
|
||||
|
||||
def show
|
||||
@assignment_policy = @inbox.assignment_policy
|
||||
end
|
||||
|
||||
def create
|
||||
# There should be only one assignment policy for an inbox.
|
||||
# If there is a new request to add an assignment policy, we will
|
||||
# delete the old one and attach the new policy
|
||||
remove_inbox_assignment_policy
|
||||
@inbox_assignment_policy = @inbox.create_inbox_assignment_policy!(assignment_policy: @assignment_policy)
|
||||
@assignment_policy = @inbox.assignment_policy
|
||||
end
|
||||
|
||||
def destroy
|
||||
remove_inbox_assignment_policy
|
||||
head :ok
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def remove_inbox_assignment_policy
|
||||
@inbox.inbox_assignment_policy&.destroy
|
||||
end
|
||||
|
||||
def fetch_inbox
|
||||
@inbox = Current.account.inboxes.find(permitted_params[:inbox_id])
|
||||
end
|
||||
|
||||
def fetch_assignment_policy
|
||||
@assignment_policy = Current.account.assignment_policies.find(permitted_params[:assignment_policy_id])
|
||||
end
|
||||
|
||||
def permitted_params
|
||||
params.permit(:assignment_policy_id, :inbox_id)
|
||||
end
|
||||
|
||||
def validate_assignment_policy
|
||||
return render_not_found_error(I18n.t('errors.assignment_policy.not_found')) unless @inbox.assignment_policy
|
||||
end
|
||||
end
|
||||
@@ -70,11 +70,9 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
|
||||
end
|
||||
|
||||
def sync_templates
|
||||
unless @inbox.channel.is_a?(Channel::Whatsapp)
|
||||
return render status: :unprocessable_entity, json: { error: 'Template sync is only available for WhatsApp channels' }
|
||||
end
|
||||
return render status: :unprocessable_entity, json: { error: 'Template sync is only available for WhatsApp channels' } unless whatsapp_channel?
|
||||
|
||||
Channels::Whatsapp::TemplatesSyncJob.perform_later(@inbox.channel)
|
||||
trigger_template_sync
|
||||
render status: :ok, json: { message: 'Template sync initiated successfully' }
|
||||
rescue StandardError => e
|
||||
render status: :internal_server_error, json: { error: e.message }
|
||||
@@ -185,6 +183,18 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
def whatsapp_channel?
|
||||
@inbox.whatsapp? || (@inbox.twilio? && @inbox.channel.whatsapp?)
|
||||
end
|
||||
|
||||
def trigger_template_sync
|
||||
if @inbox.whatsapp?
|
||||
Channels::Whatsapp::TemplatesSyncJob.perform_later(@inbox.channel)
|
||||
elsif @inbox.twilio? && @inbox.channel.whatsapp?
|
||||
Channels::Twilio::TemplatesSyncJob.perform_later(@inbox.channel)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
Api::V1::Accounts::InboxesController.prepend_mod_with('Api::V1::Accounts::InboxesController')
|
||||
|
||||
@@ -85,7 +85,8 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
|
||||
|
||||
def live_chat_widget_params
|
||||
permitted_params = params.permit(:inbox_id)
|
||||
return {} if permitted_params[:inbox_id].blank?
|
||||
return {} unless permitted_params.key?(:inbox_id)
|
||||
return { channel_web_widget_id: nil } if permitted_params[:inbox_id].blank?
|
||||
|
||||
inbox = Inbox.find(permitted_params[:inbox_id])
|
||||
return {} unless inbox.web_widget?
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
class Api::V1::Accounts::Whatsapp::AuthorizationsController < Api::V1::Accounts::BaseController
|
||||
before_action :validate_feature_enabled!
|
||||
before_action :fetch_and_validate_inbox, if: -> { params[:inbox_id].present? }
|
||||
|
||||
# POST /api/v1/accounts/:account_id/whatsapp/authorization
|
||||
@@ -65,15 +64,6 @@ class Api::V1::Accounts::Whatsapp::AuthorizationsController < Api::V1::Accounts:
|
||||
}, status: :unprocessable_entity
|
||||
end
|
||||
|
||||
def validate_feature_enabled!
|
||||
return if Current.account.feature_whatsapp_embedded_signup?
|
||||
|
||||
render json: {
|
||||
success: false,
|
||||
error: 'WhatsApp embedded signup is not enabled for this account'
|
||||
}, status: :forbidden
|
||||
end
|
||||
|
||||
def validate_embedded_signup_params!
|
||||
missing_params = []
|
||||
missing_params << 'code' if params[:code].blank?
|
||||
|
||||
68
app/controllers/api/v1/profile/mfa_controller.rb
Normal file
68
app/controllers/api/v1/profile/mfa_controller.rb
Normal file
@@ -0,0 +1,68 @@
|
||||
class Api::V1::Profile::MfaController < Api::BaseController
|
||||
before_action :check_mfa_feature_available
|
||||
before_action :check_mfa_enabled, only: [:destroy, :backup_codes]
|
||||
before_action :check_mfa_disabled, only: [:create, :verify]
|
||||
before_action :validate_otp, only: [:verify, :backup_codes, :destroy]
|
||||
before_action :validate_password, only: [:destroy]
|
||||
|
||||
def show; end
|
||||
|
||||
def create
|
||||
mfa_service.enable_two_factor!
|
||||
end
|
||||
|
||||
def verify
|
||||
@backup_codes = mfa_service.verify_and_activate!
|
||||
end
|
||||
|
||||
def destroy
|
||||
mfa_service.disable_two_factor!
|
||||
end
|
||||
|
||||
def backup_codes
|
||||
@backup_codes = mfa_service.generate_backup_codes!
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def mfa_service
|
||||
@mfa_service ||= Mfa::ManagementService.new(user: current_user)
|
||||
end
|
||||
|
||||
def check_mfa_enabled
|
||||
render_could_not_create_error(I18n.t('errors.mfa.not_enabled')) unless current_user.mfa_enabled?
|
||||
end
|
||||
|
||||
def check_mfa_feature_available
|
||||
return if Chatwoot.mfa_enabled?
|
||||
|
||||
render json: {
|
||||
error: I18n.t('errors.mfa.feature_unavailable')
|
||||
}, status: :forbidden
|
||||
end
|
||||
|
||||
def check_mfa_disabled
|
||||
render_could_not_create_error(I18n.t('errors.mfa.already_enabled')) if current_user.mfa_enabled?
|
||||
end
|
||||
|
||||
def validate_otp
|
||||
authenticated = Mfa::AuthenticationService.new(
|
||||
user: current_user,
|
||||
otp_code: mfa_params[:otp_code]
|
||||
).authenticate
|
||||
|
||||
return if authenticated
|
||||
|
||||
render_could_not_create_error(I18n.t('errors.mfa.invalid_code'))
|
||||
end
|
||||
|
||||
def validate_password
|
||||
return if current_user.valid_password?(mfa_params[:password])
|
||||
|
||||
render_could_not_create_error(I18n.t('errors.mfa.invalid_credentials'))
|
||||
end
|
||||
|
||||
def mfa_params
|
||||
params.permit(:otp_code, :password)
|
||||
end
|
||||
end
|
||||
@@ -9,7 +9,7 @@ class Api::V1::Widget::ConfigsController < Api::V1::Widget::BaseController
|
||||
private
|
||||
|
||||
def set_global_config
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL')
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'INSTALLATION_NAME')
|
||||
end
|
||||
|
||||
def set_contact
|
||||
|
||||
@@ -4,17 +4,28 @@ module SwitchLocale
|
||||
private
|
||||
|
||||
def switch_locale(&)
|
||||
# priority is for locale set in query string (mostly for widget/from js sdk)
|
||||
# Priority is for locale set in query string (mostly for widget/from js sdk)
|
||||
locale ||= params[:locale]
|
||||
|
||||
# Use the user's locale if available
|
||||
locale ||= locale_from_user
|
||||
|
||||
# Use the locale from a custom domain if applicable
|
||||
locale ||= locale_from_custom_domain
|
||||
|
||||
# if locale is not set in account, let's use DEFAULT_LOCALE env variable
|
||||
locale ||= ENV.fetch('DEFAULT_LOCALE', nil)
|
||||
|
||||
set_locale(locale, &)
|
||||
end
|
||||
|
||||
def switch_locale_using_account_locale(&)
|
||||
locale = locale_from_account(@current_account)
|
||||
# Get the locale from the user first
|
||||
locale = locale_from_user
|
||||
|
||||
# Fallback to the account's locale if the user's locale is not set
|
||||
locale ||= locale_from_account(@current_account)
|
||||
|
||||
set_locale(locale, &)
|
||||
end
|
||||
|
||||
@@ -32,6 +43,12 @@ module SwitchLocale
|
||||
@portal.default_locale
|
||||
end
|
||||
|
||||
def locale_from_user
|
||||
return unless @user
|
||||
|
||||
@user.ui_settings&.dig('locale')
|
||||
end
|
||||
|
||||
def set_locale(locale, &)
|
||||
safe_locale = validate_and_get_locale(locale)
|
||||
# Ensure locale won't bleed into other requests
|
||||
|
||||
@@ -66,7 +66,7 @@ class DashboardController < ActionController::Base
|
||||
ENABLE_ACCOUNT_SIGNUP: GlobalConfigService.load('ENABLE_ACCOUNT_SIGNUP', 'false'),
|
||||
FB_APP_ID: GlobalConfigService.load('FB_APP_ID', ''),
|
||||
INSTAGRAM_APP_ID: GlobalConfigService.load('INSTAGRAM_APP_ID', ''),
|
||||
FACEBOOK_API_VERSION: GlobalConfigService.load('FACEBOOK_API_VERSION', 'v17.0'),
|
||||
FACEBOOK_API_VERSION: GlobalConfigService.load('FACEBOOK_API_VERSION', 'v18.0'),
|
||||
WHATSAPP_APP_ID: GlobalConfigService.load('WHATSAPP_APP_ID', ''),
|
||||
WHATSAPP_CONFIGURATION_ID: GlobalConfigService.load('WHATSAPP_CONFIGURATION_ID', ''),
|
||||
IS_ENTERPRISE: ChatwootApp.enterprise?,
|
||||
|
||||
@@ -47,10 +47,8 @@ class DeviseOverrides::OmniauthCallbacksController < DeviseTokenAuth::OmniauthCa
|
||||
end
|
||||
|
||||
def get_resource_from_auth_hash # rubocop:disable Naming/AccessorMethodName
|
||||
# find the user with their email instead of UID and token
|
||||
@resource = resource_class.where(
|
||||
email: auth_hash['info']['email']
|
||||
).first
|
||||
email = auth_hash.dig('info', 'email')
|
||||
@resource = resource_class.from_email(email)
|
||||
end
|
||||
|
||||
def validate_signup_email_is_business_domain?
|
||||
@@ -75,3 +73,5 @@ class DeviseOverrides::OmniauthCallbacksController < DeviseTokenAuth::OmniauthCa
|
||||
'user'
|
||||
end
|
||||
end
|
||||
|
||||
DeviseOverrides::OmniauthCallbacksController.prepend_mod_with('DeviseOverrides::OmniauthCallbacksController')
|
||||
|
||||
@@ -44,3 +44,5 @@ class DeviseOverrides::PasswordsController < Devise::PasswordsController
|
||||
}, status: status
|
||||
end
|
||||
end
|
||||
|
||||
DeviseOverrides::PasswordsController.prepend_mod_with('DeviseOverrides::PasswordsController')
|
||||
|
||||
@@ -9,13 +9,11 @@ class DeviseOverrides::SessionsController < DeviseTokenAuth::SessionsController
|
||||
end
|
||||
|
||||
def create
|
||||
# Authenticate user via the temporary sso auth token
|
||||
if params[:sso_auth_token].present? && @resource.present?
|
||||
authenticate_resource_with_sso_token
|
||||
yield @resource if block_given?
|
||||
render_create_success
|
||||
else
|
||||
super
|
||||
return handle_mfa_verification if mfa_verification_request?
|
||||
return handle_sso_authentication if sso_authentication_request?
|
||||
|
||||
super do |resource|
|
||||
return handle_mfa_required(resource) if resource&.mfa_enabled?
|
||||
end
|
||||
end
|
||||
|
||||
@@ -25,6 +23,20 @@ class DeviseOverrides::SessionsController < DeviseTokenAuth::SessionsController
|
||||
|
||||
private
|
||||
|
||||
def mfa_verification_request?
|
||||
params[:mfa_token].present?
|
||||
end
|
||||
|
||||
def sso_authentication_request?
|
||||
params[:sso_auth_token].present? && @resource.present?
|
||||
end
|
||||
|
||||
def handle_sso_authentication
|
||||
authenticate_resource_with_sso_token
|
||||
yield @resource if block_given?
|
||||
render_create_success
|
||||
end
|
||||
|
||||
def login_page_url(error: nil)
|
||||
frontend_url = ENV.fetch('FRONTEND_URL', nil)
|
||||
|
||||
@@ -46,6 +58,41 @@ class DeviseOverrides::SessionsController < DeviseTokenAuth::SessionsController
|
||||
user = User.from_email(params[:email])
|
||||
@resource = user if user&.valid_sso_auth_token?(params[:sso_auth_token])
|
||||
end
|
||||
|
||||
def handle_mfa_required(resource)
|
||||
render json: {
|
||||
mfa_required: true,
|
||||
mfa_token: Mfa::TokenService.new(user: resource).generate_token
|
||||
}, status: :partial_content
|
||||
end
|
||||
|
||||
def handle_mfa_verification
|
||||
user = Mfa::TokenService.new(token: params[:mfa_token]).verify_token
|
||||
return render_mfa_error('errors.mfa.invalid_token', :unauthorized) unless user
|
||||
|
||||
authenticated = Mfa::AuthenticationService.new(
|
||||
user: user,
|
||||
otp_code: params[:otp_code],
|
||||
backup_code: params[:backup_code]
|
||||
).authenticate
|
||||
|
||||
return render_mfa_error('errors.mfa.invalid_code') unless authenticated
|
||||
|
||||
sign_in_mfa_user(user)
|
||||
end
|
||||
|
||||
def sign_in_mfa_user(user)
|
||||
@resource = user
|
||||
@token = @resource.create_token
|
||||
@resource.save!
|
||||
|
||||
sign_in(:user, @resource, store: false, bypass: false)
|
||||
render_create_success
|
||||
end
|
||||
|
||||
def render_mfa_error(message_key, status = :bad_request)
|
||||
render json: { error: I18n.t(message_key) }, status: status
|
||||
end
|
||||
end
|
||||
|
||||
DeviseOverrides::SessionsController.prepend_mod_with('DeviseOverrides::SessionsController')
|
||||
|
||||
@@ -58,6 +58,6 @@ class Public::Api::V1::Portals::BaseController < PublicController
|
||||
end
|
||||
|
||||
def set_global_config
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'BRAND_URL')
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'BRAND_URL', 'INSTALLATION_NAME')
|
||||
end
|
||||
end
|
||||
|
||||
@@ -5,6 +5,6 @@ class Survey::ResponsesController < ActionController::Base
|
||||
private
|
||||
|
||||
def set_global_config
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL')
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'INSTALLATION_NAME')
|
||||
end
|
||||
end
|
||||
|
||||
@@ -14,7 +14,7 @@ class WidgetsController < ActionController::Base
|
||||
private
|
||||
|
||||
def set_global_config
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'DIRECT_UPLOADS_ENABLED')
|
||||
@global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'DIRECT_UPLOADS_ENABLED', 'INSTALLATION_NAME')
|
||||
end
|
||||
|
||||
def set_web_widget
|
||||
@@ -70,7 +70,12 @@ class WidgetsController < ActionController::Base
|
||||
end
|
||||
|
||||
def allow_iframe_requests
|
||||
response.headers.delete('X-Frame-Options')
|
||||
if @web_widget.allowed_domains.blank?
|
||||
response.headers.delete('X-Frame-Options')
|
||||
else
|
||||
domains = @web_widget.allowed_domains.split(',').map(&:strip).join(' ')
|
||||
response.headers['Content-Security-Policy'] = "frame-ancestors #{domains}"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
module PortalHelper
|
||||
include UrlHelper
|
||||
def set_og_image_url(portal_name, title)
|
||||
cdn_url = GlobalConfig.get('OG_IMAGE_CDN_URL')['OG_IMAGE_CDN_URL']
|
||||
return if cdn_url.blank?
|
||||
@@ -74,6 +75,17 @@ module PortalHelper
|
||||
end
|
||||
end
|
||||
|
||||
def generate_portal_brand_url(brand_url, referer)
|
||||
url = URI.parse(brand_url.to_s)
|
||||
query_params = Rack::Utils.parse_query(url.query)
|
||||
query_params['utm_medium'] = 'helpcenter'
|
||||
query_params['utm_campaign'] = 'branding'
|
||||
query_params['utm_source'] = URI.parse(referer).host if url_valid?(referer)
|
||||
|
||||
url.query = query_params.to_query
|
||||
url.to_s
|
||||
end
|
||||
|
||||
def render_category_content(content)
|
||||
ChatwootMarkdownRenderer.new(content).render_markdown_to_plain_text
|
||||
end
|
||||
|
||||
@@ -53,13 +53,13 @@ module ReportHelper
|
||||
end
|
||||
|
||||
def resolutions
|
||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(account_id: account.id, name: :conversation_resolved,
|
||||
conversations: { status: :resolved }, created_at: range).distinct
|
||||
scope.reporting_events.where(account_id: account.id, name: :conversation_resolved,
|
||||
created_at: range)
|
||||
end
|
||||
|
||||
def bot_resolutions
|
||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(account_id: account.id, name: :conversation_bot_resolved,
|
||||
conversations: { status: :resolved }, created_at: range).distinct
|
||||
scope.reporting_events.where(account_id: account.id, name: :conversation_bot_resolved,
|
||||
created_at: range)
|
||||
end
|
||||
|
||||
def bot_handoffs
|
||||
|
||||
@@ -19,6 +19,7 @@ import {
|
||||
verifyServiceWorkerExistence,
|
||||
} from './helper/pushHelper';
|
||||
import ReconnectService from 'dashboard/helper/ReconnectService';
|
||||
import { useUISettings } from 'dashboard/composables/useUISettings';
|
||||
|
||||
export default {
|
||||
name: 'App',
|
||||
@@ -38,12 +39,14 @@ export default {
|
||||
const { accountId } = useAccount();
|
||||
// Use the font size composable (it automatically sets up the watcher)
|
||||
const { currentFontSize } = useFontSize();
|
||||
const { uiSettings } = useUISettings();
|
||||
|
||||
return {
|
||||
router,
|
||||
store,
|
||||
currentAccountId: accountId,
|
||||
currentFontSize,
|
||||
uiSettings,
|
||||
};
|
||||
},
|
||||
data() {
|
||||
@@ -88,7 +91,10 @@ export default {
|
||||
mounted() {
|
||||
this.initializeColorTheme();
|
||||
this.listenToThemeChanges();
|
||||
this.setLocale(window.chatwootConfig.selectedLocale);
|
||||
// If user locale is set, use it; otherwise use account locale
|
||||
this.setLocale(
|
||||
this.uiSettings?.locale || window.chatwootConfig.selectedLocale
|
||||
);
|
||||
},
|
||||
unmounted() {
|
||||
if (this.reconnectService) {
|
||||
@@ -114,7 +120,8 @@ export default {
|
||||
const { locale, latest_chatwoot_version: latestChatwootVersion } =
|
||||
this.getAccount(this.currentAccountId);
|
||||
const { pubsub_token: pubsubToken } = this.currentUser || {};
|
||||
this.setLocale(locale);
|
||||
// If user locale is set, use it; otherwise use account locale
|
||||
this.setLocale(this.uiSettings?.locale || locale);
|
||||
this.latestChatwootVersion = latestChatwootVersion;
|
||||
vueActionCable.init(this.store, pubsubToken);
|
||||
this.reconnectService = new ReconnectService(this.store, this.router);
|
||||
|
||||
43
app/javascript/dashboard/api/agentCapacityPolicies.js
Normal file
43
app/javascript/dashboard/api/agentCapacityPolicies.js
Normal file
@@ -0,0 +1,43 @@
|
||||
/* global axios */
|
||||
|
||||
import ApiClient from './ApiClient';
|
||||
|
||||
class AgentCapacityPolicies extends ApiClient {
|
||||
constructor() {
|
||||
super('agent_capacity_policies', { accountScoped: true });
|
||||
}
|
||||
|
||||
getUsers(policyId) {
|
||||
return axios.get(`${this.url}/${policyId}/users`);
|
||||
}
|
||||
|
||||
addUser(policyId, userData) {
|
||||
return axios.post(`${this.url}/${policyId}/users`, {
|
||||
user_id: userData.id,
|
||||
capacity: userData.capacity,
|
||||
});
|
||||
}
|
||||
|
||||
removeUser(policyId, userId) {
|
||||
return axios.delete(`${this.url}/${policyId}/users/${userId}`);
|
||||
}
|
||||
|
||||
createInboxLimit(policyId, limitData) {
|
||||
return axios.post(`${this.url}/${policyId}/inbox_limits`, {
|
||||
inbox_id: limitData.inboxId,
|
||||
conversation_limit: limitData.conversationLimit,
|
||||
});
|
||||
}
|
||||
|
||||
updateInboxLimit(policyId, limitId, limitData) {
|
||||
return axios.put(`${this.url}/${policyId}/inbox_limits/${limitId}`, {
|
||||
conversation_limit: limitData.conversationLimit,
|
||||
});
|
||||
}
|
||||
|
||||
deleteInboxLimit(policyId, limitId) {
|
||||
return axios.delete(`${this.url}/${policyId}/inbox_limits/${limitId}`);
|
||||
}
|
||||
}
|
||||
|
||||
export default new AgentCapacityPolicies();
|
||||
36
app/javascript/dashboard/api/assignmentPolicies.js
Normal file
36
app/javascript/dashboard/api/assignmentPolicies.js
Normal file
@@ -0,0 +1,36 @@
|
||||
/* global axios */
|
||||
|
||||
import ApiClient from './ApiClient';
|
||||
|
||||
class AssignmentPolicies extends ApiClient {
|
||||
constructor() {
|
||||
super('assignment_policies', { accountScoped: true });
|
||||
}
|
||||
|
||||
getInboxes(policyId) {
|
||||
return axios.get(`${this.url}/${policyId}/inboxes`);
|
||||
}
|
||||
|
||||
setInboxPolicy(inboxId, policyId) {
|
||||
return axios.post(
|
||||
`/api/v1/accounts/${this.accountIdFromRoute}/inboxes/${inboxId}/assignment_policy`,
|
||||
{
|
||||
assignment_policy_id: policyId,
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
getInboxPolicy(inboxId) {
|
||||
return axios.get(
|
||||
`/api/v1/accounts/${this.accountIdFromRoute}/inboxes/${inboxId}/assignment_policy`
|
||||
);
|
||||
}
|
||||
|
||||
removeInboxPolicy(inboxId) {
|
||||
return axios.delete(
|
||||
`/api/v1/accounts/${this.accountIdFromRoute}/inboxes/${inboxId}/assignment_policy`
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
export default new AssignmentPolicies();
|
||||
@@ -6,11 +6,11 @@ class CaptainResponses extends ApiClient {
|
||||
super('captain/assistant_responses', { accountScoped: true });
|
||||
}
|
||||
|
||||
get({ page = 1, searchKey, assistantId, documentId, status } = {}) {
|
||||
get({ page = 1, search, assistantId, documentId, status } = {}) {
|
||||
return axios.get(this.url, {
|
||||
params: {
|
||||
page,
|
||||
searchKey,
|
||||
search,
|
||||
assistant_id: assistantId,
|
||||
document_id: documentId,
|
||||
status,
|
||||
|
||||
28
app/javascript/dashboard/api/mfa.js
Normal file
28
app/javascript/dashboard/api/mfa.js
Normal file
@@ -0,0 +1,28 @@
|
||||
/* global axios */
|
||||
import ApiClient from './ApiClient';
|
||||
|
||||
class MfaAPI extends ApiClient {
|
||||
constructor() {
|
||||
super('profile/mfa', { accountScoped: false });
|
||||
}
|
||||
|
||||
enable() {
|
||||
return axios.post(`${this.url}`);
|
||||
}
|
||||
|
||||
verify(otpCode) {
|
||||
return axios.post(`${this.url}/verify`, { otp_code: otpCode });
|
||||
}
|
||||
|
||||
disable(password, otpCode) {
|
||||
return axios.delete(this.url, {
|
||||
data: { password, otp_code: otpCode },
|
||||
});
|
||||
}
|
||||
|
||||
regenerateBackupCodes(otpCode) {
|
||||
return axios.post(`${this.url}/backup_codes`, { otp_code: otpCode });
|
||||
}
|
||||
}
|
||||
|
||||
export default new MfaAPI();
|
||||
26
app/javascript/dashboard/api/samlSettings.js
Normal file
26
app/javascript/dashboard/api/samlSettings.js
Normal file
@@ -0,0 +1,26 @@
|
||||
/* global axios */
|
||||
import ApiClient from './ApiClient';
|
||||
|
||||
class SamlSettingsAPI extends ApiClient {
|
||||
constructor() {
|
||||
super('saml_settings', { accountScoped: true });
|
||||
}
|
||||
|
||||
get() {
|
||||
return axios.get(this.url);
|
||||
}
|
||||
|
||||
create(data) {
|
||||
return axios.post(this.url, { saml_settings: data });
|
||||
}
|
||||
|
||||
update(data) {
|
||||
return axios.put(this.url, { saml_settings: data });
|
||||
}
|
||||
|
||||
delete() {
|
||||
return axios.delete(this.url);
|
||||
}
|
||||
}
|
||||
|
||||
export default new SamlSettingsAPI();
|
||||
@@ -0,0 +1,98 @@
|
||||
import agentCapacityPolicies from '../agentCapacityPolicies';
|
||||
import ApiClient from '../ApiClient';
|
||||
|
||||
describe('#AgentCapacityPoliciesAPI', () => {
|
||||
it('creates correct instance', () => {
|
||||
expect(agentCapacityPolicies).toBeInstanceOf(ApiClient);
|
||||
expect(agentCapacityPolicies).toHaveProperty('get');
|
||||
expect(agentCapacityPolicies).toHaveProperty('show');
|
||||
expect(agentCapacityPolicies).toHaveProperty('create');
|
||||
expect(agentCapacityPolicies).toHaveProperty('update');
|
||||
expect(agentCapacityPolicies).toHaveProperty('delete');
|
||||
expect(agentCapacityPolicies).toHaveProperty('getUsers');
|
||||
expect(agentCapacityPolicies).toHaveProperty('addUser');
|
||||
expect(agentCapacityPolicies).toHaveProperty('removeUser');
|
||||
expect(agentCapacityPolicies).toHaveProperty('createInboxLimit');
|
||||
expect(agentCapacityPolicies).toHaveProperty('updateInboxLimit');
|
||||
expect(agentCapacityPolicies).toHaveProperty('deleteInboxLimit');
|
||||
});
|
||||
|
||||
describe('API calls', () => {
|
||||
const originalAxios = window.axios;
|
||||
const axiosMock = {
|
||||
get: vi.fn(() => Promise.resolve()),
|
||||
post: vi.fn(() => Promise.resolve()),
|
||||
put: vi.fn(() => Promise.resolve()),
|
||||
delete: vi.fn(() => Promise.resolve()),
|
||||
};
|
||||
|
||||
beforeEach(() => {
|
||||
window.axios = axiosMock;
|
||||
// Mock accountIdFromRoute
|
||||
Object.defineProperty(agentCapacityPolicies, 'accountIdFromRoute', {
|
||||
get: () => '1',
|
||||
configurable: true,
|
||||
});
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
window.axios = originalAxios;
|
||||
});
|
||||
|
||||
it('#getUsers', () => {
|
||||
agentCapacityPolicies.getUsers(123);
|
||||
expect(axiosMock.get).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/agent_capacity_policies/123/users'
|
||||
);
|
||||
});
|
||||
|
||||
it('#addUser', () => {
|
||||
const userData = { id: 456, capacity: 20 };
|
||||
agentCapacityPolicies.addUser(123, userData);
|
||||
expect(axiosMock.post).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/agent_capacity_policies/123/users',
|
||||
{
|
||||
user_id: 456,
|
||||
capacity: 20,
|
||||
}
|
||||
);
|
||||
});
|
||||
|
||||
it('#removeUser', () => {
|
||||
agentCapacityPolicies.removeUser(123, 456);
|
||||
expect(axiosMock.delete).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/agent_capacity_policies/123/users/456'
|
||||
);
|
||||
});
|
||||
|
||||
it('#createInboxLimit', () => {
|
||||
const limitData = { inboxId: 1, conversationLimit: 10 };
|
||||
agentCapacityPolicies.createInboxLimit(123, limitData);
|
||||
expect(axiosMock.post).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/agent_capacity_policies/123/inbox_limits',
|
||||
{
|
||||
inbox_id: 1,
|
||||
conversation_limit: 10,
|
||||
}
|
||||
);
|
||||
});
|
||||
|
||||
it('#updateInboxLimit', () => {
|
||||
const limitData = { conversationLimit: 15 };
|
||||
agentCapacityPolicies.updateInboxLimit(123, 789, limitData);
|
||||
expect(axiosMock.put).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/agent_capacity_policies/123/inbox_limits/789',
|
||||
{
|
||||
conversation_limit: 15,
|
||||
}
|
||||
);
|
||||
});
|
||||
|
||||
it('#deleteInboxLimit', () => {
|
||||
agentCapacityPolicies.deleteInboxLimit(123, 789);
|
||||
expect(axiosMock.delete).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/agent_capacity_policies/123/inbox_limits/789'
|
||||
);
|
||||
});
|
||||
});
|
||||
});
|
||||
@@ -0,0 +1,70 @@
|
||||
import assignmentPolicies from '../assignmentPolicies';
|
||||
import ApiClient from '../ApiClient';
|
||||
|
||||
describe('#AssignmentPoliciesAPI', () => {
|
||||
it('creates correct instance', () => {
|
||||
expect(assignmentPolicies).toBeInstanceOf(ApiClient);
|
||||
expect(assignmentPolicies).toHaveProperty('get');
|
||||
expect(assignmentPolicies).toHaveProperty('show');
|
||||
expect(assignmentPolicies).toHaveProperty('create');
|
||||
expect(assignmentPolicies).toHaveProperty('update');
|
||||
expect(assignmentPolicies).toHaveProperty('delete');
|
||||
expect(assignmentPolicies).toHaveProperty('getInboxes');
|
||||
expect(assignmentPolicies).toHaveProperty('setInboxPolicy');
|
||||
expect(assignmentPolicies).toHaveProperty('getInboxPolicy');
|
||||
expect(assignmentPolicies).toHaveProperty('removeInboxPolicy');
|
||||
});
|
||||
|
||||
describe('API calls', () => {
|
||||
const originalAxios = window.axios;
|
||||
const axiosMock = {
|
||||
get: vi.fn(() => Promise.resolve()),
|
||||
post: vi.fn(() => Promise.resolve()),
|
||||
delete: vi.fn(() => Promise.resolve()),
|
||||
};
|
||||
|
||||
beforeEach(() => {
|
||||
window.axios = axiosMock;
|
||||
// Mock accountIdFromRoute
|
||||
Object.defineProperty(assignmentPolicies, 'accountIdFromRoute', {
|
||||
get: () => '1',
|
||||
configurable: true,
|
||||
});
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
window.axios = originalAxios;
|
||||
});
|
||||
|
||||
it('#getInboxes', () => {
|
||||
assignmentPolicies.getInboxes(123);
|
||||
expect(axiosMock.get).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/assignment_policies/123/inboxes'
|
||||
);
|
||||
});
|
||||
|
||||
it('#setInboxPolicy', () => {
|
||||
assignmentPolicies.setInboxPolicy(456, 123);
|
||||
expect(axiosMock.post).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/inboxes/456/assignment_policy',
|
||||
{
|
||||
assignment_policy_id: 123,
|
||||
}
|
||||
);
|
||||
});
|
||||
|
||||
it('#getInboxPolicy', () => {
|
||||
assignmentPolicies.getInboxPolicy(456);
|
||||
expect(axiosMock.get).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/inboxes/456/assignment_policy'
|
||||
);
|
||||
});
|
||||
|
||||
it('#removeInboxPolicy', () => {
|
||||
assignmentPolicies.removeInboxPolicy(456);
|
||||
expect(axiosMock.delete).toHaveBeenCalledWith(
|
||||
'/api/v1/accounts/1/inboxes/456/assignment_policy'
|
||||
);
|
||||
});
|
||||
});
|
||||
});
|
||||
@@ -0,0 +1,116 @@
|
||||
<script setup>
|
||||
import AgentCapacityPolicyCard from './AgentCapacityPolicyCard.vue';
|
||||
|
||||
const mockUsers = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'John Smith',
|
||||
email: 'john.smith@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=1',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Sarah Johnson',
|
||||
email: 'sarah.johnson@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=2',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'Mike Chen',
|
||||
email: 'mike.chen@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=3',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Emily Davis',
|
||||
email: 'emily.davis@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=4',
|
||||
},
|
||||
{
|
||||
id: 5,
|
||||
name: 'Alex Rodriguez',
|
||||
email: 'alex.rodriguez@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=5',
|
||||
},
|
||||
];
|
||||
|
||||
const withCount = policy => ({
|
||||
...policy,
|
||||
assignedAgentCount: policy.users.length,
|
||||
});
|
||||
|
||||
const policyA = withCount({
|
||||
id: 1,
|
||||
name: 'High Volume Support',
|
||||
description:
|
||||
'Capacity-based policy for handling high conversation volumes with experienced agents',
|
||||
users: [mockUsers[0], mockUsers[1], mockUsers[2]],
|
||||
isFetchingUsers: false,
|
||||
});
|
||||
|
||||
const policyB = withCount({
|
||||
id: 2,
|
||||
name: 'Specialized Team',
|
||||
description: 'Custom capacity limits for specialized support team members',
|
||||
users: [mockUsers[3], mockUsers[4]],
|
||||
isFetchingUsers: false,
|
||||
});
|
||||
|
||||
const emptyPolicy = withCount({
|
||||
id: 3,
|
||||
name: 'New Policy',
|
||||
description: 'Recently created policy with no assigned agents yet',
|
||||
users: [],
|
||||
isFetchingUsers: false,
|
||||
});
|
||||
|
||||
const loadingPolicy = withCount({
|
||||
id: 4,
|
||||
name: 'Loading Policy',
|
||||
description: 'Policy currently loading agent information',
|
||||
users: [],
|
||||
isFetchingUsers: true,
|
||||
});
|
||||
|
||||
const onEdit = id => console.log('Edit policy:', id);
|
||||
const onDelete = id => console.log('Delete policy:', id);
|
||||
const onFetchUsers = id => console.log('Fetch users for policy:', id);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/AgentCapacityPolicyCard"
|
||||
:layout="{ type: 'grid', width: '1200px' }"
|
||||
>
|
||||
<Variant title="Multiple Cards (Various States)">
|
||||
<div class="p-4 bg-n-background">
|
||||
<div class="grid grid-cols-1 gap-4">
|
||||
<AgentCapacityPolicyCard
|
||||
v-bind="policyA"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-users="onFetchUsers"
|
||||
/>
|
||||
<AgentCapacityPolicyCard
|
||||
v-bind="policyB"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-users="onFetchUsers"
|
||||
/>
|
||||
<AgentCapacityPolicyCard
|
||||
v-bind="emptyPolicy"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-users="onFetchUsers"
|
||||
/>
|
||||
<AgentCapacityPolicyCard
|
||||
v-bind="loadingPolicy"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-users="onFetchUsers"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,86 @@
|
||||
<script setup>
|
||||
import { computed } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import CardLayout from 'dashboard/components-next/CardLayout.vue';
|
||||
import CardPopover from '../components/CardPopover.vue';
|
||||
|
||||
const props = defineProps({
|
||||
id: { type: Number, required: true },
|
||||
name: { type: String, default: '' },
|
||||
description: { type: String, default: '' },
|
||||
assignedAgentCount: { type: Number, default: 0 },
|
||||
users: { type: Array, default: () => [] },
|
||||
isFetchingUsers: { type: Boolean, default: false },
|
||||
});
|
||||
|
||||
const emit = defineEmits(['edit', 'delete', 'fetchUsers']);
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const users = computed(() => {
|
||||
return props.users.map(user => {
|
||||
return {
|
||||
name: user.name,
|
||||
key: user.id,
|
||||
email: user.email,
|
||||
avatarUrl: user.avatarUrl,
|
||||
};
|
||||
});
|
||||
});
|
||||
|
||||
const handleEdit = () => {
|
||||
emit('edit', props.id);
|
||||
};
|
||||
|
||||
const handleDelete = () => {
|
||||
emit('delete', props.id);
|
||||
};
|
||||
|
||||
const handleFetchUsers = () => {
|
||||
if (props.users?.length > 0) return;
|
||||
emit('fetchUsers', props.id);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CardLayout class="[&>div]:px-5">
|
||||
<div class="flex flex-col gap-2 relative justify-between w-full">
|
||||
<div class="flex items-center gap-3 justify-between w-full">
|
||||
<div class="flex items-center gap-3">
|
||||
<h3 class="text-base font-medium text-n-slate-12 line-clamp-1">
|
||||
{{ name }}
|
||||
</h3>
|
||||
<CardPopover
|
||||
:title="
|
||||
t('ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.INDEX.CARD.POPOVER')
|
||||
"
|
||||
icon="i-lucide-users-round"
|
||||
:count="assignedAgentCount"
|
||||
:items="users"
|
||||
:is-fetching="isFetchingUsers"
|
||||
@fetch="handleFetchUsers"
|
||||
/>
|
||||
</div>
|
||||
<div class="flex items-center gap-2">
|
||||
<Button
|
||||
:label="
|
||||
t('ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.INDEX.CARD.EDIT')
|
||||
"
|
||||
sm
|
||||
slate
|
||||
link
|
||||
class="px-2"
|
||||
@click="handleEdit"
|
||||
/>
|
||||
<div class="w-px h-2.5 bg-n-slate-5" />
|
||||
<Button icon="i-lucide-trash" sm slate ghost @click="handleDelete" />
|
||||
</div>
|
||||
</div>
|
||||
<p class="text-n-slate-11 text-sm line-clamp-1 mb-0 py-1">
|
||||
{{ description }}
|
||||
</p>
|
||||
</div>
|
||||
</CardLayout>
|
||||
</template>
|
||||
@@ -0,0 +1,63 @@
|
||||
<script setup>
|
||||
import AssignmentCard from './AssignmentCard.vue';
|
||||
|
||||
const agentAssignments = [
|
||||
{
|
||||
id: 1,
|
||||
title: 'Assignment policy',
|
||||
description: 'Manage how conversations get assigned in inboxes.',
|
||||
features: [
|
||||
{
|
||||
icon: 'i-lucide-circle-fading-arrow-up',
|
||||
label: 'Assign by conversations evenly or by available capacity',
|
||||
},
|
||||
{
|
||||
icon: 'i-lucide-scale',
|
||||
label: 'Add fair distribution rules to avoid overloading any agent',
|
||||
},
|
||||
{
|
||||
icon: 'i-lucide-inbox',
|
||||
label: 'Add inboxes to a policy - one policy per inbox',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
title: 'Agent capacity policy',
|
||||
description: 'Manage workload for agents.',
|
||||
features: [
|
||||
{
|
||||
icon: 'i-lucide-glass-water',
|
||||
label: 'Define maximum conversations per inbox',
|
||||
},
|
||||
{
|
||||
icon: 'i-lucide-circle-minus',
|
||||
label: 'Create exceptions based on labels and time',
|
||||
},
|
||||
{
|
||||
icon: 'i-lucide-users-round',
|
||||
label: 'Add agents to a policy - one policy per agent',
|
||||
},
|
||||
],
|
||||
},
|
||||
];
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/AssignmentCard"
|
||||
:layout="{ type: 'grid', width: '1000px' }"
|
||||
>
|
||||
<Variant title="Assignment Card">
|
||||
<div class="px-4 py-4 bg-n-background flex gap-6 justify-between">
|
||||
<AssignmentCard
|
||||
v-for="(item, index) in agentAssignments"
|
||||
:key="index"
|
||||
:title="item.title"
|
||||
:description="item.description"
|
||||
:features="item.features"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,49 @@
|
||||
<script setup>
|
||||
import Icon from 'dashboard/components-next/icon/Icon.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import CardLayout from 'dashboard/components-next/CardLayout.vue';
|
||||
|
||||
defineProps({
|
||||
title: { type: String, default: '' },
|
||||
description: { type: String, default: '' },
|
||||
features: { type: Array, default: () => [] },
|
||||
});
|
||||
|
||||
const emit = defineEmits(['click']);
|
||||
|
||||
const handleClick = () => {
|
||||
emit('click');
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CardLayout class="[&>div]:px-5 cursor-pointer" @click="handleClick">
|
||||
<div class="flex flex-col items-start gap-2">
|
||||
<div class="flex justify-between w-full items-center">
|
||||
<h3 class="text-n-slate-12 text-base font-medium">{{ title }}</h3>
|
||||
<Button
|
||||
xs
|
||||
slate
|
||||
ghost
|
||||
icon="i-lucide-chevron-right"
|
||||
@click.stop="handleClick"
|
||||
/>
|
||||
</div>
|
||||
<p class="text-n-slate-11 text-sm mb-0">{{ description }}</p>
|
||||
</div>
|
||||
|
||||
<ul class="flex flex-col items-start gap-3 mt-3">
|
||||
<li
|
||||
v-for="feature in features"
|
||||
:key="feature.id"
|
||||
class="flex items-center gap-3 text-sm"
|
||||
>
|
||||
<Icon
|
||||
:icon="feature.icon"
|
||||
class="text-n-slate-11 size-4 flex-shrink-0"
|
||||
/>
|
||||
{{ feature.label }}
|
||||
</li>
|
||||
</ul>
|
||||
</CardLayout>
|
||||
</template>
|
||||
@@ -0,0 +1,104 @@
|
||||
<script setup>
|
||||
import AssignmentPolicyCard from './AssignmentPolicyCard.vue';
|
||||
|
||||
const mockInboxes = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'Website Support',
|
||||
channel_type: 'Channel::WebWidget',
|
||||
inbox_type: 'Website',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Email Support',
|
||||
channel_type: 'Channel::Email',
|
||||
inbox_type: 'Email',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'WhatsApp Business',
|
||||
channel_type: 'Channel::Whatsapp',
|
||||
inbox_type: 'WhatsApp',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Facebook Messenger',
|
||||
channel_type: 'Channel::FacebookPage',
|
||||
inbox_type: 'Messenger',
|
||||
},
|
||||
];
|
||||
|
||||
const withCount = policy => ({
|
||||
...policy,
|
||||
assignedInboxCount: policy.inboxes.length,
|
||||
});
|
||||
|
||||
const policyA = withCount({
|
||||
id: 1,
|
||||
name: 'Website & Email',
|
||||
description: 'Distributes conversations evenly among available agents',
|
||||
assignmentOrder: 'round_robin',
|
||||
conversationPriority: 'high',
|
||||
enabled: true,
|
||||
inboxes: [mockInboxes[0], mockInboxes[1]],
|
||||
isFetchingInboxes: false,
|
||||
});
|
||||
|
||||
const policyB = withCount({
|
||||
id: 2,
|
||||
name: 'WhatsApp & Messenger',
|
||||
description: 'Assigns based on capacity and workload',
|
||||
assignmentOrder: 'capacity_based',
|
||||
conversationPriority: 'medium',
|
||||
enabled: true,
|
||||
inboxes: [mockInboxes[2], mockInboxes[3]],
|
||||
isFetchingInboxes: false,
|
||||
});
|
||||
|
||||
const emptyPolicy = withCount({
|
||||
id: 3,
|
||||
name: 'No Inboxes Yet',
|
||||
description: 'Policy with no assigned inboxes',
|
||||
assignmentOrder: 'manual',
|
||||
conversationPriority: 'low',
|
||||
enabled: false,
|
||||
inboxes: [],
|
||||
isFetchingInboxes: false,
|
||||
});
|
||||
|
||||
const onEdit = id => console.log('Edit policy:', id);
|
||||
const onDelete = id => console.log('Delete policy:', id);
|
||||
const onFetch = () => console.log('Fetch inboxes');
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/AssignmentPolicyCard"
|
||||
:layout="{ type: 'grid', width: '1200px' }"
|
||||
>
|
||||
<Variant title="Three Cards (Two with inboxes, One empty)">
|
||||
<div class="p-4 bg-n-background">
|
||||
<div class="grid grid-cols-1 gap-4">
|
||||
<AssignmentPolicyCard
|
||||
v-bind="policyA"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-inboxes="onFetch"
|
||||
/>
|
||||
<AssignmentPolicyCard
|
||||
v-bind="policyB"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-inboxes="onFetch"
|
||||
/>
|
||||
<AssignmentPolicyCard
|
||||
v-bind="emptyPolicy"
|
||||
@edit="onEdit"
|
||||
@delete="onDelete"
|
||||
@fetch-inboxes="onFetch"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,133 @@
|
||||
<script setup>
|
||||
import { computed } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { getInboxIconByType } from 'dashboard/helper/inbox';
|
||||
import { formatToTitleCase } from 'dashboard/helper/commons';
|
||||
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import CardLayout from 'dashboard/components-next/CardLayout.vue';
|
||||
import CardPopover from '../components/CardPopover.vue';
|
||||
|
||||
const props = defineProps({
|
||||
id: { type: Number, required: true },
|
||||
name: { type: String, default: '' },
|
||||
description: { type: String, default: '' },
|
||||
assignmentOrder: { type: String, default: '' },
|
||||
conversationPriority: { type: String, default: '' },
|
||||
assignedInboxCount: { type: Number, default: 0 },
|
||||
enabled: { type: Boolean, default: false },
|
||||
inboxes: { type: Array, default: () => [] },
|
||||
isFetchingInboxes: { type: Boolean, default: false },
|
||||
});
|
||||
|
||||
const emit = defineEmits(['edit', 'delete', 'fetchInboxes']);
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const inboxes = computed(() => {
|
||||
return props.inboxes.map(inbox => {
|
||||
return {
|
||||
name: inbox.name,
|
||||
id: inbox.id,
|
||||
icon: getInboxIconByType(inbox.channelType, inbox.medium, 'line'),
|
||||
};
|
||||
});
|
||||
});
|
||||
|
||||
const order = computed(() => {
|
||||
return formatToTitleCase(props.assignmentOrder);
|
||||
});
|
||||
|
||||
const priority = computed(() => {
|
||||
return formatToTitleCase(props.conversationPriority);
|
||||
});
|
||||
|
||||
const handleEdit = () => {
|
||||
emit('edit', props.id);
|
||||
};
|
||||
|
||||
const handleDelete = () => {
|
||||
emit('delete', props.id);
|
||||
};
|
||||
|
||||
const handleFetchInboxes = () => {
|
||||
if (props.inboxes?.length > 0) return;
|
||||
emit('fetchInboxes', props.id);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<CardLayout class="[&>div]:px-5">
|
||||
<div class="flex flex-col gap-2 relative justify-between w-full">
|
||||
<div class="flex items-center gap-3 justify-between w-full">
|
||||
<div class="flex items-center gap-3">
|
||||
<h3 class="text-base font-medium text-n-slate-12 line-clamp-1">
|
||||
{{ name }}
|
||||
</h3>
|
||||
<div class="flex items-center gap-2">
|
||||
<div class="flex items-center rounded-md bg-n-alpha-2 h-6 px-2">
|
||||
<span
|
||||
class="text-xs"
|
||||
:class="enabled ? 'text-n-teal-11' : 'text-n-slate-12'"
|
||||
>
|
||||
{{
|
||||
enabled
|
||||
? t(
|
||||
'ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.INDEX.CARD.ACTIVE'
|
||||
)
|
||||
: t(
|
||||
'ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.INDEX.CARD.INACTIVE'
|
||||
)
|
||||
}}
|
||||
</span>
|
||||
</div>
|
||||
<CardPopover
|
||||
:title="
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.INDEX.CARD.POPOVER'
|
||||
)
|
||||
"
|
||||
icon="i-lucide-inbox"
|
||||
:count="assignedInboxCount"
|
||||
:items="inboxes"
|
||||
:is-fetching="isFetchingInboxes"
|
||||
@fetch="handleFetchInboxes"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<div class="flex items-center gap-2">
|
||||
<Button
|
||||
:label="
|
||||
t('ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.INDEX.CARD.EDIT')
|
||||
"
|
||||
sm
|
||||
slate
|
||||
link
|
||||
class="px-2"
|
||||
@click="handleEdit"
|
||||
/>
|
||||
<div v-if="order" class="w-px h-2.5 bg-n-slate-5" />
|
||||
<Button icon="i-lucide-trash" sm slate ghost @click="handleDelete" />
|
||||
</div>
|
||||
</div>
|
||||
<p class="text-n-slate-11 text-sm line-clamp-1 mb-0 py-1">
|
||||
{{ description }}
|
||||
</p>
|
||||
<div class="flex items-center gap-3 py-1.5">
|
||||
<span v-if="order" class="text-n-slate-11 text-sm">
|
||||
{{
|
||||
`${t('ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.INDEX.CARD.ORDER')}:`
|
||||
}}
|
||||
<span class="text-n-slate-12">{{ order }}</span>
|
||||
</span>
|
||||
<div v-if="order" class="w-px h-3 bg-n-strong" />
|
||||
<span v-if="priority" class="text-n-slate-11 text-sm">
|
||||
{{
|
||||
`${t('ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.INDEX.CARD.PRIORITY')}:`
|
||||
}}
|
||||
<span class="text-n-slate-12">{{ priority }}</span>
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</CardLayout>
|
||||
</template>
|
||||
@@ -0,0 +1,169 @@
|
||||
<script setup>
|
||||
import { computed, ref } from 'vue';
|
||||
import { useToggle, useWindowSize, useElementBounding } from '@vueuse/core';
|
||||
import { vOnClickOutside } from '@vueuse/components';
|
||||
import { picoSearch } from '@scmmishra/pico-search';
|
||||
|
||||
import Avatar from 'next/avatar/Avatar.vue';
|
||||
import Icon from 'dashboard/components-next/icon/Icon.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
|
||||
const props = defineProps({
|
||||
label: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
searchPlaceholder: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
items: {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['add']);
|
||||
|
||||
const BUFFER_SPACE = 20;
|
||||
|
||||
const [showPopover, togglePopover] = useToggle();
|
||||
const buttonRef = ref();
|
||||
const dropdownRef = ref();
|
||||
|
||||
const searchValue = ref('');
|
||||
|
||||
const { width: windowWidth, height: windowHeight } = useWindowSize();
|
||||
const {
|
||||
top: buttonTop,
|
||||
left: buttonLeft,
|
||||
width: buttonWidth,
|
||||
height: buttonHeight,
|
||||
} = useElementBounding(buttonRef);
|
||||
const { width: dropdownWidth, height: dropdownHeight } =
|
||||
useElementBounding(dropdownRef);
|
||||
|
||||
const filteredItems = computed(() => {
|
||||
if (!searchValue.value) return props.items;
|
||||
const query = searchValue.value.toLowerCase();
|
||||
|
||||
return picoSearch(props.items, query, ['name']);
|
||||
});
|
||||
|
||||
const handleAdd = item => {
|
||||
emit('add', item);
|
||||
togglePopover(false);
|
||||
};
|
||||
|
||||
const shouldShowAbove = computed(() => {
|
||||
if (!buttonRef.value || !dropdownRef.value) return false;
|
||||
const spaceBelow =
|
||||
windowHeight.value - (buttonTop.value + buttonHeight.value);
|
||||
const spaceAbove = buttonTop.value;
|
||||
return (
|
||||
spaceBelow < dropdownHeight.value + BUFFER_SPACE && spaceAbove > spaceBelow
|
||||
);
|
||||
});
|
||||
|
||||
const shouldAlignRight = computed(() => {
|
||||
if (!buttonRef.value || !dropdownRef.value) return false;
|
||||
const spaceRight = windowWidth.value - buttonLeft.value;
|
||||
const spaceLeft = buttonLeft.value + buttonWidth.value;
|
||||
|
||||
return (
|
||||
spaceRight < dropdownWidth.value + BUFFER_SPACE && spaceLeft > spaceRight
|
||||
);
|
||||
});
|
||||
|
||||
const handleClickOutside = () => {
|
||||
if (showPopover.value) {
|
||||
togglePopover(false);
|
||||
}
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
v-on-click-outside="handleClickOutside"
|
||||
class="relative flex items-center group"
|
||||
>
|
||||
<Button
|
||||
ref="buttonRef"
|
||||
slate
|
||||
type="button"
|
||||
icon="i-lucide-plus"
|
||||
sm
|
||||
:label="label"
|
||||
@click="togglePopover(!showPopover)"
|
||||
/>
|
||||
<div
|
||||
v-if="showPopover"
|
||||
ref="dropdownRef"
|
||||
class="z-50 flex flex-col items-start absolute bg-n-alpha-3 backdrop-blur-[50px] border-0 gap-4 outline outline-1 outline-n-weak rounded-xl max-w-96 min-w-80 max-h-[20rem] overflow-y-auto py-2"
|
||||
:class="[
|
||||
shouldShowAbove ? 'bottom-full mb-2' : 'top-full mt-2',
|
||||
shouldAlignRight ? 'right-0' : 'left-0',
|
||||
]"
|
||||
>
|
||||
<div class="flex flex-col divide-y divide-n-slate-4 w-full">
|
||||
<Input
|
||||
v-model="searchValue"
|
||||
:placeholder="searchPlaceholder"
|
||||
custom-input-class="bg-transparent !outline-none w-full ltr:!pl-10 rtl:!pr-10 h-10"
|
||||
>
|
||||
<template #prefix>
|
||||
<Icon
|
||||
icon="i-lucide-search"
|
||||
class="absolute -translate-y-1/2 text-n-slate-11 size-4 top-1/2 ltr:left-3 rtl:right-3"
|
||||
/>
|
||||
</template>
|
||||
</Input>
|
||||
|
||||
<div
|
||||
v-for="item in filteredItems"
|
||||
:key="item.id"
|
||||
class="flex gap-3 min-w-0 w-full py-4 px-3 hover:bg-n-alpha-2 cursor-pointer"
|
||||
:class="{ 'items-center': item.color, 'items-start': !item.color }"
|
||||
@click="handleAdd(item)"
|
||||
>
|
||||
<Icon
|
||||
v-if="item.icon"
|
||||
:icon="item.icon"
|
||||
class="size-4 text-n-slate-12 flex-shrink-0 mt-0.5"
|
||||
/>
|
||||
<span
|
||||
v-else-if="item.color"
|
||||
:style="{ backgroundColor: item.color }"
|
||||
class="size-3 rounded-sm"
|
||||
/>
|
||||
<Avatar
|
||||
v-else
|
||||
:title="item.name"
|
||||
:src="item.avatarUrl"
|
||||
:name="item.name"
|
||||
:size="20"
|
||||
rounded-full
|
||||
/>
|
||||
<div class="flex flex-col items-start gap-2 min-w-0 flex-1">
|
||||
<div class="flex items-center gap-1 min-w-0 w-full">
|
||||
<span
|
||||
:title="item.name || item.title"
|
||||
class="text-sm text-n-slate-12 truncate min-w-0 flex-1"
|
||||
>
|
||||
{{ item.name || item.title }}
|
||||
</span>
|
||||
</div>
|
||||
<span
|
||||
v-if="item.email || item.phoneNumber"
|
||||
:title="item.email || item.phoneNumber"
|
||||
class="text-sm text-n-slate-11 truncate min-w-0 w-full block"
|
||||
>
|
||||
{{ item.email || item.phoneNumber }}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,127 @@
|
||||
<script setup>
|
||||
import { computed, watch } from 'vue';
|
||||
import { useVuelidate } from '@vuelidate/core';
|
||||
import { required, minLength } from '@vuelidate/validators';
|
||||
|
||||
import WithLabel from 'v3/components/Form/WithLabel.vue';
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
import Switch from 'dashboard/components-next/switch/Switch.vue';
|
||||
|
||||
defineProps({
|
||||
nameLabel: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
namePlaceholder: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
descriptionLabel: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
descriptionPlaceholder: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
statusLabel: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
statusPlaceholder: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['validationChange']);
|
||||
|
||||
const policyName = defineModel('policyName', {
|
||||
type: String,
|
||||
default: '',
|
||||
});
|
||||
|
||||
const description = defineModel('description', {
|
||||
type: String,
|
||||
default: '',
|
||||
});
|
||||
|
||||
const enabled = defineModel('enabled', {
|
||||
type: Boolean,
|
||||
default: true,
|
||||
});
|
||||
|
||||
const validationRules = {
|
||||
policyName: { required, minLength: minLength(1) },
|
||||
description: { required, minLength: minLength(1) },
|
||||
};
|
||||
|
||||
const v$ = useVuelidate(validationRules, { policyName, description });
|
||||
|
||||
const isValid = computed(() => !v$.value.$invalid);
|
||||
|
||||
watch(
|
||||
isValid,
|
||||
() => {
|
||||
emit('validationChange', {
|
||||
isValid: isValid.value,
|
||||
section: 'baseInfo',
|
||||
});
|
||||
},
|
||||
{ immediate: true }
|
||||
);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="flex flex-col gap-4 pb-4">
|
||||
<!-- Policy Name Field -->
|
||||
<div class="flex items-center gap-6">
|
||||
<WithLabel
|
||||
:label="nameLabel"
|
||||
name="policyName"
|
||||
class="flex items-center w-full [&>label]:min-w-[120px]"
|
||||
>
|
||||
<div class="flex-1">
|
||||
<Input
|
||||
v-model="policyName"
|
||||
type="text"
|
||||
:placeholder="namePlaceholder"
|
||||
/>
|
||||
</div>
|
||||
</WithLabel>
|
||||
</div>
|
||||
|
||||
<!-- Description Field -->
|
||||
<div class="flex items-center gap-6">
|
||||
<WithLabel
|
||||
:label="descriptionLabel"
|
||||
name="description"
|
||||
class="flex items-center w-full [&>label]:min-w-[120px]"
|
||||
>
|
||||
<div class="flex-1">
|
||||
<Input
|
||||
v-model="description"
|
||||
type="text"
|
||||
:placeholder="descriptionPlaceholder"
|
||||
/>
|
||||
</div>
|
||||
</WithLabel>
|
||||
</div>
|
||||
|
||||
<!-- Status Field -->
|
||||
<div v-if="statusLabel" class="flex items-center gap-6">
|
||||
<WithLabel
|
||||
:label="statusLabel"
|
||||
name="enabled"
|
||||
class="flex items-center w-full [&>label]:min-w-[120px]"
|
||||
>
|
||||
<div class="flex items-center gap-2">
|
||||
<Switch v-model="enabled" />
|
||||
<span class="text-sm text-n-slate-11">
|
||||
{{ statusPlaceholder }}
|
||||
</span>
|
||||
</div>
|
||||
</WithLabel>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,121 @@
|
||||
<script setup>
|
||||
import { useToggle } from '@vueuse/core';
|
||||
import { vOnClickOutside } from '@vueuse/components';
|
||||
|
||||
import Avatar from 'next/avatar/Avatar.vue';
|
||||
import Icon from 'dashboard/components-next/icon/Icon.vue';
|
||||
import Spinner from 'dashboard/components-next/spinner/Spinner.vue';
|
||||
|
||||
defineProps({
|
||||
count: {
|
||||
type: Number,
|
||||
default: 0,
|
||||
},
|
||||
title: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
icon: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
items: {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
},
|
||||
isFetching: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['fetch']);
|
||||
|
||||
const [showPopover, togglePopover] = useToggle();
|
||||
|
||||
const handleButtonClick = () => {
|
||||
emit('fetch');
|
||||
togglePopover(!showPopover.value);
|
||||
};
|
||||
|
||||
const handleClickOutside = () => {
|
||||
if (showPopover.value) {
|
||||
togglePopover(false);
|
||||
}
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
v-on-click-outside="handleClickOutside"
|
||||
class="relative flex items-center group"
|
||||
>
|
||||
<button
|
||||
v-if="count"
|
||||
class="h-6 px-2 rounded-md bg-n-alpha-2 gap-1.5 flex items-center"
|
||||
@click="handleButtonClick()"
|
||||
>
|
||||
<Icon :icon="icon" class="size-3.5 text-n-slate-12" />
|
||||
<span class="text-n-slate-12 text-sm">
|
||||
{{ count }}
|
||||
</span>
|
||||
</button>
|
||||
<div
|
||||
v-if="showPopover"
|
||||
class="top-full mt-1 ltr:left-0 rtl:right-0 z-50 flex flex-col items-start absolute bg-n-alpha-3 backdrop-blur-[50px] border-0 gap-4 outline outline-1 outline-n-weak p-3 rounded-xl max-w-96 min-w-80 max-h-[20rem] overflow-y-auto"
|
||||
>
|
||||
<div class="flex items-center gap-2.5 pb-2">
|
||||
<Icon :icon="icon" class="size-3.5" />
|
||||
<span class="text-sm text-n-slate-12 font-medium">{{ title }}</span>
|
||||
</div>
|
||||
|
||||
<div
|
||||
v-if="isFetching"
|
||||
class="flex items-center justify-center py-3 w-full text-n-slate-11"
|
||||
>
|
||||
<Spinner />
|
||||
</div>
|
||||
|
||||
<div v-else class="flex flex-col gap-4 w-full">
|
||||
<div
|
||||
v-for="item in items"
|
||||
:key="item.id"
|
||||
class="flex items-center justify-between gap-2 min-w-0 w-full"
|
||||
>
|
||||
<div class="flex items-center gap-2 min-w-0 w-full">
|
||||
<Icon
|
||||
v-if="item.icon"
|
||||
:icon="item.icon"
|
||||
class="size-4 text-n-slate-12 flex-shrink-0"
|
||||
/>
|
||||
<Avatar
|
||||
v-else
|
||||
:title="item.name"
|
||||
:src="item.avatarUrl"
|
||||
:name="item.name"
|
||||
:size="20"
|
||||
rounded-full
|
||||
/>
|
||||
<div class="flex items-center gap-1 min-w-0 flex-1">
|
||||
<span
|
||||
:title="item.name"
|
||||
class="text-sm text-n-slate-12 truncate min-w-0"
|
||||
>
|
||||
{{ item.name }}
|
||||
</span>
|
||||
<span
|
||||
v-if="item.id"
|
||||
class="text-sm text-n-slate-11 flex-shrink-0"
|
||||
>
|
||||
{{ `#${item.id}` }}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
<span v-if="item.email" class="text-sm text-n-slate-11 flex-shrink-0">
|
||||
{{ item.email }}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,90 @@
|
||||
<script setup>
|
||||
import Avatar from 'next/avatar/Avatar.vue';
|
||||
import Icon from 'dashboard/components-next/icon/Icon.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import Spinner from 'dashboard/components-next/spinner/Spinner.vue';
|
||||
|
||||
defineProps({
|
||||
items: {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
},
|
||||
isFetching: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
emptyStateMessage: {
|
||||
type: String,
|
||||
default: '',
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['delete']);
|
||||
|
||||
const handleDelete = itemId => {
|
||||
emit('delete', itemId);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
v-if="isFetching"
|
||||
class="flex items-center justify-center py-3 w-full text-n-slate-11"
|
||||
>
|
||||
<Spinner />
|
||||
</div>
|
||||
<div
|
||||
v-else-if="items.length === 0 && emptyStateMessage"
|
||||
class="custom-dashed-border flex items-center justify-center py-6 w-full"
|
||||
>
|
||||
<span class="text-sm text-n-slate-11">
|
||||
{{ emptyStateMessage }}
|
||||
</span>
|
||||
</div>
|
||||
<div v-else class="flex flex-col divide-y divide-n-weak">
|
||||
<div
|
||||
v-for="item in items"
|
||||
:key="item.id"
|
||||
class="grid grid-cols-4 items-center gap-3 min-w-0 w-full justify-between h-[3.25rem] ltr:pr-2 rtl:pl-2"
|
||||
>
|
||||
<div class="flex items-center gap-2 col-span-2">
|
||||
<Icon
|
||||
v-if="item.icon"
|
||||
:icon="item.icon"
|
||||
class="size-4 text-n-slate-12 flex-shrink-0"
|
||||
/>
|
||||
<Avatar
|
||||
v-else
|
||||
:title="item.name"
|
||||
:src="item.avatarUrl"
|
||||
:name="item.name"
|
||||
:size="20"
|
||||
rounded-full
|
||||
/>
|
||||
<span class="text-sm text-n-slate-12 truncate min-w-0">
|
||||
{{ item.name }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="flex items-start gap-2 col-span-1">
|
||||
<span
|
||||
:title="item.email || item.phoneNumber"
|
||||
class="text-sm text-n-slate-12 truncate min-w-0"
|
||||
>
|
||||
{{ item.email || item.phoneNumber }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="col-span-1 justify-end flex items-center">
|
||||
<Button
|
||||
icon="i-lucide-trash"
|
||||
slate
|
||||
ghost
|
||||
sm
|
||||
type="button"
|
||||
@click="handleDelete(item.id)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,149 @@
|
||||
<script setup>
|
||||
import { computed, ref, onMounted } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import AddDataDropdown from 'dashboard/components-next/AssignmentPolicy/components/AddDataDropdown.vue';
|
||||
import LabelItem from 'dashboard/components-next/Label/LabelItem.vue';
|
||||
import DurationInput from 'dashboard/components-next/input/DurationInput.vue';
|
||||
import { DURATION_UNITS } from 'dashboard/components-next/input/constants';
|
||||
|
||||
const props = defineProps({
|
||||
tagsList: {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
},
|
||||
});
|
||||
|
||||
const excludedLabels = defineModel('excludedLabels', {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
});
|
||||
|
||||
const excludeOlderThanMinutes = defineModel('excludeOlderThanMinutes', {
|
||||
type: Number,
|
||||
default: 10,
|
||||
});
|
||||
|
||||
// Duration limits: 10 minutes to 999 days (in minutes)
|
||||
const MIN_DURATION_MINUTES = 10;
|
||||
const MAX_DURATION_MINUTES = 1438560; // 999 days * 24 hours * 60 minutes
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const hoveredLabel = ref(null);
|
||||
const windowUnit = ref(DURATION_UNITS.MINUTES);
|
||||
|
||||
const addedTags = computed(() =>
|
||||
props.tagsList
|
||||
.filter(label => excludedLabels.value.includes(label.name))
|
||||
.map(label => ({ id: label.id, title: label.name, ...label }))
|
||||
);
|
||||
|
||||
const filteredTags = computed(() =>
|
||||
props.tagsList.filter(
|
||||
label => !addedTags.value.some(tag => tag.id === label.id)
|
||||
)
|
||||
);
|
||||
|
||||
const detectUnit = minutes => {
|
||||
const m = Number(minutes) || 0;
|
||||
if (m === 0) return DURATION_UNITS.MINUTES;
|
||||
if (m % (24 * 60) === 0) return DURATION_UNITS.DAYS;
|
||||
if (m % 60 === 0) return DURATION_UNITS.HOURS;
|
||||
return DURATION_UNITS.MINUTES;
|
||||
};
|
||||
|
||||
const onClickAddTag = tag => {
|
||||
excludedLabels.value = [...excludedLabels.value, tag.name];
|
||||
};
|
||||
|
||||
const onClickRemoveTag = tag => {
|
||||
excludedLabels.value = excludedLabels.value.filter(
|
||||
name => name !== tag.title
|
||||
);
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
windowUnit.value = detectUnit(excludeOlderThanMinutes.value);
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="py-4 flex-col flex gap-6">
|
||||
<div class="flex flex-col items-start gap-1 py-1">
|
||||
<label class="text-sm font-medium text-n-slate-12 py-1">
|
||||
{{
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.FORM.EXCLUSION_RULES.LABEL'
|
||||
)
|
||||
}}
|
||||
</label>
|
||||
<p class="mb-0 text-n-slate-11 text-sm">
|
||||
{{
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.FORM.EXCLUSION_RULES.DESCRIPTION'
|
||||
)
|
||||
}}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div class="flex flex-col items-start gap-4">
|
||||
<label class="text-sm font-medium text-n-slate-12 py-1">
|
||||
{{
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.FORM.EXCLUSION_RULES.TAGS.LABEL'
|
||||
)
|
||||
}}
|
||||
</label>
|
||||
<div
|
||||
class="flex items-start gap-2 flex-wrap"
|
||||
@mouseleave="hoveredLabel = null"
|
||||
>
|
||||
<LabelItem
|
||||
v-for="tag in addedTags"
|
||||
:key="tag.id"
|
||||
:label="tag"
|
||||
:is-hovered="hoveredLabel === tag.id"
|
||||
class="h-8"
|
||||
@remove="onClickRemoveTag"
|
||||
@hover="hoveredLabel = tag.id"
|
||||
/>
|
||||
<AddDataDropdown
|
||||
:label="
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.FORM.EXCLUSION_RULES.TAGS.ADD_TAG'
|
||||
)
|
||||
"
|
||||
:search-placeholder="
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.FORM.EXCLUSION_RULES.TAGS.DROPDOWN.SEARCH_PLACEHOLDER'
|
||||
)
|
||||
"
|
||||
:items="filteredTags"
|
||||
class="[&>button]:!text-n-blue-text [&>div]:min-w-64"
|
||||
@add="onClickAddTag"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="flex flex-col items-start gap-4">
|
||||
<label class="text-sm font-medium text-n-slate-12 py-1">
|
||||
{{
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY.FORM.EXCLUSION_RULES.DURATION.LABEL'
|
||||
)
|
||||
}}
|
||||
</label>
|
||||
<div
|
||||
class="flex items-center gap-2 flex-1 [&>select]:!bg-n-alpha-2 [&>select]:!outline-none [&>select]:hover:brightness-110"
|
||||
>
|
||||
<!-- allow 10 mins to 999 days -->
|
||||
<DurationInput
|
||||
v-model:unit="windowUnit"
|
||||
v-model:model-value="excludeOlderThanMinutes"
|
||||
:min="MIN_DURATION_MINUTES"
|
||||
:max="MAX_DURATION_MINUTES"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,86 @@
|
||||
<script setup>
|
||||
import { ref, onMounted } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
import DurationInput from 'dashboard/components-next/input/DurationInput.vue';
|
||||
import { DURATION_UNITS } from 'dashboard/components-next/input/constants';
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const fairDistributionLimit = defineModel('fairDistributionLimit', {
|
||||
type: Number,
|
||||
default: 100,
|
||||
set(value) {
|
||||
return Number(value) || 0;
|
||||
},
|
||||
});
|
||||
|
||||
const fairDistributionWindow = defineModel('fairDistributionWindow', {
|
||||
type: Number,
|
||||
default: 3600,
|
||||
set(value) {
|
||||
return Number(value) || 0;
|
||||
},
|
||||
});
|
||||
|
||||
const windowUnit = ref(DURATION_UNITS.MINUTES);
|
||||
|
||||
const detectUnit = minutes => {
|
||||
const m = Number(minutes) || 0;
|
||||
if (m === 0) return DURATION_UNITS.MINUTES;
|
||||
if (m % (24 * 60) === 0) return DURATION_UNITS.DAYS;
|
||||
if (m % 60 === 0) return DURATION_UNITS.HOURS;
|
||||
return DURATION_UNITS.MINUTES;
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
windowUnit.value = detectUnit(fairDistributionWindow.value);
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
class="flex items-start xl:items-center flex-col md:flex-row gap-4 lg:gap-3 bg-n-solid-1 p-4 outline outline-1 outline-n-weak rounded-xl"
|
||||
>
|
||||
<div class="flex items-center gap-3">
|
||||
<label class="text-sm font-medium text-n-slate-12">
|
||||
{{
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.FORM.FAIR_DISTRIBUTION.INPUT_MAX'
|
||||
)
|
||||
}}
|
||||
</label>
|
||||
<div class="flex-1">
|
||||
<Input
|
||||
v-model="fairDistributionLimit"
|
||||
type="number"
|
||||
placeholder="100"
|
||||
max="100000"
|
||||
class="w-full"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="flex sm:flex-row flex-col items-start sm:items-center gap-4">
|
||||
<label class="text-sm font-medium text-n-slate-12">
|
||||
{{
|
||||
t(
|
||||
'ASSIGNMENT_POLICY.AGENT_ASSIGNMENT_POLICY.FORM.FAIR_DISTRIBUTION.DURATION'
|
||||
)
|
||||
}}
|
||||
</label>
|
||||
|
||||
<div
|
||||
class="flex items-center gap-2 flex-1 [&>select]:!bg-n-alpha-2 [&>select]:!outline-none [&>select]:hover:brightness-110"
|
||||
>
|
||||
<!-- allow 10 mins to 999 days -->
|
||||
<DurationInput
|
||||
v-model:model-value="fairDistributionWindow"
|
||||
v-model:unit="windowUnit"
|
||||
:min="10"
|
||||
:max="1438560"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,177 @@
|
||||
<script setup>
|
||||
import { computed } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import Spinner from 'dashboard/components-next/spinner/Spinner.vue';
|
||||
import AddDataDropdown from 'dashboard/components-next/AssignmentPolicy/components/AddDataDropdown.vue';
|
||||
|
||||
const props = defineProps({
|
||||
inboxList: {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
},
|
||||
isFetching: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['delete', 'add', 'update']);
|
||||
|
||||
const inboxCapacityLimits = defineModel('inboxCapacityLimits', {
|
||||
type: Array,
|
||||
default: () => [],
|
||||
});
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const BASE_KEY = 'ASSIGNMENT_POLICY.AGENT_CAPACITY_POLICY';
|
||||
const DEFAULT_CONVERSATION_LIMIT = 10;
|
||||
const MIN_CONVERSATION_LIMIT = 1;
|
||||
const MAX_CONVERSATION_LIMIT = 100000;
|
||||
|
||||
const selectedInboxIds = computed(
|
||||
() => new Set(inboxCapacityLimits.value.map(limit => limit.inboxId))
|
||||
);
|
||||
|
||||
const availableInboxes = computed(() =>
|
||||
props.inboxList.filter(
|
||||
inbox => inbox && !selectedInboxIds.value.has(inbox.id)
|
||||
)
|
||||
);
|
||||
|
||||
const isLimitValid = limit => {
|
||||
return (
|
||||
limit.conversationLimit >= MIN_CONVERSATION_LIMIT &&
|
||||
limit.conversationLimit <= MAX_CONVERSATION_LIMIT
|
||||
);
|
||||
};
|
||||
|
||||
const inboxMap = computed(
|
||||
() => new Map(props.inboxList.map(inbox => [inbox.id, inbox]))
|
||||
);
|
||||
|
||||
const handleAddInbox = inbox => {
|
||||
emit('add', {
|
||||
inboxId: inbox.id,
|
||||
conversationLimit: DEFAULT_CONVERSATION_LIMIT,
|
||||
});
|
||||
};
|
||||
|
||||
const handleRemoveLimit = limitId => {
|
||||
emit('delete', limitId);
|
||||
};
|
||||
|
||||
const handleLimitChange = limit => {
|
||||
if (isLimitValid(limit)) {
|
||||
emit('update', limit);
|
||||
}
|
||||
};
|
||||
|
||||
const getInboxName = inboxId => {
|
||||
return inboxMap.value.get(inboxId)?.name || '';
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="py-4 flex-col flex gap-3">
|
||||
<div class="flex items-center w-full gap-8 justify-between pt-1 pb-3">
|
||||
<label class="text-sm font-medium text-n-slate-12">
|
||||
{{ t(`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.LABEL`) }}
|
||||
</label>
|
||||
|
||||
<AddDataDropdown
|
||||
:label="t(`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.ADD_BUTTON`)"
|
||||
:search-placeholder="
|
||||
t(`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.FIELD.SELECT_INBOX`)
|
||||
"
|
||||
:items="availableInboxes"
|
||||
@add="handleAddInbox"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div
|
||||
v-if="isFetching"
|
||||
class="flex items-center justify-center py-3 w-full text-n-slate-11"
|
||||
>
|
||||
<Spinner />
|
||||
</div>
|
||||
|
||||
<div
|
||||
v-else-if="!inboxCapacityLimits.length"
|
||||
class="custom-dashed-border flex items-center justify-center py-6 w-full"
|
||||
>
|
||||
<span class="text-sm text-n-slate-11">
|
||||
{{ t(`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.EMPTY_STATE`) }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div v-else class="flex-col flex gap-3">
|
||||
<div
|
||||
v-for="(limit, index) in inboxCapacityLimits"
|
||||
:key="limit.id || `temp-${index}`"
|
||||
class="flex flex-col xs:flex-row items-stretch gap-3"
|
||||
>
|
||||
<div
|
||||
class="flex items-center rounded-lg outline-1 outline cursor-not-allowed text-n-slate-11 outline-n-weak py-2.5 px-3 text-sm w-full min-w-0"
|
||||
:title="getInboxName(limit.inboxId)"
|
||||
>
|
||||
<span class="truncate min-w-0">
|
||||
{{ getInboxName(limit.inboxId) }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="flex items-center gap-3 w-full xs:w-auto">
|
||||
<div
|
||||
class="py-2.5 px-3 rounded-lg gap-2 outline outline-1 flex-1 xs:flex-shrink-0 flex items-center min-w-0"
|
||||
:class="[
|
||||
!isLimitValid(limit) ? 'outline-n-ruby-8' : 'outline-n-weak',
|
||||
]"
|
||||
>
|
||||
<label
|
||||
class="text-sm text-n-slate-12 ltr:pr-2 rtl:pl-2 truncate min-w-0 flex-shrink"
|
||||
:title="
|
||||
t(
|
||||
`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.FIELD.MAX_CONVERSATIONS`
|
||||
)
|
||||
"
|
||||
>
|
||||
{{
|
||||
t(
|
||||
`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.FIELD.MAX_CONVERSATIONS`
|
||||
)
|
||||
}}
|
||||
</label>
|
||||
|
||||
<div class="h-5 w-px bg-n-weak" />
|
||||
|
||||
<input
|
||||
v-model.number="limit.conversationLimit"
|
||||
type="number"
|
||||
:min="MIN_CONVERSATION_LIMIT"
|
||||
:max="MAX_CONVERSATION_LIMIT"
|
||||
class="reset-base bg-transparent focus:outline-none min-w-16 w-24 text-sm flex-shrink-0"
|
||||
:class="[
|
||||
!isLimitValid(limit)
|
||||
? 'placeholder:text-n-ruby-9 !text-n-ruby-9'
|
||||
: 'placeholder:text-n-slate-10 text-n-slate-12',
|
||||
]"
|
||||
:placeholder="
|
||||
t(`${BASE_KEY}.FORM.INBOX_CAPACITY_LIMIT.FIELD.SET_LIMIT`)
|
||||
"
|
||||
@blur="handleLimitChange(limit)"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<Button
|
||||
type="button"
|
||||
slate
|
||||
icon="i-lucide-trash"
|
||||
class="flex-shrink-0"
|
||||
@click="handleRemoveLimit(limit.id)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,60 @@
|
||||
<script setup>
|
||||
const props = defineProps({
|
||||
id: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
label: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
description: {
|
||||
type: String,
|
||||
required: true,
|
||||
},
|
||||
isActive: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['select']);
|
||||
|
||||
const handleChange = () => {
|
||||
if (!props.isActive) {
|
||||
emit('select', props.id);
|
||||
}
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
class="relative cursor-pointer rounded-xl outline outline-1 p-4 transition-all duration-200 bg-n-solid-1 py-4 ltr:pl-4 rtl:pr-4 ltr:pr-6 rtl:pl-6"
|
||||
:class="[
|
||||
isActive ? 'outline-n-blue-9' : 'outline-n-weak hover:outline-n-strong',
|
||||
]"
|
||||
@click="handleChange"
|
||||
>
|
||||
<div class="absolute top-4 right-4">
|
||||
<input
|
||||
:id="`${id}`"
|
||||
:checked="isActive"
|
||||
:value="id"
|
||||
:name="id"
|
||||
type="radio"
|
||||
class="h-4 w-4 border-n-slate-6 text-n-brand focus:ring-n-brand focus:ring-offset-0"
|
||||
@change="handleChange"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<!-- Content -->
|
||||
<div class="flex flex-col gap-3 items-start">
|
||||
<h3 class="text-sm font-medium text-n-slate-12">
|
||||
{{ label }}
|
||||
</h3>
|
||||
<p class="text-sm text-n-slate-11">
|
||||
{{ description }}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,92 @@
|
||||
<script setup>
|
||||
import AddDataDropdown from '../AddDataDropdown.vue';
|
||||
|
||||
const mockInboxes = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'Website Support',
|
||||
email: 'support@company.com',
|
||||
icon: 'i-lucide-globe',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Email Support',
|
||||
email: 'help@company.com',
|
||||
icon: 'i-lucide-mail',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'WhatsApp Business',
|
||||
phoneNumber: '+1 555-0123',
|
||||
icon: 'i-lucide-message-circle',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Facebook Messenger',
|
||||
email: 'messenger@company.com',
|
||||
icon: 'i-lucide-facebook',
|
||||
},
|
||||
{
|
||||
id: 5,
|
||||
name: 'Twitter DM',
|
||||
email: 'twitter@company.com',
|
||||
icon: 'i-lucide-twitter',
|
||||
},
|
||||
];
|
||||
|
||||
const mockTags = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'urgent',
|
||||
color: '#ff4757',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'bug',
|
||||
color: '#ff6b6b',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'feature-request',
|
||||
color: '#4834d4',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'documentation',
|
||||
color: '#26de81',
|
||||
},
|
||||
];
|
||||
|
||||
const handleAdd = item => {
|
||||
console.log('Add item:', item);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/AddDataDropdown"
|
||||
:layout="{ type: 'grid', width: '500px' }"
|
||||
>
|
||||
<Variant title="Basic Usage - Inboxes">
|
||||
<div class="p-8 bg-n-background flex gap-4 h-[400px] items-start">
|
||||
<AddDataDropdown
|
||||
label="Add Inbox"
|
||||
search-placeholder="Search inboxes..."
|
||||
:items="mockInboxes"
|
||||
@add="handleAdd"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Basic Usage - Tags">
|
||||
<div class="p-8 bg-n-background flex gap-4 h-[400px] items-start">
|
||||
<AddDataDropdown
|
||||
label="Add Tag"
|
||||
search-placeholder="Search tags..."
|
||||
:items="mockTags"
|
||||
@add="handleAdd"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,33 @@
|
||||
<script setup>
|
||||
import { ref } from 'vue';
|
||||
import BaseInfo from '../BaseInfo.vue';
|
||||
|
||||
const policyName = ref('Round Robin Policy');
|
||||
const description = ref(
|
||||
'Distributes conversations evenly among available agents'
|
||||
);
|
||||
const enabled = ref(true);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/BaseInfo"
|
||||
:layout="{ type: 'grid', width: '600px' }"
|
||||
>
|
||||
<Variant title="Basic Usage">
|
||||
<div class="p-8 bg-n-background">
|
||||
<BaseInfo
|
||||
v-model:policy-name="policyName"
|
||||
v-model:description="description"
|
||||
v-model:enabled="enabled"
|
||||
name-label="Policy Name"
|
||||
name-placeholder="Enter policy name"
|
||||
description-label="Description"
|
||||
description-placeholder="Enter policy description"
|
||||
status-label="Status"
|
||||
status-placeholder="Active"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,89 @@
|
||||
<script setup>
|
||||
import CardPopover from '../CardPopover.vue';
|
||||
|
||||
const mockItems = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'Website Support',
|
||||
icon: 'i-lucide-globe',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Email Support',
|
||||
icon: 'i-lucide-mail',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'WhatsApp Business',
|
||||
icon: 'i-lucide-message-circle',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Facebook Messenger',
|
||||
icon: 'i-lucide-facebook',
|
||||
},
|
||||
];
|
||||
|
||||
const mockUsers = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'John Smith',
|
||||
email: 'john.smith@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=1',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Sarah Johnson',
|
||||
email: 'sarah.johnson@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=2',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'Mike Chen',
|
||||
email: 'mike.chen@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=3',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Emily Davis',
|
||||
email: 'emily.davis@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=4',
|
||||
},
|
||||
{
|
||||
id: 5,
|
||||
name: 'Alex Rodriguez',
|
||||
email: 'alex.rodriguez@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=5',
|
||||
},
|
||||
];
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/CardPopover"
|
||||
:layout="{ type: 'grid', width: '800px' }"
|
||||
>
|
||||
<Variant title="Basic Usage">
|
||||
<div class="p-8 bg-n-background flex gap-4 h-96 items-start">
|
||||
<CardPopover
|
||||
:count="3"
|
||||
title="Added Inboxes"
|
||||
icon="i-lucide-inbox"
|
||||
:items="mockItems.slice(0, 3)"
|
||||
@fetch="() => console.log('Fetch triggered')"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
<Variant title="Basic Usage">
|
||||
<div class="p-8 bg-n-background flex gap-4 h-96 items-start">
|
||||
<CardPopover
|
||||
:count="3"
|
||||
title="Added Agents"
|
||||
icon="i-lucide-users-round"
|
||||
:items="mockUsers.slice(0, 3)"
|
||||
@fetch="() => console.log('Fetch triggered')"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,87 @@
|
||||
<script setup>
|
||||
import DataTable from '../DataTable.vue';
|
||||
|
||||
const mockItems = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'Website Support',
|
||||
email: 'support@company.com',
|
||||
icon: 'i-lucide-globe',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Email Support',
|
||||
email: 'help@company.com',
|
||||
icon: 'i-lucide-mail',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'WhatsApp Business',
|
||||
phoneNumber: '+1 555-0123',
|
||||
icon: 'i-lucide-message-circle',
|
||||
},
|
||||
];
|
||||
|
||||
const mockAgentList = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'John Doe',
|
||||
email: 'john.doe@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=1',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Jane Smith',
|
||||
email: 'jane.smith@example.com',
|
||||
avatarUrl: 'https://i.pravatar.cc/150?img=2',
|
||||
},
|
||||
];
|
||||
|
||||
const handleDelete = itemId => {
|
||||
console.log('Delete item:', itemId);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/DataTable"
|
||||
:layout="{ type: 'grid', width: '800px' }"
|
||||
>
|
||||
<Variant title="With Data">
|
||||
<div class="p-8 bg-n-background">
|
||||
<DataTable
|
||||
:items="mockItems"
|
||||
:is-fetching="false"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="With Agents">
|
||||
<div class="p-8 bg-n-background">
|
||||
<DataTable
|
||||
:items="mockAgentList"
|
||||
:is-fetching="false"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Loading State">
|
||||
<div class="p-8 bg-n-background">
|
||||
<DataTable :items="[]" is-fetching @delete="handleDelete" />
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Empty State">
|
||||
<div class="p-8 bg-n-background">
|
||||
<DataTable
|
||||
:items="[]"
|
||||
:is-fetching="false"
|
||||
empty-state-message="No items found"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,67 @@
|
||||
<script setup>
|
||||
import ExclusionRules from '../ExclusionRules.vue';
|
||||
import { ref } from 'vue';
|
||||
|
||||
const mockTagsList = [
|
||||
{
|
||||
id: 1,
|
||||
name: 'urgent',
|
||||
color: '#ff4757',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'bug',
|
||||
color: '#ff6b6b',
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'feature-request',
|
||||
color: '#4834d4',
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'documentation',
|
||||
color: '#26de81',
|
||||
},
|
||||
{
|
||||
id: 5,
|
||||
name: 'enhancement',
|
||||
color: '#2ed573',
|
||||
},
|
||||
{
|
||||
id: 6,
|
||||
name: 'question',
|
||||
color: '#ffa502',
|
||||
},
|
||||
{
|
||||
id: 7,
|
||||
name: 'duplicate',
|
||||
color: '#747d8c',
|
||||
},
|
||||
{
|
||||
id: 8,
|
||||
name: 'wontfix',
|
||||
color: '#57606f',
|
||||
},
|
||||
];
|
||||
|
||||
const excludedLabelsBasic = ref([]);
|
||||
const excludeOlderThanHoursBasic = ref(10);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/ExclusionRules"
|
||||
:layout="{ type: 'grid', width: '1200px' }"
|
||||
>
|
||||
<Variant title="Basic Usage">
|
||||
<div class="p-8 bg-n-background h-[600px]">
|
||||
<ExclusionRules
|
||||
v-model:excluded-labels="excludedLabelsBasic"
|
||||
v-model:exclude-older-than-minutes="excludeOlderThanHoursBasic"
|
||||
:tags-list="mockTagsList"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,25 @@
|
||||
<script setup>
|
||||
import { ref } from 'vue';
|
||||
import FairDistribution from '../FairDistribution.vue';
|
||||
|
||||
const fairDistributionLimit = ref(100);
|
||||
const fairDistributionWindow = ref(3600);
|
||||
const windowUnit = ref('minutes');
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/FairDistribution"
|
||||
:layout="{ type: 'grid', width: '800px' }"
|
||||
>
|
||||
<Variant title="Basic Usage">
|
||||
<div class="p-8 bg-n-background">
|
||||
<FairDistribution
|
||||
v-model:fair-distribution-limit="fairDistributionLimit"
|
||||
v-model:fair-distribution-window="fairDistributionWindow"
|
||||
v-model:window-unit="windowUnit"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,108 @@
|
||||
<script setup>
|
||||
import InboxCapacityLimits from '../InboxCapacityLimits.vue';
|
||||
import { ref } from 'vue';
|
||||
|
||||
const mockInboxList = [
|
||||
{
|
||||
value: 1,
|
||||
label: 'Website Support',
|
||||
icon: 'i-lucide-globe',
|
||||
},
|
||||
{
|
||||
value: 2,
|
||||
label: 'Email Support',
|
||||
icon: 'i-lucide-mail',
|
||||
},
|
||||
{
|
||||
value: 3,
|
||||
label: 'WhatsApp Business',
|
||||
icon: 'i-lucide-message-circle',
|
||||
},
|
||||
{
|
||||
value: 4,
|
||||
label: 'Facebook Messenger',
|
||||
icon: 'i-lucide-facebook',
|
||||
},
|
||||
{
|
||||
value: 5,
|
||||
label: 'Twitter DM',
|
||||
icon: 'i-lucide-twitter',
|
||||
},
|
||||
{
|
||||
value: 6,
|
||||
label: 'Telegram',
|
||||
icon: 'i-lucide-send',
|
||||
},
|
||||
];
|
||||
|
||||
const inboxCapacityLimitsEmpty = ref([]);
|
||||
const inboxCapacityLimitsNew = ref([
|
||||
{ id: 1, inboxId: 1, conversationLimit: 5 },
|
||||
{ inboxId: null, conversationLimit: null },
|
||||
]);
|
||||
|
||||
const handleDelete = id => {
|
||||
console.log('Delete capacity limit:', id);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/InboxCapacityLimits"
|
||||
:layout="{ type: 'grid', width: '900px' }"
|
||||
>
|
||||
<Variant title="Empty State">
|
||||
<div class="p-8 bg-n-background">
|
||||
<InboxCapacityLimits
|
||||
v-model:inbox-capacity-limits="inboxCapacityLimitsEmpty"
|
||||
:inbox-list="mockInboxList"
|
||||
:is-fetching="false"
|
||||
:is-updating="false"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Loading State">
|
||||
<div class="p-8 bg-n-background">
|
||||
<InboxCapacityLimits
|
||||
v-model:inbox-capacity-limits="inboxCapacityLimitsEmpty"
|
||||
:inbox-list="mockInboxList"
|
||||
is-fetching
|
||||
:is-updating="false"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="With New Row and existing data">
|
||||
<div class="p-8 bg-n-background">
|
||||
<InboxCapacityLimits
|
||||
v-model:inbox-capacity-limits="inboxCapacityLimitsNew"
|
||||
:inbox-list="mockInboxList"
|
||||
:is-fetching="false"
|
||||
:is-updating="false"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Interactive Demo">
|
||||
<div class="p-8 bg-n-background">
|
||||
<InboxCapacityLimits
|
||||
v-model:inbox-capacity-limits="inboxCapacityLimitsEmpty"
|
||||
:inbox-list="mockInboxList"
|
||||
:is-fetching="false"
|
||||
:is-updating="false"
|
||||
@delete="handleDelete"
|
||||
/>
|
||||
<div class="mt-4 p-4 bg-n-alpha-2 rounded-lg">
|
||||
<h4 class="text-sm font-medium mb-2">Current Limits:</h4>
|
||||
<pre class="text-xs">{{
|
||||
JSON.stringify(inboxCapacityLimitsEmpty, null, 2)
|
||||
}}</pre>
|
||||
</div>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -0,0 +1,61 @@
|
||||
<script setup>
|
||||
import { ref } from 'vue';
|
||||
import RadioCard from '../RadioCard.vue';
|
||||
|
||||
const selectedOption = ref('round_robin');
|
||||
|
||||
const handleSelect = value => {
|
||||
selectedOption.value = value;
|
||||
console.log('Selected:', value);
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Components/AgentManagementPolicy/RadioCard"
|
||||
:layout="{ type: 'grid', width: '600px' }"
|
||||
>
|
||||
<Variant title="Basic Usage">
|
||||
<div class="p-8 bg-n-background space-y-4">
|
||||
<RadioCard
|
||||
id="round_robin"
|
||||
label="Round Robin"
|
||||
description="Distributes conversations evenly among all available agents in a rotating manner"
|
||||
:is-active="selectedOption === 'round_robin'"
|
||||
@select="handleSelect"
|
||||
/>
|
||||
<RadioCard
|
||||
id="balanced"
|
||||
label="Balanced Assignment"
|
||||
description="Assigns conversations based on agent workload to maintain balance"
|
||||
:is-active="selectedOption === 'balanced'"
|
||||
@select="handleSelect"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Active State">
|
||||
<div class="p-8 bg-n-background">
|
||||
<RadioCard
|
||||
id="active_option"
|
||||
label="Active Option"
|
||||
description="This option is currently selected and active"
|
||||
is-active
|
||||
@select="handleSelect"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
|
||||
<Variant title="Inactive State">
|
||||
<div class="p-8 bg-n-background">
|
||||
<RadioCard
|
||||
id="inactive_option"
|
||||
label="Inactive Option"
|
||||
description="This option is not selected and can be clicked to activate"
|
||||
is-active
|
||||
@select="handleSelect"
|
||||
/>
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
@@ -86,8 +86,8 @@ const handleLabelAction = async ({ value }) => {
|
||||
}
|
||||
};
|
||||
|
||||
const handleRemoveLabel = labelId => {
|
||||
return handleLabelAction({ value: labelId });
|
||||
const handleRemoveLabel = label => {
|
||||
return handleLabelAction({ value: label.id });
|
||||
};
|
||||
|
||||
watch(
|
||||
|
||||
@@ -1,11 +1,13 @@
|
||||
<script setup>
|
||||
import { computed, useSlots } from 'vue';
|
||||
import { computed, useSlots, ref } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { useRoute } from 'vue-router';
|
||||
import { vOnClickOutside } from '@vueuse/components';
|
||||
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import Breadcrumb from 'dashboard/components-next/breadcrumb/Breadcrumb.vue';
|
||||
import ComposeConversation from 'dashboard/components-next/NewConversation/ComposeConversation.vue';
|
||||
import VoiceCallButton from 'dashboard/components-next/Contacts/VoiceCallButton.vue';
|
||||
|
||||
const props = defineProps({
|
||||
selectedContact: {
|
||||
@@ -24,6 +26,8 @@ const { t } = useI18n();
|
||||
const slots = useSlots();
|
||||
const route = useRoute();
|
||||
|
||||
const isContactSidebarOpen = ref(false);
|
||||
|
||||
const contactId = computed(() => route.params.contactId);
|
||||
|
||||
const selectedContactName = computed(() => {
|
||||
@@ -56,6 +60,15 @@ const handleBreadcrumbClick = () => {
|
||||
const toggleBlock = () => {
|
||||
emit('toggleBlock', isContactBlocked.value);
|
||||
};
|
||||
|
||||
const handleConversationSidebarToggle = () => {
|
||||
isContactSidebarOpen.value = !isContactSidebarOpen.value;
|
||||
};
|
||||
|
||||
const closeMobileSidebar = () => {
|
||||
if (!isContactSidebarOpen.value) return;
|
||||
isContactSidebarOpen.value = false;
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
@@ -67,7 +80,9 @@ const toggleBlock = () => {
|
||||
>
|
||||
<header class="sticky top-0 z-10 px-6 3xl:px-0">
|
||||
<div class="w-full mx-auto max-w-[40.625rem]">
|
||||
<div class="flex items-center justify-between w-full h-20 gap-2">
|
||||
<div
|
||||
class="flex flex-col xs:flex-row items-start xs:items-center justify-between w-full py-7 gap-2"
|
||||
>
|
||||
<Breadcrumb
|
||||
:items="breadcrumbItems"
|
||||
@click="handleBreadcrumbClick"
|
||||
@@ -85,6 +100,11 @@ const toggleBlock = () => {
|
||||
:disabled="isUpdating"
|
||||
@click="toggleBlock"
|
||||
/>
|
||||
<VoiceCallButton
|
||||
:phone="selectedContact?.phoneNumber"
|
||||
:label="$t('CONTACT_PANEL.CALL')"
|
||||
size="sm"
|
||||
/>
|
||||
<ComposeConversation :contact-id="contactId">
|
||||
<template #trigger="{ toggle }">
|
||||
<Button
|
||||
@@ -105,11 +125,65 @@ const toggleBlock = () => {
|
||||
</main>
|
||||
</div>
|
||||
|
||||
<!-- Desktop sidebar -->
|
||||
<div
|
||||
v-if="slots.sidebar"
|
||||
class="overflow-y-auto justify-end min-w-52 w-full py-6 max-w-md border-l border-n-weak bg-n-solid-2"
|
||||
class="hidden lg:block overflow-y-auto justify-end min-w-52 w-full py-6 max-w-md border-l border-n-weak bg-n-solid-2"
|
||||
>
|
||||
<slot name="sidebar" />
|
||||
</div>
|
||||
|
||||
<!-- Mobile sidebar container -->
|
||||
<div
|
||||
v-if="slots.sidebar"
|
||||
class="lg:hidden fixed top-0 ltr:right-0 rtl:left-0 h-full z-50 flex justify-end transition-all duration-200 ease-in-out"
|
||||
:class="isContactSidebarOpen ? 'w-full' : 'w-16'"
|
||||
>
|
||||
<!-- Toggle button -->
|
||||
<div
|
||||
v-on-click-outside="[
|
||||
closeMobileSidebar,
|
||||
{ ignore: ['#contact-sidebar-content'] },
|
||||
]"
|
||||
class="flex items-start p-1 w-fit h-fit relative order-1 xs:top-24 top-28 transition-all bg-n-solid-2 border border-n-weak duration-500 ease-in-out"
|
||||
:class="[
|
||||
isContactSidebarOpen
|
||||
? 'justify-end ltr:rounded-l-full rtl:rounded-r-full ltr:rounded-r-none rtl:rounded-l-none'
|
||||
: 'justify-center rounded-full ltr:mr-6 rtl:ml-6',
|
||||
]"
|
||||
>
|
||||
<Button
|
||||
ghost
|
||||
slate
|
||||
sm
|
||||
class="!rounded-full rtl:rotate-180"
|
||||
:class="{ 'bg-n-alpha-2': isContactSidebarOpen }"
|
||||
:icon="
|
||||
isContactSidebarOpen
|
||||
? 'i-lucide-panel-right-close'
|
||||
: 'i-lucide-panel-right-open'
|
||||
"
|
||||
data-contact-sidebar-toggle
|
||||
@click="handleConversationSidebarToggle"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<Transition
|
||||
enter-active-class="transition-transform duration-200 ease-in-out"
|
||||
leave-active-class="transition-transform duration-200 ease-in-out"
|
||||
enter-from-class="ltr:translate-x-full rtl:-translate-x-full"
|
||||
enter-to-class="ltr:translate-x-0 rtl:-translate-x-0"
|
||||
leave-from-class="ltr:translate-x-0 rtl:-translate-x-0"
|
||||
leave-to-class="ltr:translate-x-full rtl:-translate-x-full"
|
||||
>
|
||||
<div
|
||||
v-if="isContactSidebarOpen"
|
||||
id="contact-sidebar-content"
|
||||
class="order-2 w-[85%] sm:w-[50%] bg-n-solid-2 ltr:border-l rtl:border-r border-n-weak overflow-y-auto py-6 shadow-lg"
|
||||
>
|
||||
<slot name="sidebar" />
|
||||
</div>
|
||||
</Transition>
|
||||
</div>
|
||||
</section>
|
||||
</template>
|
||||
|
||||
@@ -34,13 +34,13 @@ const emit = defineEmits([
|
||||
<template>
|
||||
<header class="sticky top-0 z-10">
|
||||
<div
|
||||
class="flex items-center justify-between w-full h-20 px-6 gap-2 mx-auto max-w-[60rem]"
|
||||
class="flex items-start sm:items-center justify-between w-full py-6 px-6 gap-2 mx-auto max-w-[60rem]"
|
||||
>
|
||||
<span class="text-xl font-medium truncate text-n-slate-12">
|
||||
{{ headerTitle }}
|
||||
</span>
|
||||
<div class="flex items-center flex-shrink-0 gap-4">
|
||||
<div v-if="showSearch" class="flex items-center gap-2">
|
||||
<div class="flex items-center flex-col sm:flex-row flex-shrink-0 gap-4">
|
||||
<div v-if="showSearch" class="flex items-center gap-2 w-full">
|
||||
<Input
|
||||
:model-value="searchValue"
|
||||
type="search"
|
||||
@@ -48,6 +48,7 @@ const emit = defineEmits([
|
||||
:custom-input-class="[
|
||||
'h-8 [&:not(.focus)]:!border-transparent bg-n-alpha-2 dark:bg-n-solid-1 ltr:!pl-8 !py-1 rtl:!pr-8',
|
||||
]"
|
||||
class="w-full"
|
||||
@input="emit('search', $event.target.value)"
|
||||
>
|
||||
<template #prefix>
|
||||
@@ -58,64 +59,66 @@ const emit = defineEmits([
|
||||
</template>
|
||||
</Input>
|
||||
</div>
|
||||
<div class="flex items-center gap-2">
|
||||
<div v-if="!isLabelView && !isActiveView" class="relative">
|
||||
<div class="flex items-center flex-shrink-0 gap-4">
|
||||
<div class="flex items-center gap-2">
|
||||
<div v-if="!isLabelView && !isActiveView" class="relative">
|
||||
<Button
|
||||
id="toggleContactsFilterButton"
|
||||
:icon="
|
||||
isSegmentsView ? 'i-lucide-pen-line' : 'i-lucide-list-filter'
|
||||
"
|
||||
color="slate"
|
||||
size="sm"
|
||||
class="relative w-8"
|
||||
variant="ghost"
|
||||
@click="emit('filter')"
|
||||
>
|
||||
<div
|
||||
v-if="hasActiveFilters && !isSegmentsView"
|
||||
class="absolute top-0 right-0 w-2 h-2 rounded-full bg-n-brand"
|
||||
/>
|
||||
</Button>
|
||||
<slot name="filter" />
|
||||
</div>
|
||||
<Button
|
||||
id="toggleContactsFilterButton"
|
||||
:icon="
|
||||
isSegmentsView ? 'i-lucide-pen-line' : 'i-lucide-list-filter'
|
||||
v-if="
|
||||
hasActiveFilters &&
|
||||
!isSegmentsView &&
|
||||
!isLabelView &&
|
||||
!isActiveView
|
||||
"
|
||||
icon="i-lucide-save"
|
||||
color="slate"
|
||||
size="sm"
|
||||
class="relative w-8"
|
||||
variant="ghost"
|
||||
@click="emit('filter')"
|
||||
>
|
||||
<div
|
||||
v-if="hasActiveFilters && !isSegmentsView"
|
||||
class="absolute top-0 right-0 w-2 h-2 rounded-full bg-n-brand"
|
||||
/>
|
||||
</Button>
|
||||
<slot name="filter" />
|
||||
@click="emit('createSegment')"
|
||||
/>
|
||||
<Button
|
||||
v-if="isSegmentsView && !isLabelView && !isActiveView"
|
||||
icon="i-lucide-trash"
|
||||
color="slate"
|
||||
size="sm"
|
||||
variant="ghost"
|
||||
@click="emit('deleteSegment')"
|
||||
/>
|
||||
<ContactSortMenu
|
||||
:active-sort="activeSort"
|
||||
:active-ordering="activeOrdering"
|
||||
@update:sort="emit('update:sort', $event)"
|
||||
/>
|
||||
<ContactMoreActions
|
||||
@add="emit('add')"
|
||||
@import="emit('import')"
|
||||
@export="emit('export')"
|
||||
/>
|
||||
</div>
|
||||
<Button
|
||||
v-if="
|
||||
hasActiveFilters &&
|
||||
!isSegmentsView &&
|
||||
!isLabelView &&
|
||||
!isActiveView
|
||||
"
|
||||
icon="i-lucide-save"
|
||||
color="slate"
|
||||
size="sm"
|
||||
variant="ghost"
|
||||
@click="emit('createSegment')"
|
||||
/>
|
||||
<Button
|
||||
v-if="isSegmentsView && !isLabelView && !isActiveView"
|
||||
icon="i-lucide-trash"
|
||||
color="slate"
|
||||
size="sm"
|
||||
variant="ghost"
|
||||
@click="emit('deleteSegment')"
|
||||
/>
|
||||
<ContactSortMenu
|
||||
:active-sort="activeSort"
|
||||
:active-ordering="activeOrdering"
|
||||
@update:sort="emit('update:sort', $event)"
|
||||
/>
|
||||
<ContactMoreActions
|
||||
@add="emit('add')"
|
||||
@import="emit('import')"
|
||||
@export="emit('export')"
|
||||
/>
|
||||
<div class="w-px h-4 bg-n-strong" />
|
||||
<ComposeConversation>
|
||||
<template #trigger="{ toggle }">
|
||||
<Button :label="buttonLabel" size="sm" @click="toggle" />
|
||||
</template>
|
||||
</ComposeConversation>
|
||||
</div>
|
||||
<div class="w-px h-4 bg-n-strong" />
|
||||
<ComposeConversation>
|
||||
<template #trigger="{ toggle }">
|
||||
<Button :label="buttonLabel" size="sm" @click="toggle" />
|
||||
</template>
|
||||
</ComposeConversation>
|
||||
</div>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
@@ -62,6 +62,7 @@ const segmentsQuery = ref({});
|
||||
|
||||
const appliedFilters = useMapGetter('contacts/getAppliedContactFiltersV4');
|
||||
const contactAttributes = useMapGetter('attributes/getContactAttributes');
|
||||
const labels = useMapGetter('labels/getLabels');
|
||||
const hasActiveSegments = computed(
|
||||
() => props.activeSegment && props.segmentsId !== 0
|
||||
);
|
||||
@@ -215,6 +216,7 @@ const setParamsForEditSegmentModal = () => {
|
||||
countries,
|
||||
filterTypes: contactFilterItems,
|
||||
allCustomAttributes: useSnakeCase(contactAttributes.value),
|
||||
labels: labels.value || [],
|
||||
};
|
||||
};
|
||||
|
||||
@@ -291,17 +293,20 @@ defineExpose({
|
||||
@delete-segment="openDeleteSegmentDialog"
|
||||
>
|
||||
<template #filter>
|
||||
<ContactsFilter
|
||||
v-if="showFiltersModal"
|
||||
v-model="appliedFilter"
|
||||
:segment-name="activeSegmentName"
|
||||
:is-segment-view="hasActiveSegments"
|
||||
class="absolute mt-1 ltr:right-0 rtl:left-0 top-full"
|
||||
@apply-filter="onApplyFilter"
|
||||
@update-segment="onUpdateSegment"
|
||||
@close="closeAdvanceFiltersModal"
|
||||
@clear-filters="clearFilters"
|
||||
/>
|
||||
<div
|
||||
class="absolute mt-1 ltr:-right-52 rtl:-left-52 sm:ltr:right-0 sm:rtl:left-0 top-full"
|
||||
>
|
||||
<ContactsFilter
|
||||
v-if="showFiltersModal"
|
||||
v-model="appliedFilter"
|
||||
:segment-name="activeSegmentName"
|
||||
:is-segment-view="hasActiveSegments"
|
||||
@apply-filter="onApplyFilter"
|
||||
@update-segment="onUpdateSegment"
|
||||
@close="closeAdvanceFiltersModal"
|
||||
@clear-filters="clearFilters"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
</ContactsHeader>
|
||||
|
||||
|
||||
@@ -105,7 +105,7 @@ const handleOrderChange = value => {
|
||||
<div
|
||||
v-if="isMenuOpen"
|
||||
v-on-clickaway="() => (isMenuOpen = false)"
|
||||
class="absolute top-full mt-1 ltr:right-0 rtl:left-0 flex flex-col gap-4 bg-n-alpha-3 backdrop-blur-[100px] border border-n-weak w-72 rounded-xl p-4"
|
||||
class="absolute top-full mt-1 ltr:-right-32 rtl:-left-32 sm:ltr:right-0 sm:rtl:left-0 flex flex-col gap-4 bg-n-alpha-3 backdrop-blur-[100px] border border-n-weak w-72 rounded-xl p-4"
|
||||
>
|
||||
<div class="flex items-center justify-between gap-2">
|
||||
<span class="text-sm text-n-slate-12">
|
||||
|
||||
@@ -96,10 +96,7 @@ const openFilter = () => {
|
||||
<slot name="default" />
|
||||
</div>
|
||||
</main>
|
||||
<footer
|
||||
v-if="showPaginationFooter"
|
||||
class="sticky bottom-0 z-10 px-4 pb-4"
|
||||
>
|
||||
<footer v-if="showPaginationFooter" class="sticky bottom-0 z-0 px-4 pb-4">
|
||||
<PaginationFooter
|
||||
current-page-info="CONTACTS_LAYOUT.PAGINATION_FOOTER.SHOWING"
|
||||
:current-page="currentPage"
|
||||
|
||||
@@ -4,8 +4,8 @@ export default [
|
||||
city: 'Los Angeles',
|
||||
country: 'United States',
|
||||
description:
|
||||
"I'm Candice, a developer focusing on building web solutions. Currently, I’m working as a Product Developer at Chatwoot.",
|
||||
companyName: 'Chatwoot',
|
||||
"I'm Candice, a developer focusing on building web solutions. Currently, I’m working as a Product Developer at Lumora.",
|
||||
companyName: 'Lumora',
|
||||
countryCode: 'US',
|
||||
socialProfiles: {
|
||||
github: 'candice-dev',
|
||||
@@ -16,7 +16,7 @@ export default [
|
||||
},
|
||||
},
|
||||
availabilityStatus: 'offline',
|
||||
email: 'candice.matherson@chatwoot.com',
|
||||
email: 'candice.matherson@lumora.com',
|
||||
id: 22,
|
||||
name: 'Candice Matherson',
|
||||
phoneNumber: '+14155552671',
|
||||
|
||||
@@ -0,0 +1,91 @@
|
||||
<script setup>
|
||||
import { computed, ref, useAttrs } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { useMapGetter } from 'dashboard/composables/store';
|
||||
import { INBOX_TYPES } from 'dashboard/helper/inbox';
|
||||
import { useAlert } from 'dashboard/composables';
|
||||
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import Dialog from 'dashboard/components-next/dialog/Dialog.vue';
|
||||
|
||||
const props = defineProps({
|
||||
phone: { type: String, default: '' },
|
||||
label: { type: String, default: '' },
|
||||
icon: { type: [String, Object, Function], default: '' },
|
||||
size: { type: String, default: 'sm' },
|
||||
tooltipLabel: { type: String, default: '' },
|
||||
});
|
||||
|
||||
defineOptions({ inheritAttrs: false });
|
||||
const attrs = useAttrs();
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const inboxesList = useMapGetter('inboxes/getInboxes');
|
||||
const voiceInboxes = computed(() =>
|
||||
(inboxesList.value || []).filter(
|
||||
inbox => inbox.channel_type === INBOX_TYPES.VOICE
|
||||
)
|
||||
);
|
||||
const hasVoiceInboxes = computed(() => voiceInboxes.value.length > 0);
|
||||
|
||||
// Unified behavior: hide when no phone
|
||||
const shouldRender = computed(() => hasVoiceInboxes.value && !!props.phone);
|
||||
|
||||
const dialogRef = ref(null);
|
||||
|
||||
const onClick = () => {
|
||||
if (voiceInboxes.value.length > 1) {
|
||||
dialogRef.value?.open();
|
||||
return;
|
||||
}
|
||||
useAlert(t('CONTACT_PANEL.CALL_UNDER_DEVELOPMENT'));
|
||||
};
|
||||
|
||||
const onPickInbox = () => {
|
||||
// Placeholder until actual call wiring happens
|
||||
useAlert(t('CONTACT_PANEL.CALL_UNDER_DEVELOPMENT'));
|
||||
dialogRef.value?.close();
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<span class="contents">
|
||||
<Button
|
||||
v-if="shouldRender"
|
||||
v-tooltip.top-end="tooltipLabel || null"
|
||||
v-bind="attrs"
|
||||
:label="label"
|
||||
:icon="icon"
|
||||
:size="size"
|
||||
@click="onClick"
|
||||
/>
|
||||
|
||||
<Dialog
|
||||
v-if="shouldRender && voiceInboxes.length > 1"
|
||||
ref="dialogRef"
|
||||
:title="$t('CONTACT_PANEL.VOICE_INBOX_PICKER.TITLE')"
|
||||
show-cancel-button
|
||||
:show-confirm-button="false"
|
||||
width="md"
|
||||
>
|
||||
<div class="flex flex-col gap-2">
|
||||
<button
|
||||
v-for="inbox in voiceInboxes"
|
||||
:key="inbox.id"
|
||||
type="button"
|
||||
class="flex items-center justify-between w-full px-4 py-2 text-left rounded-lg hover:bg-n-alpha-2"
|
||||
@click="onPickInbox(inbox)"
|
||||
>
|
||||
<div class="flex items-center gap-2">
|
||||
<span class="i-ri-phone-fill text-n-slate-10" />
|
||||
<span class="text-sm text-n-slate-12">{{ inbox.name }}</span>
|
||||
</div>
|
||||
<span v-if="inbox.phone_number" class="text-xs text-n-slate-10">
|
||||
{{ inbox.phone_number }}
|
||||
</span>
|
||||
</button>
|
||||
</div>
|
||||
</Dialog>
|
||||
</span>
|
||||
</template>
|
||||
@@ -47,6 +47,7 @@ const unreadMessagesCount = computed(() => {
|
||||
</p>
|
||||
<div class="flex items-center flex-shrink-0 gap-2 pb-2">
|
||||
<Avatar
|
||||
v-if="assignee.name"
|
||||
:name="assignee.name"
|
||||
:src="assignee.thumbnail"
|
||||
:size="20"
|
||||
|
||||
@@ -96,6 +96,7 @@ defineExpose({
|
||||
/>
|
||||
</div>
|
||||
<Avatar
|
||||
v-if="assignee.name"
|
||||
:name="assignee.name"
|
||||
:src="assignee.thumbnail"
|
||||
:size="20"
|
||||
|
||||
@@ -51,12 +51,20 @@ const originalState = reactive({ ...state });
|
||||
|
||||
const liveChatWidgets = computed(() => {
|
||||
const inboxes = store.getters['inboxes/getInboxes'];
|
||||
return inboxes
|
||||
const widgetOptions = inboxes
|
||||
.filter(inbox => inbox.channel_type === 'Channel::WebWidget')
|
||||
.map(inbox => ({
|
||||
value: inbox.id,
|
||||
label: inbox.name,
|
||||
}));
|
||||
|
||||
return [
|
||||
{
|
||||
value: '',
|
||||
label: t('HELP_CENTER.PORTAL_SETTINGS.FORM.LIVE_CHAT_WIDGET.NONE_OPTION'),
|
||||
},
|
||||
...widgetOptions,
|
||||
];
|
||||
});
|
||||
|
||||
const rules = {
|
||||
@@ -108,7 +116,7 @@ watch(
|
||||
widgetColor: newVal.color,
|
||||
homePageLink: newVal.homepage_link,
|
||||
slug: newVal.slug,
|
||||
liveChatWidgetInboxId: newVal.inbox?.id,
|
||||
liveChatWidgetInboxId: newVal.inbox?.id || '',
|
||||
});
|
||||
if (newVal.logo) {
|
||||
const {
|
||||
|
||||
@@ -15,7 +15,7 @@ const props = defineProps({
|
||||
const emit = defineEmits(['remove', 'hover']);
|
||||
|
||||
const handleRemoveLabel = () => {
|
||||
emit('remove', props.label?.id);
|
||||
emit('remove', props.label);
|
||||
};
|
||||
|
||||
const handleMouseEnter = () => {
|
||||
@@ -45,6 +45,7 @@ const handleMouseEnter = () => {
|
||||
<Button
|
||||
class="transition-opacity duration-200 !h-7 ltr:rounded-r-md rtl:rounded-l-md ltr:rounded-l-none rtl:rounded-r-none w-6 bg-transparent"
|
||||
:class="{ 'opacity-0': !isHovered, 'opacity-100': isHovered }"
|
||||
type="button"
|
||||
slate
|
||||
xs
|
||||
faded
|
||||
|
||||
@@ -11,12 +11,14 @@ import { extractTextFromMarkdown } from 'dashboard/helper/editorHelper';
|
||||
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import WhatsAppOptions from './WhatsAppOptions.vue';
|
||||
import ContentTemplateSelector from './ContentTemplateSelector.vue';
|
||||
|
||||
const props = defineProps({
|
||||
attachedFiles: { type: Array, default: () => [] },
|
||||
isWhatsappInbox: { type: Boolean, default: false },
|
||||
isEmailOrWebWidgetInbox: { type: Boolean, default: false },
|
||||
isTwilioSmsInbox: { type: Boolean, default: false },
|
||||
isTwilioWhatsAppInbox: { type: Boolean, default: false },
|
||||
messageTemplates: { type: Array, default: () => [] },
|
||||
channelType: { type: String, default: '' },
|
||||
isLoading: { type: Boolean, default: false },
|
||||
@@ -32,6 +34,7 @@ const emit = defineEmits([
|
||||
'discard',
|
||||
'sendMessage',
|
||||
'sendWhatsappMessage',
|
||||
'sendTwilioMessage',
|
||||
'insertEmoji',
|
||||
'addSignature',
|
||||
'removeSignature',
|
||||
@@ -63,6 +66,20 @@ const sendWithSignature = computed(() => {
|
||||
return fetchSignatureFlagFromUISettings(props.channelType);
|
||||
});
|
||||
|
||||
const showTwilioContentTemplates = computed(() => {
|
||||
return props.isTwilioWhatsAppInbox && props.inboxId;
|
||||
});
|
||||
|
||||
const shouldShowEmojiButton = computed(() => {
|
||||
return (
|
||||
!props.isWhatsappInbox && !props.isTwilioWhatsAppInbox && !props.hasNoInbox
|
||||
);
|
||||
});
|
||||
|
||||
const isRegularMessageMode = computed(() => {
|
||||
return !props.isWhatsappInbox && !props.isTwilioWhatsAppInbox;
|
||||
});
|
||||
|
||||
const setSignature = () => {
|
||||
if (signatureToApply.value) {
|
||||
if (sendWithSignature.value) {
|
||||
@@ -125,7 +142,7 @@ const keyboardEvents = {
|
||||
action: () => {
|
||||
if (
|
||||
isEditorHotKeyEnabled('enter') &&
|
||||
!props.isWhatsappInbox &&
|
||||
isRegularMessageMode.value &&
|
||||
!props.isDropdownActive
|
||||
) {
|
||||
emit('sendMessage');
|
||||
@@ -136,7 +153,7 @@ const keyboardEvents = {
|
||||
action: () => {
|
||||
if (
|
||||
isEditorHotKeyEnabled('cmd_enter') &&
|
||||
!props.isWhatsappInbox &&
|
||||
isRegularMessageMode.value &&
|
||||
!props.isDropdownActive
|
||||
) {
|
||||
emit('sendMessage');
|
||||
@@ -158,8 +175,13 @@ useKeyboardEvents(keyboardEvents);
|
||||
:message-templates="messageTemplates"
|
||||
@send-message="emit('sendWhatsappMessage', $event)"
|
||||
/>
|
||||
<ContentTemplateSelector
|
||||
v-if="showTwilioContentTemplates"
|
||||
:inbox-id="inboxId"
|
||||
@send-message="emit('sendTwilioMessage', $event)"
|
||||
/>
|
||||
<div
|
||||
v-if="!isWhatsappInbox && !hasNoInbox"
|
||||
v-if="shouldShowEmojiButton"
|
||||
v-on-click-outside="() => (isEmojiPickerOpen = false)"
|
||||
class="relative"
|
||||
>
|
||||
@@ -172,7 +194,7 @@ useKeyboardEvents(keyboardEvents);
|
||||
/>
|
||||
<EmojiInput
|
||||
v-if="isEmojiPickerOpen"
|
||||
class="ltr:left-0 rtl:right-0 top-full mt-1.5"
|
||||
class="top-full mt-1.5 ltr:left-0 rtl:right-0"
|
||||
:on-click="onClickInsertEmoji"
|
||||
/>
|
||||
</div>
|
||||
@@ -199,7 +221,7 @@ useKeyboardEvents(keyboardEvents);
|
||||
/>
|
||||
</FileUpload>
|
||||
<Button
|
||||
v-if="hasSelectedInbox && !isWhatsappInbox"
|
||||
v-if="hasSelectedInbox && isRegularMessageMode"
|
||||
icon="i-lucide-signature"
|
||||
color="slate"
|
||||
size="sm"
|
||||
@@ -218,7 +240,7 @@ useKeyboardEvents(keyboardEvents);
|
||||
@click="emit('discard')"
|
||||
/>
|
||||
<Button
|
||||
v-if="!isWhatsappInbox"
|
||||
v-if="isRegularMessageMode"
|
||||
:label="sendButtonLabel"
|
||||
size="sm"
|
||||
class="!text-xs font-medium"
|
||||
|
||||
@@ -74,6 +74,9 @@ const inboxTypes = computed(() => ({
|
||||
isTwilioSMS:
|
||||
props.targetInbox?.channelType === INBOX_TYPES.TWILIO &&
|
||||
props.targetInbox?.medium === 'sms',
|
||||
isTwilioWhatsapp:
|
||||
props.targetInbox?.channelType === INBOX_TYPES.TWILIO &&
|
||||
props.targetInbox?.medium === 'whatsapp',
|
||||
}));
|
||||
|
||||
const whatsappMessageTemplates = computed(() =>
|
||||
@@ -261,6 +264,28 @@ const handleSendWhatsappMessage = async ({ message, templateParams }) => {
|
||||
isFromWhatsApp: true,
|
||||
});
|
||||
};
|
||||
|
||||
const handleSendTwilioMessage = async ({ message, templateParams }) => {
|
||||
const twilioMessagePayload = prepareWhatsAppMessagePayload({
|
||||
targetInbox: props.targetInbox,
|
||||
selectedContact: props.selectedContact,
|
||||
message,
|
||||
templateParams,
|
||||
currentUser: props.currentUser,
|
||||
});
|
||||
await emit('createConversation', {
|
||||
payload: twilioMessagePayload,
|
||||
isFromWhatsApp: true,
|
||||
});
|
||||
};
|
||||
|
||||
const shouldShowMessageEditor = computed(() => {
|
||||
return (
|
||||
!inboxTypes.value.isWhatsapp &&
|
||||
!showNoInboxAlert.value &&
|
||||
!inboxTypes.value.isTwilioWhatsapp
|
||||
);
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
@@ -311,7 +336,7 @@ const handleSendWhatsappMessage = async ({ message, templateParams }) => {
|
||||
/>
|
||||
|
||||
<MessageEditor
|
||||
v-if="!inboxTypes.isWhatsapp && !showNoInboxAlert"
|
||||
v-if="shouldShowMessageEditor"
|
||||
v-model="state.message"
|
||||
:message-signature="messageSignature"
|
||||
:send-with-signature="sendWithSignature"
|
||||
@@ -331,6 +356,7 @@ const handleSendWhatsappMessage = async ({ message, templateParams }) => {
|
||||
:is-whatsapp-inbox="inboxTypes.isWhatsapp"
|
||||
:is-email-or-web-widget-inbox="inboxTypes.isEmailOrWebWidget"
|
||||
:is-twilio-sms-inbox="inboxTypes.isTwilioSMS"
|
||||
:is-twilio-whats-app-inbox="inboxTypes.isTwilioWhatsapp"
|
||||
:message-templates="whatsappMessageTemplates"
|
||||
:channel-type="inboxChannelType"
|
||||
:is-loading="isCreating"
|
||||
@@ -347,6 +373,7 @@ const handleSendWhatsappMessage = async ({ message, templateParams }) => {
|
||||
@discard="$emit('discard')"
|
||||
@send-message="handleSendMessage"
|
||||
@send-whatsapp-message="handleSendWhatsappMessage"
|
||||
@send-twilio-message="handleSendTwilioMessage"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -0,0 +1,56 @@
|
||||
<script setup>
|
||||
import ContentTemplateParser from 'dashboard/components-next/content-templates/ContentTemplateParser.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
|
||||
defineProps({
|
||||
template: {
|
||||
type: Object,
|
||||
default: () => ({}),
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['sendMessage', 'back']);
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const handleSendMessage = payload => {
|
||||
emit('sendMessage', payload);
|
||||
};
|
||||
|
||||
const handleBack = () => {
|
||||
emit('back');
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
class="absolute top-full mt-1.5 max-h-[30rem] overflow-y-auto ltr:left-0 rtl:right-0 flex flex-col gap-4 px-4 pt-6 pb-5 items-start w-[28.75rem] h-auto bg-n-solid-2 border border-n-strong shadow-sm rounded-lg"
|
||||
>
|
||||
<div class="w-full">
|
||||
<ContentTemplateParser
|
||||
:template="template"
|
||||
@send-message="handleSendMessage"
|
||||
@back="handleBack"
|
||||
>
|
||||
<template #actions="{ sendMessage, goBack, disabled }">
|
||||
<div class="flex gap-3 justify-between items-end w-full h-14">
|
||||
<Button
|
||||
:label="t('CONTENT_TEMPLATES.FORM.BACK_BUTTON')"
|
||||
color="slate"
|
||||
variant="faded"
|
||||
class="w-full font-medium"
|
||||
@click="goBack"
|
||||
/>
|
||||
<Button
|
||||
:label="t('CONTENT_TEMPLATES.FORM.SEND_MESSAGE_BUTTON')"
|
||||
class="w-full font-medium"
|
||||
:disabled="disabled"
|
||||
@click="sendMessage"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
</ContentTemplateParser>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -0,0 +1,124 @@
|
||||
<script setup>
|
||||
import { computed, ref } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { useMapGetter } from 'dashboard/composables/store';
|
||||
|
||||
import Icon from 'dashboard/components-next/icon/Icon.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
import ContentTemplateForm from './ContentTemplateForm.vue';
|
||||
|
||||
const props = defineProps({
|
||||
inboxId: {
|
||||
type: Number,
|
||||
required: true,
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['sendMessage']);
|
||||
|
||||
const { t } = useI18n();
|
||||
const inbox = useMapGetter('inboxes/getInbox');
|
||||
|
||||
const searchQuery = ref('');
|
||||
const selectedTemplate = ref(null);
|
||||
const showTemplatesMenu = ref(false);
|
||||
|
||||
const contentTemplates = computed(() => {
|
||||
const inboxData = inbox.value(props.inboxId);
|
||||
return inboxData?.content_templates?.templates || [];
|
||||
});
|
||||
|
||||
const filteredTemplates = computed(() => {
|
||||
return contentTemplates.value.filter(
|
||||
template =>
|
||||
template.friendly_name
|
||||
.toLowerCase()
|
||||
.includes(searchQuery.value.toLowerCase()) &&
|
||||
template.status === 'approved'
|
||||
);
|
||||
});
|
||||
|
||||
const handleTriggerClick = () => {
|
||||
searchQuery.value = '';
|
||||
showTemplatesMenu.value = !showTemplatesMenu.value;
|
||||
};
|
||||
|
||||
const handleTemplateClick = template => {
|
||||
selectedTemplate.value = template;
|
||||
showTemplatesMenu.value = false;
|
||||
};
|
||||
|
||||
const handleBack = () => {
|
||||
selectedTemplate.value = null;
|
||||
showTemplatesMenu.value = true;
|
||||
};
|
||||
|
||||
const handleSendMessage = template => {
|
||||
emit('sendMessage', template);
|
||||
selectedTemplate.value = null;
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="relative">
|
||||
<Button
|
||||
icon="i-ph-whatsapp-logo"
|
||||
:label="t('COMPOSE_NEW_CONVERSATION.FORM.TWILIO_OPTIONS.LABEL')"
|
||||
color="slate"
|
||||
size="sm"
|
||||
:disabled="selectedTemplate"
|
||||
class="!text-xs font-medium"
|
||||
@click="handleTriggerClick"
|
||||
/>
|
||||
<div
|
||||
v-if="showTemplatesMenu"
|
||||
class="absolute top-full mt-1.5 max-h-96 overflow-y-auto ltr:left-0 rtl:right-0 flex flex-col gap-2 p-4 items-center w-[21.875rem] h-auto bg-n-solid-2 border border-n-strong shadow-sm rounded-lg"
|
||||
>
|
||||
<div class="w-full">
|
||||
<Input
|
||||
v-model="searchQuery"
|
||||
type="search"
|
||||
:placeholder="
|
||||
t('COMPOSE_NEW_CONVERSATION.FORM.TWILIO_OPTIONS.SEARCH_PLACEHOLDER')
|
||||
"
|
||||
custom-input-class="ltr:pl-10 rtl:pr-10"
|
||||
>
|
||||
<template #prefix>
|
||||
<Icon
|
||||
icon="i-lucide-search"
|
||||
class="absolute top-2 size-3.5 ltr:left-3 rtl:right-3"
|
||||
/>
|
||||
</template>
|
||||
</Input>
|
||||
</div>
|
||||
<div
|
||||
v-for="template in filteredTemplates"
|
||||
:key="template.content_sid"
|
||||
tabindex="0"
|
||||
class="flex flex-col gap-2 p-2 w-full rounded-lg cursor-pointer dark:hover:bg-n-alpha-3 hover:bg-n-alpha-1"
|
||||
@click="handleTemplateClick(template)"
|
||||
>
|
||||
<div class="flex justify-between items-center">
|
||||
<span class="text-sm text-n-slate-12">{{
|
||||
template.friendly_name
|
||||
}}</span>
|
||||
</div>
|
||||
<p class="mb-0 text-xs leading-5 text-n-slate-11 line-clamp-2">
|
||||
{{ template.body || t('CONTENT_TEMPLATES.PICKER.NO_CONTENT') }}
|
||||
</p>
|
||||
</div>
|
||||
<template v-if="filteredTemplates.length === 0">
|
||||
<p class="pt-2 w-full text-sm text-n-slate-11">
|
||||
{{ t('COMPOSE_NEW_CONVERSATION.FORM.TWILIO_OPTIONS.EMPTY_STATE') }}
|
||||
</p>
|
||||
</template>
|
||||
</div>
|
||||
<ContentTemplateForm
|
||||
v-if="selectedTemplate"
|
||||
:template="selectedTemplate"
|
||||
@send-message="handleSendMessage"
|
||||
@back="handleBack"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
@@ -1,177 +0,0 @@
|
||||
<script setup>
|
||||
import { computed, ref, onMounted } from 'vue';
|
||||
import { useVuelidate } from '@vuelidate/core';
|
||||
import { requiredIf } from '@vuelidate/validators';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
|
||||
const props = defineProps({
|
||||
template: {
|
||||
type: Object,
|
||||
required: true,
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['sendMessage', 'back']);
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const processedParams = ref({});
|
||||
|
||||
const templateName = computed(() => {
|
||||
return props.template?.name || '';
|
||||
});
|
||||
|
||||
const templateString = computed(() => {
|
||||
return props.template?.components?.find(
|
||||
component => component.type === 'BODY'
|
||||
).text;
|
||||
});
|
||||
|
||||
const processVariable = str => {
|
||||
return str.replace(/{{|}}/g, '');
|
||||
};
|
||||
|
||||
const processedString = computed(() => {
|
||||
return templateString.value.replace(/{{([^}]+)}}/g, (match, variable) => {
|
||||
const variableKey = processVariable(variable);
|
||||
return processedParams.value[variableKey] || `{{${variable}}}`;
|
||||
});
|
||||
});
|
||||
|
||||
const processedStringWithVariableHighlight = computed(() => {
|
||||
const variables = templateString.value.match(/{{([^}]+)}}/g) || [];
|
||||
|
||||
return variables.reduce((result, variable) => {
|
||||
const variableKey = processVariable(variable);
|
||||
const value = processedParams.value[variableKey] || variable;
|
||||
return result.replace(
|
||||
variable,
|
||||
`<span class="break-all text-n-slate-12">${value}</span>`
|
||||
);
|
||||
}, templateString.value);
|
||||
});
|
||||
|
||||
const rules = computed(() => {
|
||||
const paramRules = {};
|
||||
Object.keys(processedParams.value).forEach(key => {
|
||||
paramRules[key] = { required: requiredIf(true) };
|
||||
});
|
||||
return {
|
||||
processedParams: paramRules,
|
||||
};
|
||||
});
|
||||
|
||||
const v$ = useVuelidate(rules, { processedParams });
|
||||
|
||||
const getFieldErrorType = key => {
|
||||
if (!v$.value.processedParams[key]?.$error) return 'info';
|
||||
return 'error';
|
||||
};
|
||||
|
||||
const generateVariables = () => {
|
||||
const matchedVariables = templateString.value.match(/{{([^}]+)}}/g);
|
||||
if (!matchedVariables) return;
|
||||
|
||||
const finalVars = matchedVariables.map(i => processVariable(i));
|
||||
processedParams.value = finalVars.reduce((acc, variable) => {
|
||||
acc[variable] = '';
|
||||
return acc;
|
||||
}, {});
|
||||
};
|
||||
|
||||
const sendMessage = async () => {
|
||||
const isValid = await v$.value.$validate();
|
||||
if (!isValid) return;
|
||||
|
||||
const payload = {
|
||||
message: processedString.value,
|
||||
templateParams: {
|
||||
name: props.template.name,
|
||||
category: props.template.category,
|
||||
language: props.template.language,
|
||||
namespace: props.template.namespace,
|
||||
processed_params: processedParams.value,
|
||||
},
|
||||
};
|
||||
emit('sendMessage', payload);
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
generateVariables();
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
class="absolute top-full mt-1.5 max-h-[30rem] overflow-y-auto ltr:left-0 rtl:right-0 flex flex-col gap-4 px-4 pt-6 pb-5 items-start w-[28.75rem] h-auto bg-n-solid-2 border border-n-strong shadow-sm rounded-lg"
|
||||
>
|
||||
<span class="text-sm text-n-slate-12">
|
||||
{{
|
||||
t(
|
||||
'COMPOSE_NEW_CONVERSATION.FORM.WHATSAPP_OPTIONS.TEMPLATE_PARSER.TEMPLATE_NAME',
|
||||
{ templateName: templateName }
|
||||
)
|
||||
}}
|
||||
</span>
|
||||
<p
|
||||
class="mb-0 text-sm text-n-slate-11"
|
||||
v-html="processedStringWithVariableHighlight"
|
||||
/>
|
||||
|
||||
<span
|
||||
v-if="Object.keys(processedParams).length"
|
||||
class="text-sm font-medium text-n-slate-12"
|
||||
>
|
||||
{{
|
||||
t(
|
||||
'COMPOSE_NEW_CONVERSATION.FORM.WHATSAPP_OPTIONS.TEMPLATE_PARSER.VARIABLES'
|
||||
)
|
||||
}}
|
||||
</span>
|
||||
|
||||
<div
|
||||
v-for="(variable, key) in processedParams"
|
||||
:key="key"
|
||||
class="flex items-center w-full gap-2"
|
||||
>
|
||||
<span
|
||||
class="block h-8 text-sm min-w-6 text-start truncate text-n-slate-10 leading-8"
|
||||
:title="key"
|
||||
>
|
||||
{{ key }}
|
||||
</span>
|
||||
<Input
|
||||
v-model="processedParams[key]"
|
||||
custom-input-class="!h-8 w-full !bg-transparent"
|
||||
class="w-full"
|
||||
:message-type="getFieldErrorType(key)"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="flex items-end justify-between w-full gap-3 h-14">
|
||||
<Button
|
||||
:label="
|
||||
t(
|
||||
'COMPOSE_NEW_CONVERSATION.FORM.WHATSAPP_OPTIONS.TEMPLATE_PARSER.BACK'
|
||||
)
|
||||
"
|
||||
color="slate"
|
||||
variant="faded"
|
||||
class="w-full font-medium"
|
||||
@click="emit('back')"
|
||||
/>
|
||||
<Button
|
||||
:label="
|
||||
t(
|
||||
'COMPOSE_NEW_CONVERSATION.FORM.WHATSAPP_OPTIONS.TEMPLATE_PARSER.SEND_MESSAGE'
|
||||
)
|
||||
"
|
||||
class="w-full font-medium"
|
||||
@click="sendMessage"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -25,7 +25,7 @@ export const generateLabelForContactableInboxesList = ({
|
||||
channelType === INBOX_TYPES.TWILIO ||
|
||||
channelType === INBOX_TYPES.WHATSAPP
|
||||
) {
|
||||
return `${name} (${phoneNumber})`;
|
||||
return phoneNumber ? `${name} (${phoneNumber})` : name;
|
||||
}
|
||||
return name;
|
||||
};
|
||||
@@ -53,6 +53,7 @@ const transformInbox = ({
|
||||
email,
|
||||
phoneNumber,
|
||||
channelType,
|
||||
medium,
|
||||
...rest,
|
||||
});
|
||||
|
||||
|
||||
@@ -21,9 +21,17 @@ const onClick = (item, index) => {
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<nav :aria-label="t('BREADCRUMB.ARIA_LABEL')" class="flex items-center h-8">
|
||||
<ol class="flex items-center mb-0">
|
||||
<li v-for="(item, index) in items" :key="index" class="flex items-center">
|
||||
<nav
|
||||
:aria-label="t('BREADCRUMB.ARIA_LABEL')"
|
||||
class="flex items-center h-8 min-w-0"
|
||||
>
|
||||
<ol class="flex items-center mb-0 min-w-0">
|
||||
<li
|
||||
v-for="(item, index) in items"
|
||||
:key="index"
|
||||
class="flex items-center"
|
||||
:class="{ 'min-w-0 flex-1': index === items.length - 1 }"
|
||||
>
|
||||
<Icon
|
||||
v-if="index > 0"
|
||||
icon="i-lucide-chevron-right"
|
||||
@@ -40,7 +48,7 @@ const onClick = (item, index) => {
|
||||
</button>
|
||||
|
||||
<!-- The last breadcrumb item is plain text -->
|
||||
<span v-else class="text-sm truncate text-n-slate-12 max-w-56">
|
||||
<span v-else class="text-sm truncate text-n-slate-12 min-w-0 block">
|
||||
{{ item.emoji ? item.emoji : '' }} {{ item.label }}
|
||||
</span>
|
||||
</li>
|
||||
|
||||
@@ -0,0 +1,34 @@
|
||||
<script setup>
|
||||
import Guardrails from './Guardrails.vue';
|
||||
import Scenarios from './Scenarios.vue';
|
||||
import ResponseGuidelines from './ResponseGuidelines.vue';
|
||||
import Settings from './Settings.vue';
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Story
|
||||
title="Captain/AnimatingImg/AnimatingImg"
|
||||
:layout="{ type: 'grid', width: '300px' }"
|
||||
>
|
||||
<Variant title="Guardrails">
|
||||
<div class="p-4 bg-n-background w-full h-full">
|
||||
<Guardrails class="size-60" />
|
||||
</div>
|
||||
</Variant>
|
||||
<Variant title="Scenarios">
|
||||
<div class="p-4 bg-n-background w-full h-full">
|
||||
<Scenarios class="size-60" />
|
||||
</div>
|
||||
</Variant>
|
||||
<Variant title="ResponseGuidelines">
|
||||
<div class="p-4 bg-n-background w-full h-full">
|
||||
<ResponseGuidelines class="size-60" />
|
||||
</div>
|
||||
</Variant>
|
||||
<Variant title="Settings">
|
||||
<div class="p-4 bg-n-background w-full h-full">
|
||||
<Settings class="size-60" />
|
||||
</div>
|
||||
</Variant>
|
||||
</Story>
|
||||
</template>
|
||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,990 @@
|
||||
<script setup>
|
||||
import { ref } from 'vue';
|
||||
|
||||
const paused = ref(false);
|
||||
|
||||
const toggle = () => {
|
||||
paused.value = !paused.value;
|
||||
};
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div
|
||||
class="svg-wrapper relative"
|
||||
:class="{ paused }"
|
||||
role="button"
|
||||
:aria-pressed="paused"
|
||||
tabindex="0"
|
||||
@click="toggle"
|
||||
>
|
||||
<div class="absolute z-0 flex-shrink-0">
|
||||
<svg
|
||||
width="auto"
|
||||
height="auto"
|
||||
viewBox="0 0 200 156"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<g opacity="0.5">
|
||||
<g clip-path="url(#clip0_773_34322)">
|
||||
<circle cx="8" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<rect
|
||||
width="200"
|
||||
height="156"
|
||||
fill="url(#paint0_linear_773_34322)"
|
||||
/>
|
||||
<rect
|
||||
width="200"
|
||||
height="156"
|
||||
fill="url(#paint1_linear_773_34322)"
|
||||
/>
|
||||
</g>
|
||||
</g>
|
||||
<defs>
|
||||
<linearGradient
|
||||
id="paint0_linear_773_34322"
|
||||
x1="100"
|
||||
y1="0"
|
||||
x2="100"
|
||||
y2="156"
|
||||
gradientUnits="userSpaceOnUse"
|
||||
>
|
||||
<stop stop-color="var(--gradient-start)" />
|
||||
<stop
|
||||
offset="0.480769"
|
||||
stop-color="var(--gradient-start)"
|
||||
stop-opacity="0"
|
||||
/>
|
||||
<stop offset="1" stop-color="var(--gradient-start)" />
|
||||
</linearGradient>
|
||||
<linearGradient
|
||||
id="paint1_linear_773_34322"
|
||||
x1="0"
|
||||
y1="78"
|
||||
x2="200"
|
||||
y2="78"
|
||||
gradientUnits="userSpaceOnUse"
|
||||
>
|
||||
<stop stop-color="var(--gradient-start)" />
|
||||
<stop
|
||||
offset="0.480769"
|
||||
stop-color="var(--gradient-start)"
|
||||
stop-opacity="0"
|
||||
/>
|
||||
<stop offset="1" stop-color="var(--gradient-start)" />
|
||||
</linearGradient>
|
||||
<clipPath id="clip0_773_34322">
|
||||
<rect width="200" height="156" rx="16" fill="white" />
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
</div>
|
||||
<svg
|
||||
viewBox="0 0 136 108"
|
||||
aria-hidden="false"
|
||||
focusable="false"
|
||||
class="z-10 relative flex-shrink-0"
|
||||
>
|
||||
<rect width="136" height="108" fill="url(#paint0_radial_720_25701)" />
|
||||
<path
|
||||
d="M49 35.039C49 33.9129 49.9129 33 51.039 33C52.1651 33 53.0779 33.9129 53.0779 35.039V39.7513C53.0779 40.8774 52.1651 41.7902 51.039 41.7902C49.9129 41.7902 49 40.8774 49 39.7513V35.039Z"
|
||||
class="fill-n-slate-10"
|
||||
/>
|
||||
<path
|
||||
d="M55.5244 35.039C55.5244 33.9129 56.4373 33 57.5634 33C58.6895 33 59.6024 33.9129 59.6024 35.039V39.7513C59.6024 40.8774 58.6895 41.7902 57.5634 41.7902C56.4373 41.7902 55.5244 40.8774 55.5244 39.7513V35.039Z"
|
||||
class="fill-n-slate-10"
|
||||
/>
|
||||
<path
|
||||
fill-rule="evenodd"
|
||||
clip-rule="evenodd"
|
||||
d="M58.1862 24.173C50.5933 23.1017 44.4039 23.0594 36.6405 24.1683C34.678 24.4485 33.3796 24.638 32.3896 24.9273C31.4716 25.1956 30.9496 25.5155 30.5177 25.9981C29.6236 26.9974 29.5057 28.1276 29.3838 32.0695C29.264 35.9436 29.5013 39.4156 29.9507 43.3222C30.1901 45.4039 30.3549 46.8088 30.618 47.8769C30.8676 48.89 31.1694 49.4347 31.5889 49.8518C32.0117 50.2721 32.5536 50.5686 33.5482 50.8091C34.6002 51.0636 35.9806 51.2175 38.032 51.4413C44.6668 52.1652 49.6333 52.1615 56.2927 51.4451C58.369 51.2218 59.7719 51.0678 60.8396 50.815C61.8541 50.5749 62.3995 50.2805 62.8165 49.8722C63.2247 49.4724 63.5366 48.921 63.8071 47.8574C64.0894 46.7477 64.2815 45.2793 64.5594 43.1222C65.0553 39.2735 65.4123 35.8935 65.4262 32.1908C65.4411 28.232 65.3601 27.1056 64.4558 26.0608C64.019 25.5562 63.488 25.2244 62.5491 24.9481C61.5352 24.6497 60.2017 24.4573 58.1862 24.173ZM36.1664 20.849C44.2537 19.6939 50.7606 19.7391 58.6546 20.8529L58.7701 20.8692C60.6392 21.1328 62.2183 21.3555 63.4958 21.7315C64.8745 22.1372 66.0274 22.7532 66.991 23.8665C68.8087 25.9666 68.7969 28.4583 68.7811 31.7688C68.7804 31.9121 68.7797 32.057 68.7792 32.2034C68.7645 36.1184 68.3857 39.6634 67.8849 43.5506L67.8717 43.6535C67.6103 45.682 67.395 47.3537 67.0566 48.6839C66.7008 50.0827 66.169 51.282 65.1624 52.2678C64.1645 53.245 62.9854 53.7528 61.6119 54.0778C60.3149 54.3848 58.7011 54.5584 56.7558 54.7676L56.6513 54.7789C49.7551 55.5207 44.5465 55.5249 37.6684 54.7745L37.5622 54.7629C35.6442 54.5537 34.0471 54.3795 32.7599 54.0681C31.3915 53.7371 30.2202 53.219 29.2249 52.2296C28.2263 51.2369 27.7018 50.0571 27.3623 48.6788C27.0421 47.3788 26.856 45.7601 26.6318 43.8109L26.6197 43.7053C26.1588 39.6989 25.906 36.0546 26.0325 31.9659C26.037 31.8176 26.0414 31.6709 26.0458 31.5257C26.144 28.24 26.2178 25.7754 28.019 23.7623C28.9747 22.6942 30.1043 22.102 31.449 21.7089C32.6962 21.3444 34.2339 21.1249 36.0541 20.865C36.0914 20.8597 36.1288 20.8543 36.1664 20.849Z"
|
||||
class="fill-n-slate-10"
|
||||
/>
|
||||
<path
|
||||
d="M103.999 77.4062H108.999C110.104 77.4062 110.999 78.3017 110.999 79.4062C110.999 80.5108 110.104 81.4062 108.999 81.4062H103.999V77.4062Z"
|
||||
class="fill-n-slate-9"
|
||||
/>
|
||||
<path
|
||||
d="M63 77.4062H58C56.8954 77.4062 56 78.3017 56 79.4062C56 80.5108 56.8954 81.4062 58 81.4062H63V77.4062Z"
|
||||
class="fill-n-slate-9"
|
||||
/>
|
||||
<path
|
||||
d="M83.999 64.4063C85.4527 63.7978 86.4227 63.4566 87.9986 63.4508C89.5744 63.445 91.136 63.7496 92.5941 64.3472C94.0523 64.9449 95.3784 65.8239 96.4968 66.9341C97.6153 68.0442 98.5041 69.3638 99.1125 70.8175C99.721 72.2711 100.037 73.8304 100.043 75.4062"
|
||||
class="stroke-n-blue-11 fill-n-slate-2"
|
||||
stroke-width="1.6"
|
||||
/>
|
||||
<path
|
||||
d="M67.049 75.0889C67.0432 73.5131 67.3478 71.9515 67.9454 70.4934C68.5431 69.0352 69.4221 67.7091 70.5322 66.5907C71.6424 65.4723 72.962 64.5834 74.4157 63.975C75.8693 63.3666 77.4286 63.0504 79.0044 63.0445C80.5803 63.0387 82.1419 63.3433 83.6 63.941C85.0581 64.5386 86.3843 65.4176 87.5027 66.5278C88.6211 67.638 89.5099 68.9576 90.1184 70.4112C90.7268 71.8649 91.043 73.4241 91.0488 75"
|
||||
class="stroke-n-blue-11 fill-n-slate-2"
|
||||
stroke-width="1.6"
|
||||
/>
|
||||
<path
|
||||
d="M71.4659 74.9421C71.462 73.8915 71.6651 72.8505 72.0635 71.8784C72.462 70.9063 73.048 70.0222 73.7881 69.2766C74.5282 68.531 75.4079 67.9384 76.377 67.5328C77.3462 67.1272 78.3857 66.9164 79.4362 66.9125C80.4868 66.9086 81.5278 67.1117 82.4999 67.5101C83.472 67.9086 84.3561 68.4946 85.1017 69.2347C85.8473 69.9748 86.4399 70.8545 86.8455 71.8236C87.2511 72.7927 87.4619 73.8322 87.4658 74.8828"
|
||||
class="stroke-n-blue-11 fill-n-slate-2"
|
||||
stroke-width="1.6"
|
||||
/>
|
||||
<path
|
||||
d="M88.4304 67.3188C89.4809 67.3149 90.522 67.5179 91.4941 67.9164C92.4662 68.3148 93.3503 68.9008 94.0959 69.6409C94.8415 70.381 95.434 71.2608 95.8397 72.2299C96.2453 73.199 96.4561 74.2385 96.46 75.2891"
|
||||
class="stroke-n-blue-11 fill-n-slate-2"
|
||||
stroke-width="1.6"
|
||||
/>
|
||||
<path
|
||||
d="M75.9473 74.9203C75.9454 74.395 76.0469 73.8745 76.2461 73.3884C76.4453 72.9024 76.7383 72.4603 77.1084 72.0875C77.4785 71.7147 77.9183 71.4184 78.4029 71.2156C78.8874 71.0128 79.4072 70.9074 79.9325 70.9055C80.4578 70.9035 80.9783 71.0051 81.4643 71.2043C81.9504 71.4035 82.3924 71.6965 82.7652 72.0666C83.138 72.4366 83.4343 72.8765 83.6371 73.361C83.8399 73.8456 83.9453 74.3653 83.9473 74.8906"
|
||||
class="stroke-n-blue-11 fill-n-slate-2"
|
||||
stroke-width="1.6"
|
||||
/>
|
||||
<path
|
||||
d="M90.4585 71.6066C90.9445 71.8058 91.3866 72.0988 91.7594 72.4689C92.1322 72.839 92.4284 73.2788 92.6313 73.7634C92.8341 74.2479 92.9395 74.7677 92.9414 75.293"
|
||||
class="stroke-n-blue-11"
|
||||
stroke-width="1.6"
|
||||
/>
|
||||
<path
|
||||
d="M102 74.9062C103.09 74.9062 104.032 75.7999 103.973 76.957C103.853 79.29 103.335 81.588 102.439 83.751C101.939 84.9592 101.499 85.8713 100.931 86.7236C100.363 87.5763 99.6906 88.3338 98.7646 89.2598C96.6889 91.3355 93.3531 92.0527 91.0576 92.0527H76.5576C73.9588 92.0527 70.4405 91.3497 68.3506 89.2598C67.4349 88.3441 66.737 87.5905 66.1416 86.7441C65.5415 85.8911 65.0682 84.9765 64.5605 83.751C63.6646 81.588 63.1471 79.29 63.0273 76.957C62.968 75.7999 63.9098 74.9062 65 74.9062H102Z"
|
||||
class="fill-n-slate-2 stroke-n-slate-9"
|
||||
stroke-width="2"
|
||||
/>
|
||||
|
||||
<g class="default-group">
|
||||
<path
|
||||
d="M103.85 21.1431C104.824 20.8821 105.847 21.3479 106.291 22.2542L112.209 34.3514C112.793 35.544 112.143 36.9734 110.861 37.3171L100.014 40.2235C98.7313 40.5669 97.4538 39.654 97.3628 38.3295L96.4399 24.8937C96.3708 23.8871 97.0247 22.9718 97.9993 22.7107L103.85 21.1431Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M104.528 18.8482C104.84 18.7645 105.124 18.5966 105.347 18.363C106.576 17.0748 105.336 14.9834 103.616 15.4439L95.3492 17.659C93.6293 18.1203 93.6011 20.5515 95.3101 21.0523C95.6201 21.1431 95.9494 21.1468 96.2614 21.0633L104.528 18.8482Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<rect
|
||||
x="95.3496"
|
||||
y="21.2852"
|
||||
width="10.4348"
|
||||
height="2.08696"
|
||||
rx="0.695652"
|
||||
transform="rotate(-15 95.3496 21.2852)"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M103.249 14.4618C103.363 14.5616 103.464 14.6686 103.55 14.7819C103.782 15.0879 103.556 15.4819 103.185 15.5813L99.4893 16.5716L95.7936 17.5618C95.4224 17.6613 95.0296 17.4332 95.0777 17.052C95.0955 16.9109 95.1292 16.7679 95.1785 16.6242C95.3106 16.2393 95.5529 15.8568 95.8916 15.4986C96.2303 15.1403 96.6587 14.8133 97.1525 14.5362C97.6462 14.2592 98.1955 14.0375 98.7691 13.8838C99.3426 13.7301 99.9292 13.6474 100.495 13.6405C101.061 13.6336 101.596 13.7026 102.068 13.8435C102.541 13.9844 102.942 14.1945 103.249 14.4618Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
</g>
|
||||
|
||||
<g class="frame frame-1">
|
||||
<path
|
||||
d="M97.8899 24.7525C97.1819 24.0337 97.0822 22.9139 97.6517 22.0811L105.253 10.9636C106.002 9.8676 107.566 9.72734 108.498 10.6731L116.378 18.6731C117.31 19.619 117.146 21.1806 116.039 21.9133L104.809 29.3463C103.967 29.9032 102.848 29.7861 102.14 29.0673L97.8899 24.7525Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M95.5608 25.2908C95.3341 25.0607 95.0482 24.8971 94.735 24.8184C93.0078 24.3847 91.8006 26.4952 93.05 27.7641L99.0562 33.8613C100.306 35.1296 102.435 33.9543 102.027 32.2207C101.953 31.9063 101.794 31.6181 101.567 31.3879L95.5608 25.2908Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<rect
|
||||
x="102.207"
|
||||
y="32.0742"
|
||||
width="10.4348"
|
||||
height="2.08696"
|
||||
rx="0.695652"
|
||||
transform="rotate(-134.569 102.207 32.0742)"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M92.3768 28.5691C92.4071 28.4202 92.4505 28.2799 92.5066 28.1492C92.6581 27.7962 93.1124 27.7985 93.3821 28.0722L96.0671 30.7979L98.7521 33.5236C99.0218 33.7973 99.0172 34.2515 98.6619 34.3978C98.5304 34.4519 98.3895 34.4932 98.2402 34.5212C97.8402 34.5963 97.3879 34.5743 96.9092 34.4565C96.4304 34.3387 95.9346 34.1274 95.45 33.8347C94.9654 33.542 94.5015 33.1737 94.0848 32.7506C93.668 32.3276 93.3067 31.8582 93.0213 31.3692C92.7359 30.8803 92.5321 30.3813 92.4216 29.9009C92.311 29.4204 92.2958 28.9679 92.3768 28.5691Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<circle cx="85" cy="40" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="91" cy="38" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="91" cy="44" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="85" cy="46" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="79" cy="46" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="88" cy="51" r="1" class="fill-n-slate-11" />
|
||||
</g>
|
||||
|
||||
<g class="frame frame-2">
|
||||
<path
|
||||
d="M97.8879 24.7721C97.1799 24.0532 97.0803 22.9335 97.6497 22.1006L105.251 10.9832C106 9.88713 107.564 9.74688 108.496 10.6927L116.376 18.6926C117.308 19.6385 117.144 21.2001 116.037 21.9329L104.807 29.3658C103.965 29.9227 102.846 29.8056 102.138 29.0868L97.8879 24.7721Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M95.5579 25.3103C95.3311 25.0803 95.0453 24.9167 94.7321 24.838C93.0048 24.4042 91.7977 26.5148 93.0471 27.7836L99.0532 33.8808C100.303 35.1491 102.432 33.9738 102.024 32.2403C101.95 31.9259 101.791 31.6376 101.564 31.4075L95.5579 25.3103Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<rect
|
||||
x="102.205"
|
||||
y="32.0938"
|
||||
width="10.4348"
|
||||
height="2.08696"
|
||||
rx="0.695652"
|
||||
transform="rotate(-134.569 102.205 32.0938)"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M92.3758 28.5886C92.4061 28.4397 92.4495 28.2994 92.5056 28.1687C92.6572 27.8157 93.1115 27.818 93.3811 28.0917L96.0661 30.8174L98.7512 33.5431C99.0208 33.8168 99.0162 34.2711 98.661 34.4173C98.5294 34.4714 98.3885 34.5128 98.2392 34.5408C97.8392 34.6158 97.3869 34.5938 96.9082 34.476C96.4295 34.3582 95.9336 34.1469 95.449 33.8542C94.9644 33.5616 94.5005 33.1932 94.0838 32.7702C93.6671 32.3471 93.3057 31.8777 93.0203 31.3888C92.735 30.8998 92.5312 30.4009 92.4206 29.9204C92.31 29.44 92.2948 28.9874 92.3758 28.5886Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<circle cx="84" cy="42" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="90" cy="40" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="90" cy="46" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="84" cy="48" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="78" cy="48" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="87" cy="53" r="1" class="fill-n-slate-11" />
|
||||
</g>
|
||||
|
||||
<g class="frame frame-3">
|
||||
<path
|
||||
d="M97.8879 24.7486C97.1799 24.0298 97.0803 22.91 97.6497 22.0771L105.251 10.9597C106 9.86369 107.564 9.72344 108.496 10.6692L116.376 18.6692C117.308 19.6151 117.144 21.1767 116.037 21.9094L104.807 29.3424C103.965 29.8993 102.846 29.7822 102.138 29.0634L97.8879 24.7486Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M95.5579 25.2908C95.3311 25.0607 95.0453 24.8971 94.7321 24.8184C93.0048 24.3847 91.7977 26.4952 93.0471 27.7641L99.0532 33.8613C100.303 35.1296 102.432 33.9543 102.024 32.2207C101.95 31.9063 101.791 31.6181 101.564 31.3879L95.5579 25.2908Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<rect
|
||||
x="102.205"
|
||||
y="32.0703"
|
||||
width="10.4348"
|
||||
height="2.08696"
|
||||
rx="0.695652"
|
||||
transform="rotate(-134.569 102.205 32.0703)"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<path
|
||||
d="M92.3758 28.5691C92.4061 28.4202 92.4495 28.2799 92.5056 28.1492C92.6572 27.7962 93.1115 27.7985 93.3811 28.0722L96.0661 30.7979L98.7512 33.5236C99.0208 33.7973 99.0162 34.2515 98.661 34.3978C98.5294 34.4519 98.3885 34.4932 98.2392 34.5212C97.8392 34.5963 97.3869 34.5743 96.9082 34.4565C96.4295 34.3387 95.9336 34.1274 95.449 33.8347C94.9644 33.542 94.5005 33.1737 94.0838 32.7506C93.6671 32.3276 93.3057 31.8582 93.0203 31.3692C92.735 30.8803 92.5312 30.3813 92.4206 29.9009C92.31 29.4204 92.2948 28.9679 92.3758 28.5691Z"
|
||||
class="stroke-n-slate-11 fill-n-slate-2"
|
||||
stroke-width="1.43699"
|
||||
/>
|
||||
<circle cx="82" cy="44" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="88" cy="42" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="88" cy="48" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="82" cy="50" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="76" cy="50" r="1" class="fill-n-slate-11" />
|
||||
<circle cx="85" cy="55" r="1" class="fill-n-slate-11" />
|
||||
</g>
|
||||
|
||||
<defs>
|
||||
<radialGradient
|
||||
id="paint0_radial_720_25701"
|
||||
cx="0"
|
||||
cy="0"
|
||||
r="1"
|
||||
gradientUnits="userSpaceOnUse"
|
||||
gradientTransform="translate(68 54) rotate(90) scale(54 68)"
|
||||
>
|
||||
<stop
|
||||
offset="0.527769"
|
||||
stop-color="var(--gradient-end)"
|
||||
stop-opacity="0.9"
|
||||
/>
|
||||
<stop offset="1" stop-color="var(--gradient-end)" stop-opacity="0" />
|
||||
</radialGradient>
|
||||
</defs>
|
||||
</svg>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
svg {
|
||||
--gradient-start: #fcfcfd;
|
||||
--gradient-end: #fcfcfd;
|
||||
}
|
||||
|
||||
body.dark svg,
|
||||
.htw-dark svg {
|
||||
--gradient-start: #121213;
|
||||
--gradient-end: #121213;
|
||||
}
|
||||
|
||||
.svg-wrapper {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
cursor: pointer;
|
||||
user-select: none;
|
||||
-webkit-tap-highlight-color: transparent;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.default-group {
|
||||
opacity: 0;
|
||||
transition: opacity 120ms linear;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.frame {
|
||||
opacity: 0;
|
||||
animation-name: frameVisible;
|
||||
animation-duration: 600ms;
|
||||
animation-iteration-count: infinite;
|
||||
animation-timing-function: steps(1, end);
|
||||
transform-origin: center;
|
||||
}
|
||||
|
||||
.frame-1 {
|
||||
animation-delay: 0ms;
|
||||
}
|
||||
.frame-2 {
|
||||
animation-delay: 200ms;
|
||||
}
|
||||
.frame-3 {
|
||||
animation-delay: 400ms;
|
||||
}
|
||||
|
||||
@keyframes frameVisible {
|
||||
0% {
|
||||
opacity: 1;
|
||||
}
|
||||
33.333% {
|
||||
opacity: 0;
|
||||
}
|
||||
100% {
|
||||
opacity: 0;
|
||||
}
|
||||
}
|
||||
|
||||
.svg-wrapper.paused .frame {
|
||||
animation-play-state: paused;
|
||||
opacity: 0 !important;
|
||||
}
|
||||
|
||||
.svg-wrapper.paused .default-group {
|
||||
opacity: 1;
|
||||
}
|
||||
.svg-wrapper:not(.paused) .default-group {
|
||||
opacity: 0;
|
||||
}
|
||||
</style>
|
||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,752 @@
|
||||
<template>
|
||||
<div class="svg-wrapper relative" tabindex="0">
|
||||
<div class="absolute z-0 flex-shrink-0">
|
||||
<svg
|
||||
width="auto"
|
||||
height="auto"
|
||||
viewBox="0 0 200 156"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<g opacity="0.5">
|
||||
<g clip-path="url(#clip0_773_34322)">
|
||||
<circle cx="8" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="3" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="9" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="15" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="21" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="27" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="33" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="39" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="45" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="51" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="57" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="63" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="69" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="75" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="81" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="87" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="93" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="99" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="105" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="111" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="117" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="123" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="129" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="135" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="141" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="147" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="8" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="16" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="24" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="32" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="40" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="48" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="56" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="64" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="72" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="80" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="88" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="96" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="104" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="112" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="120" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="128" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="136" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="144" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="152" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="160" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="168" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="176" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="184" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<circle cx="192" cy="153" r="1" class="fill-n-blue-9" />
|
||||
<rect
|
||||
width="200"
|
||||
height="156"
|
||||
fill="url(#paint0_linear_773_34322)"
|
||||
/>
|
||||
<rect
|
||||
width="200"
|
||||
height="156"
|
||||
fill="url(#paint1_linear_773_34322)"
|
||||
/>
|
||||
</g>
|
||||
</g>
|
||||
<defs>
|
||||
<linearGradient
|
||||
id="paint0_linear_773_34322"
|
||||
x1="100"
|
||||
y1="0"
|
||||
x2="100"
|
||||
y2="156"
|
||||
gradientUnits="userSpaceOnUse"
|
||||
>
|
||||
<stop stop-color="var(--gradient-start)" />
|
||||
<stop
|
||||
offset="0.480769"
|
||||
stop-color="var(--gradient-start)"
|
||||
stop-opacity="0"
|
||||
/>
|
||||
<stop offset="1" stop-color="var(--gradient-start)" />
|
||||
</linearGradient>
|
||||
<linearGradient
|
||||
id="paint1_linear_773_34322"
|
||||
x1="0"
|
||||
y1="78"
|
||||
x2="200"
|
||||
y2="78"
|
||||
gradientUnits="userSpaceOnUse"
|
||||
>
|
||||
<stop stop-color="var(--gradient-start)" />
|
||||
<stop
|
||||
offset="0.480769"
|
||||
stop-color="var(--gradient-start)"
|
||||
stop-opacity="0"
|
||||
/>
|
||||
<stop offset="1" stop-color="var(--gradient-start)" />
|
||||
</linearGradient>
|
||||
<clipPath id="clip0_773_34322">
|
||||
<rect width="200" height="156" rx="16" fill="white" />
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
</div>
|
||||
<svg
|
||||
viewBox="0 0 136 108"
|
||||
fill="none"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
aria-hidden="false"
|
||||
focusable="false"
|
||||
class="z-10 relative flex-shrink-0"
|
||||
>
|
||||
<rect width="136" height="108" fill="url(#paint0_radial_797_91519)" />
|
||||
<path
|
||||
d="M58 48.039C58 46.9129 58.9129 46 60.039 46C61.1651 46 62.0779 46.9129 62.0779 48.039V52.7513C62.0779 53.8774 61.1651 54.7902 60.039 54.7902C58.9129 54.7902 58 53.8774 58 52.7513V48.039Z"
|
||||
class="fill-n-slate-10"
|
||||
/>
|
||||
<path
|
||||
d="M64.5244 48.039C64.5244 46.9129 65.4373 46 66.5634 46C67.6895 46 68.6024 46.9129 68.6024 48.039V52.7513C68.6024 53.8774 67.6895 54.7902 66.5634 54.7902C65.4373 54.7902 64.5244 53.8774 64.5244 52.7513V48.039Z"
|
||||
class="fill-n-slate-10"
|
||||
/>
|
||||
<path
|
||||
fill-rule="evenodd"
|
||||
clip-rule="evenodd"
|
||||
d="M79.1862 40.173C71.5933 39.1017 65.4039 39.0594 57.6405 40.1683C55.678 40.4485 54.3796 40.638 53.3896 40.9273C52.4716 41.1956 51.9496 41.5155 51.5177 41.9981C50.6236 42.9974 50.5057 44.1276 50.3838 48.0695C50.264 51.9436 50.5013 55.4156 50.9507 59.3222C51.1901 61.4039 51.3549 62.8088 51.618 63.8769C51.8676 64.89 52.1694 65.4347 52.5889 65.8518C53.0117 66.2721 53.5536 66.5686 54.5482 66.8091C55.6002 67.0636 56.9806 67.2175 59.032 67.4413C65.6668 68.1652 70.6333 68.1615 77.2927 67.4451C79.369 67.2218 80.7719 67.0678 81.8396 66.815C82.8541 66.5749 83.3995 66.2805 83.8165 65.8722C84.2247 65.4724 84.5366 64.921 84.8071 63.8574C85.0894 62.7477 85.2815 61.2793 85.5594 59.1222C86.0553 55.2735 86.4123 51.8935 86.4262 48.1908C86.4411 44.232 86.3601 43.1056 85.4558 42.0608C85.019 41.5562 84.488 41.2244 83.5491 40.9481C82.5352 40.6497 81.2017 40.4573 79.1862 40.173ZM57.1664 36.849C65.2537 35.6939 71.7606 35.7391 79.6546 36.8529L79.7701 36.8692C81.6392 37.1328 83.2183 37.3555 84.4958 37.7315C85.8745 38.1372 87.0274 38.7532 87.991 39.8665C89.8087 41.9666 89.7969 44.4583 89.7811 47.7688C89.7804 47.9121 89.7797 48.057 89.7792 48.2034C89.7645 52.1184 89.3857 55.6634 88.8849 59.5506L88.8717 59.6535C88.6103 61.682 88.395 63.3537 88.0566 64.6839C87.7008 66.0827 87.169 67.282 86.1624 68.2678C85.1645 69.245 83.9854 69.7528 82.6119 70.0778C81.3149 70.3848 79.7011 70.5584 77.7558 70.7676L77.6513 70.7789C70.7551 71.5207 65.5465 71.5249 58.6684 70.7745L58.5622 70.7629C56.6442 70.5537 55.0471 70.3795 53.7599 70.0681C52.3915 69.7371 51.2202 69.219 50.2249 68.2296C49.2263 67.2369 48.7018 66.0571 48.3623 64.6788C48.0421 63.3788 47.856 61.7601 47.6318 59.8109L47.6197 59.7053C47.1588 55.6989 46.906 52.0546 47.0325 47.9659C47.037 47.8176 47.0414 47.6709 47.0458 47.5257C47.144 44.24 47.2178 41.7754 49.019 39.7623C49.9747 38.6942 51.1043 38.102 52.449 37.7089C53.6962 37.3444 55.2339 37.1249 57.0541 36.865C57.0914 36.8597 57.1288 36.8543 57.1664 36.849Z"
|
||||
class="fill-n-slate-10"
|
||||
/>
|
||||
<defs>
|
||||
<radialGradient
|
||||
id="paint0_radial_797_91519"
|
||||
cx="0"
|
||||
cy="0"
|
||||
r="1"
|
||||
gradientUnits="userSpaceOnUse"
|
||||
gradientTransform="translate(68 54) rotate(90) scale(54 68)"
|
||||
>
|
||||
<stop
|
||||
offset="0.527769"
|
||||
stop-color="var(--gradient-end)"
|
||||
stop-opacity="0.9"
|
||||
/>
|
||||
<stop offset="1" stop-color="var(--gradient-end)" stop-opacity="0" />
|
||||
</radialGradient>
|
||||
</defs>
|
||||
</svg>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
svg {
|
||||
--gradient-start: #fcfcfd;
|
||||
--gradient-end: #fcfcfd;
|
||||
}
|
||||
|
||||
body.dark svg,
|
||||
.htw-dark svg {
|
||||
--gradient-start: #121213;
|
||||
--gradient-end: #121213;
|
||||
}
|
||||
|
||||
.svg-wrapper {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
cursor: pointer;
|
||||
user-select: none;
|
||||
-webkit-tap-highlight-color: transparent;
|
||||
outline: none;
|
||||
}
|
||||
</style>
|
||||
@@ -52,10 +52,10 @@ const handleBreadcrumbClick = item => {
|
||||
|
||||
<template>
|
||||
<section
|
||||
class="mt-4 px-10 flex flex-col w-full h-screen overflow-y-auto bg-n-background"
|
||||
class="px-6 flex flex-col w-full h-screen overflow-y-auto bg-n-background"
|
||||
>
|
||||
<div class="max-w-[60rem] mx-auto flex flex-col w-full h-full mb-4">
|
||||
<header class="mb-7 sticky top-0 z-10 bg-n-background">
|
||||
<header class="mb-7 sticky top-0 bg-n-background pt-4 z-20">
|
||||
<Breadcrumb :items="breadcrumbItems" @click="handleBreadcrumbClick" />
|
||||
</header>
|
||||
<main class="flex gap-16 w-full flex-1 pb-16">
|
||||
|
||||
@@ -4,6 +4,10 @@ import { useToggle } from '@vueuse/core';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { dynamicTime } from 'shared/helpers/timeHelper';
|
||||
import { usePolicy } from 'dashboard/composables/usePolicy';
|
||||
import {
|
||||
isPdfDocument,
|
||||
formatDocumentLink,
|
||||
} from 'shared/helpers/documentHelper';
|
||||
|
||||
import CardLayout from 'dashboard/components-next/CardLayout.vue';
|
||||
import DropdownMenu from 'dashboard/components-next/dropdown-menu/DropdownMenu.vue';
|
||||
@@ -63,6 +67,11 @@ const menuItems = computed(() => {
|
||||
|
||||
const createdAt = computed(() => dynamicTime(props.createdAt));
|
||||
|
||||
const displayLink = computed(() => formatDocumentLink(props.externalLink));
|
||||
const linkIcon = computed(() =>
|
||||
isPdfDocument(props.externalLink) ? 'i-ph-file-pdf' : 'i-ph-link-simple'
|
||||
);
|
||||
|
||||
const handleAction = ({ action, value }) => {
|
||||
toggleDropdown(false);
|
||||
emit('action', { action, value, id: props.id });
|
||||
@@ -71,14 +80,14 @@ const handleAction = ({ action, value }) => {
|
||||
|
||||
<template>
|
||||
<CardLayout>
|
||||
<div class="flex justify-between w-full gap-1">
|
||||
<div class="flex gap-1 justify-between w-full">
|
||||
<span class="text-base text-n-slate-12 line-clamp-1">
|
||||
{{ name }}
|
||||
</span>
|
||||
<div class="flex items-center gap-2">
|
||||
<div class="flex gap-2 items-center">
|
||||
<div
|
||||
v-on-clickaway="() => toggleDropdown(false)"
|
||||
class="relative flex items-center group"
|
||||
class="flex relative items-center group"
|
||||
>
|
||||
<Button
|
||||
icon="i-lucide-ellipsis-vertical"
|
||||
@@ -90,26 +99,26 @@ const handleAction = ({ action, value }) => {
|
||||
<DropdownMenu
|
||||
v-if="showActionsDropdown"
|
||||
:menu-items="menuItems"
|
||||
class="mt-1 ltr:right-0 rtl:left-0 xl:ltr:right-0 xl:rtl:left-0 top-full"
|
||||
class="top-full mt-1 ltr:right-0 rtl:left-0 xl:ltr:right-0 xl:rtl:left-0"
|
||||
@action="handleAction($event)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="flex items-center justify-between w-full gap-4">
|
||||
<div class="flex gap-4 justify-between items-center w-full">
|
||||
<span
|
||||
class="text-sm shrink-0 truncate text-n-slate-11 flex items-center gap-1"
|
||||
class="flex gap-1 items-center text-sm truncate shrink-0 text-n-slate-11"
|
||||
>
|
||||
<i class="i-woot-captain" />
|
||||
{{ assistant?.name || '' }}
|
||||
</span>
|
||||
<span
|
||||
class="text-n-slate-11 text-sm truncate flex justify-start flex-1 items-center gap-1"
|
||||
class="flex flex-1 gap-1 justify-start items-center text-sm truncate text-n-slate-11"
|
||||
>
|
||||
<i class="i-ph-link-simple shrink-0" />
|
||||
<span class="truncate">{{ externalLink }}</span>
|
||||
<i :class="linkIcon" class="shrink-0" />
|
||||
<span class="truncate">{{ displayLink }}</span>
|
||||
</span>
|
||||
<div class="shrink-0 text-sm text-n-slate-11 line-clamp-1">
|
||||
<div class="text-sm shrink-0 text-n-slate-11 line-clamp-1">
|
||||
{{ createdAt }}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<script setup>
|
||||
import { computed, h, reactive } from 'vue';
|
||||
import { computed, h, reactive, ref } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { useToggle } from '@vueuse/core';
|
||||
import { useToggle, useElementSize } from '@vueuse/core';
|
||||
import { useVuelidate } from '@vuelidate/core';
|
||||
import { required, minLength } from '@vuelidate/validators';
|
||||
import { useMessageFormatter } from 'shared/composables/useMessageFormatter';
|
||||
@@ -11,6 +11,7 @@ import TextArea from 'dashboard/components-next/textarea/TextArea.vue';
|
||||
import Editor from 'dashboard/components-next/Editor/Editor.vue';
|
||||
import CardLayout from 'dashboard/components-next/CardLayout.vue';
|
||||
import Checkbox from 'dashboard/components-next/checkbox/Checkbox.vue';
|
||||
import Icon from 'dashboard/components-next/icon/Icon.vue';
|
||||
|
||||
const props = defineProps({
|
||||
id: {
|
||||
@@ -31,7 +32,7 @@ const props = defineProps({
|
||||
},
|
||||
tools: {
|
||||
type: Array,
|
||||
required: true,
|
||||
default: () => [],
|
||||
},
|
||||
selectable: {
|
||||
type: Boolean,
|
||||
@@ -60,7 +61,13 @@ const state = reactive({
|
||||
instruction: '',
|
||||
});
|
||||
|
||||
const instructionContentRef = ref();
|
||||
|
||||
const [isEditing, toggleEditing] = useToggle();
|
||||
const [isInstructionExpanded, toggleInstructionExpanded] = useToggle();
|
||||
|
||||
const { height: contentHeight } = useElementSize(instructionContentRef);
|
||||
const needsOverlay = computed(() => contentHeight.value > 160);
|
||||
|
||||
const startEdit = () => {
|
||||
Object.assign(state, {
|
||||
@@ -111,7 +118,7 @@ const LINK_INSTRUCTION_CLASS =
|
||||
|
||||
const renderInstruction = instruction => () =>
|
||||
h('p', {
|
||||
class: `text-sm text-n-slate-12 py-4 mb-0 [&_ol]:list-decimal ${LINK_INSTRUCTION_CLASS}`,
|
||||
class: `text-sm text-n-slate-12 py-4 mb-0 prose prose-sm min-w-0 break-words max-w-none ${LINK_INSTRUCTION_CLASS}`,
|
||||
innerHTML: instruction,
|
||||
});
|
||||
</script>
|
||||
@@ -157,8 +164,38 @@ const renderInstruction = instruction => () =>
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<component :is="renderInstruction(formatMessage(instruction, false))" />
|
||||
<span class="text-sm text-n-slate-11 font-medium mb-1">
|
||||
|
||||
<div
|
||||
class="relative overflow-hidden transition-all duration-300 ease-in-out group/expandable"
|
||||
:class="{ 'cursor-pointer': needsOverlay }"
|
||||
:style="{
|
||||
maxHeight: isInstructionExpanded ? `${contentHeight}px` : '10rem',
|
||||
}"
|
||||
@click="needsOverlay ? toggleInstructionExpanded() : null"
|
||||
>
|
||||
<div ref="instructionContentRef">
|
||||
<component
|
||||
:is="renderInstruction(formatMessage(instruction, false))"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div
|
||||
class="absolute bottom-0 w-full flex items-end justify-center text-xs text-n-slate-11 bg-gradient-to-t h-40 from-n-solid-2 via-n-solid-2 via-10% to-transparent transition-all duration-500 ease-in-out px-2 py-1 rounded pointer-events-none"
|
||||
:class="{
|
||||
'visible opacity-100': !isInstructionExpanded,
|
||||
'invisible opacity-0': isInstructionExpanded || !needsOverlay,
|
||||
}"
|
||||
>
|
||||
<Icon
|
||||
icon="i-lucide-chevron-down"
|
||||
class="text-n-slate-7 mb-4 size-4 group-hover/expandable:text-n-slate-11 transition-colors duration-200"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<span
|
||||
v-if="tools?.length"
|
||||
class="text-sm text-n-slate-11 font-medium mb-1"
|
||||
>
|
||||
{{ t('CAPTAIN.ASSISTANTS.SCENARIOS.ADD.SUGGESTED.TOOLS_USED') }}
|
||||
{{ tools?.map(tool => `@${tool}`).join(', ') }}
|
||||
</span>
|
||||
|
||||
@@ -3,6 +3,7 @@ import { ref } from 'vue';
|
||||
import { useStore } from 'dashboard/composables/store';
|
||||
import { useAlert } from 'dashboard/composables';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { parseAPIErrorResponse } from 'dashboard/store/utils/api';
|
||||
|
||||
import Dialog from 'dashboard/components-next/dialog/Dialog.vue';
|
||||
import DocumentForm from './DocumentForm.vue';
|
||||
@@ -12,7 +13,6 @@ const { t } = useI18n();
|
||||
const store = useStore();
|
||||
|
||||
const dialogRef = ref(null);
|
||||
const documentForm = ref(null);
|
||||
|
||||
const i18nKey = 'CAPTAIN.DOCUMENTS.CREATE';
|
||||
|
||||
@@ -23,7 +23,7 @@ const handleSubmit = async newDocument => {
|
||||
dialogRef.value.close();
|
||||
} catch (error) {
|
||||
const errorMessage =
|
||||
error?.response?.message || t(`${i18nKey}.ERROR_MESSAGE`);
|
||||
parseAPIErrorResponse(error) || t(`${i18nKey}.ERROR_MESSAGE`);
|
||||
useAlert(errorMessage);
|
||||
}
|
||||
};
|
||||
@@ -48,11 +48,7 @@ defineExpose({ dialogRef });
|
||||
:show-confirm-button="false"
|
||||
@close="handleClose"
|
||||
>
|
||||
<DocumentForm
|
||||
ref="documentForm"
|
||||
@submit="handleSubmit"
|
||||
@cancel="handleCancel"
|
||||
/>
|
||||
<DocumentForm @submit="handleSubmit" @cancel="handleCancel" />
|
||||
<template #footer />
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
@@ -1,9 +1,10 @@
|
||||
<script setup>
|
||||
import { reactive, computed } from 'vue';
|
||||
import { reactive, computed, ref, nextTick } from 'vue';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { useVuelidate } from '@vuelidate/core';
|
||||
import { required, minLength, url } from '@vuelidate/validators';
|
||||
import { required, minLength, requiredIf, url } from '@vuelidate/validators';
|
||||
import { useMapGetter } from 'dashboard/composables/store';
|
||||
import { useAlert } from 'dashboard/composables';
|
||||
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
import Button from 'dashboard/components-next/button/Button.vue';
|
||||
@@ -11,6 +12,8 @@ import ComboBox from 'dashboard/components-next/combobox/ComboBox.vue';
|
||||
|
||||
const emit = defineEmits(['submit', 'cancel']);
|
||||
|
||||
const MAX_FILE_SIZE = 10 * 1024 * 1024; // 10MB
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const formState = {
|
||||
@@ -20,14 +23,25 @@ const formState = {
|
||||
|
||||
const initialState = {
|
||||
name: '',
|
||||
url: '',
|
||||
assistantId: null,
|
||||
documentType: 'url',
|
||||
pdfFile: null,
|
||||
};
|
||||
|
||||
const state = reactive({ ...initialState });
|
||||
const fileInputRef = ref(null);
|
||||
|
||||
const validationRules = {
|
||||
url: { required, url, minLength: minLength(1) },
|
||||
url: {
|
||||
required: requiredIf(() => state.documentType === 'url'),
|
||||
url: requiredIf(() => state.documentType === 'url' && url),
|
||||
minLength: requiredIf(() => state.documentType === 'url' && minLength(1)),
|
||||
},
|
||||
assistantId: { required },
|
||||
pdfFile: {
|
||||
required: requiredIf(() => state.documentType === 'pdf'),
|
||||
},
|
||||
};
|
||||
|
||||
const assistantList = computed(() =>
|
||||
@@ -37,10 +51,17 @@ const assistantList = computed(() =>
|
||||
}))
|
||||
);
|
||||
|
||||
const documentTypeOptions = [
|
||||
{ value: 'url', label: t('CAPTAIN.DOCUMENTS.FORM.TYPE.URL') },
|
||||
{ value: 'pdf', label: t('CAPTAIN.DOCUMENTS.FORM.TYPE.PDF') },
|
||||
];
|
||||
|
||||
const v$ = useVuelidate(validationRules, state);
|
||||
|
||||
const isLoading = computed(() => formState.uiFlags.value.creatingItem);
|
||||
|
||||
const hasPdfFileError = computed(() => v$.value.pdfFile.$error);
|
||||
|
||||
const getErrorMessage = (field, errorKey) => {
|
||||
return v$.value[field].$error
|
||||
? t(`CAPTAIN.DOCUMENTS.FORM.${errorKey}.ERROR`)
|
||||
@@ -50,14 +71,57 @@ const getErrorMessage = (field, errorKey) => {
|
||||
const formErrors = computed(() => ({
|
||||
url: getErrorMessage('url', 'URL'),
|
||||
assistantId: getErrorMessage('assistantId', 'ASSISTANT'),
|
||||
pdfFile: getErrorMessage('pdfFile', 'PDF_FILE'),
|
||||
}));
|
||||
|
||||
const handleCancel = () => emit('cancel');
|
||||
|
||||
const prepareDocumentDetails = () => ({
|
||||
external_link: state.url,
|
||||
assistant_id: state.assistantId,
|
||||
});
|
||||
const handleFileChange = event => {
|
||||
const file = event.target.files[0];
|
||||
if (file) {
|
||||
if (file.type !== 'application/pdf') {
|
||||
useAlert(t('CAPTAIN.DOCUMENTS.FORM.PDF_FILE.INVALID_TYPE'));
|
||||
event.target.value = '';
|
||||
return;
|
||||
}
|
||||
if (file.size > MAX_FILE_SIZE) {
|
||||
// 10MB
|
||||
useAlert(t('CAPTAIN.DOCUMENTS.FORM.PDF_FILE.TOO_LARGE'));
|
||||
event.target.value = '';
|
||||
return;
|
||||
}
|
||||
state.pdfFile = file;
|
||||
state.name = file.name.replace(/\.pdf$/i, '');
|
||||
}
|
||||
};
|
||||
|
||||
const openFileDialog = () => {
|
||||
// Use nextTick to ensure the ref is available
|
||||
nextTick(() => {
|
||||
if (fileInputRef.value) {
|
||||
fileInputRef.value.click();
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const prepareDocumentDetails = () => {
|
||||
const formData = new FormData();
|
||||
formData.append('document[assistant_id]', state.assistantId);
|
||||
|
||||
if (state.documentType === 'url') {
|
||||
formData.append('document[external_link]', state.url);
|
||||
formData.append('document[name]', state.name || state.url);
|
||||
} else {
|
||||
formData.append('document[pdf_file]', state.pdfFile);
|
||||
formData.append(
|
||||
'document[name]',
|
||||
state.name || state.pdfFile.name.replace('.pdf', '')
|
||||
);
|
||||
// No need to send external_link for PDF - it's auto-generated in the backend
|
||||
}
|
||||
|
||||
return formData;
|
||||
};
|
||||
|
||||
const handleSubmit = async () => {
|
||||
const isFormValid = await v$.value.$validate();
|
||||
@@ -71,13 +135,89 @@ const handleSubmit = async () => {
|
||||
|
||||
<template>
|
||||
<form class="flex flex-col gap-4" @submit.prevent="handleSubmit">
|
||||
<div class="flex flex-col gap-1">
|
||||
<label
|
||||
for="documentType"
|
||||
class="mb-0.5 text-sm font-medium text-n-slate-12"
|
||||
>
|
||||
{{ t('CAPTAIN.DOCUMENTS.FORM.TYPE.LABEL') }}
|
||||
</label>
|
||||
<ComboBox
|
||||
id="documentType"
|
||||
v-model="state.documentType"
|
||||
:options="documentTypeOptions"
|
||||
class="[&>div>button]:bg-n-alpha-black2"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<Input
|
||||
v-if="state.documentType === 'url'"
|
||||
v-model="state.url"
|
||||
:label="t('CAPTAIN.DOCUMENTS.FORM.URL.LABEL')"
|
||||
:placeholder="t('CAPTAIN.DOCUMENTS.FORM.URL.PLACEHOLDER')"
|
||||
:message="formErrors.url"
|
||||
:message-type="formErrors.url ? 'error' : 'info'"
|
||||
/>
|
||||
|
||||
<div v-if="state.documentType === 'pdf'" class="flex flex-col gap-2">
|
||||
<label class="text-sm font-medium text-n-slate-12">
|
||||
{{ t('CAPTAIN.DOCUMENTS.FORM.PDF_FILE.LABEL') }}
|
||||
</label>
|
||||
<div class="relative">
|
||||
<input
|
||||
ref="fileInputRef"
|
||||
type="file"
|
||||
accept=".pdf"
|
||||
class="hidden"
|
||||
@change="handleFileChange"
|
||||
/>
|
||||
<Button
|
||||
type="button"
|
||||
:color="hasPdfFileError ? 'ruby' : 'slate'"
|
||||
:variant="hasPdfFileError ? 'outline' : 'solid'"
|
||||
class="!w-full !h-auto !justify-between !py-4"
|
||||
@click="openFileDialog"
|
||||
>
|
||||
<template #default>
|
||||
<div class="flex gap-2 items-center">
|
||||
<div
|
||||
class="flex justify-center items-center w-10 h-10 rounded-lg bg-n-slate-3"
|
||||
>
|
||||
<i class="text-xl i-ph-file-pdf text-n-slate-11" />
|
||||
</div>
|
||||
<div class="flex flex-col flex-1 gap-1 items-start">
|
||||
<p class="m-0 text-sm font-medium text-n-slate-12">
|
||||
{{
|
||||
state.pdfFile
|
||||
? state.pdfFile.name
|
||||
: t('CAPTAIN.DOCUMENTS.FORM.PDF_FILE.CHOOSE_FILE')
|
||||
}}
|
||||
</p>
|
||||
<p class="m-0 text-xs text-n-slate-11">
|
||||
{{
|
||||
state.pdfFile
|
||||
? `${(state.pdfFile.size / 1024 / 1024).toFixed(2)} MB`
|
||||
: t('CAPTAIN.DOCUMENTS.FORM.PDF_FILE.HELP_TEXT')
|
||||
}}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<i class="i-lucide-upload text-n-slate-11" />
|
||||
</template>
|
||||
</Button>
|
||||
</div>
|
||||
<p v-if="formErrors.pdfFile" class="text-xs text-n-ruby-9">
|
||||
{{ formErrors.pdfFile }}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<Input
|
||||
v-model="state.name"
|
||||
:label="t('CAPTAIN.DOCUMENTS.FORM.NAME.LABEL')"
|
||||
:placeholder="t('CAPTAIN.DOCUMENTS.FORM.NAME.PLACEHOLDER')"
|
||||
/>
|
||||
|
||||
<div class="flex flex-col gap-1">
|
||||
<label for="assistant" class="mb-0.5 text-sm font-medium text-n-slate-12">
|
||||
{{ t('CAPTAIN.DOCUMENTS.FORM.ASSISTANT.LABEL') }}
|
||||
@@ -88,12 +228,12 @@ const handleSubmit = async () => {
|
||||
:options="assistantList"
|
||||
:has-error="!!formErrors.assistantId"
|
||||
:placeholder="t('CAPTAIN.DOCUMENTS.FORM.ASSISTANT.PLACEHOLDER')"
|
||||
class="[&>div>button]:bg-n-alpha-black2 [&>div>button:not(.focused)]:dark:outline-n-weak [&>div>button:not(.focused)]:hover:!outline-n-slate-6"
|
||||
class="[&>div>button]:bg-n-alpha-black2"
|
||||
:message="formErrors.assistantId"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="flex items-center justify-between w-full gap-3">
|
||||
<div class="flex gap-3 justify-between items-center w-full">
|
||||
<Button
|
||||
type="button"
|
||||
variant="faded"
|
||||
|
||||
@@ -96,8 +96,12 @@ watch(
|
||||
:label="selectedLabel"
|
||||
trailing-icon
|
||||
:disabled="disabled"
|
||||
class="justify-between w-full !px-3 !py-2.5 text-n-slate-12 font-normal group-hover/combobox:border-n-slate-6 [&:not(.focused)]:hover:enabled:outline-n-slate-6 [&:not(.focused)]:dark:hover:enabled:outline-n-slate-6 [&:not(.focused)]:dark:outline-n-weak focus:outline-n-brand"
|
||||
:class="{ focused: open }"
|
||||
class="justify-between w-full !px-3 !py-2.5 text-n-slate-12 font-normal group-hover/combobox:border-n-slate-6 focus:outline-n-brand"
|
||||
:class="{
|
||||
focused: open,
|
||||
'[&:not(.focused)]:dark:outline-n-weak [&:not(.focused)]:hover:enabled:outline-n-slate-6 [&:not(.focused)]:dark:hover:enabled:outline-n-slate-6':
|
||||
!hasError,
|
||||
}"
|
||||
:icon="open ? 'i-lucide-chevron-up' : 'i-lucide-chevron-down'"
|
||||
@click="toggleDropdown"
|
||||
/>
|
||||
|
||||
@@ -0,0 +1,278 @@
|
||||
<script setup>
|
||||
import { ref, computed, onMounted, watch } from 'vue';
|
||||
import { useVuelidate } from '@vuelidate/core';
|
||||
import { requiredIf } from '@vuelidate/validators';
|
||||
import { useI18n } from 'vue-i18n';
|
||||
import { extractFilenameFromUrl } from 'dashboard/helper/URLHelper';
|
||||
import { TWILIO_CONTENT_TEMPLATE_TYPES } from 'shared/constants/messages';
|
||||
|
||||
import Input from 'dashboard/components-next/input/Input.vue';
|
||||
|
||||
const props = defineProps({
|
||||
template: {
|
||||
type: Object,
|
||||
default: () => ({}),
|
||||
validator: value => {
|
||||
if (!value || typeof value !== 'object') return false;
|
||||
if (!value.friendly_name) return false;
|
||||
return true;
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const emit = defineEmits(['sendMessage', 'resetTemplate', 'back']);
|
||||
|
||||
const VARIABLE_PATTERN = /{{([^}]+)}}/g;
|
||||
|
||||
const { t } = useI18n();
|
||||
|
||||
const processedParams = ref({});
|
||||
|
||||
const languageLabel = computed(() => {
|
||||
return `${t('CONTENT_TEMPLATES.PARSER.LANGUAGE')}: ${props.template.language || 'en'}`;
|
||||
});
|
||||
|
||||
const categoryLabel = computed(() => {
|
||||
return `${t('CONTENT_TEMPLATES.PARSER.CATEGORY')}: ${props.template.category || 'utility'}`;
|
||||
});
|
||||
|
||||
const templateBody = computed(() => {
|
||||
return props.template.body || '';
|
||||
});
|
||||
|
||||
const hasMediaTemplate = computed(() => {
|
||||
return props.template.template_type === TWILIO_CONTENT_TEMPLATE_TYPES.MEDIA;
|
||||
});
|
||||
|
||||
const hasVariables = computed(() => {
|
||||
return templateBody.value?.match(VARIABLE_PATTERN) !== null;
|
||||
});
|
||||
|
||||
const mediaVariableKey = computed(() => {
|
||||
if (!hasMediaTemplate.value) return null;
|
||||
const mediaUrl = props.template?.types?.['twilio/media']?.media?.[0];
|
||||
if (!mediaUrl) return null;
|
||||
return mediaUrl.match(/{{(\d+)}}/)?.[1] ?? null;
|
||||
});
|
||||
|
||||
const hasMediaVariable = computed(() => {
|
||||
return hasMediaTemplate.value && mediaVariableKey.value !== null;
|
||||
});
|
||||
|
||||
const templateMediaUrl = computed(() => {
|
||||
if (!hasMediaTemplate.value) return '';
|
||||
|
||||
return props.template?.types?.['twilio/media']?.media?.[0] || '';
|
||||
});
|
||||
|
||||
const variablePattern = computed(() => {
|
||||
if (!hasVariables.value) return [];
|
||||
const matches = templateBody.value.match(VARIABLE_PATTERN) || [];
|
||||
return matches.map(match => match.replace(/[{}]/g, ''));
|
||||
});
|
||||
|
||||
const renderedTemplate = computed(() => {
|
||||
let rendered = templateBody.value;
|
||||
if (processedParams.value && Object.keys(processedParams.value).length > 0) {
|
||||
// Replace variables in the format {{1}}, {{2}}, etc.
|
||||
rendered = rendered.replace(VARIABLE_PATTERN, (match, variable) => {
|
||||
const cleanVariable = variable.trim();
|
||||
return processedParams.value[cleanVariable] || match;
|
||||
});
|
||||
}
|
||||
return rendered;
|
||||
});
|
||||
|
||||
const isFormInvalid = computed(() => {
|
||||
if (!hasVariables.value && !hasMediaVariable.value) return false;
|
||||
|
||||
if (hasVariables.value) {
|
||||
const hasEmptyVariable = variablePattern.value.some(
|
||||
variable => !processedParams.value[variable]
|
||||
);
|
||||
if (hasEmptyVariable) return true;
|
||||
}
|
||||
|
||||
if (
|
||||
hasMediaVariable.value &&
|
||||
mediaVariableKey.value &&
|
||||
!processedParams.value[mediaVariableKey.value]
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
});
|
||||
|
||||
const v$ = useVuelidate(
|
||||
{
|
||||
processedParams: {
|
||||
requiredIfKeysPresent: requiredIf(
|
||||
() => hasVariables.value || hasMediaVariable.value
|
||||
),
|
||||
},
|
||||
},
|
||||
{ processedParams }
|
||||
);
|
||||
|
||||
const initializeTemplateParameters = () => {
|
||||
processedParams.value = {};
|
||||
|
||||
if (hasVariables.value) {
|
||||
variablePattern.value.forEach(variable => {
|
||||
processedParams.value[variable] = '';
|
||||
});
|
||||
}
|
||||
|
||||
if (hasMediaVariable.value && mediaVariableKey.value) {
|
||||
processedParams.value[mediaVariableKey.value] = '';
|
||||
}
|
||||
};
|
||||
|
||||
const sendMessage = () => {
|
||||
v$.value.$touch();
|
||||
if (v$.value.$invalid || isFormInvalid.value) return;
|
||||
|
||||
const { friendly_name, language } = props.template;
|
||||
|
||||
// Process parameters and extract filename from media URL if needed
|
||||
const processedParameters = { ...processedParams.value };
|
||||
|
||||
// For media templates, extract filename from full URL
|
||||
if (
|
||||
hasMediaVariable.value &&
|
||||
mediaVariableKey.value &&
|
||||
processedParameters[mediaVariableKey.value]
|
||||
) {
|
||||
processedParameters[mediaVariableKey.value] = extractFilenameFromUrl(
|
||||
processedParameters[mediaVariableKey.value]
|
||||
);
|
||||
}
|
||||
|
||||
const payload = {
|
||||
message: renderedTemplate.value,
|
||||
templateParams: {
|
||||
name: friendly_name,
|
||||
language,
|
||||
processed_params: processedParameters,
|
||||
},
|
||||
};
|
||||
emit('sendMessage', payload);
|
||||
};
|
||||
|
||||
const resetTemplate = () => {
|
||||
emit('resetTemplate');
|
||||
};
|
||||
|
||||
const goBack = () => {
|
||||
emit('back');
|
||||
};
|
||||
|
||||
onMounted(initializeTemplateParameters);
|
||||
|
||||
watch(
|
||||
() => props.template,
|
||||
() => {
|
||||
initializeTemplateParameters();
|
||||
v$.value.$reset();
|
||||
},
|
||||
{ deep: true }
|
||||
);
|
||||
|
||||
defineExpose({
|
||||
processedParams,
|
||||
hasVariables,
|
||||
hasMediaTemplate,
|
||||
renderedTemplate,
|
||||
v$,
|
||||
sendMessage,
|
||||
resetTemplate,
|
||||
goBack,
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div>
|
||||
<div class="flex flex-col gap-4 p-4 mb-4 rounded-lg bg-n-alpha-black2">
|
||||
<div class="flex justify-between items-center">
|
||||
<h3 class="text-sm font-medium text-n-slate-12">
|
||||
{{ template.friendly_name }}
|
||||
</h3>
|
||||
<span class="text-xs text-n-slate-11">
|
||||
{{ languageLabel }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="flex flex-col gap-2">
|
||||
<div class="rounded-md">
|
||||
<div class="text-sm whitespace-pre-wrap text-n-slate-12">
|
||||
{{ renderedTemplate }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="text-xs text-n-slate-11">
|
||||
{{ categoryLabel }}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div v-if="hasVariables || hasMediaVariable">
|
||||
<!-- Media URL for media templates -->
|
||||
<div v-if="hasMediaVariable" class="mb-4">
|
||||
<p class="mb-2.5 text-sm font-semibold">
|
||||
{{ $t('CONTENT_TEMPLATES.PARSER.MEDIA_URL_LABEL') }}
|
||||
</p>
|
||||
<div class="flex items-center mb-2.5">
|
||||
<Input
|
||||
v-model="processedParams[mediaVariableKey]"
|
||||
type="url"
|
||||
class="flex-1"
|
||||
:placeholder="
|
||||
templateMediaUrl ||
|
||||
t('CONTENT_TEMPLATES.PARSER.MEDIA_URL_PLACEHOLDER')
|
||||
"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Body Variables Section -->
|
||||
<div v-if="hasVariables">
|
||||
<p class="mb-2.5 text-sm font-semibold">
|
||||
{{ $t('CONTENT_TEMPLATES.PARSER.VARIABLES_LABEL') }}
|
||||
</p>
|
||||
<div
|
||||
v-for="variable in variablePattern"
|
||||
:key="`variable-${variable}`"
|
||||
class="flex items-center mb-2.5"
|
||||
>
|
||||
<Input
|
||||
v-model="processedParams[variable]"
|
||||
type="text"
|
||||
class="flex-1"
|
||||
:placeholder="
|
||||
t('CONTENT_TEMPLATES.PARSER.VARIABLE_PLACEHOLDER', {
|
||||
variable: variable,
|
||||
})
|
||||
"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p
|
||||
v-if="v$.$dirty && (v$.$invalid || isFormInvalid)"
|
||||
class="p-2.5 text-center rounded-md bg-n-ruby-9/20 text-n-ruby-9"
|
||||
>
|
||||
{{ $t('CONTENT_TEMPLATES.PARSER.FORM_ERROR_MESSAGE') }}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<slot
|
||||
name="actions"
|
||||
:send-message="sendMessage"
|
||||
:reset-template="resetTemplate"
|
||||
:go-back="goBack"
|
||||
:is-valid="!v$.$invalid && !isFormInvalid"
|
||||
:disabled="isFormInvalid"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
@@ -1,25 +1,49 @@
|
||||
<script setup>
|
||||
import { ref } from 'vue';
|
||||
import { ref, nextTick, onMounted } from 'vue';
|
||||
|
||||
const emit = defineEmits(['send']);
|
||||
const message = ref('');
|
||||
const textareaRef = ref(null);
|
||||
|
||||
const adjustHeight = () => {
|
||||
if (!textareaRef.value) return;
|
||||
|
||||
// Reset height to auto to get the correct scrollHeight
|
||||
textareaRef.value.style.height = 'auto';
|
||||
// Set the height to the scrollHeight
|
||||
textareaRef.value.style.height = `${textareaRef.value.scrollHeight}px`;
|
||||
};
|
||||
|
||||
const sendMessage = () => {
|
||||
if (message.value.trim()) {
|
||||
emit('send', message.value);
|
||||
message.value = '';
|
||||
// Reset textarea height after sending
|
||||
nextTick(() => {
|
||||
adjustHeight();
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
const handleInput = () => {
|
||||
nextTick(adjustHeight);
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
nextTick(adjustHeight);
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<form class="relative" @submit.prevent="sendMessage">
|
||||
<input
|
||||
<textarea
|
||||
ref="textareaRef"
|
||||
v-model="message"
|
||||
type="text"
|
||||
:placeholder="$t('CAPTAIN.COPILOT.SEND_MESSAGE')"
|
||||
class="w-full reset-base bg-n-alpha-3 ltr:pl-4 ltr:pr-12 rtl:pl-12 rtl:pr-4 py-3 text-n-slate-11 text-sm border border-n-weak rounded-lg focus:outline-none focus:ring-1 focus:ring-n-blue-11 focus:border-n-blue-11"
|
||||
@keyup.enter="sendMessage"
|
||||
class="w-full reset-base bg-n-alpha-3 ltr:pl-4 ltr:pr-12 rtl:pl-12 rtl:pr-4 py-3 text-sm border border-n-weak rounded-lg focus:outline-0 focus:outline-none focus:ring-2 focus:ring-n-blue-11 focus:border-n-blue-11 resize-none overflow-hidden max-h-[200px] mb-0 text-n-slate-12"
|
||||
rows="1"
|
||||
@input="handleInput"
|
||||
@keydown.enter.exact.prevent="sendMessage"
|
||||
/>
|
||||
<button
|
||||
class="absolute ltr:right-1 rtl:left-1 top-1/2 -translate-y-1/2 h-9 w-10 flex items-center justify-center text-n-slate-11 hover:text-n-blue-11"
|
||||
|
||||
@@ -15,7 +15,7 @@ defineProps({
|
||||
>
|
||||
{{ title }}
|
||||
</div>
|
||||
<ul class="gap-2 grid reset-base list-none px-2">
|
||||
<ul class="gap-2 grid reset-base list-none px-2 max-h-96 overflow-y-auto">
|
||||
<slot />
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
@@ -34,14 +34,17 @@ const formatOperatorLabel = operator => {
|
||||
};
|
||||
|
||||
const formatFilterValue = value => {
|
||||
// Case 1: null, undefined, empty string
|
||||
if (!value) return '';
|
||||
|
||||
// Case 2: array → map each item, use name if present, else the item itself
|
||||
if (Array.isArray(value)) {
|
||||
return value.join(', ');
|
||||
return value.map(item => item?.name ?? item).join(', ');
|
||||
}
|
||||
if (typeof value === 'object' && value.name) {
|
||||
return value.name;
|
||||
}
|
||||
return value;
|
||||
|
||||
// Case 3: object with a "name" property → return name
|
||||
// Case 4: primitive (string, number, etc.) → return as is
|
||||
return value?.name ?? value;
|
||||
};
|
||||
</script>
|
||||
|
||||
@@ -66,6 +69,7 @@ const formatFilterValue = value => {
|
||||
</span>
|
||||
<span
|
||||
v-if="filter.values"
|
||||
:title="formatFilterValue(filter.values)"
|
||||
class="lowercase truncate text-n-slate-12"
|
||||
:class="{
|
||||
'first-letter:capitalize': shouldCapitalizeFirstLetter(
|
||||
|
||||
@@ -103,6 +103,12 @@ const validationError = computed(() => {
|
||||
);
|
||||
});
|
||||
|
||||
const inputFieldType = computed(() => {
|
||||
if (inputType.value === 'date') return 'date';
|
||||
if (inputType.value === 'number') return 'number';
|
||||
return 'text';
|
||||
});
|
||||
|
||||
const resetModelOnAttributeKeyChange = newAttributeKey => {
|
||||
/**
|
||||
* Resets the filter values and operator when the attribute key changes. This ensures that
|
||||
@@ -182,7 +188,7 @@ defineExpose({ validate });
|
||||
<Input
|
||||
v-else
|
||||
v-model="values"
|
||||
:type="inputType === 'date' ? 'date' : 'text'"
|
||||
:type="inputFieldType"
|
||||
class="[&>input]:h-8 [&>input]:py-1.5 [&>input]:outline-offset-0"
|
||||
:placeholder="t('FILTER.INPUT_PLACEHOLDER')"
|
||||
/>
|
||||
@@ -192,6 +198,7 @@ defineExpose({ validate });
|
||||
solid
|
||||
slate
|
||||
icon="i-lucide-trash"
|
||||
class="flex-shrink-0"
|
||||
@click.stop="emit('remove')"
|
||||
/>
|
||||
</div>
|
||||
|
||||
@@ -104,7 +104,7 @@ const outsideClickHandler = [
|
||||
<template>
|
||||
<div
|
||||
v-on-click-outside="outsideClickHandler"
|
||||
class="z-40 max-w-3xl lg:w-[750px] overflow-visible w-full border border-n-weak bg-n-alpha-3 backdrop-blur-[100px] shadow-lg rounded-xl p-6 grid gap-6"
|
||||
class="z-40 max-w-3xl min-w-96 lg:w-[750px] overflow-visible w-full border border-n-weak bg-n-alpha-3 backdrop-blur-[100px] shadow-lg rounded-xl p-6 grid gap-6"
|
||||
>
|
||||
<h3 class="text-base font-medium leading-6 text-n-slate-12">
|
||||
{{ filterModalHeaderTitle }}
|
||||
@@ -146,10 +146,10 @@ const outsideClickHandler = [
|
||||
</template>
|
||||
</ul>
|
||||
<div class="flex justify-between gap-2">
|
||||
<Button sm ghost blue @click="addFilter">
|
||||
<Button sm ghost blue class="flex-shrink-0" @click="addFilter">
|
||||
{{ $t('CONTACTS_LAYOUT.FILTER.BUTTONS.ADD_FILTER') }}
|
||||
</Button>
|
||||
<div class="flex gap-2">
|
||||
<div class="flex gap-2 flex-shrink-0">
|
||||
<Button sm faded slate @click="resetFilter">
|
||||
{{ $t('CONTACTS_LAYOUT.FILTER.BUTTONS.CLEAR_FILTERS') }}
|
||||
</Button>
|
||||
|
||||
@@ -50,6 +50,7 @@ export function useContactFilterContext() {
|
||||
const { t } = useI18n();
|
||||
|
||||
const contactAttributes = useMapGetter('attributes/getContactAttributes');
|
||||
const labels = useMapGetter('labels/getLabels');
|
||||
|
||||
const {
|
||||
equalityOperators,
|
||||
@@ -184,6 +185,20 @@ export function useContactFilterContext() {
|
||||
filterOperators: equalityOperators.value,
|
||||
attributeModel: 'standard',
|
||||
},
|
||||
{
|
||||
attributeKey: CONTACT_ATTRIBUTES.LABELS,
|
||||
value: CONTACT_ATTRIBUTES.LABELS,
|
||||
attributeName: t('CONTACTS_FILTER.ATTRIBUTES.LABELS'),
|
||||
label: t('CONTACTS_FILTER.ATTRIBUTES.LABELS'),
|
||||
inputType: 'multiSelect',
|
||||
options: labels.value?.map(label => ({
|
||||
id: label.title,
|
||||
name: label.title,
|
||||
})),
|
||||
dataType: 'text',
|
||||
filterOperators: equalityOperators.value,
|
||||
attributeModel: 'standard',
|
||||
},
|
||||
...customFilterTypes.value,
|
||||
]);
|
||||
|
||||
|
||||
@@ -28,6 +28,7 @@ export const CONTACT_ATTRIBUTES = {
|
||||
LAST_ACTIVITY_AT: 'last_activity_at',
|
||||
REFERER: 'referer',
|
||||
BLOCKED: 'blocked',
|
||||
LABELS: 'labels',
|
||||
};
|
||||
|
||||
/**
|
||||
|
||||
@@ -164,8 +164,8 @@ export function useConversationFilterContext() {
|
||||
value: CONVERSATION_ATTRIBUTES.DISPLAY_ID,
|
||||
attributeName: t('FILTER.ATTRIBUTES.CONVERSATION_IDENTIFIER'),
|
||||
label: t('FILTER.ATTRIBUTES.CONVERSATION_IDENTIFIER'),
|
||||
inputType: 'plainText',
|
||||
datatype: 'number',
|
||||
inputType: 'number',
|
||||
dataType: 'number',
|
||||
filterOperators: containmentOperators.value,
|
||||
attributeModel: 'standard',
|
||||
},
|
||||
@@ -179,7 +179,7 @@ export function useConversationFilterContext() {
|
||||
id: campaign.id,
|
||||
name: campaign.title,
|
||||
})),
|
||||
datatype: 'number',
|
||||
dataType: 'number',
|
||||
filterOperators: presenceOperators.value,
|
||||
attributeModel: 'standard',
|
||||
},
|
||||
|
||||
@@ -2,23 +2,23 @@ import { computed } from 'vue';
|
||||
|
||||
export function useChannelIcon(inbox) {
|
||||
const channelTypeIconMap = {
|
||||
'Channel::Api': 'i-ri-cloudy-fill',
|
||||
'Channel::Email': 'i-ri-mail-fill',
|
||||
'Channel::FacebookPage': 'i-ri-messenger-fill',
|
||||
'Channel::Line': 'i-ri-line-fill',
|
||||
'Channel::Sms': 'i-ri-chat-1-fill',
|
||||
'Channel::Telegram': 'i-ri-telegram-fill',
|
||||
'Channel::TwilioSms': 'i-ri-chat-1-fill',
|
||||
'Channel::Api': 'i-woot-api',
|
||||
'Channel::Email': 'i-woot-mail',
|
||||
'Channel::FacebookPage': 'i-woot-messenger',
|
||||
'Channel::Line': 'i-woot-line',
|
||||
'Channel::Sms': 'i-woot-sms',
|
||||
'Channel::Telegram': 'i-woot-telegram',
|
||||
'Channel::TwilioSms': 'i-woot-sms',
|
||||
'Channel::TwitterProfile': 'i-ri-twitter-x-fill',
|
||||
'Channel::WebWidget': 'i-ri-global-fill',
|
||||
'Channel::Whatsapp': 'i-ri-whatsapp-fill',
|
||||
'Channel::Instagram': 'i-ri-instagram-fill',
|
||||
'Channel::WebWidget': 'i-woot-website',
|
||||
'Channel::Whatsapp': 'i-woot-whatsapp',
|
||||
'Channel::Instagram': 'i-woot-instagram',
|
||||
'Channel::Voice': 'i-ri-phone-fill',
|
||||
};
|
||||
|
||||
const providerIconMap = {
|
||||
microsoft: 'i-ri-microsoft-fill',
|
||||
google: 'i-ri-google-fill',
|
||||
microsoft: 'i-woot-outlook',
|
||||
google: 'i-woot-gmail',
|
||||
};
|
||||
|
||||
const channelIcon = computed(() => {
|
||||
@@ -34,7 +34,7 @@ export function useChannelIcon(inbox) {
|
||||
|
||||
// Special case for Twilio whatsapp
|
||||
if (type === 'Channel::TwilioSms' && inboxDetails.medium === 'whatsapp') {
|
||||
icon = 'i-ri-whatsapp-fill';
|
||||
icon = 'i-woot-whatsapp';
|
||||
}
|
||||
|
||||
return icon ?? 'i-ri-global-fill';
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user