todolist形式的搜索框,分开组件写的,点击上下键时,框内显示当前选中的内容

### 首先  安装react 脚手架

cnpm  install  create-react-app  -g      //只需要在电脑上安装一次就好了,以后不用再下载了

### 创建项目

create-react-app  项目名( todo )

### 进入项目todo  运行项目

npm start   ---------或者下载 cnpm install yarn --save   后输入yarn  start 也可以运行项目

### 创建三个文件夹分别为:App    todo    list   

将App.js  和App.css 放入到 App文件夹中------>在到index.js中 把引入App.js的路径改一下


import App from './App/App';

在todo  list  这两个文件夹中 分别在创建一个js文件,一个css样式文件

在这里就简单的创建to.js  to.css    和    list.js    list.css

###在to.js 中写入代码


import React, {Component} from 'react'; //---->引入react模块

import "./to.css"  //---->引入to.css样式文件

class To extends Component {

    constructor(props) {

        super(props);

        this.state={ //--->初始化状态 ,所有添加的数据都从这里获取

            tt:"",

            cc:false  //--->定义一个状态为false的属性,通过属性的true,false通过三目运算符来判断

        }

    }

    ch(e){ //---->获取input框中用户当前输入的数据

        this.setState({tt:e.target.value});//--->更新状态后渲染

    }

    add(){ //添加  子组件通过事件触发的方法,通过方法的参数,将值传递给父组件

        this.props.ad(this.state.tt) //--->将当前的用户输入的数据传递给父组件

        this.state.tt="" //--->让当前用户点击添加后,input框中的值为空

    }

    render() {

        return (

{/* 这是通过兄弟组件传值的方法,把父组件中的方法调用过来*/}

{/* 获取用户的value 值得时候,需要加一个onChange事件 这是在react中的获取value值的方法 */} 

                <input className="zjb_in" type="text" value={this.state.tt}

                      onChange={this.ch.bind(this)} onFocus={this.props.bb} 

                onKeyDown={this.props.asss}/>{/* 当按键按下的时候,调用了父组件App.js中的方法*/}

                <button onClick={this.add.bind(this)}>百度一下</button>

                <div className="ss" onClick={this.props.as}></div>{/*这是让添加数据的ul框能在鼠标点击空白区域隐藏的一种方法,有点low,后面还有一种更好的办法*/}

            </div>

        );

    }

}

export default To;{/*导出文件*/}


### 在to.css中的样式,简写的,样式就不注释 了


body,div,input,p{

    margin: 0;

    padding: 0;

}

.zjb_box{

    width: 600px;

    height: 600px;

    border: 1px solid #000;

    margin: 100px auto;

    position: relative;

}

.zjb_in{

    width: 400px;

    height: 40px;

    margin: 20px 20px;

}

button{

    width: 100px;

    height: 40px;

}

.ss{

    width: 600px;

    height: 600px;

    border: 2px solid #ccc;

}


###在list.js中写入代码 


import React, {Component} from 'react';//--->引入react模块

import "./list.css"//--->引入list.css样式

class List extends Component {

    constructor(props) {

        super(props);

    }

    del(n){//删除的方法  n为当前的下标

        this.props.remove(n);//--->将当前的下标传入到父组件中

    }

    render() {

        return (

            <div className="zjb_bb" style={{display:this.props.aa?"block":"none"}}>

                <ul style={{display:this.props.aa?"block":"none"}}>{/*如果是true,就显示,false就隐藏*/}

                    {

this.props.zjb_ch.map((v,i)=>{              {/*注释this.props.zjb_ch==i?"ac":""*/}

                            return <li key={i} className={this.props.ac===i?"ac":""}>

                                {v}

                                {/*单行显示删除,可以num=0;this.state.num===i?"显示":"否则隐藏"*/}

className={this.props.ac===i?"za":" "}>删除{/*通过当前下标一样的时候,显示删除,不一样的,就为空*/}

                                {/*<div className="asas"></div>*/}

                            </li>

                        })

                    }

                </ul>

            </div>

        );

    }

}

