query[$this->ptr]); } /** * Return first record (mapper object) that matches criteria * @return object|FALSE * @param $filter string|array * @param $options array * @param $ttl int **/ function findone($filter=NULL,array $options=NULL,$ttl=0) { return ($data=$this->find($filter,$options,$ttl))?$data[0]:FALSE; } /** * Return array containing subset of records matching criteria, * total number of records in superset, specified limit, number of * subsets available, and actual subset position * @return array * @param $pos int * @param $size int * @param $filter string|array * @param $options array * @param $ttl int **/ function paginate( $pos=0,$size=10,$filter=NULL,array $options=NULL,$ttl=0) { $total=$this->count($filter,$ttl); $count=ceil($total/$size); $pos=max(0,min($pos,$count-1)); return array( 'subset'=>$this->find($filter, array_merge( $options?:array(), array('limit'=>$size,'offset'=>$pos*$size) ), $ttl ), 'total'=>$total, 'limit'=>$size, 'count'=>$count, 'pos'=>$pos<$count?$pos:0 ); } /** * Map to first record that matches criteria * @return array|FALSE * @param $filter string|array * @param $options array * @param $ttl int **/ function load($filter=NULL,array $options=NULL,$ttl=0) { return ($this->query=$this->find($filter,$options,$ttl)) && $this->skip(0)?$this->query[$this->ptr=0]:FALSE; } /** * Map to first record in cursor * @return mixed **/ function first() { return $this->skip(-$this->ptr); } /** * Map to last record in cursor * @return mixed **/ function last() { return $this->skip(($ofs=count($this->query)-$this->ptr)?$ofs-1:0); } /** * Map to nth record relative to current cursor position * @return mixed * @param $ofs int **/ function skip($ofs=1) { $this->ptr+=$ofs; return $this->ptr>-1 && $this->ptrquery)? $this->query[$this->ptr]:FALSE; } /** * Map next record * @return mixed **/ function next() { return $this->skip(); } /** * Map previous record * @return mixed **/ function prev() { return $this->skip(-1); } /** * Save mapped record * @return mixed **/ function save() { return $this->query?$this->update():$this->insert(); } /** * Delete current record * @return int|bool **/ function erase() { $this->query=array_slice($this->query,0,$this->ptr,TRUE)+ array_slice($this->query,$this->ptr,NULL,TRUE); $this->ptr=0; } /** * Define onload trigger * @return callback * @param $func callback **/ function onload($func) { return $this->trigger['load']=$func; } /** * Define beforeinsert trigger * @return callback * @param $func callback **/ function beforeinsert($func) { return $this->trigger['beforeinsert']=$func; } /** * Define afterinsert trigger * @return callback * @param $func callback **/ function afterinsert($func) { return $this->trigger['afterinsert']=$func; } /** * Define oninsert trigger * @return callback * @param $func callback **/ function oninsert($func) { return $this->afterinsert($func); } /** * Define beforeupdate trigger * @return callback * @param $func callback **/ function beforeupdate($func) { return $this->trigger['beforeupdate']=$func; } /** * Define afterupdate trigger * @return callback * @param $func callback **/ function afterupdate($func) { return $this->trigger['afterupdate']=$func; } /** * Define onupdate trigger * @return callback * @param $func callback **/ function onupdate($func) { return $this->afterupdate($func); } /** * Define beforeerase trigger * @return callback * @param $func callback **/ function beforeerase($func) { return $this->trigger['beforeerase']=$func; } /** * Define aftererase trigger * @return callback * @param $func callback **/ function aftererase($func) { return $this->trigger['aftererase']=$func; } /** * Define onerase trigger * @return callback * @param $func callback **/ function onerase($func) { return $this->aftererase($func); } /** * Reset cursor * @return NULL **/ function reset() { $this->query=array(); $this->ptr=0; } }