Merge branch '1897travel.index' of verdnatura/salix into dev
gitea/salix/dev This commit looks good
Details
gitea/salix/dev This commit looks good
Details
This commit is contained in:
commit
2330e9fb39
|
@ -254,7 +254,6 @@ class AutoSearch {
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
this.model.applyFilter(
|
||||
where ? {where} : null,
|
||||
hasParams ? userParams : null
|
||||
|
|
|
@ -0,0 +1,141 @@
|
|||
|
||||
const ParameterizedSQL = require('loopback-connector').ParameterizedSQL;
|
||||
const buildFilter = require('vn-loopback/util/filter').buildFilter;
|
||||
const mergeFilters = require('vn-loopback/util/filter').mergeFilters;
|
||||
|
||||
module.exports = Self => {
|
||||
Self.remoteMethodCtx('filter', {
|
||||
description: 'Find all instances of the model matched by filter from the data source.',
|
||||
accessType: 'READ',
|
||||
accepts: [
|
||||
{
|
||||
arg: 'filter',
|
||||
type: 'Object',
|
||||
description: 'Filter defining where, order, offset, and limit - must be a JSON-encoded string',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'search',
|
||||
type: 'String',
|
||||
description: 'Searchs the travel by id',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'id',
|
||||
type: 'Integer',
|
||||
description: 'The travel id',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'shippedFrom',
|
||||
type: 'Date',
|
||||
description: 'The shipped from date filter',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'shippedTo',
|
||||
type: 'Date',
|
||||
description: 'The shipped to date filter',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'landedFrom',
|
||||
type: 'Date',
|
||||
description: 'The landed from date filter',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'landedTo',
|
||||
type: 'Date',
|
||||
description: 'The landed to date filter',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'agencyFk',
|
||||
type: 'Number',
|
||||
description: 'The agencyModeFk id',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'warehouseOutFk',
|
||||
type: 'Number',
|
||||
description: 'The warehouseOutFk filter',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'warehouseInFk',
|
||||
type: 'Number',
|
||||
description: 'The warehouseInFk filter',
|
||||
http: {source: 'query'}
|
||||
}, {
|
||||
arg: 'totalEntries',
|
||||
type: 'Number',
|
||||
description: 'The totalEntries filter',
|
||||
http: {source: 'query'}
|
||||
}
|
||||
],
|
||||
returns: {
|
||||
type: ['Object'],
|
||||
root: true
|
||||
},
|
||||
http: {
|
||||
path: `/filter`,
|
||||
verb: 'GET'
|
||||
}
|
||||
});
|
||||
|
||||
Self.filter = async(ctx, filter) => {
|
||||
let conn = Self.dataSource.connector;
|
||||
let where = buildFilter(ctx.args, (param, value) => {
|
||||
switch (param) {
|
||||
case 'search':
|
||||
return {'t.id': value};
|
||||
case 'ref':
|
||||
return {[param]: {regexp: value}};
|
||||
case 'shippedFrom':
|
||||
return {'t.shipped': {gte: value}};
|
||||
case 'shippedTo':
|
||||
return {'t.shipped': {lte: value}};
|
||||
case 'landedFrom':
|
||||
return {'t.landed': {gte: value}};
|
||||
case 'landedTo':
|
||||
return {'t.landed': {lte: value}};
|
||||
case 'id':
|
||||
case 'agencyFk':
|
||||
case 'warehouseOutFk':
|
||||
case 'warehouseInFk':
|
||||
case 'totalEntries':
|
||||
param = `t.${param}`;
|
||||
return {[param]: value};
|
||||
}
|
||||
});
|
||||
|
||||
filter = mergeFilters(ctx.args.filter, {where});
|
||||
|
||||
let stmts = [];
|
||||
let stmt;
|
||||
stmt = new ParameterizedSQL(
|
||||
`SELECT
|
||||
t.id,
|
||||
t.shipped,
|
||||
t.landed,
|
||||
t.warehouseInFk,
|
||||
t.warehouseOutFk,
|
||||
t.agencyFk,
|
||||
t.ref,
|
||||
t.isDelivered,
|
||||
t.isReceived,
|
||||
t.m3,
|
||||
t.kg,
|
||||
t.cargoSupplierFk,
|
||||
t.totalEntries,
|
||||
am.name agencyModeName,
|
||||
win.name warehouseInName,
|
||||
wout.name warehouseOutName
|
||||
FROM vn.travel t
|
||||
JOIN vn.agencyMode am ON am.id = t.agencyFk
|
||||
JOIN vn.warehouse win ON win.id = t.warehouseInFk
|
||||
JOIN vn.warehouse wout ON wout.id = t.warehouseOutFk`
|
||||
);
|
||||
|
||||
|
||||
stmt.merge(conn.makeSuffix(filter));
|
||||
let itemsIndex = stmts.push(stmt) - 1;
|
||||
|
||||
let sql = ParameterizedSQL.join(stmts, ';');
|
||||
let result = await conn.executeStmt(sql);
|
||||
return itemsIndex === 0 ? result : result[itemsIndex];
|
||||
};
|
||||
};
|
||||
|
|
@ -0,0 +1,58 @@
|
|||
const app = require('vn-loopback/server/server');
|
||||
|
||||
describe('Travel filter()', () => {
|
||||
it('should return the travel matching "search"', async() => {
|
||||
let ctx = {
|
||||
args: {
|
||||
search: 1
|
||||
}
|
||||
};
|
||||
|
||||
let result = await app.models.Travel.filter(ctx);
|
||||
|
||||
expect(result.length).toEqual(1);
|
||||
expect(result[0].id).toEqual(1);
|
||||
});
|
||||
|
||||
it('should return the travel matching "warehouse out"', async() => {
|
||||
let ctx = {
|
||||
args: {
|
||||
warehouseOutFk: 2
|
||||
}
|
||||
};
|
||||
|
||||
let result = await app.models.Travel.filter(ctx);
|
||||
|
||||
expect(result.length).toEqual(7);
|
||||
});
|
||||
|
||||
it('should return the travel matching "total entries"', async() => {
|
||||
let ctx = {
|
||||
args: {
|
||||
totalEntries: 1,
|
||||
}
|
||||
};
|
||||
|
||||
let result = await app.models.Travel.filter(ctx);
|
||||
|
||||
expect(result.length).toEqual(5);
|
||||
});
|
||||
|
||||
it('should return the routes matching "shipped from" and "shipped to"', async() => {
|
||||
const from = new Date();
|
||||
const to = new Date();
|
||||
from.setHours(0, 0, 0, 0);
|
||||
to.setHours(23, 59, 59, 999);
|
||||
to.setDate(to.getDate() + 1);
|
||||
let ctx = {
|
||||
args: {
|
||||
shippedFrom: from,
|
||||
shippedTo: to
|
||||
}
|
||||
};
|
||||
|
||||
let result = await app.models.Travel.filter(ctx);
|
||||
|
||||
expect(result.length).toEqual(1);
|
||||
});
|
||||
});
|
|
@ -1,4 +1,5 @@
|
|||
module.exports = Self => {
|
||||
require('../methods/travel/getTravel')(Self);
|
||||
require('../methods/travel/getEntries')(Self);
|
||||
require('../methods/travel/filter')(Self);
|
||||
};
|
||||
|
|
|
@ -1,17 +1,17 @@
|
|||
<vn-crud-model
|
||||
vn-id="model"
|
||||
url="Travels"
|
||||
filter="::$ctrl.filter"
|
||||
url="Travels/filter"
|
||||
limit="20"
|
||||
data="travels">
|
||||
params="::$ctrl.params"
|
||||
data="travels"
|
||||
order="shipped DESC, landed DESC">
|
||||
</vn-crud-model>
|
||||
<vn-auto-search
|
||||
model="model"
|
||||
expr-builder="$ctrl.exprBuilder(param, value)">
|
||||
on-search="$ctrl.onSearch($params)">
|
||||
</vn-auto-search>
|
||||
<vn-data-viewer
|
||||
model="model"
|
||||
class="vn-w-lg">
|
||||
class="vn-mb-xl vn-w-xl">
|
||||
<vn-card>
|
||||
<vn-table model="model">
|
||||
<vn-thead>
|
||||
|
@ -34,11 +34,11 @@
|
|||
ui-sref="travel.card.summary({id: {{::travel.id}}})">
|
||||
<vn-td number>{{::travel.id}}</vn-td>
|
||||
<vn-td expand>{{::travel.ref}}</vn-td>
|
||||
<vn-td expand>{{::travel.agency.name}}</vn-td>
|
||||
<vn-td expand>{{::travel.warehouseOut.name}}</vn-td>
|
||||
<vn-td expand>{{::travel.agencyModeName}}</vn-td>
|
||||
<vn-td expand>{{::travel.warehouseOutName}}</vn-td>
|
||||
<vn-td center>{{::travel.shipped | date:'dd/MM/yyyy'}}</vn-td>
|
||||
<vn-td><vn-check ng-model="travel.isDelivered" disabled="true"></vn-check></vn-td>
|
||||
<vn-td expand>{{::travel.warehouseIn.name}}</vn-td>
|
||||
<vn-td expand>{{::travel.warehouseInName}}</vn-td>
|
||||
<vn-td center>{{::travel.landed | date:'dd/MM/yyyy'}}</vn-td>
|
||||
<vn-td center><vn-check ng-model="travel.isReceived" disabled="true"></vn-check></vn-td>
|
||||
<vn-td>
|
||||
|
@ -51,8 +51,8 @@
|
|||
</a>
|
||||
</vn-tbody>
|
||||
</vn-table>
|
||||
</vn-card>
|
||||
</vn-data-viewer>
|
||||
</vn-card>
|
||||
<vn-popup vn-id="summary">
|
||||
<vn-travel-summary
|
||||
travel="$ctrl.travelSelected">
|
||||
|
|
|
@ -4,44 +4,6 @@ export default class Controller {
|
|||
constructor($scope) {
|
||||
this.$ = $scope;
|
||||
this.ticketSelected = null;
|
||||
|
||||
this.filter = {
|
||||
include: [
|
||||
{
|
||||
relation: 'agency',
|
||||
scope: {fields: ['name']}
|
||||
}, {
|
||||
relation: 'warehouseIn',
|
||||
scope: {fields: ['name']}
|
||||
}, {
|
||||
relation: 'warehouseOut',
|
||||
scope: {fields: ['name']}
|
||||
}
|
||||
]
|
||||
};
|
||||
}
|
||||
|
||||
exprBuilder(param, value) {
|
||||
switch (param) {
|
||||
case 'search':
|
||||
return {id: value};
|
||||
case 'ref':
|
||||
return {[param]: {regexp: value}};
|
||||
case 'shippedFrom':
|
||||
return {shipped: {gte: value}};
|
||||
case 'shippedTo':
|
||||
return {shipped: {lte: value}};
|
||||
case 'landedFrom':
|
||||
return {landed: {gte: value}};
|
||||
case 'landedTo':
|
||||
return {landed: {lte: value}};
|
||||
case 'id':
|
||||
case 'agencyModeFk':
|
||||
case 'warehouseOutFk':
|
||||
case 'warehouseInFk':
|
||||
case 'totalEntries':
|
||||
return {[param]: value};
|
||||
}
|
||||
}
|
||||
|
||||
preview(event, travel) {
|
||||
|
@ -50,6 +12,29 @@ export default class Controller {
|
|||
event.preventDefault();
|
||||
event.stopImmediatePropagation();
|
||||
}
|
||||
|
||||
getScopeDates(days) {
|
||||
const today = new Date();
|
||||
today.setHours(0, 0, 0, 0);
|
||||
const daysOnward = new Date(today);
|
||||
daysOnward.setDate(today.getDate() + days);
|
||||
daysOnward.setHours(23, 59, 59, 999);
|
||||
return {shippedFrom: today, shippedTo: daysOnward};
|
||||
}
|
||||
|
||||
onSearch(params) {
|
||||
if (params) {
|
||||
let newParams = params;
|
||||
if (params.scopeDays) {
|
||||
const scopeDates = this.getScopeDates(params.scopeDays);
|
||||
Object.assign(newParams, scopeDates);
|
||||
} else if (Object.entries(params).length == 0)
|
||||
newParams = this.getScopeDates(1);
|
||||
|
||||
this.$.model.applyFilter(null, newParams);
|
||||
} else
|
||||
this.$.model.clear();
|
||||
}
|
||||
}
|
||||
|
||||
Controller.$inject = ['$scope'];
|
||||
|
|
|
@ -3,31 +3,62 @@ import './index.js';
|
|||
describe('Travel Component vnTravelIndex', () => {
|
||||
let $componentController;
|
||||
let controller;
|
||||
let $window;
|
||||
let travels = [{
|
||||
id: 1,
|
||||
warehouseInFk: 1,
|
||||
totalEntries: 3,
|
||||
isDelivered: false
|
||||
}, {
|
||||
id: 2,
|
||||
warehouseInFk: 1,
|
||||
total: 4,
|
||||
isDelivered: true
|
||||
}, {
|
||||
id: 3,
|
||||
warehouseInFk: 1,
|
||||
total: 2,
|
||||
isDelivered: true
|
||||
}];
|
||||
|
||||
beforeEach(angular.mock.module('travel', $translateProvider => {
|
||||
$translateProvider.translations('en', {});
|
||||
}));
|
||||
|
||||
beforeEach(angular.mock.inject(_$componentController_ => {
|
||||
beforeEach(angular.mock.inject((_$componentController_, $rootScope) => {
|
||||
$componentController = _$componentController_;
|
||||
controller = $componentController('vnTravelIndex');
|
||||
let $scope = $rootScope.$new();
|
||||
controller = $componentController('vnTravelIndex', $scope);
|
||||
controller.$.summary = {show: jasmine.createSpy('show')};
|
||||
}));
|
||||
|
||||
describe('exprBuilder()', () => {
|
||||
it('should return a formated object with the travel id in case of search', () => {
|
||||
let param = 'search';
|
||||
let value = 2;
|
||||
let result = controller.exprBuilder(param, value);
|
||||
describe('preview()', () => {
|
||||
it('should show the dialog summary', () => {
|
||||
let event = new MouseEvent('click', {
|
||||
view: $window,
|
||||
bubbles: true,
|
||||
cancelable: true
|
||||
});
|
||||
controller.preview(event, travels[0]);
|
||||
|
||||
expect(result).toEqual({id: 2});
|
||||
expect(controller.$.summary.show).toHaveBeenCalledWith();
|
||||
});
|
||||
});
|
||||
|
||||
it('should return a formated object with the warehouseInFk in case of warehouseInFk', () => {
|
||||
let param = 'warehouseInFk';
|
||||
let value = 3;
|
||||
let result = controller.exprBuilder(param, value);
|
||||
describe('getScopeDates()', () => {
|
||||
it('should return a range of dates', () => {
|
||||
let days = 2; // never put 1 or anything higher than 2
|
||||
let result = controller.getScopeDates(days);
|
||||
|
||||
expect(result).toEqual({warehouseInFk: 3});
|
||||
let from = new Date(result.shippedFrom).getTime();
|
||||
let to = new Date(result.shippedTo).getTime();
|
||||
let range = to - from;
|
||||
|
||||
const dayInMilliseconds = 24 * 60 * 60 * 1000;
|
||||
|
||||
let millisecondsPerAddedDay = dayInMilliseconds - 1;
|
||||
|
||||
expect(range - dayInMilliseconds).toEqual(dayInMilliseconds + millisecondsPerAddedDay);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
|
|
@ -30,7 +30,7 @@
|
|||
<vn-autocomplete vn-one
|
||||
label="Agency"
|
||||
ng-model="filter.agencyFk"
|
||||
url="Agencies"
|
||||
url="AgencyModes"
|
||||
show-field="name"
|
||||
value-field="id">
|
||||
</vn-autocomplete>
|
||||
|
|
Loading…
Reference in New Issue