随着技术的进步,跨平台开发已经成为了标配,在此大背景下,ASP.NET Core也应运而生。本文主要基于ASP.NET Core+Element+Sql Server开发一个校园图书管理系统为例,简述基于MVC三层架构开发的常见知识点,前三篇篇文章简单介绍了如何搭建开发框架,登录功能,主页面功能,以及书室管理,书架管理功能的实现,本篇文章继续讲解书籍管理以及借还功能相关功能的开发,仅供学习分享使用,如有不足之处,还请指正。
在本示例中,应用最多的就是如何Element中提供的组件,和控制器中业务逻辑处理,涉及知识点如下所示:
本文主要介绍书籍管理和借还管理两个功能,如下所示:
图书管理,主要是对Book表的CRUD操作,表结构如下所示:

其中BookRackId为书架ID,与BookRack表的外键。
Book表实体类是数据表的数据映射,和数据表一一对应,如下所示:
1 namespace CLMS.Entity
2 {
3 /// <summary>
4 /// 图书实体
5 /// </summary>
6 public class BookEntity
7 {
8 /// <summary>
9 /// 唯一标识
10 /// </summary>
11 public int Id { get; set; }
12
13 /// <summary>
14 /// 图书编号
15 /// </summary>
16 public string ISBN { get; set; }
17
18 /// <summary>
19 /// 图书名称
20 /// </summary>
21 public string Name { get; set; }
22
23 /// <summary>
24 /// 图书作者
25 /// </summary>
26 public string Author { get; set; }
27
28 /// <summary>
29 /// 图书出版社
30 /// </summary>
31 public string Publisher { get; set; }
32
33 /// <summary>
34 /// 出版时间
35 /// </summary>
36 public DateTime PublishDate { get; set; }
37
38 /// <summary>
39 /// 图书类型
40 /// </summary>
41 public string BookType { get; set; }
42
43 /// <summary>
44 /// 描述
45 /// </summary>
46 public string Description { get; set; }
47
48 /// <summary>
49 /// 书架ID
50 /// </summary>
51 public long BookRackId { get; set; }
52
53 /// <summary>
54 /// 创建时间
55 /// </summary>
56 public DateTime CreateTime { get; set; }
57
58 /// <summary>
59 /// 当前登录的账号的ID
60 /// </summary>
61 public int CreateUser { get; set; }
62
63 /// <summary>
64 /// 最后编辑时间
65 /// </summary>
66 public DateTime? LastEditTime { get; set; }
67
68 /// <summary>
69 /// 最后修改人
70 /// </summary>
71 public int LastEditUser { get; set; }
72 }
73 }
图书管理页面主要包括对书籍的查询,新增,编辑,删除等操作,页面布局如下所示:
1 <div id="app">
2 <template>
3 <el-breadcrumb separator-class="el-icon-arrow-right">
4 <el-breadcrumb-item>图书管理</el-breadcrumb-item>
5 <el-breadcrumb-item>图书管理</el-breadcrumb-item>
6 </el-breadcrumb>
7 <el-form :inline="true" :model="queryCondition" class="demo-form-inline" style="margin-top: 10px; border: solid;border-width: 1px;border-color: #ebeef5;padding: 10px;">
8 <el-form-item label="书籍名称">
9 <el-input v-model="queryCondition.Name" placeholder="书籍名称"></el-input>
10 </el-form-item>
11 <el-form-item label="出版社">
12 <el-input v-model="queryCondition.Publisher" placeholder="出版社"></el-input>
13 </el-form-item>
14 <el-form-item>
15 <el-button type="primary" v-on:click="handleQuery">查询</el-button>
16 </el-form-item>
17 <el-form-item>
18 <el-button type="primary" v-on:click="handleAdd">新增</el-button>
19 </el-form-item>
20 </el-form>
21
22 <el-table :data="tableData" style="width: 100%" border :default-sort="{prop: 'date', order: 'descending'}">
23 <el-table-column type="expand">
24 <template slot-scope="props">
25 <el-form label-position="left" inline class="demo-table-expand">
26 <el-form-item label="图书馆">
27 <span>{{ props.row.LibraryName }}</span>
28 </el-form-item>
29 <el-form-item label="图书室">
30 <span>{{ props.row.LibrarySubName }}</span>
31 </el-form-item>
32 <el-form-item label="排">
33 <span>{{ props.row.Row }}</span>
34 </el-form-item>
35 <el-form-item label="列">
36 <span>{{ props.row.Column }}</span>
37 </el-form-item>
38 </el-form>
39 </template>
40 </el-table-column>
41 <el-table-column prop="ISBN" label="ISBN" sortable ></el-table-column>
42 <el-table-column prop="Name" label="书籍名称" sortable ></el-table-column>
43 <el-table-column prop="Author" label="作者" sortable ></el-table-column>
44 <el-table-column prop="Publisher" label="出版社" sortable ></el-table-column>
45 <el-table-column prop="BookType" label="书籍类型" sortable ></el-table-column>
46 <el-table-column prop="CreateTime" label="上架时间" sortable ></el-table-column>
47 <el-table-column label="操作">
48 <template slot-scope="scope">
49 <el-button size="medium" type="primary" plain v-on:click="handleEdit(scope.$index,scope.row)">编辑</el-button>
50 <el-button size="medium" type="danger" v-on:click="handleDelete(scope.$index,scope.row)">删除</el-button>
51 </template>
52 </el-table-column>
53 </el-table>
54 <el-pagination background layout="prev, pager, next" :page-size="pageSize" :current-page="currentPage" :total="total" style="margin-top:10px;" v-on:current-change="handlePageChanged" v-on:prev-click="handlePrevClick" v-on:next-click="handleNextClick"></el-pagination>
55 <el-dialog title="书籍信息" :visible.sync="dialogFormVisible">
56 <el-form :model="addOrEditForm">
57 <el-form-item label="ISBN" :label-width="formLabelWidth">
58 <el-input v-model="addOrEditForm.ISBN" autocomplete="off"></el-input>
59 </el-form-item>
60 <el-form-item label="书籍名称" :label-width="formLabelWidth">
61 <el-input v-model="addOrEditForm.Name" autocomplete="off"></el-input>
62 </el-form-item>
63 <el-form-item label="书籍作者" :label-width="formLabelWidth">
64 <el-input v-model="addOrEditForm.Author" autocomplete="off"></el-input>
65 </el-form-item>
66 <el-form-item label="出版社" :label-width="formLabelWidth">
67 <el-input v-model="addOrEditForm.Publisher" autocomplete="off"></el-input>
68 </el-form-item>
69 <el-form-item label="出版时间" :label-width="formLabelWidth">
70 <el-date-picker v-model="addOrEditForm.PublishDate" type="date" placeholder="选择日期"></el-date-picker>
71 </el-form-item>
72 <el-form-item label="书籍类型" :label-width="formLabelWidth">
73 <el-select v-model="addOrEditForm.BookType" placeholder="请选择书籍类型">
74 <el-option label="技术类" value="技术类"></el-option>
75 <el-option label="科普类" value="科普类"></el-option>
76 <el-option label="文学类" value="文学类"></el-option>
77 <el-option label="社科类" value="社科类"></el-option>
78 <el-option label="语言类" value="语言类"></el-option>
79 </el-select>
80 </el-form-item>
81 <el-form-item label="书籍描述" :label-width="formLabelWidth">
82 <el-input v-model="addOrEditForm.Description" autocomplete="off"></el-input>
83 </el-form-item>
84 <el-form-item label="存放位置" :label-width="formLabelWidth">
85 <el-tag v-model="addOrEditForm.Location" style="vertical-align:middle;">{{addOrEditForm.Location}}</el-tag>
86 <el-button icon="el-icon-place" circle v-on:click="handleLocation"></el-button>
87 </el-form-item>
88 </el-form>
89 <div slot="footer" class="dialog-footer">
90 <el-button v-on:click="dialogFormVisible = false">取 消</el-button>
91 <el-button type="primary" v-on:click="handleSave">确 定</el-button>
92 </div>
93 <el-dialog title="位置信息" :visible.sync="dialogLocationVisible">
94 <el-table :data="locationData" style="width: 100%" highlight-current-row border :default-sort="{prop: 'date', order: 'descending'}" v-on:current-change="handleLocationCurrentChange">
95 <el-table-column prop="Name" label="图书馆" sortable ></el-table-column>
96 <el-table-column prop="SubName" label="图书室" sortable ></el-table-column>
97 <el-table-column prop="Row" label="排" sortable ></el-table-column>
98 <el-table-column prop="Column" label="列" sortable ></el-table-column>
99 <el-table-column prop="Description" label="描述" sortable ></el-table-column>
100 </el-table>
101 <el-pagination background layout="prev, pager, next" :page-size="locationPageSize" :current-page="locationCurrentPage" :total="locationTotal" style="margin-top:10px;" v-on:current-change="handleLocationPageChanged" v-on:prev-click="handleLocationPrevClick" v-on:next-click="handleLocationNextClick"></el-pagination>
102 <div slot="footer" class="dialog-footer">
103 <el-button v-on:click="dialogLocationVisible = false">取 消</el-button>
104 <el-button type="primary" v-on:click="handleLocationSave">确 定</el-button>
105 </div>
106 </el-dialog>
107 </el-dialog>
108
109 </template>
110 </div>
数据交互通过JS脚本进行,书写格式和VUE2.0保持一致,在页面启动时,加载所有的书室信息,并绑定到el-table对象,所以需要在mounted函数中增加调用向服务器端发出请求,当用户新增或编辑保存时,通过axios发送请求到服务端接口。
1 <script>
2 var app= new Vue({
3 el: '#app',
4 data:function() {
5 return {
6 queryCondition:{
7 Name:'',
8 Publisher:''
9 },
10 formLabelWidth: '120px',
11 addOrEditForm:{
12 Id:0,
13 ISBN: '',
14 Name: '',
15 Author: '',
16 Publisher: '',
17 PublishDate: '',
18 BookType: '',
19 Description: '',
20 BookRackId:'',
21 Location:''
22 },
23 total:0,
24 pageSize:10,
25 currentPage:1,
26 dialogFormVisible: false,
27 dialogLocationVisible:false,
28 tableData: [],
29 queryLocationCondition:{
30 Name:'',
31 Publisher:''
32 },
33 locationData:[],
34 locationTotal:0,
35 locationPageSize:5,
36 locationCurrentPage:1,
37 locationCurrentRow: null,
38 }
39 },
40 mounted:function(){
41 this.query(1);
42 },
43 methods: {
44 handleOpen(key, keyPath) {
45 console.log(key, keyPath);
46 },
47 handleClose(key, keyPath) {
48 console.log(key, keyPath);
49 },
50 formatter(row, column) {
51 return row.address;
52 },
53 handleQuery(){
54 this.query(1);
55 },
56 handlePageChanged(val){
57 this.query(val);
58 },
59 handlePrevClick(){
60 //query(this.currentPage);
61 },
62 handleNextClick(){
63 //query(this.currentPage);
64 },
65 handleLocationPageChanged(val){
66 this.queryLocation(val);
67 },
68 handleLocationPrevClick(){
69 //query(this.currentPage);
70 },
71 handleLocationNextClick(){
72 //query(this.currentPage);
73 },
74 handleAdd(){
75 this.addOrEditForm.Id=0;
76 this.addOrEditForm.ISBN= '';
77 this.addOrEditForm.Name= '';
78 this.addOrEditForm.Author= '';
79 this.addOrEditForm.Publisher= '';
80 this.addOrEditForm.PublishDate= '';
81 this.addOrEditForm.BookType= '';
82 this.addOrEditForm.Description= '';
83 this.addOrEditForm.BookRackId='';
84 this.addOrEditForm.Location='';
85 this.dialogFormVisible=true;
86 console.log("add");
87 },
88 handleEdit(index,row){
89 console.log("当前index="+index);
90 console.log(row);
91 this.addOrEditForm.Id=row.Id;
92 this.addOrEditForm.ISBN=row.ISBN;
93 this.addOrEditForm.Name=row.Name;
94 this.addOrEditForm.Author=row.Author;
95 this.addOrEditForm.Publisher=row.Publisher;
96 this.addOrEditForm.PublishDate=row.PublishDate;
97 this.addOrEditForm.BookType=row.BookType;
98 this.addOrEditForm.Description=row.Description;
99 this.addOrEditForm.BookRackId=row.BookRackId;
100 this.addOrEditForm.Location=row.LibraryName+"-"+row.LibrarySubName+"-"+row.Row+"排"+row.Column+"列";;
101 this.dialogFormVisible=true;
102 },
103 handleDelete(index,row){
104 console.log("当前index="+index);
105 console.log(row);
106 this.$confirm('确定要删除编号为'+row.Id+'的书籍吗?', '提示', {
107 confirmButtonText: '确定',
108 cancelButtonText: '取消',
109 type: 'warning'
110 }).then(() => {
111 var that=this;
112 axios.post('/Book/Delete', {
113 Id:row.Id
114 }).then(function (response) {
115 if(response.status==200){
116 var msg = response.data;
117 console.log(msg);
118 if(msg.code=="0"){
119 //刷新页面
120 that.$message({
121 type: 'success',
122 message: '删除成功!'
123 });
124 that.query(1);
125 }else{
126 that.$message.error(msg.message);
127 }
128 }
129 console.log(response);
130 }).catch(function (error) {
131 that.$message.error(error);
132 });
133 console.log("delete");
134 }).catch(() => {
135 this.$message({
136 type: 'info',
137 message: '已取消删除'
138 });
139 });
140 },
141 query(pageNum){
142 var that = this;
143 this.tableData=[];
144 console.log("query");
145 axios.get('/Book/Query', {params:{
146 Name:this.queryCondition.Name,
147 Publisher:this.queryCondition.Publisher,
148 PageSize:this.pageSize,
149 PageNum:pageNum
150 }}).then(function (response) {
151 if(response.status==200){
152 var data = response.data;
153 var count=data.count;
154 that.total = count;
155 for (let i = 0; i < data.items.length; i++) {
156 that.tableData.push({
157 Id:data.items[i].id,
158 ISBN: data.items[i].isbn,
159 Name: data.items[i].name,
160 Author: data.items[i].author,
161 Publisher: data.items[i].publisher,
162 PublishDate: data.items[i].publishDate,
163 Description:data.items[i].description,
164 BookType: data.items[i].bookType,
165 CreateTime: data.items[i].createTime,
166 LibraryName:data.items[i].libraryName,
167 LibrarySubName:data.items[i].librarySubName,
168 Row:data.items[i].row,
169 Column:data.items[i].column,
170 });
171 }
172 }
173 console.log(response);
174 }).catch(function (error) {
175 console.log(error);
176 });
177 },
178 handleLocation(){
179 this.queryLocation(1);
180 this.dialogLocationVisible=true;
181 },
182 handleLocationCurrentChange(row){
183 this.locationCurrentRow=row;
184 },
185 queryLocation(pageNum){
186 this.locationData=[];
187 var that=this;
188 console.log("location query");
189 axios.get('/BookRack/Query',{params: {
190 Name:this.queryLocationCondition.Name,
191 SubName:this.queryLocationCondition.SubName,
192 PageSize:this.locationPageSize,
193 PageNum:pageNum
194 }}).then(function (response) {
195 if(response.status==200){
196 var data = response.data;
197 var count=data.count;
198 that.locationTotal = count;
199 for (let i = 0; i < data.items.length; i++) {
200 that.locationData.push({
201 Id: data.items[i].id,
202 libraryId:data.items[i].libraryId,
203 Name: data.items[i].name,
204 SubName: data.items[i].subName,
205 Row : data.items[i].row,
206 Column : data.items[i].column,
207 Location : data.items[i].location,
208 Description:data.items[i].description,
209 CreateTime: data.items[i].createTime,
210 });
211 }
212 }
213 console.log(that.locationData);
214 console.log(response);
215 }).catch(function (error) {
216 console.log(error);
217 });
218 },
219 handleLocationSave(){
220 console.log(this.locationCurrentRow);
221 if(this.locationCurrentRow!=null){
222 this.addOrEditForm.BookRackId=this.locationCurrentRow.Id;
223 this.addOrEditForm.Location=this.locationCurrentRow.Name+"-"+this.locationCurrentRow.SubName+"-"+this.locationCurrentRow.Row+"排"+this.locationCurrentRow.Column+"列";
224 }
225 this.dialogLocationVisible=false;
226 },
227 handleSave(){
228 var that=this;
229 axios.post('/Book/Add', {
230 Id:this.addOrEditForm.Id,
231 ISBN: this.addOrEditForm.ISBN,
232 Name: this.addOrEditForm.Name,
233 Author: this.addOrEditForm.Author,
234 Publisher: this.addOrEditForm.Publisher,
235 PublishDate: this.addOrEditForm.PublishDate,
236 BookType: this.addOrEditForm.BookType,
237 Description: this.addOrEditForm.Description,
238 BookRackId:this.addOrEditForm.BookRackId,
239 }).then(function (response) {
240 if(response.status==200){
241 var msg = response.data;
242 console.log(msg);
243 if(msg.code=="0"){
244 that.dialogFormVisible=false;
245 //刷新页面
246 that.query(1);
247 }else{
248 window.alert(msg.message);
249 }
250 console.log(that.dialogFormVisible);
251 }
252 console.log(response);
253 }).catch(function (error) {
254 console.log(error);
255 });
256 console.log("save");
257 },
258 }
259 });
260 </script>
控制器主要用于响应用户的请求,与数据库交互,并返回执行的结果信息。
1 namespace CLMS.Host.Controllers
2 {
3 public class BookController : Controller
4 {
5 private DataContext dataContext;
6
7 public BookController(DataContext context) {
8 dataContext = context;
9 }
10
11 public IActionResult Index()
12 {
13 return View();
14 }
15
16 /// <summary>
17 /// 获取符合条件的查询
18 /// </summary>
19 /// <param name="Name"></param>
20 /// <param name="Publisher"></param>
21 /// <param name="pageNum"></param>
22 /// <param name="pageSize"></param>
23 /// <returns></returns>
24 [HttpGet]
25 public PagedRequest<Book> Query(string Name, string Publisher, int pageNum, int pageSize)
26 {
27 Name = string.IsNullOrEmpty(Name) ? string.Empty : Name;
28 Publisher = string.IsNullOrEmpty(Publisher) ? string.Empty : Publisher;
29 var bookEntities = dataContext.Books.Where(r => r.Name.Contains(Name) && r.Publisher.Contains(Publisher));
30 var total = bookEntities.Count();
31 var bookDtos = bookEntities.Skip((pageNum - 1) * pageSize).Take(pageSize).Select(r => new Book() { Id = r.Id, ISBN = r.ISBN, Name = r.Name, Author = r.Author, Publisher = r.Publisher, BookType = r.BookType,BookRackId=r.BookRackId,PublishDate=r.PublishDate, CreateTime = r.CreateTime,Description=r.Description }).ToList();
32
33 //位置
34 var bookRackIds = bookDtos.Select(r => r.BookRackId).ToList();
35 var locations = dataContext.BookRacks.Where(r => bookRackIds.Contains(r.Id)).Join(dataContext.Librarys, b => b.LibraryId, l => l.Id, (b, l) => new BookRack() { Name = l.Name, SubName = l.SubName, Location = l.Location, LibraryId = b.LibraryId, Id = b.Id, Row = b.Row, Column = b.Column, Description = b.Description, CreateTime = b.CreateTime }).ToList();
36
37 bookDtos.ForEach(r => {
38 var location = locations.FirstOrDefault(l => l.Id == r.BookRackId);
39 if (location != null) {
40 r.LibraryName = location.Name;
41 r.LibrarySubName=location.SubName;
42 r.Row=location.Row;
43 r.Column=location.Column;
44 }
45 });
46 //
47 return new PagedRequest<Book>()
48 {
49 count = total,
50 items = bookDtos,
51 };
52 }
53
54 [Consumes("application/json")]
55 [HttpPost]
56 public Msg Add([FromBody] Book book)
57 {
58 Msg msg = new Msg();
59 if (book == null)
60 {
61 msg.code = 1;
62 msg.message = "对象为空";
63 return msg;
64 }
65 else
66 {
67 var userId = HttpContext.Session.GetInt32("UserId");
68
69 if (book.Id > 0)
70 {
71 //更新
72 var entity = dataContext.Books.Where(r => r.Id == book.Id).FirstOrDefault();
73 if (entity != null)
74 {
75 entity.BookRackId = book.BookRackId;
76 entity.Author = book.Author;
77 entity.Publisher = book.Publisher;
78 entity.Description = book.Description;
79 entity.BookType = book.BookType;
80 entity.ISBN = book.ISBN;
81 entity.Name = book.Name;
82 entity.LastEditUser = userId.GetValueOrDefault();
83 entity.LastEditTime = DateTime.Now;
84 dataContext.Books.Update(entity);
85 dataContext.SaveChanges();
86 }
87 else
88 {
89 msg.code = 1;
90 msg.message = "修改失败";
91 return msg;
92 }
93 }
94 else
95 {
96 //新增
97 var entity = new BookEntity()
98 {
99 BookRackId = book.BookRackId,
100 Author = book.Author,
101 Publisher = book.Publisher,
102 PublishDate = book.PublishDate,
103 Description = book.Description,
104 BookType = book.BookType,
105 ISBN = book.ISBN,
106 Name = book.Name,
107 CreateTime = DateTime.Now,
108 CreateUser = userId.GetValueOrDefault(),
109 LastEditTime = DateTime.Now,
110 LastEditUser = userId.GetValueOrDefault(),
111 };
112 dataContext.Books.Add(entity);
113 dataContext.SaveChanges();
114 }
115 msg.code = 0;
116 msg.message = "success";
117 return msg;
118 }
119 }
120
121 [Consumes("application/json")]
122 [HttpPost]
123 public Msg Delete([FromBody] Book book) {
124 Msg msg = new Msg();
125 if (book == null)
126 {
127 msg.code = 1;
128 msg.message = "对象为空";
129 return msg;
130 }
131 else
132 {
133 if (book.Id > 0)
134 {
135 var entity = dataContext.Books.Where(r => r.Id == book.Id).FirstOrDefault();
136 if (entity != null)
137 {
138 dataContext.Books.Remove(entity);
139 dataContext.SaveChanges();
140 msg.code = 0;
141 msg.message = "success";
142 }
143 else {
144 msg.code = 1;
145 msg.message = "对象不存在或已被删除";
146 }
147 }
148 else {
149
150 msg.code = 1;
151 msg.message = "对象Id小于0";
152 }
153 return msg;
154 }
155 }
156 }
157 }
经过以上几个步骤,即可完成图书管理的基本操作,主要包括图书的查询,新增,编辑,删除,已经分页等功能,如下所示:

