- fixed
-
fixed issue 281 and removed customizing datetime formats. If you want to use DefaultDateTime format. Specify the sqfentity_gen version in pubspec.yaml
sqfentity_gen: 2.3.0+3
- fixed issue 279
- fixed issue 278
- New global default Date/Time Formats
- Making the viewAdd takes optional object
- some minor updates
- Added optional parameters into upsertAll method for batchCommit
-
Added optional parameters into saveAll method for batchCommit
saveAll({bool? exclusive, bool? noResult, bool? continueOnError})
- Removed dependency of flutter_datetime_picker
created UITools.showDateTimePicker() method to support another package using instead
Just create a static method as follow to call from anywhere in UITools class and customize
static Future<DateTime?> showDateTimePicker(BuildContext context,
{DateTime? currentTime,
DateTime? minTime,
DateTime? maxTime,
DateChangedCallback? onChanged,
DateChangedCallback? onConfirm,
DateCancelledCallback? onCancel}) =>
DatePicker.showDatePicker(context,
showTitleActions: true,
minTime: minTime,
maxTime: maxTime,
onChanged: onChanged,
onConfirm: onConfirm,
onCancel: onCancel,
locale: UITools.mainDatePickerLocaleType,
currentTime: currentTime,
theme: UITools.mainDatePickerTheme);
-
Added databasePath parameter into dbModel
const myDbModel = SqfEntityModel( modelName: 'MyDbModel', databaseName: 'sampleORM.db', databaseTables: [tableProduct, tableCategory, tableTodo], // This value is optional. When databasePath is null then // EntityBase uses the default path from sqflite.getDatabasesPath() databasePath: '/Volumes/Repo/MyProject/db' )
- Reduced generated models 20% less
- TableBase class and FilterBuilder classes have been abstracted as below
- bug fix
abstract class CrudOperationsBase {
Future<dynamic> save({bool ignoreBatch = true})
Future<dynamic> saveOrThrow({bool ignoreBatch = true})
Future<dynamic> saveAs({bool ignoreBatch = true})
Future<int?> upsert({bool ignoreBatch = true})
Future<BoolCommitResult> upsertAll(covariant List<dynamic> objList)
Future<BoolResult> delete([bool hardDelete = false])
Future<BoolResult> recover([bool recoverChilds = true])
}
abstract class TableBase extends CrudOperationsBase {
Map<String, dynamic> toMap({bool forQuery = false, bool forJson = false, bool forView = false});
Future<Map<String, dynamic>> toMapWithChildren([bool forQuery = false, bool forJson = false, bool forView = false]);
String toJson();
Future<String> toJsonWithChilds();
List<dynamic> toArgs();
List<dynamic toArgsWithIds();
ConjunctionBase select({List<String>? columnsToSelect, bool? getIsDeleted});
ConjunctionBase distinct({List<String>? columnsToSelect, bool? getIsDeleted});
}
Added ConjunctionBase abstract class and collected all Conjunction Keywords & toList/Single methods below
abstract class ConjunctionBase extends FluentBase {
ConjunctionBase and
ConjunctionBase or
ConjunctionBase startBlock
ConjunctionBase endBlock
ConjunctionBase where(String? whereCriteria, {dynamic parameterValue})
ConjunctionBase page(int page, int pagesize)
ConjunctionBase top(int count)
ConjunctionBase orderBy(dynamic argFields)
ConjunctionBase orderByDesc(dynamic argFields)
ConjunctionBase groupBy(dynamic argFields)
ConjunctionBase having(dynamic argFields)
/// Methods
Future<List<TableBase>> toList()
Future<BoolResult> delete([bool hardDelete = false])
Future<BoolResult> recover();
Future<BoolResult> update(Map<String, dynamic> values);
Future<TableBase?> toSingle();
Future<TableBase> toSingleOrDefault();
Future<int> toCount();
Future<String> toJson();
Future<String> toJsonWithChilds();
Future<List<dynamic>> toMapList();
Map<String, dynamic> toListPrimaryKeySQL([bool buildParams = true]);
Future<List<int>> toListPrimaryKey([bool buildParams = true]);
Future<List<dynamic>> toListObject();
Future<List<String>> toListString();
}
class FilterBase extends FluentBase {
ConjunctionBase equals(dynamic pValue)
ConjunctionBase equalsOrNull(dynamic pValue)
ConjunctionBase isNull()
ConjunctionBase contains(dynamic pValue)
ConjunctionBase startsWith(dynamic pValue)
ConjunctionBase endsWith(dynamic pValue)
ConjunctionBase between(dynamic pFirst, dynamic pLast)
ConjunctionBase greaterThan(dynamic pValue)
ConjunctionBase lessThan(dynamic pValue)
ConjunctionBase greaterThanOrEquals(dynamic pValue)
ConjunctionBase lessThanOrEquals(dynamic pValue)
ConjunctionBase inValues(dynamic pValue)
}
Fix UI
Add toSingleOrDefault() method
Add custom bundledDatabasePath and databasePath parameters to convertDatabaseToModelBase
Add closeDatabase() function
How to disconnect from database?
await MyDbModel().closeDatabase();
And reconnect:
MyDbModel().initializeDB();
Added an optional parameter ignoreBatch into save() method that sent true as a default to fix issue 214 Set ignoreBatch to false if you run more than one save() operation those are between batchStart and batchCommit
Sample:
await MyDbModel().batchStart();
for (final obj in products) {
await obj.save(ignoreBatch: false);
}
// return MyDbModel().batchCommit();
final result = await MyDbModel().batchCommit();
New features added: (by Reni Delonzek)
- Ability to specify default columns for all tables Example:
@SqfEntityBuilder(myDbModel)
const myDbModel = SqfEntityModel(
modelName: 'AppDatabase',
databaseName: 'AppDatabase.db',
databaseTables: [
table1,
table2
],
sequences: [seqIdentity],
bundledDatabasePath: null,
defaultColumns: const [
SqfEntityField('lastUpdate', DbType.datetime),
]);
All database tables (table1 and table2) will have the lastUpdate column
- Possibility of performing actions before each insert/update Example:
@SqfEntityBuilder(myDbModel)
const myDbModel = SqfEntityModel(
...
preSaveAction: getPreSaveAction,
);
Future<TableBase> getPreSaveAction(String tableName, obj) async {
// Update the lastUpdate column on all records before saving
obj.lastUpdate = DateTime.now()
return obj;
}
- Log events on failure of insert/update operation Example:
@SqfEntityBuilder(myDbModel)
const myDbModel = SqfEntityModel(
...
logFunction: getLogFunction,
);
getLogFunction(Log log) {
// Report the error to the server here
}
- Addition of saveOrThrow method, making it possible to handle throws directly Example usage with transactions:
Future<bool> saveData() async {
try {
/// Start a transaction
await AppDatabase().execSQL('BEGIN');
table1.id = await table1.saveOrThrow();
table2.table1_id = table1.id;
table2.id = table2.saveOrThrow();
/// Confirm the data in the db
return (await AppDatabase().execSQL('COMMIT')).success;
} catch (error, stackTrace) {
/// Reverses all successful operations
await AppDatabase().execSQL('ROLLBACK');
/// Cancels possible ids assigned to the object during insertions, preventing a new save attempt from accidentally performing an update
table1?.rollbackId();
table2?.rollbackId();
return false;
}
}
- Make all table classes extend from TableBase
Thanks to Reni Delonzek for these contributions.
- Migrated to null safety, min SDK is 2.12.0.
- implemented SQLChiper to encrypt DB
- Added DbType.time
- Added an option to implement an abstract class (by Amit Rotner)
-
Added Desktop support issue #59
-
Added support for sending header to the fromWebUrl() method to able using Authentication Credentials or Token issue #122
-
Added a new method named post() and postUrl() to post json to specified url with headers
-
Added VIEW support
First Define the 'Todo' constant as SqfEntityTable and generate your model
const tableTodo = SqfEntityTable(
tableName: 'todos',
primaryKeyName: 'id',
primaryKeyType: PrimaryKeyType.integer_unique,
defaultJsonUrl:
'https://jsonplaceholder.typicode.com/todos', // optional: to synchronize your table with json data from webUrl
fields: [
SqfEntityField('userId', DbType.integer),
SqfEntityField('title', DbType.text),
SqfEntityField('completed', DbType.bool, defaultValue: false),
]);
using:
final todo = Todo()
..title='test'
..userId=1
..completed = true;
final res = await todo.post(headers: {
'Content-Type': 'application/json; charset=UTF-8',
'Authorization' : 'Basic your_api_token_here'
});
You can define views like below:
const tableV_tracks = SqfEntityTable(
tableName: 'VTracks',
objectType: ObjectType.view,
fields: [
SqfEntityField('Name', DbType.text),
SqfEntityField('album', DbType.text),
SqfEntityField('media', DbType.text),
SqfEntityField('genres', DbType.text),
SqfEntityFieldRelationship(
parentTable: tableTrack,
deleteRule: DeleteRule.NO_ACTION,
fieldName: 'TrackId',
isPrimaryKeyField: false),
],
sqlStatement: '''SELECT
trackid,
track.name,
album.Title AS album,
mediatype.Name AS media,
genre.Name AS genres
FROM
track
INNER JOIN album ON Album.AlbumId = track.AlbumId
INNER JOIN mediatype ON mediatype.MediaTypeId = track.MediaTypeId
INNER JOIN genre ON genre.GenreId = track.GenreId''',
);
Get some data from the view:
final vtracs = await VTrack().select().top(5).toList();
Result:
flutter: 5 matches found
flutter: {Name: For Those About To Rock (We Salute You), album: For Those About To Rock We Salute You, media: MPEG audio file, genres: Rock, TrackId: 1}
flutter: {Name: Balls to the Wall, album: Balls to the Wall, media: Protected AAC audio file, genres: Rock, TrackId: 2}
flutter: {Name: Fast As a Shark, album: Restless and Wild, media: Protected AAC audio file, genres: Rock, TrackId: 3}
flutter: {Name: Restless and Wild, album: Restless and Wild, media: Protected AAC audio file, genres: Rock, TrackId: 4}
flutter: {Name: Princess of the Dawn, album: Restless and Wild, media: Protected AAC audio file, genres: Rock, TrackId: 5}
fixed issue #121
fixed issue #115
Added Support Collating Sequences
How to use? Set collate parameter when declaring columns like below:
SqfEntityField('name', DbType.text, collate: Collate.NOCASE)
Added ability to change columns type
## 1.3.5+2
Added a property named ignoreForFile in SqfEntityModel. You can specify the names of rules to be ignored which are specified in analysis_options.yaml file.
@SqfEntityBuilder(myDbModel)
const myDbModel = SqfEntityModel(
modelName: 'MyDbModel',
databaseName: 'sampleORM_v1.3.5+1.db',
databaseTables: [tableProduct, tableCategory, tableTodo],
ignoreForFile: [ 'avoid_unused_constructor_parameters',
'always_put_control_body_on_new_line',
'prefer_final_fields']
);
Added multicolumn index support. For more information click here
set any integer value to isIndexGroup parameter with isIndex: true example:
SqfEntityField('firtsName', DbType.text, isIndex: true, isIndexGroup: 1),
SqfEntityField('lastName', DbType.text, isIndex: true, isIndexGroup: 1),
Added SQLite Constraints and Index property to fields
NOT NULL Constraint − Ensures that a column cannot have NULL value. DEFAULT Constraint − Provides a default value for a column when none is specified. UNIQUE Constraint − Ensures that all values in a column are different. CHECK Constraint − Ensures that all values in a column satisfies certain conditions. INDEXES - Indexes are used to retrieve data from the database more quickly than otherwise.
use these parameters when declaring SqfEntityField isNotNull: true, defaultValue: 0, // specify a default value according to type of the column isUnique: true checkCondition: '(this)>0' // you can use (this) phrase instead of the column name isIndex: true
-
Define a sample table
const tableProduct = SqfEntityTable( tableName: 'product', primaryKeyName: 'id', primaryKeyType: PrimaryKeyType.integer_auto_incremental, fields: [ SqfEntityField('name',DbType.text, isUnique: true, isNotNull: true, isIndex:true), SqfEntityField('price', DbType.real, defaultValue: 1, checkCondition: '(this)>0'), ]);
-
Try to insert a row into the product table with these values:
Product product = Product(); // We didn't set any values before saving. await product.save(); print('-----TEST: NOT NULL CONSTRAINT (name must not be null)'); print(product.saveResult);
product = Product() ..name = 'product 1' ..categoryId = 1 ..price=0; // the price must be greater than 0 according to the specified CHECK CONSTRAINT
await product.save(); print('-----TEST: CHECK CONSTRAINT (price must be greater than 0)'); print(product.saveResult);
product = Product() ..name = 'product 1' ..categoryId=1 ..price=1; await product.save(); await product.saveAs(); // We saved the product again without changing the name
print('-----TEST: UNIQUE CONSTRAINT (name must be UNIQUE)'); print(product.saveResult);
flutter: -----TEST: NOT NULL CONSTRAINT (name isNotNull)
flutter: product-> Save failed. Error: "NOT NULL constraint failed: product.name"...
flutter: -----TEST: CHECK CONSTRAINT (price must be greater than 0)
flutter: product-> Save failed. Error: "CHECK constraint failed: product"...
flutter: -----TEST: UNIQUE CONSTRAINT (name must be UNIQUE)
flutter: product-> Save failed. Error: "UNIQUE constraint failed: product.name"...
fixed repeated variable declaration in tables which have more than one RelationShip to the same table
fixed bugs in nested Tables and Related tables which have more than one Primary Key
fixed bugs in MANY_TO_MANY Relationships
fixed bugs in nested ONE_TO_ONE Relationships
implemented MANY_TO_MANY Relationships
In Chinook.db there are two tables named Playlist and Track which are related with many to many relation. And there is a table called PlaylistTrack which is saved ids of those related records here. Now, you can define SqfEntityFieldRelationship column in one of these tables, like this:
Option 1: Define this SqfEntityFieldRelationship column in Track table model with the relationType is MANY_TO_MANY
SqfEntityFieldRelationship(
parentTable: tablePlaylist,
deleteRule: DeleteRule.NO_ACTION,
relationType: RelationType.MANY_TO_MANY,
manyToManyTableName: 'PlaylistTrack'),
Option 2: OR define this SqfEntityFieldRelationship column in Playlist table model
SqfEntityFieldRelationship(
parentTable: tableTrack,
deleteRule: DeleteRule.NO_ACTION,
relationType: RelationType.MANY_TO_MANY,
manyToManyTableName: 'PlaylistTrack'),
Note:
manyToManyTableName parameter is optional. SqfEntity will name it if you don't set this parameter
After generate the models you can list related records like this:
to list Tracks of a Playlist:
final playlist = await Playlist().getById(1);
final tracks = await playlist.getTracks().toList();
to list Playlists of a Track:
final track = await Track().getById(1);
final playlists = await track.getPlaylists().toList();
changed saveAll(), upsertAll(), rawInsert() methods according to whether have a primary key or not
applied following advice: hhtokpinar/sqfEntity#93 (comment)
applied isPrimaryKeyField propery to both SqfEntityField or SqfEntityFieldRelationship fields fixed some bugs
Added a new feature that allows you to set SqfEntityFieldRelationship items as primary Key set the isPrimaryKeyField parameter to true to use this feature
Sample usage:
const tableProductProperties = SqfEntityTable(
tableName: 'product_properties',
// now you do not need to set primaryKeyName If you set RelationshipField as primarykeyfield
// declare fields
fields: [
SqfEntityField('title', DbType.text),
SqfEntityField('productId', DbType.integer, isPrimaryKeyField: true),
SqfEntityFieldRelationship(fieldName: 'propertyId', parentTable: tableProperty, isPrimaryKeyField: true),
]);
Added loadParents parameter into parameters of getById, toSingle, and toList methods to preload parent of items sample using: Shop is the top element with no parent and the last element is Todo that has no children, so my simple schema is like this-> Shop > Category > Product > Todo
final todo= Todo().getById(1,loadParents:true);
print( todo.plProduct.plCategory.plShop.name );
result:
flutter: shop 1
And showed how to reach to top parent in the following figure
Extended preload option. Now preloading loads all nested children (except for parents)
added datetimeUtc dbType for SqfEntityFields
fixed bugs
Added new feature for relationships preload parent or child fields to virtual fields that created automatically starts with 'pl'
this field will be created for Product table
Category plCategory;
this field will be created for Category table
List<Product> plProducts;
Using (from parent to child): Note: You can send certain field names with preloadField parameter for preloading. For ex: toList(preload:true, preloadFields:['plField1','plField2'... etc])
final category = await Category().select()..(your optional filter)...toSingle(preload:true, preloadFields: ['plProducts']);
// you can see pre-loaded products of this category
category.plProducts[0].name
or Using (from child to parent):
final products = await Product().select().toList(preload:true);
// you see pre-loaded categories of all products
products[0].plCategory.name
Added SqfEntityFieldVirtual field type for virtual property. It will be declared in the model that only generates fields in the model class and not in the database table. They are used to store lists of preloaded rows from other tables.
Example:
When creating model:
const tablePerson = SqfEntityTable(
tableName: 'person',
primaryKeyName: 'id',
primaryKeyType: PrimaryKeyType.integer_auto_incremental,
fields: [
SqfEntityField('firstName', DbType.text),
SqfEntityField('lastName', DbType.text),
SqfEntityFieldVirtual('fullName', DbType.text),
],
customCode: '''
init()
{
fullName = '\$firstName \$lastName';
}'''
);
When using:
final personList = await Person().select().toList();
for (final person in personList) {
person.init();
print('fullName: ${person.virtualName}');
}
Also removed method called 'fromObjectList'
fixed Relationships supports text data type
Fixed some Dart Analysis hints
Applied range validator to forms (You can use minValue and maxValue property for integer, real and numeric field dbTypes)
Added One-to-One Relationships type for relationships.
as an example: to expand the product table with the properties table with one-to-one relationships
const tableProductProperties = SqfEntityTable(
tableName: 'property',
primaryKeyName: 'propertyId',
//useSoftDeleting: true,
//primaryKeyType: PrimaryKeyType.integer_auto_incremental,
fields: [
SqfEntityField('weight', DbType.real),
SqfEntityField('stockQty', DbType.numeric),
SqfEntityFieldRelationship(
//fieldName: 'productId',
parentTable: tableProduct, relationType: RelationType.ONE_TO_ONE)
],
);
after generate the code, you will see ".property" in product item properties like below: Note: In one-to-many relations There will be ".getProperty().toList()"
final product = await Product().getById(8);
product.property
..stockQty = 8
..weight = 320;
await product.save();
print(product.toMapWithChilds());
and here is DEBUG RESULT
flutter: { productId: 8, name: Notebook 15", description: 256 GB SSD, price: 10499.0, isActive: false, categoryId: 1, rownum: 8, datetime: 2019-11-21 06:22:34.512, isDeleted: false, property: {stockQty: 4, weight: 320} }
1- added equalsOrNull keyword for queries Example:
// this query lists only isActive=false
final productList = await Product().select().isActive.not.equals(true).toList();
// but this query lists isActive=false and isActive is null both
final productList = await Product().select().isActive.not.equalsOrNull(true).toList();
2- you can define customCode property of your SqfEntityTable constant for ex:
const tablePerson = SqfEntityTable(
tableName: 'person',
primaryKeyName: 'id',
primaryKeyType: PrimaryKeyType.integer_auto_incremental,
fields: [
SqfEntityField('firstName', DbType.text),
SqfEntityField('lastName', DbType.text),
],
customCode: '''
String fullName()
{
return '\$firstName \$lastName';
}
''');
modified isSaved property and to be removed when not needed
bugs fixed getById(id) -> id is null then return null instead exception and fixed .isNull() throws the exception
Converting the first character of fieldName to lowercase has been cancelled. Users should specify the field name as they wants
Added Form Generation Feature, and minValue, maxValue propery for datetime fields and fixed some bugs. Example:
@SqfEntityBuilderForm(tableCategory, formListTitleField: 'name', hasSubItems: true)
const tableCategory = SqfEntityTable(
tableName: 'category',
primaryKeyName: 'id',
primaryKeyType: PrimaryKeyType.integer_auto_incremental,
useSoftDeleting: true,
// when useSoftDeleting is true, creates a field named 'isDeleted' on the table, and set to '1' this field when item deleted (does not hard delete)
modelName:
null, // SqfEntity will set it to TableName automatically when the modelName (class name) is null
// declare fields
fields: [
SqfEntityField('name', DbType.text),
SqfEntityField('isActive', DbType.bool, defaultValue: true),
SqfEntityField('date1', DbType.datetime, defaultValue: 'DateTime.now()', minValue: '2019-01-01', maxValue: '2023-01-01'),
SqfEntityField('date2', DbType.datetime, minValue: '2020-01-01', maxValue: '2022-01-01')
]);
fixed the build error after defining multiple referance (RelationShip) to the same table
added saveResult property Note: You must re-generate your models after updating the package
Example:
final product = Product(
name: 'Notebook 12"',
description: '128 GB SSD i7',
price: 6899,
categoryId: 1);
await product.save();
print(product.saveResult.success); // bool (true/false)
print(product.saveResult.toString()); // String (message)
bugfix SequenceManager error on multi database
added unknown (text) dbType for unrecognized columns
added Date (Small Date) dbType
SqfEntityField('birthDate', DbType.date),
added DateTime dbType
SqfEntityField('birthDate', DbType.datetime),
modified sqlite dbType mapping
merged package with sqfentity_base
removed analyzer from dependencies
added FormBuilder
removed FormBuilder
added analyzer 0.38.2
sqfentity_base exported to sqfentity_base 1.0.1 package
Blob type mapped as Uint8List instead String
Added function to generate model from existing database
sqfentity_base moved into this package
some required changes applied
- Initial publish This package required for sqfentity_gen ORM for Flutter code generator Includes SqfEntity base classes and Annotation Classes