table = strtolower(get_class($this)); $this->result = null; return $this; } public function get($id) { return $this->find(array('id' => $id)); } /** * The New Find */ public function find($where = null, $sort = false, $limit = false) { $select = "*"; $q = ""; // Allow custom queries if (is_array($where)) { foreach ($where as $k => $v) { $values[] = $v; $args = explode(" ", $k); $col = array_shift($args); $op = implode(" ", $args); // No column name given, default to "id" if (empty($col)) { $col = "id"; } // No operator given, default to "=" if (empty($op)) { $op = "="; } $tmp = (empty($q)) ? ' WHERE ' : ' AND '; $q .= $tmp . $col . ' ' . $op . ' ?'; } } else { $q = "WHERE `{$this->table}`.`id` = ?"; $values = array($where); } if ($where == null) { $q = "WHERE ?"; $values = array(1); } // $schema = App::conf('tables'); // $this->schema = $schema[$this->table]; // Ugly hack until PHP 5.3 $i_sort = eval("return {$this->table}::OrderBy;"); $i_fk = eval("return {$this->table}::ForeignKey;"); $i_mtm = eval("return {$this->table}::ManyToMany;"); $i_select = eval("return {$this->table}::TableSelect;"); // $orderBy = ($sort) ? $sort : if ($sort) { list($col, $dir) = explode(' ', $sort); $sort = "ORDER BY `{$this->table}`.`$col` $dir"; } if (!$sort && $i_sort) { $sort = "ORDER BY `{$this->table}`.{$i_sort}"; } if ($i_select) { list($select_col, $args) = explode(",", $i_select); $select .= ", $args AS `{$select_col}`"; } $sql = "SELECT $select FROM `{$this->table}` $q $sort"; if ($limit) { $sql .= " LIMIT $limit"; } $stmt = DB::query($sql, $values); $i = 0; foreach ($stmt->fetchAll(PDO::FETCH_ASSOC) as $row) { $this->data[$i] = $row; $this->find_parent($row, $i); $this->find_children($row, $i); $i++; } return $this; } /** * Return all child rows for this row */ private function find_children($row, $i) { $id = $row['id']; // ID of the parent $fk = explode(",", eval("return $this->table::HasMany;")); if (empty($fk[0])) { return false; } foreach ($fk as $child) { $sql = "SELECT * FROM `$child` WHERE `{$this->table}_id` = ?"; $ref_schema = App::conf('tables'); $ref_schema = $ref_schema[$child]; if (@in_array($this->table, $ref_schema['belongsToMany'])) // m/n { $sql = "SELECT `{$child}`.*, `{$child}_{$this->table}`.*, `{$child}_{$this->table}`.id AS {$child}_{$this->table}_id, `{$child}`.* FROM `{$child}_{$this->table}`, `{$this->table}`, `$child` WHERE `{$child}_{$this->table}`.`{$this->table}_id` = `$this->table`.id AND `{$child}_{$this->table}`.`{$child}_id` = `$child`.id AND `{$this->table}`.id = ? ORDER BY `{$child}`.{$ref_schema['orderBy']}"; } else if (@in_array ($table, $ref_schema['belongsTo'])) { // 1/m $sql = "SELECT * FROM `$ref` WHERE `$ref`.`{$table}_id` = ?"; } $stmt = App::db()->prepare($sql); $stmt->execute(array($id)); $this->data[$i][$child] = $stmt->fetchAll(PDO::FETCH_ASSOC); } } /** * Find all rows for this row */ private function find_parent($row, $i) { $select = "*"; $fk = explode(",", eval("return {$this->table}::ForeignKey;")); // No parents defined if (empty($fk[0])) { return false; } foreach ($fk as $parent) { $fkey = "id"; $lkey = "{$parent}_id"; /* if ($this->schema['foreignKey'][$parent]) { list($lkey, $fkey) = explode("|", $this->schema['foreignKey'][$parent]); } */ $parent_id = $row[$lkey]; // $ref_schema = App::conf('tables'); // $ref_schema = $ref_schema[$parent]; $ref_schema = $fk['']; if ($ref_schema['select']) { foreach ($ref_schema['select'] as $a => $b) { $select .= ", $b AS `{$a}`"; } } $sql = "SELECT $select FROM `{$parent}` WHERE `{$fkey}` = ?"; $stmt = App::db()->prepare($sql); $stmt->execute(array($parent_id)); $this->data[$i][$parent] = $stmt->fetchAll(PDO::FETCH_ASSOC); } } private function find_parents() { } /** * Insert this thing in the DB and return inserted * thing */ public function insert($data = null) { if (!$data) { $data = $_POST; } if (empty($data)) { App::log("Attempted to insert empty data"); exit(App::error("Insert failed - nothing to insert")); } $insert = ""; $values = array(); foreach($data as $k => $v) { $insert .= "`{$k}`, "; $values[":{$k}"] = $v; } $insert = rtrim($insert, ", "); $val = implode(", ", array_keys($values)); $sql = "INSERT INTO `{$this->table}` ({$insert}) VALUES ({$val})"; App::log($sql); return DB::query($sql, $values); } /** * Delete this thing */ public function delete() { if (empty($_POST)) { exit(App::error("Delete without arguments")); } list($key, $value) = each($_POST); $sql = "DELETE FROM `{$this->table}` WHERE `{$key}` = ?"; return DB::query($sql, $value); } /** * Update this thing * We keep this in the Controller since it might know * more about the topmost class */ public function update($data = null, $where = null) { if (!$data) { $data = $_POST; } if (empty($data)) { exit(self::error("Update with empty parameters")); } if (empty($where)) { $where = array('id' => 'id'); } $query = ""; $values = array(); foreach ($data as $k => $v) { $query .= "`$k` = :$k, "; $values[":{$k}"] = $v; } $query = rtrim($query, ", "); list($col, $val) = each($where); $sql = "UPDATE `{$this->table}` SET $query WHERE `$col` = :$col"; return DB::query($sql, $values); } /** * Render a view */ public function render($data = null, $view = null) { // Default to the same view as the method if (!$view) { $bt = debug_backtrace(); $view = $bt[1]['function']; } if (!$view) { $view = $this->defaultAction; } if (!$data) { $data = $this->view; } $type = App::type(); $template = "../system/views/default.{$type}"; $file = "../system/views/{$this->table}/{$view}.{$type}"; if (!is_file($file)) { exit(App::error("{$this->table}_{$view}_{$type}: no such view")); } if ($data) { foreach ($data as $k => $v) { $$k = $v; } } // Capture view ob_start(); include $file; $view_contents = ob_get_contents(); ob_end_clean(); // Capture template ob_start(); include $template; $tpl_contents = ob_get_contents(); ob_end_clean(); $tpl_contents = preg_replace( '/.*?<\/title>/', "<title>{$this->pageTitle}", $tpl_contents ); echo str_replace('%%page_content%%', $view_contents, $tpl_contents); } public function match($cols, $match) { foreach ($cols as $col) { $sql .= "`{$col}`,"; } $sql = rtrim($sql, ","); $sql = "SELECT * FROM `{$this->table}` WHERE MATCH($sql) AGAINST('{$match}')"; return App::db()->query($sql); } } ?>