图书借还包括图书的借阅和归还,两个功能,主要记录借阅人,借阅时间,归还时间,以及经手人,数据表结构如下所示:

数据表实体类和数据表一一对应,主要通过EntityFrameword与数据库进行映射。如下所示:
1 namespace CLMS.Entity
2 {
3 /// <summary>
4 /// 借还记录
5 /// </summary>
6 public class CirculateEntity
7 {
8 /// <summary>
9 /// 唯一标识
10 /// </summary>
11 public int Id { get; set; }
12
13 /// <summary>
14 /// 图书标识
15 /// </summary>
16 public int BookId { get; set; }
17
18 /// <summary>
19 /// 是否归还
20 /// </summary>
21 public bool IsReturn { get; set; }
22
23 /// <summary>
24 /// 借阅人
25 /// </summary>
26 public string BorrowUser { get; set; }
27
28 /// <summary>
29 /// 借阅时间
30 /// </summary>
31 public DateTime BorrowTime { get; set; }
32
33 /// <summary>
34 /// 借阅确认人
35 /// </summary>
36 public string BorrowConfirmor { get; set; }
37
38 /// <summary>
39 /// 归还时间
40 /// </summary>
41 public DateTime? ReturnTime { get; set; }
42
43 /// <summary>
44 /// 归还确认人
45 /// </summary>
46 public string? ReturnConfirmor { get; set; }
47 }
48 }
图书借还主要包括信息查询,借阅和归还等功能,页面布局如下所示:
1 <div id="app">
2 <template>
3 <el-breadcrumb separator-class="el-icon-arrow-right">
4 <el-breadcrumb-item>图书管理</el-breadcrumb-item>
5 <el-breadcrumb-item>图书借阅及归还</el-breadcrumb-item>
6 </el-breadcrumb>
7 <el-form :inline="true" :model="queryCondition" class="demo-form-inline" style="margin-top: 10px; border: solid;border-width: 1px;border-color: #ebeef5;padding: 10px;">
8 <el-form-item label="书籍名称">
9 <el-input v-model="queryCondition.Name" placeholder="书籍名称"></el-input>
10 </el-form-item>
11 <el-form-item>
12 <el-button type="primary" v-on:click="handleQuery">查询</el-button>
13 </el-form-item>
14 <el-form-item>
15 <el-button type="primary" v-on:click="handleBorrow">借阅</el-button>
16 </el-form-item>
17 <el-form-item>
18 <el-button type="primary" v-on:click="handleReturn">归还</el-button>
19 </el-form-item>
20 </el-form>
21 <el-table :data="tableData" style="width: 100%" border :default-sort="{prop: 'BorrowTime', order: 'descending'}">
22 <el-table-column prop="Name" label="书籍名称" sortable ></el-table-column>
23 <el-table-column prop="ISBN" label="ISBN" sortable ></el-table-column>
24 <el-table-column prop="BorrowUser" label="借阅人" sortable ></el-table-column>
25 <el-table-column prop="BorrowTime" label="借阅时间" sortable ></el-table-column>
26 <el-table-column prop="BorrowConfirmor" label="借阅经手人" sortable ></el-table-column>
27 <el-table-column prop="IsReturn" label="是否归还" sortable ></el-table-column>
28 <el-table-column prop="ReturnTime" label="归还时间" sortable ></el-table-column>
29 <el-table-column prop="ReturnConfirmor" label="归还经手人" sortable ></el-table-column>
30 </el-table>
31 <el-pagination background layout="prev, pager, next" :page-size="pageSize" :current-page="currentPage" :total="total" style="margin-top:10px;" v-on:current-change="handlePageChanged" v-on:prev-click="handlePrevClick" v-on:next-click="handleNextClick"></el-pagination>
32 <el-dialog title="借阅信息" :visible.sync="dialogFormBorrowVisible">
33 <el-form :model="borrowForm">
34 <el-form-item label="ISBN" :label-width="formLabelWidth">
35 <el-input v-model="borrowForm.ISBN" autocomplete="off"></el-input>
36 </el-form-item>
37 <el-form-item label="书籍名称" :label-width="formLabelWidth">
38 <el-input v-model="borrowForm.Name" autocomplete="off"></el-input>
39 </el-form-item>
40 <el-form-item label="借阅人" :label-width="formLabelWidth">
41 <el-input v-model="borrowForm.BorrowUser" autocomplete="off"></el-input>
42 </el-form-item>
43 </el-form>
44 <div slot="footer" class="dialog-footer">
45 <el-button v-on:click="dialogFormBorrowVisible = false">取 消</el-button>
46 <el-button type="primary" v-on:click="handleSaveBorrow">确 定</el-button>
47 </div>
48 </el-dialog>
49 <el-dialog title="归还信息" :visible.sync="dialogFormReturnVisible">
50 <el-form :model="returnForm">
51 <el-form-item label="ISBN" :label-width="formLabelWidth">
52 <el-input v-model="returnForm.ISBN" autocomplete="off"></el-input>
53 </el-form-item>
54 <el-form-item label="书籍名称" :label-width="formLabelWidth">
55 <el-input v-model="returnForm.Name" autocomplete="off"></el-input>
56 </el-form-item>
57 </el-form>
58 <div slot="footer" class="dialog-footer">
59 <el-button v-on:click="dialogFormReturnVisible = false">取 消</el-button>
60 <el-button type="primary" v-on:click="handleSaveReturn">确 定</el-button>
61 </div>
62 </el-dialog>
63 </template>
64 </div>
数据交互通过JS脚本进行,书写格式和VUE2.0保持一致,在页面启动时,加载所有的图书借还信息,并绑定到el-table对象,所以需要在mounted函数中增加调用向服务器端发出请求,当用户借阅或归还保存时,通过axios发送请求到服务端接口。
1 <script>
2 var app= new Vue({
3 el: '#app',
4 data:function() {
5 return {
6 queryCondition:{
7 Name:''
8 },
9 formLabelWidth: '120px',
10 addOrEditForm:{
11 Id:0,
12 ISBN: '',
13 Name: '',
14 BorrowConfirmor: '',
15 BorrowTime: '',
16 BorrowUser: '',
17 IsReturn:'',
18 ReturnConfirmor: '',
19 ReturnTime: '',
20 },
21 borrowForm:{
22 Id:0,
23 ISBN: '',
24 Name: '',
25 BorrowUser:''
26 },
27 returnForm:{
28 Id:0,
29 ISBN: '',
30 Name: '',
31 },
32 total:0,
33 pageSize:10,
34 currentPage:1,
35 tableData: [],
36 dialogFormBorrowVisible: false,
37 dialogFormReturnVisible: false,
38 }
39 },
40 mounted:function(){
41 this.query(1);
42 },
43 methods: {
44 handleOpen(key, keyPath) {
45 console.log(key, keyPath);
46 },
47 handleClose(key, keyPath) {
48 console.log(key, keyPath);
49 },
50 formatter(row, column) {
51 return row.address;
52 },
53 handleQuery(){
54 console.log("query");
55 this.query(1);
56 },
57 handlePageChanged(val){
58 this.query(val);
59 },
60 handlePrevClick(){
61 //query(this.currentPage);
62 },
63 handleNextClick(){
64 //query(this.currentPage);
65 },
66 handleBorrow(){
67 this.dialogFormBorrowVisible=true;
68 },
69 handleReturn(){
70 this.dialogFormReturnVisible=true;
71 },
72 handleSaveBorrow(){
73 this.$confirm('确定要借阅编号为'+this.borrowForm.ISBN+'的书籍吗?', '提示', {
74 confirmButtonText: '确定',
75 cancelButtonText: '取消',
76 type: 'warning'
77 }).then(() => {
78 var that=this;
79 axios.post('/Circulate/Borrow', {
80 Id:that.borrowForm.Id,
81 ISBN:that.borrowForm.ISBN,
82 Name:that.borrowForm.Name,
83 BorrowUser:that.borrowForm.BorrowUser,
84 }).then(function (response) {
85 if(response.status==200){
86 var msg = response.data;
87 console.log(msg);
88 if(msg.code=="0"){
89 //刷新页面
90 that.dialogFormBorrowVisible=false;
91 that.$message({
92 type: 'success',
93 message: '借阅成功!'
94 });
95 that.query(1);
96 }else{
97 that.$message.error(msg.message);
98 }
99 }
100 console.log(response);
101 }).catch(function (error) {
102 that.$message.error(error);
103 });
104 console.log("delete");
105 }).catch(() => {
106 this.$message({
107 type: 'info',
108 message: '已取消借阅'
109 });
110 });
111 },
112 handleSaveReturn(){
113 this.$confirm('确定要归还编号为'+this.returnForm.ISBN+'的书籍吗?', '提示', {
114 confirmButtonText: '确定',
115 cancelButtonText: '取消',
116 type: 'warning'
117 }).then(() => {
118 var that=this;
119 axios.post('/Circulate/Return', {
120 Id:that.returnForm.Id,
121 ISBN:that.returnForm.ISBN,
122 Name:that.returnForm.Name,
123 }).then(function (response) {
124 if(response.status==200){
125 var msg = response.data;
126 console.log(msg);
127 if(msg.code=="0"){
128 //刷新页面
129 that.dialogFormReturnVisible=false;
130 that.$message({
131 type: 'success',
132 message: '归还成功!'
133 });
134 that.query(1);
135 }else{
136 that.$message.error(msg.message);
137 }
138 }
139 console.log(response);
140 }).catch(function (error) {
141 that.$message.error(error);
142 });
143 console.log("delete");
144 }).catch(() => {
145 this.$message({
146 type: 'info',
147 message: '已取消归还'
148 });
149 });
150 },
151 query(pageNum){
152 var that = this;
153 this.tableData=[];
154 console.log("query");
155 axios.get('/Circulate/Query', {params:{
156 Name:this.queryCondition.Name,
157 PageSize:this.pageSize,
158 PageNum:pageNum
159 }}).then(function (response) {
160 if(response.status==200){
161 var data = response.data;
162 var count=data.count;
163 that.total = count;
164 for (let i = 0; i < data.items.length; i++) {
165 that.tableData.push({
166 Id:data.items[i].id,
167 ISBN: data.items[i].isbn,
168 Name: data.items[i].name,
169 BorrowConfirmor: data.items[i].borrowConfirmor,
170 BorrowTime: data.items[i].borrowTime,
171 BorrowUser: data.items[i].borrowUser,
172 IsReturn:data.items[i].isReturn==true?'已归还':'未归还',
173 ReturnConfirmor: data.items[i].returnConfirmor,
174 ReturnTime: data.items[i].returnTime,
175 });
176 }
177 }
178 console.log(response);
179 }).catch(function (error) {
180 console.log(error);
181 });
182 },
183 }
184 });
185 </script>
控制器主要用于响应用户的请求,与数据库交互,并返回执行的结果信息。
1 namespace CLMS.Host.Controllers
2 {
3 /// <summary>
4 /// 借还管理
5 /// </summary>
6 public class CirculateController : Controller
7 {
8 private DataContext dataContext;
9
10 public CirculateController(DataContext context)
11 {
12 dataContext = context;
13 }
14
15 public IActionResult Index()
16 {
17 return View();
18 }
19
20 [HttpGet]
21 public PagedRequest<Circulate> Query(string Name, int pageNum, int pageSize)
22 {
23 Name = string.IsNullOrEmpty(Name) ? string.Empty : Name;
24 var dtos = dataContext.Circulates.Join(dataContext.Books, c => c.BookId, b => b.Id, (c, b) => new Circulate()
25 {
26 Id = c.Id,
27 Name = b.Name,
28 BookId = c.BookId,
29 BorrowConfirmor = c.BorrowConfirmor,
30 BorrowTime = c.BorrowTime,
31 BorrowUser = c.BorrowUser,
32 ISBN = b.ISBN,
33 IsReturn = c.IsReturn,
34 ReturnConfirmor = c.ReturnConfirmor,
35 ReturnTime = c.ReturnTime,
36 }).Where(r=>r.Name.Contains(Name));
37 var total = dtos.Count();
38 var dtos2 = dtos.Skip((pageNum - 1) * pageSize).Take(pageSize).ToList();
39 //
40 return new PagedRequest<Circulate>()
41 {
42 count = total,
43 items = dtos2,
44 };
45 }
46
47 [Consumes("application/json")]
48 [HttpPost]
49 public Msg Borrow([FromBody]Borrow borrow) {
50 Msg msg = new Msg();
51 if (borrow == null || string.IsNullOrEmpty(borrow.ISBN))
52 {
53 msg.code = 1;
54 msg.message = "书籍为空";
55 return msg;
56 }
57 var book = dataContext.Books.FirstOrDefault(r => r.ISBN == borrow.ISBN);
58 if (book == null)
59 {
60 msg.code = 1;
61 msg.message = "ISBN有误";
62 return msg;
63 }
64 var entity = dataContext.Circulates.FirstOrDefault(r => r.BookId == book.Id && r.IsReturn == false);
65 if (entity != null)
66 {
67 msg.code = 1;
68 msg.message = "书籍已被借阅";
69 return msg;
70 }
71 var userId = HttpContext.Session.GetInt32("UserId");
72 if (userId < 0) {
73 msg.code = 1;
74 msg.message = "尚未登录";
75 return msg;
76 }
77 var borrorConfirmor = dataContext.Users.FirstOrDefault(r => r.Id == userId)?.NickName;
78 var entity2 = new CirculateEntity()
79 {
80 Id = 0,
81 BookId = book.Id,
82 IsReturn = false,
83 BorrowTime = DateTime.Now,
84 BorrowUser=borrow.BorrowUser,
85 BorrowConfirmor= borrorConfirmor,
86 };
87 this.dataContext.Circulates.Add(entity2);
88 this.dataContext.SaveChanges();
89 msg.code = 0;
90 msg.message = "success";
91 return msg;
92 }
93
94 /// <summary>
95 /// 归还
96 /// </summary>
97 /// <param name="returns"></param>
98 /// <returns></returns>
99 [Consumes("application/json")]
100 [HttpPost]
101 public Msg Return([FromBody] Return returns) {
102 Msg msg = new Msg();
103 if (returns == null || string.IsNullOrEmpty(returns.ISBN))
104 {
105 msg.code = 1;
106 msg.message = "书籍为空";
107 return msg;
108 }
109 var book = dataContext.Books.FirstOrDefault(r => r.ISBN == returns.ISBN);
110 if (book == null)
111 {
112 msg.code = 1;
113 msg.message = "ISBN有误";
114 return msg;
115 }
116 var userId = HttpContext.Session.GetInt32("UserId");
117 if (userId < 0)
118 {
119 msg.code = 1;
120 msg.message = "尚未登录";
121 return msg;
122 }
123 var returnConfirmor = dataContext.Users.FirstOrDefault(r => r.Id == userId)?.NickName;
124 var entity = dataContext.Circulates.FirstOrDefault(r => r.BookId == book.Id && r.IsReturn == false);
125 if (entity != null)
126 {
127 entity.IsReturn = true;
128 entity.ReturnTime = DateTime.Now;
129 entity.ReturnConfirmor=returnConfirmor;
130 dataContext.Circulates.Update(entity);
131 dataContext.SaveChanges();
132 msg.code = 0;
133 msg.message = "success";
134 }
135 else {
136 msg.code = 1;
137 msg.message = "书籍已归还";
138 }
139 return msg;
140 }
141 }
142 }
图书借还主要包括借阅和归还,如下所示:

