Merge branch 'dev' of https://gitea.verdnatura.es/verdnatura/salix into 3750-claim_photo
gitea/salix/pipeline/head This commit looks good
Details
gitea/salix/pipeline/head This commit looks good
Details
This commit is contained in:
commit
579ff3aa4a
|
@ -7,7 +7,7 @@ RUN apt-get update \
|
||||||
curl \
|
curl \
|
||||||
ca-certificates \
|
ca-certificates \
|
||||||
gnupg2 \
|
gnupg2 \
|
||||||
libfontconfig \
|
libfontconfig lftp \
|
||||||
&& apt-get -y install xvfb gconf-service libasound2 libatk1.0-0 libc6 libcairo2 libcups2 \
|
&& apt-get -y install xvfb gconf-service libasound2 libatk1.0-0 libc6 libcairo2 libcups2 \
|
||||||
libdbus-1-3 libexpat1 libfontconfig1 libgbm1 libgcc1 libgconf-2-4 libgdk-pixbuf2.0-0 libglib2.0-0 \
|
libdbus-1-3 libexpat1 libfontconfig1 libgbm1 libgcc1 libgconf-2-4 libgdk-pixbuf2.0-0 libglib2.0-0 \
|
||||||
libgtk-3-0 libnspr4 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 \
|
libgtk-3-0 libnspr4 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 \
|
||||||
|
|
|
@ -23,7 +23,13 @@ module.exports = Self => {
|
||||||
let models = Self.app.models;
|
let models = Self.app.models;
|
||||||
|
|
||||||
let user = await models.Account.findById(userId, {
|
let user = await models.Account.findById(userId, {
|
||||||
fields: ['id', 'name', 'nickname', 'email']
|
fields: ['id', 'name', 'nickname', 'email', 'lang'],
|
||||||
|
include: {
|
||||||
|
relation: 'userConfig',
|
||||||
|
scope: {
|
||||||
|
fields: ['darkMode']
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let roles = await models.RoleMapping.find({
|
let roles = await models.RoleMapping.find({
|
||||||
|
|
|
@ -13,10 +13,9 @@ describe('Chat notifyIssue()', () => {
|
||||||
spyOn(chatModel, 'send').and.callThrough();
|
spyOn(chatModel, 'send').and.callThrough();
|
||||||
spyOn(osTicketModel, 'rawSql').and.returnValue([]);
|
spyOn(osTicketModel, 'rawSql').and.returnValue([]);
|
||||||
|
|
||||||
const response = await chatModel.notifyIssues(ctx);
|
await chatModel.notifyIssues(ctx);
|
||||||
|
|
||||||
expect(chatModel.send).not.toHaveBeenCalled();
|
expect(chatModel.send).not.toHaveBeenCalled();
|
||||||
expect(response).toBeUndefined();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it(`should return a response calling the send() method`, async() => {
|
it(`should return a response calling the send() method`, async() => {
|
||||||
|
@ -27,16 +26,15 @@ describe('Chat notifyIssue()', () => {
|
||||||
username: 'batman',
|
username: 'batman',
|
||||||
subject: 'Issue title'}
|
subject: 'Issue title'}
|
||||||
]);
|
]);
|
||||||
|
// eslint-disable-next-line max-len
|
||||||
const expectedMessage = `@all ➔ There's a new urgent ticket:\r\n[ID: *00001* - Issue title (@batman)](https://cau.verdnatura.es/scp/tickets.php?id=1)`;
|
const expectedMessage = `@all ➔ There's a new urgent ticket:\r\n[ID: *00001* - Issue title (@batman)](https://cau.verdnatura.es/scp/tickets.php?id=1)`;
|
||||||
|
|
||||||
const department = await app.models.Department.findById(departmentId);
|
const department = await app.models.Department.findById(departmentId);
|
||||||
let orgChatName = department.chatName;
|
let orgChatName = department.chatName;
|
||||||
await department.updateAttribute('chatName', 'IT');
|
await department.updateAttribute('chatName', 'IT');
|
||||||
|
|
||||||
const response = await chatModel.notifyIssues(ctx);
|
await chatModel.notifyIssues(ctx);
|
||||||
|
|
||||||
expect(response.statusCode).toEqual(200);
|
|
||||||
expect(response.message).toEqual('Fake notification sent');
|
|
||||||
expect(chatModel.send).toHaveBeenCalledWith(ctx, '#IT', expectedMessage);
|
expect(chatModel.send).toHaveBeenCalledWith(ctx, '#IT', expectedMessage);
|
||||||
|
|
||||||
// restores
|
// restores
|
||||||
|
|
|
@ -5,14 +5,13 @@ describe('Chat send()', () => {
|
||||||
let ctx = {req: {accessToken: {userId: 1}}};
|
let ctx = {req: {accessToken: {userId: 1}}};
|
||||||
let response = await app.models.Chat.send(ctx, '@salesPerson', 'I changed something');
|
let response = await app.models.Chat.send(ctx, '@salesPerson', 'I changed something');
|
||||||
|
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response).toEqual(true);
|
||||||
expect(response.message).toEqual('Fake notification sent');
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should retrun false as response', async() => {
|
it('should retrun false as response', async() => {
|
||||||
let ctx = {req: {accessToken: {userId: 18}}};
|
let ctx = {req: {accessToken: {userId: 18}}};
|
||||||
let response = await app.models.Chat.send(ctx, '@salesPerson', 'I changed something');
|
let response = await app.models.Chat.send(ctx, '@salesPerson', 'I changed something');
|
||||||
|
|
||||||
expect(response).toBeFalsy();
|
expect(response).toEqual(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,10 +20,8 @@ describe('Chat sendCheckingPresence()', () => {
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
const response = await chatModel.sendCheckingPresence(ctx, workerId, 'I changed something');
|
await chatModel.sendCheckingPresence(ctx, workerId, 'I changed something');
|
||||||
|
|
||||||
expect(response.statusCode).toEqual(200);
|
|
||||||
expect(response.message).toEqual('Fake notification sent');
|
|
||||||
expect(chatModel.send).toHaveBeenCalledWith(ctx, '@HankPym', 'I changed something');
|
expect(chatModel.send).toHaveBeenCalledWith(ctx, '@HankPym', 'I changed something');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -47,10 +45,8 @@ describe('Chat sendCheckingPresence()', () => {
|
||||||
const department = await models.Department.findById(departmentId, null, options);
|
const department = await models.Department.findById(departmentId, null, options);
|
||||||
await department.updateAttribute('chatName', 'cooler');
|
await department.updateAttribute('chatName', 'cooler');
|
||||||
|
|
||||||
const response = await chatModel.sendCheckingPresence(ctx, workerId, 'I changed something');
|
await chatModel.sendCheckingPresence(ctx, workerId, 'I changed something');
|
||||||
|
|
||||||
expect(response.statusCode).toEqual(200);
|
|
||||||
expect(response.message).toEqual('Fake notification sent');
|
|
||||||
expect(chatModel.send).toHaveBeenCalledWith(ctx, '#cooler', '@HankPym ➔ I changed something');
|
expect(chatModel.send).toHaveBeenCalledWith(ctx, '#cooler', '@HankPym ➔ I changed something');
|
||||||
|
|
||||||
await tx.rollback();
|
await tx.rollback();
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE bucket
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7, @col8, @col9, @col10, @col11, @col12)
|
||||||
|
SET
|
||||||
|
bucket_id = @col2,
|
||||||
|
bucket_type_id = @col4,
|
||||||
|
description = @col5,
|
||||||
|
x_size = @col6,
|
||||||
|
y_size = @col7,
|
||||||
|
z_size = @col8,
|
||||||
|
entry_date = STR_TO_DATE(@col10, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col11, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col12, '%Y%m%d%H%i')
|
|
@ -0,0 +1,10 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE bucket_type
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6)
|
||||||
|
SET
|
||||||
|
bucket_type_id = @col2,
|
||||||
|
description = @col3,
|
||||||
|
entry_date = STR_TO_DATE(@col4, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col5, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col6, '%Y%m%d%H%i')
|
|
@ -0,0 +1,11 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE `feature`
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7)
|
||||||
|
SET
|
||||||
|
item_id = @col2,
|
||||||
|
feature_type_id = @col3,
|
||||||
|
feature_value = @col4,
|
||||||
|
entry_date = STR_TO_DATE(@col5, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col6, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col7, '%Y%m%d%H%i')
|
|
@ -0,0 +1,10 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE genus
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6)
|
||||||
|
SET
|
||||||
|
genus_id = @col2,
|
||||||
|
latin_genus_name = @col3,
|
||||||
|
entry_date = STR_TO_DATE(@col4, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col5, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col6, '%Y%m%d%H%i')
|
|
@ -0,0 +1,13 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE item
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7, @col8, @col9, @col10, @col11, @col12)
|
||||||
|
SET
|
||||||
|
id = @col2,
|
||||||
|
product_name = @col4,
|
||||||
|
name = @col5,
|
||||||
|
plant_id = @col7,
|
||||||
|
group_id = @col9,
|
||||||
|
entry_date = STR_TO_DATE(@col10, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col11, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col12, '%Y%m%d%H%i')
|
|
@ -0,0 +1,12 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE `item_feature`
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7, @col8)
|
||||||
|
SET
|
||||||
|
item_id = @col2,
|
||||||
|
feature = @col3,
|
||||||
|
regulation_type = @col4,
|
||||||
|
presentation_order = @col5,
|
||||||
|
entry_date = STR_TO_DATE(@col6, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col7, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col8, '%Y%m%d%H%i')
|
|
@ -0,0 +1,10 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE item_group
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6)
|
||||||
|
SET
|
||||||
|
group_code = @col2,
|
||||||
|
dutch_group_description = @col3,
|
||||||
|
entry_date = STR_TO_DATE(@col4, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col5, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col6, '%Y%m%d%H%i')
|
|
@ -0,0 +1,11 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE plant
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7, @col8, @col9)
|
||||||
|
SET
|
||||||
|
plant_id = @col3,
|
||||||
|
genus_id = @col4,
|
||||||
|
specie_id = @col5,
|
||||||
|
entry_date = STR_TO_DATE(@col7, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col8, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col9, '%Y%m%d%H%i')
|
|
@ -0,0 +1,11 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE specie
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7)
|
||||||
|
SET
|
||||||
|
specie_id = @col2,
|
||||||
|
genus_id = @col3,
|
||||||
|
latin_species_name = @col4,
|
||||||
|
entry_date = STR_TO_DATE(@col5, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col6, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col7, '%Y%m%d%H%i')
|
|
@ -0,0 +1,11 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE edi.supplier
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7, @col8, @col9, @col10, @col11, @col12, @col13, @col14, @col15, @col16, @col17, @col18, @col19, @col20)
|
||||||
|
SET
|
||||||
|
GLNAddressCode = @col2,
|
||||||
|
supplier_id = @col4,
|
||||||
|
company_name = @col3,
|
||||||
|
entry_date = STR_TO_DATE(@col9, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col10, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col11, '%Y%m%d%H%i')
|
|
@ -0,0 +1,11 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE `type`
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7)
|
||||||
|
SET
|
||||||
|
type_id = @col2,
|
||||||
|
type_group_id = @col3,
|
||||||
|
description = @col4,
|
||||||
|
entry_date = STR_TO_DATE(@col5, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col6, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col7, '%Y%m%d%H%i')
|
|
@ -0,0 +1,11 @@
|
||||||
|
LOAD DATA LOCAL INFILE ?
|
||||||
|
INTO TABLE `value`
|
||||||
|
FIELDS TERMINATED BY ';'
|
||||||
|
LINES TERMINATED BY '\n' (@col1, @col2, @col3, @col4, @col5, @col6, @col7)
|
||||||
|
SET
|
||||||
|
type_id = @col2,
|
||||||
|
type_value = @col3,
|
||||||
|
type_description = @col4,
|
||||||
|
entry_date = STR_TO_DATE(@col5, '%Y%m%d'),
|
||||||
|
expiry_date = IFNULL(NULL,STR_TO_DATE(@col6, '%Y%m%d')),
|
||||||
|
change_date_time = STR_TO_DATE(@col7, '%Y%m%d%H%i')
|
|
@ -0,0 +1,155 @@
|
||||||
|
/* eslint no-console: "off" */
|
||||||
|
const path = require('path');
|
||||||
|
const fs = require('fs-extra');
|
||||||
|
|
||||||
|
module.exports = Self => {
|
||||||
|
Self.remoteMethodCtx('updateData', {
|
||||||
|
description: 'Updates schema data from external provider',
|
||||||
|
accessType: 'WRITE',
|
||||||
|
returns: {
|
||||||
|
type: 'object',
|
||||||
|
root: true
|
||||||
|
},
|
||||||
|
http: {
|
||||||
|
path: `/updateData`,
|
||||||
|
verb: 'GET'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Self.updateData = async() => {
|
||||||
|
const models = Self.app.models;
|
||||||
|
|
||||||
|
const container = await models.TempContainer.container('edi');
|
||||||
|
const tempPath = path.join(container.client.root, container.name);
|
||||||
|
|
||||||
|
const [ftpConfig] = await Self.rawSql('SELECT host, user, password FROM edi.ftpConfig');
|
||||||
|
console.debug(`Openning FTP connection to ${ftpConfig.host}...\n`);
|
||||||
|
|
||||||
|
const FtpClient = require('ftps');
|
||||||
|
const ftpClient = new FtpClient({
|
||||||
|
host: ftpConfig.host,
|
||||||
|
username: ftpConfig.user,
|
||||||
|
password: ftpConfig.password,
|
||||||
|
procotol: 'ftp'
|
||||||
|
});
|
||||||
|
|
||||||
|
const files = await Self.rawSql('SELECT fileName, toTable, file, updated FROM edi.fileConfig');
|
||||||
|
|
||||||
|
let remoteFile;
|
||||||
|
let tempDir;
|
||||||
|
let tempFile;
|
||||||
|
for (const file of files) {
|
||||||
|
try {
|
||||||
|
const fileName = file.file;
|
||||||
|
|
||||||
|
console.debug(`Downloading file ${fileName}...`);
|
||||||
|
|
||||||
|
remoteFile = `codes/${fileName}.ZIP`;
|
||||||
|
tempDir = `${tempPath}/${fileName}`;
|
||||||
|
tempFile = `${tempPath}/${fileName}.zip`;
|
||||||
|
|
||||||
|
await extractFile({
|
||||||
|
ftpClient: ftpClient,
|
||||||
|
file: file,
|
||||||
|
paths: {
|
||||||
|
remoteFile: remoteFile,
|
||||||
|
tempDir: tempDir,
|
||||||
|
tempFile: tempFile
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (fs.existsSync(tempFile))
|
||||||
|
await fs.unlink(tempFile);
|
||||||
|
|
||||||
|
await fs.rmdir(tempDir, {recursive: true});
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
async function extractFile({ftpClient, file, paths}) {
|
||||||
|
// Download the zip file
|
||||||
|
ftpClient.get(paths.remoteFile, paths.tempFile);
|
||||||
|
|
||||||
|
// Execute download command
|
||||||
|
ftpClient.exec(async(err, response) => {
|
||||||
|
if (response.error) {
|
||||||
|
console.debug(`Error downloading file... ${response.error}`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const AdmZip = require('adm-zip');
|
||||||
|
const zip = new AdmZip(paths.tempFile);
|
||||||
|
const entries = zip.getEntries();
|
||||||
|
|
||||||
|
zip.extractAllTo(paths.tempDir, false);
|
||||||
|
|
||||||
|
if (fs.existsSync(paths.tempFile))
|
||||||
|
await fs.unlink(paths.tempFile);
|
||||||
|
|
||||||
|
await dumpData({file, entries, paths});
|
||||||
|
|
||||||
|
await fs.rmdir(paths.tempDir, {recursive: true});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function dumpData({file, entries, paths}) {
|
||||||
|
const toTable = file.toTable;
|
||||||
|
const baseName = file.fileName;
|
||||||
|
|
||||||
|
for (const zipEntry of entries) {
|
||||||
|
const entryName = zipEntry.entryName;
|
||||||
|
console.log(`Reading file ${entryName}...`);
|
||||||
|
|
||||||
|
const startIndex = (entryName.length - 10);
|
||||||
|
const endIndex = (entryName.length - 4);
|
||||||
|
const dateString = entryName.substring(startIndex, endIndex);
|
||||||
|
const lastUpdated = new Date();
|
||||||
|
|
||||||
|
// Format string date to a date object
|
||||||
|
let updated = null;
|
||||||
|
if (file.updated) {
|
||||||
|
updated = new Date(file.updated);
|
||||||
|
updated.setHours(0, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
lastUpdated.setFullYear(`20${dateString.substring(4, 6)}`);
|
||||||
|
lastUpdated.setMonth(parseInt(dateString.substring(2, 4)) - 1);
|
||||||
|
lastUpdated.setDate(dateString.substring(0, 2));
|
||||||
|
lastUpdated.setHours(0, 0, 0, 0);
|
||||||
|
|
||||||
|
if (updated && lastUpdated <= updated) {
|
||||||
|
console.debug(`Table ${toTable} already updated, skipping...`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Dumping data...');
|
||||||
|
const templatePath = path.join(__dirname, `./sql/${toTable}.sql`);
|
||||||
|
const sqlTemplate = fs.readFileSync(templatePath, 'utf8');
|
||||||
|
|
||||||
|
const rawPath = path.join(paths.tempDir, entryName);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const tx = await Self.beginTransaction({});
|
||||||
|
const options = {transaction: tx};
|
||||||
|
|
||||||
|
await Self.rawSql(`DELETE FROM edi.${toTable}`, null, options);
|
||||||
|
await Self.rawSql(sqlTemplate, [rawPath], options);
|
||||||
|
await Self.rawSql(`
|
||||||
|
UPDATE edi.fileConfig
|
||||||
|
SET updated = ?
|
||||||
|
WHERE fileName = ?
|
||||||
|
`, [lastUpdated, baseName], options);
|
||||||
|
|
||||||
|
tx.commit();
|
||||||
|
} catch (error) {
|
||||||
|
tx.rollback();
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`Updated table ${toTable}\n`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
|
@ -109,6 +109,9 @@
|
||||||
},
|
},
|
||||||
"OsTicket": {
|
"OsTicket": {
|
||||||
"dataSource": "osticket"
|
"dataSource": "osticket"
|
||||||
|
},
|
||||||
|
"Edi": {
|
||||||
|
"dataSource": "vn"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,6 +71,11 @@
|
||||||
"type": "hasOne",
|
"type": "hasOne",
|
||||||
"model": "Worker",
|
"model": "Worker",
|
||||||
"foreignKey": "userFk"
|
"foreignKey": "userFk"
|
||||||
|
},
|
||||||
|
"userConfig": {
|
||||||
|
"type": "hasOne",
|
||||||
|
"model": "UserConfig",
|
||||||
|
"foreignKey": "userFk"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"acls": [
|
"acls": [
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
module.exports = Self => {
|
||||||
|
require('../methods/edi/updateData')(Self);
|
||||||
|
};
|
|
@ -0,0 +1,5 @@
|
||||||
|
{
|
||||||
|
"name": "Edi",
|
||||||
|
"base": "VnModel"
|
||||||
|
}
|
||||||
|
|
|
@ -9,20 +9,23 @@
|
||||||
"properties": {
|
"properties": {
|
||||||
"userFk": {
|
"userFk": {
|
||||||
"id": true,
|
"id": true,
|
||||||
"type": "Number",
|
"type": "number",
|
||||||
"required": true
|
"required": true
|
||||||
},
|
},
|
||||||
"warehouseFk": {
|
"warehouseFk": {
|
||||||
"type": "Number"
|
"type": "number"
|
||||||
},
|
},
|
||||||
"companyFk": {
|
"companyFk": {
|
||||||
"type": "Number"
|
"type": "number"
|
||||||
},
|
},
|
||||||
"created": {
|
"created": {
|
||||||
"type": "Date"
|
"type": "date"
|
||||||
},
|
},
|
||||||
"updated": {
|
"updated": {
|
||||||
"type": "Date"
|
"type": "date"
|
||||||
|
},
|
||||||
|
"darkMode": {
|
||||||
|
"type": "boolean"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"relations": {
|
"relations": {
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
INSERT INTO `salix`.`ACL` (model, property, accessType, permission, principalType, principalId)
|
||||||
|
VALUES ('Edi', 'updateData', 'WRITE', 'ALLOW', 'ROLE', 'employee');
|
|
@ -0,0 +1,2 @@
|
||||||
|
INSERT INTO `salix`.`ACL` (`model`, `property`, `accessType`, `permission`, `principalType`, `principalId`)
|
||||||
|
VALUES('EducationLevel', '*', '*', 'ALLOW', 'ROLE', 'employee');
|
|
@ -0,0 +1,3 @@
|
||||||
|
INSERT INTO `salix`.`ACL`
|
||||||
|
(`model`, `property`, `accessType`, `permission`, `principalType`, `principalId`)
|
||||||
|
VALUES('InvoiceInIntrastat', '*', '*', 'ALLOW', 'ROLE', 'employee');
|
|
@ -1 +1 @@
|
||||||
ALTER TABLE vn.claim ADD packages smallint(10) unsigned DEFAULT 0 NULL COMMENT 'packages received by the client';
|
ALTER TABLE `vn`.`claim` ADD packages smallint(10) unsigned DEFAULT 0 NULL COMMENT 'packages received by the client';
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
INSERT INTO `vn`.`component` (`name`,`typeFk`,`classRate`,`isRenewable`,`code`,`isRequired`)
|
||||||
|
VALUES ('maná reclamacion',7,4,0,'manaClaim',0);
|
|
@ -0,0 +1,90 @@
|
||||||
|
ALTER TABLE `vn`.`country` ADD `a3Code` INT NULL COMMENT 'Código país para a3';
|
||||||
|
|
||||||
|
UPDATE `vn`.`country` c
|
||||||
|
JOIN `vn2008`.`payroll_pais` `p` ON `p`.`pais` = `c`.`country`
|
||||||
|
SET `c`.`a3Code` = `p`.`codpais`;
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 710
|
||||||
|
WHERE `country` = 'Sud-Africa'; -- ÁFRICA DEL SUR
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 643
|
||||||
|
WHERE `country` = 'Rusia'; -- FEDERACIÓN DE RUSIA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 28
|
||||||
|
WHERE `country` = 'Antigua'; -- ANTIGUA Y BARBUDA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 840
|
||||||
|
WHERE `country` = 'USA'; -- ESTADOS UNIDOS
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 404
|
||||||
|
WHERE `country` = 'Kenya'; -- KENIA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 498
|
||||||
|
WHERE `country` = 'Moldavia'; -- REPÚBLICA DE MOLDAVIA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 826
|
||||||
|
WHERE `country` = 'Gran Bretaña'; -- REINO UNIDO
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 484
|
||||||
|
WHERE `country` = 'Mexico'; -- MÉJICO
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 716
|
||||||
|
WHERE `country` = 'Zimbawe'; -- ZINBABWE
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 203
|
||||||
|
WHERE `country` = 'Chequia'; -- REPÚBLICA CHECA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 764
|
||||||
|
WHERE `country` = 'Thailandia'; -- TAILANDIA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 276
|
||||||
|
WHERE `country` = 'Alemania'; -- REPÚBLICA FEDERAL DE ALEMANIA
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 112
|
||||||
|
WHERE `country` = 'Bielorrusia'; -- BELARUS
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 528
|
||||||
|
WHERE `country` = 'Holanda'; -- PAÍSES BAJOS
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 410
|
||||||
|
WHERE `country` = 'Corea del Sur'; -- COREA (REPÚBLICA)
|
||||||
|
|
||||||
|
UPDATE `vn`.`country`
|
||||||
|
SET `a3Code` = 724
|
||||||
|
WHERE `country` = 'España exento'; -- ESPAÑA
|
||||||
|
|
||||||
|
-- Borrar registro USA de country:
|
||||||
|
UPDATE `vn`.`supplier` `s`
|
||||||
|
SET `s`.`countryFk` = 62
|
||||||
|
WHERE `s`.`countryFk` = 12;
|
||||||
|
|
||||||
|
UPDATE `vn`.`bankEntity`
|
||||||
|
SET `countryFk` = 62
|
||||||
|
WHERE `countryFk` = 12;
|
||||||
|
|
||||||
|
DELETE FROM `vn`.`country`
|
||||||
|
WHERE `id`= 12;
|
||||||
|
|
||||||
|
UPDATE `vn2008`.`payroll_pais`
|
||||||
|
SET `pais`='COREA NORTE (REPÚBLICA DEM. POPULAR)'
|
||||||
|
WHERE `codpais`=408;
|
||||||
|
UPDATE `vn2008`.`payroll_pais`
|
||||||
|
SET `pais`='COREA SUR (REPÚBLICA) '
|
||||||
|
WHERE `codpais`=410;
|
||||||
|
|
||||||
|
RENAME TABLE `vn2008`.`payroll_pais` TO `vn2008`.`payroll_pais__`;
|
|
@ -0,0 +1,2 @@
|
||||||
|
INSERT INTO `salix`.`ACL` (`model`,`property`,`accessType`,`permission`,`principalType`,`principalId`)
|
||||||
|
VALUES ('InvoiceInIntrastat','*','*','ALLOW','ROLE','employee');
|
|
@ -0,0 +1,106 @@
|
||||||
|
DROP PROCEDURE IF EXISTS `bs`.`manaCustomerUpdate`;
|
||||||
|
|
||||||
|
DELIMITER $$
|
||||||
|
$$
|
||||||
|
CREATE DEFINER=`root`@`localhost` PROCEDURE `bs`.`manaCustomerUpdate`()
|
||||||
|
BEGIN
|
||||||
|
DECLARE vToDated DATE;
|
||||||
|
DECLARE vFromDated DATE;
|
||||||
|
DECLARE vForDeleteDated DATE;
|
||||||
|
DECLARE vManaId INT;
|
||||||
|
DECLARE vManaAutoId INT;
|
||||||
|
DECLARE vClaimManaId INT;
|
||||||
|
DECLARE vManaBankId INT;
|
||||||
|
DECLARE vManaGreugeTypeId INT;
|
||||||
|
|
||||||
|
SELECT id INTO vManaId
|
||||||
|
FROM `component` WHERE code = 'mana';
|
||||||
|
|
||||||
|
SELECT id INTO vManaAutoId
|
||||||
|
FROM `component` WHERE code = 'autoMana';
|
||||||
|
|
||||||
|
SELECT id INTO vClaimManaId
|
||||||
|
FROM `component` WHERE code = 'manaClaim';
|
||||||
|
|
||||||
|
SELECT id INTO vManaBankId
|
||||||
|
FROM `bank` WHERE code = 'mana';
|
||||||
|
|
||||||
|
SELECT id INTO vManaGreugeTypeId
|
||||||
|
FROM `greugeType` WHERE code = 'mana';
|
||||||
|
|
||||||
|
SELECT IFNULL(max(dated), '2016-01-01')
|
||||||
|
INTO vFromDated
|
||||||
|
FROM bs.manaCustomer;
|
||||||
|
|
||||||
|
DELETE
|
||||||
|
FROM bs.manaCustomer
|
||||||
|
WHERE dated = vFromDated;
|
||||||
|
|
||||||
|
SELECT IFNULL(max(dated), '2016-01-01')
|
||||||
|
INTO vFromDated
|
||||||
|
FROM bs.manaCustomer;
|
||||||
|
|
||||||
|
WHILE timestampadd(DAY,30,vFromDated) < CURDATE() DO
|
||||||
|
|
||||||
|
SELECT
|
||||||
|
timestampadd(DAY,30,vFromDated),
|
||||||
|
timestampadd(DAY,-90,vFromDated)
|
||||||
|
INTO
|
||||||
|
vToDated,
|
||||||
|
vForDeleteDated;
|
||||||
|
|
||||||
|
DELETE FROM bs.manaCustomer
|
||||||
|
WHERE dated <= vForDeleteDated;
|
||||||
|
|
||||||
|
|
||||||
|
INSERT INTO bs.manaCustomer(Id_Cliente, Mana, dated)
|
||||||
|
|
||||||
|
SELECT
|
||||||
|
Id_Cliente,
|
||||||
|
cast(sum(mana) as decimal(10,2)) as mana,
|
||||||
|
vToDated as dated
|
||||||
|
FROM
|
||||||
|
|
||||||
|
(
|
||||||
|
SELECT cs.Id_Cliente, Cantidad * Valor as mana
|
||||||
|
FROM vn2008.Tickets t
|
||||||
|
JOIN vn2008.Consignatarios cs using(Id_Consigna)
|
||||||
|
JOIN vn2008.Movimientos m on m.Id_Ticket = t.Id_Ticket
|
||||||
|
JOIN vn2008.Movimientos_componentes mc on mc.Id_Movimiento = m.Id_Movimiento
|
||||||
|
WHERE Id_Componente IN (vManaAutoId, vManaId, vClaimManaId)
|
||||||
|
AND t.Fecha > vFromDated
|
||||||
|
AND date(t.Fecha) <= vToDated
|
||||||
|
|
||||||
|
|
||||||
|
UNION ALL
|
||||||
|
|
||||||
|
SELECT r.Id_Cliente, - Entregado
|
||||||
|
FROM vn2008.Recibos r
|
||||||
|
WHERE Id_Banco = vManaBankId
|
||||||
|
AND Fechacobro > vFromDated
|
||||||
|
AND Fechacobro <= vToDated
|
||||||
|
|
||||||
|
UNION ALL
|
||||||
|
|
||||||
|
SELECT g.Id_Cliente, g.Importe
|
||||||
|
FROM vn2008.Greuges g
|
||||||
|
WHERE Greuges_type_id = vManaGreugeTypeId
|
||||||
|
AND Fecha > vFromDated
|
||||||
|
AND Fecha <= vToDated
|
||||||
|
|
||||||
|
UNION ALL
|
||||||
|
|
||||||
|
SELECT Id_Cliente, mana
|
||||||
|
FROM bs.manaCustomer
|
||||||
|
WHERE dated = vFromDated
|
||||||
|
) sub
|
||||||
|
|
||||||
|
GROUP BY Id_Cliente
|
||||||
|
HAVING Id_Cliente;
|
||||||
|
|
||||||
|
SET vFromDated = vToDated;
|
||||||
|
|
||||||
|
END WHILE;
|
||||||
|
|
||||||
|
END$$
|
||||||
|
DELIMITER ;
|
|
@ -0,0 +1,75 @@
|
||||||
|
DROP PROCEDURE IF EXISTS `vn`.`manaSpellersRequery`;
|
||||||
|
|
||||||
|
DELIMITER $$
|
||||||
|
$$
|
||||||
|
CREATE DEFINER=`root`@`localhost` PROCEDURE `vn`.`manaSpellersRequery`(vWorkerFk INTEGER)
|
||||||
|
BEGIN
|
||||||
|
/**
|
||||||
|
* Recalcula el mana consumido por un trabajador
|
||||||
|
*
|
||||||
|
* @param vWorkerFk Id Trabajador
|
||||||
|
*/
|
||||||
|
DECLARE vWorkerIsExcluded BOOLEAN;
|
||||||
|
DECLARE vFromDated DATE;
|
||||||
|
DECLARE vToDated DATE DEFAULT TIMESTAMPADD(DAY,1,CURDATE());
|
||||||
|
DECLARE vMana INT;
|
||||||
|
DECLARE vAutoMana INT;
|
||||||
|
DECLARE vClaimMana INT;
|
||||||
|
DECLARE vManaBank INT;
|
||||||
|
DECLARE vManaGreugeType INT;
|
||||||
|
|
||||||
|
SELECT COUNT(*) INTO vWorkerIsExcluded
|
||||||
|
FROM workerManaExcluded
|
||||||
|
WHERE workerFk = vWorkerFk;
|
||||||
|
|
||||||
|
IF NOT vWorkerIsExcluded THEN
|
||||||
|
SELECT id INTO vMana
|
||||||
|
FROM `component` WHERE code = 'mana';
|
||||||
|
|
||||||
|
SELECT id INTO vAutoMana
|
||||||
|
FROM `component` WHERE code = 'autoMana';
|
||||||
|
|
||||||
|
SELECT id INTO vClaimMana
|
||||||
|
FROM `component` WHERE code = 'manaClaim';
|
||||||
|
|
||||||
|
SELECT id INTO vManaBank
|
||||||
|
FROM `bank` WHERE code = 'mana';
|
||||||
|
|
||||||
|
SELECT id INTO vManaGreugeType
|
||||||
|
FROM `greugeType` WHERE code = 'mana';
|
||||||
|
|
||||||
|
SELECT max(dated) INTO vFromDated
|
||||||
|
FROM clientManaCache;
|
||||||
|
|
||||||
|
REPLACE workerMana (workerFk, amount)
|
||||||
|
SELECT vWorkerFk, sum(mana) FROM
|
||||||
|
(
|
||||||
|
SELECT s.quantity * sc.value as mana
|
||||||
|
FROM ticket t
|
||||||
|
JOIN address a ON a.id = t.addressFk
|
||||||
|
JOIN client c ON c.id = a.clientFk
|
||||||
|
JOIN sale s ON s.ticketFk = t.id
|
||||||
|
JOIN saleComponent sc ON sc.saleFk = s.id
|
||||||
|
WHERE c.salesPersonFk = vWorkerFk AND sc.componentFk IN (vMana, vAutoMana, vClaimMana)
|
||||||
|
AND t.shipped > vFromDated AND t.shipped < vToDated
|
||||||
|
UNION ALL
|
||||||
|
SELECT - r.amountPaid
|
||||||
|
FROM receipt r
|
||||||
|
JOIN client c ON c.id = r.clientFk
|
||||||
|
WHERE c.salesPersonFk = vWorkerFk AND bankFk = vManaBank
|
||||||
|
AND payed > vFromDated
|
||||||
|
UNION ALL
|
||||||
|
SELECT g.amount
|
||||||
|
FROM greuge g
|
||||||
|
JOIN client c ON c.id = g.clientFk
|
||||||
|
WHERE c.salesPersonFk = vWorkerFk AND g.greugeTypeFk = vManaGreugeType
|
||||||
|
AND g.shipped > vFromDated and g.shipped < CURDATE()
|
||||||
|
UNION ALL
|
||||||
|
SELECT cc.mana
|
||||||
|
FROM clientManaCache cc
|
||||||
|
JOIN client c ON c.id = cc.clientFk
|
||||||
|
WHERE c.salesPersonFk = vWorkerFk AND cc.dated = vFromDated
|
||||||
|
) sub;
|
||||||
|
END IF;
|
||||||
|
END$$
|
||||||
|
DELIMITER ;
|
|
@ -0,0 +1,51 @@
|
||||||
|
|
||||||
|
ALTER TABLE `vn`.`worker` MODIFY COLUMN `maritalStatus__` varchar(10) CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL;
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
SET `w`.`maritalStatus__` = NULL;
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
JOIN `vn`.`person` `p` ON `p`.`workerFk` = `w`.`id`
|
||||||
|
JOIN `postgresql`.`profile` `pr` ON `pr`.`person_id` = `p`.`id`
|
||||||
|
JOIN `vn2008`.`profile_labour_payroll` `pl` ON `pl`.`profile_id` = `pr`.`profile_id`
|
||||||
|
SET `w`.`maritalStatus__` = `pl`.`estadocivil`;
|
||||||
|
|
||||||
|
ALTER TABLE `vn`.`worker` ADD `originCountryFk` mediumint(8) unsigned NULL COMMENT 'País de origen';
|
||||||
|
ALTER TABLE `vn`.`worker` ADD `educationLevelFk` SMALLINT NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` ADD `SSN` varchar(15) NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` CHANGE `maritalStatus__` `maritalStatus` enum('S','M') CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` MODIFY COLUMN `maritalStatus` enum('S','M') CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` CHANGE `maritalStatus` maritalStatus enum('S','M') CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL AFTER sectorFk;
|
||||||
|
ALTER TABLE `vn`.`worker` ADD CONSTRAINT `worker_FK_2` FOREIGN KEY (`educationLevelFk`) REFERENCES `vn`.`educationLevel`(`id`) ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
|
ALTER TABLE `vn`.`worker` ADD CONSTRAINT `worker_FK_1` FOREIGN KEY (`originCountryFk`) REFERENCES `vn`.`country`(`id`) ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
INSERT INTO `vn`.`country` (`country`, `CEE`, `code`, `politicalCountryFk`, `isUeeMember`, `a3Code`)
|
||||||
|
VALUES
|
||||||
|
('Argentina',2,'AR',80,0,32),
|
||||||
|
('Cuba',2,'CU',81,0,192),
|
||||||
|
('Guinea Ecuatorial',2,'GQ',82,0,226),
|
||||||
|
('Guinea',2,'GN',83,0,324),
|
||||||
|
('Honduras',2,'HN',84,0,340),
|
||||||
|
('Mali',2,'ML',85,0,466),
|
||||||
|
('Nicaragua',2,'NI',86,0,558),
|
||||||
|
('Pakistán',2,'PK',87,0,586),
|
||||||
|
('Paraguay',2,'PY',88,0,600),
|
||||||
|
('Senegal',2,'SN',89,0,686),
|
||||||
|
('Uruguay',2,'UY',90,0,858),
|
||||||
|
('Venezuela',2,'VE',91,0,862),
|
||||||
|
('Bulgaria',2,'BG',92,1,100),
|
||||||
|
('Georgia',2,'GE',93,0,268);
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
JOIN `vn`.`person` `p` ON `p`.`workerFk` = `w`.`id`
|
||||||
|
JOIN `postgresql`.`profile` `pr` ON `pr`.`person_id` = `p`.`id`
|
||||||
|
JOIN `vn2008`.`profile_labour_payroll` `pl` ON `pl`.`profile_id` = `pr`.`profile_id`
|
||||||
|
JOIN `vn`.`country` `co` ON `co`.`a3Code` = `pl`.`codpais`
|
||||||
|
SET `w`.`originCountryFk` = `co`.`id`;
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
JOIN `vn`.`person` `p` ON `p`.`workerFk` = `w`.`id`
|
||||||
|
JOIN `postgresql`.`profile` `pr` ON `pr`.`person_id` = `p`.`id`
|
||||||
|
JOIN `vn2008`.`profile_labour_payroll` pl ON `pl`.`profile_id` = `pr`.`profile_id`
|
||||||
|
SET `w`.`SSN` = CONCAT(`pl`.`NSSProvincia`, `pl`.`NssNumero`, `pl`.`NssDC`);
|
||||||
|
|
||||||
|
RENAME TABLE `vn2008`.`profile_labour_payroll` TO `vn2008`.`profile_labour_payroll__`;
|
|
@ -0,0 +1,73 @@
|
||||||
|
DROP PROCEDURE IF EXISTS vn.timeControl_getError;
|
||||||
|
|
||||||
|
DELIMITER $$
|
||||||
|
$$
|
||||||
|
CREATE DEFINER=`root`@`localhost` PROCEDURE `vn`.`timeControl_getError`(vDatedFrom DATETIME, vDatedTo DATETIME)
|
||||||
|
BEGIN
|
||||||
|
/*
|
||||||
|
* @param vDatedFrom
|
||||||
|
* @param vDatedTo
|
||||||
|
* @table tmp.`user`(userFk)
|
||||||
|
* Fichadas incorrectas de las cuales no se puede calcular horas trabajadas
|
||||||
|
* @return tmp.timeControlError (id)
|
||||||
|
*/
|
||||||
|
DECLARE vDayMaxTime INTEGER;
|
||||||
|
|
||||||
|
SET @journeyCounter := 0;
|
||||||
|
SET @lastUserFk := NULL;
|
||||||
|
|
||||||
|
SELECT dayMaxTime INTO vDayMaxTime
|
||||||
|
FROM workerTimeControlConfig LIMIT 1;
|
||||||
|
|
||||||
|
DROP TEMPORARY TABLE IF EXISTS tmp.timeControl;
|
||||||
|
CREATE TEMPORARY TABLE tmp.timeControl
|
||||||
|
(INDEX(id), INDEX(journeyCounter))
|
||||||
|
ENGINE = MEMORY
|
||||||
|
SELECT sub.id,
|
||||||
|
sub.direction,
|
||||||
|
sub.timed,
|
||||||
|
IF(sub.direction = 'in' OR @hasOut OR sub.userFk <> @lastUserFk, @journeyCounter := @journeyCounter + 1, @journeyCounter) journeyCounter,
|
||||||
|
@lastUserFk := sub.userFk workerFk,
|
||||||
|
IF(sub.direction = 'out', @hasOut:= TRUE, @hasOut:= FALSE)
|
||||||
|
FROM (
|
||||||
|
SELECT DISTINCT wtc.id,
|
||||||
|
wtc.direction,
|
||||||
|
wtc.timed,
|
||||||
|
wtc.userFk
|
||||||
|
FROM workerTimeControl wtc
|
||||||
|
JOIN tmp.`user` w ON w.userFk = wtc.userFk
|
||||||
|
WHERE wtc.timed BETWEEN DATE_SUB(vDatedFrom, INTERVAL 1 DAY) AND DATE_ADD(vDatedTo, INTERVAL 1 DAY)
|
||||||
|
ORDER BY wtc.userFk, wtc.timed
|
||||||
|
) sub;
|
||||||
|
|
||||||
|
DROP TEMPORARY TABLE IF EXISTS tmp.timeControlAux;
|
||||||
|
CREATE TEMPORARY TABLE tmp.timeControlAux
|
||||||
|
(INDEX(id), INDEX(journeyCounter))
|
||||||
|
ENGINE = MEMORY
|
||||||
|
SELECT * FROM tmp.timeControl;
|
||||||
|
|
||||||
|
DROP TEMPORARY TABLE IF EXISTS tmp.timeControlError;
|
||||||
|
CREATE TEMPORARY TABLE tmp.timeControlError
|
||||||
|
(INDEX(id))
|
||||||
|
ENGINE = MEMORY
|
||||||
|
SELECT id
|
||||||
|
FROM tmp.timeControlAux tca
|
||||||
|
JOIN (SELECT journeyCounter,
|
||||||
|
UNIX_TIMESTAMP(MAX(timed)) - UNIX_TIMESTAMP(MIN(timed)) timeWork,
|
||||||
|
SUM(direction = 'in') totalIn,
|
||||||
|
SUM(direction = 'out') totalOut,
|
||||||
|
timed
|
||||||
|
FROM tmp.timeControl
|
||||||
|
GROUP BY journeyCounter
|
||||||
|
HAVING COUNT(*) MOD 2 = 1
|
||||||
|
OR totalIn <> 1
|
||||||
|
OR totalOut <> 1
|
||||||
|
OR timeWork >= vDayMaxTime
|
||||||
|
)sub ON sub.journeyCounter = tca.journeyCounter
|
||||||
|
WHERE sub.timed BETWEEN vDatedFrom AND vDatedTo;
|
||||||
|
|
||||||
|
DROP TEMPORARY TABLE IF EXISTS tmp.timeControl;
|
||||||
|
DROP TEMPORARY TABLE IF EXISTS tmp.timeControlAux;
|
||||||
|
|
||||||
|
END$$
|
||||||
|
DELIMITER ;
|
|
@ -0,0 +1,10 @@
|
||||||
|
CREATE TABLE `vn`.`clientUnpaid` (
|
||||||
|
`clientFk` int(11) NOT NULL,
|
||||||
|
`dated` date NOT NULL,
|
||||||
|
`amount` double DEFAULT 0,
|
||||||
|
PRIMARY KEY (`clientFk`),
|
||||||
|
CONSTRAINT `clientUnpaid_clientFk` FOREIGN KEY (`clientFk`) REFERENCES `client` (`id`) ON UPDATE CASCADE
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO `salix`.`ACL` (`model`, `property`, `accessType`, `permission`, `principalType`, `principalId`)
|
||||||
|
VALUES('ClientUnpaid', '*', '*', 'ALLOW', 'ROLE', 'administrative');
|
|
@ -0,0 +1 @@
|
||||||
|
ALTER TABLE `vn`.`userConfig` ADD darkMode tinyint(1) DEFAULT 1 NOT NULL COMMENT 'Salix interface dark mode';
|
|
@ -0,0 +1,6 @@
|
||||||
|
UPDATE `salix`.`ACL`
|
||||||
|
SET `property`='refund'
|
||||||
|
WHERE `model`='Sale' AND `property`='payBack';
|
||||||
|
|
||||||
|
INSERT INTO `salix`.`ACL`(`model`, `property`, `accessType`, `permission`, `principalType`, `principalId`)
|
||||||
|
VALUES('Sale', 'refundAll', 'WRITE', 'ALLOW', 'ROLE', 'employee');
|
|
@ -0,0 +1,113 @@
|
||||||
|
DROP PROCEDURE IF EXISTS vn.ticket_doRefund;
|
||||||
|
|
||||||
|
DELIMITER $$
|
||||||
|
$$
|
||||||
|
CREATE DEFINER=`root`@`localhost` PROCEDURE `vn`.`ticket_doRefund`(IN vOriginTicket INT, OUT vNewTicket INT)
|
||||||
|
BEGIN
|
||||||
|
|
||||||
|
DECLARE vDone BIT DEFAULT 0;
|
||||||
|
DECLARE vCustomer MEDIUMINT;
|
||||||
|
DECLARE vWarehouse TINYINT;
|
||||||
|
DECLARE vCompany MEDIUMINT;
|
||||||
|
DECLARE vAddress MEDIUMINT;
|
||||||
|
DECLARE vRefundAgencyMode INT;
|
||||||
|
DECLARE vItemFk INT;
|
||||||
|
DECLARE vQuantity DECIMAL (10,2);
|
||||||
|
DECLARE vConcept VARCHAR(50);
|
||||||
|
DECLARE vPrice DECIMAL (10,2);
|
||||||
|
DECLARE vDiscount TINYINT;
|
||||||
|
DECLARE vSaleNew INT;
|
||||||
|
DECLARE vSaleMain INT;
|
||||||
|
DECLARE vZoneFk INT;
|
||||||
|
DECLARE vDescription VARCHAR(50);
|
||||||
|
DECLARE vTaxClassFk INT;
|
||||||
|
DECLARE vTicketServiceTypeFk INT;
|
||||||
|
|
||||||
|
DECLARE cSales CURSOR FOR
|
||||||
|
SELECT *
|
||||||
|
FROM tmp.sale;
|
||||||
|
|
||||||
|
DECLARE cTicketServices CURSOR FOR
|
||||||
|
SELECT *
|
||||||
|
FROM tmp.ticketService;
|
||||||
|
|
||||||
|
DECLARE CONTINUE HANDLER FOR NOT FOUND SET vDone = 1;
|
||||||
|
|
||||||
|
SELECT id INTO vRefundAgencyMode
|
||||||
|
FROM agencyMode WHERE `name` = 'ABONO';
|
||||||
|
|
||||||
|
SELECT clientFk, warehouseFk, companyFk, addressFk
|
||||||
|
INTO vCustomer, vWarehouse, vCompany, vAddress
|
||||||
|
FROM ticket
|
||||||
|
WHERE id = vOriginTicket;
|
||||||
|
|
||||||
|
SELECT id INTO vZoneFk
|
||||||
|
FROM zone WHERE agencyModeFk = vRefundAgencyMode
|
||||||
|
LIMIT 1;
|
||||||
|
|
||||||
|
INSERT INTO vn.ticket (
|
||||||
|
clientFk,
|
||||||
|
shipped,
|
||||||
|
addressFk,
|
||||||
|
agencyModeFk,
|
||||||
|
nickname,
|
||||||
|
warehouseFk,
|
||||||
|
companyFk,
|
||||||
|
landed,
|
||||||
|
zoneFk
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
vCustomer,
|
||||||
|
CURDATE(),
|
||||||
|
vAddress,
|
||||||
|
vRefundAgencyMode,
|
||||||
|
a.nickname,
|
||||||
|
vWarehouse,
|
||||||
|
vCompany,
|
||||||
|
CURDATE(),
|
||||||
|
vZoneFk
|
||||||
|
FROM address a
|
||||||
|
WHERE a.id = vAddress;
|
||||||
|
|
||||||
|
SET vNewTicket = LAST_INSERT_ID();
|
||||||
|
|
||||||
|
SET vDone := 0;
|
||||||
|
OPEN cSales;
|
||||||
|
FETCH cSales INTO vSaleMain, vItemFk, vQuantity, vConcept, vPrice, vDiscount;
|
||||||
|
|
||||||
|
WHILE NOT vDone DO
|
||||||
|
|
||||||
|
INSERT INTO vn.sale(ticketFk, itemFk, quantity, concept, price, discount)
|
||||||
|
VALUES( vNewTicket, vItemFk, vQuantity, vConcept, vPrice, vDiscount );
|
||||||
|
|
||||||
|
SET vSaleNew = LAST_INSERT_ID();
|
||||||
|
|
||||||
|
INSERT INTO vn.saleComponent(saleFk,componentFk,`value`)
|
||||||
|
SELECT vSaleNew,componentFk,`value`
|
||||||
|
FROM vn.saleComponent
|
||||||
|
WHERE saleFk = vSaleMain;
|
||||||
|
|
||||||
|
FETCH cSales INTO vSaleMain, vItemFk, vQuantity, vConcept, vPrice, vDiscount;
|
||||||
|
|
||||||
|
END WHILE;
|
||||||
|
CLOSE cSales;
|
||||||
|
|
||||||
|
SET vDone := 0;
|
||||||
|
OPEN cTicketServices;
|
||||||
|
FETCH cTicketServices INTO vDescription, vQuantity, vPrice, vTaxClassFk, vTicketServiceTypeFk;
|
||||||
|
|
||||||
|
WHILE NOT vDone DO
|
||||||
|
|
||||||
|
INSERT INTO vn.ticketService(description, quantity, price, taxClassFk, ticketFk, ticketServiceTypeFk)
|
||||||
|
VALUES(vDescription, vQuantity, vPrice, vTaxClassFk, vNewTicket, vTicketServiceTypeFk);
|
||||||
|
|
||||||
|
FETCH cTicketServices INTO vDescription, vQuantity, vPrice, vTaxClassFk, vTicketServiceTypeFk;
|
||||||
|
|
||||||
|
END WHILE;
|
||||||
|
CLOSE cTicketServices;
|
||||||
|
|
||||||
|
INSERT INTO vn.ticketRefund(refundTicketFk, originalTicketFk)
|
||||||
|
VALUES(vNewTicket, vOriginTicket);
|
||||||
|
|
||||||
|
END$$
|
||||||
|
DELIMITER ;
|
|
@ -47,10 +47,36 @@ INSERT INTO `account`.`user`(`id`,`name`, `nickname`, `password`,`role`,`active`
|
||||||
INSERT INTO `account`.`account`(`id`)
|
INSERT INTO `account`.`account`(`id`)
|
||||||
SELECT id FROM `account`.`user`;
|
SELECT id FROM `account`.`user`;
|
||||||
|
|
||||||
|
INSERT INTO `vn`.`educationLevel` (`id`, `name`)
|
||||||
|
VALUES
|
||||||
|
(1, 'ESTUDIOS PRIMARIOS COMPLETOS'),
|
||||||
|
(2, 'ENSEÑANZAS DE BACHILLERATO');
|
||||||
|
|
||||||
INSERT INTO `vn`.`worker`(`id`,`code`, `firstName`, `lastName`, `userFk`, `bossFk`)
|
INSERT INTO `vn`.`worker`(`id`,`code`, `firstName`, `lastName`, `userFk`, `bossFk`)
|
||||||
SELECT id,UPPER(LPAD(role, 3, '0')), name, name, id, 9
|
SELECT id,UPPER(LPAD(role, 3, '0')), name, name, id, 9
|
||||||
FROM `vn`.`user`;
|
FROM `vn`.`user`;
|
||||||
|
|
||||||
|
ALTER TABLE `vn`.`worker` ADD `originCountryFk` mediumint(8) unsigned NULL COMMENT 'País de origen';
|
||||||
|
ALTER TABLE `vn`.`worker` ADD `educationLevelFk` SMALLINT NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` ADD `SSN` varchar(15) NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` CHANGE `maritalStatus__` `maritalStatus` enum('S','M') CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` MODIFY COLUMN `maritalStatus` enum('S','M') CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL;
|
||||||
|
ALTER TABLE `vn`.`worker` CHANGE `maritalStatus` maritalStatus enum('S','M') CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL AFTER sectorFk;
|
||||||
|
ALTER TABLE `vn`.`worker` ADD CONSTRAINT `worker_FK_2` FOREIGN KEY (`educationLevelFk`) REFERENCES `vn`.`educationLevel`(`id`) ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
|
ALTER TABLE `vn`.`worker` ADD CONSTRAINT `worker_FK_1` FOREIGN KEY (`originCountryFk`) REFERENCES `vn`.`country`(`id`) ON DELETE RESTRICT ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
SET `maritalStatus` = 'S';
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
SET `originCountryFk` = '1';
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
SET `educationLevelFk` = '2';
|
||||||
|
|
||||||
|
UPDATE `vn`.`worker` `w`
|
||||||
|
SET `SSN` = '123456789123';
|
||||||
|
|
||||||
UPDATE `vn`.`worker` SET bossFk = NULL WHERE id = 20;
|
UPDATE `vn`.`worker` SET bossFk = NULL WHERE id = 20;
|
||||||
UPDATE `vn`.`worker` SET bossFk = 20 WHERE id = 1 OR id = 9;
|
UPDATE `vn`.`worker` SET bossFk = 20 WHERE id = 1 OR id = 9;
|
||||||
UPDATE `vn`.`worker` SET bossFk = 19 WHERE id = 18;
|
UPDATE `vn`.`worker` SET bossFk = 19 WHERE id = 18;
|
||||||
|
@ -292,10 +318,10 @@ INSERT INTO `vn`.`contactChannel`(`id`, `name`)
|
||||||
INSERT INTO `vn`.`client`(`id`,`name`,`fi`,`socialName`,`contact`,`street`,`city`,`postcode`,`phone`,`mobile`,`isRelevant`,`email`,`iban`,`dueDay`,`accountingAccount`,`isEqualizated`,`provinceFk`,`hasToInvoice`,`credit`,`countryFk`,`isActive`,`gestdocFk`,`quality`,`payMethodFk`,`created`,`isToBeMailed`,`contactChannelFk`,`hasSepaVnl`,`hasCoreVnl`,`hasCoreVnh`,`riskCalculated`,`clientTypeFk`,`mailAddress`,`hasToInvoiceByAddress`,`isTaxDataChecked`,`isFreezed`,`creditInsurance`,`isCreatedAsServed`,`hasInvoiceSimplified`,`salesPersonFk`,`isVies`,`eypbc`, `businessTypeFk`)
|
INSERT INTO `vn`.`client`(`id`,`name`,`fi`,`socialName`,`contact`,`street`,`city`,`postcode`,`phone`,`mobile`,`isRelevant`,`email`,`iban`,`dueDay`,`accountingAccount`,`isEqualizated`,`provinceFk`,`hasToInvoice`,`credit`,`countryFk`,`isActive`,`gestdocFk`,`quality`,`payMethodFk`,`created`,`isToBeMailed`,`contactChannelFk`,`hasSepaVnl`,`hasCoreVnl`,`hasCoreVnh`,`riskCalculated`,`clientTypeFk`,`mailAddress`,`hasToInvoiceByAddress`,`isTaxDataChecked`,`isFreezed`,`creditInsurance`,`isCreatedAsServed`,`hasInvoiceSimplified`,`salesPersonFk`,`isVies`,`eypbc`, `businessTypeFk`)
|
||||||
VALUES
|
VALUES
|
||||||
(1101, 'Bruce Wayne', '84612325V', 'Batman', 'Alfred', '1007 Mountain Drive, Gotham', 'Silla', 46460, 1111111111, 222222222, 1, 'BruceWayne@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
(1101, 'Bruce Wayne', '84612325V', 'Batman', 'Alfred', '1007 Mountain Drive, Gotham', 'Silla', 46460, 1111111111, 222222222, 1, 'BruceWayne@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
||||||
(1102, 'Petter Parker', '87945234L', 'Spider man', 'Aunt May', '20 Ingram Street, Queens, USA', 'Silla', 46460, 1111111111, 222222222, 1, 'PetterParker@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
(1102, 'Petter Parker', '87945234L', 'Spider man', 'Aunt May', '20 Ingram Street, Queens, USA', 'Silla', 46460, 1111111111, 222222222, 1, 'PetterParker@mydomain.com', NULL, 0, 1234567890, 0, 2, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
||||||
(1103, 'Clark Kent', '06815934E', 'Super man', 'lois lane', '344 Clinton Street, Apartament 3-D', 'Silla', 46460, 1111111111, 222222222, 1, 'ClarkKent@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 0, 19, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
(1103, 'Clark Kent', '06815934E', 'Super man', 'lois lane', '344 Clinton Street, Apartament 3-D', 'Silla', 46460, 1111111111, 222222222, 1, 'ClarkKent@mydomain.com', NULL, 0, 1234567890, 0, 3, 1, 0, 19, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
||||||
(1104, 'Tony Stark', '06089160W', 'Iron man', 'Pepper Potts', '10880 Malibu Point, 90265', 'Silla', 46460, 1111111111, 222222222, 1, 'TonyStark@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
(1104, 'Tony Stark', '06089160W', 'Iron man', 'Pepper Potts', '10880 Malibu Point, 90265', 'Silla', 46460, 1111111111, 222222222, 1, 'TonyStark@mydomain.com', NULL, 0, 1234567890, 0, 2, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 18, 0, 1, 'florist'),
|
||||||
(1105, 'Max Eisenhardt', '251628698', 'Magneto', 'Rogue', 'Unknown Whereabouts', 'Silla', 46460, 1111111111, 222222222, 1, 'MaxEisenhardt@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 8, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 1, NULL, 0, 0, 18, 0, 1, 'florist'),
|
(1105, 'Max Eisenhardt', '251628698', 'Magneto', 'Rogue', 'Unknown Whereabouts', 'Silla', 46460, 1111111111, 222222222, 1, 'MaxEisenhardt@mydomain.com', NULL, 0, 1234567890, 0, 3, 1, 300, 8, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 1, NULL, 0, 0, 18, 0, 1, 'florist'),
|
||||||
(1106, 'DavidCharlesHaller', '53136686Q', 'Legion', 'Charles Xavier', 'City of New York, New York, USA', 'Silla', 46460, 1111111111, 222222222, 1, 'DavidCharlesHaller@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 0, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 19, 0, 1, 'florist'),
|
(1106, 'DavidCharlesHaller', '53136686Q', 'Legion', 'Charles Xavier', 'City of New York, New York, USA', 'Silla', 46460, 1111111111, 222222222, 1, 'DavidCharlesHaller@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 0, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 0, NULL, 0, 0, 19, 0, 1, 'florist'),
|
||||||
(1107, 'Hank Pym', '09854837G', 'Ant man', 'Hawk', 'Anthill, San Francisco, California', 'Silla', 46460, 1111111111, 222222222, 1, 'HankPym@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 0, 0, NULL, 0, 0, 19, 0, 1, 'florist'),
|
(1107, 'Hank Pym', '09854837G', 'Ant man', 'Hawk', 'Anthill, San Francisco, California', 'Silla', 46460, 1111111111, 222222222, 1, 'HankPym@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 0, 0, NULL, 0, 0, 19, 0, 1, 'florist'),
|
||||||
(1108, 'Charles Xavier', '22641921P', 'Professor X', 'Beast', '3800 Victory Pkwy, Cincinnati, OH 45207, USA', 'Silla', 46460, 1111111111, 222222222, 1, 'CharlesXavier@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 1, NULL, 0, 0, 19, 0, 1, 'florist'),
|
(1108, 'Charles Xavier', '22641921P', 'Professor X', 'Beast', '3800 Victory Pkwy, Cincinnati, OH 45207, USA', 'Silla', 46460, 1111111111, 222222222, 1, 'CharlesXavier@mydomain.com', NULL, 0, 1234567890, 0, 1, 1, 300, 1, 1, NULL, 10, 5,CURDATE(), 1, 5, 1, 1, 1, '0000-00-00', 1, 1, 1, 1, 1, NULL, 0, 0, 19, 0, 1, 'florist'),
|
||||||
|
@ -2434,6 +2460,13 @@ INSERT INTO `vn`.`invoiceInTax` (`invoiceInFk`, `taxableBase`, `expenceFk`, `for
|
||||||
(6, 29.95, '7001000000', NULL, 7, 20),
|
(6, 29.95, '7001000000', NULL, 7, 20),
|
||||||
(7, 58.64, '6210000567', NULL, 8, 20);
|
(7, 58.64, '6210000567', NULL, 8, 20);
|
||||||
|
|
||||||
|
INSERT INTO `vn`.`invoiceInIntrastat` (`invoiceInFk`, `net`, `intrastatFk`, `amount`, `stems`, `countryFk`)
|
||||||
|
VALUES
|
||||||
|
(1, 30.50, 5080000, 10.00, 162, 5),
|
||||||
|
(1, 10, 6021010, 20.00, 205, 5),
|
||||||
|
(2, 13.20, 5080000, 15.00, 580, 5),
|
||||||
|
(2, 16.10, 6021010, 25.00, 80, 5);
|
||||||
|
|
||||||
INSERT INTO `vn`.`ticketRecalc`(`ticketFk`)
|
INSERT INTO `vn`.`ticketRecalc`(`ticketFk`)
|
||||||
SELECT `id`
|
SELECT `id`
|
||||||
FROM `vn`.`ticket` t
|
FROM `vn`.`ticket` t
|
||||||
|
|
|
@ -181,12 +181,12 @@ let actions = {
|
||||||
},
|
},
|
||||||
|
|
||||||
reloadSection: async function(state) {
|
reloadSection: async function(state) {
|
||||||
await this.click('vn-icon[icon="preview"]');
|
await this.click('vn-icon[icon="launch"]');
|
||||||
await this.accessToSection(state);
|
await this.accessToSection(state);
|
||||||
},
|
},
|
||||||
|
|
||||||
forceReloadSection: async function(sectionRoute) {
|
forceReloadSection: async function(sectionRoute) {
|
||||||
await this.waitToClick('vn-icon[icon="preview"]');
|
await this.waitToClick('vn-icon[icon="launch"]');
|
||||||
await this.waitToClick('button[response="accept"]');
|
await this.waitToClick('button[response="accept"]');
|
||||||
await this.waitForSelector('vn-card.summary');
|
await this.waitForSelector('vn-card.summary');
|
||||||
await this.waitToClick(`vn-left-menu li > a[ui-sref="${sectionRoute}"]`);
|
await this.waitToClick(`vn-left-menu li > a[ui-sref="${sectionRoute}"]`);
|
||||||
|
|
|
@ -321,6 +321,12 @@ export default {
|
||||||
deleteFirstPhone: 'vn-client-contact vn-icon[icon="delete"]',
|
deleteFirstPhone: 'vn-client-contact vn-icon[icon="delete"]',
|
||||||
saveButton: 'button[type=submit]'
|
saveButton: 'button[type=submit]'
|
||||||
},
|
},
|
||||||
|
clientUnpaid: {
|
||||||
|
hasDataCheckBox: 'vn-client-unpaid vn-check[ng-model="watcher.hasData"]',
|
||||||
|
dated: 'vn-client-unpaid vn-date-picker[ng-model="$ctrl.clientUnpaid.dated"]',
|
||||||
|
amount: 'vn-client-unpaid vn-input-number[ng-model="$ctrl.clientUnpaid.amount"]',
|
||||||
|
saveButton: 'vn-submit[label="Save"]'
|
||||||
|
},
|
||||||
itemsIndex: {
|
itemsIndex: {
|
||||||
createItemButton: `vn-float-button`,
|
createItemButton: `vn-float-button`,
|
||||||
firstSearchResult: 'vn-item-index tbody tr:nth-child(1)',
|
firstSearchResult: 'vn-item-index tbody tr:nth-child(1)',
|
||||||
|
@ -570,7 +576,7 @@ export default {
|
||||||
moreMenuUnmarkReseved: 'vn-item[name="unreserve"]',
|
moreMenuUnmarkReseved: 'vn-item[name="unreserve"]',
|
||||||
moreMenuUpdateDiscount: 'vn-item[name="discount"]',
|
moreMenuUpdateDiscount: 'vn-item[name="discount"]',
|
||||||
moreMenuRecalculatePrice: 'vn-item[name="calculatePrice"]',
|
moreMenuRecalculatePrice: 'vn-item[name="calculatePrice"]',
|
||||||
moreMenuPayBack: 'vn-item[name="payBack"]',
|
moreMenuRefund: 'vn-item[name="refund"]',
|
||||||
moreMenuUpdateDiscountInput: 'vn-input-number[ng-model="$ctrl.edit.discount"] input',
|
moreMenuUpdateDiscountInput: 'vn-input-number[ng-model="$ctrl.edit.discount"] input',
|
||||||
transferQuantityInput: '.vn-popover.shown vn-table > div > vn-tbody > vn-tr > vn-td-editable > span > text',
|
transferQuantityInput: '.vn-popover.shown vn-table > div > vn-tbody > vn-tr > vn-td-editable > span > text',
|
||||||
transferQuantityCell: '.vn-popover.shown vn-table > div > vn-tbody > vn-tr > vn-td-editable',
|
transferQuantityCell: '.vn-popover.shown vn-table > div > vn-tbody > vn-tr > vn-td-editable',
|
||||||
|
@ -585,6 +591,7 @@ export default {
|
||||||
firstSalePriceInput: '.vn-popover.shown input[ng-model="$ctrl.field"]',
|
firstSalePriceInput: '.vn-popover.shown input[ng-model="$ctrl.field"]',
|
||||||
firstSaleDiscount: 'vn-ticket-sale vn-table vn-tr:nth-child(1) > vn-td:nth-child(10) > span',
|
firstSaleDiscount: 'vn-ticket-sale vn-table vn-tr:nth-child(1) > vn-td:nth-child(10) > span',
|
||||||
firstSaleDiscountInput: '.vn-popover.shown [ng-model="$ctrl.field"]',
|
firstSaleDiscountInput: '.vn-popover.shown [ng-model="$ctrl.field"]',
|
||||||
|
saveSaleDiscountButton: '.vn-popover.shown vn-button[label="Save"]',
|
||||||
firstSaleImport: 'vn-ticket-sale:nth-child(1) vn-td:nth-child(11)',
|
firstSaleImport: 'vn-ticket-sale:nth-child(1) vn-td:nth-child(11)',
|
||||||
firstSaleReservedIcon: 'vn-ticket-sale vn-tr:nth-child(1) > vn-td:nth-child(2) > vn-icon:nth-child(3)',
|
firstSaleReservedIcon: 'vn-ticket-sale vn-tr:nth-child(1) > vn-td:nth-child(2) > vn-icon:nth-child(3)',
|
||||||
firstSaleColour: 'vn-ticket-sale vn-tr:nth-child(1) vn-fetched-tags section',
|
firstSaleColour: 'vn-ticket-sale vn-tr:nth-child(1) vn-fetched-tags section',
|
||||||
|
@ -957,7 +964,7 @@ export default {
|
||||||
supplierRef: 'vn-invoice-in-summary vn-label-value:nth-child(2) > section > span'
|
supplierRef: 'vn-invoice-in-summary vn-label-value:nth-child(2) > section > span'
|
||||||
},
|
},
|
||||||
invoiceInDescriptor: {
|
invoiceInDescriptor: {
|
||||||
summaryIcon: 'vn-invoice-in-descriptor a[title="Preview"]',
|
summaryIcon: 'vn-invoice-in-descriptor a[title="Go to module summary"]',
|
||||||
moreMenu: 'vn-invoice-in-descriptor vn-icon-button[icon=more_vert]',
|
moreMenu: 'vn-invoice-in-descriptor vn-icon-button[icon=more_vert]',
|
||||||
moreMenuDeleteInvoiceIn: '.vn-menu [name="deleteInvoice"]',
|
moreMenuDeleteInvoiceIn: '.vn-menu [name="deleteInvoice"]',
|
||||||
moreMenuCloneInvoiceIn: '.vn-menu [name="cloneInvoice"]',
|
moreMenuCloneInvoiceIn: '.vn-menu [name="cloneInvoice"]',
|
||||||
|
|
|
@ -28,7 +28,7 @@ describe('Client defaulter path', () => {
|
||||||
const salesPersonName =
|
const salesPersonName =
|
||||||
await page.waitToGetProperty(selectors.clientDefaulter.firstSalesPersonName, 'innerText');
|
await page.waitToGetProperty(selectors.clientDefaulter.firstSalesPersonName, 'innerText');
|
||||||
|
|
||||||
expect(clientName).toEqual('Batman');
|
expect(clientName).toEqual('Ororo Munroe');
|
||||||
expect(salesPersonName).toEqual('salesPersonNick');
|
expect(salesPersonName).toEqual('salesPersonNick');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
import selectors from '../../helpers/selectors.js';
|
||||||
|
import getBrowser from '../../helpers/puppeteer';
|
||||||
|
|
||||||
|
describe('Client unpaid path', () => {
|
||||||
|
let browser;
|
||||||
|
let page;
|
||||||
|
|
||||||
|
beforeAll(async() => {
|
||||||
|
browser = await getBrowser();
|
||||||
|
page = browser.page;
|
||||||
|
await page.loginAndModule('administrative', 'client');
|
||||||
|
await page.accessToSearchResult('Charles Xavier');
|
||||||
|
await page.accessToSection('client.card.unpaid');
|
||||||
|
await page.waitForState('client.card.unpaid');
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(async() => {
|
||||||
|
await browser.close();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should set cliet unpaid', async() => {
|
||||||
|
await page.waitToClick(selectors.clientUnpaid.hasDataCheckBox);
|
||||||
|
|
||||||
|
await page.pickDate(selectors.clientUnpaid.dated);
|
||||||
|
await page.write(selectors.clientUnpaid.amount, '500');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should save unpaid', async() => {
|
||||||
|
await page.waitToClick(selectors.clientUnpaid.saveButton);
|
||||||
|
const message = await page.waitForSnackbar();
|
||||||
|
|
||||||
|
expect(message.text).toContain('Data saved!');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should confirm the unpaid have been saved', async() => {
|
||||||
|
await page.reloadSection('client.card.unpaid');
|
||||||
|
const result = await page.waitToGetProperty(selectors.clientUnpaid.amount, 'value');
|
||||||
|
|
||||||
|
expect(result).toEqual('500');
|
||||||
|
});
|
||||||
|
});
|
|
@ -99,8 +99,8 @@ describe('Worker time control path', () => {
|
||||||
|
|
||||||
expect(result).toEqual(scanTime);
|
expect(result).toEqual(scanTime);
|
||||||
});
|
});
|
||||||
// 3736 check proc vn.timeControl_calculate
|
|
||||||
xit(`should check Hank Pym worked 6:40 hours`, async() => {
|
it(`should check Hank Pym worked 6:40 hours`, async() => {
|
||||||
await page.waitForTextInElement(selectors.workerTimeControl.mondayWorkedHours, '06:40 h.');
|
await page.waitForTextInElement(selectors.workerTimeControl.mondayWorkedHours, '06:40 h.');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -175,7 +175,8 @@ describe('Ticket Edit sale path', () => {
|
||||||
it('should update the discount', async() => {
|
it('should update the discount', async() => {
|
||||||
await page.waitToClick(selectors.ticketSales.firstSaleDiscount);
|
await page.waitToClick(selectors.ticketSales.firstSaleDiscount);
|
||||||
await page.waitForSelector(selectors.ticketSales.firstSaleDiscountInput);
|
await page.waitForSelector(selectors.ticketSales.firstSaleDiscountInput);
|
||||||
await page.type(selectors.ticketSales.firstSaleDiscountInput, '50\u000d');
|
await page.type(selectors.ticketSales.firstSaleDiscountInput, '50');
|
||||||
|
await page.waitToClick(selectors.ticketSales.saveSaleDiscountButton);
|
||||||
const message = await page.waitForSnackbar();
|
const message = await page.waitForSnackbar();
|
||||||
|
|
||||||
expect(message.text).toContain('Data saved!');
|
expect(message.text).toContain('Data saved!');
|
||||||
|
@ -212,10 +213,10 @@ describe('Ticket Edit sale path', () => {
|
||||||
await page.accessToSection('ticket.card.sale');
|
await page.accessToSection('ticket.card.sale');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should select the third sale and create a pay back', async() => {
|
it('should select the third sale and create a refund', async() => {
|
||||||
await page.waitToClick(selectors.ticketSales.firstSaleCheckbox);
|
await page.waitToClick(selectors.ticketSales.firstSaleCheckbox);
|
||||||
await page.waitToClick(selectors.ticketSales.moreMenu);
|
await page.waitToClick(selectors.ticketSales.moreMenu);
|
||||||
await page.waitToClick(selectors.ticketSales.moreMenuPayBack);
|
await page.waitToClick(selectors.ticketSales.moreMenuRefund);
|
||||||
await page.waitForState('ticket.card.sale');
|
await page.waitForState('ticket.card.sale');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -134,7 +134,7 @@ describe('Travel descriptor path', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should navigate to the summary and then clone the travel and its entries using the descriptor menu to get redirected to the cloned travel basic data', async() => {
|
it('should navigate to the summary and then clone the travel and its entries using the descriptor menu to get redirected to the cloned travel basic data', async() => {
|
||||||
await page.waitToClick('vn-icon[icon="preview"]'); // summary icon
|
await page.waitToClick('vn-icon[icon="launch"]');
|
||||||
await page.waitForState('travel.card.summary');
|
await page.waitForState('travel.card.summary');
|
||||||
await page.waitForTimeout(1000);
|
await page.waitForTimeout(1000);
|
||||||
await page.waitToClick(selectors.travelDescriptor.dotMenu);
|
await page.waitToClick(selectors.travelDescriptor.dotMenu);
|
||||||
|
|
|
@ -23,10 +23,21 @@
|
||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.icon-agency-term:before {
|
||||||
|
content: "\e950";
|
||||||
|
}
|
||||||
|
.icon-deaulter:before {
|
||||||
|
content: "\e94b";
|
||||||
|
}
|
||||||
.icon-100:before {
|
.icon-100:before {
|
||||||
content: "\e95a";
|
content: "\e95a";
|
||||||
}
|
}
|
||||||
|
.icon-history:before {
|
||||||
|
content: "\e968";
|
||||||
|
}
|
||||||
|
.icon-Person:before {
|
||||||
|
content: "\e901";
|
||||||
|
}
|
||||||
.icon-accessory:before {
|
.icon-accessory:before {
|
||||||
content: "\e90a";
|
content: "\e90a";
|
||||||
}
|
}
|
||||||
|
@ -74,6 +85,7 @@
|
||||||
}
|
}
|
||||||
.icon-bucket:before {
|
.icon-bucket:before {
|
||||||
content: "\e97a";
|
content: "\e97a";
|
||||||
|
color: #000;
|
||||||
}
|
}
|
||||||
.icon-buscaman:before {
|
.icon-buscaman:before {
|
||||||
content: "\e93b";
|
content: "\e93b";
|
||||||
|
@ -83,26 +95,32 @@
|
||||||
}
|
}
|
||||||
.icon-calc_volum .path1:before {
|
.icon-calc_volum .path1:before {
|
||||||
content: "\e915";
|
content: "\e915";
|
||||||
|
color: rgb(0, 0, 0);
|
||||||
}
|
}
|
||||||
.icon-calc_volum .path2:before {
|
.icon-calc_volum .path2:before {
|
||||||
content: "\e916";
|
content: "\e916";
|
||||||
margin-left: -1em;
|
margin-left: -1em;
|
||||||
|
color: rgb(0, 0, 0);
|
||||||
}
|
}
|
||||||
.icon-calc_volum .path3:before {
|
.icon-calc_volum .path3:before {
|
||||||
content: "\e917";
|
content: "\e917";
|
||||||
margin-left: -1em;
|
margin-left: -1em;
|
||||||
|
color: rgb(0, 0, 0);
|
||||||
}
|
}
|
||||||
.icon-calc_volum .path4:before {
|
.icon-calc_volum .path4:before {
|
||||||
content: "\e918";
|
content: "\e918";
|
||||||
margin-left: -1em;
|
margin-left: -1em;
|
||||||
|
color: rgb(0, 0, 0);
|
||||||
}
|
}
|
||||||
.icon-calc_volum .path5:before {
|
.icon-calc_volum .path5:before {
|
||||||
content: "\e919";
|
content: "\e919";
|
||||||
margin-left: -1em;
|
margin-left: -1em;
|
||||||
|
color: rgb(0, 0, 0);
|
||||||
}
|
}
|
||||||
.icon-calc_volum .path6:before {
|
.icon-calc_volum .path6:before {
|
||||||
content: "\e91a";
|
content: "\e91a";
|
||||||
margin-left: -1em;
|
margin-left: -1em;
|
||||||
|
color: rgb(255, 255, 255);
|
||||||
}
|
}
|
||||||
.icon-calendar:before {
|
.icon-calendar:before {
|
||||||
content: "\e93d";
|
content: "\e93d";
|
||||||
|
@ -137,9 +155,6 @@
|
||||||
.icon-credit:before {
|
.icon-credit:before {
|
||||||
content: "\e927";
|
content: "\e927";
|
||||||
}
|
}
|
||||||
.icon-defaulter:before {
|
|
||||||
content: "\e94b";
|
|
||||||
}
|
|
||||||
.icon-deletedTicket:before {
|
.icon-deletedTicket:before {
|
||||||
content: "\e935";
|
content: "\e935";
|
||||||
}
|
}
|
||||||
|
@ -206,9 +221,6 @@
|
||||||
.icon-headercol:before {
|
.icon-headercol:before {
|
||||||
content: "\e958";
|
content: "\e958";
|
||||||
}
|
}
|
||||||
.icon-history:before {
|
|
||||||
content: "\e968";
|
|
||||||
}
|
|
||||||
.icon-info:before {
|
.icon-info:before {
|
||||||
content: "\e952";
|
content: "\e952";
|
||||||
}
|
}
|
||||||
|
@ -281,9 +293,6 @@
|
||||||
.icon-pbx:before {
|
.icon-pbx:before {
|
||||||
content: "\e93c";
|
content: "\e93c";
|
||||||
}
|
}
|
||||||
.icon-Person:before {
|
|
||||||
content: "\e901";
|
|
||||||
}
|
|
||||||
.icon-pets:before {
|
.icon-pets:before {
|
||||||
content: "\e947";
|
content: "\e947";
|
||||||
}
|
}
|
||||||
|
|
Binary file not shown.
File diff suppressed because one or more lines are too long
Before Width: | Height: | Size: 158 KiB After Width: | Height: | Size: 160 KiB |
Binary file not shown.
Binary file not shown.
|
@ -4,8 +4,14 @@
|
||||||
vn-descriptor-content > .descriptor {
|
vn-descriptor-content > .descriptor {
|
||||||
width: 256px;
|
width: 256px;
|
||||||
|
|
||||||
& > .header > a:first-child {
|
& > .header {
|
||||||
visibility: hidden;
|
a:first-child {
|
||||||
}
|
display: none;
|
||||||
|
}
|
||||||
|
vn-icon-button:nth-child(2) {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,10 +12,15 @@
|
||||||
name="goToModuleIndex">
|
name="goToModuleIndex">
|
||||||
<vn-icon icon="{{$ctrl.moduleMap[$ctrl.module].icon}}"></vn-icon>
|
<vn-icon icon="{{$ctrl.moduleMap[$ctrl.module].icon}}"></vn-icon>
|
||||||
</a>
|
</a>
|
||||||
|
<vn-icon-button
|
||||||
|
translate-attr="{title: 'Show summary'}"
|
||||||
|
icon="preview"
|
||||||
|
vn-click-stop="$ctrl.summary.show()">
|
||||||
|
</vn-icon-button>
|
||||||
<a
|
<a
|
||||||
translate-attr="{title: 'Preview'}"
|
translate-attr="{title: 'Go to module summary'}"
|
||||||
ui-sref="{{::$ctrl.summaryState}}({id: $ctrl.descriptor.id})">
|
ui-sref="{{::$ctrl.summaryState}}({id: $ctrl.descriptor.id})">
|
||||||
<vn-icon icon="preview"></vn-icon>
|
<vn-icon icon="launch"></vn-icon>
|
||||||
</a>
|
</a>
|
||||||
<vn-icon-button ng-if="!$ctrl.$transclude.isSlotFilled('dotMenu')"
|
<vn-icon-button ng-if="!$ctrl.$transclude.isSlotFilled('dotMenu')"
|
||||||
ng-class="::{invisible: !$ctrl.$transclude.isSlotFilled('menu')}"
|
ng-class="::{invisible: !$ctrl.$transclude.isSlotFilled('menu')}"
|
||||||
|
|
|
@ -125,7 +125,8 @@ ngModule.vnComponent('vnDescriptorContent', {
|
||||||
module: '@',
|
module: '@',
|
||||||
baseState: '@?',
|
baseState: '@?',
|
||||||
description: '<',
|
description: '<',
|
||||||
descriptor: '<?'
|
descriptor: '<?',
|
||||||
|
summary: '<?'
|
||||||
},
|
},
|
||||||
transclude: {
|
transclude: {
|
||||||
body: 'slotBody',
|
body: 'slotBody',
|
||||||
|
|
|
@ -60,6 +60,9 @@ vn-descriptor-content {
|
||||||
font-size: 1.75rem;
|
font-size: 1.75rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
& > vn-icon-button:nth-child(2) {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
& > .body {
|
& > .body {
|
||||||
display: block;
|
display: block;
|
||||||
|
|
|
@ -13,6 +13,8 @@ Preview: Vista previa
|
||||||
Profile: Perfil
|
Profile: Perfil
|
||||||
Push on applications menu: Para abrir un módulo pulsa en el menú de aplicaciones
|
Push on applications menu: Para abrir un módulo pulsa en el menú de aplicaciones
|
||||||
Go to module index: Ir al índice del módulo
|
Go to module index: Ir al índice del módulo
|
||||||
|
Go to module summary: Ir a la vista previa del módulo
|
||||||
|
Show summary: Mostrar vista previa
|
||||||
What is new: Novedades de la versión
|
What is new: Novedades de la versión
|
||||||
Settings: Ajustes
|
Settings: Ajustes
|
||||||
|
|
||||||
|
|
|
@ -219,7 +219,7 @@
|
||||||
"The worker has a marked absence that day": "El trabajador tiene marcada una ausencia ese día",
|
"The worker has a marked absence that day": "El trabajador tiene marcada una ausencia ese día",
|
||||||
"You can not modify is pay method checked": "No se puede modificar el campo método de pago validado",
|
"You can not modify is pay method checked": "No se puede modificar el campo método de pago validado",
|
||||||
"Can't transfer claimed sales": "No puedes transferir lineas reclamadas",
|
"Can't transfer claimed sales": "No puedes transferir lineas reclamadas",
|
||||||
"You don't have privileges to create pay back": "No tienes permisos para crear un abono",
|
"You don't have privileges to create refund": "No tienes permisos para crear un abono",
|
||||||
"The item is required": "El artículo es requerido",
|
"The item is required": "El artículo es requerido",
|
||||||
"The agency is already assigned to another autonomous": "La agencia ya está asignada a otro autónomo",
|
"The agency is already assigned to another autonomous": "La agencia ya está asignada a otro autónomo",
|
||||||
"date in the future": "Fecha en el futuro",
|
"date in the future": "Fecha en el futuro",
|
||||||
|
|
|
@ -15,7 +15,8 @@
|
||||||
"legacyUtcDateProcessing": false,
|
"legacyUtcDateProcessing": false,
|
||||||
"timezone": "local",
|
"timezone": "local",
|
||||||
"connectTimeout": 40000,
|
"connectTimeout": 40000,
|
||||||
"acquireTimeout": 20000
|
"acquireTimeout": 20000,
|
||||||
|
"waitForConnections": true
|
||||||
},
|
},
|
||||||
"osticket": {
|
"osticket": {
|
||||||
"connector": "memory",
|
"connector": "memory",
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<vn-descriptor-content
|
<vn-descriptor-content
|
||||||
module="account"
|
module="account"
|
||||||
description="$ctrl.user.nickname">
|
description="$ctrl.user.nickname"
|
||||||
|
summary="$ctrl.$.summary">
|
||||||
<slot-menu>
|
<slot-menu>
|
||||||
<vn-item
|
<vn-item
|
||||||
ng-click="deleteUser.show()"
|
ng-click="deleteUser.show()"
|
||||||
|
@ -173,3 +174,6 @@
|
||||||
<button response="accept" translate>Change password</button>
|
<button response="accept" translate>Change password</button>
|
||||||
</tpl-buttons>
|
</tpl-buttons>
|
||||||
</vn-dialog>
|
</vn-dialog>
|
||||||
|
<vn-popup vn-id="summary">
|
||||||
|
<vn-user-summary user="$ctrl.user"></vn-user-summary>
|
||||||
|
</vn-popup>
|
|
@ -27,7 +27,7 @@ module.exports = Self => {
|
||||||
http: {source: 'query'}
|
http: {source: 'query'}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
arg: 'client',
|
arg: 'clientName',
|
||||||
type: 'string',
|
type: 'string',
|
||||||
description: 'The worker name',
|
description: 'The worker name',
|
||||||
http: {source: 'query'}
|
http: {source: 'query'}
|
||||||
|
@ -94,11 +94,11 @@ module.exports = Self => {
|
||||||
? {'cl.id': value}
|
? {'cl.id': value}
|
||||||
: {
|
: {
|
||||||
or: [
|
or: [
|
||||||
{'cl.socialName': {like: `%${value}%`}}
|
{'cl.clientName': {like: `%${value}%`}}
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
case 'client':
|
case 'clientName':
|
||||||
return {'cl.socialName': {like: `%${value}%`}};
|
return {'cl.clientName': {like: `%${value}%`}};
|
||||||
case 'clientFk':
|
case 'clientFk':
|
||||||
return {'cl.clientFk': value};
|
return {'cl.clientFk': value};
|
||||||
case 'id':
|
case 'id':
|
||||||
|
@ -128,7 +128,7 @@ module.exports = Self => {
|
||||||
SELECT
|
SELECT
|
||||||
cl.id,
|
cl.id,
|
||||||
cl.clientFk,
|
cl.clientFk,
|
||||||
c.socialName,
|
c.name AS clientName,
|
||||||
cl.workerFk,
|
cl.workerFk,
|
||||||
u.name AS workerName,
|
u.name AS workerName,
|
||||||
cs.description,
|
cs.description,
|
||||||
|
|
|
@ -25,7 +25,7 @@ describe('claim filter()', () => {
|
||||||
try {
|
try {
|
||||||
const options = {transaction: tx};
|
const options = {transaction: tx};
|
||||||
|
|
||||||
const result = await app.models.Claim.filter({args: {filter: {}, search: 'Iron man'}}, null, options);
|
const result = await app.models.Claim.filter({args: {filter: {}, search: 'Tony Stark'}}, null, options);
|
||||||
|
|
||||||
expect(result.length).toEqual(1);
|
expect(result.length).toEqual(1);
|
||||||
expect(result[0].id).toEqual(4);
|
expect(result[0].id).toEqual(4);
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<vn-descriptor-content
|
<vn-descriptor-content
|
||||||
module="claim"
|
module="claim"
|
||||||
description="$ctrl.claim.client.name">
|
description="$ctrl.claim.client.name"
|
||||||
|
summary="$ctrl.$.summary">
|
||||||
<slot-menu>
|
<slot-menu>
|
||||||
<vn-item
|
<vn-item
|
||||||
ng-click="$ctrl.showPickupOrder()"
|
ng-click="$ctrl.showPickupOrder()"
|
||||||
|
@ -96,4 +97,7 @@
|
||||||
</vn-worker-descriptor-popover>
|
</vn-worker-descriptor-popover>
|
||||||
<vn-ticket-descriptor-popover
|
<vn-ticket-descriptor-popover
|
||||||
vn-id="ticketDescriptor">
|
vn-id="ticketDescriptor">
|
||||||
</vn-ticket-descriptor-popover>
|
</vn-ticket-descriptor-popover>
|
||||||
|
<vn-popup vn-id="summary">
|
||||||
|
<vn-claim-summary claim="$ctrl.claim"></vn-claim-summary>
|
||||||
|
</vn-popup>
|
|
@ -10,16 +10,16 @@
|
||||||
<table>
|
<table>
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th field="id" shrink>
|
<th field="clientFk" shrink>
|
||||||
<span translate>Id</span>
|
<span translate>Id</span>
|
||||||
</th>
|
</th>
|
||||||
<th field="clientFk">
|
<th field="clientName">
|
||||||
<span translate>Client</span>
|
<span translate>Client</span>
|
||||||
</th>
|
</th>
|
||||||
<th field="created" center shrink-date>
|
<th field="created" center shrink-date>
|
||||||
<span translate>Created</span>
|
<span translate>Created</span>
|
||||||
</th>
|
</th>
|
||||||
<th field="salesPersonFk">
|
<th field="workerFk">
|
||||||
<span translate>Worker</span>
|
<span translate>Worker</span>
|
||||||
</th>
|
</th>
|
||||||
<th field="claimStateFk">
|
<th field="claimStateFk">
|
||||||
|
@ -40,7 +40,7 @@
|
||||||
<span
|
<span
|
||||||
vn-click-stop="clientDescriptor.show($event, claim.clientFk)"
|
vn-click-stop="clientDescriptor.show($event, claim.clientFk)"
|
||||||
class="link">
|
class="link">
|
||||||
{{::claim.socialName}}
|
{{::claim.clientName}}
|
||||||
</span>
|
</span>
|
||||||
</td>
|
</td>
|
||||||
<td center shrink-date>{{::claim.created | date:'dd/MM/yyyy'}}</td>
|
<td center shrink-date>{{::claim.created | date:'dd/MM/yyyy'}}</td>
|
||||||
|
|
|
@ -11,11 +11,11 @@ class Controller extends Section {
|
||||||
},
|
},
|
||||||
columns: [
|
columns: [
|
||||||
{
|
{
|
||||||
field: 'clientFk',
|
field: 'clientName',
|
||||||
autocomplete: {
|
autocomplete: {
|
||||||
url: 'Clients',
|
url: 'Clients',
|
||||||
showField: 'socialName',
|
showField: 'name',
|
||||||
valueField: 'socialName'
|
valueField: 'name'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -46,21 +46,12 @@ class Controller extends Section {
|
||||||
|
|
||||||
exprBuilder(param, value) {
|
exprBuilder(param, value) {
|
||||||
switch (param) {
|
switch (param) {
|
||||||
|
case 'clientName':
|
||||||
|
return {'cl.clientName': {like: `%${value}%`}};
|
||||||
case 'clientFk':
|
case 'clientFk':
|
||||||
return {['cl.socialName']: value};
|
|
||||||
case 'id':
|
|
||||||
case 'claimStateFk':
|
case 'claimStateFk':
|
||||||
case 'priority':
|
case 'workerFk':
|
||||||
return {[`cl.${param}`]: value};
|
return {[`cl.${param}`]: value};
|
||||||
case 'salesPersonFk':
|
|
||||||
case 'attenderFk':
|
|
||||||
return {'cl.workerFk': value};
|
|
||||||
case 'created':
|
|
||||||
value.setHours(0, 0, 0, 0);
|
|
||||||
to = new Date(value);
|
|
||||||
to.setHours(23, 59, 59, 999);
|
|
||||||
|
|
||||||
return {'cl.created': {between: [value, to]}};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
<vn-textfield
|
<vn-textfield
|
||||||
vn-one
|
vn-one
|
||||||
label="Client"
|
label="Client"
|
||||||
ng-model="filter.client">
|
ng-model="filter.clientName">
|
||||||
</vn-textfield>
|
</vn-textfield>
|
||||||
</vn-horizontal>
|
</vn-horizontal>
|
||||||
<vn-horizontal>
|
<vn-horizontal>
|
||||||
|
|
|
@ -6,6 +6,12 @@ describe('Address updateAddress', () => {
|
||||||
const provinceId = 5;
|
const provinceId = 5;
|
||||||
const incotermsId = 'FAS';
|
const incotermsId = 'FAS';
|
||||||
const customAgentOneId = 1;
|
const customAgentOneId = 1;
|
||||||
|
const employeeId = 1;
|
||||||
|
const ctx = {
|
||||||
|
req: {
|
||||||
|
accessToken: {userId: employeeId}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
it('should throw the non uee member error if no incoterms is defined', async() => {
|
it('should throw the non uee member error if no incoterms is defined', async() => {
|
||||||
const tx = await models.Client.beginTransaction({});
|
const tx = await models.Client.beginTransaction({});
|
||||||
|
@ -14,11 +20,9 @@ describe('Address updateAddress', () => {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const options = {transaction: tx};
|
const options = {transaction: tx};
|
||||||
const ctx = {
|
ctx.args = {
|
||||||
args: {
|
provinceFk: provinceId,
|
||||||
provinceFk: provinceId,
|
customsAgentFk: customAgentOneId
|
||||||
customsAgentFk: customAgentOneId
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
||||||
|
@ -40,11 +44,9 @@ describe('Address updateAddress', () => {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const options = {transaction: tx};
|
const options = {transaction: tx};
|
||||||
const ctx = {
|
ctx.args = {
|
||||||
args: {
|
provinceFk: provinceId,
|
||||||
provinceFk: provinceId,
|
incotermsFk: incotermsId
|
||||||
incotermsFk: incotermsId
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
||||||
|
@ -66,13 +68,11 @@ describe('Address updateAddress', () => {
|
||||||
const options = {transaction: tx};
|
const options = {transaction: tx};
|
||||||
|
|
||||||
const expectedResult = 'My edited address';
|
const expectedResult = 'My edited address';
|
||||||
const ctx = {
|
ctx.args = {
|
||||||
args: {
|
provinceFk: provinceId,
|
||||||
provinceFk: provinceId,
|
nickname: expectedResult,
|
||||||
nickname: expectedResult,
|
incotermsFk: incotermsId,
|
||||||
incotermsFk: incotermsId,
|
customsAgentFk: customAgentOneId
|
||||||
customsAgentFk: customAgentOneId
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
||||||
|
@ -88,6 +88,48 @@ describe('Address updateAddress', () => {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should return an error for a user without enough privileges', async() => {
|
||||||
|
const tx = await models.Client.beginTransaction({});
|
||||||
|
|
||||||
|
try {
|
||||||
|
const options = {transaction: tx};
|
||||||
|
ctx.args = {
|
||||||
|
isLogifloraAllowed: true
|
||||||
|
};
|
||||||
|
|
||||||
|
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
||||||
|
await tx.rollback();
|
||||||
|
} catch (e) {
|
||||||
|
await tx.rollback();
|
||||||
|
error = e;
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(error.message).toEqual(`You don't have enough privileges`);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should update isLogifloraAllowed', async() => {
|
||||||
|
const tx = await models.Client.beginTransaction({});
|
||||||
|
const salesAssistantId = 21;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const options = {transaction: tx};
|
||||||
|
ctx.req.accessToken.userId = salesAssistantId;
|
||||||
|
ctx.args = {
|
||||||
|
isLogifloraAllowed: true
|
||||||
|
};
|
||||||
|
|
||||||
|
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
||||||
|
const address = await models.Address.findById(addressId, null, options);
|
||||||
|
|
||||||
|
expect(address.isLogifloraAllowed).toEqual(true);
|
||||||
|
|
||||||
|
await tx.rollback();
|
||||||
|
} catch (e) {
|
||||||
|
await tx.rollback();
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
it('should update the address', async() => {
|
it('should update the address', async() => {
|
||||||
const tx = await models.Client.beginTransaction({});
|
const tx = await models.Client.beginTransaction({});
|
||||||
|
|
||||||
|
@ -95,10 +137,8 @@ describe('Address updateAddress', () => {
|
||||||
const options = {transaction: tx};
|
const options = {transaction: tx};
|
||||||
|
|
||||||
const expectedResult = 'My second time edited address';
|
const expectedResult = 'My second time edited address';
|
||||||
const ctx = {
|
ctx.args = {
|
||||||
args: {
|
nickname: expectedResult
|
||||||
nickname: expectedResult
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
await models.Client.updateAddress(ctx, clientId, addressId, options);
|
||||||
|
|
|
@ -68,6 +68,10 @@ module.exports = function(Self) {
|
||||||
{
|
{
|
||||||
arg: 'isEqualizated',
|
arg: 'isEqualizated',
|
||||||
type: 'boolean'
|
type: 'boolean'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
arg: 'isLogifloraAllowed',
|
||||||
|
type: 'boolean'
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
returns: {
|
returns: {
|
||||||
|
@ -83,11 +87,16 @@ module.exports = function(Self) {
|
||||||
Self.updateAddress = async(ctx, clientId, addressId, options) => {
|
Self.updateAddress = async(ctx, clientId, addressId, options) => {
|
||||||
const models = Self.app.models;
|
const models = Self.app.models;
|
||||||
const args = ctx.args;
|
const args = ctx.args;
|
||||||
|
const userId = ctx.req.accessToken.userId;
|
||||||
const myOptions = {};
|
const myOptions = {};
|
||||||
|
const isSalesAssistant = await models.Account.hasRole(userId, 'salesAssistant', myOptions);
|
||||||
|
|
||||||
if (typeof options == 'object')
|
if (typeof options == 'object')
|
||||||
Object.assign(myOptions, options);
|
Object.assign(myOptions, options);
|
||||||
|
|
||||||
|
if (args.isLogifloraAllowed && !isSalesAssistant)
|
||||||
|
throw new UserError(`You don't have enough privileges`);
|
||||||
|
|
||||||
const address = await models.Address.findOne({
|
const address = await models.Address.findOne({
|
||||||
where: {
|
where: {
|
||||||
id: addressId,
|
id: addressId,
|
||||||
|
|
|
@ -56,7 +56,7 @@ module.exports = Self => {
|
||||||
FROM (
|
FROM (
|
||||||
SELECT
|
SELECT
|
||||||
DISTINCT c.id clientFk,
|
DISTINCT c.id clientFk,
|
||||||
c.socialName clientName,
|
c.name clientName,
|
||||||
c.salesPersonFk,
|
c.salesPersonFk,
|
||||||
u.nickname salesPersonName,
|
u.nickname salesPersonName,
|
||||||
d.amount,
|
d.amount,
|
||||||
|
|
|
@ -47,12 +47,12 @@ describe('defaulter filter()', () => {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const options = {transaction: tx};
|
const options = {transaction: tx};
|
||||||
const ctx = {req: {accessToken: {userId: authUserId}}, args: {search: 'spider'}};
|
const ctx = {req: {accessToken: {userId: authUserId}}, args: {search: 'Petter Parker'}};
|
||||||
|
|
||||||
const result = await models.Defaulter.filter(ctx, null, options);
|
const result = await models.Defaulter.filter(ctx, null, options);
|
||||||
const firstRow = result[0];
|
const firstRow = result[0];
|
||||||
|
|
||||||
expect(firstRow.clientName).toEqual('Spider man');
|
expect(firstRow.clientName).toEqual('Petter Parker');
|
||||||
|
|
||||||
await tx.rollback();
|
await tx.rollback();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|
|
@ -44,6 +44,9 @@
|
||||||
"ClientType": {
|
"ClientType": {
|
||||||
"dataSource": "vn"
|
"dataSource": "vn"
|
||||||
},
|
},
|
||||||
|
"ClientUnpaid": {
|
||||||
|
"dataSource": "vn"
|
||||||
|
},
|
||||||
"Defaulter": {
|
"Defaulter": {
|
||||||
"dataSource": "vn"
|
"dataSource": "vn"
|
||||||
},
|
},
|
||||||
|
|
|
@ -50,6 +50,9 @@
|
||||||
},
|
},
|
||||||
"isEqualizated": {
|
"isEqualizated": {
|
||||||
"type": "boolean"
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"isLogifloraAllowed": {
|
||||||
|
"type": "boolean"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"validations": [],
|
"validations": [],
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
{
|
||||||
|
"name": "ClientUnpaid",
|
||||||
|
"base": "VnModel",
|
||||||
|
"options": {
|
||||||
|
"mysql": {
|
||||||
|
"table": "clientUnpaid"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"properties": {
|
||||||
|
"clientFk": {
|
||||||
|
"type": "number",
|
||||||
|
"id": true
|
||||||
|
},
|
||||||
|
"dated": {
|
||||||
|
"type": "date"
|
||||||
|
},
|
||||||
|
"amount": {
|
||||||
|
"type": "Number"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"relations": {
|
||||||
|
"client": {
|
||||||
|
"type": "belongsTo",
|
||||||
|
"model": "Client",
|
||||||
|
"foreignKey": "clientFk"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -38,7 +38,13 @@
|
||||||
label="Is equalizated"
|
label="Is equalizated"
|
||||||
ng-model="$ctrl.address.isEqualizated"
|
ng-model="$ctrl.address.isEqualizated"
|
||||||
vn-acl="administrative, salesAssistant">
|
vn-acl="administrative, salesAssistant">
|
||||||
</vn-check>
|
</vn-check>
|
||||||
|
<vn-check
|
||||||
|
vn-one
|
||||||
|
label="Is Logiflora allowed"
|
||||||
|
ng-model="$ctrl.address.isLogifloraAllowed"
|
||||||
|
vn-acl="salesAssistant">
|
||||||
|
</vn-check>
|
||||||
</vn-horizontal>
|
</vn-horizontal>
|
||||||
<vn-horizontal>
|
<vn-horizontal>
|
||||||
<vn-textfield
|
<vn-textfield
|
||||||
|
|
|
@ -60,6 +60,11 @@
|
||||||
ng-model="address.isEqualizated"
|
ng-model="address.isEqualizated"
|
||||||
disabled="true">
|
disabled="true">
|
||||||
</vn-check>
|
</vn-check>
|
||||||
|
<vn-check
|
||||||
|
vn-one label="Is Logiflora allowed"
|
||||||
|
ng-model="address.isLogifloraAllowed"
|
||||||
|
disabled="true">
|
||||||
|
</vn-check>
|
||||||
</vn-one>
|
</vn-one>
|
||||||
<vn-vertical
|
<vn-vertical
|
||||||
vn-one
|
vn-one
|
||||||
|
|
|
@ -17,6 +17,7 @@ class Controller extends Section {
|
||||||
'phone',
|
'phone',
|
||||||
'mobile',
|
'mobile',
|
||||||
'isEqualizated',
|
'isEqualizated',
|
||||||
|
'isLogifloraAllowed',
|
||||||
'postalCode'
|
'postalCode'
|
||||||
],
|
],
|
||||||
order: [
|
order: [
|
||||||
|
|
|
@ -26,4 +26,5 @@ Mobile: Móvil
|
||||||
|
|
||||||
# Common
|
# Common
|
||||||
Fiscal name: Nombre fiscal
|
Fiscal name: Nombre fiscal
|
||||||
Street: Dirección fiscal
|
Street: Dirección fiscal
|
||||||
|
Is Logiflora allowed: Compra directa en Holanda
|
|
@ -6,8 +6,11 @@ class Controller extends Dialog {
|
||||||
super($element, $, $transclude);
|
super($element, $, $transclude);
|
||||||
|
|
||||||
this.vnReport = vnReport;
|
this.vnReport = vnReport;
|
||||||
|
|
||||||
|
const tomorrow = new Date();
|
||||||
|
tomorrow.setDate(tomorrow.getDate() + 1);
|
||||||
this.receipt = {
|
this.receipt = {
|
||||||
payed: new Date()
|
payed: tomorrow
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -49,7 +49,7 @@
|
||||||
model="model">
|
model="model">
|
||||||
</vn-multi-check>
|
</vn-multi-check>
|
||||||
</th>
|
</th>
|
||||||
<th field="clientName">
|
<th field="clientFk">
|
||||||
<span translate>Client</span>
|
<span translate>Client</span>
|
||||||
</th>
|
</th>
|
||||||
<th field="salesPersonFk">
|
<th field="salesPersonFk">
|
||||||
|
|
|
@ -13,11 +13,11 @@ export default class Controller extends Section {
|
||||||
},
|
},
|
||||||
columns: [
|
columns: [
|
||||||
{
|
{
|
||||||
field: 'clientName',
|
field: 'clientFk',
|
||||||
autocomplete: {
|
autocomplete: {
|
||||||
url: 'Clients',
|
url: 'Clients',
|
||||||
showField: 'socialName',
|
showField: 'name',
|
||||||
valueField: 'socialName'
|
valueField: 'id'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -114,7 +114,7 @@ export default class Controller extends Section {
|
||||||
switch (param) {
|
switch (param) {
|
||||||
case 'creditInsurance':
|
case 'creditInsurance':
|
||||||
case 'amount':
|
case 'amount':
|
||||||
case 'clientName':
|
case 'clientFk':
|
||||||
case 'workerFk':
|
case 'workerFk':
|
||||||
case 'salesPersonFk':
|
case 'salesPersonFk':
|
||||||
return {[`d.${param}`]: value};
|
return {[`d.${param}`]: value};
|
||||||
|
|
|
@ -104,15 +104,15 @@ describe('client defaulter', () => {
|
||||||
|
|
||||||
describe('exprBuilder()', () => {
|
describe('exprBuilder()', () => {
|
||||||
it('should search by sales person', () => {
|
it('should search by sales person', () => {
|
||||||
let expr = controller.exprBuilder('salesPersonFk', '5');
|
const expr = controller.exprBuilder('salesPersonFk', '5');
|
||||||
|
|
||||||
expect(expr).toEqual({'d.salesPersonFk': '5'});
|
expect(expr).toEqual({'d.salesPersonFk': '5'});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should search by client name', () => {
|
it('should search by client', () => {
|
||||||
let expr = controller.exprBuilder('clientName', '1foo');
|
const expr = controller.exprBuilder('clientFk', '5');
|
||||||
|
|
||||||
expect(expr).toEqual({'d.clientName': '1foo'});
|
expect(expr).toEqual({'d.clientFk': '5'});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<vn-descriptor-content
|
<vn-descriptor-content
|
||||||
module="client"
|
module="client"
|
||||||
description="$ctrl.client.name">
|
description="$ctrl.client.name"
|
||||||
|
summary="$ctrl.$.summary">
|
||||||
<slot-menu>
|
<slot-menu>
|
||||||
<a class="vn-item"
|
<a class="vn-item"
|
||||||
ui-sref="ticket.create({clientFk: $ctrl.client.id})"
|
ui-sref="ticket.create({clientFk: $ctrl.client.id})"
|
||||||
|
@ -118,4 +119,7 @@
|
||||||
</vn-client-sms>
|
</vn-client-sms>
|
||||||
<vn-worker-descriptor-popover
|
<vn-worker-descriptor-popover
|
||||||
vn-id="workerDescriptor">
|
vn-id="workerDescriptor">
|
||||||
</vn-worker-descriptor-popover>
|
</vn-worker-descriptor-popover>
|
||||||
|
<vn-popup vn-id="summary">
|
||||||
|
<vn-client-summary client="$ctrl.client"></vn-client-summary>
|
||||||
|
</vn-popup>
|
||||||
|
|
|
@ -46,3 +46,4 @@ import './consumption';
|
||||||
import './consumption-search-panel';
|
import './consumption-search-panel';
|
||||||
import './defaulter';
|
import './defaulter';
|
||||||
import './notification';
|
import './notification';
|
||||||
|
import './unpaid';
|
||||||
|
|
|
@ -61,4 +61,5 @@ Log: Historial
|
||||||
Consumption: Consumo
|
Consumption: Consumo
|
||||||
Compensation Account: Cuenta para compensar
|
Compensation Account: Cuenta para compensar
|
||||||
Amount to return: Cantidad a devolver
|
Amount to return: Cantidad a devolver
|
||||||
Delivered amount: Cantidad entregada
|
Delivered amount: Cantidad entregada
|
||||||
|
Unpaid: Impagado
|
|
@ -23,6 +23,7 @@ export default class Client extends ModuleMain {
|
||||||
case 'id':
|
case 'id':
|
||||||
case 'fi':
|
case 'fi':
|
||||||
case 'postcode':
|
case 'postcode':
|
||||||
|
case 'provinceFk':
|
||||||
case 'salesPersonFk':
|
case 'salesPersonFk':
|
||||||
return {[param]: value};
|
return {[param]: value};
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,7 +32,8 @@
|
||||||
{"state": "client.card.creditInsurance.index", "icon": "icon-solunion"},
|
{"state": "client.card.creditInsurance.index", "icon": "icon-solunion"},
|
||||||
{"state": "client.card.contact", "icon": "contact_phone"},
|
{"state": "client.card.contact", "icon": "contact_phone"},
|
||||||
{"state": "client.card.webPayment", "icon": "icon-onlinepayment"},
|
{"state": "client.card.webPayment", "icon": "icon-onlinepayment"},
|
||||||
{"state": "client.card.dms.index", "icon": "cloud_upload"}
|
{"state": "client.card.dms.index", "icon": "cloud_upload"},
|
||||||
|
{"state": "client.card.unpaid", "icon": "icon-defaulter"}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -374,6 +375,12 @@
|
||||||
"state": "client.notification",
|
"state": "client.notification",
|
||||||
"component": "vn-client-notification",
|
"component": "vn-client-notification",
|
||||||
"description": "Notifications"
|
"description": "Notifications"
|
||||||
|
}, {
|
||||||
|
"url": "/unpaid",
|
||||||
|
"state": "client.card.unpaid",
|
||||||
|
"component": "vn-client-unpaid",
|
||||||
|
"acl": ["administrative"],
|
||||||
|
"description": "Unpaid"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,6 +50,15 @@
|
||||||
ng-model="filter.postcode">
|
ng-model="filter.postcode">
|
||||||
</vn-textfield>
|
</vn-textfield>
|
||||||
</vn-horizontal>
|
</vn-horizontal>
|
||||||
|
<vn-horizontal>
|
||||||
|
<vn-autocomplete
|
||||||
|
ng-model="filter.provinceFk"
|
||||||
|
url="Provinces"
|
||||||
|
show-field="name"
|
||||||
|
value-field="id"
|
||||||
|
label="Province">
|
||||||
|
</vn-autocomplete>
|
||||||
|
</vn-horizontal>
|
||||||
<vn-horizontal>
|
<vn-horizontal>
|
||||||
<vn-textfield
|
<vn-textfield
|
||||||
vn-one
|
vn-one
|
||||||
|
|
|
@ -0,0 +1,51 @@
|
||||||
|
<div>
|
||||||
|
<vn-watcher
|
||||||
|
vn-id="watcher"
|
||||||
|
url="ClientUnpaids"
|
||||||
|
data="$ctrl.clientUnpaid"
|
||||||
|
id-value="$ctrl.$params.id"
|
||||||
|
id-field="clientFk"
|
||||||
|
form="form">
|
||||||
|
</vn-watcher>
|
||||||
|
<form
|
||||||
|
name="form"
|
||||||
|
ng-submit="watcher.submit()"
|
||||||
|
class="vn-w-md">
|
||||||
|
<vn-card class="vn-pa-lg">
|
||||||
|
<vn-vertical>
|
||||||
|
<vn-check
|
||||||
|
label="Unpaid client"
|
||||||
|
ng-model="watcher.hasData"
|
||||||
|
on-change="$ctrl.setDefaultDate(watcher.hasData)">
|
||||||
|
</vn-check>
|
||||||
|
</vn-vertical>
|
||||||
|
<vn-horizontal
|
||||||
|
ng-if="watcher.hasData">
|
||||||
|
<vn-date-picker
|
||||||
|
label="Date"
|
||||||
|
ng-model="$ctrl.clientUnpaid.dated"
|
||||||
|
vn-focus>
|
||||||
|
</vn-date-picker>
|
||||||
|
<vn-input-number
|
||||||
|
vn-focus
|
||||||
|
label="Amount"
|
||||||
|
ng-model="$ctrl.clientUnpaid.amount"
|
||||||
|
step="0.01"
|
||||||
|
required>
|
||||||
|
</vn-input-number>
|
||||||
|
</vn-horizontal>
|
||||||
|
</vn-card>
|
||||||
|
<vn-button-bar>
|
||||||
|
<vn-submit
|
||||||
|
disabled="!watcher.dataChanged()"
|
||||||
|
label="Save">
|
||||||
|
</vn-submit>
|
||||||
|
<vn-button
|
||||||
|
class="cancel"
|
||||||
|
label="Undo changes"
|
||||||
|
disabled="!watcher.dataChanged()"
|
||||||
|
ng-click="watcher.loadOriginalData()">
|
||||||
|
</vn-button>
|
||||||
|
</vn-button-bar>
|
||||||
|
</form>
|
||||||
|
</div>
|
|
@ -0,0 +1,14 @@
|
||||||
|
import ngModule from '../module';
|
||||||
|
import Section from 'salix/components/section';
|
||||||
|
|
||||||
|
export default class Controller extends Section {
|
||||||
|
setDefaultDate(hasData) {
|
||||||
|
if (hasData && !this.clientUnpaid.dated)
|
||||||
|
this.clientUnpaid.dated = new Date();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ngModule.vnComponent('vnClientUnpaid', {
|
||||||
|
template: require('./index.html'),
|
||||||
|
controller: Controller
|
||||||
|
});
|
|
@ -0,0 +1,38 @@
|
||||||
|
import './index';
|
||||||
|
|
||||||
|
describe('client unpaid', () => {
|
||||||
|
describe('Component vnClientUnpaid', () => {
|
||||||
|
let controller;
|
||||||
|
|
||||||
|
beforeEach(ngModule('client'));
|
||||||
|
|
||||||
|
beforeEach(inject($componentController => {
|
||||||
|
const $element = angular.element('<vn-client-unpaid></vn-client-unpaid>');
|
||||||
|
controller = $componentController('vnClientUnpaid', {$element});
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe('setDefaultDate()', () => {
|
||||||
|
it(`should not set today date if has dated`, () => {
|
||||||
|
const hasData = true;
|
||||||
|
const yesterday = new Date();
|
||||||
|
yesterday.setDate(yesterday.getDate() - 1);
|
||||||
|
|
||||||
|
controller.clientUnpaid = {
|
||||||
|
dated: yesterday
|
||||||
|
};
|
||||||
|
controller.setDefaultDate(hasData);
|
||||||
|
|
||||||
|
expect(controller.clientUnpaid.dated).toEqual(yesterday);
|
||||||
|
});
|
||||||
|
|
||||||
|
it(`should set today if not has dated`, () => {
|
||||||
|
const hasData = true;
|
||||||
|
|
||||||
|
controller.clientUnpaid = {};
|
||||||
|
controller.setDefaultDate(hasData);
|
||||||
|
|
||||||
|
expect(controller.clientUnpaid.dated).toBeDefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1 @@
|
||||||
|
Unpaid client: Cliente impagado
|
|
@ -1,6 +1,7 @@
|
||||||
<vn-descriptor-content
|
<vn-descriptor-content
|
||||||
module="entry"
|
module="entry"
|
||||||
description="$ctrl.entry.supplier.nickname">
|
description="$ctrl.entry.supplier.nickname"
|
||||||
|
summary="$ctrl.$.summary">
|
||||||
<slot-menu>
|
<slot-menu>
|
||||||
<vn-item
|
<vn-item
|
||||||
ng-click="$ctrl.showEntryReport()"
|
ng-click="$ctrl.showEntryReport()"
|
||||||
|
@ -59,3 +60,6 @@
|
||||||
</div>
|
</div>
|
||||||
</slot-body>
|
</slot-body>
|
||||||
</vn-descriptor-content>
|
</vn-descriptor-content>
|
||||||
|
<vn-popup vn-id="summary">
|
||||||
|
<vn-entry-summary entry="$ctrl.entry"></vn-entry-summary>
|
||||||
|
</vn-popup>
|
|
@ -64,6 +64,34 @@ module.exports = Self => {
|
||||||
}]
|
}]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
relation: 'invoiceInIntrastat',
|
||||||
|
scope: {
|
||||||
|
fields: [
|
||||||
|
'id',
|
||||||
|
'invoiceInFk',
|
||||||
|
'net',
|
||||||
|
'intrastatFk',
|
||||||
|
'amount',
|
||||||
|
'stems',
|
||||||
|
'countryFk',
|
||||||
|
'statisticalValue'],
|
||||||
|
include: [{
|
||||||
|
relation: 'intrastat',
|
||||||
|
scope: {
|
||||||
|
fields: [
|
||||||
|
'id',
|
||||||
|
'description']
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
relation: 'country',
|
||||||
|
scope: {
|
||||||
|
fields: ['code']
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
relation: 'invoiceInTax',
|
relation: 'invoiceInTax',
|
||||||
scope: {
|
scope: {
|
||||||
|
|
|
@ -8,6 +8,9 @@
|
||||||
"InvoiceInDueDay": {
|
"InvoiceInDueDay": {
|
||||||
"dataSource": "vn"
|
"dataSource": "vn"
|
||||||
},
|
},
|
||||||
|
"InvoiceInIntrastat": {
|
||||||
|
"dataSource": "vn"
|
||||||
|
},
|
||||||
"InvoiceInLog": {
|
"InvoiceInLog": {
|
||||||
"dataSource": "vn"
|
"dataSource": "vn"
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
{
|
||||||
|
"name": "InvoiceInIntrastat",
|
||||||
|
"base": "VnModel",
|
||||||
|
"options": {
|
||||||
|
"mysql": {
|
||||||
|
"table": "invoiceInIntrastat"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"properties": {
|
||||||
|
"id": {
|
||||||
|
"id": true,
|
||||||
|
"type": "number",
|
||||||
|
"description": "Identifier"
|
||||||
|
},
|
||||||
|
"invoiceInFk": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"net": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"intrastatFk": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"amount": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"stems": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"countryFk": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"statisticalValue": {
|
||||||
|
"type": "number"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"relations": {
|
||||||
|
"intrastat": {
|
||||||
|
"type": "belongsTo",
|
||||||
|
"model": "Intrastat",
|
||||||
|
"foreignKey": "intrastatFk"
|
||||||
|
},
|
||||||
|
"country": {
|
||||||
|
"type": "belongsTo",
|
||||||
|
"model": "Country",
|
||||||
|
"foreignKey": "countryFk"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -64,6 +64,11 @@
|
||||||
"model": "InvoiceInDueDay",
|
"model": "InvoiceInDueDay",
|
||||||
"foreignKey": "invoiceInFk"
|
"foreignKey": "invoiceInFk"
|
||||||
},
|
},
|
||||||
|
"invoiceInIntrastat": {
|
||||||
|
"type": "hasMany",
|
||||||
|
"model": "InvoiceInIntrastat",
|
||||||
|
"foreignKey": "invoiceInFk"
|
||||||
|
},
|
||||||
"invoiceInTax": {
|
"invoiceInTax": {
|
||||||
"type": "hasMany",
|
"type": "hasMany",
|
||||||
"model": "InvoiceInTax",
|
"model": "InvoiceInTax",
|
||||||
|
|
|
@ -1,4 +1,7 @@
|
||||||
<vn-descriptor-content module="invoiceIn" description="$ctrl.invoiceIn.supplierRef">
|
<vn-descriptor-content
|
||||||
|
module="invoiceIn"
|
||||||
|
description="$ctrl.invoiceIn.supplierRef"
|
||||||
|
summary="$ctrl.$.summary">
|
||||||
<slot-menu>
|
<slot-menu>
|
||||||
<vn-item
|
<vn-item
|
||||||
ng-click="$ctrl.checkToBook()"
|
ng-click="$ctrl.checkToBook()"
|
||||||
|
@ -76,4 +79,7 @@
|
||||||
vn-id="confirm-toBookAnyway"
|
vn-id="confirm-toBookAnyway"
|
||||||
message="Are you sure you want to book this invoice?"
|
message="Are you sure you want to book this invoice?"
|
||||||
on-accept="$ctrl.onAcceptToBook()">
|
on-accept="$ctrl.onAcceptToBook()">
|
||||||
</vn-confirm>
|
</vn-confirm>
|
||||||
|
<vn-popup vn-id="summary">
|
||||||
|
<vn-invoice-in-summary invoice-in="$ctrl.invoiceIn"></vn-invoice-in-summary>
|
||||||
|
</vn-popup>
|
|
@ -10,5 +10,6 @@ import './summary';
|
||||||
import './basic-data';
|
import './basic-data';
|
||||||
import './tax';
|
import './tax';
|
||||||
import './dueDay';
|
import './dueDay';
|
||||||
|
import './intrastat';
|
||||||
import './create';
|
import './create';
|
||||||
import './log';
|
import './log';
|
||||||
|
|
|
@ -0,0 +1,100 @@
|
||||||
|
<vn-crud-model
|
||||||
|
vn-id="model"
|
||||||
|
url="InvoiceInIntrastats"
|
||||||
|
data="$ctrl.invoceInIntrastat"
|
||||||
|
link="{invoiceInFk: $ctrl.$params.id}"
|
||||||
|
auto-load="true"
|
||||||
|
on-data-change="$ctrl.calculateTotals()">
|
||||||
|
</vn-crud-model>
|
||||||
|
<vn-crud-model
|
||||||
|
auto-load="true"
|
||||||
|
url="Countries"
|
||||||
|
data="countries"
|
||||||
|
order="country">
|
||||||
|
</vn-crud-model>
|
||||||
|
<vn-crud-model
|
||||||
|
auto-load="true"
|
||||||
|
url="Intrastats"
|
||||||
|
data="intrastats"
|
||||||
|
order="id">
|
||||||
|
</vn-crud-model>
|
||||||
|
<vn-watcher
|
||||||
|
vn-id="watcher"
|
||||||
|
data="$ctrl.invoceInIntrastat"
|
||||||
|
form="form">
|
||||||
|
</vn-watcher>
|
||||||
|
<vn-card
|
||||||
|
class="vn-mb-md vn-pa-lg vn-w-lg"
|
||||||
|
style="text-align: right"
|
||||||
|
ng-if="$ctrl.invoceInIntrastat.length > 0">
|
||||||
|
<vn-label-value label="Total amount"
|
||||||
|
value="{{$ctrl.amountTotal | currency: 'EUR':2}}">
|
||||||
|
</vn-label-value>
|
||||||
|
<vn-label-value label="Total net"
|
||||||
|
value="{{$ctrl.netTotal}}">
|
||||||
|
</vn-label-value>
|
||||||
|
<vn-label-value label="Total stems"
|
||||||
|
value="{{$ctrl.stemsTotal}}">
|
||||||
|
</vn-label-value>
|
||||||
|
</vn-card>
|
||||||
|
<form name="form" ng-submit="$ctrl.onSubmit()">
|
||||||
|
<vn-card class="vn-pa-lg">
|
||||||
|
<vn-horizontal ng-repeat="intrastat in $ctrl.invoceInIntrastat">
|
||||||
|
<vn-autocomplete vn-three
|
||||||
|
label="Code"
|
||||||
|
data="intrastats"
|
||||||
|
ng-model="intrastat.intrastatFk"
|
||||||
|
show-field="description"
|
||||||
|
rule
|
||||||
|
vn-focus>
|
||||||
|
<tpl-item>{{id | zeroFill:8}}: {{description}}</tpl-item>
|
||||||
|
</vn-autocomplete>
|
||||||
|
<vn-input-number
|
||||||
|
label="Amount"
|
||||||
|
ng-model="intrastat.amount"
|
||||||
|
step="0.01"
|
||||||
|
rule>
|
||||||
|
</vn-input-number>
|
||||||
|
<vn-input-number
|
||||||
|
label="Net"
|
||||||
|
ng-model="intrastat.net"
|
||||||
|
step="0.01"
|
||||||
|
rule>
|
||||||
|
</vn-input-number>
|
||||||
|
<vn-input-number
|
||||||
|
label="Stems"
|
||||||
|
ng-model="intrastat.stems"
|
||||||
|
rule>
|
||||||
|
</vn-input-number>
|
||||||
|
<vn-autocomplete
|
||||||
|
label="Country"
|
||||||
|
data="countries"
|
||||||
|
ng-model="intrastat.countryFk"
|
||||||
|
show-field="code"
|
||||||
|
rule>
|
||||||
|
</vn-autocomplete>
|
||||||
|
<vn-none>
|
||||||
|
<vn-icon-button
|
||||||
|
vn-tooltip="Remove due day"
|
||||||
|
icon="delete"
|
||||||
|
ng-click="$ctrl.deleteIntrastat($index)"
|
||||||
|
tabindex="-1">
|
||||||
|
</vn-icon-button>
|
||||||
|
</vn-none>
|
||||||
|
</vn-horizontal>
|
||||||
|
<vn-one>
|
||||||
|
<vn-icon-button
|
||||||
|
vn-bind="+"
|
||||||
|
vn-tooltip="Add due day"
|
||||||
|
icon="add_circle"
|
||||||
|
ng-click="$ctrl.add()">
|
||||||
|
</vn-icon-button>
|
||||||
|
</vn-one>
|
||||||
|
</vn-card>
|
||||||
|
<vn-button-bar>
|
||||||
|
<vn-submit
|
||||||
|
disabled="!watcher.dataChanged()"
|
||||||
|
label="Save">
|
||||||
|
</vn-submit>
|
||||||
|
</vn-button-bar>
|
||||||
|
</form>
|
|
@ -0,0 +1,60 @@
|
||||||
|
import ngModule from '../module';
|
||||||
|
import Section from 'salix/components/section';
|
||||||
|
|
||||||
|
class Controller extends Section {
|
||||||
|
set invoceInIntrastat(value) {
|
||||||
|
this._invoceInIntrastat = value;
|
||||||
|
|
||||||
|
if (value) this.calculateTotals();
|
||||||
|
}
|
||||||
|
|
||||||
|
get invoceInIntrastat() {
|
||||||
|
return this._invoceInIntrastat;
|
||||||
|
}
|
||||||
|
|
||||||
|
calculateTotals() {
|
||||||
|
this.amountTotal = 0.0;
|
||||||
|
this.netTotal = 0.0;
|
||||||
|
this.stemsTotal = 0.0;
|
||||||
|
if (!this.invoceInIntrastat) return;
|
||||||
|
|
||||||
|
this.invoceInIntrastat.forEach(intrastat => {
|
||||||
|
this.amountTotal += intrastat.amount;
|
||||||
|
this.netTotal += intrastat.net;
|
||||||
|
this.stemsTotal += intrastat.stems;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
add() {
|
||||||
|
this.$.model.insert({});
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteIntrastat($index) {
|
||||||
|
this.$.model.remove($index);
|
||||||
|
this.$.model.save().then(() => {
|
||||||
|
this.vnApp.showSuccess(this.$t('Data saved!'));
|
||||||
|
this.calculateTotals();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onSubmit() {
|
||||||
|
this.$.watcher.check();
|
||||||
|
this.$.model.save().then(() => {
|
||||||
|
this.$.watcher.notifySaved();
|
||||||
|
this.$.watcher.updateOriginalData();
|
||||||
|
this.calculateTotals();
|
||||||
|
this.card.reload();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ngModule.vnComponent('vnInvoiceInIntrastat', {
|
||||||
|
template: require('./index.html'),
|
||||||
|
controller: Controller,
|
||||||
|
require: {
|
||||||
|
card: '^vnInvoiceInCard'
|
||||||
|
},
|
||||||
|
bindings: {
|
||||||
|
invoiceIn: '<'
|
||||||
|
}
|
||||||
|
});
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue