1.引入 antd 组件 Form 无样式:忘记在入口文件 index.js 中导入 antd 的样式
import "antd/dist/antd.less" ;
2.报错 import in body of module;reorder to top.(import/first):import 语句置顶,不能位于其他代码下面
3.react 脚手架配置代理:编写 setupProxy.js 配置具体代理规则,确保灵活性.而不是在 package.json 中写死"proxy":url
5.路由链接(Link)与注册路由(Route)一一对应
6.在 LeftNav 组件,用 antd 的 menu 组件实现目录和二级目录以及刷新后选中目录依然保持选中功能的实现:(1).参考官方文档 (2).注意用 constructor(){} 钩子函数【componentWillMount 钩子函数 即将弃用】导入 munuList,若不导入将无法实现刷新后选中的二级目录不展开的问题
/* 在第一次render()之前执行一次
为第一个render()准备数据(必须同步的) */
constructor ( props ) {
super ( props ) ;
this . menuNodes = this . getMenuNodes ( menuList ) ;
}
8.在 header 组件中,使用高德地图 API ),申请key的时候注意 服务平台选 web 服务 而不要选 web 端,使用 ip 定位以及天气查询 API
9.在 header 组件中,得不到天气数据:跨域问题 -- 引入第三方库 jsonp
以下代码位于src / api / index . js ;
//ip定位
export const reqIP = ( ) => {
return new Promise ( ( resolve , reject ) => {
const url = `https://restapi.amap.com/v3/ip?output=json&key=4739c89722d5f12caf2ce96d8d022978` ;
jsonp ( url , { } , ( err , data ) => {
console . log ( "IP jsonp():" , err , data ) ;
if ( ! err && data . status === "1" ) {
const { adcode } = data ;
resolve ( { adcode } ) ;
} else {
message . error ( "获取天气信息失败" ) ;
}
} ) ;
} ) ;
} ;
//根据IP地址获取天气信息
export const reqWeather = ( city ) => {
return new Promise ( ( resolve , reject ) => {
const url = `https://restapi.amap.com/v3/weather/weatherInfo?key=4739c89722d5f12caf2ce96d8d022978&output=json&city=${ city } &extentions=base` ;
//发送jsonp请求
jsonp ( url , { } , ( err , data ) => {
console . log ( "Weather jsonp():" , err , data ) ;
if ( ! err && data . status === "1" ) {
//取出数据
const { province, city, weather, temperature } = data . lives [ 0 ] ;
resolve ( { province, city, weather, temperature } ) ;
} else {
message . error ( "获取天气信息失败" ) ;
}
} ) ;
} ) ;
} ;
10.在 category 组件中,引入 antd 的 Table 组件,修改鼠标移入移出表格行背景色的默认变化的方法:修改样式,【找到样式名】的方法见博客
以下代码位于category的同级目录下 . / index . less
. ant - table - tbody > tr . ant - table - row :hover > td ,
. ant - table - tbody > tr > td . ant - table - cell - row - hover {
background : #d4f0e7 ;
}
11.实现 category 组件异步显示一级分类目录的过程中,接口拿不到数据,获取分类列表失败:通过加入 console.log()打印数据的方式定位到了问题,数据在 result.data.data 里,而不是 result.data 里,后者导致 undefined.此外 result.status 的验证方式是 result.status===200,而不是视频课中的 result.status===0.
getCategories = async ( ) => {
//发请求前,显示loading转圈效果
this . setState ( { loading : true } ) ;
//发异步ajax请求获取数据,由于返回promise对象,用async+await阻塞获取
const result = await reqCategories ( "0" ) ;
/* 下行是测试代码 */
// console.log(result, result.data.status);
//请求完成后,去掉loading转圈效果
this . setState ( { loading : false } ) ;
if ( result . status === 200 ) {
const categories = result . data . data ; //数据在result.data.data中
this . setState ( { categories } ) ;
} else {
message . error ( "获取分类列表失败" ) ;
}
} ;
12.实现第 11 条中的功能后,第 10 条中的问题又出现了,自定义样式被默认样式覆盖:在自定义样式后加上!important 强制应用
以下代码位于category的同级目录下 . / index . less
. ant - table - tbody > tr . ant - table - row :hover > td ,
. ant - table - tbody > tr > td . ant - table - cell - row - hover {
background : #d4f0e7 ! important ;
}
13.【见第 19 条】取不到二级分类列表的数据:经过分析定位,借助 Navicat 图形化操作给数据库 category 集合每个商品对象添加一个 id 键值对
14.category 组件添加和更新表单用到 antd v4 版本的 Form 组件,使用方法与视频课有出入,参照 login 组件中 Form 的用法,都需要实现默认值初始化与表单输入检查认证的功能
15.vscode 进行 git push 提交代码时,被 reject,报错如下:远程分支上存在本地分支中不存在的提交,往往是多人协作开发过程中遇到的问题,可以先 fetch 再 merge,也就是 pull,把远程分支上的提交合并到本地分支之后再 push。如果你确定远程分支上那些提交都不需要了,那么直接 git push origin master -f,强行让本地分支覆盖远程分支。
! [rejected] master -> master (fetch first)
error: failed to push some refs to ' [email protected] :qzmly100/repository-.git'
hint: Updates were rejected because the remote contains work that you do
hint: not have locally. This is usually caused by another repository pushing
hint: to the same ref. You may want to first integrate the remote changes
hint: (e.g., ' git pull ...' ) before pushing again.
hint: See the ' Note about fast-forwards' in ' git push --help' for details.
16.category 组件更新分类功能中当前行修改分类操作 form 表单中的 input 输入框无法更新 :在对应的 Modal 组件标签中添加 destroyOnClose 属性 ,用于关闭时销毁重置数据
17.category 组件更新修改分类的回调函数父子组件通信问题(antd 4 + Modal 中表单 Form 提交验证的解决方案) :antd4.x 通过 ref 绑定获取 form 实例传过去.
category . jsx中添加setForm如下
< UpdateForm categoryName = { category ? category . name : "" }
setForm = { ( form ) => ( this . form = form ) } / >
再在update - form中:
创建ref,formRef = React . createRef ( ) ;
在钩子函数中初始化
componentDidMount ( ) {
// console.log(this.formRef, this.props);
this . props . setForm ( this . formRef . current ) ;
}
然后再传到render函数中的表单 < Form ref = { this . formRef} >
18.数据库更新操作失败,input 框的输入没拿到,显示为 undefined,而 API 经过 postman 测试也并未出问题:?
19.对于上述第 13 条,其实每个文档都有_id 属性,无需另外自行创建 id 键值对。问题出在数据库的导入操作,_id 字段在 Navicat 自动导入 collection 的时候没有去手动选择 ObjectId 类型,默认选贼 String 类型
20.category 组件中添加失败,后台显示缺少必要的 path:'name'。排查中发现定义 ajax API 接口有失误,去掉参数列表外的{},问题得到解决。但第 18 条更新列表名称功能仍未实现。
21.productHome 组件搜索分页功能,点击搜索按钮无数据返回,product 数组为空:检查发现请求参数 pageSize 为 null,修改 ajax 请求接口参数数量为 4 个变量,而不是按照视频教学的传一个对象,这是因为参数使用了全局常量 PAGE_SIZE,使用对象传递取不到值.ProductHome 组件中使用该请求函数也做相应修改
以下代码位于src / api / index . js ;
//搜索商品分页列表,searchType搜索的类型--productName/productDesc
//传4个变量pageNum, pageSize, searchName, searchType,而不是一个对象{pageNum, pageSize, searchName, searchType}.
export const reqSearchProducts = ( pageNum , pageSize , searchName , searchType ) =>
ajax ( "/base/manage/product/search" , {
pageNum,
pageSize,
[ searchType ] : searchName , //参数名作为属性名加[]号
} ) ;
22.productHone 组件商品上架下架功能失败,api 经 postman 测试无问题,但是点击 Button 按钮却无反应,数据库也未得到修改:经弹幕提示,初始化对应的表格时将 dataIndex:"status"注销,render(product)传入 products 问题得到解决,或如 render(status,product)传入 2 个参数,不注销 dataIndex:"status"也可解决问题,关键是 antd 文档 相关 API 不够熟悉
23.ProductAddUpdate 表单验证参照 Loin.jsx,用的是 antd v4 的 API 实现
以下代码位于pages / product / addUpdate . jsx
注意最新 ( antd v4 ) 自定义表单验证器的写法 , 返回的是一个Promise对象 ;
/* 自定义验证器中的验证价格函数 */
validatePrice = async ( rull , value , callback ) => {
if ( value > 0 ) {
return Promise . resolve ( ) ;
} else {
return Promise . reject ( "请输入正确的价格!" ) ;
}
} ;
24.PictruesWall 组件用 antd 的 Upload 组件实现图片上传功能,其中的 action 的值为 url 上传地址,记得在 url 路径前加上/base 来跨域【setupProxy.js 中配置代理】
25.修改商品失败:读取 imgs 里图片路径搞错并且没有使用跨域
修改utils / contants . js的常量BASE_IMG_URL ;
× export const BASE_IMG_URL = "http://localhost:5000/manage/img/upload/" ;
√ export const BASE_IMG_URL = "/base/upload/" ;
26.添加商品失败,后台显示 Error [ValidationError]: products validation failed: pCategoryId: Path pCategoryId
is required., categoryId: Path categoryId
is required.at D:\Chrome 下载\尚硅谷前端\谷粒后台资料\code\admin-server_final\routers\index.js:205:15:在商品对象的构造中去掉 categories 属性,添加 categoryId, pCategoryId 属性
以下代码位于pages / product / addUpdate . jsx line26 ;
const product = { name, desc, price, categoryId, pCategoryId, imgs } ;
27.role 组件添加角色的实现(role.jsx line52 以及 add-form.jsx)可参考 cartegory 组件添加分类列表(category.jsx line109 以及 add-form.jsx),都是父组件获取子组件表单数据
28.role 组件添加角色失败,后台显示 MongooseError [ValidatorError]: Path name
is required:修改 ajax 请求文件 api/index.js 中 reqAddRole 函数的参数名 name 为 roleName,保持与 API 文档一致
//添加角色
export const reqAddRole = ( roleName ) =>
ajax ( "/base/manage/role/add" , { roleName } , "POST" ) ;
29.role 组件中对 role 对象 Date()类型数据【创建时间,授权时间】格式转换:需对 antd 官方文档熟悉,借助 render 属性实现
以下代码位于pages / role / role . jsx line23
{
title : "创建时间" ,
dataIndex : "create_time" ,
render : formatDate ,
} ,
{
title : "授权时间" ,
dataIndex : "auth_time" ,
render : formatDate ,
} ,
30.user.jsx 组件中,为实现根据每个用户的 role_id 找到对应的角色名称时,控制台报错无法读取 role_id:initRoleName 需要参数 roles,要放在 getUser 函数中初始化,getUser 函数再放到 componentDidMount,而不是直接将 initRoleNames 放到 componentDidMount 钩子中
getUsers = async ( ) => {
const result = await reqUsers ( ) ;
// console.log(result);
if ( result . data . status === 0 ) {
message . success ( "获取用户列表成功" ) ;
const { users, roles } = result . data . data ;
this . initRoleNames ( roles ) ;
this . setState ( { users, roles } ) ;
} else {
message . error ( "获取用户列表失败" ) ;
}
} ;
31.user.jsx 组件删除用户失败:ajax 函数参数错误,userId 需用{}包裹起来--{uesrId},要传递的是一个对象
以下代码位于src / api / index . js ;
//删除用户
export const reqDeleteUser = ( userId ) =>
ajax ( "/base/manage/user/delete" , { userId } , "POST" ) ;