Compare commits
17 Commits
feature/IO
...
feature/IO
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
02a1d39ca4 | ||
|
|
d50efe9890 | ||
|
|
789af9fb30 | ||
|
|
18da11f593 | ||
|
|
0ca1937140 | ||
|
|
7898d5a595 | ||
|
|
e8aa4f2dac | ||
|
|
271d15d969 | ||
|
|
136d52ec0b | ||
|
|
90b9ab906a | ||
|
|
02fd5e1701 | ||
|
|
2dcd05c593 | ||
|
|
0e9414cbc8 | ||
|
|
ddc235637a | ||
|
|
60549597cb | ||
|
|
9fe94481e7 | ||
|
|
b78d0253de |
@@ -5,7 +5,6 @@ orbs:
|
|||||||
aws-s3: circleci/aws-s3@4.0.0
|
aws-s3: circleci/aws-s3@4.0.0
|
||||||
aws-cli: circleci/aws-cli@4.0
|
aws-cli: circleci/aws-cli@4.0
|
||||||
eb: circleci/aws-elastic-beanstalk@2.0.1
|
eb: circleci/aws-elastic-beanstalk@2.0.1
|
||||||
jira: circleci/jira@2.1.0
|
|
||||||
jobs:
|
jobs:
|
||||||
imex-api-deploy:
|
imex-api-deploy:
|
||||||
docker:
|
docker:
|
||||||
@@ -19,12 +18,6 @@ jobs:
|
|||||||
eb status --verbose
|
eb status --verbose
|
||||||
eb deploy
|
eb deploy
|
||||||
eb status
|
eb status
|
||||||
- jira/notify:
|
|
||||||
environment: Production (ImEX) - API
|
|
||||||
environment_type: production
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
imex-hasura-migrate:
|
imex-hasura-migrate:
|
||||||
docker:
|
docker:
|
||||||
@@ -40,16 +33,11 @@ jobs:
|
|||||||
- run:
|
- run:
|
||||||
name: Execute migration
|
name: Execute migration
|
||||||
command: |
|
command: |
|
||||||
curl -L https://github.com/hasura/graphql-engine/raw/stable/cli/get.sh | bash
|
npm install hasura-cli -g
|
||||||
hasura migrate apply --endpoint https://db.imex.online/ --admin-secret << parameters.secret >>
|
hasura migrate apply --endpoint https://db.imex.online/ --admin-secret << parameters.secret >>
|
||||||
hasura metadata apply --endpoint https://db.imex.online/ --admin-secret << parameters.secret >>
|
hasura metadata apply --endpoint https://db.imex.online/ --admin-secret << parameters.secret >>
|
||||||
hasura metadata reload --endpoint https://db.imex.online/ --admin-secret << parameters.secret >>
|
hasura metadata reload --endpoint https://db.imex.online/ --admin-secret << parameters.secret >>
|
||||||
- jira/notify:
|
|
||||||
environment: Production (ImEX) - Hasura
|
|
||||||
environment_type: production
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
imex-app-build:
|
imex-app-build:
|
||||||
docker:
|
docker:
|
||||||
- image: cimg/node:18.18.2
|
- image: cimg/node:18.18.2
|
||||||
@@ -74,7 +62,6 @@ jobs:
|
|||||||
to: "s3://imex-online-production/"
|
to: "s3://imex-online-production/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
|
|
||||||
|
|
||||||
imex-app-beta-build:
|
imex-app-beta-build:
|
||||||
docker:
|
docker:
|
||||||
- image: cimg/node:18.18.2
|
- image: cimg/node:18.18.2
|
||||||
@@ -99,12 +86,6 @@ jobs:
|
|||||||
from: dist
|
from: dist
|
||||||
to: "s3://imex-online-beta/"
|
to: "s3://imex-online-beta/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
- jira/notify:
|
|
||||||
environment: Production (ImEX) - Front End
|
|
||||||
environment_type: production
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
rome-api-deploy:
|
rome-api-deploy:
|
||||||
docker:
|
docker:
|
||||||
@@ -118,12 +99,7 @@ jobs:
|
|||||||
eb status --verbose
|
eb status --verbose
|
||||||
eb deploy
|
eb deploy
|
||||||
eb status
|
eb status
|
||||||
- jira/notify:
|
|
||||||
environment: Production (Rome) - API
|
|
||||||
environment_type: production
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
rome-hasura-migrate:
|
rome-hasura-migrate:
|
||||||
docker:
|
docker:
|
||||||
- image: cimg/node:18.18.2
|
- image: cimg/node:18.18.2
|
||||||
@@ -138,16 +114,11 @@ jobs:
|
|||||||
- run:
|
- run:
|
||||||
name: Execute migration
|
name: Execute migration
|
||||||
command: |
|
command: |
|
||||||
curl -L https://github.com/hasura/graphql-engine/raw/stable/cli/get.sh | bash
|
npm install hasura-cli -g
|
||||||
hasura migrate apply --endpoint https://db.romeonline.io/ --admin-secret << parameters.secret >>
|
hasura migrate apply --endpoint https://db.romeonline.io/ --admin-secret << parameters.secret >>
|
||||||
hasura metadata apply --endpoint https://db.romeonline.io/ --admin-secret << parameters.secret >>
|
hasura metadata apply --endpoint https://db.romeonline.io/ --admin-secret << parameters.secret >>
|
||||||
hasura metadata reload --endpoint https://db.romeonline.io/ --admin-secret << parameters.secret >>
|
hasura metadata reload --endpoint https://db.romeonline.io/ --admin-secret << parameters.secret >>
|
||||||
- jira/notify:
|
|
||||||
environment: Production (Rome) - Hasura
|
|
||||||
environment_type: production
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
rome-app-build:
|
rome-app-build:
|
||||||
docker:
|
docker:
|
||||||
- image: cimg/node:18.18.2
|
- image: cimg/node:18.18.2
|
||||||
@@ -172,12 +143,6 @@ jobs:
|
|||||||
from: dist
|
from: dist
|
||||||
to: "s3://rome-online-production/"
|
to: "s3://rome-online-production/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
- jira/notify:
|
|
||||||
environment: Production (Rome) - Front End
|
|
||||||
environment_type: production
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
promanager-app-build:
|
promanager-app-build:
|
||||||
docker:
|
docker:
|
||||||
@@ -203,12 +168,6 @@ jobs:
|
|||||||
from: dist
|
from: dist
|
||||||
to: "s3://promanager-production/"
|
to: "s3://promanager-production/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
- jira/notify:
|
|
||||||
environment: Production (ProManager) - Front End
|
|
||||||
environment_type: production
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
test-rome-hasura-migrate:
|
test-rome-hasura-migrate:
|
||||||
docker:
|
docker:
|
||||||
@@ -224,18 +183,10 @@ jobs:
|
|||||||
- run:
|
- run:
|
||||||
name: Execute migration
|
name: Execute migration
|
||||||
command: |
|
command: |
|
||||||
curl -L https://github.com/hasura/graphql-engine/raw/stable/cli/get.sh | bash
|
npm install hasura-cli -g
|
||||||
hasura migrate apply --endpoint https://db.test.romeonline.io/ --admin-secret << parameters.secret >>
|
hasura migrate apply --endpoint https://db.test.romeonline.io/ --admin-secret << parameters.secret >>
|
||||||
sleep 5
|
|
||||||
hasura metadata apply --endpoint https://db.test.romeonline.io/ --admin-secret << parameters.secret >>
|
hasura metadata apply --endpoint https://db.test.romeonline.io/ --admin-secret << parameters.secret >>
|
||||||
sleep 10
|
|
||||||
hasura metadata reload --endpoint https://db.test.romeonline.io/ --admin-secret << parameters.secret >>
|
hasura metadata reload --endpoint https://db.test.romeonline.io/ --admin-secret << parameters.secret >>
|
||||||
- jira/notify:
|
|
||||||
environment: Test (Rome) - Hasura
|
|
||||||
environment_type: testing
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
test-rome-app-build:
|
test-rome-app-build:
|
||||||
docker:
|
docker:
|
||||||
@@ -261,12 +212,6 @@ jobs:
|
|||||||
from: dist
|
from: dist
|
||||||
to: "s3://rome-online-test/"
|
to: "s3://rome-online-test/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
- jira/notify:
|
|
||||||
environment: Test (Rome) - Front End
|
|
||||||
environment_type: testing
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
test-promanager-app-build:
|
test-promanager-app-build:
|
||||||
docker:
|
docker:
|
||||||
@@ -292,12 +237,6 @@ jobs:
|
|||||||
from: dist
|
from: dist
|
||||||
to: "s3://promanager-testing/"
|
to: "s3://promanager-testing/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
- jira/notify:
|
|
||||||
environment: Test (ProManager) - Front End
|
|
||||||
environment_type: testing
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
test-hasura-migrate:
|
test-hasura-migrate:
|
||||||
docker:
|
docker:
|
||||||
@@ -313,18 +252,10 @@ jobs:
|
|||||||
- run:
|
- run:
|
||||||
name: Execute migration
|
name: Execute migration
|
||||||
command: |
|
command: |
|
||||||
curl -L https://github.com/hasura/graphql-engine/raw/stable/cli/get.sh | bash
|
npm install hasura-cli -g
|
||||||
hasura migrate apply --endpoint https://db.test.bodyshop.app/ --admin-secret << parameters.secret >>
|
hasura migrate apply --endpoint https://db.test.bodyshop.app/ --admin-secret << parameters.secret >>
|
||||||
sleep 15
|
|
||||||
hasura metadata apply --endpoint https://db.test.bodyshop.app/ --admin-secret << parameters.secret >>
|
hasura metadata apply --endpoint https://db.test.bodyshop.app/ --admin-secret << parameters.secret >>
|
||||||
sleep 30
|
|
||||||
hasura metadata reload --endpoint https://db.test.bodyshop.app/ --admin-secret << parameters.secret >>
|
hasura metadata reload --endpoint https://db.test.bodyshop.app/ --admin-secret << parameters.secret >>
|
||||||
- jira/notify:
|
|
||||||
environment: Test (ImEX) - Hasura
|
|
||||||
environment_type: testing
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
imex-test-app-build:
|
imex-test-app-build:
|
||||||
docker:
|
docker:
|
||||||
@@ -371,12 +302,7 @@ jobs:
|
|||||||
from: dist
|
from: dist
|
||||||
to: "s3://imex-online-test-beta/"
|
to: "s3://imex-online-test-beta/"
|
||||||
arguments: "--exclude '*.map'"
|
arguments: "--exclude '*.map'"
|
||||||
- jira/notify:
|
|
||||||
environment: Test (ImEX) - Front End
|
|
||||||
environment_type: testing
|
|
||||||
pipeline_id: << pipeline.id >>
|
|
||||||
job_type: deployment
|
|
||||||
pipeline_number: << pipeline.number >>
|
|
||||||
|
|
||||||
admin-app-build:
|
admin-app-build:
|
||||||
docker:
|
docker:
|
||||||
@@ -427,7 +353,7 @@ workflows:
|
|||||||
secret: ${HASURA_PROD_SECRET}
|
secret: ${HASURA_PROD_SECRET}
|
||||||
filters:
|
filters:
|
||||||
branches:
|
branches:
|
||||||
only: master-AIO
|
only: master
|
||||||
- rome-api-deploy:
|
- rome-api-deploy:
|
||||||
filters:
|
filters:
|
||||||
branches:
|
branches:
|
||||||
@@ -437,7 +363,7 @@ workflows:
|
|||||||
branches:
|
branches:
|
||||||
only: master-AIO
|
only: master-AIO
|
||||||
- rome-hasura-migrate:
|
- rome-hasura-migrate:
|
||||||
secret: ${HASURA_ROME_PROD_SECRET}
|
secret: ${HASURA_PROD_SECRET}
|
||||||
filters:
|
filters:
|
||||||
branches:
|
branches:
|
||||||
only: master-AIO
|
only: master-AIO
|
||||||
|
|||||||
@@ -1,59 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
||||||
<title>IMEX IO Extractor</title>
|
|
||||||
<style>
|
|
||||||
body {
|
|
||||||
font-family: Arial, sans-serif;
|
|
||||||
padding: 20px;
|
|
||||||
}
|
|
||||||
textarea {
|
|
||||||
width: 100%;
|
|
||||||
height: 200px;
|
|
||||||
}
|
|
||||||
.output-box {
|
|
||||||
margin-top: 20px;
|
|
||||||
padding: 10px;
|
|
||||||
border: 1px solid #ccc;
|
|
||||||
background-color: #f9f9f9;
|
|
||||||
min-height: 40px;
|
|
||||||
}
|
|
||||||
.copy-button {
|
|
||||||
margin-top: 10px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<h1>IMEX IO Extractor</h1>
|
|
||||||
<textarea id="inputText" placeholder="Paste your text here..."></textarea>
|
|
||||||
<br>
|
|
||||||
<button onclick="extractIO()">Extract</button>
|
|
||||||
|
|
||||||
<div class="output-box" id="outputBox" contenteditable="true"></div>
|
|
||||||
<button class="copy-button" onclick="copyToClipboard()">Copy to Clipboard</button>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
function extractIO() {
|
|
||||||
const inputText = document.getElementById('inputText').value;
|
|
||||||
const ioNumbers = [...new Set(inputText.match(/IO-\d{4}/g))] // Extract unique IO-#### matches
|
|
||||||
.map(io => ({ io, num: parseInt(io.split('-')[1]) })) // Extract number part for sorting
|
|
||||||
.sort((a, b) => a.num - b.num) // Sort by the number
|
|
||||||
.map(item => item.io); // Extract sorted IO-####
|
|
||||||
|
|
||||||
document.getElementById('outputBox').innerText = ioNumbers.join(', '); // Display horizontally
|
|
||||||
}
|
|
||||||
|
|
||||||
function copyToClipboard() {
|
|
||||||
const outputBox = document.getElementById('outputBox');
|
|
||||||
const range = document.createRange();
|
|
||||||
range.selectNodeContents(outputBox);
|
|
||||||
const selection = window.getSelection();
|
|
||||||
selection.removeAllRanges();
|
|
||||||
selection.addRange(range);
|
|
||||||
document.execCommand('copy');
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,20 +0,0 @@
|
|||||||
-----BEGIN CERTIFICATE-----
|
|
||||||
MIIDWzCCAkOgAwIBAgIUD/QBSAXy/AlJ/cS4DaPWJLpChxgwDQYJKoZIhvcNAQEL
|
|
||||||
BQAwPTELMAkGA1UEBhMCQ0ExCzAJBgNVBAgMAk9OMSEwHwYDVQQKDBhJbnRlcm5l
|
|
||||||
dCBXaWRnaXRzIFB0eSBMdGQwHhcNMjQwOTA5MTU0MjA1WhcNMjUwOTA5MTU0MjA1
|
|
||||||
WjA9MQswCQYDVQQGEwJDQTELMAkGA1UECAwCT04xITAfBgNVBAoMGEludGVybmV0
|
|
||||||
IFdpZGdpdHMgUHR5IEx0ZDCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEB
|
|
||||||
AKSd0l7NJCNBwvtPU+dVPQkteg0AfC3sGqRnZMQteCRVa2oIgC4NoF3A9BK/yHbF
|
|
||||||
ZF25OnXTck5vzc8yb3v73ndfTD9ASKNoiaZE84/GFBsxqlKR8cs0qVwzuAsdijMv
|
|
||||||
vlMPNlMRyE1Rb7nR6HXGkPXNyxgMko03NXPkvIje9zRudm0Lf8L4q/hPyPkS7Mrm
|
|
||||||
/uQfAAJe+xFcupkEX2XY7r0x1C+z6E8lA1UcuhK3SHdW7CWYqp1vU5/dnnUiXwCa
|
|
||||||
GiC6Y1bCJB0pDAVISzy3JUDdINZdiqGR+y8ho3pstChf2mp/76s3N9eG9KA/qaFK
|
|
||||||
BrGk2PvCoZ8/Aj1aMsRYFHECAwEAAaNTMFEwHQYDVR0OBBYEFDLJ2fbWP4VUJgOp
|
|
||||||
PSs+NGHcVgRmMB8GA1UdIwQYMBaAFDLJ2fbWP4VUJgOpPSs+NGHcVgRmMA8GA1Ud
|
|
||||||
EwEB/wQFMAMBAf8wDQYJKoZIhvcNAQELBQADggEBABfv5ut/y03atq0NMB0jeDY4
|
|
||||||
AvW4ukk0k1svyqxFZCw9o7m2lHb/IjmVrZG1Sj4JWrrSv0s02ccb26/t6vazNa5L
|
|
||||||
Powe3eyfHgfjTZJmgs8hyeMwKS0wWk/SPuu9JDhIJakiquqD+UVBGkHpP+XYvhDv
|
|
||||||
vhS2XRlW+aEjpUmr1oCyyrc6WbzrYRNadqEsn/AxwcMyUbht3Ugjkg+OpidcTIQp
|
|
||||||
5lv63waKo6I1vQofzBQ3L7JYsKo8kC0vAP7wkLxvzBii335uZJzzpFYFVOyVNezi
|
|
||||||
dJdazPbRYbXz4LjltdEn/SNfRuKX8ZRiN2OSo7OfSrZaMTS87SfCSFJGgQM8Yrk=
|
|
||||||
-----END CERTIFICATE-----
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
-----BEGIN PRIVATE KEY-----
|
|
||||||
MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQCkndJezSQjQcL7
|
|
||||||
T1PnVT0JLXoNAHwt7BqkZ2TELXgkVWtqCIAuDaBdwPQSv8h2xWRduTp103JOb83P
|
|
||||||
Mm97+953X0w/QEijaImmRPOPxhQbMapSkfHLNKlcM7gLHYozL75TDzZTEchNUW+5
|
|
||||||
0eh1xpD1zcsYDJKNNzVz5LyI3vc0bnZtC3/C+Kv4T8j5EuzK5v7kHwACXvsRXLqZ
|
|
||||||
BF9l2O69MdQvs+hPJQNVHLoSt0h3VuwlmKqdb1Of3Z51Il8AmhogumNWwiQdKQwF
|
|
||||||
SEs8tyVA3SDWXYqhkfsvIaN6bLQoX9pqf++rNzfXhvSgP6mhSgaxpNj7wqGfPwI9
|
|
||||||
WjLEWBRxAgMBAAECggEAUNpHYlLFxh9dokujPUMreF+Cy/IKDBAkQc2au5RNpyLh
|
|
||||||
YDIOqw/8TTAhcTgLQPLQygvZP9f8E7RsVLFD+pSJ/v2qmIJ9au1Edor1Sg+S/oxV
|
|
||||||
SLrwFMunx2aLpcH7iAqSI3+cQg7A3+D4zD7iOz6tIl3Su9wo+v073tFhHKTOrEwv
|
|
||||||
Qgr9Jf3viIiKV1ym+uQEVQndocfsj46FnFpXTQ2qs7kAF6FgAOLDGfQQwzkiqEBD
|
|
||||||
NsqsDmbYIx6foZL+DEz1ZVO2M5B+xxpbNK82KwuQilVpimW8ui4LZHCe+RIFzt9+
|
|
||||||
BK6KGlLpSEwTFliivI3nahy18JzskZsfyah0CPZlQQKBgQDVv+A0qIPGvOP3Sx+9
|
|
||||||
HyeQCV23SkvvSvw8p8pMB0gvwv63YdJ7N/rJzBGS6YUHFWWZZgEeTgkJ6VJvoe0r
|
|
||||||
8JL1el9uSUa7f0eayjmFBOGuzpktNVdIn2Tg7A9MWA4JqPNNC69RMOh86ewGD4J3
|
|
||||||
a8Hz2a1bHxAmy/AZt2ukypY6eQKBgQDFJ7kqeOPkRBz9WbALRgVIXo8YWf5di0sQ
|
|
||||||
r0HC03GAISHQ725A2IFBPHJWeqj0jaMiIZD0y+Obgp7KAskrJaLfsd7Ug775kFfw
|
|
||||||
oUI9UAl6kRuPKvm3BaVAm46SQm+56VsgxTi73YN0NUp75THHZgAJjepF9zSpVJxq
|
|
||||||
VY9DjEGruQKBgQCQCpGIatcCol/tUg69X7VFd0pULhkl1J5OMbQ9r9qRdRI5eg5h
|
|
||||||
QsQaIQ7mtb8TmvOwf/DY/zVQHI+U8sXlCmW+TwzoQTENQSR7xzMj1LpRFqBaustr
|
|
||||||
AR72A537kItFLzll/i3SxOam5uxK2UDOQSuerF4KPdCglGXkrpo3nt3F4QKBgQCa
|
|
||||||
RArPAOjQo7tLQfJN3+wiRFsTYtd1uphx5bA/EdOtvj8HjVFnzADXWsTchf3N3UXY
|
|
||||||
XwtdgGwIMpys1KEz8a8P+c2x26SDAj7NOmDqOMYx8Xju/WGHpBM6Cn30U6e4gK+d
|
|
||||||
ZLSPyzQgqdIuP5hDvbwpvbGiLVw3Ys1BJtGCuSxpgQJ/eHnRiuSi5Zq5jGg+GpA+
|
|
||||||
FEEc9NCy772rR+4uzEOqyIwqewffqzSuVWuIsY/8MP3fh+NDxl/mU6cB5QVeD54Z
|
|
||||||
JZUKwmpM26muiM6WvVnM4ExPdSGA2+l4pZjby/KKd6F/w0tgZ1jb9Pb2/0vN3qVA
|
|
||||||
Y4U4XNTMt2fxUACqiL4SHA==
|
|
||||||
-----END PRIVATE KEY-----
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
VITE_APP_GRAPHQL_ENDPOINT=https://db.dev.imex.online/v1/graphql
|
VITE_APP_GRAPHQL_ENDPOINT=https://db.dev.bodyshop.app/v1/graphql
|
||||||
VITE_APP_GRAPHQL_ENDPOINT_WS=wss://db.dev.imex.online/v1/graphql
|
VITE_APP_GRAPHQL_ENDPOINT_WS=wss://db.dev.bodyshop.app/v1/graphql
|
||||||
VITE_APP_GA_CODE=231099835
|
VITE_APP_GA_CODE=231099835
|
||||||
VITE_APP_FIREBASE_CONFIG={"apiKey":"AIzaSyDPLT8GiDHDR1R4nI66Qi0BY1aYviDPioc","authDomain":"imex-dev.firebaseapp.com","databaseURL":"https://imex-dev.firebaseio.com","projectId":"imex-dev","storageBucket":"imex-dev.appspot.com","messagingSenderId":"759548147434","appId":"1:759548147434:web:e8239868a48ceb36700993","measurementId":"G-K5XRBVVB4S"}
|
VITE_APP_FIREBASE_CONFIG={"apiKey":"AIzaSyDPLT8GiDHDR1R4nI66Qi0BY1aYviDPioc","authDomain":"imex-dev.firebaseapp.com","databaseURL":"https://imex-dev.firebaseio.com","projectId":"imex-dev","storageBucket":"imex-dev.appspot.com","messagingSenderId":"759548147434","appId":"1:759548147434:web:e8239868a48ceb36700993","measurementId":"G-K5XRBVVB4S"}
|
||||||
VITE_APP_CLOUDINARY_ENDPOINT_API=https://api.cloudinary.com/v1_1/io-test
|
VITE_APP_CLOUDINARY_ENDPOINT_API=https://api.cloudinary.com/v1_1/io-test
|
||||||
@@ -8,7 +8,7 @@ VITE_APP_CLOUDINARY_API_KEY=957865933348715
|
|||||||
VITE_APP_CLOUDINARY_THUMB_TRANSFORMATIONS=c_fill,h_250,w_250
|
VITE_APP_CLOUDINARY_THUMB_TRANSFORMATIONS=c_fill,h_250,w_250
|
||||||
VITE_APP_FIREBASE_PUBLIC_VAPID_KEY='BG3tzU7L2BXlGZ_3VLK4PNaRceoEXEnmHfxcVbRMF5o5g05ejslhVPki9kBM9cBBT-08Ad9kN3HSpS6JmrWD6h4'
|
VITE_APP_FIREBASE_PUBLIC_VAPID_KEY='BG3tzU7L2BXlGZ_3VLK4PNaRceoEXEnmHfxcVbRMF5o5g05ejslhVPki9kBM9cBBT-08Ad9kN3HSpS6JmrWD6h4'
|
||||||
VITE_APP_STRIPE_PUBLIC_KEY=pk_test_51GqB4TJl3nQjrZ0wCQWAxAhlNF8jKe0tipIa6ExBaxwJGitwvFsIZUEua4dUzaMIAuXp4qwYHXx7lgjyQSwP0Pe900vzm38C7g
|
VITE_APP_STRIPE_PUBLIC_KEY=pk_test_51GqB4TJl3nQjrZ0wCQWAxAhlNF8jKe0tipIa6ExBaxwJGitwvFsIZUEua4dUzaMIAuXp4qwYHXx7lgjyQSwP0Pe900vzm38C7g
|
||||||
VITE_APP_AXIOS_BASE_API_URL=/api/
|
VITE_APP_AXIOS_BASE_API_URL=http://localhost:4000
|
||||||
VITE_APP_REPORTS_SERVER_URL=https://reports3.test.imex.online
|
VITE_APP_REPORTS_SERVER_URL=https://reports3.test.imex.online
|
||||||
VITE_APP_SPLIT_API=ts615lqgnmk84thn72uk18uu5pgce6e0l4rc
|
VITE_APP_SPLIT_API=ts615lqgnmk84thn72uk18uu5pgce6e0l4rc
|
||||||
VITE_APP_INSTANCE=IMEX
|
VITE_APP_INSTANCE=IMEX
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
VITE_APP_GRAPHQL_ENDPOINT=https://db.dev.imex.online/v1/graphql
|
VITE_APP_GRAPHQL_ENDPOINT=https://db.dev.bodyshop.app/v1/graphql
|
||||||
VITE_APP_GRAPHQL_ENDPOINT_WS=wss://db.dev.imex.online/v1/graphql
|
VITE_APP_GRAPHQL_ENDPOINT_WS=wss://db.dev.bodyshop.app/v1/graphql
|
||||||
VITE_APP_GA_CODE=231099835
|
VITE_APP_GA_CODE=231099835
|
||||||
VITE_APP_FIREBASE_CONFIG={"apiKey":"AIzaSyDPLT8GiDHDR1R4nI66Qi0BY1aYviDPioc","authDomain":"imex-dev.firebaseapp.com","databaseURL":"https://imex-dev.firebaseio.com","projectId":"imex-dev","storageBucket":"imex-dev.appspot.com","messagingSenderId":"759548147434","appId":"1:759548147434:web:e8239868a48ceb36700993","measurementId":"G-K5XRBVVB4S"}
|
VITE_APP_FIREBASE_CONFIG={"apiKey":"AIzaSyDPLT8GiDHDR1R4nI66Qi0BY1aYviDPioc","authDomain":"imex-dev.firebaseapp.com","databaseURL":"https://imex-dev.firebaseio.com","projectId":"imex-dev","storageBucket":"imex-dev.appspot.com","messagingSenderId":"759548147434","appId":"1:759548147434:web:e8239868a48ceb36700993","measurementId":"G-K5XRBVVB4S"}
|
||||||
VITE_APP_CLOUDINARY_ENDPOINT_API=https://api.cloudinary.com/v1_1/io-test
|
VITE_APP_CLOUDINARY_ENDPOINT_API=https://api.cloudinary.com/v1_1/io-test
|
||||||
@@ -8,7 +8,7 @@ VITE_APP_CLOUDINARY_API_KEY=957865933348715
|
|||||||
VITE_APP_CLOUDINARY_THUMB_TRANSFORMATIONS=c_fill,h_250,w_250
|
VITE_APP_CLOUDINARY_THUMB_TRANSFORMATIONS=c_fill,h_250,w_250
|
||||||
VITE_APP_FIREBASE_PUBLIC_VAPID_KEY='BG3tzU7L2BXlGZ_3VLK4PNaRceoEXEnmHfxcVbRMF5o5g05ejslhVPki9kBM9cBBT-08Ad9kN3HSpS6JmrWD6h4'
|
VITE_APP_FIREBASE_PUBLIC_VAPID_KEY='BG3tzU7L2BXlGZ_3VLK4PNaRceoEXEnmHfxcVbRMF5o5g05ejslhVPki9kBM9cBBT-08Ad9kN3HSpS6JmrWD6h4'
|
||||||
VITE_APP_STRIPE_PUBLIC_KEY=pk_test_51GqB4TJl3nQjrZ0wCQWAxAhlNF8jKe0tipIa6ExBaxwJGitwvFsIZUEua4dUzaMIAuXp4qwYHXx7lgjyQSwP0Pe900vzm38C7g
|
VITE_APP_STRIPE_PUBLIC_KEY=pk_test_51GqB4TJl3nQjrZ0wCQWAxAhlNF8jKe0tipIa6ExBaxwJGitwvFsIZUEua4dUzaMIAuXp4qwYHXx7lgjyQSwP0Pe900vzm38C7g
|
||||||
VITE_APP_AXIOS_BASE_API_URL=/api/
|
VITE_APP_AXIOS_BASE_API_URL=http://localhost:4000
|
||||||
VITE_APP_REPORTS_SERVER_URL=https://reports3.test.imex.online
|
VITE_APP_REPORTS_SERVER_URL=https://reports3.test.imex.online
|
||||||
VITE_APP_SPLIT_API=ts615lqgnmk84thn72uk18uu5pgce6e0l4rc
|
VITE_APP_SPLIT_API=ts615lqgnmk84thn72uk18uu5pgce6e0l4rc
|
||||||
VITE_APP_INSTANCE=PROMANAGER
|
VITE_APP_INSTANCE=PROMANAGER
|
||||||
|
|||||||
@@ -1,15 +1,14 @@
|
|||||||
VITE_APP_GRAPHQL_ENDPOINT=https://db.dev.imex.online/v1/graphql
|
VITE_APP_GRAPHQL_ENDPOINT=https://db.dev.bodyshop.app/v1/graphql
|
||||||
VITE_APP_GRAPHQL_ENDPOINT_WS=wss://db.dev.imex.online/v1/graphql
|
VITE_APP_GRAPHQL_ENDPOINT_WS=wss://db.dev.bodyshop.app/v1/graphql
|
||||||
VITE_APP_GA_CODE=231099835
|
VITE_APP_GA_CODE=231099835
|
||||||
# VITE_APP_FIREBASE_CONFIG={ "apiKey": "AIzaSyAuLQR9SV5LsVxjU8wh9hvFLdhcAHU6cxE", "authDomain": "rome-prod-1.firebaseapp.com", "projectId": "rome-prod-1", "storageBucket": "rome-prod-1.appspot.com", "messagingSenderId": "147786367145", "appId": "1:147786367145:web:9d4cba68071c3f29a8a9b8", "measurementId": "G-G8Z9DRHTZS"}
|
VITE_APP_FIREBASE_CONFIG={ "apiKey": "AIzaSyAuLQR9SV5LsVxjU8wh9hvFLdhcAHU6cxE", "authDomain": "rome-prod-1.firebaseapp.com", "projectId": "rome-prod-1", "storageBucket": "rome-prod-1.appspot.com", "messagingSenderId": "147786367145", "appId": "1:147786367145:web:9d4cba68071c3f29a8a9b8", "measurementId": "G-G8Z9DRHTZS"}
|
||||||
VITE_APP_FIREBASE_CONFIG={"apiKey":"AIzaSyDPLT8GiDHDR1R4nI66Qi0BY1aYviDPioc","authDomain":"imex-dev.firebaseapp.com","databaseURL":"https://imex-dev.firebaseio.com","projectId":"imex-dev","storageBucket":"imex-dev.appspot.com","messagingSenderId":"759548147434","appId":"1:759548147434:web:e8239868a48ceb36700993","measurementId":"G-K5XRBVVB4S"}
|
|
||||||
VITE_APP_CLOUDINARY_ENDPOINT_API=https://api.cloudinary.com/v1_1/io-test
|
VITE_APP_CLOUDINARY_ENDPOINT_API=https://api.cloudinary.com/v1_1/io-test
|
||||||
VITE_APP_CLOUDINARY_ENDPOINT=https://res.cloudinary.com/io-test
|
VITE_APP_CLOUDINARY_ENDPOINT=https://res.cloudinary.com/io-test
|
||||||
VITE_APP_CLOUDINARY_API_KEY=957865933348715
|
VITE_APP_CLOUDINARY_API_KEY=957865933348715
|
||||||
VITE_APP_CLOUDINARY_THUMB_TRANSFORMATIONS=c_fill,h_250,w_250
|
VITE_APP_CLOUDINARY_THUMB_TRANSFORMATIONS=c_fill,h_250,w_250
|
||||||
VITE_APP_FIREBASE_PUBLIC_VAPID_KEY='BP1B7ZTYpn-KMt6nOxlld6aS8Skt3Q7ZLEqP0hAvGHxG4UojPYiXZ6kPlzZkUC5jH-EcWXomTLtmadAIxurfcHo'
|
VITE_APP_FIREBASE_PUBLIC_VAPID_KEY='BP1B7ZTYpn-KMt6nOxlld6aS8Skt3Q7ZLEqP0hAvGHxG4UojPYiXZ6kPlzZkUC5jH-EcWXomTLtmadAIxurfcHo'
|
||||||
VITE_APP_STRIPE_PUBLIC_KEY=pk_test_51GqB4TJl3nQjrZ0wCQWAxAhlNF8jKe0tipIa6ExBaxwJGitwvFsIZUEua4dUzaMIAuXp4qwYHXx7lgjyQSwP0Pe900vzm38C7g
|
VITE_APP_STRIPE_PUBLIC_KEY=pk_test_51GqB4TJl3nQjrZ0wCQWAxAhlNF8jKe0tipIa6ExBaxwJGitwvFsIZUEua4dUzaMIAuXp4qwYHXx7lgjyQSwP0Pe900vzm38C7g
|
||||||
VITE_APP_AXIOS_BASE_API_URL=/api/
|
VITE_APP_AXIOS_BASE_API_URL=http://localhost:4000
|
||||||
VITE_APP_REPORTS_SERVER_URL=https://reports3.test.imex.online
|
VITE_APP_REPORTS_SERVER_URL=https://reports3.test.imex.online
|
||||||
VITE_APP_SPLIT_API=ts615lqgnmk84thn72uk18uu5pgce6e0l4rc
|
VITE_APP_SPLIT_API=ts615lqgnmk84thn72uk18uu5pgce6e0l4rc
|
||||||
VITE_APP_COUNTRY=USA
|
VITE_APP_COUNTRY=USA
|
||||||
|
|||||||
1
client/.gitignore
vendored
1
client/.gitignore
vendored
@@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
# Sentry Config File
|
# Sentry Config File
|
||||||
.sentryclirc
|
.sentryclirc
|
||||||
/dev-dist
|
|
||||||
|
|||||||
@@ -12,6 +12,6 @@ module.exports = defineConfig({
|
|||||||
setupNodeEvents(on, config) {
|
setupNodeEvents(on, config) {
|
||||||
return require("./cypress/plugins/index.js")(on, config);
|
return require("./cypress/plugins/index.js")(on, config);
|
||||||
},
|
},
|
||||||
baseUrl: "https://localhost:3000"
|
baseUrl: "http://localhost:3000"
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,9 +2,6 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8"/>
|
<meta charset="utf-8"/>
|
||||||
<meta http-equiv="Cache-Control" content="no-cache, no-store, must-revalidate">
|
|
||||||
<meta http-equiv="Pragma" content="no-cache">
|
|
||||||
<meta http-equiv="Expires" content="0">
|
|
||||||
<% if (env.VITE_APP_INSTANCE === 'IMEX') { %>
|
<% if (env.VITE_APP_INSTANCE === 'IMEX') { %>
|
||||||
<link rel="icon" href="/favicon.png"/>
|
<link rel="icon" href="/favicon.png"/>
|
||||||
<% } %> <% if (env.VITE_APP_INSTANCE === 'ROME') { %>
|
<% } %> <% if (env.VITE_APP_INSTANCE === 'ROME') { %>
|
||||||
@@ -17,7 +14,7 @@
|
|||||||
<meta name="theme-color" content="#1690ff"/>
|
<meta name="theme-color" content="#1690ff"/>
|
||||||
<!-- <link rel="apple-touch-icon" href="logo192.png" /> -->
|
<!-- <link rel="apple-touch-icon" href="logo192.png" /> -->
|
||||||
<!-- TODO:AIo Update the individual logos for each.-->
|
<!-- TODO:AIo Update the individual logos for each.-->
|
||||||
<link rel="apple-touch-icon" href="/logo192.png"/>
|
<link rel="apple-touch-icon" href="public/logo192.png"/>
|
||||||
<link rel="mask-icon" href="/mask-icon.svg" color="#FFFFFF">
|
<link rel="mask-icon" href="/mask-icon.svg" color="#FFFFFF">
|
||||||
<!--
|
<!--
|
||||||
manifest.json provides metadata used when your web app is installed on a
|
manifest.json provides metadata used when your web app is installed on a
|
||||||
@@ -49,23 +46,77 @@
|
|||||||
<% } %> <% if (env.VITE_APP_INSTANCE === 'ROME') { %>
|
<% } %> <% if (env.VITE_APP_INSTANCE === 'ROME') { %>
|
||||||
<meta name="description" content="Rome Online"/>
|
<meta name="description" content="Rome Online"/>
|
||||||
<title>Rome Online</title>
|
<title>Rome Online</title>
|
||||||
<script type="text/javascript" id="zsiqchat">
|
|
||||||
var $zoho = $zoho || {};
|
<!--Use the below code snippet to provide real time updates to the live chat plugin without the need of copying and paste each time to your website when changes are made via PBX-->
|
||||||
$zoho.salesiq = $zoho.salesiq || {
|
|
||||||
widgetcode: "siq01bb8ac617280bdacddfeb528f07734dadc64ef3f05efef9f769c1ec171af666",
|
<call-us-selector phonesystem-url=https://rometech.east.3cx.us:5001
|
||||||
values: {},
|
party="LiveChat528346"></call-us-selector>
|
||||||
ready: function () {
|
|
||||||
}
|
<!--Incase you don't want real time updates to the live chat plugin when options are changed, use the below code snippet. Please note that each time you change the settings you will need to copy and paste the snippet code to your website-->
|
||||||
};
|
|
||||||
var d = document;
|
<!--<call-us
|
||||||
s = d.createElement("script");
|
|
||||||
s.type = "text/javascript";
|
phonesystem-url=https://rometech.east.3cx.us:5001
|
||||||
s.id = "zsiqscript";
|
|
||||||
s.defer = true;
|
style="position:fixed;font-size:16px;line-height:17px;z-index: 99999;right: 20px; bottom: 20px;"
|
||||||
s.src = "https://salesiq.zohopublic.com/widget";
|
|
||||||
t = d.getElementsByTagName("script")[0];
|
id="wp-live-chat-by-3CX"
|
||||||
t.parentNode.insertBefore(s, t);
|
|
||||||
</script>
|
minimized="true"
|
||||||
|
|
||||||
|
animation-style="noanimation"
|
||||||
|
|
||||||
|
party="LiveChat528346"
|
||||||
|
|
||||||
|
minimized-style="bubbleright"
|
||||||
|
|
||||||
|
allow-call="true"
|
||||||
|
|
||||||
|
allow-video="false"
|
||||||
|
|
||||||
|
allow-soundnotifications="true"
|
||||||
|
|
||||||
|
enable-mute="true"
|
||||||
|
|
||||||
|
enable-onmobile="true"
|
||||||
|
|
||||||
|
offline-enabled="true"
|
||||||
|
|
||||||
|
enable="true"
|
||||||
|
|
||||||
|
ignore-queueownership="false"
|
||||||
|
|
||||||
|
authentication="both"
|
||||||
|
|
||||||
|
show-operator-actual-name="true"
|
||||||
|
|
||||||
|
aknowledge-received="true"
|
||||||
|
|
||||||
|
gdpr-enabled="false"
|
||||||
|
|
||||||
|
message-userinfo-format="name"
|
||||||
|
|
||||||
|
message-dateformat="both"
|
||||||
|
|
||||||
|
lang="browser"
|
||||||
|
|
||||||
|
button-icon-type="default"
|
||||||
|
|
||||||
|
greeting-visibility="none"
|
||||||
|
|
||||||
|
greeting-offline-visibility="none"
|
||||||
|
|
||||||
|
chat-delay="2000"
|
||||||
|
|
||||||
|
enable-direct-call="true"
|
||||||
|
|
||||||
|
enable-ga="false"
|
||||||
|
|
||||||
|
></call-us>-->
|
||||||
|
|
||||||
|
<script defer src=https://downloads-global.3cx.com/downloads/livechatandtalk/v1/callus.js
|
||||||
|
id="tcx-callus-js" charset="utf-8"></script>
|
||||||
|
|
||||||
|
|
||||||
<% } %> <% if (env.VITE_APP_INSTANCE === 'PROMANAGER') { %>
|
<% } %> <% if (env.VITE_APP_INSTANCE === 'PROMANAGER') { %>
|
||||||
<title>ProManager</title>
|
<title>ProManager</title>
|
||||||
|
|||||||
1692
client/package-lock.json
generated
1692
client/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -9,17 +9,17 @@
|
|||||||
"proxy": "http://localhost:4000",
|
"proxy": "http://localhost:4000",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ant-design/pro-layout": "^7.19.12",
|
"@ant-design/pro-layout": "^7.19.12",
|
||||||
"@apollo/client": "^3.11.8",
|
"@apollo/client": "^3.11.6",
|
||||||
"@emotion/is-prop-valid": "^1.3.1",
|
"@emotion/is-prop-valid": "^1.3.0",
|
||||||
"@fingerprintjs/fingerprintjs": "^4.5.0",
|
"@fingerprintjs/fingerprintjs": "^4.4.3",
|
||||||
"@jsreport/browser-client": "^3.1.0",
|
"@jsreport/browser-client": "^3.1.0",
|
||||||
"@reduxjs/toolkit": "^2.2.7",
|
"@reduxjs/toolkit": "^2.2.7",
|
||||||
"@sentry/cli": "^2.36.2",
|
"@sentry/cli": "^2.34.1",
|
||||||
"@sentry/react": "^7.114.0",
|
"@sentry/react": "^7.114.0",
|
||||||
"@splitsoftware/splitio-react": "^1.13.0",
|
"@splitsoftware/splitio-react": "^1.12.1",
|
||||||
"@tanem/react-nprogress": "^5.0.51",
|
"@tanem/react-nprogress": "^5.0.51",
|
||||||
"@vitejs/plugin-react": "^4.3.1",
|
"@vitejs/plugin-react": "^4.3.1",
|
||||||
"antd": "^5.20.1",
|
"antd": "^5.20.5",
|
||||||
"apollo-link-logger": "^2.0.1",
|
"apollo-link-logger": "^2.0.1",
|
||||||
"apollo-link-sentry": "^3.3.0",
|
"apollo-link-sentry": "^3.3.0",
|
||||||
"autosize": "^6.0.1",
|
"autosize": "^6.0.1",
|
||||||
@@ -32,14 +32,14 @@
|
|||||||
"dotenv": "^16.4.5",
|
"dotenv": "^16.4.5",
|
||||||
"env-cmd": "^10.1.0",
|
"env-cmd": "^10.1.0",
|
||||||
"exifr": "^7.1.3",
|
"exifr": "^7.1.3",
|
||||||
"firebase": "^10.13.2",
|
"firebase": "^10.13.1",
|
||||||
"graphql": "^16.9.0",
|
"graphql": "^16.9.0",
|
||||||
"i18next": "^23.15.1",
|
"i18next": "^23.14.0",
|
||||||
"i18next-browser-languagedetector": "^8.0.0",
|
"i18next-browser-languagedetector": "^8.0.0",
|
||||||
"immutability-helper": "^3.1.1",
|
"immutability-helper": "^3.1.1",
|
||||||
"libphonenumber-js": "^1.11.9",
|
"libphonenumber-js": "^1.11.7",
|
||||||
"logrocket": "^8.1.2",
|
"logrocket": "^8.1.2",
|
||||||
"markerjs2": "^2.32.2",
|
"markerjs2": "^2.32.1",
|
||||||
"memoize-one": "^6.0.0",
|
"memoize-one": "^6.0.0",
|
||||||
"normalize-url": "^8.0.1",
|
"normalize-url": "^8.0.1",
|
||||||
"object-hash": "^3.0.0",
|
"object-hash": "^3.0.0",
|
||||||
@@ -47,7 +47,7 @@
|
|||||||
"query-string": "^9.1.0",
|
"query-string": "^9.1.0",
|
||||||
"raf-schd": "^4.0.3",
|
"raf-schd": "^4.0.3",
|
||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-big-calendar": "^1.14.1",
|
"react-big-calendar": "^1.13.4",
|
||||||
"react-color": "^2.19.3",
|
"react-color": "^2.19.3",
|
||||||
"react-cookie": "^7.2.0",
|
"react-cookie": "^7.2.0",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^18.3.1",
|
||||||
@@ -58,15 +58,15 @@
|
|||||||
"react-icons": "^5.3.0",
|
"react-icons": "^5.3.0",
|
||||||
"react-image-lightbox": "^5.1.4",
|
"react-image-lightbox": "^5.1.4",
|
||||||
"react-markdown": "^9.0.1",
|
"react-markdown": "^9.0.1",
|
||||||
"react-number-format": "^5.4.2",
|
"react-number-format": "^5.4.1",
|
||||||
"react-popopo": "^2.1.9",
|
"react-popopo": "^2.1.9",
|
||||||
"react-product-fruits": "^2.2.61",
|
"react-product-fruits": "^2.2.61",
|
||||||
"react-redux": "^9.1.2",
|
"react-redux": "^9.1.2",
|
||||||
"react-resizable": "^3.0.5",
|
"react-resizable": "^3.0.5",
|
||||||
"react-router-dom": "^6.26.2",
|
"react-router-dom": "^6.26.1",
|
||||||
"react-sticky": "^6.0.3",
|
"react-sticky": "^6.0.3",
|
||||||
"react-virtualized": "^9.22.5",
|
"react-virtualized": "^9.22.5",
|
||||||
"react-virtuoso": "^4.10.4",
|
"react-virtuoso": "^4.10.2",
|
||||||
"recharts": "^2.12.7",
|
"recharts": "^2.12.7",
|
||||||
"redux": "^5.0.1",
|
"redux": "^5.0.1",
|
||||||
"redux-actions": "^3.0.3",
|
"redux-actions": "^3.0.3",
|
||||||
@@ -74,26 +74,22 @@
|
|||||||
"redux-saga": "^1.3.0",
|
"redux-saga": "^1.3.0",
|
||||||
"redux-state-sync": "^3.1.4",
|
"redux-state-sync": "^3.1.4",
|
||||||
"reselect": "^5.1.1",
|
"reselect": "^5.1.1",
|
||||||
"sass": "^1.79.3",
|
"sass": "^1.78.0",
|
||||||
"socket.io-client": "^4.8.0",
|
"socket.io-client": "^4.7.5",
|
||||||
"styled-components": "^6.1.13",
|
"styled-components": "^6.1.13",
|
||||||
"subscriptions-transport-ws": "^0.11.0",
|
"subscriptions-transport-ws": "^0.11.0",
|
||||||
"use-memo-one": "^1.1.3",
|
"use-memo-one": "^1.1.3",
|
||||||
"userpilot": "^1.3.6",
|
"userpilot": "^1.3.5",
|
||||||
"vite-plugin-ejs": "^1.7.0",
|
"vite-plugin-ejs": "^1.7.0",
|
||||||
"web-vitals": "^3.5.2"
|
"web-vitals": "^3.5.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"postinstall": "echo 'when updating react-big-calendar, remember to check to localizer in the calendar wrapper'",
|
|
||||||
"analyze": "source-map-explorer 'build/static/js/*.js'",
|
"analyze": "source-map-explorer 'build/static/js/*.js'",
|
||||||
"start": "vite",
|
"start": "vite",
|
||||||
"build": "dotenvx run --env-file=.env.development.imex -- vite build",
|
"build": "dotenvx run --env-file=.env.development.imex -- vite build",
|
||||||
"start:imex": "dotenvx run --env-file=.env.development.imex -- vite",
|
"start:imex": "dotenvx run --env-file=.env.development.imex -- vite",
|
||||||
"start:rome": "dotenvx run --env-file=.env.development.rome -- vite",
|
"start:rome": "dotenvx run --env-file=.env.development.rome -- vite",
|
||||||
"start:promanager": "dotenvx run --env-file=.env.development.promanager -- vite",
|
"start:promanager": "dotenvx run --env-file=.env.development.promanager -- vite",
|
||||||
"preview:imex": "dotenvx run --env-file=.env.development.imex -- vite preview",
|
|
||||||
"preview:rome": "dotenvx run --env-file=.env.development.rome -- vite preview",
|
|
||||||
"preview:promanager": "dotenvx run --env-file=.env.development.promanager -- vite preview",
|
|
||||||
"build:test:imex": "env-cmd -f .env.test.imex npm run build",
|
"build:test:imex": "env-cmd -f .env.test.imex npm run build",
|
||||||
"build:test:rome": "env-cmd -f .env.test.rome npm run build",
|
"build:test:rome": "env-cmd -f .env.test.rome npm run build",
|
||||||
"build:test:promanager": "env-cmd -f .env.test.promanager npm run build",
|
"build:test:promanager": "env-cmd -f .env.test.promanager npm run build",
|
||||||
@@ -134,30 +130,29 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/plugin-proposal-private-property-in-object": "^7.21.11",
|
"@babel/plugin-proposal-private-property-in-object": "^7.21.11",
|
||||||
"@babel/preset-react": "^7.24.7",
|
"@babel/preset-react": "^7.24.7",
|
||||||
"@dotenvx/dotenvx": "^1.14.1",
|
"@dotenvx/dotenvx": "^1.13.3",
|
||||||
"@emotion/babel-plugin": "^11.12.0",
|
"@emotion/babel-plugin": "^11.12.0",
|
||||||
"@emotion/react": "^11.13.3",
|
"@emotion/react": "^11.13.3",
|
||||||
"@sentry/webpack-plugin": "^2.22.4",
|
"@sentry/webpack-plugin": "^2.22.4",
|
||||||
"@testing-library/cypress": "^10.0.2",
|
"@testing-library/cypress": "^10.0.2",
|
||||||
"browserslist": "^4.23.3",
|
"browserslist": "^4.23.3",
|
||||||
"browserslist-to-esbuild": "^2.1.1",
|
"browserslist-to-esbuild": "^2.1.1",
|
||||||
"chalk": "^5.3.0",
|
|
||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
"cypress": "^13.14.2",
|
"cypress": "^13.14.1",
|
||||||
"eslint": "^8.57.0",
|
"eslint": "^8.57.0",
|
||||||
"eslint-config-react-app": "^7.0.1",
|
"eslint-config-react-app": "^7.0.1",
|
||||||
"eslint-plugin-cypress": "^2.15.1",
|
"eslint-plugin-cypress": "^2.15.1",
|
||||||
"memfs": "^4.12.0",
|
"memfs": "^4.11.1",
|
||||||
"os-browserify": "^0.3.0",
|
"os-browserify": "^0.3.0",
|
||||||
"react-error-overlay": "6.0.11",
|
"react-error-overlay": "6.0.11",
|
||||||
"redux-logger": "^3.0.6",
|
"redux-logger": "^3.0.6",
|
||||||
"source-map-explorer": "^2.5.3",
|
"source-map-explorer": "^2.5.3",
|
||||||
"vite": "^5.4.7",
|
"vite": "^5.4.3",
|
||||||
"vite-plugin-babel": "^1.2.0",
|
"vite-plugin-babel": "^1.2.0",
|
||||||
"vite-plugin-eslint": "^1.8.1",
|
"vite-plugin-eslint": "^1.8.1",
|
||||||
|
"vite-plugin-legacy": "^2.1.0",
|
||||||
"vite-plugin-node-polyfills": "^0.22.0",
|
"vite-plugin-node-polyfills": "^0.22.0",
|
||||||
"vite-plugin-pwa": "^0.20.5",
|
"vite-plugin-pwa": "^0.20.4",
|
||||||
"vite-plugin-style-import": "^2.0.0",
|
"vite-plugin-style-import": "^2.0.0"
|
||||||
"workbox-window": "^7.1.0"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,10 +18,10 @@ import { checkUserSession } from "../redux/user/user.actions";
|
|||||||
import { selectBodyshop, selectCurrentEula, selectCurrentUser } from "../redux/user/user.selectors";
|
import { selectBodyshop, selectCurrentEula, selectCurrentUser } from "../redux/user/user.selectors";
|
||||||
import PrivateRoute from "../components/PrivateRoute";
|
import PrivateRoute from "../components/PrivateRoute";
|
||||||
import "./App.styles.scss";
|
import "./App.styles.scss";
|
||||||
|
import handleBeta from "../utils/handleBeta";
|
||||||
import Eula from "../components/eula/eula.component";
|
import Eula from "../components/eula/eula.component";
|
||||||
import InstanceRenderMgr from "../utils/instanceRenderMgr";
|
import InstanceRenderMgr from "../utils/instanceRenderMgr";
|
||||||
import ProductFruitsWrapper from "./ProductFruitsWrapper.jsx";
|
import ProductFruitsWrapper from "./ProductFruitsWrapper.jsx";
|
||||||
import { SocketProvider } from "../contexts/SocketIO/socketContext.jsx";
|
|
||||||
|
|
||||||
const ResetPassword = lazy(() => import("../pages/reset-password/reset-password.component"));
|
const ResetPassword = lazy(() => import("../pages/reset-password/reset-password.component"));
|
||||||
const ManagePage = lazy(() => import("../pages/manage/manage.page.container"));
|
const ManagePage = lazy(() => import("../pages/manage/manage.page.container"));
|
||||||
@@ -108,6 +108,8 @@ export function App({ bodyshop, checkUserSession, currentUser, online, setOnline
|
|||||||
return <LoadingSpinner message={t("general.labels.loggingin")} />;
|
return <LoadingSpinner message={t("general.labels.loggingin")} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleBeta();
|
||||||
|
|
||||||
if (!online) {
|
if (!online) {
|
||||||
return (
|
return (
|
||||||
<Result
|
<Result
|
||||||
@@ -202,9 +204,7 @@ export function App({ bodyshop, checkUserSession, currentUser, online, setOnline
|
|||||||
path="/manage/*"
|
path="/manage/*"
|
||||||
element={
|
element={
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
<SocketProvider bodyshop={bodyshop}>
|
<PrivateRoute isAuthorized={currentUser.authorized} />
|
||||||
<PrivateRoute isAuthorized={currentUser.authorized} />
|
|
||||||
</SocketProvider>
|
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -214,9 +214,7 @@ export function App({ bodyshop, checkUserSession, currentUser, online, setOnline
|
|||||||
path="/tech/*"
|
path="/tech/*"
|
||||||
element={
|
element={
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
<SocketProvider bodyshop={bodyshop}>
|
<PrivateRoute isAuthorized={currentUser.authorized} />
|
||||||
<PrivateRoute isAuthorized={currentUser.authorized} />
|
|
||||||
</SocketProvider>
|
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -98,7 +98,7 @@ export function BillDetailEditcontainer({ setPartsOrderContext, insertAuditTrail
|
|||||||
});
|
});
|
||||||
|
|
||||||
billlines.forEach((billline) => {
|
billlines.forEach((billline) => {
|
||||||
const { deductedfromlbr, inventories, jobline, original_actual_price, create_ppc, ...il } = billline;
|
const { deductedfromlbr, inventories, jobline, ...il } = billline;
|
||||||
delete il.__typename;
|
delete il.__typename;
|
||||||
|
|
||||||
if (il.id) {
|
if (il.id) {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { DeleteFilled, CopyFilled } from "@ant-design/icons";
|
import { DeleteFilled } from "@ant-design/icons";
|
||||||
import { useLazyQuery, useMutation } from "@apollo/client";
|
import { useLazyQuery, useMutation } from "@apollo/client";
|
||||||
import { Button, Card, Col, Form, Input, Row, Space, Spin, Statistic, message, notification } from "antd";
|
import { Button, Card, Col, Form, Input, Row, Space, Spin, Statistic, notification } from "antd";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
@@ -14,12 +14,10 @@ import { selectBodyshop } from "../../redux/user/user.selectors";
|
|||||||
import AuditTrailMapping from "../../utils/AuditTrailMappings";
|
import AuditTrailMapping from "../../utils/AuditTrailMappings";
|
||||||
import CurrencyFormItemComponent from "../form-items-formatted/currency-form-item.component";
|
import CurrencyFormItemComponent from "../form-items-formatted/currency-form-item.component";
|
||||||
import JobSearchSelectComponent from "../job-search-select/job-search-select.component";
|
import JobSearchSelectComponent from "../job-search-select/job-search-select.component";
|
||||||
import { getCurrentUser } from "../../firebase/firebase.utils";
|
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
cardPaymentModal: selectCardPayment,
|
cardPaymentModal: selectCardPayment,
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop
|
||||||
currentUser: getCurrentUser
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
const mapDispatchToProps = (dispatch) => ({
|
||||||
@@ -27,17 +25,11 @@ const mapDispatchToProps = (dispatch) => ({
|
|||||||
toggleModalVisible: () => dispatch(toggleModalVisible("cardPayment"))
|
toggleModalVisible: () => dispatch(toggleModalVisible("cardPayment"))
|
||||||
});
|
});
|
||||||
|
|
||||||
const CardPaymentModalComponent = ({
|
const CardPaymentModalComponent = ({ bodyshop, cardPaymentModal, toggleModalVisible, insertAuditTrail }) => {
|
||||||
bodyshop,
|
|
||||||
currentUser,
|
|
||||||
cardPaymentModal,
|
|
||||||
toggleModalVisible,
|
|
||||||
insertAuditTrail
|
|
||||||
}) => {
|
|
||||||
const { context, actions } = cardPaymentModal;
|
const { context, actions } = cardPaymentModal;
|
||||||
|
|
||||||
const [form] = Form.useForm();
|
const [form] = Form.useForm();
|
||||||
const [paymentLink, setPaymentLink] = useState();
|
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
// const [insertPayment] = useMutation(INSERT_NEW_PAYMENT);
|
// const [insertPayment] = useMutation(INSERT_NEW_PAYMENT);
|
||||||
const [insertPaymentResponse] = useMutation(INSERT_PAYMENT_RESPONSE);
|
const [insertPaymentResponse] = useMutation(INSERT_PAYMENT_RESPONSE);
|
||||||
@@ -45,7 +37,7 @@ const CardPaymentModalComponent = ({
|
|||||||
|
|
||||||
const [, { data, refetch, queryLoading }] = useLazyQuery(QUERY_RO_AND_OWNER_BY_JOB_PKS, {
|
const [, { data, refetch, queryLoading }] = useLazyQuery(QUERY_RO_AND_OWNER_BY_JOB_PKS, {
|
||||||
variables: { jobids: [context.jobid] },
|
variables: { jobids: [context.jobid] },
|
||||||
skip: !context?.jobid
|
skip: true
|
||||||
});
|
});
|
||||||
|
|
||||||
//Initialize the intellipay window.
|
//Initialize the intellipay window.
|
||||||
@@ -59,7 +51,8 @@ const CardPaymentModalComponent = ({
|
|||||||
//2024-04-25: Nothing is going to happen here anymore. We'll completely rely on the callback.
|
//2024-04-25: Nothing is going to happen here anymore. We'll completely rely on the callback.
|
||||||
//Add a slight delay to allow the refetch to properly get the data.
|
//Add a slight delay to allow the refetch to properly get the data.
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
if (actions && actions.refetch && typeof actions.refetch === "function") actions.refetch();
|
if (actions && actions.refetch && typeof actions.refetch === "function")
|
||||||
|
actions.refetch();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
toggleModalVisible();
|
toggleModalVisible();
|
||||||
}, 750);
|
}, 750);
|
||||||
@@ -93,6 +86,7 @@ const CardPaymentModalComponent = ({
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const handleIntelliPayCharge = async () => {
|
const handleIntelliPayCharge = async () => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
//Validate
|
//Validate
|
||||||
@@ -107,7 +101,7 @@ const CardPaymentModalComponent = ({
|
|||||||
const response = await axios.post("/intellipay/lightbox_credentials", {
|
const response = await axios.post("/intellipay/lightbox_credentials", {
|
||||||
bodyshop,
|
bodyshop,
|
||||||
refresh: !!window.intellipay,
|
refresh: !!window.intellipay,
|
||||||
paymentSplitMeta: form.getFieldsValue()
|
paymentSplitMeta: form.getFieldsValue(),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (window.intellipay) {
|
if (window.intellipay) {
|
||||||
@@ -132,42 +126,6 @@ const CardPaymentModalComponent = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleIntelliPayChargeShortLink = async () => {
|
|
||||||
setLoading(true);
|
|
||||||
//Validate
|
|
||||||
try {
|
|
||||||
await form.validateFields();
|
|
||||||
} catch (error) {
|
|
||||||
setLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const { payments } = form.getFieldsValue();
|
|
||||||
const response = await axios.post("/intellipay/generate_payment_url", {
|
|
||||||
bodyshop,
|
|
||||||
amount: payments?.reduce((acc, val) => {
|
|
||||||
return acc + (val?.amount || 0);
|
|
||||||
}, 0),
|
|
||||||
account: payments && data && data.jobs.length > 0 ? data.jobs.map((j) => j.ro_number).join(", ") : null,
|
|
||||||
comment: btoa(JSON.stringify({ payments, userEmail: currentUser.email })),
|
|
||||||
paymentSplitMeta: form.getFieldsValue()
|
|
||||||
});
|
|
||||||
if (response.data) {
|
|
||||||
setPaymentLink(response.data?.shorUrl);
|
|
||||||
navigator.clipboard.writeText(response.data?.shorUrl);
|
|
||||||
message.success(t("general.actions.copied"));
|
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
} catch (error) {
|
|
||||||
notification.open({
|
|
||||||
type: "error",
|
|
||||||
message: t("job_payments.notifications.error.openingip")
|
|
||||||
});
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card title="Card Payment">
|
<Card title="Card Payment">
|
||||||
<Spin spinning={loading}>
|
<Spin spinning={loading}>
|
||||||
@@ -244,14 +202,16 @@ const CardPaymentModalComponent = ({
|
|||||||
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
shouldUpdate={(prevValues, curValues) =>
|
shouldUpdate={(prevValues, curValues) =>
|
||||||
prevValues.payments?.map((p) => p?.jobid + p?.amount).join() !==
|
prevValues.payments?.map((p) => p?.jobid).join() !== curValues.payments?.map((p) => p?.jobid).join()
|
||||||
curValues.payments?.map((p) => p?.jobid + p?.amount).join()
|
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{() => {
|
{() => {
|
||||||
//If all of the job ids have been fileld in, then query and update the IP field.
|
//If all of the job ids have been fileld in, then query and update the IP field.
|
||||||
const { payments } = form.getFieldsValue();
|
const { payments } = form.getFieldsValue();
|
||||||
if (payments?.length > 0 && payments?.filter((p) => p?.jobid).length === payments?.length) {
|
if (
|
||||||
|
payments?.length > 0 &&
|
||||||
|
payments?.filter((p) => p?.jobid).length === payments?.length
|
||||||
|
) {
|
||||||
refetch({ jobids: payments.map((p) => p.jobid) });
|
refetch({ jobids: payments.map((p) => p.jobid) });
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
@@ -286,6 +246,7 @@ const CardPaymentModalComponent = ({
|
|||||||
const totalAmountToCharge = payments?.reduce((acc, val) => {
|
const totalAmountToCharge = payments?.reduce((acc, val) => {
|
||||||
return acc + (val?.amount || 0);
|
return acc + (val?.amount || 0);
|
||||||
}, 0);
|
}, 0);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Space style={{ float: "right" }}>
|
<Space style={{ float: "right" }}>
|
||||||
<Statistic title="Amount To Charge" value={totalAmountToCharge} precision={2} />
|
<Statistic title="Amount To Charge" value={totalAmountToCharge} precision={2} />
|
||||||
@@ -312,36 +273,11 @@ const CardPaymentModalComponent = ({
|
|||||||
>
|
>
|
||||||
{t("job_payments.buttons.proceedtopayment")}
|
{t("job_payments.buttons.proceedtopayment")}
|
||||||
</Button>
|
</Button>
|
||||||
<Space direction="vertical" align="center">
|
|
||||||
<Button
|
|
||||||
type="primary"
|
|
||||||
// data-ipayname="submit"
|
|
||||||
className="ipayfield"
|
|
||||||
loading={queryLoading || loading}
|
|
||||||
disabled={!(totalAmountToCharge > 0)}
|
|
||||||
onClick={handleIntelliPayChargeShortLink}
|
|
||||||
>
|
|
||||||
{t("job_payments.buttons.create_short_link")}
|
|
||||||
</Button>
|
|
||||||
</Space>
|
|
||||||
</Space>
|
</Space>
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
</Form>
|
</Form>
|
||||||
{paymentLink && (
|
|
||||||
<Space
|
|
||||||
style={{ cursor: "pointer", float: "right" }}
|
|
||||||
align="end"
|
|
||||||
onClick={() => {
|
|
||||||
navigator.clipboard.writeText(paymentLink);
|
|
||||||
message.success(t("general.actions.copied"));
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div>{paymentLink}</div>
|
|
||||||
<CopyFilled />
|
|
||||||
</Space>
|
|
||||||
)}
|
|
||||||
</Spin>
|
</Spin>
|
||||||
</Card>
|
</Card>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import Icon, {
|
|||||||
FileFilled,
|
FileFilled,
|
||||||
HomeFilled,
|
HomeFilled,
|
||||||
ImportOutlined,
|
ImportOutlined,
|
||||||
|
InfoCircleOutlined,
|
||||||
LineChartOutlined,
|
LineChartOutlined,
|
||||||
PaperClipOutlined,
|
PaperClipOutlined,
|
||||||
PhoneOutlined,
|
PhoneOutlined,
|
||||||
@@ -26,8 +27,8 @@ import Icon, {
|
|||||||
UserOutlined
|
UserOutlined
|
||||||
} from "@ant-design/icons";
|
} from "@ant-design/icons";
|
||||||
import { useSplitTreatments } from "@splitsoftware/splitio-react";
|
import { useSplitTreatments } from "@splitsoftware/splitio-react";
|
||||||
import { Layout, Menu } from "antd";
|
import { Layout, Menu, Switch, Tooltip } from "antd";
|
||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { BsKanban } from "react-icons/bs";
|
import { BsKanban } from "react-icons/bs";
|
||||||
import { FaCalendarAlt, FaCarCrash, FaCreditCard, FaFileInvoiceDollar, FaTasks } from "react-icons/fa";
|
import { FaCalendarAlt, FaCarCrash, FaCreditCard, FaFileInvoiceDollar, FaTasks } from "react-icons/fa";
|
||||||
@@ -42,6 +43,7 @@ import { selectRecentItems, selectSelectedHeader } from "../../redux/application
|
|||||||
import { setModalContext } from "../../redux/modals/modals.actions";
|
import { setModalContext } from "../../redux/modals/modals.actions";
|
||||||
import { signOutStart } from "../../redux/user/user.actions";
|
import { signOutStart } from "../../redux/user/user.actions";
|
||||||
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
||||||
|
import { checkBeta, handleBeta, setBeta } from "../../utils/handleBeta";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
import { HasFeatureAccess } from "../feature-wrapper/feature-wrapper.component";
|
import { HasFeatureAccess } from "../feature-wrapper/feature-wrapper.component";
|
||||||
|
|
||||||
@@ -113,21 +115,19 @@ function Header({
|
|||||||
names: ["ImEXPay", "DmsAp", "Simple_Inventory"],
|
names: ["ImEXPay", "DmsAp", "Simple_Inventory"],
|
||||||
splitKey: bodyshop && bodyshop.imexshopid
|
splitKey: bodyshop && bodyshop.imexshopid
|
||||||
});
|
});
|
||||||
|
const [betaSwitch, setBetaSwitch] = useState(false);
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
const deleteBetaCookie = () => {
|
useEffect(() => {
|
||||||
const cookieExists = document.cookie.split("; ").some((row) => row.startsWith(`betaSwitchImex=`));
|
const isBeta = checkBeta();
|
||||||
if (cookieExists) {
|
setBetaSwitch(isBeta);
|
||||||
const domain = window.location.hostname.split(".").slice(-2).join(".");
|
}, []);
|
||||||
document.cookie = `betaSwitchImex=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/; domain=.${domain}`;
|
|
||||||
console.log(`betaSwitchImex cookie deleted`);
|
|
||||||
} else {
|
|
||||||
console.log(`betaSwitchImex cookie does not exist`);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
deleteBetaCookie();
|
const betaSwitchChange = (checked) => {
|
||||||
|
setBeta(checked);
|
||||||
|
setBetaSwitch(checked);
|
||||||
|
handleBeta();
|
||||||
|
};
|
||||||
|
|
||||||
const accountingChildren = [];
|
const accountingChildren = [];
|
||||||
|
|
||||||
@@ -695,6 +695,31 @@ function Header({
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
|
InstanceRenderManager({
|
||||||
|
executeFunction: true,
|
||||||
|
args: [],
|
||||||
|
imex: () => {
|
||||||
|
menuItems.push({
|
||||||
|
key: "beta-switch",
|
||||||
|
id: "header-beta-switch",
|
||||||
|
style: { marginLeft: "auto" },
|
||||||
|
label: (
|
||||||
|
<Tooltip
|
||||||
|
title={`A more modern ${InstanceRenderManager({
|
||||||
|
imex: t("titles.imexonline"),
|
||||||
|
rome: t("titles.romeonline"),
|
||||||
|
promanager: t("titles.promanager")
|
||||||
|
})} is ready for you to try! You can switch back at any time.`}
|
||||||
|
>
|
||||||
|
<InfoCircleOutlined />
|
||||||
|
<span style={{ marginRight: 8 }}>Try the new app</span>
|
||||||
|
<Switch checked={betaSwitch} onChange={betaSwitchChange} />
|
||||||
|
</Tooltip>
|
||||||
|
)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Layout.Header>
|
<Layout.Header>
|
||||||
<Menu
|
<Menu
|
||||||
|
|||||||
@@ -141,14 +141,10 @@ export function JobTotalsTableTotals({ bodyshop, job }) {
|
|||||||
key: t("jobs.fields.ded_amt"),
|
key: t("jobs.fields.ded_amt"),
|
||||||
total: job.job_totals.totals.custPayable.deductible
|
total: job.job_totals.totals.custPayable.deductible
|
||||||
},
|
},
|
||||||
...(InstanceRenderManager({
|
// {
|
||||||
imex: [{
|
// key: t("jobs.fields.federal_tax_payable"),
|
||||||
key: t("jobs.fields.federal_tax_payable"),
|
// total: job.job_totals.totals.custPayable.federal_tax,
|
||||||
total: job.job_totals.totals.custPayable.federal_tax
|
// },
|
||||||
}],
|
|
||||||
rome: [],
|
|
||||||
promanager: "USE_ROME"
|
|
||||||
})),
|
|
||||||
{
|
{
|
||||||
key: t("jobs.fields.other_amount_payable"),
|
key: t("jobs.fields.other_amount_payable"),
|
||||||
total: job.job_totals.totals.custPayable.other_customer_amount
|
total: job.job_totals.totals.custPayable.other_customer_amount
|
||||||
|
|||||||
@@ -250,8 +250,8 @@ export function JobsList({ bodyshop }) {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: t("jobs.labels.estimator"),
|
title: t("jobs.labels.estimator"),
|
||||||
dataIndex: "estimator",
|
dataIndex: "jobs.labels.estimator",
|
||||||
key: "estimator",
|
key: "jobs.labels.estimator",
|
||||||
ellipsis: true,
|
ellipsis: true,
|
||||||
responsive: ["xl"],
|
responsive: ["xl"],
|
||||||
sorter: (a, b) =>
|
sorter: (a, b) =>
|
||||||
|
|||||||
@@ -27,10 +27,6 @@ export default function PartsOrderModalPriceChange({ form, field }) {
|
|||||||
key: "25",
|
key: "25",
|
||||||
label: t("parts_orders.labels.discount", { percent: "25%" })
|
label: t("parts_orders.labels.discount", { percent: "25%" })
|
||||||
},
|
},
|
||||||
{
|
|
||||||
key: "40",
|
|
||||||
label: t("parts_orders.labels.discount", { percent: "40%" })
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
key: "custom",
|
key: "custom",
|
||||||
label: (
|
label: (
|
||||||
|
|||||||
@@ -8,12 +8,11 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
import { openChatByPhone, setMessage } from "../../redux/messaging/messaging.actions";
|
import { openChatByPhone, setMessage } from "../../redux/messaging/messaging.actions";
|
||||||
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
import { selectBodyshop } from "../../redux/user/user.selectors";
|
||||||
import CurrencyFormItemComponent from "../form-items-formatted/currency-form-item.component";
|
import CurrencyFormItemComponent from "../form-items-formatted/currency-form-item.component";
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop
|
||||||
currentUser: selectCurrentUser
|
|
||||||
});
|
});
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
const mapDispatchToProps = (dispatch) => ({
|
||||||
openChatByPhone: (phone) => dispatch(openChatByPhone(phone)),
|
openChatByPhone: (phone) => dispatch(openChatByPhone(phone)),
|
||||||
@@ -21,7 +20,7 @@ const mapDispatchToProps = (dispatch) => ({
|
|||||||
});
|
});
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(PaymentsGenerateLink);
|
export default connect(mapStateToProps, mapDispatchToProps)(PaymentsGenerateLink);
|
||||||
|
|
||||||
export function PaymentsGenerateLink({ bodyshop, currentUser, callback, job, openChatByPhone, setMessage }) {
|
export function PaymentsGenerateLink({ bodyshop, callback, job, openChatByPhone, setMessage }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [form] = Form.useForm();
|
const [form] = Form.useForm();
|
||||||
|
|
||||||
@@ -31,35 +30,29 @@ export function PaymentsGenerateLink({ bodyshop, currentUser, callback, job, ope
|
|||||||
|
|
||||||
const handleFinish = async ({ amount }) => {
|
const handleFinish = async ({ amount }) => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
let p;
|
|
||||||
try {
|
const p = parsePhoneNumber(job.ownr_ph1, "CA");
|
||||||
p = parsePhoneNumber(job.ownr_ph1 || "", "CA");
|
|
||||||
} catch (error) {
|
|
||||||
console.log("Unable to parse phone number");
|
|
||||||
}
|
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
const response = await axios.post("/intellipay/generate_payment_url", {
|
const response = await axios.post("/intellipay/generate_payment_url", {
|
||||||
bodyshop,
|
bodyshop,
|
||||||
amount: amount,
|
amount: amount,
|
||||||
account: job.ro_number,
|
account: job.ro_number,
|
||||||
comment: btoa(JSON.stringify({ payments: [{ jobid: job.id, amount }], userEmail: currentUser.email }))
|
invoice: job.id
|
||||||
});
|
});
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
setPaymentLink(response.data.shorUrl);
|
setPaymentLink(response.data.shorUrl);
|
||||||
|
|
||||||
if (p) {
|
openChatByPhone({
|
||||||
openChatByPhone({
|
phone_num: p.formatInternational(),
|
||||||
phone_num: p.formatInternational(),
|
jobid: job.id
|
||||||
jobid: job.id
|
});
|
||||||
});
|
setMessage(
|
||||||
setMessage(
|
t("payments.labels.smspaymentreminder", {
|
||||||
t("payments.labels.smspaymentreminder", {
|
shopname: bodyshop.shopname,
|
||||||
shopname: bodyshop.shopname,
|
amount: amount,
|
||||||
amount: amount,
|
payment_link: response.data.shorUrl
|
||||||
payment_link: response.data.shorUrl
|
})
|
||||||
})
|
);
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
//Add in confirmation & errors.
|
//Add in confirmation & errors.
|
||||||
if (callback) callback();
|
if (callback) callback();
|
||||||
|
|||||||
@@ -6,11 +6,11 @@ import {
|
|||||||
PauseCircleOutlined
|
PauseCircleOutlined
|
||||||
} from "@ant-design/icons";
|
} from "@ant-design/icons";
|
||||||
import { Card, Col, Row, Space, Tooltip } from "antd";
|
import { Card, Col, Row, Space, Tooltip } from "antd";
|
||||||
import Dinero from "dinero.js";
|
|
||||||
import React, { useMemo } from "react";
|
import React, { useMemo } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
import { DateTimeFormatter } from "../../utils/DateFormatter";
|
import { DateTimeFormatter } from "../../utils/DateFormatter";
|
||||||
|
import Dinero from "dinero.js";
|
||||||
|
|
||||||
import ProductionAlert from "../production-list-columns/production-list-columns.alert.component";
|
import ProductionAlert from "../production-list-columns/production-list-columns.alert.component";
|
||||||
import ProductionListColumnProductionNote from "../production-list-columns/production-list-columns.productionnote.component";
|
import ProductionListColumnProductionNote from "../production-list-columns/production-list-columns.productionnote.component";
|
||||||
@@ -18,8 +18,8 @@ import ProductionSubletsManageComponent from "../production-sublets-manage/produ
|
|||||||
|
|
||||||
import dayjs from "../../utils/day";
|
import dayjs from "../../utils/day";
|
||||||
|
|
||||||
import JobPartsQueueCount from "../job-parts-queue-count/job-parts-queue-count.component";
|
|
||||||
import OwnerNameDisplay from "../owner-name-display/owner-name-display.component";
|
import OwnerNameDisplay from "../owner-name-display/owner-name-display.component";
|
||||||
|
import JobPartsQueueCount from "../job-parts-queue-count/job-parts-queue-count.component";
|
||||||
|
|
||||||
const cardColor = (ssbuckets, totalHrs) => {
|
const cardColor = (ssbuckets, totalHrs) => {
|
||||||
const bucket = ssbuckets.find((bucket) => bucket.gte <= totalHrs && (!bucket.lt || bucket.lt > totalHrs));
|
const bucket = ssbuckets.find((bucket) => bucket.gte <= totalHrs && (!bucket.lt || bucket.lt > totalHrs));
|
||||||
@@ -213,13 +213,21 @@ const EstimatorToolTip = ({ metadata, cardSettings }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const SubtotalTooltip = ({ metadata, cardSettings, t }) => {
|
const SubtotalTooltip = ({ metadata, cardSettings, t }) => {
|
||||||
const dineroAmount = Dinero(metadata?.job_totals?.totals?.subtotal ?? Dinero()).toFormat();
|
const amount = metadata?.job_totals?.totals?.subtotal?.amount;
|
||||||
|
const dineroAmount = amount ? Dinero({ amount: parseInt(amount * 100) }).toFormat("0,0.00") : null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
cardSettings?.subtotal && (
|
cardSettings?.subtotal && (
|
||||||
<Col span={cardSettings.compact ? 24 : 12}>
|
<Col span={cardSettings.compact ? 24 : 12}>
|
||||||
<EllipsesToolTip title={`${dineroAmount}`} kiosk={cardSettings.kiosk}>
|
<EllipsesToolTip
|
||||||
{dineroAmount}
|
title={!!amount ? `${t("production.statistics.currency_symbol")}${dineroAmount}` : null}
|
||||||
|
kiosk={cardSettings.kiosk}
|
||||||
|
>
|
||||||
|
{!!amount ? (
|
||||||
|
<span>{`${t("production.statistics.currency_symbol")}${dineroAmount}`}</span>
|
||||||
|
) : (
|
||||||
|
<span> </span>
|
||||||
|
)}
|
||||||
</EllipsesToolTip>
|
</EllipsesToolTip>
|
||||||
</Col>
|
</Col>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
import React, { useEffect, useMemo, useRef } from "react";
|
import React, { useEffect, useMemo } from "react";
|
||||||
import { useQuery, useSubscription } from "@apollo/client";
|
import { useQuery, useSubscription } from "@apollo/client";
|
||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
import {
|
import { QUERY_JOBS_IN_PRODUCTION, SUBSCRIPTION_JOBS_IN_PRODUCTION } from "../../graphql/jobs.queries";
|
||||||
QUERY_JOBS_IN_PRODUCTION,
|
|
||||||
SUBSCRIPTION_JOBS_IN_PRODUCTION,
|
|
||||||
SUBSCRIPTION_JOBS_IN_PRODUCTION_VIEW
|
|
||||||
} from "../../graphql/jobs.queries";
|
|
||||||
import { QUERY_KANBAN_SETTINGS } from "../../graphql/user.queries";
|
import { QUERY_KANBAN_SETTINGS } from "../../graphql/user.queries";
|
||||||
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
||||||
import ProductionBoardKanbanComponent from "./production-board-kanban.component";
|
import ProductionBoardKanbanComponent from "./production-board-kanban.component";
|
||||||
@@ -16,9 +12,7 @@ const mapStateToProps = createStructuredSelector({
|
|||||||
currentUser: selectCurrentUser
|
currentUser: selectCurrentUser
|
||||||
});
|
});
|
||||||
|
|
||||||
function ProductionBoardKanbanContainer({ bodyshop, currentUser, subscriptionType = "direct" }) {
|
function ProductionBoardKanbanContainer({ bodyshop, currentUser }) {
|
||||||
const fired = useRef(false); // useRef to keep track of whether the subscription fired
|
|
||||||
|
|
||||||
const combinedStatuses = useMemo(
|
const combinedStatuses = useMemo(
|
||||||
() => [
|
() => [
|
||||||
...bodyshop.md_ro_statuses.production_statuses,
|
...bodyshop.md_ro_statuses.production_statuses,
|
||||||
@@ -34,12 +28,9 @@ function ProductionBoardKanbanContainer({ bodyshop, currentUser, subscriptionTyp
|
|||||||
onError: (error) => console.error(`Error fetching jobs in production: ${error.message}`)
|
onError: (error) => console.error(`Error fetching jobs in production: ${error.message}`)
|
||||||
});
|
});
|
||||||
|
|
||||||
const { data: updatedJobs } = useSubscription(
|
const { data: updatedJobs } = useSubscription(SUBSCRIPTION_JOBS_IN_PRODUCTION, {
|
||||||
subscriptionType === "view" ? SUBSCRIPTION_JOBS_IN_PRODUCTION_VIEW : SUBSCRIPTION_JOBS_IN_PRODUCTION,
|
onError: (error) => console.error(`Error subscribing to jobs in production: ${error.message}`)
|
||||||
{
|
});
|
||||||
onError: (error) => console.error(`Error subscribing to jobs in production: ${error.message}`)
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
const { loading: associationSettingsLoading, data: associationSettings } = useQuery(QUERY_KANBAN_SETTINGS, {
|
const { loading: associationSettingsLoading, data: associationSettings } = useQuery(QUERY_KANBAN_SETTINGS, {
|
||||||
variables: { email: currentUser.email },
|
variables: { email: currentUser.email },
|
||||||
@@ -49,15 +40,10 @@ function ProductionBoardKanbanContainer({ bodyshop, currentUser, subscriptionTyp
|
|||||||
// const currentReducerData = useSelector((state) => (state.trello.lanes ? state.trello : {}));
|
// const currentReducerData = useSelector((state) => (state.trello.lanes ? state.trello : {}));
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!updatedJobs) {
|
if (updatedJobs && data) {
|
||||||
return;
|
refetch().catch((err) => console.error(`Error re-fetching jobs in production: ${err.message}`));
|
||||||
}
|
}
|
||||||
if (!fired.current) {
|
}, [updatedJobs, data, refetch]);
|
||||||
fired.current = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
refetch().catch((err) => console.error(`Error re-fetching jobs in production: ${err.message}`));
|
|
||||||
}, [updatedJobs, refetch]);
|
|
||||||
|
|
||||||
const filteredAssociationSettings = useMemo(() => {
|
const filteredAssociationSettings = useMemo(() => {
|
||||||
return associationSettings?.associations[0] || null;
|
return associationSettings?.associations[0] || null;
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import { Card, Statistic } from "antd";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import PropTypes from "prop-types";
|
import PropTypes from "prop-types";
|
||||||
import { defaultKanbanSettings, statisticsItems } from "./settings/defaultKanbanSettings.js";
|
import { defaultKanbanSettings, statisticsItems } from "./settings/defaultKanbanSettings.js";
|
||||||
import Dinero from "dinero.js";
|
|
||||||
|
|
||||||
export const StatisticType = {
|
export const StatisticType = {
|
||||||
HOURS: "hours",
|
HOURS: "hours",
|
||||||
@@ -33,21 +32,7 @@ const ProductionStatistics = ({ data, cardSettings, reducerData }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const calculateTotalAmount = (items, key) => {
|
const calculateTotalAmount = (items, key) => {
|
||||||
return items.reduce((acc, item) => acc.add(Dinero(item[key]?.totals?.subtotal ?? Dinero())), Dinero({ amount: 0 }));
|
return items.reduce((acc, item) => acc + (item[key]?.totals?.subtotal?.amount || 0), 0);
|
||||||
};
|
|
||||||
|
|
||||||
const calculateReducerTotalAmount = (lanes, key) => {
|
|
||||||
return lanes.reduce(
|
|
||||||
(acc, lane) => {
|
|
||||||
return acc.add(
|
|
||||||
lane.cards.reduce(
|
|
||||||
(laneAcc, card) => laneAcc.add(Dinero(card.metadata[key]?.totals?.subtotal ?? Dinero())),
|
|
||||||
Dinero({ amount: 0 })
|
|
||||||
)
|
|
||||||
);
|
|
||||||
},
|
|
||||||
Dinero({ amount: 0 })
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const calculateReducerTotal = (lanes, key, subKey) => {
|
const calculateReducerTotal = (lanes, key, subKey) => {
|
||||||
@@ -58,6 +43,14 @@ const ProductionStatistics = ({ data, cardSettings, reducerData }) => {
|
|||||||
}, 0);
|
}, 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const calculateReducerTotalAmount = (lanes, key) => {
|
||||||
|
return lanes.reduce((acc, lane) => {
|
||||||
|
return (
|
||||||
|
acc + lane.cards.reduce((laneAcc, card) => laneAcc + (card.metadata[key]?.totals?.subtotal?.amount || 0), 0)
|
||||||
|
);
|
||||||
|
}, 0);
|
||||||
|
};
|
||||||
|
|
||||||
const formatValue = (value, type) => {
|
const formatValue = (value, type) => {
|
||||||
if (type === StatisticType.JOBS) {
|
if (type === StatisticType.JOBS) {
|
||||||
return value.toFixed(0);
|
return value.toFixed(0);
|
||||||
@@ -94,15 +87,9 @@ const ProductionStatistics = ({ data, cardSettings, reducerData }) => {
|
|||||||
const totalAmountInProduction = useMemo(() => {
|
const totalAmountInProduction = useMemo(() => {
|
||||||
if (!cardSettings.totalAmountInProduction) return null;
|
if (!cardSettings.totalAmountInProduction) return null;
|
||||||
const total = calculateTotalAmount(data, "job_totals");
|
const total = calculateTotalAmount(data, "job_totals");
|
||||||
return total.toFormat("$0,0.00");
|
return parseFloat(total.toFixed(2));
|
||||||
}, [data, cardSettings.totalAmountInProduction]);
|
}, [data, cardSettings.totalAmountInProduction]);
|
||||||
|
|
||||||
const totalAmountOnBoard = useMemo(() => {
|
|
||||||
if (!reducerData || !cardSettings.totalAmountOnBoard) return null;
|
|
||||||
const total = calculateReducerTotalAmount(reducerData.lanes, "job_totals");
|
|
||||||
return total.toFormat("$0,0.00");
|
|
||||||
}, [reducerData, cardSettings.totalAmountOnBoard]);
|
|
||||||
|
|
||||||
const totalHrsOnBoard = useMemo(() => {
|
const totalHrsOnBoard = useMemo(() => {
|
||||||
if (!reducerData || !cardSettings.totalHrsOnBoard) return null;
|
if (!reducerData || !cardSettings.totalHrsOnBoard) return null;
|
||||||
const total =
|
const total =
|
||||||
@@ -131,6 +118,12 @@ const ProductionStatistics = ({ data, cardSettings, reducerData }) => {
|
|||||||
[reducerData, cardSettings.jobsOnBoard]
|
[reducerData, cardSettings.jobsOnBoard]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const totalAmountOnBoard = useMemo(() => {
|
||||||
|
if (!reducerData || !cardSettings.totalAmountOnBoard) return null;
|
||||||
|
const total = calculateReducerTotalAmount(reducerData.lanes, "job_totals");
|
||||||
|
return parseFloat(total.toFixed(2));
|
||||||
|
}, [reducerData, cardSettings.totalAmountOnBoard]);
|
||||||
|
|
||||||
const tasksInProduction = useMemo(() => {
|
const tasksInProduction = useMemo(() => {
|
||||||
if (!data || !cardSettings.tasksInProduction) return null;
|
if (!data || !cardSettings.tasksInProduction) return null;
|
||||||
return data.reduce((acc, item) => acc + (item.tasks_aggregate?.aggregate?.count || 0), 0);
|
return data.reduce((acc, item) => acc + (item.tasks_aggregate?.aggregate?.count || 0), 0);
|
||||||
@@ -198,6 +191,7 @@ const ProductionStatistics = ({ data, cardSettings, reducerData }) => {
|
|||||||
<Statistic
|
<Statistic
|
||||||
title={t(`production.statistics.${stat.label}`)}
|
title={t(`production.statistics.${stat.label}`)}
|
||||||
value={formatValue(stat.value, stat.type)}
|
value={formatValue(stat.value, stat.type)}
|
||||||
|
prefix={stat.type === StatisticType.AMOUNT ? t("production.statistics.currency_symbol") : undefined}
|
||||||
suffix={
|
suffix={
|
||||||
stat.type === StatisticType.HOURS
|
stat.type === StatisticType.HOURS
|
||||||
? t("production.statistics.hours")
|
? t("production.statistics.hours")
|
||||||
|
|||||||
@@ -17,6 +17,7 @@
|
|||||||
border-radius: 5px 5px 0 0;
|
border-radius: 5px 5px 0 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.production-alert {
|
.production-alert {
|
||||||
background: transparent;
|
background: transparent;
|
||||||
border: none;
|
border: none;
|
||||||
@@ -69,8 +70,3 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.clone.is-dragging .ant-card {
|
|
||||||
border: #1890ff 2px solid !important;
|
|
||||||
border-radius: 12px;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -25,8 +25,8 @@ function getFurthestAway({ pageBorderBox, draggable, candidates }) {
|
|||||||
const axis = candidate.axis;
|
const axis = candidate.axis;
|
||||||
const target = patch(
|
const target = patch(
|
||||||
candidate.axis.line,
|
candidate.axis.line,
|
||||||
// use the center of the list on the main axis
|
// use the current center of the dragging item on the main axis
|
||||||
candidate.page.borderBox.center[axis.line],
|
pageBorderBox.center[axis.line],
|
||||||
// use the center of the list on the cross axis
|
// use the center of the list on the cross axis
|
||||||
candidate.page.borderBox.center[axis.crossAxisLine]
|
candidate.page.borderBox.center[axis.crossAxisLine]
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import getBodyElement from "../get-body-element";
|
|||||||
const isEqual = (base) => (value) => base === value;
|
const isEqual = (base) => (value) => base === value;
|
||||||
const isScroll = isEqual("scroll");
|
const isScroll = isEqual("scroll");
|
||||||
const isAuto = isEqual("auto");
|
const isAuto = isEqual("auto");
|
||||||
const isOverlay = isEqual("overlay");
|
|
||||||
const isVisible = isEqual("visible");
|
const isVisible = isEqual("visible");
|
||||||
const isEither = (overflow, fn) => fn(overflow.overflowX) || fn(overflow.overflowY);
|
const isEither = (overflow, fn) => fn(overflow.overflowX) || fn(overflow.overflowY);
|
||||||
const isBoth = (overflow, fn) => fn(overflow.overflowX) && fn(overflow.overflowY);
|
const isBoth = (overflow, fn) => fn(overflow.overflowX) && fn(overflow.overflowY);
|
||||||
@@ -15,7 +14,7 @@ const isElementScrollable = (el) => {
|
|||||||
overflowX: style.overflowX,
|
overflowX: style.overflowX,
|
||||||
overflowY: style.overflowY
|
overflowY: style.overflowY
|
||||||
};
|
};
|
||||||
return isEither(overflow, isScroll) || isEither(overflow, isAuto) || isEither(overflow, isOverlay);
|
return isEither(overflow, isScroll) || isEither(overflow, isAuto);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Special case for a body element
|
// Special case for a body element
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ function getSelector(contextId) {
|
|||||||
return `[${attributes.dragHandle.contextId}="${contextId}"]`;
|
return `[${attributes.dragHandle.contextId}="${contextId}"]`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function findClosestDragHandleFromEvent(contextId, event) {
|
function findClosestDragHandleFromEvent(contextId, event) {
|
||||||
const target = event.target;
|
const target = event.target;
|
||||||
if (!isElement(target)) {
|
if (!isElement(target)) {
|
||||||
warning("event.target must be a Element");
|
warning("event.target must be a Element");
|
||||||
|
|||||||
@@ -240,14 +240,11 @@ export default function useTouchSensor(api) {
|
|||||||
y: clientY
|
y: clientY
|
||||||
};
|
};
|
||||||
|
|
||||||
const handle = api.findClosestDragHandle(event);
|
|
||||||
invariant(handle, "Touch sensor unable to find drag handle");
|
|
||||||
|
|
||||||
// unbind this event handler
|
// unbind this event handler
|
||||||
unbindEventsRef.current();
|
unbindEventsRef.current();
|
||||||
|
|
||||||
// eslint-disable-next-line no-use-before-define
|
// eslint-disable-next-line no-use-before-define
|
||||||
startPendingDrag(actions, point, handle);
|
startPendingDrag(actions, point);
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
// not including stop or startPendingDrag as it is not defined initially
|
// not including stop or startPendingDrag as it is not defined initially
|
||||||
@@ -291,7 +288,7 @@ export default function useTouchSensor(api) {
|
|||||||
}
|
}
|
||||||
}, [stop]);
|
}, [stop]);
|
||||||
const bindCapturingEvents = useCallback(
|
const bindCapturingEvents = useCallback(
|
||||||
function bindCapturingEvents(target) {
|
function bindCapturingEvents() {
|
||||||
const options = {
|
const options = {
|
||||||
capture: true,
|
capture: true,
|
||||||
passive: false
|
passive: false
|
||||||
@@ -310,7 +307,7 @@ export default function useTouchSensor(api) {
|
|||||||
// Old behaviour:
|
// Old behaviour:
|
||||||
// https://gist.github.com/parris/dda613e3ae78f14eb2dc9fa0f4bfce3d
|
// https://gist.github.com/parris/dda613e3ae78f14eb2dc9fa0f4bfce3d
|
||||||
// https://stackoverflow.com/questions/33298828/touch-move-event-dont-fire-after-touch-start-target-is-removed
|
// https://stackoverflow.com/questions/33298828/touch-move-event-dont-fire-after-touch-start-target-is-removed
|
||||||
const unbindTarget = bindEvents(target, getHandleBindings(args), options);
|
const unbindTarget = bindEvents(window, getHandleBindings(args), options);
|
||||||
const unbindWindow = bindEvents(window, getWindowBindings(args), options);
|
const unbindWindow = bindEvents(window, getWindowBindings(args), options);
|
||||||
unbindEventsRef.current = function unbindAll() {
|
unbindEventsRef.current = function unbindAll() {
|
||||||
unbindTarget();
|
unbindTarget();
|
||||||
@@ -333,7 +330,7 @@ export default function useTouchSensor(api) {
|
|||||||
[getPhase, setPhase]
|
[getPhase, setPhase]
|
||||||
);
|
);
|
||||||
const startPendingDrag = useCallback(
|
const startPendingDrag = useCallback(
|
||||||
function startPendingDrag(actions, point, target) {
|
function startPendingDrag(actions, point) {
|
||||||
invariant(getPhase().type === "IDLE", "Expected to move from IDLE to PENDING drag");
|
invariant(getPhase().type === "IDLE", "Expected to move from IDLE to PENDING drag");
|
||||||
const longPressTimerId = setTimeout(startDragging, timeForLongPress);
|
const longPressTimerId = setTimeout(startDragging, timeForLongPress);
|
||||||
setPhase({
|
setPhase({
|
||||||
@@ -342,7 +339,7 @@ export default function useTouchSensor(api) {
|
|||||||
actions,
|
actions,
|
||||||
longPressTimerId
|
longPressTimerId
|
||||||
});
|
});
|
||||||
bindCapturingEvents(target);
|
bindCapturingEvents();
|
||||||
},
|
},
|
||||||
[bindCapturingEvents, getPhase, setPhase, startDragging]
|
[bindCapturingEvents, getPhase, setPhase, startDragging]
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -23,9 +23,7 @@ import getBorderBoxCenterPosition from "../get-border-box-center-position";
|
|||||||
import { warning } from "../../dev-warning";
|
import { warning } from "../../dev-warning";
|
||||||
import useLayoutEffect from "../use-isomorphic-layout-effect";
|
import useLayoutEffect from "../use-isomorphic-layout-effect";
|
||||||
import { noop } from "../../empty";
|
import { noop } from "../../empty";
|
||||||
import findClosestDraggableIdFromEvent, {
|
import findClosestDraggableIdFromEvent from "./find-closest-draggable-id-from-event";
|
||||||
findClosestDragHandleFromEvent
|
|
||||||
} from "./find-closest-draggable-id-from-event";
|
|
||||||
import findDraggable from "../get-elements/find-draggable";
|
import findDraggable from "../get-elements/find-draggable";
|
||||||
import bindEvents from "../event-bindings/bind-events";
|
import bindEvents from "../event-bindings/bind-events";
|
||||||
|
|
||||||
@@ -341,9 +339,6 @@ export default function useSensorMarshal({ contextId, store, registry, customSen
|
|||||||
}),
|
}),
|
||||||
[contextId, lockAPI, registry, store]
|
[contextId, lockAPI, registry, store]
|
||||||
);
|
);
|
||||||
|
|
||||||
const findClosestDragHandle = useCallback((event) => findClosestDragHandleFromEvent(contextId, event), [contextId]);
|
|
||||||
|
|
||||||
const findClosestDraggableId = useCallback((event) => findClosestDraggableIdFromEvent(contextId, event), [contextId]);
|
const findClosestDraggableId = useCallback((event) => findClosestDraggableIdFromEvent(contextId, event), [contextId]);
|
||||||
const findOptionsForDraggable = useCallback(
|
const findOptionsForDraggable = useCallback(
|
||||||
(id) => {
|
(id) => {
|
||||||
@@ -375,18 +370,9 @@ export default function useSensorMarshal({ contextId, store, registry, customSen
|
|||||||
findClosestDraggableId,
|
findClosestDraggableId,
|
||||||
findOptionsForDraggable,
|
findOptionsForDraggable,
|
||||||
tryReleaseLock,
|
tryReleaseLock,
|
||||||
isLockClaimed,
|
isLockClaimed
|
||||||
findClosestDragHandle
|
|
||||||
}),
|
}),
|
||||||
[
|
[canGetLock, tryGetLock, findClosestDraggableId, findOptionsForDraggable, tryReleaseLock, isLockClaimed]
|
||||||
canGetLock,
|
|
||||||
tryGetLock,
|
|
||||||
findClosestDraggableId,
|
|
||||||
findOptionsForDraggable,
|
|
||||||
tryReleaseLock,
|
|
||||||
isLockClaimed,
|
|
||||||
findClosestDragHandle
|
|
||||||
]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
// Bad ass
|
// Bad ass
|
||||||
|
|||||||
@@ -83,13 +83,7 @@ const getFinalStyles = (contextId) => {
|
|||||||
return {
|
return {
|
||||||
selector: getSelector(attributes.draggable.contextId),
|
selector: getSelector(attributes.draggable.contextId),
|
||||||
styles: {
|
styles: {
|
||||||
dragging: `
|
dragging: transition,
|
||||||
${transition}
|
|
||||||
user-select: none;
|
|
||||||
-webkit-user-select: none;
|
|
||||||
-moz-user-select: none;
|
|
||||||
-ms-user-select: none;
|
|
||||||
`,
|
|
||||||
dropAnimating: transition,
|
dropAnimating: transition,
|
||||||
userCancel: transition
|
userCancel: transition
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -67,9 +67,7 @@ export default function useStyleMarshal(contextId, nonce) {
|
|||||||
const remove = (ref) => {
|
const remove = (ref) => {
|
||||||
const current = ref.current;
|
const current = ref.current;
|
||||||
invariant(current, "Cannot unmount ref as it is not set");
|
invariant(current, "Cannot unmount ref as it is not set");
|
||||||
if (getHead().contains(current)) {
|
getHead().removeChild(current);
|
||||||
getHead().removeChild(current);
|
|
||||||
}
|
|
||||||
ref.current = null;
|
ref.current = null;
|
||||||
};
|
};
|
||||||
remove(alwaysRef);
|
remove(alwaysRef);
|
||||||
|
|||||||
@@ -298,16 +298,6 @@ const r = ({ technician, state, activeStatuses, data, bodyshop, refetch, treatme
|
|||||||
ellipsis: true,
|
ellipsis: true,
|
||||||
sorter: (a, b) => statusSort(a.status, b.status, activeStatuses),
|
sorter: (a, b) => statusSort(a.status, b.status, activeStatuses),
|
||||||
sortOrder: state.sortedInfo.columnKey === "status" && state.sortedInfo.order,
|
sortOrder: state.sortedInfo.columnKey === "status" && state.sortedInfo.order,
|
||||||
filters:
|
|
||||||
activeStatuses
|
|
||||||
?.map((s) => {
|
|
||||||
return {
|
|
||||||
text: s || "No Status*",
|
|
||||||
value: [s]
|
|
||||||
};
|
|
||||||
})
|
|
||||||
.sort((a, b) => statusSort(a.text, b.text, activeStatuses)) || [],
|
|
||||||
onFilter: (value, record) => value.includes(record.status),
|
|
||||||
render: (text, record) => <ProductionListColumnStatus record={record} />
|
render: (text, record) => <ProductionListColumnStatus record={record} />
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -4,13 +4,12 @@ import {
|
|||||||
QUERY_EXACT_JOB_IN_PRODUCTION,
|
QUERY_EXACT_JOB_IN_PRODUCTION,
|
||||||
QUERY_EXACT_JOBS_IN_PRODUCTION,
|
QUERY_EXACT_JOBS_IN_PRODUCTION,
|
||||||
QUERY_JOBS_IN_PRODUCTION,
|
QUERY_JOBS_IN_PRODUCTION,
|
||||||
SUBSCRIPTION_JOBS_IN_PRODUCTION,
|
SUBSCRIPTION_JOBS_IN_PRODUCTION
|
||||||
SUBSCRIPTION_JOBS_IN_PRODUCTION_VIEW
|
|
||||||
} from "../../graphql/jobs.queries";
|
} from "../../graphql/jobs.queries";
|
||||||
import ProductionListTable from "./production-list-table.component";
|
import ProductionListTable from "./production-list-table.component";
|
||||||
import _ from "lodash";
|
import _ from "lodash";
|
||||||
|
|
||||||
export default function ProductionListTableContainer({ subscriptionType = "direct" }) {
|
export default function ProductionListTableContainer() {
|
||||||
const { refetch, loading, data } = useQuery(QUERY_JOBS_IN_PRODUCTION, {
|
const { refetch, loading, data } = useQuery(QUERY_JOBS_IN_PRODUCTION, {
|
||||||
pollInterval: 3600000,
|
pollInterval: 3600000,
|
||||||
fetchPolicy: "network-only",
|
fetchPolicy: "network-only",
|
||||||
@@ -18,9 +17,7 @@ export default function ProductionListTableContainer({ subscriptionType = "direc
|
|||||||
});
|
});
|
||||||
const client = useApolloClient();
|
const client = useApolloClient();
|
||||||
const [joblist, setJoblist] = useState([]);
|
const [joblist, setJoblist] = useState([]);
|
||||||
const { data: updatedJobs } = useSubscription(
|
const { data: updatedJobs } = useSubscription(SUBSCRIPTION_JOBS_IN_PRODUCTION);
|
||||||
subscriptionType === "view" ? SUBSCRIPTION_JOBS_IN_PRODUCTION_VIEW : SUBSCRIPTION_JOBS_IN_PRODUCTION
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!(data && data.jobs)) return;
|
if (!(data && data.jobs)) return;
|
||||||
|
|||||||
@@ -34,34 +34,28 @@ export function ReportCenterModalComponent({ reportCenterModal, bodyshop }) {
|
|||||||
const [form] = Form.useForm();
|
const [form] = Form.useForm();
|
||||||
const [search, setSearch] = useState("");
|
const [search, setSearch] = useState("");
|
||||||
const {
|
const {
|
||||||
treatments: { Enhanced_Payroll, ADPPayroll }
|
treatments: { Enhanced_Payroll }
|
||||||
} = useSplitTreatments({
|
} = useSplitTreatments({
|
||||||
attributes: {},
|
attributes: {},
|
||||||
names: ["Enhanced_Payroll", "ADPPayroll"],
|
names: ["Enhanced_Payroll"],
|
||||||
splitKey: bodyshop.imexshopid
|
splitKey: bodyshop.imexshopid
|
||||||
});
|
});
|
||||||
|
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const Templates = TemplateList("report_center");
|
const Templates = TemplateList("report_center");
|
||||||
const ReportsList = Object.keys(Templates)
|
const ReportsList =
|
||||||
.map((key) => Templates[key])
|
Enhanced_Payroll.treatment === "on"
|
||||||
.filter((temp) => {
|
? Object.keys(Templates)
|
||||||
const enhancedPayrollOn = Enhanced_Payroll.treatment === "on";
|
.map((key) => {
|
||||||
const adpPayrollOn = ADPPayroll.treatment === "on";
|
return Templates[key];
|
||||||
|
})
|
||||||
if (enhancedPayrollOn && adpPayrollOn) {
|
.filter((temp) => temp.enhanced_payroll === undefined || temp.enhanced_payroll === true)
|
||||||
return temp.enhanced_payroll !== false || temp.adp_payroll !== false;
|
: Object.keys(Templates)
|
||||||
}
|
.map((key) => {
|
||||||
if (enhancedPayrollOn) {
|
return Templates[key];
|
||||||
return temp.enhanced_payroll !== false && temp.adp_payroll !== true;
|
})
|
||||||
}
|
.filter((temp) => temp.enhanced_payroll === undefined || temp.enhanced_payroll === false);
|
||||||
if (adpPayrollOn) {
|
|
||||||
return temp.adp_payroll !== false && temp.enhanced_payroll !== true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return temp.enhanced_payroll !== true && temp.adp_payroll !== true;
|
|
||||||
});
|
|
||||||
|
|
||||||
const { open } = reportCenterModal;
|
const { open } = reportCenterModal;
|
||||||
|
|
||||||
@@ -110,7 +104,7 @@ export function ReportCenterModalComponent({ reportCenterModal, bodyshop }) {
|
|||||||
to: values.to,
|
to: values.to,
|
||||||
subject: Templates[values.key]?.subject
|
subject: Templates[values.key]?.subject
|
||||||
},
|
},
|
||||||
values.sendbytext === "text" ? values.sendbytext : values.sendbyexcel === "excel" ? "x" : values.sendby === "email" ? "e" : "p",
|
values.sendbyexcel === "excel" ? "x" : values.sendby === "email" ? "e" : "p",
|
||||||
id
|
id
|
||||||
);
|
);
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
@@ -297,15 +291,7 @@ export function ReportCenterModalComponent({ reportCenterModal, bodyshop }) {
|
|||||||
</Radio.Group>
|
</Radio.Group>
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
);
|
);
|
||||||
if (reporttype === "text")
|
if (reporttype !== "excel")
|
||||||
return (
|
|
||||||
<Form.Item label={t("general.labels.sendby")} name="sendbytext" initialValue="text">
|
|
||||||
<Radio.Group>
|
|
||||||
<Radio value="text">{t("general.labels.text")}</Radio>
|
|
||||||
</Radio.Group>
|
|
||||||
</Form.Item>
|
|
||||||
);
|
|
||||||
if (reporttype !== "excel" || reporttype !== "text")
|
|
||||||
return (
|
return (
|
||||||
<Form.Item label={t("general.labels.sendby")} name="sendby" initialValue="print">
|
<Form.Item label={t("general.labels.sendby")} name="sendby" initialValue="print">
|
||||||
<Radio.Group>
|
<Radio.Group>
|
||||||
|
|||||||
@@ -1,505 +0,0 @@
|
|||||||
import isBetween from "dayjs/plugin/isBetween";
|
|
||||||
import isSameOrAfter from "dayjs/plugin/isSameOrAfter";
|
|
||||||
import isSameOrBefore from "dayjs/plugin/isSameOrBefore";
|
|
||||||
import localeData from "dayjs/plugin/localeData";
|
|
||||||
import localizedFormat from "dayjs/plugin/localizedFormat";
|
|
||||||
import minMax from "dayjs/plugin/minMax";
|
|
||||||
import utc from "dayjs/plugin/utc";
|
|
||||||
import { DateLocalizer } from "react-big-calendar";
|
|
||||||
|
|
||||||
function arrayWithHoles(arr) {
|
|
||||||
if (Array.isArray(arr)) return arr;
|
|
||||||
}
|
|
||||||
|
|
||||||
function iterableToArrayLimit(arr, i) {
|
|
||||||
if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return;
|
|
||||||
var _arr = [];
|
|
||||||
var _n = true;
|
|
||||||
var _d = false;
|
|
||||||
var _e = undefined;
|
|
||||||
try {
|
|
||||||
for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) {
|
|
||||||
_arr.push(_s.value);
|
|
||||||
if (i && _arr.length === i) break;
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
_d = true;
|
|
||||||
_e = err;
|
|
||||||
} finally {
|
|
||||||
try {
|
|
||||||
if (!_n && _i["return"] != null) _i["return"]();
|
|
||||||
} finally {
|
|
||||||
if (_d) throw _e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return _arr;
|
|
||||||
}
|
|
||||||
|
|
||||||
function unsupportedIterableToArray(o, minLen) {
|
|
||||||
if (!o) return;
|
|
||||||
if (typeof o === "string") return arrayLikeToArray(o, minLen);
|
|
||||||
var n = Object.prototype.toString.call(o).slice(8, -1);
|
|
||||||
if (n === "Object" && o.constructor) n = o.constructor.name;
|
|
||||||
if (n === "Map" || n === "Set") return Array.from(o);
|
|
||||||
if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return arrayLikeToArray(o, minLen);
|
|
||||||
}
|
|
||||||
|
|
||||||
function arrayLikeToArray(arr, len) {
|
|
||||||
if (len == null || len > arr.length) len = arr.length;
|
|
||||||
for (var i = 0, arr2 = new Array(len); i < len; i++) {
|
|
||||||
arr2[i] = arr[i];
|
|
||||||
}
|
|
||||||
return arr2;
|
|
||||||
}
|
|
||||||
|
|
||||||
function nonIterableRest() {
|
|
||||||
throw new TypeError(
|
|
||||||
"Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function _slicedToArray(arr, i) {
|
|
||||||
return arrayWithHoles(arr) || iterableToArrayLimit(arr, i) || unsupportedIterableToArray(arr, i) || nonIterableRest();
|
|
||||||
}
|
|
||||||
|
|
||||||
function fixUnit(unit) {
|
|
||||||
var datePart = unit ? unit.toLowerCase() : unit;
|
|
||||||
if (datePart === "FullYear") {
|
|
||||||
datePart = "year";
|
|
||||||
} else if (!datePart) {
|
|
||||||
datePart = undefined;
|
|
||||||
}
|
|
||||||
return datePart;
|
|
||||||
}
|
|
||||||
|
|
||||||
var timeRangeFormat = function timeRangeFormat(_ref3, culture, local) {
|
|
||||||
var start = _ref3.start,
|
|
||||||
end = _ref3.end;
|
|
||||||
return local.format(start, "LT", culture) + " – " + local.format(end, "LT", culture);
|
|
||||||
};
|
|
||||||
var timeRangeStartFormat = function timeRangeStartFormat(_ref4, culture, local) {
|
|
||||||
var start = _ref4.start;
|
|
||||||
return local.format(start, "LT", culture) + " – ";
|
|
||||||
};
|
|
||||||
var timeRangeEndFormat = function timeRangeEndFormat(_ref5, culture, local) {
|
|
||||||
var end = _ref5.end;
|
|
||||||
return " – " + local.format(end, "LT", culture);
|
|
||||||
};
|
|
||||||
var weekRangeFormat = function weekRangeFormat(_ref, culture, local) {
|
|
||||||
var start = _ref.start,
|
|
||||||
end = _ref.end;
|
|
||||||
return (
|
|
||||||
local.format(start, "MMMM DD", culture) +
|
|
||||||
" – " +
|
|
||||||
// updated to use this localizer 'eq()' method
|
|
||||||
local.format(end, local.eq(start, end, "month") ? "DD" : "MMMM DD", culture)
|
|
||||||
);
|
|
||||||
};
|
|
||||||
var dateRangeFormat = function dateRangeFormat(_ref2, culture, local) {
|
|
||||||
var start = _ref2.start,
|
|
||||||
end = _ref2.end;
|
|
||||||
return local.format(start, "L", culture) + " – " + local.format(end, "L", culture);
|
|
||||||
};
|
|
||||||
|
|
||||||
var formats = {
|
|
||||||
dateFormat: "DD",
|
|
||||||
dayFormat: "DD ddd",
|
|
||||||
weekdayFormat: "ddd",
|
|
||||||
selectRangeFormat: timeRangeFormat,
|
|
||||||
eventTimeRangeFormat: timeRangeFormat,
|
|
||||||
eventTimeRangeStartFormat: timeRangeStartFormat,
|
|
||||||
eventTimeRangeEndFormat: timeRangeEndFormat,
|
|
||||||
timeGutterFormat: "LT",
|
|
||||||
monthHeaderFormat: "MMMM YYYY",
|
|
||||||
dayHeaderFormat: "dddd MMM DD",
|
|
||||||
dayRangeHeaderFormat: weekRangeFormat,
|
|
||||||
agendaHeaderFormat: dateRangeFormat,
|
|
||||||
agendaDateFormat: "ddd MMM DD",
|
|
||||||
agendaTimeFormat: "LT",
|
|
||||||
agendaTimeRangeFormat: timeRangeFormat
|
|
||||||
};
|
|
||||||
|
|
||||||
const localizer = (dayjsLib) => {
|
|
||||||
// load dayjs plugins
|
|
||||||
dayjsLib.extend(isBetween);
|
|
||||||
dayjsLib.extend(isSameOrAfter);
|
|
||||||
dayjsLib.extend(isSameOrBefore);
|
|
||||||
dayjsLib.extend(localeData);
|
|
||||||
dayjsLib.extend(localizedFormat);
|
|
||||||
dayjsLib.extend(minMax);
|
|
||||||
dayjsLib.extend(utc);
|
|
||||||
var locale = function locale(dj, c) {
|
|
||||||
return c ? dj.locale(c) : dj;
|
|
||||||
};
|
|
||||||
|
|
||||||
// if the timezone plugin is loaded,
|
|
||||||
// then use the timezone aware version
|
|
||||||
|
|
||||||
//TODO This was the issue entirely...
|
|
||||||
// var dayjs = dayjsLib.tz ? dayjsLib.tz : dayjsLib;
|
|
||||||
var dayjs = dayjsLib;
|
|
||||||
|
|
||||||
function getTimezoneOffset(date) {
|
|
||||||
// ensures this gets cast to timezone
|
|
||||||
return dayjs(date).toDate().getTimezoneOffset();
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDstOffset(start, end) {
|
|
||||||
var _st$tz$$x$$timezone;
|
|
||||||
// convert to dayjs, in case
|
|
||||||
var st = dayjs(start);
|
|
||||||
var ed = dayjs(end);
|
|
||||||
// if not using the dayjs timezone plugin
|
|
||||||
if (!dayjs.tz) {
|
|
||||||
return st.toDate().getTimezoneOffset() - ed.toDate().getTimezoneOffset();
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* If a default timezone has been applied, then
|
|
||||||
* use this to get the proper timezone offset, otherwise default
|
|
||||||
* the timezone to the browser local
|
|
||||||
*/
|
|
||||||
var tzName =
|
|
||||||
(_st$tz$$x$$timezone = st.tz().$x.$timezone) !== null && _st$tz$$x$$timezone !== void 0
|
|
||||||
? _st$tz$$x$$timezone
|
|
||||||
: dayjsLib.tz.guess();
|
|
||||||
// invert offsets to be inline with moment.js
|
|
||||||
var startOffset = -dayjs.tz(+st, tzName).utcOffset();
|
|
||||||
var endOffset = -dayjs.tz(+ed, tzName).utcOffset();
|
|
||||||
return startOffset - endOffset;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDayStartDstOffset(start) {
|
|
||||||
var dayStart = dayjs(start).startOf("day");
|
|
||||||
return getDstOffset(dayStart, start);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*** BEGIN localized date arithmetic methods with dayjs ***/
|
|
||||||
function defineComparators(a, b, unit) {
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
var dtA = datePart ? dayjs(a).startOf(datePart) : dayjs(a);
|
|
||||||
var dtB = datePart ? dayjs(b).startOf(datePart) : dayjs(b);
|
|
||||||
return [dtA, dtB, datePart];
|
|
||||||
}
|
|
||||||
|
|
||||||
function startOf() {
|
|
||||||
var date = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : null;
|
|
||||||
var unit = arguments.length > 1 ? arguments[1] : undefined;
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
if (datePart) {
|
|
||||||
return dayjs(date).startOf(datePart).toDate();
|
|
||||||
}
|
|
||||||
return dayjs(date).toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function endOf() {
|
|
||||||
var date = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : null;
|
|
||||||
var unit = arguments.length > 1 ? arguments[1] : undefined;
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
if (datePart) {
|
|
||||||
return dayjs(date).endOf(datePart).toDate();
|
|
||||||
}
|
|
||||||
return dayjs(date).toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
// dayjs comparison operations *always* convert both sides to dayjs objects
|
|
||||||
// prior to running the comparisons
|
|
||||||
function eq(a, b, unit) {
|
|
||||||
var _defineComparators = defineComparators(a, b, unit),
|
|
||||||
_defineComparators2 = _slicedToArray(_defineComparators, 3),
|
|
||||||
dtA = _defineComparators2[0],
|
|
||||||
dtB = _defineComparators2[1],
|
|
||||||
datePart = _defineComparators2[2];
|
|
||||||
return dtA.isSame(dtB, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function neq(a, b, unit) {
|
|
||||||
return !eq(a, b, unit);
|
|
||||||
}
|
|
||||||
|
|
||||||
function gt(a, b, unit) {
|
|
||||||
var _defineComparators3 = defineComparators(a, b, unit),
|
|
||||||
_defineComparators4 = _slicedToArray(_defineComparators3, 3),
|
|
||||||
dtA = _defineComparators4[0],
|
|
||||||
dtB = _defineComparators4[1],
|
|
||||||
datePart = _defineComparators4[2];
|
|
||||||
return dtA.isAfter(dtB, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function lt(a, b, unit) {
|
|
||||||
var _defineComparators5 = defineComparators(a, b, unit),
|
|
||||||
_defineComparators6 = _slicedToArray(_defineComparators5, 3),
|
|
||||||
dtA = _defineComparators6[0],
|
|
||||||
dtB = _defineComparators6[1],
|
|
||||||
datePart = _defineComparators6[2];
|
|
||||||
return dtA.isBefore(dtB, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function gte(a, b, unit) {
|
|
||||||
var _defineComparators7 = defineComparators(a, b, unit),
|
|
||||||
_defineComparators8 = _slicedToArray(_defineComparators7, 3),
|
|
||||||
dtA = _defineComparators8[0],
|
|
||||||
dtB = _defineComparators8[1],
|
|
||||||
datePart = _defineComparators8[2];
|
|
||||||
return dtA.isSameOrBefore(dtB, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function lte(a, b, unit) {
|
|
||||||
var _defineComparators9 = defineComparators(a, b, unit),
|
|
||||||
_defineComparators10 = _slicedToArray(_defineComparators9, 3),
|
|
||||||
dtA = _defineComparators10[0],
|
|
||||||
dtB = _defineComparators10[1],
|
|
||||||
datePart = _defineComparators10[2];
|
|
||||||
return dtA.isSameOrBefore(dtB, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function inRange(day, min, max) {
|
|
||||||
var unit = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : "day";
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
var djDay = dayjs(day);
|
|
||||||
var djMin = dayjs(min);
|
|
||||||
var djMax = dayjs(max);
|
|
||||||
return djDay.isBetween(djMin, djMax, datePart, "[]");
|
|
||||||
}
|
|
||||||
|
|
||||||
function min(dateA, dateB) {
|
|
||||||
var dtA = dayjs(dateA);
|
|
||||||
var dtB = dayjs(dateB);
|
|
||||||
var minDt = dayjsLib.min(dtA, dtB);
|
|
||||||
return minDt.toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function max(dateA, dateB) {
|
|
||||||
var dtA = dayjs(dateA);
|
|
||||||
var dtB = dayjs(dateB);
|
|
||||||
var maxDt = dayjsLib.max(dtA, dtB);
|
|
||||||
return maxDt.toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function merge(date, time) {
|
|
||||||
if (!date && !time) return null;
|
|
||||||
var tm = dayjs(time).format("HH:mm:ss");
|
|
||||||
var dt = dayjs(date).startOf("day").format("MM/DD/YYYY");
|
|
||||||
// We do it this way to avoid issues when timezone switching
|
|
||||||
return dayjsLib("".concat(dt, " ").concat(tm), "MM/DD/YYYY HH:mm:ss").toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function add(date, adder, unit) {
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
return dayjs(date).add(adder, datePart).toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function range(start, end) {
|
|
||||||
var unit = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : "day";
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
// because the add method will put these in tz, we have to start that way
|
|
||||||
var current = dayjs(start).toDate();
|
|
||||||
var days = [];
|
|
||||||
while (lte(current, end)) {
|
|
||||||
days.push(current);
|
|
||||||
current = add(current, 1, datePart);
|
|
||||||
}
|
|
||||||
return days;
|
|
||||||
}
|
|
||||||
|
|
||||||
function ceil(date, unit) {
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
var floor = startOf(date, datePart);
|
|
||||||
return eq(floor, date) ? floor : add(floor, 1, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function diff(a, b) {
|
|
||||||
var unit = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : "day";
|
|
||||||
var datePart = fixUnit(unit);
|
|
||||||
// don't use 'defineComparators' here, as we don't want to mutate the values
|
|
||||||
var dtA = dayjs(a);
|
|
||||||
var dtB = dayjs(b);
|
|
||||||
return dtB.diff(dtA, datePart);
|
|
||||||
}
|
|
||||||
|
|
||||||
function minutes(date) {
|
|
||||||
var dt = dayjs(date);
|
|
||||||
return dt.minutes();
|
|
||||||
}
|
|
||||||
|
|
||||||
function firstOfWeek(culture) {
|
|
||||||
var data = culture ? dayjsLib.localeData(culture) : dayjsLib.localeData();
|
|
||||||
return data ? data.firstDayOfWeek() : 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
function firstVisibleDay(date) {
|
|
||||||
return dayjs(date).startOf("month").startOf("week").toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function lastVisibleDay(date) {
|
|
||||||
return dayjs(date).endOf("month").endOf("week").toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
function visibleDays(date) {
|
|
||||||
var current = firstVisibleDay(date);
|
|
||||||
var last = lastVisibleDay(date);
|
|
||||||
var days = [];
|
|
||||||
while (lte(current, last)) {
|
|
||||||
days.push(current);
|
|
||||||
current = add(current, 1, "d");
|
|
||||||
}
|
|
||||||
return days;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*** END localized date arithmetic methods with dayjs ***/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Moved from TimeSlots.js, this method overrides the method of the same name
|
|
||||||
* in the localizer.js, using dayjs to construct the js Date
|
|
||||||
* @param {Date} dt - date to start with
|
|
||||||
* @param {Number} minutesFromMidnight
|
|
||||||
* @param {Number} offset
|
|
||||||
* @returns {Date}
|
|
||||||
*/
|
|
||||||
function getSlotDate(dt, minutesFromMidnight, offset) {
|
|
||||||
return dayjs(dt)
|
|
||||||
.startOf("day")
|
|
||||||
.minute(minutesFromMidnight + offset)
|
|
||||||
.toDate();
|
|
||||||
}
|
|
||||||
|
|
||||||
// dayjs will automatically handle DST differences in it's calculations
|
|
||||||
function getTotalMin(start, end) {
|
|
||||||
return diff(start, end, "minutes");
|
|
||||||
}
|
|
||||||
|
|
||||||
function getMinutesFromMidnight(start) {
|
|
||||||
var dayStart = dayjs(start).startOf("day");
|
|
||||||
var day = dayjs(start);
|
|
||||||
return day.diff(dayStart, "minutes") + getDayStartDstOffset(start);
|
|
||||||
}
|
|
||||||
|
|
||||||
// These two are used by DateSlotMetrics
|
|
||||||
function continuesPrior(start, first) {
|
|
||||||
var djStart = dayjs(start);
|
|
||||||
var djFirst = dayjs(first);
|
|
||||||
return djStart.isBefore(djFirst, "day");
|
|
||||||
}
|
|
||||||
|
|
||||||
function continuesAfter(start, end, last) {
|
|
||||||
var djEnd = dayjs(end);
|
|
||||||
var djLast = dayjs(last);
|
|
||||||
return djEnd.isSameOrAfter(djLast, "minutes");
|
|
||||||
}
|
|
||||||
|
|
||||||
function daySpan(start, end) {
|
|
||||||
var startDay = dayjs(start);
|
|
||||||
var endDay = dayjs(end);
|
|
||||||
return endDay.diff(startDay, "day");
|
|
||||||
}
|
|
||||||
|
|
||||||
// These two are used by eventLevels
|
|
||||||
function sortEvents(_ref6) {
|
|
||||||
var _ref6$evtA = _ref6.evtA,
|
|
||||||
aStart = _ref6$evtA.start,
|
|
||||||
aEnd = _ref6$evtA.end,
|
|
||||||
aAllDay = _ref6$evtA.allDay,
|
|
||||||
_ref6$evtB = _ref6.evtB,
|
|
||||||
bStart = _ref6$evtB.start,
|
|
||||||
bEnd = _ref6$evtB.end,
|
|
||||||
bAllDay = _ref6$evtB.allDay;
|
|
||||||
var startSort = +startOf(aStart, "day") - +startOf(bStart, "day");
|
|
||||||
var durA = daySpan(aStart, aEnd);
|
|
||||||
var durB = daySpan(bStart, bEnd);
|
|
||||||
return (
|
|
||||||
startSort ||
|
|
||||||
// sort by start Day first
|
|
||||||
durB - durA ||
|
|
||||||
// events spanning multiple days go first
|
|
||||||
!!bAllDay - !!aAllDay ||
|
|
||||||
// then allDay single day events
|
|
||||||
+aStart - +bStart ||
|
|
||||||
// then sort by start time *don't need dayjs conversion here
|
|
||||||
+aEnd - +bEnd // then sort by end time *don't need dayjs conversion here either
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function inEventRange(_ref7) {
|
|
||||||
var _ref7$event = _ref7.event,
|
|
||||||
start = _ref7$event.start,
|
|
||||||
end = _ref7$event.end,
|
|
||||||
_ref7$range = _ref7.range,
|
|
||||||
rangeStart = _ref7$range.start,
|
|
||||||
rangeEnd = _ref7$range.end;
|
|
||||||
var startOfDay = dayjs(start).startOf("day");
|
|
||||||
var eEnd = dayjs(end);
|
|
||||||
var rStart = dayjs(rangeStart);
|
|
||||||
var rEnd = dayjs(rangeEnd);
|
|
||||||
var startsBeforeEnd = startOfDay.isSameOrBefore(rEnd, "day");
|
|
||||||
// when the event is zero duration we need to handle a bit differently
|
|
||||||
var sameMin = !startOfDay.isSame(eEnd, "minutes");
|
|
||||||
var endsAfterStart = sameMin ? eEnd.isAfter(rStart, "minutes") : eEnd.isSameOrAfter(rStart, "minutes");
|
|
||||||
return startsBeforeEnd && endsAfterStart;
|
|
||||||
}
|
|
||||||
|
|
||||||
function isSameDate(date1, date2) {
|
|
||||||
var dt = dayjs(date1);
|
|
||||||
var dt2 = dayjs(date2);
|
|
||||||
return dt.isSame(dt2, "day");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method, called once in the localizer constructor, is used by eventLevels
|
|
||||||
* 'eventSegments()' to assist in determining the 'span' of the event in the display,
|
|
||||||
* specifically when using a timezone that is greater than the browser native timezone.
|
|
||||||
* @returns number
|
|
||||||
*/
|
|
||||||
function browserTZOffset() {
|
|
||||||
/**
|
|
||||||
* Date.prototype.getTimezoneOffset horrifically flips the positive/negative from
|
|
||||||
* what you see in it's string, so we have to jump through some hoops to get a value
|
|
||||||
* we can actually compare.
|
|
||||||
*/
|
|
||||||
var dt = new Date();
|
|
||||||
var neg = /-/.test(dt.toString()) ? "-" : "";
|
|
||||||
var dtOffset = dt.getTimezoneOffset();
|
|
||||||
var comparator = Number("".concat(neg).concat(Math.abs(dtOffset)));
|
|
||||||
// dayjs correctly provides positive/negative offset, as expected
|
|
||||||
var mtOffset = dayjs().utcOffset();
|
|
||||||
return mtOffset > comparator ? 1 : 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return new DateLocalizer({
|
|
||||||
formats: formats,
|
|
||||||
firstOfWeek: firstOfWeek,
|
|
||||||
firstVisibleDay: firstVisibleDay,
|
|
||||||
lastVisibleDay: lastVisibleDay,
|
|
||||||
visibleDays: visibleDays,
|
|
||||||
format: function format(value, _format, culture) {
|
|
||||||
return locale(dayjs(value), culture).format(_format);
|
|
||||||
},
|
|
||||||
lt: lt,
|
|
||||||
lte: lte,
|
|
||||||
gt: gt,
|
|
||||||
gte: gte,
|
|
||||||
eq: eq,
|
|
||||||
neq: neq,
|
|
||||||
merge: merge,
|
|
||||||
inRange: inRange,
|
|
||||||
startOf: startOf,
|
|
||||||
endOf: endOf,
|
|
||||||
range: range,
|
|
||||||
add: add,
|
|
||||||
diff: diff,
|
|
||||||
ceil: ceil,
|
|
||||||
min: min,
|
|
||||||
max: max,
|
|
||||||
minutes: minutes,
|
|
||||||
getSlotDate: getSlotDate,
|
|
||||||
getTimezoneOffset: getTimezoneOffset,
|
|
||||||
getDstOffset: getDstOffset,
|
|
||||||
getTotalMin: getTotalMin,
|
|
||||||
getMinutesFromMidnight: getMinutesFromMidnight,
|
|
||||||
continuesPrior: continuesPrior,
|
|
||||||
continuesAfter: continuesAfter,
|
|
||||||
sortEvents: sortEvents,
|
|
||||||
inEventRange: inEventRange,
|
|
||||||
isSameDate: isSameDate,
|
|
||||||
browserTZOffset: browserTZOffset
|
|
||||||
});
|
|
||||||
};
|
|
||||||
export default localizer;
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import dayjs from "../../utils/day";
|
import dayjs from "../../utils/day";
|
||||||
import queryString from "query-string";
|
import queryString from "query-string";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { Calendar } from "react-big-calendar";
|
import { Calendar, dayjsLocalizer } from "react-big-calendar";
|
||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { Link, useLocation, useNavigate } from "react-router-dom";
|
import { Link, useLocation, useNavigate } from "react-router-dom";
|
||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
@@ -14,13 +14,12 @@ import { selectProblemJobs } from "../../redux/application/application.selectors
|
|||||||
import { Alert, Collapse, Space } from "antd";
|
import { Alert, Collapse, Space } from "antd";
|
||||||
import { Trans, useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
import local from "./localizer";
|
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop,
|
||||||
problemJobs: selectProblemJobs
|
problemJobs: selectProblemJobs
|
||||||
});
|
});
|
||||||
const localizer = local(dayjs);
|
const localizer = dayjsLocalizer(dayjs);
|
||||||
|
|
||||||
export function ScheduleCalendarWrapperComponent({
|
export function ScheduleCalendarWrapperComponent({
|
||||||
bodyshop,
|
bodyshop,
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ export const CalculateWorkingDaysThisMonth = () => dayjs().endOf("month").busine
|
|||||||
|
|
||||||
export const CalculateWorkingDaysInPeriod = (start, end) => dayjs(end).businessDiff(dayjs(start));
|
export const CalculateWorkingDaysInPeriod = (start, end) => dayjs(end).businessDiff(dayjs(start));
|
||||||
|
|
||||||
export const CalculateWorkingDaysAsOfToday = () => dayjs().endOf("day").businessDiff(dayjs().startOf("month"));
|
export const CalculateWorkingDaysAsOfToday = () => dayjs().businessDaysInMonth().length;
|
||||||
|
|
||||||
export const CalculateWorkingDaysLastMonth = () =>
|
export const CalculateWorkingDaysLastMonth = () =>
|
||||||
dayjs().subtract(1, "month").endOf("month").businessDaysInMonth().length;
|
dayjs().subtract(1, "month").endOf("month").businessDaysInMonth().length;
|
||||||
|
|||||||
@@ -20,7 +20,6 @@ import ShopInfoTaskPresets from "./shop-info.task-presets.component";
|
|||||||
import queryString from "query-string";
|
import queryString from "query-string";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
import ShopInfoRoGuard from "./shop-info.roguard.component";
|
import ShopInfoRoGuard from "./shop-info.roguard.component";
|
||||||
import ShopInfoIntellipay from "./shop-intellipay-config.component";
|
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop
|
bodyshop: selectBodyshop
|
||||||
@@ -136,17 +135,6 @@ export function ShopInfoComponent({ bodyshop, form, saveLoading }) {
|
|||||||
],
|
],
|
||||||
rome: "USE_IMEX",
|
rome: "USE_IMEX",
|
||||||
promanager: []
|
promanager: []
|
||||||
}),
|
|
||||||
...InstanceRenderManager({
|
|
||||||
imex: [],
|
|
||||||
rome: [
|
|
||||||
{
|
|
||||||
key: "intellipay",
|
|
||||||
label: t("bodyshop.labels.intellipay"),
|
|
||||||
children: <ShopInfoIntellipay form={form} />
|
|
||||||
}
|
|
||||||
],
|
|
||||||
promanager: []
|
|
||||||
})
|
})
|
||||||
];
|
];
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -7,13 +7,13 @@ import { connect } from "react-redux";
|
|||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
import { selectBodyshop } from "../../redux/user/user.selectors";
|
import { selectBodyshop } from "../../redux/user/user.selectors";
|
||||||
import DatePickerRanges from "../../utils/DatePickerRanges";
|
import DatePickerRanges from "../../utils/DatePickerRanges";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
|
||||||
import FeatureWrapper, { HasFeatureAccess } from "../feature-wrapper/feature-wrapper.component";
|
|
||||||
import CurrencyInput from "../form-items-formatted/currency-form-item.component";
|
import CurrencyInput from "../form-items-formatted/currency-form-item.component";
|
||||||
import FormItemEmail from "../form-items-formatted/email-form-item.component";
|
import FormItemEmail from "../form-items-formatted/email-form-item.component";
|
||||||
import PhoneFormItem, { PhoneItemFormatterValidation } from "../form-items-formatted/phone-form-item.component";
|
import PhoneFormItem, { PhoneItemFormatterValidation } from "../form-items-formatted/phone-form-item.component";
|
||||||
import FormListMoveArrows from "../form-list-move-arrows/form-list-move-arrows.component";
|
import FormListMoveArrows from "../form-list-move-arrows/form-list-move-arrows.component";
|
||||||
import LayoutFormRow from "../layout-form-row/layout-form-row.component";
|
import LayoutFormRow from "../layout-form-row/layout-form-row.component";
|
||||||
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
|
import FeatureWrapper, { HasFeatureAccess } from "../feature-wrapper/feature-wrapper.component";
|
||||||
// TODO: Client Update, this might break
|
// TODO: Client Update, this might break
|
||||||
const timeZonesList = Intl.supportedValuesOf("timeZone");
|
const timeZonesList = Intl.supportedValuesOf("timeZone");
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
@@ -28,10 +28,10 @@ export function ShopInfoGeneral({ form, bodyshop }) {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
const {
|
const {
|
||||||
treatments: { ClosingPeriod, ADPPayroll }
|
treatments: { ClosingPeriod }
|
||||||
} = useSplitTreatments({
|
} = useSplitTreatments({
|
||||||
attributes: {},
|
attributes: {},
|
||||||
names: ["ClosingPeriod", "ADPPayroll"],
|
names: ["ClosingPeriod"],
|
||||||
splitKey: bodyshop && bodyshop.imexshopid
|
splitKey: bodyshop && bodyshop.imexshopid
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -98,6 +98,7 @@ export function ShopInfoGeneral({ form, bodyshop }) {
|
|||||||
<Form.Item label={t("bodyshop.fields.email")} name="email">
|
<Form.Item label={t("bodyshop.fields.email")} name="email">
|
||||||
<Input />
|
<Input />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.phone")}
|
label={t("bodyshop.fields.phone")}
|
||||||
name="phone"
|
name="phone"
|
||||||
@@ -355,22 +356,14 @@ export function ShopInfoGeneral({ form, bodyshop }) {
|
|||||||
<Select mode="tags" />
|
<Select mode="tags" />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
{ClosingPeriod.treatment === "on" && (
|
{ClosingPeriod.treatment === "on" && (
|
||||||
<Form.Item
|
<>
|
||||||
name={["accountingconfig", "ClosingPeriod"]}
|
<Form.Item
|
||||||
label={t("bodyshop.fields.closingperiod")} //{t("reportcenter.labels.dates")}
|
name={["accountingconfig", "ClosingPeriod"]}
|
||||||
>
|
label={t("bodyshop.fields.closingperiod")} //{t("reportcenter.labels.dates")}
|
||||||
<DatePicker.RangePicker format="MM/DD/YYYY" presets={DatePickerRanges} />
|
>
|
||||||
</Form.Item>
|
<DatePicker.RangePicker format="MM/DD/YYYY" presets={DatePickerRanges} />
|
||||||
)}
|
</Form.Item>
|
||||||
{ADPPayroll.treatment === "on" && (
|
</>
|
||||||
<Form.Item name={["accountingconfig", "companyCode"]} label={t("bodyshop.fields.companycode")}>
|
|
||||||
<Input />
|
|
||||||
</Form.Item>
|
|
||||||
)}
|
|
||||||
{ADPPayroll.treatment === "on" && (
|
|
||||||
<Form.Item name={["accountingconfig", "batchID"]} label={t("bodyshop.fields.batchid")}>
|
|
||||||
<Input />
|
|
||||||
</Form.Item>
|
|
||||||
)}
|
)}
|
||||||
</LayoutFormRow>
|
</LayoutFormRow>
|
||||||
</FeatureWrapper>
|
</FeatureWrapper>
|
||||||
|
|||||||
@@ -30,226 +30,219 @@ export function ShopInfoRbacComponent({ form, bodyshop }) {
|
|||||||
return (
|
return (
|
||||||
<RbacWrapper action="shop:rbac">
|
<RbacWrapper action="shop:rbac">
|
||||||
<LayoutFormRow>
|
<LayoutFormRow>
|
||||||
{...HasFeatureAccess({ featureName: "export", bodyshop })
|
{...HasFeatureAccess({ featureName: "export", bodyshop }) ? [
|
||||||
? [
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.accounting.exportlog")}
|
||||||
label={t("bodyshop.fields.rbac.accounting.exportlog")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "accounting:exportlog"]}
|
||||||
name={["md_rbac", "accounting:exportlog"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.accounting.payables")}
|
||||||
label={t("bodyshop.fields.rbac.accounting.payables")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "accounting:payables"]}
|
||||||
name={["md_rbac", "accounting:payables"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.accounting.payments")}
|
||||||
label={t("bodyshop.fields.rbac.accounting.payments")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "accounting:payments"]}
|
||||||
name={["md_rbac", "accounting:payments"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.accounting.receivables")}
|
||||||
label={t("bodyshop.fields.rbac.accounting.receivables")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "accounting:receivables"]}
|
||||||
name={["md_rbac", "accounting:receivables"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>
|
||||||
</Form.Item>
|
]:[]}
|
||||||
]
|
{...HasFeatureAccess({ featureName: "bills", bodyshop }) ? [
|
||||||
: []}
|
<Form.Item
|
||||||
{...HasFeatureAccess({ featureName: "bills", bodyshop })
|
label={t("bodyshop.fields.rbac.bills.delete")}
|
||||||
? [
|
rules={[
|
||||||
<Form.Item
|
{
|
||||||
label={t("bodyshop.fields.rbac.bills.delete")}
|
required: true
|
||||||
rules={[
|
//message: t("general.validation.required"),
|
||||||
{
|
}
|
||||||
required: true
|
]}
|
||||||
//message: t("general.validation.required"),
|
name={["md_rbac", "bills:delete"]}
|
||||||
}
|
>
|
||||||
]}
|
<InputNumber />
|
||||||
name={["md_rbac", "bills:delete"]}
|
</Form.Item>,
|
||||||
>
|
<Form.Item
|
||||||
<InputNumber />
|
label={t("bodyshop.fields.rbac.bills.enter")}
|
||||||
</Form.Item>,
|
rules={[
|
||||||
<Form.Item
|
{
|
||||||
label={t("bodyshop.fields.rbac.bills.enter")}
|
required: true
|
||||||
rules={[
|
//message: t("general.validation.required"),
|
||||||
{
|
}
|
||||||
required: true
|
]}
|
||||||
//message: t("general.validation.required"),
|
name={["md_rbac", "bills:enter"]}
|
||||||
}
|
>
|
||||||
]}
|
<InputNumber />
|
||||||
name={["md_rbac", "bills:enter"]}
|
</Form.Item>,
|
||||||
>
|
<Form.Item
|
||||||
<InputNumber />
|
label={t("bodyshop.fields.rbac.bills.list")}
|
||||||
</Form.Item>,
|
rules={[
|
||||||
<Form.Item
|
{
|
||||||
label={t("bodyshop.fields.rbac.bills.list")}
|
required: true
|
||||||
rules={[
|
//message: t("general.validation.required"),
|
||||||
{
|
}
|
||||||
required: true
|
]}
|
||||||
//message: t("general.validation.required"),
|
name={["md_rbac", "bills:list"]}
|
||||||
}
|
>
|
||||||
]}
|
<InputNumber />
|
||||||
name={["md_rbac", "bills:list"]}
|
</Form.Item>,
|
||||||
>
|
<Form.Item
|
||||||
<InputNumber />
|
label={t("bodyshop.fields.rbac.bills.reexport")}
|
||||||
</Form.Item>,
|
rules={[
|
||||||
<Form.Item
|
{
|
||||||
label={t("bodyshop.fields.rbac.bills.reexport")}
|
required: true
|
||||||
rules={[
|
//message: t("general.validation.required"),
|
||||||
{
|
}
|
||||||
required: true
|
]}
|
||||||
//message: t("general.validation.required"),
|
name={["md_rbac", "bills:reexport"]}
|
||||||
}
|
>
|
||||||
]}
|
<InputNumber />
|
||||||
name={["md_rbac", "bills:reexport"]}
|
</Form.Item>,
|
||||||
>
|
<Form.Item
|
||||||
<InputNumber />
|
label={t("bodyshop.fields.rbac.bills.view")}
|
||||||
</Form.Item>,
|
rules={[
|
||||||
<Form.Item
|
{
|
||||||
label={t("bodyshop.fields.rbac.bills.view")}
|
required: true
|
||||||
rules={[
|
//message: t("general.validation.required"),
|
||||||
{
|
}
|
||||||
required: true
|
]}
|
||||||
//message: t("general.validation.required"),
|
name={["md_rbac", "bills:view"]}
|
||||||
}
|
>
|
||||||
]}
|
<InputNumber />
|
||||||
name={["md_rbac", "bills:view"]}
|
</Form.Item>
|
||||||
>
|
]:[]}
|
||||||
<InputNumber />
|
|
||||||
</Form.Item>
|
{...HasFeatureAccess({ featureName: "courtesycars", bodyshop }) ? [
|
||||||
]
|
<Form.Item
|
||||||
: []}
|
label={t("bodyshop.fields.rbac.contracts.create")}
|
||||||
{...HasFeatureAccess({ featureName: "courtesycars", bodyshop })
|
rules={[
|
||||||
? [
|
{
|
||||||
<Form.Item
|
required: true
|
||||||
label={t("bodyshop.fields.rbac.contracts.create")}
|
//message: t("general.validation.required"),
|
||||||
rules={[
|
}
|
||||||
{
|
]}
|
||||||
required: true
|
name={["md_rbac", "contracts:create"]}
|
||||||
//message: t("general.validation.required"),
|
>
|
||||||
}
|
<InputNumber />
|
||||||
]}
|
</Form.Item>,
|
||||||
name={["md_rbac", "contracts:create"]}
|
<Form.Item
|
||||||
>
|
label={t("bodyshop.fields.rbac.contracts.detail")}
|
||||||
<InputNumber />
|
rules={[
|
||||||
</Form.Item>,
|
{
|
||||||
<Form.Item
|
required: true
|
||||||
label={t("bodyshop.fields.rbac.contracts.detail")}
|
//message: t("general.validation.required"),
|
||||||
rules={[
|
}
|
||||||
{
|
]}
|
||||||
required: true
|
name={["md_rbac", "contracts:detail"]}
|
||||||
//message: t("general.validation.required"),
|
>
|
||||||
}
|
<InputNumber />
|
||||||
]}
|
</Form.Item>,
|
||||||
name={["md_rbac", "contracts:detail"]}
|
<Form.Item
|
||||||
>
|
label={t("bodyshop.fields.rbac.contracts.list")}
|
||||||
<InputNumber />
|
rules={[
|
||||||
</Form.Item>,
|
{
|
||||||
<Form.Item
|
required: true
|
||||||
label={t("bodyshop.fields.rbac.contracts.list")}
|
//message: t("general.validation.required"),
|
||||||
rules={[
|
}
|
||||||
{
|
]}
|
||||||
required: true
|
name={["md_rbac", "contracts:list"]}
|
||||||
//message: t("general.validation.required"),
|
>
|
||||||
}
|
<InputNumber />
|
||||||
]}
|
</Form.Item>,
|
||||||
name={["md_rbac", "contracts:list"]}
|
<Form.Item
|
||||||
>
|
label={t("bodyshop.fields.rbac.courtesycar.create")}
|
||||||
<InputNumber />
|
rules={[
|
||||||
</Form.Item>,
|
{
|
||||||
<Form.Item
|
required: true
|
||||||
label={t("bodyshop.fields.rbac.courtesycar.create")}
|
//message: t("general.validation.required"),
|
||||||
rules={[
|
}
|
||||||
{
|
]}
|
||||||
required: true
|
name={["md_rbac", "courtesycar:create"]}
|
||||||
//message: t("general.validation.required"),
|
>
|
||||||
}
|
<InputNumber />
|
||||||
]}
|
</Form.Item>,
|
||||||
name={["md_rbac", "courtesycar:create"]}
|
<Form.Item
|
||||||
>
|
label={t("bodyshop.fields.rbac.courtesycar.detail")}
|
||||||
<InputNumber />
|
rules={[
|
||||||
</Form.Item>,
|
{
|
||||||
<Form.Item
|
required: true
|
||||||
label={t("bodyshop.fields.rbac.courtesycar.detail")}
|
//message: t("general.validation.required"),
|
||||||
rules={[
|
}
|
||||||
{
|
]}
|
||||||
required: true
|
name={["md_rbac", "courtesycar:detail"]}
|
||||||
//message: t("general.validation.required"),
|
>
|
||||||
}
|
<InputNumber />
|
||||||
]}
|
</Form.Item>,
|
||||||
name={["md_rbac", "courtesycar:detail"]}
|
<Form.Item
|
||||||
>
|
label={t("bodyshop.fields.rbac.courtesycar.list")}
|
||||||
<InputNumber />
|
rules={[
|
||||||
</Form.Item>,
|
{
|
||||||
<Form.Item
|
required: true
|
||||||
label={t("bodyshop.fields.rbac.courtesycar.list")}
|
//message: t("general.validation.required"),
|
||||||
rules={[
|
}
|
||||||
{
|
]}
|
||||||
required: true
|
name={["md_rbac", "courtesycar:list"]}
|
||||||
//message: t("general.validation.required"),
|
>
|
||||||
}
|
<InputNumber />
|
||||||
]}
|
</Form.Item>
|
||||||
name={["md_rbac", "courtesycar:list"]}
|
]:[]}
|
||||||
>
|
{...HasFeatureAccess({ featureName: "csi", bodyshop }) ? [
|
||||||
<InputNumber />
|
<Form.Item
|
||||||
</Form.Item>
|
label={t("bodyshop.fields.rbac.csi.export")}
|
||||||
]
|
rules={[
|
||||||
: []}
|
{
|
||||||
{...HasFeatureAccess({ featureName: "csi", bodyshop })
|
required: true
|
||||||
? [
|
//message: t("general.validation.required"),
|
||||||
<Form.Item
|
}
|
||||||
label={t("bodyshop.fields.rbac.csi.export")}
|
]}
|
||||||
rules={[
|
name={["md_rbac", "csi:export"]}
|
||||||
{
|
>
|
||||||
required: true
|
<InputNumber />
|
||||||
//message: t("general.validation.required"),
|
</Form.Item>,
|
||||||
}
|
<Form.Item
|
||||||
]}
|
label={t("bodyshop.fields.rbac.csi.page")}
|
||||||
name={["md_rbac", "csi:export"]}
|
rules={[
|
||||||
>
|
{
|
||||||
<InputNumber />
|
required: true
|
||||||
</Form.Item>,
|
//message: t("general.validation.required"),
|
||||||
<Form.Item
|
}
|
||||||
label={t("bodyshop.fields.rbac.csi.page")}
|
]}
|
||||||
rules={[
|
name={["md_rbac", "csi:page"]}
|
||||||
{
|
>
|
||||||
required: true
|
<InputNumber />
|
||||||
//message: t("general.validation.required"),
|
</Form.Item>
|
||||||
}
|
]:[]}
|
||||||
]}
|
|
||||||
name={["md_rbac", "csi:page"]}
|
|
||||||
>
|
|
||||||
<InputNumber />
|
|
||||||
</Form.Item>
|
|
||||||
]
|
|
||||||
: []}
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.rbac.employees.page")}
|
label={t("bodyshop.fields.rbac.employees.page")}
|
||||||
rules={[
|
rules={[
|
||||||
@@ -262,18 +255,6 @@ export function ShopInfoRbacComponent({ form, bodyshop }) {
|
|||||||
>
|
>
|
||||||
<InputNumber />
|
<InputNumber />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
<Form.Item
|
|
||||||
label={t("bodyshop.fields.rbac.employee_teams.page")}
|
|
||||||
rules={[
|
|
||||||
{
|
|
||||||
required: true
|
|
||||||
//message: t("general.validation.required"),
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
name={["md_rbac", "employee_teams:page"]}
|
|
||||||
>
|
|
||||||
<InputNumber />
|
|
||||||
</Form.Item>
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.rbac.jobs.admin")}
|
label={t("bodyshop.fields.rbac.jobs.admin")}
|
||||||
rules={[
|
rules={[
|
||||||
@@ -454,6 +435,31 @@ export function ShopInfoRbacComponent({ form, bodyshop }) {
|
|||||||
>
|
>
|
||||||
<InputNumber />
|
<InputNumber />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.rbac.employees.page")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
}
|
||||||
|
]}
|
||||||
|
name={["md_rbac", "employees:page"]}
|
||||||
|
>
|
||||||
|
<InputNumber />
|
||||||
|
</Form.Item>
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.rbac.employee_teams.page")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
}
|
||||||
|
]}
|
||||||
|
name={["md_rbac", "employee_teams:page"]}
|
||||||
|
>
|
||||||
|
<InputNumber />
|
||||||
|
</Form.Item>
|
||||||
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.rbac.payments.enter")}
|
label={t("bodyshop.fields.rbac.payments.enter")}
|
||||||
rules={[
|
rules={[
|
||||||
@@ -516,6 +522,7 @@ export function ShopInfoRbacComponent({ form, bodyshop }) {
|
|||||||
<InputNumber />
|
<InputNumber />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.rbac.production.list")}
|
label={t("bodyshop.fields.rbac.production.list")}
|
||||||
rules={[
|
rules={[
|
||||||
@@ -554,118 +561,128 @@ export function ShopInfoRbacComponent({ form, bodyshop }) {
|
|||||||
<InputNumber />
|
<InputNumber />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
)}
|
)}
|
||||||
{...HasFeatureAccess({ featureName: "timetickets", bodyshop })
|
{...HasFeatureAccess({ featureName: "timetickets", bodyshop }) ? [
|
||||||
? [
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.shiftclock.view")}
|
||||||
label={t("bodyshop.fields.rbac.shiftclock.view")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "shiftclock:view"]}
|
||||||
name={["md_rbac", "shiftclock:view"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.shop.config")}
|
||||||
label={t("bodyshop.fields.rbac.shop.config")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "shop:config"]}
|
||||||
name={["md_rbac", "shop:config"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.timetickets.edit")}
|
||||||
label={t("bodyshop.fields.rbac.timetickets.edit")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "timetickets:edit"]}
|
||||||
name={["md_rbac", "timetickets:edit"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.timetickets.shiftedit")}
|
||||||
label={t("bodyshop.fields.rbac.timetickets.shiftedit")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "timetickets:shiftedit"]}
|
||||||
name={["md_rbac", "timetickets:shiftedit"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.timetickets.editcommitted")}
|
||||||
label={t("bodyshop.fields.rbac.timetickets.editcommitted")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "timetickets:editcommitted"]}
|
||||||
name={["md_rbac", "timetickets:editcommitted"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.ttapprovals.view")}
|
||||||
label={t("bodyshop.fields.rbac.ttapprovals.view")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "ttapprovals:view"]}
|
||||||
name={["md_rbac", "ttapprovals:view"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.ttapprovals.approve")}
|
||||||
label={t("bodyshop.fields.rbac.ttapprovals.approve")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "ttapprovals:approve"]}
|
||||||
name={["md_rbac", "ttapprovals:approve"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.timetickets.enter")}
|
||||||
label={t("bodyshop.fields.rbac.timetickets.enter")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "timetickets:enter"]}
|
||||||
name={["md_rbac", "timetickets:enter"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>,
|
<Form.Item
|
||||||
<Form.Item
|
label={t("bodyshop.fields.rbac.timetickets.list")}
|
||||||
label={t("bodyshop.fields.rbac.timetickets.list")}
|
rules={[
|
||||||
rules={[
|
{
|
||||||
{
|
required: true
|
||||||
required: true
|
//message: t("general.validation.required"),
|
||||||
//message: t("general.validation.required"),
|
}
|
||||||
}
|
]}
|
||||||
]}
|
name={["md_rbac", "timetickets:list"]}
|
||||||
name={["md_rbac", "timetickets:list"]}
|
>
|
||||||
>
|
<InputNumber />
|
||||||
<InputNumber />
|
</Form.Item>,
|
||||||
</Form.Item>
|
<Form.Item
|
||||||
]
|
label={t("bodyshop.fields.rbac.timetickets.shiftedit")}
|
||||||
: []}
|
rules={[
|
||||||
|
{
|
||||||
|
required: true
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
}
|
||||||
|
]}
|
||||||
|
name={["md_rbac", "timetickets:shiftedit"]}
|
||||||
|
>
|
||||||
|
<InputNumber />
|
||||||
|
</Form.Item>
|
||||||
|
]:[]}
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.rbac.shop.vendors")}
|
label={t("bodyshop.fields.rbac.shop.vendors")}
|
||||||
rules={[
|
rules={[
|
||||||
@@ -740,6 +757,7 @@ export function ShopInfoRbacComponent({ form, bodyshop }) {
|
|||||||
<InputNumber />
|
<InputNumber />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.rbac.users.editaccess")}
|
label={t("bodyshop.fields.rbac.users.editaccess")}
|
||||||
rules={[
|
rules={[
|
||||||
|
|||||||
@@ -1,54 +0,0 @@
|
|||||||
import { Alert, Form, InputNumber, Switch } from "antd";
|
|
||||||
import React from "react";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import LayoutFormRow from "../layout-form-row/layout-form-row.component";
|
|
||||||
|
|
||||||
import { connect } from "react-redux";
|
|
||||||
import { createStructuredSelector } from "reselect";
|
|
||||||
import { selectBodyshop } from "../../redux/user/user.selectors";
|
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
|
||||||
bodyshop: selectBodyshop
|
|
||||||
});
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
|
||||||
//setUserLanguage: language => dispatch(setUserLanguage(language))
|
|
||||||
});
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(ShopInfoIntellipay);
|
|
||||||
|
|
||||||
export function ShopInfoIntellipay({ bodyshop, form }) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Form.Item dependencies={[["intellipay_config", "enable_cash_discount"]]}>
|
|
||||||
{() => {
|
|
||||||
const { intellipay_config } = form.getFieldsValue();
|
|
||||||
|
|
||||||
if (intellipay_config?.enable_cash_discount)
|
|
||||||
return <Alert message={t("bodyshop.labels.intellipay_cash_discount")} />;
|
|
||||||
}}
|
|
||||||
</Form.Item>
|
|
||||||
|
|
||||||
<LayoutFormRow noDivider>
|
|
||||||
<Form.Item
|
|
||||||
label={t("bodyshop.fields.intellipay_config.enable_cash_discount")}
|
|
||||||
valuePropName="checked"
|
|
||||||
name={["intellipay_config", "enable_cash_discount"]}
|
|
||||||
>
|
|
||||||
<Switch />
|
|
||||||
</Form.Item>
|
|
||||||
<Form.Item
|
|
||||||
label={t("bodyshop.fields.intellipay_config.cash_discount_percentage")}
|
|
||||||
valuePropName="checked"
|
|
||||||
dependencies={[["intellipay_config", "enable_cash_discount"]]}
|
|
||||||
name={["intellipay_config", "cash_discount_percentage"]}
|
|
||||||
rules={[
|
|
||||||
({ getFieldsValue }) => ({ required: form.getFieldValue(["intellipay_config", "enable_cash_discount"]) })
|
|
||||||
]}
|
|
||||||
>
|
|
||||||
<InputNumber min={0} max={100} precision={1} suffix='%'/>
|
|
||||||
</Form.Item>
|
|
||||||
</LayoutFormRow>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -12,7 +12,7 @@ import { DateFormatter, DateTimeFormatter } from "../../utils/DateFormatter";
|
|||||||
import { onlyUnique } from "../../utils/arrayHelper";
|
import { onlyUnique } from "../../utils/arrayHelper";
|
||||||
import dayjs from "../../utils/day";
|
import dayjs from "../../utils/day";
|
||||||
import { alphaSort, dateSort } from "../../utils/sorters";
|
import { alphaSort, dateSort } from "../../utils/sorters";
|
||||||
import { HasRbacAccess } from "../rbac-wrapper/rbac-wrapper.component";
|
import RbacWrapper, { HasRbacAccess } from "../rbac-wrapper/rbac-wrapper.component";
|
||||||
import TimeTicketEnterButton from "../time-ticket-enter-button/time-ticket-enter-button.component";
|
import TimeTicketEnterButton from "../time-ticket-enter-button/time-ticket-enter-button.component";
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
@@ -52,10 +52,6 @@ export function TimeTicketList({
|
|||||||
splitKey: bodyshop.imexshopid
|
splitKey: bodyshop.imexshopid
|
||||||
});
|
});
|
||||||
|
|
||||||
const canEditCommittedTimeTickets = HasRbacAccess({ bodyshop, authLevel, action: "timetickets:editcommitted" });
|
|
||||||
const canEditTimeTickets = HasRbacAccess({ bodyshop, authLevel, action: "timetickets:edit" });
|
|
||||||
const canEditShiftTickets = HasRbacAccess({ bodyshop, authLevel, action: "timetickets:shiftedit" });
|
|
||||||
|
|
||||||
const totals = useMemo(() => {
|
const totals = useMemo(() => {
|
||||||
if (timetickets)
|
if (timetickets)
|
||||||
return timetickets.reduce(
|
return timetickets.reduce(
|
||||||
@@ -69,18 +65,6 @@ export function TimeTicketList({
|
|||||||
return { productivehrs: 0, actualhrs: 0 };
|
return { productivehrs: 0, actualhrs: 0 };
|
||||||
}, [timetickets]);
|
}, [timetickets]);
|
||||||
|
|
||||||
const isDisabled = (record) => {
|
|
||||||
if (disabled === true || !record.id) return true;
|
|
||||||
|
|
||||||
const isShiftTicket = !record.ciecacode;
|
|
||||||
const isCommitted = record.committed_at;
|
|
||||||
if (isShiftTicket) {
|
|
||||||
return !(canEditShiftTickets && (!isCommitted || canEditCommittedTimeTickets));
|
|
||||||
}
|
|
||||||
|
|
||||||
return !(canEditTimeTickets && (!isCommitted || canEditCommittedTimeTickets));
|
|
||||||
};
|
|
||||||
|
|
||||||
const columns = [
|
const columns = [
|
||||||
...(Enhanced_Payroll.treatment === "on"
|
...(Enhanced_Payroll.treatment === "on"
|
||||||
? [
|
? [
|
||||||
@@ -257,16 +241,59 @@ export function TimeTicketList({
|
|||||||
</TimeTicketEnterButton>
|
</TimeTicketEnterButton>
|
||||||
)}
|
)}
|
||||||
{!techConsole && (
|
{!techConsole && (
|
||||||
<TimeTicketEnterButton
|
<RbacWrapper
|
||||||
actions={{ refetch }}
|
action="timetickets:edit"
|
||||||
context={{
|
noauth={() => {
|
||||||
id: record.id,
|
return <div />;
|
||||||
timeticket: record
|
|
||||||
}}
|
}}
|
||||||
disabled={isDisabled(record)}
|
|
||||||
>
|
>
|
||||||
<EditFilled />
|
<TimeTicketEnterButton
|
||||||
</TimeTicketEnterButton>
|
actions={{ refetch }}
|
||||||
|
context={{
|
||||||
|
id: record.id,
|
||||||
|
timeticket: record
|
||||||
|
}}
|
||||||
|
disabled={
|
||||||
|
record.ciecacode
|
||||||
|
? record.committed_at
|
||||||
|
? HasRbacAccess({
|
||||||
|
bodyshop,
|
||||||
|
authLevel: authLevel,
|
||||||
|
action: "timetickets:editcommitted"
|
||||||
|
}) &&
|
||||||
|
HasRbacAccess({
|
||||||
|
bodyshop,
|
||||||
|
authLevel: authLevel,
|
||||||
|
action: "timetickets:edit"
|
||||||
|
})
|
||||||
|
: HasRbacAccess({
|
||||||
|
bodyshop,
|
||||||
|
authLevel: authLevel,
|
||||||
|
action: "timetickets:edit"
|
||||||
|
})
|
||||||
|
: record.committed_at
|
||||||
|
? HasRbacAccess({
|
||||||
|
bodyshop,
|
||||||
|
authLevel: authLevel,
|
||||||
|
action: "timetickets:editcommitted"
|
||||||
|
}) &&
|
||||||
|
HasRbacAccess({
|
||||||
|
bodyshop,
|
||||||
|
authLevel: authLevel,
|
||||||
|
action: "timetickets:shiftedit"
|
||||||
|
})
|
||||||
|
: HasRbacAccess({
|
||||||
|
bodyshop,
|
||||||
|
authLevel: authLevel,
|
||||||
|
action: "timetickets:shiftedit"
|
||||||
|
})
|
||||||
|
? disabled
|
||||||
|
: !record.jobid
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<EditFilled />
|
||||||
|
</TimeTicketEnterButton>
|
||||||
|
</RbacWrapper>
|
||||||
)}
|
)}
|
||||||
</Space>
|
</Space>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -329,9 +329,7 @@ export function LaborAllocationContainer({ jobid, loading, lineTicketData, hideT
|
|||||||
timetickets={lineTicketData.timetickets}
|
timetickets={lineTicketData.timetickets}
|
||||||
adjustments={lineTicketData.jobs_by_pk.lbr_adjustments}
|
adjustments={lineTicketData.jobs_by_pk.lbr_adjustments}
|
||||||
/>
|
/>
|
||||||
{!hideTimeTickets && (
|
{!hideTimeTickets && <TimeTicketList loading={loading} timetickets={lineTicketData.timetickets} techConsole />}
|
||||||
<TimeTicketList loading={loading} timetickets={jobid ? lineTicketData.timetickets : []} techConsole />
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,13 @@
|
|||||||
import { AlertOutlined } from "@ant-design/icons";
|
import { AlertOutlined } from "@ant-design/icons";
|
||||||
import { Alert, Button, Col, notification, Row, Space } from "antd";
|
import { Alert, Button, Col, Row, Space } from "antd";
|
||||||
import i18n from "i18next";
|
import i18n from "i18next";
|
||||||
import React, { useCallback, useEffect, useState } from "react";
|
import React, { useEffect } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
import { selectUpdateAvailable } from "../../redux/application/application.selectors";
|
import { selectUpdateAvailable } from "../../redux/application/application.selectors";
|
||||||
import { useRegisterSW } from "virtual:pwa-register/react";
|
import { useRegisterSW } from "virtual:pwa-register/react";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
import useCountDown from "../../utils/countdownHook";
|
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
updateAvailable: selectUpdateAvailable
|
updateAvailable: selectUpdateAvailable
|
||||||
@@ -20,15 +19,6 @@ const mapDispatchToProps = (dispatch) => ({
|
|||||||
|
|
||||||
export function UpdateAlert({ updateAvailable }) {
|
export function UpdateAlert({ updateAvailable }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [timerStarted, setTimerStarted] = useState(false);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [
|
|
||||||
timeLeft,
|
|
||||||
{
|
|
||||||
start //pause, resume, reset
|
|
||||||
}
|
|
||||||
] = useCountDown(180000, 1000);
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
offlineReady: [offlineReady],
|
offlineReady: [offlineReady],
|
||||||
needRefresh: [needRefresh],
|
needRefresh: [needRefresh],
|
||||||
@@ -41,7 +31,7 @@ export function UpdateAlert({ updateAvailable }) {
|
|||||||
() => {
|
() => {
|
||||||
r.update();
|
r.update();
|
||||||
},
|
},
|
||||||
30 * 60 * 1000
|
10 * 60 * 1000
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -50,43 +40,11 @@ export function UpdateAlert({ updateAvailable }) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const ReloadNewVersion = useCallback(() => {
|
|
||||||
setLoading(true);
|
|
||||||
updateServiceWorker(true);
|
|
||||||
setTimeout(() => {
|
|
||||||
window.location.reload(true);
|
|
||||||
}, 5000);
|
|
||||||
}, [updateServiceWorker]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (needRefresh) {
|
if (import.meta.env.DEV) {
|
||||||
start();
|
console.log(`SW Status => Refresh? ${needRefresh} - offlineReady? ${offlineReady}`);
|
||||||
setTimerStarted(true);
|
|
||||||
}
|
}
|
||||||
}, [start, needRefresh, offlineReady]);
|
}, [needRefresh, offlineReady]);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (needRefresh && timerStarted && timeLeft < 60000) {
|
|
||||||
notification.open({
|
|
||||||
type: "warning",
|
|
||||||
closable: false,
|
|
||||||
duration: 65000,
|
|
||||||
key: "autoupdate",
|
|
||||||
message: t("general.actions.autoupdate", {
|
|
||||||
time: (timeLeft / 1000).toFixed(0),
|
|
||||||
app: InstanceRenderManager({
|
|
||||||
imex: "$t(titles.imexonline)",
|
|
||||||
rome: "$t(titles.romeonline)",
|
|
||||||
promanager: "$t(titles.promanager)"
|
|
||||||
})
|
|
||||||
}),
|
|
||||||
placement: "bottomRight"
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (needRefresh && timerStarted && timeLeft <= 0) {
|
|
||||||
ReloadNewVersion();
|
|
||||||
}
|
|
||||||
}, [timeLeft, t, needRefresh, ReloadNewVersion, timerStarted]);
|
|
||||||
|
|
||||||
if (!needRefresh) return null;
|
if (!needRefresh) return null;
|
||||||
|
|
||||||
@@ -117,10 +75,9 @@ export function UpdateAlert({ updateAvailable }) {
|
|||||||
<Button onClick={() => window.open("https://imex-online.noticeable.news/", "_blank")}>
|
<Button onClick={() => window.open("https://imex-online.noticeable.news/", "_blank")}>
|
||||||
{i18n.t("general.actions.viewreleasenotes")}
|
{i18n.t("general.actions.viewreleasenotes")}
|
||||||
</Button>
|
</Button>
|
||||||
<Button loading={loading} type="primary" onClick={() => ReloadNewVersion()}>
|
<Button type="primary" onClick={() => updateServiceWorker(true)}>
|
||||||
{i18n.t("general.actions.refresh")} {`(${(timeLeft / 1000).toFixed(0)} s)`}
|
{i18n.t("general.actions.refresh")}
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={() => start(300000)}>{i18n.t("general.actions.delay")}</Button>
|
|
||||||
</Space>
|
</Space>
|
||||||
</Col>
|
</Col>
|
||||||
</Row>
|
</Row>
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import PhoneNumberFormatter from "../../utils/PhoneFormatter";
|
|||||||
|
|
||||||
const { Option } = Select;
|
const { Option } = Select;
|
||||||
|
|
||||||
// To be used as a form element only.
|
//To be used as a form element only.
|
||||||
|
|
||||||
const VendorSearchSelect = ({ value, onChange, options, onSelect, disabled, preferredMake, showPhone }, ref) => {
|
const VendorSearchSelect = ({ value, onChange, options, onSelect, disabled, preferredMake, showPhone }, ref) => {
|
||||||
const [option, setOption] = useState(value);
|
const [option, setOption] = useState(value);
|
||||||
@@ -33,25 +33,9 @@ const VendorSearchSelect = ({ value, onChange, options, onSelect, disabled, pref
|
|||||||
if (!value || !options) return label;
|
if (!value || !options) return label;
|
||||||
const discount = options?.find((o) => o.id === value)?.discount;
|
const discount = options?.find((o) => o.id === value)?.discount;
|
||||||
return (
|
return (
|
||||||
<div
|
<div className="imex-flex-row" style={{ width: "100%" }}>
|
||||||
style={{
|
<div style={{ flex: 1 }}>{label}</div>
|
||||||
display: "flex",
|
|
||||||
alignItems: "center",
|
|
||||||
flexWrap: "nowrap",
|
|
||||||
width: "100%"
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
flex: 1,
|
|
||||||
minWidth: 0,
|
|
||||||
overflow: "hidden",
|
|
||||||
textOverflow: "ellipsis",
|
|
||||||
whiteSpace: "nowrap"
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{label}
|
|
||||||
</div>
|
|
||||||
{discount && discount !== 0 ? <Tag color="green">{`${discount * 100}%`}</Tag> : null}
|
{discount && discount !== 0 ? <Tag color="green">{`${discount * 100}%`}</Tag> : null}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -61,67 +45,36 @@ const VendorSearchSelect = ({ value, onChange, options, onSelect, disabled, pref
|
|||||||
optionFilterProp="name"
|
optionFilterProp="name"
|
||||||
onSelect={onSelect}
|
onSelect={onSelect}
|
||||||
disabled={disabled || false}
|
disabled={disabled || false}
|
||||||
optionLabelProp="name"
|
optionLabelProp={"name"}
|
||||||
>
|
>
|
||||||
{favorites &&
|
{favorites
|
||||||
favorites.map((o) => (
|
? favorites.map((o) => (
|
||||||
<Option key={`favorite-${o.id}`} value={o.id} name={o.name} discount={o.discount}>
|
<Option key={`favorite-${o.id}`} value={o.id} name={o.name} discount={o.discount}>
|
||||||
<div
|
<div className="imex-flex-row">
|
||||||
style={{
|
<div style={{ flex: 1 }}>{o.name}</div>
|
||||||
display: "flex",
|
<Space style={{ marginLeft: "1rem" }}>
|
||||||
alignItems: "center",
|
<HeartOutlined style={{ color: "red" }} />
|
||||||
flexWrap: "nowrap",
|
{o.phone && showPhone && <PhoneNumberFormatter>{o.phone}</PhoneNumberFormatter>}
|
||||||
width: "100%"
|
{o.discount && o.discount !== 0 ? <Tag color="green">{`${o.discount * 100}%`}</Tag> : null}
|
||||||
}}
|
</Space>
|
||||||
>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
flex: 1,
|
|
||||||
minWidth: 0,
|
|
||||||
overflow: "hidden",
|
|
||||||
textOverflow: "ellipsis",
|
|
||||||
whiteSpace: "nowrap"
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{o.name}
|
|
||||||
</div>
|
</div>
|
||||||
<Space style={{ marginLeft: "1rem" }}>
|
</Option>
|
||||||
<HeartOutlined style={{ color: "red" }} />
|
))
|
||||||
{o.phone && showPhone && <PhoneNumberFormatter>{o.phone}</PhoneNumberFormatter>}
|
: null}
|
||||||
{o.discount && o.discount !== 0 ? <Tag color="green">{`${o.discount * 100}%`}</Tag> : null}
|
{options
|
||||||
</Space>
|
? options.map((o) => (
|
||||||
</div>
|
<Option key={o.id} value={o.id} name={o.name} discount={o.discount}>
|
||||||
</Option>
|
<div className="imex-flex-row" style={{ width: "100%" }}>
|
||||||
))}
|
<div style={{ flex: 1 }}>{o.name}</div>
|
||||||
{options &&
|
|
||||||
options.map((o) => (
|
<Space style={{ marginLeft: "1rem" }}>
|
||||||
<Option key={o.id} value={o.id} name={o.name} discount={o.discount}>
|
{o.phone && showPhone && <PhoneNumberFormatter>{o.phone}</PhoneNumberFormatter>}
|
||||||
<div
|
{o.discount && o.discount !== 0 ? <Tag color="green">{`${o.discount * 100}%`}</Tag> : null}
|
||||||
style={{
|
</Space>
|
||||||
display: "flex",
|
|
||||||
alignItems: "center",
|
|
||||||
flexWrap: "nowrap",
|
|
||||||
width: "100%"
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
flex: 1,
|
|
||||||
minWidth: 0,
|
|
||||||
overflow: "hidden",
|
|
||||||
textOverflow: "ellipsis",
|
|
||||||
whiteSpace: "nowrap"
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{o.name}
|
|
||||||
</div>
|
</div>
|
||||||
<Space style={{ marginLeft: "1rem" }}>
|
</Option>
|
||||||
{o.phone && showPhone && <PhoneNumberFormatter>{o.phone}</PhoneNumberFormatter>}
|
))
|
||||||
{o.discount && o.discount !== 0 ? <Tag color="green">{`${o.discount * 100}%`}</Tag> : null}
|
: null}
|
||||||
</Space>
|
|
||||||
</div>
|
|
||||||
</Option>
|
|
||||||
))}
|
|
||||||
</Select>
|
</Select>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,13 +0,0 @@
|
|||||||
import React, { createContext } from "react";
|
|
||||||
import useSocket from "./useSocket"; // Import the custom hook
|
|
||||||
|
|
||||||
// Create the SocketContext
|
|
||||||
const SocketContext = createContext(null);
|
|
||||||
|
|
||||||
export const SocketProvider = ({ children, bodyshop }) => {
|
|
||||||
const { socket, clientId } = useSocket(bodyshop);
|
|
||||||
|
|
||||||
return <SocketContext.Provider value={{ socket, clientId }}> {children}</SocketContext.Provider>;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default SocketContext;
|
|
||||||
@@ -1,83 +0,0 @@
|
|||||||
import { useEffect, useState } from "react";
|
|
||||||
import SocketIO from "socket.io-client";
|
|
||||||
import { auth } from "../../firebase/firebase.utils";
|
|
||||||
|
|
||||||
const useSocket = (bodyshop) => {
|
|
||||||
const [socket, setSocket] = useState(null);
|
|
||||||
const [clientId, setClientId] = useState(null);
|
|
||||||
const [token, setToken] = useState(null);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
// Listener for token changes
|
|
||||||
const unsubscribe = auth.onIdTokenChanged(async (user) => {
|
|
||||||
if (user) {
|
|
||||||
const newToken = await user.getIdToken();
|
|
||||||
setToken(newToken);
|
|
||||||
} else {
|
|
||||||
setToken(null);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Clean up the listener on unmount
|
|
||||||
return () => unsubscribe();
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (bodyshop && bodyshop.id && token) {
|
|
||||||
const endpoint = import.meta.env.PROD ? import.meta.env.VITE_APP_AXIOS_BASE_API_URL : "";
|
|
||||||
|
|
||||||
const socketInstance = SocketIO(endpoint, {
|
|
||||||
path: "/wss",
|
|
||||||
withCredentials: true,
|
|
||||||
auth: { token }, // Use the current token
|
|
||||||
reconnectionAttempts: Infinity,
|
|
||||||
reconnectionDelay: 2000,
|
|
||||||
reconnectionDelayMax: 10000
|
|
||||||
});
|
|
||||||
|
|
||||||
setSocket(socketInstance);
|
|
||||||
|
|
||||||
const handleBodyshopMessage = (message) => {
|
|
||||||
console.log(`Received message for bodyshop ${bodyshop.id}:`, message);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleConnect = () => {
|
|
||||||
console.log("Socket connected:", socketInstance.id);
|
|
||||||
socketInstance.emit("join-bodyshop-room", bodyshop.id);
|
|
||||||
setClientId(socketInstance.id);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleReconnect = (attempt) => {
|
|
||||||
console.log(`Socket reconnected after ${attempt} attempts`);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleConnectionError = (err) => {
|
|
||||||
console.error("Socket connection error:", err);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDisconnect = () => {
|
|
||||||
console.log("Socket disconnected");
|
|
||||||
};
|
|
||||||
|
|
||||||
socketInstance.on("connect", handleConnect);
|
|
||||||
socketInstance.on("reconnect", handleReconnect);
|
|
||||||
socketInstance.on("connect_error", handleConnectionError);
|
|
||||||
socketInstance.on("disconnect", handleDisconnect);
|
|
||||||
socketInstance.on("bodyshop-message", handleBodyshopMessage);
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
socketInstance.emit("leave-bodyshop-room", bodyshop.id);
|
|
||||||
socketInstance.off("connect", handleConnect);
|
|
||||||
socketInstance.off("reconnect", handleReconnect);
|
|
||||||
socketInstance.off("connect_error", handleConnectionError);
|
|
||||||
socketInstance.off("disconnect", handleDisconnect);
|
|
||||||
socketInstance.off("bodyshop-message", handleBodyshopMessage);
|
|
||||||
socketInstance.disconnect();
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}, [bodyshop, token]); // Include 'token' in dependencies to re-run effect when it changes
|
|
||||||
|
|
||||||
return { socket, clientId };
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useSocket;
|
|
||||||
@@ -5,6 +5,7 @@ import { getFirestore } from "firebase/firestore";
|
|||||||
import { getMessaging, getToken, onMessage } from "firebase/messaging";
|
import { getMessaging, getToken, onMessage } from "firebase/messaging";
|
||||||
import { store } from "../redux/store";
|
import { store } from "../redux/store";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
|
import { checkBeta } from "../utils/handleBeta";
|
||||||
|
|
||||||
const config = JSON.parse(import.meta.env.VITE_APP_FIREBASE_CONFIG);
|
const config = JSON.parse(import.meta.env.VITE_APP_FIREBASE_CONFIG);
|
||||||
initializeApp(config);
|
initializeApp(config);
|
||||||
@@ -87,7 +88,7 @@ export const logImEXEvent = (eventName, additionalParams, stateProp = null) => {
|
|||||||
operationName: eventName,
|
operationName: eventName,
|
||||||
variables: additionalParams,
|
variables: additionalParams,
|
||||||
dbevent: false,
|
dbevent: false,
|
||||||
env: `master-AIO|${import.meta.env.VITE_APP_GIT_SHA_DATE}`
|
env: checkBeta() ? "beta" : "master"
|
||||||
});
|
});
|
||||||
// console.log(
|
// console.log(
|
||||||
// "%c[Analytics]",
|
// "%c[Analytics]",
|
||||||
|
|||||||
@@ -138,8 +138,7 @@ export const QUERY_BODYSHOP = gql`
|
|||||||
tt_enforce_hours_for_tech_console
|
tt_enforce_hours_for_tech_console
|
||||||
md_tasks_presets
|
md_tasks_presets
|
||||||
use_paint_scale_data
|
use_paint_scale_data
|
||||||
intellipay_config
|
md_ro_guard
|
||||||
md_ro_guard
|
|
||||||
employee_teams(order_by: { name: asc }, where: { active: { _eq: true } }) {
|
employee_teams(order_by: { name: asc }, where: { active: { _eq: true } }) {
|
||||||
id
|
id
|
||||||
name
|
name
|
||||||
@@ -267,8 +266,7 @@ export const UPDATE_SHOP = gql`
|
|||||||
enforce_conversion_category
|
enforce_conversion_category
|
||||||
tt_enforce_hours_for_tech_console
|
tt_enforce_hours_for_tech_console
|
||||||
md_tasks_presets
|
md_tasks_presets
|
||||||
intellipay_config
|
md_ro_guard
|
||||||
md_ro_guard
|
|
||||||
employee_teams(order_by: { name: asc }, where: { active: { _eq: true } }) {
|
employee_teams(order_by: { name: asc }, where: { active: { _eq: true } }) {
|
||||||
id
|
id
|
||||||
name
|
name
|
||||||
|
|||||||
@@ -2461,14 +2461,6 @@ export const SUBSCRIPTION_JOBS_IN_PRODUCTION = gql`
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
export const SUBSCRIPTION_JOBS_IN_PRODUCTION_VIEW = gql`
|
|
||||||
subscription SUBSCRIPTION_JOBS_IN_PRODUCTION_VIEW {
|
|
||||||
jobs: jobs_inproduction {
|
|
||||||
id
|
|
||||||
updated_at
|
|
||||||
}
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
export const QUERY_JOBS_IN_PRODUCTION = gql`
|
export const QUERY_JOBS_IN_PRODUCTION = gql`
|
||||||
query QUERY_JOBS_IN_PRODUCTION {
|
query QUERY_JOBS_IN_PRODUCTION {
|
||||||
|
|||||||
@@ -23,14 +23,17 @@ const mapDispatchToProps = (dispatch) => ({
|
|||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(DmsContainer);
|
export default connect(mapStateToProps, mapDispatchToProps)(DmsContainer);
|
||||||
|
|
||||||
export const socket = SocketIO(import.meta.env.PROD ? import.meta.env.VITE_APP_AXIOS_BASE_API_URL : "", {
|
export const socket = SocketIO(
|
||||||
path: "/ws",
|
import.meta.env.PROD ? import.meta.env.VITE_APP_AXIOS_BASE_API_URL : window.location.origin,
|
||||||
withCredentials: true,
|
{
|
||||||
auth: async (callback) => {
|
path: "/ws",
|
||||||
const token = auth.currentUser && (await auth.currentUser.getIdToken());
|
withCredentials: true,
|
||||||
callback({ token });
|
auth: async (callback) => {
|
||||||
|
const token = auth.currentUser && (await auth.currentUser.getIdToken());
|
||||||
|
callback({ token });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
);
|
||||||
|
|
||||||
export function DmsContainer({ bodyshop, setBreadcrumbs, setSelectedHeader }) {
|
export function DmsContainer({ bodyshop, setBreadcrumbs, setSelectedHeader }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ const mapDispatchToProps = (dispatch) => ({
|
|||||||
export default connect(mapStateToProps, mapDispatchToProps)(DmsContainer);
|
export default connect(mapStateToProps, mapDispatchToProps)(DmsContainer);
|
||||||
|
|
||||||
export const socket = SocketIO(
|
export const socket = SocketIO(
|
||||||
import.meta.env.PROD ? import.meta.env.VITE_APP_AXIOS_BASE_API_URL : "", // for dev testing,
|
import.meta.env.PROD ? import.meta.env.VITE_APP_AXIOS_BASE_API_URL : "http://localhost:4000", // for dev testing,
|
||||||
{
|
{
|
||||||
path: "/ws",
|
path: "/ws",
|
||||||
withCredentials: true,
|
withCredentials: true,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { FloatButton, Layout, Spin } from "antd";
|
import { FloatButton, Layout, Spin } from "antd";
|
||||||
// import preval from "preval.macro";
|
// import preval from "preval.macro";
|
||||||
import React, { lazy, Suspense, useContext, useEffect, useState } from "react";
|
import React, { lazy, Suspense, useEffect, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { Link, Route, Routes } from "react-router-dom";
|
import { Link, Route, Routes } from "react-router-dom";
|
||||||
@@ -19,7 +19,6 @@ import PartnerPingComponent from "../../components/partner-ping/partner-ping.com
|
|||||||
import PrintCenterModalContainer from "../../components/print-center-modal/print-center-modal.container";
|
import PrintCenterModalContainer from "../../components/print-center-modal/print-center-modal.container";
|
||||||
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component";
|
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component";
|
||||||
import { requestForToken } from "../../firebase/firebase.utils";
|
import { requestForToken } from "../../firebase/firebase.utils";
|
||||||
import SocketContext from "../../contexts/SocketIO/socketContext.jsx";
|
|
||||||
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
||||||
|
|
||||||
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
||||||
@@ -111,7 +110,6 @@ const mapDispatchToProps = (dispatch) => ({});
|
|||||||
export function Manage({ conflict, bodyshop }) {
|
export function Manage({ conflict, bodyshop }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [chatVisible] = useState(false);
|
const [chatVisible] = useState(false);
|
||||||
const { socket, clientId } = useContext(SocketContext);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const widgetId = InstanceRenderManager({
|
const widgetId = InstanceRenderManager({
|
||||||
@@ -131,7 +129,6 @@ export function Manage({ conflict, bodyshop }) {
|
|||||||
promanager: t("titles.promanager")
|
promanager: t("titles.promanager")
|
||||||
});
|
});
|
||||||
}, [t]);
|
}, [t]);
|
||||||
|
|
||||||
const AppRouteTable = (
|
const AppRouteTable = (
|
||||||
<Suspense
|
<Suspense
|
||||||
fallback={
|
fallback={
|
||||||
@@ -572,13 +569,6 @@ export function Manage({ conflict, bodyshop }) {
|
|||||||
else if (bodyshop && bodyshop.sub_status !== "active") PageContent = <ShopSubStatusComponent />;
|
else if (bodyshop && bodyshop.sub_status !== "active") PageContent = <ShopSubStatusComponent />;
|
||||||
else PageContent = AppRouteTable;
|
else PageContent = AppRouteTable;
|
||||||
|
|
||||||
const broadcastMessage = () => {
|
|
||||||
if (socket && bodyshop && bodyshop.id) {
|
|
||||||
console.log(`Broadcasting message to bodyshop ${bodyshop.id}:`);
|
|
||||||
socket.emit("broadcast-to-bodyshop", bodyshop.id, `Hello from ${clientId}`);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{import.meta.env.PROD && <ChatAffixContainer bodyshop={bodyshop} chatVisible={chatVisible} />}
|
{import.meta.env.PROD && <ChatAffixContainer bodyshop={bodyshop} chatVisible={chatVisible} />}
|
||||||
@@ -604,7 +594,7 @@ export function Manage({ conflict, bodyshop }) {
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div style={{ display: "flex" }}>
|
<div style={{ display: "flex" }}>
|
||||||
<div onClick={broadcastMessage}>
|
<div>
|
||||||
{`${InstanceRenderManager({
|
{`${InstanceRenderManager({
|
||||||
imex: t("titles.imexonline"),
|
imex: t("titles.imexonline"),
|
||||||
rome: t("titles.romeonline"),
|
rome: t("titles.romeonline"),
|
||||||
|
|||||||
@@ -1,26 +1,6 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import ProductionBoardKanbanContainer from "../../components/production-board-kanban/production-board-kanban.container";
|
import ProductionBoardKanbanContainer from "../../components/production-board-kanban/production-board-kanban.container";
|
||||||
import { connect } from "react-redux";
|
|
||||||
import { createStructuredSelector } from "reselect";
|
|
||||||
import { selectBodyshop } from "../../redux/user/user.selectors";
|
|
||||||
import { useSplitTreatments } from "@splitsoftware/splitio-react";
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
|
||||||
//currentUser: selectCurrentUser
|
|
||||||
bodyshop: selectBodyshop
|
|
||||||
});
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
|
||||||
//setUserLanguage: language => dispatch(setUserLanguage(language))
|
|
||||||
});
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(ProductionBoardComponent);
|
|
||||||
|
|
||||||
export function ProductionBoardComponent({ bodyshop }) {
|
export default function ProductionBoardComponent() {
|
||||||
const {
|
return <ProductionBoardKanbanContainer />;
|
||||||
treatments: { Production_Use_View }
|
|
||||||
} = useSplitTreatments({
|
|
||||||
attributes: {},
|
|
||||||
names: ["Production_Use_View"],
|
|
||||||
splitKey: bodyshop && bodyshop.imexshopid
|
|
||||||
});
|
|
||||||
|
|
||||||
return <ProductionBoardKanbanContainer subscriptionType={Production_Use_View.treatment} />;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,31 +2,11 @@ import React from "react";
|
|||||||
import NoteUpsertModal from "../../components/note-upsert-modal/note-upsert-modal.container";
|
import NoteUpsertModal from "../../components/note-upsert-modal/note-upsert-modal.container";
|
||||||
import ProductionListTable from "../../components/production-list-table/production-list-table.container";
|
import ProductionListTable from "../../components/production-list-table/production-list-table.container";
|
||||||
|
|
||||||
import { connect } from "react-redux";
|
export default function ProductionListComponent() {
|
||||||
import { createStructuredSelector } from "reselect";
|
|
||||||
import { selectBodyshop } from "../../redux/user/user.selectors";
|
|
||||||
import { useSplitTreatments } from "@splitsoftware/splitio-react";
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
|
||||||
bodyshop: selectBodyshop
|
|
||||||
});
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
|
||||||
//setUserLanguage: language => dispatch(setUserLanguage(language))
|
|
||||||
});
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(ProductionListComponent);
|
|
||||||
|
|
||||||
export function ProductionListComponent({ bodyshop }) {
|
|
||||||
const {
|
|
||||||
treatments: { Production_Use_View }
|
|
||||||
} = useSplitTreatments({
|
|
||||||
attributes: {},
|
|
||||||
names: ["Production_Use_View"],
|
|
||||||
splitKey: bodyshop && bodyshop.imexshopid
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<NoteUpsertModal />
|
<NoteUpsertModal />
|
||||||
<ProductionListTable subscriptionType={Production_Use_View.treatment} />
|
<ProductionListTable />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,21 +1,21 @@
|
|||||||
import { Tabs } from "antd";
|
import { Tabs } from "antd";
|
||||||
import queryString from "query-string";
|
|
||||||
import React, { useEffect } from "react";
|
import React, { useEffect } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { connect } from "react-redux";
|
|
||||||
import { useLocation, useNavigate } from "react-router-dom";
|
import { useLocation, useNavigate } from "react-router-dom";
|
||||||
import { createStructuredSelector } from "reselect";
|
import queryString from "query-string";
|
||||||
import RbacWrapper from "../../components/rbac-wrapper/rbac-wrapper.component";
|
import { useTranslation } from "react-i18next";
|
||||||
import ShopCsiConfig from "../../components/shop-csi-config/shop-csi-config.component";
|
|
||||||
import ShopEmployeesContainer from "../../components/shop-employees/shop-employees.container";
|
import ShopEmployeesContainer from "../../components/shop-employees/shop-employees.container";
|
||||||
import ShopInfoContainer from "../../components/shop-info/shop-info.container";
|
import ShopInfoContainer from "../../components/shop-info/shop-info.container";
|
||||||
import ShopInfoUsersComponent from "../../components/shop-users/shop-users.component";
|
import ShopCsiConfig from "../../components/shop-csi-config/shop-csi-config.component";
|
||||||
|
import RbacWrapper from "../../components/rbac-wrapper/rbac-wrapper.component";
|
||||||
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
|
import { connect } from "react-redux";
|
||||||
|
import { createStructuredSelector } from "reselect";
|
||||||
import { setBreadcrumbs, setSelectedHeader } from "../../redux/application/application.actions";
|
import { setBreadcrumbs, setSelectedHeader } from "../../redux/application/application.actions";
|
||||||
import { selectBodyshop } from "../../redux/user/user.selectors";
|
import { selectBodyshop } from "../../redux/user/user.selectors";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
import ShopInfoUsersComponent from "../../components/shop-users/shop-users.component";
|
||||||
|
|
||||||
import { HasFeatureAccess } from "../../components/feature-wrapper/feature-wrapper.component";
|
|
||||||
import ShopTeamsContainer from "../../components/shop-teams/shop-teams.container";
|
import ShopTeamsContainer from "../../components/shop-teams/shop-teams.container";
|
||||||
|
import { HasFeatureAccess } from "../../components/feature-wrapper/feature-wrapper.component";
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop
|
bodyshop: selectBodyshop
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import { useTranslation } from "react-i18next";
|
|||||||
import RbacWrapperComponent from "../../components/rbac-wrapper/rbac-wrapper.component";
|
import RbacWrapperComponent from "../../components/rbac-wrapper/rbac-wrapper.component";
|
||||||
import TechLookupJobsList from "../../components/tech-lookup-jobs-list/tech-lookup-jobs-list.component";
|
import TechLookupJobsList from "../../components/tech-lookup-jobs-list/tech-lookup-jobs-list.component";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr";
|
||||||
import TechLookupJobsDrawer from "../../components/tech-lookup-jobs-drawer/tech-lookup-jobs-drawer.component";
|
|
||||||
|
|
||||||
export default function TechLookupContainer() {
|
export default function TechLookupContainer() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -21,7 +20,6 @@ export default function TechLookupContainer() {
|
|||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<RbacWrapperComponent action="jobs:list-active">
|
<RbacWrapperComponent action="jobs:list-active">
|
||||||
<TechLookupJobsDrawer />
|
|
||||||
<TechLookupJobsList />
|
<TechLookupJobsList />
|
||||||
</RbacWrapperComponent>
|
</RbacWrapperComponent>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import ErrorBoundary from "../../components/error-boundary/error-boundary.compon
|
|||||||
import FeatureWrapper from "../../components/feature-wrapper/feature-wrapper.component";
|
import FeatureWrapper from "../../components/feature-wrapper/feature-wrapper.component";
|
||||||
import LoadingSpinner from "../../components/loading-spinner/loading-spinner.component";
|
import LoadingSpinner from "../../components/loading-spinner/loading-spinner.component";
|
||||||
import TechHeader from "../../components/tech-header/tech-header.component";
|
import TechHeader from "../../components/tech-header/tech-header.component";
|
||||||
|
import TechLookupJobsDrawer from "../../components/tech-lookup-jobs-drawer/tech-lookup-jobs-drawer.component";
|
||||||
import TechSider from "../../components/tech-sider/tech-sider.component";
|
import TechSider from "../../components/tech-sider/tech-sider.component";
|
||||||
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
||||||
import { selectTechnician } from "../../redux/tech/tech.selectors";
|
import { selectTechnician } from "../../redux/tech/tech.selectors";
|
||||||
@@ -67,7 +68,7 @@ export function TechPage({ technician }) {
|
|||||||
<Layout>
|
<Layout>
|
||||||
<UpdateAlert />
|
<UpdateAlert />
|
||||||
<TechHeader />
|
<TechHeader />
|
||||||
|
<TechLookupJobsDrawer />
|
||||||
<TaskUpsertModalContainer />
|
<TaskUpsertModalContainer />
|
||||||
<Content className="tech-content-container">
|
<Content className="tech-content-container">
|
||||||
<ErrorBoundary>
|
<ErrorBoundary>
|
||||||
|
|||||||
@@ -36,8 +36,7 @@ export function* openChatByPhone({ payload }) {
|
|||||||
data: { conversations }
|
data: { conversations }
|
||||||
} = yield client.query({
|
} = yield client.query({
|
||||||
query: CONVERSATION_ID_BY_PHONE,
|
query: CONVERSATION_ID_BY_PHONE,
|
||||||
variables: { phone: p.number },
|
variables: { phone: p.number }
|
||||||
fetchPolicy: 'no-cache'
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (conversations.length === 0) {
|
if (conversations.length === 0) {
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import {
|
|||||||
signInWithEmailAndPassword,
|
signInWithEmailAndPassword,
|
||||||
signOut
|
signOut
|
||||||
} from "firebase/auth";
|
} from "firebase/auth";
|
||||||
import { arrayUnion, doc, getDoc, setDoc, updateDoc } from "firebase/firestore";
|
import { doc, getDoc, setDoc } from "firebase/firestore";
|
||||||
import { getToken } from "firebase/messaging";
|
import { getToken } from "firebase/messaging";
|
||||||
import i18next from "i18next";
|
import i18next from "i18next";
|
||||||
import LogRocket from "logrocket";
|
import LogRocket from "logrocket";
|
||||||
@@ -48,7 +48,6 @@ import {
|
|||||||
validatePasswordResetSuccess
|
validatePasswordResetSuccess
|
||||||
} from "./user.actions";
|
} from "./user.actions";
|
||||||
import UserActionTypes from "./user.types";
|
import UserActionTypes from "./user.types";
|
||||||
import cleanAxios from "../../utils/CleanAxios";
|
|
||||||
|
|
||||||
const fpPromise = FingerprintJS.load();
|
const fpPromise = FingerprintJS.load();
|
||||||
|
|
||||||
@@ -178,24 +177,10 @@ export function* setInstanceIdSaga({ payload: uid }) {
|
|||||||
// Get the visitor identifier when you need it.
|
// Get the visitor identifier when you need it.
|
||||||
const fp = yield fpPromise;
|
const fp = yield fpPromise;
|
||||||
const result = yield fp.get();
|
const result = yield fp.get();
|
||||||
const res = yield cleanAxios.get("https://api.ipify.org/?format=json");
|
yield setDoc(userInstanceRef, {
|
||||||
const udoc = yield getDoc(userInstanceRef);
|
timestamp: new Date(),
|
||||||
|
fingerprint: result.visitorId
|
||||||
if (!udoc.data()) {
|
});
|
||||||
yield setDoc(userInstanceRef, {
|
|
||||||
timestamp: new Date(),
|
|
||||||
fingerprint: result.visitorId,
|
|
||||||
//totalFingerprint: result,
|
|
||||||
ip: [res.data.ip]
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
yield updateDoc(userInstanceRef, {
|
|
||||||
timestamp: new Date(),
|
|
||||||
fingerprint: result.visitorId,
|
|
||||||
//totalFingerprint: result,
|
|
||||||
ip: arrayUnion(res.data.ip)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
yield put(setLocalFingerprint(result.visitorId));
|
yield put(setLocalFingerprint(result.visitorId));
|
||||||
yield delay(5 * 60 * 1000);
|
yield delay(5 * 60 * 1000);
|
||||||
|
|||||||
@@ -230,7 +230,7 @@
|
|||||||
"markexported": "Mark Exported",
|
"markexported": "Mark Exported",
|
||||||
"markforreexport": "Mark for Re-export",
|
"markforreexport": "Mark for Re-export",
|
||||||
"new": "New Bill",
|
"new": "New Bill",
|
||||||
"nobilllines": "",
|
"nobilllines": "This part has not yet been recieved.",
|
||||||
"noneselected": "No bill selected.",
|
"noneselected": "No bill selected.",
|
||||||
"onlycmforinvoiced": "Only credit memos can be entered for any Job that has been invoiced, exported, or voided.",
|
"onlycmforinvoiced": "Only credit memos can be entered for any Job that has been invoiced, exported, or voided.",
|
||||||
"printlabels": "Print Labels",
|
"printlabels": "Print Labels",
|
||||||
@@ -270,9 +270,9 @@
|
|||||||
"testrender": "Test Render"
|
"testrender": "Test Render"
|
||||||
},
|
},
|
||||||
"errors": {
|
"errors": {
|
||||||
"creatingdefaultview": "Error creating default view.",
|
|
||||||
"loading": "Unable to load shop details. Please call technical support.",
|
"loading": "Unable to load shop details. Please call technical support.",
|
||||||
"saving": "Error encountered while saving. {{message}}"
|
"saving": "Error encountered while saving. {{message}}",
|
||||||
|
"creatingdefaultview": "Error creating default view."
|
||||||
},
|
},
|
||||||
"fields": {
|
"fields": {
|
||||||
"ReceivableCustomField": "QBO Receivable Custom Field {{number}}",
|
"ReceivableCustomField": "QBO Receivable Custom Field {{number}}",
|
||||||
@@ -285,21 +285,19 @@
|
|||||||
},
|
},
|
||||||
"appt_length": "Default Appointment Length",
|
"appt_length": "Default Appointment Length",
|
||||||
"attach_pdf_to_email": "Attach PDF copy to sent emails?",
|
"attach_pdf_to_email": "Attach PDF copy to sent emails?",
|
||||||
"batchid": "ADP Batch ID",
|
|
||||||
"bill_allow_post_to_closed": "Allow Bills to be posted to Closed Jobs",
|
"bill_allow_post_to_closed": "Allow Bills to be posted to Closed Jobs",
|
||||||
"bill_federal_tax_rate": "Bills - Federal Tax Rate %",
|
"bill_federal_tax_rate": "Bills - Federal Tax Rate %",
|
||||||
"bill_local_tax_rate": "Bill - Local Tax Rate %",
|
"bill_local_tax_rate": "Bill - Local Tax Rate %",
|
||||||
"bill_state_tax_rate": "Bill - Provincial/State Tax Rate %",
|
"bill_state_tax_rate": "Bill - Provincial/State Tax Rate %",
|
||||||
"city": "City",
|
"city": "City",
|
||||||
"closingperiod": "Closing Period",
|
"closingperiod": "Closing Period",
|
||||||
"companycode": "ADP Company Code",
|
|
||||||
"country": "Country",
|
"country": "Country",
|
||||||
"dailybodytarget": "Scoreboard - Daily Body Target",
|
"dailybodytarget": "Scoreboard - Daily Body Target",
|
||||||
"dailypainttarget": "Scoreboard - Daily Paint Target",
|
"dailypainttarget": "Scoreboard - Daily Paint Target",
|
||||||
"default_adjustment_rate": "Default Labor Deduction Adjustment Rate",
|
"default_adjustment_rate": "Default Labor Deduction Adjustment Rate",
|
||||||
"deliver": {
|
"deliver": {
|
||||||
"require_actual_delivery_date": "Require Actual Delivery",
|
"templates": "Delivery Templates",
|
||||||
"templates": "Delivery Templates"
|
"require_actual_delivery_date": "Require Actual Delivery"
|
||||||
},
|
},
|
||||||
"dms": {
|
"dms": {
|
||||||
"apcontrol": "AP Control Number",
|
"apcontrol": "AP Control Number",
|
||||||
@@ -332,10 +330,6 @@
|
|||||||
"next_contact_hours": "Automatic Next Contact Date - Hours from Intake",
|
"next_contact_hours": "Automatic Next Contact Date - Hours from Intake",
|
||||||
"templates": "Intake Templates"
|
"templates": "Intake Templates"
|
||||||
},
|
},
|
||||||
"intellipay_config": {
|
|
||||||
"cash_discount_percentage": "Cash Discount %",
|
|
||||||
"enable_cash_discount": "Enable Cash Discounting"
|
|
||||||
},
|
|
||||||
"invoice_federal_tax_rate": "Invoices - Federal Tax Rate",
|
"invoice_federal_tax_rate": "Invoices - Federal Tax Rate",
|
||||||
"invoice_local_tax_rate": "Invoices - Local Tax Rate",
|
"invoice_local_tax_rate": "Invoices - Local Tax Rate",
|
||||||
"invoice_state_tax_rate": "Invoices - State Tax Rate",
|
"invoice_state_tax_rate": "Invoices - State Tax Rate",
|
||||||
@@ -667,8 +661,6 @@
|
|||||||
"filehandlers": "Adjusters",
|
"filehandlers": "Adjusters",
|
||||||
"insurancecos": "Insurance Companies",
|
"insurancecos": "Insurance Companies",
|
||||||
"intakechecklist": "Intake Checklist",
|
"intakechecklist": "Intake Checklist",
|
||||||
"intellipay": "IntelliPay",
|
|
||||||
"intellipay_cash_discount": "Please ensure that cash discounting has been enabled on your merchant account. Reach out to IntelliPay Support if you need assistance. ",
|
|
||||||
"jobstatuses": "Job Statuses",
|
"jobstatuses": "Job Statuses",
|
||||||
"laborrates": "Labor Rates",
|
"laborrates": "Labor Rates",
|
||||||
"licensing": "Licensing",
|
"licensing": "Licensing",
|
||||||
@@ -708,10 +700,10 @@
|
|||||||
"workingdays": "Working Days"
|
"workingdays": "Working Days"
|
||||||
},
|
},
|
||||||
"successes": {
|
"successes": {
|
||||||
"areyousure": "Are you sure you want to continue?",
|
|
||||||
"defaultviewcreated": "Default view created successfully.",
|
|
||||||
"save": "Shop configuration saved successfully. ",
|
"save": "Shop configuration saved successfully. ",
|
||||||
"unsavedchanges": "Unsaved changes will be lost. Are you sure you want to continue?"
|
"unsavedchanges": "Unsaved changes will be lost. Are you sure you want to continue?",
|
||||||
|
"areyousure": "Are you sure you want to continue?",
|
||||||
|
"defaultviewcreated": "Default view created successfully."
|
||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"centermustexist": "The chosen responsibility center does not exist.",
|
"centermustexist": "The chosen responsibility center does not exist.",
|
||||||
@@ -1141,8 +1133,8 @@
|
|||||||
},
|
},
|
||||||
"general": {
|
"general": {
|
||||||
"actions": {
|
"actions": {
|
||||||
|
"defaults": "Defaults",
|
||||||
"add": "Add",
|
"add": "Add",
|
||||||
"autoupdate": "{{app}} will automatically update in {{time}} seconds. Please save all changes.",
|
|
||||||
"calculate": "Calculate",
|
"calculate": "Calculate",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"clear": "Clear",
|
"clear": "Clear",
|
||||||
@@ -1150,8 +1142,6 @@
|
|||||||
"copied": "Copied!",
|
"copied": "Copied!",
|
||||||
"copylink": "Copy Link",
|
"copylink": "Copy Link",
|
||||||
"create": "Create",
|
"create": "Create",
|
||||||
"defaults": "Defaults",
|
|
||||||
"delay": "Delay Update (5 mins)",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"deleteall": "Delete All",
|
"deleteall": "Delete All",
|
||||||
"deselectall": "Deselect All",
|
"deselectall": "Deselect All",
|
||||||
@@ -1163,12 +1153,10 @@
|
|||||||
"print": "Print",
|
"print": "Print",
|
||||||
"refresh": "Refresh",
|
"refresh": "Refresh",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"remove_alert": "Are you sure you want to dismiss the alert?",
|
|
||||||
"reset": "Reset your changes.",
|
"reset": "Reset your changes.",
|
||||||
"resetpassword": "Reset Password",
|
"resetpassword": "Reset Password",
|
||||||
"save": "Save",
|
"save": "Save",
|
||||||
"saveandnew": "Save and New",
|
"saveandnew": "Save and New",
|
||||||
"saveas": "Save As",
|
|
||||||
"selectall": "Select All",
|
"selectall": "Select All",
|
||||||
"send": "Send",
|
"send": "Send",
|
||||||
"sendbysms": "Send by SMS",
|
"sendbysms": "Send by SMS",
|
||||||
@@ -1176,7 +1164,9 @@
|
|||||||
"submit": "Submit",
|
"submit": "Submit",
|
||||||
"tryagain": "Try Again",
|
"tryagain": "Try Again",
|
||||||
"view": "View",
|
"view": "View",
|
||||||
"viewreleasenotes": "See What's Changed"
|
"viewreleasenotes": "See What's Changed",
|
||||||
|
"remove_alert": "Are you sure you want to dismiss the alert?",
|
||||||
|
"saveas": "Save As"
|
||||||
},
|
},
|
||||||
"errors": {
|
"errors": {
|
||||||
"fcm": "You must allow notification permissions to have real time messaging. Click to try again.",
|
"fcm": "You must allow notification permissions to have real time messaging. Click to try again.",
|
||||||
@@ -1191,6 +1181,7 @@
|
|||||||
"vehicle": "Vehicle"
|
"vehicle": "Vehicle"
|
||||||
},
|
},
|
||||||
"labels": {
|
"labels": {
|
||||||
|
"unsavedchanges": "Unsaved changes.",
|
||||||
"actions": "Actions",
|
"actions": "Actions",
|
||||||
"areyousure": "Are you sure?",
|
"areyousure": "Are you sure?",
|
||||||
"barcode": "Barcode",
|
"barcode": "Barcode",
|
||||||
@@ -1259,7 +1250,6 @@
|
|||||||
"tuesday": "Tuesday",
|
"tuesday": "Tuesday",
|
||||||
"tvmode": "TV Mode",
|
"tvmode": "TV Mode",
|
||||||
"unknown": "Unknown",
|
"unknown": "Unknown",
|
||||||
"unsavedchanges": "Unsaved changes.",
|
|
||||||
"username": "Username",
|
"username": "Username",
|
||||||
"view": "View",
|
"view": "View",
|
||||||
"wednesday": "Wednesday",
|
"wednesday": "Wednesday",
|
||||||
@@ -1373,7 +1363,6 @@
|
|||||||
},
|
},
|
||||||
"job_payments": {
|
"job_payments": {
|
||||||
"buttons": {
|
"buttons": {
|
||||||
"create_short_link": "Generate Short Link",
|
|
||||||
"goback": "Go Back",
|
"goback": "Go Back",
|
||||||
"proceedtopayment": "Proceed to Payment",
|
"proceedtopayment": "Proceed to Payment",
|
||||||
"refundpayment": "Refund Payment"
|
"refundpayment": "Refund Payment"
|
||||||
@@ -2750,6 +2739,41 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"production": {
|
"production": {
|
||||||
|
"constants": {
|
||||||
|
"main_profile": "Default"
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"small": "Small",
|
||||||
|
"medium": "Medium",
|
||||||
|
"large": "Large",
|
||||||
|
"vertical": "Vertical",
|
||||||
|
"horizontal": "Horizontal"
|
||||||
|
},
|
||||||
|
"settings": {
|
||||||
|
"layout": "Layout",
|
||||||
|
"information": "Information",
|
||||||
|
"statistics_title": "Statistics",
|
||||||
|
"board_settings": "Board Settings",
|
||||||
|
"filters_title": "Filters",
|
||||||
|
"filters": {
|
||||||
|
"md_ins_cos": "Insurance Companies",
|
||||||
|
"md_estimators": "Estimators"
|
||||||
|
},
|
||||||
|
"statistics": {
|
||||||
|
"total_hours_in_production": "Hours in Production",
|
||||||
|
"total_lab_in_production": "Body Hours in Production",
|
||||||
|
"total_lar_in_production": "Refinish Hours in Production",
|
||||||
|
"total_amount_in_production": "Dollars in Production",
|
||||||
|
"jobs_in_production": "Jobs in Production",
|
||||||
|
"total_hours_on_board": "Hours on Board",
|
||||||
|
"total_lab_on_board": "Body Hours on Board",
|
||||||
|
"total_lar_on_board": "Refinish Hours on Board",
|
||||||
|
"total_amount_on_board": "Dollars on Board",
|
||||||
|
"total_jobs_on_board": "Jobs on Board",
|
||||||
|
"tasks_in_production": "Tasks in Production",
|
||||||
|
"tasks_on_board": "Tasks on Board"
|
||||||
|
}
|
||||||
|
},
|
||||||
"actions": {
|
"actions": {
|
||||||
"addcolumns": "Add Columns",
|
"addcolumns": "Add Columns",
|
||||||
"bodypriority-clear": "Clear Body Priority",
|
"bodypriority-clear": "Clear Body Priority",
|
||||||
@@ -2764,23 +2788,29 @@
|
|||||||
"suspend": "Suspend",
|
"suspend": "Suspend",
|
||||||
"unsuspend": "Unsuspend"
|
"unsuspend": "Unsuspend"
|
||||||
},
|
},
|
||||||
"constants": {
|
|
||||||
"main_profile": "Default"
|
|
||||||
},
|
|
||||||
"errors": {
|
"errors": {
|
||||||
"boardupdate": "Error encountered updating Job. {{message}}",
|
"boardupdate": "Error encountered updating Job. {{message}}",
|
||||||
"name_exists": "A Profile with this name already exists. Please choose a different name.",
|
|
||||||
"name_required": "Profile name is required.",
|
|
||||||
"removing": "Error removing from production board. {{error}}",
|
"removing": "Error removing from production board. {{error}}",
|
||||||
"settings": "Error saving board settings: {{error}}"
|
"settings": "Error saving board settings: {{error}}",
|
||||||
|
"name_exists": "A Profile with this name already exists. Please choose a different name.",
|
||||||
|
"name_required": "Profile name is required."
|
||||||
},
|
},
|
||||||
"labels": {
|
"labels": {
|
||||||
|
"kiosk_mode": "Kiosk Mode",
|
||||||
|
"on": "On",
|
||||||
|
"off": "Off",
|
||||||
|
"wide": "Wide",
|
||||||
|
"tall": "Tall",
|
||||||
|
"vertical": "Vertical",
|
||||||
|
"horizontal": "Horizontal",
|
||||||
|
"orientation": "Board Orientation",
|
||||||
|
"card_size": "Card Size",
|
||||||
|
"model_info": "Vehicle Info",
|
||||||
"actual_in": "Actual In",
|
"actual_in": "Actual In",
|
||||||
"addnewprofile": "Add New Profile",
|
|
||||||
"alert": "Alert",
|
"alert": "Alert",
|
||||||
|
"tasks": "Tasks",
|
||||||
"alertoff": "Remove alert from Job",
|
"alertoff": "Remove alert from Job",
|
||||||
"alerton": "Add alert to Job",
|
"alerton": "Add alert to Job",
|
||||||
"alerts": "Alerts",
|
|
||||||
"ats": "Alternative Transportation",
|
"ats": "Alternative Transportation",
|
||||||
"bodyhours": "B",
|
"bodyhours": "B",
|
||||||
"bodypriority": "B/P",
|
"bodypriority": "B/P",
|
||||||
@@ -2790,7 +2820,6 @@
|
|||||||
"qbo_usa": "QBO USA"
|
"qbo_usa": "QBO USA"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"card_size": "Card Size",
|
|
||||||
"cardcolor": "Colored Cards",
|
"cardcolor": "Colored Cards",
|
||||||
"cardsettings": "Card Settings",
|
"cardsettings": "Card Settings",
|
||||||
"clm_no": "Claim Number",
|
"clm_no": "Claim Number",
|
||||||
@@ -2799,88 +2828,48 @@
|
|||||||
"detailpriority": "D/P",
|
"detailpriority": "D/P",
|
||||||
"employeeassignments": "Employee Assignments",
|
"employeeassignments": "Employee Assignments",
|
||||||
"employeesearch": "Employee Search",
|
"employeesearch": "Employee Search",
|
||||||
"estimator": "Estimator",
|
|
||||||
"horizontal": "Horizontal",
|
|
||||||
"ins_co_nm": "Insurance Company Name",
|
"ins_co_nm": "Insurance Company Name",
|
||||||
"jobdetail": "Job Details",
|
"jobdetail": "Job Details",
|
||||||
"kiosk_mode": "Kiosk Mode",
|
|
||||||
"laborhrs": "Labor Hours",
|
"laborhrs": "Labor Hours",
|
||||||
"legend": "Legend:",
|
"legend": "Legend:",
|
||||||
"model_info": "Vehicle Info",
|
|
||||||
"note": "Production Note",
|
"note": "Production Note",
|
||||||
"off": "Off",
|
|
||||||
"on": "On",
|
|
||||||
"orientation": "Board Orientation",
|
|
||||||
"ownr_nm": "Customer Name",
|
"ownr_nm": "Customer Name",
|
||||||
"paintpriority": "P/P",
|
"paintpriority": "P/P",
|
||||||
"partsstatus": "Parts Status",
|
"partsstatus": "Parts Status",
|
||||||
|
"estimator": "Estimator",
|
||||||
|
"subtotal": "Subtotal",
|
||||||
"production_note": "Production Note",
|
"production_note": "Production Note",
|
||||||
"refinishhours": "R",
|
"refinishhours": "R",
|
||||||
"scheduled_completion": "Scheduled Completion",
|
"scheduled_completion": "Scheduled Completion",
|
||||||
"selectview": "Select a View",
|
"selectview": "Select a View",
|
||||||
"stickyheader": "Sticky Header (BETA)",
|
"stickyheader": "Sticky Header (BETA)",
|
||||||
"sublets": "Sublets",
|
"sublets": "Sublets",
|
||||||
"subtotal": "Subtotal",
|
|
||||||
"tall": "Tall",
|
|
||||||
"tasks": "Tasks",
|
|
||||||
"totalhours": "Total Hrs ",
|
"totalhours": "Total Hrs ",
|
||||||
"touchtime": "T/T",
|
"touchtime": "T/T",
|
||||||
"vertical": "Vertical",
|
|
||||||
"viewname": "View Name",
|
"viewname": "View Name",
|
||||||
"wide": "Wide"
|
"alerts": "Alerts",
|
||||||
},
|
"addnewprofile": "Add New Profile"
|
||||||
"options": {
|
|
||||||
"horizontal": "Horizontal",
|
|
||||||
"large": "Large",
|
|
||||||
"medium": "Medium",
|
|
||||||
"small": "Small",
|
|
||||||
"vertical": "Vertical"
|
|
||||||
},
|
|
||||||
"settings": {
|
|
||||||
"board_settings": "Board Settings",
|
|
||||||
"filters": {
|
|
||||||
"md_estimators": "Estimators",
|
|
||||||
"md_ins_cos": "Insurance Companies"
|
|
||||||
},
|
|
||||||
"filters_title": "Filters",
|
|
||||||
"information": "Information",
|
|
||||||
"layout": "Layout",
|
|
||||||
"statistics": {
|
|
||||||
"jobs_in_production": "Jobs in Production",
|
|
||||||
"tasks_in_production": "Tasks in Production",
|
|
||||||
"tasks_on_board": "Tasks on Board",
|
|
||||||
"total_amount_in_production": "Dollars in Production",
|
|
||||||
"total_amount_on_board": "Dollars on Board",
|
|
||||||
"total_hours_in_production": "Hours in Production",
|
|
||||||
"total_hours_on_board": "Hours on Board",
|
|
||||||
"total_jobs_on_board": "Jobs on Board",
|
|
||||||
"total_lab_in_production": "Body Hours in Production",
|
|
||||||
"total_lab_on_board": "Body Hours on Board",
|
|
||||||
"total_lar_in_production": "Refinish Hours in Production",
|
|
||||||
"total_lar_on_board": "Refinish Hours on Board"
|
|
||||||
},
|
|
||||||
"statistics_title": "Statistics"
|
|
||||||
},
|
|
||||||
"statistics": {
|
|
||||||
"currency_symbol": "$",
|
|
||||||
"hours": "Hours",
|
|
||||||
"jobs": "Jobs",
|
|
||||||
"jobs_in_production": "Jobs in Production",
|
|
||||||
"tasks": "Tasks",
|
|
||||||
"tasks_in_production": "Tasks in Production",
|
|
||||||
"tasks_on_board": "Tasks on Board",
|
|
||||||
"total_amount_in_production": "Dollars in Production",
|
|
||||||
"total_amount_on_board": "Dollars on Board",
|
|
||||||
"total_hours_in_production": "Hours in Production",
|
|
||||||
"total_hours_on_board": "Hours on Board",
|
|
||||||
"total_jobs_on_board": "Jobs on Board",
|
|
||||||
"total_lab_in_production": "Body Hours in Production",
|
|
||||||
"total_lab_on_board": "Body Hours on Board",
|
|
||||||
"total_lar_in_production": "Refinish Hours in Production",
|
|
||||||
"total_lar_on_board": "Refinish Hours on Board"
|
|
||||||
},
|
},
|
||||||
"successes": {
|
"successes": {
|
||||||
"removed": "Job removed from production."
|
"removed": "Job removed from production."
|
||||||
|
},
|
||||||
|
"statistics": {
|
||||||
|
"total_hours_in_production": "Hours in Production",
|
||||||
|
"total_lab_in_production": "Body Hours in Production",
|
||||||
|
"total_lar_in_production": "Refinish Hours in Production",
|
||||||
|
"total_amount_in_production": "Dollars in Production",
|
||||||
|
"jobs_in_production": "Jobs in Production",
|
||||||
|
"total_hours_on_board": "Hours on Board",
|
||||||
|
"total_lab_on_board": "Body Hours on Board",
|
||||||
|
"total_lar_on_board": "Refinish Hours on Board",
|
||||||
|
"total_amount_on_board": "Dollars on Board",
|
||||||
|
"total_jobs_on_board": "Jobs on Board",
|
||||||
|
"tasks_in_production": "Tasks in Production",
|
||||||
|
"tasks_on_board": "Tasks on Board",
|
||||||
|
"tasks": "Tasks",
|
||||||
|
"hours": "Hours",
|
||||||
|
"currency_symbol": "$",
|
||||||
|
"jobs": "Jobs"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"profile": {
|
"profile": {
|
||||||
@@ -2938,8 +2927,6 @@
|
|||||||
"vendor": "Vendor"
|
"vendor": "Vendor"
|
||||||
},
|
},
|
||||||
"templates": {
|
"templates": {
|
||||||
"adp_payroll_flat": "ADP Payroll - Flat Rate",
|
|
||||||
"adp_payroll_straight": "ADP Payroll - Straight Time",
|
|
||||||
"anticipated_revenue": "Anticipated Revenue",
|
"anticipated_revenue": "Anticipated Revenue",
|
||||||
"ar_aging": "AR Aging",
|
"ar_aging": "AR Aging",
|
||||||
"attendance_detail": "Attendance (All Employees)",
|
"attendance_detail": "Attendance (All Employees)",
|
||||||
@@ -3431,18 +3418,6 @@
|
|||||||
"vehicledetail": "Vehicle Details {{vehicle}} | {{app}}",
|
"vehicledetail": "Vehicle Details {{vehicle}} | {{app}}",
|
||||||
"vehicles": "All Vehicles | {{app}}"
|
"vehicles": "All Vehicles | {{app}}"
|
||||||
},
|
},
|
||||||
"trello": {
|
|
||||||
"labels": {
|
|
||||||
"add_card": "Add Card",
|
|
||||||
"add_lane": "Add Lane",
|
|
||||||
"cancel": "Cancel",
|
|
||||||
"delete_lane": "Delete Lane",
|
|
||||||
"description": "Description",
|
|
||||||
"label": "Label",
|
|
||||||
"lane_actions": "Lane Actions",
|
|
||||||
"title": "Title"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"tt_approvals": {
|
"tt_approvals": {
|
||||||
"actions": {
|
"actions": {
|
||||||
"approveselected": "Approve Selected"
|
"approveselected": "Approve Selected"
|
||||||
@@ -3581,6 +3556,18 @@
|
|||||||
"validation": {
|
"validation": {
|
||||||
"unique_vendor_name": "You must enter a unique vendor name."
|
"unique_vendor_name": "You must enter a unique vendor name."
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"trello": {
|
||||||
|
"labels": {
|
||||||
|
"add_card": "Add Card",
|
||||||
|
"add_lane": "Add Lane",
|
||||||
|
"delete_lane": "Delete Lane",
|
||||||
|
"lane_actions": "Lane Actions",
|
||||||
|
"title": "Title",
|
||||||
|
"description": "Description",
|
||||||
|
"label": "Label",
|
||||||
|
"cancel": "Cancel"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -270,9 +270,9 @@
|
|||||||
"testrender": ""
|
"testrender": ""
|
||||||
},
|
},
|
||||||
"errors": {
|
"errors": {
|
||||||
"creatingdefaultview": "",
|
|
||||||
"loading": "No se pueden cargar los detalles de la tienda. Por favor llame al soporte técnico.",
|
"loading": "No se pueden cargar los detalles de la tienda. Por favor llame al soporte técnico.",
|
||||||
"saving": ""
|
"saving": "",
|
||||||
|
"creatingdefaultview": ""
|
||||||
},
|
},
|
||||||
"fields": {
|
"fields": {
|
||||||
"ReceivableCustomField": "",
|
"ReceivableCustomField": "",
|
||||||
@@ -285,21 +285,19 @@
|
|||||||
},
|
},
|
||||||
"appt_length": "",
|
"appt_length": "",
|
||||||
"attach_pdf_to_email": "",
|
"attach_pdf_to_email": "",
|
||||||
"batchid": "",
|
|
||||||
"bill_allow_post_to_closed": "",
|
"bill_allow_post_to_closed": "",
|
||||||
"bill_federal_tax_rate": "",
|
"bill_federal_tax_rate": "",
|
||||||
"bill_local_tax_rate": "",
|
"bill_local_tax_rate": "",
|
||||||
"bill_state_tax_rate": "",
|
"bill_state_tax_rate": "",
|
||||||
"city": "",
|
"city": "",
|
||||||
"closingperiod": "",
|
"closingperiod": "",
|
||||||
"companycode": "",
|
|
||||||
"country": "",
|
"country": "",
|
||||||
"dailybodytarget": "",
|
"dailybodytarget": "",
|
||||||
"dailypainttarget": "",
|
"dailypainttarget": "",
|
||||||
"default_adjustment_rate": "",
|
"default_adjustment_rate": "",
|
||||||
"deliver": {
|
"deliver": {
|
||||||
"require_actual_delivery_date": "",
|
"templates": "",
|
||||||
"templates": ""
|
"require_actual_delivery_date": ""
|
||||||
},
|
},
|
||||||
"dms": {
|
"dms": {
|
||||||
"apcontrol": "",
|
"apcontrol": "",
|
||||||
@@ -332,10 +330,6 @@
|
|||||||
"next_contact_hours": "",
|
"next_contact_hours": "",
|
||||||
"templates": ""
|
"templates": ""
|
||||||
},
|
},
|
||||||
"intellipay_config": {
|
|
||||||
"cash_discount_percentage": "",
|
|
||||||
"enable_cash_discount": ""
|
|
||||||
},
|
|
||||||
"invoice_federal_tax_rate": "",
|
"invoice_federal_tax_rate": "",
|
||||||
"invoice_local_tax_rate": "",
|
"invoice_local_tax_rate": "",
|
||||||
"invoice_state_tax_rate": "",
|
"invoice_state_tax_rate": "",
|
||||||
@@ -667,8 +661,6 @@
|
|||||||
"filehandlers": "",
|
"filehandlers": "",
|
||||||
"insurancecos": "",
|
"insurancecos": "",
|
||||||
"intakechecklist": "",
|
"intakechecklist": "",
|
||||||
"intellipay": "",
|
|
||||||
"intellipay_cash_discount": "",
|
|
||||||
"jobstatuses": "",
|
"jobstatuses": "",
|
||||||
"laborrates": "",
|
"laborrates": "",
|
||||||
"licensing": "",
|
"licensing": "",
|
||||||
@@ -708,10 +700,10 @@
|
|||||||
"workingdays": ""
|
"workingdays": ""
|
||||||
},
|
},
|
||||||
"successes": {
|
"successes": {
|
||||||
"areyousure": "",
|
|
||||||
"defaultviewcreated": "",
|
|
||||||
"save": "",
|
"save": "",
|
||||||
"unsavedchanges": ""
|
"unsavedchanges": "",
|
||||||
|
"areyousure": "",
|
||||||
|
"defaultviewcreated": ""
|
||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"centermustexist": "",
|
"centermustexist": "",
|
||||||
@@ -1141,8 +1133,8 @@
|
|||||||
},
|
},
|
||||||
"general": {
|
"general": {
|
||||||
"actions": {
|
"actions": {
|
||||||
|
"defaults": "defaults",
|
||||||
"add": "",
|
"add": "",
|
||||||
"autoupdate": "",
|
|
||||||
"calculate": "",
|
"calculate": "",
|
||||||
"cancel": "",
|
"cancel": "",
|
||||||
"clear": "",
|
"clear": "",
|
||||||
@@ -1150,8 +1142,6 @@
|
|||||||
"copied": "",
|
"copied": "",
|
||||||
"copylink": "",
|
"copylink": "",
|
||||||
"create": "",
|
"create": "",
|
||||||
"defaults": "defaults",
|
|
||||||
"delay": "",
|
|
||||||
"delete": "Borrar",
|
"delete": "Borrar",
|
||||||
"deleteall": "",
|
"deleteall": "",
|
||||||
"deselectall": "",
|
"deselectall": "",
|
||||||
@@ -1163,12 +1153,10 @@
|
|||||||
"print": "",
|
"print": "",
|
||||||
"refresh": "",
|
"refresh": "",
|
||||||
"remove": "",
|
"remove": "",
|
||||||
"remove_alert": "",
|
|
||||||
"reset": " Restablecer a original.",
|
"reset": " Restablecer a original.",
|
||||||
"resetpassword": "",
|
"resetpassword": "",
|
||||||
"save": "Salvar",
|
"save": "Salvar",
|
||||||
"saveandnew": "",
|
"saveandnew": "",
|
||||||
"saveas": "",
|
|
||||||
"selectall": "",
|
"selectall": "",
|
||||||
"send": "",
|
"send": "",
|
||||||
"sendbysms": "",
|
"sendbysms": "",
|
||||||
@@ -1176,7 +1164,9 @@
|
|||||||
"submit": "",
|
"submit": "",
|
||||||
"tryagain": "",
|
"tryagain": "",
|
||||||
"view": "",
|
"view": "",
|
||||||
"viewreleasenotes": ""
|
"viewreleasenotes": "",
|
||||||
|
"remove_alert": "",
|
||||||
|
"saveas": ""
|
||||||
},
|
},
|
||||||
"errors": {
|
"errors": {
|
||||||
"fcm": "",
|
"fcm": "",
|
||||||
@@ -1191,6 +1181,7 @@
|
|||||||
"vehicle": ""
|
"vehicle": ""
|
||||||
},
|
},
|
||||||
"labels": {
|
"labels": {
|
||||||
|
"unsavedchanges": "",
|
||||||
"actions": "Comportamiento",
|
"actions": "Comportamiento",
|
||||||
"areyousure": "",
|
"areyousure": "",
|
||||||
"barcode": "código de barras",
|
"barcode": "código de barras",
|
||||||
@@ -1259,7 +1250,6 @@
|
|||||||
"tuesday": "",
|
"tuesday": "",
|
||||||
"tvmode": "",
|
"tvmode": "",
|
||||||
"unknown": "Desconocido",
|
"unknown": "Desconocido",
|
||||||
"unsavedchanges": "",
|
|
||||||
"username": "",
|
"username": "",
|
||||||
"view": "",
|
"view": "",
|
||||||
"wednesday": "",
|
"wednesday": "",
|
||||||
@@ -1373,7 +1363,6 @@
|
|||||||
},
|
},
|
||||||
"job_payments": {
|
"job_payments": {
|
||||||
"buttons": {
|
"buttons": {
|
||||||
"create_short_link": "",
|
|
||||||
"goback": "",
|
"goback": "",
|
||||||
"proceedtopayment": "",
|
"proceedtopayment": "",
|
||||||
"refundpayment": ""
|
"refundpayment": ""
|
||||||
@@ -2750,6 +2739,41 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"production": {
|
"production": {
|
||||||
|
"constants": {
|
||||||
|
"main_profile": ""
|
||||||
|
},
|
||||||
|
"options": {
|
||||||
|
"small": "",
|
||||||
|
"medium": "",
|
||||||
|
"large": "",
|
||||||
|
"vertical": "",
|
||||||
|
"horizontal": ""
|
||||||
|
},
|
||||||
|
"settings": {
|
||||||
|
"layout": "",
|
||||||
|
"information": "",
|
||||||
|
"statistics_title": "",
|
||||||
|
"board_settings": "",
|
||||||
|
"filters_title": "",
|
||||||
|
"filters": {
|
||||||
|
"md_ins_cos": "",
|
||||||
|
"md_estimators": ""
|
||||||
|
},
|
||||||
|
"statistics": {
|
||||||
|
"total_hours_in_production": "",
|
||||||
|
"total_lab_in_production": "",
|
||||||
|
"total_lar_in_production": "",
|
||||||
|
"total_amount_in_production": "",
|
||||||
|
"jobs_in_production": "",
|
||||||
|
"total_hours_on_board": "",
|
||||||
|
"total_lab_on_board": "",
|
||||||
|
"total_lar_on_board": "",
|
||||||
|
"total_amount_on_board": "",
|
||||||
|
"total_jobs_on_board": "",
|
||||||
|
"tasks_in_production": "",
|
||||||
|
"tasks_on_board": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
"actions": {
|
"actions": {
|
||||||
"addcolumns": "",
|
"addcolumns": "",
|
||||||
"bodypriority-clear": "",
|
"bodypriority-clear": "",
|
||||||
@@ -2764,23 +2788,29 @@
|
|||||||
"suspend": "",
|
"suspend": "",
|
||||||
"unsuspend": ""
|
"unsuspend": ""
|
||||||
},
|
},
|
||||||
"constants": {
|
|
||||||
"main_profile": ""
|
|
||||||
},
|
|
||||||
"errors": {
|
"errors": {
|
||||||
"boardupdate": "",
|
"boardupdate": "",
|
||||||
"name_exists": "",
|
|
||||||
"name_required": "",
|
|
||||||
"removing": "",
|
"removing": "",
|
||||||
"settings": ""
|
"settings": "",
|
||||||
|
"name_exists": "",
|
||||||
|
"name_required": ""
|
||||||
},
|
},
|
||||||
"labels": {
|
"labels": {
|
||||||
|
"kiosk_mode": "",
|
||||||
|
"on": "",
|
||||||
|
"off": "",
|
||||||
|
"wide": "",
|
||||||
|
"tall": "",
|
||||||
|
"vertical": "",
|
||||||
|
"horizontal": "",
|
||||||
|
"orientation": "",
|
||||||
|
"card_size": "",
|
||||||
|
"model_info": "",
|
||||||
"actual_in": "",
|
"actual_in": "",
|
||||||
"addnewprofile": "",
|
|
||||||
"alert": "",
|
"alert": "",
|
||||||
|
"tasks": "",
|
||||||
"alertoff": "",
|
"alertoff": "",
|
||||||
"alerton": "",
|
"alerton": "",
|
||||||
"alerts": "",
|
|
||||||
"ats": "",
|
"ats": "",
|
||||||
"bodyhours": "",
|
"bodyhours": "",
|
||||||
"bodypriority": "",
|
"bodypriority": "",
|
||||||
@@ -2790,7 +2820,6 @@
|
|||||||
"qbo_usa": ""
|
"qbo_usa": ""
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"card_size": "",
|
|
||||||
"cardcolor": "",
|
"cardcolor": "",
|
||||||
"cardsettings": "",
|
"cardsettings": "",
|
||||||
"clm_no": "",
|
"clm_no": "",
|
||||||
@@ -2799,88 +2828,48 @@
|
|||||||
"detailpriority": "",
|
"detailpriority": "",
|
||||||
"employeeassignments": "",
|
"employeeassignments": "",
|
||||||
"employeesearch": "",
|
"employeesearch": "",
|
||||||
"estimator": "",
|
|
||||||
"horizontal": "",
|
|
||||||
"ins_co_nm": "",
|
"ins_co_nm": "",
|
||||||
"jobdetail": "",
|
"jobdetail": "",
|
||||||
"kiosk_mode": "",
|
|
||||||
"laborhrs": "",
|
"laborhrs": "",
|
||||||
"legend": "",
|
"legend": "",
|
||||||
"model_info": "",
|
|
||||||
"note": "",
|
"note": "",
|
||||||
"off": "",
|
|
||||||
"on": "",
|
|
||||||
"orientation": "",
|
|
||||||
"ownr_nm": "",
|
"ownr_nm": "",
|
||||||
"paintpriority": "",
|
"paintpriority": "",
|
||||||
"partsstatus": "",
|
"partsstatus": "",
|
||||||
|
"estimator": "",
|
||||||
|
"subtotal": "",
|
||||||
"production_note": "",
|
"production_note": "",
|
||||||
"refinishhours": "",
|
"refinishhours": "",
|
||||||
"scheduled_completion": "",
|
"scheduled_completion": "",
|
||||||
"selectview": "",
|
"selectview": "",
|
||||||
"stickyheader": "",
|
"stickyheader": "",
|
||||||
"sublets": "",
|
"sublets": "",
|
||||||
"subtotal": "",
|
|
||||||
"tall": "",
|
|
||||||
"tasks": "",
|
|
||||||
"totalhours": "",
|
"totalhours": "",
|
||||||
"touchtime": "",
|
"touchtime": "",
|
||||||
"vertical": "",
|
|
||||||
"viewname": "",
|
"viewname": "",
|
||||||
"wide": ""
|
"alerts": "",
|
||||||
},
|
"addnewprofile": ""
|
||||||
"options": {
|
|
||||||
"horizontal": "",
|
|
||||||
"large": "",
|
|
||||||
"medium": "",
|
|
||||||
"small": "",
|
|
||||||
"vertical": ""
|
|
||||||
},
|
|
||||||
"settings": {
|
|
||||||
"board_settings": "",
|
|
||||||
"filters": {
|
|
||||||
"md_estimators": "",
|
|
||||||
"md_ins_cos": ""
|
|
||||||
},
|
|
||||||
"filters_title": "",
|
|
||||||
"information": "",
|
|
||||||
"layout": "",
|
|
||||||
"statistics": {
|
|
||||||
"jobs_in_production": "",
|
|
||||||
"tasks_in_production": "",
|
|
||||||
"tasks_on_board": "",
|
|
||||||
"total_amount_in_production": "",
|
|
||||||
"total_amount_on_board": "",
|
|
||||||
"total_hours_in_production": "",
|
|
||||||
"total_hours_on_board": "",
|
|
||||||
"total_jobs_on_board": "",
|
|
||||||
"total_lab_in_production": "",
|
|
||||||
"total_lab_on_board": "",
|
|
||||||
"total_lar_in_production": "",
|
|
||||||
"total_lar_on_board": ""
|
|
||||||
},
|
|
||||||
"statistics_title": ""
|
|
||||||
},
|
|
||||||
"statistics": {
|
|
||||||
"currency_symbol": "",
|
|
||||||
"hours": "",
|
|
||||||
"jobs": "",
|
|
||||||
"jobs_in_production": "",
|
|
||||||
"tasks": "",
|
|
||||||
"tasks_in_production": "",
|
|
||||||
"tasks_on_board": "",
|
|
||||||
"total_amount_in_production": "",
|
|
||||||
"total_amount_on_board": "",
|
|
||||||
"total_hours_in_production": "",
|
|
||||||
"total_hours_on_board": "",
|
|
||||||
"total_jobs_on_board": "",
|
|
||||||
"total_lab_in_production": "",
|
|
||||||
"total_lab_on_board": "",
|
|
||||||
"total_lar_in_production": "",
|
|
||||||
"total_lar_on_board": ""
|
|
||||||
},
|
},
|
||||||
"successes": {
|
"successes": {
|
||||||
"removed": ""
|
"removed": ""
|
||||||
|
},
|
||||||
|
"statistics": {
|
||||||
|
"total_hours_in_production": "",
|
||||||
|
"total_lab_in_production": "",
|
||||||
|
"total_lar_in_production": "",
|
||||||
|
"total_amount_in_production": "",
|
||||||
|
"jobs_in_production": "",
|
||||||
|
"total_hours_on_board": "",
|
||||||
|
"total_lab_on_board": "",
|
||||||
|
"total_lar_on_board": "",
|
||||||
|
"total_amount_on_board": "",
|
||||||
|
"total_jobs_on_board": "",
|
||||||
|
"tasks_in_production": "",
|
||||||
|
"tasks_on_board": "",
|
||||||
|
"tasks": "",
|
||||||
|
"hours": "",
|
||||||
|
"currency_symbol": "",
|
||||||
|
"jobs": ""
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"profile": {
|
"profile": {
|
||||||
@@ -2938,8 +2927,6 @@
|
|||||||
"vendor": ""
|
"vendor": ""
|
||||||
},
|
},
|
||||||
"templates": {
|
"templates": {
|
||||||
"adp_payroll_flat": "",
|
|
||||||
"adp_payroll_straight": "",
|
|
||||||
"anticipated_revenue": "",
|
"anticipated_revenue": "",
|
||||||
"ar_aging": "",
|
"ar_aging": "",
|
||||||
"attendance_detail": "",
|
"attendance_detail": "",
|
||||||
@@ -3431,18 +3418,6 @@
|
|||||||
"vehicledetail": "Detalles del vehículo {{vehicle}} | {{app}}",
|
"vehicledetail": "Detalles del vehículo {{vehicle}} | {{app}}",
|
||||||
"vehicles": "Todos los vehiculos | {{app}}"
|
"vehicles": "Todos los vehiculos | {{app}}"
|
||||||
},
|
},
|
||||||
"trello": {
|
|
||||||
"labels": {
|
|
||||||
"add_card": "",
|
|
||||||
"add_lane": "",
|
|
||||||
"cancel": "",
|
|
||||||
"delete_lane": "",
|
|
||||||
"description": "",
|
|
||||||
"label": "",
|
|
||||||
"lane_actions": "",
|
|
||||||
"title": ""
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"tt_approvals": {
|
"tt_approvals": {
|
||||||
"actions": {
|
"actions": {
|
||||||
"approveselected": ""
|
"approveselected": ""
|
||||||
@@ -3581,6 +3556,18 @@
|
|||||||
"validation": {
|
"validation": {
|
||||||
"unique_vendor_name": ""
|
"unique_vendor_name": ""
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"trello": {
|
||||||
|
"labels": {
|
||||||
|
"add_card": "",
|
||||||
|
"add_lane": "",
|
||||||
|
"delete_lane": "",
|
||||||
|
"lane_actions": "",
|
||||||
|
"title": "",
|
||||||
|
"description": "",
|
||||||
|
"label": "",
|
||||||
|
"cancel": ""
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -2,9 +2,9 @@ import axios from "axios";
|
|||||||
import { auth } from "../firebase/firebase.utils";
|
import { auth } from "../firebase/firebase.utils";
|
||||||
import InstanceRenderManager from "./instanceRenderMgr";
|
import InstanceRenderManager from "./instanceRenderMgr";
|
||||||
|
|
||||||
axios.defaults.baseURL = import.meta.env.DEV
|
axios.defaults.baseURL =
|
||||||
? "/api/"
|
import.meta.env.VITE_APP_AXIOS_BASE_API_URL ||
|
||||||
: import.meta.env.VITE_APP_AXIOS_BASE_API_URL || "https://api.imex.online/";
|
(import.meta.env.MODE === "production" ? "https://api.imex.online/" : "http://localhost:4000/");
|
||||||
|
|
||||||
export const axiosAuthInterceptorId = axios.interceptors.request.use(
|
export const axiosAuthInterceptorId = axios.interceptors.request.use(
|
||||||
async (config) => {
|
async (config) => {
|
||||||
|
|||||||
@@ -2158,32 +2158,6 @@ export const TemplateList = (type, context) => {
|
|||||||
field: i18n.t("tasks.fields.created_at")
|
field: i18n.t("tasks.fields.created_at")
|
||||||
},
|
},
|
||||||
group: "jobs"
|
group: "jobs"
|
||||||
},
|
|
||||||
adp_payroll_flat: {
|
|
||||||
title: i18n.t("reportcenter.templates.adp_payroll_flat"),
|
|
||||||
subject: i18n.t("reportcenter.templates.adp_payroll_flat"),
|
|
||||||
key: "adp_payroll_flat",
|
|
||||||
reporttype: "text",
|
|
||||||
disabled: false,
|
|
||||||
rangeFilter: {
|
|
||||||
object: i18n.t("reportcenter.labels.objects.timetickets"),
|
|
||||||
field: i18n.t("timetickets.fields.committed_at")
|
|
||||||
},
|
|
||||||
group: "payroll",
|
|
||||||
adp_payroll: true
|
|
||||||
},
|
|
||||||
adp_payroll_straight: {
|
|
||||||
title: i18n.t("reportcenter.templates.adp_payroll_straight"),
|
|
||||||
subject: i18n.t("reportcenter.templates.adp_payroll_straight"),
|
|
||||||
key: "adp_payroll_straight",
|
|
||||||
reporttype: "text",
|
|
||||||
disabled: false,
|
|
||||||
rangeFilter: {
|
|
||||||
object: i18n.t("reportcenter.labels.objects.timetickets"),
|
|
||||||
field: i18n.t("timetickets.fields.date")
|
|
||||||
},
|
|
||||||
group: "payroll",
|
|
||||||
adp_payroll: true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
: {}),
|
: {}),
|
||||||
|
|||||||
@@ -1,84 +0,0 @@
|
|||||||
import React from "react";
|
|
||||||
|
|
||||||
const useCountDown = (timeToCount = 60 * 1000, interval = 1000) => {
|
|
||||||
const [timeLeft, setTimeLeft] = React.useState(0);
|
|
||||||
const timer = React.useRef({});
|
|
||||||
|
|
||||||
const run = (ts) => {
|
|
||||||
if (!timer.current.started) {
|
|
||||||
timer.current.started = ts;
|
|
||||||
timer.current.lastInterval = ts;
|
|
||||||
}
|
|
||||||
|
|
||||||
const localInterval = Math.min(interval, timer.current.timeLeft || Infinity);
|
|
||||||
if (ts - timer.current.lastInterval >= localInterval) {
|
|
||||||
timer.current.lastInterval += localInterval;
|
|
||||||
setTimeLeft((timeLeft) => {
|
|
||||||
timer.current.timeLeft = timeLeft - localInterval;
|
|
||||||
return timer.current.timeLeft;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ts - timer.current.started < timer.current.timeToCount) {
|
|
||||||
timer.current.requestId = window.requestAnimationFrame(run);
|
|
||||||
} else {
|
|
||||||
timer.current = {};
|
|
||||||
setTimeLeft(0);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const start = React.useCallback(
|
|
||||||
(ttc) => {
|
|
||||||
window.cancelAnimationFrame(timer.current.requestId);
|
|
||||||
|
|
||||||
const newTimeToCount = ttc !== undefined ? ttc : timeToCount;
|
|
||||||
timer.current.started = null;
|
|
||||||
timer.current.lastInterval = null;
|
|
||||||
timer.current.timeToCount = newTimeToCount;
|
|
||||||
timer.current.requestId = window.requestAnimationFrame(run);
|
|
||||||
|
|
||||||
setTimeLeft(newTimeToCount);
|
|
||||||
},
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const pause = React.useCallback(() => {
|
|
||||||
window.cancelAnimationFrame(timer.current.requestId);
|
|
||||||
timer.current.started = null;
|
|
||||||
timer.current.lastInterval = null;
|
|
||||||
timer.current.timeToCount = timer.current.timeLeft;
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const resume = React.useCallback(
|
|
||||||
() => {
|
|
||||||
if (!timer.current.started && timer.current.timeLeft > 0) {
|
|
||||||
window.cancelAnimationFrame(timer.current.requestId);
|
|
||||||
timer.current.requestId = window.requestAnimationFrame(run);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const reset = React.useCallback(() => {
|
|
||||||
if (timer.current.timeLeft) {
|
|
||||||
window.cancelAnimationFrame(timer.current.requestId);
|
|
||||||
timer.current = {};
|
|
||||||
setTimeLeft(0);
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const actions = React.useMemo(
|
|
||||||
() => ({ start, pause, resume, reset }), // eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
return () => window.cancelAnimationFrame(timer.current.requestId);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return [timeLeft, actions];
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useCountDown;
|
|
||||||
47
client/src/utils/handleBeta.js
Normal file
47
client/src/utils/handleBeta.js
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
export const BETA_KEY = "betaSwitchImex";
|
||||||
|
|
||||||
|
export const checkBeta = () => {
|
||||||
|
const cookie = document.cookie.split("; ").find((row) => row.startsWith(BETA_KEY));
|
||||||
|
return cookie ? cookie.split("=")[1] === "true" : false;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setBeta = (value) => {
|
||||||
|
const domain = window.location.hostname.split(".").slice(-2).join(".");
|
||||||
|
document.cookie = `${BETA_KEY}=${value}; path=/; domain=.${domain}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const handleBeta = () => {
|
||||||
|
if (window.location.hostname.startsWith("localhost")) {
|
||||||
|
console.log("Not on beta or test, so no need to handle beta.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const isBeta = checkBeta();
|
||||||
|
const currentHostName = window.location.hostname;
|
||||||
|
|
||||||
|
// Determine if the host name starts with "beta" or "www.beta"
|
||||||
|
const isBetaHost = currentHostName.startsWith("beta.");
|
||||||
|
const isBetaHostWithWWW = currentHostName.startsWith("www.beta.");
|
||||||
|
|
||||||
|
if (isBeta) {
|
||||||
|
// If beta is on and we are not on a beta domain, redirect to the beta version
|
||||||
|
if (!isBetaHost && !isBetaHostWithWWW) {
|
||||||
|
const newHostName = currentHostName.startsWith("www.")
|
||||||
|
? `www.beta.${currentHostName.replace(/^www\./, "")}`
|
||||||
|
: `beta.${currentHostName}`;
|
||||||
|
const href = `${window.location.protocol}//${newHostName}${window.location.pathname}${window.location.search}${window.location.hash}`;
|
||||||
|
window.location.replace(href);
|
||||||
|
}
|
||||||
|
// Otherwise, if beta is on and we're already on a beta domain, stay there
|
||||||
|
} else {
|
||||||
|
// If beta is off and we are on a beta domain, redirect to the non-beta version
|
||||||
|
if (isBetaHost || isBetaHostWithWWW) {
|
||||||
|
const newHostName = currentHostName.replace(/^www\.beta\./, "www.").replace(/^beta\./, "");
|
||||||
|
const href = `${window.location.protocol}//${newHostName}${window.location.pathname}${window.location.search}${window.location.hash}`;
|
||||||
|
window.location.replace(href);
|
||||||
|
}
|
||||||
|
// Otherwise, if beta is off and we're not on a beta domain, stay there
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export default handleBeta;
|
||||||
@@ -3,22 +3,16 @@ import { promises as fsPromises } from "fs";
|
|||||||
import { createRequire } from "module";
|
import { createRequire } from "module";
|
||||||
import * as path from "path";
|
import * as path from "path";
|
||||||
import * as url from "url";
|
import * as url from "url";
|
||||||
import { createLogger, defineConfig } from "vite";
|
import { defineConfig } from "vite";
|
||||||
import { ViteEjsPlugin } from "vite-plugin-ejs";
|
import { ViteEjsPlugin } from "vite-plugin-ejs";
|
||||||
import eslint from "vite-plugin-eslint";
|
import eslint from "vite-plugin-eslint";
|
||||||
import { VitePWA } from "vite-plugin-pwa";
|
import { VitePWA } from "vite-plugin-pwa";
|
||||||
import InstanceRenderManager from "./src/utils/instanceRenderMgr";
|
import InstanceRenderManager from "./src/utils/instanceRenderMgr";
|
||||||
import chalk from "chalk";
|
|
||||||
//import { visualizer } from "rollup-plugin-visualizer";
|
|
||||||
|
|
||||||
process.env.VITE_APP_GIT_SHA_DATE = new Date().toLocaleString("en-US", {
|
process.env.VITE_APP_GIT_SHA_DATE = new Date().toLocaleString("en-US", {
|
||||||
timeZone: "America/Los_Angeles"
|
timeZone: "America/Los_Angeles"
|
||||||
});
|
});
|
||||||
|
|
||||||
const getFormattedTimestamp = () =>
|
|
||||||
new Date().toLocaleTimeString("en-US", { hour12: true }).replace("AM", "a.m.").replace("PM", "p.m.");
|
|
||||||
|
|
||||||
/** This is a hack around react-virtualized, should be removed when switching to react-virtuoso */
|
|
||||||
const WRONG_CODE = `import { bpfrpt_proptype_WindowScroller } from "../WindowScroller.js";`;
|
const WRONG_CODE = `import { bpfrpt_proptype_WindowScroller } from "../WindowScroller.js";`;
|
||||||
|
|
||||||
function reactVirtualizedFix() {
|
function reactVirtualizedFix() {
|
||||||
@@ -38,16 +32,10 @@ function reactVirtualizedFix() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
/** End of hack */
|
|
||||||
|
|
||||||
export const logger = createLogger("info", {
|
|
||||||
allowClearScreen: false
|
|
||||||
});
|
|
||||||
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
base: "/",
|
base: "/",
|
||||||
plugins: [
|
plugins: [
|
||||||
//visualizer(),
|
|
||||||
ViteEjsPlugin((viteConfig) => ({ env: viteConfig.env })),
|
ViteEjsPlugin((viteConfig) => ({ env: viteConfig.env })),
|
||||||
VitePWA({
|
VitePWA({
|
||||||
injectRegister: "auto",
|
injectRegister: "auto",
|
||||||
@@ -111,6 +99,7 @@ export default defineConfig({
|
|||||||
reactVirtualizedFix(),
|
reactVirtualizedFix(),
|
||||||
react(),
|
react(),
|
||||||
eslint()
|
eslint()
|
||||||
|
// CompressionPlugin(), //Cloudfront already compresses assets, so not needed.
|
||||||
],
|
],
|
||||||
define: {
|
define: {
|
||||||
APP_VERSION: JSON.stringify(process.env.npm_package_version)
|
APP_VERSION: JSON.stringify(process.env.npm_package_version)
|
||||||
@@ -118,69 +107,7 @@ export default defineConfig({
|
|||||||
server: {
|
server: {
|
||||||
host: true,
|
host: true,
|
||||||
port: 3000,
|
port: 3000,
|
||||||
open: true,
|
open: true
|
||||||
proxy: {
|
|
||||||
"/ws": {
|
|
||||||
target: "ws://localhost:4000",
|
|
||||||
rewriteWsOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
ws: true
|
|
||||||
},
|
|
||||||
"/wss": {
|
|
||||||
target: "ws://localhost:4000",
|
|
||||||
rewriteWsOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
ws: true
|
|
||||||
},
|
|
||||||
"/api": {
|
|
||||||
target: "http://localhost:4000",
|
|
||||||
changeOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
ws: false,
|
|
||||||
rewrite: (path) => {
|
|
||||||
const replacedValue = path.replace(/^\/api/, "");
|
|
||||||
logger.info(
|
|
||||||
`${chalk.grey.bold(getFormattedTimestamp())} ${chalk.cyan.bold("[vite]")} ${chalk.green.bold("[API]")} ${chalk.blue(replacedValue)}`
|
|
||||||
);
|
|
||||||
return replacedValue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
https: {
|
|
||||||
key: await fsPromises.readFile("../certs/key.pem"),
|
|
||||||
cert: await fsPromises.readFile("../certs/cert.pem"),
|
|
||||||
allowHTTP1: false // Force HTTP/2
|
|
||||||
}
|
|
||||||
},
|
|
||||||
preview: {
|
|
||||||
port: 6000,
|
|
||||||
host: true,
|
|
||||||
open: true,
|
|
||||||
https: {
|
|
||||||
key: await fsPromises.readFile("../certs/key.pem"),
|
|
||||||
cert: await fsPromises.readFile("../certs/cert.pem"),
|
|
||||||
allowHTTP1: false // Force HTTP/2
|
|
||||||
},
|
|
||||||
proxy: {
|
|
||||||
"/ws": {
|
|
||||||
target: "ws://localhost:4000",
|
|
||||||
rewriteWsOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
ws: true
|
|
||||||
},
|
|
||||||
"/wss": {
|
|
||||||
target: "ws://localhost:4000",
|
|
||||||
rewriteWsOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
ws: true
|
|
||||||
},
|
|
||||||
"/api": {
|
|
||||||
target: "http://localhost:4000",
|
|
||||||
changeOrigin: true,
|
|
||||||
secure: false,
|
|
||||||
ws: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
build: {
|
build: {
|
||||||
rollupOptions: {
|
rollupOptions: {
|
||||||
@@ -188,63 +115,17 @@ export default defineConfig({
|
|||||||
manualChunks: {
|
manualChunks: {
|
||||||
antd: ["antd"],
|
antd: ["antd"],
|
||||||
"react-redux": ["react-redux"],
|
"react-redux": ["react-redux"],
|
||||||
redux: ["redux"],
|
redux: ["redux"]
|
||||||
lodash: ["lodash"],
|
|
||||||
"@sentry/react": ["@sentry/react"],
|
|
||||||
"@splitsoftware/splitio-react": ["@splitsoftware/splitio-react"],
|
|
||||||
logrocket: ["logrocket"],
|
|
||||||
"firebase/app": ["firebase/app"],
|
|
||||||
"firebase/firestore": ["firebase/firestore"],
|
|
||||||
"firebase/firestore/lite": ["firebase/firestore/lite"],
|
|
||||||
"firebase/auth": ["firebase/auth"],
|
|
||||||
"firebase/functions": ["firebase/functions"],
|
|
||||||
"firebase/storage": ["firebase/storage"],
|
|
||||||
"firebase/database": ["firebase/database"],
|
|
||||||
"firebase/remote-config": ["firebase/remote-config"],
|
|
||||||
"firebase/performance": ["firebase/performance"],
|
|
||||||
"@firebase/app": ["@firebase/app"],
|
|
||||||
"@firebase/firestore": ["@firebase/firestore"],
|
|
||||||
"@firebase/firestore/lite": ["@firebase/firestore/lite"],
|
|
||||||
"@firebase/auth": ["@firebase/auth"],
|
|
||||||
"@firebase/functions": ["@firebase/functions"],
|
|
||||||
"@firebase/storage": ["@firebase/storage"],
|
|
||||||
"@firebase/database": ["@firebase/database"],
|
|
||||||
"@firebase/remote-config": ["@firebase/remote-config"],
|
|
||||||
"@firebase/performance": ["@firebase/performance"],
|
|
||||||
markerjs2: ["markerjs2"],
|
|
||||||
"@apollo/client": ["@apollo/client"],
|
|
||||||
"libphonenumber-js": ["libphonenumber-js"]
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
optimizeDeps: {
|
optimizeDeps: {
|
||||||
include: [
|
include: ["react", "react-dom", "antd", "@apollo/client", "@reduxjs/toolkit", "axios", "react-grid-layout"],
|
||||||
"react",
|
|
||||||
"react-dom",
|
|
||||||
"antd",
|
|
||||||
"lodash",
|
|
||||||
"@sentry/react",
|
|
||||||
"@apollo/client",
|
|
||||||
"@reduxjs/toolkit",
|
|
||||||
"axios",
|
|
||||||
"react-router-dom",
|
|
||||||
"dayjs",
|
|
||||||
"redux",
|
|
||||||
"react-redux"
|
|
||||||
],
|
|
||||||
esbuildOptions: {
|
esbuildOptions: {
|
||||||
loader: {
|
loader: {
|
||||||
".js": "jsx"
|
".js": "jsx"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
|
||||||
css: {
|
|
||||||
preprocessorOptions: {
|
|
||||||
scss: {
|
|
||||||
api: "modern-compiler",
|
|
||||||
quietDeps: true // Quite Deprecation Warnings, should be disabled occasionally before major upgrades
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
Must set the environment variables using:
|
Must set the environment variables using:
|
||||||
|
|
||||||
firebase functions:config:set auth.graphql_endpoint="https://db.dev.imex.online/v1/graphql"
|
firebase functions:config:set auth.graphql_endpoint="https://db.dev.bodyshop.app/v1/graphql"
|
||||||
auth.hasura_secret_admin_key="Dev-BodyShopApp!"
|
auth.hasura_secret_admin_key="Dev-BodyShopApp!"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
version: 2
|
version: 2
|
||||||
endpoint: https://db.dev.imex.online
|
endpoint: https://db.dev.bodyshop.app
|
||||||
admin_secret: Dev-BodyShopApp!
|
admin_secret: Dev-BodyShopApp!
|
||||||
metadata_directory: metadata
|
metadata_directory: metadata
|
||||||
actions:
|
actions:
|
||||||
|
|||||||
@@ -918,7 +918,6 @@
|
|||||||
- bill_tax_rates
|
- bill_tax_rates
|
||||||
- cdk_configuration
|
- cdk_configuration
|
||||||
- cdk_dealerid
|
- cdk_dealerid
|
||||||
- chatterid
|
|
||||||
- city
|
- city
|
||||||
- claimscorpid
|
- claimscorpid
|
||||||
- convenient_company
|
- convenient_company
|
||||||
@@ -940,7 +939,6 @@
|
|||||||
- inhousevendorid
|
- inhousevendorid
|
||||||
- insurance_vendor_id
|
- insurance_vendor_id
|
||||||
- intakechecklist
|
- intakechecklist
|
||||||
- intellipay_config
|
|
||||||
- jc_hourly_rates
|
- jc_hourly_rates
|
||||||
- jobsizelimit
|
- jobsizelimit
|
||||||
- last_name_first
|
- last_name_first
|
||||||
@@ -1042,7 +1040,6 @@
|
|||||||
- inhousevendorid
|
- inhousevendorid
|
||||||
- insurance_vendor_id
|
- insurance_vendor_id
|
||||||
- intakechecklist
|
- intakechecklist
|
||||||
- intellipay_config
|
|
||||||
- jc_hourly_rates
|
- jc_hourly_rates
|
||||||
- last_name_first
|
- last_name_first
|
||||||
- localmediaserverhttp
|
- localmediaserverhttp
|
||||||
@@ -4243,63 +4240,6 @@
|
|||||||
- active:
|
- active:
|
||||||
_eq: true
|
_eq: true
|
||||||
event_triggers:
|
event_triggers:
|
||||||
- name: job_modified
|
|
||||||
definition:
|
|
||||||
enable_manual: false
|
|
||||||
update:
|
|
||||||
columns:
|
|
||||||
- clm_no
|
|
||||||
- v_make_desc
|
|
||||||
- date_next_contact
|
|
||||||
- status
|
|
||||||
- employee_csr
|
|
||||||
- employee_prep
|
|
||||||
- clm_total
|
|
||||||
- suspended
|
|
||||||
- employee_body
|
|
||||||
- ro_number
|
|
||||||
- actual_in
|
|
||||||
- ownr_co_nm
|
|
||||||
- v_model_yr
|
|
||||||
- comment
|
|
||||||
- job_totals
|
|
||||||
- v_vin
|
|
||||||
- ownr_fn
|
|
||||||
- scheduled_completion
|
|
||||||
- special_coverage_policy
|
|
||||||
- v_color
|
|
||||||
- ca_gst_registrant
|
|
||||||
- scheduled_delivery
|
|
||||||
- actual_delivery
|
|
||||||
- actual_completion
|
|
||||||
- kanbanparent
|
|
||||||
- est_ct_fn
|
|
||||||
- employee_refinish
|
|
||||||
- ownr_ph1
|
|
||||||
- date_last_contacted
|
|
||||||
- alt_transport
|
|
||||||
- inproduction
|
|
||||||
- est_ct_ln
|
|
||||||
- production_vars
|
|
||||||
- category
|
|
||||||
- v_model_desc
|
|
||||||
- date_invoiced
|
|
||||||
- est_co_nm
|
|
||||||
- ownr_ln
|
|
||||||
retry_conf:
|
|
||||||
interval_sec: 10
|
|
||||||
num_retries: 0
|
|
||||||
timeout_sec: 60
|
|
||||||
webhook_from_env: HASURA_API_URL
|
|
||||||
headers:
|
|
||||||
- name: event-secret
|
|
||||||
value_from_env: EVENT_SECRET
|
|
||||||
request_transform:
|
|
||||||
method: POST
|
|
||||||
query_params: {}
|
|
||||||
template_engine: Kriti
|
|
||||||
url: '{{$base_url}}/job/job-updated'
|
|
||||||
version: 2
|
|
||||||
- name: job_status_transition
|
- name: job_status_transition
|
||||||
definition:
|
definition:
|
||||||
enable_manual: true
|
enable_manual: true
|
||||||
@@ -4359,35 +4299,6 @@
|
|||||||
template_engine: Kriti
|
template_engine: Kriti
|
||||||
url: '{{$base_url}}/opensearch'
|
url: '{{$base_url}}/opensearch'
|
||||||
version: 2
|
version: 2
|
||||||
- table:
|
|
||||||
name: jobs_inproduction
|
|
||||||
schema: public
|
|
||||||
object_relationships:
|
|
||||||
- name: bodyshop
|
|
||||||
using:
|
|
||||||
manual_configuration:
|
|
||||||
column_mapping:
|
|
||||||
shopid: id
|
|
||||||
insertion_order: null
|
|
||||||
remote_table:
|
|
||||||
name: bodyshops
|
|
||||||
schema: public
|
|
||||||
select_permissions:
|
|
||||||
- role: user
|
|
||||||
permission:
|
|
||||||
columns:
|
|
||||||
- id
|
|
||||||
- shopid
|
|
||||||
- updated_at
|
|
||||||
filter:
|
|
||||||
bodyshop:
|
|
||||||
associations:
|
|
||||||
_and:
|
|
||||||
- user:
|
|
||||||
authid:
|
|
||||||
_eq: X-Hasura-User-Id
|
|
||||||
- active:
|
|
||||||
_eq: true
|
|
||||||
- table:
|
- table:
|
||||||
name: masterdata
|
name: masterdata
|
||||||
schema: public
|
schema: public
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
-- Could not auto-generate a down migration.
|
|
||||||
-- Please write an appropriate down migration for the SQL below:
|
|
||||||
-- alter table "public"."bodyshops" add column "intellipay_config" jsonb
|
|
||||||
-- not null default jsonb_build_object();
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
alter table "public"."bodyshops" add column "intellipay_config" jsonb
|
|
||||||
not null default jsonb_build_object();
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
-- Could not auto-generate a down migration.
|
|
||||||
-- Please write an appropriate down migration for the SQL below:
|
|
||||||
-- alter table "public"."bodyshops" add column "chatterid" text
|
|
||||||
-- null;
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
alter table "public"."bodyshops" add column "chatterid" text
|
|
||||||
null;
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "courtesycars_idx_fleet" on
|
|
||||||
"public"."courtesycars" using btree ("fleetnumber");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."courtesycars_idx_fleet";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_ownrfn" on
|
|
||||||
"public"."jobs" using gin ("ownr_fn");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_ownrfn";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_ownrln" on
|
|
||||||
"public"."jobs" using gin ("ownr_ln");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_ownrln";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "jobs_idx_iouparent" on
|
|
||||||
"public"."jobs" using btree ("iouparent");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."jobs_idx_iouparent";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_ronumber" on
|
|
||||||
"public"."jobs" using gin ("ro_number");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_ronumber";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_clmno" on
|
|
||||||
"public"."jobs" using gin ("clm_no");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_clmno";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_vmodeldesc" on
|
|
||||||
"public"."jobs" using gin ("v_model_desc");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_vmodeldesc";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_vmakedesc" on
|
|
||||||
"public"."jobs" using gin ("v_make_desc");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_vmakedesc";
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
CREATE INDEX "idx_jobs_plateno" on
|
|
||||||
"public"."jobs" using gin ("plate_no");
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP INDEX IF EXISTS "public"."idx_jobs_plateno";
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
-- Could not auto-generate a down migration.
|
|
||||||
-- Please write an appropriate down migration for the SQL below:
|
|
||||||
-- CREATE
|
|
||||||
-- OR REPLACE VIEW "public"."jobs_inproduction" AS
|
|
||||||
-- SELECT
|
|
||||||
-- j.id,
|
|
||||||
-- j.updated_at
|
|
||||||
-- FROM
|
|
||||||
-- jobs j
|
|
||||||
-- WHERE
|
|
||||||
-- j.inproduction=true;
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
CREATE
|
|
||||||
OR REPLACE VIEW "public"."jobs_inproduction" AS
|
|
||||||
SELECT
|
|
||||||
j.id,
|
|
||||||
j.updated_at
|
|
||||||
FROM
|
|
||||||
jobs j
|
|
||||||
WHERE
|
|
||||||
j.inproduction=true;
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
-- Could not auto-generate a down migration.
|
|
||||||
-- Please write an appropriate down migration for the SQL below:
|
|
||||||
-- CREATE OR REPLACE VIEW "public"."jobs_inproduction" AS
|
|
||||||
-- SELECT j.id,
|
|
||||||
-- j.updated_at,
|
|
||||||
-- j.shopid
|
|
||||||
-- FROM jobs j
|
|
||||||
-- WHERE (j.inproduction = true);
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
CREATE OR REPLACE VIEW "public"."jobs_inproduction" AS
|
|
||||||
SELECT j.id,
|
|
||||||
j.updated_at,
|
|
||||||
j.shopid
|
|
||||||
FROM jobs j
|
|
||||||
WHERE (j.inproduction = true);
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
-- Could not auto-generate a down migration.
|
|
||||||
-- Please write an appropriate down migration for the SQL below:
|
|
||||||
-- CREATE OR REPLACE VIEW "public"."jobs_inproduction" AS
|
|
||||||
-- SELECT j.id,
|
|
||||||
-- j.updated_at,
|
|
||||||
-- j.shopid
|
|
||||||
-- FROM jobs j
|
|
||||||
-- WHERE (j.inproduction = true);
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
CREATE OR REPLACE VIEW "public"."jobs_inproduction" AS
|
|
||||||
SELECT j.id,
|
|
||||||
j.updated_at,
|
|
||||||
j.shopid
|
|
||||||
FROM jobs j
|
|
||||||
WHERE (j.inproduction = true);
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user