Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Boxiang Sun
gitlab-ce
Commits
01ef0b64
Commit
01ef0b64
authored
May 09, 2018
by
Dennis Tang
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
use component mixin to refactor components
parent
5a2f0913
Changes
7
Show whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
118 additions
and
194 deletions
+118
-194
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gcp_dropdown_mixin.js
...ts/gke_cluster_dropdowns/components/gcp_dropdown_mixin.js
+62
-0
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gke_machine_type_dropdown.vue
...luster_dropdowns/components/gke_machine_type_dropdown.vue
+12
-67
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gke_project_id_dropdown.vue
..._cluster_dropdowns/components/gke_project_id_dropdown.vue
+29
-61
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gke_zone_dropdown.vue
...ts/gke_cluster_dropdowns/components/gke_zone_dropdown.vue
+11
-62
spec/javascripts/projects/gke_cluster_dropdowns/components/gke_machine_type_dropdown_spec.js
...er_dropdowns/components/gke_machine_type_dropdown_spec.js
+1
-1
spec/javascripts/projects/gke_cluster_dropdowns/components/gke_project_id_dropdown_spec.js
...ster_dropdowns/components/gke_project_id_dropdown_spec.js
+2
-2
spec/javascripts/projects/gke_cluster_dropdowns/components/gke_zone_dropdown_spec.js
...ke_cluster_dropdowns/components/gke_zone_dropdown_spec.js
+1
-1
No files found.
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gcp_dropdown_mixin.js
0 → 100644
View file @
01ef0b64
import
_
from
'
underscore
'
;
import
LoadingIcon
from
'
~/vue_shared/components/loading_icon.vue
'
;
import
DropdownSearchInput
from
'
~/vue_shared/components/dropdown/dropdown_search_input.vue
'
;
import
DropdownHiddenInput
from
'
~/vue_shared/components/dropdown/dropdown_hidden_input.vue
'
;
import
DropdownButton
from
'
~/vue_shared/components/dropdown/dropdown_button.vue
'
;
import
store
from
'
../stores
'
;
export
default
{
store
,
components
:
{
LoadingIcon
,
DropdownButton
,
DropdownSearchInput
,
DropdownHiddenInput
,
},
props
:
{
fieldId
:
{
type
:
String
,
required
:
true
,
},
fieldName
:
{
type
:
String
,
required
:
true
,
},
defaultValue
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
},
data
()
{
return
{
isLoading
:
false
,
hasErrors
:
false
,
searchQuery
:
''
,
};
},
computed
:
{
results
()
{
return
this
.
items
.
filter
(
item
=>
item
.
name
.
toLowerCase
().
indexOf
(
this
.
searchQuery
)
>
-
1
);
},
},
methods
:
{
fetchSuccessHandler
()
{
if
(
this
.
defaultValue
)
{
const
itemToSelect
=
_
.
find
(
this
.
items
,
item
=>
item
.
name
===
this
.
defaultValue
);
if
(
itemToSelect
)
{
this
.
setItem
(
itemToSelect
.
name
);
}
}
this
.
isLoading
=
false
;
this
.
hasErrors
=
false
;
},
fetchFailureHandler
()
{
this
.
isLoading
=
false
;
this
.
hasErrors
=
true
;
},
},
};
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gke_machine_type_dropdown.vue
View file @
01ef0b64
...
@@ -2,46 +2,15 @@
...
@@ -2,46 +2,15 @@
import
_
from
'
underscore
'
;
import
_
from
'
underscore
'
;
import
{
s__
}
from
'
~/locale
'
;
import
{
s__
}
from
'
~/locale
'
;
import
{
mapState
,
mapGetters
,
mapActions
}
from
'
vuex
'
;
import
{
mapState
,
mapGetters
,
mapActions
}
from
'
vuex
'
;
import
LoadingIcon
from
'
~/vue_shared/components/loading_icon.vue
'
;
import
DropdownSearchInput
from
'
~/vue_shared/components/dropdown/dropdown_search_input.vue
'
;
import
DropdownHiddenInput
from
'
~/vue_shared/components/dropdown/dropdown_hidden_input.vue
'
;
import
DropdownButton
from
'
~/vue_shared/components/dropdown/dropdown_button.vue
'
;
import
store
from
'
../stores
'
;
import
gcpDropdownMixin
from
'
./gcp_dropdown_mixin
'
;
export
default
{
export
default
{
name
:
'
GkeMachineTypeDropdown
'
,
name
:
'
GkeMachineTypeDropdown
'
,
store
,
mixins
:
[
gcpDropdownMixin
],
components
:
{
LoadingIcon
,
DropdownButton
,
DropdownSearchInput
,
DropdownHiddenInput
,
},
props
:
{
fieldId
:
{
type
:
String
,
required
:
true
,
},
fieldName
:
{
type
:
String
,
required
:
true
,
},
defaultValue
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
},
data
()
{
return
{
isLoading
:
false
,
hasErrors
:
false
,
searchQuery
:
''
,
};
},
computed
:
{
computed
:
{
...
mapState
([
'
selectedProject
'
,
'
selectedZone
'
,
'
selectedMachineType
'
,
'
machineTypes
'
]),
...
mapState
([
'
selectedProject
'
,
'
selectedZone
'
,
'
selectedMachineType
'
]),
...
mapState
({
items
:
'
machineTypes
'
}),
...
mapGetters
([
'
hasProject
'
,
'
hasZone
'
,
'
hasMachineType
'
]),
...
mapGetters
([
'
hasProject
'
,
'
hasZone
'
,
'
hasMachineType
'
]),
allDropdownsSelected
()
{
allDropdownsSelected
()
{
return
this
.
hasProject
&&
this
.
hasZone
&&
this
.
hasMachineType
;
return
this
.
hasProject
&&
this
.
hasZone
&&
this
.
hasMachineType
;
...
@@ -49,11 +18,6 @@ export default {
...
@@ -49,11 +18,6 @@ export default {
isDisabled
()
{
isDisabled
()
{
return
!
this
.
selectedProject
||
!
this
.
selectedZone
;
return
!
this
.
selectedProject
||
!
this
.
selectedZone
;
},
},
results
()
{
return
this
.
machineTypes
.
filter
(
item
=>
item
.
name
.
toLowerCase
().
indexOf
(
this
.
searchQuery
)
>
-
1
,
);
},
toggleText
()
{
toggleText
()
{
if
(
this
.
isLoading
)
{
if
(
this
.
isLoading
)
{
return
s__
(
'
ClusterIntegration|Fetching machine types
'
);
return
s__
(
'
ClusterIntegration|Fetching machine types
'
);
...
@@ -80,38 +44,19 @@ export default {
...
@@ -80,38 +44,19 @@ export default {
},
},
watch
:
{
watch
:
{
selectedZone
()
{
selectedZone
()
{
this
.
fetchMachineTypes
();
this
.
isLoading
=
true
;
this
.
getMachineTypes
()
.
then
(
this
.
fetchSuccessHandler
)
.
catch
(
this
.
fetchFailureHandler
);
},
},
selectedMachineType
()
{
selectedMachineType
()
{
this
.
enableSubmit
();
this
.
enableSubmit
();
},
},
},
},
methods
:
{
methods
:
{
...
mapActions
([
'
setMachineType
'
,
'
getMachineTypes
'
]),
...
mapActions
([
'
getMachineTypes
'
]),
fetchMachineTypes
()
{
...
mapActions
({
setItem
:
'
setMachineType
'
}),
this
.
isLoading
=
true
;
this
.
getMachineTypes
()
.
then
(()
=>
{
if
(
this
.
defaultValue
)
{
const
machineTypeToSelect
=
_
.
find
(
this
.
machineTypes
,
item
=>
item
.
name
===
this
.
defaultValue
,
);
if
(
machineTypeToSelect
)
{
this
.
setMachineType
(
machineTypeToSelect
.
name
);
}
}
this
.
isLoading
=
false
;
this
.
hasErrors
=
false
;
})
.
catch
(()
=>
{
this
.
isLoading
=
false
;
this
.
hasErrors
=
true
;
});
},
enableSubmit
()
{
enableSubmit
()
{
if
(
this
.
allDropdownsSelected
)
{
if
(
this
.
allDropdownsSelected
)
{
const
submitButtonEl
=
document
.
querySelector
(
'
.js-gke-cluster-creation-submit
'
);
const
submitButtonEl
=
document
.
querySelector
(
'
.js-gke-cluster-creation-submit
'
);
...
@@ -154,7 +99,7 @@ export default {
...
@@ -154,7 +99,7 @@ export default {
v-for=
"result in results"
v-for=
"result in results"
:key=
"result.id"
:key=
"result.id"
>
>
<button
@
click.prevent=
"set
MachineType
(result.name)"
>
<button
@
click.prevent=
"set
Item
(result.name)"
>
{{
result
.
name
}}
{{
result
.
name
}}
</button>
</button>
</li>
</li>
...
...
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gke_project_id_dropdown.vue
View file @
01ef0b64
...
@@ -2,59 +2,32 @@
...
@@ -2,59 +2,32 @@
import
_
from
'
underscore
'
;
import
_
from
'
underscore
'
;
import
{
s__
,
sprintf
}
from
'
~/locale
'
;
import
{
s__
,
sprintf
}
from
'
~/locale
'
;
import
{
mapState
,
mapGetters
,
mapActions
}
from
'
vuex
'
;
import
{
mapState
,
mapGetters
,
mapActions
}
from
'
vuex
'
;
import
LoadingIcon
from
'
~/vue_shared/components/loading_icon.vue
'
;
import
DropdownSearchInput
from
'
~/vue_shared/components/dropdown/dropdown_search_input.vue
'
;
import
DropdownHiddenInput
from
'
~/vue_shared/components/dropdown/dropdown_hidden_input.vue
'
;
import
DropdownButton
from
'
~/vue_shared/components/dropdown/dropdown_button.vue
'
;
import
store
from
'
../stores
'
;
import
gcpDropdownMixin
from
'
./gcp_dropdown_mixin
'
;
export
default
{
export
default
{
name
:
'
GkeProjectIdDropdown
'
,
name
:
'
GkeProjectIdDropdown
'
,
store
,
mixins
:
[
gcpDropdownMixin
],
components
:
{
LoadingIcon
,
DropdownButton
,
DropdownSearchInput
,
DropdownHiddenInput
,
},
props
:
{
props
:
{
docsUrl
:
{
docsUrl
:
{
type
:
String
,
type
:
String
,
required
:
true
,
required
:
true
,
},
},
fieldId
:
{
type
:
String
,
required
:
true
,
},
fieldName
:
{
type
:
String
,
required
:
true
,
},
defaultValue
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
},
},
data
()
{
data
()
{
return
{
return
{
isLoading
:
true
,
isLoading
:
true
,
hasErrors
:
false
,
searchQuery
:
''
,
};
};
},
},
computed
:
{
computed
:
{
...
mapState
([
'
selectedProject
'
,
'
projects
'
]),
...
mapState
([
'
selectedProject
'
]),
...
mapState
({
items
:
'
projects
'
}),
...
mapGetters
([
'
hasProject
'
]),
...
mapGetters
([
'
hasProject
'
]),
hasOneProject
()
{
hasOneProject
()
{
return
this
.
project
s
.
length
===
1
;
return
this
.
item
s
.
length
===
1
;
},
},
isDisabled
()
{
isDisabled
()
{
return
this
.
projects
.
length
<
2
;
return
this
.
items
.
length
<
2
;
},
results
()
{
return
this
.
projects
.
filter
(
item
=>
item
.
name
.
toLowerCase
().
indexOf
(
this
.
searchQuery
)
>
-
1
);
},
},
noSearchResultsText
()
{
noSearchResultsText
()
{
return
s__
(
'
ClusterIntegration|No projects matched your search
'
);
return
s__
(
'
ClusterIntegration|No projects matched your search
'
);
...
@@ -68,7 +41,7 @@ export default {
...
@@ -68,7 +41,7 @@ export default {
return
this
.
selectedProject
.
name
;
return
this
.
selectedProject
.
name
;
}
}
return
!
this
.
project
s
.
length
return
!
this
.
item
s
.
length
?
s__
(
'
ClusterIntegration|No projects found
'
)
?
s__
(
'
ClusterIntegration|No projects found
'
)
:
s__
(
'
ClusterIntegration|Select project
'
);
:
s__
(
'
ClusterIntegration|Select project
'
);
},
},
...
@@ -82,7 +55,7 @@ export default {
...
@@ -82,7 +55,7 @@ export default {
'
ClusterIntegration|We were unable to fetch any projects. Ensure that you have a project on %{docsLinkStart}Google Cloud Platform%{docsLinkEnd}.
'
;
'
ClusterIntegration|We were unable to fetch any projects. Ensure that you have a project on %{docsLinkStart}Google Cloud Platform%{docsLinkEnd}.
'
;
}
}
message
=
this
.
project
s
.
length
message
=
this
.
item
s
.
length
?
'
ClusterIntegration|To use a new project, first create one on %{docsLinkStart}Google Cloud Platform%{docsLinkEnd}.
'
?
'
ClusterIntegration|To use a new project, first create one on %{docsLinkStart}Google Cloud Platform%{docsLinkEnd}.
'
:
'
ClusterIntegration|To create a cluster, first create a project on %{docsLinkStart}Google Cloud Platform%{docsLinkEnd}.
'
;
:
'
ClusterIntegration|To create a cluster, first create a project on %{docsLinkStart}Google Cloud Platform%{docsLinkEnd}.
'
;
...
@@ -99,24 +72,16 @@ export default {
...
@@ -99,24 +72,16 @@ export default {
},
},
},
},
created
()
{
created
()
{
this
.
fetchProjects
();
},
methods
:
{
...
mapActions
([
'
setProject
'
,
'
getProjects
'
]),
fetchProjects
()
{
this
.
getProjects
()
this
.
getProjects
()
.
then
(()
=>
{
.
then
(()
=>
{
if
(
this
.
defaultValue
)
{
if
(
this
.
defaultValue
)
{
const
projectToSelect
=
_
.
find
(
const
projectToSelect
=
_
.
find
(
this
.
items
,
item
=>
item
.
projectId
===
this
.
defaultValue
);
this
.
projects
,
item
=>
item
.
projectId
===
this
.
defaultValue
,
);
if
(
projectToSelect
)
{
if
(
projectToSelect
)
{
this
.
setProject
(
projectToSelect
);
this
.
setItem
(
projectToSelect
);
}
}
}
else
if
(
this
.
project
s
.
length
===
1
)
{
}
else
if
(
this
.
item
s
.
length
===
1
)
{
this
.
setProject
(
this
.
project
s
[
0
]);
this
.
setItem
(
this
.
item
s
[
0
]);
}
}
this
.
isLoading
=
false
;
this
.
isLoading
=
false
;
...
@@ -127,6 +92,9 @@ export default {
...
@@ -127,6 +92,9 @@ export default {
this
.
hasErrors
=
true
;
this
.
hasErrors
=
true
;
});
});
},
},
methods
:
{
...
mapActions
([
'
getProjects
'
]),
...
mapActions
({
setItem
:
'
setProject
'
}),
},
},
};
};
</
script
>
</
script
>
...
@@ -164,7 +132,7 @@ export default {
...
@@ -164,7 +132,7 @@ export default {
v-for=
"result in results"
v-for=
"result in results"
:key=
"result.project_number"
:key=
"result.project_number"
>
>
<button
@
click.prevent=
"set
Project
(result)"
>
<button
@
click.prevent=
"set
Item
(result)"
>
{{
result
.
name
}}
{{
result
.
name
}}
</button>
</button>
</li>
</li>
...
...
app/assets/javascripts/projects/gke_cluster_dropdowns/components/gke_zone_dropdown.vue
View file @
01ef0b64
<
script
>
<
script
>
import
_
from
'
underscore
'
;
import
{
s__
}
from
'
~/locale
'
;
import
{
s__
}
from
'
~/locale
'
;
import
{
mapState
,
mapGetters
,
mapActions
}
from
'
vuex
'
;
import
{
mapState
,
mapGetters
,
mapActions
}
from
'
vuex
'
;
import
LoadingIcon
from
'
~/vue_shared/components/loading_icon.vue
'
;
import
DropdownSearchInput
from
'
~/vue_shared/components/dropdown/dropdown_search_input.vue
'
;
import
DropdownHiddenInput
from
'
~/vue_shared/components/dropdown/dropdown_hidden_input.vue
'
;
import
DropdownButton
from
'
~/vue_shared/components/dropdown/dropdown_button.vue
'
;
import
store
from
'
../stores
'
;
import
gcpDropdownMixin
from
'
./gcp_dropdown_mixin
'
;
export
default
{
export
default
{
name
:
'
GkeZoneDropdown
'
,
name
:
'
GkeZoneDropdown
'
,
store
,
mixins
:
[
gcpDropdownMixin
],
components
:
{
LoadingIcon
,
DropdownButton
,
DropdownSearchInput
,
DropdownHiddenInput
,
},
props
:
{
fieldId
:
{
type
:
String
,
required
:
true
,
},
fieldName
:
{
type
:
String
,
required
:
true
,
},
defaultValue
:
{
type
:
String
,
required
:
false
,
default
:
''
,
},
},
data
()
{
return
{
isLoading
:
false
,
hasErrors
:
false
,
searchQuery
:
''
,
};
},
computed
:
{
computed
:
{
...
mapState
([
'
selectedProject
'
,
'
selectedZone
'
,
'
zones
'
]),
...
mapState
([
'
selectedProject
'
,
'
selectedZone
'
]),
...
mapState
({
items
:
'
zones
'
}),
...
mapGetters
([
'
hasProject
'
]),
...
mapGetters
([
'
hasProject
'
]),
isDisabled
()
{
isDisabled
()
{
return
!
this
.
hasProject
;
return
!
this
.
hasProject
;
},
},
results
()
{
return
this
.
zones
.
filter
(
item
=>
item
.
name
.
toLowerCase
().
indexOf
(
this
.
searchQuery
)
>
-
1
);
},
toggleText
()
{
toggleText
()
{
if
(
this
.
isLoading
)
{
if
(
this
.
isLoading
)
{
return
s__
(
'
ClusterIntegration|Fetching zones
'
);
return
s__
(
'
ClusterIntegration|Fetching zones
'
);
...
@@ -71,33 +36,17 @@ export default {
...
@@ -71,33 +36,17 @@ export default {
},
},
watch
:
{
watch
:
{
selectedProject
()
{
selectedProject
()
{
this
.
fetchZones
();
},
},
methods
:
{
...
mapActions
([
'
setZone
'
,
'
getZones
'
]),
fetchZones
()
{
this
.
isLoading
=
true
;
this
.
isLoading
=
true
;
this
.
getZones
()
this
.
getZones
()
.
then
(()
=>
{
.
then
(
this
.
fetchSuccessHandler
)
if
(
this
.
defaultValue
)
{
.
catch
(
this
.
fetchFailureHandler
);
const
zoneToSelect
=
_
.
find
(
this
.
zones
,
item
=>
item
.
name
===
this
.
defaultValue
);
if
(
zoneToSelect
)
{
this
.
setZone
(
zoneToSelect
.
name
);
}
}
this
.
isLoading
=
false
;
this
.
hasErrors
=
false
;
})
.
catch
(()
=>
{
this
.
isLoading
=
false
;
this
.
hasErrors
=
true
;
});
},
},
},
},
methods
:
{
...
mapActions
([
'
getZones
'
]),
...
mapActions
({
setItem
:
'
setZone
'
}),
},
};
};
</
script
>
</
script
>
...
@@ -130,7 +79,7 @@ export default {
...
@@ -130,7 +79,7 @@ export default {
v-for=
"result in results"
v-for=
"result in results"
:key=
"result.id"
:key=
"result.id"
>
>
<button
@
click.prevent=
"set
Zone
(result.name)"
>
<button
@
click.prevent=
"set
Item
(result.name)"
>
{{
result
.
name
}}
{{
result
.
name
}}
</button>
</button>
</li>
</li>
...
...
spec/javascripts/projects/gke_cluster_dropdowns/components/gke_machine_type_dropdown_spec.js
View file @
01ef0b64
...
@@ -73,7 +73,7 @@ describe('GkeMachineTypeDropdown', () => {
...
@@ -73,7 +73,7 @@ describe('GkeMachineTypeDropdown', () => {
});
});
it
(
'
returns machine type name if machine type selected
'
,
()
=>
{
it
(
'
returns machine type name if machine type selected
'
,
()
=>
{
vm
.
set
MachineType
(
selectedMachineTypeMock
);
vm
.
set
Item
(
selectedMachineTypeMock
);
expect
(
vm
.
toggleText
).
toBe
(
selectedMachineTypeMock
);
expect
(
vm
.
toggleText
).
toBe
(
selectedMachineTypeMock
);
});
});
...
...
spec/javascripts/projects/gke_cluster_dropdowns/components/gke_project_id_dropdown_spec.js
View file @
01ef0b64
...
@@ -45,7 +45,7 @@ describe('GkeProjectIdDropdown', () => {
...
@@ -45,7 +45,7 @@ describe('GkeProjectIdDropdown', () => {
it
(
'
returns default toggle text
'
,
done
=>
{
it
(
'
returns default toggle text
'
,
done
=>
{
vm
.
$nextTick
(()
=>
{
vm
.
$nextTick
(()
=>
{
vm
.
$nextTick
(()
=>
{
vm
.
$nextTick
(()
=>
{
vm
.
set
Project
(
emptyProjectMock
);
vm
.
set
Item
(
emptyProjectMock
);
expect
(
vm
.
toggleText
).
toBe
(
LABELS
.
DEFAULT
);
expect
(
vm
.
toggleText
).
toBe
(
LABELS
.
DEFAULT
);
done
();
done
();
...
@@ -65,7 +65,7 @@ describe('GkeProjectIdDropdown', () => {
...
@@ -65,7 +65,7 @@ describe('GkeProjectIdDropdown', () => {
it
(
'
returns empty toggle text
'
,
done
=>
{
it
(
'
returns empty toggle text
'
,
done
=>
{
vm
.
$nextTick
(()
=>
{
vm
.
$nextTick
(()
=>
{
vm
.
$store
.
commit
(
SET_PROJECTS
,
[]);
vm
.
$store
.
commit
(
SET_PROJECTS
,
[]);
vm
.
set
Project
(
emptyProjectMock
);
vm
.
set
Item
(
emptyProjectMock
);
vm
.
$nextTick
(()
=>
{
vm
.
$nextTick
(()
=>
{
expect
(
vm
.
toggleText
).
toBe
(
LABELS
.
EMPTY
);
expect
(
vm
.
toggleText
).
toBe
(
LABELS
.
EMPTY
);
...
...
spec/javascripts/projects/gke_cluster_dropdowns/components/gke_zone_dropdown_spec.js
View file @
01ef0b64
...
@@ -55,7 +55,7 @@ describe('GkeZoneDropdown', () => {
...
@@ -55,7 +55,7 @@ describe('GkeZoneDropdown', () => {
});
});
it
(
'
returns project name if project selected
'
,
()
=>
{
it
(
'
returns project name if project selected
'
,
()
=>
{
vm
.
set
Zone
(
selectedZoneMock
);
vm
.
set
Item
(
selectedZoneMock
);
expect
(
vm
.
toggleText
).
toBe
(
selectedZoneMock
);
expect
(
vm
.
toggleText
).
toBe
(
selectedZoneMock
);
});
});
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment