track selecting and unselecting an option When updating a table - php

I have three tables (I simplified it in this question) :
Table 1
id | Name
-----------
1 | John
2 | Smith
Table 2
id | Title
-----------
1 | Developer
2 | Web Developer
3 | A New Title
Table 3 (links between 1 and 2)
idName | idTitle
-----------
1 | 1
1 | 2
2 | 1
My problem is with the update page. My HTML is <select multiple> ... options ... </select> and I am using chosen to select and deselect options. I am trying to do this with PHP only.
Lets say that we have the following scenario:
The administrator wanted to remove the 'Developer' Title for 'John' and add 'New Title' for 'John'.
He/She will deselect Title id 1 and select id 3. When He/She submits the form I will get two select values: id 1 (the one that he selected) and id 2 (the one that was already there). But I will not get id 3 because he deselected it.
What I am struggling with is : when the user posted the new selected values the ones that were deselected were not submitted with the form. There is no way for me to track the ones that were deselected so I can delete them from my table. How do you update your table with the new changes then? Do you delete what is already existed in that table and add the ids again? Is there a better option?
UPDATE :
It seems #wander answer is less destructive than the other ones. However, #wonder did not explain how to compute step 4 in his answer. to distinguesh between new selected options, already existing selected options, and deselected options.

There's no need to submit the deselected one. e.g.
John has two titles(Developer and Web Developer) saved in database.
The administrator opens the page, selects 'New Title', deselects 'Developer' and clicks submit.
Now on server side, we can get
the title list of John stored in database: Developer and Web Developer
title list submitted from the users: Web Developer and New Title
We compare the two title lists and figure out: we shall delete Developer title and add New Title on John.

You could add hidden fields before the select menu which will cause 0/falsey values to also be sent to PHP.
<input type="hidden" name="stuff[]" value="0" />
<input type="hidden" name="stuff[]" value="0" />
<input type="hidden" name="stuff[]" value="0" />
<select multiple name="stuff[]">
<option value="1">1</option>
<option value="2">2</option>
<option value="3">3</option>
</select>
This is similar to how you would send unchecked check boxes to the server:
https://stackoverflow.com/a/1992745/268074 (note the answer without JS)

You can dump your old values in a hidden select, so you'll get them when handling your form.
<!-- Invisible select field -->
<select multiple name="oldData[]" style="display:none;" aria-hidden="true">
<option value="1">1</option>
<option value="2" selected>2</option>
<option value="3" selected>3</option>
</select>
<!-- Real select field -->
<select multiple name="data[]">
<option value="1">1</option>
<option value="2">2</option>
<option value="3">3</option>
</select>
Note: I was inspired by #Petah's answer, don't forget to upvote him :-).

Let's say we have $old_titles list with titles from db, and $new_titles titles from submit.
$add = array_diff($new_titles, $old_titles);
$delete = array_diff($old_titles, $new_titles);
$add - list of titles to add, $delete - list of titles to delete.
Add action:
$dbh->prepare('INSERT INTO table3(idName, idTitle) VALUES (:idName, :idTitle)');
foreach($add as $titleId) {
$dbh->execute(['idName' => $userId, 'idTitle' => $titleId]);
}
Delete action:
$dbh->prepare('DELETE FROM table3 WHERE idName = :idName AND idTitle = :idTitle');
foreach($delete as $titleId) {
$dbh->execute(['idName' => $userId, 'idTitle' => $titleId]);
}

