Merge pull request #4993 from KanchanChauhan/collaborativeprojectmanagement
[WIP] Collaborative Project Management first commit
This commit is contained in:
commit
bd8d288255
@ -921,7 +921,7 @@
|
|||||||
"issingle": 0,
|
"issingle": 0,
|
||||||
"istable": 0,
|
"istable": 0,
|
||||||
"max_attachments": 4,
|
"max_attachments": 4,
|
||||||
"modified": "2016-03-10 05:10:21.779365",
|
"modified": "2016-03-15 05:10:21.779365",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Projects",
|
"module": "Projects",
|
||||||
"name": "Project",
|
"name": "Project",
|
||||||
|
@ -124,6 +124,18 @@ class Project(Document):
|
|||||||
|
|
||||||
self.total_purchase_cost = total_purchase_cost and total_purchase_cost[0][0] or 0
|
self.total_purchase_cost = total_purchase_cost and total_purchase_cost[0][0] or 0
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def get_project_list(doctype, txt, filters, limit_start, limit_page_length=20):
|
||||||
|
from frappe.templates.pages.list import get_list
|
||||||
|
return get_list(doctype, txt, filters, limit_start, limit_page_length, ignore_permissions=True)
|
||||||
|
|
||||||
|
def get_list_context(context=None):
|
||||||
|
return {
|
||||||
|
"title": _("My Projects"),
|
||||||
|
"get_list": get_project_list,
|
||||||
|
"row_template": "templates/includes/project_row.html"
|
||||||
|
}
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_cost_center_name(project):
|
def get_cost_center_name(project):
|
||||||
return frappe.db.get_value("Project", project, "cost_center")
|
return frappe.db.get_value("Project", project, "cost_center")
|
||||||
|
@ -437,3 +437,6 @@ def get_address_territory(address_name):
|
|||||||
break
|
break
|
||||||
|
|
||||||
return territory
|
return territory
|
||||||
|
|
||||||
|
def show_terms(doc):
|
||||||
|
return doc.tc_name
|
||||||
|
@ -30,9 +30,10 @@ def update_website_context(context):
|
|||||||
|
|
||||||
def update_my_account_context(context):
|
def update_my_account_context(context):
|
||||||
context["my_account_list"].extend([
|
context["my_account_list"].extend([
|
||||||
|
{"label": _("Projects"), "url": "project"},
|
||||||
{"label": _("Orders"), "url": "orders"},
|
{"label": _("Orders"), "url": "orders"},
|
||||||
{"label": _("Invoices"), "url": "invoices"},
|
{"label": _("Invoices"), "url": "invoices"},
|
||||||
{"label": _("Shipments"), "url": "shipments"},
|
{"label": _("Shipments"), "url": "shipments"},
|
||||||
{"label": _("Issues"), "url": "issues"},
|
{"label": _("Issues"), "url": "issues"},
|
||||||
{"label": _("Addresses"), "url": "addresses"},
|
{"label": _("Addresses"), "url": "addresses"}
|
||||||
])
|
])
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Subject",
|
"label": "Subject",
|
||||||
@ -40,6 +41,7 @@
|
|||||||
"fieldtype": "Select",
|
"fieldtype": "Select",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Series",
|
"label": "Series",
|
||||||
@ -64,6 +66,7 @@
|
|||||||
"fieldtype": "Data",
|
"fieldtype": "Data",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 1,
|
"in_filter": 1,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Subject",
|
"label": "Subject",
|
||||||
@ -87,6 +90,7 @@
|
|||||||
"fieldtype": "Column Break",
|
"fieldtype": "Column Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -110,6 +114,7 @@
|
|||||||
"fieldtype": "Select",
|
"fieldtype": "Select",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"label": "Status",
|
"label": "Status",
|
||||||
@ -137,6 +142,7 @@
|
|||||||
"fieldtype": "Data",
|
"fieldtype": "Data",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 1,
|
"in_filter": 1,
|
||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"label": "Raised By (Email)",
|
"label": "Raised By (Email)",
|
||||||
@ -163,6 +169,7 @@
|
|||||||
"fieldtype": "Fold",
|
"fieldtype": "Fold",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -185,6 +192,7 @@
|
|||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -209,6 +217,7 @@
|
|||||||
"fieldtype": "Text",
|
"fieldtype": "Text",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Description",
|
"label": "Description",
|
||||||
@ -234,6 +243,7 @@
|
|||||||
"fieldtype": "Column Break",
|
"fieldtype": "Column Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -258,6 +268,7 @@
|
|||||||
"fieldtype": "Datetime",
|
"fieldtype": "Datetime",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Resolution Date",
|
"label": "Resolution Date",
|
||||||
@ -283,6 +294,7 @@
|
|||||||
"fieldtype": "Datetime",
|
"fieldtype": "Datetime",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "First Responded On",
|
"label": "First Responded On",
|
||||||
@ -306,6 +318,7 @@
|
|||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "",
|
"label": "",
|
||||||
@ -330,6 +343,7 @@
|
|||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Lead",
|
"label": "Lead",
|
||||||
@ -354,6 +368,7 @@
|
|||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Contact",
|
"label": "Contact",
|
||||||
@ -378,6 +393,7 @@
|
|||||||
"fieldtype": "Column Break",
|
"fieldtype": "Column Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -401,6 +417,7 @@
|
|||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 1,
|
"in_filter": 1,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Customer",
|
"label": "Customer",
|
||||||
@ -427,6 +444,7 @@
|
|||||||
"fieldtype": "Data",
|
"fieldtype": "Data",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 1,
|
"in_filter": 1,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Customer Name",
|
"label": "Customer Name",
|
||||||
@ -452,6 +470,7 @@
|
|||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -476,6 +495,7 @@
|
|||||||
"fieldtype": "Small Text",
|
"fieldtype": "Small Text",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Resolution Details",
|
"label": "Resolution Details",
|
||||||
@ -502,6 +522,7 @@
|
|||||||
"fieldtype": "Column Break",
|
"fieldtype": "Column Break",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"length": 0,
|
"length": 0,
|
||||||
@ -526,6 +547,7 @@
|
|||||||
"fieldtype": "Date",
|
"fieldtype": "Date",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Opening Date",
|
"label": "Opening Date",
|
||||||
@ -551,6 +573,7 @@
|
|||||||
"fieldtype": "Time",
|
"fieldtype": "Time",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Opening Time",
|
"label": "Opening Time",
|
||||||
@ -576,6 +599,7 @@
|
|||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Company",
|
"label": "Company",
|
||||||
@ -600,6 +624,7 @@
|
|||||||
"fieldtype": "Data",
|
"fieldtype": "Data",
|
||||||
"hidden": 1,
|
"hidden": 1,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Content Type",
|
"label": "Content Type",
|
||||||
@ -623,6 +648,7 @@
|
|||||||
"fieldtype": "Attach",
|
"fieldtype": "Attach",
|
||||||
"hidden": 1,
|
"hidden": 1,
|
||||||
"ignore_user_permissions": 0,
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
"in_filter": 0,
|
"in_filter": 0,
|
||||||
"in_list_view": 0,
|
"in_list_view": 0,
|
||||||
"label": "Attachment",
|
"label": "Attachment",
|
||||||
@ -638,6 +664,32 @@
|
|||||||
"search_index": 0,
|
"search_index": 0,
|
||||||
"set_only_once": 0,
|
"set_only_once": 0,
|
||||||
"unique": 0
|
"unique": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"fieldname": "project",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"hidden": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_filter": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"label": "Project",
|
||||||
|
"length": 0,
|
||||||
|
"no_copy": 0,
|
||||||
|
"options": "Project",
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"set_only_once": 0,
|
||||||
|
"unique": 0
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"hide_heading": 0,
|
"hide_heading": 0,
|
||||||
@ -650,7 +702,7 @@
|
|||||||
"issingle": 0,
|
"issingle": 0,
|
||||||
"istable": 0,
|
"istable": 0,
|
||||||
"max_attachments": 0,
|
"max_attachments": 0,
|
||||||
"modified": "2016-02-03 01:08:48.180242",
|
"modified": "2016-03-09 18:31:10.404549",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Support",
|
"module": "Support",
|
||||||
"name": "Issue",
|
"name": "Issue",
|
||||||
@ -661,6 +713,26 @@
|
|||||||
"apply_user_permissions": 1,
|
"apply_user_permissions": 1,
|
||||||
"cancel": 0,
|
"cancel": 0,
|
||||||
"create": 1,
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"if_owner": 0,
|
||||||
|
"import": 0,
|
||||||
|
"permlevel": 0,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "Projects User",
|
||||||
|
"set_user_permissions": 0,
|
||||||
|
"share": 1,
|
||||||
|
"submit": 0,
|
||||||
|
"write": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"amend": 0,
|
||||||
|
"apply_user_permissions": 0,
|
||||||
|
"cancel": 0,
|
||||||
|
"create": 1,
|
||||||
"delete": 0,
|
"delete": 0,
|
||||||
"email": 1,
|
"email": 1,
|
||||||
"export": 0,
|
"export": 0,
|
||||||
@ -670,6 +742,8 @@
|
|||||||
"print": 1,
|
"print": 1,
|
||||||
"read": 1,
|
"read": 1,
|
||||||
"report": 1,
|
"report": 1,
|
||||||
|
"restrict": 0,
|
||||||
|
"restricted": 0,
|
||||||
"role": "Customer",
|
"role": "Customer",
|
||||||
"set_user_permissions": 0,
|
"set_user_permissions": 0,
|
||||||
"share": 1,
|
"share": 1,
|
||||||
@ -690,6 +764,8 @@
|
|||||||
"print": 1,
|
"print": 1,
|
||||||
"read": 1,
|
"read": 1,
|
||||||
"report": 1,
|
"report": 1,
|
||||||
|
"restrict": 0,
|
||||||
|
"restricted": 0,
|
||||||
"role": "Support Team",
|
"role": "Support Team",
|
||||||
"set_user_permissions": 0,
|
"set_user_permissions": 0,
|
||||||
"share": 1,
|
"share": 1,
|
||||||
|
@ -23,3 +23,10 @@
|
|||||||
margin-top: 80px;
|
margin-top: 80px;
|
||||||
margin-bottom: 60px;
|
margin-bottom: 60px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.cart-link {
|
||||||
|
margin-top: 40px;
|
||||||
|
text-align: right;
|
||||||
|
|
||||||
|
|
||||||
|
}
|
@ -147,3 +147,8 @@ $(document).ready(function() {
|
|||||||
$(".cart-icon").hide();
|
$(".cart-icon").hide();
|
||||||
shopping_cart.bind_events();
|
shopping_cart.bind_events();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function show_terms() {
|
||||||
|
var html = $(".cart-terms").html();
|
||||||
|
frappe.msgprint(html);
|
||||||
|
}
|
||||||
|
29
erpnext/templates/includes/project_row.html
Normal file
29
erpnext/templates/includes/project_row.html
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
{% if doc.status=="Open" %}
|
||||||
|
<div class="web-list-item">
|
||||||
|
<a class="no-decoration" href="/projects?project={{ doc.name }}">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-xs-4">
|
||||||
|
|
||||||
|
{{ doc.name }}
|
||||||
|
</div>
|
||||||
|
<div class="col-xs-4">
|
||||||
|
{% if doc.percent_complete %}
|
||||||
|
<div class="progress" style="margin-bottom: 0!important;">
|
||||||
|
<div class="progress-bar progress-bar-warning" role="progressbar"
|
||||||
|
aria-valuenow="{{ doc.percent_complete|round|int }}"
|
||||||
|
aria-valuemin="0" aria-valuemax="100" style="width:{{ doc.percent_complete|round|int }}%;">
|
||||||
|
{{ doc.percent_complete|round|int }}% Complete
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<span class="indicator {{ "red" if doc.status=="Open" else "darkgrey" }}">
|
||||||
|
{{ doc.status }}</span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<div class="col-xs-4 text-right small text-muted">
|
||||||
|
{{ frappe.utils.pretty_date(doc.modified) }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
20
erpnext/templates/includes/project_search_box.html
Normal file
20
erpnext/templates/includes/project_search_box.html
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
<div class="project-search">
|
||||||
|
<input type="text" id="project-search"
|
||||||
|
placeholder="Search..."><span style="position:relative;
|
||||||
|
left:-19px;"> <a href="/projects?project={{doc.name}}" class="octicon octicon-x text-extra-muted" title="Clear" ></a> </span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
frappe.ready(function() {
|
||||||
|
var thread = null;
|
||||||
|
function findResult(t) {
|
||||||
|
window.location.href="/projects?project={{doc.name}}&q=" + t;
|
||||||
|
}
|
||||||
|
|
||||||
|
$("#project-search").keyup(function() {
|
||||||
|
clearTimeout(thread);
|
||||||
|
var $this = $(this); thread = setTimeout(function(){findResult($this.val())}, 1000);
|
||||||
|
});
|
||||||
|
$(".form-search").on("submit", function() { return false; });
|
||||||
|
});
|
||||||
|
</script>
|
144
erpnext/templates/includes/projects.css
Normal file
144
erpnext/templates/includes/projects.css
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
/* CSS used here will be applied after bootstrap.css */
|
||||||
|
|
||||||
|
.underline {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-container .indicator {
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-container .project-item {
|
||||||
|
padding-top: 5px;
|
||||||
|
padding-bottom: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.row-header {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
padding-bottom: 8px;
|
||||||
|
border-bottom: 2px solid #d1d8dd;
|
||||||
|
margin: 0!important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content_display{
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.task-btn, .issue-btn, .timelog-btn{
|
||||||
|
padding: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered {
|
||||||
|
position: relative;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered:before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
display: block;
|
||||||
|
width: 3px;
|
||||||
|
background: #f5f5f6;
|
||||||
|
/*left: 50%;*/
|
||||||
|
top: 0px;
|
||||||
|
bottom: 0px;
|
||||||
|
margin-left: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry {
|
||||||
|
position: relative;
|
||||||
|
/*width: 50%;
|
||||||
|
float: right;*/
|
||||||
|
margin-top: 5px;
|
||||||
|
margin-left: 10px;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
clear: both;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner {
|
||||||
|
position: relative;
|
||||||
|
margin-left: -3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon {
|
||||||
|
margin-top:10px;
|
||||||
|
background: #fff;
|
||||||
|
color: #737881;
|
||||||
|
display: block;
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
-webkit-background-clip: padding-box;
|
||||||
|
-moz-background-clip: padding;
|
||||||
|
background-clip: padding-box;
|
||||||
|
-webkit-border-radius: 20px;
|
||||||
|
-moz-border-radius: 20px;
|
||||||
|
border-radius: 20px;
|
||||||
|
text-align: center;
|
||||||
|
-moz-box-shadow: 0 0 0 2px #f5f5f6;
|
||||||
|
-webkit-box-shadow: 0 0 0 2px #f5f5f6;
|
||||||
|
box-shadow: 0 0 0 2px #f5f5f6;
|
||||||
|
line-height: 30px;
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon.bg-primary {
|
||||||
|
background-color: #303641;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon.bg-secondary {
|
||||||
|
background-color: #ee4749;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon.bg-success {
|
||||||
|
background-color: #98d85b;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon.bg-info {
|
||||||
|
background-color: #21a9e1;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon.bg-warning {
|
||||||
|
background-color: #ffa00a;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-icon.bg-danger {
|
||||||
|
background-color: #ff5858;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-label {
|
||||||
|
position: relative;
|
||||||
|
background: #f5f5f6;
|
||||||
|
padding: 1em;
|
||||||
|
margin-left: 40px;
|
||||||
|
-webkit-background-clip: padding-box;
|
||||||
|
-moz-background-clip: padding;
|
||||||
|
background-clip: padding-box;
|
||||||
|
-webkit-border-radius: 3px;
|
||||||
|
-moz-border-radius: 3px;
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-centered .timeline-entry .timeline-entry-inner .timeline-label:after {
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
border-style: solid;
|
||||||
|
border-width: 9px 9px 9px 0;
|
||||||
|
border-color: transparent #f5f5f6 transparent transparent;
|
||||||
|
left: 0;
|
||||||
|
top: 10px;
|
||||||
|
margin-left: -9px;
|
||||||
|
}
|
31
erpnext/templates/includes/projects/project_issues.html
Normal file
31
erpnext/templates/includes/projects/project_issues.html
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{% for issue in doc.issues %}
|
||||||
|
<div class='issue'>
|
||||||
|
<div class='row project-item'>
|
||||||
|
<div class='col-xs-9'>
|
||||||
|
<a class="no-decoration" href="/issues-view?name={{ issue.name}}">
|
||||||
|
<span class="indicator {{ "red" if issue.status=="Open" else "green" }}">
|
||||||
|
{% if issue.status == "Closed" %}
|
||||||
|
{{ issue.subject }} resolved {{ frappe.utils.pretty_date(issue.resolution_date) }}
|
||||||
|
{% else %}
|
||||||
|
{{ issue.subject }} raised on {{ issue.opening_date }}
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class='col-xs-3'>
|
||||||
|
<div class='pull-right'>
|
||||||
|
{% if issue.todo %}
|
||||||
|
<span class="avatar avatar-small" title="{{ issue.todo.owner }}"> <img src="{{ issue.todo.user_image }}"></span>
|
||||||
|
{% else %}
|
||||||
|
<span class="avatar avatar-small avatar-empty"></span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<div class='pull-right' style='padding-right:10px;'>
|
||||||
|
{% if issue.status != "Closed" %}
|
||||||
|
<span class="text-extra-muted "> <i class="octicon octicon-x issue-x" title="Close" id = "{{issue.name}}"></i> </span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
30
erpnext/templates/includes/projects/project_tasks.html
Normal file
30
erpnext/templates/includes/projects/project_tasks.html
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
{% for task in doc.tasks %}
|
||||||
|
<div class='task'>
|
||||||
|
<div class='row project-item'>
|
||||||
|
<div class='col-xs-9'>
|
||||||
|
<a class="no-decoration" href="/tasks-view?name={{ task.name}}">
|
||||||
|
<span class="indicator {{ "orange" if task.status=="Open" else "green" }}">
|
||||||
|
{% if task.status == "Closed" %}
|
||||||
|
{{ task.subject }} completed on {{ task.closing_date }}
|
||||||
|
{% else %}
|
||||||
|
{{ task.subject }}
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class='col-xs-3'>
|
||||||
|
<div class='pull-right'>
|
||||||
|
{% if task.todo %} <span class="avatar avatar-small" title="{{ task.todo.owner }}"> <img src="{{ task.todo.user_image }}"></span>
|
||||||
|
{% else %}
|
||||||
|
<span class="avatar avatar-small avatar-empty"></span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<div class='pull-right' style='padding-right:10px;'>
|
||||||
|
{% if task.status != "Closed" %}
|
||||||
|
<span class="text-extra-muted "> <i class="octicon octicon-x task-x" title="Close" id = "{{task.name}}"></i> </span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
20
erpnext/templates/includes/projects/project_timelogs.html
Normal file
20
erpnext/templates/includes/projects/project_timelogs.html
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{% for timelog in doc.timelogs %}
|
||||||
|
<div class='timelog'>
|
||||||
|
<div class='row project-item'>
|
||||||
|
<div class='col-xs-9'>
|
||||||
|
<a class="no-decoration" href="/timelog_info?timelog={{ timelog.name}}">
|
||||||
|
<span class="indicator {{ "green" if timelog.status=="Draft" else "blue" if timelog.status=="Submitted" else "darkgrey"}}">
|
||||||
|
|
||||||
|
{{ timelog.title }}: From {{ frappe.format_date(timelog.from_time) }} to {{ frappe.format_date(timelog.to_time) }}
|
||||||
|
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class='col-xs-3'>
|
||||||
|
<div class='pull-right'>
|
||||||
|
<span class="avatar avatar-small" title="{{ timelog.modified_by }}"> <img src="{{ timelog.user_image }}"></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
17
erpnext/templates/includes/projects/timeline.html
Normal file
17
erpnext/templates/includes/projects/timeline.html
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
{% for timeline in doc.timelines %}
|
||||||
|
<article class="timeline-entry">
|
||||||
|
<div class="timeline-entry-inner">
|
||||||
|
<div class="timeline-icon {{ "bg-danger" if ((timeline.reference_name).startswith('ISS') and (timeline.subject).startswith('Open')) else "bg-warning" if ((timeline.reference_name).startswith('TASK') and (timeline.subject).startswith('Open')) else "bg-success" if (timeline.subject).startswith('Closed') else "bg-info"}}">
|
||||||
|
</div>
|
||||||
|
<div class="timeline-label">
|
||||||
|
<span class="avatar avatar-small" title="{{ timeline.modified_by }}"> <img src="{{ timeline.user_image }}"></span>
|
||||||
|
<span class='indicator'>
|
||||||
|
{{timeline.reference_name}} {{timeline.subject }}
|
||||||
|
</span>
|
||||||
|
<span class='indicator pull-right'>
|
||||||
|
{{ frappe.utils.pretty_date(timeline.creation) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
{% endfor %}
|
@ -61,6 +61,15 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="cart-totals">
|
<div id="cart-totals">
|
||||||
</div>
|
</div>
|
||||||
|
{% if doc.tc_name %}
|
||||||
|
<div class="cart-terms" style="display: none;" title={{doc.tc_name}}>
|
||||||
|
{{doc.tc_name}}
|
||||||
|
{{doc.terms}}
|
||||||
|
</div>
|
||||||
|
<div class="cart-link">
|
||||||
|
<a href="#" onclick="show_terms();return false;">*Terms and Conditions</a>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
<div class="cart-addresses">
|
<div class="cart-addresses">
|
||||||
{% include "templates/includes/cart/cart_address.html" %}
|
{% include "templates/includes/cart/cart_address.html" %}
|
||||||
</div>
|
</div>
|
||||||
|
2
erpnext/templates/pages/cart_terms.html
Normal file
2
erpnext/templates/pages/cart_terms.html
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
|
||||||
|
<div>{{doc.terms}}</div>
|
109
erpnext/templates/pages/projects.html
Normal file
109
erpnext/templates/pages/projects.html
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
{% extends "templates/web.html" %}
|
||||||
|
|
||||||
|
{% block title %}{{ doc.project_name }}{% endblock %}
|
||||||
|
|
||||||
|
{% block breadcrumbs %}
|
||||||
|
<div class="page-breadcrumbs" data-html-block="breadcrumbs">
|
||||||
|
<ul class="breadcrumb">
|
||||||
|
<li>
|
||||||
|
<span class="icon icon-angle-left"></span>
|
||||||
|
<a href="/project">Projects</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block header %}
|
||||||
|
<h1 class= "title">
|
||||||
|
{{ doc.project_name }}
|
||||||
|
</h1>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block style %}
|
||||||
|
<style>
|
||||||
|
{% include "templates/includes/projects.css" %}
|
||||||
|
</style>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
{% block page_content %}
|
||||||
|
|
||||||
|
{% include 'templates/includes/project_search_box.html' %}
|
||||||
|
|
||||||
|
{% if frappe.form_dict.q %}
|
||||||
|
<p class="text-muted"> <a href="/projects?project={{doc.name}}" class="text-muted">
|
||||||
|
Filtered by "{{ frappe.form_dict.q }}" Clear</a></p>
|
||||||
|
{% else %}
|
||||||
|
<h3>{{ _("Activity Feed") }}</h3>
|
||||||
|
<div class='project-timelines timeline-centered'>
|
||||||
|
{% include "erpnext/templates/includes/projects/timeline.html" %}
|
||||||
|
</div>
|
||||||
|
{% if doc.timelines|length > 9 %}
|
||||||
|
<p><a class='more-timelines small underline'>{{ _("More") }}</a><p>
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if doc.tasks %}
|
||||||
|
<div class='project-tasks-section'>
|
||||||
|
<div>
|
||||||
|
<a class="btn btn-xs btn-primary pull-right"
|
||||||
|
href='/tasks?new=1&project={{ doc.project_name }}'>New</a>
|
||||||
|
<h3>{{ _("Tasks") }}</h3>
|
||||||
|
</div>
|
||||||
|
<div class="btn-group btn-toggle">
|
||||||
|
<button class="btn btn-xs btn-open-tasks" style="float:left;">Open</button>
|
||||||
|
<button class="btn btn-xs btn-closed-tasks">Close</button>
|
||||||
|
</div>
|
||||||
|
<div class='project-tasks'>
|
||||||
|
{% include "erpnext/templates/includes/projects/project_tasks.html" %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if doc.tasks|length > 4 %}
|
||||||
|
<p><a id= 'more-tasks' class='more-tasks small underline'>{{ _("More") }}</a><p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<p class="text-muted">No tasks</p>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if doc.issues %}
|
||||||
|
<div class='project-issues-section'>
|
||||||
|
<div>
|
||||||
|
<a class="btn btn-xs btn-primary pull-right"
|
||||||
|
href='/issues?new=1&project={{ doc.project_name }}'>New</a>
|
||||||
|
<h3>{{ _("Issues") }}</h3>
|
||||||
|
</div>
|
||||||
|
<div class="btn-group btn-toggle">
|
||||||
|
<button class="btn btn-xs btn-open-issues" style="float:left;">Open</button>
|
||||||
|
<button class="btn btn-xs btn-closed-issues">Close</button>
|
||||||
|
</div>
|
||||||
|
<div class='project-issues'>
|
||||||
|
{% include "erpnext/templates/includes/projects/project_issues.html" %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if doc.issues|length > 4 %}
|
||||||
|
<p><a id='more-issues' class='more-issues small underline'>{{ _("More") }}</a><p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<p class="text-muted">No Issues</p>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if doc.timelogs %}
|
||||||
|
<h3>{{ _("Time Logs") }}</h3>
|
||||||
|
<div class='project-timelogs'>
|
||||||
|
{% include "erpnext/templates/includes/projects/project_timelogs.html" %}
|
||||||
|
</div>
|
||||||
|
{% if doc.timelogs|length > 1 %}
|
||||||
|
<p><a class='more-timelogs small underline'>{{ _("More") }}</a><p>
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
<p class="text-muted">No time logs</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{% include "erpnext/templates/pages/projects.js" %}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{% endblock %}
|
120
erpnext/templates/pages/projects.js
Normal file
120
erpnext/templates/pages/projects.js
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
frappe.ready(function() {
|
||||||
|
$( window ).load(function() {
|
||||||
|
$(".btn-open-tasks").click();
|
||||||
|
$(".btn-open-issues").click();
|
||||||
|
});
|
||||||
|
|
||||||
|
$('.btn-closed-tasks').click(function() {
|
||||||
|
reload_items('closed','tasks');
|
||||||
|
});
|
||||||
|
|
||||||
|
$('.btn-open-tasks').click(function() {
|
||||||
|
reload_items('open','tasks');
|
||||||
|
});
|
||||||
|
|
||||||
|
$('.btn-closed-issues').click(function() {
|
||||||
|
reload_items('closed','issues');
|
||||||
|
});
|
||||||
|
|
||||||
|
$('.btn-open-issues').click(function() {
|
||||||
|
reload_items('open','issues');
|
||||||
|
});
|
||||||
|
|
||||||
|
var start = 10;
|
||||||
|
$(".more-tasks").click(function() {
|
||||||
|
more_items('tasks', true);
|
||||||
|
});
|
||||||
|
|
||||||
|
$(".more-issues").click(function() {
|
||||||
|
more_items('issues', true);
|
||||||
|
});
|
||||||
|
|
||||||
|
$(".more-timelogs").click(function() {
|
||||||
|
more_items('timelogs', false);
|
||||||
|
});
|
||||||
|
|
||||||
|
$(".more-timelines").click(function() {
|
||||||
|
more_items('timelines', false);
|
||||||
|
});
|
||||||
|
|
||||||
|
$( ".project-tasks" ).on('click', '.task-x', function() {
|
||||||
|
var item_name = $(this).attr('id');
|
||||||
|
close_item('task', item_name);
|
||||||
|
});
|
||||||
|
|
||||||
|
$( ".project-issues" ).on('click', '.issue-x', function() {
|
||||||
|
var item_name = $(this).attr('id');
|
||||||
|
close_item('issue', item_name);
|
||||||
|
});
|
||||||
|
|
||||||
|
var reload_items = function(item_status, item) {
|
||||||
|
$.ajax({
|
||||||
|
method: "GET",
|
||||||
|
url: "/",
|
||||||
|
dataType: "json",
|
||||||
|
data: {
|
||||||
|
cmd: "erpnext.templates.pages.projects.get_"+ item +"_html",
|
||||||
|
project: '{{ doc.name }}',
|
||||||
|
item_status: item_status,
|
||||||
|
},
|
||||||
|
dataType: "json",
|
||||||
|
success: function(data) {
|
||||||
|
$('.project-'+ item).html(data.message);
|
||||||
|
|
||||||
|
$('.project-'+ item +'-section .btn-group .btn-primary').removeClass('btn-primary');
|
||||||
|
$('.btn-'+ item_status +'-'+ item).addClass( "btn-primary" );
|
||||||
|
$(".more-"+ item).toggle(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
var more_items = function(item, item_status){
|
||||||
|
if(item_status)
|
||||||
|
{
|
||||||
|
var item_status = $('.project-'+ item +'-section .btn-group .btn-primary').hasClass('btn-closed-'+ item)
|
||||||
|
? 'closed' : 'open';
|
||||||
|
}
|
||||||
|
$.ajax({
|
||||||
|
method: "GET",
|
||||||
|
url: "/",
|
||||||
|
dataType: "json",
|
||||||
|
data: {
|
||||||
|
cmd: "erpnext.templates.pages.projects.get_"+ item +"_html",
|
||||||
|
project: '{{ doc.name }}',
|
||||||
|
start: start,
|
||||||
|
item_status: item_status,
|
||||||
|
},
|
||||||
|
dataType: "json",
|
||||||
|
success: function(data) {
|
||||||
|
$(data.message).appendTo('.project-'+ item);
|
||||||
|
if(typeof data.message == 'undefined') {
|
||||||
|
$(".more-"+ item).toggle(false);
|
||||||
|
}
|
||||||
|
start = start+10;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var close_item = function(item, item_name){
|
||||||
|
var args = {
|
||||||
|
project: '{{ doc.name }}',
|
||||||
|
item_name: item_name,
|
||||||
|
}
|
||||||
|
frappe.call({
|
||||||
|
btn: this,
|
||||||
|
type: "POST",
|
||||||
|
method: "erpnext.templates.pages.projects.set_"+ item +"_status",
|
||||||
|
args: args,
|
||||||
|
callback: function(r) {
|
||||||
|
if(r.exc) {
|
||||||
|
if(r._server_messages)
|
||||||
|
frappe.msgprint(r._server_messages);
|
||||||
|
} else {
|
||||||
|
$(this).remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
142
erpnext/templates/pages/projects.py
Normal file
142
erpnext/templates/pages/projects.py
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
||||||
|
# License: GNU General Public License v3. See license.txt
|
||||||
|
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
import frappe
|
||||||
|
|
||||||
|
from frappe import _
|
||||||
|
|
||||||
|
def get_context(context):
|
||||||
|
context.no_cache = 1
|
||||||
|
|
||||||
|
project = frappe.get_doc('Project', frappe.form_dict.project)
|
||||||
|
|
||||||
|
project.has_permission('read')
|
||||||
|
|
||||||
|
context.issues = frappe.get_all('Issue', filters={'project': project.project_name},
|
||||||
|
fields=['subject', 'opening_date', 'resolution_date', 'status', 'name', 'resolution_details','modified','modified_by'])
|
||||||
|
|
||||||
|
project.tasks = get_tasks(project.name, start=0, search=frappe.form_dict.get("q"))
|
||||||
|
|
||||||
|
project.timelogs = get_timelogs(project.name, start=0, search=frappe.form_dict.get("q"))
|
||||||
|
|
||||||
|
project.issues = get_issues(project.name, start=0, search=frappe.form_dict.get("q"))
|
||||||
|
|
||||||
|
project.timelines = get_timeline(project.project_name, start=0)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
context.doc = project
|
||||||
|
|
||||||
|
|
||||||
|
def get_timeline(project, start=10):
|
||||||
|
issue_names = '({0})'.format(", ".join(["'{0}'".format(i.name) for i in get_issues(project)]))
|
||||||
|
|
||||||
|
|
||||||
|
timelines = frappe.db.sql("""
|
||||||
|
select
|
||||||
|
sender_full_name,
|
||||||
|
subject, communication_date, comment_type, name, creation, modified_by, reference_doctype, reference_name,
|
||||||
|
_liked_by, comment_type, _comments
|
||||||
|
from
|
||||||
|
tabCommunication
|
||||||
|
where
|
||||||
|
(reference_doctype='Project' and reference_name=%s)
|
||||||
|
or (timeline_doctype='Project' and timeline_name=%s)
|
||||||
|
or (reference_doctype='Issue' and reference_name IN {issue_names})
|
||||||
|
order by
|
||||||
|
modified DESC limit {start}, {limit}""".format(
|
||||||
|
issue_names=issue_names, start=start, limit=10),
|
||||||
|
(project, project), as_dict=True);
|
||||||
|
for timeline in timelines:
|
||||||
|
timeline.user_image = frappe.db.get_value('User', timeline.modified_by, 'user_image')
|
||||||
|
return timelines
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def get_timelines_html(project, start=0):
|
||||||
|
return frappe.render_template("erpnext/templates/includes/projects/timeline.html",
|
||||||
|
{"doc": {"timelines": get_timeline(project, start)}}, is_path=True)
|
||||||
|
|
||||||
|
def get_issue_list(project):
|
||||||
|
return [issue.name for issue in get_issues(project)]
|
||||||
|
|
||||||
|
def get_tasks(project, start=0, search=None, item_status=None):
|
||||||
|
filters = {"project": project}
|
||||||
|
if search:
|
||||||
|
filters["subject"] = ("like", "%{0}%".format(search))
|
||||||
|
if item_status:
|
||||||
|
filters = {"status": item_status}
|
||||||
|
tasks = frappe.get_all("Task", filters=filters,
|
||||||
|
fields=["name", "subject", "status", "exp_start_date", "exp_end_date", "priority"],
|
||||||
|
limit_start=start, limit_page_length=10)
|
||||||
|
|
||||||
|
for task in tasks:
|
||||||
|
task.todo = frappe.get_all('ToDo',filters={'reference_name':task.name, 'reference_type':'Task'},
|
||||||
|
fields=["assigned_by", "owner", "modified", "modified_by"])
|
||||||
|
if task.todo:
|
||||||
|
task.todo=task.todo[0]
|
||||||
|
task.todo.user_image = frappe.db.get_value('User', task.todo.owner, 'user_image')
|
||||||
|
|
||||||
|
return tasks
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def get_tasks_html(project, start=0, item_status=None):
|
||||||
|
return frappe.render_template("erpnext/templates/includes/projects/project_tasks.html",
|
||||||
|
{"doc": {"tasks": get_tasks(project, start, item_status=item_status)}}, is_path=True)
|
||||||
|
|
||||||
|
|
||||||
|
def get_issues(project, start=0, search=None, item_status=None):
|
||||||
|
filters = {"project": project}
|
||||||
|
if search:
|
||||||
|
filters["subject"] = ("like", "%{0}%".format(search))
|
||||||
|
if item_status:
|
||||||
|
filters = {"status": item_status}
|
||||||
|
issues = frappe.get_all("Issue", filters=filters,
|
||||||
|
fields=["name", "subject", "status", "opening_date", "resolution_date", "resolution_details"],
|
||||||
|
order_by='modified desc',
|
||||||
|
limit_start=start, limit_page_length=10)
|
||||||
|
|
||||||
|
for issue in issues:
|
||||||
|
issue.todo = frappe.get_all('ToDo',filters={'reference_name':issue.name, 'reference_type':'Issue'},
|
||||||
|
fields=["assigned_by", "owner", "modified", "modified_by"])
|
||||||
|
if issue.todo:
|
||||||
|
issue.todo=issue.todo[0]
|
||||||
|
issue.todo.user_image = frappe.db.get_value('User', issue.todo.owner, 'user_image')
|
||||||
|
|
||||||
|
return issues
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def get_issues_html(project, start=0, item_status=None):
|
||||||
|
return frappe.render_template("erpnext/templates/includes/projects/project_issues.html",
|
||||||
|
{"doc": {"issues": get_issues(project, start, item_status=item_status)}}, is_path=True)
|
||||||
|
|
||||||
|
def get_timelogs(project, start=0, search=None):
|
||||||
|
filters = {"project": project}
|
||||||
|
if search:
|
||||||
|
filters["title"] = ("like", "%{0}%".format(search))
|
||||||
|
|
||||||
|
timelogs = frappe.get_all('Time Log', filters=filters,
|
||||||
|
fields=['name','title','task','activity_type','from_time','to_time','hours','status','modified','modified_by'],
|
||||||
|
limit_start=start, limit_page_length=10)
|
||||||
|
for timelog in timelogs:
|
||||||
|
timelog.user_image = frappe.db.get_value('User', timelog.modified_by, 'user_image')
|
||||||
|
return timelogs
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def get_timelogs_html(project, start=0):
|
||||||
|
return frappe.render_template("erpnext/templates/includes/projects/project_timelogs.html",
|
||||||
|
{"doc": {"timelogs": get_timelogs(project, start)}}, is_path=True)
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def set_task_status(project, item_name):
|
||||||
|
task = frappe.get_doc("Task", item_name)
|
||||||
|
task.status = 'Closed'
|
||||||
|
task.save(ignore_permissions=True)
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def set_issue_status(project, item_name):
|
||||||
|
issue = frappe.get_doc("Issue", item_name)
|
||||||
|
issue.status = 'Closed'
|
||||||
|
issue.save(ignore_permissions=True)
|
||||||
|
|
||||||
|
|
149
erpnext/templates/pages/task_info.html
Normal file
149
erpnext/templates/pages/task_info.html
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
{% extends "templates/web.html" %}
|
||||||
|
{% block title %} {{ doc.name }} {% endblock %}
|
||||||
|
{% block breadcrumbs %}
|
||||||
|
<div class="page-breadcrumbs" data-html-block="breadcrumbs">
|
||||||
|
<ul class="breadcrumb">
|
||||||
|
<li>
|
||||||
|
<span class="icon icon-angle-left"></span>
|
||||||
|
<a href="/projects?project={{ doc.project }}">{{ doc.project }}</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
|
{% block page_content %}
|
||||||
|
<div class="row">
|
||||||
|
<div class=" col-sm-8 ">
|
||||||
|
<h1> {{ doc.subject }} </h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col-sm-4">
|
||||||
|
<div class="page-header-actions-block" data-html-block="header-actions">
|
||||||
|
<button type="submit" class="btn btn-primary btn-sm btn-form-submit">
|
||||||
|
Update</button>
|
||||||
|
<a href="tasks" class="btn btn-default btn-sm">
|
||||||
|
Cancel</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="page-content-block">
|
||||||
|
<form role="form" data-web-form="tasks">
|
||||||
|
|
||||||
|
<input type="hidden" name="web_form" value="tasks">
|
||||||
|
<input type="hidden" name="doctype" value="Task">
|
||||||
|
<input type="hidden" name="name" value="TASK00056">
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-12" style="max-width: 500px;">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="project" class="control-label text-muted small">Project</label>
|
||||||
|
<input type="text" class="form-control" name="project" readonly value= "{{ doc.project }}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="subject" class="control-label text-muted small">Subject</label>
|
||||||
|
<input type="text" class="form-control" name="subject" readonly value="{{ doc.subject }}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="description" class="control-label text-muted small">Details</label>
|
||||||
|
<textarea class="form-control" style="height: 200px;" name="description">{{ doc.description }}</textarea>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="priority" class="control-label text-muted small">Priority</label>
|
||||||
|
<input type="text" class="form-control" name="priority" readonly value="{{ doc.priority }}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="exp_start_date" class="control-label text-muted small">Expected Start Date</label>
|
||||||
|
<input type="text" class="form-control hasDatepicker" name="exp_start_date" readonly value="{{ doc.exp_start_date }}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="exp_end_date" class="control-label text-muted small">Expected End Date</label>
|
||||||
|
<input type="text" class="form-control hasDatepicker" name="exp_end_date" readonly value="{{ doc.exp_end_date }}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="status" class="control-label text-muted small">Status</label>
|
||||||
|
<select class="form-control" name="status" id="status" data-label="Status" data-fieldtype="Select">
|
||||||
|
<option value="Open" selected="selected">
|
||||||
|
Open</option><option value="Working">
|
||||||
|
Working</option><option value="Pending Review">
|
||||||
|
Pending Review</option><option value="Overdue">
|
||||||
|
Overdue</option><option value="Closed">
|
||||||
|
Closed</option><option value="Cancelled">
|
||||||
|
Cancelled</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="comments">
|
||||||
|
<h3>Comments</h3>
|
||||||
|
<div class="no-comment">
|
||||||
|
{% for comment in comments %}
|
||||||
|
<p class="text-muted">{{comment.sender_full_name}} : {{comment.subject}} on {{comment.communication_date.strftime('%Y-%m-%d')}}</p>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
<div class="comment-form-wrapper">
|
||||||
|
<a class="add-comment btn btn-default btn-sm">Add Comment</a>
|
||||||
|
<div style="display: none;" id="comment-form">
|
||||||
|
<p>Add Comment</p>
|
||||||
|
<form>
|
||||||
|
<fieldset>
|
||||||
|
<textarea class="form-control" name="comment" rows="5" placeholder="Comment"></textarea>
|
||||||
|
<p>
|
||||||
|
<button class="btn btn-primary btn-sm" id="submit-comment">Submit</button>
|
||||||
|
</p>
|
||||||
|
</fieldset>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
frappe.ready(function() {
|
||||||
|
var n_comments = $(".comment-row").length;
|
||||||
|
$(".add-comment").click(function() {
|
||||||
|
$(this).toggle(false);
|
||||||
|
$("#comment-form").toggle();
|
||||||
|
$("#comment-form textarea").val("");
|
||||||
|
})
|
||||||
|
$("#submit-comment").click(function() {
|
||||||
|
var args = {
|
||||||
|
comment_by_fullname: "test",
|
||||||
|
comment_by: "admin@localhost.com",
|
||||||
|
comment: $("[name='comment']").val(),
|
||||||
|
reference_doctype: "Task",
|
||||||
|
reference_name: "TASK00069",
|
||||||
|
comment_type: "Comment",
|
||||||
|
page_name: "tasks",
|
||||||
|
}
|
||||||
|
|
||||||
|
frappe.call({
|
||||||
|
btn: this,
|
||||||
|
type: "POST",
|
||||||
|
method: "frappe.templates.includes.comments.comments.add_comment",
|
||||||
|
args: args,
|
||||||
|
callback: function(r) {
|
||||||
|
if(r.exc) {
|
||||||
|
if(r._server_messages)
|
||||||
|
frappe.msgprint(r._server_messages);
|
||||||
|
} else {
|
||||||
|
$(r.message).appendTo("#comment-list");
|
||||||
|
$(".no-comment, .add-comment").toggle(false);
|
||||||
|
$("#comment-form")
|
||||||
|
.replaceWith('<div class="text-muted">Thank you for your comment!</div>')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return false;
|
||||||
|
})
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{% endblock %}
|
14
erpnext/templates/pages/task_info.py
Normal file
14
erpnext/templates/pages/task_info.py
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
from __future__ import unicode_literals
|
||||||
|
import frappe
|
||||||
|
|
||||||
|
from frappe import _
|
||||||
|
|
||||||
|
def get_context(context):
|
||||||
|
context.no_cache = 1
|
||||||
|
|
||||||
|
task = frappe.get_doc('Task', frappe.form_dict.task)
|
||||||
|
|
||||||
|
context.comments = frappe.get_all('Communication', filters={'reference_name': task.name, 'comment_type': 'comment'},
|
||||||
|
fields=['subject', 'sender_full_name', 'communication_date'])
|
||||||
|
|
||||||
|
context.doc = task
|
48
erpnext/templates/pages/timelog_info.html
Normal file
48
erpnext/templates/pages/timelog_info.html
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
{% extends "templates/web.html" %}
|
||||||
|
{% block title %} {{ doc.name }} {% endblock %}
|
||||||
|
{% block breadcrumbs %}
|
||||||
|
<div class="page-breadcrumbs" data-html-block="breadcrumbs">
|
||||||
|
<ul class="breadcrumb">
|
||||||
|
<li>
|
||||||
|
<span class="icon icon-angle-left"></span>
|
||||||
|
<a href="/projects?project={{ doc.project }}">{{ doc.project }}</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block page_content %}
|
||||||
|
<div class=" col-sm-8 ">
|
||||||
|
<h1> {{ doc.name }} </h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="page-content-block">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-12" style="max-width: 500px;">
|
||||||
|
<label for="project" class="control-label text-muted small">Project</label>
|
||||||
|
<input type="text" class="form-control" name="project" readonly value= "{{ doc.project }}">
|
||||||
|
|
||||||
|
<label for="activity_type" class="control-label text-muted small">Activity Type</label>
|
||||||
|
<input type="text" class="form-control" name="activity_type" readonly value= "{{ doc.activity_type }}">
|
||||||
|
|
||||||
|
<label for="task" class="control-label text-muted small">Task</label>
|
||||||
|
<input type="text" class="form-control" name="task" readonly value= "{{ doc.task }}">
|
||||||
|
|
||||||
|
<label for="from_time" class="control-label text-muted small">From Time</label>
|
||||||
|
<input type="text" class="form-control" name="from_time" readonly value= "{{ doc.from_time }}">
|
||||||
|
|
||||||
|
<label for="to_time" class="control-label text-muted small">To Time</label>
|
||||||
|
<input type="text" class="form-control" name="to_time" readonly value= "{{ doc.to_time }}">
|
||||||
|
|
||||||
|
<label for="to_time" class="control-label text-muted small">Hours</label>
|
||||||
|
<input type="text" class="form-control" name="Hours" readonly value= "{{ doc.hours }}">
|
||||||
|
|
||||||
|
<label for="status" class="control-label text-muted small">Status</label>
|
||||||
|
<input type="text" class="form-control" name="status" readonly value= "{{ doc.status }}">
|
||||||
|
|
||||||
|
<label for="Note" class="control-label text-muted small">Note</label>
|
||||||
|
<textarea class="form-control" name="Hours" readonly> {{ doc.note }} </textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
11
erpnext/templates/pages/timelog_info.py
Normal file
11
erpnext/templates/pages/timelog_info.py
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
from __future__ import unicode_literals
|
||||||
|
import frappe
|
||||||
|
|
||||||
|
from frappe import _
|
||||||
|
|
||||||
|
def get_context(context):
|
||||||
|
context.no_cache = 1
|
||||||
|
|
||||||
|
timelog = frappe.get_doc('Time Log', frappe.form_dict.timelog)
|
||||||
|
|
||||||
|
context.doc = timelog
|
Loading…
Reference in New Issue
Block a user