장고:글쓰기: 두 판 사이의 차이
둘러보기로 이동
검색으로 이동
편집 요약 없음 |
|||
(사용자 3명의 중간 판 13개는 보이지 않습니다) | |||
1번째 줄: | 1번째 줄: | ||
{{장고}} | {{장고}} | ||
=글 보기= | |||
리스트에서 글을 클릭하면 글의 상세내용을 볼 수 있어야 한다. | 리스트에서 글을 클릭하면 글의 상세내용을 볼 수 있어야 한다. | ||
{| class="wikitable" | {| class="wikitable" | ||
163번째 줄: | 47번째 줄: | ||
<code><nowiki>{{object.subject}}</nowiki></code> 형태의 이름을 써서 변수를 사용할 수 있다. | <code><nowiki>{{object.subject}}</nowiki></code> 형태의 이름을 써서 변수를 사용할 수 있다. | ||
|} | |} | ||
==글 수정== | |||
==글 | |||
{| class="wikitable" | {| class="wikitable" | ||
! | ! | ||
172번째 줄: | 54번째 줄: | ||
|- | |- | ||
|template작성 | |template작성 | ||
| colspan="2" |위에서 만든 | | colspan="2" |위에서 만든 detail.html을 수정한다. 글 수정을 위한 링크를 추가.<syntaxhighlight lang="html"> | ||
<a href="{% url '앱이름: | {% if request.user == question.author %}<!--해당 작성자 일때만 수정버튼이 보이게끔!--> | ||
<a href="{% url '앱이름:modify' question.id%}" class="btn btn-primary">수정</a> | |||
{% endif %} | |||
</syntaxhighlight>제네릭 뷰를 쓰는 경우 넘겨주는 변수에 <code>pk=question.id</code> 형태로 넣어주어야 한다. | |||
</syntaxhighlight>< | |||
|- | |- | ||
|view 작성 | |view 작성 | ||
| | |글 작성과 동일하게 get처리와 post 처리를 한 함수에서 다룬다.<syntaxhighlight lang="python"> | ||
from .forms import QuestionForm#위에서 작성한 폼 불러오기 | |||
from .models import Question #모델을 불러온다. | |||
from django.contrib import messages#넌필드 오류를 반환하기 위한 것 | |||
def modify(request, question_id):#이름을 update로 해도 괜찮았을 듯하다. | |||
question = get_object_or_404(Question, pk=question_id) | |||
if request.user != question.author: | |||
messages.error(request, '수정권한이 없습니다') | |||
return redirect('앱이름:detail', question_id=question.id) | |||
if request.method == "POST": | |||
if request.method == | form = QuestionForm(request.POST, instance=question)#받은 내용을 객체에 담는다. | ||
form = QuestionForm(request.POST) # | if form.is_valid(): | ||
if form.is_valid(): | |||
question = form.save(commit=False) | question = form.save(commit=False) | ||
question.author = request.user | |||
question.modify_date = timezone.now() # 수정일시 자동 저장 | |||
question.author = request.user | |||
question. | |||
question.save() | question.save() | ||
return redirect(' | return redirect('앱이름:detail', question_id=question.id) | ||
else:# | else:#GET으로 요청된 경우. | ||
form = QuestionForm() | form = QuestionForm(instance=question)#해당 모델의 내용을 가져온다! | ||
context = {'form': form} | context = {'form': form} | ||
return render(request, 'create.html', context) | return render(request, 'create.html', context) | ||
</syntaxhighlight> | </syntaxhighlight>messages.error 가 있으면 넌필드 오류를 반환한다. | ||
|<syntaxhighlight lang="python"> | |<syntaxhighlight lang="python"> | ||
from django.views import generic | from django.views import generic | ||
from django.urls import reverse_lazy | from django.urls import reverse_lazy | ||
class | class update(generic.UpdateView): | ||
model = Question | model = Question | ||
feilds = ['필드명', '필드명',...]#작성할 필드명 지정 | feilds = ['필드명', '필드명',...]#작성할 필드명 지정 | ||
success_url=reverse_lazy('앱이름:detail')#작성에 성공한 경우 보낼 탬플릿. | success_url=reverse_lazy('앱이름:detail')#작성에 성공한 경우 보낼 탬플릿. | ||
template_name_suffix=' | template_name_suffix='_update' | ||
#사용하는 탬플릿 명을 '모델명_create.html'로 바꾼다는 의미. 접미사만 바꾼다. | #사용하는 탬플릿 명을 '모델명_create.html'로 바꾼다는 의미. 접미사만 바꾼다. | ||
#기본 탬플릿은 '모델명_form.html'로 나타난다. | #기본 탬플릿은 '모델명_form.html'로 나타난다. | ||
def form_valid(self, form): # 폼에 이상이 없으면 실행. | |||
temp = form.save(commit=False) # 임시 저장. 폼 외의 다른 내용을 조작하고 싶을 때 사용한다. | |||
조작 | |||
temp.save() # 최종 저장 | |||
return super().form_valid(form) | |||
</syntaxhighlight> | </syntaxhighlight> | ||
|- | |- | ||
|template작성 | |template작성 | ||
| colspan="2" | | | colspan="2" |글 작성과 동일한 탬플릿을 사용한다. 생성칸에 기존의 데이터만 form에 담아서 보내면 되니까!<syntaxhighlight lang="html"> | ||
{% extends '연장할탬플릿.html' %} | {% extends '연장할탬플릿.html' %} | ||
244번째 줄: | 116번째 줄: | ||
</form> | </form> | ||
{% endblock %} | {% endblock %} | ||
</syntaxhighlight> | </syntaxhighlight> | ||
|} | |} | ||
== | ==글 삭제(+에러메시지 띄우기)== | ||
{| class="wikitable" | {| class="wikitable" | ||
! | ! | ||
310번째 줄: | 126번째 줄: | ||
|- | |- | ||
|template작성 | |template작성 | ||
| colspan="2" |위에서 만든 detail.html을 수정한다. 글 수정을 위한 링크를 추가.<syntaxhighlight lang="html"> | +JQuery | ||
<a href="{% url '앱이름: | | colspan="2" |위에서 만든 detail.html을 수정한다. 글 수정을 위한 링크를 추가. | ||
</syntaxhighlight>제네릭 뷰를 쓰는 경우 넘겨주는 변수에 <code>pk=question.id</code> 형태로 넣어주어야 한다. | 글 수정과 동일하게, 편집자만 접근할 수 있게끔 if 태그 안에 넣어준다.<syntaxhighlight lang="html"> | ||
<a href="{% url '앱이름:delete' question.id%}" class="btn btn-primary">삭제</a> | |||
</syntaxhighlight>제네릭 뷰를 쓰는 경우, 넘겨주는 변수에 <code>pk=question.id</code> 형태로 넣어주어야 한다. | |||
이번엔 특수한 기능을 위해 JQuery도 함께 작성한다. 에러 메시지를 띄우기 위한 확인창을 구현하기 위해서. | |||
기존의 작성법과 조금 다르다. 다음과 같이 수정해주자.<syntaxhighlight lang="html"> | |||
<a href="#" data-uri="{% url '앱이름:delete' question.id %}" class="delete">삭제</a> | |||
<!-- data-uri 속성은 jQuery에서 $(this).data('uri') 형태로 받아가기 위해서 작성--> | |||
</ | |||
= | <script type='text/javascript'> | ||
{ | $(document).ready(function(){ | ||
$(".delete").on('click', function() { //delete를 포함하는 클래스가 있으면 작동한다. | |||
if(confirm("님, 정말로 지움??")) { | |||
location.href = $(this).data('uri'); | |||
} | |||
}); | |||
}); | |||
</script> | |||
< | <!-- 취소를 누르면 아무 일도 하지 않는다.--> | ||
</syntaxhighlight>JQuery가 포함되어있지 않으면 사용할 수 없다. | |||
|- | |- | ||
|view 작성 | |view 작성 | ||
| | |<syntaxhighlight lang="python"> | ||
from | from .models import Question #모델을 불러온다. | ||
@login_required(login_url='membership:login') | |||
def delete(request, question_id): | |||
question = get_object_or_404(Question, pk=question_id) | |||
if request.user != question.author: | |||
messages.error(request, '삭제권한이 없습니다') | |||
return redirect('앱이름:detail', question_id=question.id) | |||
question.delete() | |||
return redirect('앱이름:index') | |||
return | |||
</syntaxhighlight> | </syntaxhighlight> | ||
|<syntaxhighlight lang="python"> | |<syntaxhighlight lang="python"> | ||
411번째 줄: | 178번째 줄: | ||
|- | |- | ||
|template작성 | |template작성 | ||
|글 작성에서 사용한 탬플릿에서 버튼 이름만 바꾼다.<syntaxhighlight lang="html"> | |글 삭제 확인페이지.(JQuery를 이용한 경우엔 해당사항 없음.) | ||
글 작성에서 사용한 탬플릿에서 버튼 이름만 바꾼다. | |||
(post요청을 통해 들어온 경우에만 삭제하게끔 view 변형 필요.)<syntaxhighlight lang="html"> | |||
{% extends '연장할탬플릿.html' %} | {% extends '연장할탬플릿.html' %} | ||
422번째 줄: | 192번째 줄: | ||
{% endblock %} | {% endblock %} | ||
</syntaxhighlight> | </syntaxhighlight> | ||
|글 삭제 확인페이지로 이동한다.<syntaxhighlight lang="html"> | |글 삭제 확인페이지로 이동한다.(JQuery를 이용한 경우엔 해당사항 없음.)<syntaxhighlight lang="html"> | ||
{% extends '연장할탬플릿.html' %} | {% extends '연장할탬플릿.html' %} | ||
{% block %} | |||
{% block content%} | |||
<form action="" method='post'> | <form action="" method='post'> | ||
{% csrf_token %} | {% csrf_token %} |
2021년 6월 25일 (금) 14:26 기준 최신판
장고! 웹 프레임워크! 틀:장고
글 보기[편집 | 원본 편집]
리스트에서 글을 클릭하면 글의 상세내용을 볼 수 있어야 한다.
일반적으로 코드를 짤 경우 | 제네릭뷰(클래스형 뷰)를 쓰는 경우 | |
---|---|---|
view 작성 | 글 하나의 내용을 본다.
get_object_or_404를 쓴다. id에 해당하는 객체가 없으면 서버에서 에러를 일으키는데, 이는 서버에러로 잡힌다. 에러의 방향성을 명확히 지정해주기 위해. 404에러를 부르게끔!from django.shortcuts import render, get_object_or_404
from .models import * #모델을 불러온다.
def detail(request,question_id):#url에서 매핑된 question_id가 전달된다.
question=get_object_or_404(Question, pk=question_id)
context={'question':question}
return render(request, 'detail.html', context)
|
from django.views import generic
class DetailView(generic.DetailView):
model = Question
#템플릿 명이 명시적으로 지정되지 않은 경우에는 자동으로 모델명_detail.html을 템플릿명으로 사용
제네릭뷰에 경우, 변수명이 지정되어 있다. urls.py에서 <int:pk> 형태로 기입받는다. |
template작성 | 경로에 맞게 해주면 되는데, 위 뷰의 경우엔 /앱이름/template/detail.html 에 만들어준다.
만들어진 html파일의 body에 다음과 같이 넣어준다. 상황에 맞게 표를 만들든, 목차를 만들든 html을 짜면 될 터. {{변수.속성}} 형태로 필요한 데이터를 가져온다.<h1>{{ question.subject }}</h1>
<div>
{{ question.content }}
</div>
|
제네릭 뷰를 쓸 땐 탬플릿에 모델이 object라는 이름으로 전달된다.
|
글 수정[편집 | 원본 편집]
일반적으로 코드를 짤 경우 | 제네릭뷰(클래스형 뷰)를 쓰는 경우 | |
---|---|---|
template작성 | 위에서 만든 detail.html을 수정한다. 글 수정을 위한 링크를 추가.{% if request.user == question.author %}<!--해당 작성자 일때만 수정버튼이 보이게끔!-->
<a href="{% url '앱이름:modify' question.id%}" class="btn btn-primary">수정</a>
{% endif %}
pk=question.id 형태로 넣어주어야 한다.
| |
view 작성 | 글 작성과 동일하게 get처리와 post 처리를 한 함수에서 다룬다.from .forms import QuestionForm#위에서 작성한 폼 불러오기
from .models import Question #모델을 불러온다.
from django.contrib import messages#넌필드 오류를 반환하기 위한 것
def modify(request, question_id):#이름을 update로 해도 괜찮았을 듯하다.
question = get_object_or_404(Question, pk=question_id)
if request.user != question.author:
messages.error(request, '수정권한이 없습니다')
return redirect('앱이름:detail', question_id=question.id)
if request.method == "POST":
form = QuestionForm(request.POST, instance=question)#받은 내용을 객체에 담는다.
if form.is_valid():
question = form.save(commit=False)
question.author = request.user
question.modify_date = timezone.now() # 수정일시 자동 저장
question.save()
return redirect('앱이름:detail', question_id=question.id)
else:#GET으로 요청된 경우.
form = QuestionForm(instance=question)#해당 모델의 내용을 가져온다!
context = {'form': form}
return render(request, 'create.html', context)
|
from django.views import generic
from django.urls import reverse_lazy
class update(generic.UpdateView):
model = Question
feilds = ['필드명', '필드명',...]#작성할 필드명 지정
success_url=reverse_lazy('앱이름:detail')#작성에 성공한 경우 보낼 탬플릿.
template_name_suffix='_update'
#사용하는 탬플릿 명을 '모델명_create.html'로 바꾼다는 의미. 접미사만 바꾼다.
#기본 탬플릿은 '모델명_form.html'로 나타난다.
def form_valid(self, form): # 폼에 이상이 없으면 실행.
temp = form.save(commit=False) # 임시 저장. 폼 외의 다른 내용을 조작하고 싶을 때 사용한다.
조작
temp.save() # 최종 저장
return super().form_valid(form)
|
template작성 | 글 작성과 동일한 탬플릿을 사용한다. 생성칸에 기존의 데이터만 form에 담아서 보내면 되니까!{% extends '연장할탬플릿.html' %}
{% block content %}
<form method="post">
{% csrf_token %}
{{ form.as_p }}
<input type="submit" value='나타낼 텍스트'>
</form>
{% endblock %}
|
글 삭제(+에러메시지 띄우기)[편집 | 원본 편집]
일반적으로 코드를 짤 경우 | 제네릭뷰(클래스형 뷰)를 쓰는 경우 | |
---|---|---|
template작성
+JQuery |
위에서 만든 detail.html을 수정한다. 글 수정을 위한 링크를 추가.
글 수정과 동일하게, 편집자만 접근할 수 있게끔 if 태그 안에 넣어준다.<a href="{% url '앱이름:delete' question.id%}" class="btn btn-primary">삭제</a>
pk=question.id 형태로 넣어주어야 한다.
<a href="#" data-uri="{% url '앱이름:delete' question.id %}" class="delete">삭제</a>
<!-- data-uri 속성은 jQuery에서 $(this).data('uri') 형태로 받아가기 위해서 작성-->
<script type='text/javascript'>
$(document).ready(function(){
$(".delete").on('click', function() { //delete를 포함하는 클래스가 있으면 작동한다.
if(confirm("님, 정말로 지움??")) {
location.href = $(this).data('uri');
}
});
});
</script>
<!-- 취소를 누르면 아무 일도 하지 않는다.-->
| |
view 작성 | from .models import Question #모델을 불러온다.
@login_required(login_url='membership:login')
def delete(request, question_id):
question = get_object_or_404(Question, pk=question_id)
if request.user != question.author:
messages.error(request, '삭제권한이 없습니다')
return redirect('앱이름:detail', question_id=question.id)
question.delete()
return redirect('앱이름:index')
|
from django.views import generic
from django.urls import reverse_lazy
class create(generic.DeleteView):
model = Question
success_url=reverse_lazy('앱이름:list')#작성에 성공한 경우 보낼 탬플릿.
template_name_suffix='_delete'
#사용하는 탬플릿 명을 '모델명_delete.html'로 바꾼다는 의미. 접미사만 바꾼다.
#기본 탬플릿은 '모델명_confirm_delete.html'로 나타난다.
|
template작성 | 글 삭제 확인페이지.(JQuery를 이용한 경우엔 해당사항 없음.)
글 작성에서 사용한 탬플릿에서 버튼 이름만 바꾼다. (post요청을 통해 들어온 경우에만 삭제하게끔 view 변형 필요.){% extends '연장할탬플릿.html' %}
{% block content %}
<form method="post">
{% csrf_token %}
{{ form.as_p }}
<input type="submit" value='나타낼 텍스트'>
</form>
{% endblock %}
|
글 삭제 확인페이지로 이동한다.(JQuery를 이용한 경우엔 해당사항 없음.){% extends '연장할탬플릿.html' %}
{% block content%}
<form action="" method='post'>
{% csrf_token %}
<div class="alert alert-danget">나타낼텍스트 {{나타낼변수}} 자유롭게 사용.</div>
<input type="submit" value="Delete" class="btn btn-danget"
</form>
{% endblock %}
|
tip[편집 | 원본 편집]
뷰가 많아지면 /앱/views/posting_view.py 따위의 이름으로 뷰를 나누어 작성하는 것도 전략이다.