Not sure why everybody keeps using hidden input fields to track the existing data. When you POST your form, you can just retrieve the original values from the database. That's a lot safer then depending on user input, because even hidden fields can be altered. I know it won't matter too much in this particular situation, but I think it's best to always use the best practise.
Now in PHP you have two options. The first would be to delete all relations and then add the new relations. The second would be to delete all that wasn't posted and add/update the relations that where.
Although the second options might seem to be the "least destructive" as you call it, it is the easiest and least error prone way. The only reason I would abandon this tactic is when I would store extra data in the relations table. EG: date added (when did somebody get into a function) or added by user.
How would you go about it (method 2)?
Keep in mind that I use a method that tries to continue whenever possible. Another approach would be to halt when false data is posted.
<?php
if (array_key_exists("relations",$_POST) && is_array($_POST["relations"])) {
//no duplicates, you dont want to store the same function twice
//only integers
$list = array_unique(array_filter($_POST["relations"], "is_int"));
$csv = implode(',', $list);
//get the userid hoewever you normally would
$userId = 1
//Delete existing
$query = "DELETE FROM Table3 WHERE idname=".$userId
//insert all posted functions
//I select the id's from the actual table so it will never inserted a none existing ID.
//this approach never has duplicates, so the duplicate check in the beginning is redundant now, but I leave it incase I change this.
$query = "INSERT INTO Table3 (idName, idTitle) SELECT ".$userId.", Id FROM Table2 WHERE id IN (".$csv.")".
//execute queries
} else {
//nothing posted, delete all relations
}
?>
Now if you only want to insert new data and remove none-existing
<?php
if (array_key_exists("relations",$_POST) && is_array($_POST["relations"])) {
//no duplicates, you dont want to store the same function twice
//only integers
$list = array_unique(array_filter($_POST["relations"], "is_int"));
$csv = implode(',', $list);
//get the userid hoewever you normally would
$userId = 1
//Delete existing
$query = "DELETE FROM Table3 WHERE idname=".$userId." AND idTitle NOT IN (".$csv.")"
//only insert new functions
//I select the id's from the actual table so it will never inserted a none existing ID.
//this approach never has duplicates, so the duplicate check in the beginning is redundant now, but I leave it incase I change this.
$query = "INSERT INTO Table3 (idName, idTitle) SELECT ".$userId.", Id FROM Table2 WHERE id IN (".$csv.") AND id NOT IN (SELECT idTitle FROM table3 WHERE idName=".$userId.")".
//execute queries
} else {
//nothing posted, delete all relations
}
?>

The only better way is the simplest one, remember KIS
on update you just need to do following
DELETE FROM table3 WHERE idName=??
and then insert all selected idTitles again in table3

This can be done in two SQL statement without knowing which value has been de-selected. Following these 2 steps
1) In your PHP, create a comma separated list of all selected option. For example: 2, 3.
2) Then execute the following statements:
DELETE FROM tblLink WHERE idName = $id AND idTitle NOT IN ($list);
INSERT IGNORE tblLink (idName, idTitle)
SELECT $id, id FROM tblTitle WHERE id IN ($list);
Note: for shake of simplification, I use $id and $list in my example. When you intent to use it, you should properly prepare and bind parameter properly. Make sure that idName and idTitle are primary key to make it work.

To my opinion you have to check what actions should be performed first. This is a small example that you can run here http://sandbox.onlinephpfunctions.com/code/0fa11b3c9f846e344c912f5a3e994eea5e9cac34.
First run a select on Table 3 for idName=1 the output would be an array of idTitle's (1,2). In the following example the $array1 is the output of such a select statement and $array2 is the array of new idTitle's for the same idName that will have to be updated or deleted or inserted. If for example the new idTitles for a user are more or less than the previous ones then there you should have some sort of control over it in order to decide what queries you will run to the database.
$array1 = array(1,7,3,5);// initial idTitles try it on the above link (php sandbox) with more or less values
$array2 = array(1,4,3,6);//new idTitles try it on the above link (php sandbox) with more or less values
$count1=count($array1);
$count2=count($array2);
$diff=$count1-$count2;
if ($diff==0) {
$result1=array_values(array_diff_assoc($array2,$array1));
$result2=array_values(array_diff_assoc($array1,$array2));
$countr=count($result1);// or $result2 it is the same since they have the same length
$cr=0;
while($cr < $countr) {
print_r("update tblname set val=$result1[$cr] where id=theid and val=$result2[$cr]\n");
$cr++;
}
}
if ($diff!=0) {
$result1=array_diff($array2,$array1);
$result2=array_diff($array1,$array2);
if(count($result2)>0) {
foreach($result2 as $r2) {
print_r("delete from tblname where id=theid and val=$r2\n");
}
}
foreach($result1 as $r1) {
print_r("insert into tblname where id=theid and val=$r1\n");
}
}

