forked from immense/knockout-tinymce
-
Notifications
You must be signed in to change notification settings - Fork 0
/
knockout-tinymce.coffee
98 lines (76 loc) · 3.25 KB
/
knockout-tinymce.coffee
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
(($, ko) ->
binding =
after: [
"attr"
"value"
]
defaults: {}
extensions: {}
init: (element, valueAccessor, allBindings, viewModel, bindingContext) ->
throw "valueAccessor must be writeable and observable" unless ko.isWriteableObservable(valueAccessor())
# Get custom configuration object from the 'wysiwygConfig' binding, more settings here... http://www.tinymce.com/wiki.php/Configuration
options = (if allBindings.has("tinymceConfig") then allBindings.get("tinymceConfig") else null)
# Get any extensions that have been enabled for this instance.
ext = (if allBindings.has("tinymceExtensions") then allBindings.get("tinymceExtensions") else [])
settings = configure(binding["defaults"], ext, options, arguments)
# Ensure the valueAccessor's value has been applied to the underlying element, before instanciating the tinymce plugin
$(element)[if $(element).is('input, textarea') then 'text' else 'html'] valueAccessor()()
# Defer TinyMCE instantiation
setTimeout (->
$(element).tinymce settings
return
), 0
# To prevent a memory leak, ensure that the underlying element's disposal destroys it's associated editor.
ko.utils["domNodeDisposal"].addDisposeCallback element, ->
tinymce = $(element).tinymce()
tinymce.remove() if tinymce
return
controlsDescendantBindings: true
update: (element, valueAccessor, allBindings, viewModel, bindingContext) ->
tinymce = $(element).tinymce()
value = valueAccessor()()
if tinymce
if tinymce.getContent() isnt value
tinymce.setContent value
return
configure = (defaults, extensions, options, args) ->
# Apply global configuration over TinyMCE defaults
config = $.extend(true, {}, defaults)
if options
# Concatenate element specific configuration
ko.utils.objectForEach options, (property) ->
if Object::toString.call(options[property]) is "[object Array]"
config[property] = [] unless config[property]
options[property] = ko.utils.arrayGetDistinctValues(config[property].concat(options[property]))
return
$.extend true, config, options
# Ensure paste functionality
unless config["plugins"]
config["plugins"] = ["paste"]
else config["plugins"].push "paste" if $.inArray("paste", config["plugins"]) is -1
# Define change handler
applyChange = (editor) ->
# Ensure the valueAccessor state to achieve a realtime responsive UI.
editor.on "change keyup nodechange", (e) ->
# Update the valueAccessor
args[1]() editor.getContent()
# Run all applied extensions
for name of extensions
binding["extensions"][extensions[name]] editor, e, args[2], args[4] if extensions.hasOwnProperty(name)
return
return
if typeof (config["setup"]) is "function"
setup = config["setup"]
# Concatenate setup functionality with the change handler
config["setup"] = (editor) ->
setup editor
applyChange editor
return
else
# Apply change handler
config["setup"] = applyChange
config
# Export the binding
ko.bindingHandlers["tinymce"] = binding
return
) jQuery, ko