mirror of
https://github.com/OrchardCMS/Orchard.git
synced 2025-10-15 19:54:57 +08:00
Improving media library browsing
- Adding 'Recent' menu item - Adding sorting - Adding filtering --HG-- branch : 1.x
This commit is contained in:
@@ -11,19 +11,24 @@ using Orchard.Mvc;
|
|||||||
using Orchard.Themes;
|
using Orchard.Themes;
|
||||||
using Orchard.UI.Navigation;
|
using Orchard.UI.Navigation;
|
||||||
using Orchard.Utility.Extensions;
|
using Orchard.Utility.Extensions;
|
||||||
|
using Orchard.ContentManagement.MetaData;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace Orchard.MediaLibrary.Controllers {
|
namespace Orchard.MediaLibrary.Controllers {
|
||||||
[ValidateInput(false)]
|
[ValidateInput(false)]
|
||||||
public class AdminController : Controller {
|
public class AdminController : Controller {
|
||||||
private readonly IMediaLibraryService _mediaLibraryService;
|
private readonly IMediaLibraryService _mediaLibraryService;
|
||||||
private readonly INavigationManager _navigationManager;
|
private readonly INavigationManager _navigationManager;
|
||||||
|
private readonly IContentDefinitionManager _contentDefinitionManager;
|
||||||
|
|
||||||
public AdminController(
|
public AdminController(
|
||||||
IOrchardServices services,
|
IOrchardServices services,
|
||||||
IMediaLibraryService mediaLibraryService,
|
IMediaLibraryService mediaLibraryService,
|
||||||
INavigationManager navigationManager ) {
|
INavigationManager navigationManager,
|
||||||
|
IContentDefinitionManager contentDefinitionManager) {
|
||||||
_mediaLibraryService = mediaLibraryService;
|
_mediaLibraryService = mediaLibraryService;
|
||||||
_navigationManager = navigationManager;
|
_navigationManager = navigationManager;
|
||||||
|
_contentDefinitionManager = contentDefinitionManager;
|
||||||
Services = services;
|
Services = services;
|
||||||
|
|
||||||
T = NullLocalizer.Instance;
|
T = NullLocalizer.Instance;
|
||||||
@@ -35,17 +40,22 @@ namespace Orchard.MediaLibrary.Controllers {
|
|||||||
public ILogger Logger { get; set; }
|
public ILogger Logger { get; set; }
|
||||||
|
|
||||||
public ActionResult Index(int? id, bool dialog = false) {
|
public ActionResult Index(int? id, bool dialog = false) {
|
||||||
|
string stereotype;
|
||||||
|
var mediaTypes = new List<string>();
|
||||||
|
|
||||||
|
foreach(var contentTypeDefinition in _contentDefinitionManager.ListTypeDefinitions()) {
|
||||||
|
if (contentTypeDefinition.Settings.TryGetValue("Stereotype", out stereotype) && stereotype == "Media")
|
||||||
|
mediaTypes.Add(contentTypeDefinition.Name);
|
||||||
|
}
|
||||||
|
|
||||||
var viewModel = new MediaManagerIndexViewModel {
|
var viewModel = new MediaManagerIndexViewModel {
|
||||||
DialogMode = dialog,
|
DialogMode = dialog,
|
||||||
Folders = _mediaLibraryService.GetMediaFolders(),
|
Folders = _mediaLibraryService.GetMediaFolders(),
|
||||||
Folder = id,
|
Folder = id,
|
||||||
Hierarchy = id.HasValue ? _mediaLibraryService.GetMediaFolderHierarchy(id.Value) : Enumerable.Empty<MediaFolder>()
|
Hierarchy = id.HasValue ? _mediaLibraryService.GetMediaFolderHierarchy(id.Value) : Enumerable.Empty<MediaFolder>(),
|
||||||
|
MediaTypes = mediaTypes.ToArray()
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!id.HasValue && viewModel.Folders.Any()) {
|
|
||||||
viewModel.Folder = viewModel.Folders.First().TermId;
|
|
||||||
}
|
|
||||||
|
|
||||||
return View(viewModel);
|
return View(viewModel);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -55,20 +65,21 @@ namespace Orchard.MediaLibrary.Controllers {
|
|||||||
|
|
||||||
var hierarchy = _mediaLibraryService.GetMediaFolderHierarchy(id);
|
var hierarchy = _mediaLibraryService.GetMediaFolderHierarchy(id);
|
||||||
|
|
||||||
|
|
||||||
var viewModel = new MediaManagerImportViewModel {
|
var viewModel = new MediaManagerImportViewModel {
|
||||||
DialogMode = dialog,
|
DialogMode = dialog,
|
||||||
Menu = mediaProviderMenu,
|
Menu = mediaProviderMenu,
|
||||||
Hierarchy = hierarchy.ToReadOnlyCollection(),
|
Hierarchy = hierarchy.ToReadOnlyCollection(),
|
||||||
ImageSets = imageSets
|
ImageSets = imageSets,
|
||||||
};
|
};
|
||||||
|
|
||||||
return View(viewModel);
|
return View(viewModel);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Themed(false)]
|
[Themed(false)]
|
||||||
public ActionResult MediaItems(int id, int skip = 0, int count = 0) {
|
public ActionResult MediaItems(int id, int skip = 0, int count = 0, string order = "created", string mediaType = "") {
|
||||||
var mediaParts = _mediaLibraryService.GetMediaContentItemsForLocation(id, skip, count);
|
var mediaParts = _mediaLibraryService.GetMediaContentItems(id, skip, count, order, mediaType);
|
||||||
var mediaPartsCount = _mediaLibraryService.GetMediaContentItemsCountForLocation(id);
|
var mediaPartsCount = _mediaLibraryService.GetMediaContentItemsCount(id, mediaType);
|
||||||
|
|
||||||
var mediaItems = mediaParts.Select(x => new MediaManagerMediaItemViewModel {
|
var mediaItems = mediaParts.Select(x => new MediaManagerMediaItemViewModel {
|
||||||
MediaPart = x,
|
MediaPart = x,
|
||||||
@@ -83,6 +94,24 @@ namespace Orchard.MediaLibrary.Controllers {
|
|||||||
return View(viewModel);
|
return View(viewModel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Themed(false)]
|
||||||
|
public ActionResult RecentMediaItems(int skip = 0, int count = 0, string order = "created", string mediaType = "") {
|
||||||
|
var mediaParts = _mediaLibraryService.GetMediaContentItems(skip, count, order, mediaType);
|
||||||
|
var mediaPartsCount = _mediaLibraryService.GetMediaContentItemsCount(mediaType);
|
||||||
|
|
||||||
|
var mediaItems = mediaParts.Select(x => new MediaManagerMediaItemViewModel {
|
||||||
|
MediaPart = x,
|
||||||
|
Shape = Services.ContentManager.BuildDisplay(x, "Thumbnail")
|
||||||
|
}).ToList();
|
||||||
|
|
||||||
|
var viewModel = new MediaManagerMediaItemsViewModel {
|
||||||
|
MediaItems = mediaItems,
|
||||||
|
MediaItemsCount = mediaPartsCount
|
||||||
|
};
|
||||||
|
|
||||||
|
return View("MediaItems", viewModel);
|
||||||
|
}
|
||||||
|
|
||||||
[Themed(false)]
|
[Themed(false)]
|
||||||
public ActionResult MediaItem(int id, string displayType = "SummaryAdmin") {
|
public ActionResult MediaItem(int id, string displayType = "SummaryAdmin") {
|
||||||
var contentItem = Services.ContentManager.Get(id, VersionOptions.Latest);
|
var contentItem = Services.ContentManager.Get(id, VersionOptions.Latest);
|
||||||
|
@@ -23,12 +23,22 @@ namespace Orchard.MediaLibrary {
|
|||||||
.Column<string>("Resource", c => c.WithLength(2048))
|
.Column<string>("Resource", c => c.WithLength(2048))
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// create the "Media Location" taxonomy
|
||||||
var taxonomy = _contentManager.New<TaxonomyPart>("Taxonomy");
|
var taxonomy = _contentManager.New<TaxonomyPart>("Taxonomy");
|
||||||
taxonomy.IsInternal = true;
|
taxonomy.IsInternal = true;
|
||||||
taxonomy.Name = MediaLibraryService.MediaLocation;
|
taxonomy.Name = MediaLibraryService.MediaLocation;
|
||||||
|
|
||||||
_contentManager.Create(taxonomy);
|
_contentManager.Create(taxonomy);
|
||||||
|
|
||||||
|
// create the "Media" term
|
||||||
|
var term = _contentManager.New<TermPart>(taxonomy.TermTypeName);
|
||||||
|
term.TaxonomyId = taxonomy.Id;
|
||||||
|
term.Container = taxonomy;
|
||||||
|
term.Name = "Media";
|
||||||
|
term.Path = "/";
|
||||||
|
|
||||||
|
_contentManager.Create(term, VersionOptions.Published);
|
||||||
|
|
||||||
ContentDefinitionManager.AlterTypeDefinition("Image", td => td
|
ContentDefinitionManager.AlterTypeDefinition("Image", td => td
|
||||||
.DisplayedAs("Image")
|
.DisplayedAs("Image")
|
||||||
.WithSetting("Stereotype", "Media")
|
.WithSetting("Stereotype", "Media")
|
||||||
@@ -65,21 +75,17 @@ namespace Orchard.MediaLibrary {
|
|||||||
.WithPart("TitlePart")
|
.WithPart("TitlePart")
|
||||||
);
|
);
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int UpdateFrom1() {
|
|
||||||
|
|
||||||
ContentDefinitionManager.AlterTypeDefinition("OEmbed", td => td
|
ContentDefinitionManager.AlterTypeDefinition("OEmbed", td => td
|
||||||
.DisplayedAs("External Media")
|
.DisplayedAs("External Media")
|
||||||
.WithSetting("Stereotype", "Media")
|
.WithSetting("Stereotype", "Media")
|
||||||
.WithPart("CommonPart")
|
.WithPart("CommonPart")
|
||||||
.WithPart("MediaPart")
|
.WithPart("MediaPart")
|
||||||
.WithPart("OEmbedPart")
|
.WithPart("OEmbedPart")
|
||||||
.WithPart("TitlePart")
|
.WithPart("TitlePart")
|
||||||
);
|
);
|
||||||
|
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -19,8 +19,10 @@ namespace Orchard.MediaLibrary.Services {
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
IEnumerable<string> GetMediaTypes();
|
IEnumerable<string> GetMediaTypes();
|
||||||
|
|
||||||
IEnumerable<MediaPart> GetMediaContentItemsForLocation(int? locationId, int skip, int count);
|
IEnumerable<MediaPart> GetMediaContentItems(int skip, int count, string order, string mediaType);
|
||||||
int GetMediaContentItemsCountForLocation(int? locationId);
|
IEnumerable<MediaPart> GetMediaContentItems(int folder, int skip, int count, string order, string mediaType);
|
||||||
|
int GetMediaContentItemsCount(string mediaType);
|
||||||
|
int GetMediaContentItemsCount(int folder, string mediaType);
|
||||||
|
|
||||||
MediaPart ImportStream(int termId, Stream stream, string filename);
|
MediaPart ImportStream(int termId, Stream stream, string filename);
|
||||||
|
|
||||||
|
@@ -9,6 +9,7 @@ using Orchard.MediaLibrary.Factories;
|
|||||||
using Orchard.MediaLibrary.Models;
|
using Orchard.MediaLibrary.Models;
|
||||||
using Orchard.Taxonomies.Models;
|
using Orchard.Taxonomies.Models;
|
||||||
using Orchard.Taxonomies.Services;
|
using Orchard.Taxonomies.Services;
|
||||||
|
using Orchard.Core.Title.Models;
|
||||||
|
|
||||||
namespace Orchard.MediaLibrary.Services {
|
namespace Orchard.MediaLibrary.Services {
|
||||||
public class MediaLibraryService : IMediaLibraryService {
|
public class MediaLibraryService : IMediaLibraryService {
|
||||||
@@ -37,24 +38,26 @@ namespace Orchard.MediaLibrary.Services {
|
|||||||
var taxonomy = GetMediaLocationTaxonomy();
|
var taxonomy = GetMediaLocationTaxonomy();
|
||||||
|
|
||||||
var terms = _taxonomyService.GetTerms(taxonomy.Id);
|
var terms = _taxonomyService.GetTerms(taxonomy.Id);
|
||||||
var result = new List<MediaFolder>();
|
var rootFolders = new List<MediaFolder>();
|
||||||
|
var index = new Dictionary<int, MediaFolder>();
|
||||||
|
|
||||||
_taxonomyService.CreateHierarchy(terms, (parent, child) => {
|
_taxonomyService.CreateHierarchy(terms, (parent, child) => {
|
||||||
|
MediaFolder parentFolder;
|
||||||
|
MediaFolder childFolder = CreateMediaFolder(child.TermPart);
|
||||||
|
index.Add(child.TermPart.Id, childFolder);
|
||||||
|
|
||||||
// adding to root
|
// adding to root
|
||||||
if (parent.TermPart == null) {
|
if (parent.TermPart != null) {
|
||||||
result.Add(CreateMediaFolder(child.TermPart));
|
parentFolder = index.ContainsKey(parent.TermPart.Id) ? index[parent.TermPart.Id] : null;
|
||||||
|
parentFolder.Folders.Add(childFolder);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
var seek = result.FirstOrDefault(x => x.TermId == parent.TermPart.Id);
|
rootFolders.Add(childFolder);
|
||||||
if (seek != null) {
|
|
||||||
seek.Folders.Add(CreateMediaFolder(child.TermPart));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return result;
|
return rootFolders;
|
||||||
}
|
}
|
||||||
|
|
||||||
public MediaFolder GetMediaFolder(int id) {
|
public MediaFolder GetMediaFolder(int id) {
|
||||||
@@ -90,33 +93,64 @@ namespace Orchard.MediaLibrary.Services {
|
|||||||
return _contentManager.Query<MediaPart, MediaPartRecord>();
|
return _contentManager.Query<MediaPart, MediaPartRecord>();
|
||||||
}
|
}
|
||||||
|
|
||||||
public IEnumerable<MediaPart> GetMediaContentItemsForLocation(int? locationId, int skip, int count) {
|
public IEnumerable<MediaPart> GetMediaContentItems(int folder, int skip, int count, string order, string mediaType) {
|
||||||
if (locationId.HasValue) {
|
var query = _contentManager.Query<MediaPart>();
|
||||||
return _contentManager.Query<MediaPart, MediaPartRecord>()
|
|
||||||
.Where(m => m.TermPartRecord.Id == locationId)
|
if (!String.IsNullOrEmpty(mediaType)) {
|
||||||
.Join<CommonPartRecord>()
|
query = query.ForType(new[] { mediaType });
|
||||||
.OrderByDescending(x => x.CreatedUtc)
|
|
||||||
.Slice(skip, count)
|
|
||||||
.ToArray();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return _contentManager.Query<MediaPart, MediaPartRecord>()
|
if (folder > 0) {
|
||||||
.Where(m => m.TermPartRecord == null)
|
query = query.Join<MediaPartRecord>().Where(m => m.TermPartRecord.Id == folder);
|
||||||
.Join<CommonPartRecord>()
|
}
|
||||||
|
|
||||||
|
switch(order) {
|
||||||
|
case "title":
|
||||||
|
return query.Join<TitlePartRecord>()
|
||||||
|
.OrderBy(x => x.Title)
|
||||||
|
.Slice(skip, count)
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
|
case "modified":
|
||||||
|
return query.Join<CommonPartRecord>()
|
||||||
|
.OrderByDescending(x => x.ModifiedUtc)
|
||||||
|
.Slice(skip, count)
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
|
case "published":
|
||||||
|
return query.Join<CommonPartRecord>()
|
||||||
|
.OrderByDescending(x => x.PublishedUtc)
|
||||||
|
.Slice(skip, count)
|
||||||
|
.ToArray();
|
||||||
|
|
||||||
|
default:
|
||||||
|
return query.Join<CommonPartRecord>()
|
||||||
.OrderByDescending(x => x.CreatedUtc)
|
.OrderByDescending(x => x.CreatedUtc)
|
||||||
.Slice(skip, count)
|
.Slice(skip, count)
|
||||||
.ToArray();
|
.ToArray();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int GetMediaContentItemsCountForLocation(int? locationId) {
|
public IEnumerable<MediaPart> GetMediaContentItems(int skip, int count, string order, string mediaType) {
|
||||||
if (locationId.HasValue) {
|
return GetMediaContentItems(-1, skip, count, order, mediaType);
|
||||||
return _contentManager.Query<MediaPart, MediaPartRecord>()
|
}
|
||||||
.Where(m => m.TermPartRecord.Id == locationId)
|
|
||||||
.Count();
|
public int GetMediaContentItemsCount(int folder, string mediaType) {
|
||||||
|
var query = _contentManager.Query<MediaPart>();
|
||||||
|
|
||||||
|
if (!String.IsNullOrEmpty(mediaType)) {
|
||||||
|
query = query.ForType(new[] { mediaType });
|
||||||
}
|
}
|
||||||
return _contentManager.Query<MediaPart, MediaPartRecord>()
|
|
||||||
.Where(m => m.TermPartRecord == null)
|
if (folder > 0) {
|
||||||
.Count();
|
query = query.Join<MediaPartRecord>().Where(m => m.TermPartRecord.Id == folder);
|
||||||
|
}
|
||||||
|
|
||||||
|
return query.Count();
|
||||||
|
}
|
||||||
|
|
||||||
|
public int GetMediaContentItemsCount(string mediaType) {
|
||||||
|
return GetMediaContentItemsCount(-1, mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
public MediaPart ImportStream(int termId, Stream stream, string filename) {
|
public MediaPart ImportStream(int termId, Stream stream, string filename) {
|
||||||
|
@@ -49,6 +49,10 @@
|
|||||||
float: right;
|
float: right;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#media-library-toolbar label {
|
||||||
|
display: inline;
|
||||||
|
}
|
||||||
|
|
||||||
#media-library-main {
|
#media-library-main {
|
||||||
display: table-row;
|
display: table-row;
|
||||||
border-bottom: 1px solid #e0e0e0;
|
border-bottom: 1px solid #e0e0e0;
|
||||||
@@ -63,9 +67,25 @@
|
|||||||
border-right: 1px solid #e0e0e0;
|
border-right: 1px solid #e0e0e0;
|
||||||
}
|
}
|
||||||
#media-library-main-navigation > ul { /* sub-navigations, e.g. folders */
|
#media-library-main-navigation > ul { /* sub-navigations, e.g. folders */
|
||||||
padding-left: 16px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#button-recent {
|
||||||
|
display: block;
|
||||||
|
padding:2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#button-recent:hover, #button-recent.selected {
|
||||||
|
cursor: pointer;
|
||||||
|
background-color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#button-recent i {
|
||||||
|
color: rgb(75,75,75);
|
||||||
|
padding-left:3px;
|
||||||
|
padding-right: 5px;
|
||||||
|
font-size:14px;
|
||||||
|
}
|
||||||
|
|
||||||
.media-library-folder-title {
|
.media-library-folder-title {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
-moz-box-sizing: border-box;
|
-moz-box-sizing: border-box;
|
||||||
@@ -82,17 +102,17 @@
|
|||||||
background-color: #e0e0e0;
|
background-color: #e0e0e0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.media-library-navigation-folder-icon {
|
|
||||||
background-image: url('');
|
|
||||||
background-position: left;
|
|
||||||
background-repeat: no-repeat;
|
|
||||||
padding-left: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.media-library-navigation-folder-link {
|
.media-library-navigation-folder-link {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.media-library-navigation-folder-link i {
|
||||||
|
color: #808080;
|
||||||
|
padding-left:3px;
|
||||||
|
padding-right: 5px;
|
||||||
|
font-size:14px;
|
||||||
|
}
|
||||||
|
|
||||||
.media-library-folder ul {
|
.media-library-folder ul {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
@@ -7,5 +7,6 @@ namespace Orchard.MediaLibrary.ViewModels {
|
|||||||
public IEnumerable<MediaFolder> Hierarchy { get; set; }
|
public IEnumerable<MediaFolder> Hierarchy { get; set; }
|
||||||
public int? Folder { get; set; }
|
public int? Folder { get; set; }
|
||||||
public bool DialogMode { get; set; }
|
public bool DialogMode { get; set; }
|
||||||
|
public string[] MediaTypes { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -9,7 +9,8 @@
|
|||||||
Script.Require("jQueryUI_Droppable");
|
Script.Require("jQueryUI_Droppable");
|
||||||
Script.Include("knockout-2.2.1.js");
|
Script.Include("knockout-2.2.1.js");
|
||||||
Script.Include("history.js");
|
Script.Include("history.js");
|
||||||
|
Style.Require("FontAwesome");
|
||||||
|
|
||||||
if (Model.DialogMode) {
|
if (Model.DialogMode) {
|
||||||
Style.Include("dialog-mode.css");
|
Style.Include("dialog-mode.css");
|
||||||
}
|
}
|
||||||
@@ -20,28 +21,42 @@
|
|||||||
<div id="media-library">
|
<div id="media-library">
|
||||||
<div id="media-library-toolbar">
|
<div id="media-library-toolbar">
|
||||||
<a href="#" data-bind="visible: displayed(), click: importMedia" class="button" id="button-import">@T("Import")</a>
|
<a href="#" data-bind="visible: displayed(), click: importMedia" class="button" id="button-import">@T("Import")</a>
|
||||||
|
<a href="#" data-bind="visible: displayed(), attr: { href: '@HttpUtility.JavaScriptStringEncode(Url.Action("Create", "Folder", new { area = "Orchard.MediaLibrary"}))/' + displayed() }" class="button" id="button-create-folder">@T("Create Folder")</a>
|
||||||
|
<a href="#" data-bind="visible: displayed(), attr: { href: '@HttpUtility.JavaScriptStringEncode(Url.Action("Edit", "Folder", new { area = "Orchard.MediaLibrary"}))/' + displayed() }" class="button" id="button-edit-folder">@T("Edit Folder")</a>
|
||||||
|
|
||||||
|
<label for="filterMediaType">@T("Show")</label>
|
||||||
|
<select id="filterMediaType" name="FilteredMediaType" data-bind="selectedOptions: mediaType">
|
||||||
|
@Html.SelectOption("", true, T("Any (show all)").ToString())
|
||||||
|
@foreach(var mediaType in Model.MediaTypes) {
|
||||||
|
@Html.SelectOption(mediaType, false, mediaType)
|
||||||
|
}
|
||||||
|
</select>
|
||||||
|
|
||||||
|
<label for="orderMedia">@T("Ordered by")</label>
|
||||||
|
<select data-bind="selectedOptions: orderMedia" id="orderMedia" name="OrderMedia">
|
||||||
|
@Html.SelectOption("title", false, T("title").ToString())
|
||||||
|
@Html.SelectOption("created", true, T("recently created").ToString())
|
||||||
|
@Html.SelectOption("published", false, T("recently published").ToString())
|
||||||
|
@Html.SelectOption("modified", false, T("recently modified").ToString())
|
||||||
|
</select>
|
||||||
|
|
||||||
<div id="media-library-toolbar-actions">
|
<div id="media-library-toolbar-actions">
|
||||||
<a href="@Url.Action("Create", "Folder", new {id = viewModel.Folder, area = "Orchard.MediaLibrary"})" class="button" id="button-create-folder">@T("Create Folder")</a>
|
|
||||||
<a href="#" data-bind="attr: { href: '@HttpUtility.JavaScriptStringEncode(Url.Action("Edit", "Folder", new { area = "Orchard.MediaLibrary"}))/' + displayed() }" class="button" id="button-edit-folder">@T("Edit Folder")</a>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="media-library-main">
|
<div id="media-library-main">
|
||||||
<div id="media-library-main-navigation">
|
<div id="media-library-main-navigation">
|
||||||
<ul>
|
<ul>
|
||||||
|
<li><a href="#" data-bind="click: selectRecent, css: { selected: !displayed() }" id="button-recent"><i class="icon-time"></i>@T("Recent")</a>
|
||||||
<li id="media-library-main-navigation-tree">
|
<li id="media-library-main-navigation-tree">
|
||||||
<h3>@T("Categories")</h3>
|
<ul>
|
||||||
|
|
||||||
<ul class="">
|
|
||||||
@foreach (var folder in viewModel.Folders) {
|
@foreach (var folder in viewModel.Folders) {
|
||||||
<li>
|
<li>
|
||||||
@Display.Partial(TemplateName: "MediaManagerFolder", Model: folder)
|
@Display.Partial(TemplateName: "MediaManagerFolder", Model: folder)
|
||||||
</li>
|
</li>
|
||||||
}
|
}
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
</li>
|
</li>
|
||||||
<li><h3>@T("Recent")</h3></li>
|
|
||||||
<li><h3>@T("Starred")</h3></li>
|
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
<div id="media-library-main-list-wrapper">
|
<div id="media-library-main-list-wrapper">
|
||||||
@@ -148,7 +163,9 @@
|
|||||||
self.displayed = ko.observable();
|
self.displayed = ko.observable();
|
||||||
self.pendingRequest = ko.observable(false);
|
self.pendingRequest = ko.observable(false);
|
||||||
self.mediaItemsCount = 0;
|
self.mediaItemsCount = 0;
|
||||||
|
self.orderMedia = ko.observableArray(['created']);
|
||||||
|
self.mediaType = ko.observableArray([]);
|
||||||
|
|
||||||
self.getMediaItems = function (id, max) {
|
self.getMediaItems = function (id, max) {
|
||||||
if (self.pendingRequest()) {
|
if (self.pendingRequest()) {
|
||||||
return;
|
return;
|
||||||
@@ -161,7 +178,7 @@
|
|||||||
|
|
||||||
self.pendingRequest(true);
|
self.pendingRequest(true);
|
||||||
|
|
||||||
var url = '@HttpUtility.JavaScriptStringEncode(Url.Action("MediaItems", "Admin"))/' + id + '?skip=' + self.results().length + '&count=' + max;
|
var url = '@HttpUtility.JavaScriptStringEncode(Url.Action("MediaItems", "Admin"))/' + id + '?skip=' + self.results().length + '&count=' + max + '&order=' + self.orderMedia() + '&mediaType=' + self.mediaType();
|
||||||
|
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "GET",
|
type: "GET",
|
||||||
@@ -235,6 +252,51 @@
|
|||||||
window.history.pushState({ action: 'displayFolder', folder: id }, '', '?folder=' + id);
|
window.history.pushState({ action: 'displayFolder', folder: id }, '', '?folder=' + id);
|
||||||
self.displayFolder(id);
|
self.displayFolder(id);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
self.selectRecent = function () {
|
||||||
|
window.history.pushState({ action: 'selectRecent' }, '', '?recent');
|
||||||
|
|
||||||
|
self.results([]);
|
||||||
|
self.displayed(null);
|
||||||
|
var max = 20;
|
||||||
|
|
||||||
|
if (self.pendingRequest()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self.results().length > 0 && self.results().length >= self.mediaItemsCount) {
|
||||||
|
console.log('no more content, mediaItemsCount: ' + self.mediaItemsCount);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.pendingRequest(true);
|
||||||
|
|
||||||
|
var url = '@HttpUtility.JavaScriptStringEncode(Url.Action("RecentMediaItems", "Admin"))?skip=' + self.results().length + '&count=' + max + '&order=' + self.orderMedia() + '&mediaType=' + self.mediaType();
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
type: "GET",
|
||||||
|
url: url,
|
||||||
|
}).done(function (data) {
|
||||||
|
var mediaItems = data.mediaItems;
|
||||||
|
self.mediaItemsCount = data.mediaItemsCount;
|
||||||
|
for (var i = 0; i < mediaItems.length; i++) {
|
||||||
|
var item = new MediaPartViewModel(mediaItems[i]);
|
||||||
|
self.results.push(item);
|
||||||
|
|
||||||
|
// pre-select result which are already part of the selection
|
||||||
|
var selection = self.selection();
|
||||||
|
for (var j = 0; j < selection.length; j++) {
|
||||||
|
if (selection[j].data.id == item.data.id) {
|
||||||
|
viewModel.toggleSelect(item, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}).fail(function(data) {
|
||||||
|
console.error(data);
|
||||||
|
}).always(function () {
|
||||||
|
self.pendingRequest(false);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
self.toggleSelect = function (searchResult, force) {
|
self.toggleSelect = function (searchResult, force) {
|
||||||
var index = $.inArray(searchResult, self.selection());
|
var index = $.inArray(searchResult, self.selection());
|
||||||
@@ -264,7 +326,13 @@
|
|||||||
self.scrolled = function(data, event) {
|
self.scrolled = function(data, event) {
|
||||||
var elem = event.target;
|
var elem = event.target;
|
||||||
if (elem.scrollTop > (elem.scrollHeight - elem.offsetHeight - 300)) {
|
if (elem.scrollTop > (elem.scrollHeight - elem.offsetHeight - 300)) {
|
||||||
self.getMediaItems(self.displayed(), 20);
|
if(seld.displayed()) {
|
||||||
|
self.getMediaItems(self.displayed(), 20);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// todo, infinite scrolling for 'recent'
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -272,6 +340,24 @@
|
|||||||
var url = '@HttpUtility.JavaScriptStringEncode(Url.Action("Import", "Admin"))/' + self.displayed();
|
var url = '@HttpUtility.JavaScriptStringEncode(Url.Action("Import", "Admin"))/' + self.displayed();
|
||||||
window.location = url;
|
window.location = url;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
self.orderMedia.subscribe(function (newValue) {
|
||||||
|
if(self.displayed()) {
|
||||||
|
self.selectFolder(self.displayed());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
self.selectRecent();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
self.mediaType.subscribe(function (newValue) {
|
||||||
|
if(self.displayed()) {
|
||||||
|
self.selectFolder(self.displayed());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
self.selectRecent();
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
var viewModel = new MediaIndexViewModel();
|
var viewModel = new MediaIndexViewModel();
|
||||||
@@ -283,11 +369,21 @@
|
|||||||
window.history.pushState({ action: 'displayFolder', folder: @viewModel.Folder.Value }, '', '?folder=@viewModel.Folder.Value');
|
window.history.pushState({ action: 'displayFolder', folder: @viewModel.Folder.Value }, '', '?folder=@viewModel.Folder.Value');
|
||||||
</text>
|
</text>
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
<text>
|
||||||
|
viewModel.selectRecent();
|
||||||
|
window.history.pushState({ action: 'selectRecent' }, '', '?recent');
|
||||||
|
</text>
|
||||||
|
}
|
||||||
|
|
||||||
window.onpopstate = function (event) {
|
window.onpopstate = function (event) {
|
||||||
if (event && event.state && event.state.action == 'displayFolder') {
|
if (event && event.state && event.state.action == 'displayFolder') {
|
||||||
viewModel.displayFolder(event.state.folder);
|
viewModel.displayFolder(event.state.folder);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (event && event.state && event.state.action == 'selectRecent') {
|
||||||
|
viewModel.selectRecent();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
$("#media-library-main-list").on("mousedown", "li", function (e) {
|
$("#media-library-main-list").on("mousedown", "li", function (e) {
|
||||||
@@ -366,9 +462,11 @@
|
|||||||
},
|
},
|
||||||
}).done(function (result) {
|
}).done(function (result) {
|
||||||
if (result) {
|
if (result) {
|
||||||
viewModel.results.remove(function(item) {
|
if(viewModel.displayed()) {
|
||||||
return ids.indexOf(item.data.id) != -1;
|
viewModel.results.remove(function(item) {
|
||||||
});
|
return ids.indexOf(item.data.id) != -1;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
viewModel.clearSelection();
|
viewModel.clearSelection();
|
||||||
} else {
|
} else {
|
||||||
|
@@ -2,8 +2,7 @@
|
|||||||
|
|
||||||
<div class="media-library-folder">
|
<div class="media-library-folder">
|
||||||
<div class="media-library-folder-title" data-bind="click: function(data, event) { selectFolder(@Model.TermId); }, css: { selected: displayed() == $element.getAttribute('data-term-id') }" data-term-id="@Model.TermId" >
|
<div class="media-library-folder-title" data-bind="click: function(data, event) { selectFolder(@Model.TermId); }, css: { selected: displayed() == $element.getAttribute('data-term-id') }" data-term-id="@Model.TermId" >
|
||||||
<span class="media-library-navigation-folder-icon"> </span>
|
<a data-bind="disable: $root.pendingRequest" href="#" class="media-library-navigation-folder-link" data-mediapath="@Model.MediaPath"><i class="icon-folder-close-alt"></i>@Model.Name</a>
|
||||||
<a data-bind="disable: $root.pendingRequest" href="#" class="media-library-navigation-folder-link" data-mediapath="@Model.MediaPath">@Model.Name</a>
|
|
||||||
</div>
|
</div>
|
||||||
@if (Model.Folders.Any()) {
|
@if (Model.Folders.Any()) {
|
||||||
<ul>
|
<ul>
|
||||||
|
Reference in New Issue
Block a user