以上就是校园图书管理系统的图书管理及图书借还功能实现,功能正在开发完善中,后续功能再继续介绍。旨在抛砖引玉,一起学习,共同进步。
我正在使用i18n从头开始构建一个多语言网络应用程序,虽然我自己可以处理一大堆yml文件,但我说的语言(非常)有限,最终我想寻求外部帮助帮助。我想知道这里是否有人在使用UI插件/gem(与django上的django-rosetta不同)来处理多个翻译器,其中一些翻译器不愿意或无法处理存储库中的100多个文件,处理语言数据。谢谢&问候,安德拉斯(如果您已经在rubyonrails-talk上遇到了这个问题,我们深表歉意) 最佳答案 有一个rails3branchofthetolkgem在github上。您可以通过在Gemfi
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm
我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI
我安装了ruby版本管理器,并将RVM安装的ruby实现设置为默认值,这样'哪个ruby'显示'~/.rvm/ruby-1.8.6-p383/bin/ruby'但是当我在emacs中打开inf-ruby缓冲区时,它使用安装在/usr/bin中的ruby。有没有办法让emacs像shell一样尊重ruby的路径?谢谢! 最佳答案 我创建了一个emacs扩展来将rvm集成到emacs中。如果您有兴趣,可以在这里获取:http://github.com/senny/rvm.el
这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub
我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain
是否有简单的方法来更改默认ISO格式(yyyy-mm-dd)的ActiveAdmin日期过滤器显示格式? 最佳答案 您可以像这样为日期选择器提供额外的选项,而不是覆盖js:=f.input:my_date,as::datepicker,datepicker_options:{dateFormat:"mm/dd/yy"} 关于ruby-on-rails-事件管理员日期过滤器日期格式自定义,我们在StackOverflow上找到一个类似的问题: https://s
无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD
电脑0x0000001A蓝屏错误怎么U盘重装系统教学分享。有用户电脑开机之后遇到了系统蓝屏的情况。系统蓝屏问题很多时候都是系统bug,只有通过重装系统来进行解决。那么蓝屏问题如何通过U盘重装新系统来解决呢?来看看以下的详细操作方法教学吧。 准备工作: 1、U盘一个(尽量使用8G以上的U盘)。 2、一台正常联网可使用的电脑。 3、ghost或ISO系统镜像文件(Win10系统下载_Win10专业版_windows10正式版下载-系统之家)。 4、在本页面下载U盘启动盘制作工具:系统之家U盘启动工具。 U盘启动盘制作步骤: 注意:制作期间,U盘会被格式化,因此U盘中的重要文件请注