0
class Lab_Group(models.Model):
    group = models.CharField(max_length=100, unique=True,)

class Lab(models.Model):
    laboratory = models.CharField(max_length=50, unique=True)
    group = models.ForeignKey(Lab_Group, on_delete=models.CASCADE)

class LabRequest(models.Model):
    ticket = models.ForeignKey(Ticket, on_delete=models.CASCADE)
    lab_test = models.ManyToManyField(Lab)

This is my view

def LabRequestToGenerateView(request, pk):
    doctor = Doctor.objects.filter(user_id=request.user.id, active='Yes')

    for doc in doctor:
        ticket = get_object_or_404(
            Ticket, pk=pk, has_cancelled=0, is_active=0, doctor=doc.id)
        lab_group = Lab_Group.objects.all()
        if request.POST.get('lab_test'):
            obj = LabRequest()
            obj.lab_test = request.POST.get('lab_test')
            obj.ticket = ticket
            obj.created_by = request.user.id
            obj.save()
            obj.save_m2m()
            messages.success(
                request, 'Lab Request has been sent successfully.')
            return redirect('/dashboard/lab-request')           
    context = {
        'title': 'Laboratory Request',
        'valueBtn': 'Save',
        'form': form,
        'rl': rl,
        'lab_group': lab_group

    }
    return render(request, 'dashboard/laboratory_request.html', context)

This is my template

<form action="." method="POST">
    {% csrf_token %}
    <div class="row">
        {% for lab_group in lab_group %}
        <div class="col-3 border">
            <label for="">{{ lab_group|upper }}</label>
            {% for lab in lab_group.lab_set.all %}
            {% if lab.active == 'Yes' %}
            <div class="form-check">
                <input type="checkbox" value="{{ lab.id }}" name="lab_test">
                {{ lab }}
            </div>
            {% endif %}
            {% endfor %}
        </div>
        {% endfor %}

    </div>
    <div class="form-group float-right">
        <button type="submit" name="submit" class="btn btn-success btn-sm" value=" {{ valueBtn }} "> <i
                class="fa fa-save"></i>
            Save</button>
    </div>
</form>

When I try to save it display this error:

Direct assignment to the forward side of a many-to-many set is prohibited. Use lab_test.set() instead.

I'm stuck this for a week, please feel free to ask me anything. thank you.

MD. Khairul Basar
  • 4,976
  • 14
  • 41
  • 59
Ahmed Ibrahim
  • 477
  • 1
  • 7
  • 18

1 Answers1

2

This answer solves your problem. Yet here is a solution.

def LabRequestToGenerateView(request, pk):
    doctors = Doctor.objects.filter(user_id=request.user.id, active='Yes')

    for doc in doctors:
        ticket = get_object_or_404(Ticket, pk=pk, has_cancelled=0, is_active=0, doctor=doc.id)
        lab_id = request.POST.get('lab_test')
        if lab_id is not None:
            obj = LabRequest.objects.create(ticket=ticket, created_by=request.user.id)
            lab = Lab.objects.get(id=lab_id)
            obj.lab_test.add(lab)
            obj.save()
            messages.success(request, 'Lab Request has been sent successfully.')
            return redirect('/dashboard/lab-request')

    lab_group = Lab_Group.objects.all()
    context = {
        'title': 'Laboratory Request',
        'valueBtn': 'Save',
        'form': form,
        'rl': rl,
        'lab_group': lab_group

    }
    return render(request, 'dashboard/laboratory_request.html', context)
MD. Khairul Basar
  • 4,976
  • 14
  • 41
  • 59
  • But this way is only getting 1 data, how about if I want more like 3 data checked and want to save to database like the old example? If I change to getlist('lab_test) it says this error: `int() argument must be a string, a bytes-like object or a number, not 'list'` If I change to like this ` lab = Lab.objects.filter(id__in=lab_id)` it says `int() argument must be a string, a bytes-like object or a number, not 'QuerySet'` – Ahmed Ibrahim Dec 10 '19 at 06:51
  • 1
    use `filter()` or add all and then save. Follow the linked question. – MD. Khairul Basar Dec 10 '19 at 06:53