How I would do it would extend the 1st table
id | Name | Title IDs
1 | John | 1,2
2 | Smith | 1,3
Each time the row is updated I would simply just replace the title_ids fields and be done with it.

Related

Transferring data from int to string

A user puts in their information via a scroll down menu and selects an option a couple of times. When they do this, it is saved as a INT rather then a STRING. This is because some calculations need to be run on the informaton. So essentially it looks like this:
<select name="option1">
<option value="0">Chair</option>
<option value="10">Table</option>
<option value="20">Counter</option>
<option value="30">Toaster</option>
<option value="40">Oven</option>
<option value="50">Microwave</option>
</select>
<select name="option2">
<option value="0">Red</option>
<option value="2">Blue</option>
<option value="4">Green</option>
<option value="6">Yellow</option>
<option value="8">Orange</option>
These values are added together to make a sum. So a blue(2) table(10) would have a value of 12.
My problem arises when I want users to be able to edit this data. I want to display the data they currently have, however it is now saved as a INT. How would I make it so that the information is displayed back in text format based on a the value in the database. (ie. a value of 38 would output "Orange Toaster)?
*I am doing this in PHP but I am not necassarily looking for code, just an idea on how to do this.
You can save the options in database with the following table structure.
Table Name: selection_options
id(pk) option_name option_value category
1 chair 0 option1
2 Table 10 option1
3 Toaster 30 option1
.
.
6 red 0 option2
7 Orange 8 option2
.
.
so on
For creating the selection box option1 execute the query as below:
$option1Array = "SELECT * FROM selection_options WHERE category = 'option1'" ;
This will give you all the options available in the category option1
Do same for the option2 now your selection boxes will be changed as below:
<select name="option1">
<?php foreach($option1Array as $option1) {?>
<option value="<?php echo $option1['id']; ?>"><?php echo $option1['option_name']; ?></option>
<?php }?>
</select>
<select name="option2">
<?php foreach($option2Array as $option2) { ?>
<option value="<?php echo $option2['id']; ?>"><?php echo $option2['option_name']; ?></option>
<?php } ?>
</select>
Now you have to save this data in the database as below:
Table Name: users_selected_options
id user_id option1 option2
1 1 3 7
.
.
id: Primary key of the table
user_id: Foreign key of the user table
option1: Foreign key of the selection_options
option2: Foreign key of the selection_options
No you want to show what options user has selected fire a execute join query you will get the result.
SELECT uso . * , so.option_name, so1.option_name, (so.option_value + so1.option_value) AS total
FROM users_selected_options AS uso
JOIN selection_options AS so ON uso.option1 = so.id
JOIN selection_options AS so1 ON uso.option2 = so1.id
WHERE user_id =1
LIMIT 0 , 30
The above query will give you the records as below:
id user_id option1 option2 option_name option_name total
1 1 3 4 Toaster Orange 38
So now you can easily iterate over the result and show the appropriate output to the user.
This solution is might be lengthy for you but it will helps in future when you want to make the
application dynamic as.
Now you can easily provide the interface to add / edit the values
and options names.
2.If the option names and values are being changed
you don't have to worry about as we are getting the SUM directly by
the query.
3.Now you can able to save the multiple data against the
same user.
4.In future if you want to provide option3 for the
selection then you can do this change easily and you just have to
update the query not in the entire code.
To start you will need to find the remainder when divided by 10.
if($numberX%10 == 4) { echo "Green"; }
You need a table to save all of the options eg:
id info
-- ----
0 Red Chair
2 Blue Chair
...
But I might be better saved in 2 tables, why do you sum them up as one int?
OK since you said no need to add them up, try two tables:
id furniture
-- -------
10 ...
id color
-- -----
2 Blue
For non-table solution, you save above values into an array eg
$furniture = array(
10 => '...',
...
);
$colors = array(
2 => 'Blue',
...
);
It is not a good idea to make some ID value as 0
You can use math functions.
Use division that you can find remainder which would be your option2 and use subtraction (integer - remainder) = your option1

checkbox inputs and DB search

I'm tagging Movies and store this into a Database.
a tag_id could be a car, train, boat and details could be color or specific types.
DB
movie_id | tag_id | tag_detailsid
2612 | 75 | 1
2612 | 10 | 3
2612 | 12 | 2
The tags are submitted via a form with checkboxes (all checkboxes checked are added to the db)
Now..
How do I keep track with checkboxes I uncheck..at a later stage
So looking at the above example.. for movie_id 2612, I uncheck tag 12 with id 2 as details.
So $_POST holds only 75-1 and 10-3....12-2 should be deleted from dB.
So I thought.. I simply go through the dB with a while loop and compare the db value with the values I get from the checkboxes (via the $_Post method)..
I count the values in the $_Post and it shows 2 (checkboxes checked).
Database however holds 3 entries ($numberofrecords) for that specific Movie.
My script so far...
$sql_query = "Select tag_id, tag_details_id FROM movie_tags WHERE movie_id = '2612";
$report = MYSQL_QUERY($sql_query);
$numberofrecords = mysql_num_rows ($report);
while ($report_row = mysql_fetch_array($report))
{
$thistag_id = $report_row['tag_id'];
$tag_details_id = $report_row['tag_details_id'];
foreach($_POST as $checkbox_id => $value)
{
if ($thistag_id == $checkbox_id) // check if DB tag equals checkbox value
{
echo "checkbox value is checked equals value in DB";
}
}
}
How do I track the record I need to delete from the database?
Sometimes the easiest solution is the way to go.
Unless you have a compelling reason NOT to, you can simply delete all of the tag records before saving, then insert only those that were checked:
sample code using OP's selected db driver
$sql = 'DELETE FROM movie_tags WHERE movie_id=2612';
mysql_query($sql);
foreach($_POST as $checkbox_id => $value) {
// save your records here
// To show you the code I'd need to see the HTML for the checkboxes.
}
NOTE: You should not be using mysql_. Use PDO / mysqli instead: Why shouldn't I use mysql_* functions in PHP?
A simple solution would be to delete every rows based on your primary key such as movie_id in your case and do insert in loop for every checked checkboxes. That is:
Delete rows based on movie_id
Loop through POST data and do insert

PHP SQL update checkboxes in db table

I have a update.php page which checks checkbox values and updates the data in the DB.
For example, the sport_table looks like this (primary key is a combination of MemberID-Sport):
MemberID | Sport
John | Football
John | Rugby
John | Cricket
Paul | Football
Paul | Rugby
Mike | Cricket
So what I want to do is get get the value from the checkboxes on my webform and update them as necesary in the db table.
I have tried the following code:
$sport = $_POST["sport"];
for ($i=0; $i < sizeof($sport); $i++) {
$sql = "UPDATE sport_table
SET sport='$sport[$i]', MemberID='$username'";
$result = mysqli_query($conn, $sql);
But receive an errors duplicate keys when checking/unchecking boxes and then clicking update.
Making a few assumptions, this is where you might need to be looking:
First, if using checkboxes, you need to allow for multiple answers in the submission with HTML that looks something like this:
<input type="checkbox" name="sport[]" value="Football">Football</input>
Note the [] in the field name. This ensures PHP will parse the field as an array which can have multiple values.
Second, PHP needs to process the array if it exists and always validate user input. Learn about SQL Injection, Never take content direct from the request and use in an SQL query. It will save you a heap of bother one day.
$validSports=array('Football','Rugby','Cricket','Football');
// $_POST['sport'] will not be set if no boxes are checked
if(isset($_POST['sport']) &&
is_array($_POST['sport'])) {
// Remove previous entries from the table for this member.
$sql = "DELETE FROM sport_table WHERE MemberID='$username'";
$result = mysqli_query($conn, $sql);
foreach($_POST['sport'] as $sport) {
// See how we validate the user input here, there are other ways
if(in_array($sport, $validSports)) {
$sql = "INSERT INTO sport_table
SET sport='$sport[$i]', MemberID='$username'";
$result = mysqli_query($conn, $sql);
}
}
}
Thirdly, I've missed out the example code to keep it simple, but in practice, always check the $result value from mysqli_query and if $result === false then you need to handle the error - probably by writing to a log somewhere.
Finally, check the indexes on your sport_table to make it fast. An index on MemberID would be sufficient for the use cases above although a unique index on (MemberId,sport) may be more appropriate to mitigate against data duplication.

SELECT * FROM table WHERE column = ??? in PHP with unknown number of values for?

I'm trying to show some stuff from my database. There's this column called Genre which can be one out of five different values. Now I want to show only the data for the genres the user has selected by checking checkboxes. The user can choose more than one genre. I know how to show one, two or three genres by using WHERE... AND... but I don't know how many genres the user will select! This is what I use to show data when a checkbox 'actie' is checked:
if (isset($_GET["actie"]))
{
$stmt = $db->prepare("SELECT ProductID,Afbeelding,Product,Prijs,Beschrijving FROM Producten WHERE Genre='Actie' order by Product ASC");
}
How can I do this for multiple checkboxes so that it shows all data for when for instance 'actie' and 'sports' are checked? I'm confused since I don't know how many checkboxes a user will select, none, one, two, three, four or five.
Thanks in advance for any help!
You need to make sure that all your checkboxes have the same name property followed by []
i.e
<input type="checkbox" name="mycheckbox[]" value="gen1">
<input type="checkbox" name="mycheckbox[]" value="gen2">
<input type="checkbox" name="mycheckbox[]" value="gen3">
Then when you submit the form, all selected checkboxes with the same common name will be available as $_POST['mycheckbox'] or $_GET['mycheckbox']; and their corresponding values will be gen1,gen2 or gen3 depending on what you set in the value field.
$genres = $_POST['mycheckbox'];//creates an array of selected checkboxes values gen1,gen2,gen3
Then use MySQL WHERE IN clause. You will have to convert the array to a string of 'gen1,gen2,..' by using implode().
WHERE IN correct syntax: SELECT * FROM my_table WHERE Genre IN ('gen1','gen2','gen3',...)
if (isset($_GET["actie"]))
{
$genres_str = "'" .implode( "','" ,$genres ) ."'";//converts the array into 'gen1,gen2,gen3,..'
$stmt = $db->prepare("SELECT ProductID,Afbeelding,Product,Prijs,Beschrijving FROM Producten WHERE Genre IN (". $genres_str .") order by Product ASC");
//don't forget to execute the query
}
Hope this helps!

Let user create table based on existing table

After registering at my site, I want the user to select some elements from a table that already exists in my DB, and add them to "their" column in another table.
Say this is the existing table in a MySQL DB:
ID Item Color
1 Car Red
2 Apple Green
3 Trophy Gold
4 Suit Black
I would want the user to fill out a form where they:
Are presented with a dropdown list, to choose items from (based on the existing table).
When they have chosen up to x amount of items, they submit their "inventory" which..
Adds the selected items to their own column in a table that holds "user_inventory"
So this second table (user_inventory) should look something like this:
User_id item_id1 item_id2 item_id3
1 4 3 1
2 3 1 4
3 2 4 1
4 2 4 3
I don't expect you to write the code for me or anything, but I would be thrilled if you could answer these questions:
Is this possible?
Can you direct me to a similar type of thread or article that helps me write the code?
If you can not direct me anywhere, please help me in whichever way you see fit.
It is possible.
Firstly, read from the "existing" db table to create the select menu, example:
<select name="item">
<?php
$sql = "SELECT * FROM existing";
$query = mysqli_query($mysqli, $sql);
while ($result = mysqli_fetch_array($query)) {
$item = $result['item'];
echo "<option value='$item'>$item</option>";
}
?>
</select>
This will create the "Item" select menu, I'm sure you can figure out how to do the other menus by yourself, all that is left now is to submit and store into the new table which I assume you already know how to do.
if(isset($_POST['yoursubmitname'])) {
$item = $_POST['item'];
/// RUN THE INSERT COMMAND HERE //
}

Categories