Connecting GraphQL Service Data with React Grid Component
2 Apr 202424 minutes to read
GraphQL is a powerful query language for APIs, designed to provide a more efficient alternative to traditional REST APIs. It allows you to precisely fetch the data you need, reducing over-fetching and under-fetching of data. GraphQL provides a flexible and expressive syntax for querying, enabling clients to request only the specific data they require.
Syncfusion’s Grid component seamlessly integrates with GraphQL servers using the GraphQLAdaptor in the DataManager. This specialized adaptor simplifies the interaction between the Syncfusion Grid and GraphQL servers, allowing efficient data retrieval with support for various operations like CRUD (Create, Read, Update, Delete), paging, sorting, and filtering.
This section describes a step-by-step process for retrieving data from GraphQL service using GraphQLAdaptor
, then binding it to the React Grid component to facilitate data and CRUD operations.
Configure GraphQL Server
To configure a GraphQL server for use with Syncfusion React Grid, you need to follow several steps:
Step 1: Create service for GraphQL
-
Create a new folder named GraphQLServer specifically for your GraphQL server.
-
Install the GraphQL server using the graph pack npm package. Open your terminal and navigate to the server folder, then run:
npm i graphpack
-
To utilize Syncfusion’s
ej2-data
package, you need to include it as a dependency in your project’spackage.json
file. Here’s how you can mention it in the configuration:{ "name": "graphql-server", "version": "1.0.0", "description": "", "scripts": { "dev": "graphpack --port 4205", "build": "graphpack build" }, "devDependencies": { "graphpack": "^1.0.9" }, "dependencies": { "@syncfusion/ej2-data": "24.1.41" } }
-
Create a database file (src/db.js) to store your data.
export let OrderData = [
{
OrderID: 10248, CustomerID: 'VINET', EmployeeID: 5, OrderDate: new Date("07 12 1996 02:00:23"),
ShipName: 'Vins et alcools Chevalier', ShipCity: 'Reims', ShipAddress: '59 rue de l Abbaye',
ShipRegion: 'CJ', ShipPostalCode: '51100', ShipCountry: 'France', Freight: 32.38, Verified: !0
},
{
OrderID: 10249, CustomerID: 'TOMSP', EmployeeID: 6, OrderDate: new Date("07 12 1996 00:03:23"),
ShipName: 'Toms Spezialitäten', ShipCity: 'Münster', ShipAddress: 'Luisenstr. 48',
ShipRegion: 'CJ', ShipPostalCode: '44087', ShipCountry: 'Germany', Freight: 11.61, Verified: !1
},
{
OrderID: 10250, CustomerID: 'HANAR', EmployeeID: 4, OrderDate: new Date("07 12 1996 00:00:23"),
ShipName: 'Hanari Carnes', ShipCity: 'Rio de Janeiro', ShipAddress: 'Rua do Paço, 67',
ShipRegion: 'RJ', ShipPostalCode: '05454-876', ShipCountry: 'Brazil', Freight: 65.83,Verified: !0
},
....
];
Ensure that the GraphQL server is properly configured and dependencies are installed to proceed with the next steps.
Step 2: Schema definition for GraphQL server
In the context of GraphQL, a schema defines the structure of the data that clients can query from the server. It serves as a contract between the client and the server, outlining the types of data available, the operations that can be performed, and how the data is related.
When integrating GraphQL with the Syncfusion Grid, defining a schema involves specifying the types of data the Grid expects to receive from the GraphQL server, along with any additional parameters for operations like sorting, filtering, and paging.
Here’s how you can define a schema for the Syncfusion Grid:
-
Define Types: Create types representing the structure of data retrieved from GraphQL queries. Since the
GraphQLAdaptor
in Syncfusion extends fromUrlAdaptor
, it expects a JSON response with specific properties:- result: An array containing the data entities.
- count: The total number of records.
- aggregates: Contains total aggregate data(optional).
For example, if your Grid displays orders, you might define types for ReturnType and Order:
type ReturnType { result: [Order] count: Int aggregates: String # Total records aggregates } type Order { OrderID: Int! CustomerID: String EmployeeID: Int Freight: Int ShipCity: String ShipCountry: String }
-
Define Queries: Define queries that can be made to retrieve data from the server. In the case of a Grid, you may define a query to fetch orders, accepting parameters such as
DataManager
for advanced data operations. To utilizeDatamanager
, you need to install packages from@syncfusion/ej2-data
type Query { getOrders(datamanager: DataManager): ReturnType }
-
Define DataManager Input: Define input types for
DataManager
, specifying parameters for sorting, filtering, paging, aggregates, etc., to be used in queries. The query parameters will be send in a string format which contains the below details.Parameters Description requiresCounts
If it is true then the total count of records will be included in response. skip
Holds the number of records to skip. take
Holds the number of records to take. sorted
Contains details about current sorted column and its direction. where
Contains details about current filter column name and its constraints. group
Contains details about current Grouped column names. search
Contains details about current search data. aggregates
Contains details about aggregate data. input DataManager { skip: Int take: Int sorted: [Sort] group: [String] table: String select: [String] where: String search: String requiresCounts: Boolean aggregates: [Aggregate] params: String }
Create a schema file (e.g., src/schema.graphql) in your GraphQL server project and write the schema definition there.
#Grid Sort direction
input Sort {
name: String
direction: String
}
#Grid aggregates type
input Aggregate {
field: String!
type: String!
}
# Represents parameters for querying data, including sorting, filtering, etc.
input DataManager {
skip: Int
take: Int
sorted: [Sort]
group: [String]
table: String
select: [String]
where: String
search: String
requiresCounts: Boolean
aggregates: [Aggregate]
params: String
}
# Represents an order type
type Order {
OrderID: Int!
CustomerID: String
EmployeeID: Int
Freight: Int
ShipCity: String
ShipCountry: String
}
# Represents the result of a query, including the data and count
type ReturnType {
result: [Order]
count: Int
aggregates: String
}
# Represents a query to fetch orders with specified data manager parameters
type Query {
getOrders(datamanager: DataManager): ReturnType
}
Step 3: Implement Resolvers
To handle GraphQL queries and fetch data from your database, you need to create resolver functions. These resolver functions will be responsible for processing GraphQL queries and returning the appropriate data. In order to return data based on the grid expected result and count, utilize DataUtil
from @syncfusion/ej2-data
package.
Create a resolver file(src/resolvers.js) and implement the following code.
import { OrderData } from "./db";
import { DataUtil } from "@syncfusion/ej2-data";
const resolvers = {
Query: {
getOrders: (parent, { datamanager }, context, info) => {
var ret = DataUtil.processData(datamanager, OrderData);
return ret;
}
}
};
export default resolvers;
Step 4: Run GraphQL Server
Install required packages and start the GraphQL server by running the following commands in your terminal:
npm install
npm run dev
The server will be hosted at http://localhost:xxxx/. (where xxxx represents the port number).
Connecting grid to an GraphQL service
To integrate GraphQL with the Syncfusion Grid in your React application, follow these steps:
Step 1: Create an Syncfusion React Grid:
Start a new React application using below command.
ng new GridClient
This command will prompt you for a few settings for the new project, such as whether to add React routing and which stylesheet format to use.
cd GridClient
Step 2: Adding Syncfusion Grid package
To use Syncfusion Grid component and Syncfusion Datamanager, install the packages using the below command.
npm i @syncfusion/ej2-data
npm install @syncfusion/ej2-react-grids --save
Step 3: Adding CSS reference
The following CSS files are available in ../node_modules/@syncfusion
package folder. This can be referenced in [src/styles.css] using following code.
[styles.css]
@import '../node_modules/@syncfusion/ej2-base/styles/material.css';
@import '../node_modules/@syncfusion/ej2-buttons/styles/material.css';
@import '../node_modules/@syncfusion/ej2-calendars/styles/material.css';
@import '../node_modules/@syncfusion/ej2-dropdowns/styles/material.css';
@import '../node_modules/@syncfusion/ej2-inputs/styles/material.css';
@import '../node_modules/@syncfusion/ej2-navigations/styles/material.css';
@import '../node_modules/@syncfusion/ej2-popups/styles/material.css';
@import '../node_modules/@syncfusion/ej2-splitbuttons/styles/material.css';
@import '../node_modules/@syncfusion/ej2-notifications/styles/material.css';
@import '../node_modules/@syncfusion/ej2-react-grids/styles/material.css';
Step 4: Configure DataManager with GraphQLAdaptor:
Set up the DataManager
with the GraphQLAdaptor
to communicate with the GraphQL server. Then define the GraphQL query to specify the expected data structure and retrieval parameters:
[App.tsx]
import { DataManager, GraphQLAdaptor } from '@syncfusion/ej2-data';
import { GridComponent, ColumnsDirective, ColumnDirective } from '@syncfusion/ej2-react-grids';
function App() {
// Set the adaptor as UrlAdaptor
const data = new DataManager({
url: "http://localhost:****/", // **** represents the port number
adaptor: new GraphQLAdaptor({
response: {
result: 'getOrders.result',// Retrieve the actual order data
count: 'getOrders.count' // Retrieve the total count of orders
},
// GraphQL query to fetch orders
query: `query getOrders($datamanager: DataManager) {
getOrders(datamanager: $datamanager) {
count,
result{
OrderID, CustomerID, EmployeeID, ShipCountry}
}
}`,
})
});
return <GridComponent dataSource={data} height={320}>
<ColumnsDirective>
<ColumnDirective field='OrderID' headerText='Order ID' isPrimaryKey={true} width='150'textAlign='Right'></ColumnDirective>
<ColumnDirective field='CustomerID' headerText='Customer ID' width='150'></ColumnDirective>
<ColumnDirective field='EmployeeID' headerText='Employee ID' width='150'></ColumnDirective>
<ColumnDirective field='ShipCity' headerText='Ship City' width='150'/>
</ColumnsDirective>
</GridComponent>
};
export default App;
Step 5: Run the Application:
Once the GraphQL server is running, assign its URL (e.g., http://localhost:xxxx/) to the dataManager.url
property of the DataManager
in your React application.
npm i
npm start
By following these steps, you will successfully integrate GraphQL with the Syncfusion Grid in your React application. Ensure that the GraphQL server is running smoothly and is accessible at the specified URL.
Handling filtering operation
To handle filter operation in the Syncfusion Grid using the GraphQLAdaptor, by utilizing the datamanager.where
parameters and executing the filter operation with the where method. This feature allows you to efficiently filter through the grid’s data and retrieve relevant information based on specified criteria.
In the image below, you can see the values of datamanager.where
parameters:
import { OrderData } from "./db";
import { DataUtil, Query, DataManager } from "@syncfusion/ej2-data";
const resolvers = {
Query: {
getOrders: (parent, { datamanager }, context, info) => {
let orders = [...OrderData];
const query = new Query();
const performFiltering = (filterString) => {
const filter = JSON.parse(filterString);
// Iterating over each predicate
filter[0].predicates.forEach(predicate => {
const field = predicate.field;
const operator = predicate.operator;
const value = predicate.value;
query.where(field, operator, value);
});
}
// Perform filtering
if (datamanager.where) {
performFiltering(datamanager.where);
}
orders = new DataManager(orders).executeLocal(query);
var count = orders.length;
return { result: orders, count: count }; // Return result and count
},
},
};
export default resolvers;
import { DataManager, GraphQLAdaptor } from '@syncfusion/ej2-data';
import { ColumnDirective, ColumnsDirective, GridComponent, Filter, Inject } from '@syncfusion/ej2-react-grids';
function App() {
const data = new DataManager({
url: "http://localhost:xxxx/", // Here xxxx denotes port number
adaptor: new GraphQLAdaptor({
response: {
result: 'getOrders.result',
count: 'getOrders.count'
},
query: `query getOrders($datamanager: DataManager) {
getOrders(datamanager: $datamanager) {
count,
result{
OrderID, CustomerID, ShipCity, ShipCountry}
}
}`,
})
});
return <GridComponent dataSource={data} allowFiltering={true} height={320}>
<ColumnsDirective>
<ColumnDirective field='OrderID' headerText='Order ID' isPrimaryKey={true} width='150'textAlign='Right'></ColumnDirective>
<ColumnDirective field='CustomerID' headerText='Customer ID' width='150'></ColumnDirective>
<ColumnDirective field='ShipCity' headerText='Ship City' width='150'/>
<ColumnDirective field='ShipCountry' headerText='Ship Country' width='150'></ColumnDirective>
</ColumnsDirective>
<Inject services={[Filter]} />
</GridComponent>
};
export default App;
Handling search operation
To handle search operation in the Syncfusion Grid using the GraphQLAdaptor, by utilizing the datamanager.search
parameters and executing the search operation with the search method. This feature allows users to efficiently search through the grid’s data and retrieve relevant information based on specified criteria.
In the image below, you can see the values of datamanager.search
parameters:
import { OrderData } from "./db";
import { DataUtil, Query, DataManager } from "@syncfusion/ej2-data";
const resolvers = {
Query: {
getOrders: (parent, { datamanager }, context, info) => {
let orders = [...OrderData];
const query = new Query();
const performSearching = (searchParam) => {
const { fields, key } = JSON.parse(searchParam)[0];
query.search(key, fields);
}
// Perform Searching
if (datamanager.search) {
performSearching(datamanager.search);
}
orders = new DataManager(orders).executeLocal(query);
var count = orders.length;
return { result: orders, count: count }; // Return result and count
},
},
};
export default resolvers;
import { DataManager, GraphQLAdaptor } from '@syncfusion/ej2-data';
import { ColumnDirective, ColumnsDirective, GridComponent, ToolbarItems, Toolbar, Inject } from '@syncfusion/ej2-react-grids';
function App() {
const data = new DataManager({
url: "http://localhost:xxxx/", // Here xxxx denotes port number
adaptor: new GraphQLAdaptor({
response: {
result: 'getOrders.result',
count: 'getOrders.count'
},
query: `query getOrders($datamanager: DataManager) {
getOrders(datamanager: $datamanager) {
count,
result{
OrderID, CustomerID, ShipCity, ShipCountry}
}
}`,
})
});
const toolbar: ToolbarItems[] = ['Search'];
return <GridComponent dataSource={data} toolbar={toolbar} height={320}>
<ColumnsDirective>
<ColumnDirective field='OrderID' headerText='Order ID' isPrimaryKey={true} width='150'textAlign='Right'></ColumnDirective>
<ColumnDirective field='CustomerID' headerText='Customer ID' width='150'></ColumnDirective>
<ColumnDirective field='ShipCity' headerText='Ship City' width='150'/>
<ColumnDirective field='ShipCountry' headerText='Ship Country' width='150'></ColumnDirective>
</ColumnsDirective>
<Inject services={[Toolbar]} />
</GridComponent>
};
export default App;
Handling sorting operation
To handle sort operation in the Syncfusion Grid using the GraphQLAdaptor, by utilizing the datamanager.sorted
parameters and executing the sort operation with the sortBy method. This feature allows users to efficiently sort grid data based on specified criteria.
In the image below, you can see the values of datamanager.sorted
parameters:
import { OrderData } from "./db";
import { DataUtil, Query, DataManager } from "@syncfusion/ej2-data";
const resolvers = {
Query: {
getOrders: (parent, { datamanager }, context, info) => {
let orders = [...OrderData];
const query = new Query();
const performSorting = (sorted) => {
for (let i = 0; i < sorted.length; i++) {
const { name, direction } = sorted[i];
query.sortBy(name, direction);
}
}
// Perform sorting
if (datamanager.sorted) {
performSorting(datamanager.sorted);
}
orders = new DataManager(orders).executeLocal(query);
var count = orders.length;
return { result: orders, count: count }; // Return result and count
},
},
};
export default resolvers;
import { DataManager, GraphQLAdaptor } from '@syncfusion/ej2-data';
import { ColumnDirective, ColumnsDirective, GridComponent, Sort, Inject } from '@syncfusion/ej2-react-grids';
function App() {
const data = new DataManager({
url: "http://localhost:4205/",
adaptor: new GraphQLAdaptor({
response: {
result: 'getOrders.result',
count: 'getOrders.count'
},
query: `query getOrders($datamanager: DataManager) {
getOrders(datamanager: $datamanager) {
count,
result{
OrderID, CustomerID, ShipCity, ShipCountry}
}
}`,
})
});
return <GridComponent dataSource={data} allowSorting={true} height={320}>
<ColumnsDirective>
<ColumnDirective field='OrderID' headerText='Order ID' isPrimaryKey={true} width='150'textAlign='Right'></ColumnDirective>
<ColumnDirective field='CustomerID' headerText='Customer ID' width='150'></ColumnDirective>
<ColumnDirective field='ShipCity' headerText='Ship City' width='150'/>
<ColumnDirective field='ShipCountry' headerText='Ship Country' width='150'></ColumnDirective>
</ColumnsDirective>
<Inject services={[Sort]} />
</GridComponent>
};
export default App;
Handling paging operation
To handle page operation in the Syncfusion Grid using the GraphQLAdaptor, by utilizing the datamanager.skip
and datamanager.take
parameters and executing the paging with the page method. This feature allows users to navigate through large datasets efficiently by dividing them into pages.
In the image below, you can see the value of datamanager.skip
and datamanager.take
parameters:
import { OrderData } from "./db";
import { DataUtil, Query, DataManager } from "@syncfusion/ej2-data";
const resolvers = {
Query: {
getOrders: (parent, { datamanager }, context, info) => {
let orders = [...OrderData];
const query = new Query();
// Perform paging
if (datamanager.skip && datamanager.take) {
const pageSkip = datamanager.skip / datamanager.take + 1;
const pageTake = datamanager.take;
query.page(pageSkip, pageTake);
} else if (datamanager.skip === 0 && datamanager.take) {
query.page(1, datamanager.take);
}
const currentResult = new DataManager(orders).executeLocal(query);
return { result: currentResult, count: count }; // Return result and count
},
},
};
export default resolvers;
import { DataManager, GraphQLAdaptor } from '@syncfusion/ej2-data';
import { ColumnDirective, ColumnsDirective, GridComponent, Page, PageSettingsModel, Inject } from '@syncfusion/ej2-react-grids';
function App() {
const data = new DataManager({
url: "http://localhost:xxxx/", // Here xxxx denotes port number
adaptor: new GraphQLAdaptor({
response: {
result: 'getOrders.result',
count: 'getOrders.count'
},
query: `query getOrders($datamanager: DataManager) {
getOrders(datamanager: $datamanager) {
count,
result{
OrderID, CustomerID, ShipCity, ShipCountry}
}
}`,
})
});
const pageSettings: PageSettingsModel = {pageSize: 12}
return <GridComponent dataSource={data} allowPaging={true} pageSettings={pageSettings} height={320}>
<ColumnsDirective>
<ColumnDirective field='OrderID' headerText='Order ID' isPrimaryKey={true} width='150'textAlign='Right'></ColumnDirective>
<ColumnDirective field='CustomerID' headerText='Customer ID' width='150'></ColumnDirective>
<ColumnDirective field='ShipCity' headerText='Ship City' width='150'/>
<ColumnDirective field='ShipCountry' headerText='Ship Country' width='150'></ColumnDirective>
</ColumnsDirective>
<Inject services={[Page]} />
</GridComponent>
};
export default App;
Handling CRUD operations
Syncfusion Grid seamlessly integrates with GraphQL servers using the GraphQLAdaptor
, enabling efficient CRUD (Create, Read, Update, Delete) operations on your data. The below steps explains how to perform CRUD actions using GraphQLAdaptor
in Syncfusion Grid.
Insert operation
Adding a new record to the database involves the following steps:
-
Define the createOrder mutation in your GraphQL schema to handle creating a new order. This mutation should accept an OrderInput object containing the new order details.
[schema.graphql] input OrderInput { OrderID: Int! CustomerID: String Freight: Float ShipCity: String ShipCountry: String } type Mutation { createOrder(value: OrderInput): Order! }
-
Implement the createOrder resolver function in your resolver file. This function should add the new order data to your data source and return the newly created order object.
[resolver.js] Mutation: { createOrder: (parent, { value }, context, info) => { const newOrder = value; OrderData.push(newOrder); return newOrder; }, }
-
Configure the getMutation function in your
GraphQLAdaptor
to return the appropriate GraphQL mutation query string based on the insert action. This query string should reference the createOrder mutation defined in your schema.[App.tsx] // mutation for performing CRUD getMutation: function (action: any): string { if (action === 'insert') { return `mutation CreateOrderMutation($value: OrderInput!){ createOrder(value: $value){ OrderID, CustomerID, Freight, ShipCity, ShipCountry }}`; } }
Update operation
Updating an existing record in the database involves the following steps:
-
Define the updateOrder mutation in your GraphQL schema to handle updating an order. This mutation should accept three arguments:
- key: The unique identifier of the order to be updated.
- keyColumn: The name of the column containing the unique identifier.
- value: An OrderInput object containing the updated order details.
[schema.graphql] type Order { OrderID: Int! CustomerID: String Freight: Float ShipCity: String ShipCountry: String } input OrderInput { OrderID: Int! CustomerID: String Freight: Float ShipCity: String ShipCountry: String } type Mutation { updateOrder(key: Int!, keyColumn: String, value: OrderInput): Order }
-
Implement the updateOrder resolver function in your resolver file. This function should find the order based on the provided key and keyColumn, update its properties with the values from the value argument, and return the updated order object.
[resolver.js] Mutation: { updateOrder: (parent, { key, keyColumn, value }, context, info) => { let updatedOrder = OrderData.find(order => order.OrderID === parseInt(key)); updatedOrder.CustomerID = value.CustomerID; updatedOrder.EmployeeID = value.EmployeeID; updatedOrder.Freight = value.Freight; updatedOrder.ShipCity = value.ShipCity; updatedOrder.ShipCountry = value.ShipCountry; return updatedOrder; // Make sure to return the updated order. }, }
-
Configure the getMutation function in your
GraphQLAdaptor
to return the appropriate GraphQL mutation query string based on the update action. This query string should reference the updateOrder mutation defined in your schema.[App.tsx] // mutation for performing CRUD getMutation: function (action: any): string { if (action === 'update') { return `mutation UpdateOrderMutation($key: Int!, $keyColumn: String,$value: OrderInput){ updateOrder(key: $key, keyColumn: $keyColumn, value: $value) { OrderID, CustomerID, Freight, ShipCity, ShipCountry }}`; } }
Delete Operation
Deleting a record from the database involves the following steps:
-
Define the deleteOrder mutation in your GraphQL schema to handle deleting an order. This mutation should accept three arguments similar to the updateOrder mutation.
[schema.graphql] type Order { OrderID: Int! CustomerID: String Freight: Float ShipCity: String ShipCountry: String } input OrderInput { OrderID: Int! CustomerID: String Freight: Float ShipCity: String ShipCountry: String } type Mutation { deleteOrder(key: Int!, keyColumn: String, value: OrderInput): Order! }
-
Implement the deleteOrder resolver function in your resolver file. This function should find the order based on the provided key and keyColumn, remove it from your data source, and return the deleted order object.
[resolver.js] Mutation: { deleteOrder: (parent, { key, keyColumn, value }, context, info) => { const orderIndex = OrderData.findIndex(order => order.OrderID === parseInt(key)); if (orderIndex === -1) throw new Error("Order not found." + value); const deletedOrders = OrderData.splice(orderIndex, 1); return deletedOrders[0]; } }
-
Configure the getMutation function in your
GraphQLAdaptor
to return the appropriate GraphQL mutation query string based on the delete action. This query string should reference the deleteOrder mutation defined in your schema.
.[App.tsx] // mutation for performing CRUD getMutation: function (action: any): string { if (action === 'delete') { return `mutation RemoveOrderMutation($key: Int!, $keyColumn: String, $value: OrderInput){ deleteOrder(key: $key, keyColumn: $keyColumn, value: $value) { OrderID, CustomerID, Freight, ShipCity, ShipCountry }}`; } }
Normal/Inline editing is the default edit mode for the Grid component. To enable CRUD operations, ensure that the isPrimaryKey property is set to true for a specific Grid Column, ensuring that its value is unique.
import { OrderData } from "./db";
import { DataUtil } from "@syncfusion/ej2-data";
const resolvers = {
Query: {
getOrders: (parent, { datamanager }, context, info) => {
var ret = DataUtil.processData(datamanager, OrderData);
return ret;
}
},
Mutation: {
createOrder: (parent, { value }, context, info) => {
const newOrder = value;
OrderData.push(newOrder);
return newOrder;
},
updateOrder: (parent, { key, keyColumn, value }, context, info) => {
let updatedOrder = OrderData.find(order => order.OrderID === parseInt(key));
updatedOrder.CustomerID = value.CustomerID;
updatedOrder.EmployeeID = value.EmployeeID;
updatedOrder.Freight = value.Freight;
updatedOrder.ShipCity = value.ShipCity;
updatedOrder.ShipCountry = value.ShipCountry;
return updatedOrder; // Make sure to return the updated order.
},
deleteOrder: (parent, { key, keyColumn, value }, context, info) => {
const orderIndex = OrderData.findIndex(order => order.OrderID === parseInt(key));
if (orderIndex === -1) throw new Error("Order not found." + value);
const deletedOrders = OrderData.splice(orderIndex, 1);
return deletedOrders[0];
}
}
};
export default resolvers;
#Grid Sort direction
input Sort {
name: String
direction: String
}
#Grid aggregates type
input Aggregate {
field: String!
type: String!
}
#Syncfusion DataManager query params
input DataManager {
skip: Int
take: Int
sorted: [Sort]
group: [String]
table: String
select: [String]
where: String
search: String
requiresCounts: Boolean,
aggregates: [Aggregate],
params: String
}
#Grid field names
input OrderInput {
OrderID: Int!
CustomerID: String
ShipCity: String
ShipCountry: String
}
type Order {
OrderID: Int!
CustomerID: String
ShipCity: String
ShipCountry: String
}
#Need to return type as 'result (i.e, current pager data)' and count (i.e., total number of records in your database)
type ReturnType {
result: [Order]
count: Int
aggregates: String
}
type Query {
getOrders(datamanager: DataManager): ReturnType
}
type Mutation {
createOrder(value: OrderInput): Order!
updateOrder(key: Int!, keyColumn: String, value: OrderInput): Order
deleteOrder(key: Int!, keyColumn: String, value: OrderInput): Order!
}
import { DataManager, GraphQLAdaptor } from '@syncfusion/ej2-data';
import { ColumnDirective, ColumnsDirective, GridComponent, ToolbarItems, Toolbar, Inject, EditSettingsModel, Edit } from '@syncfusion/ej2-react-grids';
function App() {
const data = new DataManager({
url: "http://localhost:xxxx/", // Here xxxx denotes port number
adaptor: new GraphQLAdaptor({
response: {
result: 'getOrders.result',
count: 'getOrders.count'
},
query: `query getOrders($datamanager: DataManager) {
getOrders(datamanager: $datamanager) {
count,
result{
OrderID, CustomerID, ShipCity, ShipCountry}
}
}`,
// mutation for performing CRUD
getMutation: function (action: any): string {
if (action === 'insert') {
return `mutation CreateOrderMutation($value: OrderInput!){
createOrder(value: $value){
OrderID, CustomerID, ShipCity, ShipCountry
}}`;
}
if (action === 'update') {
return `mutation UpdateOrderMutation($key: Int!, $keyColumn: String,$value: OrderInput){
updateOrder(key: $key, keyColumn: $keyColumn, value: $value) {
OrderID, CustomerID, ShipCity, ShipCountry
}
}`;
} else {
return `mutation RemoveOrderMutation($key: Int!, $keyColumn: String, $value: OrderInput){
deleteOrder(key: $key, keyColumn: $keyColumn, value: $value) {
OrderID, CustomerID, ShipCity, ShipCountry
}
}`;
}
}
})
});
const editSettings: EditSettingsModel = { allowAdding: true, allowDeleting: true, allowEditing: true, mode: 'Normal' };
const toolbar: ToolbarItems[] = ['Add', 'Delete', 'Update', 'Cancel'];
return <GridComponent dataSource={data} toolbar={toolbar} editSettings={editSettings} height={320}>
<ColumnsDirective>
<ColumnDirective field='OrderID' headerText='Order ID' isPrimaryKey={true} width='150'textAlign='Right'></ColumnDirective>
<ColumnDirective field='CustomerID' headerText='Customer ID' width='150'></ColumnDirective>
<ColumnDirective field='ShipCity' headerText='Ship City' width='150'/>
<ColumnDirective field='ShipCountry' headerText='Ship Country' width='150'></ColumnDirective>
</ColumnsDirective>
<Inject services={[Toolbar, Edit]} />
</GridComponent>
};
export default App;