export default List;{/*导出*/}

###在list.css中写入样式


*{

    margin: 0;

    padding: 0;

    list-style: none;

}

.zjb_bb{

    position: absolute;

    width: 402px;

    top: 164px;

    left: 286px;

    /*border: 1px solid #000;*/

}

.zjb_bb>ul{

    display: none;

    /*height: 400px;*/

    border: 1px solid #000;

}

.ac{

    background: red;

}

.za{

    float: right;

    color: #000;

    margin-right: 20px;

}

.asas{

    width: 1000px;

    height: 1000px;

    border: 2px solid greenyellow;

}


###最后回到主文件App.js中


import React from 'react';            //--->引入react模块

import To from "../todo/To";          //--->引入to.js文件

import List from "../list/list";      //--->引入list.js文件

class App extends React.Component {

  constructor(props) {

    super(props);

    var arr0=localStorage.getItem("arr");  //---> 获取本地的储存

    if(arr0){  //本地储存中如果有数据的话就会执行if中的true

      var arr1=JSON.parse(arr0);  //--->将字符串转换为对象

    }else{

      var arr1=[]; //--->如果本地储存中没有数据,就是null,先执行if中的else 创建一个新数组, 之后再渲染,输出本地储存,在渲染到页面上

    }

    this.state={

      list:arr1,

      cc:false,

      num:0

    }

  }

  add1(v){ //--->添加的方法  v就是to.js中传过来的input框中用户输入的值

    var list=this.state.list; //获取当前的初始状态中的list

    list.push(v); //将用户输入的值添加到list中

    var list1=JSON.stringify(list);//--->将当前的list是拼接的对象转换为字符串

    localStorage.setItem("arr",list1);//--->将数据储存到本地内存中

    this.setState({list}) //--->更新状态后 页面又渲染了一次

    // alert(this.state.list)

  }

  del1(n){//删除的方法  n为list.js中传递过来的下标

    var list=this.state.list;

    list.splice(n,1);

    var list1=JSON.stringify(list);

    localStorage.setItem("arr",list1);

    this.setState({list})

  }

  bb1(){//通过cc的属性来判断,当点击后让cc的状态为true

    this.setState({cc:true})

  }

  ass(){

    this.setState({cc:false})

  }

  asa(e){//键盘事件,上下键

    if(e.keyCode===38){

      console.log(e.keyCode);

      var n=this.state.num;//--->通过num判断,在list.js中如果num和下标i相等的话,就让ac的class样式显示

      n--;

      if(n<0){

        n=this.state.list.length-1

      }

      this.setState({num:n,tt:this.state.list[n]})

    }

    if(e.keyCode===40){

      // alert(11)

      var n=this.state.num;

      n++;

      if(n>this.state.list.length-1){

        n=0

      }

      this.setState({num:n,tt:this.state.list[n]})

    }

  }

  // dcc(e){

  //    if(e.target.nodeName=="DIV"){--->获取当前div的节点DIV,能获取到就改变cc的boolearn值

  //        this.setState({cc:false})

  //    }

  //隐藏

  hide=(e)=>{//--->通过获取当前的大div的节点来改变cc的true和false

    console.log(e.target.nodeName)

    if(e.target.nodeName=="DIV"){

      this.setState({cc:false})

    }

  }

  // }

  render() {

    return (

        <div>

          <To ad={this.add1.bind(this)} bb={this.bb1.bind(this)}

          as={this.ass.bind(this)} ak={this.hide.bind(this)}

          cc={this.state.cc} asss={this.asa.bind(this)}></To>{/*这是to.js中的组件*/}

          <List zjb_ch={this.state.list} remove={this.del1.bind(this)}

          aa={this.state.cc} ac={this.state.num}></List>{/*这是list.js中的组件*/}

        </div>

    );

  }

}

export default App;


###如果有喜欢研究的志同道合的朋友,欢迎一起交流!!!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容